UNPKG

2.68 kBJavaScriptView Raw
1var checkParameters = require('./precondition')
2var defaultEncoding = require('./default-encoding')
3var sync = require('./sync')
4var Buffer = require('safe-buffer').Buffer
5
6var ZERO_BUF
7var subtle = global.crypto && global.crypto.subtle
8var toBrowser = {
9 'sha': 'SHA-1',
10 'sha-1': 'SHA-1',
11 'sha1': 'SHA-1',
12 'sha256': 'SHA-256',
13 'sha-256': 'SHA-256',
14 'sha384': 'SHA-384',
15 'sha-384': 'SHA-384',
16 'sha-512': 'SHA-512',
17 'sha512': 'SHA-512'
18}
19var checks = []
20function checkNative (algo) {
21 if (global.process && !global.process.browser) {
22 return Promise.resolve(false)
23 }
24 if (!subtle || !subtle.importKey || !subtle.deriveBits) {
25 return Promise.resolve(false)
26 }
27 if (checks[algo] !== undefined) {
28 return checks[algo]
29 }
30 ZERO_BUF = ZERO_BUF || Buffer.alloc(8)
31 var prom = browserPbkdf2(ZERO_BUF, ZERO_BUF, 10, 128, algo)
32 .then(function () {
33 return true
34 }).catch(function () {
35 return false
36 })
37 checks[algo] = prom
38 return prom
39}
40function browserPbkdf2 (password, salt, iterations, length, algo) {
41 return subtle.importKey(
42 'raw', password, {name: 'PBKDF2'}, false, ['deriveBits']
43 ).then(function (key) {
44 return subtle.deriveBits({
45 name: 'PBKDF2',
46 salt: salt,
47 iterations: iterations,
48 hash: {
49 name: algo
50 }
51 }, key, length << 3)
52 }).then(function (res) {
53 return Buffer.from(res)
54 })
55}
56function resolvePromise (promise, callback) {
57 promise.then(function (out) {
58 process.nextTick(function () {
59 callback(null, out)
60 })
61 }, function (e) {
62 process.nextTick(function () {
63 callback(e)
64 })
65 })
66}
67module.exports = function (password, salt, iterations, keylen, digest, callback) {
68 if (!Buffer.isBuffer(password)) password = Buffer.from(password, defaultEncoding)
69 if (!Buffer.isBuffer(salt)) salt = Buffer.from(salt, defaultEncoding)
70
71 checkParameters(iterations, keylen)
72 if (typeof digest === 'function') {
73 callback = digest
74 digest = undefined
75 }
76 if (typeof callback !== 'function') throw new Error('No callback provided to pbkdf2')
77
78 digest = digest || 'sha1'
79 var algo = toBrowser[digest.toLowerCase()]
80 if (!algo || typeof global.Promise !== 'function') {
81 return process.nextTick(function () {
82 var out
83 try {
84 out = sync(password, salt, iterations, keylen, digest)
85 } catch (e) {
86 return callback(e)
87 }
88 callback(null, out)
89 })
90 }
91 resolvePromise(checkNative(algo).then(function (resp) {
92 if (resp) {
93 return browserPbkdf2(password, salt, iterations, keylen, algo)
94 } else {
95 return sync(password, salt, iterations, keylen, digest)
96 }
97 }), callback)
98}