const item = chunk[i]; const codeUnit = StringPrototypeCharCodeAt(item, 0); if (this[kPendingHighSurrogate] !== null) { const highSurrogate = this[kPendingHighSurrogate]; this[kPendingHighSurrogate] = null; if (0xDC00 <= codeUnit && codeUnit <= 0xDFFF) { finalChunk += highSurrogate + item; continue; } finalChunk += '\uFFFD'; } if (0xD800 <= codeUnit && codeUnit <= 0xDBFF) { this[kPendingHighSurrogate] = item; continue; } if (0xDC00 <= codeUnit && codeUnit <= 0xDFFF) { finalChunk += '\uFFFD'; continue; } finalChunk += item; } if (finalChunk) { const value = this[kHandle].encode(finalChunk); controller.enqueue(value); } }, flush: (controller) => { // https://encoding.spec.whatwg.org/#encode-and-flush if (this[kPendingHighSurrogate] !== null) { controller.enqueue(new Uint8Array([0xEF, 0xBF, 0xBD])); } }, }); } /** * @readonly * @type {string} */ get encoding() { if (!isTextEncoderStream(this)) throw new ERR_INVALID_THIS('TextEncoderStream'); return this[kHandle].encoding; } /** * @readonly * @type {ReadableStream} */ get readable() { if (!isTextEncoderStream(this)) throw new ERR_INVALID_THIS('TextEncoderStream'); return this[kTransform].readable; } /** * @readonly * @type {WritableStream} */ get writable() { if (!isTextEncoderStream(this)) throw new ERR_INVALID_THIS('TextEncoderStream'); return this[kTransform].writable; } [kInspect](depth, options) { if (!isTextEncoderStream(this)) throw new ERR_INVALID_THIS('TextEncoderStream'); return customInspect(depth, options, 'TextEncoderStream', { encoding: this[kHandle].encoding, readable: this[kTransform].readable, writable: this[kTransform].writable, }); } } class TextDecoderStream { /** * @param {string} [encoding] * @param {{ * fatal? : boolean, * ignoreBOM? : boolean, * }} [options] */ constructor(encoding = 'utf-8', options = kEmptyObject) { this[kType] = 'TextDecoderStream'; this[kHandle] = new TextDecoder(encoding, options); this[kTransform] = new TransformStream({ transform: (chunk, controller) => { const value = this[kHandle].decode(chunk, { stream: true }); if (value) controller.enqueue(value); }, flush: (controller) => { const value = this[kHandle].decode(); if (value) controller.enqueue(value); controller.terminate(); }, }); } /** * @readonly * @type {string} */ get encoding() { if (!isTextDecoderStream(this)) throw new ERR_INVALID_THIS('TextDecoderStream'); return this[kHandle].encoding; } /** * @readonly * @type {boolean} */ get fatal() { if (!isTextDecoderStream(this)) throw new ERR_INVALID_THIS('TextDecoderStream'); return this[kHandle].fatal; } /** * @readonly * @type {boolean} */ get ignoreBOM() { if (!isTextDecoderStream(this)) throw new ERR_INVALID_THIS('TextDecoderStream'); return this[kHandle].ignoreBOM; } /** * @readonly * @type {ReadableStream} */ get readable() { if (!isTextDecoderStream(this)) throw new ERR_INVALID_THIS('TextDecoderStream'); return this[kTransform].readable; } /** * @readonly * @type {WritableStream} */ get writable() { if (!isTextDecoderStream(this)) throw new ERR_INVALID_THIS('TextDecoderStream'); return this[kTransform].writable; } [kInspect](depth, options) { if (!isTextDecoderStream(this)) throw new ERR_INVALID_THIS('TextDecoderStream'); return customInspect(depth, options, 'TextDecoderStream', { encoding: this[kHandle].encoding, fatal: this[kHandle].fatal, ignoreBOM: this[kHandle].ignoreBOM, readable: this[kTransform].readable, writable: this[kTransform].writable, }); } } ObjectDefineProperties(TextEncoderStream.prototype, { encoding: kEnumerableProperty, readable: kEnumerableProperty, writable: kEnumerableProperty, }); ObjectDefineProperties(TextDecoderStream.prototype, { encoding: kEnumerableProperty, fatal: kEnumerableProperty, ignoreBOM: kEnumerableProperty, readable: kEnumerableProperty, writable: kEnumerableProperty, }); module.exports = { TextEncoderStream, TextDecoderStream, };