| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122 |
- 'use strict';
- var Buffer = require('safe-buffer').Buffer;
- var checkParameters = require('./precondition');
- var defaultEncoding = require('./default-encoding');
- var sync = require('./sync');
- var toBuffer = require('./to-buffer');
- var ZERO_BUF;
- var subtle = global.crypto && global.crypto.subtle;
- var toBrowser = {
- sha: 'SHA-1',
- 'sha-1': 'SHA-1',
- sha1: 'SHA-1',
- sha256: 'SHA-256',
- 'sha-256': 'SHA-256',
- sha384: 'SHA-384',
- 'sha-384': 'SHA-384',
- 'sha-512': 'SHA-512',
- sha512: 'SHA-512'
- };
- var checks = [];
- var nextTick;
- function getNextTick() {
- if (nextTick) {
- return nextTick;
- }
- if (global.process && global.process.nextTick) {
- nextTick = global.process.nextTick;
- } else if (global.queueMicrotask) {
- nextTick = global.queueMicrotask;
- } else if (global.setImmediate) {
- nextTick = global.setImmediate;
- } else {
- nextTick = global.setTimeout;
- }
- return nextTick;
- }
- function browserPbkdf2(password, salt, iterations, length, algo) {
- return subtle.importKey('raw', password, { name: 'PBKDF2' }, false, ['deriveBits']).then(function (key) {
- return subtle.deriveBits({
- name: 'PBKDF2',
- salt: salt,
- iterations: iterations,
- hash: {
- name: algo
- }
- }, key, length << 3);
- }).then(function (res) {
- return Buffer.from(res);
- });
- }
- function checkNative(algo) {
- if (global.process && !global.process.browser) {
- return Promise.resolve(false);
- }
- if (!subtle || !subtle.importKey || !subtle.deriveBits) {
- return Promise.resolve(false);
- }
- if (checks[algo] !== undefined) {
- return checks[algo];
- }
- ZERO_BUF = ZERO_BUF || Buffer.alloc(8);
- var prom = browserPbkdf2(ZERO_BUF, ZERO_BUF, 10, 128, algo)
- .then(
- function () { return true; },
- function () { return false; }
- );
- checks[algo] = prom;
- return prom;
- }
- function resolvePromise(promise, callback) {
- promise.then(function (out) {
- getNextTick()(function () {
- callback(null, out);
- });
- }, function (e) {
- getNextTick()(function () {
- callback(e);
- });
- });
- }
- module.exports = function (password, salt, iterations, keylen, digest, callback) {
- if (typeof digest === 'function') {
- callback = digest;
- digest = undefined;
- }
- checkParameters(iterations, keylen);
- password = toBuffer(password, defaultEncoding, 'Password');
- salt = toBuffer(salt, defaultEncoding, 'Salt');
- if (typeof callback !== 'function') {
- throw new Error('No callback provided to pbkdf2');
- }
- digest = digest || 'sha1';
- var algo = toBrowser[digest.toLowerCase()];
- if (!algo || typeof global.Promise !== 'function') {
- getNextTick()(function () {
- var out;
- try {
- out = sync(password, salt, iterations, keylen, digest);
- } catch (e) {
- callback(e);
- return;
- }
- callback(null, out);
- });
- return;
- }
- resolvePromise(checkNative(algo).then(function (resp) {
- if (resp) {
- return browserPbkdf2(password, salt, iterations, keylen, algo);
- }
- return sync(password, salt, iterations, keylen, digest);
- }), callback);
- };
|