mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-02 12:47:41 +02:00
123 lines
4.5 KiB
TypeScript
123 lines
4.5 KiB
TypeScript
import { Crypto, CryptoBuffer, Digest, CipherAlgorithm, EncryptionResult, EncryptionParameters } from '@joplin/lib/services/e2ee/types';
|
|
import QuickCrypto from 'react-native-quick-crypto';
|
|
import type { CipherGCMOptions, CipherGCM, DecipherGCM } from 'crypto';
|
|
import {
|
|
generateNonce as generateNonceShared,
|
|
increaseNonce as increaseNonceShared,
|
|
setRandomBytesImplementation,
|
|
} from '@joplin/lib/services/e2ee/cryptoShared';
|
|
|
|
type DigestNameMap = Record<Digest, string>;
|
|
const digestNameMap: DigestNameMap = {
|
|
[Digest.sha1]: 'sha1',
|
|
[Digest.sha256]: 'sha256',
|
|
[Digest.sha384]: 'sha384',
|
|
[Digest.sha512]: 'sha512',
|
|
};
|
|
|
|
const pbkdf2Raw = (password: string, salt: CryptoBuffer, iterations: number, keylen: number, digest: Digest): Promise<CryptoBuffer> => {
|
|
return new Promise((resolve, reject) => {
|
|
QuickCrypto.pbkdf2(password, salt, iterations, keylen, digest, (error, result) => {
|
|
if (error) {
|
|
reject(error);
|
|
} else {
|
|
resolve(result);
|
|
}
|
|
});
|
|
});
|
|
};
|
|
|
|
const encryptRaw = (data: CryptoBuffer, algorithm: CipherAlgorithm, key: CryptoBuffer, iv: CryptoBuffer, authTagLength: number, associatedData: CryptoBuffer) => {
|
|
|
|
const cipher = QuickCrypto.createCipheriv(algorithm, key, iv, { authTagLength: authTagLength } as CipherGCMOptions) as CipherGCM;
|
|
|
|
cipher.setAAD(associatedData, { plaintextLength: Buffer.byteLength(data) });
|
|
|
|
const encryptedData = [cipher.update(data), cipher.final()];
|
|
const authTag = cipher.getAuthTag();
|
|
|
|
return Buffer.concat([encryptedData[0], encryptedData[1], authTag]);
|
|
};
|
|
|
|
const decryptRaw = (data: CryptoBuffer, algorithm: CipherAlgorithm, key: CryptoBuffer, iv: CryptoBuffer, authTagLength: number, associatedData: CryptoBuffer) => {
|
|
|
|
const decipher = QuickCrypto.createDecipheriv(algorithm, key, iv, { authTagLength: authTagLength } as CipherGCMOptions) as DecipherGCM;
|
|
|
|
const authTag = data.subarray(-authTagLength);
|
|
const encryptedData = data.subarray(0, data.byteLength - authTag.byteLength);
|
|
decipher.setAuthTag(authTag);
|
|
decipher.setAAD(associatedData, { plaintextLength: Buffer.byteLength(data) });
|
|
|
|
try {
|
|
return Buffer.concat([decipher.update(encryptedData), decipher.final()]);
|
|
} catch (error) {
|
|
throw new Error(`Authentication failed! ${error}`);
|
|
}
|
|
};
|
|
|
|
const crypto: Crypto = {
|
|
|
|
randomBytes: async (size: number) => {
|
|
return new Promise((resolve, reject) => {
|
|
QuickCrypto.randomBytes(size, (error, result) => {
|
|
if (error) {
|
|
reject(error);
|
|
} else {
|
|
resolve(result);
|
|
}
|
|
});
|
|
});
|
|
},
|
|
|
|
digest: async (algorithm: Digest, data: Uint8Array) => {
|
|
const hash = QuickCrypto.createHash(digestNameMap[algorithm]);
|
|
hash.update(data);
|
|
return hash.digest();
|
|
},
|
|
|
|
encrypt: async (password: string, salt: CryptoBuffer, data: CryptoBuffer, encryptionParameters: EncryptionParameters) => {
|
|
|
|
// Parameters in EncryptionParameters won't appear in result
|
|
const result: EncryptionResult = {
|
|
salt: salt.toString('base64'),
|
|
iv: '',
|
|
ct: '', // cipherText
|
|
};
|
|
|
|
// 96 bits IV
|
|
// "For IVs, it is recommended that implementations restrict support to the length of 96 bits, to promote interoperability, efficiency, and simplicity of design." - NIST SP 800-38D
|
|
const iv = await crypto.randomBytes(12);
|
|
|
|
const key = await pbkdf2Raw(password, salt, encryptionParameters.iterationCount, encryptionParameters.keyLength, encryptionParameters.digestAlgorithm);
|
|
const encrypted = encryptRaw(data, encryptionParameters.cipherAlgorithm, key, iv, encryptionParameters.authTagLength, encryptionParameters.associatedData);
|
|
|
|
result.iv = iv.toString('base64');
|
|
result.ct = encrypted.toString('base64');
|
|
|
|
return result;
|
|
},
|
|
|
|
decrypt: async (password: string, data: EncryptionResult, encryptionParameters: EncryptionParameters) => {
|
|
|
|
const salt = Buffer.from(data.salt, 'base64');
|
|
const iv = Buffer.from(data.iv, 'base64');
|
|
|
|
const key = await pbkdf2Raw(password, salt, encryptionParameters.iterationCount, encryptionParameters.keyLength, encryptionParameters.digestAlgorithm);
|
|
const decrypted = decryptRaw(Buffer.from(data.ct, 'base64'), encryptionParameters.cipherAlgorithm, key, iv, encryptionParameters.authTagLength, encryptionParameters.associatedData);
|
|
|
|
return decrypted;
|
|
},
|
|
|
|
encryptString: async (password: string, salt: CryptoBuffer, data: string, encoding: BufferEncoding, encryptionParameters: EncryptionParameters) => {
|
|
return crypto.encrypt(password, salt, Buffer.from(data, encoding), encryptionParameters);
|
|
},
|
|
|
|
generateNonce: generateNonceShared,
|
|
|
|
increaseNonce: increaseNonceShared,
|
|
};
|
|
|
|
setRandomBytesImplementation(crypto.randomBytes);
|
|
|
|
export default crypto;
|