Skip to content

Commit

Permalink
feat(ext-json): add extended JSON support to the bson library
Browse files Browse the repository at this point in the history
  • Loading branch information
mbroadst committed Oct 1, 2018
1 parent 10e5f00 commit d6b71ab
Showing 1 changed file with 297 additions and 0 deletions.
297 changes: 297 additions & 0 deletions lib/extended_json.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,297 @@
'use strict';

// const Buffer = require('buffer').Buffer;
// const Map = require('./map');
const Long = require('./long');
const Double = require('./double');
const Timestamp = require('./timestamp');
const ObjectId = require('./objectid');
const BSONRegExp = require('./regexp');
const Symbol = require('./symbol');
const Int32 = require('./int_32');
const Code = require('./code');
const Decimal128 = require('./decimal128');
const MinKey = require('./min_key');
const MaxKey = require('./max_key');
const DBRef = require('./db_ref');
const Binary = require('./binary');

// all the types where we don't need to do any special processing and can just pass the EJSON
//straight to type.fromExtendedJSON
const keysToCodecs = {
$oid: ObjectId,
$binary: Binary,
$symbol: Symbol,
$numberInt: Int32,
$numberDecimal: Decimal128,
$numberDouble: Double,
$numberLong: Long,
$minKey: MinKey,
$maxKey: MaxKey,
$regularExpression: BSONRegExp,
$timestamp: Timestamp
};

// const BSONTypes = Object.keys(BSON);

function deserializeValue(self, key, value, options) {
if (typeof value === 'number') {
// if it's an integer, should interpret as smallest BSON integer
// that can represent it exactly. (if out of range, interpret as double.)
if (Math.floor(value) === value) {
let int32Range = value >= BSON_INT32_MIN && value <= BSON_INT32_MAX,
int64Range = value >= BSON_INT64_MIN && value <= BSON_INT64_MAX;

if (int32Range) return options.strict ? new Int32(value) : value;
if (int64Range) return options.strict ? new Long.fromNumber(value) : value;
}
// If the number is a non-integer or out of integer range, should interpret as BSON Double.
return new Double(value);
}

// from here on out we're looking for bson types, so bail if its not an object
if (value == null || typeof value !== 'object') return value;

// upgrade deprecated undefined to null
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++) {
let c = keysToCodecs[keys[i]];
if (c) return c.fromExtendedJSON(value, options);
}

if (value.$date != null) {
const d = value.$date;
const date = new Date();

if (typeof d === 'string') date.setTime(Date.parse(d));
else if (d instanceof Long) date.setTime(d.toNumber());
else if (typeof d === 'number' && options.relaxed) date.setTime(d);
return date;
}

if (value.$code != null) {
let copy = Object.assign({}, value);
if (value.$scope) {
copy.$scope = deserializeValue(self, null, value.$scope);
}

return Code.fromExtendedJSON(value);
}

if (value.$ref != null || value.$dbPointer != null) {
let v = value.$ref ? value : value.$dbPointer;

// we run into this in a "degenerate EJSON" case (with $id and $ref order flipped)
// because of the order JSON.parse goes through the document
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;
});

// only make DBRef if $ keys are all valid
if (valid) return DBRef.fromExtendedJSON(v);
}

return value;
}

/**
* Parse an Extended JSON string, constructing the JavaScript value or object described by that
* string.
*
* @param {string} text
* @param {object} [options] Optional settings
* @param {boolean} [options.relaxed=true] Attempt to return native JS types where possible, rather than BSON types (if true)
* @return {object}
*
* @example
* const EJSON = require('mongodb-extjson');
* const text = '{ "int32": { "$numberInt": "10" } }';
*
* // prints { int32: { [String: '10'] _bsontype: 'Int32', value: '10' } }
* console.log(EJSON.parse(text, { relaxed: false }));
*
* // prints { int32: 10 }
* console.log(EJSON.parse(text));
*/
function parse(text, options) {
options = Object.assign({}, { relaxed: true }, options);

// relaxed implies not strict
if (typeof options.relaxed === 'boolean') options.strict = !options.relaxed;
if (typeof options.strict === 'boolean') options.relaxed = !options.strict;

return JSON.parse(text, (key, value) => deserializeValue(this, key, value, options));
}

//
// Serializer
//

// MAX INT32 boundaries
const BSON_INT32_MAX = 0x7fffffff,
BSON_INT32_MIN = -0x80000000,
BSON_INT64_MAX = 0x7fffffffffffffff,
BSON_INT64_MIN = -0x8000000000000000;

