|
|
|
@ -6,6 +6,8 @@ import { encryptWithSenderKey } from '../../session/medium_group/ratchet';
|
|
|
|
|
import { PubKey } from '../types';
|
|
|
|
|
import { StringUtils } from '../utils';
|
|
|
|
|
|
|
|
|
|
import * as libsodiumwrappers from 'libsodium-wrappers';
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Add padding to a message buffer
|
|
|
|
|
* @param messageBuffer The buffer to add padding to.
|
|
|
|
@ -69,6 +71,76 @@ export async function encrypt(
|
|
|
|
|
return encryptUsingSealedSender(device, innerCipherText);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function getSodium(): Promise<typeof libsodiumwrappers> {
|
|
|
|
|
await libsodiumwrappers.ready;
|
|
|
|
|
return libsodiumwrappers;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const concatUInt8Array = (...args: Array<Uint8Array>): Uint8Array => {
|
|
|
|
|
const totalLength = args.reduce((acc, current) => acc + current.length, 0);
|
|
|
|
|
|
|
|
|
|
const concatted = new Uint8Array(totalLength);
|
|
|
|
|
let currentIndex = 0;
|
|
|
|
|
args.forEach(arr => {
|
|
|
|
|
concatted.set(arr, currentIndex);
|
|
|
|
|
currentIndex += arr.length;
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
return concatted;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
export async function encryptUsingSessionProtocol(
|
|
|
|
|
recipientHexEncodedX25519PublicKey: PubKey,
|
|
|
|
|
plaintext: Uint8Array
|
|
|
|
|
): Promise<Uint8Array> {
|
|
|
|
|
const userED25519KeyPairHex = await UserUtil.getUserED25519KeyPair();
|
|
|
|
|
if (!userED25519KeyPairHex) {
|
|
|
|
|
throw new Error("Couldn't find user ED25519 key pair.");
|
|
|
|
|
}
|
|
|
|
|
const sodium = await getSodium();
|
|
|
|
|
|
|
|
|
|
const recipientX25519PublicKeyWithoutPrefix = PubKey.remove05PrefixIfNeeded(
|
|
|
|
|
recipientHexEncodedX25519PublicKey.key
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const recipientX25519PublicKey = new Uint8Array(
|
|
|
|
|
StringUtils.fromHex(recipientX25519PublicKeyWithoutPrefix)
|
|
|
|
|
);
|
|
|
|
|
const userED25519PubKeyBytes = new Uint8Array(
|
|
|
|
|
StringUtils.fromHex(userED25519KeyPairHex.pubKey)
|
|
|
|
|
);
|
|
|
|
|
const userED25519SecretKeyBytes = new Uint8Array(
|
|
|
|
|
StringUtils.fromHex(userED25519KeyPairHex.privKey)
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// merge all arrays into one
|
|
|
|
|
const data = concatUInt8Array(
|
|
|
|
|
plaintext,
|
|
|
|
|
userED25519PubKeyBytes,
|
|
|
|
|
recipientX25519PublicKey
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const signature = sodium.crypto_sign(data, userED25519SecretKeyBytes);
|
|
|
|
|
if (!signature) {
|
|
|
|
|
throw new Error("Couldn't sign message");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const dataForBoxSeal = concatUInt8Array(
|
|
|
|
|
plaintext,
|
|
|
|
|
userED25519PubKeyBytes,
|
|
|
|
|
signature
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
const ciphertext = sodium.crypto_box_seal(
|
|
|
|
|
dataForBoxSeal,
|
|
|
|
|
recipientX25519PublicKey
|
|
|
|
|
);
|
|
|
|
|
if (!ciphertext) {
|
|
|
|
|
throw new Error("Couldn't encrypt message.");
|
|
|
|
|
}
|
|
|
|
|
return ciphertext;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export async function encryptForMediumGroup(
|
|
|
|
|
device: PubKey,
|
|
|
|
|
plainTextBuffer: Uint8Array
|
|
|
|
|