UNPKG

node-aescrypt

Version:

A node implementation of the AES Crypt <https://www.aescrypt.com/> file encryption format.

207 lines 16.4 kB
import { createDecipheriv } from 'crypto'; import { Transform } from 'stream'; import { AESCRYPT_FILE_FORMAT_VERSION, getHMAC, getKey, toStream, withStream, } from './util'; /** * Decrypt a Buffer that is in the AES Crypt file format. * * Create a stream transformer that takes [Readable stream](https://nodejs.org/api/stream.html) * that is encrypted in the * [AES Crypt file format](https://www.aescrypt.com/aes_file_format.html) and * decrypts it passing it on as a Readable stream. */ export class Decrypt extends Transform { constructor(password, options) { super(options); this.password = password; this.decipher = null; this.hmac = null; this.mode = 0; this.buffer = Buffer.alloc(0); } static get MODE_FILE_HEADER() { return 0; } static get MODE_EXTESIONS() { return 1; } static get MODE_CREDENTIALS() { return 2; } static get MODE_DECRYPT() { return 3; } // Create a small helper static method if you just want to decrypt a whole // Buffer all at once. static buffer(password, buffer) { return new Promise((resolve, reject) => { toStream(buffer) .pipe(new Decrypt(password)) .pipe(withStream(contents => { resolve(contents); })) .on('error', reject); }); } _transform(chunk, _, callback) { this.buffer = Buffer.concat([this.buffer, chunk]); let error = null; // Move through the various sections of the file format and raise an error // If anything is malformed. if (this.mode === Decrypt.MODE_FILE_HEADER) { error = this._modeFileHeader(); } if (!error && this.mode === Decrypt.MODE_EXTESIONS) { error = this._modeExtensions(); } if (!error && this.mode === Decrypt.MODE_CREDENTIALS) { error = this._modeCredentials(); } // Finally ready to decrypt the contents. if (!error && this.mode === Decrypt.MODE_DECRYPT) { // We need to reserve 33 bytes (+ 16 for the padding) of the end for the file-size-modulo-16 and HMAC. if (this.buffer.length > 49) { const encChunk = this.buffer.slice(0, -49); // This is unnecessary, but makes tslint keep quiet. if (this.decipher == null) { return; } if (this.hmac == null) { return; } this.hmac.update(encChunk); this.push(this.decipher.update(encChunk)); this.buffer = this.buffer.slice(-49); } } if (error) { callback(error); } else { callback(); } } _flush(callback) { let error = null; // If we never got to the decryption mode, something went terribly wrong. // Most likely, there is a problem in the extensions, and we never found // the end. if (this.mode !== Decrypt.MODE_DECRYPT) { error = new Error('Error: Message has been altered or password is incorrect'); } else { // We are at the end of the file. Let's hash that last remaining cipher text // and check the HMAC. const encChunk = this.buffer.slice(0, 16); const lenMod16 = this.buffer.readUInt8(16); const encHMACActual = this.buffer.slice(17); // This is unnecessary, but makes tslint keep quiet. if (this.decipher == null) { return; } if (this.hmac == null) { return; } this.hmac.update(encChunk); const encHMACExpected = this.hmac.digest(); // Validately the integrity of the cipher text. if (encHMACExpected.compare(encHMACActual) !== 0) { error = new Error('Error: Message has been altered or password is incorrect'); } // Validate the padding length (or more accurately, the length of the last block minus the padding). else if (lenMod16 > 16) { error = new Error('Error: Message has been altered or password is incorrect'); } else { // Decrypt the last block and send it on its way. const decChunk = Buffer.concat([ this.decipher.update(encChunk), this.decipher.final(), ]).slice(0, lenMod16); this.push(decChunk); } } if (error) { callback(error); } else { callback(); } } _modeFileHeader() { if (this.buffer.length >= 5) { const type = this.buffer.slice(0, 3).toString(); const version = this.buffer.readUInt8(3); if (type !== 'AES') { return new Error('Error: Bad file header (not aescrypt file or is corrupted?'); } // We only understand the version 2 of the AES Crypt file format as described // at https://www.aescrypt.com/aes_file_format.html. if (version !== AESCRYPT_FILE_FORMAT_VERSION) { return new Error('Error: Unsupported AES file version'); } this.buffer = this.buffer.slice(5); this.mode = Decrypt.MODE_EXTESIONS; } return null; } _modeExtensions() { let i = 0; // Search through the buffer to the length. // If we can't find the end of the extensions in the current buffer, let it // buffer a little more. while (i + 1 < this.buffer.length) { const extLen = this.buffer.readUInt16BE(i); i += 2; // If this extension has a length, fast-forward past it. if (extLen > 0) { i += extLen; } // If this is a zero length extension, we are done. else { this.buffer = this.buffer.slice(i); this.mode = Decrypt.MODE_CREDENTIALS; break; } } return null; } _modeCredentials() { if (this.buffer.length >= 96) { const credIV = this.buffer.slice(0, 16); const credKey = getKey(credIV, this.password); const credDecipher = this._getDecipher(credKey, credIV); credDecipher.setAutoPadding(false); const credBlock = this.buffer.slice(16, 64); const credHMACActual = this.buffer.slice(64, 96); const credHMACExpected = getHMAC(credKey) .update(credBlock) .digest(); // First we check the HMAC signature of the encrypted credentials block. // This ensures nothing was tampered with. It also has the added benefit // of checking the password early on in the decryption process. if (credHMACExpected.compare(credHMACActual) !== 0) { return new Error('Error: Message has been altered or password is incorrect'); } // Decrypt the credentials we need for the rest of the contents. const decryptedCredBlock = Buffer.concat([ credDecipher.update(credBlock), credDecipher.final(), ]); const encIV = decryptedCredBlock.slice(0, 16); const encKey = decryptedCredBlock.slice(16, 48); // Create our main workhorses using the decrypted credentials. this.decipher = this._getDecipher(encKey, encIV); this.hmac = getHMAC(encKey); delete this.password; // Don't need this anymore. this.buffer = this.buffer.slice(96); this.mode = Decrypt.MODE_DECRYPT; } return null; } _getDecipher(key, iv) { const encDecipher = createDecipheriv('aes-256-cbc', key, iv); encDecipher.setAutoPadding(false); return encDecipher; } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decrypt.js","sourceRoot":"","sources":["../../../src/lib/decrypt.ts"],"names":[],"mappings":"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"}