| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169 |
- 'use strict';
- const {Readable, Writable, Duplex, Transform} = require('stream');
- function Final(value) {
- this.value = value;
- }
- function Many(values) {
- this.values = values;
- }
- const processData = (result, stream) => {
- if (result instanceof Final) {
- result = result.value;
- } else if (result instanceof Many) {
- result = result.values;
- }
- if (result !== undefined && result !== null) {
- if (result instanceof Array) {
- result.forEach(value => value !== undefined && value !== null && stream.push(value));
- } else {
- stream.push(result);
- }
- }
- };
- const wrapFunction = fn =>
- new Transform({
- writableObjectMode: true,
- readableObjectMode: true,
- transform(chunk, encoding, callback) {
- try {
- const result = fn.call(this, chunk, encoding);
- if (result && typeof result.then == 'function') {
- // Promise
- result.then(result => (processData(result, this), callback(null)), error => callback(error));
- return;
- }
- if (result && typeof result.next == 'function') {
- // generator
- while (true) {
- const data = result.next();
- processData(data.value, this);
- if (data.done) break;
- }
- } else {
- processData(result, this);
- }
- callback(null);
- } catch (error) {
- callback(error);
- }
- }
- });
- const wrapArray = array =>
- new Transform({
- writableObjectMode: true,
- readableObjectMode: true,
- transform(chunk, encoding, callback) {
- try {
- let value = chunk;
- for (let i = 0; i < array.length; ++i) {
- const result = array[i].call(this, value, encoding);
- if (result instanceof Final) {
- value = result.value;
- break;
- }
- value = result;
- }
- processData(value, this);
- callback(null);
- } catch (error) {
- callback(error);
- }
- }
- });
- const convertToTransform = fn => {
- if (typeof fn === 'function') return wrapFunction(fn);
- if (fn instanceof Array) return fn.length ? wrapArray(fn) : 0;
- return null;
- };
- class Chain extends Duplex {
- constructor(fns, options) {
- super(options || {writableObjectMode: true, readableObjectMode: true});
- if (!(fns instanceof Array) || !fns.length) {
- throw Error("Chain's argument should be a non-empty array.");
- }
- this.streams = fns
- .map((fn, index) => {
- if (typeof fn === 'function' || fn instanceof Array) return Chain.convertToTransform(fn);
- if (
- fn instanceof Duplex ||
- fn instanceof Transform ||
- (!index && fn instanceof Readable) ||
- (index === fns.length - 1 && fn instanceof Writable)
- ) {
- return fn;
- }
- throw Error('Arguments should be functions or streams.');
- })
- .filter(s => s);
- this.input = this.streams[0];
- this.output = this.streams.reduce((output, stream) => (output && output.pipe(stream)) || stream);
- if (!(this.input instanceof Writable)) {
- this._write = (_1, _2, callback) => callback(null);
- this._final = callback => callback(null); // unavailable in Node 6
- this.input.on('end', () => this.end());
- }
- if (this.output instanceof Readable) {
- this.output.on('data', chunk => !this.push(chunk) && this.output.pause());
- this.output.on('end', () => this.push(null));
- } else {
- this._read = () => {}; // nop
- this.resume();
- this.output.on('finish', () => this.push(null));
- }
- // connect events
- if (!options || !options.skipEvents) {
- this.streams.forEach(stream => stream.on('error', error => this.emit('error', error)));
- }
- }
- _write(chunk, encoding, callback) {
- let error = null;
- try {
- this.input.write(chunk, encoding, e => callback(e || error));
- } catch (e) {
- error = e;
- }
- }
- _final(callback) {
- // unavailable in Node 6
- let error = null;
- try {
- this.input.end(null, null, e => callback(e || error));
- } catch (e) {
- error = e;
- }
- }
- _read() {
- this.output.resume();
- }
- static chain(fns, options) {
- return new Chain(fns, options);
- }
- static final(value) {
- return new Final(value);
- }
- static many(values) {
- return new Many(values);
- }
- static convertToTransform(fn) {
- if (typeof fn === 'function') return wrapFunction(fn);
- if (fn instanceof Array) return fn.length ? wrapArray(fn) : 0;
- return null;
- }
- }
- Chain.make = Chain.chain;
- Chain.make.Constructor = Chain;
- module.exports = Chain;
|