Spaces:
Runtime error
Runtime error
| var BSON = (function (exports) { | |
| ; | |
| const TypedArrayPrototypeGetSymbolToStringTag = (() => { | |
| const g = Object.getOwnPropertyDescriptor(Object.getPrototypeOf(Uint8Array.prototype), Symbol.toStringTag).get; | |
| return (value) => g.call(value); | |
| })(); | |
| function isUint8Array(value) { | |
| return TypedArrayPrototypeGetSymbolToStringTag(value) === 'Uint8Array'; | |
| } | |
| function isAnyArrayBuffer(value) { | |
| return (typeof value === 'object' && | |
| value != null && | |
| Symbol.toStringTag in value && | |
| (value[Symbol.toStringTag] === 'ArrayBuffer' || | |
| value[Symbol.toStringTag] === 'SharedArrayBuffer')); | |
| } | |
| function isRegExp(regexp) { | |
| return regexp instanceof RegExp || Object.prototype.toString.call(regexp) === '[object RegExp]'; | |
| } | |
| function isMap(value) { | |
| return (typeof value === 'object' && | |
| value != null && | |
| Symbol.toStringTag in value && | |
| value[Symbol.toStringTag] === 'Map'); | |
| } | |
| function isDate(date) { | |
| return date instanceof Date || Object.prototype.toString.call(date) === '[object Date]'; | |
| } | |
| function defaultInspect(x, _options) { | |
| return JSON.stringify(x, (k, v) => { | |
| if (typeof v === 'bigint') { | |
| return { $numberLong: `${v}` }; | |
| } | |
| else if (isMap(v)) { | |
| return Object.fromEntries(v); | |
| } | |
| return v; | |
| }); | |
| } | |
| function getStylizeFunction(options) { | |
| const stylizeExists = options != null && | |
| typeof options === 'object' && | |
| 'stylize' in options && | |
| typeof options.stylize === 'function'; | |
| if (stylizeExists) { | |
| return options.stylize; | |
| } | |
| } | |
| const BSON_MAJOR_VERSION = 7; | |
| const BSON_VERSION_SYMBOL = Symbol.for('@@mdb.bson.version'); | |
| const BSON_INT32_MAX = 0x7fffffff; | |
| const BSON_INT32_MIN = -2147483648; | |
| const BSON_INT64_MAX = Math.pow(2, 63) - 1; | |
| const BSON_INT64_MIN = -Math.pow(2, 63); | |
| const JS_INT_MAX = Math.pow(2, 53); | |
| const JS_INT_MIN = -Math.pow(2, 53); | |
| const BSON_DATA_NUMBER = 1; | |
| const BSON_DATA_STRING = 2; | |
| const BSON_DATA_OBJECT = 3; | |
| const BSON_DATA_ARRAY = 4; | |
| const BSON_DATA_BINARY = 5; | |
| const BSON_DATA_UNDEFINED = 6; | |
| const BSON_DATA_OID = 7; | |
| const BSON_DATA_BOOLEAN = 8; | |
| const BSON_DATA_DATE = 9; | |
| const BSON_DATA_NULL = 10; | |
| const BSON_DATA_REGEXP = 11; | |
| const BSON_DATA_DBPOINTER = 12; | |
| const BSON_DATA_CODE = 13; | |
| const BSON_DATA_SYMBOL = 14; | |
| const BSON_DATA_CODE_W_SCOPE = 15; | |
| const BSON_DATA_INT = 16; | |
| const BSON_DATA_TIMESTAMP = 17; | |
| const BSON_DATA_LONG = 18; | |
| const BSON_DATA_DECIMAL128 = 19; | |
| const BSON_DATA_MIN_KEY = 0xff; | |
| const BSON_DATA_MAX_KEY = 0x7f; | |
| const BSON_BINARY_SUBTYPE_DEFAULT = 0; | |
| const BSON_BINARY_SUBTYPE_UUID_NEW = 4; | |
| const BSONType = Object.freeze({ | |
| double: 1, | |
| string: 2, | |
| object: 3, | |
| array: 4, | |
| binData: 5, | |
| undefined: 6, | |
| objectId: 7, | |
| bool: 8, | |
| date: 9, | |
| null: 10, | |
| regex: 11, | |
| dbPointer: 12, | |
| javascript: 13, | |
| symbol: 14, | |
| javascriptWithScope: 15, | |
| int: 16, | |
| timestamp: 17, | |
| long: 18, | |
| decimal: 19, | |
| minKey: -1, | |
| maxKey: 127 | |
| }); | |
| class BSONError extends Error { | |
| get bsonError() { | |
| return true; | |
| } | |
| get name() { | |
| return 'BSONError'; | |
| } | |
| constructor(message, options) { | |
| super(message, options); | |
| } | |
| static isBSONError(value) { | |
| return (value != null && | |
| typeof value === 'object' && | |
| 'bsonError' in value && | |
| value.bsonError === true && | |
| 'name' in value && | |
| 'message' in value && | |
| 'stack' in value); | |
| } | |
| } | |
| class BSONVersionError extends BSONError { | |
| get name() { | |
| return 'BSONVersionError'; | |
| } | |
| constructor() { | |
| super(`Unsupported BSON version, bson types must be from bson ${BSON_MAJOR_VERSION}.x.x`); | |
| } | |
| } | |
| class BSONRuntimeError extends BSONError { | |
| get name() { | |
| return 'BSONRuntimeError'; | |
| } | |
| constructor(message) { | |
| super(message); | |
| } | |
| } | |
| class BSONOffsetError extends BSONError { | |
| get name() { | |
| return 'BSONOffsetError'; | |
| } | |
| offset; | |
| constructor(message, offset, options) { | |
| super(`${message}. offset: ${offset}`, options); | |
| this.offset = offset; | |
| } | |
| } | |
| let TextDecoderFatal; | |
| let TextDecoderNonFatal; | |
| function parseUtf8(buffer, start, end, fatal) { | |
| if (fatal) { | |
| TextDecoderFatal ??= new TextDecoder('utf8', { fatal: true }); | |
| try { | |
| return TextDecoderFatal.decode(buffer.subarray(start, end)); | |
| } | |
| catch (cause) { | |
| throw new BSONError('Invalid UTF-8 string in BSON document', { cause }); | |
| } | |
| } | |
| TextDecoderNonFatal ??= new TextDecoder('utf8', { fatal: false }); | |
| return TextDecoderNonFatal.decode(buffer.subarray(start, end)); | |
| } | |
| function tryReadBasicLatin(uint8array, start, end) { | |
| if (uint8array.length === 0) { | |
| return ''; | |
| } | |
| const stringByteLength = end - start; | |
| if (stringByteLength === 0) { | |
| return ''; | |
| } | |
| if (stringByteLength > 20) { | |
| return null; | |
| } | |
| if (stringByteLength === 1 && uint8array[start] < 128) { | |
| return String.fromCharCode(uint8array[start]); | |
| } | |
| if (stringByteLength === 2 && uint8array[start] < 128 && uint8array[start + 1] < 128) { | |
| return String.fromCharCode(uint8array[start]) + String.fromCharCode(uint8array[start + 1]); | |
| } | |
| if (stringByteLength === 3 && | |
| uint8array[start] < 128 && | |
| uint8array[start + 1] < 128 && | |
| uint8array[start + 2] < 128) { | |
| return (String.fromCharCode(uint8array[start]) + | |
| String.fromCharCode(uint8array[start + 1]) + | |
| String.fromCharCode(uint8array[start + 2])); | |
| } | |
| const latinBytes = []; | |
| for (let i = start; i < end; i++) { | |
| const byte = uint8array[i]; | |
| if (byte > 127) { | |
| return null; | |
| } | |
| latinBytes.push(byte); | |
| } | |
| return String.fromCharCode(...latinBytes); | |
| } | |
| function tryWriteBasicLatin(destination, source, offset) { | |
| if (source.length === 0) | |
| return 0; | |
| if (source.length > 25) | |
| return null; | |
| if (destination.length - offset < source.length) | |
| return null; | |
| for (let charOffset = 0, destinationOffset = offset; charOffset < source.length; charOffset++, destinationOffset++) { | |
| const char = source.charCodeAt(charOffset); | |
| if (char > 127) | |
| return null; | |
| destination[destinationOffset] = char; | |
| } | |
| return source.length; | |
| } | |
| function nodejsMathRandomBytes(byteLength) { | |
| return nodeJsByteUtils.fromNumberArray(Array.from({ length: byteLength }, () => Math.floor(Math.random() * 256))); | |
| } | |
| function nodejsSecureRandomBytes(byteLength) { | |
| return crypto.getRandomValues(nodeJsByteUtils.allocate(byteLength)); | |
| } | |
| const nodejsRandomBytes = (() => { | |
| const { crypto } = globalThis; | |
| if (crypto != null && typeof crypto.getRandomValues === 'function') { | |
| return nodejsSecureRandomBytes; | |
| } | |
| else { | |
| return nodejsMathRandomBytes; | |
| } | |
| })(); | |
| const nodeJsByteUtils = { | |
| isUint8Array: isUint8Array, | |
| toLocalBufferType(potentialBuffer) { | |
| if (Buffer.isBuffer(potentialBuffer)) { | |
| return potentialBuffer; | |
| } | |
| if (ArrayBuffer.isView(potentialBuffer)) { | |
| return Buffer.from(potentialBuffer.buffer, potentialBuffer.byteOffset, potentialBuffer.byteLength); | |
| } | |
| const stringTag = potentialBuffer?.[Symbol.toStringTag] ?? Object.prototype.toString.call(potentialBuffer); | |
| if (stringTag === 'ArrayBuffer' || | |
| stringTag === 'SharedArrayBuffer' || | |
| stringTag === '[object ArrayBuffer]' || | |
| stringTag === '[object SharedArrayBuffer]') { | |
| return Buffer.from(potentialBuffer); | |
| } | |
| throw new BSONError(`Cannot create Buffer from the passed potentialBuffer.`); | |
| }, | |
| allocate(size) { | |
| return Buffer.alloc(size); | |
| }, | |
| allocateUnsafe(size) { | |
| return Buffer.allocUnsafe(size); | |
| }, | |
| compare(a, b) { | |
| return nodeJsByteUtils.toLocalBufferType(a).compare(b); | |
| }, | |
| concat(list) { | |
| return Buffer.concat(list); | |
| }, | |
| copy(source, target, targetStart, sourceStart, sourceEnd) { | |
| return nodeJsByteUtils | |
| .toLocalBufferType(source) | |
| .copy(target, targetStart ?? 0, sourceStart ?? 0, sourceEnd ?? source.length); | |
| }, | |
| equals(a, b) { | |
| return nodeJsByteUtils.toLocalBufferType(a).equals(b); | |
| }, | |
| fromNumberArray(array) { | |
| return Buffer.from(array); | |
| }, | |
| fromBase64(base64) { | |
| return Buffer.from(base64, 'base64'); | |
| }, | |
| fromUTF8(utf8) { | |
| return Buffer.from(utf8, 'utf8'); | |
| }, | |
| toBase64(buffer) { | |
| return nodeJsByteUtils.toLocalBufferType(buffer).toString('base64'); | |
| }, | |
| fromISO88591(codePoints) { | |
| return Buffer.from(codePoints, 'binary'); | |
| }, | |
| toISO88591(buffer) { | |
| return nodeJsByteUtils.toLocalBufferType(buffer).toString('binary'); | |
| }, | |
| fromHex(hex) { | |
| return Buffer.from(hex, 'hex'); | |
| }, | |
| toHex(buffer) { | |
| return nodeJsByteUtils.toLocalBufferType(buffer).toString('hex'); | |
| }, | |
| toUTF8(buffer, start, end, fatal) { | |
| const basicLatin = end - start <= 20 ? tryReadBasicLatin(buffer, start, end) : null; | |
| if (basicLatin != null) { | |
| return basicLatin; | |
| } | |
| const string = nodeJsByteUtils.toLocalBufferType(buffer).toString('utf8', start, end); | |
| if (fatal) { | |
| for (let i = 0; i < string.length; i++) { | |
| if (string.charCodeAt(i) === 0xfffd) { | |
| parseUtf8(buffer, start, end, true); | |
| break; | |
| } | |
| } | |
| } | |
| return string; | |
| }, | |
| utf8ByteLength(input) { | |
| return Buffer.byteLength(input, 'utf8'); | |
| }, | |
| encodeUTF8Into(buffer, source, byteOffset) { | |
| const latinBytesWritten = tryWriteBasicLatin(buffer, source, byteOffset); | |
| if (latinBytesWritten != null) { | |
| return latinBytesWritten; | |
| } | |
| return nodeJsByteUtils.toLocalBufferType(buffer).write(source, byteOffset, undefined, 'utf8'); | |
| }, | |
| randomBytes: nodejsRandomBytes, | |
| swap32(buffer) { | |
| return nodeJsByteUtils.toLocalBufferType(buffer).swap32(); | |
| } | |
| }; | |
| function isReactNative() { | |
| const { navigator } = globalThis; | |
| return typeof navigator === 'object' && navigator.product === 'ReactNative'; | |
| } | |
| function webMathRandomBytes(byteLength) { | |
| if (byteLength < 0) { | |
| throw new RangeError(`The argument 'byteLength' is invalid. Received ${byteLength}`); | |
| } | |
| return webByteUtils.fromNumberArray(Array.from({ length: byteLength }, () => Math.floor(Math.random() * 256))); | |
| } | |
| const webRandomBytes = (() => { | |
| const { crypto } = globalThis; | |
| if (crypto != null && typeof crypto.getRandomValues === 'function') { | |
| return (byteLength) => { | |
| return crypto.getRandomValues(webByteUtils.allocate(byteLength)); | |
| }; | |
| } | |
| else { | |
| if (isReactNative()) { | |
| const { console } = globalThis; | |
| console?.warn?.('BSON: For React Native please polyfill crypto.getRandomValues, e.g. using: https://www.npmjs.com/package/react-native-get-random-values.'); | |
| } | |
| return webMathRandomBytes; | |
| } | |
| })(); | |
| const HEX_DIGIT = /(\d|[a-f])/i; | |
| const webByteUtils = { | |
| isUint8Array: isUint8Array, | |
| toLocalBufferType(potentialUint8array) { | |
| const stringTag = potentialUint8array?.[Symbol.toStringTag] ?? | |
| Object.prototype.toString.call(potentialUint8array); | |
| if (stringTag === 'Uint8Array') { | |
| return potentialUint8array; | |
| } | |
| if (ArrayBuffer.isView(potentialUint8array)) { | |
| return new Uint8Array(potentialUint8array.buffer.slice(potentialUint8array.byteOffset, potentialUint8array.byteOffset + potentialUint8array.byteLength)); | |
| } | |
| if (stringTag === 'ArrayBuffer' || | |
| stringTag === 'SharedArrayBuffer' || | |
| stringTag === '[object ArrayBuffer]' || | |
| stringTag === '[object SharedArrayBuffer]') { | |
| return new Uint8Array(potentialUint8array); | |
| } | |
| throw new BSONError(`Cannot make a Uint8Array from passed potentialBuffer.`); | |
| }, | |
| allocate(size) { | |
| if (typeof size !== 'number') { | |
| throw new TypeError(`The "size" argument must be of type number. Received ${String(size)}`); | |
| } | |
| return new Uint8Array(size); | |
| }, | |
| allocateUnsafe(size) { | |
| return webByteUtils.allocate(size); | |
| }, | |
| compare(uint8Array, otherUint8Array) { | |
| if (uint8Array === otherUint8Array) | |
| return 0; | |
| const len = Math.min(uint8Array.length, otherUint8Array.length); | |
| for (let i = 0; i < len; i++) { | |
| if (uint8Array[i] < otherUint8Array[i]) | |
| return -1; | |
| if (uint8Array[i] > otherUint8Array[i]) | |
| return 1; | |
| } | |
| if (uint8Array.length < otherUint8Array.length) | |
| return -1; | |
| if (uint8Array.length > otherUint8Array.length) | |
| return 1; | |
| return 0; | |
| }, | |
| concat(uint8Arrays) { | |
| if (uint8Arrays.length === 0) | |
| return webByteUtils.allocate(0); | |
| let totalLength = 0; | |
| for (const uint8Array of uint8Arrays) { | |
| totalLength += uint8Array.length; | |
| } | |
| const result = webByteUtils.allocate(totalLength); | |
| let offset = 0; | |
| for (const uint8Array of uint8Arrays) { | |
| result.set(uint8Array, offset); | |
| offset += uint8Array.length; | |
| } | |
| return result; | |
| }, | |
| copy(source, target, targetStart, sourceStart, sourceEnd) { | |
| if (sourceEnd !== undefined && sourceEnd < 0) { | |
| throw new RangeError(`The value of "sourceEnd" is out of range. It must be >= 0. Received ${sourceEnd}`); | |
| } | |
| sourceEnd = sourceEnd ?? source.length; | |
| if (sourceStart !== undefined && (sourceStart < 0 || sourceStart > sourceEnd)) { | |
| throw new RangeError(`The value of "sourceStart" is out of range. It must be >= 0 and <= ${sourceEnd}. Received ${sourceStart}`); | |
| } | |
| sourceStart = sourceStart ?? 0; | |
| if (targetStart !== undefined && targetStart < 0) { | |
| throw new RangeError(`The value of "targetStart" is out of range. It must be >= 0. Received ${targetStart}`); | |
| } | |
| targetStart = targetStart ?? 0; | |
| const srcSlice = source.subarray(sourceStart, sourceEnd); | |
| const maxLen = Math.min(srcSlice.length, target.length - targetStart); | |
| if (maxLen <= 0) { | |
| return 0; | |
| } | |
| target.set(srcSlice.subarray(0, maxLen), targetStart); | |
| return maxLen; | |
| }, | |
| equals(uint8Array, otherUint8Array) { | |
| if (uint8Array.byteLength !== otherUint8Array.byteLength) { | |
| return false; | |
| } | |
| for (let i = 0; i < uint8Array.byteLength; i++) { | |
| if (uint8Array[i] !== otherUint8Array[i]) { | |
| return false; | |
| } | |
| } | |
| return true; | |
| }, | |
| fromNumberArray(array) { | |
| return Uint8Array.from(array); | |
| }, | |
| fromBase64(base64) { | |
| return Uint8Array.from(atob(base64), c => c.charCodeAt(0)); | |
| }, | |
| fromUTF8(utf8) { | |
| return new TextEncoder().encode(utf8); | |
| }, | |
| toBase64(uint8array) { | |
| return btoa(webByteUtils.toISO88591(uint8array)); | |
| }, | |
| fromISO88591(codePoints) { | |
| return Uint8Array.from(codePoints, c => c.charCodeAt(0) & 0xff); | |
| }, | |
| toISO88591(uint8array) { | |
| return Array.from(Uint16Array.from(uint8array), b => String.fromCharCode(b)).join(''); | |
| }, | |
| fromHex(hex) { | |
| const evenLengthHex = hex.length % 2 === 0 ? hex : hex.slice(0, hex.length - 1); | |
| const buffer = []; | |
| for (let i = 0; i < evenLengthHex.length; i += 2) { | |
| const firstDigit = evenLengthHex[i]; | |
| const secondDigit = evenLengthHex[i + 1]; | |
| if (!HEX_DIGIT.test(firstDigit)) { | |
| break; | |
| } | |
| if (!HEX_DIGIT.test(secondDigit)) { | |
| break; | |
| } | |
| const hexDigit = Number.parseInt(`${firstDigit}${secondDigit}`, 16); | |
| buffer.push(hexDigit); | |
| } | |
| return Uint8Array.from(buffer); | |
| }, | |
| toHex(uint8array) { | |
| return Array.from(uint8array, byte => byte.toString(16).padStart(2, '0')).join(''); | |
| }, | |
| toUTF8(uint8array, start, end, fatal) { | |
| const basicLatin = end - start <= 20 ? tryReadBasicLatin(uint8array, start, end) : null; | |
| if (basicLatin != null) { | |
| return basicLatin; | |
| } | |
| return parseUtf8(uint8array, start, end, fatal); | |
| }, | |
| utf8ByteLength(input) { | |
| return new TextEncoder().encode(input).byteLength; | |
| }, | |
| encodeUTF8Into(uint8array, source, byteOffset) { | |
| const bytes = new TextEncoder().encode(source); | |
| uint8array.set(bytes, byteOffset); | |
| return bytes.byteLength; | |
| }, | |
| randomBytes: webRandomBytes, | |
| swap32(buffer) { | |
| if (buffer.length % 4 !== 0) { | |
| throw new RangeError('Buffer size must be a multiple of 32-bits'); | |
| } | |
| for (let i = 0; i < buffer.length; i += 4) { | |
| const byte0 = buffer[i]; | |
| const byte1 = buffer[i + 1]; | |
| const byte2 = buffer[i + 2]; | |
| const byte3 = buffer[i + 3]; | |
| buffer[i] = byte3; | |
| buffer[i + 1] = byte2; | |
| buffer[i + 2] = byte1; | |
| buffer[i + 3] = byte0; | |
| } | |
| return buffer; | |
| } | |
| }; | |
| const hasGlobalBuffer = typeof Buffer === 'function' && Buffer.prototype?._isBuffer !== true; | |
| const ByteUtils = hasGlobalBuffer ? nodeJsByteUtils : webByteUtils; | |
| const bsonType = Symbol.for('@@mdb.bson.type'); | |
| class BSONValue { | |
| get [bsonType]() { | |
| return this._bsontype; | |
| } | |
| get [BSON_VERSION_SYMBOL]() { | |
| return BSON_MAJOR_VERSION; | |
| } | |
| [Symbol.for('nodejs.util.inspect.custom')](depth, options, inspect) { | |
| return this.inspect(depth, options, inspect); | |
| } | |
| } | |
| const FLOAT = new Float64Array(1); | |
| const FLOAT_BYTES = new Uint8Array(FLOAT.buffer, 0, 8); | |
| FLOAT[0] = -1; | |
| const isBigEndian = FLOAT_BYTES[7] === 0; | |
| const NumberUtils = { | |
| isBigEndian, | |
| getNonnegativeInt32LE(source, offset) { | |
| if (source[offset + 3] > 127) { | |
| throw new RangeError(`Size cannot be negative at offset: ${offset}`); | |
| } | |
| return (source[offset] | | |
| (source[offset + 1] << 8) | | |
| (source[offset + 2] << 16) | | |
| (source[offset + 3] << 24)); | |
| }, | |
| getInt32LE(source, offset) { | |
| return (source[offset] | | |
| (source[offset + 1] << 8) | | |
| (source[offset + 2] << 16) | | |
| (source[offset + 3] << 24)); | |
| }, | |
| getUint32LE(source, offset) { | |
| return (source[offset] + | |
| source[offset + 1] * 256 + | |
| source[offset + 2] * 65536 + | |
| source[offset + 3] * 16777216); | |
| }, | |
| getUint32BE(source, offset) { | |
| return (source[offset + 3] + | |
| source[offset + 2] * 256 + | |
| source[offset + 1] * 65536 + | |
| source[offset] * 16777216); | |
| }, | |
| getBigInt64LE(source, offset) { | |
| const hi = BigInt(source[offset + 4] + | |
| source[offset + 5] * 256 + | |
| source[offset + 6] * 65536 + | |
| (source[offset + 7] << 24)); | |
| const lo = BigInt(source[offset] + | |
| source[offset + 1] * 256 + | |
| source[offset + 2] * 65536 + | |
| source[offset + 3] * 16777216); | |
| return (hi << 32n) + lo; | |
| }, | |
| getFloat64LE: isBigEndian | |
| ? (source, offset) => { | |
| FLOAT_BYTES[7] = source[offset]; | |
| FLOAT_BYTES[6] = source[offset + 1]; | |
| FLOAT_BYTES[5] = source[offset + 2]; | |
| FLOAT_BYTES[4] = source[offset + 3]; | |
| FLOAT_BYTES[3] = source[offset + 4]; | |
| FLOAT_BYTES[2] = source[offset + 5]; | |
| FLOAT_BYTES[1] = source[offset + 6]; | |
| FLOAT_BYTES[0] = source[offset + 7]; | |
| return FLOAT[0]; | |
| } | |
| : (source, offset) => { | |
| FLOAT_BYTES[0] = source[offset]; | |
| FLOAT_BYTES[1] = source[offset + 1]; | |
| FLOAT_BYTES[2] = source[offset + 2]; | |
| FLOAT_BYTES[3] = source[offset + 3]; | |
| FLOAT_BYTES[4] = source[offset + 4]; | |
| FLOAT_BYTES[5] = source[offset + 5]; | |
| FLOAT_BYTES[6] = source[offset + 6]; | |
| FLOAT_BYTES[7] = source[offset + 7]; | |
| return FLOAT[0]; | |
| }, | |
| setInt32BE(destination, offset, value) { | |
| destination[offset + 3] = value; | |
| value >>>= 8; | |
| destination[offset + 2] = value; | |
| value >>>= 8; | |
| destination[offset + 1] = value; | |
| value >>>= 8; | |
| destination[offset] = value; | |
| return 4; | |
| }, | |
| setInt32LE(destination, offset, value) { | |
| destination[offset] = value; | |
| value >>>= 8; | |
| destination[offset + 1] = value; | |
| value >>>= 8; | |
| destination[offset + 2] = value; | |
| value >>>= 8; | |
| destination[offset + 3] = value; | |
| return 4; | |
| }, | |
| setBigInt64LE(destination, offset, value) { | |
| const mask32bits = 0xffffffffn; | |
| let lo = Number(value & mask32bits); | |
| destination[offset] = lo; | |
| lo >>= 8; | |
| destination[offset + 1] = lo; | |
| lo >>= 8; | |
| destination[offset + 2] = lo; | |
| lo >>= 8; | |
| destination[offset + 3] = lo; | |
| let hi = Number((value >> 32n) & mask32bits); | |
| destination[offset + 4] = hi; | |
| hi >>= 8; | |
| destination[offset + 5] = hi; | |
| hi >>= 8; | |
| destination[offset + 6] = hi; | |
| hi >>= 8; | |
| destination[offset + 7] = hi; | |
| return 8; | |
| }, | |
| setFloat64LE: isBigEndian | |
| ? (destination, offset, value) => { | |
| FLOAT[0] = value; | |
| destination[offset] = FLOAT_BYTES[7]; | |
| destination[offset + 1] = FLOAT_BYTES[6]; | |
| destination[offset + 2] = FLOAT_BYTES[5]; | |
| destination[offset + 3] = FLOAT_BYTES[4]; | |
| destination[offset + 4] = FLOAT_BYTES[3]; | |
| destination[offset + 5] = FLOAT_BYTES[2]; | |
| destination[offset + 6] = FLOAT_BYTES[1]; | |
| destination[offset + 7] = FLOAT_BYTES[0]; | |
| return 8; | |
| } | |
| : (destination, offset, value) => { | |
| FLOAT[0] = value; | |
| destination[offset] = FLOAT_BYTES[0]; | |
| destination[offset + 1] = FLOAT_BYTES[1]; | |
| destination[offset + 2] = FLOAT_BYTES[2]; | |
| destination[offset + 3] = FLOAT_BYTES[3]; | |
| destination[offset + 4] = FLOAT_BYTES[4]; | |
| destination[offset + 5] = FLOAT_BYTES[5]; | |
| destination[offset + 6] = FLOAT_BYTES[6]; | |
| destination[offset + 7] = FLOAT_BYTES[7]; | |
| return 8; | |
| } | |
| }; | |
| class Binary extends BSONValue { | |
| get _bsontype() { | |
| return 'Binary'; | |
| } | |
| static BSON_BINARY_SUBTYPE_DEFAULT = 0; | |
| static BUFFER_SIZE = 256; | |
| static SUBTYPE_DEFAULT = 0; | |
| static SUBTYPE_FUNCTION = 1; | |
| static SUBTYPE_BYTE_ARRAY = 2; | |
| static SUBTYPE_UUID_OLD = 3; | |
| static SUBTYPE_UUID = 4; | |
| static SUBTYPE_MD5 = 5; | |
| static SUBTYPE_ENCRYPTED = 6; | |
| static SUBTYPE_COLUMN = 7; | |
| static SUBTYPE_SENSITIVE = 8; | |
| static SUBTYPE_VECTOR = 9; | |
| static SUBTYPE_USER_DEFINED = 128; | |
| static VECTOR_TYPE = Object.freeze({ | |
| Int8: 0x03, | |
| Float32: 0x27, | |
| PackedBit: 0x10 | |
| }); | |
| buffer; | |
| sub_type; | |
| position; | |
| constructor(buffer, subType) { | |
| super(); | |
| if (!(buffer == null) && | |
| typeof buffer === 'string' && | |
| !ArrayBuffer.isView(buffer) && | |
| !isAnyArrayBuffer(buffer) && | |
| !Array.isArray(buffer)) { | |
| throw new BSONError('Binary can only be constructed from Uint8Array or number[]'); | |
| } | |
| this.sub_type = subType ?? Binary.BSON_BINARY_SUBTYPE_DEFAULT; | |
| if (buffer == null) { | |
| this.buffer = ByteUtils.allocate(Binary.BUFFER_SIZE); | |
| this.position = 0; | |
| } | |
| else { | |
| this.buffer = Array.isArray(buffer) | |
| ? ByteUtils.fromNumberArray(buffer) | |
| : ByteUtils.toLocalBufferType(buffer); | |
| this.position = this.buffer.byteLength; | |
| } | |
| } | |
| put(byteValue) { | |
| if (typeof byteValue === 'string' && byteValue.length !== 1) { | |
| throw new BSONError('only accepts single character String'); | |
| } | |
| else if (typeof byteValue !== 'number' && byteValue.length !== 1) | |
| throw new BSONError('only accepts single character Uint8Array or Array'); | |
| let decodedByte; | |
| if (typeof byteValue === 'string') { | |
| decodedByte = byteValue.charCodeAt(0); | |
| } | |
| else if (typeof byteValue === 'number') { | |
| decodedByte = byteValue; | |
| } | |
| else { | |
| decodedByte = byteValue[0]; | |
| } | |
| if (decodedByte < 0 || decodedByte > 255) { | |
| throw new BSONError('only accepts number in a valid unsigned byte range 0-255'); | |
| } | |
| if (this.buffer.byteLength > this.position) { | |
| this.buffer[this.position++] = decodedByte; | |
| } | |
| else { | |
| const newSpace = ByteUtils.allocate(Binary.BUFFER_SIZE + this.buffer.length); | |
| newSpace.set(this.buffer, 0); | |
| this.buffer = newSpace; | |
| this.buffer[this.position++] = decodedByte; | |
| } | |
| } | |
| write(sequence, offset) { | |
| offset = typeof offset === 'number' ? offset : this.position; | |
| if (this.buffer.byteLength < offset + sequence.length) { | |
| const newSpace = ByteUtils.allocate(this.buffer.byteLength + sequence.length); | |
| newSpace.set(this.buffer, 0); | |
| this.buffer = newSpace; | |
| } | |
| if (ArrayBuffer.isView(sequence)) { | |
| this.buffer.set(ByteUtils.toLocalBufferType(sequence), offset); | |
| this.position = | |
| offset + sequence.byteLength > this.position ? offset + sequence.length : this.position; | |
| } | |
| else if (typeof sequence === 'string') { | |
| throw new BSONError('input cannot be string'); | |
| } | |
| } | |
| read(position, length) { | |
| length = length && length > 0 ? length : this.position; | |
| const end = position + length; | |
| return this.buffer.subarray(position, end > this.position ? this.position : end); | |
| } | |
| value() { | |
| return this.buffer.length === this.position | |
| ? this.buffer | |
| : this.buffer.subarray(0, this.position); | |
| } | |
| length() { | |
| return this.position; | |
| } | |
| toJSON() { | |
| return ByteUtils.toBase64(this.buffer.subarray(0, this.position)); | |
| } | |
| toString(encoding) { | |
| if (encoding === 'hex') | |
| return ByteUtils.toHex(this.buffer.subarray(0, this.position)); | |
| if (encoding === 'base64') | |
| return ByteUtils.toBase64(this.buffer.subarray(0, this.position)); | |
| if (encoding === 'utf8' || encoding === 'utf-8') | |
| return ByteUtils.toUTF8(this.buffer, 0, this.position, false); | |
| return ByteUtils.toUTF8(this.buffer, 0, this.position, false); | |
| } | |
| toExtendedJSON(options) { | |
| options = options || {}; | |
| if (this.sub_type === Binary.SUBTYPE_VECTOR) { | |
| validateBinaryVector(this); | |
| } | |
| const base64String = ByteUtils.toBase64(this.buffer); | |
| const subType = Number(this.sub_type).toString(16); | |
| if (options.legacy) { | |
| return { | |
| $binary: base64String, | |
| $type: subType.length === 1 ? '0' + subType : subType | |
| }; | |
| } | |
| return { | |
| $binary: { | |
| base64: base64String, | |
| subType: subType.length === 1 ? '0' + subType : subType | |
| } | |
| }; | |
| } | |
| toUUID() { | |
| if (this.sub_type === Binary.SUBTYPE_UUID) { | |
| return new UUID(this.buffer.subarray(0, this.position)); | |
| } | |
| throw new BSONError(`Binary sub_type "${this.sub_type}" is not supported for converting to UUID. Only "${Binary.SUBTYPE_UUID}" is currently supported.`); | |
| } | |
| static createFromHexString(hex, subType) { | |
| return new Binary(ByteUtils.fromHex(hex), subType); | |
| } | |
| static createFromBase64(base64, subType) { | |
| return new Binary(ByteUtils.fromBase64(base64), subType); | |
| } | |
| static fromExtendedJSON(doc, options) { | |
| options = options || {}; | |
| let data; | |
| let type; | |
| if ('$binary' in doc) { | |
| if (options.legacy && typeof doc.$binary === 'string' && '$type' in doc) { | |
| type = doc.$type ? parseInt(doc.$type, 16) : 0; | |
| data = ByteUtils.fromBase64(doc.$binary); | |
| } | |
| else { | |
| if (typeof doc.$binary !== 'string') { | |
| type = doc.$binary.subType ? parseInt(doc.$binary.subType, 16) : 0; | |
| data = ByteUtils.fromBase64(doc.$binary.base64); | |
| } | |
| } | |
| } | |
| else if ('$uuid' in doc) { | |
| type = 4; | |
| data = UUID.bytesFromString(doc.$uuid); | |
| } | |
| if (!data) { | |
| throw new BSONError(`Unexpected Binary Extended JSON format ${JSON.stringify(doc)}`); | |
| } | |
| return type === BSON_BINARY_SUBTYPE_UUID_NEW ? new UUID(data) : new Binary(data, type); | |
| } | |
| inspect(depth, options, inspect) { | |
| inspect ??= defaultInspect; | |
| const base64 = ByteUtils.toBase64(this.buffer.subarray(0, this.position)); | |
| const base64Arg = inspect(base64, options); | |
| const subTypeArg = inspect(this.sub_type, options); | |
| return `Binary.createFromBase64(${base64Arg}, ${subTypeArg})`; | |
| } | |
| toInt8Array() { | |
| if (this.sub_type !== Binary.SUBTYPE_VECTOR) { | |
| throw new BSONError('Binary sub_type is not Vector'); | |
| } | |
| if (this.buffer[0] !== Binary.VECTOR_TYPE.Int8) { | |
| throw new BSONError('Binary datatype field is not Int8'); | |
| } | |
| validateBinaryVector(this); | |
| return new Int8Array(this.buffer.buffer.slice(this.buffer.byteOffset + 2, this.buffer.byteOffset + this.position)); | |
| } | |
| toFloat32Array() { | |
| if (this.sub_type !== Binary.SUBTYPE_VECTOR) { | |
| throw new BSONError('Binary sub_type is not Vector'); | |
| } | |
| if (this.buffer[0] !== Binary.VECTOR_TYPE.Float32) { | |
| throw new BSONError('Binary datatype field is not Float32'); | |
| } | |
| validateBinaryVector(this); | |
| const floatBytes = new Uint8Array(this.buffer.buffer.slice(this.buffer.byteOffset + 2, this.buffer.byteOffset + this.position)); | |
| if (NumberUtils.isBigEndian) | |
| ByteUtils.swap32(floatBytes); | |
| return new Float32Array(floatBytes.buffer); | |
| } | |
| toPackedBits() { | |
| if (this.sub_type !== Binary.SUBTYPE_VECTOR) { | |
| throw new BSONError('Binary sub_type is not Vector'); | |
| } | |
| if (this.buffer[0] !== Binary.VECTOR_TYPE.PackedBit) { | |
| throw new BSONError('Binary datatype field is not packed bit'); | |
| } | |
| validateBinaryVector(this); | |
| return new Uint8Array(this.buffer.buffer.slice(this.buffer.byteOffset + 2, this.buffer.byteOffset + this.position)); | |
| } | |
| toBits() { | |
| if (this.sub_type !== Binary.SUBTYPE_VECTOR) { | |
| throw new BSONError('Binary sub_type is not Vector'); | |
| } | |
| if (this.buffer[0] !== Binary.VECTOR_TYPE.PackedBit) { | |
| throw new BSONError('Binary datatype field is not packed bit'); | |
| } | |
| validateBinaryVector(this); | |
| const byteCount = this.length() - 2; | |
| const bitCount = byteCount * 8 - this.buffer[1]; | |
| const bits = new Int8Array(bitCount); | |
| for (let bitOffset = 0; bitOffset < bits.length; bitOffset++) { | |
| const byteOffset = (bitOffset / 8) | 0; | |
| const byte = this.buffer[byteOffset + 2]; | |
| const shift = 7 - (bitOffset % 8); | |
| const bit = (byte >> shift) & 1; | |
| bits[bitOffset] = bit; | |
| } | |
| return bits; | |
| } | |
| static fromInt8Array(array) { | |
| const buffer = ByteUtils.allocate(array.byteLength + 2); | |
| buffer[0] = Binary.VECTOR_TYPE.Int8; | |
| buffer[1] = 0; | |
| const intBytes = new Uint8Array(array.buffer, array.byteOffset, array.byteLength); | |
| buffer.set(intBytes, 2); | |
| const bin = new this(buffer, this.SUBTYPE_VECTOR); | |
| validateBinaryVector(bin); | |
| return bin; | |
| } | |
| static fromFloat32Array(array) { | |
| const binaryBytes = ByteUtils.allocate(array.byteLength + 2); | |
| binaryBytes[0] = Binary.VECTOR_TYPE.Float32; | |
| binaryBytes[1] = 0; | |
| const floatBytes = new Uint8Array(array.buffer, array.byteOffset, array.byteLength); | |
| binaryBytes.set(floatBytes, 2); | |
| if (NumberUtils.isBigEndian) | |
| ByteUtils.swap32(new Uint8Array(binaryBytes.buffer, 2)); | |
| const bin = new this(binaryBytes, this.SUBTYPE_VECTOR); | |
| validateBinaryVector(bin); | |
| return bin; | |
| } | |
| static fromPackedBits(array, padding = 0) { | |
| const buffer = ByteUtils.allocate(array.byteLength + 2); | |
| buffer[0] = Binary.VECTOR_TYPE.PackedBit; | |
| buffer[1] = padding; | |
| buffer.set(array, 2); | |
| const bin = new this(buffer, this.SUBTYPE_VECTOR); | |
| validateBinaryVector(bin); | |
| return bin; | |
| } | |
| static fromBits(bits) { | |
| const byteLength = (bits.length + 7) >>> 3; | |
| const bytes = new Uint8Array(byteLength + 2); | |
| bytes[0] = Binary.VECTOR_TYPE.PackedBit; | |
| const remainder = bits.length % 8; | |
| bytes[1] = remainder === 0 ? 0 : 8 - remainder; | |
| for (let bitOffset = 0; bitOffset < bits.length; bitOffset++) { | |
| const byteOffset = bitOffset >>> 3; | |
| const bit = bits[bitOffset]; | |
| if (bit !== 0 && bit !== 1) { | |
| throw new BSONError(`Invalid bit value at ${bitOffset}: must be 0 or 1, found ${bits[bitOffset]}`); | |
| } | |
| if (bit === 0) | |
| continue; | |
| const shift = 7 - (bitOffset % 8); | |
| bytes[byteOffset + 2] |= bit << shift; | |
| } | |
| return new this(bytes, Binary.SUBTYPE_VECTOR); | |
| } | |
| } | |
| function validateBinaryVector(vector) { | |
| if (vector.sub_type !== Binary.SUBTYPE_VECTOR) | |
| return; | |
| const size = vector.position; | |
| const datatype = vector.buffer[0]; | |
| const padding = vector.buffer[1]; | |
| if ((datatype === Binary.VECTOR_TYPE.Float32 || datatype === Binary.VECTOR_TYPE.Int8) && | |
| padding !== 0) { | |
| throw new BSONError('Invalid Vector: padding must be zero for int8 and float32 vectors'); | |
| } | |
| if (datatype === Binary.VECTOR_TYPE.Float32) { | |
| if (size !== 0 && size - 2 !== 0 && (size - 2) % 4 !== 0) { | |
| throw new BSONError('Invalid Vector: Float32 vector must contain a multiple of 4 bytes'); | |
| } | |
| } | |
| if (datatype === Binary.VECTOR_TYPE.PackedBit && padding !== 0 && size === 2) { | |
| throw new BSONError('Invalid Vector: padding must be zero for packed bit vectors that are empty'); | |
| } | |
| if (datatype === Binary.VECTOR_TYPE.PackedBit && padding > 7) { | |
| throw new BSONError(`Invalid Vector: padding must be a value between 0 and 7. found: ${padding}`); | |
| } | |
| } | |
| const UUID_BYTE_LENGTH = 16; | |
| const UUID_WITHOUT_DASHES = /^[0-9A-F]{32}$/i; | |
| const UUID_WITH_DASHES = /^[0-9A-F]{8}-[0-9A-F]{4}-[0-9A-F]{4}-[0-9A-F]{4}-[0-9A-F]{12}$/i; | |
| class UUID extends Binary { | |
| constructor(input) { | |
| let bytes; | |
| if (input == null) { | |
| bytes = UUID.generate(); | |
| } | |
| else if (input instanceof UUID) { | |
| bytes = ByteUtils.toLocalBufferType(new Uint8Array(input.buffer)); | |
| } | |
| else if (ArrayBuffer.isView(input) && input.byteLength === UUID_BYTE_LENGTH) { | |
| bytes = ByteUtils.toLocalBufferType(input); | |
| } | |
| else if (typeof input === 'string') { | |
| bytes = UUID.bytesFromString(input); | |
| } | |
| else { | |
| throw new BSONError('Argument passed in UUID constructor must be a UUID, a 16 byte Buffer or a 32/36 character hex string (dashes excluded/included, format: xxxxxxxx-xxxx-xxxx-xxxx-xxxxxxxxxxxx).'); | |
| } | |
| super(bytes, BSON_BINARY_SUBTYPE_UUID_NEW); | |
| } | |
| get id() { | |
| return this.buffer; | |
| } | |
| set id(value) { | |
| this.buffer = value; | |
| } | |
| toHexString(includeDashes = true) { | |
| if (includeDashes) { | |
| return [ | |
| ByteUtils.toHex(this.buffer.subarray(0, 4)), | |
| ByteUtils.toHex(this.buffer.subarray(4, 6)), | |
| ByteUtils.toHex(this.buffer.subarray(6, 8)), | |
| ByteUtils.toHex(this.buffer.subarray(8, 10)), | |
| ByteUtils.toHex(this.buffer.subarray(10, 16)) | |
| ].join('-'); | |
| } | |
| return ByteUtils.toHex(this.buffer); | |
| } | |
| toString(encoding) { | |
| if (encoding === 'hex') | |
| return ByteUtils.toHex(this.id); | |
| if (encoding === 'base64') | |
| return ByteUtils.toBase64(this.id); | |
| return this.toHexString(); | |
| } | |
| toJSON() { | |
| return this.toHexString(); | |
| } | |
| equals(otherId) { | |
| if (!otherId) { | |
| return false; | |
| } | |
| if (otherId instanceof UUID) { | |
| return ByteUtils.equals(otherId.id, this.id); | |
| } | |
| try { | |
| return ByteUtils.equals(new UUID(otherId).id, this.id); | |
| } | |
| catch { | |
| return false; | |
| } | |
| } | |
| toBinary() { | |
| return new Binary(this.id, Binary.SUBTYPE_UUID); | |
| } | |
| static generate() { | |
| const bytes = ByteUtils.randomBytes(UUID_BYTE_LENGTH); | |
| bytes[6] = (bytes[6] & 0x0f) | 0x40; | |
| bytes[8] = (bytes[8] & 0x3f) | 0x80; | |
| return bytes; | |
| } | |
| static isValid(input) { | |
| if (!input) { | |
| return false; | |
| } | |
| if (typeof input === 'string') { | |
| return UUID.isValidUUIDString(input); | |
| } | |
| if (isUint8Array(input)) { | |
| return input.byteLength === UUID_BYTE_LENGTH; | |
| } | |
| return (input._bsontype === 'Binary' && | |
| input.sub_type === this.SUBTYPE_UUID && | |
| input.buffer.byteLength === 16); | |
| } | |
| static createFromHexString(hexString) { | |
| const buffer = UUID.bytesFromString(hexString); | |
| return new UUID(buffer); | |
| } | |
| static createFromBase64(base64) { | |
| return new UUID(ByteUtils.fromBase64(base64)); | |
| } | |
| static bytesFromString(representation) { | |
| if (!UUID.isValidUUIDString(representation)) { | |
| throw new BSONError('UUID string representation must be 32 hex digits or canonical hyphenated representation'); | |
| } | |
| return ByteUtils.fromHex(representation.replace(/-/g, '')); | |
| } | |
| static isValidUUIDString(representation) { | |
| return UUID_WITHOUT_DASHES.test(representation) || UUID_WITH_DASHES.test(representation); | |
| } | |
| inspect(depth, options, inspect) { | |
| inspect ??= defaultInspect; | |
| return `new UUID(${inspect(this.toHexString(), options)})`; | |
| } | |
| } | |
| class Code extends BSONValue { | |
| get _bsontype() { | |
| return 'Code'; | |
| } | |
| code; | |
| scope; | |
| constructor(code, scope) { | |
| super(); | |
| this.code = code.toString(); | |
| this.scope = scope ?? null; | |
| } | |
| toJSON() { | |
| if (this.scope != null) { | |
| return { code: this.code, scope: this.scope }; | |
| } | |
| return { code: this.code }; | |
| } | |
| toExtendedJSON() { | |
| if (this.scope) { | |
| return { $code: this.code, $scope: this.scope }; | |
| } | |
| return { $code: this.code }; | |
| } | |
| static fromExtendedJSON(doc) { | |
| return new Code(doc.$code, doc.$scope); | |
| } | |
| inspect(depth, options, inspect) { | |
| inspect ??= defaultInspect; | |
| let parametersString = inspect(this.code, options); | |
| const multiLineFn = parametersString.includes('\n'); | |
| if (this.scope != null) { | |
| parametersString += `,${multiLineFn ? '\n' : ' '}${inspect(this.scope, options)}`; | |
| } | |
| const endingNewline = multiLineFn && this.scope === null; | |
| return `new Code(${multiLineFn ? '\n' : ''}${parametersString}${endingNewline ? '\n' : ''})`; | |
| } | |
| } | |
| function isDBRefLike(value) { | |
| return (value != null && | |
| typeof value === 'object' && | |
| '$id' in value && | |
| value.$id != null && | |
| '$ref' in value && | |
| typeof value.$ref === 'string' && | |
| (!('$db' in value) || ('$db' in value && typeof value.$db === 'string'))); | |
| } | |
| class DBRef extends BSONValue { | |
| get _bsontype() { | |
| return 'DBRef'; | |
| } | |
| collection; | |
| oid; | |
| db; | |
| fields; | |
| constructor(collection, oid, db, fields) { | |
| super(); | |
| const parts = collection.split('.'); | |
| if (parts.length === 2) { | |
| db = parts.shift(); | |
| collection = parts.shift(); | |
| } | |
| this.collection = collection; | |
| this.oid = oid; | |
| this.db = db; | |
| this.fields = fields || {}; | |
| } | |
| get namespace() { | |
| return this.collection; | |
| } | |
| set namespace(value) { | |
| this.collection = value; | |
| } | |
| toJSON() { | |
| const o = Object.assign({ | |
| $ref: this.collection, | |
| $id: this.oid | |
| }, this.fields); | |
| if (this.db != null) | |
| o.$db = this.db; | |
| return o; | |
| } | |
| toExtendedJSON(options) { | |
| options = options || {}; | |
| let o = { | |
| $ref: this.collection, | |
| $id: this.oid | |
| }; | |
| if (options.legacy) { | |
| return o; | |
| } | |
| if (this.db) | |
| o.$db = this.db; | |
| o = Object.assign(o, this.fields); | |
| return o; | |
| } | |
| static fromExtendedJSON(doc) { | |
| const copy = Object.assign({}, doc); | |
| delete copy.$ref; | |
| delete copy.$id; | |
| delete copy.$db; | |
| return new DBRef(doc.$ref, doc.$id, doc.$db, copy); | |
| } | |
| inspect(depth, options, inspect) { | |
| inspect ??= defaultInspect; | |
| const args = [ | |
| inspect(this.namespace, options), | |
| inspect(this.oid, options), | |
| ...(this.db ? [inspect(this.db, options)] : []), | |
| ...(Object.keys(this.fields).length > 0 ? [inspect(this.fields, options)] : []) | |
| ]; | |
| args[1] = inspect === defaultInspect ? `new ObjectId(${args[1]})` : args[1]; | |
| return `new DBRef(${args.join(', ')})`; | |
| } | |
| } | |
| function removeLeadingZerosAndExplicitPlus(str) { | |
| if (str === '') { | |
| return str; | |
| } | |
| let startIndex = 0; | |
| const isNegative = str[startIndex] === '-'; | |
| const isExplicitlyPositive = str[startIndex] === '+'; | |
| if (isExplicitlyPositive || isNegative) { | |
| startIndex += 1; | |
| } | |
| let foundInsignificantZero = false; | |
| for (; startIndex < str.length && str[startIndex] === '0'; ++startIndex) { | |
| foundInsignificantZero = true; | |
| } | |
| if (!foundInsignificantZero) { | |
| return isExplicitlyPositive ? str.slice(1) : str; | |
| } | |
| return `${isNegative ? '-' : ''}${str.length === startIndex ? '0' : str.slice(startIndex)}`; | |
| } | |
| function validateStringCharacters(str, radix) { | |
| radix = radix ?? 10; | |
| const validCharacters = '0123456789abcdefghijklmnopqrstuvwxyz'.slice(0, radix); | |
| const regex = new RegExp(`[^-+${validCharacters}]`, 'i'); | |
| return regex.test(str) ? false : str; | |
| } | |
| let wasm = undefined; | |
| try { | |
| wasm = new WebAssembly.Instance(new WebAssembly.Module(new Uint8Array([0, 97, 115, 109, 1, 0, 0, 0, 1, 13, 2, 96, 0, 1, 127, 96, 4, 127, 127, 127, 127, 1, 127, 3, 7, 6, 0, 1, 1, 1, 1, 1, 6, 6, 1, 127, 1, 65, 0, 11, 7, 50, 6, 3, 109, 117, 108, 0, 1, 5, 100, 105, 118, 95, 115, 0, 2, 5, 100, 105, 118, 95, 117, 0, 3, 5, 114, 101, 109, 95, 115, 0, 4, 5, 114, 101, 109, 95, 117, 0, 5, 8, 103, 101, 116, 95, 104, 105, 103, 104, 0, 0, 10, 191, 1, 6, 4, 0, 35, 0, 11, 36, 1, 1, 126, 32, 0, 173, 32, 1, 173, 66, 32, 134, 132, 32, 2, 173, 32, 3, 173, 66, 32, 134, 132, 126, 34, 4, 66, 32, 135, 167, 36, 0, 32, 4, 167, 11, 36, 1, 1, 126, 32, 0, 173, 32, 1, 173, 66, 32, 134, 132, 32, 2, 173, 32, 3, 173, 66, 32, 134, 132, 127, 34, 4, 66, 32, 135, 167, 36, 0, 32, 4, 167, 11, 36, 1, 1, 126, 32, 0, 173, 32, 1, 173, 66, 32, 134, 132, 32, 2, 173, 32, 3, 173, 66, 32, 134, 132, 128, 34, 4, 66, 32, 135, 167, 36, 0, 32, 4, 167, 11, 36, 1, 1, 126, 32, 0, 173, 32, 1, 173, 66, 32, 134, 132, 32, 2, 173, 32, 3, 173, 66, 32, 134, 132, 129, 34, 4, 66, 32, 135, 167, 36, 0, 32, 4, 167, 11, 36, 1, 1, 126, 32, 0, 173, 32, 1, 173, 66, 32, 134, 132, 32, 2, 173, 32, 3, 173, 66, 32, 134, 132, 130, 34, 4, 66, 32, 135, 167, 36, 0, 32, 4, 167, 11])), {}).exports; | |
| } | |
| catch { | |
| } | |
| const TWO_PWR_16_DBL = 1 << 16; | |
| const TWO_PWR_24_DBL = 1 << 24; | |
| const TWO_PWR_32_DBL = TWO_PWR_16_DBL * TWO_PWR_16_DBL; | |
| const TWO_PWR_64_DBL = TWO_PWR_32_DBL * TWO_PWR_32_DBL; | |
| const TWO_PWR_63_DBL = TWO_PWR_64_DBL / 2; | |
| const INT_CACHE = {}; | |
| const UINT_CACHE = {}; | |
| const MAX_INT64_STRING_LENGTH = 20; | |
| const DECIMAL_REG_EX = /^(\+?0|(\+|-)?[1-9][0-9]*)$/; | |
| class Long extends BSONValue { | |
| get _bsontype() { | |
| return 'Long'; | |
| } | |
| get __isLong__() { | |
| return true; | |
| } | |
| high; | |
| low; | |
| unsigned; | |
| constructor(lowOrValue = 0, highOrUnsigned, unsigned) { | |
| super(); | |
| const unsignedBool = typeof highOrUnsigned === 'boolean' ? highOrUnsigned : Boolean(unsigned); | |
| const high = typeof highOrUnsigned === 'number' ? highOrUnsigned : 0; | |
| const res = typeof lowOrValue === 'string' | |
| ? Long.fromString(lowOrValue, unsignedBool) | |
| : typeof lowOrValue === 'bigint' | |
| ? Long.fromBigInt(lowOrValue, unsignedBool) | |
| : { low: lowOrValue | 0, high: high | 0, unsigned: unsignedBool }; | |
| this.low = res.low; | |
| this.high = res.high; | |
| this.unsigned = res.unsigned; | |
| } | |
| static TWO_PWR_24 = Long.fromInt(TWO_PWR_24_DBL); | |
| static MAX_UNSIGNED_VALUE = Long.fromBits(0xffffffff | 0, 0xffffffff | 0, true); | |
| static ZERO = Long.fromInt(0); | |
| static UZERO = Long.fromInt(0, true); | |
| static ONE = Long.fromInt(1); | |
| static UONE = Long.fromInt(1, true); | |
| static NEG_ONE = Long.fromInt(-1); | |
| static MAX_VALUE = Long.fromBits(0xffffffff | 0, 0x7fffffff | 0, false); | |
| static MIN_VALUE = Long.fromBits(0, 0x80000000 | 0, false); | |
| static fromBits(lowBits, highBits, unsigned) { | |
| return new Long(lowBits, highBits, unsigned); | |
| } | |
| static fromInt(value, unsigned) { | |
| let obj, cachedObj, cache; | |
| if (unsigned) { | |
| value >>>= 0; | |
| if ((cache = 0 <= value && value < 256)) { | |
| cachedObj = UINT_CACHE[value]; | |
| if (cachedObj) | |
| return cachedObj; | |
| } | |
| obj = Long.fromBits(value, (value | 0) < 0 ? -1 : 0, true); | |
| if (cache) | |
| UINT_CACHE[value] = obj; | |
| return obj; | |
| } | |
| else { | |
| value |= 0; | |
| if ((cache = -128 <= value && value < 128)) { | |
| cachedObj = INT_CACHE[value]; | |
| if (cachedObj) | |
| return cachedObj; | |
| } | |
| obj = Long.fromBits(value, value < 0 ? -1 : 0, false); | |
| if (cache) | |
| INT_CACHE[value] = obj; | |
| return obj; | |
| } | |
| } | |
| static fromNumber(value, unsigned) { | |
| if (isNaN(value)) | |
| return unsigned ? Long.UZERO : Long.ZERO; | |
| if (unsigned) { | |
| if (value < 0) | |
| return Long.UZERO; | |
| if (value >= TWO_PWR_64_DBL) | |
| return Long.MAX_UNSIGNED_VALUE; | |
| } | |
| else { | |
| if (value <= -TWO_PWR_63_DBL) | |
| return Long.MIN_VALUE; | |
| if (value + 1 >= TWO_PWR_63_DBL) | |
| return Long.MAX_VALUE; | |
| } | |
| if (value < 0) | |
| return Long.fromNumber(-value, unsigned).neg(); | |
| return Long.fromBits(value % TWO_PWR_32_DBL | 0, (value / TWO_PWR_32_DBL) | 0, unsigned); | |
| } | |
| static fromBigInt(value, unsigned) { | |
| const FROM_BIGINT_BIT_MASK = 0xffffffffn; | |
| const FROM_BIGINT_BIT_SHIFT = 32n; | |
| return new Long(Number(value & FROM_BIGINT_BIT_MASK), Number((value >> FROM_BIGINT_BIT_SHIFT) & FROM_BIGINT_BIT_MASK), unsigned); | |
| } | |
| static _fromString(str, unsigned, radix) { | |
| if (str.length === 0) | |
| throw new BSONError('empty string'); | |
| if (radix < 2 || 36 < radix) | |
| throw new BSONError('radix'); | |
| let p; | |
| if ((p = str.indexOf('-')) > 0) | |
| throw new BSONError('interior hyphen'); | |
| else if (p === 0) { | |
| return Long._fromString(str.substring(1), unsigned, radix).neg(); | |
| } | |
| const radixToPower = Long.fromNumber(Math.pow(radix, 8)); | |
| let result = Long.ZERO; | |
| for (let i = 0; i < str.length; i += 8) { | |
| const size = Math.min(8, str.length - i), value = parseInt(str.substring(i, i + size), radix); | |
| if (size < 8) { | |
| const power = Long.fromNumber(Math.pow(radix, size)); | |
| result = result.mul(power).add(Long.fromNumber(value)); | |
| } | |
| else { | |
| result = result.mul(radixToPower); | |
| result = result.add(Long.fromNumber(value)); | |
| } | |
| } | |
| result.unsigned = unsigned; | |
| return result; | |
| } | |
| static fromStringStrict(str, unsignedOrRadix, radix) { | |
| let unsigned = false; | |
| if (typeof unsignedOrRadix === 'number') { | |
| ((radix = unsignedOrRadix), (unsignedOrRadix = false)); | |
| } | |
| else { | |
| unsigned = !!unsignedOrRadix; | |
| } | |
| radix ??= 10; | |
| if (str.trim() !== str) { | |
| throw new BSONError(`Input: '${str}' contains leading and/or trailing whitespace`); | |
| } | |
| if (!validateStringCharacters(str, radix)) { | |
| throw new BSONError(`Input: '${str}' contains invalid characters for radix: ${radix}`); | |
| } | |
| const cleanedStr = removeLeadingZerosAndExplicitPlus(str); | |
| const result = Long._fromString(cleanedStr, unsigned, radix); | |
| if (result.toString(radix).toLowerCase() !== cleanedStr.toLowerCase()) { | |
| throw new BSONError(`Input: ${str} is not representable as ${result.unsigned ? 'an unsigned' : 'a signed'} 64-bit Long ${radix != null ? `with radix: ${radix}` : ''}`); | |
| } | |
| return result; | |
| } | |
| static fromString(str, unsignedOrRadix, radix) { | |
| let unsigned = false; | |
| if (typeof unsignedOrRadix === 'number') { | |
| ((radix = unsignedOrRadix), (unsignedOrRadix = false)); | |
| } | |
| else { | |
| unsigned = !!unsignedOrRadix; | |
| } | |
| radix ??= 10; | |
| if (str === 'NaN' && radix < 24) { | |
| return Long.ZERO; | |
| } | |
| else if ((str === 'Infinity' || str === '+Infinity' || str === '-Infinity') && radix < 35) { | |
| return Long.ZERO; | |
| } | |
| return Long._fromString(str, unsigned, radix); | |
| } | |
| static fromBytes(bytes, unsigned, le) { | |
| return le ? Long.fromBytesLE(bytes, unsigned) : Long.fromBytesBE(bytes, unsigned); | |
| } | |
| static fromBytesLE(bytes, unsigned) { | |
| return new Long(bytes[0] | (bytes[1] << 8) | (bytes[2] << 16) | (bytes[3] << 24), bytes[4] | (bytes[5] << 8) | (bytes[6] << 16) | (bytes[7] << 24), unsigned); | |
| } | |
| static fromBytesBE(bytes, unsigned) { | |
| return new Long((bytes[4] << 24) | (bytes[5] << 16) | (bytes[6] << 8) | bytes[7], (bytes[0] << 24) | (bytes[1] << 16) | (bytes[2] << 8) | bytes[3], unsigned); | |
| } | |
| static isLong(value) { | |
| return (value != null && | |
| typeof value === 'object' && | |
| '__isLong__' in value && | |
| value.__isLong__ === true); | |
| } | |
| static fromValue(val, unsigned) { | |
| if (typeof val === 'number') | |
| return Long.fromNumber(val, unsigned); | |
| if (typeof val === 'string') | |
| return Long.fromString(val, unsigned); | |
| return Long.fromBits(val.low, val.high, typeof unsigned === 'boolean' ? unsigned : val.unsigned); | |
| } | |
| add(addend) { | |
| if (!Long.isLong(addend)) | |
| addend = Long.fromValue(addend); | |
| const a48 = this.high >>> 16; | |
| const a32 = this.high & 0xffff; | |
| const a16 = this.low >>> 16; | |
| const a00 = this.low & 0xffff; | |
| const b48 = addend.high >>> 16; | |
| const b32 = addend.high & 0xffff; | |
| const b16 = addend.low >>> 16; | |
| const b00 = addend.low & 0xffff; | |
| let c48 = 0, c32 = 0, c16 = 0, c00 = 0; | |
| c00 += a00 + b00; | |
| c16 += c00 >>> 16; | |
| c00 &= 0xffff; | |
| c16 += a16 + b16; | |
| c32 += c16 >>> 16; | |
| c16 &= 0xffff; | |
| c32 += a32 + b32; | |
| c48 += c32 >>> 16; | |
| c32 &= 0xffff; | |
| c48 += a48 + b48; | |
| c48 &= 0xffff; | |
| return Long.fromBits((c16 << 16) | c00, (c48 << 16) | c32, this.unsigned); | |
| } | |
| and(other) { | |
| if (!Long.isLong(other)) | |
| other = Long.fromValue(other); | |
| return Long.fromBits(this.low & other.low, this.high & other.high, this.unsigned); | |
| } | |
| compare(other) { | |
| if (!Long.isLong(other)) | |
| other = Long.fromValue(other); | |
| if (this.eq(other)) | |
| return 0; | |
| const thisNeg = this.isNegative(), otherNeg = other.isNegative(); | |
| if (thisNeg && !otherNeg) | |
| return -1; | |
| if (!thisNeg && otherNeg) | |
| return 1; | |
| if (!this.unsigned) | |
| return this.sub(other).isNegative() ? -1 : 1; | |
| return other.high >>> 0 > this.high >>> 0 || | |
| (other.high === this.high && other.low >>> 0 > this.low >>> 0) | |
| ? -1 | |
| : 1; | |
| } | |
| comp(other) { | |
| return this.compare(other); | |
| } | |
| divide(divisor) { | |
| if (!Long.isLong(divisor)) | |
| divisor = Long.fromValue(divisor); | |
| if (divisor.isZero()) | |
| throw new BSONError('division by zero'); | |
| if (wasm) { | |
| if (!this.unsigned && | |
| this.high === -2147483648 && | |
| divisor.low === -1 && | |
| divisor.high === -1) { | |
| return this; | |
| } | |
| const low = (this.unsigned ? wasm.div_u : wasm.div_s)(this.low, this.high, divisor.low, divisor.high); | |
| return Long.fromBits(low, wasm.get_high(), this.unsigned); | |
| } | |
| if (this.isZero()) | |
| return this.unsigned ? Long.UZERO : Long.ZERO; | |
| let approx, rem, res; | |
| if (!this.unsigned) { | |
| if (this.eq(Long.MIN_VALUE)) { | |
| if (divisor.eq(Long.ONE) || divisor.eq(Long.NEG_ONE)) | |
| return Long.MIN_VALUE; | |
| else if (divisor.eq(Long.MIN_VALUE)) | |
| return Long.ONE; | |
| else { | |
| const halfThis = this.shr(1); | |
| approx = halfThis.div(divisor).shl(1); | |
| if (approx.eq(Long.ZERO)) { | |
| return divisor.isNegative() ? Long.ONE : Long.NEG_ONE; | |
| } | |
| else { | |
| rem = this.sub(divisor.mul(approx)); | |
| res = approx.add(rem.div(divisor)); | |
| return res; | |
| } | |
| } | |
| } | |
| else if (divisor.eq(Long.MIN_VALUE)) | |
| return this.unsigned ? Long.UZERO : Long.ZERO; | |
| if (this.isNegative()) { | |
| if (divisor.isNegative()) | |
| return this.neg().div(divisor.neg()); | |
| return this.neg().div(divisor).neg(); | |
| } | |
| else if (divisor.isNegative()) | |
| return this.div(divisor.neg()).neg(); | |
| res = Long.ZERO; | |
| } | |
| else { | |
| if (!divisor.unsigned) | |
| divisor = divisor.toUnsigned(); | |
| if (divisor.gt(this)) | |
| return Long.UZERO; | |
| if (divisor.gt(this.shru(1))) | |
| return Long.UONE; | |
| res = Long.UZERO; | |
| } | |
| rem = this; | |
| while (rem.gte(divisor)) { | |
| approx = Math.max(1, Math.floor(rem.toNumber() / divisor.toNumber())); | |
| const log2 = Math.ceil(Math.log(approx) / Math.LN2); | |
| const delta = log2 <= 48 ? 1 : Math.pow(2, log2 - 48); | |
| let approxRes = Long.fromNumber(approx); | |
| let approxRem = approxRes.mul(divisor); | |
| while (approxRem.isNegative() || approxRem.gt(rem)) { | |
| approx -= delta; | |
| approxRes = Long.fromNumber(approx, this.unsigned); | |
| approxRem = approxRes.mul(divisor); | |
| } | |
| if (approxRes.isZero()) | |
| approxRes = Long.ONE; | |
| res = res.add(approxRes); | |
| rem = rem.sub(approxRem); | |
| } | |
| return res; | |
| } | |
| div(divisor) { | |
| return this.divide(divisor); | |
| } | |
| equals(other) { | |
| if (!Long.isLong(other)) | |
| other = Long.fromValue(other); | |
| if (this.unsigned !== other.unsigned && this.high >>> 31 === 1 && other.high >>> 31 === 1) | |
| return false; | |
| return this.high === other.high && this.low === other.low; | |
| } | |
| eq(other) { | |
| return this.equals(other); | |
| } | |
| getHighBits() { | |
| return this.high; | |
| } | |
| getHighBitsUnsigned() { | |
| return this.high >>> 0; | |
| } | |
| getLowBits() { | |
| return this.low; | |
| } | |
| getLowBitsUnsigned() { | |
| return this.low >>> 0; | |
| } | |
| getNumBitsAbs() { | |
| if (this.isNegative()) { | |
| return this.eq(Long.MIN_VALUE) ? 64 : this.neg().getNumBitsAbs(); | |
| } | |
| const val = this.high !== 0 ? this.high : this.low; | |
| let bit; | |
| for (bit = 31; bit > 0; bit--) | |
| if ((val & (1 << bit)) !== 0) | |
| break; | |
| return this.high !== 0 ? bit + 33 : bit + 1; | |
| } | |
| greaterThan(other) { | |
| return this.comp(other) > 0; | |
| } | |
| gt(other) { | |
| return this.greaterThan(other); | |
| } | |
| greaterThanOrEqual(other) { | |
| return this.comp(other) >= 0; | |
| } | |
| gte(other) { | |
| return this.greaterThanOrEqual(other); | |
| } | |
| ge(other) { | |
| return this.greaterThanOrEqual(other); | |
| } | |
| isEven() { | |
| return (this.low & 1) === 0; | |
| } | |
| isNegative() { | |
| return !this.unsigned && this.high < 0; | |
| } | |
| isOdd() { | |
| return (this.low & 1) === 1; | |
| } | |
| isPositive() { | |
| return this.unsigned || this.high >= 0; | |
| } | |
| isZero() { | |
| return this.high === 0 && this.low === 0; | |
| } | |
| lessThan(other) { | |
| return this.comp(other) < 0; | |
| } | |
| lt(other) { | |
| return this.lessThan(other); | |
| } | |
| lessThanOrEqual(other) { | |
| return this.comp(other) <= 0; | |
| } | |
| lte(other) { | |
| return this.lessThanOrEqual(other); | |
| } | |
| modulo(divisor) { | |
| if (!Long.isLong(divisor)) | |
| divisor = Long.fromValue(divisor); | |
| if (wasm) { | |
| const low = (this.unsigned ? wasm.rem_u : wasm.rem_s)(this.low, this.high, divisor.low, divisor.high); | |
| return Long.fromBits(low, wasm.get_high(), this.unsigned); | |
| } | |
| return this.sub(this.div(divisor).mul(divisor)); | |
| } | |
| mod(divisor) { | |
| return this.modulo(divisor); | |
| } | |
| rem(divisor) { | |
| return this.modulo(divisor); | |
| } | |
| multiply(multiplier) { | |
| if (this.isZero()) | |
| return Long.ZERO; | |
| if (!Long.isLong(multiplier)) | |
| multiplier = Long.fromValue(multiplier); | |
| if (wasm) { | |
| const low = wasm.mul(this.low, this.high, multiplier.low, multiplier.high); | |
| return Long.fromBits(low, wasm.get_high(), this.unsigned); | |
| } | |
| if (multiplier.isZero()) | |
| return Long.ZERO; | |
| if (this.eq(Long.MIN_VALUE)) | |
| return multiplier.isOdd() ? Long.MIN_VALUE : Long.ZERO; | |
| if (multiplier.eq(Long.MIN_VALUE)) | |
| return this.isOdd() ? Long.MIN_VALUE : Long.ZERO; | |
| if (this.isNegative()) { | |
| if (multiplier.isNegative()) | |
| return this.neg().mul(multiplier.neg()); | |
| else | |
| return this.neg().mul(multiplier).neg(); | |
| } | |
| else if (multiplier.isNegative()) | |
| return this.mul(multiplier.neg()).neg(); | |
| if (this.lt(Long.TWO_PWR_24) && multiplier.lt(Long.TWO_PWR_24)) | |
| return Long.fromNumber(this.toNumber() * multiplier.toNumber(), this.unsigned); | |
| const a48 = this.high >>> 16; | |
| const a32 = this.high & 0xffff; | |
| const a16 = this.low >>> 16; | |
| const a00 = this.low & 0xffff; | |
| const b48 = multiplier.high >>> 16; | |
| const b32 = multiplier.high & 0xffff; | |
| const b16 = multiplier.low >>> 16; | |
| const b00 = multiplier.low & 0xffff; | |
| let c48 = 0, c32 = 0, c16 = 0, c00 = 0; | |
| c00 += a00 * b00; | |
| c16 += c00 >>> 16; | |
| c00 &= 0xffff; | |
| c16 += a16 * b00; | |
| c32 += c16 >>> 16; | |
| c16 &= 0xffff; | |
| c16 += a00 * b16; | |
| c32 += c16 >>> 16; | |
| c16 &= 0xffff; | |
| c32 += a32 * b00; | |
| c48 += c32 >>> 16; | |
| c32 &= 0xffff; | |
| c32 += a16 * b16; | |
| c48 += c32 >>> 16; | |
| c32 &= 0xffff; | |
| c32 += a00 * b32; | |
| c48 += c32 >>> 16; | |
| c32 &= 0xffff; | |
| c48 += a48 * b00 + a32 * b16 + a16 * b32 + a00 * b48; | |
| c48 &= 0xffff; | |
| return Long.fromBits((c16 << 16) | c00, (c48 << 16) | c32, this.unsigned); | |
| } | |
| mul(multiplier) { | |
| return this.multiply(multiplier); | |
| } | |
| negate() { | |
| if (!this.unsigned && this.eq(Long.MIN_VALUE)) | |
| return Long.MIN_VALUE; | |
| return this.not().add(Long.ONE); | |
| } | |
| neg() { | |
| return this.negate(); | |
| } | |
| not() { | |
| return Long.fromBits(~this.low, ~this.high, this.unsigned); | |
| } | |
| notEquals(other) { | |
| return !this.equals(other); | |
| } | |
| neq(other) { | |
| return this.notEquals(other); | |
| } | |
| ne(other) { | |
| return this.notEquals(other); | |
| } | |
| or(other) { | |
| if (!Long.isLong(other)) | |
| other = Long.fromValue(other); | |
| return Long.fromBits(this.low | other.low, this.high | other.high, this.unsigned); | |
| } | |
| shiftLeft(numBits) { | |
| if (Long.isLong(numBits)) | |
| numBits = numBits.toInt(); | |
| if ((numBits &= 63) === 0) | |
| return this; | |
| else if (numBits < 32) | |
| return Long.fromBits(this.low << numBits, (this.high << numBits) | (this.low >>> (32 - numBits)), this.unsigned); | |
| else | |
| return Long.fromBits(0, this.low << (numBits - 32), this.unsigned); | |
| } | |
| shl(numBits) { | |
| return this.shiftLeft(numBits); | |
| } | |
| shiftRight(numBits) { | |
| if (Long.isLong(numBits)) | |
| numBits = numBits.toInt(); | |
| if ((numBits &= 63) === 0) | |
| return this; | |
| else if (numBits < 32) | |
| return Long.fromBits((this.low >>> numBits) | (this.high << (32 - numBits)), this.high >> numBits, this.unsigned); | |
| else | |
| return Long.fromBits(this.high >> (numBits - 32), this.high >= 0 ? 0 : -1, this.unsigned); | |
| } | |
| shr(numBits) { | |
| return this.shiftRight(numBits); | |
| } | |
| shiftRightUnsigned(numBits) { | |
| if (Long.isLong(numBits)) | |
| numBits = numBits.toInt(); | |
| numBits &= 63; | |
| if (numBits === 0) | |
| return this; | |
| else { | |
| const high = this.high; | |
| if (numBits < 32) { | |
| const low = this.low; | |
| return Long.fromBits((low >>> numBits) | (high << (32 - numBits)), high >>> numBits, this.unsigned); | |
| } | |
| else if (numBits === 32) | |
| return Long.fromBits(high, 0, this.unsigned); | |
| else | |
| return Long.fromBits(high >>> (numBits - 32), 0, this.unsigned); | |
| } | |
| } | |
| shr_u(numBits) { | |
| return this.shiftRightUnsigned(numBits); | |
| } | |
| shru(numBits) { | |
| return this.shiftRightUnsigned(numBits); | |
| } | |
| subtract(subtrahend) { | |
| if (!Long.isLong(subtrahend)) | |
| subtrahend = Long.fromValue(subtrahend); | |
| return this.add(subtrahend.neg()); | |
| } | |
| sub(subtrahend) { | |
| return this.subtract(subtrahend); | |
| } | |
| toInt() { | |
| return this.unsigned ? this.low >>> 0 : this.low; | |
| } | |
| toNumber() { | |
| if (this.unsigned) | |
| return (this.high >>> 0) * TWO_PWR_32_DBL + (this.low >>> 0); | |
| return this.high * TWO_PWR_32_DBL + (this.low >>> 0); | |
| } | |
| toBigInt() { | |
| return BigInt(this.toString()); | |
| } | |
| toBytes(le) { | |
| return le ? this.toBytesLE() : this.toBytesBE(); | |
| } | |
| toBytesLE() { | |
| const hi = this.high, lo = this.low; | |
| return [ | |
| lo & 0xff, | |
| (lo >>> 8) & 0xff, | |
| (lo >>> 16) & 0xff, | |
| lo >>> 24, | |
| hi & 0xff, | |
| (hi >>> 8) & 0xff, | |
| (hi >>> 16) & 0xff, | |
| hi >>> 24 | |
| ]; | |
| } | |
| toBytesBE() { | |
| const hi = this.high, lo = this.low; | |
| return [ | |
| hi >>> 24, | |
| (hi >>> 16) & 0xff, | |
| (hi >>> 8) & 0xff, | |
| hi & 0xff, | |
| lo >>> 24, | |
| (lo >>> 16) & 0xff, | |
| (lo >>> 8) & 0xff, | |
| lo & 0xff | |
| ]; | |
| } | |
| toSigned() { | |
| if (!this.unsigned) | |
| return this; | |
| return Long.fromBits(this.low, this.high, false); | |
| } | |
| toString(radix) { | |
| radix = radix || 10; | |
| if (radix < 2 || 36 < radix) | |
| throw new BSONError('radix'); | |
| if (this.isZero()) | |
| return '0'; | |
| if (this.isNegative()) { | |
| if (this.eq(Long.MIN_VALUE)) { | |
| const radixLong = Long.fromNumber(radix), div = this.div(radixLong), rem1 = div.mul(radixLong).sub(this); | |
| return div.toString(radix) + rem1.toInt().toString(radix); | |
| } | |
| else | |
| return '-' + this.neg().toString(radix); | |
| } | |
| const radixToPower = Long.fromNumber(Math.pow(radix, 6), this.unsigned); | |
| let rem = this; | |
| let result = ''; | |
| while (true) { | |
| const remDiv = rem.div(radixToPower); | |
| const intval = rem.sub(remDiv.mul(radixToPower)).toInt() >>> 0; | |
| let digits = intval.toString(radix); | |
| rem = remDiv; | |
| if (rem.isZero()) { | |
| return digits + result; | |
| } | |
| else { | |
| while (digits.length < 6) | |
| digits = '0' + digits; | |
| result = '' + digits + result; | |
| } | |
| } | |
| } | |
| toUnsigned() { | |
| if (this.unsigned) | |
| return this; | |
| return Long.fromBits(this.low, this.high, true); | |
| } | |
| xor(other) { | |
| if (!Long.isLong(other)) | |
| other = Long.fromValue(other); | |
| return Long.fromBits(this.low ^ other.low, this.high ^ other.high, this.unsigned); | |
| } | |
| eqz() { | |
| return this.isZero(); | |
| } | |
| le(other) { | |
| return this.lessThanOrEqual(other); | |
| } | |
| toExtendedJSON(options) { | |
| if (options && options.relaxed) | |
| return this.toNumber(); | |
| return { $numberLong: this.toString() }; | |
| } | |
| static fromExtendedJSON(doc, options) { | |
| const { useBigInt64 = false, relaxed = true } = { ...options }; | |
| if (doc.$numberLong.length > MAX_INT64_STRING_LENGTH) { | |
| throw new BSONError('$numberLong string is too long'); | |
| } | |
| if (!DECIMAL_REG_EX.test(doc.$numberLong)) { | |
| throw new BSONError(`$numberLong string "${doc.$numberLong}" is in an invalid format`); | |
| } | |
| if (useBigInt64) { | |
| const bigIntResult = BigInt(doc.$numberLong); | |
| return BigInt.asIntN(64, bigIntResult); | |
| } | |
| const longResult = Long.fromString(doc.$numberLong); | |
| if (relaxed) { | |
| return longResult.toNumber(); | |
| } | |
| return longResult; | |
| } | |
| inspect(depth, options, inspect) { | |
| inspect ??= defaultInspect; | |
| const longVal = inspect(this.toString(), options); | |
| const unsignedVal = this.unsigned ? `, ${inspect(this.unsigned, options)}` : ''; | |
| return `new Long(${longVal}${unsignedVal})`; | |
| } | |
| } | |
| const PARSE_STRING_REGEXP = /^(\+|-)?(\d+|(\d*\.\d*))?(E|e)?([-+])?(\d+)?$/; | |
| const PARSE_INF_REGEXP = /^(\+|-)?(Infinity|inf)$/i; | |
| const PARSE_NAN_REGEXP = /^(\+|-)?NaN$/i; | |
| const EXPONENT_MAX = 6111; | |
| const EXPONENT_MIN = -6176; | |
| const EXPONENT_BIAS = 6176; | |
| const MAX_DIGITS = 34; | |
| const NAN_BUFFER = ByteUtils.fromNumberArray([ | |
| 0x7c, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 | |
| ].reverse()); | |
| const INF_NEGATIVE_BUFFER = ByteUtils.fromNumberArray([ | |
| 0xf8, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 | |
| ].reverse()); | |
| const INF_POSITIVE_BUFFER = ByteUtils.fromNumberArray([ | |
| 0x78, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00 | |
| ].reverse()); | |
| const EXPONENT_REGEX = /^([-+])?(\d+)?$/; | |
| const COMBINATION_MASK = 0x1f; | |
| const EXPONENT_MASK = 0x3fff; | |
| const COMBINATION_INFINITY = 30; | |
| const COMBINATION_NAN = 31; | |
| function isDigit(value) { | |
| return !isNaN(parseInt(value, 10)); | |
| } | |
| function divideu128(value) { | |
| const DIVISOR = Long.fromNumber(1000 * 1000 * 1000); | |
| let _rem = Long.fromNumber(0); | |
| if (!value.parts[0] && !value.parts[1] && !value.parts[2] && !value.parts[3]) { | |
| return { quotient: value, rem: _rem }; | |
| } | |
| for (let i = 0; i <= 3; i++) { | |
| _rem = _rem.shiftLeft(32); | |
| _rem = _rem.add(new Long(value.parts[i], 0)); | |
| value.parts[i] = _rem.div(DIVISOR).low; | |
| _rem = _rem.modulo(DIVISOR); | |
| } | |
| return { quotient: value, rem: _rem }; | |
| } | |
| function multiply64x2(left, right) { | |
| if (!left && !right) { | |
| return { high: Long.fromNumber(0), low: Long.fromNumber(0) }; | |
| } | |
| const leftHigh = left.shiftRightUnsigned(32); | |
| const leftLow = new Long(left.getLowBits(), 0); | |
| const rightHigh = right.shiftRightUnsigned(32); | |
| const rightLow = new Long(right.getLowBits(), 0); | |
| let productHigh = leftHigh.multiply(rightHigh); | |
| let productMid = leftHigh.multiply(rightLow); | |
| const productMid2 = leftLow.multiply(rightHigh); | |
| let productLow = leftLow.multiply(rightLow); | |
| productHigh = productHigh.add(productMid.shiftRightUnsigned(32)); | |
| productMid = new Long(productMid.getLowBits(), 0) | |
| .add(productMid2) | |
| .add(productLow.shiftRightUnsigned(32)); | |
| productHigh = productHigh.add(productMid.shiftRightUnsigned(32)); | |
| productLow = productMid.shiftLeft(32).add(new Long(productLow.getLowBits(), 0)); | |
| return { high: productHigh, low: productLow }; | |
| } | |
| function lessThan(left, right) { | |
| const uhleft = left.high >>> 0; | |
| const uhright = right.high >>> 0; | |
| if (uhleft < uhright) { | |
| return true; | |
| } | |
| else if (uhleft === uhright) { | |
| const ulleft = left.low >>> 0; | |
| const ulright = right.low >>> 0; | |
| if (ulleft < ulright) | |
| return true; | |
| } | |
| return false; | |
| } | |
| function invalidErr(string, message) { | |
| throw new BSONError(`"${string}" is not a valid Decimal128 string - ${message}`); | |
| } | |
| class Decimal128 extends BSONValue { | |
| get _bsontype() { | |
| return 'Decimal128'; | |
| } | |
| bytes; | |
| constructor(bytes) { | |
| super(); | |
| if (typeof bytes === 'string') { | |
| this.bytes = Decimal128.fromString(bytes).bytes; | |
| } | |
| else if (bytes instanceof Uint8Array || isUint8Array(bytes)) { | |
| if (bytes.byteLength !== 16) { | |
| throw new BSONError('Decimal128 must take a Buffer of 16 bytes'); | |
| } | |
| this.bytes = bytes; | |
| } | |
| else { | |
| throw new BSONError('Decimal128 must take a Buffer or string'); | |
| } | |
| } | |
| static fromString(representation) { | |
| return Decimal128._fromString(representation, { allowRounding: false }); | |
| } | |
| static fromStringWithRounding(representation) { | |
| return Decimal128._fromString(representation, { allowRounding: true }); | |
| } | |
| static _fromString(representation, options) { | |
| let isNegative = false; | |
| let sawSign = false; | |
| let sawRadix = false; | |
| let foundNonZero = false; | |
| let significantDigits = 0; | |
| let nDigitsRead = 0; | |
| let nDigits = 0; | |
| let radixPosition = 0; | |
| let firstNonZero = 0; | |
| const digits = [0]; | |
| let nDigitsStored = 0; | |
| let digitsInsert = 0; | |
| let lastDigit = 0; | |
| let exponent = 0; | |
| let significandHigh = new Long(0, 0); | |
| let significandLow = new Long(0, 0); | |
| let biasedExponent = 0; | |
| let index = 0; | |
| if (representation.length >= 7000) { | |
| throw new BSONError('' + representation + ' not a valid Decimal128 string'); | |
| } | |
| const stringMatch = representation.match(PARSE_STRING_REGEXP); | |
| const infMatch = representation.match(PARSE_INF_REGEXP); | |
| const nanMatch = representation.match(PARSE_NAN_REGEXP); | |
| if ((!stringMatch && !infMatch && !nanMatch) || representation.length === 0) { | |
| throw new BSONError('' + representation + ' not a valid Decimal128 string'); | |
| } | |
| if (stringMatch) { | |
| const unsignedNumber = stringMatch[2]; | |
| const e = stringMatch[4]; | |
| const expSign = stringMatch[5]; | |
| const expNumber = stringMatch[6]; | |
| if (e && expNumber === undefined) | |
| invalidErr(representation, 'missing exponent power'); | |
| if (e && unsignedNumber === undefined) | |
| invalidErr(representation, 'missing exponent base'); | |
| if (e === undefined && (expSign || expNumber)) { | |
| invalidErr(representation, 'missing e before exponent'); | |
| } | |
| } | |
| if (representation[index] === '+' || representation[index] === '-') { | |
| sawSign = true; | |
| isNegative = representation[index++] === '-'; | |
| } | |
| if (!isDigit(representation[index]) && representation[index] !== '.') { | |
| if (representation[index] === 'i' || representation[index] === 'I') { | |
| return new Decimal128(isNegative ? INF_NEGATIVE_BUFFER : INF_POSITIVE_BUFFER); | |
| } | |
| else if (representation[index] === 'N') { | |
| return new Decimal128(NAN_BUFFER); | |
| } | |
| } | |
| while (isDigit(representation[index]) || representation[index] === '.') { | |
| if (representation[index] === '.') { | |
| if (sawRadix) | |
| invalidErr(representation, 'contains multiple periods'); | |
| sawRadix = true; | |
| index = index + 1; | |
| continue; | |
| } | |
| if (nDigitsStored < MAX_DIGITS) { | |
| if (representation[index] !== '0' || foundNonZero) { | |
| if (!foundNonZero) { | |
| firstNonZero = nDigitsRead; | |
| } | |
| foundNonZero = true; | |
| digits[digitsInsert++] = parseInt(representation[index], 10); | |
| nDigitsStored = nDigitsStored + 1; | |
| } | |
| } | |
| if (foundNonZero) | |
| nDigits = nDigits + 1; | |
| if (sawRadix) | |
| radixPosition = radixPosition + 1; | |
| nDigitsRead = nDigitsRead + 1; | |
| index = index + 1; | |
| } | |
| if (sawRadix && !nDigitsRead) | |
| throw new BSONError('' + representation + ' not a valid Decimal128 string'); | |
| if (representation[index] === 'e' || representation[index] === 'E') { | |
| const match = representation.substr(++index).match(EXPONENT_REGEX); | |
| if (!match || !match[2]) | |
| return new Decimal128(NAN_BUFFER); | |
| exponent = parseInt(match[0], 10); | |
| index = index + match[0].length; | |
| } | |
| if (representation[index]) | |
| return new Decimal128(NAN_BUFFER); | |
| if (!nDigitsStored) { | |
| digits[0] = 0; | |
| nDigits = 1; | |
| nDigitsStored = 1; | |
| significantDigits = 0; | |
| } | |
| else { | |
| lastDigit = nDigitsStored - 1; | |
| significantDigits = nDigits; | |
| if (significantDigits !== 1) { | |
| while (representation[firstNonZero + significantDigits - 1 + Number(sawSign) + Number(sawRadix)] === '0') { | |
| significantDigits = significantDigits - 1; | |
| } | |
| } | |
| } | |
| if (exponent <= radixPosition && radixPosition > exponent + (1 << 14)) { | |
| exponent = EXPONENT_MIN; | |
| } | |
| else { | |
| exponent = exponent - radixPosition; | |
| } | |
| while (exponent > EXPONENT_MAX) { | |
| lastDigit = lastDigit + 1; | |
| if (lastDigit >= MAX_DIGITS) { | |
| if (significantDigits === 0) { | |
| exponent = EXPONENT_MAX; | |
| break; | |
| } | |
| invalidErr(representation, 'overflow'); | |
| } | |
| exponent = exponent - 1; | |
| } | |
| if (options.allowRounding) { | |
| while (exponent < EXPONENT_MIN || nDigitsStored < nDigits) { | |
| if (lastDigit === 0 && significantDigits < nDigitsStored) { | |
| exponent = EXPONENT_MIN; | |
| significantDigits = 0; | |
| break; | |
| } | |
| if (nDigitsStored < nDigits) { | |
| nDigits = nDigits - 1; | |
| } | |
| else { | |
| lastDigit = lastDigit - 1; | |
| } | |
| if (exponent < EXPONENT_MAX) { | |
| exponent = exponent + 1; | |
| } | |
| else { | |
| const digitsString = digits.join(''); | |
| if (digitsString.match(/^0+$/)) { | |
| exponent = EXPONENT_MAX; | |
| break; | |
| } | |
| invalidErr(representation, 'overflow'); | |
| } | |
| } | |
| if (lastDigit + 1 < significantDigits) { | |
| let endOfString = nDigitsRead; | |
| if (sawRadix) { | |
| firstNonZero = firstNonZero + 1; | |
| endOfString = endOfString + 1; | |
| } | |
| if (sawSign) { | |
| firstNonZero = firstNonZero + 1; | |
| endOfString = endOfString + 1; | |
| } | |
| const roundDigit = parseInt(representation[firstNonZero + lastDigit + 1], 10); | |
| let roundBit = 0; | |
| if (roundDigit >= 5) { | |
| roundBit = 1; | |
| if (roundDigit === 5) { | |
| roundBit = digits[lastDigit] % 2 === 1 ? 1 : 0; | |
| for (let i = firstNonZero + lastDigit + 2; i < endOfString; i++) { | |
| if (parseInt(representation[i], 10)) { | |
| roundBit = 1; | |
| break; | |
| } | |
| } | |
| } | |
| } | |
| if (roundBit) { | |
| let dIdx = lastDigit; | |
| for (; dIdx >= 0; dIdx--) { | |
| if (++digits[dIdx] > 9) { | |
| digits[dIdx] = 0; | |
| if (dIdx === 0) { | |
| if (exponent < EXPONENT_MAX) { | |
| exponent = exponent + 1; | |
| digits[dIdx] = 1; | |
| } | |
| else { | |
| return new Decimal128(isNegative ? INF_NEGATIVE_BUFFER : INF_POSITIVE_BUFFER); | |
| } | |
| } | |
| } | |
| else { | |
| break; | |
| } | |
| } | |
| } | |
| } | |
| } | |
| else { | |
| while (exponent < EXPONENT_MIN || nDigitsStored < nDigits) { | |
| if (lastDigit === 0) { | |
| if (significantDigits === 0) { | |
| exponent = EXPONENT_MIN; | |
| break; | |
| } | |
| invalidErr(representation, 'exponent underflow'); | |
| } | |
| if (nDigitsStored < nDigits) { | |
| if (representation[nDigits - 1 + Number(sawSign) + Number(sawRadix)] !== '0' && | |
| significantDigits !== 0) { | |
| invalidErr(representation, 'inexact rounding'); | |
| } | |
| nDigits = nDigits - 1; | |
| } | |
| else { | |
| if (digits[lastDigit] !== 0) { | |
| invalidErr(representation, 'inexact rounding'); | |
| } | |
| lastDigit = lastDigit - 1; | |
| } | |
| if (exponent < EXPONENT_MAX) { | |
| exponent = exponent + 1; | |
| } | |
| else { | |
| invalidErr(representation, 'overflow'); | |
| } | |
| } | |
| if (lastDigit + 1 < significantDigits) { | |
| if (sawRadix) { | |
| firstNonZero = firstNonZero + 1; | |
| } | |
| if (sawSign) { | |
| firstNonZero = firstNonZero + 1; | |
| } | |
| const roundDigit = parseInt(representation[firstNonZero + lastDigit + 1], 10); | |
| if (roundDigit !== 0) { | |
| invalidErr(representation, 'inexact rounding'); | |
| } | |
| } | |
| } | |
| significandHigh = Long.fromNumber(0); | |
| significandLow = Long.fromNumber(0); | |
| if (significantDigits === 0) { | |
| significandHigh = Long.fromNumber(0); | |
| significandLow = Long.fromNumber(0); | |
| } | |
| else if (lastDigit < 17) { | |
| let dIdx = 0; | |
| significandLow = Long.fromNumber(digits[dIdx++]); | |
| significandHigh = new Long(0, 0); | |
| for (; dIdx <= lastDigit; dIdx++) { | |
| significandLow = significandLow.multiply(Long.fromNumber(10)); | |
| significandLow = significandLow.add(Long.fromNumber(digits[dIdx])); | |
| } | |
| } | |
| else { | |
| let dIdx = 0; | |
| significandHigh = Long.fromNumber(digits[dIdx++]); | |
| for (; dIdx <= lastDigit - 17; dIdx++) { | |
| significandHigh = significandHigh.multiply(Long.fromNumber(10)); | |
| significandHigh = significandHigh.add(Long.fromNumber(digits[dIdx])); | |
| } | |
| significandLow = Long.fromNumber(digits[dIdx++]); | |
| for (; dIdx <= lastDigit; dIdx++) { | |
| significandLow = significandLow.multiply(Long.fromNumber(10)); | |
| significandLow = significandLow.add(Long.fromNumber(digits[dIdx])); | |
| } | |
| } | |
| const significand = multiply64x2(significandHigh, Long.fromString('100000000000000000')); | |
| significand.low = significand.low.add(significandLow); | |
| if (lessThan(significand.low, significandLow)) { | |
| significand.high = significand.high.add(Long.fromNumber(1)); | |
| } | |
| biasedExponent = exponent + EXPONENT_BIAS; | |
| const dec = { low: Long.fromNumber(0), high: Long.fromNumber(0) }; | |
| if (significand.high.shiftRightUnsigned(49).and(Long.fromNumber(1)).equals(Long.fromNumber(1))) { | |
| dec.high = dec.high.or(Long.fromNumber(0x3).shiftLeft(61)); | |
| dec.high = dec.high.or(Long.fromNumber(biasedExponent).and(Long.fromNumber(0x3fff).shiftLeft(47))); | |
| dec.high = dec.high.or(significand.high.and(Long.fromNumber(0x7fffffffffff))); | |
| } | |
| else { | |
| dec.high = dec.high.or(Long.fromNumber(biasedExponent & 0x3fff).shiftLeft(49)); | |
| dec.high = dec.high.or(significand.high.and(Long.fromNumber(0x1ffffffffffff))); | |
| } | |
| dec.low = significand.low; | |
| if (isNegative) { | |
| dec.high = dec.high.or(Long.fromString('9223372036854775808')); | |
| } | |
| const buffer = ByteUtils.allocateUnsafe(16); | |
| index = 0; | |
| buffer[index++] = dec.low.low & 0xff; | |
| buffer[index++] = (dec.low.low >> 8) & 0xff; | |
| buffer[index++] = (dec.low.low >> 16) & 0xff; | |
| buffer[index++] = (dec.low.low >> 24) & 0xff; | |
| buffer[index++] = dec.low.high & 0xff; | |
| buffer[index++] = (dec.low.high >> 8) & 0xff; | |
| buffer[index++] = (dec.low.high >> 16) & 0xff; | |
| buffer[index++] = (dec.low.high >> 24) & 0xff; | |
| buffer[index++] = dec.high.low & 0xff; | |
| buffer[index++] = (dec.high.low >> 8) & 0xff; | |
| buffer[index++] = (dec.high.low >> 16) & 0xff; | |
| buffer[index++] = (dec.high.low >> 24) & 0xff; | |
| buffer[index++] = dec.high.high & 0xff; | |
| buffer[index++] = (dec.high.high >> 8) & 0xff; | |
| buffer[index++] = (dec.high.high >> 16) & 0xff; | |
| buffer[index++] = (dec.high.high >> 24) & 0xff; | |
| return new Decimal128(buffer); | |
| } | |
| toString() { | |
| let biased_exponent; | |
| let significand_digits = 0; | |
| const significand = new Array(36); | |
| for (let i = 0; i < significand.length; i++) | |
| significand[i] = 0; | |
| let index = 0; | |
| let is_zero = false; | |
| let significand_msb; | |
| let significand128 = { parts: [0, 0, 0, 0] }; | |
| let j, k; | |
| const string = []; | |
| index = 0; | |
| const buffer = this.bytes; | |
| const low = buffer[index++] | (buffer[index++] << 8) | (buffer[index++] << 16) | (buffer[index++] << 24); | |
| const midl = buffer[index++] | (buffer[index++] << 8) | (buffer[index++] << 16) | (buffer[index++] << 24); | |
| const midh = buffer[index++] | (buffer[index++] << 8) | (buffer[index++] << 16) | (buffer[index++] << 24); | |
| const high = buffer[index++] | (buffer[index++] << 8) | (buffer[index++] << 16) | (buffer[index++] << 24); | |
| index = 0; | |
| const dec = { | |
| low: new Long(low, midl), | |
| high: new Long(midh, high) | |
| }; | |
| if (dec.high.lessThan(Long.ZERO)) { | |
| string.push('-'); | |
| } | |
| const combination = (high >> 26) & COMBINATION_MASK; | |
| if (combination >> 3 === 3) { | |
| if (combination === COMBINATION_INFINITY) { | |
| return string.join('') + 'Infinity'; | |
| } | |
| else if (combination === COMBINATION_NAN) { | |
| return 'NaN'; | |
| } | |
| else { | |
| biased_exponent = (high >> 15) & EXPONENT_MASK; | |
| significand_msb = 0x08 + ((high >> 14) & 0x01); | |
| } | |
| } | |
| else { | |
| significand_msb = (high >> 14) & 0x07; | |
| biased_exponent = (high >> 17) & EXPONENT_MASK; | |
| } | |
| const exponent = biased_exponent - EXPONENT_BIAS; | |
| significand128.parts[0] = (high & 0x3fff) + ((significand_msb & 0xf) << 14); | |
| significand128.parts[1] = midh; | |
| significand128.parts[2] = midl; | |
| significand128.parts[3] = low; | |
| if (significand128.parts[0] === 0 && | |
| significand128.parts[1] === 0 && | |
| significand128.parts[2] === 0 && | |
| significand128.parts[3] === 0) { | |
| is_zero = true; | |
| } | |
| else { | |
| for (k = 3; k >= 0; k--) { | |
| let least_digits = 0; | |
| const result = divideu128(significand128); | |
| significand128 = result.quotient; | |
| least_digits = result.rem.low; | |
| if (!least_digits) | |
| continue; | |
| for (j = 8; j >= 0; j--) { | |
| significand[k * 9 + j] = least_digits % 10; | |
| least_digits = Math.floor(least_digits / 10); | |
| } | |
| } | |
| } | |
| if (is_zero) { | |
| significand_digits = 1; | |
| significand[index] = 0; | |
| } | |
| else { | |
| significand_digits = 36; | |
| while (!significand[index]) { | |
| significand_digits = significand_digits - 1; | |
| index = index + 1; | |
| } | |
| } | |
| const scientific_exponent = significand_digits - 1 + exponent; | |
| if (scientific_exponent >= 34 || scientific_exponent <= -7 || exponent > 0) { | |
| if (significand_digits > 34) { | |
| string.push(`${0}`); | |
| if (exponent > 0) | |
| string.push(`E+${exponent}`); | |
| else if (exponent < 0) | |
| string.push(`E${exponent}`); | |
| return string.join(''); | |
| } | |
| string.push(`${significand[index++]}`); | |
| significand_digits = significand_digits - 1; | |
| if (significand_digits) { | |
| string.push('.'); | |
| } | |
| for (let i = 0; i < significand_digits; i++) { | |
| string.push(`${significand[index++]}`); | |
| } | |
| string.push('E'); | |
| if (scientific_exponent > 0) { | |
| string.push(`+${scientific_exponent}`); | |
| } | |
| else { | |
| string.push(`${scientific_exponent}`); | |
| } | |
| } | |
| else { | |
| if (exponent >= 0) { | |
| for (let i = 0; i < significand_digits; i++) { | |
| string.push(`${significand[index++]}`); | |
| } | |
| } | |
| else { | |
| let radix_position = significand_digits + exponent; | |
| if (radix_position > 0) { | |
| for (let i = 0; i < radix_position; i++) { | |
| string.push(`${significand[index++]}`); | |
| } | |
| } | |
| else { | |
| string.push('0'); | |
| } | |
| string.push('.'); | |
| while (radix_position++ < 0) { | |
| string.push('0'); | |
| } | |
| for (let i = 0; i < significand_digits - Math.max(radix_position - 1, 0); i++) { | |
| string.push(`${significand[index++]}`); | |
| } | |
| } | |
| } | |
| return string.join(''); | |
| } | |
| toJSON() { | |
| return { $numberDecimal: this.toString() }; | |
| } | |
| toExtendedJSON() { | |
| return { $numberDecimal: this.toString() }; | |
| } | |
| static fromExtendedJSON(doc) { | |
| return Decimal128.fromString(doc.$numberDecimal); | |
| } | |
| inspect(depth, options, inspect) { | |
| inspect ??= defaultInspect; | |
| const d128string = inspect(this.toString(), options); | |
| return `new Decimal128(${d128string})`; | |
| } | |
| } | |
| class Double extends BSONValue { | |
| get _bsontype() { | |
| return 'Double'; | |
| } | |
| value; | |
| constructor(value) { | |
| super(); | |
| if (value instanceof Number) { | |
| value = value.valueOf(); | |
| } | |
| this.value = +value; | |
| } | |
| static fromString(value) { | |
| const coercedValue = Number(value); | |
| if (value === 'NaN') | |
| return new Double(NaN); | |
| if (value === 'Infinity') | |
| return new Double(Infinity); | |
| if (value === '-Infinity') | |
| return new Double(-Infinity); | |
| if (!Number.isFinite(coercedValue)) { | |
| throw new BSONError(`Input: ${value} is not representable as a Double`); | |
| } | |
| if (value.trim() !== value) { | |
| throw new BSONError(`Input: '${value}' contains whitespace`); | |
| } | |
| if (value === '') { | |
| throw new BSONError(`Input is an empty string`); | |
| } | |
| if (/[^-0-9.+eE]/.test(value)) { | |
| throw new BSONError(`Input: '${value}' is not in decimal or exponential notation`); | |
| } | |
| return new Double(coercedValue); | |
| } | |
| valueOf() { | |
| return this.value; | |
| } | |
| toJSON() { | |
| return this.value; | |
| } | |
| toString(radix) { | |
| return this.value.toString(radix); | |
| } | |
| toExtendedJSON(options) { | |
| if (options && (options.legacy || (options.relaxed && isFinite(this.value)))) { | |
| return this.value; | |
| } | |
| if (Object.is(Math.sign(this.value), -0)) { | |
| return { $numberDouble: '-0.0' }; | |
| } | |
| return { | |
| $numberDouble: Number.isInteger(this.value) ? this.value.toFixed(1) : this.value.toString() | |
| }; | |
| } | |
| static fromExtendedJSON(doc, options) { | |
| const doubleValue = parseFloat(doc.$numberDouble); | |
| return options && options.relaxed ? doubleValue : new Double(doubleValue); | |
| } | |
| inspect(depth, options, inspect) { | |
| inspect ??= defaultInspect; | |
| return `new Double(${inspect(this.value, options)})`; | |
| } | |
| } | |
| class Int32 extends BSONValue { | |
| get _bsontype() { | |
| return 'Int32'; | |
| } | |
| value; | |
| constructor(value) { | |
| super(); | |
| if (value instanceof Number) { | |
| value = value.valueOf(); | |
| } | |
| this.value = +value | 0; | |
| } | |
| static fromString(value) { | |
| const cleanedValue = removeLeadingZerosAndExplicitPlus(value); | |
| const coercedValue = Number(value); | |
| if (BSON_INT32_MAX < coercedValue) { | |
| throw new BSONError(`Input: '${value}' is larger than the maximum value for Int32`); | |
| } | |
| else if (BSON_INT32_MIN > coercedValue) { | |
| throw new BSONError(`Input: '${value}' is smaller than the minimum value for Int32`); | |
| } | |
| else if (!Number.isSafeInteger(coercedValue)) { | |
| throw new BSONError(`Input: '${value}' is not a safe integer`); | |
| } | |
| else if (coercedValue.toString() !== cleanedValue) { | |
| throw new BSONError(`Input: '${value}' is not a valid Int32 string`); | |
| } | |
| return new Int32(coercedValue); | |
| } | |
| valueOf() { | |
| return this.value; | |
| } | |
| toString(radix) { | |
| return this.value.toString(radix); | |
| } | |
| toJSON() { | |
| return this.value; | |
| } | |
| toExtendedJSON(options) { | |
| if (options && (options.relaxed || options.legacy)) | |
| return this.value; | |
| return { $numberInt: this.value.toString() }; | |
| } | |
| static fromExtendedJSON(doc, options) { | |
| return options && options.relaxed ? parseInt(doc.$numberInt, 10) : new Int32(doc.$numberInt); | |
| } | |
| inspect(depth, options, inspect) { | |
| inspect ??= defaultInspect; | |
| return `new Int32(${inspect(this.value, options)})`; | |
| } | |
| } | |
| class MaxKey extends BSONValue { | |
| get _bsontype() { | |
| return 'MaxKey'; | |
| } | |
| toExtendedJSON() { | |
| return { $maxKey: 1 }; | |
| } | |
| static fromExtendedJSON() { | |
| return new MaxKey(); | |
| } | |
| inspect() { | |
| return 'new MaxKey()'; | |
| } | |
| } | |
| class MinKey extends BSONValue { | |
| get _bsontype() { | |
| return 'MinKey'; | |
| } | |
| toExtendedJSON() { | |
| return { $minKey: 1 }; | |
| } | |
| static fromExtendedJSON() { | |
| return new MinKey(); | |
| } | |
| inspect() { | |
| return 'new MinKey()'; | |
| } | |
| } | |
| let PROCESS_UNIQUE = null; | |
| const __idCache = new WeakMap(); | |
| class ObjectId extends BSONValue { | |
| get _bsontype() { | |
| return 'ObjectId'; | |
| } | |
| static index = Math.floor(Math.random() * 0xffffff); | |
| static cacheHexString; | |
| buffer; | |
| constructor(inputId) { | |
| super(); | |
| let workingId; | |
| if (typeof inputId === 'object' && inputId && 'id' in inputId) { | |
| if (typeof inputId.id !== 'string' && !ArrayBuffer.isView(inputId.id)) { | |
| throw new BSONError('Argument passed in must have an id that is of type string or Buffer'); | |
| } | |
| if ('toHexString' in inputId && typeof inputId.toHexString === 'function') { | |
| workingId = ByteUtils.fromHex(inputId.toHexString()); | |
| } | |
| else { | |
| workingId = inputId.id; | |
| } | |
| } | |
| else { | |
| workingId = inputId; | |
| } | |
| if (workingId == null) { | |
| this.buffer = ObjectId.generate(); | |
| } | |
| else if (ArrayBuffer.isView(workingId) && workingId.byteLength === 12) { | |
| this.buffer = ByteUtils.toLocalBufferType(workingId); | |
| } | |
| else if (typeof workingId === 'string') { | |
| if (ObjectId.validateHexString(workingId)) { | |
| this.buffer = ByteUtils.fromHex(workingId); | |
| if (ObjectId.cacheHexString) { | |
| __idCache.set(this, workingId); | |
| } | |
| } | |
| else { | |
| throw new BSONError('input must be a 24 character hex string, 12 byte Uint8Array, or an integer'); | |
| } | |
| } | |
| else { | |
| throw new BSONError('Argument passed in does not match the accepted types'); | |
| } | |
| } | |
| get id() { | |
| return this.buffer; | |
| } | |
| set id(value) { | |
| this.buffer = value; | |
| if (ObjectId.cacheHexString) { | |
| __idCache.set(this, ByteUtils.toHex(value)); | |
| } | |
| } | |
| static validateHexString(string) { | |
| if (string?.length !== 24) | |
| return false; | |
| for (let i = 0; i < 24; i++) { | |
| const char = string.charCodeAt(i); | |
| if ((char >= 48 && char <= 57) || | |
| (char >= 97 && char <= 102) || | |
| (char >= 65 && char <= 70)) { | |
| continue; | |
| } | |
| return false; | |
| } | |
| return true; | |
| } | |
| toHexString() { | |
| if (ObjectId.cacheHexString) { | |
| const __id = __idCache.get(this); | |
| if (__id) | |
| return __id; | |
| } | |
| const hexString = ByteUtils.toHex(this.id); | |
| if (ObjectId.cacheHexString) { | |
| __idCache.set(this, hexString); | |
| } | |
| return hexString; | |
| } | |
| static getInc() { | |
| return (ObjectId.index = (ObjectId.index + 1) % 0xffffff); | |
| } | |
| static generate(time) { | |
| if ('number' !== typeof time) { | |
| time = Math.floor(Date.now() / 1000); | |
| } | |
| const inc = ObjectId.getInc(); | |
| const buffer = ByteUtils.allocateUnsafe(12); | |
| NumberUtils.setInt32BE(buffer, 0, time); | |
| if (PROCESS_UNIQUE === null) { | |
| PROCESS_UNIQUE = ByteUtils.randomBytes(5); | |
| } | |
| buffer[4] = PROCESS_UNIQUE[0]; | |
| buffer[5] = PROCESS_UNIQUE[1]; | |
| buffer[6] = PROCESS_UNIQUE[2]; | |
| buffer[7] = PROCESS_UNIQUE[3]; | |
| buffer[8] = PROCESS_UNIQUE[4]; | |
| buffer[11] = inc & 0xff; | |
| buffer[10] = (inc >> 8) & 0xff; | |
| buffer[9] = (inc >> 16) & 0xff; | |
| return buffer; | |
| } | |
| toString(encoding) { | |
| if (encoding === 'base64') | |
| return ByteUtils.toBase64(this.id); | |
| if (encoding === 'hex') | |
| return this.toHexString(); | |
| return this.toHexString(); | |
| } | |
| toJSON() { | |
| return this.toHexString(); | |
| } | |
| static is(variable) { | |
| return (variable != null && | |
| typeof variable === 'object' && | |
| '_bsontype' in variable && | |
| variable._bsontype === 'ObjectId'); | |
| } | |
| equals(otherId) { | |
| if (otherId === undefined || otherId === null) { | |
| return false; | |
| } | |
| if (ObjectId.is(otherId)) { | |
| return (this.buffer[11] === otherId.buffer[11] && ByteUtils.equals(this.buffer, otherId.buffer)); | |
| } | |
| if (typeof otherId === 'string') { | |
| return otherId.toLowerCase() === this.toHexString(); | |
| } | |
| if (typeof otherId === 'object' && typeof otherId.toHexString === 'function') { | |
| const otherIdString = otherId.toHexString(); | |
| const thisIdString = this.toHexString(); | |
| return typeof otherIdString === 'string' && otherIdString.toLowerCase() === thisIdString; | |
| } | |
| return false; | |
| } | |
| getTimestamp() { | |
| const timestamp = new Date(); | |
| const time = NumberUtils.getUint32BE(this.buffer, 0); | |
| timestamp.setTime(Math.floor(time) * 1000); | |
| return timestamp; | |
| } | |
| static createPk() { | |
| return new ObjectId(); | |
| } | |
| serializeInto(uint8array, index) { | |
| uint8array[index] = this.buffer[0]; | |
| uint8array[index + 1] = this.buffer[1]; | |
| uint8array[index + 2] = this.buffer[2]; | |
| uint8array[index + 3] = this.buffer[3]; | |
| uint8array[index + 4] = this.buffer[4]; | |
| uint8array[index + 5] = this.buffer[5]; | |
| uint8array[index + 6] = this.buffer[6]; | |
| uint8array[index + 7] = this.buffer[7]; | |
| uint8array[index + 8] = this.buffer[8]; | |
| uint8array[index + 9] = this.buffer[9]; | |
| uint8array[index + 10] = this.buffer[10]; | |
| uint8array[index + 11] = this.buffer[11]; | |
| return 12; | |
| } | |
| static createFromTime(time) { | |
| const buffer = ByteUtils.allocate(12); | |
| for (let i = 11; i >= 4; i--) | |
| buffer[i] = 0; | |
| NumberUtils.setInt32BE(buffer, 0, time); | |
| return new ObjectId(buffer); | |
| } | |
| static createFromHexString(hexString) { | |
| if (hexString?.length !== 24) { | |
| throw new BSONError('hex string must be 24 characters'); | |
| } | |
| return new ObjectId(ByteUtils.fromHex(hexString)); | |
| } | |
| static createFromBase64(base64) { | |
| if (base64?.length !== 16) { | |
| throw new BSONError('base64 string must be 16 characters'); | |
| } | |
| return new ObjectId(ByteUtils.fromBase64(base64)); | |
| } | |
| static isValid(id) { | |
| if (id == null) | |
| return false; | |
| if (typeof id === 'string') | |
| return ObjectId.validateHexString(id); | |
| try { | |
| new ObjectId(id); | |
| return true; | |
| } | |
| catch { | |
| return false; | |
| } | |
| } | |
| toExtendedJSON() { | |
| if (this.toHexString) | |
| return { $oid: this.toHexString() }; | |
| return { $oid: this.toString('hex') }; | |
| } | |
| static fromExtendedJSON(doc) { | |
| return new ObjectId(doc.$oid); | |
| } | |
| isCached() { | |
| return ObjectId.cacheHexString && __idCache.has(this); | |
| } | |
| inspect(depth, options, inspect) { | |
| inspect ??= defaultInspect; | |
| return `new ObjectId(${inspect(this.toHexString(), options)})`; | |
| } | |
| } | |
| function internalCalculateObjectSize(object, serializeFunctions, ignoreUndefined) { | |
| let totalLength = 4 + 1; | |
| if (Array.isArray(object)) { | |
| for (let i = 0; i < object.length; i++) { | |
| totalLength += calculateElement(i.toString(), object[i], serializeFunctions, true, ignoreUndefined); | |
| } | |
| } | |
| else { | |
| if (typeof object?.toBSON === 'function') { | |
| object = object.toBSON(); | |
| } | |
| for (const key of Object.keys(object)) { | |
| totalLength += calculateElement(key, object[key], serializeFunctions, false, ignoreUndefined); | |
| } | |
| } | |
| return totalLength; | |
| } | |
| function calculateElement(name, value, serializeFunctions = false, isArray = false, ignoreUndefined = false) { | |
| if (typeof value?.toBSON === 'function') { | |
| value = value.toBSON(); | |
| } | |
| switch (typeof value) { | |
| case 'string': | |
| return 1 + ByteUtils.utf8ByteLength(name) + 1 + 4 + ByteUtils.utf8ByteLength(value) + 1; | |
| case 'number': | |
| if (Math.floor(value) === value && | |
| value >= JS_INT_MIN && | |
| value <= JS_INT_MAX) { | |
| if (value >= BSON_INT32_MIN && value <= BSON_INT32_MAX) { | |
| return (name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + (4 + 1); | |
| } | |
| else { | |
| return (name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + (8 + 1); | |
| } | |
| } | |
| else { | |
| return (name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + (8 + 1); | |
| } | |
| case 'undefined': | |
| if (isArray || !ignoreUndefined) | |
| return (name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + 1; | |
| return 0; | |
| case 'boolean': | |
| return (name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + (1 + 1); | |
| case 'object': | |
| if (value != null && | |
| typeof value._bsontype === 'string' && | |
| value[BSON_VERSION_SYMBOL] !== BSON_MAJOR_VERSION) { | |
| throw new BSONVersionError(); | |
| } | |
| else if (value == null || value._bsontype === 'MinKey' || value._bsontype === 'MaxKey') { | |
| return (name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + 1; | |
| } | |
| else if (value._bsontype === 'ObjectId') { | |
| return (name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + (12 + 1); | |
| } | |
| else if (value instanceof Date || isDate(value)) { | |
| return (name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + (8 + 1); | |
| } | |
| else if (ArrayBuffer.isView(value) || | |
| value instanceof ArrayBuffer || | |
| isAnyArrayBuffer(value)) { | |
| return ((name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + (1 + 4 + 1) + value.byteLength); | |
| } | |
| else if (value._bsontype === 'Long' || | |
| value._bsontype === 'Double' || | |
| value._bsontype === 'Timestamp') { | |
| return (name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + (8 + 1); | |
| } | |
| else if (value._bsontype === 'Decimal128') { | |
| return (name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + (16 + 1); | |
| } | |
| else if (value._bsontype === 'Code') { | |
| if (value.scope != null && Object.keys(value.scope).length > 0) { | |
| return ((name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + | |
| 1 + | |
| 4 + | |
| 4 + | |
| ByteUtils.utf8ByteLength(value.code.toString()) + | |
| 1 + | |
| internalCalculateObjectSize(value.scope, serializeFunctions, ignoreUndefined)); | |
| } | |
| else { | |
| return ((name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + | |
| 1 + | |
| 4 + | |
| ByteUtils.utf8ByteLength(value.code.toString()) + | |
| 1); | |
| } | |
| } | |
| else if (value._bsontype === 'Binary') { | |
| const binary = value; | |
| if (binary.sub_type === Binary.SUBTYPE_BYTE_ARRAY) { | |
| return ((name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + | |
| (binary.position + 1 + 4 + 1 + 4)); | |
| } | |
| else { | |
| return ((name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + (binary.position + 1 + 4 + 1)); | |
| } | |
| } | |
| else if (value._bsontype === 'Symbol') { | |
| return ((name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + | |
| ByteUtils.utf8ByteLength(value.value) + | |
| 4 + | |
| 1 + | |
| 1); | |
| } | |
| else if (value._bsontype === 'DBRef') { | |
| const ordered_values = Object.assign({ | |
| $ref: value.collection, | |
| $id: value.oid | |
| }, value.fields); | |
| if (value.db != null) { | |
| ordered_values['$db'] = value.db; | |
| } | |
| return ((name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + | |
| 1 + | |
| internalCalculateObjectSize(ordered_values, serializeFunctions, ignoreUndefined)); | |
| } | |
| else if (value instanceof RegExp || isRegExp(value)) { | |
| return ((name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + | |
| 1 + | |
| ByteUtils.utf8ByteLength(value.source) + | |
| 1 + | |
| (value.global ? 1 : 0) + | |
| (value.ignoreCase ? 1 : 0) + | |
| (value.multiline ? 1 : 0) + | |
| 1); | |
| } | |
| else if (value._bsontype === 'BSONRegExp') { | |
| return ((name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + | |
| 1 + | |
| ByteUtils.utf8ByteLength(value.pattern) + | |
| 1 + | |
| ByteUtils.utf8ByteLength(value.options) + | |
| 1); | |
| } | |
| else { | |
| return ((name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + | |
| internalCalculateObjectSize(value, serializeFunctions, ignoreUndefined) + | |
| 1); | |
| } | |
| case 'function': | |
| if (serializeFunctions) { | |
| return ((name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + | |
| 1 + | |
| 4 + | |
| ByteUtils.utf8ByteLength(value.toString()) + | |
| 1); | |
| } | |
| return 0; | |
| case 'bigint': | |
| return (name != null ? ByteUtils.utf8ByteLength(name) + 1 : 0) + (8 + 1); | |
| case 'symbol': | |
| return 0; | |
| default: | |
| throw new BSONError(`Unrecognized JS type: ${typeof value}`); | |
| } | |
| } | |
| function alphabetize(str) { | |
| return str.split('').sort().join(''); | |
| } | |
| class BSONRegExp extends BSONValue { | |
| get _bsontype() { | |
| return 'BSONRegExp'; | |
| } | |
| pattern; | |
| options; | |
| constructor(pattern, options) { | |
| super(); | |
| this.pattern = pattern; | |
| this.options = alphabetize(options ?? ''); | |
| if (this.pattern.indexOf('\x00') !== -1) { | |
| throw new BSONError(`BSON Regex patterns cannot contain null bytes, found: ${JSON.stringify(this.pattern)}`); | |
| } | |
| if (this.options.indexOf('\x00') !== -1) { | |
| throw new BSONError(`BSON Regex options cannot contain null bytes, found: ${JSON.stringify(this.options)}`); | |
| } | |
| for (let i = 0; i < this.options.length; i++) { | |
| if (!(this.options[i] === 'i' || | |
| this.options[i] === 'm' || | |
| this.options[i] === 'x' || | |
| this.options[i] === 'l' || | |
| this.options[i] === 's' || | |
| this.options[i] === 'u')) { | |
| throw new BSONError(`The regular expression option [${this.options[i]}] is not supported`); | |
| } | |
| } | |
| } | |
| static parseOptions(options) { | |
| return options ? options.split('').sort().join('') : ''; | |
| } | |
| toExtendedJSON(options) { | |
| options = options || {}; | |
| if (options.legacy) { | |
| return { $regex: this.pattern, $options: this.options }; | |
| } | |
| return { $regularExpression: { pattern: this.pattern, options: this.options } }; | |
| } | |
| static fromExtendedJSON(doc) { | |
| if ('$regex' in doc) { | |
| if (typeof doc.$regex !== 'string') { | |
| if (doc.$regex._bsontype === 'BSONRegExp') { | |
| return doc; | |
| } | |
| } | |
| else { | |
| return new BSONRegExp(doc.$regex, BSONRegExp.parseOptions(doc.$options)); | |
| } | |
| } | |
| if ('$regularExpression' in doc) { | |
| return new BSONRegExp(doc.$regularExpression.pattern, BSONRegExp.parseOptions(doc.$regularExpression.options)); | |
| } | |
| throw new BSONError(`Unexpected BSONRegExp EJSON object form: ${JSON.stringify(doc)}`); | |
| } | |
| inspect(depth, options, inspect) { | |
| const stylize = getStylizeFunction(options) ?? (v => v); | |
| inspect ??= defaultInspect; | |
| const pattern = stylize(inspect(this.pattern), 'regexp'); | |
| const flags = stylize(inspect(this.options), 'regexp'); | |
| return `new BSONRegExp(${pattern}, ${flags})`; | |
| } | |
| } | |
| class BSONSymbol extends BSONValue { | |
| get _bsontype() { | |
| return 'BSONSymbol'; | |
| } | |
| value; | |
| constructor(value) { | |
| super(); | |
| this.value = value; | |
| } | |
| valueOf() { | |
| return this.value; | |
| } | |
| toString() { | |
| return this.value; | |
| } | |
| toJSON() { | |
| return this.value; | |
| } | |
| toExtendedJSON() { | |
| return { $symbol: this.value }; | |
| } | |
| static fromExtendedJSON(doc) { | |
| return new BSONSymbol(doc.$symbol); | |
| } | |
| inspect(depth, options, inspect) { | |
| inspect ??= defaultInspect; | |
| return `new BSONSymbol(${inspect(this.value, options)})`; | |
| } | |
| } | |
| const LongWithoutOverridesClass = Long; | |
| class Timestamp extends LongWithoutOverridesClass { | |
| get _bsontype() { | |
| return 'Timestamp'; | |
| } | |
| get [bsonType]() { | |
| return 'Timestamp'; | |
| } | |
| static MAX_VALUE = Long.MAX_UNSIGNED_VALUE; | |
| get i() { | |
| return this.low >>> 0; | |
| } | |
| get t() { | |
| return this.high >>> 0; | |
| } | |
| constructor(low) { | |
| if (low == null) { | |
| super(0, 0, true); | |
| } | |
| else if (typeof low === 'bigint') { | |
| super(low, true); | |
| } | |
| else if (Long.isLong(low)) { | |
| super(low.low, low.high, true); | |
| } | |
| else if (typeof low === 'object' && 't' in low && 'i' in low) { | |
| if (typeof low.t !== 'number' && (typeof low.t !== 'object' || low.t._bsontype !== 'Int32')) { | |
| throw new BSONError('Timestamp constructed from { t, i } must provide t as a number'); | |
| } | |
| if (typeof low.i !== 'number' && (typeof low.i !== 'object' || low.i._bsontype !== 'Int32')) { | |
| throw new BSONError('Timestamp constructed from { t, i } must provide i as a number'); | |
| } | |
| const t = Number(low.t); | |
| const i = Number(low.i); | |
| if (t < 0 || Number.isNaN(t)) { | |
| throw new BSONError('Timestamp constructed from { t, i } must provide a positive t'); | |
| } | |
| if (i < 0 || Number.isNaN(i)) { | |
| throw new BSONError('Timestamp constructed from { t, i } must provide a positive i'); | |
| } | |
| if (t > 0xffff_ffff) { | |
| throw new BSONError('Timestamp constructed from { t, i } must provide t equal or less than uint32 max'); | |
| } | |
| if (i > 0xffff_ffff) { | |
| throw new BSONError('Timestamp constructed from { t, i } must provide i equal or less than uint32 max'); | |
| } | |
| super(i, t, true); | |
| } | |
| else { | |
| throw new BSONError('A Timestamp can only be constructed with: bigint, Long, or { t: number; i: number }'); | |
| } | |
| } | |
| toJSON() { | |
| return { | |
| $timestamp: this.toString() | |
| }; | |
| } | |
| static fromInt(value) { | |
| return new Timestamp(Long.fromInt(value, true)); | |
| } | |
| static fromNumber(value) { | |
| return new Timestamp(Long.fromNumber(value, true)); | |
| } | |
| static fromBits(lowBits, highBits) { | |
| return new Timestamp({ i: lowBits, t: highBits }); | |
| } | |
| static fromString(str, optRadix) { | |
| return new Timestamp(Long.fromString(str, true, optRadix)); | |
| } | |
| toExtendedJSON() { | |
| return { $timestamp: { t: this.t, i: this.i } }; | |
| } | |
| static fromExtendedJSON(doc) { | |
| const i = Long.isLong(doc.$timestamp.i) | |
| ? doc.$timestamp.i.getLowBitsUnsigned() | |
| : doc.$timestamp.i; | |
| const t = Long.isLong(doc.$timestamp.t) | |
| ? doc.$timestamp.t.getLowBitsUnsigned() | |
| : doc.$timestamp.t; | |
| return new Timestamp({ t, i }); | |
| } | |
| inspect(depth, options, inspect) { | |
| inspect ??= defaultInspect; | |
| const t = inspect(this.t, options); | |
| const i = inspect(this.i, options); | |
| return `new Timestamp({ t: ${t}, i: ${i} })`; | |
| } | |
| } | |
| const JS_INT_MAX_LONG = Long.fromNumber(JS_INT_MAX); | |
| const JS_INT_MIN_LONG = Long.fromNumber(JS_INT_MIN); | |
| function internalDeserialize(buffer, options, isArray) { | |
| options = options == null ? {} : options; | |
| const index = options && options.index ? options.index : 0; | |
| const size = NumberUtils.getInt32LE(buffer, index); | |
| if (size < 5) { | |
| throw new BSONError(`bson size must be >= 5, is ${size}`); | |
| } | |
| if (options.allowObjectSmallerThanBufferSize && buffer.length < size) { | |
| throw new BSONError(`buffer length ${buffer.length} must be >= bson size ${size}`); | |
| } | |
| if (!options.allowObjectSmallerThanBufferSize && buffer.length !== size) { | |
| throw new BSONError(`buffer length ${buffer.length} must === bson size ${size}`); | |
| } | |
| if (size + index > buffer.byteLength) { | |
| throw new BSONError(`(bson size ${size} + options.index ${index} must be <= buffer length ${buffer.byteLength})`); | |
| } | |
| if (buffer[index + size - 1] !== 0) { | |
| throw new BSONError("One object, sized correctly, with a spot for an EOO, but the EOO isn't 0x00"); | |
| } | |
| return deserializeObject(buffer, index, options, isArray); | |
| } | |
| const allowedDBRefKeys = /^\$ref$|^\$id$|^\$db$/; | |
| function deserializeObject(buffer, index, options, isArray = false) { | |
| const fieldsAsRaw = options['fieldsAsRaw'] == null ? null : options['fieldsAsRaw']; | |
| const raw = options['raw'] == null ? false : options['raw']; | |
| const bsonRegExp = typeof options['bsonRegExp'] === 'boolean' ? options['bsonRegExp'] : false; | |
| const promoteBuffers = options.promoteBuffers ?? false; | |
| const promoteLongs = options.promoteLongs ?? true; | |
| const promoteValues = options.promoteValues ?? true; | |
| const useBigInt64 = options.useBigInt64 ?? false; | |
| if (useBigInt64 && !promoteValues) { | |
| throw new BSONError('Must either request bigint or Long for int64 deserialization'); | |
| } | |
| if (useBigInt64 && !promoteLongs) { | |
| throw new BSONError('Must either request bigint or Long for int64 deserialization'); | |
| } | |
| const validation = options.validation == null ? { utf8: true } : options.validation; | |
| let globalUTFValidation = true; | |
| let validationSetting; | |
| let utf8KeysSet; | |
| const utf8ValidatedKeys = validation.utf8; | |
| if (typeof utf8ValidatedKeys === 'boolean') { | |
| validationSetting = utf8ValidatedKeys; | |
| } | |
| else { | |
| globalUTFValidation = false; | |
| const utf8ValidationValues = Object.keys(utf8ValidatedKeys).map(function (key) { | |
| return utf8ValidatedKeys[key]; | |
| }); | |
| if (utf8ValidationValues.length === 0) { | |
| throw new BSONError('UTF-8 validation setting cannot be empty'); | |
| } | |
| if (typeof utf8ValidationValues[0] !== 'boolean') { | |
| throw new BSONError('Invalid UTF-8 validation option, must specify boolean values'); | |
| } | |
| validationSetting = utf8ValidationValues[0]; | |
| if (!utf8ValidationValues.every(item => item === validationSetting)) { | |
| throw new BSONError('Invalid UTF-8 validation option - keys must be all true or all false'); | |
| } | |
| } | |
| if (!globalUTFValidation) { | |
| utf8KeysSet = new Set(); | |
| for (const key of Object.keys(utf8ValidatedKeys)) { | |
| utf8KeysSet.add(key); | |
| } | |
| } | |
| const startIndex = index; | |
| if (buffer.length < 5) | |
| throw new BSONError('corrupt bson message < 5 bytes long'); | |
| const size = NumberUtils.getInt32LE(buffer, index); | |
| index += 4; | |
| if (size < 5 || size > buffer.length) | |
| throw new BSONError('corrupt bson message'); | |
| const object = isArray ? [] : {}; | |
| let arrayIndex = 0; | |
| let isPossibleDBRef = isArray ? false : null; | |
| while (true) { | |
| const elementType = buffer[index++]; | |
| if (elementType === 0) | |
| break; | |
| let i = index; | |
| while (buffer[i] !== 0x00 && i < buffer.length) { | |
| i++; | |
| } | |
| if (i >= buffer.byteLength) | |
| throw new BSONError('Bad BSON Document: illegal CString'); | |
| const name = isArray ? arrayIndex++ : ByteUtils.toUTF8(buffer, index, i, false); | |
| let shouldValidateKey = true; | |
| if (globalUTFValidation || utf8KeysSet?.has(name)) { | |
| shouldValidateKey = validationSetting; | |
| } | |
| else { | |
| shouldValidateKey = !validationSetting; | |
| } | |
| if (isPossibleDBRef !== false && name[0] === '$') { | |
| isPossibleDBRef = allowedDBRefKeys.test(name); | |
| } | |
| let value; | |
| index = i + 1; | |
| if (elementType === BSON_DATA_STRING) { | |
| const stringSize = NumberUtils.getInt32LE(buffer, index); | |
| index += 4; | |
| if (stringSize <= 0 || | |
| stringSize > buffer.length - index || | |
| buffer[index + stringSize - 1] !== 0) { | |
| throw new BSONError('bad string length in bson'); | |
| } | |
| value = ByteUtils.toUTF8(buffer, index, index + stringSize - 1, shouldValidateKey); | |
| index = index + stringSize; | |
| } | |
| else if (elementType === BSON_DATA_OID) { | |
| const oid = ByteUtils.allocateUnsafe(12); | |
| for (let i = 0; i < 12; i++) | |
| oid[i] = buffer[index + i]; | |
| value = new ObjectId(oid); | |
| index = index + 12; | |
| } | |
| else if (elementType === BSON_DATA_INT && promoteValues === false) { | |
| value = new Int32(NumberUtils.getInt32LE(buffer, index)); | |
| index += 4; | |
| } | |
| else if (elementType === BSON_DATA_INT) { | |
| value = NumberUtils.getInt32LE(buffer, index); | |
| index += 4; | |
| } | |
| else if (elementType === BSON_DATA_NUMBER) { | |
| value = NumberUtils.getFloat64LE(buffer, index); | |
| index += 8; | |
| if (promoteValues === false) | |
| value = new Double(value); | |
| } | |
| else if (elementType === BSON_DATA_DATE) { | |
| const lowBits = NumberUtils.getInt32LE(buffer, index); | |
| const highBits = NumberUtils.getInt32LE(buffer, index + 4); | |
| index += 8; | |
| value = new Date(new Long(lowBits, highBits).toNumber()); | |
| } | |
| else if (elementType === BSON_DATA_BOOLEAN) { | |
| if (buffer[index] !== 0 && buffer[index] !== 1) | |
| throw new BSONError('illegal boolean type value'); | |
| value = buffer[index++] === 1; | |
| } | |
| else if (elementType === BSON_DATA_OBJECT) { | |
| const _index = index; | |
| const objectSize = NumberUtils.getInt32LE(buffer, index); | |
| if (objectSize <= 0 || objectSize > buffer.length - index) | |
| throw new BSONError('bad embedded document length in bson'); | |
| if (raw) { | |
| value = buffer.subarray(index, index + objectSize); | |
| } | |
| else { | |
| let objectOptions = options; | |
| if (!globalUTFValidation) { | |
| objectOptions = { ...options, validation: { utf8: shouldValidateKey } }; | |
| } | |
| value = deserializeObject(buffer, _index, objectOptions, false); | |
| } | |
| index = index + objectSize; | |
| } | |
| else if (elementType === BSON_DATA_ARRAY) { | |
| const _index = index; | |
| const objectSize = NumberUtils.getInt32LE(buffer, index); | |
| let arrayOptions = options; | |
| const stopIndex = index + objectSize; | |
| if (fieldsAsRaw && fieldsAsRaw[name]) { | |
| arrayOptions = { ...options, raw: true }; | |
| } | |
| if (!globalUTFValidation) { | |
| arrayOptions = { ...arrayOptions, validation: { utf8: shouldValidateKey } }; | |
| } | |
| value = deserializeObject(buffer, _index, arrayOptions, true); | |
| index = index + objectSize; | |
| if (buffer[index - 1] !== 0) | |
| throw new BSONError('invalid array terminator byte'); | |
| if (index !== stopIndex) | |
| throw new BSONError('corrupted array bson'); | |
| } | |
| else if (elementType === BSON_DATA_UNDEFINED) { | |
| value = undefined; | |
| } | |
| else if (elementType === BSON_DATA_NULL) { | |
| value = null; | |
| } | |
| else if (elementType === BSON_DATA_LONG) { | |
| if (useBigInt64) { | |
| value = NumberUtils.getBigInt64LE(buffer, index); | |
| index += 8; | |
| } | |
| else { | |
| const lowBits = NumberUtils.getInt32LE(buffer, index); | |
| const highBits = NumberUtils.getInt32LE(buffer, index + 4); | |
| index += 8; | |
| const long = new Long(lowBits, highBits); | |
| if (promoteLongs && promoteValues === true) { | |
| value = | |
| long.lessThanOrEqual(JS_INT_MAX_LONG) && long.greaterThanOrEqual(JS_INT_MIN_LONG) | |
| ? long.toNumber() | |
| : long; | |
| } | |
| else { | |
| value = long; | |
| } | |
| } | |
| } | |
| else if (elementType === BSON_DATA_DECIMAL128) { | |
| const bytes = ByteUtils.allocateUnsafe(16); | |
| for (let i = 0; i < 16; i++) | |
| bytes[i] = buffer[index + i]; | |
| index = index + 16; | |
| value = new Decimal128(bytes); | |
| } | |
| else if (elementType === BSON_DATA_BINARY) { | |
| let binarySize = NumberUtils.getInt32LE(buffer, index); | |
| index += 4; | |
| const totalBinarySize = binarySize; | |
| const subType = buffer[index++]; | |
| if (binarySize < 0) | |
| throw new BSONError('Negative binary type element size found'); | |
| if (binarySize > buffer.byteLength) | |
| throw new BSONError('Binary type size larger than document size'); | |
| if (subType === Binary.SUBTYPE_BYTE_ARRAY) { | |
| binarySize = NumberUtils.getInt32LE(buffer, index); | |
| index += 4; | |
| if (binarySize < 0) | |
| throw new BSONError('Negative binary type element size found for subtype 0x02'); | |
| if (binarySize > totalBinarySize - 4) | |
| throw new BSONError('Binary type with subtype 0x02 contains too long binary size'); | |
| if (binarySize < totalBinarySize - 4) | |
| throw new BSONError('Binary type with subtype 0x02 contains too short binary size'); | |
| } | |
| if (promoteBuffers && promoteValues) { | |
| value = ByteUtils.toLocalBufferType(buffer.subarray(index, index + binarySize)); | |
| } | |
| else { | |
| value = new Binary(buffer.subarray(index, index + binarySize), subType); | |
| if (subType === BSON_BINARY_SUBTYPE_UUID_NEW && UUID.isValid(value)) { | |
| value = value.toUUID(); | |
| } | |
| } | |
| index = index + binarySize; | |
| } | |
| else if (elementType === BSON_DATA_REGEXP && bsonRegExp === false) { | |
| i = index; | |
| while (buffer[i] !== 0x00 && i < buffer.length) { | |
| i++; | |
| } | |
| if (i >= buffer.length) | |
| throw new BSONError('Bad BSON Document: illegal CString'); | |
| const source = ByteUtils.toUTF8(buffer, index, i, false); | |
| index = i + 1; | |
| i = index; | |
| while (buffer[i] !== 0x00 && i < buffer.length) { | |
| i++; | |
| } | |
| if (i >= buffer.length) | |
| throw new BSONError('Bad BSON Document: illegal CString'); | |
| const regExpOptions = ByteUtils.toUTF8(buffer, index, i, false); | |
| index = i + 1; | |
| const optionsArray = new Array(regExpOptions.length); | |
| for (i = 0; i < regExpOptions.length; i++) { | |
| switch (regExpOptions[i]) { | |
| case 'm': | |
| optionsArray[i] = 'm'; | |
| break; | |
| case 's': | |
| optionsArray[i] = 'g'; | |
| break; | |
| case 'i': | |
| optionsArray[i] = 'i'; | |
| break; | |
| } | |
| } | |
| value = new RegExp(source, optionsArray.join('')); | |
| } | |
| else if (elementType === BSON_DATA_REGEXP && bsonRegExp === true) { | |
| i = index; | |
| while (buffer[i] !== 0x00 && i < buffer.length) { | |
| i++; | |
| } | |
| if (i >= buffer.length) | |
| throw new BSONError('Bad BSON Document: illegal CString'); | |
| const source = ByteUtils.toUTF8(buffer, index, i, false); | |
| index = i + 1; | |
| i = index; | |
| while (buffer[i] !== 0x00 && i < buffer.length) { | |
| i++; | |
| } | |
| if (i >= buffer.length) | |
| throw new BSONError('Bad BSON Document: illegal CString'); | |
| const regExpOptions = ByteUtils.toUTF8(buffer, index, i, false); | |
| index = i + 1; | |
| value = new BSONRegExp(source, regExpOptions); | |
| } | |
| else if (elementType === BSON_DATA_SYMBOL) { | |
| const stringSize = NumberUtils.getInt32LE(buffer, index); | |
| index += 4; | |
| if (stringSize <= 0 || | |
| stringSize > buffer.length - index || | |
| buffer[index + stringSize - 1] !== 0) { | |
| throw new BSONError('bad string length in bson'); | |
| } | |
| const symbol = ByteUtils.toUTF8(buffer, index, index + stringSize - 1, shouldValidateKey); | |
| value = promoteValues ? symbol : new BSONSymbol(symbol); | |
| index = index + stringSize; | |
| } | |
| else if (elementType === BSON_DATA_TIMESTAMP) { | |
| value = new Timestamp({ | |
| i: NumberUtils.getUint32LE(buffer, index), | |
| t: NumberUtils.getUint32LE(buffer, index + 4) | |
| }); | |
| index += 8; | |
| } | |
| else if (elementType === BSON_DATA_MIN_KEY) { | |
| value = new MinKey(); | |
| } | |
| else if (elementType === BSON_DATA_MAX_KEY) { | |
| value = new MaxKey(); | |
| } | |
| else if (elementType === BSON_DATA_CODE) { | |
| const stringSize = NumberUtils.getInt32LE(buffer, index); | |
| index += 4; | |
| if (stringSize <= 0 || | |
| stringSize > buffer.length - index || | |
| buffer[index + stringSize - 1] !== 0) { | |
| throw new BSONError('bad string length in bson'); | |
| } | |
| const functionString = ByteUtils.toUTF8(buffer, index, index + stringSize - 1, shouldValidateKey); | |
| value = new Code(functionString); | |
| index = index + stringSize; | |
| } | |
| else if (elementType === BSON_DATA_CODE_W_SCOPE) { | |
| const totalSize = NumberUtils.getInt32LE(buffer, index); | |
| index += 4; | |
| if (totalSize < 4 + 4 + 4 + 1) { | |
| throw new BSONError('code_w_scope total size shorter minimum expected length'); | |
| } | |
| const stringSize = NumberUtils.getInt32LE(buffer, index); | |
| index += 4; | |
| if (stringSize <= 0 || | |
| stringSize > buffer.length - index || | |
| buffer[index + stringSize - 1] !== 0) { | |
| throw new BSONError('bad string length in bson'); | |
| } | |
| const functionString = ByteUtils.toUTF8(buffer, index, index + stringSize - 1, shouldValidateKey); | |
| index = index + stringSize; | |
| const _index = index; | |
| const objectSize = NumberUtils.getInt32LE(buffer, index); | |
| const scopeObject = deserializeObject(buffer, _index, options, false); | |
| index = index + objectSize; | |
| if (totalSize < 4 + 4 + objectSize + stringSize) { | |
| throw new BSONError('code_w_scope total size is too short, truncating scope'); | |
| } | |
| if (totalSize > 4 + 4 + objectSize + stringSize) { | |
| throw new BSONError('code_w_scope total size is too long, clips outer document'); | |
| } | |
| value = new Code(functionString, scopeObject); | |
| } | |
| else if (elementType === BSON_DATA_DBPOINTER) { | |
| const stringSize = NumberUtils.getInt32LE(buffer, index); | |
| index += 4; | |
| if (stringSize <= 0 || | |
| stringSize > buffer.length - index || | |
| buffer[index + stringSize - 1] !== 0) | |
| throw new BSONError('bad string length in bson'); | |
| const namespace = ByteUtils.toUTF8(buffer, index, index + stringSize - 1, shouldValidateKey); | |
| index = index + stringSize; | |
| const oidBuffer = ByteUtils.allocateUnsafe(12); | |
| for (let i = 0; i < 12; i++) | |
| oidBuffer[i] = buffer[index + i]; | |
| const oid = new ObjectId(oidBuffer); | |
| index = index + 12; | |
| value = new DBRef(namespace, oid); | |
| } | |
| else { | |
| throw new BSONError(`Detected unknown BSON type ${elementType.toString(16)} for fieldname "${name}"`); | |
| } | |
| if (name === '__proto__') { | |
| Object.defineProperty(object, name, { | |
| value, | |
| writable: true, | |
| enumerable: true, | |
| configurable: true | |
| }); | |
| } | |
| else { | |
| object[name] = value; | |
| } | |
| } | |
| if (size !== index - startIndex) { | |
| if (isArray) | |
| throw new BSONError('corrupt array bson'); | |
| throw new BSONError('corrupt object bson'); | |
| } | |
| if (!isPossibleDBRef) | |
| return object; | |
| if (isDBRefLike(object)) { | |
| const copy = Object.assign({}, object); | |
| delete copy.$ref; | |
| delete copy.$id; | |
| delete copy.$db; | |
| return new DBRef(object.$ref, object.$id, object.$db, copy); | |
| } | |
| return object; | |
| } | |
| const regexp = /\x00/; | |
| const ignoreKeys = new Set(['$db', '$ref', '$id', '$clusterTime']); | |
| function serializeString(buffer, key, value, index) { | |
| buffer[index++] = BSON_DATA_STRING; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes + 1; | |
| buffer[index - 1] = 0; | |
| const size = ByteUtils.encodeUTF8Into(buffer, value, index + 4); | |
| NumberUtils.setInt32LE(buffer, index, size + 1); | |
| index = index + 4 + size; | |
| buffer[index++] = 0; | |
| return index; | |
| } | |
| function serializeNumber(buffer, key, value, index) { | |
| const isNegativeZero = Object.is(value, -0); | |
| const type = !isNegativeZero && | |
| Number.isSafeInteger(value) && | |
| value <= BSON_INT32_MAX && | |
| value >= BSON_INT32_MIN | |
| ? BSON_DATA_INT | |
| : BSON_DATA_NUMBER; | |
| buffer[index++] = type; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0x00; | |
| if (type === BSON_DATA_INT) { | |
| index += NumberUtils.setInt32LE(buffer, index, value); | |
| } | |
| else { | |
| index += NumberUtils.setFloat64LE(buffer, index, value); | |
| } | |
| return index; | |
| } | |
| function serializeBigInt(buffer, key, value, index) { | |
| buffer[index++] = BSON_DATA_LONG; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index += numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| index += NumberUtils.setBigInt64LE(buffer, index, value); | |
| return index; | |
| } | |
| function serializeNull(buffer, key, _, index) { | |
| buffer[index++] = BSON_DATA_NULL; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| return index; | |
| } | |
| function serializeBoolean(buffer, key, value, index) { | |
| buffer[index++] = BSON_DATA_BOOLEAN; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| buffer[index++] = value ? 1 : 0; | |
| return index; | |
| } | |
| function serializeDate(buffer, key, value, index) { | |
| buffer[index++] = BSON_DATA_DATE; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| const dateInMilis = Long.fromNumber(value.getTime()); | |
| const lowBits = dateInMilis.getLowBits(); | |
| const highBits = dateInMilis.getHighBits(); | |
| index += NumberUtils.setInt32LE(buffer, index, lowBits); | |
| index += NumberUtils.setInt32LE(buffer, index, highBits); | |
| return index; | |
| } | |
| function serializeRegExp(buffer, key, value, index) { | |
| buffer[index++] = BSON_DATA_REGEXP; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| if (value.source && value.source.match(regexp) != null) { | |
| throw new BSONError('value ' + value.source + ' must not contain null bytes'); | |
| } | |
| index = index + ByteUtils.encodeUTF8Into(buffer, value.source, index); | |
| buffer[index++] = 0x00; | |
| if (value.ignoreCase) | |
| buffer[index++] = 0x69; | |
| if (value.global) | |
| buffer[index++] = 0x73; | |
| if (value.multiline) | |
| buffer[index++] = 0x6d; | |
| buffer[index++] = 0x00; | |
| return index; | |
| } | |
| function serializeBSONRegExp(buffer, key, value, index) { | |
| buffer[index++] = BSON_DATA_REGEXP; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| if (value.pattern.match(regexp) != null) { | |
| throw new BSONError('pattern ' + value.pattern + ' must not contain null bytes'); | |
| } | |
| index = index + ByteUtils.encodeUTF8Into(buffer, value.pattern, index); | |
| buffer[index++] = 0x00; | |
| const sortedOptions = value.options.split('').sort().join(''); | |
| index = index + ByteUtils.encodeUTF8Into(buffer, sortedOptions, index); | |
| buffer[index++] = 0x00; | |
| return index; | |
| } | |
| function serializeMinMax(buffer, key, value, index) { | |
| if (value === null) { | |
| buffer[index++] = BSON_DATA_NULL; | |
| } | |
| else if (value._bsontype === 'MinKey') { | |
| buffer[index++] = BSON_DATA_MIN_KEY; | |
| } | |
| else { | |
| buffer[index++] = BSON_DATA_MAX_KEY; | |
| } | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| return index; | |
| } | |
| function serializeObjectId(buffer, key, value, index) { | |
| buffer[index++] = BSON_DATA_OID; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| index += value.serializeInto(buffer, index); | |
| return index; | |
| } | |
| function serializeBuffer(buffer, key, value, index) { | |
| buffer[index++] = BSON_DATA_BINARY; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| const size = value.length; | |
| index += NumberUtils.setInt32LE(buffer, index, size); | |
| buffer[index++] = BSON_BINARY_SUBTYPE_DEFAULT; | |
| if (size <= 16) { | |
| for (let i = 0; i < size; i++) | |
| buffer[index + i] = value[i]; | |
| } | |
| else { | |
| buffer.set(value, index); | |
| } | |
| index = index + size; | |
| return index; | |
| } | |
| function serializeObject(buffer, key, value, index, checkKeys, depth, serializeFunctions, ignoreUndefined, path) { | |
| if (path.has(value)) { | |
| throw new BSONError('Cannot convert circular structure to BSON'); | |
| } | |
| path.add(value); | |
| buffer[index++] = Array.isArray(value) ? BSON_DATA_ARRAY : BSON_DATA_OBJECT; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| const endIndex = serializeInto(buffer, value, checkKeys, index, depth + 1, serializeFunctions, ignoreUndefined, path); | |
| path.delete(value); | |
| return endIndex; | |
| } | |
| function serializeDecimal128(buffer, key, value, index) { | |
| buffer[index++] = BSON_DATA_DECIMAL128; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| for (let i = 0; i < 16; i++) | |
| buffer[index + i] = value.bytes[i]; | |
| return index + 16; | |
| } | |
| function serializeLong(buffer, key, value, index) { | |
| buffer[index++] = | |
| value._bsontype === 'Long' ? BSON_DATA_LONG : BSON_DATA_TIMESTAMP; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| const lowBits = value.getLowBits(); | |
| const highBits = value.getHighBits(); | |
| index += NumberUtils.setInt32LE(buffer, index, lowBits); | |
| index += NumberUtils.setInt32LE(buffer, index, highBits); | |
| return index; | |
| } | |
| function serializeInt32(buffer, key, value, index) { | |
| value = value.valueOf(); | |
| buffer[index++] = BSON_DATA_INT; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| index += NumberUtils.setInt32LE(buffer, index, value); | |
| return index; | |
| } | |
| function serializeDouble(buffer, key, value, index) { | |
| buffer[index++] = BSON_DATA_NUMBER; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| index += NumberUtils.setFloat64LE(buffer, index, value.value); | |
| return index; | |
| } | |
| function serializeFunction(buffer, key, value, index) { | |
| buffer[index++] = BSON_DATA_CODE; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| const functionString = value.toString(); | |
| const size = ByteUtils.encodeUTF8Into(buffer, functionString, index + 4) + 1; | |
| NumberUtils.setInt32LE(buffer, index, size); | |
| index = index + 4 + size - 1; | |
| buffer[index++] = 0; | |
| return index; | |
| } | |
| function serializeCode(buffer, key, value, index, checkKeys = false, depth = 0, serializeFunctions = false, ignoreUndefined = true, path) { | |
| if (value.scope && typeof value.scope === 'object') { | |
| buffer[index++] = BSON_DATA_CODE_W_SCOPE; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| let startIndex = index; | |
| const functionString = value.code; | |
| index = index + 4; | |
| const codeSize = ByteUtils.encodeUTF8Into(buffer, functionString, index + 4) + 1; | |
| NumberUtils.setInt32LE(buffer, index, codeSize); | |
| buffer[index + 4 + codeSize - 1] = 0; | |
| index = index + codeSize + 4; | |
| const endIndex = serializeInto(buffer, value.scope, checkKeys, index, depth + 1, serializeFunctions, ignoreUndefined, path); | |
| index = endIndex - 1; | |
| const totalSize = endIndex - startIndex; | |
| startIndex += NumberUtils.setInt32LE(buffer, startIndex, totalSize); | |
| buffer[index++] = 0; | |
| } | |
| else { | |
| buffer[index++] = BSON_DATA_CODE; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| const functionString = value.code.toString(); | |
| const size = ByteUtils.encodeUTF8Into(buffer, functionString, index + 4) + 1; | |
| NumberUtils.setInt32LE(buffer, index, size); | |
| index = index + 4 + size - 1; | |
| buffer[index++] = 0; | |
| } | |
| return index; | |
| } | |
| function serializeBinary(buffer, key, value, index) { | |
| buffer[index++] = BSON_DATA_BINARY; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| const data = value.buffer; | |
| let size = value.position; | |
| if (value.sub_type === Binary.SUBTYPE_BYTE_ARRAY) | |
| size = size + 4; | |
| index += NumberUtils.setInt32LE(buffer, index, size); | |
| buffer[index++] = value.sub_type; | |
| if (value.sub_type === Binary.SUBTYPE_BYTE_ARRAY) { | |
| size = size - 4; | |
| index += NumberUtils.setInt32LE(buffer, index, size); | |
| } | |
| if (value.sub_type === Binary.SUBTYPE_VECTOR) { | |
| validateBinaryVector(value); | |
| } | |
| if (size <= 16) { | |
| for (let i = 0; i < size; i++) | |
| buffer[index + i] = data[i]; | |
| } | |
| else { | |
| buffer.set(data, index); | |
| } | |
| index = index + value.position; | |
| return index; | |
| } | |
| function serializeSymbol(buffer, key, value, index) { | |
| buffer[index++] = BSON_DATA_SYMBOL; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| const size = ByteUtils.encodeUTF8Into(buffer, value.value, index + 4) + 1; | |
| NumberUtils.setInt32LE(buffer, index, size); | |
| index = index + 4 + size - 1; | |
| buffer[index++] = 0; | |
| return index; | |
| } | |
| function serializeDBRef(buffer, key, value, index, depth, serializeFunctions, path) { | |
| buffer[index++] = BSON_DATA_OBJECT; | |
| const numberOfWrittenBytes = ByteUtils.encodeUTF8Into(buffer, key, index); | |
| index = index + numberOfWrittenBytes; | |
| buffer[index++] = 0; | |
| let startIndex = index; | |
| let output = { | |
| $ref: value.collection || value.namespace, | |
| $id: value.oid | |
| }; | |
| if (value.db != null) { | |
| output.$db = value.db; | |
| } | |
| output = Object.assign(output, value.fields); | |
| const endIndex = serializeInto(buffer, output, false, index, depth + 1, serializeFunctions, true, path); | |
| const size = endIndex - startIndex; | |
| startIndex += NumberUtils.setInt32LE(buffer, index, size); | |
| return endIndex; | |
| } | |
| function serializeInto(buffer, object, checkKeys, startingIndex, depth, serializeFunctions, ignoreUndefined, path) { | |
| if (path == null) { | |
| if (object == null) { | |
| buffer[0] = 0x05; | |
| buffer[1] = 0x00; | |
| buffer[2] = 0x00; | |
| buffer[3] = 0x00; | |
| buffer[4] = 0x00; | |
| return 5; | |
| } | |
| if (Array.isArray(object)) { | |
| throw new BSONError('serialize does not support an array as the root input'); | |
| } | |
| if (typeof object !== 'object') { | |
| throw new BSONError('serialize does not support non-object as the root input'); | |
| } | |
| else if ('_bsontype' in object && typeof object._bsontype === 'string') { | |
| throw new BSONError(`BSON types cannot be serialized as a document`); | |
| } | |
| else if (isDate(object) || | |
| isRegExp(object) || | |
| isUint8Array(object) || | |
| isAnyArrayBuffer(object)) { | |
| throw new BSONError(`date, regexp, typedarray, and arraybuffer cannot be BSON documents`); | |
| } | |
| path = new Set(); | |
| } | |
| path.add(object); | |
| let index = startingIndex + 4; | |
| if (Array.isArray(object)) { | |
| for (let i = 0; i < object.length; i++) { | |
| const key = `${i}`; | |
| let value = object[i]; | |
| if (typeof value?.toBSON === 'function') { | |
| value = value.toBSON(); | |
| } | |
| const type = typeof value; | |
| if (value === undefined) { | |
| index = serializeNull(buffer, key, value, index); | |
| } | |
| else if (value === null) { | |
| index = serializeNull(buffer, key, value, index); | |
| } | |
| else if (type === 'string') { | |
| index = serializeString(buffer, key, value, index); | |
| } | |
| else if (type === 'number') { | |
| index = serializeNumber(buffer, key, value, index); | |
| } | |
| else if (type === 'bigint') { | |
| index = serializeBigInt(buffer, key, value, index); | |
| } | |
| else if (type === 'boolean') { | |
| index = serializeBoolean(buffer, key, value, index); | |
| } | |
| else if (type === 'object' && value._bsontype == null) { | |
| if (value instanceof Date || isDate(value)) { | |
| index = serializeDate(buffer, key, value, index); | |
| } | |
| else if (value instanceof Uint8Array || isUint8Array(value)) { | |
| index = serializeBuffer(buffer, key, value, index); | |
| } | |
| else if (value instanceof RegExp || isRegExp(value)) { | |
| index = serializeRegExp(buffer, key, value, index); | |
| } | |
| else { | |
| index = serializeObject(buffer, key, value, index, checkKeys, depth, serializeFunctions, ignoreUndefined, path); | |
| } | |
| } | |
| else if (type === 'object') { | |
| if (value[BSON_VERSION_SYMBOL] !== BSON_MAJOR_VERSION) { | |
| throw new BSONVersionError(); | |
| } | |
| else if (value._bsontype === 'ObjectId') { | |
| index = serializeObjectId(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'Decimal128') { | |
| index = serializeDecimal128(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'Long' || value._bsontype === 'Timestamp') { | |
| index = serializeLong(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'Double') { | |
| index = serializeDouble(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'Code') { | |
| index = serializeCode(buffer, key, value, index, checkKeys, depth, serializeFunctions, ignoreUndefined, path); | |
| } | |
| else if (value._bsontype === 'Binary') { | |
| index = serializeBinary(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'BSONSymbol') { | |
| index = serializeSymbol(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'DBRef') { | |
| index = serializeDBRef(buffer, key, value, index, depth, serializeFunctions, path); | |
| } | |
| else if (value._bsontype === 'BSONRegExp') { | |
| index = serializeBSONRegExp(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'Int32') { | |
| index = serializeInt32(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'MinKey' || value._bsontype === 'MaxKey') { | |
| index = serializeMinMax(buffer, key, value, index); | |
| } | |
| else if (typeof value._bsontype !== 'undefined') { | |
| throw new BSONError(`Unrecognized or invalid _bsontype: ${String(value._bsontype)}`); | |
| } | |
| } | |
| else if (type === 'function' && serializeFunctions) { | |
| index = serializeFunction(buffer, key, value, index); | |
| } | |
| } | |
| } | |
| else if (object instanceof Map || isMap(object)) { | |
| const iterator = object.entries(); | |
| let done = false; | |
| while (!done) { | |
| const entry = iterator.next(); | |
| done = !!entry.done; | |
| if (done) | |
| continue; | |
| const key = entry.value ? entry.value[0] : undefined; | |
| let value = entry.value ? entry.value[1] : undefined; | |
| if (typeof value?.toBSON === 'function') { | |
| value = value.toBSON(); | |
| } | |
| const type = typeof value; | |
| if (typeof key === 'string' && !ignoreKeys.has(key)) { | |
| if (key.match(regexp) != null) { | |
| throw new BSONError('key ' + key + ' must not contain null bytes'); | |
| } | |
| if (checkKeys) { | |
| if ('$' === key[0]) { | |
| throw new BSONError('key ' + key + " must not start with '$'"); | |
| } | |
| else if (key.includes('.')) { | |
| throw new BSONError('key ' + key + " must not contain '.'"); | |
| } | |
| } | |
| } | |
| if (value === undefined) { | |
| if (ignoreUndefined === false) | |
| index = serializeNull(buffer, key, value, index); | |
| } | |
| else if (value === null) { | |
| index = serializeNull(buffer, key, value, index); | |
| } | |
| else if (type === 'string') { | |
| index = serializeString(buffer, key, value, index); | |
| } | |
| else if (type === 'number') { | |
| index = serializeNumber(buffer, key, value, index); | |
| } | |
| else if (type === 'bigint') { | |
| index = serializeBigInt(buffer, key, value, index); | |
| } | |
| else if (type === 'boolean') { | |
| index = serializeBoolean(buffer, key, value, index); | |
| } | |
| else if (type === 'object' && value._bsontype == null) { | |
| if (value instanceof Date || isDate(value)) { | |
| index = serializeDate(buffer, key, value, index); | |
| } | |
| else if (value instanceof Uint8Array || isUint8Array(value)) { | |
| index = serializeBuffer(buffer, key, value, index); | |
| } | |
| else if (value instanceof RegExp || isRegExp(value)) { | |
| index = serializeRegExp(buffer, key, value, index); | |
| } | |
| else { | |
| index = serializeObject(buffer, key, value, index, checkKeys, depth, serializeFunctions, ignoreUndefined, path); | |
| } | |
| } | |
| else if (type === 'object') { | |
| if (value[BSON_VERSION_SYMBOL] !== BSON_MAJOR_VERSION) { | |
| throw new BSONVersionError(); | |
| } | |
| else if (value._bsontype === 'ObjectId') { | |
| index = serializeObjectId(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'Decimal128') { | |
| index = serializeDecimal128(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'Long' || value._bsontype === 'Timestamp') { | |
| index = serializeLong(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'Double') { | |
| index = serializeDouble(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'Code') { | |
| index = serializeCode(buffer, key, value, index, checkKeys, depth, serializeFunctions, ignoreUndefined, path); | |
| } | |
| else if (value._bsontype === 'Binary') { | |
| index = serializeBinary(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'BSONSymbol') { | |
| index = serializeSymbol(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'DBRef') { | |
| index = serializeDBRef(buffer, key, value, index, depth, serializeFunctions, path); | |
| } | |
| else if (value._bsontype === 'BSONRegExp') { | |
| index = serializeBSONRegExp(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'Int32') { | |
| index = serializeInt32(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'MinKey' || value._bsontype === 'MaxKey') { | |
| index = serializeMinMax(buffer, key, value, index); | |
| } | |
| else if (typeof value._bsontype !== 'undefined') { | |
| throw new BSONError(`Unrecognized or invalid _bsontype: ${String(value._bsontype)}`); | |
| } | |
| } | |
| else if (type === 'function' && serializeFunctions) { | |
| index = serializeFunction(buffer, key, value, index); | |
| } | |
| } | |
| } | |
| else { | |
| if (typeof object?.toBSON === 'function') { | |
| object = object.toBSON(); | |
| if (object != null && typeof object !== 'object') { | |
| throw new BSONError('toBSON function did not return an object'); | |
| } | |
| } | |
| for (const key of Object.keys(object)) { | |
| let value = object[key]; | |
| if (typeof value?.toBSON === 'function') { | |
| value = value.toBSON(); | |
| } | |
| const type = typeof value; | |
| if (typeof key === 'string' && !ignoreKeys.has(key)) { | |
| if (key.match(regexp) != null) { | |
| throw new BSONError('key ' + key + ' must not contain null bytes'); | |
| } | |
| if (checkKeys) { | |
| if ('$' === key[0]) { | |
| throw new BSONError('key ' + key + " must not start with '$'"); | |
| } | |
| else if (key.includes('.')) { | |
| throw new BSONError('key ' + key + " must not contain '.'"); | |
| } | |
| } | |
| } | |
| if (value === undefined) { | |
| if (ignoreUndefined === false) | |
| index = serializeNull(buffer, key, value, index); | |
| } | |
| else if (value === null) { | |
| index = serializeNull(buffer, key, value, index); | |
| } | |
| else if (type === 'string') { | |
| index = serializeString(buffer, key, value, index); | |
| } | |
| else if (type === 'number') { | |
| index = serializeNumber(buffer, key, value, index); | |
| } | |
| else if (type === 'bigint') { | |
| index = serializeBigInt(buffer, key, value, index); | |
| } | |
| else if (type === 'boolean') { | |
| index = serializeBoolean(buffer, key, value, index); | |
| } | |
| else if (type === 'object' && value._bsontype == null) { | |
| if (value instanceof Date || isDate(value)) { | |
| index = serializeDate(buffer, key, value, index); | |
| } | |
| else if (value instanceof Uint8Array || isUint8Array(value)) { | |
| index = serializeBuffer(buffer, key, value, index); | |
| } | |
| else if (value instanceof RegExp || isRegExp(value)) { | |
| index = serializeRegExp(buffer, key, value, index); | |
| } | |
| else { | |
| index = serializeObject(buffer, key, value, index, checkKeys, depth, serializeFunctions, ignoreUndefined, path); | |
| } | |
| } | |
| else if (type === 'object') { | |
| if (value[BSON_VERSION_SYMBOL] !== BSON_MAJOR_VERSION) { | |
| throw new BSONVersionError(); | |
| } | |
| else if (value._bsontype === 'ObjectId') { | |
| index = serializeObjectId(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'Decimal128') { | |
| index = serializeDecimal128(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'Long' || value._bsontype === 'Timestamp') { | |
| index = serializeLong(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'Double') { | |
| index = serializeDouble(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'Code') { | |
| index = serializeCode(buffer, key, value, index, checkKeys, depth, serializeFunctions, ignoreUndefined, path); | |
| } | |
| else if (value._bsontype === 'Binary') { | |
| index = serializeBinary(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'BSONSymbol') { | |
| index = serializeSymbol(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'DBRef') { | |
| index = serializeDBRef(buffer, key, value, index, depth, serializeFunctions, path); | |
| } | |
| else if (value._bsontype === 'BSONRegExp') { | |
| index = serializeBSONRegExp(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'Int32') { | |
| index = serializeInt32(buffer, key, value, index); | |
| } | |
| else if (value._bsontype === 'MinKey' || value._bsontype === 'MaxKey') { | |
| index = serializeMinMax(buffer, key, value, index); | |
| } | |
| else if (typeof value._bsontype !== 'undefined') { | |
| throw new BSONError(`Unrecognized or invalid _bsontype: ${String(value._bsontype)}`); | |
| } | |
| } | |
| else if (type === 'function' && serializeFunctions) { | |
| index = serializeFunction(buffer, key, value, index); | |
| } | |
| } | |
| } | |
| path.delete(object); | |
| buffer[index++] = 0x00; | |
| const size = index - startingIndex; | |
| startingIndex += NumberUtils.setInt32LE(buffer, startingIndex, size); | |
| return index; | |
| } | |
| function isBSONType(value) { | |
| return (value != null && | |
| typeof value === 'object' && | |
| '_bsontype' in value && | |
| typeof value._bsontype === 'string'); | |
| } | |
| const keysToCodecs = { | |
| $oid: ObjectId, | |
| $binary: Binary, | |
| $uuid: Binary, | |
| $symbol: BSONSymbol, | |
| $numberInt: Int32, | |
| $numberDecimal: Decimal128, | |
| $numberDouble: Double, | |
| $numberLong: Long, | |
| $minKey: MinKey, | |
| $maxKey: MaxKey, | |
| $regex: BSONRegExp, | |
| $regularExpression: BSONRegExp, | |
| $timestamp: Timestamp | |
| }; | |
| function deserializeValue(value, options = {}) { | |
| if (typeof value === 'number') { | |
| const in32BitRange = value <= BSON_INT32_MAX && value >= BSON_INT32_MIN; | |
| const in64BitRange = value <= BSON_INT64_MAX && value >= BSON_INT64_MIN; | |
| if (options.relaxed || options.legacy) { | |
| return value; | |
| } | |
| if (Number.isInteger(value) && !Object.is(value, -0)) { | |
| if (in32BitRange) { | |
| return new Int32(value); | |
| } | |
| if (in64BitRange) { | |
| if (options.useBigInt64) { | |
| return BigInt(value); | |
| } | |
| return Long.fromNumber(value); | |
| } | |
| } | |
| return new Double(value); | |
| } | |
| if (value == null || typeof value !== 'object') | |
| return value; | |
| if (value.$undefined) | |
| return null; | |
| const keys = Object.keys(value).filter(k => k.startsWith('$') && value[k] != null); | |
| for (let i = 0; i < keys.length; i++) { | |
| const c = keysToCodecs[keys[i]]; | |
| if (c) | |
| return c.fromExtendedJSON(value, options); | |
| } | |
| if (value.$date != null) { | |
| const d = value.$date; | |
| const date = new Date(); | |
| if (options.legacy) { | |
| if (typeof d === 'number') | |
| date.setTime(d); | |
| else if (typeof d === 'string') | |
| date.setTime(Date.parse(d)); | |
| else if (typeof d === 'bigint') | |
| date.setTime(Number(d)); | |
| else | |
| throw new BSONRuntimeError(`Unrecognized type for EJSON date: ${typeof d}`); | |
| } | |
| else { | |
| if (typeof d === 'string') | |
| date.setTime(Date.parse(d)); | |
| else if (Long.isLong(d)) | |
| date.setTime(d.toNumber()); | |
| else if (typeof d === 'number' && options.relaxed) | |
| date.setTime(d); | |
| else if (typeof d === 'bigint') | |
| date.setTime(Number(d)); | |
| else | |
| throw new BSONRuntimeError(`Unrecognized type for EJSON date: ${typeof d}`); | |
| } | |
| return date; | |
| } | |
| if (value.$code != null) { | |
| const copy = Object.assign({}, value); | |
| if (value.$scope) { | |
| copy.$scope = deserializeValue(value.$scope); | |
| } | |
| return Code.fromExtendedJSON(value); | |
| } | |
| if (isDBRefLike(value) || value.$dbPointer) { | |
| const v = value.$ref ? value : value.$dbPointer; | |
| if (v instanceof DBRef) | |
| return v; | |
| const dollarKeys = Object.keys(v).filter(k => k.startsWith('$')); | |
| let valid = true; | |
| dollarKeys.forEach(k => { | |
| if (['$ref', '$id', '$db'].indexOf(k) === -1) | |
| valid = false; | |
| }); | |
| if (valid) | |
| return DBRef.fromExtendedJSON(v); | |
| } | |
| return value; | |
| } | |
| function serializeArray(array, options) { | |
| return array.map((v, index) => { | |
| options.seenObjects.push({ propertyName: `index ${index}`, obj: null }); | |
| try { | |
| return serializeValue(v, options); | |
| } | |
| finally { | |
| options.seenObjects.pop(); | |
| } | |
| }); | |
| } | |
| function getISOString(date) { | |
| const isoStr = date.toISOString(); | |
| return date.getUTCMilliseconds() !== 0 ? isoStr : isoStr.slice(0, -5) + 'Z'; | |
| } | |
| function serializeValue(value, options) { | |
| if (value instanceof Map || isMap(value)) { | |
| const obj = Object.create(null); | |
| for (const [k, v] of value) { | |
| if (typeof k !== 'string') { | |
| throw new BSONError('Can only serialize maps with string keys'); | |
| } | |
| obj[k] = v; | |
| } | |
| return serializeValue(obj, options); | |
| } | |
| if ((typeof value === 'object' || typeof value === 'function') && value !== null) { | |
| const index = options.seenObjects.findIndex(entry => entry.obj === value); | |
| if (index !== -1) { | |
| const props = options.seenObjects.map(entry => entry.propertyName); | |
| const leadingPart = props | |
| .slice(0, index) | |
| .map(prop => `${prop} -> `) | |
| .join(''); | |
| const alreadySeen = props[index]; | |
| const circularPart = ' -> ' + | |
| props | |
| .slice(index + 1, props.length - 1) | |
| .map(prop => `${prop} -> `) | |
| .join(''); | |
| const current = props[props.length - 1]; | |
| const leadingSpace = ' '.repeat(leadingPart.length + alreadySeen.length / 2); | |
| const dashes = '-'.repeat(circularPart.length + (alreadySeen.length + current.length) / 2 - 1); | |
| throw new BSONError('Converting circular structure to EJSON:\n' + | |
| ` ${leadingPart}${alreadySeen}${circularPart}${current}\n` + | |
| ` ${leadingSpace}\\${dashes}/`); | |
| } | |
| options.seenObjects[options.seenObjects.length - 1].obj = value; | |
| } | |
| if (Array.isArray(value)) | |
| return serializeArray(value, options); | |
| if (value === undefined) | |
| return options.ignoreUndefined ? undefined : null; | |
| if (value instanceof Date || isDate(value)) { | |
| const dateNum = value.getTime(), inRange = dateNum > -1 && dateNum < 253402318800000; | |
| if (options.legacy) { | |
| return options.relaxed && inRange | |
| ? { $date: value.getTime() } | |
| : { $date: getISOString(value) }; | |
| } | |
| return options.relaxed && inRange | |
| ? { $date: getISOString(value) } | |
| : { $date: { $numberLong: value.getTime().toString() } }; | |
| } | |
| if (typeof value === 'number' && (!options.relaxed || !isFinite(value))) { | |
| if (Number.isInteger(value) && !Object.is(value, -0)) { | |
| if (value >= BSON_INT32_MIN && value <= BSON_INT32_MAX) { | |
| return { $numberInt: value.toString() }; | |
| } | |
| if (value >= BSON_INT64_MIN && value <= BSON_INT64_MAX) { | |
| return { $numberLong: value.toString() }; | |
| } | |
| } | |
| return { $numberDouble: Object.is(value, -0) ? '-0.0' : value.toString() }; | |
| } | |
| if (typeof value === 'bigint') { | |
| if (!options.relaxed) { | |
| return { $numberLong: BigInt.asIntN(64, value).toString() }; | |
| } | |
| return Number(BigInt.asIntN(64, value)); | |
| } | |
| if (value instanceof RegExp || isRegExp(value)) { | |
| let flags = value.flags; | |
| if (flags === undefined) { | |
| const match = value.toString().match(/[gimuy]*$/); | |
| if (match) { | |
| flags = match[0]; | |
| } | |
| } | |
| const rx = new BSONRegExp(value.source, flags); | |
| return rx.toExtendedJSON(options); | |
| } | |
| if (value != null && typeof value === 'object') | |
| return serializeDocument(value, options); | |
| return value; | |
| } | |
| const BSON_TYPE_MAPPINGS = { | |
| Binary: (o) => new Binary(o.value(), o.sub_type), | |
| Code: (o) => new Code(o.code, o.scope), | |
| DBRef: (o) => new DBRef(o.collection || o.namespace, o.oid, o.db, o.fields), | |
| Decimal128: (o) => new Decimal128(o.bytes), | |
| Double: (o) => new Double(o.value), | |
| Int32: (o) => new Int32(o.value), | |
| Long: (o) => Long.fromBits(o.low != null ? o.low : o.low_, o.low != null ? o.high : o.high_, o.low != null ? o.unsigned : o.unsigned_), | |
| MaxKey: () => new MaxKey(), | |
| MinKey: () => new MinKey(), | |
| ObjectId: (o) => new ObjectId(o), | |
| BSONRegExp: (o) => new BSONRegExp(o.pattern, o.options), | |
| BSONSymbol: (o) => new BSONSymbol(o.value), | |
| Timestamp: (o) => Timestamp.fromBits(o.low, o.high) | |
| }; | |
| function serializeDocument(doc, options) { | |
| if (doc == null || typeof doc !== 'object') | |
| throw new BSONError('not an object instance'); | |
| const bsontype = doc._bsontype; | |
| if (typeof bsontype === 'undefined') { | |
| const _doc = {}; | |
| for (const name of Object.keys(doc)) { | |
| options.seenObjects.push({ propertyName: name, obj: null }); | |
| try { | |
| const value = serializeValue(doc[name], options); | |
| if (name === '__proto__') { | |
| Object.defineProperty(_doc, name, { | |
| value, | |
| writable: true, | |
| enumerable: true, | |
| configurable: true | |
| }); | |
| } | |
| else { | |
| _doc[name] = value; | |
| } | |
| } | |
| finally { | |
| options.seenObjects.pop(); | |
| } | |
| } | |
| return _doc; | |
| } | |
| else if (doc != null && | |
| typeof doc === 'object' && | |
| typeof doc._bsontype === 'string' && | |
| doc[BSON_VERSION_SYMBOL] !== BSON_MAJOR_VERSION) { | |
| throw new BSONVersionError(); | |
| } | |
| else if (isBSONType(doc)) { | |
| let outDoc = doc; | |
| if (typeof outDoc.toExtendedJSON !== 'function') { | |
| const mapper = BSON_TYPE_MAPPINGS[doc._bsontype]; | |
| if (!mapper) { | |
| throw new BSONError('Unrecognized or invalid _bsontype: ' + doc._bsontype); | |
| } | |
| outDoc = mapper(outDoc); | |
| } | |
| if (bsontype === 'Code' && outDoc.scope) { | |
| outDoc = new Code(outDoc.code, serializeValue(outDoc.scope, options)); | |
| } | |
| else if (bsontype === 'DBRef' && outDoc.oid) { | |
| outDoc = new DBRef(serializeValue(outDoc.collection, options), serializeValue(outDoc.oid, options), serializeValue(outDoc.db, options), serializeValue(outDoc.fields, options)); | |
| } | |
| return outDoc.toExtendedJSON(options); | |
| } | |
| else { | |
| throw new BSONError('_bsontype must be a string, but was: ' + typeof bsontype); | |
| } | |
| } | |
| function parse(text, options) { | |
| const ejsonOptions = { | |
| useBigInt64: options?.useBigInt64 ?? false, | |
| relaxed: options?.relaxed ?? true, | |
| legacy: options?.legacy ?? false | |
| }; | |
| return JSON.parse(text, (key, value) => { | |
| if (key.indexOf('\x00') !== -1) { | |
| throw new BSONError(`BSON Document field names cannot contain null bytes, found: ${JSON.stringify(key)}`); | |
| } | |
| return deserializeValue(value, ejsonOptions); | |
| }); | |
| } | |
| function stringify(value, replacer, space, options) { | |
| if (space != null && typeof space === 'object') { | |
| options = space; | |
| space = 0; | |
| } | |
| if (replacer != null && typeof replacer === 'object' && !Array.isArray(replacer)) { | |
| options = replacer; | |
| replacer = undefined; | |
| space = 0; | |
| } | |
| const serializeOptions = Object.assign({ relaxed: true, legacy: false }, options, { | |
| seenObjects: [{ propertyName: '(root)', obj: null }] | |
| }); | |
| const doc = serializeValue(value, serializeOptions); | |
| return JSON.stringify(doc, replacer, space); | |
| } | |
| function EJSONserialize(value, options) { | |
| options = options || {}; | |
| return JSON.parse(stringify(value, options)); | |
| } | |
| function EJSONdeserialize(ejson, options) { | |
| options = options || {}; | |
| return parse(JSON.stringify(ejson), options); | |
| } | |
| const EJSON = Object.create(null); | |
| EJSON.parse = parse; | |
| EJSON.stringify = stringify; | |
| EJSON.serialize = EJSONserialize; | |
| EJSON.deserialize = EJSONdeserialize; | |
| Object.freeze(EJSON); | |
| const BSONElementType = { | |
| double: 1, | |
| string: 2, | |
| object: 3, | |
| array: 4, | |
| binData: 5, | |
| undefined: 6, | |
| objectId: 7, | |
| bool: 8, | |
| date: 9, | |
| null: 10, | |
| regex: 11, | |
| dbPointer: 12, | |
| javascript: 13, | |
| symbol: 14, | |
| javascriptWithScope: 15, | |
| int: 16, | |
| timestamp: 17, | |
| long: 18, | |
| decimal: 19, | |
| minKey: 255, | |
| maxKey: 127 | |
| }; | |
| function getSize(source, offset) { | |
| try { | |
| return NumberUtils.getNonnegativeInt32LE(source, offset); | |
| } | |
| catch (cause) { | |
| throw new BSONOffsetError('BSON size cannot be negative', offset, { cause }); | |
| } | |
| } | |
| function findNull(bytes, offset) { | |
| let nullTerminatorOffset = offset; | |
| for (; bytes[nullTerminatorOffset] !== 0x00; nullTerminatorOffset++) | |
| ; | |
| if (nullTerminatorOffset === bytes.length - 1) { | |
| throw new BSONOffsetError('Null terminator not found', offset); | |
| } | |
| return nullTerminatorOffset; | |
| } | |
| function parseToElements(bytes, startOffset = 0) { | |
| startOffset ??= 0; | |
| if (bytes.length < 5) { | |
| throw new BSONOffsetError(`Input must be at least 5 bytes, got ${bytes.length} bytes`, startOffset); | |
| } | |
| const documentSize = getSize(bytes, startOffset); | |
| if (documentSize > bytes.length - startOffset) { | |
| throw new BSONOffsetError(`Parsed documentSize (${documentSize} bytes) does not match input length (${bytes.length} bytes)`, startOffset); | |
| } | |
| if (bytes[startOffset + documentSize - 1] !== 0x00) { | |
| throw new BSONOffsetError('BSON documents must end in 0x00', startOffset + documentSize); | |
| } | |
| const elements = []; | |
| let offset = startOffset + 4; | |
| while (offset <= documentSize + startOffset) { | |
| const type = bytes[offset]; | |
| offset += 1; | |
| if (type === 0) { | |
| if (offset - startOffset !== documentSize) { | |
| throw new BSONOffsetError(`Invalid 0x00 type byte`, offset); | |
| } | |
| break; | |
| } | |
| const nameOffset = offset; | |
| const nameLength = findNull(bytes, offset) - nameOffset; | |
| offset += nameLength + 1; | |
| let length; | |
| if (type === BSONElementType.double || | |
| type === BSONElementType.long || | |
| type === BSONElementType.date || | |
| type === BSONElementType.timestamp) { | |
| length = 8; | |
| } | |
| else if (type === BSONElementType.int) { | |
| length = 4; | |
| } | |
| else if (type === BSONElementType.objectId) { | |
| length = 12; | |
| } | |
| else if (type === BSONElementType.decimal) { | |
| length = 16; | |
| } | |
| else if (type === BSONElementType.bool) { | |
| length = 1; | |
| } | |
| else if (type === BSONElementType.null || | |
| type === BSONElementType.undefined || | |
| type === BSONElementType.maxKey || | |
| type === BSONElementType.minKey) { | |
| length = 0; | |
| } | |
| else if (type === BSONElementType.regex) { | |
| length = findNull(bytes, findNull(bytes, offset) + 1) + 1 - offset; | |
| } | |
| else if (type === BSONElementType.object || | |
| type === BSONElementType.array || | |
| type === BSONElementType.javascriptWithScope) { | |
| length = getSize(bytes, offset); | |
| } | |
| else if (type === BSONElementType.string || | |
| type === BSONElementType.binData || | |
| type === BSONElementType.dbPointer || | |
| type === BSONElementType.javascript || | |
| type === BSONElementType.symbol) { | |
| length = getSize(bytes, offset) + 4; | |
| if (type === BSONElementType.binData) { | |
| length += 1; | |
| } | |
| if (type === BSONElementType.dbPointer) { | |
| length += 12; | |
| } | |
| } | |
| else { | |
| throw new BSONOffsetError(`Invalid 0x${type.toString(16).padStart(2, '0')} type byte`, offset); | |
| } | |
| if (length > documentSize) { | |
| throw new BSONOffsetError('value reports length larger than document', offset); | |
| } | |
| elements.push([type, nameOffset, nameLength, offset, length]); | |
| offset += length; | |
| } | |
| return elements; | |
| } | |
| const onDemand = Object.create(null); | |
| onDemand.parseToElements = parseToElements; | |
| onDemand.ByteUtils = ByteUtils; | |
| onDemand.NumberUtils = NumberUtils; | |
| Object.freeze(onDemand); | |
| const MAXSIZE = 1024 * 1024 * 17; | |
| let buffer = ByteUtils.allocate(MAXSIZE); | |
| function setInternalBufferSize(size) { | |
| if (buffer.length < size) { | |
| buffer = ByteUtils.allocate(size); | |
| } | |
| } | |
| function serialize(object, options = {}) { | |
| const checkKeys = typeof options.checkKeys === 'boolean' ? options.checkKeys : false; | |
| const serializeFunctions = typeof options.serializeFunctions === 'boolean' ? options.serializeFunctions : false; | |
| const ignoreUndefined = typeof options.ignoreUndefined === 'boolean' ? options.ignoreUndefined : true; | |
| const minInternalBufferSize = typeof options.minInternalBufferSize === 'number' ? options.minInternalBufferSize : MAXSIZE; | |
| if (buffer.length < minInternalBufferSize) { | |
| buffer = ByteUtils.allocate(minInternalBufferSize); | |
| } | |
| const serializationIndex = serializeInto(buffer, object, checkKeys, 0, 0, serializeFunctions, ignoreUndefined, null); | |
| const finishedBuffer = ByteUtils.allocateUnsafe(serializationIndex); | |
| finishedBuffer.set(buffer.subarray(0, serializationIndex), 0); | |
| return finishedBuffer; | |
| } | |
| function serializeWithBufferAndIndex(object, finalBuffer, options = {}) { | |
| const checkKeys = typeof options.checkKeys === 'boolean' ? options.checkKeys : false; | |
| const serializeFunctions = typeof options.serializeFunctions === 'boolean' ? options.serializeFunctions : false; | |
| const ignoreUndefined = typeof options.ignoreUndefined === 'boolean' ? options.ignoreUndefined : true; | |
| const startIndex = typeof options.index === 'number' ? options.index : 0; | |
| const serializationIndex = serializeInto(buffer, object, checkKeys, 0, 0, serializeFunctions, ignoreUndefined, null); | |
| finalBuffer.set(buffer.subarray(0, serializationIndex), startIndex); | |
| return startIndex + serializationIndex - 1; | |
| } | |
| function deserialize(buffer, options = {}) { | |
| return internalDeserialize(ByteUtils.toLocalBufferType(buffer), options); | |
| } | |
| function calculateObjectSize(object, options = {}) { | |
| options = options || {}; | |
| const serializeFunctions = typeof options.serializeFunctions === 'boolean' ? options.serializeFunctions : false; | |
| const ignoreUndefined = typeof options.ignoreUndefined === 'boolean' ? options.ignoreUndefined : true; | |
| return internalCalculateObjectSize(object, serializeFunctions, ignoreUndefined); | |
| } | |
| function deserializeStream(data, startIndex, numberOfDocuments, documents, docStartIndex, options) { | |
| const internalOptions = Object.assign({ allowObjectSmallerThanBufferSize: true, index: 0 }, options); | |
| const bufferData = ByteUtils.toLocalBufferType(data); | |
| let index = startIndex; | |
| for (let i = 0; i < numberOfDocuments; i++) { | |
| const size = NumberUtils.getInt32LE(bufferData, index); | |
| internalOptions.index = index; | |
| documents[docStartIndex + i] = internalDeserialize(bufferData, internalOptions); | |
| index = index + size; | |
| } | |
| return index; | |
| } | |
| var bson = /*#__PURE__*/Object.freeze({ | |
| __proto__: null, | |
| BSONError: BSONError, | |
| BSONOffsetError: BSONOffsetError, | |
| BSONRegExp: BSONRegExp, | |
| BSONRuntimeError: BSONRuntimeError, | |
| BSONSymbol: BSONSymbol, | |
| BSONType: BSONType, | |
| BSONValue: BSONValue, | |
| BSONVersionError: BSONVersionError, | |
| Binary: Binary, | |
| ByteUtils: ByteUtils, | |
| Code: Code, | |
| DBRef: DBRef, | |
| Decimal128: Decimal128, | |
| Double: Double, | |
| EJSON: EJSON, | |
| Int32: Int32, | |
| Long: Long, | |
| MaxKey: MaxKey, | |
| MinKey: MinKey, | |
| NumberUtils: NumberUtils, | |
| ObjectId: ObjectId, | |
| Timestamp: Timestamp, | |
| UUID: UUID, | |
| bsonType: bsonType, | |
| calculateObjectSize: calculateObjectSize, | |
| deserialize: deserialize, | |
| deserializeStream: deserializeStream, | |
| onDemand: onDemand, | |
| serialize: serialize, | |
| serializeWithBufferAndIndex: serializeWithBufferAndIndex, | |
| setInternalBufferSize: setInternalBufferSize | |
| }); | |
| exports.BSON = bson; | |
| exports.BSONError = BSONError; | |
| exports.BSONOffsetError = BSONOffsetError; | |
| exports.BSONRegExp = BSONRegExp; | |
| exports.BSONRuntimeError = BSONRuntimeError; | |
| exports.BSONSymbol = BSONSymbol; | |
| exports.BSONType = BSONType; | |
| exports.BSONValue = BSONValue; | |
| exports.BSONVersionError = BSONVersionError; | |
| exports.Binary = Binary; | |
| exports.ByteUtils = ByteUtils; | |
| exports.Code = Code; | |
| exports.DBRef = DBRef; | |
| exports.Decimal128 = Decimal128; | |
| exports.Double = Double; | |
| exports.EJSON = EJSON; | |
| exports.Int32 = Int32; | |
| exports.Long = Long; | |
| exports.MaxKey = MaxKey; | |
| exports.MinKey = MinKey; | |
| exports.NumberUtils = NumberUtils; | |
| exports.ObjectId = ObjectId; | |
| exports.Timestamp = Timestamp; | |
| exports.UUID = UUID; | |
| exports.bsonType = bsonType; | |
| exports.calculateObjectSize = calculateObjectSize; | |
| exports.deserialize = deserialize; | |
| exports.deserializeStream = deserializeStream; | |
| exports.onDemand = onDemand; | |
| exports.serialize = serialize; | |
| exports.serializeWithBufferAndIndex = serializeWithBufferAndIndex; | |
| exports.setInternalBufferSize = setInternalBufferSize; | |
| return exports; | |
| })({}); | |
| //# sourceMappingURL=bson.bundle.js.map | |