/* 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();
      });
    });
  });
});