async.js 2.6 KB

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