0
0
mirror of https://github.com/nodejs/node.git synced 2024-12-01 16:10:02 +01:00
nodejs/test/parallel/test-stream-readable-destroy.js
Mathias Buus 5e3f51648e stream: updated streams error handling
This improves error handling for streams in a few ways.

1. It ensures that no user defined methods (_read, _write, ...) are run
after .destroy has been called.
2. It introduces an explicit error to tell the user if they are write to
write, etc to the stream after it has been destroyed.
3. It makes streams always emit close as the last thing after they have
been destroyed
4. Changes the default _destroy to not gracefully end streams.

It also updates net, http2, zlib and fs to the new error handling.

PR-URL: https://github.com/nodejs/node/pull/18438
Reviewed-By: Matteo Collina <matteo.collina@gmail.com>
Reviewed-By: James M Snell <jasnell@gmail.com>
Reviewed-By: Ruben Bridgewater <ruben@bridgewater.de>
Reviewed-By: Anna Henningsen <anna@addaleax.net>
2018-03-06 13:31:56 +01:00

192 lines
3.6 KiB
JavaScript

'use strict';
const common = require('../common');
const { Readable } = require('stream');
const assert = require('assert');
const { inherits } = require('util');
{
const read = new Readable({
read() {}
});
read.resume();
read.on('close', common.mustCall());
read.destroy();
assert.strictEqual(read.destroyed, true);
}
{
const read = new Readable({
read() {}
});
read.resume();
const expected = new Error('kaboom');
read.on('end', common.mustNotCall('no end event'));
read.on('close', common.mustCall());
read.on('error', common.mustCall((err) => {
assert.strictEqual(err, expected);
}));
read.destroy(expected);
assert.strictEqual(read.destroyed, true);
}
{
const read = new Readable({
read() {}
});
read._destroy = common.mustCall(function(err, cb) {
assert.strictEqual(err, expected);
cb(err);
});
const expected = new Error('kaboom');
read.on('end', common.mustNotCall('no end event'));
read.on('close', common.mustCall());
read.on('error', common.mustCall((err) => {
assert.strictEqual(err, expected);
}));
read.destroy(expected);
assert.strictEqual(read.destroyed, true);
}
{
const read = new Readable({
read() {},
destroy: common.mustCall(function(err, cb) {
assert.strictEqual(err, expected);
cb();
})
});
const expected = new Error('kaboom');
read.on('end', common.mustNotCall('no end event'));
// error is swallowed by the custom _destroy
read.on('error', common.mustNotCall('no error event'));
read.on('close', common.mustCall());
read.destroy(expected);
assert.strictEqual(read.destroyed, true);
}
{
const read = new Readable({
read() {}
});
read._destroy = common.mustCall(function(err, cb) {
assert.strictEqual(err, null);
cb();
});
read.destroy();
assert.strictEqual(read.destroyed, true);
}
{
const read = new Readable({
read() {}
});
read.resume();
read._destroy = common.mustCall(function(err, cb) {
assert.strictEqual(err, null);
process.nextTick(() => {
this.push(null);
cb();
});
});
const fail = common.mustNotCall('no end event');
read.on('end', fail);
read.on('close', common.mustCall());
read.destroy();
read.removeListener('end', fail);
read.on('end', common.mustCall());
assert.strictEqual(read.destroyed, true);
}
{
const read = new Readable({
read() {}
});
const expected = new Error('kaboom');
read._destroy = common.mustCall(function(err, cb) {
assert.strictEqual(err, null);
cb(expected);
});
read.on('end', common.mustNotCall('no end event'));
read.on('error', common.mustCall((err) => {
assert.strictEqual(err, expected);
}));
read.destroy();
assert.strictEqual(read.destroyed, true);
}
{
const read = new Readable({
read() {}
});
read.resume();
read.destroyed = true;
assert.strictEqual(read.destroyed, true);
// the internal destroy() mechanism should not be triggered
read.on('end', common.mustNotCall());
read.destroy();
}
{
function MyReadable() {
assert.strictEqual(this.destroyed, false);
this.destroyed = false;
Readable.call(this);
}
inherits(MyReadable, Readable);
new MyReadable();
}
{
// destroy and destroy callback
const read = new Readable({
read() {}
});
read.resume();
const expected = new Error('kaboom');
read.on('close', common.mustCall());
read.destroy(expected, common.mustCall(function(err) {
assert.strictEqual(expected, err);
}));
}
{
const read = new Readable({
read() {}
});
read.destroy();
read.push('hi');
read.on('data', common.mustNotCall());
}