12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788 |
- (function(){
- var C = (typeof window === 'undefined') ? require('./Crypto').Crypto : window.Crypto;
- // Shortcuts
- var util = C.util,
- charenc = C.charenc,
- UTF8 = charenc.UTF8,
- Binary = charenc.Binary;
- if (!C.nextTick) {
- // node.js has setTime out but prefer process.nextTick
- if (typeof process != 'undefined' && typeof process.nextTick !== 'undefined') {
- C.nextTick = process.nextTick;
- } else if (typeof setTimeout !== 'undefined') {
- C.nextTick = function (callback) {
- setTimeout(callback, 0);
- };
- }
- }
- C.PBKDF2Async = function (password, salt, keylen, callback, options) {
- // Convert to byte arrays
- if (password.constructor == String) password = UTF8.stringToBytes(password);
- if (salt.constructor == String) salt = UTF8.stringToBytes(salt);
- /* else, assume byte arrays already */
- // Defaults
- var hasher = options && options.hasher || C.SHA1,
- iterations = options && options.iterations || 1;
- // Progress callback option
- var progressChangeHandler = options && options.onProgressChange;
- var totalIterations = Math.ceil(keylen / hasher._digestsize) * iterations;
- function fireProgressChange(currentIteration) {
- if (progressChangeHandler) {
- var iterationsSoFar = derivedKeyBytes.length / hasher._digestsize * iterations + currentIteration;
- setTimeout(function () {
- progressChangeHandler(Math.round(iterationsSoFar / totalIterations * 100));
- }, 0);
- }
- }
- // Pseudo-random function
- function PRF(password, salt) {
- return C.HMAC(hasher, salt, password, { asBytes: true });
- }
- var nextTick = C.nextTick;
- // Generate key
- var derivedKeyBytes = [],
- blockindex = 1;
- var outer, inner;
- nextTick(outer = function () {
- if (derivedKeyBytes.length < keylen) {
- var block = PRF(password, salt.concat(util.wordsToBytes([blockindex])));
- fireProgressChange(1);
- var u = block, i = 1;
- nextTick(inner = function () {
- if (i < iterations) {
- u = PRF(password, u);
- for (var j = 0; j < block.length; j++) block[j] ^= u[j];
- i++;
- fireProgressChange(i);
- nextTick(inner);
- } else {
- derivedKeyBytes = derivedKeyBytes.concat(block);
- blockindex++;
- nextTick(outer);
- }
- });
- } else {
- // Truncate excess bytes
- derivedKeyBytes.length = keylen;
- callback(
- options && options.asBytes ? derivedKeyBytes :
- options && options.asString ? Binary.bytesToString(derivedKeyBytes) :
- util.bytesToHex(derivedKeyBytes));
- }
- });
- };
- })();
|