@zkp2p/reclaim-witness-sdk
Version:
<div> <div> <img src="https://raw.githubusercontent.com/reclaimprotocol/.github/main/assets/banners/Attestor-Core.png" /> </div> </div>
191 lines • 18.2 kB
JavaScript
Object.defineProperty(exports, "__esModule", { value: true });
exports.processHandshake = processHandshake;
const tls_1 = require("@reclaimprotocol/tls");
const parse_certificate_1 = require("@reclaimprotocol/tls/lib/utils/parse-certificate");
const api_1 = require("../../proto/api");
const utils_1 = require("../../utils");
const RECORD_LENGTH_BYTES = 3;
/**
* Verifies server cert chain and removes handshake messages from transcript
* @param receipt
* @param logger
*/
async function processHandshake(receipt, logger) {
let currentPacketIdx = 0;
let readPacketIdx = 0;
let handshakeData = Uint8Array.from([]);
let packetData;
const handshakeRawMessages = [];
const certificates = [];
let cipherSuite = undefined;
let tlsVersion = undefined;
let serverRandom = undefined;
let clientRandom = undefined;
let serverFinishedIdx = -1;
let clientFinishedIdx = -1;
let certVerified = false;
let hostname = undefined;
let clientChangeCipherSpecMsgIdx = -1;
let serverChangeCipherSpecMsgIdx = -1;
while ((packetData = await readPacket())) {
const { type, content } = packetData;
switch (type) {
case tls_1.SUPPORTED_RECORD_TYPE_MAP.CLIENT_HELLO:
const clientHello = (0, tls_1.parseClientHello)(handshakeRawMessages[0]);
clientRandom = clientHello.serverRandom;
const { SERVER_NAME: sni } = clientHello.extensions;
hostname = sni === null || sni === void 0 ? void 0 : sni.serverName;
if (!hostname) {
throw new Error('client hello has no SNI');
}
break;
case tls_1.SUPPORTED_RECORD_TYPE_MAP.SERVER_HELLO:
const serverHello = await (0, tls_1.parseServerHello)(content);
cipherSuite = serverHello.cipherSuite;
tlsVersion = serverHello.serverTlsVersion;
serverRandom = serverHello.serverRandom;
logger.info({ serverTLSVersion: tlsVersion, cipherSuite }, 'extracted server hello params');
break;
case tls_1.SUPPORTED_RECORD_TYPE_MAP.CERTIFICATE:
const parseResult = (0, tls_1.parseCertificates)(content, { version: tlsVersion });
certificates.push(...parseResult.certificates);
break;
case tls_1.SUPPORTED_RECORD_TYPE_MAP.CERTIFICATE_VERIFY:
const signature = (0, tls_1.parseServerCertificateVerify)(content);
if (!(certificates === null || certificates === void 0 ? void 0 : certificates.length)) {
throw new Error('No provider certificates received');
}
const signatureData = await (0, tls_1.getSignatureDataTls13)(handshakeRawMessages.slice(0, -1), cipherSuite);
await (0, tls_1.verifyCertificateSignature)({
...signature,
publicKey: certificates[0].getPublicKey(),
signatureData,
});
await (0, parse_certificate_1.verifyCertificateChain)(certificates, hostname);
logger.info({ host: hostname }, 'verified provider certificate chain');
certVerified = true;
break;
case tls_1.SUPPORTED_RECORD_TYPE_MAP.SERVER_KEY_SHARE:
if (!(certificates === null || certificates === void 0 ? void 0 : certificates.length)) {
throw new Error('No provider certificates received');
}
const keyShare = await (0, tls_1.processServerKeyShare)(content);
const signatureData12 = await (0, tls_1.getSignatureDataTls12)({
clientRandom: clientRandom,
serverRandom: serverRandom,
curveType: keyShare.publicKeyType,
publicKey: keyShare.publicKey,
});
// verify signature
await (0, tls_1.verifyCertificateSignature)({
signature: keyShare.signatureBytes,
algorithm: keyShare.signatureAlgorithm,
publicKey: certificates[0].getPublicKey(),
signatureData: signatureData12,
});
await (0, parse_certificate_1.verifyCertificateChain)(certificates, hostname);
logger.info({ host: hostname }, 'verified provider certificate chain');
certVerified = true;
break;
case tls_1.SUPPORTED_RECORD_TYPE_MAP.FINISHED:
if (receipt[readPacketIdx].sender === api_1.TranscriptMessageSenderType.TRANSCRIPT_MESSAGE_SENDER_TYPE_CLIENT) {
clientFinishedIdx = readPacketIdx;
}
else {
serverFinishedIdx = readPacketIdx;
}
break;
}
}
if (!certVerified) {
throw new Error('No provider certificates received');
}
if (tlsVersion === 'TLS1_3' && serverFinishedIdx < 0) {
throw new Error('server finished message not found');
}
if (tlsVersion === 'TLS1_2' && (serverChangeCipherSpecMsgIdx < 0 || clientChangeCipherSpecMsgIdx < 0)) {
throw new Error('change cipher spec message not found');
}
async function readPacket(getMoreData = false) {
var _a;
if (currentPacketIdx > (receipt.length - 1)) {
return;
}
if (certVerified && serverFinishedIdx > 0 && clientFinishedIdx > 0) {
return;
}
readPacketIdx = currentPacketIdx;
if (!(handshakeData === null || handshakeData === void 0 ? void 0 : handshakeData.length) || getMoreData) {
let newHandshakeData;
const { message, reveal, sender } = receipt[currentPacketIdx];
const recordHeader = message.slice(0, 5);
const content = getWithoutHeader(message);
if (message[0] === tls_1.PACKET_TYPE['CHANGE_CIPHER_SPEC']) { //skip change cipher spec message
if (sender === api_1.TranscriptMessageSenderType.TRANSCRIPT_MESSAGE_SENDER_TYPE_CLIENT) {
clientChangeCipherSpecMsgIdx = currentPacketIdx;
}
else {
serverChangeCipherSpecMsgIdx = currentPacketIdx;
}
currentPacketIdx++;
return await readPacket();
}
if (message[0] === tls_1.PACKET_TYPE['WRAPPED_RECORD'] ||
(serverChangeCipherSpecMsgIdx > 0 && sender === api_1.TranscriptMessageSenderType.TRANSCRIPT_MESSAGE_SENDER_TYPE_SERVER) ||
(clientChangeCipherSpecMsgIdx > 0 && sender === api_1.TranscriptMessageSenderType.TRANSCRIPT_MESSAGE_SENDER_TYPE_CLIENT)) { // encrypted
if (!tlsVersion || !cipherSuite) {
throw new Error('Could not find cipherSuite to use');
}
if (!((_a = reveal === null || reveal === void 0 ? void 0 : reveal.directReveal) === null || _a === void 0 ? void 0 : _a.key)) {
throw new Error('no direct reveal for handshake packet');
}
const { plaintext } = await (0, utils_1.decryptDirect)(reveal === null || reveal === void 0 ? void 0 : reveal.directReveal, cipherSuite, recordHeader, tlsVersion, content);
newHandshakeData = plaintext;
if (tlsVersion === 'TLS1_3') {
newHandshakeData = newHandshakeData.slice(0, -1);
}
}
else {
newHandshakeData = content;
}
handshakeData = (0, tls_1.concatenateUint8Arrays)([handshakeData, newHandshakeData]);
}
const type = handshakeData[0];
const content = readWithLength(handshakeData.slice(1), RECORD_LENGTH_BYTES);
if (!content) {
logger.warn('missing bytes from packet');
currentPacketIdx++;
return await readPacket(true);
}
const totalLength = 1 + RECORD_LENGTH_BYTES + content.length;
handshakeRawMessages.push(handshakeData.slice(0, totalLength));
handshakeData = handshakeData.slice(totalLength);
if (!handshakeData.length) {
currentPacketIdx++;
}
return { type, content };
}
const nextMsgIndex = Math.max(serverFinishedIdx, clientFinishedIdx) + 1;
return {
tlsVersion: tlsVersion,
cipherSuite: cipherSuite,
hostname: hostname,
nextMsgIndex
};
}
function getWithoutHeader(message) {
// strip the record header (xx 03 03 xx xx)
return message.slice(5);
}
function readWithLength(data, lengthBytes = 2) {
const dataView = (0, tls_1.uint8ArrayToDataView)(data);
const length = lengthBytes === 1
? dataView.getUint8(0)
: dataView.getUint16(lengthBytes === 3 ? 1 : 0);
if (data.length < lengthBytes + length) {
return undefined;
}
return data.slice(lengthBytes, lengthBytes + length);
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"process-handshake.js","sourceRoot":"","sources":["../../../src/server/utils/process-handshake.ts"],"names":[],"mappings":"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"}
;