// Copyright 2018-2021 the Deno authors. All rights reserved. MIT license. // @ts-check /// /// /// /// /// /// "use strict"; ((window) => { const core = Deno.core; const webidl = window.__bootstrap.webidl; class TextDecoder { /** @type {string} */ #encoding; /** @type {boolean} */ #fatal; /** @type {boolean} */ #ignoreBOM; /** @type {number | null} */ #rid = null; /** * * @param {string} label * @param {TextDecoderOptions} options */ constructor(label = "utf-8", options = {}) { const prefix = "Failed to construct 'TextDecoder'"; label = webidl.converters.DOMString(label, { prefix, context: "Argument 1", }); options = webidl.converters.TextDecoderOptions(options, { prefix, context: "Argument 2", }); const encoding = core.opSync("op_encoding_normalize_label", label); this.#encoding = encoding; this.#fatal = options.fatal; this.#ignoreBOM = options.ignoreBOM; this[webidl.brand] = webidl.brand; } /** @returns {string} */ get encoding() { webidl.assertBranded(this, TextDecoder); return this.#encoding; } /** @returns {boolean} */ get fatal() { webidl.assertBranded(this, TextDecoder); return this.#fatal; } /** @returns {boolean} */ get ignoreBOM() { webidl.assertBranded(this, TextDecoder); return this.#ignoreBOM; } /** * @param {BufferSource} [input] * @param {TextDecodeOptions} options */ decode(input = new Uint8Array(), options = {}) { webidl.assertBranded(this, TextDecoder); const prefix = "Failed to execute 'decode' on 'TextDecoder'"; if (input !== undefined) { input = webidl.converters.BufferSource(input, { prefix, context: "Argument 1", allowShared: true, }); } options = webidl.converters.TextDecodeOptions(options, { prefix, context: "Argument 2", }); // TODO(lucacasonato): add fast path for non-streaming decoder & decode if (this.#rid === null) { this.#rid = core.opSync("op_encoding_new_decoder", { label: this.#encoding, fatal: this.#fatal, ignoreBom: this.#ignoreBOM, }); } try { if (ArrayBuffer.isView(input)) { input = new Uint8Array( input.buffer, input.byteOffset, input.byteLength, ); } else { input = new Uint8Array(input); } return core.opSync("op_encoding_decode", new Uint8Array(input), { rid: this.#rid, stream: options.stream, }); } finally { if (!options.stream) { core.close(this.#rid); this.#rid = null; } } } get [Symbol.toStringTag]() { return "TextDecoder"; } } Object.defineProperty(TextDecoder.prototype, "encoding", { enumerable: true, configurable: true, }); Object.defineProperty(TextDecoder.prototype, "fatal", { enumerable: true, configurable: true, }); Object.defineProperty(TextDecoder.prototype, "ignoreBOM", { enumerable: true, configurable: true, }); Object.defineProperty(TextDecoder.prototype, "decode", { enumerable: true, writable: true, configurable: true, }); class TextEncoder { constructor() { this[webidl.brand] = webidl.brand; } /** @returns {string} */ get encoding() { webidl.assertBranded(this, TextEncoder); return "utf-8"; } /** * @param {string} input * @returns {Uint8Array} */ encode(input = "") { webidl.assertBranded(this, TextEncoder); const prefix = "Failed to execute 'encode' on 'TextEncoder'"; input = webidl.converters.USVString(input, { prefix, context: "Argument 1", }); return core.encode(input); } /** * @param {string} source * @param {Uint8Array} destination * @returns {TextEncoderEncodeIntoResult} */ encodeInto(source, destination) { webidl.assertBranded(this, TextEncoder); const prefix = "Failed to execute 'encodeInto' on 'TextEncoder'"; source = webidl.converters.USVString(source, { prefix, context: "Argument 1", }); destination = webidl.converters.Uint8Array(destination, { prefix, context: "Argument 2", allowShared: true, }); return core.opSync("op_encoding_encode_into", source, destination); } get [Symbol.toStringTag]() { return "TextEncoder"; } } Object.defineProperty(TextEncoder.prototype, "encoding", { enumerable: true, configurable: true, }); Object.defineProperty(TextEncoder.prototype, "encode", { enumerable: true, writable: true, configurable: true, }); Object.defineProperty(TextEncoder.prototype, "encodeInto", { enumerable: true, writable: true, configurable: true, }); class TextDecoderStream { /** @type {TextDecoder} */ #decoder; /** @type {TransformStream} */ #transform; /** * * @param {string} label * @param {TextDecoderOptions} options */ constructor(label = "utf-8", options = {}) { const prefix = "Failed to construct 'TextDecoderStream'"; label = webidl.converters.DOMString(label, { prefix, context: "Argument 1", }); options = webidl.converters.TextDecoderOptions(options, { prefix, context: "Argument 2", }); this.#decoder = new TextDecoder(label, options); this.#transform = new TransformStream({ // The transform and flush functions need access to TextDecoderStream's // `this`, so they are defined as functions rather than methods. transform: (chunk, controller) => { try { chunk = webidl.converters.BufferSource(chunk, { allowShared: true, }); const decoded = this.#decoder.decode(chunk, { stream: true }); if (decoded) { controller.enqueue(decoded); } return Promise.resolve(); } catch (err) { return Promise.reject(err); } }, flush: (controller) => { try { const final = this.#decoder.decode(); if (final) { controller.enqueue(final); } return Promise.resolve(); } catch (err) { return Promise.reject(err); } }, }); this[webidl.brand] = webidl.brand; } /** @returns {string} */ get encoding() { webidl.assertBranded(this, TextDecoderStream); return this.#decoder.encoding; } /** @returns {boolean} */ get fatal() { webidl.assertBranded(this, TextDecoderStream); return this.#decoder.fatal; } /** @returns {boolean} */ get ignoreBOM() { webidl.assertBranded(this, TextDecoderStream); return this.#decoder.ignoreBOM; } /** @returns {ReadableStream} */ get readable() { webidl.assertBranded(this, TextDecoderStream); return this.#transform.readable; } /** @returns {WritableStream} */ get writable() { webidl.assertBranded(this, TextDecoderStream); return this.#transform.writable; } get [Symbol.toStringTag]() { return "TextDecoderStream"; } } Object.defineProperty(TextDecoderStream.prototype, "encoding", { enumerable: true, configurable: true, }); Object.defineProperty(TextDecoderStream.prototype, "fatal", { enumerable: true, configurable: true, }); Object.defineProperty(TextDecoderStream.prototype, "ignoreBOM", { enumerable: true, configurable: true, }); Object.defineProperty(TextDecoderStream.prototype, "readable", { enumerable: true, configurable: true, }); Object.defineProperty(TextDecoderStream.prototype, "writable", { enumerable: true, configurable: true, }); class TextEncoderStream { /** @type {string | null} */ #pendingHighSurrogate = null; /** @type {TransformStream} */ #transform; constructor() { this.#transform = new TransformStream({ // The transform and flush functions need access to TextEncoderStream's // `this`, so they are defined as functions rather than methods. transform: (chunk, controller) => { try { chunk = webidl.converters.DOMString(chunk); if (this.#pendingHighSurrogate !== null) { chunk = this.#pendingHighSurrogate + chunk; } const lastCodeUnit = chunk.charCodeAt(chunk.length - 1); if (0xD800 <= lastCodeUnit && lastCodeUnit <= 0xDBFF) { this.#pendingHighSurrogate = chunk.slice(-1); chunk = chunk.slice(0, -1); } else { this.#pendingHighSurrogate = null; } if (chunk) { controller.enqueue(core.encode(chunk)); } return Promise.resolve(); } catch (err) { return Promise.reject(err); } }, flush: (controller) => { try { if (this.#pendingHighSurrogate !== null) { controller.enqueue(new Uint8Array([0xEF, 0xBF, 0xBD])); } return Promise.resolve(); } catch (err) { return Promise.reject(err); } }, }); this[webidl.brand] = webidl.brand; } /** @returns {string} */ get encoding() { webidl.assertBranded(this, TextEncoderStream); return "utf-8"; } /** @returns {ReadableStream} */ get readable() { webidl.assertBranded(this, TextEncoderStream); return this.#transform.readable; } /** @returns {WritableStream} */ get writable() { webidl.assertBranded(this, TextEncoderStream); return this.#transform.writable; } get [Symbol.toStringTag]() { return "TextEncoderStream"; } } Object.defineProperty(TextEncoderStream.prototype, "encoding", { enumerable: true, configurable: true, }); Object.defineProperty(TextEncoderStream.prototype, "readable", { enumerable: true, configurable: true, }); Object.defineProperty(TextEncoderStream.prototype, "writable", { enumerable: true, configurable: true, }); webidl.converters.TextDecoderOptions = webidl.createDictionaryConverter( "TextDecoderOptions", [ { key: "fatal", converter: webidl.converters.boolean, defaultValue: false, }, { key: "ignoreBOM", converter: webidl.converters.boolean, defaultValue: false, }, ], ); webidl.converters.TextDecodeOptions = webidl.createDictionaryConverter( "TextDecodeOptions", [ { key: "stream", converter: webidl.converters.boolean, defaultValue: false, }, ], ); /** * @param {Uint8Array} bytes */ function decode(bytes, encoding) { const BOMEncoding = BOMSniff(bytes); let start = 0; if (BOMEncoding !== null) { encoding = BOMEncoding; if (BOMEncoding === "UTF-8") start = 3; else start = 2; } return new TextDecoder(encoding).decode(bytes.slice(start)); } /** * @param {Uint8Array} bytes */ function BOMSniff(bytes) { const BOM = bytes.subarray(0, 3); if (BOM[0] === 0xEF && BOM[1] === 0xBB && BOM[2] === 0xBF) { return "UTF-8"; } if (BOM[0] === 0xFE && BOM[1] === 0xFF) return "UTF-16BE"; if (BOM[0] === 0xFF && BOM[1] === 0xFE) return "UTF-16LE"; return null; } window.__bootstrap.encoding = { TextEncoder, TextDecoder, TextEncoderStream, TextDecoderStream, decode, }; })(this);