// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license. // This code closely follows the WHATWG Stream Specification // See: https://streams.spec.whatwg.org/ // // There are some parts that are not fully implemented, and there are some // comments which point to steps of the specification that are not implemented. ((window) => { /* eslint-disable @typescript-eslint/no-explicit-any,require-await */ const { cloneValue, setFunctionName } = window.__bootstrap.webUtil; const { assert, AssertionError } = window.__bootstrap.util; const { customInspect, inspect } = window.__bootstrap.console; const sym = { abortAlgorithm: Symbol("abortAlgorithm"), abortSteps: Symbol("abortSteps"), asyncIteratorReader: Symbol("asyncIteratorReader"), autoAllocateChunkSize: Symbol("autoAllocateChunkSize"), backpressure: Symbol("backpressure"), backpressureChangePromise: Symbol("backpressureChangePromise"), byobRequest: Symbol("byobRequest"), cancelAlgorithm: Symbol("cancelAlgorithm"), cancelSteps: Symbol("cancelSteps"), closeAlgorithm: Symbol("closeAlgorithm"), closedPromise: Symbol("closedPromise"), closeRequest: Symbol("closeRequest"), closeRequested: Symbol("closeRequested"), controlledReadableByteStream: Symbol( "controlledReadableByteStream", ), controlledReadableStream: Symbol("controlledReadableStream"), controlledTransformStream: Symbol("controlledTransformStream"), controlledWritableStream: Symbol("controlledWritableStream"), disturbed: Symbol("disturbed"), errorSteps: Symbol("errorSteps"), flushAlgorithm: Symbol("flushAlgorithm"), forAuthorCode: Symbol("forAuthorCode"), inFlightWriteRequest: Symbol("inFlightWriteRequest"), inFlightCloseRequest: Symbol("inFlightCloseRequest"), isFakeDetached: Symbol("isFakeDetached"), ownerReadableStream: Symbol("ownerReadableStream"), ownerWritableStream: Symbol("ownerWritableStream"), pendingAbortRequest: Symbol("pendingAbortRequest"), preventCancel: Symbol("preventCancel"), pullAgain: Symbol("pullAgain"), pullAlgorithm: Symbol("pullAlgorithm"), pulling: Symbol("pulling"), pullSteps: Symbol("pullSteps"), queue: Symbol("queue"), queueTotalSize: Symbol("queueTotalSize"), readable: Symbol("readable"), readableStreamController: Symbol("readableStreamController"), reader: Symbol("reader"), readRequests: Symbol("readRequests"), readyPromise: Symbol("readyPromise"), started: Symbol("started"), state: Symbol("state"), storedError: Symbol("storedError"), strategyHWM: Symbol("strategyHWM"), strategySizeAlgorithm: Symbol("strategySizeAlgorithm"), transformAlgorithm: Symbol("transformAlgorithm"), transformStreamController: Symbol("transformStreamController"), writableStreamController: Symbol("writableStreamController"), writeAlgorithm: Symbol("writeAlgorithm"), writable: Symbol("writable"), writer: Symbol("writer"), writeRequests: Symbol("writeRequests"), }; class ReadableByteStreamController { constructor() { throw new TypeError( "ReadableByteStreamController's constructor cannot be called.", ); } get byobRequest() { return undefined; } get desiredSize() { if (!isReadableByteStreamController(this)) { throw new TypeError("Invalid ReadableByteStreamController."); } return readableByteStreamControllerGetDesiredSize(this); } close() { if (!isReadableByteStreamController(this)) { throw new TypeError("Invalid ReadableByteStreamController."); } if (this[sym.closeRequested]) { throw new TypeError("Closed already requested."); } if (this[sym.controlledReadableByteStream][sym.state] !== "readable") { throw new TypeError( "ReadableByteStreamController's stream is not in a readable state.", ); } readableByteStreamControllerClose(this); } enqueue(chunk) { if (!isReadableByteStreamController(this)) { throw new TypeError("Invalid ReadableByteStreamController."); } if (this[sym.closeRequested]) { throw new TypeError("Closed already requested."); } if (this[sym.controlledReadableByteStream][sym.state] !== "readable") { throw new TypeError( "ReadableByteStreamController's stream is not in a readable state.", ); } if (!ArrayBuffer.isView(chunk)) { throw new TypeError( "You can only enqueue array buffer views when using a ReadableByteStreamController", ); } if (isDetachedBuffer(chunk.buffer)) { throw new TypeError( "Cannot enqueue a view onto a detached ArrayBuffer", ); } readableByteStreamControllerEnqueue(this, chunk); } error(error) { if (!isReadableByteStreamController(this)) { throw new TypeError("Invalid ReadableByteStreamController."); } readableByteStreamControllerError(this, error); } [sym.cancelSteps](reason) { // 3.11.5.1.1 If this.[[pendingPullIntos]] is not empty, resetQueue(this); const result = this[sym.cancelAlgorithm](reason); readableByteStreamControllerClearAlgorithms(this); return result; } [sym.pullSteps]() { const stream = this[sym.controlledReadableByteStream]; assert(readableStreamHasDefaultReader(stream)); if (this[sym.queueTotalSize] > 0) { assert(readableStreamGetNumReadRequests(stream) === 0); const entry = this[sym.queue].shift(); assert(entry); this[sym.queueTotalSize] -= entry.size; readableByteStreamControllerHandleQueueDrain(this); const view = new Uint8Array(entry.value, entry.offset, entry.size); return Promise.resolve( readableStreamCreateReadResult( view, false, stream[sym.reader][sym.forAuthorCode], ), ); } // 3.11.5.2.5 If autoAllocateChunkSize is not undefined, const promise = readableStreamAddReadRequest(stream); readableByteStreamControllerCallPullIfNeeded(this); return promise; } [customInspect]() { return `${this.constructor.name} { byobRequest: ${ String(this.byobRequest) }, desiredSize: ${String(this.desiredSize)} }`; } } class ReadableStreamDefaultController { constructor() { throw new TypeError( "ReadableStreamDefaultController's constructor cannot be called.", ); } get desiredSize() { if (!isReadableStreamDefaultController(this)) { throw new TypeError("Invalid ReadableStreamDefaultController."); } return readableStreamDefaultControllerGetDesiredSize(this); } close() { if (!isReadableStreamDefaultController(this)) { throw new TypeError("Invalid ReadableStreamDefaultController."); } if (!readableStreamDefaultControllerCanCloseOrEnqueue(this)) { throw new TypeError( "ReadableStreamDefaultController cannot close or enqueue.", ); } readableStreamDefaultControllerClose(this); } enqueue(chunk) { if (!isReadableStreamDefaultController(this)) { throw new TypeError("Invalid ReadableStreamDefaultController."); } if (!readableStreamDefaultControllerCanCloseOrEnqueue(this)) { throw new TypeError("ReadableSteamController cannot enqueue."); } return readableStreamDefaultControllerEnqueue(this, chunk); } error(error) { if (!isReadableStreamDefaultController(this)) { throw new TypeError("Invalid ReadableStreamDefaultController."); } readableStreamDefaultControllerError(this, error); } [sym.cancelSteps](reason) { resetQueue(this); const result = this[sym.cancelAlgorithm](reason); readableStreamDefaultControllerClearAlgorithms(this); return result; } [sym.pullSteps]() { const stream = this[sym.controlledReadableStream]; if (this[sym.queue].length) { const chunk = dequeueValue(this); if (this[sym.closeRequested] && this[sym.queue].length === 0) { readableStreamDefaultControllerClearAlgorithms(this); readableStreamClose(stream); } else { readableStreamDefaultControllerCallPullIfNeeded(this); } return Promise.resolve( readableStreamCreateReadResult( chunk, false, stream[sym.reader][sym.forAuthorCode], ), ); } const pendingPromise = readableStreamAddReadRequest(stream); readableStreamDefaultControllerCallPullIfNeeded(this); return pendingPromise; } [customInspect]() { return `${this.constructor.name} { desiredSize: ${ String(this.desiredSize) } }`; } } class ReadableStreamDefaultReader { constructor(stream) { if (!isReadableStream(stream)) { throw new TypeError("stream is not a ReadableStream."); } if (isReadableStreamLocked(stream)) { throw new TypeError("stream is locked."); } readableStreamReaderGenericInitialize(this, stream); this[sym.readRequests] = []; } get closed() { if (!isReadableStreamDefaultReader(this)) { return Promise.reject( new TypeError("Invalid ReadableStreamDefaultReader."), ); } return ( this[sym.closedPromise].promise ?? Promise.reject(new TypeError("Invalid reader.")) ); } cancel(reason) { if (!isReadableStreamDefaultReader(this)) { return Promise.reject( new TypeError("Invalid ReadableStreamDefaultReader."), ); } if (!this[sym.ownerReadableStream]) { return Promise.reject(new TypeError("Invalid reader.")); } return readableStreamReaderGenericCancel(this, reason); } read() { if (!isReadableStreamDefaultReader(this)) { return Promise.reject( new TypeError("Invalid ReadableStreamDefaultReader."), ); } if (!this[sym.ownerReadableStream]) { return Promise.reject(new TypeError("Invalid reader.")); } return readableStreamDefaultReaderRead(this); } releaseLock() { if (!isReadableStreamDefaultReader(this)) { throw new TypeError("Invalid ReadableStreamDefaultReader."); } if (this[sym.ownerReadableStream] === undefined) { return; } if (this[sym.readRequests].length) { throw new TypeError("Cannot release lock with pending read requests."); } readableStreamReaderGenericRelease(this); } [customInspect]() { return `${this.constructor.name} { closed: Promise }`; } } const AsyncIteratorPrototype = Object .getPrototypeOf(Object.getPrototypeOf(async function* () {}).prototype); const ReadableStreamAsyncIteratorPrototype = Object.setPrototypeOf({ next() { if (!isReadableStreamAsyncIterator(this)) { return Promise.reject( new TypeError("invalid ReadableStreamAsyncIterator."), ); } const reader = this[sym.asyncIteratorReader]; if (!reader[sym.ownerReadableStream]) { return Promise.reject( new TypeError("reader owner ReadableStream is undefined."), ); } return readableStreamDefaultReaderRead(reader).then((result) => { assert(typeof result === "object"); const { done } = result; assert(typeof done === "boolean"); if (done) { readableStreamReaderGenericRelease(reader); } const { value } = result; return readableStreamCreateReadResult(value, done, true); }); }, return( value, ) { if (!isReadableStreamAsyncIterator(this)) { return Promise.reject( new TypeError("invalid ReadableStreamAsyncIterator."), ); } const reader = this[sym.asyncIteratorReader]; if (!reader[sym.ownerReadableStream]) { return Promise.reject( new TypeError("reader owner ReadableStream is undefined."), ); } if (reader[sym.readRequests].length) { return Promise.reject( new TypeError("reader has outstanding read requests."), ); } if (!this[sym.preventCancel]) { const result = readableStreamReaderGenericCancel(reader, value); readableStreamReaderGenericRelease(reader); return result.then(() => readableStreamCreateReadResult(value, true, true) ); } readableStreamReaderGenericRelease(reader); return Promise.resolve( readableStreamCreateReadResult(value, true, true), ); }, }, AsyncIteratorPrototype); class ReadableStream { constructor( underlyingSource = {}, strategy = {}, ) { initializeReadableStream(this); const { size } = strategy; let { highWaterMark } = strategy; const { type } = underlyingSource; if (underlyingSource.type == "bytes") { if (size !== undefined) { throw new RangeError( `When underlying source is "bytes", strategy.size must be undefined.`, ); } highWaterMark = validateAndNormalizeHighWaterMark(highWaterMark ?? 0); setUpReadableByteStreamControllerFromUnderlyingSource( this, underlyingSource, highWaterMark, ); } else if (type === undefined) { const sizeAlgorithm = makeSizeAlgorithmFromSizeFunction(size); highWaterMark = validateAndNormalizeHighWaterMark(highWaterMark ?? 1); setUpReadableStreamDefaultControllerFromUnderlyingSource( this, underlyingSource, highWaterMark, sizeAlgorithm, ); } else { throw new RangeError( `Valid values for underlyingSource are "bytes" or undefined. Received: "${type}".`, ); } } get locked() { if (!isReadableStream(this)) { throw new TypeError("Invalid ReadableStream."); } return isReadableStreamLocked(this); } cancel(reason) { if (!isReadableStream(this)) { return Promise.reject(new TypeError("Invalid ReadableStream.")); } if (isReadableStreamLocked(this)) { return Promise.reject( new TypeError("Cannot cancel a locked ReadableStream."), ); } return readableStreamCancel(this, reason); } getIterator({ preventCancel, } = {}) { if (!isReadableStream(this)) { throw new TypeError("Invalid ReadableStream."); } const reader = acquireReadableStreamDefaultReader(this); const iterator = Object.create(ReadableStreamAsyncIteratorPrototype); iterator[sym.asyncIteratorReader] = reader; iterator[sym.preventCancel] = Boolean(preventCancel); return iterator; } getReader({ mode } = {}) { if (!isReadableStream(this)) { throw new TypeError("Invalid ReadableStream."); } if (mode === undefined) { return acquireReadableStreamDefaultReader(this, true); } mode = String(mode); // 3.2.5.4.4 If mode is "byob", return ? AcquireReadableStreamBYOBReader(this, true). throw new RangeError(`Unsupported mode "${mode}"`); } pipeThrough( { writable, readable, }, { preventClose, preventAbort, preventCancel, signal } = {}, ) { if (!isReadableStream(this)) { throw new TypeError("Invalid ReadableStream."); } if (!isWritableStream(writable)) { throw new TypeError("writable is not a valid WritableStream."); } if (!isReadableStream(readable)) { throw new TypeError("readable is not a valid ReadableStream."); } preventClose = Boolean(preventClose); preventAbort = Boolean(preventAbort); preventCancel = Boolean(preventCancel); if (signal && !(signal instanceof AbortSignal)) { throw new TypeError("Invalid signal."); } if (isReadableStreamLocked(this)) { throw new TypeError("ReadableStream is locked."); } if (isWritableStreamLocked(writable)) { throw new TypeError("writable is locked."); } const promise = readableStreamPipeTo( this, writable, preventClose, preventAbort, preventCancel, signal, ); setPromiseIsHandledToTrue(promise); return readable; } pipeTo( dest, { preventClose, preventAbort, preventCancel, signal } = {}, ) { if (!isReadableStream(this)) { return Promise.reject(new TypeError("Invalid ReadableStream.")); } if (!isWritableStream(dest)) { return Promise.reject( new TypeError("dest is not a valid WritableStream."), ); } preventClose = Boolean(preventClose); preventAbort = Boolean(preventAbort); preventCancel = Boolean(preventCancel); if (signal && !(signal instanceof AbortSignal)) { return Promise.reject(new TypeError("Invalid signal.")); } if (isReadableStreamLocked(this)) { return Promise.reject(new TypeError("ReadableStream is locked.")); } if (isWritableStreamLocked(dest)) { return Promise.reject(new TypeError("dest is locked.")); } return readableStreamPipeTo( this, dest, preventClose, preventAbort, preventCancel, signal, ); } tee() { if (!isReadableStream(this)) { throw new TypeError("Invalid ReadableStream."); } return readableStreamTee(this, false); } [customInspect]() { return `${this.constructor.name} { locked: ${String(this.locked)} }`; } [Symbol.asyncIterator]( options = {}, ) { return this.getIterator(options); } } class TransformStream { constructor( transformer = {}, writableStrategy = {}, readableStrategy = {}, ) { const writableSizeFunction = writableStrategy.size; let writableHighWaterMark = writableStrategy.highWaterMark; const readableSizeFunction = readableStrategy.size; let readableHighWaterMark = readableStrategy.highWaterMark; const writableType = transformer.writableType; if (writableType !== undefined) { throw new RangeError( `Expected transformer writableType to be undefined, received "${ String(writableType) }"`, ); } const writableSizeAlgorithm = makeSizeAlgorithmFromSizeFunction( writableSizeFunction, ); if (writableHighWaterMark === undefined) { writableHighWaterMark = 1; } writableHighWaterMark = validateAndNormalizeHighWaterMark( writableHighWaterMark, ); const readableType = transformer.readableType; if (readableType !== undefined) { throw new RangeError( `Expected transformer readableType to be undefined, received "${ String(readableType) }"`, ); } const readableSizeAlgorithm = makeSizeAlgorithmFromSizeFunction( readableSizeFunction, ); if (readableHighWaterMark === undefined) { readableHighWaterMark = 1; } readableHighWaterMark = validateAndNormalizeHighWaterMark( readableHighWaterMark, ); const startPromise = getDeferred(); initializeTransformStream( this, startPromise.promise, writableHighWaterMark, writableSizeAlgorithm, readableHighWaterMark, readableSizeAlgorithm, ); // the brand check expects this, and the brand check occurs in the following // but the property hasn't been defined. Object.defineProperty(this, sym.transformStreamController, { value: undefined, writable: true, configurable: true, }); setUpTransformStreamDefaultControllerFromTransformer(this, transformer); const startResult = invokeOrNoop( transformer, "start", this[sym.transformStreamController], ); startPromise.resolve(startResult); } get readable() { if (!isTransformStream(this)) { throw new TypeError("Invalid TransformStream."); } return this[sym.readable]; } get writable() { if (!isTransformStream(this)) { throw new TypeError("Invalid TransformStream."); } return this[sym.writable]; } [customInspect]() { return `${this.constructor.name} {\n readable: ${ inspect(this.readable) }\n writable: ${inspect(this.writable)}\n}`; } } class TransformStreamDefaultController { constructor() { throw new TypeError( "TransformStreamDefaultController's constructor cannot be called.", ); } get desiredSize() { if (!isTransformStreamDefaultController(this)) { throw new TypeError("Invalid TransformStreamDefaultController."); } const readableController = this[sym.controlledTransformStream][ sym.readable ][sym.readableStreamController]; return readableStreamDefaultControllerGetDesiredSize( readableController, ); } enqueue(chunk) { if (!isTransformStreamDefaultController(this)) { throw new TypeError("Invalid TransformStreamDefaultController."); } transformStreamDefaultControllerEnqueue(this, chunk); } error(reason) { if (!isTransformStreamDefaultController(this)) { throw new TypeError("Invalid TransformStreamDefaultController."); } transformStreamDefaultControllerError(this, reason); } terminate() { if (!isTransformStreamDefaultController(this)) { throw new TypeError("Invalid TransformStreamDefaultController."); } transformStreamDefaultControllerTerminate(this); } [customInspect]() { return `${this.constructor.name} { desiredSize: ${ String(this.desiredSize) } }`; } } class WritableStreamDefaultController { constructor() { throw new TypeError( "WritableStreamDefaultController's constructor cannot be called.", ); } error(e) { if (!isWritableStreamDefaultController(this)) { throw new TypeError("Invalid WritableStreamDefaultController."); } const state = this[sym.controlledWritableStream][sym.state]; if (state !== "writable") { return; } writableStreamDefaultControllerError(this, e); } [sym.abortSteps](reason) { const result = this[sym.abortAlgorithm](reason); writableStreamDefaultControllerClearAlgorithms(this); return result; } [sym.errorSteps]() { resetQueue(this); } [customInspect]() { return `${this.constructor.name} { }`; } } class WritableStreamDefaultWriter { constructor(stream) { if (!isWritableStream(stream)) { throw new TypeError("Invalid stream."); } if (isWritableStreamLocked(stream)) { throw new TypeError("Cannot create a writer for a locked stream."); } this[sym.ownerWritableStream] = stream; stream[sym.writer] = this; const state = stream[sym.state]; if (state === "writable") { if ( !writableStreamCloseQueuedOrInFlight(stream) && stream[sym.backpressure] ) { this[sym.readyPromise] = getDeferred(); } else { this[sym.readyPromise] = { promise: Promise.resolve() }; } this[sym.closedPromise] = getDeferred(); } else if (state === "erroring") { this[sym.readyPromise] = { promise: Promise.reject(stream[sym.storedError]), }; setPromiseIsHandledToTrue(this[sym.readyPromise].promise); this[sym.closedPromise] = getDeferred(); } else if (state === "closed") { this[sym.readyPromise] = { promise: Promise.resolve() }; this[sym.closedPromise] = { promise: Promise.resolve() }; } else { assert(state === "errored"); const storedError = stream[sym.storedError]; this[sym.readyPromise] = { promise: Promise.reject(storedError) }; setPromiseIsHandledToTrue(this[sym.readyPromise].promise); this[sym.closedPromise] = { promise: Promise.reject(storedError) }; setPromiseIsHandledToTrue(this[sym.closedPromise].promise); } } get closed() { if (!isWritableStreamDefaultWriter(this)) { return Promise.reject( new TypeError("Invalid WritableStreamDefaultWriter."), ); } return this[sym.closedPromise].promise; } get desiredSize() { if (!isWritableStreamDefaultWriter(this)) { throw new TypeError("Invalid WritableStreamDefaultWriter."); } if (!this[sym.ownerWritableStream]) { throw new TypeError("WritableStreamDefaultWriter has no owner."); } return writableStreamDefaultWriterGetDesiredSize(this); } get ready() { if (!isWritableStreamDefaultWriter(this)) { return Promise.reject( new TypeError("Invalid WritableStreamDefaultWriter."), ); } return this[sym.readyPromise].promise; } abort(reason) { if (!isWritableStreamDefaultWriter(this)) { return Promise.reject( new TypeError("Invalid WritableStreamDefaultWriter."), ); } if (!this[sym.ownerWritableStream]) { Promise.reject( new TypeError("WritableStreamDefaultWriter has no owner."), ); } return writableStreamDefaultWriterAbort(this, reason); } close() { if (!isWritableStreamDefaultWriter(this)) { return Promise.reject( new TypeError("Invalid WritableStreamDefaultWriter."), ); } const stream = this[sym.ownerWritableStream]; if (!stream) { Promise.reject( new TypeError("WritableStreamDefaultWriter has no owner."), ); } if (writableStreamCloseQueuedOrInFlight(stream)) { Promise.reject( new TypeError("Stream is in an invalid state to be closed."), ); } return writableStreamDefaultWriterClose(this); } releaseLock() { if (!isWritableStreamDefaultWriter(this)) { throw new TypeError("Invalid WritableStreamDefaultWriter."); } const stream = this[sym.ownerWritableStream]; if (!stream) { return; } assert(stream[sym.writer]); writableStreamDefaultWriterRelease(this); } write(chunk) { if (!isWritableStreamDefaultWriter(this)) { return Promise.reject( new TypeError("Invalid WritableStreamDefaultWriter."), ); } if (!this[sym.ownerWritableStream]) { Promise.reject( new TypeError("WritableStreamDefaultWriter has no owner."), ); } return writableStreamDefaultWriterWrite(this, chunk); } [customInspect]() { return `${this.constructor.name} { closed: Promise, desiredSize: ${ String(this.desiredSize) }, ready: Promise }`; } } class WritableStream { constructor( underlyingSink = {}, strategy = {}, ) { initializeWritableStream(this); const size = strategy.size; let highWaterMark = strategy.highWaterMark ?? 1; const { type } = underlyingSink; if (type !== undefined) { throw new RangeError(`Sink type of "${String(type)}" not supported.`); } const sizeAlgorithm = makeSizeAlgorithmFromSizeFunction(size); highWaterMark = validateAndNormalizeHighWaterMark(highWaterMark); setUpWritableStreamDefaultControllerFromUnderlyingSink( this, underlyingSink, highWaterMark, sizeAlgorithm, ); } get locked() { if (!isWritableStream(this)) { throw new TypeError("Invalid WritableStream."); } return isWritableStreamLocked(this); } abort(reason) { if (!isWritableStream(this)) { return Promise.reject(new TypeError("Invalid WritableStream.")); } if (isWritableStreamLocked(this)) { return Promise.reject( new TypeError("Cannot abort a locked WritableStream."), ); } return writableStreamAbort(this, reason); } close() { if (!isWritableStream(this)) { return Promise.reject(new TypeError("Invalid WritableStream.")); } if (isWritableStreamLocked(this)) { return Promise.reject( new TypeError("Cannot abort a locked WritableStream."), ); } if (writableStreamCloseQueuedOrInFlight(this)) { return Promise.reject( new TypeError("Cannot close an already closing WritableStream."), ); } return writableStreamClose(this); } getWriter() { if (!isWritableStream(this)) { throw new TypeError("Invalid WritableStream."); } return acquireWritableStreamDefaultWriter(this); } [customInspect]() { return `${this.constructor.name} { locked: ${String(this.locked)} }`; } } function acquireReadableStreamDefaultReader( stream, forAuthorCode = false, ) { const reader = new ReadableStreamDefaultReader(stream); reader[sym.forAuthorCode] = forAuthorCode; return reader; } function acquireWritableStreamDefaultWriter( stream, ) { return new WritableStreamDefaultWriter(stream); } function call( fn, v, args, ) { return Function.prototype.apply.call(fn, v, args); } function createAlgorithmFromUnderlyingMethod( underlyingObject, methodName, algoArgCount, ...extraArgs ) { const method = underlyingObject[methodName]; if (method) { if (!isCallable(method)) { throw new TypeError("method is not callable"); } if (algoArgCount === 0) { return async () => call(method, underlyingObject, extraArgs); } else { return async (arg) => { const fullArgs = [arg, ...extraArgs]; return call(method, underlyingObject, fullArgs); }; } } return async () => undefined; } function createReadableStream( startAlgorithm, pullAlgorithm, cancelAlgorithm, highWaterMark = 1, sizeAlgorithm = () => 1, ) { highWaterMark = validateAndNormalizeHighWaterMark(highWaterMark); const stream = Object.create( ReadableStream.prototype, ); initializeReadableStream(stream); const controller = Object.create( ReadableStreamDefaultController.prototype, ); setUpReadableStreamDefaultController( stream, controller, startAlgorithm, pullAlgorithm, cancelAlgorithm, highWaterMark, sizeAlgorithm, ); return stream; } function createWritableStream( startAlgorithm, writeAlgorithm, closeAlgorithm, abortAlgorithm, highWaterMark = 1, sizeAlgorithm = () => 1, ) { highWaterMark = validateAndNormalizeHighWaterMark(highWaterMark); const stream = Object.create(WritableStream.prototype); initializeWritableStream(stream); const controller = Object.create( WritableStreamDefaultController.prototype, ); setUpWritableStreamDefaultController( stream, controller, startAlgorithm, writeAlgorithm, closeAlgorithm, abortAlgorithm, highWaterMark, sizeAlgorithm, ); return stream; } function dequeueValue(container) { assert(sym.queue in container && sym.queueTotalSize in container); assert(container[sym.queue].length); const pair = container[sym.queue].shift(); container[sym.queueTotalSize] -= pair.size; if (container[sym.queueTotalSize] <= 0) { container[sym.queueTotalSize] = 0; } return pair.value; } function enqueueValueWithSize( container, value, size, ) { assert(sym.queue in container && sym.queueTotalSize in container); size = Number(size); if (!isFiniteNonNegativeNumber(size)) { throw new RangeError("size must be a finite non-negative number."); } container[sym.queue].push({ value, size }); container[sym.queueTotalSize] += size; } /** Non-spec mechanism to "unwrap" a promise and store it to be resolved * later. */ function getDeferred() { let resolve; let reject; const promise = new Promise((res, rej) => { resolve = res; reject = rej; }); return { promise, resolve: resolve, reject: reject }; } function initializeReadableStream( stream, ) { stream[sym.state] = "readable"; stream[sym.reader] = stream[sym.storedError] = undefined; stream[sym.disturbed] = false; } function initializeTransformStream( stream, startPromise, writableHighWaterMark, writableSizeAlgorithm, readableHighWaterMark, readableSizeAlgorithm, ) { const startAlgorithm = () => startPromise; const writeAlgorithm = (chunk) => transformStreamDefaultSinkWriteAlgorithm(stream, chunk); const abortAlgorithm = (reason) => transformStreamDefaultSinkAbortAlgorithm(stream, reason); const closeAlgorithm = () => transformStreamDefaultSinkCloseAlgorithm(stream); stream[sym.writable] = createWritableStream( startAlgorithm, writeAlgorithm, closeAlgorithm, abortAlgorithm, writableHighWaterMark, writableSizeAlgorithm, ); const pullAlgorithm = () => transformStreamDefaultSourcePullAlgorithm(stream); const cancelAlgorithm = (reason) => { transformStreamErrorWritableAndUnblockWrite(stream, reason); return Promise.resolve(undefined); }; stream[sym.readable] = createReadableStream( startAlgorithm, pullAlgorithm, cancelAlgorithm, readableHighWaterMark, readableSizeAlgorithm, ); stream[sym.backpressure] = stream[sym.backpressureChangePromise] = undefined; transformStreamSetBackpressure(stream, true); Object.defineProperty(stream, sym.transformStreamController, { value: undefined, configurable: true, }); } function initializeWritableStream( stream, ) { stream[sym.state] = "writable"; stream[sym.storedError] = stream[sym.writer] = stream[ sym.writableStreamController ] = stream[sym.inFlightWriteRequest] = stream[sym.closeRequest] = stream[ sym.inFlightCloseRequest ] = stream[sym.pendingAbortRequest] = undefined; stream[sym.writeRequests] = []; stream[sym.backpressure] = false; } function invokeOrNoop( o, p, ...args ) { assert(o); const method = o[p]; if (!method) { return undefined; } return call(method, o, args); } function isCallable(value) { return typeof value === "function"; } function isDetachedBuffer(value) { return sym.isFakeDetached in value; } function isFiniteNonNegativeNumber(v) { return Number.isFinite(v) && (v) >= 0; } function isReadableByteStreamController( x, ) { return !( typeof x !== "object" || x === null || !(sym.controlledReadableByteStream in x) ); } function isReadableStream(x) { return !( typeof x !== "object" || x === null || !(sym.readableStreamController in x) ); } function isReadableStreamAsyncIterator( x, ) { if (typeof x !== "object" || x === null) { return false; } return sym.asyncIteratorReader in x; } function isReadableStreamDefaultController( x, ) { return !( typeof x !== "object" || x === null || !(sym.controlledReadableStream in x) ); } function isReadableStreamDefaultReader( x, ) { return !(typeof x !== "object" || x === null || !(sym.readRequests in x)); } function isReadableStreamLocked(stream) { assert(isReadableStream(stream)); return !!stream[sym.reader]; } function isReadableStreamDisturbed(stream) { assert(isReadableStream(stream)); return !!stream[sym.disturbed]; } function isTransformStream(x) { return !( typeof x !== "object" || x === null || !(sym.transformStreamController in x) ); } function isTransformStreamDefaultController( x, ) { return !( typeof x !== "object" || x === null || !(sym.controlledTransformStream in x) ); } function isWritableStream(x) { return !( typeof x !== "object" || x === null || !(sym.writableStreamController in x) ); } function isWritableStreamDefaultController( x, ) { return !( typeof x !== "object" || x === null || !(sym.controlledWritableStream in x) ); } function isWritableStreamDefaultWriter( x, ) { return !( typeof x !== "object" || x === null || !(sym.ownerWritableStream in x) ); } function isWritableStreamLocked(stream) { assert(isWritableStream(stream)); return stream[sym.writer] !== undefined; } function makeSizeAlgorithmFromSizeFunction( size, ) { if (size === undefined) { return () => 1; } if (typeof size !== "function") { throw new TypeError("size must be callable."); } return (chunk) => { return size.call(undefined, chunk); }; } function peekQueueValue(container) { assert(sym.queue in container && sym.queueTotalSize in container); assert(container[sym.queue].length); const [pair] = container[sym.queue]; return pair.value; } function readableByteStreamControllerShouldCallPull( controller, ) { const stream = controller[sym.controlledReadableByteStream]; if ( stream[sym.state] !== "readable" || controller[sym.closeRequested] || !controller[sym.started] ) { return false; } if ( readableStreamHasDefaultReader(stream) && readableStreamGetNumReadRequests(stream) > 0 ) { return true; } // 3.13.25.6 If ! ReadableStreamHasBYOBReader(stream) is true and ! // ReadableStreamGetNumReadIntoRequests(stream) > 0, return true. const desiredSize = readableByteStreamControllerGetDesiredSize(controller); assert(desiredSize !== null); return desiredSize > 0; } function readableByteStreamControllerCallPullIfNeeded( controller, ) { const shouldPull = readableByteStreamControllerShouldCallPull(controller); if (!shouldPull) { return; } if (controller[sym.pulling]) { controller[sym.pullAgain] = true; return; } assert(controller[sym.pullAgain] === false); controller[sym.pulling] = true; const pullPromise = controller[sym.pullAlgorithm](); setPromiseIsHandledToTrue( pullPromise.then( () => { controller[sym.pulling] = false; if (controller[sym.pullAgain]) { controller[sym.pullAgain] = false; readableByteStreamControllerCallPullIfNeeded(controller); } }, (e) => { readableByteStreamControllerError(controller, e); }, ), ); } function readableByteStreamControllerClearAlgorithms( controller, ) { controller[sym.pullAlgorithm] = undefined; controller[sym.cancelAlgorithm] = undefined; } function readableByteStreamControllerClose( controller, ) { const stream = controller[sym.controlledReadableByteStream]; if (controller[sym.closeRequested] || stream[sym.state] !== "readable") { return; } if (controller[sym.queueTotalSize] > 0) { controller[sym.closeRequested] = true; return; } // 3.13.6.4 If controller.[[pendingPullIntos]] is not empty, (BYOB Support) readableByteStreamControllerClearAlgorithms(controller); readableStreamClose(stream); } function readableByteStreamControllerEnqueue( controller, chunk, ) { const stream = controller[sym.controlledReadableByteStream]; if (controller[sym.closeRequested] || stream[sym.state] !== "readable") { return; } const { buffer, byteOffset, byteLength } = chunk; const transferredBuffer = transferArrayBuffer(buffer); if (readableStreamHasDefaultReader(stream)) { if (readableStreamGetNumReadRequests(stream) === 0) { readableByteStreamControllerEnqueueChunkToQueue( controller, transferredBuffer, byteOffset, byteLength, ); } else { assert(controller[sym.queue].length === 0); const transferredView = new Uint8Array( transferredBuffer, byteOffset, byteLength, ); readableStreamFulfillReadRequest(stream, transferredView, false); } // 3.13.9.8 Otherwise, if ! ReadableStreamHasBYOBReader(stream) is true } else { assert(!isReadableStreamLocked(stream)); readableByteStreamControllerEnqueueChunkToQueue( controller, transferredBuffer, byteOffset, byteLength, ); } readableByteStreamControllerCallPullIfNeeded(controller); } function readableByteStreamControllerEnqueueChunkToQueue( controller, buffer, byteOffset, byteLength, ) { controller[sym.queue].push({ value: buffer, offset: byteOffset, size: byteLength, }); controller[sym.queueTotalSize] += byteLength; } function readableByteStreamControllerError( controller, e, ) { const stream = controller[sym.controlledReadableByteStream]; if (stream[sym.state] !== "readable") { return; } // 3.13.11.3 Perform ! ReadableByteStreamControllerClearPendingPullIntos(controller). resetQueue(controller); readableByteStreamControllerClearAlgorithms(controller); readableStreamError(stream, e); } function readableByteStreamControllerGetDesiredSize( controller, ) { const stream = controller[sym.controlledReadableByteStream]; const state = stream[sym.state]; if (state === "errored") { return null; } if (state === "closed") { return 0; } return controller[sym.strategyHWM] - controller[sym.queueTotalSize]; } function readableByteStreamControllerHandleQueueDrain( controller, ) { assert( controller[sym.controlledReadableByteStream][sym.state] === "readable", ); if ( controller[sym.queueTotalSize] === 0 && controller[sym.closeRequested] ) { readableByteStreamControllerClearAlgorithms(controller); readableStreamClose(controller[sym.controlledReadableByteStream]); } else { readableByteStreamControllerCallPullIfNeeded(controller); } } function readableStreamAddReadRequest( stream, ) { assert(isReadableStreamDefaultReader(stream[sym.reader])); assert(stream[sym.state] === "readable"); const promise = getDeferred(); stream[sym.reader][sym.readRequests].push(promise); return promise.promise; } function readableStreamCancel( stream, reason, ) { stream[sym.disturbed] = true; if (stream[sym.state] === "closed") { return Promise.resolve(); } if (stream[sym.state] === "errored") { return Promise.reject(stream[sym.storedError]); } readableStreamClose(stream); return stream[sym.readableStreamController][sym.cancelSteps](reason).then( () => undefined, ); } function readableStreamClose(stream) { assert(stream[sym.state] === "readable"); stream[sym.state] = "closed"; const reader = stream[sym.reader]; if (!reader) { return; } if (isReadableStreamDefaultReader(reader)) { for (const readRequest of reader[sym.readRequests]) { assert(readRequest.resolve); readRequest.resolve( readableStreamCreateReadResult( undefined, true, reader[sym.forAuthorCode], ), ); } reader[sym.readRequests] = []; } const resolve = reader[sym.closedPromise].resolve; assert(resolve); resolve(); } function readableStreamCreateReadResult( value, done, forAuthorCode, ) { const prototype = forAuthorCode ? Object.prototype : null; assert(typeof done === "boolean"); const obj = Object.create(prototype); Object.defineProperties(obj, { value: { value, writable: true, enumerable: true, configurable: true }, done: { value: done, writable: true, enumerable: true, configurable: true, }, }); return obj; } function readableStreamDefaultControllerCallPullIfNeeded( controller, ) { const shouldPull = readableStreamDefaultControllerShouldCallPull( controller, ); if (!shouldPull) { return; } if (controller[sym.pulling]) { controller[sym.pullAgain] = true; return; } assert(controller[sym.pullAgain] === false); controller[sym.pulling] = true; const pullPromise = controller[sym.pullAlgorithm](); pullPromise.then( () => { controller[sym.pulling] = false; if (controller[sym.pullAgain]) { controller[sym.pullAgain] = false; readableStreamDefaultControllerCallPullIfNeeded(controller); } }, (e) => { readableStreamDefaultControllerError(controller, e); }, ); } function readableStreamDefaultControllerCanCloseOrEnqueue( controller, ) { const state = controller[sym.controlledReadableStream][sym.state]; return !controller[sym.closeRequested] && state === "readable"; } function readableStreamDefaultControllerClearAlgorithms( controller, ) { controller[sym.pullAlgorithm] = undefined; controller[sym.cancelAlgorithm] = undefined; controller[sym.strategySizeAlgorithm] = undefined; } function readableStreamDefaultControllerClose( controller, ) { if (!readableStreamDefaultControllerCanCloseOrEnqueue(controller)) { return; } const stream = controller[sym.controlledReadableStream]; controller[sym.closeRequested] = true; if (controller[sym.queue].length === 0) { readableStreamDefaultControllerClearAlgorithms(controller); readableStreamClose(stream); } } function readableStreamDefaultControllerEnqueue( controller, chunk, ) { if (!readableStreamDefaultControllerCanCloseOrEnqueue(controller)) { return; } const stream = controller[sym.controlledReadableStream]; if ( isReadableStreamLocked(stream) && readableStreamGetNumReadRequests(stream) > 0 ) { readableStreamFulfillReadRequest(stream, chunk, false); } else { try { const chunkSize = controller[sym.strategySizeAlgorithm](chunk); enqueueValueWithSize(controller, chunk, chunkSize); } catch (err) { readableStreamDefaultControllerError(controller, err); throw err; } } readableStreamDefaultControllerCallPullIfNeeded(controller); } function readableStreamDefaultControllerGetDesiredSize( controller, ) { const stream = controller[sym.controlledReadableStream]; const state = stream[sym.state]; if (state === "errored") { return null; } if (state === "closed") { return 0; } return controller[sym.strategyHWM] - controller[sym.queueTotalSize]; } function readableStreamDefaultControllerError( controller, e, ) { const stream = controller[sym.controlledReadableStream]; if (stream[sym.state] !== "readable") { return; } resetQueue(controller); readableStreamDefaultControllerClearAlgorithms(controller); readableStreamError(stream, e); } function readableStreamDefaultControllerHasBackpressure( controller, ) { return readableStreamDefaultControllerShouldCallPull(controller); } function readableStreamDefaultControllerShouldCallPull( controller, ) { const stream = controller[sym.controlledReadableStream]; if ( !readableStreamDefaultControllerCanCloseOrEnqueue(controller) || controller[sym.started] === false ) { return false; } if ( isReadableStreamLocked(stream) && readableStreamGetNumReadRequests(stream) > 0 ) { return true; } const desiredSize = readableStreamDefaultControllerGetDesiredSize( controller, ); assert(desiredSize !== null); return desiredSize > 0; } function readableStreamDefaultReaderRead( reader, ) { const stream = reader[sym.ownerReadableStream]; assert(stream); stream[sym.disturbed] = true; if (stream[sym.state] === "closed") { return Promise.resolve( readableStreamCreateReadResult( undefined, true, reader[sym.forAuthorCode], ), ); } if (stream[sym.state] === "errored") { return Promise.reject(stream[sym.storedError]); } assert(stream[sym.state] === "readable"); return (stream[ sym.readableStreamController ])[sym.pullSteps](); } function readableStreamError(stream, e) { assert(isReadableStream(stream)); assert(stream[sym.state] === "readable"); stream[sym.state] = "errored"; stream[sym.storedError] = e; const reader = stream[sym.reader]; if (reader === undefined) { return; } if (isReadableStreamDefaultReader(reader)) { for (const readRequest of reader[sym.readRequests]) { assert(readRequest.reject); readRequest.reject(e); readRequest.reject = undefined; readRequest.resolve = undefined; } reader[sym.readRequests] = []; } // 3.5.6.8 Otherwise, support BYOB Reader reader[sym.closedPromise].reject(e); reader[sym.closedPromise].reject = undefined; reader[sym.closedPromise].resolve = undefined; setPromiseIsHandledToTrue(reader[sym.closedPromise].promise); } function readableStreamFulfillReadRequest( stream, chunk, done, ) { const reader = stream[sym.reader]; const readRequest = reader[sym.readRequests].shift(); assert(readRequest.resolve); readRequest.resolve( readableStreamCreateReadResult(chunk, done, reader[sym.forAuthorCode]), ); } function readableStreamGetNumReadRequests( stream, ) { return stream[sym.reader]?.[sym.readRequests].length ?? 0; } function readableStreamHasDefaultReader( stream, ) { const reader = stream[sym.reader]; return !(reader === undefined || !isReadableStreamDefaultReader(reader)); } function readableStreamPipeTo( source, dest, preventClose, preventAbort, preventCancel, signal, ) { assert(isReadableStream(source)); assert(isWritableStream(dest)); assert( typeof preventClose === "boolean" && typeof preventAbort === "boolean" && typeof preventCancel === "boolean", ); assert(signal === undefined || signal instanceof AbortSignal); assert(!isReadableStreamLocked(source)); assert(!isWritableStreamLocked(dest)); const reader = acquireReadableStreamDefaultReader(source); const writer = acquireWritableStreamDefaultWriter(dest); source[sym.disturbed] = true; let shuttingDown = false; const promise = getDeferred(); let abortAlgorithm; if (signal) { abortAlgorithm = () => { const error = new DOMException("Abort signal received.", "AbortSignal"); const actions = []; if (!preventAbort) { actions.push(() => { if (dest[sym.state] === "writable") { return writableStreamAbort(dest, error); } else { return Promise.resolve(undefined); } }); } if (!preventCancel) { actions.push(() => { if (source[sym.state] === "readable") { return readableStreamCancel(source, error); } else { return Promise.resolve(undefined); } }); } shutdownWithAction( () => Promise.all(actions.map((action) => action())), true, error, ); }; if (signal.aborted) { abortAlgorithm(); return promise.promise; } signal.addEventListener("abort", abortAlgorithm); } let currentWrite = Promise.resolve(); // At this point, the spec becomes non-specific and vague. Most of the rest // of this code is based on the reference implementation that is part of the // specification. This is why the functions are only scoped to this function // to ensure they don't leak into the spec compliant parts. function isOrBecomesClosed( stream, promise, action, ) { if (stream[sym.state] === "closed") { action(); } else { setPromiseIsHandledToTrue(promise.then(action)); } } function isOrBecomesErrored( stream, promise, action, ) { if (stream[sym.state] === "errored") { action(stream[sym.storedError]); } else { setPromiseIsHandledToTrue(promise.catch((error) => action(error))); } } function finalize(isError, error) { writableStreamDefaultWriterRelease(writer); readableStreamReaderGenericRelease(reader); if (signal) { signal.removeEventListener("abort", abortAlgorithm); } if (isError) { promise.reject(error); } else { promise.resolve(); } } function waitForWritesToFinish() { const oldCurrentWrite = currentWrite; return currentWrite.then(() => oldCurrentWrite !== currentWrite ? waitForWritesToFinish() : undefined ); } function shutdownWithAction( action, originalIsError, originalError, ) { function doTheRest() { setPromiseIsHandledToTrue( action().then( () => finalize(originalIsError, originalError), (newError) => finalize(true, newError), ), ); } if (shuttingDown) { return; } shuttingDown = true; if ( dest[sym.state] === "writable" && writableStreamCloseQueuedOrInFlight(dest) === false ) { setPromiseIsHandledToTrue(waitForWritesToFinish().then(doTheRest)); } else { doTheRest(); } } function shutdown(isError, error) { if (shuttingDown) { return; } shuttingDown = true; if ( dest[sym.state] === "writable" && !writableStreamCloseQueuedOrInFlight(dest) ) { setPromiseIsHandledToTrue( waitForWritesToFinish().then(() => finalize(isError, error)), ); } finalize(isError, error); } function pipeStep() { if (shuttingDown) { return Promise.resolve(true); } return writer[sym.readyPromise].promise.then(() => { return readableStreamDefaultReaderRead(reader).then( ({ value, done }) => { if (done === true) { return true; } currentWrite = writableStreamDefaultWriterWrite( writer, value, ).then(undefined, () => {}); return false; }, ); }); } function pipeLoop() { return new Promise((resolveLoop, rejectLoop) => { function next(done) { if (done) { resolveLoop(undefined); } else { setPromiseIsHandledToTrue(pipeStep().then(next, rejectLoop)); } } next(false); }); } isOrBecomesErrored( source, reader[sym.closedPromise].promise, (storedError) => { if (!preventAbort) { shutdownWithAction( () => writableStreamAbort(dest, storedError), true, storedError, ); } else { shutdown(true, storedError); } }, ); isOrBecomesErrored( dest, writer[sym.closedPromise].promise, (storedError) => { if (!preventCancel) { shutdownWithAction( () => readableStreamCancel(source, storedError), true, storedError, ); } else { shutdown(true, storedError); } }, ); isOrBecomesClosed(source, reader[sym.closedPromise].promise, () => { if (!preventClose) { shutdownWithAction(() => writableStreamDefaultWriterCloseWithErrorPropagation(writer) ); } }); if ( writableStreamCloseQueuedOrInFlight(dest) || dest[sym.state] === "closed" ) { const destClosed = new TypeError( "The destination writable stream closed before all data could be piped to it.", ); if (!preventCancel) { shutdownWithAction( () => readableStreamCancel(source, destClosed), true, destClosed, ); } else { shutdown(true, destClosed); } } setPromiseIsHandledToTrue(pipeLoop()); return promise.promise; } function readableStreamReaderGenericCancel( reader, reason, ) { const stream = reader[sym.ownerReadableStream]; assert(stream); return readableStreamCancel(stream, reason); } function readableStreamReaderGenericInitialize( reader, stream, ) { reader[sym.forAuthorCode] = true; reader[sym.ownerReadableStream] = stream; stream[sym.reader] = reader; if (stream[sym.state] === "readable") { reader[sym.closedPromise] = getDeferred(); } else if (stream[sym.state] === "closed") { reader[sym.closedPromise] = { promise: Promise.resolve() }; } else { assert(stream[sym.state] === "errored"); reader[sym.closedPromise] = { promise: Promise.reject(stream[sym.storedError]), }; setPromiseIsHandledToTrue(reader[sym.closedPromise].promise); } } function readableStreamReaderGenericRelease( reader, ) { assert(reader[sym.ownerReadableStream]); assert(reader[sym.ownerReadableStream][sym.reader] === reader); const closedPromise = reader[sym.closedPromise]; if (reader[sym.ownerReadableStream][sym.state] === "readable") { assert(closedPromise.reject); closedPromise.reject(new TypeError("ReadableStream state is readable.")); } else { closedPromise.promise = Promise.reject( new TypeError("Reading is closed."), ); delete closedPromise.reject; delete closedPromise.resolve; } setPromiseIsHandledToTrue(closedPromise.promise); reader[sym.ownerReadableStream][sym.reader] = undefined; reader[sym.ownerReadableStream] = undefined; } function readableStreamTee( stream, cloneForBranch2, ) { assert(isReadableStream(stream)); assert(typeof cloneForBranch2 === "boolean"); const reader = acquireReadableStreamDefaultReader(stream); let reading = false; let canceled1 = false; let canceled2 = false; let reason1 = undefined; let reason2 = undefined; /* eslint-disable prefer-const */ let branch1; let branch2; /* eslint-enable prefer-const */ const cancelPromise = getDeferred(); const pullAlgorithm = () => { if (reading) { return Promise.resolve(); } reading = true; const readPromise = readableStreamDefaultReaderRead(reader).then( (result) => { reading = false; assert(typeof result === "object"); const { done } = result; assert(typeof done === "boolean"); if (done) { if (!canceled1) { readableStreamDefaultControllerClose( branch1[ sym.readableStreamController ], ); } if (!canceled2) { readableStreamDefaultControllerClose( branch2[ sym.readableStreamController ], ); } return; } const { value } = result; const value1 = value; let value2 = value; if (!canceled2 && cloneForBranch2) { value2 = cloneValue(value2); } if (!canceled1) { readableStreamDefaultControllerEnqueue( branch1[ sym.readableStreamController ], value1, ); } if (!canceled2) { readableStreamDefaultControllerEnqueue( branch2[ sym.readableStreamController ], value2, ); } }, ); setPromiseIsHandledToTrue(readPromise); return Promise.resolve(); }; const cancel1Algorithm = (reason) => { canceled1 = true; reason1 = reason; if (canceled2) { const compositeReason = [reason1, reason2]; const cancelResult = readableStreamCancel(stream, compositeReason); cancelPromise.resolve(cancelResult); } return cancelPromise.promise; }; const cancel2Algorithm = (reason) => { canceled2 = true; reason2 = reason; if (canceled1) { const compositeReason = [reason1, reason2]; const cancelResult = readableStreamCancel(stream, compositeReason); cancelPromise.resolve(cancelResult); } return cancelPromise.promise; }; const startAlgorithm = () => undefined; branch1 = createReadableStream( startAlgorithm, pullAlgorithm, cancel1Algorithm, ); branch2 = createReadableStream( startAlgorithm, pullAlgorithm, cancel2Algorithm, ); setPromiseIsHandledToTrue( reader[sym.closedPromise].promise.catch((r) => { readableStreamDefaultControllerError( branch1[ sym.readableStreamController ], r, ); readableStreamDefaultControllerError( branch2[ sym.readableStreamController ], r, ); }), ); return [branch1, branch2]; } function resetQueue(container) { assert(sym.queue in container && sym.queueTotalSize in container); container[sym.queue] = []; container[sym.queueTotalSize] = 0; } /** An internal function which mimics the behavior of setting the promise to * handled in JavaScript. In this situation, an assertion failure, which * shouldn't happen will get thrown, instead of swallowed. */ function setPromiseIsHandledToTrue(promise) { promise.then(undefined, (e) => { if (e && e instanceof AssertionError) { queueMicrotask(() => { throw e; }); } }); } function setUpReadableByteStreamController( stream, controller, startAlgorithm, pullAlgorithm, cancelAlgorithm, highWaterMark, autoAllocateChunkSize, ) { assert(stream[sym.readableStreamController] === undefined); if (autoAllocateChunkSize !== undefined) { assert(Number.isInteger(autoAllocateChunkSize)); assert(autoAllocateChunkSize >= 0); } controller[sym.controlledReadableByteStream] = stream; controller[sym.pulling] = controller[sym.pullAgain] = false; controller[sym.byobRequest] = undefined; controller[sym.queue] = []; controller[sym.queueTotalSize] = 0; controller[sym.closeRequested] = controller[sym.started] = false; controller[sym.strategyHWM] = validateAndNormalizeHighWaterMark( highWaterMark, ); controller[sym.pullAlgorithm] = pullAlgorithm; controller[sym.cancelAlgorithm] = cancelAlgorithm; controller[sym.autoAllocateChunkSize] = autoAllocateChunkSize; // 3.13.26.12 Set controller.[[pendingPullIntos]] to a new empty List. stream[sym.readableStreamController] = controller; const startResult = startAlgorithm(); const startPromise = Promise.resolve(startResult); setPromiseIsHandledToTrue( startPromise.then( () => { controller[sym.started] = true; assert(!controller[sym.pulling]); assert(!controller[sym.pullAgain]); readableByteStreamControllerCallPullIfNeeded(controller); }, (r) => { readableByteStreamControllerError(controller, r); }, ), ); } function setUpReadableByteStreamControllerFromUnderlyingSource( stream, underlyingByteSource, highWaterMark, ) { assert(underlyingByteSource); const controller = Object.create( ReadableByteStreamController.prototype, ); const startAlgorithm = () => { return invokeOrNoop(underlyingByteSource, "start", controller); }; const pullAlgorithm = createAlgorithmFromUnderlyingMethod( underlyingByteSource, "pull", 0, controller, ); setFunctionName(pullAlgorithm, "[[pullAlgorithm]]"); const cancelAlgorithm = createAlgorithmFromUnderlyingMethod( underlyingByteSource, "cancel", 1, ); setFunctionName(cancelAlgorithm, "[[cancelAlgorithm]]"); // 3.13.27.6 Let autoAllocateChunkSize be ? GetV(underlyingByteSource, "autoAllocateChunkSize"). const autoAllocateChunkSize = undefined; setUpReadableByteStreamController( stream, controller, startAlgorithm, pullAlgorithm, cancelAlgorithm, highWaterMark, autoAllocateChunkSize, ); } function setUpReadableStreamDefaultController( stream, controller, startAlgorithm, pullAlgorithm, cancelAlgorithm, highWaterMark, sizeAlgorithm, ) { assert(stream[sym.readableStreamController] === undefined); controller[sym.controlledReadableStream] = stream; controller[sym.queue] = []; controller[sym.queueTotalSize] = 0; controller[sym.started] = controller[sym.closeRequested] = controller[ sym.pullAgain ] = controller[sym.pulling] = false; controller[sym.strategySizeAlgorithm] = sizeAlgorithm; controller[sym.strategyHWM] = highWaterMark; controller[sym.pullAlgorithm] = pullAlgorithm; controller[sym.cancelAlgorithm] = cancelAlgorithm; stream[sym.readableStreamController] = controller; const startResult = startAlgorithm(); const startPromise = Promise.resolve(startResult); setPromiseIsHandledToTrue( startPromise.then( () => { controller[sym.started] = true; assert(controller[sym.pulling] === false); assert(controller[sym.pullAgain] === false); readableStreamDefaultControllerCallPullIfNeeded(controller); }, (r) => { readableStreamDefaultControllerError(controller, r); }, ), ); } function setUpReadableStreamDefaultControllerFromUnderlyingSource( stream, underlyingSource, highWaterMark, sizeAlgorithm, ) { assert(underlyingSource); const controller = Object.create( ReadableStreamDefaultController.prototype, ); const startAlgorithm = () => invokeOrNoop(underlyingSource, "start", controller); const pullAlgorithm = createAlgorithmFromUnderlyingMethod( underlyingSource, "pull", 0, controller, ); setFunctionName(pullAlgorithm, "[[pullAlgorithm]]"); const cancelAlgorithm = createAlgorithmFromUnderlyingMethod( underlyingSource, "cancel", 1, ); setFunctionName(cancelAlgorithm, "[[cancelAlgorithm]]"); setUpReadableStreamDefaultController( stream, controller, startAlgorithm, pullAlgorithm, cancelAlgorithm, highWaterMark, sizeAlgorithm, ); } function setUpTransformStreamDefaultController( stream, controller, transformAlgorithm, flushAlgorithm, ) { assert(isTransformStream(stream)); assert(stream[sym.transformStreamController] === undefined); controller[sym.controlledTransformStream] = stream; stream[sym.transformStreamController] = controller; controller[sym.transformAlgorithm] = transformAlgorithm; controller[sym.flushAlgorithm] = flushAlgorithm; } function setUpTransformStreamDefaultControllerFromTransformer( stream, transformer, ) { assert(transformer); const controller = Object.create( TransformStreamDefaultController.prototype, ); let transformAlgorithm = (chunk) => { try { transformStreamDefaultControllerEnqueue( controller, // it defaults to no tranformation, so I is assumed to be O chunk, ); } catch (e) { return Promise.reject(e); } return Promise.resolve(); }; const transformMethod = transformer.transform; if (transformMethod) { if (typeof transformMethod !== "function") { throw new TypeError("tranformer.transform must be callable."); } transformAlgorithm = async (chunk) => call(transformMethod, transformer, [chunk, controller]); } const flushAlgorithm = createAlgorithmFromUnderlyingMethod( transformer, "flush", 0, controller, ); setUpTransformStreamDefaultController( stream, controller, transformAlgorithm, flushAlgorithm, ); } function setUpWritableStreamDefaultController( stream, controller, startAlgorithm, writeAlgorithm, closeAlgorithm, abortAlgorithm, highWaterMark, sizeAlgorithm, ) { assert(isWritableStream(stream)); assert(stream[sym.writableStreamController] === undefined); controller[sym.controlledWritableStream] = stream; stream[sym.writableStreamController] = controller; controller[sym.queue] = []; controller[sym.queueTotalSize] = 0; controller[sym.started] = false; controller[sym.strategySizeAlgorithm] = sizeAlgorithm; controller[sym.strategyHWM] = highWaterMark; controller[sym.writeAlgorithm] = writeAlgorithm; controller[sym.closeAlgorithm] = closeAlgorithm; controller[sym.abortAlgorithm] = abortAlgorithm; const backpressure = writableStreamDefaultControllerGetBackpressure( controller, ); writableStreamUpdateBackpressure(stream, backpressure); const startResult = startAlgorithm(); const startPromise = Promise.resolve(startResult); setPromiseIsHandledToTrue( startPromise.then( () => { assert( stream[sym.state] === "writable" || stream[sym.state] === "erroring", ); controller[sym.started] = true; writableStreamDefaultControllerAdvanceQueueIfNeeded(controller); }, (r) => { assert( stream[sym.state] === "writable" || stream[sym.state] === "erroring", ); controller[sym.started] = true; writableStreamDealWithRejection(stream, r); }, ), ); } function setUpWritableStreamDefaultControllerFromUnderlyingSink( stream, underlyingSink, highWaterMark, sizeAlgorithm, ) { assert(underlyingSink); const controller = Object.create( WritableStreamDefaultController.prototype, ); const startAlgorithm = () => { return invokeOrNoop(underlyingSink, "start", controller); }; const writeAlgorithm = createAlgorithmFromUnderlyingMethod( underlyingSink, "write", 1, controller, ); setFunctionName(writeAlgorithm, "[[writeAlgorithm]]"); const closeAlgorithm = createAlgorithmFromUnderlyingMethod( underlyingSink, "close", 0, ); setFunctionName(closeAlgorithm, "[[closeAlgorithm]]"); const abortAlgorithm = createAlgorithmFromUnderlyingMethod( underlyingSink, "abort", 1, ); setFunctionName(abortAlgorithm, "[[abortAlgorithm]]"); setUpWritableStreamDefaultController( stream, controller, startAlgorithm, writeAlgorithm, closeAlgorithm, abortAlgorithm, highWaterMark, sizeAlgorithm, ); } function transformStreamDefaultControllerClearAlgorithms( controller, ) { controller[sym.transformAlgorithm] = undefined; controller[sym.flushAlgorithm] = undefined; } function transformStreamDefaultControllerEnqueue( controller, chunk, ) { const stream = controller[sym.controlledTransformStream]; const readableController = stream[sym.readable][ sym.readableStreamController ]; if (!readableStreamDefaultControllerCanCloseOrEnqueue(readableController)) { throw new TypeError( "TransformStream's readable controller cannot be closed or enqueued.", ); } try { readableStreamDefaultControllerEnqueue(readableController, chunk); } catch (e) { transformStreamErrorWritableAndUnblockWrite(stream, e); throw stream[sym.readable][sym.storedError]; } const backpressure = readableStreamDefaultControllerHasBackpressure( readableController, ); if (backpressure) { transformStreamSetBackpressure(stream, true); } } function transformStreamDefaultControllerError( controller, e, ) { transformStreamError(controller[sym.controlledTransformStream], e); } function transformStreamDefaultControllerPerformTransform( controller, chunk, ) { const transformPromise = controller[sym.transformAlgorithm](chunk); return transformPromise.then(undefined, (r) => { transformStreamError(controller[sym.controlledTransformStream], r); throw r; }); } function transformStreamDefaultSinkAbortAlgorithm( stream, reason, ) { transformStreamError(stream, reason); return Promise.resolve(undefined); } function transformStreamDefaultSinkCloseAlgorithm( stream, ) { const readable = stream[sym.readable]; const controller = stream[sym.transformStreamController]; const flushPromise = controller[sym.flushAlgorithm](); transformStreamDefaultControllerClearAlgorithms(controller); return flushPromise.then( () => { if (readable[sym.state] === "errored") { throw readable[sym.storedError]; } const readableController = readable[ sym.readableStreamController ]; if ( readableStreamDefaultControllerCanCloseOrEnqueue(readableController) ) { readableStreamDefaultControllerClose(readableController); } }, (r) => { transformStreamError(stream, r); throw readable[sym.storedError]; }, ); } function transformStreamDefaultSinkWriteAlgorithm( stream, chunk, ) { assert(stream[sym.writable][sym.state] === "writable"); const controller = stream[sym.transformStreamController]; if (stream[sym.backpressure]) { const backpressureChangePromise = stream[sym.backpressureChangePromise]; assert(backpressureChangePromise); return backpressureChangePromise.promise.then(() => { const writable = stream[sym.writable]; const state = writable[sym.state]; if (state === "erroring") { throw writable[sym.storedError]; } assert(state === "writable"); return transformStreamDefaultControllerPerformTransform( controller, chunk, ); }); } return transformStreamDefaultControllerPerformTransform(controller, chunk); } function transformStreamDefaultSourcePullAlgorithm( stream, ) { assert(stream[sym.backpressure] === true); assert(stream[sym.backpressureChangePromise] !== undefined); transformStreamSetBackpressure(stream, false); return stream[sym.backpressureChangePromise].promise; } function transformStreamError( stream, e, ) { readableStreamDefaultControllerError( stream[sym.readable][ sym.readableStreamController ], e, ); transformStreamErrorWritableAndUnblockWrite(stream, e); } function transformStreamDefaultControllerTerminate( controller, ) { const stream = controller[sym.controlledTransformStream]; const readableController = stream[sym.readable][ sym.readableStreamController ]; readableStreamDefaultControllerClose(readableController); const error = new TypeError("TransformStream is closed."); transformStreamErrorWritableAndUnblockWrite(stream, error); } function transformStreamErrorWritableAndUnblockWrite( stream, e, ) { transformStreamDefaultControllerClearAlgorithms( stream[sym.transformStreamController], ); writableStreamDefaultControllerErrorIfNeeded( stream[sym.writable][sym.writableStreamController], e, ); if (stream[sym.backpressure]) { transformStreamSetBackpressure(stream, false); } } function transformStreamSetBackpressure( stream, backpressure, ) { assert(stream[sym.backpressure] !== backpressure); if (stream[sym.backpressureChangePromise] !== undefined) { stream[sym.backpressureChangePromise].resolve(undefined); } stream[sym.backpressureChangePromise] = getDeferred(); stream[sym.backpressure] = backpressure; } function transferArrayBuffer(buffer) { assert(!isDetachedBuffer(buffer)); const transferredIshVersion = buffer.slice(0); Object.defineProperty(buffer, "byteLength", { get() { return 0; }, }); buffer[sym.isFakeDetached] = true; return transferredIshVersion; } function validateAndNormalizeHighWaterMark( highWaterMark, ) { highWaterMark = Number(highWaterMark); if (Number.isNaN(highWaterMark) || highWaterMark < 0) { throw new RangeError( `highWaterMark must be a positive number or Infinity. Received: ${highWaterMark}.`, ); } return highWaterMark; } function writableStreamAbort( stream, reason, ) { const state = stream[sym.state]; if (state === "closed" || state === "errored") { return Promise.resolve(undefined); } if (stream[sym.pendingAbortRequest]) { return stream[sym.pendingAbortRequest].promise.promise; } assert(state === "writable" || state === "erroring"); let wasAlreadyErroring = false; if (state === "erroring") { wasAlreadyErroring = true; reason = undefined; } const promise = getDeferred(); stream[sym.pendingAbortRequest] = { promise, reason, wasAlreadyErroring }; if (wasAlreadyErroring === false) { writableStreamStartErroring(stream, reason); } return promise.promise; } function writableStreamAddWriteRequest( stream, ) { assert(isWritableStream(stream)); assert(stream[sym.state] === "writable"); const promise = getDeferred(); stream[sym.writeRequests].push(promise); return promise.promise; } function writableStreamClose( stream, ) { const state = stream[sym.state]; if (state === "closed" || state === "errored") { return Promise.reject( new TypeError( "Cannot close an already closed or errored WritableStream.", ), ); } assert(!writableStreamCloseQueuedOrInFlight(stream)); const promise = getDeferred(); stream[sym.closeRequest] = promise; const writer = stream[sym.writer]; if (writer && stream[sym.backpressure] && state === "writable") { writer[sym.readyPromise].resolve(); writer[sym.readyPromise].resolve = undefined; writer[sym.readyPromise].reject = undefined; } writableStreamDefaultControllerClose(stream[sym.writableStreamController]); return promise.promise; } function writableStreamCloseQueuedOrInFlight( stream, ) { return !( stream[sym.closeRequest] === undefined && stream[sym.inFlightCloseRequest] === undefined ); } function writableStreamDealWithRejection( stream, error, ) { const state = stream[sym.state]; if (state === "writable") { writableStreamStartErroring(stream, error); return; } assert(state === "erroring"); writableStreamFinishErroring(stream); } function writableStreamDefaultControllerAdvanceQueueIfNeeded( controller, ) { const stream = controller[sym.controlledWritableStream]; if (!controller[sym.started]) { return; } if (stream[sym.inFlightWriteRequest]) { return; } const state = stream[sym.state]; assert(state !== "closed" && state !== "errored"); if (state === "erroring") { writableStreamFinishErroring(stream); return; } if (!controller[sym.queue].length) { return; } const writeRecord = peekQueueValue(controller); if (writeRecord === "close") { writableStreamDefaultControllerProcessClose(controller); } else { writableStreamDefaultControllerProcessWrite( controller, writeRecord.chunk, ); } } function writableStreamDefaultControllerClearAlgorithms( controller, ) { controller[sym.writeAlgorithm] = undefined; controller[sym.closeAlgorithm] = undefined; controller[sym.abortAlgorithm] = undefined; controller[sym.strategySizeAlgorithm] = undefined; } function writableStreamDefaultControllerClose( controller, ) { enqueueValueWithSize(controller, "close", 0); writableStreamDefaultControllerAdvanceQueueIfNeeded(controller); } function writableStreamDefaultControllerError( controller, error, ) { const stream = controller[sym.controlledWritableStream]; assert(stream[sym.state] === "writable"); writableStreamDefaultControllerClearAlgorithms(controller); writableStreamStartErroring(stream, error); } function writableStreamDefaultControllerErrorIfNeeded( controller, error, ) { if (controller[sym.controlledWritableStream][sym.state] === "writable") { writableStreamDefaultControllerError(controller, error); } } function writableStreamDefaultControllerGetBackpressure( controller, ) { const desiredSize = writableStreamDefaultControllerGetDesiredSize( controller, ); return desiredSize <= 0; } function writableStreamDefaultControllerGetChunkSize( controller, chunk, ) { let returnValue; try { returnValue = controller[sym.strategySizeAlgorithm](chunk); } catch (e) { writableStreamDefaultControllerErrorIfNeeded(controller, e); return 1; } return returnValue; } function writableStreamDefaultControllerGetDesiredSize( controller, ) { return controller[sym.strategyHWM] - controller[sym.queueTotalSize]; } function writableStreamDefaultControllerProcessClose( controller, ) { const stream = controller[sym.controlledWritableStream]; writableStreamMarkCloseRequestInFlight(stream); dequeueValue(controller); assert(controller[sym.queue].length === 0); const sinkClosePromise = controller[sym.closeAlgorithm](); writableStreamDefaultControllerClearAlgorithms(controller); setPromiseIsHandledToTrue( sinkClosePromise.then( () => { writableStreamFinishInFlightClose(stream); }, (reason) => { writableStreamFinishInFlightCloseWithError(stream, reason); }, ), ); } function writableStreamDefaultControllerProcessWrite( controller, chunk, ) { const stream = controller[sym.controlledWritableStream]; writableStreamMarkFirstWriteRequestInFlight(stream); const sinkWritePromise = controller[sym.writeAlgorithm](chunk); setPromiseIsHandledToTrue( sinkWritePromise.then( () => { writableStreamFinishInFlightWrite(stream); const state = stream[sym.state]; assert(state === "writable" || state === "erroring"); dequeueValue(controller); if ( !writableStreamCloseQueuedOrInFlight(stream) && state === "writable" ) { const backpressure = writableStreamDefaultControllerGetBackpressure( controller, ); writableStreamUpdateBackpressure(stream, backpressure); } writableStreamDefaultControllerAdvanceQueueIfNeeded(controller); }, (reason) => { if (stream[sym.state] === "writable") { writableStreamDefaultControllerClearAlgorithms(controller); } writableStreamFinishInFlightWriteWithError(stream, reason); }, ), ); } function writableStreamDefaultControllerWrite( controller, chunk, chunkSize, ) { const writeRecord = { chunk }; try { enqueueValueWithSize(controller, writeRecord, chunkSize); } catch (e) { writableStreamDefaultControllerErrorIfNeeded(controller, e); return; } const stream = controller[sym.controlledWritableStream]; if ( !writableStreamCloseQueuedOrInFlight(stream) && stream[sym.state] === "writable" ) { const backpressure = writableStreamDefaultControllerGetBackpressure( controller, ); writableStreamUpdateBackpressure(stream, backpressure); } writableStreamDefaultControllerAdvanceQueueIfNeeded(controller); } function writableStreamDefaultWriterAbort( writer, reason, ) { const stream = writer[sym.ownerWritableStream]; assert(stream); return writableStreamAbort(stream, reason); } function writableStreamDefaultWriterClose( writer, ) { const stream = writer[sym.ownerWritableStream]; assert(stream); return writableStreamClose(stream); } function writableStreamDefaultWriterCloseWithErrorPropagation( writer, ) { const stream = writer[sym.ownerWritableStream]; assert(stream); const state = stream[sym.state]; if (writableStreamCloseQueuedOrInFlight(stream) || state === "closed") { return Promise.resolve(); } if (state === "errored") { return Promise.reject(stream[sym.storedError]); } assert(state === "writable" || state === "erroring"); return writableStreamDefaultWriterClose(writer); } function writableStreamDefaultWriterEnsureClosePromiseRejected( writer, error, ) { if (writer[sym.closedPromise].reject) { writer[sym.closedPromise].reject(error); } else { writer[sym.closedPromise] = { promise: Promise.reject(error), }; } setPromiseIsHandledToTrue(writer[sym.closedPromise].promise); } function writableStreamDefaultWriterEnsureReadyPromiseRejected( writer, error, ) { if (writer[sym.readyPromise].reject) { writer[sym.readyPromise].reject(error); writer[sym.readyPromise].reject = undefined; writer[sym.readyPromise].resolve = undefined; } else { writer[sym.readyPromise] = { promise: Promise.reject(error), }; } setPromiseIsHandledToTrue(writer[sym.readyPromise].promise); } function writableStreamDefaultWriterWrite( writer, chunk, ) { const stream = writer[sym.ownerWritableStream]; assert(stream); const controller = stream[sym.writableStreamController]; assert(controller); const chunkSize = writableStreamDefaultControllerGetChunkSize( controller, chunk, ); if (stream !== writer[sym.ownerWritableStream]) { return Promise.reject("Writer has incorrect WritableStream."); } const state = stream[sym.state]; if (state === "errored") { return Promise.reject(stream[sym.storedError]); } if (writableStreamCloseQueuedOrInFlight(stream) || state === "closed") { return Promise.reject(new TypeError("The stream is closed or closing.")); } if (state === "erroring") { return Promise.reject(stream[sym.storedError]); } assert(state === "writable"); const promise = writableStreamAddWriteRequest(stream); writableStreamDefaultControllerWrite(controller, chunk, chunkSize); return promise; } function writableStreamDefaultWriterGetDesiredSize( writer, ) { const stream = writer[sym.ownerWritableStream]; const state = stream[sym.state]; if (state === "errored" || state === "erroring") { return null; } if (state === "closed") { return 0; } return writableStreamDefaultControllerGetDesiredSize( stream[sym.writableStreamController], ); } function writableStreamDefaultWriterRelease( writer, ) { const stream = writer[sym.ownerWritableStream]; assert(stream); assert(stream[sym.writer] === writer); const releasedError = new TypeError( "Writer was released and can no longer be used to monitor the stream's closedness.", ); writableStreamDefaultWriterEnsureReadyPromiseRejected( writer, releasedError, ); writableStreamDefaultWriterEnsureClosePromiseRejected( writer, releasedError, ); stream[sym.writer] = undefined; writer[sym.ownerWritableStream] = undefined; } function writableStreamFinishErroring(stream) { assert(stream[sym.state] === "erroring"); assert(!writableStreamHasOperationMarkedInFlight(stream)); stream[sym.state] = "errored"; stream[sym.writableStreamController][sym.errorSteps](); const storedError = stream[sym.storedError]; for (const writeRequest of stream[sym.writeRequests]) { assert(writeRequest.reject); writeRequest.reject(storedError); } stream[sym.writeRequests] = []; if (!stream[sym.pendingAbortRequest]) { writableStreamRejectCloseAndClosedPromiseIfNeeded(stream); return; } const abortRequest = stream[sym.pendingAbortRequest]; assert(abortRequest); stream[sym.pendingAbortRequest] = undefined; if (abortRequest.wasAlreadyErroring) { assert(abortRequest.promise.reject); abortRequest.promise.reject(storedError); writableStreamRejectCloseAndClosedPromiseIfNeeded(stream); return; } const promise = stream[sym.writableStreamController][sym.abortSteps]( abortRequest.reason, ); setPromiseIsHandledToTrue( promise.then( () => { assert(abortRequest.promise.resolve); abortRequest.promise.resolve(); writableStreamRejectCloseAndClosedPromiseIfNeeded(stream); }, (reason) => { assert(abortRequest.promise.reject); abortRequest.promise.reject(reason); writableStreamRejectCloseAndClosedPromiseIfNeeded(stream); }, ), ); } function writableStreamFinishInFlightClose( stream, ) { assert(stream[sym.inFlightCloseRequest]); stream[sym.inFlightCloseRequest]?.resolve(); stream[sym.inFlightCloseRequest] = undefined; const state = stream[sym.state]; assert(state === "writable" || state === "erroring"); if (state === "erroring") { stream[sym.storedError] = undefined; if (stream[sym.pendingAbortRequest]) { stream[sym.pendingAbortRequest].promise.resolve(); stream[sym.pendingAbortRequest] = undefined; } } stream[sym.state] = "closed"; const writer = stream[sym.writer]; if (writer) { writer[sym.closedPromise].resolve(); } assert(stream[sym.pendingAbortRequest] === undefined); assert(stream[sym.storedError] === undefined); } function writableStreamFinishInFlightCloseWithError( stream, error, ) { assert(stream[sym.inFlightCloseRequest]); stream[sym.inFlightCloseRequest]?.reject(error); stream[sym.inFlightCloseRequest] = undefined; assert( stream[sym.state] === "writable" || stream[sym.state] === "erroring", ); if (stream[sym.pendingAbortRequest]) { stream[sym.pendingAbortRequest]?.promise.reject(error); stream[sym.pendingAbortRequest] = undefined; } writableStreamDealWithRejection(stream, error); } function writableStreamFinishInFlightWrite( stream, ) { assert(stream[sym.inFlightWriteRequest]); stream[sym.inFlightWriteRequest].resolve(); stream[sym.inFlightWriteRequest] = undefined; } function writableStreamFinishInFlightWriteWithError( stream, error, ) { assert(stream[sym.inFlightWriteRequest]); stream[sym.inFlightWriteRequest].reject(error); stream[sym.inFlightWriteRequest] = undefined; assert( stream[sym.state] === "writable" || stream[sym.state] === "erroring", ); writableStreamDealWithRejection(stream, error); } function writableStreamHasOperationMarkedInFlight( stream, ) { return !( stream[sym.inFlightWriteRequest] === undefined && stream[sym.inFlightCloseRequest] === undefined ); } function writableStreamMarkCloseRequestInFlight( stream, ) { assert(stream[sym.inFlightCloseRequest] === undefined); assert(stream[sym.closeRequest] !== undefined); stream[sym.inFlightCloseRequest] = stream[sym.closeRequest]; stream[sym.closeRequest] = undefined; } function writableStreamMarkFirstWriteRequestInFlight( stream, ) { assert(stream[sym.inFlightWriteRequest] === undefined); assert(stream[sym.writeRequests].length); const writeRequest = stream[sym.writeRequests].shift(); stream[sym.inFlightWriteRequest] = writeRequest; } function writableStreamRejectCloseAndClosedPromiseIfNeeded( stream, ) { assert(stream[sym.state] === "errored"); if (stream[sym.closeRequest]) { assert(stream[sym.inFlightCloseRequest] === undefined); stream[sym.closeRequest].reject(stream[sym.storedError]); stream[sym.closeRequest] = undefined; } const writer = stream[sym.writer]; if (writer) { writer[sym.closedPromise].reject(stream[sym.storedError]); setPromiseIsHandledToTrue(writer[sym.closedPromise].promise); } } function writableStreamStartErroring( stream, reason, ) { assert(stream[sym.storedError] === undefined); assert(stream[sym.state] === "writable"); const controller = stream[sym.writableStreamController]; assert(controller); stream[sym.state] = "erroring"; stream[sym.storedError] = reason; const writer = stream[sym.writer]; if (writer) { writableStreamDefaultWriterEnsureReadyPromiseRejected(writer, reason); } if ( !writableStreamHasOperationMarkedInFlight(stream) && controller[sym.started] ) { writableStreamFinishErroring(stream); } } function writableStreamUpdateBackpressure( stream, backpressure, ) { assert(stream[sym.state] === "writable"); assert(!writableStreamCloseQueuedOrInFlight(stream)); const writer = stream[sym.writer]; if (writer && backpressure !== stream[sym.backpressure]) { if (backpressure) { writer[sym.readyPromise] = getDeferred(); } else { assert(backpressure === false); writer[sym.readyPromise].resolve(); writer[sym.readyPromise].resolve = undefined; writer[sym.readyPromise].reject = undefined; } } stream[sym.backpressure] = backpressure; } /* eslint-enable */ window.__bootstrap.streams = { ReadableStream, TransformStream, WritableStream, isReadableStreamDisturbed, }; })(this);