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.
		
		
		
		
		
			
		
			
				
					237 lines
				
				7.1 KiB
			
		
		
			
		
	
	
					237 lines
				
				7.1 KiB
			| 
								 
											3 years ago
										 
									 | 
							
								'use strict';
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								var nodePreGyp = require('@mapbox/node-pre-gyp');
							 | 
						||
| 
								 | 
							
								var path = require('path');
							 | 
						||
| 
								 | 
							
								var binding_path = nodePreGyp.find(path.resolve(path.join(__dirname, './package.json')));
							 | 
						||
| 
								 | 
							
								var bindings = require(binding_path);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								var crypto = require('crypto');
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								var promises = require('./promises');
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/// generate a salt (sync)
							 | 
						||
| 
								 | 
							
								/// @param {Number} [rounds] number of rounds (default 10)
							 | 
						||
| 
								 | 
							
								/// @return {String} salt
							 | 
						||
| 
								 | 
							
								module.exports.genSaltSync = function genSaltSync(rounds, minor) {
							 | 
						||
| 
								 | 
							
								    // default 10 rounds
							 | 
						||
| 
								 | 
							
								    if (!rounds) {
							 | 
						||
| 
								 | 
							
								        rounds = 10;
							 | 
						||
| 
								 | 
							
								    } else if (typeof rounds !== 'number') {
							 | 
						||
| 
								 | 
							
								        throw new Error('rounds must be a number');
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if(!minor) {
							 | 
						||
| 
								 | 
							
								        minor = 'b';
							 | 
						||
| 
								 | 
							
								    } else if(minor !== 'b' && minor !== 'a') {
							 | 
						||
| 
								 | 
							
								        throw new Error('minor must be either "a" or "b"');
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    return bindings.gen_salt_sync(minor, rounds, crypto.randomBytes(16));
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/// generate a salt
							 | 
						||
| 
								 | 
							
								/// @param {Number} [rounds] number of rounds (default 10)
							 | 
						||
| 
								 | 
							
								/// @param {Function} cb callback(err, salt)
							 | 
						||
| 
								 | 
							
								module.exports.genSalt = function genSalt(rounds, minor, cb) {
							 | 
						||
| 
								 | 
							
								    var error;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // if callback is first argument, then use defaults for others
							 | 
						||
| 
								 | 
							
								    if (typeof arguments[0] === 'function') {
							 | 
						||
| 
								 | 
							
								        // have to set callback first otherwise arguments are overriden
							 | 
						||
| 
								 | 
							
								        cb = arguments[0];
							 | 
						||
| 
								 | 
							
								        rounds = 10;
							 | 
						||
| 
								 | 
							
								        minor = 'b';
							 | 
						||
| 
								 | 
							
								    // callback is second argument
							 | 
						||
| 
								 | 
							
								    } else if (typeof arguments[1] === 'function') {
							 | 
						||
| 
								 | 
							
								        // have to set callback first otherwise arguments are overriden
							 | 
						||
| 
								 | 
							
								        cb = arguments[1];
							 | 
						||
| 
								 | 
							
								        minor = 'b';
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (!cb) {
							 | 
						||
| 
								 | 
							
								        return promises.promise(genSalt, this, [rounds, minor]);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // default 10 rounds
							 | 
						||
| 
								 | 
							
								    if (!rounds) {
							 | 
						||
| 
								 | 
							
								        rounds = 10;
							 | 
						||
| 
								 | 
							
								    } else if (typeof rounds !== 'number') {
							 | 
						||
| 
								 | 
							
								        // callback error asynchronously
							 | 
						||
| 
								 | 
							
								        error = new Error('rounds must be a number');
							 | 
						||
| 
								 | 
							
								        return process.nextTick(function() {
							 | 
						||
| 
								 | 
							
								            cb(error);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if(!minor) {
							 | 
						||
| 
								 | 
							
								        minor = 'b'
							 | 
						||
| 
								 | 
							
								    } else if(minor !== 'b' && minor !== 'a') {
							 | 
						||
| 
								 | 
							
								        error = new Error('minor must be either "a" or "b"');
							 | 
						||
| 
								 | 
							
								        return process.nextTick(function() {
							 | 
						||
| 
								 | 
							
								            cb(error);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    crypto.randomBytes(16, function(error, randomBytes) {
							 | 
						||
| 
								 | 
							
								        if (error) {
							 | 
						||
| 
								 | 
							
								            cb(error);
							 | 
						||
| 
								 | 
							
								            return;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        bindings.gen_salt(minor, rounds, randomBytes, cb);
							 | 
						||
| 
								 | 
							
								    });
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/// hash data using a salt
							 | 
						||
| 
								 | 
							
								/// @param {String|Buffer} data the data to encrypt
							 | 
						||
| 
								 | 
							
								/// @param {String} salt the salt to use when hashing
							 | 
						||
| 
								 | 
							
								/// @return {String} hash
							 | 
						||
| 
								 | 
							
								module.exports.hashSync = function hashSync(data, salt) {
							 | 
						||
| 
								 | 
							
								    if (data == null || salt == null) {
							 | 
						||
| 
								 | 
							
								        throw new Error('data and salt arguments required');
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (!(typeof data === 'string' || data instanceof Buffer) || (typeof salt !== 'string' && typeof salt !== 'number')) {
							 | 
						||
| 
								 | 
							
								        throw new Error('data must be a string or Buffer and salt must either be a salt string or a number of rounds');
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (typeof salt === 'number') {
							 | 
						||
| 
								 | 
							
								        salt = module.exports.genSaltSync(salt);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    return bindings.encrypt_sync(data, salt);
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/// hash data using a salt
							 | 
						||
| 
								 | 
							
								/// @param {String|Buffer} data the data to encrypt
							 | 
						||
| 
								 | 
							
								/// @param {String} salt the salt to use when hashing
							 | 
						||
| 
								 | 
							
								/// @param {Function} cb callback(err, hash)
							 | 
						||
| 
								 | 
							
								module.exports.hash = function hash(data, salt, cb) {
							 | 
						||
| 
								 | 
							
								    var error;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (typeof data === 'function') {
							 | 
						||
| 
								 | 
							
								        error = new Error('data must be a string or Buffer and salt must either be a salt string or a number of rounds');
							 | 
						||
| 
								 | 
							
								        return process.nextTick(function() {
							 | 
						||
| 
								 | 
							
								            data(error);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (typeof salt === 'function') {
							 | 
						||
| 
								 | 
							
								        error = new Error('data must be a string or Buffer and salt must either be a salt string or a number of rounds');
							 | 
						||
| 
								 | 
							
								        return process.nextTick(function() {
							 | 
						||
| 
								 | 
							
								            salt(error);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // cb exists but is not a function
							 | 
						||
| 
								 | 
							
								    // return a rejecting promise
							 | 
						||
| 
								 | 
							
								    if (cb && typeof cb !== 'function') {
							 | 
						||
| 
								 | 
							
								        return promises.reject(new Error('cb must be a function or null to return a Promise'));
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (!cb) {
							 | 
						||
| 
								 | 
							
								        return promises.promise(hash, this, [data, salt]);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (data == null || salt == null) {
							 | 
						||
| 
								 | 
							
								        error = new Error('data and salt arguments required');
							 | 
						||
| 
								 | 
							
								        return process.nextTick(function() {
							 | 
						||
| 
								 | 
							
								            cb(error);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (!(typeof data === 'string' || data instanceof Buffer) || (typeof salt !== 'string' && typeof salt !== 'number')) {
							 | 
						||
| 
								 | 
							
								        error = new Error('data must be a string or Buffer and salt must either be a salt string or a number of rounds');
							 | 
						||
| 
								 | 
							
								        return process.nextTick(function() {
							 | 
						||
| 
								 | 
							
								            cb(error);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (typeof salt === 'number') {
							 | 
						||
| 
								 | 
							
								        return module.exports.genSalt(salt, function(err, salt) {
							 | 
						||
| 
								 | 
							
								            return bindings.encrypt(data, salt, cb);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    return bindings.encrypt(data, salt, cb);
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/// compare raw data to hash
							 | 
						||
| 
								 | 
							
								/// @param {String|Buffer} data the data to hash and compare
							 | 
						||
| 
								 | 
							
								/// @param {String} hash expected hash
							 | 
						||
| 
								 | 
							
								/// @return {bool} true if hashed data matches hash
							 | 
						||
| 
								 | 
							
								module.exports.compareSync = function compareSync(data, hash) {
							 | 
						||
| 
								 | 
							
								    if (data == null || hash == null) {
							 | 
						||
| 
								 | 
							
								        throw new Error('data and hash arguments required');
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (!(typeof data === 'string' || data instanceof Buffer) || typeof hash !== 'string') {
							 | 
						||
| 
								 | 
							
								        throw new Error('data must be a string or Buffer and hash must be a string');
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    return bindings.compare_sync(data, hash);
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/// compare raw data to hash
							 | 
						||
| 
								 | 
							
								/// @param {String|Buffer} data the data to hash and compare
							 | 
						||
| 
								 | 
							
								/// @param {String} hash expected hash
							 | 
						||
| 
								 | 
							
								/// @param {Function} cb callback(err, matched) - matched is true if hashed data matches hash
							 | 
						||
| 
								 | 
							
								module.exports.compare = function compare(data, hash, cb) {
							 | 
						||
| 
								 | 
							
								    var error;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (typeof data === 'function') {
							 | 
						||
| 
								 | 
							
								        error = new Error('data and hash arguments required');
							 | 
						||
| 
								 | 
							
								        return process.nextTick(function() {
							 | 
						||
| 
								 | 
							
								            data(error);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (typeof hash === 'function') {
							 | 
						||
| 
								 | 
							
								        error = new Error('data and hash arguments required');
							 | 
						||
| 
								 | 
							
								        return process.nextTick(function() {
							 | 
						||
| 
								 | 
							
								            hash(error);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    // cb exists but is not a function
							 | 
						||
| 
								 | 
							
								    // return a rejecting promise
							 | 
						||
| 
								 | 
							
								    if (cb && typeof cb !== 'function') {
							 | 
						||
| 
								 | 
							
								        return promises.reject(new Error('cb must be a function or null to return a Promise'));
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (!cb) {
							 | 
						||
| 
								 | 
							
								        return promises.promise(compare, this, [data, hash]);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (data == null || hash == null) {
							 | 
						||
| 
								 | 
							
								        error = new Error('data and hash arguments required');
							 | 
						||
| 
								 | 
							
								        return process.nextTick(function() {
							 | 
						||
| 
								 | 
							
								            cb(error);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (!(typeof data === 'string' || data instanceof Buffer) || typeof hash !== 'string') {
							 | 
						||
| 
								 | 
							
								        error = new Error('data and hash must be strings');
							 | 
						||
| 
								 | 
							
								        return process.nextTick(function() {
							 | 
						||
| 
								 | 
							
								            cb(error);
							 | 
						||
| 
								 | 
							
								        });
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    return bindings.compare(data, hash, cb);
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/// @param {String} hash extract rounds from this hash
							 | 
						||
| 
								 | 
							
								/// @return {Number} the number of rounds used to encrypt a given hash
							 | 
						||
| 
								 | 
							
								module.exports.getRounds = function getRounds(hash) {
							 | 
						||
| 
								 | 
							
								    if (hash == null) {
							 | 
						||
| 
								 | 
							
								        throw new Error('hash argument required');
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    if (typeof hash !== 'string') {
							 | 
						||
| 
								 | 
							
								        throw new Error('hash must be a string');
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    return bindings.get_rounds(hash);
							 | 
						||
| 
								 | 
							
								};
							 |