UNPKG

sanka-baileyss

Version:

WhatsApp API Modification By Sanka_Vollerei

99 lines (94 loc) 4.55 kB
"use strict" Object.defineProperty(exports, "__esModule", { value: true }) const crypto_1 = require("@itsukichan/libsignal-node/src/crypto") const queue_job_1 = require("./queue-job") const sender_key_message_1 = require("./sender-key-message") class GroupCipher { constructor(senderKeyStore, senderKeyName) { this.senderKeyStore = senderKeyStore this.senderKeyName = senderKeyName } queueJob(awaitable) { return queue_job_1.queueJob(this.senderKeyName.toString(), awaitable) } async encrypt(paddedPlaintext) { return await this.queueJob(async () => { const record = await this.senderKeyStore.loadSenderKey(this.senderKeyName) if (!record) { throw new Error('No SenderKeyRecord found for encryption') } const senderKeyState = record.getSenderKeyState() if (!senderKeyState) { throw new Error('No session to encrypt message') } const iteration = senderKeyState.getSenderChainKey().getIteration() const senderKey = this.getSenderKey(senderKeyState, iteration === 0 ? 0 : iteration + 1) const ciphertext = await this.getCipherText(senderKey.getIv(), senderKey.getCipherKey(), paddedPlaintext) const senderKeyMessage = new sender_key_message_1.SenderKeyMessage(senderKeyState.getKeyId(), senderKey.getIteration(), ciphertext, senderKeyState.getSigningKeyPrivate()) await this.senderKeyStore.storeSenderKey(this.senderKeyName, record) return senderKeyMessage.serialize() }) } async decrypt(senderKeyMessageBytes) { return await this.queueJob(async () => { const record = await this.senderKeyStore.loadSenderKey(this.senderKeyName) if (!record) { throw new Error('No SenderKeyRecord found for decryption') } const senderKeyMessage = new sender_key_message_1.SenderKeyMessage(null, null, null, null, senderKeyMessageBytes) const senderKeyState = record.getSenderKeyState(senderKeyMessage.getKeyId()) if (!senderKeyState) { throw new Error('No session found to decrypt message') } senderKeyMessage.verifySignature(senderKeyState.getSigningKeyPublic()) const senderKey = this.getSenderKey(senderKeyState, senderKeyMessage.getIteration()) const plaintext = await this.getPlainText(senderKey.getIv(), senderKey.getCipherKey(), senderKeyMessage.getCipherText()) await this.senderKeyStore.storeSenderKey(this.senderKeyName, record) return plaintext }) } getSenderKey(senderKeyState, iteration) { let senderChainKey = senderKeyState.getSenderChainKey() if (senderChainKey.getIteration() > iteration) { if (senderKeyState.hasSenderMessageKey(iteration)) { const messageKey = senderKeyState.removeSenderMessageKey(iteration) if (!messageKey) { throw new Error('No sender message key found for iteration') } return messageKey } throw new Error(`Received message with old counter: ${senderChainKey.getIteration()}, ${iteration}`) } if (iteration - senderChainKey.getIteration() > 2000) { throw new Error('Over 2000 messages into the future!') } while (senderChainKey.getIteration() < iteration) { senderKeyState.addSenderMessageKey(senderChainKey.getSenderMessageKey()) senderChainKey = senderChainKey.getNext() } senderKeyState.setSenderChainKey(senderChainKey.getNext()) return senderChainKey.getSenderMessageKey() } async getPlainText(iv, key, ciphertext) { try { return crypto_1.decrypt(key, ciphertext, iv) } catch (e) { throw new Error('InvalidMessageException') } } async getCipherText(iv, key, plaintext) { try { const ivBuffer = typeof iv === 'string' ? Buffer.from(iv, 'base64') : iv const keyBuffer = typeof key === 'string' ? Buffer.from(key, 'base64') : key const plaintextBuffer = typeof plaintext === 'string' ? Buffer.from(plaintext) : plaintext return crypto_1.encrypt(keyBuffer, plaintextBuffer, ivBuffer) } catch (e) { throw new Error('InvalidMessageException') } } } module.exports = { GroupCipher }