Skip to content

Commit e943cdb

Browse files
feat(NODE-6086): add Double.fromString() method (#671)
1 parent 5a21889 commit e943cdb

File tree

2 files changed

+83
-0
lines changed

2 files changed

+83
-0
lines changed

src/double.ts

+33
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
import { BSONValue } from './bson_value';
2+
import { BSONError } from './error';
23
import type { EJSONOptions } from './extended_json';
34
import { type InspectFn, defaultInspect } from './parser/utils';
45

@@ -32,6 +33,38 @@ export class Double extends BSONValue {
3233
this.value = +value;
3334
}
3435

36+
/**
37+
* Attempt to create an double type from string.
38+
*
39+
* This method will throw a BSONError on any string input that is not representable as a IEEE-754 64-bit double.
40+
* Notably, this method will also throw on the following string formats:
41+
* - Strings in non-decimal formats (exponent notation, binary, hex, or octal digits)
42+
* - Strings with characters other than numeric, floating point, or leading sign characters (Note: 'Infinity', '-Infinity', and 'NaN' input strings are still allowed)
43+
* - Strings with leading and/or trailing whitespace
44+
*
45+
* Strings with leading zeros, however, are also allowed
46+
*
47+
* @param value - the string we want to represent as an double.
48+
*/
49+
static fromString(value: string): Double {
50+
const coercedValue = Number(value);
51+
const nonFiniteValidInputs = ['Infinity', '-Infinity', 'NaN'];
52+
53+
if (value.trim() !== value) {
54+
throw new BSONError(`Input: '${value}' contains whitespace`);
55+
} else if (value === '') {
56+
throw new BSONError(`Input is an empty string`);
57+
} else if (/[^-0-9.]/.test(value) && !nonFiniteValidInputs.includes(value)) {
58+
throw new BSONError(`Input: '${value}' contains invalid characters`);
59+
} else if (
60+
(!Number.isFinite(coercedValue) && !nonFiniteValidInputs.includes(value)) ||
61+
(Number.isNaN(coercedValue) && value !== 'NaN')
62+
) {
63+
throw new BSONError(`Input: ${value} is not representable as a Double`); // generic case
64+
}
65+
return new Double(coercedValue);
66+
}
67+
3568
/**
3669
* Access the number value.
3770
*

test/node/double.test.ts

+50
Original file line numberDiff line numberDiff line change
@@ -225,6 +225,56 @@ describe('BSON Double Precision', function () {
225225
});
226226
});
227227
});
228+
229+
describe('fromString', () => {
230+
const acceptedInputs = [
231+
['zero', '0', 0],
232+
['non-leading zeros', '45000000', 45000000],
233+
['zero with leading zeros', '000000.0000', 0],
234+
['positive leading zeros', '000000867.1', 867.1],
235+
['negative leading zeros', '-00007.980', -7.98],
236+
['positive integer with decimal', '2.0', 2],
237+
['zero with decimal', '0.0', 0.0],
238+
['Infinity', 'Infinity', Infinity],
239+
['-Infinity', '-Infinity', -Infinity],
240+
['NaN', 'NaN', NaN],
241+
['basic floating point', '-4.556000', -4.556],
242+
['negative zero', '-0', -0]
243+
];
244+
245+
const errorInputs = [
246+
['commas', '34,450', 'contains invalid characters'],
247+
['exponentiation notation', '1.34e16', 'contains invalid characters'],
248+
['octal', '0o1', 'contains invalid characters'],
249+
['binary', '0b1', 'contains invalid characters'],
250+
['hex', '0x1', 'contains invalid characters'],
251+
['empty string', '', 'is an empty string'],
252+
['leading and trailing whitespace', ' 89 ', 'contains whitespace'],
253+
['fake positive infinity', '2e308', 'contains invalid characters'],
254+
['fake negative infinity', '-2e308', 'contains invalid characters'],
255+
['fraction', '3/4', 'contains invalid characters'],
256+
['foo', 'foo', 'contains invalid characters']
257+
];
258+
259+
for (const [testName, value, expectedDouble] of acceptedInputs) {
260+
context(`when the input is ${testName}`, () => {
261+
it(`should successfully return a Double representation`, () => {
262+
if (value === 'NaN') {
263+
expect(isNaN(Double.fromString(value))).to.be.true;
264+
} else {
265+
expect(Double.fromString(value).value).to.equal(expectedDouble);
266+
}
267+
});
268+
});
269+
}
270+
for (const [testName, value, expectedErrMsg] of errorInputs) {
271+
context(`when the input is ${testName}`, () => {
272+
it(`should throw an error containing '${expectedErrMsg}'`, () => {
273+
expect(() => Double.fromString(value)).to.throw(BSON.BSONError, expectedErrMsg);
274+
});
275+
});
276+
}
277+
});
228278
});
229279

230280
function serializeThenDeserialize(value) {

0 commit comments

Comments
 (0)