deno.land / x / hono@v4.2.5 / utils / stream.ts
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172export class StreamingApi { private writer: WritableStreamDefaultWriter<Uint8Array> private encoder: TextEncoder private writable: WritableStream private abortSubscribers: (() => void | Promise<void>)[] = [] responseReadable: ReadableStream
constructor(writable: WritableStream, _readable: ReadableStream) { this.writable = writable this.writer = writable.getWriter() this.encoder = new TextEncoder()
const reader = _readable.getReader()
// in case the user disconnects, let the reader know to cancel // this in-turn results in responseReadable being closed // and writeSSE method no longer blocks indefinitely this.abortSubscribers.push(async () => { await reader.cancel() })
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() }
onAbort(listener: () => void | Promise<void>) { this.abortSubscribers.push(listener) }}
Version Info