PHP WebShell
Текущая директория: /opt/BitGoJS/modules/bitgo/dist/test/v2/unit/internal/tssUtils/ecdsaMPCv2
Просмотр файла: createKeychains.js
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const assert = require("assert");
const nock = require("nock");
const openpgp = require("openpgp");
const crypto = require("crypto");
const sdk_test_1 = require("@bitgo/sdk-test");
const sdk_core_1 = require("@bitgo/sdk-core");
const sdk_lib_mpc_1 = require("@bitgo/sdk-lib-mpc");
const public_types_1 = require("@bitgo/public-types");
const src_1 = require("../../../../../../src");
const v1Fixtures = require("./fixtures/mpcv1KeyShares");
describe('TSS Ecdsa MPCv2 Utils:', async function () {
const coinName = 'hteth';
const walletId = '5b34252f1bf349930e34020a00000000';
const enterpriseId = '6449153a6f6bc20006d66771cdbe15d3';
let storedUserCommitment2;
let storedBackupCommitment2;
let storedBitgoCommitment2;
let bgUrl;
let tssUtils;
let wallet;
let bitgo;
let baseCoin;
let bitGoGgpKey;
let constants;
let bitgoGpgPrvKey;
let userGpgPubKey;
let backupGpgPubKey;
let bitgoGpgPubKey;
beforeEach(async function () {
nock.cleanAll();
await nockGetBitgoPublicKeyBasedOnFeatureFlags(coinName, enterpriseId, bitGoGgpKey);
nock(bgUrl).get('/api/v1/client/constants').times(16).reply(200, { ttl: 3600, constants });
});
before(async function () {
bitGoGgpKey = await openpgp.generateKey({
userIDs: [
{
name: 'bitgo',
email: 'bitgo@test.com',
},
],
curve: 'secp256k1',
});
constants = {
mpc: {
bitgoPublicKey: bitGoGgpKey.publicKey,
bitgoMPCv2PublicKey: bitGoGgpKey.publicKey,
},
};
bitgoGpgPubKey = {
partyId: 2,
gpgKey: bitGoGgpKey.publicKey,
};
bitgoGpgPrvKey = {
partyId: 2,
gpgKey: bitGoGgpKey.privateKey,
};
bitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' });
bitgo.initializeTestVars();
baseCoin = bitgo.coin(coinName);
bgUrl = sdk_core_1.common.Environments[bitgo.getEnv()].uri;
const walletData = {
id: walletId,
enterprise: enterpriseId,
coin: coinName,
coinSpecific: {},
multisigType: 'tss',
};
wallet = new sdk_core_1.Wallet(bitgo, baseCoin, walletData);
tssUtils = new sdk_core_1.ECDSAUtils.EcdsaMPCv2Utils(bitgo, baseCoin, wallet);
});
after(function () {
nock.cleanAll();
});
describe('Retrofit MPCv1 to MPCv2 keys', async function () {
it('should generate TSS MPCv2 keys from MPCv1 keys and sign a message', async function () {
const retrofitData = tssUtils.getMpcV2RetrofitDataFromMpcV1Keys({
mpcv1UserKeyShare: JSON.stringify(v1Fixtures.mockUserSigningMaterial),
mpcv1BackupKeyShare: JSON.stringify(v1Fixtures.mockBackupSigningMaterial),
});
const bitgoRetrofitData = {
xiList: retrofitData.mpcv2UserKeyShare.xiList,
xShare: v1Fixtures.mockBitGoShareSigningMaterial.xShare,
};
const [user, backup, bitgo] = await sdk_lib_mpc_1.DklsUtils.generateDKGKeyShares(retrofitData.mpcv2UserKeyShare, retrofitData.mpcv2BackupKeyShare, bitgoRetrofitData);
assert.ok(bitgo.getKeyShare());
const messageToSign = crypto.createHash('sha256').update(Buffer.from('ffff', 'hex')).digest();
const derivationPath = 'm/999/988/0/0';
const signature = await sdk_lib_mpc_1.DklsUtils.executeTillRound(5, new sdk_lib_mpc_1.DklsDsg.Dsg(user.getKeyShare(), 0, derivationPath, messageToSign), new sdk_lib_mpc_1.DklsDsg.Dsg(backup.getKeyShare(), 1, derivationPath, messageToSign));
const convertedSignature = sdk_lib_mpc_1.DklsUtils.verifyAndConvertDklsSignature(Buffer.from('ffff', 'hex'), signature, v1Fixtures.mockBitGoShareSigningMaterial.xShare.y + v1Fixtures.mockBitGoShareSigningMaterial.xShare.chaincode, derivationPath);
assert.ok(convertedSignature);
convertedSignature.split(':').length.should.equal(4);
});
});
describe('TSS key chains', async function () {
it('should generate TSS MPCv2 keys', async function () {
const bitgoSession = new sdk_lib_mpc_1.DklsDkg.Dkg(3, 2, 2);
const round1Nock = await nockKeyGenRound1(bitgoSession, 1);
const round2Nock = await nockKeyGenRound2(bitgoSession, 1);
const round3Nock = await nockKeyGenRound3(bitgoSession, 1);
const addKeyNock = await nockAddKeyChain(coinName, 3);
const params = {
passphrase: 'test',
enterprise: enterpriseId,
originalPasscodeEncryptionCode: '123456',
};
const { userKeychain, backupKeychain, bitgoKeychain } = await tssUtils.createKeychains(params);
assert.ok(round1Nock.isDone());
assert.ok(round2Nock.isDone());
assert.ok(round3Nock.isDone());
assert.ok(addKeyNock.isDone());
assert.ok(userKeychain);
assert.equal(userKeychain.source, 'user');
assert.ok(userKeychain.commonKeychain);
assert.ok(sdk_core_1.ECDSAUtils.EcdsaMPCv2Utils.validateCommonKeychainPublicKey(userKeychain.commonKeychain));
assert.ok(userKeychain.encryptedPrv);
assert.ok(bitgo.decrypt({ input: userKeychain.encryptedPrv, password: params.passphrase }));
assert.ok(backupKeychain);
assert.equal(backupKeychain.source, 'backup');
assert.ok(backupKeychain.commonKeychain);
assert.ok(sdk_core_1.ECDSAUtils.EcdsaMPCv2Utils.validateCommonKeychainPublicKey(backupKeychain.commonKeychain));
assert.ok(backupKeychain.encryptedPrv);
assert.ok(bitgo.decrypt({ input: backupKeychain.encryptedPrv, password: params.passphrase }));
assert.ok(bitgoKeychain);
assert.equal(bitgoKeychain.source, 'bitgo');
});
it('should generate TSS MPCv2 keys for retrofit', async function () {
const xiList = [
Array.from((0, sdk_lib_mpc_1.bigIntToBufferBE)(BigInt(1), 32)),
Array.from((0, sdk_lib_mpc_1.bigIntToBufferBE)(BigInt(2), 32)),
Array.from((0, sdk_lib_mpc_1.bigIntToBufferBE)(BigInt(3), 32)),
];
const bitgoRetrofitData = {
xiList,
xShare: v1Fixtures.mockBitGoShareSigningMaterial.xShare,
};
const bitgoSession = new sdk_lib_mpc_1.DklsDkg.Dkg(3, 2, sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.BITGO, undefined, bitgoRetrofitData);
const round1Nock = await nockKeyGenRound1(bitgoSession, 1);
const round2Nock = await nockKeyGenRound2(bitgoSession, 1);
const round3Nock = await nockKeyGenRound3(bitgoSession, 1);
const addKeyNock = await nockAddKeyChain(coinName, 3);
const params = {
passphrase: 'test',
enterprise: enterpriseId,
originalPasscodeEncryptionCode: '123456',
retrofit: {
decryptedUserKey: JSON.stringify(v1Fixtures.mockUserSigningMaterial),
decryptedBackupKey: JSON.stringify(v1Fixtures.mockBackupSigningMaterial),
walletId: '123',
},
};
const { userKeychain, backupKeychain, bitgoKeychain } = await tssUtils.createKeychains(params);
assert.ok(round1Nock.isDone());
assert.ok(round2Nock.isDone());
assert.ok(round3Nock.isDone());
assert.ok(addKeyNock.isDone());
assert.ok(userKeychain);
assert.equal(userKeychain.source, 'user');
assert.ok(userKeychain.commonKeychain);
assert.ok(sdk_core_1.ECDSAUtils.EcdsaMPCv2Utils.validateCommonKeychainPublicKey(userKeychain.commonKeychain));
assert.ok(userKeychain.encryptedPrv);
assert.ok(bitgo.decrypt({ input: userKeychain.encryptedPrv, password: params.passphrase }));
assert.ok(backupKeychain);
assert.equal(backupKeychain.source, 'backup');
assert.ok(backupKeychain.commonKeychain);
assert.ok(sdk_core_1.ECDSAUtils.EcdsaMPCv2Utils.validateCommonKeychainPublicKey(backupKeychain.commonKeychain));
assert.ok(backupKeychain.encryptedPrv);
assert.ok(bitgo.decrypt({ input: backupKeychain.encryptedPrv, password: params.passphrase }));
assert.ok(bitgoKeychain);
assert.equal(bitgoKeychain.source, 'bitgo');
});
it('should create TSS key chains', async function () {
const nockPromises = [
nockKeychain({
coin: coinName,
keyChain: { id: '1', pub: '1', type: 'tss', reducedEncryptedPrv: '' },
source: 'user',
}),
nockKeychain({
coin: coinName,
keyChain: { id: '2', pub: '2', type: 'tss', reducedEncryptedPrv: '' },
source: 'backup',
}),
nockKeychain({
coin: coinName,
keyChain: { id: '3', pub: '3', type: 'tss', reducedEncryptedPrv: '' },
source: 'bitgo',
}),
];
const [nockedUserKeychain, nockedBackupKeychain, nockedBitGoKeychain] = await Promise.all(nockPromises);
const bitgoKeychainPromise = tssUtils.createParticipantKeychain(sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.BITGO, 'test');
const usersKeychainPromise = tssUtils.createParticipantKeychain(sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.USER, 'test', Buffer.from('test'), Buffer.from('test'), 'passphrase', 'test');
const backupKeychainPromise = tssUtils.createParticipantKeychain(sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.BACKUP, 'test', Buffer.from('test'), Buffer.from('test'), 'passphrase', 'test');
const [userKeychain, backupKeychain, bitgoKeychain] = await Promise.all([
usersKeychainPromise,
backupKeychainPromise,
bitgoKeychainPromise,
]);
({ ...userKeychain, reducedEncryptedPrv: '' }).should.deepEqual(nockedUserKeychain);
({ ...backupKeychain, reducedEncryptedPrv: '' }).should.deepEqual(nockedBackupKeychain);
({ ...bitgoKeychain, reducedEncryptedPrv: '' }).should.deepEqual(nockedBitGoKeychain);
});
it('should create TSS MPCv2 key chains with OVCs', async function () {
const MPCv2SMCUtils = new sdk_core_1.ECDSAUtils.MPCv2SMCUtils(bitgo, baseCoin);
const bitgoSession = new sdk_lib_mpc_1.DklsDkg.Dkg(3, 2, 2);
const round1Nock = await nockKeyGenRound1(bitgoSession, 1);
const round2Nock = await nockKeyGenRound2(bitgoSession, 1);
const round3Nock = await nockKeyGenRound3(bitgoSession, 1);
const addKeyNock = await nockAddKeyChain(coinName, 3);
// OVC 1 - User GPG key
const userGgpKey = await openpgp.generateKey({
userIDs: [
{
name: 'user',
email: 'user@test.com',
},
],
curve: 'secp256k1',
});
const userGpgPrvKey = {
partyId: 0,
gpgKey: userGgpKey.privateKey,
};
// Round 1 User
const userSession = new sdk_lib_mpc_1.DklsDkg.Dkg(3, 2, 0);
let OVC1ToOVC2Round1Payload;
{
const userBroadcastMsg1Unsigned = await userSession.initDkg();
const userMsgs1Signed = await sdk_lib_mpc_1.DklsComms.encryptAndAuthOutgoingMessages({ broadcastMessages: [sdk_lib_mpc_1.DklsTypes.serializeBroadcastMessage(userBroadcastMsg1Unsigned)], p2pMessages: [] }, [], [userGpgPrvKey]);
const userMsg1 = userMsgs1Signed.broadcastMessages.find((m) => m.from === 0);
assert(userMsg1, 'userMsg1 not found');
OVC1ToOVC2Round1Payload = {
tssVersion: '0.0.1',
walletType: public_types_1.WalletTypeEnum.tss,
coin: 'eth',
state: public_types_1.KeyCreationMPCv2StateEnum.WaitingForOVC2Round1Data,
ovc: {
[public_types_1.OVCIndexEnum.ONE]: {
gpgPubKey: userGgpKey.publicKey,
ovcMsg1: userMsg1,
},
},
};
}
// OVC 2 - Backup GPG key
const backupGgpKey = await openpgp.generateKey({
userIDs: [
{
name: 'backup',
email: 'backup@test.com',
},
],
curve: 'secp256k1',
});
const backupGpgPrvKey = {
partyId: 1,
gpgKey: backupGgpKey.privateKey,
};
// Round 1 Backup
const backupSession = new sdk_lib_mpc_1.DklsDkg.Dkg(3, 2, 1);
let OVC2ToBitgoRound1Payload;
{
assert(OVC1ToOVC2Round1Payload.state === 0, 'OVC1ToOVC2Round1Payload.state should be 0');
const backupBroadcastMsg1Unsigned = await backupSession.initDkg();
const backupMsgs1Signed = await sdk_lib_mpc_1.DklsComms.encryptAndAuthOutgoingMessages({ broadcastMessages: [sdk_lib_mpc_1.DklsTypes.serializeBroadcastMessage(backupBroadcastMsg1Unsigned)], p2pMessages: [] }, [], [backupGpgPrvKey]);
const backupMsg1 = backupMsgs1Signed.broadcastMessages.find((m) => m.from === 1);
assert(backupMsg1, 'backupMsg1 not found');
OVC2ToBitgoRound1Payload = {
...OVC1ToOVC2Round1Payload,
state: public_types_1.KeyCreationMPCv2StateEnum.WaitingForBitgoRound1Data,
ovc: {
...OVC1ToOVC2Round1Payload.ovc,
[public_types_1.OVCIndexEnum.TWO]: {
gpgPubKey: backupGgpKey.publicKey,
ovcMsg1: backupMsg1,
},
},
};
}
// Round 1 BitGo
const bitgoToOVC1Round1Payload = await MPCv2SMCUtils.keyGenRound1('testId', OVC2ToBitgoRound1Payload);
// Round 2 User
let OVC1ToOVC2Round2Payload;
{
assert(bitgoToOVC1Round1Payload.state === 2, 'bitgoToOVC1Round1Payload.state should be 2');
const toUserRound1BroadcastMessages = await sdk_lib_mpc_1.DklsComms.decryptAndVerifyIncomingMessages({
p2pMessages: [],
broadcastMessages: [
bitgoToOVC1Round1Payload.ovc[public_types_1.OVCIndexEnum.TWO].ovcMsg1,
bitgoToOVC1Round1Payload.platform.bitgoMsg1,
],
}, [bitgoGpgPubKey, { partyId: 1, gpgKey: bitgoToOVC1Round1Payload.ovc[public_types_1.OVCIndexEnum.TWO].gpgPubKey }], [userGpgPrvKey]);
const userRound2P2PMessages = userSession.handleIncomingMessages({
p2pMessages: [],
broadcastMessages: toUserRound1BroadcastMessages.broadcastMessages.map(sdk_lib_mpc_1.DklsTypes.deserializeBroadcastMessage),
});
const userRound2Messages = await sdk_lib_mpc_1.DklsComms.encryptAndAuthOutgoingMessages(sdk_lib_mpc_1.DklsTypes.serializeMessages(userRound2P2PMessages), [{ partyId: 1, gpgKey: bitgoToOVC1Round1Payload.ovc[public_types_1.OVCIndexEnum.TWO].gpgPubKey }, bitgoGpgPubKey], [userGpgPrvKey]);
const userToBackupMsg2 = userRound2Messages.p2pMessages.find((m) => m.from === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.USER && m.to === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.BACKUP);
assert(userToBackupMsg2, 'userToBackupMsg2 not found');
const userToBitgoMsg2 = userRound2Messages.p2pMessages.find((m) => m.from === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.USER && m.to === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.BITGO);
assert(userToBitgoMsg2, 'userToBitgoMsg2 not found');
OVC1ToOVC2Round2Payload = {
...bitgoToOVC1Round1Payload,
state: public_types_1.KeyCreationMPCv2StateEnum.WaitingForOVC2Round2Data,
ovc: {
...bitgoToOVC1Round1Payload.ovc,
[public_types_1.OVCIndexEnum.ONE]: Object.assign(bitgoToOVC1Round1Payload.ovc[public_types_1.OVCIndexEnum.ONE], {
ovcToBitgoMsg2: userToBitgoMsg2,
ovcToOvcMsg2: userToBackupMsg2,
}),
},
};
}
// Round 2 Backup
let OVC2ToBitgoRound2Payload;
{
assert(OVC1ToOVC2Round2Payload.state === 3, 'bitgoToOVC1Round1Payload.state should be 3');
const toBackupRound1BroadcastMessages = await sdk_lib_mpc_1.DklsComms.decryptAndVerifyIncomingMessages({
p2pMessages: [],
broadcastMessages: [
bitgoToOVC1Round1Payload.ovc[public_types_1.OVCIndexEnum.ONE].ovcMsg1,
bitgoToOVC1Round1Payload.platform.bitgoMsg1,
],
}, [bitgoGpgPubKey, { partyId: 0, gpgKey: OVC1ToOVC2Round2Payload.ovc[public_types_1.OVCIndexEnum.ONE].gpgPubKey }], [backupGpgPrvKey]);
const backupRound2P2PMessages = backupSession.handleIncomingMessages({
p2pMessages: [],
broadcastMessages: toBackupRound1BroadcastMessages.broadcastMessages.map(sdk_lib_mpc_1.DklsTypes.deserializeBroadcastMessage),
});
const backupRound2Messages = await sdk_lib_mpc_1.DklsComms.encryptAndAuthOutgoingMessages(sdk_lib_mpc_1.DklsTypes.serializeMessages(backupRound2P2PMessages), [{ partyId: 0, gpgKey: bitgoToOVC1Round1Payload.ovc[public_types_1.OVCIndexEnum.ONE].gpgPubKey }, bitgoGpgPubKey], [backupGpgPrvKey]);
const backupToUserMsg2 = backupRound2Messages.p2pMessages.find((m) => m.from === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.BACKUP && m.to === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.USER);
assert(backupToUserMsg2, 'backupToUserMsg2 not found');
const backupToBitgoMsg2 = backupRound2Messages.p2pMessages.find((m) => m.from === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.BACKUP && m.to === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.BITGO);
assert(backupToBitgoMsg2, 'backupToBitgoMsg2 not found');
OVC2ToBitgoRound2Payload = {
...OVC1ToOVC2Round2Payload,
state: public_types_1.KeyCreationMPCv2StateEnum.WaitingForBitgoRound2Data,
ovc: {
...OVC1ToOVC2Round2Payload.ovc,
[public_types_1.OVCIndexEnum.TWO]: Object.assign(OVC1ToOVC2Round2Payload.ovc[public_types_1.OVCIndexEnum.TWO], {
ovcToBitgoMsg2: backupToBitgoMsg2,
ovcToOvcMsg2: backupToUserMsg2,
}),
},
};
}
// Round 2 BitGo
// call bitgo round 2
const bitgoToOVC1Round2Payload = await MPCv2SMCUtils.keyGenRound2('testId', OVC2ToBitgoRound2Payload);
// Round 3A User
let OVC1ToOVC2Round3Payload;
{
assert(bitgoToOVC1Round2Payload.state === 5, 'bitgoToOVC1Round2Payload.state should be 5');
const toUserRound2P2PMessages = await sdk_lib_mpc_1.DklsComms.decryptAndVerifyIncomingMessages({
p2pMessages: [
bitgoToOVC1Round2Payload.ovc[public_types_1.OVCIndexEnum.TWO].ovcToOvcMsg2,
bitgoToOVC1Round2Payload.platform.ovc[public_types_1.OVCIndexEnum.ONE].bitgoToOvcMsg2,
],
broadcastMessages: [],
}, [bitgoGpgPubKey, { partyId: 1, gpgKey: bitgoToOVC1Round2Payload.ovc[public_types_1.OVCIndexEnum.TWO].gpgPubKey }], [userGpgPrvKey]);
const userRound3AP2PMessages = userSession.handleIncomingMessages({
p2pMessages: toUserRound2P2PMessages.p2pMessages.map(sdk_lib_mpc_1.DklsTypes.deserializeP2PMessage),
broadcastMessages: [],
}).p2pMessages;
const userRound3AMessages = await sdk_lib_mpc_1.DklsComms.encryptAndAuthOutgoingMessages(sdk_lib_mpc_1.DklsTypes.serializeMessages({
p2pMessages: userRound3AP2PMessages,
broadcastMessages: [],
}), [{ partyId: 1, gpgKey: bitgoToOVC1Round2Payload.ovc[public_types_1.OVCIndexEnum.TWO].gpgPubKey }, bitgoGpgPubKey], [userGpgPrvKey]);
const userToBitgoMsg3 = userRound3AMessages.p2pMessages.find((m) => m.from === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.USER && m.to === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.BITGO);
assert(userToBitgoMsg3, 'userToBitgoMsg3 not found');
const userToBackupMsg3 = userRound3AMessages.p2pMessages.find((m) => m.from === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.USER && m.to === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.BACKUP);
assert(userToBackupMsg3, 'userToBackupMsg3 not found');
OVC1ToOVC2Round3Payload = {
...bitgoToOVC1Round2Payload,
state: public_types_1.KeyCreationMPCv2StateEnum.WaitingForOVC2Round3Data,
ovc: {
...bitgoToOVC1Round2Payload.ovc,
[public_types_1.OVCIndexEnum.ONE]: Object.assign(bitgoToOVC1Round2Payload.ovc[public_types_1.OVCIndexEnum.ONE], {
ovcToBitgoMsg3: userToBitgoMsg3,
ovcToOvcMsg3: userToBackupMsg3,
}),
},
};
}
// Round 3 Backup
let OVC2ToOVC1Round3Payload;
{
assert(OVC1ToOVC2Round3Payload.state === 6, 'OVC1ToOVC2Round3Payload.state should be 6');
const toBackupRound3P2PMessages = await sdk_lib_mpc_1.DklsComms.decryptAndVerifyIncomingMessages({
p2pMessages: [
OVC1ToOVC2Round3Payload.ovc[public_types_1.OVCIndexEnum.ONE].ovcToOvcMsg2,
OVC1ToOVC2Round3Payload.platform.ovc[public_types_1.OVCIndexEnum.TWO].bitgoToOvcMsg2,
],
broadcastMessages: [],
}, [bitgoGpgPubKey, { partyId: 0, gpgKey: OVC1ToOVC2Round3Payload.ovc[public_types_1.OVCIndexEnum.ONE].gpgPubKey }], [backupGpgPrvKey]);
const backupRound3P2PMessages = backupSession.handleIncomingMessages({
p2pMessages: toBackupRound3P2PMessages.p2pMessages.map(sdk_lib_mpc_1.DklsTypes.deserializeP2PMessage),
broadcastMessages: [],
});
const backupRound3Messages = await sdk_lib_mpc_1.DklsComms.encryptAndAuthOutgoingMessages(sdk_lib_mpc_1.DklsTypes.serializeMessages(backupRound3P2PMessages), [{ partyId: 0, gpgKey: OVC1ToOVC2Round3Payload.ovc[public_types_1.OVCIndexEnum.ONE].gpgPubKey }, bitgoGpgPubKey], [backupGpgPrvKey]);
const backupToBitgoMsg3 = backupRound3Messages.p2pMessages.find((m) => m.from === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.BACKUP && m.to === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.BITGO);
assert(backupToBitgoMsg3, 'backupToBitgoMsg3 not found');
const backupToUserMsg3 = backupRound3Messages.p2pMessages.find((m) => m.from === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.BACKUP && m.to === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.USER);
assert(backupToUserMsg3, 'backupToUserMsg3 not found');
const toBackupRound3Messages = await sdk_lib_mpc_1.DklsComms.decryptAndVerifyIncomingMessages({
p2pMessages: [
{
...OVC1ToOVC2Round3Payload.ovc[public_types_1.OVCIndexEnum.ONE].ovcToOvcMsg3,
commitment: OVC1ToOVC2Round3Payload.ovc[public_types_1.OVCIndexEnum.ONE].ovcToOvcMsg2.commitment,
},
{
...OVC1ToOVC2Round3Payload.platform.ovc[public_types_1.OVCIndexEnum.TWO].bitgoToOvcMsg3,
commitment: OVC1ToOVC2Round3Payload.platform.bitgoCommitment2,
},
],
broadcastMessages: [],
}, [bitgoGpgPubKey, { partyId: 0, gpgKey: OVC1ToOVC2Round3Payload.ovc[public_types_1.OVCIndexEnum.ONE].gpgPubKey }], [backupGpgPrvKey]);
const backupRound4Messages = backupSession.handleIncomingMessages({
p2pMessages: toBackupRound3Messages.p2pMessages.map(sdk_lib_mpc_1.DklsTypes.deserializeP2PMessage),
broadcastMessages: [],
}).broadcastMessages;
const backupRound4BroadcastMessages = await sdk_lib_mpc_1.DklsComms.encryptAndAuthOutgoingMessages(sdk_lib_mpc_1.DklsTypes.serializeMessages({
p2pMessages: [],
broadcastMessages: backupRound4Messages,
}), [], [backupGpgPrvKey]);
const backupMsg4 = backupRound4BroadcastMessages.broadcastMessages.find((m) => m.from === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.BACKUP);
assert(backupMsg4, 'backupMsg4 not found');
OVC2ToOVC1Round3Payload = {
...OVC1ToOVC2Round3Payload,
state: public_types_1.KeyCreationMPCv2StateEnum.WaitingForOVC1Round3bData,
ovc: {
...OVC1ToOVC2Round3Payload.ovc,
[public_types_1.OVCIndexEnum.TWO]: Object.assign(OVC1ToOVC2Round3Payload.ovc[public_types_1.OVCIndexEnum.TWO], {
ovcToOvcMsg3: backupToUserMsg3,
ovcToBitgoMsg3: backupToBitgoMsg3,
ovcMsg4: backupMsg4,
}),
},
};
}
// Round 3B User
let OVC1ToBitgoRound3BPayload;
{
assert(OVC2ToOVC1Round3Payload.state === 7, 'OVC2ToOVC1Round3Payload.state should be 7');
const toUserRound4Messages = await sdk_lib_mpc_1.DklsComms.decryptAndVerifyIncomingMessages({
p2pMessages: [
{
...OVC2ToOVC1Round3Payload.ovc[public_types_1.OVCIndexEnum.TWO].ovcToOvcMsg3,
commitment: OVC2ToOVC1Round3Payload.ovc[public_types_1.OVCIndexEnum.TWO].ovcToOvcMsg2.commitment,
},
{
...OVC2ToOVC1Round3Payload.platform.ovc[public_types_1.OVCIndexEnum.ONE].bitgoToOvcMsg3,
commitment: OVC2ToOVC1Round3Payload.platform.bitgoCommitment2,
},
],
broadcastMessages: [],
}, [bitgoGpgPubKey, { partyId: 1, gpgKey: OVC2ToOVC1Round3Payload.ovc[public_types_1.OVCIndexEnum.TWO].gpgPubKey }], [userGpgPrvKey]);
const userRound4BroadcastMessages = userSession.handleIncomingMessages({
p2pMessages: toUserRound4Messages.p2pMessages.map(sdk_lib_mpc_1.DklsTypes.deserializeP2PMessage),
broadcastMessages: [],
}).broadcastMessages;
assert(userRound4BroadcastMessages.length === 1, 'userRound4BroadcastMessages length should be 1');
const userRound4Messages = await sdk_lib_mpc_1.DklsComms.encryptAndAuthOutgoingMessages(sdk_lib_mpc_1.DklsTypes.serializeMessages({
p2pMessages: [],
broadcastMessages: userRound4BroadcastMessages,
}), [], [userGpgPrvKey]);
const userMsg4 = userRound4Messages.broadcastMessages.find((m) => m.from === sdk_core_1.ECDSAUtils.MPCv2PartiesEnum.USER);
assert(userMsg4, 'userMsg4 not found');
OVC1ToBitgoRound3BPayload = {
...OVC2ToOVC1Round3Payload,
state: public_types_1.KeyCreationMPCv2StateEnum.WaitingForBitgoRound3Data,
ovc: {
...OVC2ToOVC1Round3Payload.ovc,
[public_types_1.OVCIndexEnum.ONE]: Object.assign(OVC2ToOVC1Round3Payload.ovc[public_types_1.OVCIndexEnum.ONE], {
ovcMsg4: userMsg4,
}),
},
};
}
// Round 3 BitGo
// creates bitgo keychain
const bitgoToOVC1Round3Payload = await MPCv2SMCUtils.keyGenRound3('testId', OVC1ToBitgoRound3BPayload);
// Round 4 User
let userCommonKeychain;
let OVC1ToOVC2Round4Payload;
{
assert(bitgoToOVC1Round3Payload.state === 9, 'bitgoToOVC1Round3Payload.state should be 9');
assert(bitgoToOVC1Round3Payload.bitGoKeyId, 'bitgoToOVC1Round3Payload.bitGoKeyId not found');
const toUserBitgoRound3Msg = await sdk_lib_mpc_1.DklsComms.decryptAndVerifyIncomingMessages({
p2pMessages: [],
broadcastMessages: [
bitgoToOVC1Round3Payload.ovc[public_types_1.OVCIndexEnum.TWO].ovcMsg4,
bitgoToOVC1Round3Payload.platform.bitgoMsg4,
],
}, [bitgoGpgPubKey, { partyId: 1, gpgKey: bitgoToOVC1Round3Payload.ovc[public_types_1.OVCIndexEnum.TWO].gpgPubKey }], [userGpgPrvKey]);
userSession.handleIncomingMessages({
p2pMessages: [],
broadcastMessages: toUserBitgoRound3Msg.broadcastMessages.map(sdk_lib_mpc_1.DklsTypes.deserializeBroadcastMessage),
});
const userPrivateMaterial = userSession.getKeyShare();
userCommonKeychain = sdk_lib_mpc_1.DklsTypes.getCommonKeychain(userPrivateMaterial);
assert.equal(bitgoToOVC1Round3Payload.platform.commonKeychain, userCommonKeychain, 'User and Bitgo Common keychains do not match');
const userPrv = userPrivateMaterial.toString('base64');
assert(userPrv, 'userPrv not found');
OVC1ToOVC2Round4Payload = {
bitgoKeyId: bitgoToOVC1Round3Payload.bitGoKeyId,
...bitgoToOVC1Round3Payload,
state: public_types_1.KeyCreationMPCv2StateEnum.WaitingForOVC2GenerateKey,
};
}
// Round 4 Backup
let backupCommonKeychain;
{
assert(OVC1ToOVC2Round4Payload.state === 10, 'OVC1ToOVC2Round4Payload.state should be 10');
assert(OVC1ToOVC2Round4Payload.bitgoKeyId, 'OVC1ToOVC2Round4Payload.bitGoKeyId not found');
const toBackupBitgoRound3Msg = await sdk_lib_mpc_1.DklsComms.decryptAndVerifyIncomingMessages({
p2pMessages: [],
broadcastMessages: [
OVC1ToOVC2Round4Payload.ovc[public_types_1.OVCIndexEnum.ONE].ovcMsg4,
OVC1ToOVC2Round4Payload.platform.bitgoMsg4,
],
}, [bitgoGpgPubKey, { partyId: 0, gpgKey: OVC1ToOVC2Round4Payload.ovc[public_types_1.OVCIndexEnum.ONE].gpgPubKey }], [backupGpgPrvKey]);
backupSession.handleIncomingMessages({
p2pMessages: [],
broadcastMessages: toBackupBitgoRound3Msg.broadcastMessages.map(sdk_lib_mpc_1.DklsTypes.deserializeBroadcastMessage),
});
const backupPrivateMaterial = backupSession.getKeyShare();
backupCommonKeychain = sdk_lib_mpc_1.DklsTypes.getCommonKeychain(backupPrivateMaterial);
assert.equal(OVC1ToOVC2Round4Payload.platform.commonKeychain, backupCommonKeychain, 'Backup and Bitgo Common keychains do not match');
const backupPrv = backupPrivateMaterial.toString('base64');
assert(backupPrv, 'backupPrv not found');
}
// Round 4 BitGo
// creates user and backup keychain
const keychains = await MPCv2SMCUtils.uploadClientKeys(bitgoToOVC1Round3Payload.bitGoKeyId, userCommonKeychain, backupCommonKeychain);
assert.deepEqual(keychains.userKeychain, {
commonKeychain: userCommonKeychain,
type: 'tss',
source: 'user',
id: 'user',
});
assert.deepEqual(keychains.backupKeychain, {
commonKeychain: backupCommonKeychain,
type: 'tss',
source: 'backup',
id: 'backup',
});
assert.ok(round1Nock.isDone());
assert.ok(round2Nock.isDone());
assert.ok(round3Nock.isDone());
assert.ok(addKeyNock.isDone());
});
it('should throw for MPCv2 SMC utils if the state is invalid', async function () {
const MPCv2SMCUtils = new sdk_core_1.ECDSAUtils.MPCv2SMCUtils(bitgo, baseCoin);
const invalidPayload = {
state: public_types_1.KeyCreationMPCv2StateEnum.WaitingForOVC1Round2Data,
};
await assert.rejects(async () => await MPCv2SMCUtils.keyGenRound1('testId', invalidPayload), {
message: `Invalid state for round 1, expected: ${public_types_1.KeyCreationMPCv2StateEnum.WaitingForBitgoRound1Data}, got: ${public_types_1.KeyCreationMPCv2StateEnum.WaitingForOVC1Round2Data}`,
});
await assert.rejects(async () => await MPCv2SMCUtils.keyGenRound2('testId', invalidPayload), {
message: `Invalid state for round 2, expected: ${public_types_1.KeyCreationMPCv2StateEnum.WaitingForBitgoRound2Data}, got: ${public_types_1.KeyCreationMPCv2StateEnum.WaitingForOVC1Round2Data}`,
});
await assert.rejects(async () => await MPCv2SMCUtils.keyGenRound3('testId', invalidPayload), {
message: `Invalid state for round 3, expected: ${public_types_1.KeyCreationMPCv2StateEnum.WaitingForBitgoRound3Data}, got: ${public_types_1.KeyCreationMPCv2StateEnum.WaitingForOVC1Round2Data}`,
});
});
});
async function nockKeychain(params, times = 1) {
nock(bgUrl)
.post(`/api/v2/${params.coin}/key`, (body) => {
return body.keyType === 'tss' && body.source === params.source;
})
.times(times)
.reply(200, params.keyChain);
return params.keyChain;
}
async function nockGetBitgoPublicKeyBasedOnFeatureFlags(coin, enterpriseId, bitgoGpgKeyPair) {
const bitgoGPGPublicKeyResponse = {
name: 'irrelevant',
publicKey: bitgoGpgKeyPair.publicKey,
mpcv2PublicKey: bitgoGpgKeyPair.publicKey,
enterpriseId,
};
nock(bgUrl).get(`/api/v2/${coin}/tss/pubkey`).query({ enterpriseId }).reply(200, bitgoGPGPublicKeyResponse);
return bitgoGPGPublicKeyResponse;
}
async function nockKeyGenRound1(bitgoSession, times = 1) {
return nock(bgUrl)
.post(`/api/v2/mpc/generatekey`, (body) => body.round === 'MPCv2-R1')
.times(times)
.reply(200, async (uri, { payload }) => {
const { userGpgPublicKey, backupGpgPublicKey, userMsg1, backupMsg1 } = payload;
userGpgPubKey = {
partyId: 0,
gpgKey: userGpgPublicKey,
};
backupGpgPubKey = {
partyId: 1,
gpgKey: backupGpgPublicKey,
};
const bitgoBroadcastMsg1Unsigned = await bitgoSession.initDkg();
const bitgoMsgs1Signed = await sdk_lib_mpc_1.DklsComms.encryptAndAuthOutgoingMessages({ broadcastMessages: [sdk_lib_mpc_1.DklsTypes.serializeBroadcastMessage(bitgoBroadcastMsg1Unsigned)], p2pMessages: [] }, [], [bitgoGpgPrvKey]);
const bitgoMsg1 = bitgoMsgs1Signed.broadcastMessages.find((m) => m.from === 2);
assert(bitgoMsg1, 'bitgoMsg1 not found');
const round1IncomingMsgs = await sdk_lib_mpc_1.DklsComms.decryptAndVerifyIncomingMessages({
p2pMessages: [],
broadcastMessages: [
{ from: 0, payload: userMsg1 },
{ from: 1, payload: backupMsg1 },
],
}, [userGpgPubKey, backupGpgPubKey], [bitgoGpgPrvKey]);
const round2Messages = sdk_lib_mpc_1.DklsTypes.serializeMessages(bitgoSession.handleIncomingMessages(sdk_lib_mpc_1.DklsTypes.deserializeMessages(round1IncomingMsgs)));
const round2SignedMessages = await sdk_lib_mpc_1.DklsComms.encryptAndAuthOutgoingMessages(round2Messages, [userGpgPubKey, backupGpgPubKey], [bitgoGpgPrvKey]);
const bitgoToUserMsg2 = round2SignedMessages.p2pMessages.find((m) => m.to === 0);
const bitgoToBackupMsg2 = round2SignedMessages.p2pMessages.find((m) => m.to === 1);
assert(bitgoToUserMsg2, 'bitgoToUserMsg2 not found');
assert(bitgoToBackupMsg2, 'bitgoToBackupMsg2 not found');
assert(bitgoToUserMsg2.commitment, 'bitgoToUserMsg2.commitment not found');
storedBitgoCommitment2 = bitgoToUserMsg2?.commitment;
return {
sessionId: 'testid',
bitgoMsg1: { from: 2, ...bitgoMsg1.payload },
bitgoToBackupMsg2: {
from: 2,
to: 1,
encryptedMessage: bitgoToBackupMsg2.payload.encryptedMessage,
signature: bitgoToBackupMsg2.payload.signature,
},
bitgoToUserMsg2: {
from: 2,
to: 0,
encryptedMessage: bitgoToUserMsg2.payload.encryptedMessage,
signature: bitgoToUserMsg2.payload.signature,
},
walletGpgPubKeySigs: 'something',
};
});
}
async function nockKeyGenRound2(bitgoSession, times = 1) {
return nock(bgUrl)
.post(`/api/v2/mpc/generatekey`, (body) => body.round === 'MPCv2-R2')
.times(times)
.reply(200, async (uri, { payload }) => {
const { sessionId, userMsg2, backupMsg2, userCommitment2, backupCommitment2 } = payload;
storedUserCommitment2 = userCommitment2;
storedBackupCommitment2 = backupCommitment2;
const round2IncomingMsgs = await sdk_lib_mpc_1.DklsComms.decryptAndVerifyIncomingMessages({
p2pMessages: [
{
from: userMsg2.from,
to: userMsg2.to,
payload: { signature: userMsg2.signature, encryptedMessage: userMsg2.encryptedMessage },
},
{
from: backupMsg2.from,
to: backupMsg2.to,
payload: { signature: backupMsg2.signature, encryptedMessage: backupMsg2.encryptedMessage },
},
],
broadcastMessages: [],
}, [userGpgPubKey, backupGpgPubKey], [bitgoGpgPrvKey]);
const round3Messages = sdk_lib_mpc_1.DklsTypes.serializeMessages(bitgoSession.handleIncomingMessages(sdk_lib_mpc_1.DklsTypes.deserializeMessages(round2IncomingMsgs)));
const round3SignedMessages = await sdk_lib_mpc_1.DklsComms.encryptAndAuthOutgoingMessages(round3Messages, [userGpgPubKey, backupGpgPubKey], [bitgoGpgPrvKey]);
const bitgoToUserMsg3 = round3SignedMessages.p2pMessages.find((m) => m.to === 0);
const bitgoToBackupMsg3 = round3SignedMessages.p2pMessages.find((m) => m.to === 1);
assert(bitgoToUserMsg3, 'bitgoToUserMsg3 not found');
assert(bitgoToBackupMsg3, 'bitgoToBackupMsg3 not found');
return {
sessionId,
bitgoCommitment2: storedBitgoCommitment2,
bitgoToUserMsg3: {
from: 2,
to: 0,
encryptedMessage: bitgoToUserMsg3.payload.encryptedMessage,
signature: bitgoToUserMsg3.payload.signature,
},
bitgoToBackupMsg3: {
from: 2,
to: 1,
encryptedMessage: bitgoToBackupMsg3.payload.encryptedMessage,
signature: bitgoToBackupMsg3.payload.signature,
},
};
});
}
async function nockKeyGenRound3(bitgoSession, times = 1) {
return nock(bgUrl)
.post(`/api/v2/mpc/generatekey`, (body) => body.round === 'MPCv2-R3')
.times(times)
.reply(200, async (uri, { payload }) => {
const { sessionId, userMsg3, userMsg4, backupMsg3, backupMsg4 } = payload;
const round3IncomingMsgs = await sdk_lib_mpc_1.DklsComms.decryptAndVerifyIncomingMessages({
p2pMessages: [
{
from: userMsg3.from,
to: userMsg3.to,
payload: { signature: userMsg3.signature, encryptedMessage: userMsg3.encryptedMessage },
commitment: storedUserCommitment2,
},
{
from: backupMsg3.from,
to: backupMsg3.to,
payload: { signature: backupMsg3.signature, encryptedMessage: backupMsg3.encryptedMessage },
commitment: storedBackupCommitment2,
},
],
broadcastMessages: [],
}, [userGpgPubKey, backupGpgPubKey], [bitgoGpgPrvKey]);
const round4Messages = sdk_lib_mpc_1.DklsTypes.serializeMessages(bitgoSession.handleIncomingMessages(sdk_lib_mpc_1.DklsTypes.deserializeMessages(round3IncomingMsgs)));
const round4SignedMessages = await sdk_lib_mpc_1.DklsComms.encryptAndAuthOutgoingMessages(round4Messages, [], [bitgoGpgPrvKey]);
const bitgoMsg4 = round4SignedMessages.broadcastMessages.find((m) => m.from === 2);
assert(bitgoMsg4, 'bitgoMsg4 not found');
const round4IncomingMsgs = await sdk_lib_mpc_1.DklsComms.decryptAndVerifyIncomingMessages({
p2pMessages: [],
broadcastMessages: [
{
from: userMsg4.from,
payload: { signature: userMsg4.signature, message: userMsg4.message },
},
{
from: backupMsg4.from,
payload: { signature: backupMsg4.signature, message: backupMsg4.message },
},
],
}, [userGpgPubKey, backupGpgPubKey], []);
bitgoSession.handleIncomingMessages(sdk_lib_mpc_1.DklsTypes.deserializeMessages(round4IncomingMsgs));
const keyShare = bitgoSession.getKeyShare();
const commonKeychain = sdk_lib_mpc_1.DklsTypes.getCommonKeychain(keyShare);
return {
sessionId,
commonKeychain: commonKeychain,
bitgoMsg4: { from: 2, ...bitgoMsg4.payload },
};
});
}
async function nockAddKeyChain(coin, times = 1) {
return nock('https://bitgo.fakeurl')
.post(`/api/v2/${coin}/key`, (body) => body.keyType === 'tss' && body.isMPCv2)
.times(times)
.reply(200, async (uri, requestBody) => {
const key = {
id: requestBody.source,
source: requestBody.source,
type: requestBody.keyType,
commonKeychain: requestBody.commonKeychain,
encryptedPrv: requestBody.encryptedPrv,
};
// nock gets
nock('https://bitgo.fakeurl').get(`/api/v2/${coin}/key/${requestBody.source}`).reply(200, key);
return key;
});
}
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"createKeychains.js","sourceRoot":"","sources":["../../../../../../../test/v2/unit/internal/tssUtils/ecdsaMPCv2/createKeychains.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\nimport * as nock from 'nock';\nimport * as openpgp from 'openpgp';\nimport * as crypto from 'crypto';\n\nimport { TestableBG, TestBitGo } from '@bitgo/sdk-test';\nimport { AddKeychainOptions, BaseCoin, common, ECDSAUtils, Keychain, Wallet } from '@bitgo/sdk-core';\nimport { bigIntToBufferBE, DklsComms, DklsDkg, DklsDsg, DklsTypes, DklsUtils } from '@bitgo/sdk-lib-mpc';\nimport {\n  KeyCreationMPCv2StateEnum,\n  MPCv2KeyGenRound1Request,\n  MPCv2KeyGenRound1Response,\n  MPCv2KeyGenRound2Request,\n  MPCv2KeyGenRound2Response,\n  MPCv2KeyGenRound3Request,\n  MPCv2KeyGenRound3Response,\n  OVC1ToBitgoRound3Payload,\n  OVC1ToOVC2Round1Payload,\n  OVC1ToOVC2Round2Payload,\n  OVC1ToOVC2Round3Payload,\n  OVC2ToBitgoRound1Payload,\n  OVC2ToBitgoRound2Payload,\n  OVC2ToOVC1Round3Payload,\n  OVCIndexEnum,\n  WalletTypeEnum,\n} from '@bitgo/public-types';\nimport { NonEmptyString } from 'io-ts-types';\nimport { BitGo, BitgoGPGPublicKey } from '../../../../../../src';\nimport * as v1Fixtures from './fixtures/mpcv1KeyShares';\n\ndescribe('TSS Ecdsa MPCv2 Utils:', async function () {\n  const coinName = 'hteth';\n  const walletId = '5b34252f1bf349930e34020a00000000';\n  const enterpriseId = '6449153a6f6bc20006d66771cdbe15d3';\n  let storedUserCommitment2: string;\n  let storedBackupCommitment2: string;\n  let storedBitgoCommitment2: string;\n\n  let bgUrl: string;\n  let tssUtils: ECDSAUtils.EcdsaMPCv2Utils;\n  let wallet: Wallet;\n  let bitgo: TestableBG & BitGo;\n  let baseCoin: BaseCoin;\n  let bitGoGgpKey: openpgp.SerializedKeyPair<string> & {\n    revocationCertificate: string;\n  };\n  let constants: { mpc: { bitgoPublicKey: string; bitgoMPCv2PublicKey: string } };\n  let bitgoGpgPrvKey: { partyId: number; gpgKey: string };\n  let userGpgPubKey: { partyId: number; gpgKey: string };\n  let backupGpgPubKey: { partyId: number; gpgKey: string };\n  let bitgoGpgPubKey: { partyId: number; gpgKey: string };\n\n  beforeEach(async function () {\n    nock.cleanAll();\n    await nockGetBitgoPublicKeyBasedOnFeatureFlags(coinName, enterpriseId, bitGoGgpKey);\n    nock(bgUrl).get('/api/v1/client/constants').times(16).reply(200, { ttl: 3600, constants });\n  });\n\n  before(async function () {\n    bitGoGgpKey = await openpgp.generateKey({\n      userIDs: [\n        {\n          name: 'bitgo',\n          email: 'bitgo@test.com',\n        },\n      ],\n      curve: 'secp256k1',\n    });\n    constants = {\n      mpc: {\n        bitgoPublicKey: bitGoGgpKey.publicKey,\n        bitgoMPCv2PublicKey: bitGoGgpKey.publicKey,\n      },\n    };\n\n    bitgoGpgPubKey = {\n      partyId: 2,\n      gpgKey: bitGoGgpKey.publicKey,\n    };\n\n    bitgoGpgPrvKey = {\n      partyId: 2,\n      gpgKey: bitGoGgpKey.privateKey,\n    };\n\n    bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n    bitgo.initializeTestVars();\n\n    baseCoin = bitgo.coin(coinName);\n\n    bgUrl = common.Environments[bitgo.getEnv()].uri;\n\n    const walletData = {\n      id: walletId,\n      enterprise: enterpriseId,\n      coin: coinName,\n      coinSpecific: {},\n      multisigType: 'tss',\n    };\n    wallet = new Wallet(bitgo, baseCoin, walletData);\n    tssUtils = new ECDSAUtils.EcdsaMPCv2Utils(bitgo, baseCoin, wallet);\n  });\n\n  after(function () {\n    nock.cleanAll();\n  });\n\n  describe('Retrofit MPCv1 to MPCv2 keys', async function () {\n    it('should generate TSS MPCv2 keys from MPCv1 keys and sign a message', async function () {\n      const retrofitData = tssUtils.getMpcV2RetrofitDataFromMpcV1Keys({\n        mpcv1UserKeyShare: JSON.stringify(v1Fixtures.mockUserSigningMaterial),\n        mpcv1BackupKeyShare: JSON.stringify(v1Fixtures.mockBackupSigningMaterial),\n      });\n      const bitgoRetrofitData: DklsTypes.RetrofitData = {\n        xiList: retrofitData.mpcv2UserKeyShare.xiList,\n        xShare: v1Fixtures.mockBitGoShareSigningMaterial.xShare,\n      };\n      const [user, backup, bitgo] = await DklsUtils.generateDKGKeyShares(\n        retrofitData.mpcv2UserKeyShare,\n        retrofitData.mpcv2BackupKeyShare,\n        bitgoRetrofitData\n      );\n      assert.ok(bitgo.getKeyShare());\n      const messageToSign = crypto.createHash('sha256').update(Buffer.from('ffff', 'hex')).digest();\n      const derivationPath = 'm/999/988/0/0';\n      const signature = await DklsUtils.executeTillRound(\n        5,\n        new DklsDsg.Dsg(user.getKeyShare(), 0, derivationPath, messageToSign),\n        new DklsDsg.Dsg(backup.getKeyShare(), 1, derivationPath, messageToSign)\n      );\n      const convertedSignature = DklsUtils.verifyAndConvertDklsSignature(\n        Buffer.from('ffff', 'hex'),\n        signature as DklsTypes.DeserializedDklsSignature,\n        v1Fixtures.mockBitGoShareSigningMaterial.xShare.y + v1Fixtures.mockBitGoShareSigningMaterial.xShare.chaincode,\n        derivationPath\n      );\n      assert.ok(convertedSignature);\n      convertedSignature.split(':').length.should.equal(4);\n    });\n  });\n\n  describe('TSS key chains', async function () {\n    it('should generate TSS MPCv2 keys', async function () {\n      const bitgoSession = new DklsDkg.Dkg(3, 2, 2);\n\n      const round1Nock = await nockKeyGenRound1(bitgoSession, 1);\n      const round2Nock = await nockKeyGenRound2(bitgoSession, 1);\n      const round3Nock = await nockKeyGenRound3(bitgoSession, 1);\n      const addKeyNock = await nockAddKeyChain(coinName, 3);\n      const params = {\n        passphrase: 'test',\n        enterprise: enterpriseId,\n        originalPasscodeEncryptionCode: '123456',\n      };\n      const { userKeychain, backupKeychain, bitgoKeychain } = await tssUtils.createKeychains(params);\n      assert.ok(round1Nock.isDone());\n      assert.ok(round2Nock.isDone());\n      assert.ok(round3Nock.isDone());\n      assert.ok(addKeyNock.isDone());\n\n      assert.ok(userKeychain);\n      assert.equal(userKeychain.source, 'user');\n      assert.ok(userKeychain.commonKeychain);\n      assert.ok(ECDSAUtils.EcdsaMPCv2Utils.validateCommonKeychainPublicKey(userKeychain.commonKeychain));\n      assert.ok(userKeychain.encryptedPrv);\n      assert.ok(bitgo.decrypt({ input: userKeychain.encryptedPrv, password: params.passphrase }));\n\n      assert.ok(backupKeychain);\n      assert.equal(backupKeychain.source, 'backup');\n      assert.ok(backupKeychain.commonKeychain);\n      assert.ok(ECDSAUtils.EcdsaMPCv2Utils.validateCommonKeychainPublicKey(backupKeychain.commonKeychain));\n      assert.ok(backupKeychain.encryptedPrv);\n      assert.ok(bitgo.decrypt({ input: backupKeychain.encryptedPrv, password: params.passphrase }));\n\n      assert.ok(bitgoKeychain);\n      assert.equal(bitgoKeychain.source, 'bitgo');\n    });\n\n    it('should generate TSS MPCv2 keys for retrofit', async function () {\n      const xiList = [\n        Array.from(bigIntToBufferBE(BigInt(1), 32)),\n        Array.from(bigIntToBufferBE(BigInt(2), 32)),\n        Array.from(bigIntToBufferBE(BigInt(3), 32)),\n      ];\n      const bitgoRetrofitData: DklsTypes.RetrofitData = {\n        xiList,\n        xShare: v1Fixtures.mockBitGoShareSigningMaterial.xShare,\n      };\n      const bitgoSession = new DklsDkg.Dkg(3, 2, ECDSAUtils.MPCv2PartiesEnum.BITGO, undefined, bitgoRetrofitData);\n\n      const round1Nock = await nockKeyGenRound1(bitgoSession, 1);\n      const round2Nock = await nockKeyGenRound2(bitgoSession, 1);\n      const round3Nock = await nockKeyGenRound3(bitgoSession, 1);\n      const addKeyNock = await nockAddKeyChain(coinName, 3);\n      const params: Parameters<typeof tssUtils.createKeychains>[0] = {\n        passphrase: 'test',\n        enterprise: enterpriseId,\n        originalPasscodeEncryptionCode: '123456',\n        retrofit: {\n          decryptedUserKey: JSON.stringify(v1Fixtures.mockUserSigningMaterial),\n          decryptedBackupKey: JSON.stringify(v1Fixtures.mockBackupSigningMaterial),\n          walletId: '123',\n        },\n      };\n      const { userKeychain, backupKeychain, bitgoKeychain } = await tssUtils.createKeychains(params);\n      assert.ok(round1Nock.isDone());\n      assert.ok(round2Nock.isDone());\n      assert.ok(round3Nock.isDone());\n      assert.ok(addKeyNock.isDone());\n\n      assert.ok(userKeychain);\n      assert.equal(userKeychain.source, 'user');\n      assert.ok(userKeychain.commonKeychain);\n      assert.ok(ECDSAUtils.EcdsaMPCv2Utils.validateCommonKeychainPublicKey(userKeychain.commonKeychain));\n      assert.ok(userKeychain.encryptedPrv);\n      assert.ok(bitgo.decrypt({ input: userKeychain.encryptedPrv, password: params.passphrase }));\n\n      assert.ok(backupKeychain);\n      assert.equal(backupKeychain.source, 'backup');\n      assert.ok(backupKeychain.commonKeychain);\n      assert.ok(ECDSAUtils.EcdsaMPCv2Utils.validateCommonKeychainPublicKey(backupKeychain.commonKeychain));\n      assert.ok(backupKeychain.encryptedPrv);\n      assert.ok(bitgo.decrypt({ input: backupKeychain.encryptedPrv, password: params.passphrase }));\n\n      assert.ok(bitgoKeychain);\n      assert.equal(bitgoKeychain.source, 'bitgo');\n    });\n\n    it('should create TSS key chains', async function () {\n      const nockPromises = [\n        nockKeychain({\n          coin: coinName,\n          keyChain: { id: '1', pub: '1', type: 'tss', reducedEncryptedPrv: '' },\n          source: 'user',\n        }),\n        nockKeychain({\n          coin: coinName,\n          keyChain: { id: '2', pub: '2', type: 'tss', reducedEncryptedPrv: '' },\n          source: 'backup',\n        }),\n        nockKeychain({\n          coin: coinName,\n          keyChain: { id: '3', pub: '3', type: 'tss', reducedEncryptedPrv: '' },\n          source: 'bitgo',\n        }),\n      ];\n      const [nockedUserKeychain, nockedBackupKeychain, nockedBitGoKeychain] = await Promise.all(nockPromises);\n\n      const bitgoKeychainPromise = tssUtils.createParticipantKeychain(ECDSAUtils.MPCv2PartiesEnum.BITGO, 'test');\n      const usersKeychainPromise = tssUtils.createParticipantKeychain(\n        ECDSAUtils.MPCv2PartiesEnum.USER,\n        'test',\n        Buffer.from('test'),\n        Buffer.from('test'),\n        'passphrase',\n        'test'\n      );\n      const backupKeychainPromise = tssUtils.createParticipantKeychain(\n        ECDSAUtils.MPCv2PartiesEnum.BACKUP,\n        'test',\n        Buffer.from('test'),\n        Buffer.from('test'),\n        'passphrase',\n        'test'\n      );\n\n      const [userKeychain, backupKeychain, bitgoKeychain] = await Promise.all([\n        usersKeychainPromise,\n        backupKeychainPromise,\n        bitgoKeychainPromise,\n      ]);\n\n      ({ ...userKeychain, reducedEncryptedPrv: '' }).should.deepEqual(nockedUserKeychain);\n      ({ ...backupKeychain, reducedEncryptedPrv: '' }).should.deepEqual(nockedBackupKeychain);\n      ({ ...bitgoKeychain, reducedEncryptedPrv: '' }).should.deepEqual(nockedBitGoKeychain);\n    });\n\n    it('should create TSS MPCv2 key chains with OVCs', async function () {\n      const MPCv2SMCUtils = new ECDSAUtils.MPCv2SMCUtils(bitgo, baseCoin);\n      const bitgoSession = new DklsDkg.Dkg(3, 2, 2);\n\n      const round1Nock = await nockKeyGenRound1(bitgoSession, 1);\n      const round2Nock = await nockKeyGenRound2(bitgoSession, 1);\n      const round3Nock = await nockKeyGenRound3(bitgoSession, 1);\n      const addKeyNock = await nockAddKeyChain(coinName, 3);\n\n      // OVC 1 - User GPG key\n      const userGgpKey = await openpgp.generateKey({\n        userIDs: [\n          {\n            name: 'user',\n            email: 'user@test.com',\n          },\n        ],\n        curve: 'secp256k1',\n      });\n      const userGpgPrvKey = {\n        partyId: 0,\n        gpgKey: userGgpKey.privateKey,\n      };\n\n      // Round 1 User\n      const userSession = new DklsDkg.Dkg(3, 2, 0);\n      let OVC1ToOVC2Round1Payload: OVC1ToOVC2Round1Payload;\n      {\n        const userBroadcastMsg1Unsigned = await userSession.initDkg();\n        const userMsgs1Signed = await DklsComms.encryptAndAuthOutgoingMessages(\n          { broadcastMessages: [DklsTypes.serializeBroadcastMessage(userBroadcastMsg1Unsigned)], p2pMessages: [] },\n          [],\n          [userGpgPrvKey]\n        );\n        const userMsg1 = userMsgs1Signed.broadcastMessages.find((m) => m.from === 0);\n        assert(userMsg1, 'userMsg1 not found');\n\n        OVC1ToOVC2Round1Payload = {\n          tssVersion: '0.0.1' as NonEmptyString,\n          walletType: WalletTypeEnum.tss,\n          coin: 'eth' as NonEmptyString,\n          state: KeyCreationMPCv2StateEnum.WaitingForOVC2Round1Data,\n          ovc: {\n            [OVCIndexEnum.ONE]: {\n              gpgPubKey: userGgpKey.publicKey as NonEmptyString,\n              ovcMsg1: userMsg1 as any,\n            },\n          },\n        };\n      }\n\n      // OVC 2 - Backup GPG key\n      const backupGgpKey = await openpgp.generateKey({\n        userIDs: [\n          {\n            name: 'backup',\n            email: 'backup@test.com',\n          },\n        ],\n        curve: 'secp256k1',\n      });\n\n      const backupGpgPrvKey = {\n        partyId: 1,\n        gpgKey: backupGgpKey.privateKey,\n      };\n      // Round 1 Backup\n      const backupSession = new DklsDkg.Dkg(3, 2, 1);\n      let OVC2ToBitgoRound1Payload: OVC2ToBitgoRound1Payload;\n      {\n        assert(OVC1ToOVC2Round1Payload.state === 0, 'OVC1ToOVC2Round1Payload.state should be 0');\n        const backupBroadcastMsg1Unsigned = await backupSession.initDkg();\n        const backupMsgs1Signed = await DklsComms.encryptAndAuthOutgoingMessages(\n          { broadcastMessages: [DklsTypes.serializeBroadcastMessage(backupBroadcastMsg1Unsigned)], p2pMessages: [] },\n          [],\n          [backupGpgPrvKey]\n        );\n        const backupMsg1 = backupMsgs1Signed.broadcastMessages.find((m) => m.from === 1);\n        assert(backupMsg1, 'backupMsg1 not found');\n\n        OVC2ToBitgoRound1Payload = {\n          ...OVC1ToOVC2Round1Payload,\n          state: KeyCreationMPCv2StateEnum.WaitingForBitgoRound1Data,\n          ovc: {\n            ...OVC1ToOVC2Round1Payload.ovc,\n            [OVCIndexEnum.TWO]: {\n              gpgPubKey: backupGgpKey.publicKey as NonEmptyString,\n              ovcMsg1: backupMsg1 as any,\n            },\n          },\n        };\n      }\n\n      // Round 1 BitGo\n      const bitgoToOVC1Round1Payload = await MPCv2SMCUtils.keyGenRound1('testId', OVC2ToBitgoRound1Payload);\n\n      // Round 2 User\n      let OVC1ToOVC2Round2Payload: OVC1ToOVC2Round2Payload;\n      {\n        assert(bitgoToOVC1Round1Payload.state === 2, 'bitgoToOVC1Round1Payload.state should be 2');\n        const toUserRound1BroadcastMessages = await DklsComms.decryptAndVerifyIncomingMessages(\n          {\n            p2pMessages: [],\n            broadcastMessages: [\n              bitgoToOVC1Round1Payload.ovc[OVCIndexEnum.TWO].ovcMsg1,\n              bitgoToOVC1Round1Payload.platform.bitgoMsg1,\n            ],\n          },\n          [bitgoGpgPubKey, { partyId: 1, gpgKey: bitgoToOVC1Round1Payload.ovc[OVCIndexEnum.TWO].gpgPubKey }],\n          [userGpgPrvKey]\n        );\n\n        const userRound2P2PMessages = userSession.handleIncomingMessages({\n          p2pMessages: [],\n          broadcastMessages: toUserRound1BroadcastMessages.broadcastMessages.map(DklsTypes.deserializeBroadcastMessage),\n        });\n        const userRound2Messages = await DklsComms.encryptAndAuthOutgoingMessages(\n          DklsTypes.serializeMessages(userRound2P2PMessages),\n          [{ partyId: 1, gpgKey: bitgoToOVC1Round1Payload.ovc[OVCIndexEnum.TWO].gpgPubKey }, bitgoGpgPubKey],\n          [userGpgPrvKey]\n        );\n        const userToBackupMsg2 = userRound2Messages.p2pMessages.find(\n          (m) => m.from === ECDSAUtils.MPCv2PartiesEnum.USER && m.to === ECDSAUtils.MPCv2PartiesEnum.BACKUP\n        );\n        assert(userToBackupMsg2, 'userToBackupMsg2 not found');\n        const userToBitgoMsg2 = userRound2Messages.p2pMessages.find(\n          (m) => m.from === ECDSAUtils.MPCv2PartiesEnum.USER && m.to === ECDSAUtils.MPCv2PartiesEnum.BITGO\n        );\n        assert(userToBitgoMsg2, 'userToBitgoMsg2 not found');\n\n        OVC1ToOVC2Round2Payload = {\n          ...bitgoToOVC1Round1Payload,\n          state: KeyCreationMPCv2StateEnum.WaitingForOVC2Round2Data,\n          ovc: {\n            ...bitgoToOVC1Round1Payload.ovc,\n            [OVCIndexEnum.ONE]: Object.assign(bitgoToOVC1Round1Payload.ovc[OVCIndexEnum.ONE], {\n              ovcToBitgoMsg2: userToBitgoMsg2,\n              ovcToOvcMsg2: userToBackupMsg2,\n            }) as any,\n          },\n        };\n      }\n\n      // Round 2 Backup\n      let OVC2ToBitgoRound2Payload: OVC2ToBitgoRound2Payload;\n\n      {\n        assert(OVC1ToOVC2Round2Payload.state === 3, 'bitgoToOVC1Round1Payload.state should be 3');\n        const toBackupRound1BroadcastMessages = await DklsComms.decryptAndVerifyIncomingMessages(\n          {\n            p2pMessages: [],\n            broadcastMessages: [\n              bitgoToOVC1Round1Payload.ovc[OVCIndexEnum.ONE].ovcMsg1,\n              bitgoToOVC1Round1Payload.platform.bitgoMsg1,\n            ],\n          },\n          [bitgoGpgPubKey, { partyId: 0, gpgKey: OVC1ToOVC2Round2Payload.ovc[OVCIndexEnum.ONE].gpgPubKey }],\n          [backupGpgPrvKey]\n        );\n\n        const backupRound2P2PMessages = backupSession.handleIncomingMessages({\n          p2pMessages: [],\n          broadcastMessages: toBackupRound1BroadcastMessages.broadcastMessages.map(\n            DklsTypes.deserializeBroadcastMessage\n          ),\n        });\n        const backupRound2Messages = await DklsComms.encryptAndAuthOutgoingMessages(\n          DklsTypes.serializeMessages(backupRound2P2PMessages),\n          [{ partyId: 0, gpgKey: bitgoToOVC1Round1Payload.ovc[OVCIndexEnum.ONE].gpgPubKey }, bitgoGpgPubKey],\n          [backupGpgPrvKey]\n        );\n        const backupToUserMsg2 = backupRound2Messages.p2pMessages.find(\n          (m) => m.from === ECDSAUtils.MPCv2PartiesEnum.BACKUP && m.to === ECDSAUtils.MPCv2PartiesEnum.USER\n        );\n        assert(backupToUserMsg2, 'backupToUserMsg2 not found');\n        const backupToBitgoMsg2 = backupRound2Messages.p2pMessages.find(\n          (m) => m.from === ECDSAUtils.MPCv2PartiesEnum.BACKUP && m.to === ECDSAUtils.MPCv2PartiesEnum.BITGO\n        );\n        assert(backupToBitgoMsg2, 'backupToBitgoMsg2 not found');\n\n        OVC2ToBitgoRound2Payload = {\n          ...OVC1ToOVC2Round2Payload,\n          state: KeyCreationMPCv2StateEnum.WaitingForBitgoRound2Data,\n          ovc: {\n            ...OVC1ToOVC2Round2Payload.ovc,\n            [OVCIndexEnum.TWO]: Object.assign(OVC1ToOVC2Round2Payload.ovc[OVCIndexEnum.TWO], {\n              ovcToBitgoMsg2: backupToBitgoMsg2,\n              ovcToOvcMsg2: backupToUserMsg2,\n            }) as any,\n          },\n        };\n      }\n\n      // Round 2 BitGo\n      // call bitgo round 2\n      const bitgoToOVC1Round2Payload = await MPCv2SMCUtils.keyGenRound2('testId', OVC2ToBitgoRound2Payload);\n\n      // Round 3A User\n      let OVC1ToOVC2Round3Payload: OVC1ToOVC2Round3Payload;\n      {\n        assert(bitgoToOVC1Round2Payload.state === 5, 'bitgoToOVC1Round2Payload.state should be 5');\n        const toUserRound2P2PMessages = await DklsComms.decryptAndVerifyIncomingMessages(\n          {\n            p2pMessages: [\n              bitgoToOVC1Round2Payload.ovc[OVCIndexEnum.TWO].ovcToOvcMsg2,\n              bitgoToOVC1Round2Payload.platform.ovc[OVCIndexEnum.ONE].bitgoToOvcMsg2,\n            ],\n            broadcastMessages: [],\n          },\n          [bitgoGpgPubKey, { partyId: 1, gpgKey: bitgoToOVC1Round2Payload.ovc[OVCIndexEnum.TWO].gpgPubKey }],\n          [userGpgPrvKey]\n        );\n        const userRound3AP2PMessages = userSession.handleIncomingMessages({\n          p2pMessages: toUserRound2P2PMessages.p2pMessages.map(DklsTypes.deserializeP2PMessage),\n          broadcastMessages: [],\n        }).p2pMessages;\n\n        const userRound3AMessages = await DklsComms.encryptAndAuthOutgoingMessages(\n          DklsTypes.serializeMessages({\n            p2pMessages: userRound3AP2PMessages,\n            broadcastMessages: [],\n          }),\n          [{ partyId: 1, gpgKey: bitgoToOVC1Round2Payload.ovc[OVCIndexEnum.TWO].gpgPubKey }, bitgoGpgPubKey],\n          [userGpgPrvKey]\n        );\n\n        const userToBitgoMsg3 = userRound3AMessages.p2pMessages.find(\n          (m) => m.from === ECDSAUtils.MPCv2PartiesEnum.USER && m.to === ECDSAUtils.MPCv2PartiesEnum.BITGO\n        );\n        assert(userToBitgoMsg3, 'userToBitgoMsg3 not found');\n        const userToBackupMsg3 = userRound3AMessages.p2pMessages.find(\n          (m) => m.from === ECDSAUtils.MPCv2PartiesEnum.USER && m.to === ECDSAUtils.MPCv2PartiesEnum.BACKUP\n        );\n        assert(userToBackupMsg3, 'userToBackupMsg3 not found');\n\n        OVC1ToOVC2Round3Payload = {\n          ...bitgoToOVC1Round2Payload,\n          state: KeyCreationMPCv2StateEnum.WaitingForOVC2Round3Data,\n          ovc: {\n            ...bitgoToOVC1Round2Payload.ovc,\n            [OVCIndexEnum.ONE]: Object.assign(bitgoToOVC1Round2Payload.ovc[OVCIndexEnum.ONE], {\n              ovcToBitgoMsg3: userToBitgoMsg3,\n              ovcToOvcMsg3: userToBackupMsg3,\n            }) as any,\n          },\n        };\n      }\n\n      // Round 3 Backup\n      let OVC2ToOVC1Round3Payload: OVC2ToOVC1Round3Payload;\n      {\n        assert(OVC1ToOVC2Round3Payload.state === 6, 'OVC1ToOVC2Round3Payload.state should be 6');\n        const toBackupRound3P2PMessages = await DklsComms.decryptAndVerifyIncomingMessages(\n          {\n            p2pMessages: [\n              OVC1ToOVC2Round3Payload.ovc[OVCIndexEnum.ONE].ovcToOvcMsg2,\n              OVC1ToOVC2Round3Payload.platform.ovc[OVCIndexEnum.TWO].bitgoToOvcMsg2,\n            ],\n            broadcastMessages: [],\n          },\n          [bitgoGpgPubKey, { partyId: 0, gpgKey: OVC1ToOVC2Round3Payload.ovc[OVCIndexEnum.ONE].gpgPubKey }],\n          [backupGpgPrvKey]\n        );\n\n        const backupRound3P2PMessages = backupSession.handleIncomingMessages({\n          p2pMessages: toBackupRound3P2PMessages.p2pMessages.map(DklsTypes.deserializeP2PMessage),\n          broadcastMessages: [],\n        });\n\n        const backupRound3Messages = await DklsComms.encryptAndAuthOutgoingMessages(\n          DklsTypes.serializeMessages(backupRound3P2PMessages),\n          [{ partyId: 0, gpgKey: OVC1ToOVC2Round3Payload.ovc[OVCIndexEnum.ONE].gpgPubKey }, bitgoGpgPubKey],\n          [backupGpgPrvKey]\n        );\n\n        const backupToBitgoMsg3 = backupRound3Messages.p2pMessages.find(\n          (m) => m.from === ECDSAUtils.MPCv2PartiesEnum.BACKUP && m.to === ECDSAUtils.MPCv2PartiesEnum.BITGO\n        );\n        assert(backupToBitgoMsg3, 'backupToBitgoMsg3 not found');\n        const backupToUserMsg3 = backupRound3Messages.p2pMessages.find(\n          (m) => m.from === ECDSAUtils.MPCv2PartiesEnum.BACKUP && m.to === ECDSAUtils.MPCv2PartiesEnum.USER\n        );\n        assert(backupToUserMsg3, 'backupToUserMsg3 not found');\n\n        const toBackupRound3Messages = await DklsComms.decryptAndVerifyIncomingMessages(\n          {\n            p2pMessages: [\n              {\n                ...OVC1ToOVC2Round3Payload.ovc[OVCIndexEnum.ONE].ovcToOvcMsg3,\n                commitment: OVC1ToOVC2Round3Payload.ovc[OVCIndexEnum.ONE].ovcToOvcMsg2.commitment,\n              },\n              {\n                ...OVC1ToOVC2Round3Payload.platform.ovc[OVCIndexEnum.TWO].bitgoToOvcMsg3,\n                commitment: OVC1ToOVC2Round3Payload.platform.bitgoCommitment2,\n              },\n            ],\n            broadcastMessages: [],\n          },\n          [bitgoGpgPubKey, { partyId: 0, gpgKey: OVC1ToOVC2Round3Payload.ovc[OVCIndexEnum.ONE].gpgPubKey }],\n          [backupGpgPrvKey]\n        );\n\n        const backupRound4Messages = backupSession.handleIncomingMessages({\n          p2pMessages: toBackupRound3Messages.p2pMessages.map(DklsTypes.deserializeP2PMessage),\n          broadcastMessages: [],\n        }).broadcastMessages;\n\n        const backupRound4BroadcastMessages = await DklsComms.encryptAndAuthOutgoingMessages(\n          DklsTypes.serializeMessages({\n            p2pMessages: [],\n            broadcastMessages: backupRound4Messages,\n          }),\n          [],\n          [backupGpgPrvKey]\n        );\n\n        const backupMsg4 = backupRound4BroadcastMessages.broadcastMessages.find(\n          (m) => m.from === ECDSAUtils.MPCv2PartiesEnum.BACKUP\n        );\n        assert(backupMsg4, 'backupMsg4 not found');\n\n        OVC2ToOVC1Round3Payload = {\n          ...OVC1ToOVC2Round3Payload,\n          state: KeyCreationMPCv2StateEnum.WaitingForOVC1Round3bData,\n          ovc: {\n            ...OVC1ToOVC2Round3Payload.ovc,\n            [OVCIndexEnum.TWO]: Object.assign(OVC1ToOVC2Round3Payload.ovc[OVCIndexEnum.TWO], {\n              ovcToOvcMsg3: backupToUserMsg3,\n              ovcToBitgoMsg3: backupToBitgoMsg3,\n              ovcMsg4: backupMsg4,\n            }) as any,\n          },\n        };\n      }\n\n      // Round 3B User\n      let OVC1ToBitgoRound3BPayload: OVC1ToBitgoRound3Payload;\n      {\n        assert(OVC2ToOVC1Round3Payload.state === 7, 'OVC2ToOVC1Round3Payload.state should be 7');\n        const toUserRound4Messages = await DklsComms.decryptAndVerifyIncomingMessages(\n          {\n            p2pMessages: [\n              {\n                ...OVC2ToOVC1Round3Payload.ovc[OVCIndexEnum.TWO].ovcToOvcMsg3,\n                commitment: OVC2ToOVC1Round3Payload.ovc[OVCIndexEnum.TWO].ovcToOvcMsg2.commitment,\n              },\n              {\n                ...OVC2ToOVC1Round3Payload.platform.ovc[OVCIndexEnum.ONE].bitgoToOvcMsg3,\n                commitment: OVC2ToOVC1Round3Payload.platform.bitgoCommitment2,\n              },\n            ],\n            broadcastMessages: [],\n          },\n          [bitgoGpgPubKey, { partyId: 1, gpgKey: OVC2ToOVC1Round3Payload.ovc[OVCIndexEnum.TWO].gpgPubKey }],\n          [userGpgPrvKey]\n        );\n\n        const userRound4BroadcastMessages = userSession.handleIncomingMessages({\n          p2pMessages: toUserRound4Messages.p2pMessages.map(DklsTypes.deserializeP2PMessage),\n          broadcastMessages: [],\n        }).broadcastMessages;\n        assert(userRound4BroadcastMessages.length === 1, 'userRound4BroadcastMessages length should be 1');\n\n        const userRound4Messages = await DklsComms.encryptAndAuthOutgoingMessages(\n          DklsTypes.serializeMessages({\n            p2pMessages: [],\n            broadcastMessages: userRound4BroadcastMessages,\n          }),\n          [],\n          [userGpgPrvKey]\n        );\n\n        const userMsg4 = userRound4Messages.broadcastMessages.find((m) => m.from === ECDSAUtils.MPCv2PartiesEnum.USER);\n        assert(userMsg4, 'userMsg4 not found');\n\n        OVC1ToBitgoRound3BPayload = {\n          ...OVC2ToOVC1Round3Payload,\n          state: KeyCreationMPCv2StateEnum.WaitingForBitgoRound3Data,\n          ovc: {\n            ...OVC2ToOVC1Round3Payload.ovc,\n            [OVCIndexEnum.ONE]: Object.assign(OVC2ToOVC1Round3Payload.ovc[OVCIndexEnum.ONE], {\n              ovcMsg4: userMsg4,\n            }) as any,\n          },\n        };\n      }\n\n      // Round 3 BitGo\n      // creates bitgo keychain\n      const bitgoToOVC1Round3Payload = await MPCv2SMCUtils.keyGenRound3('testId', OVC1ToBitgoRound3BPayload);\n\n      // Round 4 User\n      let userCommonKeychain: string;\n      let OVC1ToOVC2Round4Payload;\n      {\n        assert(bitgoToOVC1Round3Payload.state === 9, 'bitgoToOVC1Round3Payload.state should be 9');\n        assert(bitgoToOVC1Round3Payload.bitGoKeyId, 'bitgoToOVC1Round3Payload.bitGoKeyId not found');\n        const toUserBitgoRound3Msg = await DklsComms.decryptAndVerifyIncomingMessages(\n          {\n            p2pMessages: [],\n            broadcastMessages: [\n              bitgoToOVC1Round3Payload.ovc[OVCIndexEnum.TWO].ovcMsg4,\n              bitgoToOVC1Round3Payload.platform.bitgoMsg4,\n            ],\n          },\n          [bitgoGpgPubKey, { partyId: 1, gpgKey: bitgoToOVC1Round3Payload.ovc[OVCIndexEnum.TWO].gpgPubKey }],\n          [userGpgPrvKey]\n        );\n\n        userSession.handleIncomingMessages({\n          p2pMessages: [],\n          broadcastMessages: toUserBitgoRound3Msg.broadcastMessages.map(DklsTypes.deserializeBroadcastMessage),\n        });\n\n        const userPrivateMaterial = userSession.getKeyShare();\n        userCommonKeychain = DklsTypes.getCommonKeychain(userPrivateMaterial);\n        assert.equal(\n          bitgoToOVC1Round3Payload.platform.commonKeychain,\n          userCommonKeychain,\n          'User and Bitgo Common keychains do not match'\n        );\n        const userPrv = userPrivateMaterial.toString('base64');\n        assert(userPrv, 'userPrv not found');\n\n        OVC1ToOVC2Round4Payload = {\n          bitgoKeyId: bitgoToOVC1Round3Payload.bitGoKeyId,\n          ...bitgoToOVC1Round3Payload,\n          state: KeyCreationMPCv2StateEnum.WaitingForOVC2GenerateKey,\n        };\n      }\n\n      // Round 4 Backup\n      let backupCommonKeychain: string;\n      {\n        assert(OVC1ToOVC2Round4Payload.state === 10, 'OVC1ToOVC2Round4Payload.state should be 10');\n        assert(OVC1ToOVC2Round4Payload.bitgoKeyId, 'OVC1ToOVC2Round4Payload.bitGoKeyId not found');\n\n        const toBackupBitgoRound3Msg = await DklsComms.decryptAndVerifyIncomingMessages(\n          {\n            p2pMessages: [],\n            broadcastMessages: [\n              OVC1ToOVC2Round4Payload.ovc[OVCIndexEnum.ONE].ovcMsg4,\n              OVC1ToOVC2Round4Payload.platform.bitgoMsg4,\n            ],\n          },\n          [bitgoGpgPubKey, { partyId: 0, gpgKey: OVC1ToOVC2Round4Payload.ovc[OVCIndexEnum.ONE].gpgPubKey }],\n          [backupGpgPrvKey]\n        );\n\n        backupSession.handleIncomingMessages({\n          p2pMessages: [],\n          broadcastMessages: toBackupBitgoRound3Msg.broadcastMessages.map(DklsTypes.deserializeBroadcastMessage),\n        });\n\n        const backupPrivateMaterial = backupSession.getKeyShare();\n        backupCommonKeychain = DklsTypes.getCommonKeychain(backupPrivateMaterial);\n        assert.equal(\n          OVC1ToOVC2Round4Payload.platform.commonKeychain,\n          backupCommonKeychain,\n          'Backup and Bitgo Common keychains do not match'\n        );\n        const backupPrv = backupPrivateMaterial.toString('base64');\n        assert(backupPrv, 'backupPrv not found');\n      }\n\n      // Round 4 BitGo\n      // creates user and backup keychain\n      const keychains = await MPCv2SMCUtils.uploadClientKeys(\n        bitgoToOVC1Round3Payload.bitGoKeyId,\n        userCommonKeychain,\n        backupCommonKeychain\n      );\n      assert.deepEqual(keychains.userKeychain, {\n        commonKeychain: userCommonKeychain,\n        type: 'tss',\n        source: 'user',\n        id: 'user',\n      });\n      assert.deepEqual(keychains.backupKeychain, {\n        commonKeychain: backupCommonKeychain,\n        type: 'tss',\n        source: 'backup',\n        id: 'backup',\n      });\n      assert.ok(round1Nock.isDone());\n      assert.ok(round2Nock.isDone());\n      assert.ok(round3Nock.isDone());\n      assert.ok(addKeyNock.isDone());\n    });\n\n    it('should throw for MPCv2 SMC utils if the state is invalid', async function () {\n      const MPCv2SMCUtils = new ECDSAUtils.MPCv2SMCUtils(bitgo, baseCoin);\n      const invalidPayload = {\n        state: KeyCreationMPCv2StateEnum.WaitingForOVC1Round2Data,\n      } as any;\n\n      await assert.rejects(async () => await MPCv2SMCUtils.keyGenRound1('testId', invalidPayload), {\n        message: `Invalid state for round 1, expected: ${KeyCreationMPCv2StateEnum.WaitingForBitgoRound1Data}, got: ${KeyCreationMPCv2StateEnum.WaitingForOVC1Round2Data}`,\n      });\n      await assert.rejects(async () => await MPCv2SMCUtils.keyGenRound2('testId', invalidPayload), {\n        message: `Invalid state for round 2, expected: ${KeyCreationMPCv2StateEnum.WaitingForBitgoRound2Data}, got: ${KeyCreationMPCv2StateEnum.WaitingForOVC1Round2Data}`,\n      });\n      await assert.rejects(async () => await MPCv2SMCUtils.keyGenRound3('testId', invalidPayload), {\n        message: `Invalid state for round 3, expected: ${KeyCreationMPCv2StateEnum.WaitingForBitgoRound3Data}, got: ${KeyCreationMPCv2StateEnum.WaitingForOVC1Round2Data}`,\n      });\n    });\n  });\n\n  async function nockKeychain(\n    params: {\n      coin: string;\n      keyChain: Keychain;\n      source: 'user' | 'backup' | 'bitgo';\n    },\n    times = 1\n  ): Promise<Keychain> {\n    nock(bgUrl)\n      .post(`/api/v2/${params.coin}/key`, (body) => {\n        return body.keyType === 'tss' && body.source === params.source;\n      })\n      .times(times)\n      .reply(200, params.keyChain);\n\n    return params.keyChain;\n  }\n\n  async function nockGetBitgoPublicKeyBasedOnFeatureFlags(\n    coin: string,\n    enterpriseId: string,\n    bitgoGpgKeyPair: openpgp.SerializedKeyPair<string>\n  ): Promise<BitgoGPGPublicKey> {\n    const bitgoGPGPublicKeyResponse: BitgoGPGPublicKey = {\n      name: 'irrelevant',\n      publicKey: bitgoGpgKeyPair.publicKey,\n      mpcv2PublicKey: bitgoGpgKeyPair.publicKey,\n      enterpriseId,\n    };\n    nock(bgUrl).get(`/api/v2/${coin}/tss/pubkey`).query({ enterpriseId }).reply(200, bitgoGPGPublicKeyResponse);\n\n    return bitgoGPGPublicKeyResponse;\n  }\n\n  async function nockKeyGenRound1(bitgoSession: DklsDkg.Dkg, times = 1) {\n    return nock(bgUrl)\n      .post(`/api/v2/mpc/generatekey`, (body) => body.round === 'MPCv2-R1')\n      .times(times)\n      .reply(\n        200,\n        async (uri, { payload }: { payload: MPCv2KeyGenRound1Request }): Promise<MPCv2KeyGenRound1Response> => {\n          const { userGpgPublicKey, backupGpgPublicKey, userMsg1, backupMsg1 } = payload;\n          userGpgPubKey = {\n            partyId: 0,\n            gpgKey: userGpgPublicKey,\n          };\n          backupGpgPubKey = {\n            partyId: 1,\n            gpgKey: backupGpgPublicKey,\n          };\n\n          const bitgoBroadcastMsg1Unsigned = await bitgoSession.initDkg();\n          const bitgoMsgs1Signed = await DklsComms.encryptAndAuthOutgoingMessages(\n            { broadcastMessages: [DklsTypes.serializeBroadcastMessage(bitgoBroadcastMsg1Unsigned)], p2pMessages: [] },\n            [],\n            [bitgoGpgPrvKey]\n          );\n          const bitgoMsg1 = bitgoMsgs1Signed.broadcastMessages.find((m) => m.from === 2);\n          assert(bitgoMsg1, 'bitgoMsg1 not found');\n\n          const round1IncomingMsgs = await DklsComms.decryptAndVerifyIncomingMessages(\n            {\n              p2pMessages: [],\n              broadcastMessages: [\n                { from: 0, payload: userMsg1 },\n                { from: 1, payload: backupMsg1 },\n              ],\n            },\n            [userGpgPubKey, backupGpgPubKey],\n            [bitgoGpgPrvKey]\n          );\n\n          const round2Messages = DklsTypes.serializeMessages(\n            bitgoSession.handleIncomingMessages(DklsTypes.deserializeMessages(round1IncomingMsgs))\n          );\n\n          const round2SignedMessages = await DklsComms.encryptAndAuthOutgoingMessages(\n            round2Messages,\n            [userGpgPubKey, backupGpgPubKey],\n            [bitgoGpgPrvKey]\n          );\n\n          const bitgoToUserMsg2 = round2SignedMessages.p2pMessages.find((m) => m.to === 0);\n          const bitgoToBackupMsg2 = round2SignedMessages.p2pMessages.find((m) => m.to === 1);\n          assert(bitgoToUserMsg2, 'bitgoToUserMsg2 not found');\n          assert(bitgoToBackupMsg2, 'bitgoToBackupMsg2 not found');\n          assert(bitgoToUserMsg2.commitment, 'bitgoToUserMsg2.commitment not found');\n\n          storedBitgoCommitment2 = bitgoToUserMsg2?.commitment;\n          return {\n            sessionId: 'testid' as NonEmptyString,\n            bitgoMsg1: { from: 2, ...bitgoMsg1.payload },\n            bitgoToBackupMsg2: {\n              from: 2,\n              to: 1,\n              encryptedMessage: bitgoToBackupMsg2.payload.encryptedMessage,\n              signature: bitgoToBackupMsg2.payload.signature,\n            },\n            bitgoToUserMsg2: {\n              from: 2,\n              to: 0,\n              encryptedMessage: bitgoToUserMsg2.payload.encryptedMessage,\n              signature: bitgoToUserMsg2.payload.signature,\n            },\n            walletGpgPubKeySigs: 'something' as NonEmptyString,\n          };\n        }\n      );\n  }\n\n  async function nockKeyGenRound2(bitgoSession: DklsDkg.Dkg, times = 1) {\n    return nock(bgUrl)\n      .post(`/api/v2/mpc/generatekey`, (body) => body.round === 'MPCv2-R2')\n      .times(times)\n      .reply(\n        200,\n        async (uri, { payload }: { payload: MPCv2KeyGenRound2Request }): Promise<MPCv2KeyGenRound2Response> => {\n          const { sessionId, userMsg2, backupMsg2, userCommitment2, backupCommitment2 } = payload;\n          storedUserCommitment2 = userCommitment2;\n          storedBackupCommitment2 = backupCommitment2;\n          const round2IncomingMsgs = await DklsComms.decryptAndVerifyIncomingMessages(\n            {\n              p2pMessages: [\n                {\n                  from: userMsg2.from,\n                  to: userMsg2.to,\n                  payload: { signature: userMsg2.signature, encryptedMessage: userMsg2.encryptedMessage },\n                },\n                {\n                  from: backupMsg2.from,\n                  to: backupMsg2.to,\n                  payload: { signature: backupMsg2.signature, encryptedMessage: backupMsg2.encryptedMessage },\n                },\n              ],\n              broadcastMessages: [],\n            },\n            [userGpgPubKey, backupGpgPubKey],\n            [bitgoGpgPrvKey]\n          );\n\n          const round3Messages = DklsTypes.serializeMessages(\n            bitgoSession.handleIncomingMessages(DklsTypes.deserializeMessages(round2IncomingMsgs))\n          );\n\n          const round3SignedMessages = await DklsComms.encryptAndAuthOutgoingMessages(\n            round3Messages,\n            [userGpgPubKey, backupGpgPubKey],\n            [bitgoGpgPrvKey]\n          );\n\n          const bitgoToUserMsg3 = round3SignedMessages.p2pMessages.find((m) => m.to === 0);\n          const bitgoToBackupMsg3 = round3SignedMessages.p2pMessages.find((m) => m.to === 1);\n          assert(bitgoToUserMsg3, 'bitgoToUserMsg3 not found');\n          assert(bitgoToBackupMsg3, 'bitgoToBackupMsg3 not found');\n\n          return {\n            sessionId,\n            bitgoCommitment2: storedBitgoCommitment2 as NonEmptyString,\n            bitgoToUserMsg3: {\n              from: 2,\n              to: 0,\n              encryptedMessage: bitgoToUserMsg3.payload.encryptedMessage,\n              signature: bitgoToUserMsg3.payload.signature,\n            },\n            bitgoToBackupMsg3: {\n              from: 2,\n              to: 1,\n              encryptedMessage: bitgoToBackupMsg3.payload.encryptedMessage,\n              signature: bitgoToBackupMsg3.payload.signature,\n            },\n          };\n        }\n      );\n  }\n\n  async function nockKeyGenRound3(bitgoSession: DklsDkg.Dkg, times = 1) {\n    return nock(bgUrl)\n      .post(`/api/v2/mpc/generatekey`, (body) => body.round === 'MPCv2-R3')\n      .times(times)\n      .reply(\n        200,\n        async (uri, { payload }: { payload: MPCv2KeyGenRound3Request }): Promise<MPCv2KeyGenRound3Response> => {\n          const { sessionId, userMsg3, userMsg4, backupMsg3, backupMsg4 } = payload;\n\n          const round3IncomingMsgs = await DklsComms.decryptAndVerifyIncomingMessages(\n            {\n              p2pMessages: [\n                {\n                  from: userMsg3.from,\n                  to: userMsg3.to,\n                  payload: { signature: userMsg3.signature, encryptedMessage: userMsg3.encryptedMessage },\n                  commitment: storedUserCommitment2,\n                },\n                {\n                  from: backupMsg3.from,\n                  to: backupMsg3.to,\n                  payload: { signature: backupMsg3.signature, encryptedMessage: backupMsg3.encryptedMessage },\n                  commitment: storedBackupCommitment2,\n                },\n              ],\n              broadcastMessages: [],\n            },\n            [userGpgPubKey, backupGpgPubKey],\n            [bitgoGpgPrvKey]\n          );\n\n          const round4Messages = DklsTypes.serializeMessages(\n            bitgoSession.handleIncomingMessages(DklsTypes.deserializeMessages(round3IncomingMsgs))\n          );\n          const round4SignedMessages = await DklsComms.encryptAndAuthOutgoingMessages(\n            round4Messages,\n            [],\n            [bitgoGpgPrvKey]\n          );\n          const bitgoMsg4 = round4SignedMessages.broadcastMessages.find((m) => m.from === 2);\n          assert(bitgoMsg4, 'bitgoMsg4 not found');\n\n          const round4IncomingMsgs = await DklsComms.decryptAndVerifyIncomingMessages(\n            {\n              p2pMessages: [],\n              broadcastMessages: [\n                {\n                  from: userMsg4.from,\n                  payload: { signature: userMsg4.signature, message: userMsg4.message },\n                },\n                {\n                  from: backupMsg4.from,\n\n                  payload: { signature: backupMsg4.signature, message: backupMsg4.message },\n                },\n              ],\n            },\n            [userGpgPubKey, backupGpgPubKey],\n            []\n          );\n          bitgoSession.handleIncomingMessages(DklsTypes.deserializeMessages(round4IncomingMsgs));\n          const keyShare = bitgoSession.getKeyShare();\n          const commonKeychain = DklsTypes.getCommonKeychain(keyShare);\n\n          return {\n            sessionId,\n            commonKeychain: commonKeychain as NonEmptyString,\n            bitgoMsg4: { from: 2, ...bitgoMsg4.payload },\n          };\n        }\n      );\n  }\n\n  async function nockAddKeyChain(coin: string, times = 1) {\n    return nock('https://bitgo.fakeurl')\n      .post(`/api/v2/${coin}/key`, (body) => body.keyType === 'tss' && body.isMPCv2)\n      .times(times)\n      .reply(200, async (uri, requestBody: AddKeychainOptions) => {\n        const key = {\n          id: requestBody.source,\n          source: requestBody.source,\n          type: requestBody.keyType,\n          commonKeychain: requestBody.commonKeychain,\n          encryptedPrv: requestBody.encryptedPrv,\n        };\n        // nock gets\n        nock('https://bitgo.fakeurl').get(`/api/v2/${coin}/key/${requestBody.source}`).reply(200, key);\n        return key;\n      });\n  }\n});\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!