You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
364 lines
7.5 KiB
364 lines
7.5 KiB
'use strict';
|
|
|
|
/*!
|
|
* Module dependencies.
|
|
*/
|
|
|
|
var Buffer = require('safe-buffer').Buffer;
|
|
var RegExpClone = require('regexp-clone');
|
|
|
|
var specialProperties = ['__proto__', 'constructor', 'prototype'];
|
|
|
|
/**
|
|
* Clones objects
|
|
*
|
|
* @param {Object} obj the object to clone
|
|
* @param {Object} options
|
|
* @return {Object} the cloned object
|
|
* @api private
|
|
*/
|
|
|
|
var clone = exports.clone = function clone(obj, options) {
|
|
if (obj === undefined || obj === null)
|
|
return obj;
|
|
|
|
if (Array.isArray(obj))
|
|
return exports.cloneArray(obj, options);
|
|
|
|
if (obj.constructor) {
|
|
if (/ObjectI[dD]$/.test(obj.constructor.name)) {
|
|
return 'function' == typeof obj.clone
|
|
? obj.clone()
|
|
: new obj.constructor(obj.id);
|
|
}
|
|
|
|
if (obj.constructor.name === 'ReadPreference') {
|
|
return new obj.constructor(obj.mode, clone(obj.tags, options));
|
|
}
|
|
|
|
if ('Binary' == obj._bsontype && obj.buffer && obj.value) {
|
|
return 'function' == typeof obj.clone
|
|
? obj.clone()
|
|
: new obj.constructor(obj.value(true), obj.sub_type);
|
|
}
|
|
|
|
if ('Date' === obj.constructor.name || 'Function' === obj.constructor.name)
|
|
return new obj.constructor(+obj);
|
|
|
|
if ('RegExp' === obj.constructor.name)
|
|
return RegExpClone(obj);
|
|
|
|
if ('Buffer' === obj.constructor.name)
|
|
return exports.cloneBuffer(obj);
|
|
}
|
|
|
|
if (isObject(obj))
|
|
return exports.cloneObject(obj, options);
|
|
|
|
if (obj.valueOf)
|
|
return obj.valueOf();
|
|
};
|
|
|
|
/*!
|
|
* ignore
|
|
*/
|
|
|
|
exports.cloneObject = function cloneObject(obj, options) {
|
|
var minimize = options && options.minimize;
|
|
var ret = {};
|
|
var hasKeys;
|
|
var val;
|
|
|
|
for (const k of Object.keys(obj)) {
|
|
// Not technically prototype pollution because this wouldn't merge properties
|
|
// onto `Object.prototype`, but avoid properties like __proto__ as a precaution.
|
|
if (specialProperties.indexOf(k) !== -1) {
|
|
continue;
|
|
}
|
|
|
|
val = clone(obj[k], options);
|
|
|
|
if (!minimize || ('undefined' !== typeof val)) {
|
|
hasKeys || (hasKeys = true);
|
|
ret[k] = val;
|
|
}
|
|
}
|
|
|
|
return minimize
|
|
? hasKeys && ret
|
|
: ret;
|
|
};
|
|
|
|
exports.cloneArray = function cloneArray(arr, options) {
|
|
var ret = [];
|
|
for (var i = 0, l = arr.length; i < l; i++)
|
|
ret.push(clone(arr[i], options));
|
|
return ret;
|
|
};
|
|
|
|
/**
|
|
* process.nextTick helper.
|
|
*
|
|
* Wraps the given `callback` in a try/catch. If an error is
|
|
* caught it will be thrown on nextTick.
|
|
*
|
|
* node-mongodb-native had a habit of state corruption when
|
|
* an error was immediately thrown from within a collection
|
|
* method (find, update, etc) callback.
|
|
*
|
|
* @param {Function} [callback]
|
|
* @api private
|
|
*/
|
|
|
|
exports.tick = function tick(callback) {
|
|
if ('function' !== typeof callback) return;
|
|
return function() {
|
|
// callbacks should always be fired on the next
|
|
// turn of the event loop. A side benefit is
|
|
// errors thrown from executing the callback
|
|
// will not cause drivers state to be corrupted
|
|
// which has historically been a problem.
|
|
var args = arguments;
|
|
soon(function() {
|
|
callback.apply(this, args);
|
|
});
|
|
};
|
|
};
|
|
|
|
/**
|
|
* Merges `from` into `to` without overwriting existing properties.
|
|
*
|
|
* @param {Object} to
|
|
* @param {Object} from
|
|
* @api private
|
|
*/
|
|
|
|
exports.merge = function merge(to, from) {
|
|
var keys = Object.keys(from),
|
|
i = keys.length,
|
|
key;
|
|
|
|
while (i--) {
|
|
key = keys[i];
|
|
if (specialProperties.indexOf(key) !== -1) {
|
|
continue;
|
|
}
|
|
if ('undefined' === typeof to[key]) {
|
|
to[key] = from[key];
|
|
} else {
|
|
if (exports.isObject(from[key])) {
|
|
merge(to[key], from[key]);
|
|
} else {
|
|
to[key] = from[key];
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Same as merge but clones the assigned values.
|
|
*
|
|
* @param {Object} to
|
|
* @param {Object} from
|
|
* @api private
|
|
*/
|
|
|
|
exports.mergeClone = function mergeClone(to, from) {
|
|
var keys = Object.keys(from),
|
|
i = keys.length,
|
|
key;
|
|
|
|
while (i--) {
|
|
key = keys[i];
|
|
if (specialProperties.indexOf(key) !== -1) {
|
|
continue;
|
|
}
|
|
if ('undefined' === typeof to[key]) {
|
|
to[key] = clone(from[key]);
|
|
} else {
|
|
if (exports.isObject(from[key])) {
|
|
mergeClone(to[key], from[key]);
|
|
} else {
|
|
to[key] = clone(from[key]);
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
/**
|
|
* Read pref helper (mongo 2.2 drivers support this)
|
|
*
|
|
* Allows using aliases instead of full preference names:
|
|
*
|
|
* p primary
|
|
* pp primaryPreferred
|
|
* s secondary
|
|
* sp secondaryPreferred
|
|
* n nearest
|
|
*
|
|
* @param {String} pref
|
|
*/
|
|
|
|
exports.readPref = function readPref(pref) {
|
|
switch (pref) {
|
|
case 'p':
|
|
pref = 'primary';
|
|
break;
|
|
case 'pp':
|
|
pref = 'primaryPreferred';
|
|
break;
|
|
case 's':
|
|
pref = 'secondary';
|
|
break;
|
|
case 'sp':
|
|
pref = 'secondaryPreferred';
|
|
break;
|
|
case 'n':
|
|
pref = 'nearest';
|
|
break;
|
|
}
|
|
|
|
return pref;
|
|
};
|
|
|
|
|
|
/**
|
|
* Read Concern helper (mongo 3.2 drivers support this)
|
|
*
|
|
* Allows using string to specify read concern level:
|
|
*
|
|
* local 3.2+
|
|
* available 3.6+
|
|
* majority 3.2+
|
|
* linearizable 3.4+
|
|
* snapshot 4.0+
|
|
*
|
|
* @param {String|Object} concern
|
|
*/
|
|
|
|
exports.readConcern = function readConcern(concern) {
|
|
if ('string' === typeof concern) {
|
|
switch (concern) {
|
|
case 'l':
|
|
concern = 'local';
|
|
break;
|
|
case 'a':
|
|
concern = 'available';
|
|
break;
|
|
case 'm':
|
|
concern = 'majority';
|
|
break;
|
|
case 'lz':
|
|
concern = 'linearizable';
|
|
break;
|
|
case 's':
|
|
concern = 'snapshot';
|
|
break;
|
|
}
|
|
concern = { level: concern };
|
|
}
|
|
return concern;
|
|
};
|
|
|
|
/**
|
|
* Object.prototype.toString.call helper
|
|
*/
|
|
|
|
var _toString = Object.prototype.toString;
|
|
exports.toString = function(arg) {
|
|
return _toString.call(arg);
|
|
};
|
|
|
|
/**
|
|
* Determines if `arg` is an object.
|
|
*
|
|
* @param {Object|Array|String|Function|RegExp|any} arg
|
|
* @return {Boolean}
|
|
*/
|
|
|
|
var isObject = exports.isObject = function(arg) {
|
|
return '[object Object]' == exports.toString(arg);
|
|
};
|
|
|
|
/**
|
|
* Determines if `arg` is an array.
|
|
*
|
|
* @param {Object}
|
|
* @return {Boolean}
|
|
* @see nodejs utils
|
|
*/
|
|
|
|
exports.isArray = function(arg) {
|
|
return Array.isArray(arg) ||
|
|
'object' == typeof arg && '[object Array]' == exports.toString(arg);
|
|
};
|
|
|
|
/**
|
|
* Object.keys helper
|
|
*/
|
|
|
|
exports.keys = Object.keys;
|
|
|
|
/**
|
|
* Basic Object.create polyfill.
|
|
* Only one argument is supported.
|
|
*
|
|
* Based on https://developer.mozilla.org/en-US/docs/JavaScript/Reference/Global_Objects/Object/create
|
|
*/
|
|
|
|
exports.create = 'function' == typeof Object.create
|
|
? Object.create
|
|
: create;
|
|
|
|
function create(proto) {
|
|
if (arguments.length > 1) {
|
|
throw new Error('Adding properties is not supported');
|
|
}
|
|
|
|
function F() {}
|
|
F.prototype = proto;
|
|
return new F;
|
|
}
|
|
|
|
/**
|
|
* inheritance
|
|
*/
|
|
|
|
exports.inherits = function(ctor, superCtor) {
|
|
ctor.prototype = exports.create(superCtor.prototype);
|
|
ctor.prototype.constructor = ctor;
|
|
};
|
|
|
|
/**
|
|
* nextTick helper
|
|
* compat with node 0.10 which behaves differently than previous versions
|
|
*/
|
|
|
|
var soon = exports.soon = 'function' == typeof setImmediate
|
|
? setImmediate
|
|
: process.nextTick;
|
|
|
|
/**
|
|
* Clones the contents of a buffer.
|
|
*
|
|
* @param {Buffer} buff
|
|
* @return {Buffer}
|
|
*/
|
|
|
|
exports.cloneBuffer = function(buff) {
|
|
var dupe = Buffer.alloc(buff.length);
|
|
buff.copy(dupe, 0, 0, buff.length);
|
|
return dupe;
|
|
};
|
|
|
|
/**
|
|
* Check if this object is an arguments object
|
|
*
|
|
* @param {Any} v
|
|
* @return {Boolean}
|
|
*/
|
|
|
|
exports.isArgumentsObject = function(v) {
|
|
return Object.prototype.toString.call(v) === '[object Arguments]';
|
|
};
|