// 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 };