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.
		
		
		
		
		
			
		
			
				
					736 lines
				
				17 KiB
			
		
		
			
		
	
	
					736 lines
				
				17 KiB
			| 
											3 years ago
										 | /* | ||
|  |  * lib/jsprim.js: utilities for primitive JavaScript types | ||
|  |  */ | ||
|  | 
 | ||
|  | var mod_assert = require('assert-plus'); | ||
|  | var mod_util = require('util'); | ||
|  | 
 | ||
|  | var mod_extsprintf = require('extsprintf'); | ||
|  | var mod_verror = require('verror'); | ||
|  | var mod_jsonschema = require('json-schema'); | ||
|  | 
 | ||
|  | /* | ||
|  |  * Public interface | ||
|  |  */ | ||
|  | exports.deepCopy = deepCopy; | ||
|  | exports.deepEqual = deepEqual; | ||
|  | exports.isEmpty = isEmpty; | ||
|  | exports.hasKey = hasKey; | ||
|  | exports.forEachKey = forEachKey; | ||
|  | exports.pluck = pluck; | ||
|  | exports.flattenObject = flattenObject; | ||
|  | exports.flattenIter = flattenIter; | ||
|  | exports.validateJsonObject = validateJsonObjectJS; | ||
|  | exports.validateJsonObjectJS = validateJsonObjectJS; | ||
|  | exports.randElt = randElt; | ||
|  | exports.extraProperties = extraProperties; | ||
|  | exports.mergeObjects = mergeObjects; | ||
|  | 
 | ||
|  | exports.startsWith = startsWith; | ||
|  | exports.endsWith = endsWith; | ||
|  | 
 | ||
|  | exports.parseInteger = parseInteger; | ||
|  | 
 | ||
|  | exports.iso8601 = iso8601; | ||
|  | exports.rfc1123 = rfc1123; | ||
|  | exports.parseDateTime = parseDateTime; | ||
|  | 
 | ||
|  | exports.hrtimediff = hrtimeDiff; | ||
|  | exports.hrtimeDiff = hrtimeDiff; | ||
|  | exports.hrtimeAccum = hrtimeAccum; | ||
|  | exports.hrtimeAdd = hrtimeAdd; | ||
|  | exports.hrtimeNanosec = hrtimeNanosec; | ||
|  | exports.hrtimeMicrosec = hrtimeMicrosec; | ||
|  | exports.hrtimeMillisec = hrtimeMillisec; | ||
|  | 
 | ||
|  | 
 | ||
