Skip to content

Commit 4cc7907

Browse files
jkremstargos
authored andcommittedFeb 10, 2025
zlib: add zstd support
Fixes: #48412 PR-URL: #52100 Reviewed-By: Yagiz Nizipli <yagiz@nizipli.com>
1 parent 380a8d8 commit 4cc7907

22 files changed

+997
-21
lines changed
 

‎benchmark/zlib/creation.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ const zlib = require('zlib');
55
const bench = common.createBenchmark(main, {
66
type: [
77
'Deflate', 'DeflateRaw', 'Inflate', 'InflateRaw', 'Gzip', 'Gunzip', 'Unzip',
8-
'BrotliCompress', 'BrotliDecompress',
8+
'BrotliCompress', 'BrotliDecompress', 'ZstdCompress', 'ZstdDecompress',
99
],
1010
options: ['true', 'false'],
1111
n: [5e5],

‎benchmark/zlib/pipe.js

+10-5
Original file line numberDiff line numberDiff line change
@@ -7,22 +7,27 @@ const bench = common.createBenchmark(main, {
77
inputLen: [1024],
88
duration: [5],
99
type: ['string', 'buffer'],
10-
algorithm: ['gzip', 'brotli'],
10+
algorithm: ['gzip', 'brotli', 'zstd'],
1111
}, {
1212
test: {
1313
inputLen: 1024,
1414
duration: 0.2,
1515
},
1616
});
1717

18+
const algorithms = {
19+
'gzip': [zlib.createGzip, zlib.createGunzip],
20+
'brotli': [zlib.createBrotliCompress, zlib.createBrotliDecompress],
21+
'zstd': [zlib.createZstdCompress, zlib.createZstdDecompress],
22+
};
23+
1824
function main({ inputLen, duration, type, algorithm }) {
1925
const buffer = Buffer.alloc(inputLen, fs.readFileSync(__filename));
2026
const chunk = type === 'buffer' ? buffer : buffer.toString('utf8');
2127

22-
const input = algorithm === 'gzip' ?
23-
zlib.createGzip() : zlib.createBrotliCompress();
24-
const output = algorithm === 'gzip' ?
25-
zlib.createGunzip() : zlib.createBrotliDecompress();
28+
const [createCompress, createUncompress] = algorithms[algorithm];
29+
const input = createCompress();
30+
const output = createUncompress();
2631

2732
let readFromOutput = 0;
2833
input.pipe(output);

‎doc/api/errors.md

+6
Original file line numberDiff line numberDiff line change
@@ -3314,6 +3314,12 @@ The requested functionality is not supported in worker threads.
33143314

33153315
Creation of a [`zlib`][] object failed due to incorrect configuration.
33163316

3317+
<a id="ERR_ZSTD_INVALID_PARAM"></a>
3318+
3319+
### `ERR_ZSTD_INVALID_PARAM`
3320+
3321+
An invalid parameter key was passed during construction of a Zstd stream.
3322+
33173323
<a id="HPE_CHUNK_EXTENSIONS_OVERFLOW"></a>
33183324

33193325
### `HPE_CHUNK_EXTENSIONS_OVERFLOW`

‎doc/api/zlib.md

+175-4
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@
77
<!-- source_link=lib/zlib.js -->
88

99
The `node:zlib` module provides compression functionality implemented using
10-
Gzip, Deflate/Inflate, and Brotli.
10+
Gzip, Deflate/Inflate, Brotli, and Zstd.
1111

1212
To access it:
1313

@@ -220,8 +220,8 @@ operations be cached to avoid duplication of effort.
220220

221221
## Compressing HTTP requests and responses
222222

223-
The `node:zlib` module can be used to implement support for the `gzip`, `deflate`
224-
and `br` content-encoding mechanisms defined by
223+
The `node:zlib` module can be used to implement support for the `gzip`, `deflate`,
224+
`br`, and `zstd` content-encoding mechanisms defined by
225225
[HTTP](https://tools.ietf.org/html/rfc7230#section-4.2).
226226

227227
The HTTP [`Accept-Encoding`][] header is used within an HTTP request to identify
@@ -284,7 +284,7 @@ const { pipeline } = require('node:stream');
284284
const request = http.get({ host: 'example.com',
285285
path: '/',
286286
port: 80,
287-
headers: { 'Accept-Encoding': 'br,gzip,deflate' } });
287+
headers: { 'Accept-Encoding': 'br,gzip,deflate,zstd' } });
288288
request.on('response', (response) => {
289289
const output = fs.createWriteStream('example.com_index.html');
290290

@@ -306,6 +306,9 @@ request.on('response', (response) => {
306306
case 'deflate':
307307
pipeline(response, zlib.createInflate(), output, onError);
308308
break;
309+
case 'zstd':
310+
pipeline(response, zlib.createZstdDecompress(), output, onError);
311+
break;
309312
default:
310313
pipeline(response, output, onError);
311314
break;
@@ -396,6 +399,9 @@ http.createServer((request, response) => {
396399
} else if (/\bbr\b/.test(acceptEncoding)) {
397400
response.writeHead(200, { 'Content-Encoding': 'br' });
398401
pipeline(raw, zlib.createBrotliCompress(), response, onError);
402+
} else if (/\bzstd\b/.test(acceptEncoding)) {
403+
response.writeHead(200, { 'Content-Encoding': 'zstd' });
404+
pipeline(raw, zlib.createZstdCompress(), response, onError);
399405
} else {
400406
response.writeHead(200, {});
401407
pipeline(raw, response, onError);
@@ -416,6 +422,7 @@ const buffer = Buffer.from('eJzT0yMA', 'base64');
416422
zlib.unzip(
417423
buffer,
418424
// For Brotli, the equivalent is zlib.constants.BROTLI_OPERATION_FLUSH.
425+
// For Zstd, the equivalent is zlib.constants.ZSTD_e_flush.
419426
{ finishFlush: zlib.constants.Z_SYNC_FLUSH },
420427
(err, buffer) => {
421428
if (err) {
@@ -487,6 +494,16 @@ these options have different ranges than the zlib ones:
487494

488495
See [below][Brotli parameters] for more details on Brotli-specific options.
489496

497+
### For Zstd-based streams
498+
499+
There are equivalents to the zlib options for Zstd-based streams, although
500+
these options have different ranges than the zlib ones:
501+
502+
* zlib's `level` option matches Zstd's `ZSTD_c_compressionLevel` option.
503+
* zlib's `windowBits` option matches Zstd's `ZSTD_c_windowLog` option.
504+
505+
See [below][Zstd parameters] for more details on Zstd-specific options.
506+
490507
## Flushing
491508

492509
Calling [`.flush()`][] on a compression stream will make `zlib` return as much
@@ -701,6 +718,50 @@ These advanced options are available for controlling decompression:
701718
* Boolean flag enabling “Large Window Brotli” mode (not compatible with the
702719
Brotli format as standardized in [RFC 7932][]).
703720

721+
### Zstd constants
722+
723+
<!-- YAML
724+
added: REPLACEME
725+
-->
726+
727+
There are several options and other constants available for Zstd-based
728+
streams:
729+
730+
#### Flush operations
731+
732+
The following values are valid flush operations for Zstd-based streams:
733+
734+
* `zlib.constants.ZSTD_e_continue` (default for all operations)
735+
* `zlib.constants.ZSTD_e_flush` (default when calling `.flush()`)
736+
* `zlib.constants.ZSTD_e_end` (default for the last chunk)
737+
738+
#### Compressor options
739+
740+
There are several options that can be set on Zstd encoders, affecting
741+
compression efficiency and speed. Both the keys and the values can be accessed
742+
as properties of the `zlib.constants` object.
743+
744+
The most important options are:
745+
746+
* `ZSTD_c_compressionLevel`
747+
* Set compression parameters according to pre-defined cLevel table. Default
748+
level is ZSTD\_CLEVEL\_DEFAULT==3.
749+
750+
#### Pledged Source Size
751+
752+
It's possible to specify the expected total size of the uncompressed input via
753+
`opts.pledgedSrcSize`. If the size doesn't match at the end of the input,
754+
compression will fail with the code `ZSTD_error_srcSize_wrong`.
755+
756+
#### Decompressor options
757+
758+
These advanced options are available for controlling decompression:
759+
760+
* `ZSTD_d_windowLogMax`
761+
* Select a size limit (in power of 2) beyond which the streaming API will
762+
refuse to allocate memory buffer in order to protect the host from
763+
unreasonable memory requirements.
764+
704765
## Class: `Options`
705766

706767
<!-- YAML
@@ -962,6 +1023,51 @@ added: v0.7.0
9621023
Reset the compressor/decompressor to factory defaults. Only applicable to
9631024
the inflate and deflate algorithms.
9641025

1026+
## Class: `ZstdOptions`
1027+
1028+
<!-- YAML
1029+
added: REPLACEME
1030+
-->
1031+
1032+
<!--type=misc-->
1033+
1034+
Each Zstd-based class takes an `options` object. All options are optional.
1035+
1036+
* `flush` {integer} **Default:** `zlib.constants.ZSTD_e_continue`
1037+
* `finishFlush` {integer} **Default:** `zlib.constants.ZSTD_e_end`
1038+
* `chunkSize` {integer} **Default:** `16 * 1024`
1039+
* `params` {Object} Key-value object containing indexed [Zstd parameters][].
1040+
* `maxOutputLength` {integer} Limits output size when using
1041+
[convenience methods][]. **Default:** [`buffer.kMaxLength`][]
1042+
1043+
For example:
1044+
1045+
```js
1046+
const stream = zlib.createZstdCompress({
1047+
chunkSize: 32 * 1024,
1048+
params: {
1049+
[zlib.constants.ZSTD_c_compressionLevel]: 10,
1050+
[zlib.constants.ZSTD_c_checksumFlag]: 1,
1051+
},
1052+
});
1053+
```
1054+
1055+
## Class: `zlib.ZstdCompress`
1056+
1057+
<!-- YAML
1058+
added: REPLACEME
1059+
-->
1060+
1061+
Compress data using the Zstd algorithm.
1062+
1063+
## Class: `zlib.ZstdDecompress`
1064+
1065+
<!-- YAML
1066+
added: REPLACEME
1067+
-->
1068+
1069+
Decompress data using the Zstd algorithm.
1070+
9651071
## `zlib.constants`
9661072

9671073
<!-- YAML
@@ -1135,6 +1241,26 @@ added: v0.5.8
11351241

11361242
Creates and returns a new [`Unzip`][] object.
11371243

1244+
## `zlib.createZstdCompress([options])`
1245+
1246+
<!-- YAML
1247+
added: REPLACEME
1248+
-->
1249+
1250+
* `options` {zstd options}
1251+
1252+
Creates and returns a new [`ZstdCompress`][] object.
1253+
1254+
## `zlib.createZstdDecompress([options])`
1255+
1256+
<!-- YAML
1257+
added: REPLACEME
1258+
-->
1259+
1260+
* `options` {zstd options}
1261+
1262+
Creates and returns a new [`ZstdDecompress`][] object.
1263+
11381264
## Convenience methods
11391265

11401266
<!--type=misc-->
@@ -1481,11 +1607,54 @@ changes:
14811607

14821608
Decompress a chunk of data with [`Unzip`][].
14831609

1610+
### `zlib.zstdCompress(buffer[, options], callback)`
1611+
1612+
<!-- YAML
1613+
added: REPLACEME
1614+
-->
1615+
1616+
* `buffer` {Buffer|TypedArray|DataView|ArrayBuffer|string}
1617+
* `options` {zstd options}
1618+
* `callback` {Function}
1619+
1620+
### `zlib.zstdCompressSync(buffer[, options])`
1621+
1622+
<!-- YAML
1623+
added: REPLACEME
1624+
-->
1625+
1626+
* `buffer` {Buffer|TypedArray|DataView|ArrayBuffer|string}
1627+
* `options` {zstd options}
1628+
1629+
Compress a chunk of data with [`ZstdCompress`][].
1630+
1631+
### `zlib.zstdDecompress(buffer[, options], callback)`
1632+
1633+
<!-- YAML
1634+
added: REPLACEME
1635+
-->
1636+
1637+
* `buffer` {Buffer|TypedArray|DataView|ArrayBuffer|string}
1638+
* `options` {zstd options}
1639+
* `callback` {Function}
1640+
1641+
### `zlib.zstdDecompressSync(buffer[, options])`
1642+
1643+
<!-- YAML
1644+
added: REPLACEME
1645+
-->
1646+
1647+
* `buffer` {Buffer|TypedArray|DataView|ArrayBuffer|string}
1648+
* `options` {zstd options}
1649+
1650+
Decompress a chunk of data with [`ZstdDecompress`][].
1651+
14841652
[Brotli parameters]: #brotli-constants
14851653
[Cyclic redundancy check]: https://en.wikipedia.org/wiki/Cyclic_redundancy_check
14861654
[Memory usage tuning]: #memory-usage-tuning
14871655
[RFC 7932]: https://www.rfc-editor.org/rfc/rfc7932.txt
14881656
[Streams API]: stream.md
1657+
[Zstd parameters]: #zstd-constants
14891658
[`.flush()`]: #zlibflushkind-callback
14901659
[`Accept-Encoding`]: https://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.3
14911660
[`BrotliCompress`]: #class-zlibbrotlicompress
@@ -1498,6 +1667,8 @@ Decompress a chunk of data with [`Unzip`][].
14981667
[`InflateRaw`]: #class-zlibinflateraw
14991668
[`Inflate`]: #class-zlibinflate
15001669
[`Unzip`]: #class-zlibunzip
1670+
[`ZstdCompress`]: #class-zlibzstdcompress
1671+
[`ZstdDecompress`]: #class-zlibzstddecompress
15011672
[`buffer.kMaxLength`]: buffer.md#bufferkmaxlength
15021673
[`deflateInit2` and `inflateInit2`]: https://zlib.net/manual.html#Advanced
15031674
[`stream.Transform`]: stream.md#class-streamtransform

‎lib/internal/errors.js

+1
Original file line numberDiff line numberDiff line change
@@ -1889,3 +1889,4 @@ E('ERR_WORKER_UNSERIALIZABLE_ERROR',
18891889
'Serializing an uncaught exception failed', Error);
18901890
E('ERR_WORKER_UNSUPPORTED_OPERATION',
18911891
'%s is not supported in workers', TypeError);
1892+
E('ERR_ZSTD_INVALID_PARAM', '%s is not a valid zstd parameter', RangeError);

‎lib/zlib.js

+103-4
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,7 @@ const {
4242
ERR_BUFFER_TOO_LARGE,
4343
ERR_INVALID_ARG_TYPE,
4444
ERR_OUT_OF_RANGE,
45+
ERR_ZSTD_INVALID_PARAM,
4546
},
4647
genericNodeError,
4748
} = require('internal/errors');
@@ -80,9 +81,12 @@ const {
8081
// Node's compression stream modes (node_zlib_mode)
8182
DEFLATE, DEFLATERAW, INFLATE, INFLATERAW, GZIP, GUNZIP, UNZIP,
8283
BROTLI_DECODE, BROTLI_ENCODE,
84+
ZSTD_COMPRESS, ZSTD_DECOMPRESS,
8385
// Brotli operations (~flush levels)
8486
BROTLI_OPERATION_PROCESS, BROTLI_OPERATION_FLUSH,
8587
BROTLI_OPERATION_FINISH, BROTLI_OPERATION_EMIT_METADATA,
88+
// Zstd end directives (~flush levels)
89+
ZSTD_e_continue, ZSTD_e_flush, ZSTD_e_end,
8690
} = constants;
8791

8892
// Translation table for return codes.
@@ -189,9 +193,11 @@ function zlibOnError(message, errno, code) {
189193
const FLUSH_BOUND = [
190194
[ Z_NO_FLUSH, Z_BLOCK ],
191195
[ BROTLI_OPERATION_PROCESS, BROTLI_OPERATION_EMIT_METADATA ],
196+
[ ZSTD_e_continue, ZSTD_e_end ],
192197
];
193198
const FLUSH_BOUND_IDX_NORMAL = 0;
194199
const FLUSH_BOUND_IDX_BROTLI = 1;
200+
const FLUSH_BOUND_IDX_ZSTD = 2;
195201

196202
// The base class for all Zlib-style streams.
197203
function ZlibBase(opts, mode, handle, { flush, finishFlush, fullFlush }) {
@@ -200,13 +206,15 @@ function ZlibBase(opts, mode, handle, { flush, finishFlush, fullFlush }) {
200206
// The ZlibBase class is not exported to user land, the mode should only be
201207
// passed in by us.
202208
assert(typeof mode === 'number');
203-
assert(mode >= DEFLATE && mode <= BROTLI_ENCODE);
209+
assert(mode >= DEFLATE && mode <= ZSTD_DECOMPRESS);
204210

205211
let flushBoundIdx;
206-
if (mode !== BROTLI_ENCODE && mode !== BROTLI_DECODE) {
207-
flushBoundIdx = FLUSH_BOUND_IDX_NORMAL;
208-
} else {
212+
if (mode === BROTLI_ENCODE || mode === BROTLI_DECODE) {
209213
flushBoundIdx = FLUSH_BOUND_IDX_BROTLI;
214+
} else if (mode === ZSTD_COMPRESS || mode === ZSTD_DECOMPRESS) {
215+
flushBoundIdx = FLUSH_BOUND_IDX_ZSTD;
216+
} else {
217+
flushBoundIdx = FLUSH_BOUND_IDX_NORMAL;
210218
}
211219

212220
if (opts) {
@@ -817,6 +825,89 @@ ObjectSetPrototypeOf(BrotliDecompress.prototype, Brotli.prototype);
817825
ObjectSetPrototypeOf(BrotliDecompress, Brotli);
818826

819827

828+
const zstdDefaultOpts = {
829+
flush: ZSTD_e_continue,
830+
finishFlush: ZSTD_e_end,
831+
fullFlush: ZSTD_e_flush,
832+
};
833+
function Zstd(opts, mode, initParamsArray, maxParam) {
834+
assert(mode === ZSTD_COMPRESS || mode === ZSTD_DECOMPRESS);
835+
836+
initParamsArray.fill(-1);
837+
if (opts?.params) {
838+
ObjectKeys(opts.params).forEach((origKey) => {
839+
const key = +origKey;
840+
if (NumberIsNaN(key) || key < 0 || key > maxParam ||
841+
(initParamsArray[key] | 0) !== -1) {
842+
throw new ERR_ZSTD_INVALID_PARAM(origKey);
843+
}
844+
845+
const value = opts.params[origKey];
846+
if (typeof value !== 'number' && typeof value !== 'boolean') {
847+
throw new ERR_INVALID_ARG_TYPE('options.params[key]',
848+
'number', opts.params[origKey]);
849+
}
850+
initParamsArray[key] = value;
851+
});
852+
}
853+
854+
const handle = mode === ZSTD_COMPRESS ?
855+
new binding.ZstdCompress() : new binding.ZstdDecompress();
856+
857+
const pledgedSrcSize = opts?.pledgedSrcSize ?? undefined;
858+
859+
this._writeState = new Uint32Array(2);
860+
handle.init(
861+
initParamsArray,
862+
pledgedSrcSize,
863+
this._writeState,
864+
processCallback,
865+
);
866+
867+
ReflectApply(ZlibBase, this, [opts, mode, handle, zstdDefaultOpts]);
868+
}
869+
ObjectSetPrototypeOf(Zstd.prototype, ZlibBase.prototype);
870+
ObjectSetPrototypeOf(Zstd, ZlibBase);
871+
872+
873+
const kMaxZstdCParam = MathMax(...ObjectKeys(constants).map(
874+
(key) => (key.startsWith('ZSTD_c_') ?
875+
constants[key] :
876+
0),
877+
));
878+
879+
const zstdInitCParamsArray = new Uint32Array(kMaxZstdCParam + 1);
880+
881+
function ZstdCompress(opts) {
882+
if (!(this instanceof ZstdCompress))
883+
return new ZstdCompress(opts);
884+
885+
ReflectApply(Zstd, this,
886+
[opts, ZSTD_COMPRESS, zstdInitCParamsArray, kMaxZstdCParam]);
887+
}
888+
ObjectSetPrototypeOf(ZstdCompress.prototype, Zstd.prototype);
889+
ObjectSetPrototypeOf(ZstdCompress, Zstd);
890+
891+
892+
const kMaxZstdDParam = MathMax(...ObjectKeys(constants).map(
893+
(key) => (key.startsWith('ZSTD_d_') ?
894+
constants[key] :
895+
0),
896+
));
897+
898+
const zstdInitDParamsArray = new Uint32Array(kMaxZstdDParam + 1);
899+
900+
function ZstdDecompress(opts) {
901+
if (!(this instanceof ZstdDecompress))
902+
return new ZstdDecompress(opts);
903+
904+
ReflectApply(Zstd, this,
905+
[opts, ZSTD_DECOMPRESS, zstdInitDParamsArray, kMaxZstdDParam]);
906+
}
907+
ObjectSetPrototypeOf(ZstdDecompress.prototype, Zstd.prototype);
908+
ObjectSetPrototypeOf(ZstdDecompress, Zstd);
909+
910+
820911
function createProperty(ctor) {
821912
return {
822913
__proto__: null,
@@ -855,6 +946,8 @@ module.exports = {
855946
Unzip,
856947
BrotliCompress,
857948
BrotliDecompress,
949+
ZstdCompress,
950+
ZstdDecompress,
858951

859952
// Convenience methods.
860953
// compress/decompress a string or buffer in one step.
@@ -876,6 +969,10 @@ module.exports = {
876969
brotliCompressSync: createConvenienceMethod(BrotliCompress, true),
877970
brotliDecompress: createConvenienceMethod(BrotliDecompress, false),
878971
brotliDecompressSync: createConvenienceMethod(BrotliDecompress, true),
972+
zstdCompress: createConvenienceMethod(ZstdCompress, false),
973+
zstdCompressSync: createConvenienceMethod(ZstdCompress, true),
974+
zstdDecompress: createConvenienceMethod(ZstdDecompress, false),
975+
zstdDecompressSync: createConvenienceMethod(ZstdDecompress, true),
879976
};
880977

881978
ObjectDefineProperties(module.exports, {
@@ -888,6 +985,8 @@ ObjectDefineProperties(module.exports, {
888985
createUnzip: createProperty(Unzip),
889986
createBrotliCompress: createProperty(BrotliCompress),
890987
createBrotliDecompress: createProperty(BrotliDecompress),
988+
createZstdCompress: createProperty(ZstdCompress),
989+
createZstdDecompress: createProperty(ZstdDecompress),
891990
constants: {
892991
__proto__: null,
893992
configurable: false,

‎src/node_zlib.cc

+382-2
Large diffs are not rendered by default.

‎test/fixtures/person.jpg.zst

44.3 KB
Binary file not shown.

‎test/parallel/test-zlib-convenience-methods.js

+2
Original file line numberDiff line numberDiff line change
@@ -54,6 +54,8 @@ for (const [type, expect] of [
5454
['deflateRaw', 'inflateRaw', 'DeflateRaw', 'InflateRaw'],
5555
['brotliCompress', 'brotliDecompress',
5656
'BrotliCompress', 'BrotliDecompress'],
57+
['zstdCompress', 'zstdDecompress',
58+
'ZstdCompress', 'ZstdDecompress'],
5759
]) {
5860
zlib[method[0]](expect, opts, common.mustCall((err, result) => {
5961
zlib[method[1]](result, opts, common.mustCall((err, result) => {

‎test/parallel/test-zlib-empty-buffer.js

+2
Original file line numberDiff line numberDiff line change
@@ -11,10 +11,12 @@ const emptyBuffer = Buffer.alloc(0);
1111
[ zlib.deflateSync, zlib.inflateSync, 'deflate sync' ],
1212
[ zlib.gzipSync, zlib.gunzipSync, 'gzip sync' ],
1313
[ zlib.brotliCompressSync, zlib.brotliDecompressSync, 'br sync' ],
14+
[ zlib.zstdCompressSync, zlib.zstdDecompressSync, 'zstd sync' ],
1415
[ promisify(zlib.deflateRaw), promisify(zlib.inflateRaw), 'raw' ],
1516
[ promisify(zlib.deflate), promisify(zlib.inflate), 'deflate' ],
1617
[ promisify(zlib.gzip), promisify(zlib.gunzip), 'gzip' ],
1718
[ promisify(zlib.brotliCompress), promisify(zlib.brotliDecompress), 'br' ],
19+
[ promisify(zlib.zstdCompress), promisify(zlib.zstdDecompress), 'zstd' ],
1820
]) {
1921
const compressed = await compress(emptyBuffer);
2022
const decompressed = await decompress(compressed);

‎test/parallel/test-zlib-invalid-input.js

+1
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,7 @@ const unzips = [
4040
zlib.Inflate(),
4141
zlib.InflateRaw(),
4242
zlib.BrotliDecompress(),
43+
zlib.ZstdDecompress(),
4344
];
4445

4546
nonStringInputs.forEach(common.mustCall((input) => {

‎test/parallel/test-zlib-random-byte-pipes.js

+1
Original file line numberDiff line numberDiff line change
@@ -144,6 +144,7 @@ class HashStream extends Stream {
144144
for (const [ createCompress, createDecompress ] of [
145145
[ zlib.createGzip, zlib.createGunzip ],
146146
[ zlib.createBrotliCompress, zlib.createBrotliDecompress ],
147+
[ zlib.createZstdCompress, zlib.createZstdDecompress ],
147148
]) {
148149
const inp = new RandomReadStream({ total: 1024, block: 256, jitter: 16 });
149150
const out = new HashStream();

‎test/parallel/test-zlib-write-after-flush.js

+4-3
Original file line numberDiff line numberDiff line change
@@ -28,9 +28,10 @@ const zlib = require('node:zlib');
2828
const { test } = require('node:test');
2929

3030
test('zlib should accept writing after flush', async () => {
31-
for (const [createCompress, createDecompress] of [
32-
[zlib.createGzip, zlib.createGunzip],
33-
[zlib.createBrotliCompress, zlib.createBrotliDecompress],
31+
for (const [ createCompress, createDecompress ] of [
32+
[ zlib.createGzip, zlib.createGunzip ],
33+
[ zlib.createBrotliCompress, zlib.createBrotliDecompress ],
34+
[ zlib.createZstdCompress, zlib.createZstdDecompress ],
3435
]) {
3536
const { promise, resolve, reject } = Promise.withResolvers();
3637
const gzip = createCompress();

‎test/parallel/test-zlib-zero-byte.js

+7-2
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,13 @@ const zlib = require('node:zlib');
2828
const { test } = require('node:test');
2929

3030
test('zlib should properly handle zero byte input', async () => {
31-
for (const Compressor of [zlib.Gzip, zlib.BrotliCompress]) {
31+
const compressors = [
32+
[zlib.Gzip, 20],
33+
[zlib.BrotliCompress, 1],
34+
[zlib.ZstdCompress, 9],
35+
];
36+
37+
for (const [Compressor, expected] of compressors) {
3238
const { promise, resolve, reject } = Promise.withResolvers();
3339
const gz = Compressor();
3440
const emptyBuffer = Buffer.alloc(0);
@@ -38,7 +44,6 @@ test('zlib should properly handle zero byte input', async () => {
3844
});
3945
gz.on('error', reject);
4046
gz.on('end', function() {
41-
const expected = Compressor === zlib.Gzip ? 20 : 1;
4247
assert.strictEqual(received, expected,
4348
`${received}, ${expected}, ${Compressor.name}`);
4449
resolve();

‎test/parallel/test-zlib-zstd-flush.js

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
'use strict';
2+
require('../common');
3+
const assert = require('assert');
4+
const zlib = require('zlib');
5+
const fixtures = require('../common/fixtures');
6+
7+
const file = fixtures.readSync('person.jpg');
8+
const chunkSize = 16;
9+
const compress = new zlib.ZstdCompress();
10+
11+
const chunk = file.slice(0, chunkSize);
12+
const expectedFull = Buffer.from('KLUv/QBYgAAA/9j/4AAQSkZJRgABAQEASA==', 'base64');
13+
let actualFull;
14+
15+
compress.write(chunk, function() {
16+
compress.flush(function() {
17+
const bufs = [];
18+
let buf;
19+
while ((buf = compress.read()) !== null)
20+
bufs.push(buf);
21+
actualFull = Buffer.concat(bufs);
22+
});
23+
});
24+
25+
process.once('exit', function() {
26+
assert.deepStrictEqual(actualFull.toString('base64'), expectedFull.toString('base64'));
27+
assert.deepStrictEqual(actualFull, expectedFull);
28+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
'use strict';
2+
// Test compressing and uncompressing a string with zstd
3+
4+
const common = require('../common');
5+
const assert = require('assert');
6+
const zlib = require('zlib');
7+
8+
const inputString = 'ΩΩLorem ipsum dolor sit amet, consectetur adipiscing eli' +
9+
't. Morbi faucibus, purus at gravida dictum, libero arcu ' +
10+
'convallis lacus, in commodo libero metus eu nisi. Nullam' +
11+
' commodo, neque nec porta placerat, nisi est fermentum a' +
12+
'ugue, vitae gravida tellus sapien sit amet tellus. Aenea' +
13+
'n non diam orci. Proin quis elit turpis. Suspendisse non' +
14+
' diam ipsum. Suspendisse nec ullamcorper odio. Vestibulu' +
15+
'm arcu mi, sodales non suscipit id, ultrices ut massa. S' +
16+
'ed ac sem sit amet arcu malesuada fermentum. Nunc sed. ';
17+
const compressedString = 'KLUv/QRYRQkA9tc9H6AlhTb/z/7/gbTI3kaWLKnbCtkZu/hXm0j' +
18+
'FpNz/VQM2ADMANQBHTuQOpIYzfVv7XGwXrpoIfgXNAB98xW4wV3' +
19+
'vnCF2bjcvWZF2wIZ1vr1mSHHvPHU0TgMGBwUFrF0xqReWcWPO8z' +
20+
'Ny6wMwFUilN+Lg987Zvs2GSRMy6uYvtovK9Uuhgst6l9FQrXLnA' +
21+
'5gpZL7PdI8bO9sDH3tHm73XBzaUK+LjSPNKRmzQ3ZMYEPozdof1' +
22+
'2KcZGfIcLa0PTsdkYqhGcAx/E9mWa8EGEeq0Qou2LTmzgg3YJz/' +
23+
'21OuXSF+TOd662d60Qyb04xC5dOF4b8JFH8mpHAxAAELu3tg1oa' +
24+
'bBEIWaRHdE0l/+0RdEWWIVMAku8TgbiX/4bU+OpLo4UuY1FKDR8' +
25+
'RgBc';
26+
27+
zlib.zstdCompress(inputString, common.mustCall((err, buffer) => {
28+
assert(inputString.length > buffer.length);
29+
30+
zlib.zstdDecompress(buffer, common.mustCall((err, buffer) => {
31+
assert.strictEqual(buffer.toString(), inputString);
32+
}));
33+
}));
34+
35+
const buffer = Buffer.from(compressedString, 'base64');
36+
zlib.zstdDecompress(buffer, common.mustCall((err, buffer) => {
37+
assert.strictEqual(buffer.toString(), inputString);
38+
}));
+34
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,34 @@
1+
'use strict';
2+
// Test unzipping a file that was created with a non-node zstd lib,
3+
// piped in as fast as possible.
4+
//
5+
// The compressed fixture was created using the reference CLI:
6+
// $ zstd -19 test/fixtures/person.jpg -o test/fixtures/person.jpg.zst
7+
8+
const common = require('../common');
9+
const assert = require('assert');
10+
const zlib = require('zlib');
11+
const fixtures = require('../common/fixtures');
12+
13+
const tmpdir = require('../common/tmpdir');
14+
tmpdir.refresh();
15+
16+
const decompress = new zlib.ZstdDecompress();
17+
18+
const fs = require('fs');
19+
20+
const fixture = fixtures.path('person.jpg.zst');
21+
const unzippedFixture = fixtures.path('person.jpg');
22+
const outputFile = tmpdir.resolve('person.jpg');
23+
const expect = fs.readFileSync(unzippedFixture);
24+
const inp = fs.createReadStream(fixture);
25+
const out = fs.createWriteStream(outputFile);
26+
27+
inp.pipe(decompress).pipe(out);
28+
out.on('close', common.mustCall(() => {
29+
const actual = fs.readFileSync(outputFile);
30+
assert.strictEqual(actual.length, expect.length);
31+
for (let i = 0, l = actual.length; i < l; i++) {
32+
assert.strictEqual(actual[i], expect[i], `byte[${i}]`);
33+
}
34+
}));
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
'use strict';
2+
require('../common');
3+
4+
// This test ensures that zlib throws a RangeError if the final buffer needs to
5+
// be larger than kMaxLength and concatenation fails.
6+
// https://github.com/nodejs/node/pull/1811
7+
8+
const assert = require('assert');
9+
10+
// Change kMaxLength for zlib to trigger the error without having to allocate
11+
// large Buffers.
12+
const buffer = require('buffer');
13+
const oldkMaxLength = buffer.kMaxLength;
14+
buffer.kMaxLength = 64;
15+
const zlib = require('zlib');
16+
buffer.kMaxLength = oldkMaxLength;
17+
18+
// "a".repeat(128), compressed using zstd.
19+
const encoded = Buffer.from('KLUv/SCARQAAEGFhAQA7BVg=', 'base64');
20+
21+
// Async
22+
zlib.zstdDecompress(encoded, function(err) {
23+
assert.ok(err instanceof RangeError);
24+
});
25+
26+
// Sync
27+
assert.throws(function() {
28+
zlib.zstdDecompressSync(encoded);
29+
}, RangeError);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
'use strict';
2+
const common = require('../common');
3+
const assert = require('assert');
4+
const zlib = require('zlib');
5+
6+
function compressWithPledgedSrcSize({ pledgedSrcSize, actualSrcSize }) {
7+
return new Promise((resolve, reject) => {
8+
const compressor = zlib.createZstdCompress({ pledgedSrcSize });
9+
compressor.on('error', (e) => {
10+
reject(e);
11+
});
12+
compressor.on('end', resolve);
13+
compressor.write('x'.repeat(actualSrcSize), () => {
14+
compressor.end();
15+
compressor.resume();
16+
});
17+
}).then(() => {
18+
// Compression should only succeed if sizes match
19+
assert.strictEqual(pledgedSrcSize, actualSrcSize);
20+
}, (error) => {
21+
assert.strictEqual(error.code, 'ZSTD_error_srcSize_wrong');
22+
// Size error should only happen when sizes do not match
23+
assert.notStrictEqual(pledgedSrcSize, actualSrcSize);
24+
}).then(common.mustCall());
25+
}
26+
27+
compressWithPledgedSrcSize({ pledgedSrcSize: 0, actualSrcSize: 0 });
28+
29+
compressWithPledgedSrcSize({ pledgedSrcSize: 0, actualSrcSize: 42 });
30+
31+
compressWithPledgedSrcSize({ pledgedSrcSize: 13, actualSrcSize: 42 });
32+
33+
compressWithPledgedSrcSize({ pledgedSrcSize: 42, actualSrcSize: 0 });
34+
35+
compressWithPledgedSrcSize({ pledgedSrcSize: 42, actualSrcSize: 13 });
36+
37+
compressWithPledgedSrcSize({ pledgedSrcSize: 42, actualSrcSize: 42 });

‎test/parallel/test-zlib-zstd.js

+134
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,134 @@
1+
'use strict';
2+
require('../common');
3+
const fixtures = require('../common/fixtures');
4+
const assert = require('assert');
5+
const zlib = require('zlib');
6+
7+
// Test some zstd-specific properties of the zstd streams that can not
8+
// be easily covered through expanding zlib-only tests.
9+
10+
const sampleBuffer = fixtures.readSync('/pss-vectors.json');
11+
12+
{
13+
// Test setting the quality parameter at stream creation:
14+
const sizes = [];
15+
for (let quality = 1;
16+
quality <= 22;
17+
quality++) {
18+
const encoded = zlib.zstdCompressSync(sampleBuffer, {
19+
params: {
20+
[zlib.constants.ZSTD_c_compressionLevel]: quality
21+
}
22+
});
23+
sizes.push(encoded.length);
24+
}
25+
26+
// Increasing quality should roughly correspond to decreasing compressed size:
27+
for (let i = 0; i < sizes.length - 1; i++) {
28+
assert(sizes[i + 1] <= sizes[i] * 1.05, sizes); // 5 % margin of error.
29+
}
30+
assert(sizes[0] > sizes[sizes.length - 1], sizes);
31+
}
32+
33+
{
34+
// Test that setting out-of-bounds option values or keys fails.
35+
assert.throws(() => {
36+
zlib.createZstdCompress({
37+
params: {
38+
10000: 0
39+
}
40+
});
41+
}, {
42+
code: 'ERR_ZSTD_INVALID_PARAM',
43+
name: 'RangeError',
44+
message: '10000 is not a valid zstd parameter'
45+
});
46+
47+
// Test that accidentally using duplicate keys fails.
48+
assert.throws(() => {
49+
zlib.createZstdCompress({
50+
params: {
51+
'0': 0,
52+
'00': 0
53+
}
54+
});
55+
}, {
56+
code: 'ERR_ZSTD_INVALID_PARAM',
57+
name: 'RangeError',
58+
message: '00 is not a valid zstd parameter'
59+
});
60+
61+
assert.throws(() => {
62+
zlib.createZstdCompress({
63+
params: {
64+
// This param must be a valid ZSTD_strategy value.
65+
[zlib.constants.ZSTD_c_strategy]: 130
66+
}
67+
});
68+
}, {
69+
code: 'ERR_ZLIB_INITIALIZATION_FAILED',
70+
name: 'Error',
71+
message: 'Setting parameter failed'
72+
});
73+
74+
// Test that setting out-of-bounds option values or keys fails.
75+
assert.throws(() => {
76+
zlib.createZstdDecompress({
77+
params: {
78+
10000: 0
79+
}
80+
});
81+
}, {
82+
code: 'ERR_ZSTD_INVALID_PARAM',
83+
name: 'RangeError',
84+
message: '10000 is not a valid zstd parameter'
85+
});
86+
87+
// Test that accidentally using duplicate keys fails.
88+
assert.throws(() => {
89+
zlib.createZstdDecompress({
90+
params: {
91+
'0': 0,
92+
'00': 0
93+
}
94+
});
95+
}, {
96+
code: 'ERR_ZSTD_INVALID_PARAM',
97+
name: 'RangeError',
98+
message: '00 is not a valid zstd parameter'
99+
});
100+
101+
assert.throws(() => {
102+
zlib.createZstdDecompress({
103+
params: {
104+
// This param must be >= 10 (ZSTD_WINDOWLOG_ABSOLUTEMIN).
105+
[zlib.constants.ZSTD_d_windowLogMax]: 1
106+
}
107+
});
108+
}, {
109+
code: 'ERR_ZLIB_INITIALIZATION_FAILED',
110+
name: 'Error',
111+
message: 'Setting parameter failed'
112+
});
113+
}
114+
115+
{
116+
// Test options.flush range
117+
assert.throws(() => {
118+
zlib.zstdCompressSync('', { flush: zlib.constants.Z_FINISH });
119+
}, {
120+
code: 'ERR_OUT_OF_RANGE',
121+
name: 'RangeError',
122+
message: 'The value of "options.flush" is out of range. It must be >= 0 ' +
123+
'and <= 2. Received 4',
124+
});
125+
126+
assert.throws(() => {
127+
zlib.zstdCompressSync('', { finishFlush: zlib.constants.Z_FINISH });
128+
}, {
129+
code: 'ERR_OUT_OF_RANGE',
130+
name: 'RangeError',
131+
message: 'The value of "options.finishFlush" is out of range. It must be ' +
132+
'>= 0 and <= 2. Received 4',
133+
});
134+
}

‎test/parallel/test-zlib.js

+1
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,7 @@ let zlibPairs = [
4242
[zlib.Gzip, zlib.Unzip],
4343
[zlib.DeflateRaw, zlib.InflateRaw],
4444
[zlib.BrotliCompress, zlib.BrotliDecompress],
45+
[zlib.ZstdCompress, zlib.ZstdDecompress],
4546
];
4647

4748
// How fast to trickle through the slowstream

‎tools/doc/type-parser.mjs

+1
Original file line numberDiff line numberDiff line change
@@ -252,6 +252,7 @@ const customTypesMap = {
252252
'X509Certificate': 'crypto.html#class-x509certificate',
253253

254254
'zlib options': 'zlib.html#class-options',
255+
'zstd options': 'zlib.html#class-zstdoptions',
255256

256257
'ReadableStream':
257258
'webstreams.html#class-readablestream',

0 commit comments

Comments
 (0)
Please sign in to comment.