mirror of
https://github.com/nodejs/node.git
synced 2025-05-03 02:06:12 +00:00

PR-URL: https://github.com/nodejs/node/pull/52070 Refs: https://github.com/nodejs/node/pull/51044/files#r1522362983 Reviewed-By: Vinícius Lourenço Claro Cardoso <contact@viniciusl.com.br> Reviewed-By: Luigi Pinca <luigipinca@gmail.com>
226 lines
5.8 KiB
JavaScript
226 lines
5.8 KiB
JavaScript
'use strict';
|
|
|
|
const {
|
|
ObjectSetPrototypeOf,
|
|
ReflectApply,
|
|
StringPrototypeToLowerCase,
|
|
Symbol,
|
|
} = primordials;
|
|
|
|
const {
|
|
Hash: _Hash,
|
|
HashJob,
|
|
Hmac: _Hmac,
|
|
kCryptoJobAsync,
|
|
oneShotDigest,
|
|
} = internalBinding('crypto');
|
|
|
|
const {
|
|
getStringOption,
|
|
jobPromise,
|
|
normalizeHashName,
|
|
validateMaxBufferLength,
|
|
kHandle,
|
|
getCachedHashId,
|
|
getHashCache,
|
|
} = require('internal/crypto/util');
|
|
|
|
const {
|
|
prepareSecretKey,
|
|
} = require('internal/crypto/keys');
|
|
|
|
const {
|
|
lazyDOMException,
|
|
normalizeEncoding,
|
|
encodingsMap,
|
|
} = require('internal/util');
|
|
|
|
const {
|
|
Buffer,
|
|
} = require('buffer');
|
|
|
|
const {
|
|
codes: {
|
|
ERR_CRYPTO_HASH_FINALIZED,
|
|
ERR_CRYPTO_HASH_UPDATE_FAILED,
|
|
ERR_INVALID_ARG_TYPE,
|
|
ERR_INVALID_ARG_VALUE,
|
|
},
|
|
} = require('internal/errors');
|
|
|
|
const {
|
|
validateEncoding,
|
|
validateString,
|
|
validateUint32,
|
|
} = require('internal/validators');
|
|
|
|
const {
|
|
isArrayBufferView,
|
|
} = require('internal/util/types');
|
|
|
|
const LazyTransform = require('internal/streams/lazy_transform');
|
|
|
|
const kState = Symbol('kState');
|
|
const kFinalized = Symbol('kFinalized');
|
|
|
|
function Hash(algorithm, options) {
|
|
if (!new.target)
|
|
return new Hash(algorithm, options);
|
|
const isCopy = algorithm instanceof _Hash;
|
|
if (!isCopy)
|
|
validateString(algorithm, 'algorithm');
|
|
const xofLen = typeof options === 'object' && options !== null ?
|
|
options.outputLength : undefined;
|
|
if (xofLen !== undefined)
|
|
validateUint32(xofLen, 'options.outputLength');
|
|
// Lookup the cached ID from JS land because it's faster than decoding
|
|
// the string in C++ land.
|
|
const algorithmId = isCopy ? -1 : getCachedHashId(algorithm);
|
|
this[kHandle] = new _Hash(algorithm, xofLen, algorithmId, getHashCache());
|
|
this[kState] = {
|
|
[kFinalized]: false,
|
|
};
|
|
ReflectApply(LazyTransform, this, [options]);
|
|
}
|
|
|
|
ObjectSetPrototypeOf(Hash.prototype, LazyTransform.prototype);
|
|
ObjectSetPrototypeOf(Hash, LazyTransform);
|
|
|
|
Hash.prototype.copy = function copy(options) {
|
|
const state = this[kState];
|
|
if (state[kFinalized])
|
|
throw new ERR_CRYPTO_HASH_FINALIZED();
|
|
|
|
return new Hash(this[kHandle], options);
|
|
};
|
|
|
|
Hash.prototype._transform = function _transform(chunk, encoding, callback) {
|
|
this[kHandle].update(chunk, encoding);
|
|
callback();
|
|
};
|
|
|
|
Hash.prototype._flush = function _flush(callback) {
|
|
this.push(this[kHandle].digest());
|
|
callback();
|
|
};
|
|
|
|
Hash.prototype.update = function update(data, encoding) {
|
|
const state = this[kState];
|
|
if (state[kFinalized])
|
|
throw new ERR_CRYPTO_HASH_FINALIZED();
|
|
|
|
if (typeof data === 'string') {
|
|
validateEncoding(data, encoding);
|
|
} else if (!isArrayBufferView(data)) {
|
|
throw new ERR_INVALID_ARG_TYPE(
|
|
'data', ['string', 'Buffer', 'TypedArray', 'DataView'], data);
|
|
}
|
|
|
|
if (!this[kHandle].update(data, encoding))
|
|
throw new ERR_CRYPTO_HASH_UPDATE_FAILED();
|
|
return this;
|
|
};
|
|
|
|
|
|
Hash.prototype.digest = function digest(outputEncoding) {
|
|
const state = this[kState];
|
|
if (state[kFinalized])
|
|
throw new ERR_CRYPTO_HASH_FINALIZED();
|
|
|
|
// Explicit conversion of truthy values for backward compatibility.
|
|
const ret = this[kHandle].digest(outputEncoding && `${outputEncoding}`);
|
|
state[kFinalized] = true;
|
|
return ret;
|
|
};
|
|
|
|
function Hmac(hmac, key, options) {
|
|
if (!(this instanceof Hmac))
|
|
return new Hmac(hmac, key, options);
|
|
validateString(hmac, 'hmac');
|
|
const encoding = getStringOption(options, 'encoding');
|
|
key = prepareSecretKey(key, encoding);
|
|
this[kHandle] = new _Hmac();
|
|
this[kHandle].init(hmac, key);
|
|
this[kState] = {
|
|
[kFinalized]: false,
|
|
};
|
|
ReflectApply(LazyTransform, this, [options]);
|
|
}
|
|
|
|
ObjectSetPrototypeOf(Hmac.prototype, LazyTransform.prototype);
|
|
ObjectSetPrototypeOf(Hmac, LazyTransform);
|
|
|
|
Hmac.prototype.update = Hash.prototype.update;
|
|
|
|
Hmac.prototype.digest = function digest(outputEncoding) {
|
|
const state = this[kState];
|
|
|
|
if (state[kFinalized]) {
|
|
const buf = Buffer.from('');
|
|
if (outputEncoding && outputEncoding !== 'buffer')
|
|
return buf.toString(outputEncoding);
|
|
return buf;
|
|
}
|
|
|
|
// Explicit conversion of truthy values for backward compatibility.
|
|
const ret = this[kHandle].digest(outputEncoding && `${outputEncoding}`);
|
|
state[kFinalized] = true;
|
|
return ret;
|
|
};
|
|
|
|
Hmac.prototype._flush = Hash.prototype._flush;
|
|
Hmac.prototype._transform = Hash.prototype._transform;
|
|
|
|
// Implementation for WebCrypto subtle.digest()
|
|
|
|
async function asyncDigest(algorithm, data) {
|
|
validateMaxBufferLength(data, 'data');
|
|
|
|
switch (algorithm.name) {
|
|
case 'SHA-1':
|
|
// Fall through
|
|
case 'SHA-256':
|
|
// Fall through
|
|
case 'SHA-384':
|
|
// Fall through
|
|
case 'SHA-512':
|
|
return jobPromise(() => new HashJob(
|
|
kCryptoJobAsync,
|
|
normalizeHashName(algorithm.name),
|
|
data));
|
|
}
|
|
|
|
throw lazyDOMException('Unrecognized algorithm name', 'NotSupportedError');
|
|
}
|
|
|
|
function hash(algorithm, input, outputEncoding = 'hex') {
|
|
validateString(algorithm, 'algorithm');
|
|
if (typeof input !== 'string' && !isArrayBufferView(input)) {
|
|
throw new ERR_INVALID_ARG_TYPE('input', ['Buffer', 'TypedArray', 'DataView', 'string'], input);
|
|
}
|
|
let normalized = outputEncoding;
|
|
// Fast case: if it's 'hex', we don't need to validate it further.
|
|
if (outputEncoding !== 'hex') {
|
|
validateString(outputEncoding, 'outputEncoding');
|
|
normalized = normalizeEncoding(outputEncoding);
|
|
// If the encoding is invalid, normalizeEncoding() returns undefined.
|
|
if (normalized === undefined) {
|
|
// normalizeEncoding() doesn't handle 'buffer'.
|
|
if (StringPrototypeToLowerCase(outputEncoding) === 'buffer') {
|
|
normalized = 'buffer';
|
|
} else {
|
|
throw new ERR_INVALID_ARG_VALUE('outputEncoding', outputEncoding);
|
|
}
|
|
}
|
|
}
|
|
return oneShotDigest(algorithm, getCachedHashId(algorithm), getHashCache(),
|
|
input, normalized, encodingsMap[normalized]);
|
|
}
|
|
|
|
module.exports = {
|
|
Hash,
|
|
Hmac,
|
|
asyncDigest,
|
|
hash,
|
|
};
|