454 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
		
		
			
		
	
	
			454 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
|  | 'use strict'; | |||
|  | 
 | |||
|  | var crypto = require('crypto'); | |||
|  | 
 | |||
|  | /** | |||
|  |  * Exported function | |||
|  |  * | |||
|  |  * Options: | |||
|  |  * | |||
|  |  *  - `algorithm` hash algo to be used by this instance: *'sha1', 'md5' | |||
|  |  *  - `excludeValues` {true|*false} hash object keys, values ignored | |||
|  |  *  - `encoding` hash encoding, supports 'buffer', '*hex', 'binary', 'base64' | |||
|  |  *  - `ignoreUnknown` {true|*false} ignore unknown object types | |||
|  |  *  - `replacer` optional function that replaces values before hashing | |||
|  |  *  - `respectFunctionProperties` {*true|false} consider function properties when hashing | |||
|  |  *  - `respectFunctionNames` {*true|false} consider 'name' property of functions for hashing | |||
|  |  *  - `respectType` {*true|false} Respect special properties (prototype, constructor) | |||
|  |  *    when hashing to distinguish between types | |||
|  |  *  - `unorderedArrays` {true|*false} Sort all arrays before hashing | |||
|  |  *  - `unorderedSets` {*true|false} Sort `Set` and `Map` instances before hashing | |||
|  |  *  * = default | |||
|  |  * | |||
|  |  * @param {object} object value to hash | |||
|  |  * @param {object} options hashing options | |||
|  |  * @return {string} hash value | |||
|  |  * @api public | |||
|  |  */ | |||
|  | exports = module.exports = objectHash; | |||
|  | 
 | |||
|  | function objectHash(object, options){ | |||
|  |   options = applyDefaults(object, options); | |||
|  | 
 | |||
|  |   return hash(object, options); | |||
|  | } | |||
|  | 
 | |||
|  | /** | |||
|  |  * Exported sugar methods | |||
|  |  * | |||
|  |  * @param {object} object value to hash | |||
|  |  * @return {string} hash value | |||
|  |  * @api public | |||
|  |  */ | |||
|  | exports.sha1 = function(object){ | |||
|  |   return objectHash(object); | |||
|  | }; | |||
|  | exports.keys = function(object){ | |||
|  |   return objectHash(object, {excludeValues: true, algorithm: 'sha1', encoding: 'hex'}); | |||
|  | }; | |||
|  | exports.MD5 = function(object){ | |||
|  |   return objectHash(object, {algorithm: 'md5', encoding: 'hex'}); | |||
|  | }; | |||
|  | exports.keysMD5 = function(object){ | |||
|  |   return objectHash(object, {algorithm: 'md5', encoding: 'hex', excludeValues: true}); | |||
|  | }; | |||
|  | 
 | |||
|  | // Internals
 | |||
|  | var hashes = crypto.getHashes ? crypto.getHashes().slice() : ['sha1', 'md5']; | |||
|  | hashes.push('passthrough'); | |||
|  | var encodings = ['buffer', 'hex', 'binary', 'base64']; | |||
|  | 
 | |||
