| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775 |
- /* eslint-disable no-use-before-define */
- /**
- * Base class for inheritance.
- */
- export class Base {
- /**
- * Extends this object and runs the init method.
- * Arguments to create() will be passed to init().
- *
- * @return {Object} The new object.
- *
- * @static
- *
- * @example
- *
- * var instance = MyType.create();
- */
- static create(...args) {
- return new this(...args);
- }
- /**
- * Copies properties into this object.
- *
- * @param {Object} properties The properties to mix in.
- *
- * @example
- *
- * MyType.mixIn({
- * field: 'value'
- * });
- */
- mixIn(properties) {
- return Object.assign(this, properties);
- }
- /**
- * Creates a copy of this object.
- *
- * @return {Object} The clone.
- *
- * @example
- *
- * var clone = instance.clone();
- */
- clone() {
- const clone = new this.constructor();
- Object.assign(clone, this);
- return clone;
- }
- }
- /**
- * An array of 32-bit words.
- *
- * @property {Array} words The array of 32-bit words.
- * @property {number} sigBytes The number of significant bytes in this word array.
- */
- export class WordArray extends Base {
- /**
- * Initializes a newly created word array.
- *
- * @param {Array} words (Optional) An array of 32-bit words.
- * @param {number} sigBytes (Optional) The number of significant bytes in the words.
- *
- * @example
- *
- * var wordArray = CryptoJS.lib.WordArray.create();
- * var wordArray = CryptoJS.lib.WordArray.create([0x00010203, 0x04050607]);
- * var wordArray = CryptoJS.lib.WordArray.create([0x00010203, 0x04050607], 6);
- */
- constructor(words = [], sigBytes = words.length * 4) {
- super();
- let typedArray = words;
- // Convert buffers to uint8
- if (typedArray instanceof ArrayBuffer) {
- typedArray = new Uint8Array(typedArray);
- }
- // Convert other array views to uint8
- if (
- typedArray instanceof Int8Array
- || typedArray instanceof Uint8ClampedArray
- || typedArray instanceof Int16Array
- || typedArray instanceof Uint16Array
- || typedArray instanceof Int32Array
- || typedArray instanceof Uint32Array
- || typedArray instanceof Float32Array
- || typedArray instanceof Float64Array
- ) {
- typedArray = new Uint8Array(typedArray.buffer, typedArray.byteOffset, typedArray.byteLength);
- }
- // Handle Uint8Array
- if (typedArray instanceof Uint8Array) {
- // Shortcut
- const typedArrayByteLength = typedArray.byteLength;
- // Extract bytes
- const _words = [];
- for (let i = 0; i < typedArrayByteLength; i += 1) {
- _words[i >>> 2] |= typedArray[i] << (24 - (i % 4) * 8);
- }
- // Initialize this word array
- this.words = _words;
- this.sigBytes = typedArrayByteLength;
- } else {
- // Else call normal init
- this.words = words;
- this.sigBytes = sigBytes;
- }
- }
- /**
- * Creates a word array filled with random bytes.
- *
- * @param {number} nBytes The number of random bytes to generate.
- *
- * @return {WordArray} The random word array.
- *
- * @static
- *
- * @example
- *
- * var wordArray = CryptoJS.lib.WordArray.random(16);
- */
- static random(nBytes) {
- const words = [];
- const r = (m_w) => {
- let _m_w = m_w;
- let _m_z = 0x3ade68b1;
- const mask = 0xffffffff;
- return () => {
- _m_z = (0x9069 * (_m_z & 0xFFFF) + (_m_z >> 0x10)) & mask;
- _m_w = (0x4650 * (_m_w & 0xFFFF) + (_m_w >> 0x10)) & mask;
- let result = ((_m_z << 0x10) + _m_w) & mask;
- result /= 0x100000000;
- result += 0.5;
- return result * (Math.random() > 0.5 ? 1 : -1);
- };
- };
- for (let i = 0, rcache; i < nBytes; i += 4) {
- const _r = r((rcache || Math.random()) * 0x100000000);
- rcache = _r() * 0x3ade67b7;
- words.push((_r() * 0x100000000) | 0);
- }
- return new WordArray(words, nBytes);
- }
- /**
- * Converts this word array to a string.
- *
- * @param {Encoder} encoder (Optional) The encoding strategy to use. Default: CryptoJS.enc.Hex
- *
- * @return {string} The stringified word array.
- *
- * @example
- *
- * var string = wordArray + '';
- * var string = wordArray.toString();
- * var string = wordArray.toString(CryptoJS.enc.Utf8);
- */
- toString(encoder = Hex) {
- return encoder.stringify(this);
- }
- /**
- * Concatenates a word array to this word array.
- *
- * @param {WordArray} wordArray The word array to append.
- *
- * @return {WordArray} This word array.
- *
- * @example
- *
- * wordArray1.concat(wordArray2);
- */
- concat(wordArray) {
- // Shortcuts
- const thisWords = this.words;
- const thatWords = wordArray.words;
- const thisSigBytes = this.sigBytes;
- const thatSigBytes = wordArray.sigBytes;
- // Clamp excess bits
- this.clamp();
- // Concat
- if (thisSigBytes % 4) {
- // Copy one byte at a time
- for (let i = 0; i < thatSigBytes; i += 1) {
- const thatByte = (thatWords[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
- thisWords[(thisSigBytes + i) >>> 2] |= thatByte << (24 - ((thisSigBytes + i) % 4) * 8);
- }
- } else {
- // Copy one word at a time
- for (let i = 0; i < thatSigBytes; i += 4) {
- thisWords[(thisSigBytes + i) >>> 2] = thatWords[i >>> 2];
- }
- }
- this.sigBytes += thatSigBytes;
- // Chainable
- return this;
- }
- /**
- * Removes insignificant bits.
- *
- * @example
- *
- * wordArray.clamp();
- */
- clamp() {
- // Shortcuts
- const { words, sigBytes } = this;
- // Clamp
- words[sigBytes >>> 2] &= 0xffffffff << (32 - (sigBytes % 4) * 8);
- words.length = Math.ceil(sigBytes / 4);
- }
- /**
- * Creates a copy of this word array.
- *
- * @return {WordArray} The clone.
- *
- * @example
- *
- * var clone = wordArray.clone();
- */
- clone() {
- const clone = super.clone.call(this);
- clone.words = this.words.slice(0);
- return clone;
- }
- }
- /**
- * Hex encoding strategy.
- */
- export const Hex = {
- /**
- * Converts a word array to a hex string.
- *
- * @param {WordArray} wordArray The word array.
- *
- * @return {string} The hex string.
- *
- * @static
- *
- * @example
- *
- * var hexString = CryptoJS.enc.Hex.stringify(wordArray);
- */
- stringify(wordArray) {
- // Shortcuts
- const { words, sigBytes } = wordArray;
- // Convert
- const hexChars = [];
- for (let i = 0; i < sigBytes; i += 1) {
- const bite = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
- hexChars.push((bite >>> 4).toString(16));
- hexChars.push((bite & 0x0f).toString(16));
- }
- return hexChars.join('');
- },
- /**
- * Converts a hex string to a word array.
- *
- * @param {string} hexStr The hex string.
- *
- * @return {WordArray} The word array.
- *
- * @static
- *
- * @example
- *
- * var wordArray = CryptoJS.enc.Hex.parse(hexString);
- */
- parse(hexStr) {
- // Shortcut
- const hexStrLength = hexStr.length;
- // Convert
- const words = [];
- for (let i = 0; i < hexStrLength; i += 2) {
- words[i >>> 3] |= parseInt(hexStr.substr(i, 2), 16) << (24 - (i % 8) * 4);
- }
- return new WordArray(words, hexStrLength / 2);
- },
- };
- /**
- * Latin1 encoding strategy.
- */
- export const Latin1 = {
- /**
- * Converts a word array to a Latin1 string.
- *
- * @param {WordArray} wordArray The word array.
- *
- * @return {string} The Latin1 string.
- *
- * @static
- *
- * @example
- *
- * var latin1String = CryptoJS.enc.Latin1.stringify(wordArray);
- */
- stringify(wordArray) {
- // Shortcuts
- const { words, sigBytes } = wordArray;
- // Convert
- const latin1Chars = [];
- for (let i = 0; i < sigBytes; i += 1) {
- const bite = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
- latin1Chars.push(String.fromCharCode(bite));
- }
- return latin1Chars.join('');
- },
- /**
- * Converts a Latin1 string to a word array.
- *
- * @param {string} latin1Str The Latin1 string.
- *
- * @return {WordArray} The word array.
- *
- * @static
- *
- * @example
- *
- * var wordArray = CryptoJS.enc.Latin1.parse(latin1String);
- */
- parse(latin1Str) {
- // Shortcut
- const latin1StrLength = latin1Str.length;
- // Convert
- const words = [];
- for (let i = 0; i < latin1StrLength; i += 1) {
- words[i >>> 2] |= (latin1Str.charCodeAt(i) & 0xff) << (24 - (i % 4) * 8);
- }
- return new WordArray(words, latin1StrLength);
- },
- };
- /**
- * UTF-8 encoding strategy.
- */
- export const Utf8 = {
- /**
- * Converts a word array to a UTF-8 string.
- *
- * @param {WordArray} wordArray The word array.
- *
- * @return {string} The UTF-8 string.
- *
- * @static
- *
- * @example
- *
- * var utf8String = CryptoJS.enc.Utf8.stringify(wordArray);
- */
- stringify(wordArray) {
- try {
- return decodeURIComponent(escape(Latin1.stringify(wordArray)));
- } catch (e) {
- throw new Error('Malformed UTF-8 data');
- }
- },
- /**
- * Converts a UTF-8 string to a word array.
- *
- * @param {string} utf8Str The UTF-8 string.
- *
- * @return {WordArray} The word array.
- *
- * @static
- *
- * @example
- *
- * var wordArray = CryptoJS.enc.Utf8.parse(utf8String);
- */
- parse(utf8Str) {
- return Latin1.parse(unescape(encodeURIComponent(utf8Str)));
- },
- };
- /**
- * Abstract buffered block algorithm template.
- *
- * The property blockSize must be implemented in a concrete subtype.
- *
- * @property {number} _minBufferSize
- *
- * The number of blocks that should be kept unprocessed in the buffer. Default: 0
- */
- export class BufferedBlockAlgorithm extends Base {
- constructor() {
- super();
- this._minBufferSize = 0;
- }
- /**
- * Resets this block algorithm's data buffer to its initial state.
- *
- * @example
- *
- * bufferedBlockAlgorithm.reset();
- */
- reset() {
- // Initial values
- this._data = new WordArray();
- this._nDataBytes = 0;
- }
- /**
- * Adds new data to this block algorithm's buffer.
- *
- * @param {WordArray|string} data
- *
- * The data to append. Strings are converted to a WordArray using UTF-8.
- *
- * @example
- *
- * bufferedBlockAlgorithm._append('data');
- * bufferedBlockAlgorithm._append(wordArray);
- */
- _append(data) {
- let m_data = data;
- // Convert string to WordArray, else assume WordArray already
- if (typeof m_data === 'string') {
- m_data = Utf8.parse(m_data);
- }
- // Append
- this._data.concat(m_data);
- this._nDataBytes += m_data.sigBytes;
- }
- /**
- * Processes available data blocks.
- *
- * This method invokes _doProcessBlock(offset), which must be implemented by a concrete subtype.
- *
- * @param {boolean} doFlush Whether all blocks and partial blocks should be processed.
- *
- * @return {WordArray} The processed data.
- *
- * @example
- *
- * var processedData = bufferedBlockAlgorithm._process();
- * var processedData = bufferedBlockAlgorithm._process(!!'flush');
- */
- _process(doFlush) {
- let processedWords;
- // Shortcuts
- const { _data: data, blockSize } = this;
- const dataWords = data.words;
- const dataSigBytes = data.sigBytes;
- const blockSizeBytes = blockSize * 4;
- // Count blocks ready
- let nBlocksReady = dataSigBytes / blockSizeBytes;
- if (doFlush) {
- // Round up to include partial blocks
- nBlocksReady = Math.ceil(nBlocksReady);
- } else {
- // Round down to include only full blocks,
- // less the number of blocks that must remain in the buffer
- nBlocksReady = Math.max((nBlocksReady | 0) - this._minBufferSize, 0);
- }
- // Count words ready
- const nWordsReady = nBlocksReady * blockSize;
- // Count bytes ready
- const nBytesReady = Math.min(nWordsReady * 4, dataSigBytes);
- // Process blocks
- if (nWordsReady) {
- for (let offset = 0; offset < nWordsReady; offset += blockSize) {
- // Perform concrete-algorithm logic
- this._doProcessBlock(dataWords, offset);
- }
- // Remove processed words
- processedWords = dataWords.splice(0, nWordsReady);
- data.sigBytes -= nBytesReady;
- }
- // Return processed words
- return new WordArray(processedWords, nBytesReady);
- }
- /**
- * Creates a copy of this object.
- *
- * @return {Object} The clone.
- *
- * @example
- *
- * var clone = bufferedBlockAlgorithm.clone();
- */
- clone() {
- const clone = super.clone.call(this);
- clone._data = this._data.clone();
- return clone;
- }
- }
- /**
- * Abstract hasher template.
- *
- * @property {number} blockSize
- *
- * The number of 32-bit words this hasher operates on. Default: 16 (512 bits)
- */
- export class Hasher extends BufferedBlockAlgorithm {
- constructor(cfg) {
- super();
- this.blockSize = 512 / 32;
- /**
- * Configuration options.
- */
- this.cfg = Object.assign(new Base(), cfg);
- // Set initial values
- this.reset();
- }
- /**
- * Creates a shortcut function to a hasher's object interface.
- *
- * @param {Hasher} SubHasher The hasher to create a helper for.
- *
- * @return {Function} The shortcut function.
- *
- * @static
- *
- * @example
- *
- * var SHA256 = CryptoJS.lib.Hasher._createHelper(CryptoJS.algo.SHA256);
- */
- static _createHelper(SubHasher) {
- return (message, cfg) => new SubHasher(cfg).finalize(message);
- }
- /**
- * Creates a shortcut function to the HMAC's object interface.
- *
- * @param {Hasher} SubHasher The hasher to use in this HMAC helper.
- *
- * @return {Function} The shortcut function.
- *
- * @static
- *
- * @example
- *
- * var HmacSHA256 = CryptoJS.lib.Hasher._createHmacHelper(CryptoJS.algo.SHA256);
- */
- static _createHmacHelper(SubHasher) {
- return (message, key) => new HMAC(SubHasher, key).finalize(message);
- }
- /**
- * Resets this hasher to its initial state.
- *
- * @example
- *
- * hasher.reset();
- */
- reset() {
- // Reset data buffer
- super.reset.call(this);
- // Perform concrete-hasher logic
- this._doReset();
- }
- /**
- * Updates this hasher with a message.
- *
- * @param {WordArray|string} messageUpdate The message to append.
- *
- * @return {Hasher} This hasher.
- *
- * @example
- *
- * hasher.update('message');
- * hasher.update(wordArray);
- */
- update(messageUpdate) {
- // Append
- this._append(messageUpdate);
- // Update the hash
- this._process();
- // Chainable
- return this;
- }
- /**
- * Finalizes the hash computation.
- * Note that the finalize operation is effectively a destructive, read-once operation.
- *
- * @param {WordArray|string} messageUpdate (Optional) A final message update.
- *
- * @return {WordArray} The hash.
- *
- * @example
- *
- * var hash = hasher.finalize();
- * var hash = hasher.finalize('message');
- * var hash = hasher.finalize(wordArray);
- */
- finalize(messageUpdate) {
- // Final message update
- if (messageUpdate) {
- this._append(messageUpdate);
- }
- // Perform concrete-hasher logic
- const hash = this._doFinalize();
- return hash;
- }
- }
- /**
- * HMAC algorithm.
- */
- export class HMAC extends Base {
- /**
- * Initializes a newly created HMAC.
- *
- * @param {Hasher} SubHasher The hash algorithm to use.
- * @param {WordArray|string} key The secret key.
- *
- * @example
- *
- * var hmacHasher = CryptoJS.algo.HMAC.create(CryptoJS.algo.SHA256, key);
- */
- constructor(SubHasher, key) {
- super();
- const hasher = new SubHasher();
- this._hasher = hasher;
- // Convert string to WordArray, else assume WordArray already
- let _key = key;
- if (typeof _key === 'string') {
- _key = Utf8.parse(_key);
- }
- // Shortcuts
- const hasherBlockSize = hasher.blockSize;
- const hasherBlockSizeBytes = hasherBlockSize * 4;
- // Allow arbitrary length keys
- if (_key.sigBytes > hasherBlockSizeBytes) {
- _key = hasher.finalize(key);
- }
- // Clamp excess bits
- _key.clamp();
- // Clone key for inner and outer pads
- const oKey = _key.clone();
- this._oKey = oKey;
- const iKey = _key.clone();
- this._iKey = iKey;
- // Shortcuts
- const oKeyWords = oKey.words;
- const iKeyWords = iKey.words;
- // XOR keys with pad constants
- for (let i = 0; i < hasherBlockSize; i += 1) {
- oKeyWords[i] ^= 0x5c5c5c5c;
- iKeyWords[i] ^= 0x36363636;
- }
- oKey.sigBytes = hasherBlockSizeBytes;
- iKey.sigBytes = hasherBlockSizeBytes;
- // Set initial values
- this.reset();
- }
- /**
- * Resets this HMAC to its initial state.
- *
- * @example
- *
- * hmacHasher.reset();
- */
- reset() {
- // Shortcut
- const hasher = this._hasher;
- // Reset
- hasher.reset();
- hasher.update(this._iKey);
- }
- /**
- * Updates this HMAC with a message.
- *
- * @param {WordArray|string} messageUpdate The message to append.
- *
- * @return {HMAC} This HMAC instance.
- *
- * @example
- *
- * hmacHasher.update('message');
- * hmacHasher.update(wordArray);
- */
- update(messageUpdate) {
- this._hasher.update(messageUpdate);
- // Chainable
- return this;
- }
- /**
- * Finalizes the HMAC computation.
- * Note that the finalize operation is effectively a destructive, read-once operation.
- *
- * @param {WordArray|string} messageUpdate (Optional) A final message update.
- *
- * @return {WordArray} The HMAC.
- *
- * @example
- *
- * var hmac = hmacHasher.finalize();
- * var hmac = hmacHasher.finalize('message');
- * var hmac = hmacHasher.finalize(wordArray);
- */
- finalize(messageUpdate) {
- // Shortcut
- const hasher = this._hasher;
- // Compute HMAC
- const innerHash = hasher.finalize(messageUpdate);
- hasher.reset();
- const hmac = hasher.finalize(this._oKey.clone().concat(innerHash));
- return hmac;
- }
- }
|