'use strict'; const {Duplex} = require('stream'); class Pump { constructor(iterator, readable) { this.iterator = iterator; this.readable = readable; this.done = false; } start() { if (this.done) return Promise.resolve(); for (;;) { const item = this.iterator.next(); if (item.done) return Promise.resolve(); if (!this.readable.push(item.value)) break; } return new Promise((resolve, reject) => { this.resolve = resolve; this.reject = reject; }); } resume() { if (!this.resolve) return; for (;;) { const item = this.iterator.next(); if (item.done) { this.done = true; this.resolve(); return; } if (!this.readable.push(item.value)) break; } } } function* dump(value, options, processed) { if (!processed) { if (typeof value?.toJSON == 'function') { value = value.toJSON(''); } if (options.replacer) { value = options.replacer('', value); } } switch (typeof value) { case 'function': case 'symbol': case 'undefined': return; case 'number': if (isNaN(value) || !isFinite(value)) { yield {name: 'nullValue', value: null}; } value = String(value); if (options.streamNumbers) { yield {name: 'startNumber'}; yield {name: 'numberChunk', value}; yield {name: 'endNumber'}; } if (options.packNumbers) { yield {name: 'numberValue', value}; } return; case 'string': if (options.streamStrings) { yield {name: 'startString'}; yield {name: 'stringChunk', value}; yield {name: 'endString'}; } if (options.packStrings) { yield {name: 'stringValue', value}; } return; case 'boolean': yield value ? {name: 'trueValue', value: true} : {name: 'falseValue', value: false}; return; case 'object': break; default: return; // skip anything else } // null if (value === null) { yield {name: 'nullValue', value: null}; return; } // Array if (Array.isArray(value)) { yield {name: 'startArray'}; for (let i = 0; i < value.length; ++i) { let v = value[i]; if (typeof v?.toJSON == 'function') { v = v.toJSON(String(i)); } if (options.replacer) { v = options.replacer(String(i), v); } switch (typeof v) { case 'function': case 'symbol': case 'undefined': v = null; // force null break; } yield* dump(v, options, true); } yield {name: 'endArray'}; return; } // Object yield {name: 'startObject'}; for (let [k, v] of Object.entries(value)) { if (options.dict && options.dict[k] !== 1) continue; if (typeof v?.toJSON == 'function') { v = v.toJSON(k); } if (options.replacer) { v = options.replacer(k, v); } switch (typeof v) { case 'function': case 'symbol': case 'undefined': continue; } if (options.streamKeys) { yield {name: 'startKey'}; yield {name: 'stringChunk', value: k}; yield {name: 'endKey'}; } if (options.packKeys) { yield {name: 'keyValue', value: k}; } yield* dump(v, options, true); } yield {name: 'endObject'}; } const kOptions = Symbol('options'), kPump = Symbol('pump'); class Disassembler extends Duplex { static make(options) { return new Disassembler(options); } constructor(options) { super(Object.assign({}, options, {writableObjectMode: true, readableObjectMode: true})); const opt = {packKeys: true, packStrings: true, packNumbers: true, streamKeys: true, streamStrings: true, streamNumbers: true}; if (options) { 'packValues' in options && (opt.packKeys = opt.packStrings = opt.packNumbers = options.packValues); 'packKeys' in options && (opt.packKeys = options.packKeys); 'packStrings' in options && (opt.packStrings = options.packStrings); 'packNumbers' in options && (opt.packNumbers = options.packNumbers); 'streamValues' in options && (opt.streamKeys = opt.streamStrings = opt.streamNumbers = options.streamValues); 'streamKeys' in options && (opt.streamKeys = options.streamKeys); 'streamStrings' in options && (opt.streamStrings = options.streamStrings); 'streamNumbers' in options && (opt.streamNumbers = options.streamNumbers); if (typeof options.replacer == 'function') { opt.replacer = options.replacer; } else if (Array.isArray(options.replacer)) { opt.dict = options.replacer.reduce((acc, k) => ((acc[k] = 1), acc), {}); } } !opt.packKeys && (opt.streamKeys = true); !opt.packStrings && (opt.streamStrings = true); !opt.packNumbers && (opt.streamNumbers = true); this[kOptions] = opt; } _read() { this[kPump]?.resume(); } _write(chunk, _, callback) { const iterator = dump(chunk, this[kOptions]), pump = (this[kPump] = new Pump(iterator, this)); pump.start().then( () => ((this[kPump] = null), callback()), error => ((this[kPump] = null), callback(error)) ); } _final(callback) { this.push(null); callback(); } } Disassembler.disassembler = Disassembler.make; Disassembler.make.Constructor = Disassembler; module.exports = Disassembler; module.exports.dump = dump;