2020-10-30 20:34:04 +00:00
|
|
|
// Copyright 2020 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2020-04-15 23:12:28 +00:00
|
|
|
import {
|
2020-09-09 02:25:05 +00:00
|
|
|
AuthCredential,
|
|
|
|
ClientZkAuthOperations,
|
|
|
|
ClientZkGroupCipher,
|
2020-04-15 23:12:28 +00:00
|
|
|
ClientZkProfileOperations,
|
|
|
|
FFICompatArray,
|
|
|
|
FFICompatArrayType,
|
2020-09-09 02:25:05 +00:00
|
|
|
GroupMasterKey,
|
|
|
|
GroupSecretParams,
|
2020-04-15 23:12:28 +00:00
|
|
|
ProfileKey,
|
2020-09-09 02:25:05 +00:00
|
|
|
ProfileKeyCiphertext,
|
2020-10-06 17:06:34 +00:00
|
|
|
ProfileKeyCredential,
|
2020-09-09 02:25:05 +00:00
|
|
|
ProfileKeyCredentialPresentation,
|
2020-04-15 23:12:28 +00:00
|
|
|
ProfileKeyCredentialRequestContext,
|
|
|
|
ProfileKeyCredentialResponse,
|
|
|
|
ServerPublicParams,
|
2020-09-09 02:25:05 +00:00
|
|
|
UuidCiphertext,
|
2020-04-15 23:12:28 +00:00
|
|
|
} from 'zkgroup';
|
2021-06-22 14:46:42 +00:00
|
|
|
import * as Bytes from '../Bytes';
|
2020-04-15 23:12:28 +00:00
|
|
|
|
2020-09-14 21:56:35 +00:00
|
|
|
export * from 'zkgroup';
|
2020-09-09 02:25:05 +00:00
|
|
|
|
2021-06-22 14:46:42 +00:00
|
|
|
export function uint8ArrayToCompatArray(
|
|
|
|
buffer: Uint8Array
|
2020-04-15 23:12:28 +00:00
|
|
|
): FFICompatArrayType {
|
2021-06-22 14:46:42 +00:00
|
|
|
return new FFICompatArray(Buffer.from(buffer));
|
2020-04-15 23:12:28 +00:00
|
|
|
}
|
|
|
|
|
2021-06-22 14:46:42 +00:00
|
|
|
export function compatArrayToUint8Array(
|
2020-04-15 23:12:28 +00:00
|
|
|
compatArray: FFICompatArrayType
|
2021-06-22 14:46:42 +00:00
|
|
|
): Uint8Array {
|
|
|
|
return compatArray.buffer;
|
2020-04-15 23:12:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function base64ToCompatArray(base64: string): FFICompatArrayType {
|
2021-06-22 14:46:42 +00:00
|
|
|
return uint8ArrayToCompatArray(Bytes.fromBase64(base64));
|
2020-04-15 23:12:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function compatArrayToBase64(compatArray: FFICompatArrayType): string {
|
2021-06-22 14:46:42 +00:00
|
|
|
return Bytes.toBase64(compatArrayToUint8Array(compatArray));
|
2020-04-15 23:12:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function compatArrayToHex(compatArray: FFICompatArrayType): string {
|
2021-06-22 14:46:42 +00:00
|
|
|
return Bytes.toHex(compatArrayToUint8Array(compatArray));
|
2020-04-15 23:12:28 +00:00
|
|
|
}
|
|
|
|
|
2020-09-09 02:25:05 +00:00
|
|
|
// Scenarios
|
|
|
|
|
|
|
|
export function decryptGroupBlob(
|
|
|
|
clientZkGroupCipher: ClientZkGroupCipher,
|
2021-06-22 14:46:42 +00:00
|
|
|
ciphertext: Uint8Array
|
|
|
|
): Uint8Array {
|
|
|
|
return compatArrayToUint8Array(
|
|
|
|
clientZkGroupCipher.decryptBlob(uint8ArrayToCompatArray(ciphertext))
|
2020-09-09 02:25:05 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function decryptProfileKeyCredentialPresentation(
|
|
|
|
clientZkGroupCipher: ClientZkGroupCipher,
|
2021-06-22 14:46:42 +00:00
|
|
|
presentationBuffer: Uint8Array
|
|
|
|
): { profileKey: Uint8Array; uuid: string } {
|
2020-09-09 02:25:05 +00:00
|
|
|
const presentation = new ProfileKeyCredentialPresentation(
|
2021-06-22 14:46:42 +00:00
|
|
|
uint8ArrayToCompatArray(presentationBuffer)
|
2020-09-09 02:25:05 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
const uuidCiphertext = presentation.getUuidCiphertext();
|
|
|
|
const uuid = clientZkGroupCipher.decryptUuid(uuidCiphertext);
|
|
|
|
|
|
|
|
const profileKeyCiphertext = presentation.getProfileKeyCiphertext();
|
|
|
|
const profileKey = clientZkGroupCipher.decryptProfileKey(
|
|
|
|
profileKeyCiphertext,
|
|
|
|
uuid
|
|
|
|
);
|
|
|
|
|
|
|
|
return {
|
2021-06-22 14:46:42 +00:00
|
|
|
profileKey: compatArrayToUint8Array(profileKey.serialize()),
|
2020-09-09 02:25:05 +00:00
|
|
|
uuid,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export function decryptProfileKey(
|
|
|
|
clientZkGroupCipher: ClientZkGroupCipher,
|
2021-06-22 14:46:42 +00:00
|
|
|
profileKeyCiphertextBuffer: Uint8Array,
|
2020-09-09 02:25:05 +00:00
|
|
|
uuid: string
|
2021-06-22 14:46:42 +00:00
|
|
|
): Uint8Array {
|
2020-09-09 02:25:05 +00:00
|
|
|
const profileKeyCiphertext = new ProfileKeyCiphertext(
|
2021-06-22 14:46:42 +00:00
|
|
|
uint8ArrayToCompatArray(profileKeyCiphertextBuffer)
|
2020-09-09 02:25:05 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
const profileKey = clientZkGroupCipher.decryptProfileKey(
|
|
|
|
profileKeyCiphertext,
|
|
|
|
uuid
|
|
|
|
);
|
|
|
|
|
2021-06-22 14:46:42 +00:00
|
|
|
return compatArrayToUint8Array(profileKey.serialize());
|
2020-09-09 02:25:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function decryptUuid(
|
|
|
|
clientZkGroupCipher: ClientZkGroupCipher,
|
2021-06-22 14:46:42 +00:00
|
|
|
uuidCiphertextBuffer: Uint8Array
|
2020-09-09 02:25:05 +00:00
|
|
|
): string {
|
|
|
|
const uuidCiphertext = new UuidCiphertext(
|
2021-06-22 14:46:42 +00:00
|
|
|
uint8ArrayToCompatArray(uuidCiphertextBuffer)
|
2020-09-09 02:25:05 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
return clientZkGroupCipher.decryptUuid(uuidCiphertext);
|
|
|
|
}
|
|
|
|
|
2020-04-15 23:12:28 +00:00
|
|
|
export function deriveProfileKeyVersion(
|
|
|
|
profileKeyBase64: string,
|
|
|
|
uuid: string
|
2020-09-14 21:56:35 +00:00
|
|
|
): string {
|
2020-04-15 23:12:28 +00:00
|
|
|
const profileKeyArray = base64ToCompatArray(profileKeyBase64);
|
|
|
|
const profileKey = new ProfileKey(profileKeyArray);
|
|
|
|
|
|
|
|
const profileKeyVersion = profileKey.getProfileKeyVersion(uuid);
|
|
|
|
|
|
|
|
return profileKeyVersion.toString();
|
|
|
|
}
|
|
|
|
|
2020-09-14 21:56:35 +00:00
|
|
|
export function deriveGroupPublicParams(
|
2021-06-22 14:46:42 +00:00
|
|
|
groupSecretParamsBuffer: Uint8Array
|
|
|
|
): Uint8Array {
|
2020-09-09 02:25:05 +00:00
|
|
|
const groupSecretParams = new GroupSecretParams(
|
2021-06-22 14:46:42 +00:00
|
|
|
uint8ArrayToCompatArray(groupSecretParamsBuffer)
|
2020-09-09 02:25:05 +00:00
|
|
|
);
|
2020-04-15 23:12:28 +00:00
|
|
|
|
2021-06-22 14:46:42 +00:00
|
|
|
return compatArrayToUint8Array(
|
2020-09-09 02:25:05 +00:00
|
|
|
groupSecretParams.getPublicParams().serialize()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-06-22 14:46:42 +00:00
|
|
|
export function deriveGroupID(groupSecretParamsBuffer: Uint8Array): Uint8Array {
|
2020-09-09 02:25:05 +00:00
|
|
|
const groupSecretParams = new GroupSecretParams(
|
2021-06-22 14:46:42 +00:00
|
|
|
uint8ArrayToCompatArray(groupSecretParamsBuffer)
|
2020-09-09 02:25:05 +00:00
|
|
|
);
|
|
|
|
|
2021-06-22 14:46:42 +00:00
|
|
|
return compatArrayToUint8Array(
|
2020-11-18 15:15:42 +00:00
|
|
|
groupSecretParams.getPublicParams().getGroupIdentifier().serialize()
|
2020-09-09 02:25:05 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function deriveGroupSecretParams(
|
2021-06-22 14:46:42 +00:00
|
|
|
masterKeyBuffer: Uint8Array
|
|
|
|
): Uint8Array {
|
2020-09-09 02:25:05 +00:00
|
|
|
const masterKey = new GroupMasterKey(
|
2021-06-22 14:46:42 +00:00
|
|
|
uint8ArrayToCompatArray(masterKeyBuffer)
|
2020-09-09 02:25:05 +00:00
|
|
|
);
|
|
|
|
const groupSecretParams = GroupSecretParams.deriveFromMasterKey(masterKey);
|
|
|
|
|
2021-06-22 14:46:42 +00:00
|
|
|
return compatArrayToUint8Array(groupSecretParams.serialize());
|
2020-09-09 02:25:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function encryptGroupBlob(
|
|
|
|
clientZkGroupCipher: ClientZkGroupCipher,
|
2021-06-22 14:46:42 +00:00
|
|
|
plaintext: Uint8Array
|
|
|
|
): Uint8Array {
|
|
|
|
return compatArrayToUint8Array(
|
|
|
|
clientZkGroupCipher.encryptBlob(uint8ArrayToCompatArray(plaintext))
|
2020-09-09 02:25:05 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function encryptUuid(
|
|
|
|
clientZkGroupCipher: ClientZkGroupCipher,
|
|
|
|
uuidPlaintext: string
|
2021-06-22 14:46:42 +00:00
|
|
|
): Uint8Array {
|
2020-09-09 02:25:05 +00:00
|
|
|
const uuidCiphertext = clientZkGroupCipher.encryptUuid(uuidPlaintext);
|
|
|
|
|
2021-06-22 14:46:42 +00:00
|
|
|
return compatArrayToUint8Array(uuidCiphertext.serialize());
|
2020-04-15 23:12:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function generateProfileKeyCredentialRequest(
|
|
|
|
clientZkProfileCipher: ClientZkProfileOperations,
|
|
|
|
uuid: string,
|
|
|
|
profileKeyBase64: string
|
|
|
|
): { context: ProfileKeyCredentialRequestContext; requestHex: string } {
|
|
|
|
const profileKeyArray = base64ToCompatArray(profileKeyBase64);
|
|
|
|
const profileKey = new ProfileKey(profileKeyArray);
|
|
|
|
|
|
|
|
const context = clientZkProfileCipher.createProfileKeyCredentialRequestContext(
|
|
|
|
uuid,
|
|
|
|
profileKey
|
|
|
|
);
|
|
|
|
const request = context.getRequest();
|
|
|
|
const requestArray = request.serialize();
|
|
|
|
|
|
|
|
return {
|
|
|
|
context,
|
|
|
|
requestHex: compatArrayToHex(requestArray),
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2020-09-09 02:25:05 +00:00
|
|
|
export function getAuthCredentialPresentation(
|
|
|
|
clientZkAuthOperations: ClientZkAuthOperations,
|
|
|
|
authCredentialBase64: string,
|
|
|
|
groupSecretParamsBase64: string
|
2021-06-22 14:46:42 +00:00
|
|
|
): Uint8Array {
|
2020-09-09 02:25:05 +00:00
|
|
|
const authCredential = new AuthCredential(
|
|
|
|
base64ToCompatArray(authCredentialBase64)
|
|
|
|
);
|
|
|
|
const secretParams = new GroupSecretParams(
|
|
|
|
base64ToCompatArray(groupSecretParamsBase64)
|
|
|
|
);
|
|
|
|
|
|
|
|
const presentation = clientZkAuthOperations.createAuthCredentialPresentation(
|
|
|
|
secretParams,
|
|
|
|
authCredential
|
|
|
|
);
|
2021-06-22 14:46:42 +00:00
|
|
|
return compatArrayToUint8Array(presentation.serialize());
|
2020-09-09 02:25:05 +00:00
|
|
|
}
|
|
|
|
|
2020-10-06 17:06:34 +00:00
|
|
|
export function createProfileKeyCredentialPresentation(
|
|
|
|
clientZkProfileCipher: ClientZkProfileOperations,
|
|
|
|
profileKeyCredentialBase64: string,
|
|
|
|
groupSecretParamsBase64: string
|
2021-06-22 14:46:42 +00:00
|
|
|
): Uint8Array {
|
2020-10-06 17:06:34 +00:00
|
|
|
const profileKeyCredentialArray = base64ToCompatArray(
|
|
|
|
profileKeyCredentialBase64
|
|
|
|
);
|
|
|
|
const profileKeyCredential = new ProfileKeyCredential(
|
|
|
|
profileKeyCredentialArray
|
|
|
|
);
|
|
|
|
const secretParams = new GroupSecretParams(
|
|
|
|
base64ToCompatArray(groupSecretParamsBase64)
|
|
|
|
);
|
|
|
|
|
|
|
|
const presentation = clientZkProfileCipher.createProfileKeyCredentialPresentation(
|
|
|
|
secretParams,
|
|
|
|
profileKeyCredential
|
|
|
|
);
|
|
|
|
|
2021-06-22 14:46:42 +00:00
|
|
|
return compatArrayToUint8Array(presentation.serialize());
|
2020-10-06 17:06:34 +00:00
|
|
|
}
|
|
|
|
|
2020-09-09 02:25:05 +00:00
|
|
|
export function getClientZkAuthOperations(
|
|
|
|
serverPublicParamsBase64: string
|
|
|
|
): ClientZkAuthOperations {
|
|
|
|
const serverPublicParams = new ServerPublicParams(
|
|
|
|
base64ToCompatArray(serverPublicParamsBase64)
|
|
|
|
);
|
|
|
|
|
|
|
|
return new ClientZkAuthOperations(serverPublicParams);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getClientZkGroupCipher(
|
|
|
|
groupSecretParamsBase64: string
|
|
|
|
): ClientZkGroupCipher {
|
|
|
|
const serverPublicParams = new GroupSecretParams(
|
|
|
|
base64ToCompatArray(groupSecretParamsBase64)
|
|
|
|
);
|
|
|
|
|
|
|
|
return new ClientZkGroupCipher(serverPublicParams);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getClientZkProfileOperations(
|
|
|
|
serverPublicParamsBase64: string
|
|
|
|
): ClientZkProfileOperations {
|
|
|
|
const serverPublicParams = new ServerPublicParams(
|
|
|
|
base64ToCompatArray(serverPublicParamsBase64)
|
|
|
|
);
|
|
|
|
|
|
|
|
return new ClientZkProfileOperations(serverPublicParams);
|
|
|
|
}
|
|
|
|
|
2020-04-15 23:12:28 +00:00
|
|
|
export function handleProfileKeyCredential(
|
|
|
|
clientZkProfileCipher: ClientZkProfileOperations,
|
|
|
|
context: ProfileKeyCredentialRequestContext,
|
|
|
|
responseBase64: string
|
|
|
|
): string {
|
2020-09-09 02:25:05 +00:00
|
|
|
const response = new ProfileKeyCredentialResponse(
|
|
|
|
base64ToCompatArray(responseBase64)
|
|
|
|
);
|
2020-04-15 23:12:28 +00:00
|
|
|
const profileKeyCredential = clientZkProfileCipher.receiveProfileKeyCredential(
|
|
|
|
context,
|
|
|
|
response
|
|
|
|
);
|
|
|
|
|
|
|
|
const credentialArray = profileKeyCredential.serialize();
|
|
|
|
|
|
|
|
return compatArrayToBase64(credentialArray);
|
|
|
|
}
|