|  | function applyDefaults(object, sourceOptions){ | |||
|  |   sourceOptions = sourceOptions || {}; | |||
|  | 
 | |||
|  |   // create a copy rather than mutating
 | |||
|  |   var options = {}; | |||
|  |   options.algorithm = sourceOptions.algorithm || 'sha1'; | |||
|  |   options.encoding = sourceOptions.encoding || 'hex'; | |||
|  |   options.excludeValues = sourceOptions.excludeValues ? true : false; | |||
|  |   options.algorithm = options.algorithm.toLowerCase(); | |||
|  |   options.encoding = options.encoding.toLowerCase(); | |||
|  |   options.ignoreUnknown = sourceOptions.ignoreUnknown !== true ? false : true; // default to false
 | |||
|  |   options.respectType = sourceOptions.respectType === false ? false : true; // default to true
 | |||
|  |   options.respectFunctionNames = sourceOptions.respectFunctionNames === false ? false : true; | |||
|  |   options.respectFunctionProperties = sourceOptions.respectFunctionProperties === false ? false : true; | |||
|  |   options.unorderedArrays = sourceOptions.unorderedArrays !== true ? false : true; // default to false
 | |||
|  |   options.unorderedSets = sourceOptions.unorderedSets === false ? false : true; // default to false
 | |||
|  |   options.unorderedObjects = sourceOptions.unorderedObjects === false ? false : true; // default to true
 | |||
|  |   options.replacer = sourceOptions.replacer || undefined; | |||
|  |   options.excludeKeys = sourceOptions.excludeKeys || undefined; | |||
|  | 
 | |||
|  |   if(typeof object === 'undefined') { | |||
|  |     throw new Error('Object argument required.'); | |||
|  |   } | |||
|  | 
 | |||
|  |   // if there is a case-insensitive match in the hashes list, accept it
 | |||
|  |   // (i.e. SHA256 for sha256)
 | |||
|  |   for (var i = 0; i < hashes.length; ++i) { | |||
|  |     if (hashes[i].toLowerCase() === options.algorithm.toLowerCase()) { | |||
|  |       options.algorithm = hashes[i]; | |||
|  |     } | |||
|  |   } | |||
|  | 
 | |||
|  |   if(hashes.indexOf(options.algorithm) === -1){ | |||
|  |     throw new Error('Algorithm "' + options.algorithm + '"  not supported. ' + | |||
|  |       'supported values: ' + hashes.join(', ')); | |||
|  |   } | |||
|  | 
 | |||
|  |   if(encodings.indexOf(options.encoding) === -1 && | |||
|  |      options.algorithm !== 'passthrough'){ | |||
|  |     throw new Error('Encoding "' + options.encoding + '"  not supported. ' + | |||
|  |       'supported values: ' + encodings.join(', ')); | |||
|  |   } | |||
|  | 
 | |||
|  |   return options; | |||
|  | } | |||
|  | 
 | |||
|  | /** Check if the given function is a native function */ | |||
|  | function isNativeFunction(f) { | |||
|  |   if ((typeof f) !== 'function') { | |||
|  |     return false; | |||
|  |   } | |||
|  |   var exp = /^function\s+\w*\s*\(\s*\)\s*{\s+\[native code\]\s+}$/i; | |||
|  |   return exp.exec(Function.prototype.toString.call(f)) != null; | |||
|  | } | |||
|  | 
 | |||
|  | function hash(object, options) { | |||
|  |   var hashingStream; | |||
|  | 
 | |||
|  |   if (options.algorithm !== 'passthrough') { | |||
|  |     hashingStream = crypto.createHash(options.algorithm); | |||
|  |   } else { | |||
|  |     hashingStream = new PassThrough(); | |||
|  |   } | |||
|  | 
 | |||
|  |   if (typeof hashingStream.write === 'undefined') { | |||
|  |     hashingStream.write = hashingStream.update; | |||
|  |     hashingStream.end   = hashingStream.update; | |||
|  |   } | |||
|  | 
 | |||
|  |   var hasher = typeHasher(options, hashingStream); | |||
|  |   hasher.dispatch(object); | |||
|  |   if (!hashingStream.update) { | |||
|  |     hashingStream.end(''); | |||
|  |   } | |||
|  | 
 | |||
|  |   if (hashingStream.digest) { | |||
|  |     return hashingStream.digest(options.encoding === 'buffer' ? undefined : options.encoding); | |||
|  |   } | |||
|  | 
 | |||
|  |   var buf = hashingStream.read(); | |||
|  |   if (options.encoding === 'buffer') { | |||
|  |     return buf; | |||
|  |   } | |||
|  | 
 | |||
|  |   return buf.toString(options.encoding); | |||
|  | } | |||
|  | 
 | |||
|  | /** | |||
|  |  * Expose streaming API | |||
|  |  * | |||
|  |  * @param {object} object  Value to serialize | |||
|  |  * @param {object} options  Options, as for hash() | |||
|  |  * @param {object} stream  A stream to write the serializiation to | |||
|  |  * @api public | |||
|  |  */ | |||
|  | exports.writeToStream = function(object, options, stream) { | |||
|  |   if (typeof stream === 'undefined') { | |||
|  |     stream = options; | |||
|  |     options = {}; | |||
|  |   } | |||
|  | 
 | |||
|  |   options = applyDefaults(object, options); | |||
|  | 
 | |||
|  |   return typeHasher(options, stream).dispatch(object); | |||
|  | }; | |||
|  | 
 | |||