/**
* Converts a BSON document to an Extended JSON string, optionally replacing values if a replacer
* function is specified or optionally including only the specified properties if a replacer array
* is specified.
*
* @param {object} value The value to convert to extended JSON
* @param {function|array} [replacer] A function that alters the behavior of the stringification process, or an array of String and Number objects that serve as a whitelist for selecting/filtering the properties of the value object to be included in the JSON string. If this value is null or not provided, all properties of the object are included in the resulting JSON string
* @param {string|number} [space] A String or Number object that's used to insert white space into the output JSON string for readability purposes.
* @param {object} [options] Optional settings
* @param {boolean} [options.relaxed=true] Enabled Extended JSON's `relaxed` mode
* @returns {string}
*
* @example
* const EJSON = require('mongodb-extjson');
* const Int32 = require('mongodb').Int32;
* const doc = { int32: new Int32(10) };
*
* // prints '{"int32":{"$numberInt":"10"}}'
* console.log(EJSON.stringify(doc, { relaxed: false }));
*
* // prints '{"int32":10}'
* console.log(EJSON.stringify(doc));
*/
function stringify(value, replacer, space, options) {
if (space != null && typeof space === 'object') (options = space), (space = 0);
if (replacer != null && typeof replacer === 'object')
(options = replacer), (replacer = null), (space = 0);
options = Object.assign({}, { relaxed: true }, options);

const doc = Array.isArray(value)
? serializeArray(value, options)
: serializeDocument(value, options);

return JSON.stringify(doc, replacer, space);
}

/**
* Serializes an object to an Extended JSON string, and reparse it as a JavaScript object.
*
* @param {object} bson The object to serialize
* @param {object} [options] Optional settings passed to the `stringify` function
* @return {object}
*/
function serialize(bson, options) {
options = options || {};
return JSON.parse(stringify(bson, options));
}

/**
* Deserializes an Extended JSON object into a plain JavaScript object with native/BSON types
*
* @param {object} ejson The Extended JSON object to deserialize
* @param {object} [options] Optional settings passed to the parse method
* @return {object}
*/
function deserialize(ejson, options) {
options = options || {};
return parse(JSON.stringify(ejson), options);
}

function serializeArray(array, options) {
return array.map(v => serializeValue(v, options));
}

function getISOString(date) {
const isoStr = date.toISOString();
// we should only show milliseconds in timestamp if they're non-zero
return date.getUTCMilliseconds() !== 0 ? isoStr : isoStr.slice(0, -5) + 'Z';
}

function serializeValue(value, options) {
if (Array.isArray(value)) return serializeArray(value, options);

if (value === undefined) return null;

if (value instanceof Date) {
let dateNum = value.getTime(),
// is it in year range 1970-9999?
inRange = dateNum > -1 && dateNum < 253402318800000;

return options.relaxed && inRange
? { $date: getISOString(value) }
: { $date: { $numberLong: value.getTime().toString() } };
}

if (typeof value === 'number' && !options.relaxed) {
// it's an integer
if (Math.floor(value) === value) {
let int32Range = value >= BSON_INT32_MIN && value <= BSON_INT32_MAX,
int64Range = value >= BSON_INT64_MIN && value <= BSON_INT64_MAX;

// interpret as being of the smallest BSON integer type that can represent the number exactly
if (int32Range) return { $numberInt: value.toString() };
if (int64Range) return { $numberLong: value.toString() };
}
return { $numberDouble: value.toString() };
}

if (value != null && typeof value === 'object') return serializeDocument(value, options);
return value;
}

function serializeDocument(doc, options) {
if (doc == null || typeof doc !== 'object') throw new Error('not an object instance');

// the document itself is a BSON type
if (doc._bsontype && typeof doc.toExtendedJSON === 'function') {
if (doc._bsontype === 'Code' && doc.scope) {
doc.scope = serializeDocument(doc.scope, options);
} else if (doc._bsontype === 'DBRef' && doc.oid) {
doc.oid = serializeDocument(doc.oid, options);
}

return doc.toExtendedJSON(options);
}

// the document is an object with nested BSON types
const _doc = {};
for (let name in doc) {
let val = doc[name];
if (Array.isArray(val)) {
_doc[name] = serializeArray(val, options);
} else if (val != null && typeof val.toExtendedJSON === 'function') {
if (val._bsontype === 'Code' && val.scope) {
val.scope = serializeDocument(val.scope, options);
} else if (val._bsontype === 'DBRef' && val.oid) {
val.oid = serializeDocument(val.oid, options);
}

_doc[name] = val.toExtendedJSON(options);
} else if (val instanceof Date) {
_doc[name] = serializeValue(val, options);
} else if (val != null && typeof val === 'object') {
_doc[name] = serializeDocument(val, options);
}
_doc[name] = serializeValue(val, options);
if (val instanceof RegExp) {
let flags = val.flags;
if (flags === undefined) {
flags = val.toString().match(/[gimuy]*$/)[0];
}

const rx = new BSONRegExp(val.source, flags);
_doc[name] = rx.toExtendedJSON();
}
}

return _doc;
}

module.exports = {
parse,
deserialize,
serialize,
stringify
};

0 comments on commit d6b71ab

Please sign in to comment.