379 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			379 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
/* Copyright 2017 Mozilla Foundation
 | 
						|
 *
 | 
						|
 * Licensed under the Apache License, Version 2.0 (the "License");
 | 
						|
 * you may not use this file except in compliance with the License.
 | 
						|
 * You may obtain a copy of the License at
 | 
						|
 *
 | 
						|
 *     http://www.apache.org/licenses/LICENSE-2.0
 | 
						|
 *
 | 
						|
 * Unless required by applicable law or agreed to in writing, software
 | 
						|
 * distributed under the License is distributed on an "AS IS" BASIS,
 | 
						|
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
						|
 * See the License for the specific language governing permissions and
 | 
						|
 * limitations under the License.
 | 
						|
 */
 | 
						|
 | 
						|
import { createPromiseCapability, MessageHandler } from '../../src/shared/util';
 | 
						|
 | 
						|
describe('util_stream', function () {
 | 
						|
  // Temporary fake port for sending messages between main and worker.
 | 
						|
  class FakePort {
 | 
						|
    constructor() {
 | 
						|
      this._listeners = [];
 | 
						|
      this._deferred = Promise.resolve(undefined);
 | 
						|
    }
 | 
						|
 | 
						|
    postMessage(obj) {
 | 
						|
      let event = { data: obj, };
 | 
						|
      this._deferred.then(() => {
 | 
						|
        this._listeners.forEach(function (listener) {
 | 
						|
          listener.call(this, event);
 | 
						|
        }, this);
 | 
						|
      });
 | 
						|
    }
 | 
						|
 | 
						|
    addEventListener(name, listener) {
 | 
						|
      this._listeners.push(listener);
 | 
						|
    }
 | 
						|
 | 
						|
    removeEventListener(name, listener) {
 | 
						|
      let i = this._listeners.indexOf(listener);
 | 
						|
      this._listeners.splice(i, 1);
 | 
						|
    }
 | 
						|
 | 
						|
    terminate() {
 | 
						|
      this._listeners = [];
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  // Sleep function to wait for sometime, similar to setTimeout but faster.
 | 
						|
  function sleep(ticks) {
 | 
						|
    return Promise.resolve().then(() => {
 | 
						|
      return (ticks && sleep(ticks - 1));
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  describe('sendWithStream', function () {
 | 
						|
    it('should return a ReadableStream', function () {
 | 
						|
      let port = new FakePort();
 | 
						|
      let messageHandler1 = new MessageHandler('main', 'worker', port);
 | 
						|
      let readable = messageHandler1.sendWithStream('fakeHandler');
 | 
						|
      // Check if readable is an instance of ReadableStream.
 | 
						|
      expect(typeof readable).toEqual('object');
 | 
						|
      expect(typeof readable.getReader).toEqual('function');
 | 
						|
    });
 | 
						|
 | 
						|
    it('should read using a reader', function (done) {
 | 
						|
      let log = '';
 | 
						|
      let port = new FakePort();
 | 
						|
      let messageHandler1 = new MessageHandler('main', 'worker', port);
 | 
						|
      let messageHandler2 = new MessageHandler('worker', 'main', port);
 | 
						|
      messageHandler2.on('fakeHandler', (data, sink) => {
 | 
						|
        sink.onPull = function () {
 | 
						|
          log += 'p';
 | 
						|
        };
 | 
						|
        sink.onCancel = function (reason) {
 | 
						|
          log += 'c';
 | 
						|
        };
 | 
						|
        sink.ready.then(() => {
 | 
						|
          sink.enqueue('hi');
 | 
						|
          return sink.ready;
 | 
						|
        }).then(() => {
 | 
						|
          sink.close();
 | 
						|
        });
 | 
						|
        return sleep(5);
 | 
						|
      });
 | 
						|
      let readable = messageHandler1.sendWithStream('fakeHandler', {}, {
 | 
						|
        highWaterMark: 1,
 | 
						|
        size() {
 | 
						|
          return 1;
 | 
						|
        },
 | 
						|
      });
 | 
						|
      let reader = readable.getReader();
 | 
						|
      sleep(10).then(() => {
 | 
						|
        expect(log).toEqual('');
 | 
						|
        return reader.read();
 | 
						|
      }).then((result) => {
 | 
						|
        expect(log).toEqual('p');
 | 
						|
        expect(result.value).toEqual('hi');
 | 
						|
        expect(result.done).toEqual(false);
 | 
						|
        return sleep(10);
 | 
						|
      }).then(() => {
 | 
						|
        return reader.read();
 | 
						|
      }).then((result) => {
 | 
						|
        expect(result.value).toEqual(undefined);
 | 
						|
        expect(result.done).toEqual(true);
 | 
						|
        done();
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('should not read any data when cancelled', function (done) {
 | 
						|
      let log = '';
 | 
						|
      let port = new FakePort();
 | 
						|
      let messageHandler2 = new MessageHandler('worker', 'main', port);
 | 
						|
      messageHandler2.on('fakeHandler', (data, sink) => {
 | 
						|
        sink.onPull = function () {
 | 
						|
          log += 'p';
 | 
						|
        };
 | 
						|
        sink.onCancel = function (reason) {
 | 
						|
          log += 'c';
 | 
						|
        };
 | 
						|
        log += '0';
 | 
						|
        sink.ready.then(() => {
 | 
						|
          log += '1';
 | 
						|
          sink.enqueue([1, 2, 3, 4], 4);
 | 
						|
          return sink.ready;
 | 
						|
        }).then(() => {
 | 
						|
          log += '2';
 | 
						|
          sink.enqueue([5, 6, 7, 8], 4);
 | 
						|
          return sink.ready;
 | 
						|
        }).then(() => {
 | 
						|
          log += '3';
 | 
						|
          sink.close();
 | 
						|
        }, () => {
 | 
						|
          log += '4';
 | 
						|
        });
 | 
						|
      });
 | 
						|
      let messageHandler1 = new MessageHandler('main', 'worker', port);
 | 
						|
      let readable = messageHandler1.sendWithStream('fakeHandler', {}, {
 | 
						|
        highWaterMark: 4,
 | 
						|
        size(arr) {
 | 
						|
          return arr.length;
 | 
						|
        },
 | 
						|
      });
 | 
						|
 | 
						|
      let reader = readable.getReader();
 | 
						|
      sleep(10).then(() => {
 | 
						|
        expect(log).toEqual('01');
 | 
						|
        return reader.read();
 | 
						|
      }).then((result) => {
 | 
						|
        expect(result.value).toEqual([1, 2, 3, 4]);
 | 
						|
        expect(result.done).toEqual(false);
 | 
						|
        return sleep(10);
 | 
						|
      }).then(() => {
 | 
						|
        expect(log).toEqual('01p2');
 | 
						|
        return reader.cancel();
 | 
						|
      }).then(() => {
 | 
						|
        expect(log).toEqual('01p2c4');
 | 
						|
        done();
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('should not read when errored', function(done) {
 | 
						|
      let log = '';
 | 
						|
      let port = new FakePort();
 | 
						|
      let messageHandler2 = new MessageHandler('worker', 'main', port);
 | 
						|
      messageHandler2.on('fakeHandler', (data, sink) => {
 | 
						|
        sink.onPull = function () {
 | 
						|
          log += 'p';
 | 
						|
        };
 | 
						|
        sink.onCancel = function (reason) {
 | 
						|
          log += 'c';
 | 
						|
        };
 | 
						|
        sink.ready.then(() => {
 | 
						|
          sink.enqueue([1, 2, 3, 4], 4);
 | 
						|
          return sink.ready;
 | 
						|
        }).then(() => {
 | 
						|
          log += 'error';
 | 
						|
          sink.error('error');
 | 
						|
        });
 | 
						|
      });
 | 
						|
      let messageHandler1 = new MessageHandler('main', 'worker', port);
 | 
						|
      let readable = messageHandler1.sendWithStream('fakeHandler', {}, {
 | 
						|
        highWaterMark: 4,
 | 
						|
        size(arr) {
 | 
						|
          return arr.length;
 | 
						|
        },
 | 
						|
      });
 | 
						|
 | 
						|
      let reader = readable.getReader();
 | 
						|
 | 
						|
      sleep(10).then(() => {
 | 
						|
        expect(log).toEqual('');
 | 
						|
        return reader.read();
 | 
						|
      }).then((result) => {
 | 
						|
        expect(result.value).toEqual([1, 2, 3, 4]);
 | 
						|
        expect(result.done).toEqual(false);
 | 
						|
        return reader.read();
 | 
						|
      }).then(() => {
 | 
						|
      }, (reason) => {
 | 
						|
        expect(reason).toEqual('error');
 | 
						|
        done();
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('should read data with blocking promise', function (done) {
 | 
						|
      let log = '';
 | 
						|
      let port = new FakePort();
 | 
						|
      let messageHandler2 = new MessageHandler('worker', 'main', port);
 | 
						|
      messageHandler2.on('fakeHandler', (data, sink) => {
 | 
						|
        sink.onPull = function () {
 | 
						|
          log += 'p';
 | 
						|
        };
 | 
						|
        sink.onCancel = function (reason) {
 | 
						|
          log += 'c';
 | 
						|
        };
 | 
						|
        log += '0';
 | 
						|
        sink.ready.then(() => {
 | 
						|
          log += '1';
 | 
						|
          sink.enqueue([1, 2, 3, 4], 4);
 | 
						|
          return sink.ready;
 | 
						|
        }).then(() => {
 | 
						|
          log += '2';
 | 
						|
          sink.enqueue([5, 6, 7, 8], 4);
 | 
						|
          return sink.ready;
 | 
						|
        }).then(() => {
 | 
						|
          sink.close();
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      let messageHandler1 = new MessageHandler('main', 'worker', port);
 | 
						|
      let readable = messageHandler1.sendWithStream('fakeHandler', {}, {
 | 
						|
        highWaterMark: 4,
 | 
						|
        size(arr) {
 | 
						|
          return arr.length;
 | 
						|
        },
 | 
						|
      });
 | 
						|
 | 
						|
      let reader = readable.getReader();
 | 
						|
      // Sleep for 10ms, so that read() is not unblocking the ready promise.
 | 
						|
      // Chain all read() to stream in sequence.
 | 
						|
      sleep(10).then(() => {
 | 
						|
        expect(log).toEqual('01');
 | 
						|
        return reader.read();
 | 
						|
      }).then((result) => {
 | 
						|
        expect(result.value).toEqual([1, 2, 3, 4]);
 | 
						|
        expect(result.done).toEqual(false);
 | 
						|
        return sleep(10);
 | 
						|
      }).then(() => {
 | 
						|
        expect(log).toEqual('01p2');
 | 
						|
        return reader.read();
 | 
						|
      }).then((result) => {
 | 
						|
        expect(result.value).toEqual([5, 6, 7, 8]);
 | 
						|
        expect(result.done).toEqual(false);
 | 
						|
        return sleep(10);
 | 
						|
      }).then(() => {
 | 
						|
        expect(log).toEqual('01p2p');
 | 
						|
        return reader.read();
 | 
						|
      }).then((result) => {
 | 
						|
        expect(result.value).toEqual(undefined);
 | 
						|
        expect(result.done).toEqual(true);
 | 
						|
        done();
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('should read data with blocking promise and buffer whole data' +
 | 
						|
       ' into stream', function (done) {
 | 
						|
      let log = '';
 | 
						|
      let port = new FakePort();
 | 
						|
      let messageHandler2 = new MessageHandler('worker', 'main', port);
 | 
						|
      messageHandler2.on('fakeHandler', (data, sink) => {
 | 
						|
        sink.onPull = function () {
 | 
						|
          log += 'p';
 | 
						|
        };
 | 
						|
        sink.onCancel = function (reason) {
 | 
						|
          log += 'c';
 | 
						|
        };
 | 
						|
        log += '0';
 | 
						|
        sink.ready.then(() => {
 | 
						|
          log += '1';
 | 
						|
          sink.enqueue([1, 2, 3, 4], 4);
 | 
						|
          return sink.ready;
 | 
						|
        }).then(() => {
 | 
						|
          log += '2';
 | 
						|
          sink.enqueue([5, 6, 7, 8], 4);
 | 
						|
          return sink.ready;
 | 
						|
        }).then(() => {
 | 
						|
          sink.close();
 | 
						|
        });
 | 
						|
        return sleep(10);
 | 
						|
      });
 | 
						|
 | 
						|
      let messageHandler1 = new MessageHandler('main', 'worker', port);
 | 
						|
      let readable = messageHandler1.sendWithStream('fakeHandler', {}, {
 | 
						|
        highWaterMark: 8,
 | 
						|
        size(arr) {
 | 
						|
          return arr.length;
 | 
						|
        },
 | 
						|
      });
 | 
						|
 | 
						|
      let reader = readable.getReader();
 | 
						|
 | 
						|
      sleep(10).then(() => {
 | 
						|
        expect(log).toEqual('012');
 | 
						|
        return reader.read();
 | 
						|
      }).then((result) => {
 | 
						|
        expect(result.value).toEqual([1, 2, 3, 4]);
 | 
						|
        expect(result.done).toEqual(false);
 | 
						|
        return sleep(10);
 | 
						|
      }).then(() => {
 | 
						|
        expect(log).toEqual('012p');
 | 
						|
        return reader.read();
 | 
						|
      }).then((result) => {
 | 
						|
        expect(result.value).toEqual([5, 6, 7, 8]);
 | 
						|
        expect(result.done).toEqual(false);
 | 
						|
        return sleep(10);
 | 
						|
      }).then(() => {
 | 
						|
        expect(log).toEqual('012p');
 | 
						|
        return reader.read();
 | 
						|
      }).then((result) => {
 | 
						|
        expect(result.value).toEqual(undefined);
 | 
						|
        expect(result.done).toEqual(true);
 | 
						|
        done();
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it('should ignore any pull after close is called', function (done) {
 | 
						|
      let log = '';
 | 
						|
      let port = new FakePort();
 | 
						|
      let capability = createPromiseCapability();
 | 
						|
      let messageHandler2 = new MessageHandler('worker', 'main', port);
 | 
						|
      messageHandler2.on('fakeHandler', (data, sink) => {
 | 
						|
        sink.onPull = function () {
 | 
						|
          log += 'p';
 | 
						|
        };
 | 
						|
        sink.onCancel = function (reason) {
 | 
						|
          log += 'c';
 | 
						|
        };
 | 
						|
        log += '0';
 | 
						|
        sink.ready.then(() => {
 | 
						|
          log += '1';
 | 
						|
          sink.enqueue([1, 2, 3, 4], 4);
 | 
						|
        });
 | 
						|
        return capability.promise.then(() => {
 | 
						|
          sink.close();
 | 
						|
        });
 | 
						|
      });
 | 
						|
 | 
						|
      let messageHandler1 = new MessageHandler('main', 'worker', port);
 | 
						|
      let readable = messageHandler1.sendWithStream('fakeHandler', {}, {
 | 
						|
        highWaterMark: 10,
 | 
						|
        size(arr) {
 | 
						|
          return arr.length;
 | 
						|
        },
 | 
						|
      });
 | 
						|
 | 
						|
      let reader = readable.getReader();
 | 
						|
 | 
						|
      sleep(10).then(() => {
 | 
						|
        expect(log).toEqual('01');
 | 
						|
        capability.resolve();
 | 
						|
        return capability.promise.then(() => {
 | 
						|
          return reader.read();
 | 
						|
        });
 | 
						|
      }).then((result) => {
 | 
						|
        expect(result.value).toEqual([1, 2, 3, 4]);
 | 
						|
        expect(result.done).toEqual(false);
 | 
						|
        return sleep(10);
 | 
						|
      }).then(() => {
 | 
						|
        expect(log).toEqual('01');
 | 
						|
        return reader.read();
 | 
						|
      }).then((result) => {
 | 
						|
        expect(result.value).toEqual(undefined);
 | 
						|
        expect(result.done).toEqual(true);
 | 
						|
        done();
 | 
						|
      });
 | 
						|
    });
 | 
						|
  });
 | 
						|
});
 |