UNPKG

1.58 kBJavaScriptView Raw
1'use strict'
2var inherits = require('inherits')
3var Legacy = require('./legacy')
4var Base = require('cipher-base')
5var Buffer = require('safe-buffer').Buffer
6var md5 = require('create-hash/md5')
7var RIPEMD160 = require('ripemd160')
8
9var sha = require('sha.js')
10
11var ZEROS = Buffer.alloc(128)
12
13function Hmac (alg, key) {
14 Base.call(this, 'digest')
15 if (typeof key === 'string') {
16 key = Buffer.from(key)
17 }
18
19 var blocksize = (alg === 'sha512' || alg === 'sha384') ? 128 : 64
20
21 this._alg = alg
22 this._key = key
23 if (key.length > blocksize) {
24 var hash = alg === 'rmd160' ? new RIPEMD160() : sha(alg)
25 key = hash.update(key).digest()
26 } else if (key.length < blocksize) {
27 key = Buffer.concat([key, ZEROS], blocksize)
28 }
29
30 var ipad = this._ipad = Buffer.allocUnsafe(blocksize)
31 var opad = this._opad = Buffer.allocUnsafe(blocksize)
32
33 for (var i = 0; i < blocksize; i++) {
34 ipad[i] = key[i] ^ 0x36
35 opad[i] = key[i] ^ 0x5C
36 }
37 this._hash = alg === 'rmd160' ? new RIPEMD160() : sha(alg)
38 this._hash.update(ipad)
39}
40
41inherits(Hmac, Base)
42
43Hmac.prototype._update = function (data) {
44 this._hash.update(data)
45}
46
47Hmac.prototype._final = function () {
48 var h = this._hash.digest()
49 var hash = this._alg === 'rmd160' ? new RIPEMD160() : sha(this._alg)
50 return hash.update(this._opad).update(h).digest()
51}
52
53module.exports = function createHmac (alg, key) {
54 alg = alg.toLowerCase()
55 if (alg === 'rmd160' || alg === 'ripemd160') {
56 return new Hmac('rmd160', key)
57 }
58 if (alg === 'md5') {
59 return new Legacy(md5, key)
60 }
61 return new Hmac(alg, key)
62}