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]';
 | 
						|
};
 |