|  | function typeHasher(options, writeTo, context){ | |||
|  |   context = context || []; | |||
|  |   var write = function(str) { | |||
|  |     if (writeTo.update) { | |||
|  |       return writeTo.update(str, 'utf8'); | |||
|  |     } else { | |||
|  |       return writeTo.write(str, 'utf8'); | |||
|  |     } | |||
|  |   }; | |||
|  | 
 | |||
|  |   return { | |||
|  |     dispatch: function(value){ | |||
|  |       if (options.replacer) { | |||
|  |         value = options.replacer(value); | |||
|  |       } | |||
|  | 
 | |||
|  |       var type = typeof value; | |||
|  |       if (value === null) { | |||
|  |         type = 'null'; | |||
|  |       } | |||
|  | 
 | |||
|  |       //console.log("[DEBUG] Dispatch: ", value, "->", type, " -> ", "_" + type);
 | |||
|  | 
 | |||
|  |       return this['_' + type](value); | |||
|  |     }, | |||
|  |     _object: function(object) { | |||
|  |       var pattern = (/\[object (.*)\]/i); | |||
|  |       var objString = Object.prototype.toString.call(object); | |||
|  |       var objType = pattern.exec(objString); | |||
|  |       if (!objType) { // object type did not match [object ...]
 | |||
|  |         objType = 'unknown:[' + objString + ']'; | |||
|  |       } else { | |||
|  |         objType = objType[1]; // take only the class name
 | |||
|  |       } | |||
|  | 
 | |||
|  |       objType = objType.toLowerCase(); | |||
|  | 
 | |||
|  |       var objectNumber = null; | |||
|  | 
 | |||
|  |       if ((objectNumber = context.indexOf(object)) >= 0) { | |||
|  |         return this.dispatch('[CIRCULAR:' + objectNumber + ']'); | |||
|  |       } else { | |||
|  |         context.push(object); | |||
|  |       } | |||
|  | 
 | |||
|  |       if (typeof Buffer !== 'undefined' && Buffer.isBuffer && Buffer.isBuffer(object)) { | |||
|  |         write('buffer:'); | |||
|  |         return write(object); | |||
|  |       } | |||
|  | 
 | |||
|  |       if(objType !== 'object' && objType !== 'function' && objType !== 'asyncfunction') { | |||
|  |         if(this['_' + objType]) { | |||
|  |           this['_' + objType](object); | |||
|  |         } else if (options.ignoreUnknown) { | |||
|  |           return write('[' + objType + ']'); | |||
|  |         } else { | |||
|  |           throw new Error('Unknown object type "' + objType + '"'); | |||
|  |         } | |||
|  |       }else{ | |||
|  |         var keys = Object.keys(object); | |||
|  |         if (options.unorderedObjects) { | |||
|  |           keys = keys.sort(); | |||
|  |         } | |||
|  |         // Make sure to incorporate special properties, so
 | |||
|  |         // Types with different prototypes will produce
 | |||
|  |         // a different hash and objects derived from
 | |||
|  |         // different functions (`new Foo`, `new Bar`) will
 | |||
|  |         // produce different hashes.
 | |||
|  |         // We never do this for native functions since some
 | |||
|  |         // seem to break because of that.
 | |||
|  |         if (options.respectType !== false && !isNativeFunction(object)) { | |||
|  |           keys.splice(0, 0, 'prototype', '__proto__', 'constructor'); | |||
|  |         } | |||
|  | 
 | |||
|  |         if (options.excludeKeys) { | |||
|  |           keys = keys.filter(function(key) { return !options.excludeKeys(key); }); | |||
|  |         } | |||
|  | 
 | |||
|  |         write('object:' + keys.length + ':'); | |||
|  |         var self = this; | |||
|  |         return keys.forEach(function(key){ | |||
|  |           self.dispatch(key); | |||
|  |           write(':'); | |||
|  |           if(!options.excludeValues) { | |||
|  |             self.dispatch(object[key]); | |||
|  |           } | |||
|  |           write(','); | |||
|  |         }); | |||
|  |       } | |||
|  |     }, | |||
|  |     _array: function(arr, unordered){ | |||
|  |       unordered = typeof unordered !== 'undefined' ? unordered : | |||
|  |         options.unorderedArrays !== false; // default to options.unorderedArrays
 | |||
|  | 
 | |||
|  |       var self = this; | |||
|  |       write('array:' + arr.length + ':'); | |||
|  |       if (!unordered || arr.length <= 1) { | |||
|  |         return arr.forEach(function(entry) { | |||
|  |           return self.dispatch(entry); | |||
|  |         }); | |||
|  |       } | |||
|  | 
 | |||
|  |       // the unordered case is a little more complicated:
 | |||
|  |       // since there is no canonical ordering on objects,
 | |||
|  |       // i.e. {a:1} < {a:2} and {a:1} > {a:2} are both false,
 | |||
|  |       // we first serialize each entry using a PassThrough stream
 | |||
|  |       // before sorting.
 | |||
|  |       // also: we can’t use the same context array for all entries
 | |||
|  |       // since the order of hashing should *not* matter. instead,
 | |||
|  |       // we keep track of the additions to a copy of the context array
 | |||
|  |       // and add all of them to the global context array when we’re done
 | |||
|  |       var contextAdditions = []; | |||
|  |       var entries = arr.map(function(entry) { | |||
|  |         var strm = new PassThrough(); | |||
|  |         var localContext = context.slice(); // make copy
 | |||
|  |         var hasher = typeHasher(options, strm, localContext); | |||
|  |         hasher.dispatch(entry); | |||
|  |         // take only what was added to localContext and append it to contextAdditions
 | |||
|  |         contextAdditions = contextAdditions.concat(localContext.slice(context.length)); | |||
|  |         return strm.read().toString(); | |||
|  |       }); | |||
|  |       context = context.concat(contextAdditions); | |||
|  |       entries.sort(); | |||
|  |       return this._array(entries, false); | |||
|  |     }, | |||
|  |     _date: function(date){ | |||
|  |       return write('date:' + date.toJSON()); | |||
|  |     }, | |||
|  |     _symbol: function(sym){ | |||
|  |       return write('symbol:' + sym.toString()); | |||
|  |     }, | |||
|  |     _error: function(err){ | |||
|  |       return write('error:' + err.toString()); | |||
|  |     }, | |||
|  |     _boolean: function(bool){ | |||
|  |       return write('bool:' + bool.toString()); | |||
|  |     }, | |||
|  |     _string: function(string){ | |||
|  |       write('string:' + string.length + ':'); | |||
|  |       write(string.toString()); | |||
|  |     }, | |||
|  |     _function: function(fn){ | |||
|  |       write('fn:'); | |||
|  |       if (isNativeFunction(fn)) { | |||
|  |         this.dispatch('[native]'); | |||
|  |       } else { | |||
|  |         this.dispatch(fn.toString()); | |||
|  |       } | |||
|  | 
 | |||
|  |       if (options.respectFunctionNames !== false) { | |||
|  |         // Make sure we can still distinguish native functions
 | |||
|  |         // by their name, otherwise String and Function will
 | |||
|  |         // have the same hash
 | |||
|  |         this.dispatch("function-name:" + String(fn.name)); | |||
|  |       } | |||
|  | 
 | |||
|  |       if (options.respectFunctionProperties) { | |||
|  |         this._object(fn); | |||
|  |       } | |||
|  |     }, | |||
|  |     _number: function(number){ | |||
|  |       return write('number:' + number.toString()); | |||
|  |     }, | |||
|  |     _xml: function(xml){ | |||
|  |       return write('xml:' + xml.toString()); | |||
|  |     }, | |||
|  |     _null: function() { | |||
|  |       return write('Null'); | |||
|  |     }, | |||
|  |     _undefined: function() { | |||
|  |       return write('Undefined'); | |||
|  |     }, | |||
|  |     _regexp: function(regex){ | |||
|  |       return write('regex:' + regex.toString()); | |||
|  |     }, | |||
|  |     _uint8array: function(arr){ | |||
|  |       write('uint8array:'); | |||
|  |       return this.dispatch(Array.prototype.slice.call(arr)); | |||
|  |     }, | |||
|  |     _uint8clampedarray: function(arr){ | |||
|  |       write('uint8clampedarray:'); | |||
|  |       return this.dispatch(Array.prototype.slice.call(arr)); | |||
|  |     }, | |||
|  |     _int8array: function(arr){ | |||
|  |       write('int8array:'); | |||
|  |       return this.dispatch(Array.prototype.slice.call(arr)); | |||
|  |     }, | |||
|  |     _uint16array: function(arr){ | |||
|  |       write('uint16array:'); | |||
|  |       return this.dispatch(Array.prototype.slice.call(arr)); | |||
|  |     }, | |||
|  |     _int16array: function(arr){ | |||
|  |       write('int16array:'); | |||
|  |       return this.dispatch(Array.prototype.slice.call(arr)); | |||
|  |     }, | |||
|  |     _uint32array: function(arr){ | |||
|  |       write('uint32array:'); | |||
|  |       return this.dispatch(Array.prototype.slice.call(arr)); | |||
|  |     }, | |||
|  |     _int32array: function(arr){ | |||
|  |       write('int32array:'); | |||
|  |       return this.dispatch(Array.prototype.slice.call(arr)); | |||
|  |     }, | |||
|  |     _float32array: function(arr){ | |||
|  |       write('float32array:'); | |||
|  |       return this.dispatch(Array.prototype.slice.call(arr)); | |||
|  |     }, | |||
|  |     _float64array: function(arr){ | |||
|  |       write('float64array:'); | |||
|  |       return this.dispatch(Array.prototype.slice.call(arr)); | |||
|  |     }, | |||
|  |     _arraybuffer: function(arr){ | |||
|  |       write('arraybuffer:'); | |||
|  |       return this.dispatch(new Uint8Array(arr)); | |||
|  |     }, | |||
|  |     _url: function(url) { | |||
|  |       return write('url:' + url.toString(), 'utf8'); | |||
|  |     }, | |||
|  |     _map: function(map) { | |||
|  |       write('map:'); | |||
|  |       var arr = Array.from(map); | |||
|  |       return this._array(arr, options.unorderedSets !== false); | |||
|  |     }, | |||
|  |     _set: function(set) { | |||
|  |       write('set:'); | |||
|  |       var arr = Array.from(set); | |||
|  |       return this._array(arr, options.unorderedSets !== false); | |||
|  |     }, | |||
|  |     _file: function(file) { | |||
|  |       write('file:'); | |||
|  |       return this.dispatch([file.name, file.size, file.type, file.lastModfied]); | |||
|  |     }, | |||
|  |     _blob: function() { | |||
|  |       if (options.ignoreUnknown) { | |||
|  |         return write('[blob]'); | |||
|  |       } | |||
|  | 
 | |||
|  |       throw Error('Hashing Blob objects is currently not supported\n' + | |||
|  |         '(see https://github.com/puleos/object-hash/issues/26)\n' + | |||
|  |         'Use "options.replacer" or "options.ignoreUnknown"\n'); | |||
|  |     }, | |||
|  |     _domwindow: function() { return write('domwindow'); }, | |||
|  |     _bigint: function(number){ | |||
|  |       return write('bigint:' + number.toString()); | |||
|  |     }, | |||
|  |     /* Node.js standard native objects */ | |||
|  |     _process: function() { return write('process'); }, | |||
|  |     _timer: function() { return write('timer'); }, | |||
|  |     _pipe: function() { return write('pipe'); }, | |||
|  |     _tcp: function() { return write('tcp'); }, | |||
|  |     _udp: function() { return write('udp'); }, | |||
|  |     _tty: function() { return write('tty'); }, | |||
|  |     _statwatcher: function() { return write('statwatcher'); }, | |||
|  |     _securecontext: function() { return write('securecontext'); }, | |||
|  |     _connection: function() { return write('connection'); }, | |||
|  |     _zlib: function() { return write('zlib'); }, | |||
|  |     _context: function() { return write('context'); }, | |||
|  |     _nodescript: function() { return write('nodescript'); }, | |||
|  |     _httpparser: function() { return write('httpparser'); }, | |||
|  |     _dataview: function() { return write('dataview'); }, | |||
|  |     _signal: function() { return write('signal'); }, | |||
|  |     _fsevent: function() { return write('fsevent'); }, | |||
|  |     _tlswrap: function() { return write('tlswrap'); }, | |||
|  |   }; | |||
|  | } | |||
|  | 
 | |||
|  | // Mini-implementation of stream.PassThrough
 | |||
|  | // We are far from having need for the full implementation, and we can
 | |||
|  | // make assumptions like "many writes, then only one final read"
 | |||
|  | // and we can ignore encoding specifics
 | |||
|  | function PassThrough() { | |||
|  |   return { | |||
|  |     buf: '', | |||
|  | 
 | |||
|  |     write: function(b) { | |||
|  |       this.buf += b; | |||
|  |     }, | |||
|  | 
 | |||
|  |     end: function(b) { | |||
|  |       this.buf += b; | |||
|  |     }, | |||
|  | 
 | |||
|  |     read: function() { | |||
|  |       return this.buf; | |||
|  |     } | |||
|  |   }; | |||
|  | } |