UNPKG

@whatwg-node/node-fetch

Version:

Fetch API implementation for Node

148 lines (147 loc) 4.74 kB
import { Writable } from 'node:stream'; import { fakePromise } from './utils.js'; export class PonyfillWritableStream { writable; constructor(underlyingSink) { if (underlyingSink instanceof Writable) { this.writable = underlyingSink; } else if (underlyingSink) { const writable = new Writable({ write(chunk, _encoding, callback) { try { const result = underlyingSink.write?.(chunk, controller); if (result instanceof Promise) { result.then(() => { callback(); }, err => { callback(err); }); } else { callback(); } } catch (err) { callback(err); } }, final(callback) { const result = underlyingSink.close?.(); if (result instanceof Promise) { result.then(() => { callback(); }, err => { callback(err); }); } else { callback(); } }, }); this.writable = writable; const abortCtrl = new AbortController(); const controller = { signal: abortCtrl.signal, error(e) { writable.destroy(e); }, }; writable.once('error', err => abortCtrl.abort(err)); writable.once('close', () => abortCtrl.abort()); } else { this.writable = new Writable(); } } getWriter() { const writable = this.writable; return { closed: new Promise(resolve => { writable.once('close', () => { resolve(undefined); }); }), get desiredSize() { return writable.writableLength; }, ready: new Promise(resolve => { writable.once('drain', () => { resolve(undefined); }); }), releaseLock() { // no-op }, write(chunk) { if (chunk == null) { return fakePromise(); } return new Promise((resolve, reject) => { writable.write(chunk, (err) => { if (err) { reject(err); } else { resolve(); } }); }); }, close() { if (!writable.errored && writable.closed) { return fakePromise(); } return new Promise((resolve, reject) => { if (writable.errored) { reject(writable.errored); } else { writable.end((err) => { if (err) { reject(err); } else { resolve(); } }); } }); }, abort(reason) { return new Promise(resolve => { writable.destroy(reason); writable.once('close', resolve); }); }, }; } close() { if (!this.writable.errored && this.writable.closed) { return fakePromise(); } return new Promise((resolve, reject) => { if (this.writable.errored) { reject(this.writable.errored); } else { this.writable.end((err) => { if (err) { reject(err); } else { resolve(); } }); } }); } abort(reason) { return new Promise(resolve => { this.writable.destroy(reason); this.writable.once('close', resolve); }); } locked = false; }