Skip to content

Commit

Permalink
fix(NODE-3821): nullish check before using toBSON override function (#…
Browse files Browse the repository at this point in the history
  • Loading branch information
cplepage authored Jan 4, 2022
1 parent 95e8293 commit 1d898b6
Show file tree
Hide file tree
Showing 3 changed files with 89 additions and 10 deletions.
4 changes: 2 additions & 2 deletions src/parser/calculate_size.ts
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ export function calculateObjectSize(
} else {
// If we have toBSON defined, override the current object

if (object.toBSON) {
if (typeof object?.toBSON === 'function') {
object = object.toBSON();
}

Expand All @@ -47,7 +47,7 @@ function calculateElement(
ignoreUndefined = false
) {
// If we have toBSON defined, override the current object
if (value && value.toBSON) {
if (typeof value?.toBSON === 'function') {
value = value.toBSON();
}

Expand Down
14 changes: 6 additions & 8 deletions src/parser/serializer.ts
Original file line number Diff line number Diff line change
Expand Up @@ -767,8 +767,7 @@ export function serializeInto(
let value = object[i];

// Is there an override value
if (value && value.toBSON) {
if (typeof value.toBSON !== 'function') throw new BSONTypeError('toBSON is not a function');
if (typeof value?.toBSON === 'function') {
value = value.toBSON();
}

Expand Down Expand Up @@ -947,20 +946,19 @@ export function serializeInto(
}
}
} else {
// Did we provide a custom serialization method
if (object.toBSON) {
if (typeof object.toBSON !== 'function') throw new BSONTypeError('toBSON is not a function');
if (typeof object?.toBSON === 'function') {
// Provided a custom serialization method
object = object.toBSON();
if (object != null && typeof object !== 'object')
if (object != null && typeof object !== 'object') {
throw new BSONTypeError('toBSON function did not return an object');
}
}

// Iterate over all the keys
for (const key in object) {
let value = object[key];
// Is there an override value
if (value && value.toBSON) {
if (typeof value.toBSON !== 'function') throw new BSONTypeError('toBSON is not a function');
if (typeof value?.toBSON === 'function') {
value = value.toBSON();
}

Expand Down
81 changes: 81 additions & 0 deletions test/node/to_bson_test.js
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,8 @@
const BSON = require('../register-bson');
const ObjectId = BSON.ObjectId;

const BigInt = global.BigInt;

describe('toBSON', function () {
/**
* @ignore
Expand Down Expand Up @@ -129,4 +131,83 @@ describe('toBSON', function () {
expect(true).to.equal(test2);
done();
});

describe('when used on global existing types', () => {
beforeEach(() => {
Number.prototype.toBSON = () => 'hello';
String.prototype.toBSON = () => 'hello';
Boolean.prototype.toBSON = () => 'hello';
if (BigInt) BigInt.prototype.toBSON = () => 'hello';
});

afterEach(() => {
// remove prototype extension intended for test
delete Number.prototype.toBSON;
delete String.prototype.toBSON;
delete Boolean.prototype.toBSON;
if (BigInt) delete BigInt.prototype.toBSON;
});

const testToBSONFor = value => {
it(`should use toBSON on false-y ${typeof value} ${value === '' ? "''" : value}`, () => {
const serialized_data = BSON.serialize({ a: value });
expect(serialized_data.indexOf(Buffer.from('hello\0', 'utf8'))).to.be.greaterThan(0);

const deserialized_doc = BSON.deserialize(serialized_data);
expect(deserialized_doc).to.have.property('a', 'hello');
});
};

testToBSONFor(0);
testToBSONFor(NaN);
testToBSONFor('');
testToBSONFor(false);
if (BigInt) {
testToBSONFor(BigInt(0));
}

it('should use toBSON on false-y number in calculateObjectSize', () => {
// Normally is 20 bytes
// int32 0x04 'a\x00'
// int32 0x10 '0\x00' int32 \0
// \0
// ---------
// with toBSON is 26 bytes (hello + null)
// int32 0x04 'a\x00'
// int32 0x02 '0\x00' int32 'hello\0' \0
// \0
const sizeNestedToBSON = BSON.calculateObjectSize({ a: [0] });
expect(sizeNestedToBSON).to.equal(26);
});
});

it('should use toBSON in calculateObjectSize', () => {
const sizeTopLvlToBSON = BSON.calculateObjectSize({ toBSON: () => ({ a: 1 }) });
const sizeOfWhatToBSONReturns = BSON.calculateObjectSize({ a: 1 });
expect(sizeOfWhatToBSONReturns).to.equal(12);
expect(sizeTopLvlToBSON).to.equal(12);

const toBSONAsAKeySize = BSON.calculateObjectSize({ toBSON: { a: 1 } });
expect(toBSONAsAKeySize).to.equal(25);
});

it('should serialize to a key for non-function values', () => {
// int32 0x10 'toBSON\x00' int32 \0
const size = BSON.calculateObjectSize({ toBSON: 1 });
expect(size).to.equal(17);

const bytes = BSON.serialize({ toBSON: 1 });
expect(bytes.indexOf(Buffer.from('toBSON\0', 'utf8'))).to.be.greaterThan(0);
});

it('should still be omitted if serializeFunctions is true', () => {
const bytes = BSON.serialize(
{ toBSON: () => ({ a: 1, fn: () => ({ a: 1 }) }) },
{ serializeFunctions: true }
);
expect(bytes.indexOf(Buffer.from('a\0', 'utf8'))).to.be.greaterThan(0);
const doc = BSON.deserialize(bytes);
expect(doc).to.have.property('a', 1);
expect(doc).to.have.property('fn').that.is.instanceOf(BSON.Code);
});
});

0 comments on commit 1d898b6

Please sign in to comment.