export class StreamingApi { private writer: WritableStreamDefaultWriter private encoder: TextEncoder private writable: WritableStream private abortSubscribers: (() => void | Promise)[] = [] responseReadable: ReadableStream constructor(writable: WritableStream, _readable: ReadableStream) { this.writable = writable this.writer = writable.getWriter() this.encoder = new TextEncoder() const reader = _readable.getReader() this.responseReadable = new ReadableStream({ async pull(controller) { const { done, value } = await reader.read() done ? controller.close() : controller.enqueue(value) }, cancel: () => { this.abortSubscribers.forEach((subscriber) => subscriber()) }, }) } async write(input: Uint8Array | string) { try { if (typeof input === 'string') { input = this.encoder.encode(input) } await this.writer.write(input) } catch (e) { // Do nothing. If you want to handle errors, create a stream by yourself. } return this } async writeln(input: string) { await this.write(input + '\n') return this } sleep(ms: number) { return new Promise((res) => setTimeout(res, ms)) } async close() { try { await this.writer.close() } catch (e) { // Do nothing. If you want to handle errors, create a stream by yourself. } } async pipe(body: ReadableStream) { this.writer.releaseLock() await body.pipeTo(this.writable, { preventClose: true }) this.writer = this.writable.getWriter() } async onAbort(listener: () => void | Promise) { this.abortSubscribers.push(listener) } }