mirror of
https://github.com/denoland/deno.git
synced 2024-12-29 10:39:10 -05:00
189 lines
4.1 KiB
JavaScript
189 lines
4.1 KiB
JavaScript
// Copyright 2018-2024 the Deno authors. All rights reserved. MIT license.
|
|
// Copyright Joyent and Node contributors. All rights reserved. MIT license.
|
|
// deno-lint-ignore-file
|
|
|
|
import { aggregateTwoErrors } from "ext:deno_node/internal/errors.ts";
|
|
import * as process from "ext:deno_node/_process/process.ts";
|
|
|
|
const kDestroy = Symbol("kDestroy");
|
|
|
|
function checkError(err, w, r) {
|
|
if (err) {
|
|
// Avoid V8 leak, https://github.com/nodejs/node/pull/34103#issuecomment-652002364
|
|
err.stack; // eslint-disable-line no-unused-expressions
|
|
|
|
if (w && !w.errored) {
|
|
w.errored = err;
|
|
}
|
|
if (r && !r.errored) {
|
|
r.errored = err;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Backwards compat. cb() is undocumented and unused in core but
|
|
// unfortunately might be used by modules.
|
|
function destroy(err, cb) {
|
|
const r = this._readableState;
|
|
const w = this._writableState;
|
|
// With duplex streams we use the writable side for state.
|
|
const s = w || r;
|
|
|
|
if ((w && w.destroyed) || (r && r.destroyed)) {
|
|
if (typeof cb === "function") {
|
|
cb();
|
|
}
|
|
|
|
return this;
|
|
}
|
|
|
|
// We set destroyed to true before firing error callbacks in order
|
|
// to make it re-entrance safe in case destroy() is called within callbacks
|
|
checkError(err, w, r);
|
|
|
|
if (w) {
|
|
w.destroyed = true;
|
|
}
|
|
if (r) {
|
|
r.destroyed = true;
|
|
}
|
|
|
|
// If still constructing then defer calling _destroy.
|
|
if (!s.constructed) {
|
|
this.once(kDestroy, function (er) {
|
|
_destroy(this, aggregateTwoErrors(er, err), cb);
|
|
});
|
|
} else {
|
|
_destroy(this, err, cb);
|
|
}
|
|
|
|
return this;
|
|
}
|
|
|
|
function _destroy(self, err, cb) {
|
|
let called = false;
|
|
|
|
function onDestroy(err) {
|
|
if (called) {
|
|
return;
|
|
}
|
|
called = true;
|
|
|
|
const r = self._readableState;
|
|
const w = self._writableState;
|
|
|
|
checkError(err, w, r);
|
|
|
|
if (w) {
|
|
w.closed = true;
|
|
}
|
|
if (r) {
|
|
r.closed = true;
|
|
}
|
|
|
|
if (typeof cb === "function") {
|
|
cb(err);
|
|
}
|
|
|
|
if (err) {
|
|
process.nextTick(emitErrorCloseNT, self, err);
|
|
} else {
|
|
process.nextTick(emitCloseNT, self);
|
|
}
|
|
}
|
|
try {
|
|
const result = self._destroy(err || null, onDestroy);
|
|
if (result != null) {
|
|
const then = result.then;
|
|
if (typeof then === "function") {
|
|
then.call(
|
|
result,
|
|
function () {
|
|
process.nextTick(onDestroy, null);
|
|
},
|
|
function (err) {
|
|
process.nextTick(onDestroy, err);
|
|
},
|
|
);
|
|
}
|
|
}
|
|
} catch (err) {
|
|
onDestroy(err);
|
|
}
|
|
}
|
|
|
|
function emitErrorCloseNT(self, err) {
|
|
emitErrorNT(self, err);
|
|
emitCloseNT(self);
|
|
}
|
|
|
|
function emitCloseNT(self) {
|
|
const r = self._readableState;
|
|
const w = self._writableState;
|
|
|
|
if (w) {
|
|
w.closeEmitted = true;
|
|
}
|
|
if (r) {
|
|
r.closeEmitted = true;
|
|
}
|
|
|
|
if ((w && w.emitClose) || (r && r.emitClose)) {
|
|
self.emit("close");
|
|
}
|
|
}
|
|
|
|
function emitErrorNT(self, err) {
|
|
const r = self._readableState;
|
|
const w = self._writableState;
|
|
|
|
if ((w && w.errorEmitted) || (r && r.errorEmitted)) {
|
|
return;
|
|
}
|
|
|
|
if (w) {
|
|
w.errorEmitted = true;
|
|
}
|
|
if (r) {
|
|
r.errorEmitted = true;
|
|
}
|
|
|
|
self.emit("error", err);
|
|
}
|
|
|
|
function errorOrDestroy(stream, err, sync) {
|
|
// We have tests that rely on errors being emitted
|
|
// in the same tick, so changing this is semver major.
|
|
// For now when you opt-in to autoDestroy we allow
|
|
// the error to be emitted nextTick. In a future
|
|
// semver major update we should change the default to this.
|
|
|
|
const r = stream._readableState;
|
|
const w = stream._writableState;
|
|
|
|
if ((w && w.destroyed) || (r && r.destroyed)) {
|
|
return this;
|
|
}
|
|
|
|
if ((r && r.autoDestroy) || (w && w.autoDestroy)) {
|
|
stream.destroy(err);
|
|
} else if (err) {
|
|
// Avoid V8 leak, https://github.com/nodejs/node/pull/34103#issuecomment-652002364
|
|
err.stack; // eslint-disable-line no-unused-expressions
|
|
|
|
if (w && !w.errored) {
|
|
w.errored = err;
|
|
}
|
|
if (r && !r.errored) {
|
|
r.errored = err;
|
|
}
|
|
if (sync) {
|
|
process.nextTick(emitErrorNT, stream, err);
|
|
} else {
|
|
emitErrorNT(stream, err);
|
|
}
|
|
}
|
|
}
|
|
|
|
export default { destroy, errorOrDestroy };
|
|
export { destroy, errorOrDestroy };
|