123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236 |
- '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);
- };
|