"use strict"; // Copyright (c) Holly Stubbs (tgpholly) - Licensed under MIT // Check LICENSE in repository root for more information. // This is a mostly 1:1 feature complete implementation of node's Buffer class for browsers. // It's missing some of the static stuff but everything in the object *should* be there. class BrowserBuffer { constructor(dataOrSize) { if (typeof (dataOrSize) === "number") { this.buffer = new Uint8Array(dataOrSize); } else if (typeof (dataOrSize) === "object") { this.buffer = new Uint8Array(dataOrSize); // Convert whatever comes in to a Uint8Array. } else { this.buffer = new Uint8Array(0); // Fallback } this.dataView = new DataView(this.buffer.buffer); } static allocUnsafe(size) { return this.alloc(size); } static allocUnsafeSlow(size) { return this.alloc(size); } static alloc(size) { return new BrowserBuffer(size); } static concat(buffers, totalLength) { let joinedArrays; if (totalLength !== undefined) { joinedArrays = new Uint8Array(totalLength); } else { let arraysLength = 0; for (const buffer of buffers) { arraysLength += buffer.length; } joinedArrays = new Uint8Array(arraysLength); } let offset = 0; for (const buffer of buffers) { joinedArrays.set(buffer.buffer, offset); offset += buffer.length; } return new BrowserBuffer(joinedArrays); } static from(data) { if (typeof (data) === "string") { throw new Error("BrowserBuffer does not currently support creating buffers from strings."); } return new BrowserBuffer(data); } // TODO: Implement static of() { } static isBuffer() { } static isEncoding() { } static byteLength() { } static copyBytesFrom() { } static compare() { } get length() { return this.buffer.length; } checkRanged(value, valueName, lowRange, highRange) { if (value < lowRange || value > highRange) { throw new Error(`The value of "${valueName}" is out of range. It must be >= ${lowRange} and <= ${highRange}. Received ${value}`); } } checkValue(value, low, high) { this.checkRanged(value, "value", low, high); } checkOffset(offset) { if (offset) { this.checkRanged(offset, "offset", 0, this.buffer.length - 1); } } // Writing methods writeInt8(value, offset) { this.checkValue(value, -128, 127); this.checkOffset(offset); this.dataView.setInt8(offset, value); return this; } writeUInt8(value, offset) { this.checkValue(value, 0, 255); this.checkOffset(offset); this.dataView.setUint8(offset, value); return this; } // Little Endian writeInt16LE(value, offset) { this.checkValue(value, -32768, 32767); this.checkOffset(offset); this.dataView.setInt16(offset, value, true); return this; } writeUInt16LE(value, offset) { this.checkValue(value, 0, 65535); this.checkOffset(offset); this.dataView.setUint16(offset, value, true); return this; } writeInt32LE(value, offset) { this.checkValue(value, -2147483648, 2147483647); this.checkOffset(offset); this.dataView.setInt32(offset, value, true); return this; } writeUInt32LE(value, offset) { this.checkValue(value, 0, 4294967295); this.checkOffset(offset); this.dataView.setUint32(offset, value, true); return this; } writeBigInt64LE(value, offset) { if (typeof (value) === "number") { value = BigInt(value); } this.checkValue(value, -(2n ** 63n), 2n ** 63n); this.checkOffset(offset); this.dataView.setBigInt64(offset, value, true); return this; } writeBigUint64LE(value, offset) { if (typeof (value) === "number") { value = BigInt(value); } this.checkValue(value, 0n, 2n ** 64n); this.checkOffset(offset); this.dataView.setBigUint64(offset, value, true); return this; } writeFloatLE(value, offset) { this.checkOffset(offset); this.dataView.setFloat32(offset, value, true); return this; } writeDoubleLE(value, offset) { this.checkOffset(offset); this.dataView.setFloat64(offset, value, true); return this; } // Big Endian writeInt16BE(value, offset) { this.checkValue(value, -32768, 32767); this.checkOffset(offset); this.dataView.setInt16(offset, value, false); return this; } writeUInt16BE(value, offset) { this.checkValue(value, 0, 65535); this.checkOffset(offset); this.dataView.setUint16(offset, value, false); return this; } writeInt32BE(value, offset) { this.checkValue(value, -2147483648, 2147483647); this.checkOffset(offset); this.dataView.setInt32(offset, value, false); return this; } writeUInt32BE(value, offset) { this.checkValue(value, 0, 4294967295); this.checkOffset(offset); this.dataView.setUint32(offset, value, false); return this; } writeBigInt64BE(value, offset) { if (typeof (value) === "number") { value = BigInt(value); } this.checkValue(value, -(2n ** 63n), 2n ** 63n); this.checkOffset(offset); this.dataView.setBigInt64(offset, value, false); return this; } writeBigUint64BE(value, offset) { if (typeof (value) === "number") { value = BigInt(value); } this.checkValue(value, 0n, 2n ** 64n); this.checkOffset(offset); this.dataView.setBigUint64(offset, value, false); return this; } writeFloatBE(value, offset) { this.checkOffset(offset); this.dataView.setFloat32(offset, value, false); return this; } writeDoubleBE(value, offset) { this.checkOffset(offset); this.dataView.setFloat64(offset, value, false); return this; } // Reading methods readInt8(offset) { this.checkOffset(offset); return this.dataView.getInt8(offset); } readUInt8(offset) { this.checkOffset(offset); return this.dataView.getUint8(offset); } // Little Endian readInt16LE(offset) { this.checkOffset(offset); return this.dataView.getInt16(offset, true); } readUInt16LE(offset) { this.checkOffset(offset); return this.dataView.getUint16(offset, true); } readInt32LE(offset) { this.checkOffset(offset); return this.dataView.getInt32(offset, true); } readUInt32LE(offset) { this.checkOffset(offset); return this.dataView.getUint32(offset, true); } readBigInt64LE(offset) { this.checkOffset(offset); return this.dataView.getBigInt64(offset, true); } readBigUint64LE(offset) { this.checkOffset(offset); return this.dataView.getBigUint64(offset, true); } readFloatLE(offset) { this.checkOffset(offset); return this.dataView.getFloat32(offset, true); } readDoubleLE(offset) { this.checkOffset(offset); return this.dataView.getFloat64(offset, true); } // Big Endian readInt16BE(offset) { this.checkOffset(offset); return this.dataView.getInt16(offset, false); } readUInt16BE(offset) { this.checkOffset(offset); return this.dataView.getUint16(offset, false); } readInt32BE(offset) { this.checkOffset(offset); return this.dataView.getInt32(offset, false); } readUInt32BE(offset) { this.checkOffset(offset); return this.dataView.getUint32(offset, false); } readBigInt64BE(offset) { this.checkOffset(offset); return this.dataView.getBigInt64(offset, false); } readBigUint64BE(offset) { this.checkOffset(offset); return this.dataView.getBigUint64(offset, false); } readFloatBE(offset) { this.checkOffset(offset); return this.dataView.getFloat32(offset, false); } readDoubleBE(offset) { this.checkOffset(offset); return this.dataView.getFloat64(offset, false); } } // NOTE: Here to match node buffer, has no use. BrowserBuffer.poolSize = 8192; function getBufferClass() { if (typeof (Buffer) === "undefined") { // @ts-ignore return BrowserBuffer; } else { return Buffer; } } // Copyright (c) Holly Stubbs (tgpholly) - Licensed under MIT // Check LICENSE in repository root for more information. class ReaderBase { constructor(buffer) { this.buffer = buffer; this.offset = 0; } get readOffset() { return this.offset; } get length() { return this.buffer.length; } readBuffer(bytes) { const value = this.buffer.subarray(this.offset, this.offset + bytes); this.offset += bytes; return value; } // NOTE: This has to be a copy as the subarray is only cropped & offset // Realistically this is what we want anyway. readUint8Array(bytes) { const croppedBuffer = this.readBuffer(bytes); const newArray = new Uint8Array(croppedBuffer.length); for (let i = 0; i < croppedBuffer.length; i++) { newArray[i] = croppedBuffer[i]; } return newArray; } readByte() { const value = this.buffer.readInt8(this.offset); this.offset++; return value; } readUByte() { const value = this.buffer.readUInt8(this.offset); this.offset++; return value; } readBool() { return Boolean(this.readUByte()); } readShortString() { const length = this.readUByte(); let text = ""; for (let i = 0; i < length; i++) { text += String.fromCharCode(this.readUByte()); } return text; } readBytesAsString(bytesToRead) { let text = ""; for (let i = 0; i < bytesToRead; i++) { text += String.fromCharCode(this.readUByte()); } return text; } } // Copyright (c) Holly Stubbs (tgpholly) - Licensed under MIT // Check LICENSE in repository root for more information. class WriterBase { constructor(size = 0) { this.buffer = getBufferClass().alloc(size); this.offset = 0; this.resizable = size === 0; } get writeOffset() { return this.offset; } get length() { return this.buffer.length; } toBuffer() { return this.buffer; } toString() { return this.buffer.toString(); } writeBuffer(buffer) { this.buffer = getBufferClass().concat([this.buffer, buffer], this.buffer.length + buffer.length); return this; } writeUint8Array(array) { this.writeBuffer(getBufferClass().from(array)); return this; } writeByte(value) { if (this.resizable) { const buffer = getBufferClass().alloc(1); buffer.writeInt8(value); this.writeBuffer(buffer); } else { this.buffer.writeInt8(value, this.offset); this.offset++; } return this; } writeUByte(value) { if (this.resizable) { const buffer = getBufferClass().alloc(1); buffer.writeUInt8(value); this.writeBuffer(buffer); } else { this.buffer.writeUInt8(value, this.offset); this.offset++; } return this; } writeBool(value) { if (typeof (value) === "number") { value = Boolean(value); } this.writeUByte(value ? 1 : 0); return this; } writeStringAsBytes(text) { let buffer; if (this.resizable) { buffer = getBufferClass().alloc(text.length); } else { buffer = this.buffer; } for (let i = 0; i < text.length; i++) { buffer.writeUInt8(text.charCodeAt(i), i); } return this; } } // Copyright (c) Holly Stubbs (tgpholly) - Licensed under MIT // Check LICENSE in repository root for more information. var Endian; (function (Endian) { Endian[Endian["LE"] = 0] = "LE"; Endian[Endian["BE"] = 1] = "BE"; })(Endian || (Endian = {})); function createReader(endianness, buffer) { if (endianness === Endian.LE) { return new ReaderLE(buffer); } else { return new ReaderBE(buffer); } } function createWriter(endianness, size) { if (endianness === Endian.LE) { return new WriterLE(size); } else { return new WriterBE(size); } } // Copyright (c) Holly Stubbs (tgpholly) - Licensed under MIT // Check LICENSE in repository root for more information. class ReaderBE extends ReaderBase { readShort() { const value = this.buffer.readInt16BE(this.offset); this.offset += 2; return value; } readUShort() { const value = this.buffer.readUInt16BE(this.offset); this.offset += 2; return value; } readInt() { const value = this.buffer.readInt32BE(this.offset); this.offset += 4; return value; } readUInt() { const value = this.buffer.readUInt32BE(this.offset); this.offset += 4; return value; } readLong() { const value = this.buffer.readBigInt64BE(this.offset); this.offset += 8; return value; } readULong() { const value = this.buffer.readBigUint64BE(this.offset); this.offset += 8; return value; } readFloat() { const value = this.buffer.readFloatBE(this.offset); this.offset += 4; return value; } readDouble() { const value = this.buffer.readDoubleBE(this.offset); this.offset += 8; return value; } readString() { const length = this.readUShort(); let text = ""; for (let i = 0; i < length; i++) { text += String.fromCharCode(this.readUByte()); } return text; } readString16() { const length = this.readUShort(); let text = ""; for (let i = 0; i < length; i++) { text += String.fromCharCode(this.readUShort()); } return text; } readShortsAsString(shortsToRead) { let text = ""; for (let i = 0; i < shortsToRead; i++) { text += String.fromCharCode(this.readUShort()); } return text; } } // Copyright (c) Holly Stubbs (tgpholly) - Licensed under MIT // Check LICENSE in repository root for more information. class ReaderLE extends ReaderBase { readShort() { const value = this.buffer.readInt16LE(this.offset); this.offset += 2; return value; } readUShort() { const value = this.buffer.readUInt16LE(this.offset); this.offset += 2; return value; } readInt() { const value = this.buffer.readInt32LE(this.offset); this.offset += 4; return value; } readUInt() { const value = this.buffer.readUInt32LE(this.offset); this.offset += 4; return value; } readLong() { const value = this.buffer.readBigInt64LE(this.offset); this.offset += 8; return value; } readULong() { const value = this.buffer.readBigUint64LE(this.offset); this.offset += 8; return value; } readFloat() { const value = this.buffer.readFloatLE(this.offset); this.offset += 4; return value; } readDouble() { const value = this.buffer.readDoubleLE(this.offset); this.offset += 8; return value; } readString() { const length = this.readUShort(); let text = ""; for (let i = 0; i < length; i++) { text += String.fromCharCode(this.readUByte()); } return text; } readString16() { const length = this.readUShort(); let text = ""; for (let i = 0; i < length; i++) { text += String.fromCharCode(this.readUShort()); } return text; } readShortsAsString(shortsToRead) { let text = ""; for (let i = 0; i < shortsToRead; i++) { text += String.fromCharCode(this.readUShort()); } return text; } } // Copyright (c) Holly Stubbs (tgpholly) - Licensed under MIT // Check LICENSE in repository root for more information. class WriterBE extends WriterBase { writeShort(value) { if (this.resizable) { const buffer = getBufferClass().alloc(2); buffer.writeInt16BE(value); this.writeBuffer(buffer); } else { this.buffer.writeInt16BE(value, this.offset); this.offset += 2; } return this; } writeUShort(value) { if (this.resizable) { const buffer = getBufferClass().alloc(2); buffer.writeUInt16BE(value); this.writeBuffer(buffer); } else { this.buffer.writeUInt16BE(value, this.offset); this.offset += 2; } return this; } writeInt(value) { if (this.resizable) { const buffer = getBufferClass().alloc(4); buffer.writeInt32BE(value); this.writeBuffer(buffer); } else { this.buffer.writeInt32BE(value, this.offset); this.offset += 4; } return this; } writeUInt(value) { if (this.resizable) { const buffer = getBufferClass().alloc(4); buffer.writeUInt32BE(value); this.writeBuffer(buffer); } else { this.buffer.writeUInt32BE(value, this.offset); this.offset += 4; } return this; } writeLong(value) { if (typeof (value) !== "bigint") { value = BigInt(value); } if (this.resizable) { const buffer = getBufferClass().alloc(8); buffer.writeBigInt64BE(value); this.writeBuffer(buffer); } else { this.buffer.writeBigInt64BE(value, this.offset); this.offset += 8; } return this; } writeULong(value) { if (typeof (value) !== "bigint") { value = BigInt(value); } if (this.resizable) { const buffer = getBufferClass().alloc(8); buffer.writeBigUint64BE(value); this.writeBuffer(buffer); } else { this.buffer.writeBigUint64BE(value, this.offset); this.offset += 8; } return this; } writeFloat(value) { if (this.resizable) { const buffer = getBufferClass().alloc(4); buffer.writeFloatBE(value); this.writeBuffer(buffer); } else { this.buffer.writeFloatBE(value, this.offset); this.offset += 4; } return this; } writeDouble(value) { if (this.resizable) { const buffer = getBufferClass().alloc(8); buffer.writeDoubleBE(value); this.writeBuffer(buffer); } else { this.buffer.writeDoubleBE(value, this.offset); this.offset += 8; } return this; } writeShortString(text) { this.writeUByte(text.length); for (let i = 0; i < text.length; i++) { this.writeUByte(text.charCodeAt(i)); } return this; } writeString(text) { this.writeUShort(text.length); for (let i = 0; i < text.length; i++) { this.writeUByte(text.charCodeAt(i)); } return this; } writeString16(text) { this.writeUShort(text.length); for (let i = 0; i < text.length; i++) { this.writeUShort(text.charCodeAt(i)); } return this; } writeStringAsShorts(text) { let buffer; if (this.resizable) { buffer = getBufferClass().alloc(text.length * 2); } else { buffer = this.buffer; } for (let i = 0; i < text.length; i++) { buffer.writeUint16BE(text.charCodeAt(i), i); } return this; } } // Copyright (c) Holly Stubbs (tgpholly) - Licensed under MIT // Check LICENSE in repository root for more information. class WriterLE extends WriterBase { writeShort(value) { if (this.resizable) { const buffer = getBufferClass().alloc(2); buffer.writeInt16LE(value); this.writeBuffer(buffer); } else { this.buffer.writeInt16LE(value, this.offset); this.offset += 2; } return this; } writeUShort(value) { if (this.resizable) { const buffer = getBufferClass().alloc(2); buffer.writeUInt16LE(value); this.writeBuffer(buffer); } else { this.buffer.writeUInt16LE(value, this.offset); this.offset += 2; } return this; } writeInt(value) { if (this.resizable) { const buffer = getBufferClass().alloc(4); buffer.writeInt32LE(value); this.writeBuffer(buffer); } else { this.buffer.writeInt32LE(value, this.offset); this.offset += 4; } return this; } writeUInt(value) { if (this.resizable) { const buffer = getBufferClass().alloc(4); buffer.writeUInt32LE(value); this.writeBuffer(buffer); } else { this.buffer.writeUInt32LE(value, this.offset); this.offset += 4; } return this; } writeLong(value) { if (typeof (value) !== "bigint") { value = BigInt(value); } if (this.resizable) { const buffer = getBufferClass().alloc(8); buffer.writeBigInt64LE(value); this.writeBuffer(buffer); } else { this.buffer.writeBigInt64LE(value, this.offset); this.offset += 8; } return this; } writeULong(value) { if (typeof (value) !== "bigint") { value = BigInt(value); } if (this.resizable) { const buffer = getBufferClass().alloc(8); buffer.writeBigUint64LE(value); this.writeBuffer(buffer); } else { this.buffer.writeBigUint64LE(value, this.offset); this.offset += 8; } return this; } writeFloat(value) { if (this.resizable) { const buffer = getBufferClass().alloc(4); buffer.writeFloatLE(value); this.writeBuffer(buffer); } else { this.buffer.writeFloatLE(value, this.offset); this.offset += 4; } return this; } writeDouble(value) { if (this.resizable) { const buffer = getBufferClass().alloc(8); buffer.writeDoubleLE(value); this.writeBuffer(buffer); } else { this.buffer.writeDoubleLE(value, this.offset); this.offset += 8; } return this; } writeShortString(text) { this.writeUByte(text.length); for (let i = 0; i < text.length; i++) { this.writeUByte(text.charCodeAt(i)); } return this; } writeString(text) { this.writeUShort(text.length); for (let i = 0; i < text.length; i++) { this.writeUByte(text.charCodeAt(i)); } return this; } writeString16(text) { this.writeUShort(text.length); for (let i = 0; i < text.length; i++) { this.writeUShort(text.charCodeAt(i)); } return this; } writeStringAsShorts(text) { let buffer; if (this.resizable) { buffer = getBufferClass().alloc(text.length * 2); } else { buffer = this.buffer; } for (let i = 0; i < text.length; i++) { buffer.writeUint16LE(text.charCodeAt(i), i); } return this; } }