|  | /* | ||
|  |  * Deep copy an acyclic *basic* Javascript object.  This only handles basic | ||
|  |  * scalars (strings, numbers, booleans) and arbitrarily deep arrays and objects | ||
|  |  * containing these.  This does *not* handle instances of other classes. | ||
|  |  */ | ||
|  | function deepCopy(obj) | ||
|  | { | ||
|  | 	var ret, key; | ||
|  | 	var marker = '__deepCopy'; | ||
|  | 
 | ||
|  | 	if (obj && obj[marker]) | ||
|  | 		throw (new Error('attempted deep copy of cyclic object')); | ||
|  | 
 | ||
|  | 	if (obj && obj.constructor == Object) { | ||
|  | 		ret = {}; | ||
|  | 		obj[marker] = true; | ||
|  | 
 | ||
|  | 		for (key in obj) { | ||
|  | 			if (key == marker) | ||
|  | 				continue; | ||
|  | 
 | ||
|  | 			ret[key] = deepCopy(obj[key]); | ||
|  | 		} | ||
|  | 
 | ||
|  | 		delete (obj[marker]); | ||
|  | 		return (ret); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	if (obj && obj.constructor == Array) { | ||
|  | 		ret = []; | ||
|  | 		obj[marker] = true; | ||
|  | 
 | ||
|  | 		for (key = 0; key < obj.length; key++) | ||
|  | 			ret.push(deepCopy(obj[key])); | ||
|  | 
 | ||
|  | 		delete (obj[marker]); | ||
|  | 		return (ret); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/* | ||
|  | 	 * It must be a primitive type -- just return it. | ||
|  | 	 */ | ||
|  | 	return (obj); | ||
|  | } | ||
|  | 
 | ||
|  | function deepEqual(obj1, obj2) | ||
|  | { | ||
|  | 	if (typeof (obj1) != typeof (obj2)) | ||
|  | 		return (false); | ||
|  | 
 | ||
|  | 	if (obj1 === null || obj2 === null || typeof (obj1) != 'object') | ||
|  | 		return (obj1 === obj2); | ||
|  | 
 | ||
|  | 	if (obj1.constructor != obj2.constructor) | ||
|  | 		return (false); | ||
|  | 
 | ||
|  | 	var k; | ||
|  | 	for (k in obj1) { | ||
|  | 		if (!obj2.hasOwnProperty(k)) | ||
|  | 			return (false); | ||
|  | 
 | ||
|  | 		if (!deepEqual(obj1[k], obj2[k])) | ||
|  | 			return (false); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	for (k in obj2) { | ||
|  | 		if (!obj1.hasOwnProperty(k)) | ||
|  | 			return (false); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	return (true); | ||
|  | } | ||
|  | 
 | ||
|  | function isEmpty(obj) | ||
|  | { | ||
|  | 	var key; | ||
|  | 	for (key in obj) | ||
|  | 		return (false); | ||
|  | 	return (true); | ||
|  | } | ||
|  | 
 | ||
|  | function hasKey(obj, key) | ||
|  | { | ||
|  | 	mod_assert.equal(typeof (key), 'string'); | ||
|  | 	return (Object.prototype.hasOwnProperty.call(obj, key)); | ||
|  | } | ||
|  | 
 | ||
|  | function forEachKey(obj, callback) | ||
|  | { | ||
|  | 	for (var key in obj) { | ||
|  | 		if (hasKey(obj, key)) { | ||
|  | 			callback(key, obj[key]); | ||
|  | 		} | ||
|  | 	} | ||
|  | } | ||
|  | 
 | ||
|  | function pluck(obj, key) | ||
|  | { | ||
|  | 	mod_assert.equal(typeof (key), 'string'); | ||
|  | 	return (pluckv(obj, key)); | ||
|  | } | ||
|  | 
 | ||
|  | function pluckv(obj, key) | ||
|  | { | ||
|  | 	if (obj === null || typeof (obj) !== 'object') | ||
|  | 		return (undefined); | ||
|  | 
 | ||
|  | 	if (obj.hasOwnProperty(key)) | ||
|  | 		return (obj[key]); | ||
|  | 
 | ||
|  | 	var i = key.indexOf('.'); | ||
|  | 	if (i == -1) | ||
|  | 		return (undefined); | ||
|  | 
 | ||
|  | 	var key1 = key.substr(0, i); | ||
|  | 	if (!obj.hasOwnProperty(key1)) | ||
|  | 		return (undefined); | ||
|  | 
 | ||
|  | 	return (pluckv(obj[key1], key.substr(i + 1))); | ||
|  | } | ||
|  | 
 | ||
|  | /* | ||
|  |  * Invoke callback(row) for each entry in the array that would be returned by | ||
|  |  * flattenObject(data, depth).  This is just like flattenObject(data, | ||
|  |  * depth).forEach(callback), except that the intermediate array is never | ||
|  |  * created. | ||
|  |  */ | ||
|  | function flattenIter(data, depth, callback) | ||
|  | { | ||
|  | 	doFlattenIter(data, depth, [], callback); | ||
|  | } | ||
|  | 
 | ||
|  | function doFlattenIter(data, depth, accum, callback) | ||
|  | { | ||
|  | 	var each; | ||
|  | 	var key; | ||
|  | 
 | ||
|  | 	if (depth === 0) { | ||
|  | 		each = accum.slice(0); | ||
|  | 		each.push(data); | ||
|  | 		callback(each); | ||
|  | 		return; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	mod_assert.ok(data !== null); | ||
|  | 	mod_assert.equal(typeof (data), 'object'); | ||
|  | 	mod_assert.equal(typeof (depth), 'number'); | ||
|  | 	mod_assert.ok(depth >= 0); | ||
|  | 
 | ||
|  | 	for (key in data) { | ||
|  | 		each = accum.slice(0); | ||
|  | 		each.push(key); | ||
|  | 		doFlattenIter(data[key], depth - 1, each, callback); | ||
|  | 	} | ||
|  | } | ||
|  | 
 | ||
|  | function flattenObject(data, depth) | ||
|  | { | ||
|  | 	if (depth === 0) | ||
|  | 		return ([ data ]); | ||
|  | 
 | ||
|  | 	mod_assert.ok(data !== null); | ||
|  | 	mod_assert.equal(typeof (data), 'object'); | ||
|  | 	mod_assert.equal(typeof (depth), 'number'); | ||
|  | 	mod_assert.ok(depth >= 0); | ||
|  | 
 | ||
|  | 	var rv = []; | ||
|  | 	var key; | ||
|  | 
 | ||
|  | 	for (key in data) { | ||
|  | 		flattenObject(data[key], depth - 1).forEach(function (p) { | ||
|  | 			rv.push([ key ].concat(p)); | ||
|  | 		}); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	return (rv); | ||
|  | } | ||
|  | 
 | ||
|  | function startsWith(str, prefix) | ||
|  | { | ||
|  | 	return (str.substr(0, prefix.length) == prefix); | ||
|  | } | ||
|  | 
 | ||
|  | function endsWith(str, suffix) | ||
|  | { | ||
|  | 	return (str.substr( | ||
|  | 	    str.length - suffix.length, suffix.length) == suffix); | ||
|  | } | ||
|  | 
 | ||
|  | function iso8601(d) | ||
|  | { | ||
|  | 	if (typeof (d) == 'number') | ||
|  | 		d = new Date(d); | ||
|  | 	mod_assert.ok(d.constructor === Date); | ||
|  | 	return (mod_extsprintf.sprintf('%4d-%02d-%02dT%02d:%02d:%02d.%03dZ', | ||
|  | 	    d.getUTCFullYear(), d.getUTCMonth() + 1, d.getUTCDate(), | ||
|  | 	    d.getUTCHours(), d.getUTCMinutes(), d.getUTCSeconds(), | ||
|  | 	    d.getUTCMilliseconds())); | ||
|  | } | ||
|  | 
 | ||
|  | var RFC1123_MONTHS = [ | ||
|  |     'Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', | ||
|  |     'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec']; | ||
|  | var RFC1123_DAYS = [ | ||
|  |     'Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat']; | ||
|  | 
 | ||
|  | function rfc1123(date) { | ||
|  | 	return (mod_extsprintf.sprintf('%s, %02d %s %04d %02d:%02d:%02d GMT', | ||
|  | 	    RFC1123_DAYS[date.getUTCDay()], date.getUTCDate(), | ||
|  | 	    RFC1123_MONTHS[date.getUTCMonth()], date.getUTCFullYear(), | ||
|  | 	    date.getUTCHours(), date.getUTCMinutes(), | ||
|  | 	    date.getUTCSeconds())); | ||
|  | } | ||
|  | 
 | ||
|  | /* | ||
|  |  * Parses a date expressed as a string, as either a number of milliseconds since | ||
|  |  * the epoch or any string format that Date accepts, giving preference to the | ||
|  |  * former where these two sets overlap (e.g., small numbers). | ||
|  |  */ | ||
|  | function parseDateTime(str) | ||
|  | { | ||
|  | 	/* | ||
|  | 	 * This is irritatingly implicit, but significantly more concise than | ||
|  | 	 * alternatives.  The "+str" will convert a string containing only a | ||
|  | 	 * number directly to a Number, or NaN for other strings.  Thus, if the | ||
|  | 	 * conversion succeeds, we use it (this is the milliseconds-since-epoch | ||
|  | 	 * case).  Otherwise, we pass the string directly to the Date | ||
|  | 	 * constructor to parse. | ||
|  | 	 */ | ||
|  | 	var numeric = +str; | ||
|  | 	if (!isNaN(numeric)) { | ||
|  | 		return (new Date(numeric)); | ||
|  | 	} else { | ||
|  | 		return (new Date(str)); | ||
|  | 	} | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | /* | ||
|  |  * Number.*_SAFE_INTEGER isn't present before node v0.12, so we hardcode | ||
|  |  * the ES6 definitions here, while allowing for them to someday be higher. | ||
|  |  */ | ||
|  | var MAX_SAFE_INTEGER = Number.MAX_SAFE_INTEGER || 9007199254740991; | ||
|  | var MIN_SAFE_INTEGER = Number.MIN_SAFE_INTEGER || -9007199254740991; | ||
|  | 
 | ||
|  | 
 | ||
|  | /* | ||
|  |  * Default options for parseInteger(). | ||
|  |  */ | ||
|  | var PI_DEFAULTS = { | ||
|  | 	base: 10, | ||
|  | 	allowSign: true, | ||
|  | 	allowPrefix: false, | ||
|  | 	allowTrailing: false, | ||
|  | 	allowImprecise: false, | ||
|  | 	trimWhitespace: false, | ||
|  | 	leadingZeroIsOctal: false | ||
|  | }; | ||
|  | 
 | ||
|  | var CP_0 = 0x30; | ||
|  | var CP_9 = 0x39; | ||
|  | 
 | ||
|  | var CP_A = 0x41; | ||
|  | var CP_B = 0x42; | ||
|  | var CP_O = 0x4f; | ||
|  | var CP_T = 0x54; | ||
|  | var CP_X = 0x58; | ||
|  | var CP_Z = 0x5a; | ||
|  | 
 | ||
|  | var CP_a = 0x61; | ||
|  | var CP_b = 0x62; | ||
|  | var CP_o = 0x6f; | ||
|  | var CP_t = 0x74; | ||
|  | var CP_x = 0x78; | ||
|  | var CP_z = 0x7a; | ||
|  | 
 | ||
|  | var PI_CONV_DEC = 0x30; | ||
|  | var PI_CONV_UC = 0x37; | ||
|  | var PI_CONV_LC = 0x57; | ||
|  | 
 | ||
|  | 
 | ||
|  | /* | ||
|  |  * A stricter version of parseInt() that provides options for changing what | ||
|  |  * is an acceptable string (for example, disallowing trailing characters). | ||
|  |  */ | ||
|  | function parseInteger(str, uopts) | ||
|  | { | ||
|  | 	mod_assert.string(str, 'str'); | ||
|  | 	mod_assert.optionalObject(uopts, 'options'); | ||
|  | 
 | ||
|  | 	var baseOverride = false; | ||
|  | 	var options = PI_DEFAULTS; | ||
|  | 
 | ||
|  | 	if (uopts) { | ||
|  | 		baseOverride = hasKey(uopts, 'base'); | ||
|  | 		options = mergeObjects(options, uopts); | ||
|  | 		mod_assert.number(options.base, 'options.base'); | ||
|  | 		mod_assert.ok(options.base >= 2, 'options.base >= 2'); | ||
|  | 		mod_assert.ok(options.base <= 36, 'options.base <= 36'); | ||
|  | 		mod_assert.bool(options.allowSign, 'options.allowSign'); | ||
|  | 		mod_assert.bool(options.allowPrefix, 'options.allowPrefix'); | ||
|  | 		mod_assert.bool(options.allowTrailing, | ||
|  | 		    'options.allowTrailing'); | ||
|  | 		mod_assert.bool(options.allowImprecise, | ||
|  | 		    'options.allowImprecise'); | ||
|  | 		mod_assert.bool(options.trimWhitespace, | ||
|  | 		    'options.trimWhitespace'); | ||
|  | 		mod_assert.bool(options.leadingZeroIsOctal, | ||
|  | 		    'options.leadingZeroIsOctal'); | ||
|  | 
 | ||
|  | 		if (options.leadingZeroIsOctal) { | ||
|  | 			mod_assert.ok(!baseOverride, | ||
|  | 			    '"base" and "leadingZeroIsOctal" are ' + | ||
|  | 			    'mutually exclusive'); | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	var c; | ||
|  | 	var pbase = -1; | ||
|  | 	var base = options.base; | ||
|  | 	var start; | ||
|  | 	var mult = 1; | ||
|  | 	var value = 0; | ||
|  | 	var idx = 0; | ||
|  | 	var len = str.length; | ||
|  | 
 | ||
|  | 	/* Trim any whitespace on the left side. */ | ||
|  | 	if (options.trimWhitespace) { | ||
|  | 		while (idx < len && isSpace(str.charCodeAt(idx))) { | ||
|  | 			++idx; | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/* Check the number for a leading sign. */ | ||
|  | 	if (options.allowSign) { | ||
|  | 		if (str[idx] === '-') { | ||
|  | 			idx += 1; | ||
|  | 			mult = -1; | ||
|  | 		} else if (str[idx] === '+') { | ||
|  | 			idx += 1; | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/* Parse the base-indicating prefix if there is one. */ | ||
|  | 	if (str[idx] === '0') { | ||
|  | 		if (options.allowPrefix) { | ||
|  | 			pbase = prefixToBase(str.charCodeAt(idx + 1)); | ||
|  | 			if (pbase !== -1 && (!baseOverride || pbase === base)) { | ||
|  | 				base = pbase; | ||
|  | 				idx += 2; | ||
|  | 			} | ||
|  | 		} | ||
|  | 
 | ||
|  | 		if (pbase === -1 && options.leadingZeroIsOctal) { | ||
|  | 			base = 8; | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/* Parse the actual digits. */ | ||
|  | 	for (start = idx; idx < len; ++idx) { | ||
|  | 		c = translateDigit(str.charCodeAt(idx)); | ||
|  | 		if (c !== -1 && c < base) { | ||
|  | 			value *= base; | ||
|  | 			value += c; | ||
|  | 		} else { | ||
|  | 			break; | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/* If we didn't parse any digits, we have an invalid number. */ | ||
|  | 	if (start === idx) { | ||
|  | 		return (new Error('invalid number: ' + JSON.stringify(str))); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/* Trim any whitespace on the right side. */ | ||
|  | 	if (options.trimWhitespace) { | ||
|  | 		while (idx < len && isSpace(str.charCodeAt(idx))) { | ||
|  | 			++idx; | ||
|  | 		} | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/* Check for trailing characters. */ | ||
|  | 	if (idx < len && !options.allowTrailing) { | ||
|  | 		return (new Error('trailing characters after number: ' + | ||
|  | 		    JSON.stringify(str.slice(idx)))); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/* If our value is 0, we return now, to avoid returning -0. */ | ||
|  | 	if (value === 0) { | ||
|  | 		return (0); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/* Calculate our final value. */ | ||
|  | 	var result = value * mult; | ||
|  | 
 | ||
|  | 	/* | ||
|  | 	 * If the string represents a value that cannot be precisely represented | ||
|  | 	 * by JavaScript, then we want to check that: | ||
|  | 	 * | ||
|  | 	 * - We never increased the value past MAX_SAFE_INTEGER | ||
|  | 	 * - We don't make the result negative and below MIN_SAFE_INTEGER | ||
|  | 	 * | ||
|  | 	 * Because we only ever increment the value during parsing, there's no | ||
|  | 	 * chance of moving past MAX_SAFE_INTEGER and then dropping below it | ||
|  | 	 * again, losing precision in the process. This means that we only need | ||
|  | 	 * to do our checks here, at the end. | ||
|  | 	 */ | ||
|  | 	if (!options.allowImprecise && | ||
|  | 	    (value > MAX_SAFE_INTEGER || result < MIN_SAFE_INTEGER)) { | ||
|  | 		return (new Error('number is outside of the supported range: ' + | ||
|  | 		    JSON.stringify(str.slice(start, idx)))); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	return (result); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | /* | ||
|  |  * Interpret a character code as a base-36 digit. | ||
|  |  */ | ||
|  | function translateDigit(d) | ||
|  | { | ||
|  | 	if (d >= CP_0 && d <= CP_9) { | ||
|  | 		/* '0' to '9' -> 0 to 9 */ | ||
|  | 		return (d - PI_CONV_DEC); | ||
|  | 	} else if (d >= CP_A && d <= CP_Z) { | ||
|  | 		/* 'A' - 'Z' -> 10 to 35 */ | ||
|  | 		return (d - PI_CONV_UC); | ||
|  | 	} else if (d >= CP_a && d <= CP_z) { | ||
|  | 		/* 'a' - 'z' -> 10 to 35 */ | ||
|  | 		return (d - PI_CONV_LC); | ||
|  | 	} else { | ||
|  | 		/* Invalid character code */ | ||
|  | 		return (-1); | ||
|  | 	} | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | /* | ||
|  |  * Test if a value matches the ECMAScript definition of trimmable whitespace. | ||
|  |  */ | ||
|  | function isSpace(c) | ||
|  | { | ||
|  | 	return (c === 0x20) || | ||
|  | 	    (c >= 0x0009 && c <= 0x000d) || | ||
|  | 	    (c === 0x00a0) || | ||
|  | 	    (c === 0x1680) || | ||
|  | 	    (c === 0x180e) || | ||
|  | 	    (c >= 0x2000 && c <= 0x200a) || | ||
|  | 	    (c === 0x2028) || | ||
|  | 	    (c === 0x2029) || | ||
|  | 	    (c === 0x202f) || | ||
|  | 	    (c === 0x205f) || | ||
|  | 	    (c === 0x3000) || | ||
|  | 	    (c === 0xfeff); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | /* | ||
|  |  * Determine which base a character indicates (e.g., 'x' indicates hex). | ||
|  |  */ | ||
|  | function prefixToBase(c) | ||
|  | { | ||
|  | 	if (c === CP_b || c === CP_B) { | ||
|  | 		/* 0b/0B (binary) */ | ||
|  | 		return (2); | ||
|  | 	} else if (c === CP_o || c === CP_O) { | ||
|  | 		/* 0o/0O (octal) */ | ||
|  | 		return (8); | ||
|  | 	} else if (c === CP_t || c === CP_T) { | ||
|  | 		/* 0t/0T (decimal) */ | ||
|  | 		return (10); | ||
|  | 	} else if (c === CP_x || c === CP_X) { | ||
|  | 		/* 0x/0X (hexadecimal) */ | ||
|  | 		return (16); | ||
|  | 	} else { | ||
|  | 		/* Not a meaningful character */ | ||
|  | 		return (-1); | ||
|  | 	} | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | function validateJsonObjectJS(schema, input) | ||
|  | { | ||
|  | 	var report = mod_jsonschema.validate(input, schema); | ||
|  | 
 | ||
|  | 	if (report.errors.length === 0) | ||
|  | 		return (null); | ||
|  | 
 | ||
|  | 	/* Currently, we only do anything useful with the first error. */ | ||
|  | 	var error = report.errors[0]; | ||
|  | 
 | ||
|  | 	/* The failed property is given by a URI with an irrelevant prefix. */ | ||
|  | 	var propname = error['property']; | ||
|  | 	var reason = error['message'].toLowerCase(); | ||
|  | 	var i, j; | ||
|  | 
 | ||
|  | 	/* | ||
|  | 	 * There's at least one case where the property error message is | ||
|  | 	 * confusing at best.  We work around this here. | ||
|  | 	 */ | ||
|  | 	if ((i = reason.indexOf('the property ')) != -1 && | ||
|  | 	    (j = reason.indexOf(' is not defined in the schema and the ' + | ||
|  | 	    'schema does not allow additional properties')) != -1) { | ||
|  | 		i += 'the property '.length; | ||
|  | 		if (propname === '') | ||
|  | 			propname = reason.substr(i, j - i); | ||
|  | 		else | ||
|  | 			propname = propname + '.' + reason.substr(i, j - i); | ||
|  | 
 | ||
|  | 		reason = 'unsupported property'; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	var rv = new mod_verror.VError('property "%s": %s', propname, reason); | ||
|  | 	rv.jsv_details = error; | ||
|  | 	return (rv); | ||
|  | } | ||
|  | 
 | ||
|  | function randElt(arr) | ||
|  | { | ||
|  | 	mod_assert.ok(Array.isArray(arr) && arr.length > 0, | ||
|  | 	    'randElt argument must be a non-empty array'); | ||
|  | 
 | ||
|  | 	return (arr[Math.floor(Math.random() * arr.length)]); | ||
|  | } | ||
|  | 
 | ||
|  | function assertHrtime(a) | ||
|  | { | ||
|  | 	mod_assert.ok(a[0] >= 0 && a[1] >= 0, | ||
|  | 	    'negative numbers not allowed in hrtimes'); | ||
|  | 	mod_assert.ok(a[1] < 1e9, 'nanoseconds column overflow'); | ||
|  | } | ||
|  | 
 | ||
|  | /* | ||
|  |  * Compute the time elapsed between hrtime readings A and B, where A is later | ||
|  |  * than B.  hrtime readings come from Node's process.hrtime().  There is no | ||
|  |  * defined way to represent negative deltas, so it's illegal to diff B from A | ||
|  |  * where the time denoted by B is later than the time denoted by A.  If this | ||
|  |  * becomes valuable, we can define a representation and extend the | ||
|  |  * implementation to support it. | ||
|  |  */ | ||
|  | function hrtimeDiff(a, b) | ||
|  | { | ||
|  | 	assertHrtime(a); | ||
|  | 	assertHrtime(b); | ||
|  | 	mod_assert.ok(a[0] > b[0] || (a[0] == b[0] && a[1] >= b[1]), | ||
|  | 	    'negative differences not allowed'); | ||
|  | 
 | ||
|  | 	var rv = [ a[0] - b[0], 0 ]; | ||
|  | 
 | ||
|  | 	if (a[1] >= b[1]) { | ||
|  | 		rv[1] = a[1] - b[1]; | ||
|  | 	} else { | ||
|  | 		rv[0]--; | ||
|  | 		rv[1] = 1e9 - (b[1] - a[1]); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	return (rv); | ||
|  | } | ||
|  | 
 | ||
|  | /* | ||
|  |  * Convert a hrtime reading from the array format returned by Node's | ||
|  |  * process.hrtime() into a scalar number of nanoseconds. | ||
|  |  */ | ||
|  | function hrtimeNanosec(a) | ||
|  | { | ||
|  | 	assertHrtime(a); | ||
|  | 
 | ||
|  | 	return (Math.floor(a[0] * 1e9 + a[1])); | ||
|  | } | ||
|  | 
 | ||
|  | /* | ||
|  |  * Convert a hrtime reading from the array format returned by Node's | ||
|  |  * process.hrtime() into a scalar number of microseconds. | ||
|  |  */ | ||
|  | function hrtimeMicrosec(a) | ||
|  | { | ||
|  | 	assertHrtime(a); | ||
|  | 
 | ||
|  | 	return (Math.floor(a[0] * 1e6 + a[1] / 1e3)); | ||
|  | } | ||
|  | 
 | ||
|  | /* | ||
|  |  * Convert a hrtime reading from the array format returned by Node's | ||
|  |  * process.hrtime() into a scalar number of milliseconds. | ||
|  |  */ | ||
|  | function hrtimeMillisec(a) | ||
|  | { | ||
|  | 	assertHrtime(a); | ||
|  | 
 | ||
|  | 	return (Math.floor(a[0] * 1e3 + a[1] / 1e6)); | ||
|  | } | ||
|  | 
 | ||
|  | /* | ||
|  |  * Add two hrtime readings A and B, overwriting A with the result of the | ||
|  |  * addition.  This function is useful for accumulating several hrtime intervals | ||
|  |  * into a counter.  Returns A. | ||
|  |  */ | ||
|  | function hrtimeAccum(a, b) | ||
|  | { | ||
|  | 	assertHrtime(a); | ||
|  | 	assertHrtime(b); | ||
|  | 
 | ||
|  | 	/* | ||
|  | 	 * Accumulate the nanosecond component. | ||
|  | 	 */ | ||
|  | 	a[1] += b[1]; | ||
|  | 	if (a[1] >= 1e9) { | ||
|  | 		/* | ||
|  | 		 * The nanosecond component overflowed, so carry to the seconds | ||
|  | 		 * field. | ||
|  | 		 */ | ||
|  | 		a[0]++; | ||
|  | 		a[1] -= 1e9; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	/* | ||
|  | 	 * Accumulate the seconds component. | ||
|  | 	 */ | ||
|  | 	a[0] += b[0]; | ||
|  | 
 | ||
|  | 	return (a); | ||
|  | } | ||
|  | 
 | ||
|  | /* | ||
|  |  * Add two hrtime readings A and B, returning the result as a new hrtime array. | ||
|  |  * Does not modify either input argument. | ||
|  |  */ | ||
|  | function hrtimeAdd(a, b) | ||
|  | { | ||
|  | 	assertHrtime(a); | ||
|  | 
 | ||
|  | 	var rv = [ a[0], a[1] ]; | ||
|  | 
 | ||
|  | 	return (hrtimeAccum(rv, b)); | ||
|  | } | ||
|  | 
 | ||
|  | 
 | ||
|  | /* | ||
|  |  * Check an object for unexpected properties.  Accepts the object to check, and | ||
|  |  * an array of allowed property names (strings).  Returns an array of key names | ||
|  |  * that were found on the object, but did not appear in the list of allowed | ||
|  |  * properties.  If no properties were found, the returned array will be of | ||
|  |  * zero length. | ||
|  |  */ | ||
|  | function extraProperties(obj, allowed) | ||
|  | { | ||
|  | 	mod_assert.ok(typeof (obj) === 'object' && obj !== null, | ||
|  | 	    'obj argument must be a non-null object'); | ||
|  | 	mod_assert.ok(Array.isArray(allowed), | ||
|  | 	    'allowed argument must be an array of strings'); | ||
|  | 	for (var i = 0; i < allowed.length; i++) { | ||
|  | 		mod_assert.ok(typeof (allowed[i]) === 'string', | ||
|  | 		    'allowed argument must be an array of strings'); | ||
|  | 	} | ||
|  | 
 | ||
|  | 	return (Object.keys(obj).filter(function (key) { | ||
|  | 		return (allowed.indexOf(key) === -1); | ||
|  | 	})); | ||
|  | } | ||
|  | 
 | ||
|  | /* | ||
|  |  * Given three sets of properties "provided" (may be undefined), "overrides" | ||
|  |  * (required), and "defaults" (may be undefined), construct an object containing | ||
|  |  * the union of these sets with "overrides" overriding "provided", and | ||
|  |  * "provided" overriding "defaults".  None of the input objects are modified. | ||
|  |  */ | ||
|  | function mergeObjects(provided, overrides, defaults) | ||
|  | { | ||
|  | 	var rv, k; | ||
|  | 
 | ||
|  | 	rv = {}; | ||
|  | 	if (defaults) { | ||
|  | 		for (k in defaults) | ||
|  | 			rv[k] = defaults[k]; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	if (provided) { | ||
|  | 		for (k in provided) | ||
|  | 			rv[k] = provided[k]; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	if (overrides) { | ||
|  | 		for (k in overrides) | ||
|  | 			rv[k] = overrides[k]; | ||
|  | 	} | ||
|  | 
 | ||
|  | 	return (rv); | ||
|  | } |