Skip to content

Commit 2dd54e5

Browse files
authored
fix: Throw on BigInt type values (#397)
Presently BigInt values were being lost in serialization silently. Add case and error on BigInt values to avoid loss of data. NODE-2529
1 parent b95f059 commit 2dd54e5

File tree

5 files changed

+110
-3
lines changed

5 files changed

+110
-3
lines changed

src/long.ts

+16-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
1-
import type { Timestamp } from './timestamp';
21
import type { EJSONOptions } from './extended_json';
32
import { isObjectLike } from './parser/utils';
3+
import type { Timestamp } from './timestamp';
44

55
interface LongWASMHelpers {
66
/** Gets the high bits of the last operation performed */
@@ -191,6 +191,16 @@ export class Long {
191191
return Long.fromBits(value % TWO_PWR_32_DBL | 0, (value / TWO_PWR_32_DBL) | 0, unsigned);
192192
}
193193

194+
/**
195+
* Returns a Long representing the given value, provided that it is a finite number. Otherwise, zero is returned.
196+
* @param value - The number in question
197+
* @param unsigned - Whether unsigned or not, defaults to signed
198+
* @returns The corresponding Long value
199+
*/
200+
static fromBigInt(value: bigint, unsigned?: boolean): Long {
201+
return Long.fromString(value.toString(), unsigned);
202+
}
203+
194204
/**
195205
* Returns a Long representation of the given string, written using the specified radix.
196206
* @param str - The textual representation of the Long
@@ -797,6 +807,11 @@ export class Long {
797807
return this.high * TWO_PWR_32_DBL + (this.low >>> 0);
798808
}
799809

810+
/** Converts the Long to a BigInt (arbitrary precision). */
811+
toBigInt(): bigint {
812+
return BigInt(this.toString());
813+
}
814+
800815
/**
801816
* Converts this Long to its byte representation.
802817
* @param le - Whether little or big endian, defaults to big endian

src/parser/serializer.ts

+13-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,13 @@ import { Map } from '../map';
1515
import type { MinKey } from '../min_key';
1616
import type { ObjectId } from '../objectid';
1717
import type { BSONRegExp } from '../regexp';
18-
import { isDate, isUint8Array, normalizedFunctionString } from './utils';
18+
import {
19+
isBigInt64Array,
20+
isBigUInt64Array,
21+
isDate,
22+
isUint8Array,
23+
normalizedFunctionString
24+
} from './utils';
1925

2026
export interface SerializeOptions {
2127
/** the serializer will check if keys are valid. */
@@ -807,6 +813,8 @@ export function serializeInto(
807813
index = serializeString(buffer, key, value, index, true);
808814
} else if (typeof value === 'number') {
809815
index = serializeNumber(buffer, key, value, index, true);
816+
} else if (typeof value === 'bigint') {
817+
throw new TypeError('Unsupported type BigInt, please use Decimal128');
810818
} else if (typeof value === 'boolean') {
811819
index = serializeBoolean(buffer, key, value, index, true);
812820
} else if (value instanceof Date || isDate(value)) {
@@ -913,6 +921,8 @@ export function serializeInto(
913921
index = serializeString(buffer, key, value, index);
914922
} else if (type === 'number') {
915923
index = serializeNumber(buffer, key, value, index);
924+
} else if (type === 'bigint' || isBigInt64Array(value) || isBigUInt64Array(value)) {
925+
throw new TypeError('Unsupported type BigInt, please use Decimal128');
916926
} else if (type === 'boolean') {
917927
index = serializeBoolean(buffer, key, value, index);
918928
} else if (value instanceof Date || isDate(value)) {
@@ -1015,6 +1025,8 @@ export function serializeInto(
10151025
index = serializeString(buffer, key, value, index);
10161026
} else if (type === 'number') {
10171027
index = serializeNumber(buffer, key, value, index);
1028+
} else if (type === 'bigint') {
1029+
throw new TypeError('Unsupported type BigInt, please use Decimal128');
10181030
} else if (type === 'boolean') {
10191031
index = serializeBoolean(buffer, key, value, index);
10201032
} else if (value instanceof Date || isDate(value)) {

src/parser/utils.ts

+20
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,8 @@ function insecureRandomBytes(size: number): Uint8Array {
1818
// eslint-disable-next-line @typescript-eslint/no-explicit-any
1919
declare let window: any;
2020
declare let require: Function;
21+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
22+
declare let global: any;
2123

2224
export let randomBytes = insecureRandomBytes;
2325
if (typeof window !== 'undefined' && window.crypto && window.crypto.getRandomValues) {
@@ -40,6 +42,24 @@ export function isUint8Array(value: unknown): value is Uint8Array {
4042
return Object.prototype.toString.call(value) === '[object Uint8Array]';
4143
}
4244

45+
export function isBigInt64Array(value: unknown): value is BigInt64Array {
46+
return Object.prototype.toString.call(value) === '[object BigInt64Array]';
47+
}
48+
49+
export function isBigUInt64Array(value: unknown): value is BigUint64Array {
50+
return Object.prototype.toString.call(value) === '[object BigUint64Array]';
51+
}
52+
53+
/** Call to check if your environment has `Buffer` */
54+
export function haveBuffer(): boolean {
55+
return typeof global !== 'undefined' && typeof global.Buffer !== 'undefined';
56+
}
57+
58+
/** Callable in any environment to check if value is a Buffer */
59+
export function isBuffer(value: unknown): value is Buffer {
60+
return haveBuffer() && Buffer.isBuffer(value);
61+
}
62+
4363
// To ensure that 0.4 of node works correctly
4464
export function isDate(d: unknown): d is Date {
4565
return isObjectLike(d) && Object.prototype.toString.call(d) === '[object Date]';

test/node/bigint_tests.js

+58
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,58 @@
1+
/* globals BigInt */
2+
'use strict';
3+
4+
const BSON = require('../register-bson');
5+
6+
describe('BSON BigInt Support', function () {
7+
before(function () {
8+
try {
9+
BigInt(0);
10+
} catch (_) {
11+
this.skip('JS VM does not support BigInt');
12+
}
13+
});
14+
it('Should serialize an int that fits in int32', function () {
15+
const testDoc = { b: BigInt(32) };
16+
expect(() => BSON.serialize(testDoc)).to.throw(TypeError);
17+
18+
// const serializedDoc = BSON.serialize(testDoc);
19+
// // prettier-ignore
20+
// const resultBuffer = Buffer.from([0x0C, 0x00, 0x00, 0x00, 0x10, 0x62, 0x00, 0x20, 0x00, 0x00, 0x00, 0x00]);
21+
// const resultDoc = BSON.deserialize(serializedDoc);
22+
// expect(Array.from(serializedDoc)).to.have.members(Array.from(resultBuffer));
23+
// expect(BigInt(resultDoc.b)).to.equal(testDoc.b);
24+
});
25+
26+
it('Should serialize an int that fits in int64', function () {
27+
const testDoc = { b: BigInt(0x1ffffffff) };
28+
expect(() => BSON.serialize(testDoc)).to.throw(TypeError);
29+
30+
// const serializedDoc = BSON.serialize(testDoc);
31+
// // prettier-ignore
32+
// const resultBuffer = Buffer.from([0x10, 0x00, 0x00, 0x00, 0x12, 0x62, 0x00, 0xFF, 0xFF, 0xFF, 0xFF, 0x01, 0x00, 0x00, 0x00, 0x00]);
33+
// const resultDoc = BSON.deserialize(serializedDoc);
34+
// expect(Array.from(serializedDoc)).to.have.members(Array.from(resultBuffer));
35+
// expect(BigInt(resultDoc.b)).to.equal(testDoc.b);
36+
});
37+
38+
it('Should serialize an int that fits in decimal128', function () {
39+
const testDoc = { b: BigInt('9223372036854776001') }; // int64 max + 1
40+
expect(() => BSON.serialize(testDoc)).to.throw(TypeError);
41+
42+
// const serializedDoc = BSON.serialize(testDoc);
43+
// // prettier-ignore
44+
// const resultBuffer = Buffer.from([0x18, 0x00, 0x00, 0x00, 0x13, 0x62, 0x00, 0xC1, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x80, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x40, 0x30, 0x00]);
45+
// const resultDoc = BSON.deserialize(serializedDoc);
46+
// expect(Array.from(serializedDoc)).to.have.members(Array.from(resultBuffer));
47+
// expect(resultDoc.b._bsontype).to.equal('Decimal128');
48+
// expect(BigInt(resultDoc.b.toString())).to.equal(testDoc.b);
49+
});
50+
51+
it('Should throw if BigInt is too large to serialize', function () {
52+
const testDoc = {
53+
b: BigInt('9'.repeat(35))
54+
}; // decimal 128 can only encode 34 digits of precision
55+
expect(() => BSON.serialize(testDoc)).to.throw(TypeError);
56+
// expect(() => BSON.serialize(testDoc)).to.throw();
57+
});
58+
});

tsconfig.json

+3-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,9 @@
88
"module": "commonjs",
99
"moduleResolution": "node",
1010
"lib": [
11-
"ES2017"
11+
"ES2017",
12+
"ES2020.BigInt",
13+
"ES2017.TypedArrays"
1214
],
1315
"outDir": "lib",
1416
// We don't make use of tslib helpers

0 commit comments

Comments
 (0)