From 09ad64d66de6222e5d029ef40a93287b7f5d8275 Mon Sep 17 00:00:00 2001 From: James M Snell Date: Sat, 10 Jul 2021 20:56:56 -0700 Subject: [PATCH] stream: add CompressionStream and DecompressionStream Signed-off-by: James M Snell PR-URL: https://github.com/nodejs/node/pull/39348 Reviewed-By: Antoine du Hamel Reviewed-By: Matteo Collina --- doc/api/webstreams.md | 51 ++++++ lib/internal/webstreams/compression.js | 164 ++++++++++++++++++ lib/stream/web.js | 7 + .../test-whatwg-webstreams-compression.js | 59 +++++++ 4 files changed, 281 insertions(+) create mode 100644 lib/internal/webstreams/compression.js create mode 100644 test/parallel/test-whatwg-webstreams-compression.js diff --git a/doc/api/webstreams.md b/doc/api/webstreams.md index 0bacb7b475f094..083567aa8da002 100644 --- a/doc/api/webstreams.md +++ b/doc/api/webstreams.md @@ -1217,5 +1217,56 @@ added: REPLACEME * Type: {WritableStream} +### Class: `CompressionStream` + +#### `new CompressionStream(format)` + + +* `format` {string} One of either `'deflate'` or `'gzip'`. + +#### `compressionStream.readable` + + +* Type: {ReadableStream} + +#### `compressionStream.writable` + + +* Type: {WritableStream} + +### Class: `DecompressionStream` + + +#### `new DecompressionStream(format)` + + +* `format` {string} One of either `'deflate'` or `'gzip'`. + +#### `decompressionStream.readable` + + +* Type: {ReadableStream} + +#### `deccompressionStream.writable` + + +* Type: {WritableStream} + [Streams]: stream.md [WHATWG Streams Standard]: https://streams.spec.whatwg.org/ diff --git a/lib/internal/webstreams/compression.js b/lib/internal/webstreams/compression.js new file mode 100644 index 00000000000000..692f64af005af9 --- /dev/null +++ b/lib/internal/webstreams/compression.js @@ -0,0 +1,164 @@ +'use strict'; + +const { + ObjectDefineProperties, + Symbol, +} = primordials; + +const { + codes: { + ERR_INVALID_ARG_VALUE, + ERR_INVALID_THIS, + }, +} = require('internal/errors'); + +const { + newReadableWritablePairFromDuplex, +} = require('internal/webstreams/adapters'); + +const { + customInspect, + kEnumerableProperty, +} = require('internal/webstreams/util'); + +const { + customInspectSymbol: kInspect, +} = require('internal/util'); + +let zlib; +function lazyZlib() { + zlib ??= require('zlib'); + return zlib; +} + +const kHandle = Symbol('kHandle'); +const kTransform = Symbol('kTransform'); +const kType = Symbol('kType'); + +/** + * @typedef {import('./readablestream').ReadableStream} ReadableStream + * @typedef {import('./writablestream').WritableStream} WritableStream + */ + +function isCompressionStream(value) { + return typeof value?.[kHandle] === 'object' && + value?.[kType] === 'CompressionStream'; +} + +function isDecompressionStream(value) { + return typeof value?.[kHandle] === 'object' && + value?.[kType] === 'DecompressionStream'; +} + +class CompressionStream { + /** + * @param {'deflate'|'gzip'} format + */ + constructor(format) { + this[kType] = 'CompressionStream'; + switch (format) { + case 'deflate': + this[kHandle] = lazyZlib().createDeflate(); + break; + case 'gzip': + this[kHandle] = lazyZlib().createGzip(); + break; + default: + throw new ERR_INVALID_ARG_VALUE('format', format); + } + this[kTransform] = newReadableWritablePairFromDuplex(this[kHandle]); + } + + /** + * @readonly + * @type {ReadableStream} + */ + get readable() { + if (!isCompressionStream(this)) + throw new ERR_INVALID_THIS('CompressionStream'); + return this[kTransform].readable; + } + + /** + * @readonly + * @type {WritableStream} + */ + get writable() { + if (!isCompressionStream(this)) + throw new ERR_INVALID_THIS('CompressionStream'); + return this[kTransform].writable; + } + + [kInspect](depth, options) { + if (!isCompressionStream(this)) + throw new ERR_INVALID_THIS('CompressionStream'); + customInspect(depth, options, 'CompressionStream', { + readable: this[kTransform].readable, + writable: this[kTransform].writable, + }); + } +} + +class DecompressionStream { + /** + * @param {'deflate'|'gzip'} format + */ + constructor(format) { + this[kType] = 'DecompressionStream'; + switch (format) { + case 'deflate': + this[kHandle] = lazyZlib().createInflate(); + break; + case 'gzip': + this[kHandle] = lazyZlib().createGunzip(); + break; + default: + throw new ERR_INVALID_ARG_VALUE('format', format); + } + this[kTransform] = newReadableWritablePairFromDuplex(this[kHandle]); + } + + /** + * @readonly + * @type {ReadableStream} + */ + get readable() { + if (!isDecompressionStream(this)) + throw new ERR_INVALID_THIS('DecompressionStream'); + return this[kTransform].readable; + } + + /** + * @readonly + * @type {WritableStream} + */ + get writable() { + if (!isDecompressionStream(this)) + throw new ERR_INVALID_THIS('DecompressionStream'); + return this[kTransform].writable; + } + + [kInspect](depth, options) { + if (!isDecompressionStream(this)) + throw new ERR_INVALID_THIS('DecompressionStream'); + customInspect(depth, options, 'DecompressionStream', { + readable: this[kTransform].readable, + writable: this[kTransform].writable, + }); + } +} + +ObjectDefineProperties(CompressionStream.prototype, { + readable: kEnumerableProperty, + writable: kEnumerableProperty, +}); + +ObjectDefineProperties(DecompressionStream.prototype, { + readable: kEnumerableProperty, + writable: kEnumerableProperty, +}); + +module.exports = { + CompressionStream, + DecompressionStream, +}; diff --git a/lib/stream/web.js b/lib/stream/web.js index 06b320f001a646..5af05a8d3fcd95 100644 --- a/lib/stream/web.js +++ b/lib/stream/web.js @@ -36,6 +36,11 @@ const { TextDecoderStream, } = require('internal/webstreams/encoding'); +const { + CompressionStream, + DecompressionStream, +} = require('internal/webstreams/compression'); + module.exports = { ReadableStream, ReadableStreamDefaultReader, @@ -52,4 +57,6 @@ module.exports = { CountQueuingStrategy, TextEncoderStream, TextDecoderStream, + CompressionStream, + DecompressionStream, }; diff --git a/test/parallel/test-whatwg-webstreams-compression.js b/test/parallel/test-whatwg-webstreams-compression.js new file mode 100644 index 00000000000000..6d3d6253bd1b86 --- /dev/null +++ b/test/parallel/test-whatwg-webstreams-compression.js @@ -0,0 +1,59 @@ +// Flags: --no-warnings +'use strict'; + +const common = require('../common'); + +const { + CompressionStream, + DecompressionStream, +} = require('stream/web'); + +const assert = require('assert'); +const dec = new TextDecoder(); + +async function test(format) { + const gzip = new CompressionStream(format); + const gunzip = new DecompressionStream(format); + + gzip.readable.pipeTo(gunzip.writable).then(common.mustCall()); + + const reader = gunzip.readable.getReader(); + const writer = gzip.writable.getWriter(); + + await Promise.all([ + reader.read().then(({ value, done }) => { + assert.strictEqual(dec.decode(value), 'hello'); + }), + reader.read().then(({ done }) => assert(done)), + writer.write('hello'), + writer.close(), + ]); +} + +Promise.all(['gzip', 'deflate'].map((i) => test(i))).then(common.mustCall()); + +[1, 'hello', false, {}].forEach((i) => { + assert.throws(() => new CompressionStream(i), { + code: 'ERR_INVALID_ARG_VALUE', + }); + assert.throws(() => new DecompressionStream(i), { + code: 'ERR_INVALID_ARG_VALUE', + }); +}); + +assert.throws( + () => Reflect.get(CompressionStream.prototype, 'readable', {}), { + code: 'ERR_INVALID_THIS', + }); +assert.throws( + () => Reflect.get(CompressionStream.prototype, 'writable', {}), { + code: 'ERR_INVALID_THIS', + }); +assert.throws( + () => Reflect.get(DecompressionStream.prototype, 'readable', {}), { + code: 'ERR_INVALID_THIS', + }); +assert.throws( + () => Reflect.get(DecompressionStream.prototype, 'writable', {}), { + code: 'ERR_INVALID_THIS', + });