PHP WebShell

Текущая директория: /opt/BitGoJS/modules/bitgo/dist/test/v2/unit/internal/tssUtils

Просмотр файла: ecdsa.js

"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const assert = require("assert");
const _ = require("lodash");
const nock = require("nock");
const openpgp = require("openpgp");
const should = require("should");
const sinon = require("sinon");
const sdk_test_1 = require("@bitgo/sdk-test");
const src_1 = require("../../../../../src");
const sdk_core_1 = require("@bitgo/sdk-core");
const sdk_lib_mpc_1 = require("@bitgo/sdk-lib-mpc");
const ecdsaFixtures_1 = require("../../../fixtures/tss/ecdsaFixtures");
const common_1 = require("./common");
const helpers_1 = require("../../tss/helpers");
const utxo_lib_1 = require("@bitgo/utxo-lib");
const ecdsaNtilde_1 = require("./mocks/ecdsaNtilde");
const sdk_opensslbytes_1 = require("@bitgo/sdk-opensslbytes");
const openSSLBytes = (0, sdk_opensslbytes_1.loadWebAssembly)().buffer;
const createKeccakHash = require('keccak');
const encryptNShare = sdk_core_1.ECDSAMethods.encryptNShare;
openpgp.config.rejectCurves = new Set();
describe('TSS Ecdsa Utils:', async function () {
    const coinName = 'hteth';
    const reqId = new sdk_core_1.RequestTracer();
    const walletId = '5b34252f1bf349930e34020a00000000';
    const enterpriseId = '6449153a6f6bc20006d66771cdbe15d3';
    const enterpriseData = { id: enterpriseId, name: 'Test Enterprise' };
    let sandbox;
    let MPC;
    let bgUrl;
    let tssUtils;
    let wallet;
    let bitgo;
    let baseCoin;
    let bitgoKeyShare;
    let userKeyShare;
    let backupKeyShare;
    let bitgoPublicKey;
    let userGpgKey;
    let userLocalBackupGpgKey;
    let bitGoGPGKeyPair;
    let nockedBitGoKeychain;
    let nockedUserKeychain;
    beforeEach(async function () {
        sandbox = sinon.createSandbox();
    });
    afterEach(function () {
        sandbox.restore();
    });
    before(async function () {
        nock.cleanAll();
        MPC = new sdk_core_1.Ecdsa();
        userKeyShare = ecdsaFixtures_1.keyShares.userKeyShare;
        backupKeyShare = ecdsaFixtures_1.keyShares.backupKeyShare;
        bitgoKeyShare = ecdsaFixtures_1.keyShares.bitgoKeyShare;
        const gpgKeyPromises = [
            openpgp.generateKey({
                userIDs: [
                    {
                        name: 'test',
                        email: 'test@test.com',
                    },
                ],
                curve: 'secp256k1',
            }),
            openpgp.generateKey({
                userIDs: [
                    {
                        name: 'backup',
                        email: 'backup@test.com',
                    },
                ],
                curve: 'secp256k1',
            }),
            openpgp.generateKey({
                userIDs: [
                    {
                        name: 'bitgo',
                        email: 'bitgo@test.com',
                    },
                ],
                curve: 'secp256k1',
            }),
        ];
        [userGpgKey, userLocalBackupGpgKey, bitGoGPGKeyPair] = await Promise.all(gpgKeyPromises);
        bitgoPublicKey = await openpgp.readKey({ armoredKey: bitGoGPGKeyPair.publicKey });
        const constants = {
            mpc: {
                bitgoPublicKey: bitGoGPGKeyPair.publicKey,
            },
        };
        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;
        // TODO(WP-346): sdk-test mocks conflict so we can't use persist
        nock(bgUrl).get('/api/v1/client/constants').times(16).reply(200, { ttl: 3600, constants });
        const nockPromises = [
            nockBitgoKeychain({
                coin: coinName,
                userKeyShare,
                backupKeyShare,
                bitgoKeyShare,
                userGpgKey,
                userLocalBackupGpgKey,
                bitgoGpgKey: bitGoGPGKeyPair,
            }),
            nockKeychain({ coin: coinName, keyChain: { id: '1', pub: '', type: 'tss' }, source: 'user' }),
            nockKeychain({ coin: coinName, keyChain: { id: '2', pub: '', type: 'tss' }, source: 'backup' }),
        ];
        [nockedBitGoKeychain, nockedUserKeychain] = await Promise.all(nockPromises);
        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.EcdsaUtils(bitgo, baseCoin, wallet);
    });
    after(function () {
        nock.cleanAll();
    });
    describe('TSS key chains', async function () {
        it('should create backup key share held by BitGo', async function () {
            const enterpriseId = 'enterprise id';
            const expectedKeyShare = await nockCreateBitgoHeldBackupKeyShare(coinName, enterpriseId, userGpgKey, backupKeyShare, bitGoGPGKeyPair);
            const result = await tssUtils.createBitgoHeldBackupKeyShare(userGpgKey, enterpriseId);
            result.should.eql(expectedKeyShare);
        });
        it('should finalize backup key share held by BitGo', async function () {
            const commonKeychain = '4428';
            const originalKeyShare = await createIncompleteBitgoHeldBackupKeyShare(userGpgKey, backupKeyShare, bitGoGPGKeyPair);
            const expectedFinalKeyShare = await nockFinalizeBitgoHeldBackupKeyShare(coinName, originalKeyShare, commonKeychain, userKeyShare, bitGoGPGKeyPair, nockedBitGoKeychain);
            const result = await tssUtils.finalizeBitgoHeldBackupKeyShare(originalKeyShare.id, commonKeychain, userKeyShare, nockedBitGoKeychain, userGpgKey, bitgoPublicKey);
            result.should.eql(expectedFinalKeyShare);
        });
        it('should get the respective backup key shares based on provider', async function () {
            const enterpriseId = 'enterprise id';
            await nockCreateBitgoHeldBackupKeyShare(coinName, enterpriseId, userGpgKey, backupKeyShare, bitGoGPGKeyPair);
            const backupKeyShares = await tssUtils.createBackupKeyShares();
            should.exist(backupKeyShares.userHeldKeyShare);
            should.not.exist(backupKeyShares.bitGoHeldKeyShares);
        });
        it('should get the correct bitgo gpg key based on coin and feature flags', async function () {
            const nitroGPGKeypair = await openpgp.generateKey({
                userIDs: [
                    {
                        name: 'bitgo nitro',
                        email: 'bitgo@test.com',
                    },
                ],
            });
            const nockGPGKey = await nockGetBitgoPublicKeyBasedOnFeatureFlags(coinName, 'enterprise_id', nitroGPGKeypair);
            const bitgoGpgPublicKey = await tssUtils.getBitgoGpgPubkeyBasedOnFeatureFlags('enterprise_id');
            should.equal(nockGPGKey.publicKey, bitgoGpgPublicKey.armor());
        });
        it('getBackupEncryptedNShare should get valid encrypted n shares based on provider', async function () {
            const bitgoGpgKeyPubKey = await tssUtils.getBitgoPublicGpgKey();
            // Backup key held by user
            const backupShareHolderNew = {
                userHeldKeyShare: backupKeyShare,
            };
            const backupToBitgoEncryptedNShare = await tssUtils.getBackupEncryptedNShare(backupShareHolderNew, 3, bitgoGpgKeyPubKey.armor(), userGpgKey);
            const encryptedNShare = await encryptNShare(backupKeyShare, 3, bitgoGpgKeyPubKey.armor(), userGpgKey);
            // cant verify the encrypted shares, since they will be encrypted with diff. values
            should.equal(backupToBitgoEncryptedNShare.publicShare, encryptedNShare.publicShare);
        });
        it('should generate TSS key chains', async function () {
            const backupShareHolder = {
                userHeldKeyShare: backupKeyShare,
            };
            const backupGpgKey = userLocalBackupGpgKey;
            const bitgoKeychain = await tssUtils.createBitgoKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare: backupShareHolder,
                bitgoPublicGpgKey: bitgoPublicKey,
            });
            const usersKeyChainPromises = [
                tssUtils.createParticipantKeychain(userGpgKey, userLocalBackupGpgKey, bitgoPublicKey, 1, userKeyShare, backupKeyShare, bitgoKeychain, 'passphrase'),
                tssUtils.createParticipantKeychain(userGpgKey, userLocalBackupGpgKey, bitgoPublicKey, 2, userKeyShare, backupKeyShare, bitgoKeychain, 'passphrase'),
            ];
            const [userKeychain, backupKeychain] = await Promise.all(usersKeyChainPromises);
            bitgoKeychain.should.deepEqual(nockedBitGoKeychain);
            userKeychain.should.deepEqual(nockedUserKeychain);
            // unencrypted `prv` property should exist on backup keychain
            const keyChainPrv = JSON.parse(backupKeychain.prv ?? '');
            _.isEqual(keyChainPrv.pShare, backupKeyShare.pShare).should.be.true();
            _.isEqual(keyChainPrv.bitgoNShare, bitgoKeyShare.nShares[2]).should.be.true();
            _.isEqual(keyChainPrv.userNShare, userKeyShare.nShares[2]).should.be.true();
            should.exist(backupKeychain.encryptedPrv);
        });
        it('should generate TSS key chains with optional params', async function () {
            const enterprise = 'enterprise_id';
            const backupShareHolder = {
                userHeldKeyShare: backupKeyShare,
            };
            const backupGpgKey = userLocalBackupGpgKey;
            const bitgoKeychain = await tssUtils.createBitgoKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare: backupShareHolder,
                enterprise,
                bitgoPublicGpgKey: bitgoPublicKey,
            });
            const usersKeyChainPromises = [
                tssUtils.createParticipantKeychain(userGpgKey, userLocalBackupGpgKey, bitgoPublicKey, 1, userKeyShare, backupKeyShare, bitgoKeychain, 'passphrase', 'originalPasscodeEncryptionCode'),
                tssUtils.createParticipantKeychain(userGpgKey, userLocalBackupGpgKey, bitgoPublicKey, 2, userKeyShare, backupKeyShare, bitgoKeychain, 'passphrase'),
            ];
            const [userKeychain, backupKeychain] = await Promise.all(usersKeyChainPromises);
            bitgoKeychain.should.deepEqual(nockedBitGoKeychain);
            userKeychain.should.deepEqual(nockedUserKeychain);
            // unencrypted `prv` property should exist on backup keychain
            const keyChainPrv = JSON.parse(backupKeychain.prv ?? '');
            _.isEqual(keyChainPrv.pShare, backupKeyShare.pShare).should.be.true();
            _.isEqual(keyChainPrv.bitgoNShare, bitgoKeyShare.nShares[2]).should.be.true();
            _.isEqual(keyChainPrv.userNShare, userKeyShare.nShares[2]).should.be.true();
            should.exist(backupKeychain.encryptedPrv);
        });
        it('should fail to generate TSS key chains', async function () {
            const backupShareHolder = {
                userHeldKeyShare: backupKeyShare,
            };
            const backupGpgKey = userLocalBackupGpgKey;
            const bitgoKeychain = await tssUtils.createBitgoKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare: backupShareHolder,
                bitgoPublicGpgKey: bitgoPublicKey,
            });
            bitgoKeychain.should.deepEqual(nockedBitGoKeychain);
            const testKeyShares = ecdsaFixtures_1.otherKeyShares;
            const testCasesPromises = [
                tssUtils
                    .createParticipantKeychain(userGpgKey, userLocalBackupGpgKey, bitgoPublicKey, 1, userKeyShare, testKeyShares[0], bitgoKeychain, 'passphrase')
                    .should.be.rejectedWith('Common keychains do not match'),
                tssUtils
                    .createParticipantKeychain(userGpgKey, userLocalBackupGpgKey, bitgoPublicKey, 1, testKeyShares[1], backupKeyShare, bitgoKeychain, 'passphrase')
                    .should.be.rejectedWith('Common keychains do not match'),
                tssUtils
                    .createParticipantKeychain(userGpgKey, userLocalBackupGpgKey, bitgoPublicKey, 2, testKeyShares[2], backupKeyShare, bitgoKeychain, 'passphrase')
                    .should.be.rejectedWith('Common keychains do not match'),
                tssUtils
                    .createParticipantKeychain(userGpgKey, userLocalBackupGpgKey, bitgoPublicKey, 2, userKeyShare, testKeyShares[3], bitgoKeychain, 'passphrase')
                    .should.be.rejectedWith('Common keychains do not match'),
            ];
            await Promise.all(testCasesPromises);
        });
        it('should fail to generate TSS keychains when received invalid number of wallet signatures', async function () {
            const bitgoKeychain = await generateBitgoKeychain({
                coin: coinName,
                userKeyShare,
                backupKeyShare,
                bitgoKeyShare,
                userGpgKey,
                userLocalBackupGpgKey,
                bitgoGpgKey: bitGoGPGKeyPair,
            });
            const certsString = await (0, src_1.createSharedDataProof)(bitGoGPGKeyPair.privateKey, userGpgKey.publicKey, []);
            const certsKey = await openpgp.readKey({ armoredKey: certsString });
            const finalKey = new openpgp.PacketList();
            certsKey.toPacketList().forEach((packet) => finalKey.push(packet));
            // Once the following PR has been merged and released we no longer need the ts-ignore:
            // https://github.com/openpgpjs/openpgpjs/pull/1576
            // eslint-disable-next-line @typescript-eslint/ban-ts-comment
            // @ts-ignore
            bitgoKeychain.walletHSMGPGPublicKeySigs = openpgp.armor(openpgp.enums.armor.publicKey, finalKey.write());
            await tssUtils
                .verifyWalletSignatures(userLocalBackupGpgKey.publicKey, userLocalBackupGpgKey.publicKey, bitgoKeychain, '', 1)
                .should.be.rejectedWith(`Invalid wallet signatures`);
        });
        it('should fail to generate TSS keychains when wallet signature fingerprints do not match passed user/backup fingerprints', async function () {
            const customUserKeyShare = await MPC.keyShare(1, 2, 3);
            const customBackupKeyShare = await MPC.keyShare(2, 2, 3);
            const backupShareHolder = {
                userHeldKeyShare: customBackupKeyShare,
            };
            const backupGpgKey = userLocalBackupGpgKey;
            const bitgoKeychain = await tssUtils.createBitgoKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare: customUserKeyShare,
                backupKeyShare: backupShareHolder,
                bitgoPublicGpgKey: bitgoPublicKey,
            });
            // using the backup gpg here instead of the user gpg key to simulate that the first signature has a different
            // fingerprint from the passed in first gpg key
            await tssUtils
                .verifyWalletSignatures(userLocalBackupGpgKey.publicKey, userLocalBackupGpgKey.publicKey, bitgoKeychain, '', 1)
                .should.be.rejectedWith(`first wallet signature's fingerprint does not match passed user gpg key's fingerprint`);
            // using the user gpg here instead of the backup gpg key to simulate that the second signature has a different
            // fingerprint from the passed in second gpg key
            await tssUtils
                .verifyWalletSignatures(userGpgKey.publicKey, userGpgKey.publicKey, bitgoKeychain, '', 1)
                .should.be.rejectedWith(`second wallet signature's fingerprint does not match passed backup gpg key's fingerprint`);
        });
    });
    describe('signTxRequest:', () => {
        const txRequestId = 'randomidEcdsa';
        const txRequest = {
            txRequestId,
            transactions: [
                {
                    unsignedTx: {
                        // hteth txid: 0xc5a7bfe6b13ceae563da0f9feaa9c4ad1c101a15366a2a488828a5dd27cb9da3
                        serializedTxHex: '02f38242688084448b9b8084448b9b908301637894a1cfb9d51c0af191ff21c5f0f01723e056f7dc12865af3107a400080c0808080',
                        signableHex: '02f08242688084448b9b8084448b9b908301637894a1cfb9d51c0af191ff21c5f0f01723e056f7dc12865af3107a400080c0',
                        derivationPath: '', // Needs this when key derivation is supported
                    },
                    state: 'pendingSignature',
                    signatureShares: [],
                },
            ],
            unsignedTxs: [
                {
                    // hteth txid: 0xc5a7bfe6b13ceae563da0f9feaa9c4ad1c101a15366a2a488828a5dd27cb9da3
                    serializedTxHex: '02f38242688084448b9b8084448b9b908301637894a1cfb9d51c0af191ff21c5f0f01723e056f7dc12865af3107a400080c0808080',
                    signableHex: '02f38242688084448b9b8084448b9b908301637894a1cfb9d51c0af191ff21c5f0f01723e056f7dc12865af3107a400080c0808080',
                    derivationPath: '', // Needs this when key derivation is supported
                },
            ],
            date: new Date().toISOString(),
            intent: {
                intentType: 'payment',
            },
            latest: true,
            state: 'pendingUserSignature',
            walletType: 'hot',
            walletId: 'walletId',
            policiesChecked: true,
            version: 1,
            userId: 'userId',
        };
        let aShare, dShare, wShare, oShare, userSignShare, bitgoChallenges, enterpriseChallenges;
        beforeEach(async () => {
            // Initializing user and bitgo for creating shares for nocks
            const userSigningKey = MPC.keyCombine(userKeyShare.pShare, [bitgoKeyShare.nShares[1], backupKeyShare.nShares[1]]);
            const bitgoSigningKey = MPC.keyCombine(bitgoKeyShare.pShare, [
                userKeyShare.nShares[3],
                backupKeyShare.nShares[3],
            ]);
            const serializedEntChallenge = ecdsaNtilde_1.mockChallengeA;
            const serializedBitgoChallenge = ecdsaNtilde_1.mockChallengeB;
            const deserializedEntChallenge = sdk_lib_mpc_1.EcdsaTypes.deserializeNtildeWithProofs(serializedEntChallenge);
            sinon.stub(sdk_lib_mpc_1.EcdsaRangeProof, 'generateNtilde').resolves(deserializedEntChallenge);
            const [userToBitgoPaillierChallenge, bitgoToUserPaillierChallenge] = await Promise.all([
                sdk_lib_mpc_1.EcdsaPaillierProof.generateP((0, sdk_lib_mpc_1.hexToBigInt)(userSigningKey.yShares[3].n)),
                sdk_lib_mpc_1.EcdsaPaillierProof.generateP((0, sdk_lib_mpc_1.hexToBigInt)(bitgoSigningKey.yShares[1].n)),
            ]);
            bitgoChallenges = {
                ...serializedBitgoChallenge,
                p: sdk_lib_mpc_1.EcdsaTypes.serializePaillierChallenge({ p: bitgoToUserPaillierChallenge }).p,
                n: bitgoSigningKey.xShare.n,
            };
            enterpriseChallenges = {
                ...serializedEntChallenge,
                p: sdk_lib_mpc_1.EcdsaTypes.serializePaillierChallenge({ p: userToBitgoPaillierChallenge }).p,
                n: bitgoSigningKey.xShare.n,
            };
            sinon.stub(sdk_core_1.ECDSAUtils.EcdsaUtils.prototype, 'getEcdsaSigningChallenges').resolves({
                enterpriseChallenge: enterpriseChallenges,
                bitgoChallenge: bitgoChallenges,
            });
            const [userXShare, bitgoXShare] = [
                MPC.appendChallenge(userSigningKey.xShare, serializedEntChallenge, sdk_lib_mpc_1.EcdsaTypes.serializePaillierChallenge({ p: userToBitgoPaillierChallenge })),
                MPC.appendChallenge(bitgoSigningKey.xShare, serializedBitgoChallenge, sdk_lib_mpc_1.EcdsaTypes.serializePaillierChallenge({ p: bitgoToUserPaillierChallenge })),
            ];
            const bitgoYShare = MPC.appendChallenge(userSigningKey.yShares[3], serializedBitgoChallenge, sdk_lib_mpc_1.EcdsaTypes.serializePaillierChallenge({ p: bitgoToUserPaillierChallenge }));
            /**
             * START STEP ONE
             * 1) User creates signShare, saves wShare and sends kShare to bitgo
             * 2) Bitgo performs signConvert operation using its private xShare , yShare
             *  and KShare from user and responds back with aShare and saves bShare for later use
             */
            userSignShare = await sdk_core_1.ECDSAMethods.createUserSignShare(userXShare, bitgoYShare);
            wShare = userSignShare.wShare;
            const signatureShareOneFromUser = {
                from: sdk_core_1.SignatureShareType.USER,
                to: sdk_core_1.SignatureShareType.BITGO,
                share: sdk_core_1.ECDSAMethods.convertKShare(userSignShare.kShare).share.replace(sdk_core_1.ECDSAMethods.delimeter, ''),
            };
            const getBitgoAandBShare = await MPC.signConvertStep1({
                xShare: bitgoXShare,
                yShare: bitgoSigningKey.yShares[1], // corresponds to the user
                kShare: userSignShare.kShare,
            });
            const bitgoAshare = getBitgoAandBShare.aShare;
            aShare = bitgoAshare;
            const aShareBitgoResponse = sdk_core_1.ECDSAMethods.convertAShare(bitgoAshare).share.replace(sdk_core_1.ECDSAMethods.delimeter, '');
            const signatureShareOneFromBitgo = {
                from: sdk_core_1.SignatureShareType.BITGO,
                to: sdk_core_1.SignatureShareType.USER,
                share: aShareBitgoResponse,
            };
            await (0, common_1.nockSendSignatureShareWithResponse)({
                walletId: wallet.id(),
                txRequestId: txRequest.txRequestId,
                signatureShare: signatureShareOneFromUser,
                response: signatureShareOneFromBitgo,
                tssType: 'ecdsa',
            });
            /**  END STEP ONE */
            /**
             * START STEP TWO
             * 1) Using the aShare got from bitgo and wShare from previous step,
             * user creates gShare and muShare and sends muShare to bitgo
             * 2) Bitgo using the signConvert step using bShare from previous step
             * and muShare from user generates its gShare.
             * 3) Using the signCombine operation using gShare, Bitgo generates oShare
             * which it saves and dShare which is send back to the user.
             */
            const userGammaAndMuShares = await sdk_core_1.ECDSAMethods.createUserGammaAndMuShare(userSignShare.wShare, bitgoAshare);
            const signatureShareTwoFromUser = {
                from: sdk_core_1.SignatureShareType.USER,
                to: sdk_core_1.SignatureShareType.BITGO,
                share: sdk_core_1.ECDSAMethods.convertMuShare(userGammaAndMuShares.muShare).share.replace(sdk_core_1.ECDSAMethods.delimeter, ''),
            };
            const getBitGoGShareAndSignerIndexes = await MPC.signConvertStep3({
                bShare: getBitgoAandBShare.bShare,
                muShare: userGammaAndMuShares.muShare,
            });
            const getBitgoOShareAndDShares = MPC.signCombine({
                gShare: getBitGoGShareAndSignerIndexes.gShare,
                signIndex: {
                    i: 1,
                    j: 3,
                },
            });
            const bitgoDshare = getBitgoOShareAndDShares.dShare;
            dShare = bitgoDshare;
            const dShareBitgoResponse = bitgoDshare.delta + bitgoDshare.Gamma;
            const signatureShareTwoFromBitgo = {
                from: sdk_core_1.SignatureShareType.BITGO,
                to: sdk_core_1.SignatureShareType.USER,
                share: dShareBitgoResponse,
            };
            await (0, common_1.nockSendSignatureShareWithResponse)({
                walletId: wallet.id(),
                txRequestId: txRequest.txRequestId,
                signatureShare: signatureShareTwoFromUser,
                response: signatureShareTwoFromBitgo,
                tssType: 'ecdsa',
            });
            /**  END STEP TWO */
            /**
             * START STEP THREE
             * 1) User creates its oShare and  dShare using the  private gShare
             * from step two
             * 2) User uses the private oShare and dShare from bitgo from step
             * two to generate its signature share which it sends back along with dShare that
             * user generated from the above step
             * 3) Bitgo using its private oShare from step two and dShare from bitgo creates
             * its signature share. Using the Signature Share received from user from the above
             * step, bitgo constructs the final signature and is returned to the user
             */
            const userOmicronAndDeltaShare = await sdk_core_1.ECDSAMethods.createUserOmicronAndDeltaShare(userGammaAndMuShares.gShare);
            oShare = userOmicronAndDeltaShare.oShare;
            const signablePayload = Buffer.from(txRequest.unsignedTxs[0].signableHex, 'hex');
            const userSShare = await sdk_core_1.ECDSAMethods.createUserSignatureShare(userOmicronAndDeltaShare.oShare, bitgoDshare, signablePayload);
            const signatureShareThreeFromUser = {
                from: sdk_core_1.SignatureShareType.USER,
                to: sdk_core_1.SignatureShareType.BITGO,
                share: userSShare.R +
                    userSShare.s +
                    userSShare.y +
                    userOmicronAndDeltaShare.dShare.delta +
                    userOmicronAndDeltaShare.dShare.Gamma,
            };
            const getBitGoSShare = MPC.sign(signablePayload, getBitgoOShareAndDShares.oShare, userOmicronAndDeltaShare.dShare, createKeccakHash('keccak256'));
            const getBitGoFinalSignature = MPC.constructSignature([getBitGoSShare, userSShare]);
            const finalSigantureBitgoResponse = getBitGoFinalSignature.r + getBitGoFinalSignature.s + getBitGoFinalSignature.y;
            const signatureShareThreeFromBitgo = {
                from: sdk_core_1.SignatureShareType.BITGO,
                to: sdk_core_1.SignatureShareType.USER,
                share: finalSigantureBitgoResponse,
            };
            await (0, common_1.nockSendSignatureShareWithResponse)({
                walletId: wallet.id(),
                txRequestId: txRequest.txRequestId,
                signatureShare: signatureShareThreeFromUser,
                response: signatureShareThreeFromBitgo,
                tssType: 'ecdsa',
            });
            /* END STEP THREE */
            const signature = MPC.constructSignature([userSShare, getBitGoSShare]);
            MPC.verify(signablePayload, signature, createKeccakHash('keccak256')).should.be.true;
        });
        afterEach(async () => {
            sinon.restore();
        });
        it('signTxRequest should fail if wallet is in pendingEcdsaTssInitialization', async function () {
            sandbox.stub(wallet, 'coinSpecific').returns({
                customChangeWalletId: '',
                pendingEcdsaTssInitialization: true,
            });
            await tssUtils
                .signTxRequest({
                txRequest,
                prv: JSON.stringify({
                    pShare: userKeyShare.pShare,
                    bitgoNShare: bitgoKeyShare.nShares[1],
                    backupNShare: backupKeyShare.nShares[1],
                }),
                reqId,
            })
                .should.be.rejectedWith('Wallet is not ready for TSS ECDSA signing. Please contact your enterprise admin to finish the enterprise TSS initialization.');
        });
        it('signTxRequest should succeed with txRequest object as input', async function () {
            const sendShareSpy = sinon.spy(sdk_core_1.ECDSAMethods, 'sendShareToBitgo');
            await setupSignTxRequestNocks(false, userSignShare, aShare, dShare, enterpriseData);
            const signedTxRequest = await tssUtils.signTxRequest({
                txRequest,
                prv: JSON.stringify({
                    pShare: userKeyShare.pShare,
                    bitgoNShare: bitgoKeyShare.nShares[1],
                    backupNShare: backupKeyShare.nShares[1],
                }),
                reqId,
            });
            signedTxRequest.unsignedTxs.should.deepEqual(txRequest.unsignedTxs);
            const userGpgActual = sendShareSpy.getCalls()[0].args[10];
            userGpgActual.should.startWith('-----BEGIN PGP PUBLIC KEY BLOCK-----');
        });
        it('signTxRequest should succeed with txRequest id as input', async function () {
            const sendShareSpy = sinon.spy(sdk_core_1.ECDSAMethods, 'sendShareToBitgo');
            await setupSignTxRequestNocks(true, userSignShare, aShare, dShare, enterpriseData);
            const signedTxRequest = await tssUtils.signTxRequest({
                txRequest: txRequestId,
                prv: JSON.stringify({
                    pShare: userKeyShare.pShare,
                    bitgoNShare: bitgoKeyShare.nShares[1],
                    backupNShare: backupKeyShare.nShares[1],
                }),
                reqId,
            });
            signedTxRequest.unsignedTxs.should.deepEqual(txRequest.unsignedTxs);
            const userGpgActual = sendShareSpy.getCalls()[0].args[10];
            userGpgActual.should.startWith('-----BEGIN PGP PUBLIC KEY BLOCK-----');
        });
        it('signTxRequest should fail with wrong recipient', async function () {
            await setupSignTxRequestNocks(true, userSignShare, aShare, dShare, enterpriseData);
            await tssUtils
                .signTxRequest({
                txRequest: txRequestId,
                prv: JSON.stringify({
                    pShare: userKeyShare.pShare,
                    bitgoNShare: bitgoKeyShare.nShares[1],
                    backupNShare: backupKeyShare.nShares[1],
                }),
                reqId,
                txParams: { recipients: [{ address: '0x1234', amount: '100000000000000' }], type: 'transfer' },
            })
                .should.be.rejectedWith('destination address does not match with the recipient address');
        });
        it('signTxRequest should fail with incorrect value', async function () {
            await setupSignTxRequestNocks(true, userSignShare, aShare, dShare, enterpriseData);
            await tssUtils
                .signTxRequest({
                txRequest: txRequestId,
                prv: JSON.stringify({
                    pShare: userKeyShare.pShare,
                    bitgoNShare: bitgoKeyShare.nShares[1],
                    backupNShare: backupKeyShare.nShares[1],
                }),
                reqId,
                txParams: {
                    recipients: [{ address: '0xa1cfb9d51c0af191ff21c5f0f01723e056f7dc12', amount: '1' }],
                    type: 'transfer',
                },
            })
                .should.be.rejectedWith('the transaction amount in txPrebuild does not match the value given by client');
        });
        it('signTxRequest should fail with incorrect value for token txn', async function () {
            const signableHex = '02f86d8242681083122c9e83122cae8301e04994ebe8b46a42f05072b723b00013ff822b2af1b5cb80b844a9059cbb0000000000000000000000002b0d6cb2f8c388757f4d7ad857fccab18290dbc900000000000000000000000000000000000000000000000000000000000186a0c0';
            const serializedTxHex = '02f8708242681083122c9e83122cae8301e04994ebe8b46a42f05072b723b00013ff822b2af1b5cb80b844a9059cbb0000000000000000000000002b0d6cb2f8c388757f4d7ad857fccab18290dbc900000000000000000000000000000000000000000000000000000000000186a0c0808080';
            await setupSignTxRequestNocks(true, userSignShare, aShare, dShare, enterpriseData, {
                signableHex,
                serializedTxHex,
                apiVersion: 'full',
            });
            await tssUtils
                .signTxRequest({
                txRequest: txRequestId,
                prv: JSON.stringify({
                    pShare: userKeyShare.pShare,
                    bitgoNShare: bitgoKeyShare.nShares[1],
                    backupNShare: backupKeyShare.nShares[1],
                }),
                reqId,
                txParams: {
                    recipients: [{ address: '0x2b0d6cb2f8c388757f4d7ad857fccab18290dbc9', amount: '707' }],
                    type: 'transfer',
                },
            })
                .should.be.rejectedWith('the transaction amount in txPrebuild does not match the value given by client');
        });
        it('getOfflineSignerPaillierModulus should succeed', async function () {
            const paillierModulus = tssUtils.getOfflineSignerPaillierModulus({
                prv: JSON.stringify({
                    pShare: userKeyShare.pShare,
                    bitgoNShare: bitgoKeyShare.nShares[1],
                    backupNShare: backupKeyShare.nShares[1],
                }),
            });
            paillierModulus.userPaillierModulus.should.equal(userKeyShare.pShare.n);
        });
        it('createOfflineKShare should succeed', async function () {
            const mockPassword = 'password';
            const step1SigningMaterial = await tssUtils.createOfflineKShare({
                tssParams: {
                    txRequest,
                    reqId: reqId,
                },
                challenges: {
                    enterpriseChallenge: enterpriseChallenges,
                    bitgoChallenge: bitgoChallenges,
                },
                prv: JSON.stringify({
                    pShare: userKeyShare.pShare,
                    bitgoNShare: bitgoKeyShare.nShares[1],
                    backupNShare: backupKeyShare.nShares[1],
                }),
                requestType: src_1.RequestType.tx,
                walletPassphrase: mockPassword,
            });
            step1SigningMaterial.privateShareProof.should.startWith('-----BEGIN PGP PUBLIC KEY BLOCK-----');
            step1SigningMaterial.vssProof?.length.should.equal(userKeyShare.nShares[3].v?.length);
            step1SigningMaterial.publicShare.length.should.equal(userKeyShare.nShares[3].y.length + userKeyShare.nShares[3].chaincode.length);
            step1SigningMaterial.encryptedSignerOffsetShare.should.startWith('-----BEGIN PGP MESSAGE-----');
            step1SigningMaterial.userPublicGpgKey.should.startWith('-----BEGIN PGP PUBLIC KEY BLOCK-----');
            step1SigningMaterial.kShare.n.should.equal(userKeyShare.pShare.n);
            step1SigningMaterial.wShare.should.startWith('{"iv":');
        });
        it('createOfflineKShare should fail with txId passed', async function () {
            const mockPassword = 'password';
            await tssUtils
                .createOfflineKShare({
                tssParams: {
                    txRequest: txRequest.txRequestId,
                    reqId: reqId,
                },
                challenges: {
                    enterpriseChallenge: enterpriseChallenges,
                    bitgoChallenge: bitgoChallenges,
                },
                prv: JSON.stringify({
                    pShare: userKeyShare.pShare,
                    bitgoNShare: bitgoKeyShare.nShares[1],
                    backupNShare: backupKeyShare.nShares[1],
                }),
                requestType: src_1.RequestType.tx,
                walletPassphrase: mockPassword,
            })
                .should.be.rejectedWith('Invalid txRequest type');
        });
        // Seems to be flaky on CI, failed here: https://github.com/BitGo/BitGoJS/actions/runs/5902489990/job/16010623888?pr=3822
        xit('createOfflineMuDeltaShare should succeed', async function () {
            const mockPassword = 'password';
            const alphaLength = 1536;
            const deltaLength = 64;
            const bitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' });
            const step2SigningMaterial = await tssUtils.createOfflineMuDeltaShare({
                aShareFromBitgo: aShare,
                bitgoChallenge: bitgoChallenges,
                encryptedWShare: bitgo.encrypt({ input: JSON.stringify(wShare), password: mockPassword }),
                walletPassphrase: mockPassword,
            });
            step2SigningMaterial.muDShare.muShare.alpha.length.should.equal(alphaLength);
            step2SigningMaterial.muDShare.dShare.delta.length.should.equal(deltaLength);
            step2SigningMaterial.oShare.should.startWith('{"iv":');
        });
        it('createOfflineMuDeltaShare should fail with incorrect password', async function () {
            const mockPassword = 'password';
            const bitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' });
            await tssUtils
                .createOfflineMuDeltaShare({
                aShareFromBitgo: aShare,
                bitgoChallenge: bitgoChallenges,
                encryptedWShare: bitgo.encrypt({ input: JSON.stringify(wShare), password: mockPassword }),
                walletPassphrase: 'password1',
            })
                .should.be.rejectedWith("password error - ccm: tag doesn't match");
        });
        it('createOfflineSShare should succeed', async function () {
            const mockPassword = 'password';
            const pubKeyLength = 66;
            const privKeyLength = 64;
            const bitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' });
            const step3SigningMaterial = await tssUtils.createOfflineSShare({
                tssParams: {
                    txRequest: txRequest,
                    reqId: reqId,
                },
                dShareFromBitgo: dShare,
                encryptedOShare: bitgo.encrypt({ input: JSON.stringify(oShare), password: mockPassword }),
                walletPassphrase: mockPassword,
                requestType: src_1.RequestType.tx,
            });
            step3SigningMaterial.R.length.should.equal(pubKeyLength);
            step3SigningMaterial.y.length.should.equal(pubKeyLength);
            step3SigningMaterial.s.length.should.equal(privKeyLength);
        });
        it('createOfflineSShare should fail with txId passed', async function () {
            const mockPassword = 'password';
            const bitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' });
            await tssUtils
                .createOfflineSShare({
                tssParams: {
                    txRequest: txRequest.txRequestId,
                    reqId: reqId,
                },
                dShareFromBitgo: dShare,
                encryptedOShare: bitgo.encrypt({ input: JSON.stringify(oShare), password: mockPassword }),
                walletPassphrase: mockPassword,
                requestType: src_1.RequestType.tx,
            })
                .should.be.rejectedWith('Invalid txRequest type');
        });
        it('signTxRequest should fail with invalid user prv', async function () {
            const invalidUserKey = { ...userKeyShare, pShare: { ...userKeyShare.pShare, i: 2 } };
            await tssUtils
                .signTxRequest({
                txRequest: txRequestId,
                prv: JSON.stringify({
                    pShare: invalidUserKey.pShare,
                    bitgoNShare: bitgoKeyShare.nShares[1],
                    backupNShare: backupKeyShare.nShares[1],
                }),
                reqId,
            })
                .should.be.rejectedWith('Invalid user key');
        });
        it('signTxRequest should fail with no backupNShares', async function () {
            const getTxRequest = sandbox.stub(tssUtils, 'getTxRequest');
            getTxRequest.resolves(txRequest);
            getTxRequest.calledWith(txRequestId);
            setupSignTxRequestNocks(false, userSignShare, aShare, dShare, enterpriseData);
            await tssUtils
                .signTxRequest({
                txRequest: txRequestId,
                prv: JSON.stringify({
                    pShare: userKeyShare.pShare,
                    bitgoNShare: bitgoKeyShare.nShares[1],
                }),
                reqId,
            })
                .should.be.rejectedWith('Invalid user key - missing backupNShare');
        });
        async function setupSignTxRequestNocks(isTxRequest = true, userSignShare, aShare, dShare, enterpriseData, { signableHex, serializedTxHex, apiVersion, } = {}) {
            if (enterpriseData) {
                await (0, helpers_1.nockGetEnterprise)({ enterpriseId: enterpriseData.id, response: enterpriseData, times: 1 });
            }
            const derivationPath = '';
            sinon.stub(sdk_core_1.ECDSAMethods, 'createUserSignShare').resolves(userSignShare);
            let response = {
                txRequests: [
                    {
                        ...txRequest,
                        transactions: [
                            {
                                ...txRequest,
                                unsignedTx: {
                                    signableHex: signableHex ?? txRequest.unsignedTxs[0].signableHex,
                                    serializedTxHex: serializedTxHex ?? txRequest.unsignedTxs[0].serializedTxHex,
                                    derivationPath,
                                },
                            },
                        ],
                        apiVersion: apiVersion,
                    },
                ],
            };
            if (isTxRequest) {
                await (0, helpers_1.nockGetTxRequest)({ walletId: wallet.id(), txRequestId: txRequest.txRequestId, response: response });
            }
            const aRecord = sdk_core_1.ECDSAMethods.convertAShare(aShare);
            const signatureShares = [aRecord];
            txRequest.signatureShares = signatureShares;
            response = {
                txRequests: [
                    {
                        ...txRequest,
                        transactions: [
                            {
                                ...txRequest,
                                unsignedTx: {
                                    signableHex: txRequest.unsignedTxs[0].signableHex,
                                    serializedTxHex: txRequest.unsignedTxs[0].serializedTxHex,
                                    derivationPath,
                                },
                            },
                        ],
                        apiVersion: apiVersion,
                    },
                ],
            };
            await (0, helpers_1.nockGetTxRequest)({ walletId: wallet.id(), txRequestId: txRequest.txRequestId, response: response });
            const dRecord = sdk_core_1.ECDSAMethods.convertDShare(dShare);
            signatureShares.push(dRecord);
            response = {
                txRequests: [
                    {
                        ...txRequest,
                        transactions: [
                            {
                                ...txRequest,
                                unsignedTx: {
                                    signableHex: txRequest.unsignedTxs[0].signableHex,
                                    serializedTxHex: txRequest.unsignedTxs[0].serializedTxHex,
                                    derivationPath,
                                },
                            },
                        ],
                        apiVersion: apiVersion,
                    },
                ],
            };
            await (0, helpers_1.nockGetTxRequest)({ walletId: wallet.id(), txRequestId: txRequest.txRequestId, response: response });
            await (0, helpers_1.nockGetTxRequest)({ walletId: wallet.id(), txRequestId: txRequest.txRequestId, response: response });
        }
    });
    describe('getEcdsaSigningChallenges', function () {
        const mockWalletPaillierKey = {
            n: 'f47be4c2d8bc1e28f88c6c4da634da97d92a1c279a7b0fe7b87c337c36a27b32ce0ff0c45f16e4e15bbd20e4e640de12047eff9b1a2b98144f9a268d406bd000d192a35b6847a17e40fb85f55b314d001ff87393481cafe391807d0eb83eff9e38614b38e5f25fc4449cb01caed805584d026b5d866c723f3d4d4f1e462662f2113b1561eb2bf755b4b91d0308d8eacc439167da8b7d6e108524f226960360af00215d9614457414ebdbe8834999689e2e903208c8713ff5d9901f9eaba3aa81d705323cbbba61ba7fa9f3228f30853fb55da1b3d3ed7db1dfc6545bc96aa8d2eb848931c1b807fdfe8f65af72f68638a82fe9e22ac1f0f032e621066806a1f144b5719a5f091986867b384be6c34146c8241cbfbd781966ebbcd19e6caa27fab040e62e5a162888aa8624d046c8fe3b72244f04a7264c4a36b6366dbe7da98afb201d34be2c0d6dd11982af35bf7535582b263914725aaec280d52290527382d3ab297d746c41aacd8de98c09fcfb85a95e02de1b34d4933e51045e2f1ce8af',
            lambda: 'f47be4c2d8bc1e28f88c6c4da634da97d92a1c279a7b0fe7b87c337c36a27b32ce0ff0c45f16e4e15bbd20e4e640de12047eff9b1a2b98144f9a268d406bd000d192a35b6847a17e40fb85f55b314d001ff87393481cafe391807d0eb83eff9e38614b38e5f25fc4449cb01caed805584d026b5d866c723f3d4d4f1e462662f2113b1561eb2bf755b4b91d0308d8eacc439167da8b7d6e108524f226960360af00215d9614457414ebdbe8834999689e2e903208c8713ff5d9901f9eaba3aa7fc3d0c0bcc5bff644156ab887146d51bcee1eef70f45c486147d687ee37def1f8a16bc945eff22dd4dca3614a99158823acd9492e347f7ec79a7771024205d07f27b30cd20340e330411da8fa2da209e5cc688da94d1dbef54bfd9c69b4e99cf06d67309a3420b82c78a0fe0dd0b9c31382eae38746cfdd27fa90022a50532246c8ae1339c93e183c03bf6fd7014be3658abc73baae1fa5b86dab94b9f125395a818e54dde6235c45d3dbc032b3078e9df1cad69d8ac19a7cb6405a558b7bfba8',
        };
        const mockBitgoPaillierKey = {
            n: 'f010d294effceb8c4f96af1978ab367c4fbb272c2169317e41ae87220652cae2ce929696ee55ec6831aa6b4b3b931babc2bac9c1a20fddbca925cc99680791f7c3157b3d31256ee72c47d47db567e0f070dce121c3a4d9e003c1f1389073acb252c65d2b0723e86e3265f67a137cb1e23f4551544405644d0ae63d35f25f40becd2b693879f3bdbec3f7250791a3f3c975a5ac78a0e81dcd1a87eb2ca67010dff880b2338556275de23d9e88d21b77da0d524ddc2b394f8de00b1af0ce85f6eee2e05a184e05494d66d2c636045bf70ed15ebd0f41a8eea2920af85e6d68a0ce11fc2abbcb3cebcc3c23ec2e148c318683a5426e15b5207efd3b9b05cb919ec4340f74dff336986d0c923df10a789007b1da9daddf8edf3014e93989f30243f27f9a307d55d630cbfcd16cd6a95a41dee10c31acc293df6834ce0e3ea5b68f170bd7938ea0c2eeb788e16f30af57b3f0888fb44d3610e7eeba60e7fd8cc4a8f044718dfc6174bf4a380690dc1dc77472a48892eb3e81775540ea0acc9e89b639',
            lambda: 'f010d294effceb8c4f96af1978ab367c4fbb272c2169317e41ae87220652cae2ce929696ee55ec6831aa6b4b3b931babc2bac9c1a20fddbca925cc99680791f7c3157b3d31256ee72c47d47db567e0f070dce121c3a4d9e003c1f1389073acb252c65d2b0723e86e3265f67a137cb1e23f4551544405644d0ae63d35f25f40becd2b693879f3bdbec3f7250791a3f3c975a5ac78a0e81dcd1a87eb2ca67010dff880b2338556275de23d9e88d21b77da0d524ddc2b394f8de00b1af0ce85f6ecdef8a4bc955a28ecef7d97cded079d390e77c80998d78ad9510cbabfeb8f0a157dbfc590b4d59ee8c0b088f9d89473b557320078a117478624f5d1df36e30f320b6722a4217dcb46b978cc6c8f1a21c8a6c74bce84d82c481402c99a69b798e3c05f23350b4aade4f79784b1c09692b6a33cfba7f145597d82b799cccef620c36f1fbbe2cb4ac0ea395c476e381bc475d41722320f541ae9bf56aa4a12dff3ea7ab11174fb5b8df7429c9f57d36f8fc51e1a8c647d5b8fa0189fb8acdbd0a780',
        };
        const bitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' });
        const txRequestId = 'fakeTxRequestId';
        const rawEntChallengeWithProofs = sdk_test_1.mockSerializedChallengeWithProofs;
        let rawBitgoChallenge;
        const adminEcdhKey = bitgo.keychains().create();
        const fakeAdminEcdhKey = bitgo.keychains().create();
        const derivationPath = 'm/0/0';
        const mockedSigningKey = {
            userId: 'id',
            userEmail: 'user@bitgo.com',
            derivedPubkey: utxo_lib_1.bip32.fromBase58(adminEcdhKey.xpub).derivePath(derivationPath).publicKey.toString('hex'),
            derivationPath: derivationPath,
            ecdhKeychain: 'my keychain',
        };
        before(async function () {
            const p = await sdk_lib_mpc_1.EcdsaPaillierProof.generateP((0, sdk_lib_mpc_1.hexToBigInt)(mockWalletPaillierKey.n));
            rawBitgoChallenge = {
                ...sdk_lib_mpc_1.EcdsaTypes.serializeNtilde(sdk_lib_mpc_1.EcdsaTypes.deserializeNtilde(sdk_test_1.mockSerializedChallengeWithProofs2)),
                p: sdk_lib_mpc_1.EcdsaTypes.serializePaillierChallenge({ p }).p,
                n: mockBitgoPaillierKey.n,
            };
        });
        afterEach(function () {
            sinon.restore();
            nock.cleanAll();
        });
        it('should fetch static ent and bitgo challenges with the ent feature flag and verify them', async function () {
            await (0, helpers_1.nockGetChallenge)({ walletId, txRequestId, addendum: '/transactions/0', response: rawBitgoChallenge });
            await (0, helpers_1.nockGetSigningKey)({ enterpriseId, userId: mockedSigningKey.userId, response: mockedSigningKey, times: 1 });
            const adminSignatureEntChallenge = sdk_core_1.ECDSAUtils.EcdsaUtils.signChallenge(rawEntChallengeWithProofs, adminEcdhKey.xprv, derivationPath);
            const adminSignatureBitGoChallenge = sdk_core_1.ECDSAUtils.EcdsaUtils.signChallenge(rawBitgoChallenge, adminEcdhKey.xprv, derivationPath);
            const mockChallengesResponse = {
                enterpriseChallenge: {
                    ...rawEntChallengeWithProofs,
                    verifiers: {
                        adminSignature: adminSignatureEntChallenge.toString('hex'),
                    },
                },
                bitgoChallenge: {
                    ...rawBitgoChallenge,
                    verifiers: {
                        adminSignature: adminSignatureBitGoChallenge.toString('hex'),
                    },
                },
                createdBy: 'id',
            };
            await (0, helpers_1.nockGetChallenges)({ walletId: walletId, response: mockChallengesResponse });
            const challenges = await tssUtils.getEcdsaSigningChallenges(txRequestId, 0, mockWalletPaillierKey.n, 0);
            should.exist(challenges);
            const expectedRangeProofChallenges = {
                enterpriseChallenge: {
                    ntilde: challenges.enterpriseChallenge.ntilde,
                    h1: challenges.enterpriseChallenge.h1,
                    h2: challenges.enterpriseChallenge.h2,
                },
                bitgoChallenge: rawBitgoChallenge,
            };
            expectedRangeProofChallenges.should.deepEqual({
                enterpriseChallenge: {
                    ntilde: rawEntChallengeWithProofs.ntilde,
                    h1: rawEntChallengeWithProofs.h1,
                    h2: rawEntChallengeWithProofs.h2,
                },
                bitgoChallenge: rawBitgoChallenge,
            });
        });
        it('Fails if the enterprise challenge signature is different from the admin ecdh key', async function () {
            await (0, helpers_1.nockGetChallenge)({ walletId, txRequestId, addendum: '/transactions/0', response: rawBitgoChallenge });
            await (0, helpers_1.nockGetEnterprise)({
                enterpriseId: enterpriseData.id,
                response: {
                    ...enterpriseData,
                    featureFlags: ['useEnterpriseEcdsaTssChallenge'],
                },
                times: 1,
            });
            await (0, helpers_1.nockGetSigningKey)({ enterpriseId, userId: mockedSigningKey.userId, response: mockedSigningKey, times: 1 });
            // Bad sign
            const adminSignedEntChallenge = sdk_core_1.ECDSAUtils.EcdsaUtils.signChallenge(rawEntChallengeWithProofs, fakeAdminEcdhKey.xprv, derivationPath);
            const adminSignedBitGoChallenge = sdk_core_1.ECDSAUtils.EcdsaUtils.signChallenge(rawBitgoChallenge, adminEcdhKey.xprv, derivationPath);
            const mockChallengesResponse = {
                enterpriseChallenge: {
                    ...rawEntChallengeWithProofs,
                    verifiers: {
                        adminSignature: adminSignedEntChallenge.toString('hex'),
                    },
                },
                bitgoChallenge: {
                    ...rawBitgoChallenge,
                    verifiers: {
                        adminSignature: adminSignedBitGoChallenge.toString('hex'),
                    },
                },
                createdBy: 'id',
            };
            await (0, helpers_1.nockGetChallenges)({ walletId: walletId, response: mockChallengesResponse });
            await tssUtils
                .getEcdsaSigningChallenges(txRequestId, 0, mockWalletPaillierKey.n)
                .should.be.rejectedWith('Admin signature for enterprise challenge is not valid. Please contact your enterprise admin.');
        });
        it('Fails if the bitgo challenge signature is different from the admin ecdh key', async function () {
            await (0, helpers_1.nockGetChallenge)({ walletId, txRequestId, addendum: '/transactions/0', response: rawBitgoChallenge });
            await (0, helpers_1.nockGetEnterprise)({
                enterpriseId: enterpriseData.id,
                response: {
                    ...enterpriseData,
                    featureFlags: ['useEnterpriseEcdsaTssChallenge'],
                },
                times: 1,
            });
            await (0, helpers_1.nockGetSigningKey)({ enterpriseId, userId: mockedSigningKey.userId, response: mockedSigningKey, times: 1 });
            const adminSignedEntChallenge = sdk_core_1.ECDSAUtils.EcdsaUtils.signChallenge(rawEntChallengeWithProofs, adminEcdhKey.xprv, derivationPath);
            // Bad sign
            const adminSignedBitGoChallenge = sdk_core_1.ECDSAUtils.EcdsaUtils.signChallenge(rawBitgoChallenge, fakeAdminEcdhKey.xprv, derivationPath);
            const mockChallengesResponse = {
                enterpriseChallenge: {
                    ...rawEntChallengeWithProofs,
                    verifiers: {
                        adminSignature: adminSignedEntChallenge.toString('hex'),
                    },
                },
                bitgoChallenge: {
                    ...rawBitgoChallenge,
                    verifiers: {
                        adminSignature: adminSignedBitGoChallenge.toString('hex'),
                    },
                },
                createdBy: 'id',
            };
            await (0, helpers_1.nockGetChallenges)({ walletId: walletId, response: mockChallengesResponse });
            await tssUtils
                .getEcdsaSigningChallenges(txRequestId, 0, mockWalletPaillierKey.n)
                .should.be.rejectedWith("Admin signature for BitGo's challenge is not valid. Please contact your enterprise admin.");
        });
    });
    describe('getVerifyAndSignBitGoChallenges', function () {
        const bitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' });
        const adminEcdhKey = bitgo.keychains().create();
        const derivationPath = 'm/0/0';
        const bitgoInstChallenge = ecdsaNtilde_1.mockChallengeA;
        const bitgoNitroChallenge = ecdsaNtilde_1.mockChallengeB;
        const userPassword = 'password123';
        const encryptedXprv = bitgo.encrypt({
            password: userPassword,
            input: adminEcdhKey.xprv,
        });
        beforeEach(async function () {
            sinon.stub(bitgo, 'getSigningKeyForUser').resolves({
                userId: 'id',
                userEmail: 'user@bitgo.com',
                derivedPubkey: utxo_lib_1.bip32.fromBase58(adminEcdhKey.xpub).derivePath(derivationPath).publicKey.toString('hex'),
                derivationPath: derivationPath,
                ecdhKeychain: 'my keychain',
            });
            sinon.stub(bitgo, 'getECDHKeychain').resolves({
                encryptedXprv: encryptedXprv,
            });
        });
        afterEach(async function () {
            sinon.restore();
            nock.cleanAll();
        });
        function nockGetBitgoChallenges(response) {
            return nock(bgUrl)
                .get(`/api/v2/tss/ecdsa/challenges`)
                .times(1)
                .reply(() => [200, response]);
        }
        it('succeeds for valid bitgo proofs', async function () {
            const nockGetBitgoChallengesApi = nockGetBitgoChallenges({
                bitgoNitroHsm: bitgoNitroChallenge,
                bitgoInstitutionalHsm: bitgoInstChallenge,
            });
            await sdk_core_1.ECDSAUtils.EcdsaUtils.getVerifyAndSignBitGoChallenges(bitgo, 'ent_id', userPassword).should.not.be.rejected();
            nockGetBitgoChallengesApi.isDone().should.be.true();
        });
        it('Fails if bitgo challenge proofs are not present', async function () {
            const nockGetBitgoChallengesApi = nockGetBitgoChallenges({
                bitgoNitroHsm: {
                    ...bitgoNitroChallenge,
                    ntildeProof: undefined,
                },
                bitgoInstitutionalHsm: bitgoInstChallenge,
            });
            await sdk_core_1.ECDSAUtils.EcdsaUtils.getVerifyAndSignBitGoChallenges(bitgo, 'ent_id', userPassword).should.be.rejectedWith('Expected BitGo challenge proof to be present. Contact support@bitgo.com.');
            nockGetBitgoChallengesApi.isDone().should.be.true();
        });
        it('Fails if the user password to decrypt the ecdhkeychain is wrong', async function () {
            const nockGetBitgoChallengesApi = nockGetBitgoChallenges({
                bitgoNitroHsm: bitgoNitroChallenge,
                bitgoInstitutionalHsm: bitgoInstChallenge,
            });
            await sdk_core_1.ECDSAUtils.EcdsaUtils.getVerifyAndSignBitGoChallenges(bitgo, 'ent_id', 'bro').should.be.rejectedWith('Incorrect password. Please try again.');
            nockGetBitgoChallengesApi.isDone().should.be.true();
        });
        it('Fails bitgo challenge proofs for faulty nitro h2WrtH1 proof', async function () {
            const nockGetBitgoChallengesApi = nockGetBitgoChallenges({
                bitgoNitroHsm: {
                    ...bitgoNitroChallenge,
                    ntildeProof: {
                        ...bitgoNitroChallenge.ntildeProof,
                        h2WrtH1: bitgoNitroChallenge.ntildeProof.h1WrtH2,
                    },
                },
                bitgoInstitutionalHsm: bitgoInstChallenge,
            });
            await sdk_core_1.ECDSAUtils.EcdsaUtils.getVerifyAndSignBitGoChallenges(bitgo, 'ent_id', userPassword).should.be.rejectedWith("Failed to verify BitGo's challenge needed to enable ECDSA signing. Please contact support@bitgo.com");
            nockGetBitgoChallengesApi.isDone().should.be.true();
        });
        it('Fails bitgo challenge proofs for faulty nitro h1WrtH2 proof', async function () {
            const nockGetBitgoChallengesApi = nockGetBitgoChallenges({
                bitgoNitroHsm: {
                    ...bitgoNitroChallenge,
                    ntildeProof: {
                        ...bitgoNitroChallenge.ntildeProof,
                        h1WrtH2: bitgoNitroChallenge.ntildeProof.h2WrtH1,
                    },
                },
                bitgoInstitutionalHsm: bitgoInstChallenge,
            });
            await sdk_core_1.ECDSAUtils.EcdsaUtils.getVerifyAndSignBitGoChallenges(bitgo, 'ent_id', userPassword).should.be.rejectedWith("Failed to verify BitGo's challenge needed to enable ECDSA signing. Please contact support@bitgo.com");
            nockGetBitgoChallengesApi.isDone().should.be.true();
        });
        it('Fails bitgo challenge proofs for faulty inst h2WrtH1 proof', async function () {
            const nockGetBitgoChallengesApi = nock(bgUrl)
                .get(`/api/v2/tss/ecdsa/challenges`)
                .times(1)
                .reply(200, {
                bitgoNitroHsm: bitgoNitroChallenge,
                bitgoInstitutionalHsm: {
                    ...bitgoInstChallenge,
                    ntildeProof: {
                        ...bitgoInstChallenge.ntildeProof,
                        h2WrtH1: bitgoInstChallenge.ntildeProof.h1WrtH2,
                    },
                },
            });
            await sdk_core_1.ECDSAUtils.EcdsaUtils.getVerifyAndSignBitGoChallenges(bitgo, 'ent_id', userPassword).should.be.rejectedWith("Failed to verify BitGo's challenge needed to enable ECDSA signing. Please contact support@bitgo.com");
            nockGetBitgoChallengesApi.isDone().should.be.true();
        });
        it('Fails bitgo challenge proofs for faulty inst h1WrtH2 proof', async function () {
            const nockGetBitgoChallengesApi = nock(bgUrl)
                .get(`/api/v2/tss/ecdsa/challenges`)
                .times(1)
                .reply(200, {
                bitgoNitroHsm: bitgoNitroChallenge,
                bitgoInstitutionalHsm: {
                    ...bitgoInstChallenge,
                    ntildeProof: {
                        ...bitgoInstChallenge.ntildeProof,
                        h1WrtH2: bitgoInstChallenge.ntildeProof.h2WrtH1,
                    },
                },
            });
            await sdk_core_1.ECDSAUtils.EcdsaUtils.getVerifyAndSignBitGoChallenges(bitgo, 'ent_id', userPassword).should.be.rejectedWith("Failed to verify BitGo's challenge needed to enable ECDSA signing. Please contact support@bitgo.com");
            nockGetBitgoChallengesApi.isDone().should.be.true();
        });
    });
    describe('supportedTxRequestVersions', function () {
        it('returns only full for hot wallets', function () {
            const hotWallet = new sdk_core_1.Wallet(bitgo, baseCoin, { type: 'hot', multisigType: 'tss' });
            const hotWalletTssUtils = new sdk_core_1.ECDSAUtils.EcdsaUtils(bitgo, baseCoin, hotWallet);
            hotWalletTssUtils.supportedTxRequestVersions().should.deepEqual(['full']);
        });
        it('returns only full for cold wallets', function () {
            const coldWallet = new sdk_core_1.Wallet(bitgo, baseCoin, {
                type: 'cold',
                multisigType: 'tss',
            });
            const coldWalletTssUtils = new sdk_core_1.ECDSAUtils.EcdsaUtils(bitgo, baseCoin, coldWallet);
            coldWalletTssUtils.supportedTxRequestVersions().should.deepEqual(['full']);
        });
        it('returns only full for custodial wallets', function () {
            const custodialWallet = new sdk_core_1.Wallet(bitgo, baseCoin, { type: 'custodial', multisigType: 'tss' });
            const custodialWalletTssUtils = new sdk_core_1.ECDSAUtils.EcdsaUtils(bitgo, baseCoin, custodialWallet);
            custodialWalletTssUtils.supportedTxRequestVersions().should.deepEqual(['full']);
        });
        it('returns empty for trading wallets', function () {
            const tradingWallet = new sdk_core_1.Wallet(bitgo, baseCoin, { type: 'trading', multisigType: 'tss' });
            const tradingWalletTssUtils = new sdk_core_1.ECDSAUtils.EcdsaUtils(bitgo, baseCoin, tradingWallet);
            tradingWalletTssUtils.supportedTxRequestVersions().should.deepEqual([]);
        });
        it('returns empty for non-tss wallets', function () {
            const nonTssWalletData = { coin: 'tbtc', multisigType: 'onchain' };
            const btcCoin = bitgo.coin('tbtc');
            const nonTssWallet = new sdk_core_1.Wallet(bitgo, btcCoin, nonTssWalletData);
            const nonTssWalletTssUtils = new src_1.TssUtils(bitgo, btcCoin, nonTssWallet);
            nonTssWalletTssUtils.supportedTxRequestVersions().should.deepEqual([]);
        });
    });
    describe('initiateChallengesForEnterprise', function () {
        const bitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' });
        const adminEcdhKey = bitgo.keychains().create();
        const derivationPath = 'm/0/0';
        const bitgoInstChallenge = ecdsaNtilde_1.mockChallengeA;
        const bitgoNitroChallenge = ecdsaNtilde_1.mockChallengeB;
        const serializedEntChallenge = ecdsaNtilde_1.mockChallengeC;
        const userPassword = 'password123';
        const encryptedXprv = bitgo.encrypt({
            password: userPassword,
            input: adminEcdhKey.xprv,
        });
        beforeEach(async function () {
            sinon.stub(bitgo, 'getSigningKeyForUser').resolves({
                userId: 'id',
                userEmail: 'user@bitgo.com',
                derivedPubkey: utxo_lib_1.bip32.fromBase58(adminEcdhKey.xpub).derivePath(derivationPath).publicKey.toString('hex'),
                derivationPath: derivationPath,
                ecdhKeychain: 'my keychain',
            });
            sinon.stub(bitgo, 'getECDHKeychain').resolves({
                encryptedXprv: encryptedXprv,
            });
        });
        afterEach(async function () {
            sinon.restore();
        });
        it('should upload challenge without generating if passed in', async function () {
            const stubUploadChallenge = sinon.stub(sdk_core_1.ECDSAUtils.EcdsaUtils, 'uploadChallengesToEnterprise');
            const deserializedEntChallenge = sdk_lib_mpc_1.EcdsaTypes.deserializeNtildeWithProofs(serializedEntChallenge);
            const signedEntChallenge = sdk_core_1.ECDSAUtils.EcdsaUtils.signChallenge(serializedEntChallenge, adminEcdhKey.xprv, derivationPath);
            const signedInstChallenge = sdk_core_1.ECDSAUtils.EcdsaUtils.signChallenge(bitgoInstChallenge, adminEcdhKey.xprv, derivationPath);
            const signedNitroChallenge = sdk_core_1.ECDSAUtils.EcdsaUtils.signChallenge(bitgoNitroChallenge, adminEcdhKey.xprv, derivationPath);
            await sdk_core_1.ECDSAUtils.EcdsaUtils.initiateChallengesForEnterprise(bitgo, 'ent_id', userPassword, signedInstChallenge, signedNitroChallenge, openSSLBytes, deserializedEntChallenge).should.not.be.rejected();
            stubUploadChallenge.calledWith(bitgo, 'ent_id', serializedEntChallenge, signedEntChallenge.toString('hex'), signedInstChallenge.toString('hex'), signedNitroChallenge.toString('hex'));
        });
        it('should generate a challenge and if one is not provided', async function () {
            const stubUploadChallenge = sinon.stub(sdk_core_1.ECDSAUtils.EcdsaUtils, 'uploadChallengesToEnterprise');
            const deserializedEntChallenge = sdk_lib_mpc_1.EcdsaTypes.deserializeNtildeWithProofs(serializedEntChallenge);
            sinon.stub(sdk_lib_mpc_1.EcdsaRangeProof, 'generateNtilde').resolves(deserializedEntChallenge);
            const signedEntChallenge = sdk_core_1.ECDSAUtils.EcdsaUtils.signChallenge(serializedEntChallenge, adminEcdhKey.xprv, derivationPath);
            const signedInstChallenge = sdk_core_1.ECDSAUtils.EcdsaUtils.signChallenge(bitgoInstChallenge, adminEcdhKey.xprv, derivationPath);
            const signedNitroChallenge = sdk_core_1.ECDSAUtils.EcdsaUtils.signChallenge(bitgoNitroChallenge, adminEcdhKey.xprv, derivationPath);
            await sdk_core_1.ECDSAUtils.EcdsaUtils.initiateChallengesForEnterprise(bitgo, 'ent_id', userPassword, signedInstChallenge, signedNitroChallenge, openSSLBytes).should.not.be.rejected();
            stubUploadChallenge.calledWith(bitgo, 'ent_id', serializedEntChallenge, signedEntChallenge.toString('hex'), signedInstChallenge.toString('hex'), signedNitroChallenge.toString('hex'));
        });
    });
    it('getMessageToSignFromChallenge concatenates the challenge values only', function () {
        const challenge = ecdsaNtilde_1.mockChallengeA;
        const expectedMessageToSign = challenge.ntilde.concat(challenge.h1).concat(challenge.h2);
        const message = sdk_core_1.ECDSAUtils.EcdsaUtils.getMessageToSignFromChallenge(challenge);
        message.should.equal(expectedMessageToSign);
    });
    describe('validateCommonKeychainPublicKey', function () {
        it('validateCommonKeychainPublicKey returns correct public key', function () {
            const commonKeychain = '03f40c70545b519bb7bbc7195fd4b7d5bbfc873bfd38b18596e4b47a05b6a88d552e2e8319cb31e279b99dbe54115a983d35e86679af96d81b7478d1df368f76a8';
            const expectedPubKeyResult = `f40c70545b519bb7bbc7195fd4b7d5bbfc873bfd38b18596e4b47a05b6a88d556a10d6ab8055dc0b3a9af9dc4e42f4f9773c590afcc298d017c1b1ce29a88041`;
            const actualPubKey = sdk_core_1.ECDSAUtils.EcdsaUtils.validateCommonKeychainPublicKey(commonKeychain);
            actualPubKey.should.equal(expectedPubKeyResult);
        });
        it('validateCommonKeychainPublicKey throws correctly with invalid length', function () {
            const commonKeychain = '03f40c70548';
            should(() => sdk_core_1.ECDSAUtils.EcdsaUtils.validateCommonKeychainPublicKey(commonKeychain)).throwError('Invalid commonKeychain length, expected 130, got 11');
        });
        it('validateCommonKeychainPublicKey throws correctly with invalid commonKeychain', function () {
            const commonKeychainWithInvalidCharacters = '!@#$^0c70545b519bb7bbc7195fd4b7d5bfc873bfd38b18596e4b47a05b6a88d552e2e8319cb31e279b99dbe54115a983d35e86679af96d81b7478d1df368f76a8'; // 129 chars
            should(() => sdk_core_1.ECDSAUtils.EcdsaUtils.validateCommonKeychainPublicKey(commonKeychainWithInvalidCharacters)).throwError(/^Invalid commonKeychain, error:/);
        });
    });
    // #region Nock helpers
    async function createIncompleteBitgoHeldBackupKeyShare(userGpgKey, backupKeyShare, bitgoGpgKey) {
        const nSharePromises = [
            encryptNShare(backupKeyShare, 1, userGpgKey.publicKey, userGpgKey, false),
            encryptNShare(backupKeyShare, 3, bitgoGpgKey.publicKey, userGpgKey, false),
        ];
        const backupToUserPublicShare = Buffer.concat([
            Buffer.from(backupKeyShare.nShares[1].y, 'hex'),
            Buffer.from(backupKeyShare.nShares[1].chaincode, 'hex'),
        ]).toString('hex');
        const backupToBitgoPublicShare = Buffer.concat([
            Buffer.from(backupKeyShare.nShares[3].y, 'hex'),
            Buffer.from(backupKeyShare.nShares[3].chaincode, 'hex'),
        ]).toString('hex');
        return {
            id: '4711',
            keyShares: [
                {
                    from: 'backup',
                    to: 'user',
                    publicShare: backupToUserPublicShare,
                    privateShare: (await nSharePromises[0]).encryptedPrivateShare,
                },
                {
                    from: 'backup',
                    to: 'bitgo',
                    publicShare: backupToBitgoPublicShare,
                    privateShare: (await nSharePromises[1]).encryptedPrivateShare,
                },
            ],
        };
    }
    async function nockGetBitgoPublicKeyBasedOnFeatureFlags(coin, enterpriseId, bitgoGpgKeyPair) {
        const bitgoGPGPublicKeyResponse = {
            name: 'irrelevant',
            publicKey: bitgoGpgKeyPair.publicKey,
            enterpriseId,
        };
        nock(bgUrl).get(`/api/v2/${coin}/tss/pubkey`).query({ enterpriseId }).reply(200, bitgoGPGPublicKeyResponse);
        return bitgoGPGPublicKeyResponse;
    }
    async function nockCreateBitgoHeldBackupKeyShare(coin, enterpriseId, userGpgKey, backupKeyShare, bitgoGpgKey) {
        const keyShare = await createIncompleteBitgoHeldBackupKeyShare(userGpgKey, backupKeyShare, bitgoGpgKey);
        nock(bgUrl)
            .post(`/api/v2/${coin}/krs/backupkeys`, _.matches({ enterprise: enterpriseId, userGPGPublicKey: userGpgKey.publicKey }))
            .reply(201, keyShare);
        return keyShare;
    }
    async function nockFinalizeBitgoHeldBackupKeyShare(coin, originalKeyShare, commonKeychain, userKeyShare, userLocalBackupGpgKey, bitgoKeychain) {
        const encryptedUserToBackupKeyShare = await encryptNShare(userKeyShare, 2, userLocalBackupGpgKey.publicKey, userGpgKey, false);
        assert(bitgoKeychain.keyShares);
        const bitgoToBackupKeyShare = bitgoKeychain.keyShares.find((keyShare) => keyShare.from === 'bitgo' && keyShare.to === 'backup');
        assert(bitgoToBackupKeyShare);
        const userPublicShare = Buffer.concat([
            Buffer.from(userKeyShare.nShares[2].y, 'hex'),
            Buffer.from(userKeyShare.nShares[2].chaincode, 'hex'),
        ]).toString('hex');
        const expectedKeyShares = [
            {
                from: 'user',
                to: 'backup',
                publicShare: userPublicShare,
                // Omitting the private share, the actual encryption happens inside the function where we make the matching call
                // to this nock. We cannot recreate the same encrypted value here because gpg encryption is not deterministic
            },
            bitgoToBackupKeyShare,
        ];
        const updatedKeyShare = {
            id: originalKeyShare.id,
            commonKeychain,
            keyShares: [
                ...originalKeyShare.keyShares,
                {
                    from: 'user',
                    to: 'backup',
                    publicShare: userPublicShare,
                    privateShare: encryptedUserToBackupKeyShare.encryptedPrivateShare,
                },
                bitgoToBackupKeyShare,
            ],
        };
        nock(bgUrl)
            .put(`/api/v2/${coin}/krs/backupkeys/${originalKeyShare.id}`, _.matches({ commonKeychain, keyShares: expectedKeyShares }))
            .reply(200, updatedKeyShare);
        return updatedKeyShare;
    }
    /**
     * Helper function to generate a bitgo keychain given the full set of keyshares and GPG keys.
     * Also mocks the wallet signatures added by the HSM.
     * @param params
     */
    async function generateBitgoKeychain(params) {
        const bitgoCombined = MPC.keyCombine(params.bitgoKeyShare.pShare, [
            params.userKeyShare.nShares[3],
            params.backupKeyShare.nShares[3],
        ]);
        const userGpgKeyActual = await openpgp.readKey({ armoredKey: params.userGpgKey.publicKey });
        const backupGpgKeyActual = await openpgp.readKey({ armoredKey: params.userLocalBackupGpgKey.publicKey });
        const nSharePromises = [
            encryptNShare(params.bitgoKeyShare, 1, params.userGpgKey.publicKey, params.userGpgKey, false),
            encryptNShare(params.bitgoKeyShare, 2, params.userLocalBackupGpgKey.publicKey, params.userLocalBackupGpgKey, false),
        ];
        const [userToBitgoShare, backupToBitgoShare] = await Promise.all(nSharePromises);
        const bitgoKeychain = {
            id: '3',
            pub: '',
            commonKeychain: bitgoCombined.xShare.y + bitgoCombined.xShare.chaincode,
            keyShares: [
                {
                    from: 'bitgo',
                    to: 'user',
                    publicShare: userToBitgoShare.publicShare,
                    privateShare: userToBitgoShare.encryptedPrivateShare,
                    n: userToBitgoShare.n,
                    vssProof: userToBitgoShare.vssProof,
                    privateShareProof: userToBitgoShare.privateShareProof,
                },
                {
                    from: 'bitgo',
                    to: 'backup',
                    publicShare: backupToBitgoShare.publicShare,
                    privateShare: backupToBitgoShare.encryptedPrivateShare,
                    n: backupToBitgoShare.n,
                    vssProof: backupToBitgoShare.vssProof,
                    privateShareProof: backupToBitgoShare.privateShareProof,
                },
            ],
            type: 'tss',
        };
        const userKeyId = userGpgKeyActual.keyPacket.getFingerprint();
        const backupKeyId = backupGpgKeyActual.keyPacket.getFingerprint();
        const bitgoToUserPublicU = Buffer.from(utxo_lib_1.ecc.pointFromScalar(Buffer.from(params.bitgoKeyShare.nShares[1].u, 'hex'), true)).toString('hex') + params.bitgoKeyShare.nShares[1].chaincode;
        const bitgoToBackupPublicU = Buffer.from(utxo_lib_1.ecc.pointFromScalar(Buffer.from(params.bitgoKeyShare.nShares[2].u, 'hex'), true)).toString('hex') + params.bitgoKeyShare.nShares[2].chaincode;
        bitgoKeychain.walletHSMGPGPublicKeySigs = await (0, helpers_1.createWalletSignatures)(params.bitgoGpgKey.privateKey, params.userGpgKey.publicKey, params.userLocalBackupGpgKey.publicKey, [
            { name: 'commonKeychain', value: bitgoCombined.xShare.y + bitgoCombined.xShare.chaincode },
            { name: 'userKeyId', value: userKeyId },
            { name: 'backupKeyId', value: backupKeyId },
            { name: 'bitgoToUserPublicShare', value: bitgoToUserPublicU },
            { name: 'bitgoToBackupPublicShare', value: bitgoToBackupPublicU },
        ]);
        return bitgoKeychain;
    }
    async function nockBitgoKeychain(params) {
        const bitgoKeychain = await generateBitgoKeychain(params);
        nock(bgUrl)
            .persist()
            .post(`/api/v2/${params.coin}/key`, _.matches({ keyType: 'tss', source: 'bitgo' }))
            .reply(200, bitgoKeychain);
        return bitgoKeychain;
    }
    async function nockKeychain(params) {
        nock('https://bitgo.fakeurl')
            .persist()
            .post(`/api/v2/${params.coin}/key`, _.matches({ keyType: 'tss', source: params.source }))
            .reply(200, params.keyChain);
        return params.keyChain;
    }
    // #endregion Nock helpers
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ecdsa.js","sourceRoot":"","sources":["../../../../../../test/v2/unit/internal/tssUtils/ecdsa.ts"],"names":[],"mappings":"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","sourcesContent":["import * as assert from 'assert';\nimport * as _ from 'lodash';\nimport * as nock from 'nock';\nimport * as openpgp from 'openpgp';\nimport * as should from 'should';\nimport * as sinon from 'sinon';\n\nimport {\n  mockSerializedChallengeWithProofs,\n  mockSerializedChallengeWithProofs2,\n  TestableBG,\n  TestBitGo,\n} from '@bitgo/sdk-test';\nimport { BitGo, createSharedDataProof, TssUtils, RequestType } from '../../../../../src';\nimport {\n  BackupGpgKey,\n  BackupKeyShare,\n  BaseCoin,\n  BitgoGPGPublicKey,\n  BitgoHeldBackupKeyShare,\n  common,\n  Ecdsa,\n  ECDSA,\n  ECDSAMethods,\n  ECDSAUtils,\n  EnterpriseData,\n  Keychain,\n  RequestTracer,\n  SignatureShareRecord,\n  SignatureShareType,\n  TxRequest,\n  Wallet,\n} from '@bitgo/sdk-core';\nimport { EcdsaPaillierProof, EcdsaRangeProof, EcdsaTypes, hexToBigInt } from '@bitgo/sdk-lib-mpc';\nimport { keyShares, otherKeyShares } from '../../../fixtures/tss/ecdsaFixtures';\nimport { nockSendSignatureShareWithResponse } from './common';\nimport {\n  createWalletSignatures,\n  nockGetChallenge,\n  nockGetChallenges,\n  nockGetEnterprise,\n  nockGetSigningKey,\n  nockGetTxRequest,\n} from '../../tss/helpers';\nimport { bip32, ecc } from '@bitgo/utxo-lib';\nimport { Hash } from 'crypto';\nimport { mockChallengeA, mockChallengeB, mockChallengeC } from './mocks/ecdsaNtilde';\n\nimport { loadWebAssembly } from '@bitgo/sdk-opensslbytes';\n\nconst openSSLBytes = loadWebAssembly().buffer;\n\nconst createKeccakHash = require('keccak');\n\nconst encryptNShare = ECDSAMethods.encryptNShare;\ntype KeyShare = ECDSA.KeyShare;\n\nopenpgp.config.rejectCurves = new Set();\n\ndescribe('TSS Ecdsa Utils:', async function () {\n  const coinName = 'hteth';\n  const reqId = new RequestTracer();\n  const walletId = '5b34252f1bf349930e34020a00000000';\n  const enterpriseId = '6449153a6f6bc20006d66771cdbe15d3';\n  const enterpriseData = { id: enterpriseId, name: 'Test Enterprise' };\n\n  let sandbox: sinon.SinonSandbox;\n  let MPC: Ecdsa;\n  let bgUrl: string;\n  let tssUtils: ECDSAUtils.EcdsaUtils;\n  let wallet: Wallet;\n  let bitgo: TestableBG & BitGo;\n  let baseCoin: BaseCoin;\n  let bitgoKeyShare;\n  let userKeyShare: KeyShare;\n  let backupKeyShare: KeyShare;\n  let bitgoPublicKey: openpgp.Key;\n\n  let userGpgKey: openpgp.SerializedKeyPair<string> & {\n    revocationCertificate: string;\n  };\n  let userLocalBackupGpgKey: openpgp.SerializedKeyPair<string> & {\n    revocationCertificate: string;\n  };\n  let bitGoGPGKeyPair: openpgp.SerializedKeyPair<string> & {\n    revocationCertificate: string;\n  };\n  let nockedBitGoKeychain: Keychain;\n  let nockedUserKeychain: Keychain;\n\n  beforeEach(async function () {\n    sandbox = sinon.createSandbox();\n  });\n\n  afterEach(function () {\n    sandbox.restore();\n  });\n\n  before(async function () {\n    nock.cleanAll();\n    MPC = new Ecdsa();\n    userKeyShare = keyShares.userKeyShare;\n    backupKeyShare = keyShares.backupKeyShare;\n    bitgoKeyShare = keyShares.bitgoKeyShare;\n\n    const gpgKeyPromises = [\n      openpgp.generateKey({\n        userIDs: [\n          {\n            name: 'test',\n            email: 'test@test.com',\n          },\n        ],\n        curve: 'secp256k1',\n      }),\n      openpgp.generateKey({\n        userIDs: [\n          {\n            name: 'backup',\n            email: 'backup@test.com',\n          },\n        ],\n        curve: 'secp256k1',\n      }),\n      openpgp.generateKey({\n        userIDs: [\n          {\n            name: 'bitgo',\n            email: 'bitgo@test.com',\n          },\n        ],\n        curve: 'secp256k1',\n      }),\n    ];\n    [userGpgKey, userLocalBackupGpgKey, bitGoGPGKeyPair] = await Promise.all(gpgKeyPromises);\n    bitgoPublicKey = await openpgp.readKey({ armoredKey: bitGoGPGKeyPair.publicKey });\n    const constants = {\n      mpc: {\n        bitgoPublicKey: bitGoGPGKeyPair.publicKey,\n      },\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    // TODO(WP-346): sdk-test mocks conflict so we can't use persist\n    nock(bgUrl).get('/api/v1/client/constants').times(16).reply(200, { ttl: 3600, constants });\n\n    const nockPromises = [\n      nockBitgoKeychain({\n        coin: coinName,\n        userKeyShare,\n        backupKeyShare,\n        bitgoKeyShare,\n        userGpgKey,\n        userLocalBackupGpgKey,\n        bitgoGpgKey: bitGoGPGKeyPair,\n      }),\n      nockKeychain({ coin: coinName, keyChain: { id: '1', pub: '', type: 'tss' }, source: 'user' }),\n      nockKeychain({ coin: coinName, keyChain: { id: '2', pub: '', type: 'tss' }, source: 'backup' }),\n    ];\n    [nockedBitGoKeychain, nockedUserKeychain] = await Promise.all(nockPromises);\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.EcdsaUtils(bitgo, baseCoin, wallet);\n  });\n\n  after(function () {\n    nock.cleanAll();\n  });\n\n  describe('TSS key chains', async function () {\n    it('should create backup key share held by BitGo', async function () {\n      const enterpriseId = 'enterprise id';\n      const expectedKeyShare = await nockCreateBitgoHeldBackupKeyShare(\n        coinName,\n        enterpriseId,\n        userGpgKey,\n        backupKeyShare,\n        bitGoGPGKeyPair\n      );\n      const result = await tssUtils.createBitgoHeldBackupKeyShare(userGpgKey, enterpriseId);\n      result.should.eql(expectedKeyShare);\n    });\n\n    it('should finalize backup key share held by BitGo', async function () {\n      const commonKeychain = '4428';\n      const originalKeyShare = await createIncompleteBitgoHeldBackupKeyShare(\n        userGpgKey,\n        backupKeyShare,\n        bitGoGPGKeyPair\n      );\n      const expectedFinalKeyShare = await nockFinalizeBitgoHeldBackupKeyShare(\n        coinName,\n        originalKeyShare,\n        commonKeychain,\n        userKeyShare,\n        bitGoGPGKeyPair,\n        nockedBitGoKeychain\n      );\n\n      const result = await tssUtils.finalizeBitgoHeldBackupKeyShare(\n        originalKeyShare.id,\n        commonKeychain,\n        userKeyShare,\n        nockedBitGoKeychain,\n        userGpgKey,\n        bitgoPublicKey\n      );\n      result.should.eql(expectedFinalKeyShare);\n    });\n\n    it('should get the respective backup key shares based on provider', async function () {\n      const enterpriseId = 'enterprise id';\n      await nockCreateBitgoHeldBackupKeyShare(coinName, enterpriseId, userGpgKey, backupKeyShare, bitGoGPGKeyPair);\n      const backupKeyShares = await tssUtils.createBackupKeyShares();\n      should.exist(backupKeyShares.userHeldKeyShare);\n      should.not.exist(backupKeyShares.bitGoHeldKeyShares);\n    });\n\n    it('should get the correct bitgo gpg key based on coin and feature flags', async function () {\n      const nitroGPGKeypair = await openpgp.generateKey({\n        userIDs: [\n          {\n            name: 'bitgo nitro',\n            email: 'bitgo@test.com',\n          },\n        ],\n      });\n      const nockGPGKey = await nockGetBitgoPublicKeyBasedOnFeatureFlags(coinName, 'enterprise_id', nitroGPGKeypair);\n      const bitgoGpgPublicKey = await tssUtils.getBitgoGpgPubkeyBasedOnFeatureFlags('enterprise_id');\n      should.equal(nockGPGKey.publicKey, bitgoGpgPublicKey.armor());\n    });\n\n    it('getBackupEncryptedNShare should get valid encrypted n shares based on provider', async function () {\n      const bitgoGpgKeyPubKey = await tssUtils.getBitgoPublicGpgKey();\n      // Backup key held by user\n      const backupShareHolderNew: BackupKeyShare = {\n        userHeldKeyShare: backupKeyShare,\n      };\n      const backupToBitgoEncryptedNShare = await tssUtils.getBackupEncryptedNShare(\n        backupShareHolderNew,\n        3,\n        bitgoGpgKeyPubKey.armor(),\n        userGpgKey\n      );\n      const encryptedNShare = await encryptNShare(backupKeyShare, 3, bitgoGpgKeyPubKey.armor(), userGpgKey);\n      // cant verify the encrypted shares, since they will be encrypted with diff. values\n      should.equal(backupToBitgoEncryptedNShare.publicShare, encryptedNShare.publicShare);\n    });\n\n    it('should generate TSS key chains', async function () {\n      const backupShareHolder: BackupKeyShare = {\n        userHeldKeyShare: backupKeyShare,\n      };\n      const backupGpgKey: BackupGpgKey = userLocalBackupGpgKey;\n      const bitgoKeychain = await tssUtils.createBitgoKeychain({\n        userGpgKey,\n        backupGpgKey,\n        userKeyShare,\n        backupKeyShare: backupShareHolder,\n        bitgoPublicGpgKey: bitgoPublicKey,\n      });\n      const usersKeyChainPromises = [\n        tssUtils.createParticipantKeychain(\n          userGpgKey,\n          userLocalBackupGpgKey,\n          bitgoPublicKey,\n          1,\n          userKeyShare,\n          backupKeyShare,\n          bitgoKeychain,\n          'passphrase'\n        ),\n        tssUtils.createParticipantKeychain(\n          userGpgKey,\n          userLocalBackupGpgKey,\n          bitgoPublicKey,\n          2,\n          userKeyShare,\n          backupKeyShare,\n          bitgoKeychain,\n          'passphrase'\n        ),\n      ];\n      const [userKeychain, backupKeychain] = await Promise.all(usersKeyChainPromises);\n\n      bitgoKeychain.should.deepEqual(nockedBitGoKeychain);\n      userKeychain.should.deepEqual(nockedUserKeychain);\n\n      // unencrypted `prv` property should exist on backup keychain\n      const keyChainPrv = JSON.parse(backupKeychain.prv ?? '');\n      _.isEqual(keyChainPrv.pShare, backupKeyShare.pShare).should.be.true();\n      _.isEqual(keyChainPrv.bitgoNShare, bitgoKeyShare.nShares[2]).should.be.true();\n      _.isEqual(keyChainPrv.userNShare, userKeyShare.nShares[2]).should.be.true();\n      should.exist(backupKeychain.encryptedPrv);\n    });\n\n    it('should generate TSS key chains with optional params', async function () {\n      const enterprise = 'enterprise_id';\n      const backupShareHolder: BackupKeyShare = {\n        userHeldKeyShare: backupKeyShare,\n      };\n      const backupGpgKey: BackupGpgKey = userLocalBackupGpgKey;\n      const bitgoKeychain = await tssUtils.createBitgoKeychain({\n        userGpgKey,\n        backupGpgKey,\n        userKeyShare,\n        backupKeyShare: backupShareHolder,\n        enterprise,\n        bitgoPublicGpgKey: bitgoPublicKey,\n      });\n      const usersKeyChainPromises = [\n        tssUtils.createParticipantKeychain(\n          userGpgKey,\n          userLocalBackupGpgKey,\n          bitgoPublicKey,\n          1,\n          userKeyShare,\n          backupKeyShare,\n          bitgoKeychain,\n          'passphrase',\n          'originalPasscodeEncryptionCode'\n        ),\n        tssUtils.createParticipantKeychain(\n          userGpgKey,\n          userLocalBackupGpgKey,\n          bitgoPublicKey,\n          2,\n          userKeyShare,\n          backupKeyShare,\n          bitgoKeychain,\n          'passphrase'\n        ),\n      ];\n\n      const [userKeychain, backupKeychain] = await Promise.all(usersKeyChainPromises);\n      bitgoKeychain.should.deepEqual(nockedBitGoKeychain);\n      userKeychain.should.deepEqual(nockedUserKeychain);\n\n      // unencrypted `prv` property should exist on backup keychain\n      const keyChainPrv = JSON.parse(backupKeychain.prv ?? '');\n      _.isEqual(keyChainPrv.pShare, backupKeyShare.pShare).should.be.true();\n      _.isEqual(keyChainPrv.bitgoNShare, bitgoKeyShare.nShares[2]).should.be.true();\n      _.isEqual(keyChainPrv.userNShare, userKeyShare.nShares[2]).should.be.true();\n      should.exist(backupKeychain.encryptedPrv);\n    });\n\n    it('should fail to generate TSS key chains', async function () {\n      const backupShareHolder: BackupKeyShare = {\n        userHeldKeyShare: backupKeyShare,\n      };\n      const backupGpgKey: BackupGpgKey = userLocalBackupGpgKey;\n      const bitgoKeychain = await tssUtils.createBitgoKeychain({\n        userGpgKey,\n        backupGpgKey,\n        userKeyShare,\n        backupKeyShare: backupShareHolder,\n        bitgoPublicGpgKey: bitgoPublicKey,\n      });\n      bitgoKeychain.should.deepEqual(nockedBitGoKeychain);\n      const testKeyShares = otherKeyShares;\n      const testCasesPromises = [\n        tssUtils\n          .createParticipantKeychain(\n            userGpgKey,\n            userLocalBackupGpgKey,\n            bitgoPublicKey,\n            1,\n            userKeyShare,\n            testKeyShares[0],\n            bitgoKeychain,\n            'passphrase'\n          )\n          .should.be.rejectedWith('Common keychains do not match'),\n        tssUtils\n          .createParticipantKeychain(\n            userGpgKey,\n            userLocalBackupGpgKey,\n            bitgoPublicKey,\n            1,\n            testKeyShares[1],\n            backupKeyShare,\n            bitgoKeychain,\n            'passphrase'\n          )\n          .should.be.rejectedWith('Common keychains do not match'),\n        tssUtils\n          .createParticipantKeychain(\n            userGpgKey,\n            userLocalBackupGpgKey,\n            bitgoPublicKey,\n            2,\n            testKeyShares[2],\n            backupKeyShare,\n            bitgoKeychain,\n            'passphrase'\n          )\n          .should.be.rejectedWith('Common keychains do not match'),\n        tssUtils\n          .createParticipantKeychain(\n            userGpgKey,\n            userLocalBackupGpgKey,\n            bitgoPublicKey,\n            2,\n            userKeyShare,\n            testKeyShares[3],\n            bitgoKeychain,\n            'passphrase'\n          )\n          .should.be.rejectedWith('Common keychains do not match'),\n      ];\n      await Promise.all(testCasesPromises);\n    });\n\n    it('should fail to generate TSS keychains when received invalid number of wallet signatures', async function () {\n      const bitgoKeychain = await generateBitgoKeychain({\n        coin: coinName,\n        userKeyShare,\n        backupKeyShare,\n        bitgoKeyShare,\n        userGpgKey,\n        userLocalBackupGpgKey,\n        bitgoGpgKey: bitGoGPGKeyPair,\n      });\n\n      const certsString = await createSharedDataProof(bitGoGPGKeyPair.privateKey, userGpgKey.publicKey, []);\n      const certsKey = await openpgp.readKey({ armoredKey: certsString });\n      const finalKey = new openpgp.PacketList();\n      certsKey.toPacketList().forEach((packet) => finalKey.push(packet));\n      // Once the following PR has been merged and released we no longer need the ts-ignore:\n      // https://github.com/openpgpjs/openpgpjs/pull/1576\n      // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n      // @ts-ignore\n      bitgoKeychain.walletHSMGPGPublicKeySigs = openpgp.armor(openpgp.enums.armor.publicKey, finalKey.write());\n      await tssUtils\n        .verifyWalletSignatures(userLocalBackupGpgKey.publicKey, userLocalBackupGpgKey.publicKey, bitgoKeychain, '', 1)\n        .should.be.rejectedWith(`Invalid wallet signatures`);\n    });\n\n    it('should fail to generate TSS keychains when wallet signature fingerprints do not match passed user/backup fingerprints', async function () {\n      const customUserKeyShare = await MPC.keyShare(1, 2, 3);\n      const customBackupKeyShare = await MPC.keyShare(2, 2, 3);\n      const backupShareHolder: BackupKeyShare = {\n        userHeldKeyShare: customBackupKeyShare,\n      };\n      const backupGpgKey: BackupGpgKey = userLocalBackupGpgKey;\n\n      const bitgoKeychain = await tssUtils.createBitgoKeychain({\n        userGpgKey,\n        backupGpgKey,\n        userKeyShare: customUserKeyShare,\n        backupKeyShare: backupShareHolder,\n        bitgoPublicGpgKey: bitgoPublicKey,\n      });\n\n      // using the backup gpg here instead of the user gpg key to simulate that the first signature has a different\n      // fingerprint from the passed in first gpg key\n      await tssUtils\n        .verifyWalletSignatures(userLocalBackupGpgKey.publicKey, userLocalBackupGpgKey.publicKey, bitgoKeychain, '', 1)\n        .should.be.rejectedWith(\n          `first wallet signature's fingerprint does not match passed user gpg key's fingerprint`\n        );\n\n      // using the user gpg here instead of the backup gpg key to simulate that the second signature has a different\n      // fingerprint from the passed in second gpg key\n      await tssUtils\n        .verifyWalletSignatures(userGpgKey.publicKey, userGpgKey.publicKey, bitgoKeychain, '', 1)\n        .should.be.rejectedWith(\n          `second wallet signature's fingerprint does not match passed backup gpg key's fingerprint`\n        );\n    });\n  });\n\n  describe('signTxRequest:', () => {\n    const txRequestId = 'randomidEcdsa';\n    const txRequest: TxRequest = {\n      txRequestId,\n      transactions: [\n        {\n          unsignedTx: {\n            // hteth txid: 0xc5a7bfe6b13ceae563da0f9feaa9c4ad1c101a15366a2a488828a5dd27cb9da3\n            serializedTxHex:\n              '02f38242688084448b9b8084448b9b908301637894a1cfb9d51c0af191ff21c5f0f01723e056f7dc12865af3107a400080c0808080',\n            signableHex:\n              '02f08242688084448b9b8084448b9b908301637894a1cfb9d51c0af191ff21c5f0f01723e056f7dc12865af3107a400080c0',\n            derivationPath: '', // Needs this when key derivation is supported\n          },\n          state: 'pendingSignature',\n          signatureShares: [],\n        },\n      ],\n      unsignedTxs: [\n        {\n          // hteth txid: 0xc5a7bfe6b13ceae563da0f9feaa9c4ad1c101a15366a2a488828a5dd27cb9da3\n          serializedTxHex:\n            '02f38242688084448b9b8084448b9b908301637894a1cfb9d51c0af191ff21c5f0f01723e056f7dc12865af3107a400080c0808080',\n          signableHex:\n            '02f38242688084448b9b8084448b9b908301637894a1cfb9d51c0af191ff21c5f0f01723e056f7dc12865af3107a400080c0808080',\n          derivationPath: '', // Needs this when key derivation is supported\n        },\n      ],\n      date: new Date().toISOString(),\n      intent: {\n        intentType: 'payment',\n      },\n      latest: true,\n      state: 'pendingUserSignature',\n      walletType: 'hot',\n      walletId: 'walletId',\n      policiesChecked: true,\n      version: 1,\n      userId: 'userId',\n    };\n    let aShare, dShare, wShare, oShare, userSignShare, bitgoChallenges, enterpriseChallenges;\n\n    beforeEach(async () => {\n      // Initializing user and bitgo for creating shares for nocks\n      const userSigningKey = MPC.keyCombine(userKeyShare.pShare, [bitgoKeyShare.nShares[1], backupKeyShare.nShares[1]]);\n      const bitgoSigningKey = MPC.keyCombine(bitgoKeyShare.pShare, [\n        userKeyShare.nShares[3],\n        backupKeyShare.nShares[3],\n      ]);\n\n      const serializedEntChallenge = mockChallengeA;\n      const serializedBitgoChallenge = mockChallengeB;\n\n      const deserializedEntChallenge = EcdsaTypes.deserializeNtildeWithProofs(serializedEntChallenge);\n      sinon.stub(EcdsaRangeProof, 'generateNtilde').resolves(deserializedEntChallenge);\n\n      const [userToBitgoPaillierChallenge, bitgoToUserPaillierChallenge] = await Promise.all([\n        EcdsaPaillierProof.generateP(hexToBigInt(userSigningKey.yShares[3].n)),\n        EcdsaPaillierProof.generateP(hexToBigInt(bitgoSigningKey.yShares[1].n)),\n      ]);\n\n      bitgoChallenges = {\n        ...serializedBitgoChallenge,\n        p: EcdsaTypes.serializePaillierChallenge({ p: bitgoToUserPaillierChallenge }).p,\n        n: bitgoSigningKey.xShare.n,\n      };\n      enterpriseChallenges = {\n        ...serializedEntChallenge,\n        p: EcdsaTypes.serializePaillierChallenge({ p: userToBitgoPaillierChallenge }).p,\n        n: bitgoSigningKey.xShare.n,\n      };\n      sinon.stub(ECDSAUtils.EcdsaUtils.prototype, 'getEcdsaSigningChallenges').resolves({\n        enterpriseChallenge: enterpriseChallenges,\n        bitgoChallenge: bitgoChallenges,\n      });\n\n      const [userXShare, bitgoXShare] = [\n        MPC.appendChallenge(\n          userSigningKey.xShare,\n          serializedEntChallenge,\n          EcdsaTypes.serializePaillierChallenge({ p: userToBitgoPaillierChallenge })\n        ),\n        MPC.appendChallenge(\n          bitgoSigningKey.xShare,\n          serializedBitgoChallenge,\n          EcdsaTypes.serializePaillierChallenge({ p: bitgoToUserPaillierChallenge })\n        ),\n      ];\n      const bitgoYShare = MPC.appendChallenge(\n        userSigningKey.yShares[3],\n        serializedBitgoChallenge,\n        EcdsaTypes.serializePaillierChallenge({ p: bitgoToUserPaillierChallenge })\n      );\n      /**\n       * START STEP ONE\n       * 1) User creates signShare, saves wShare and sends kShare to bitgo\n       * 2) Bitgo performs signConvert operation using its private xShare , yShare\n       *  and KShare from user and responds back with aShare and saves bShare for later use\n       */\n      userSignShare = await ECDSAMethods.createUserSignShare(userXShare, bitgoYShare);\n      wShare = userSignShare.wShare;\n      const signatureShareOneFromUser: SignatureShareRecord = {\n        from: SignatureShareType.USER,\n        to: SignatureShareType.BITGO,\n        share: ECDSAMethods.convertKShare(userSignShare.kShare).share.replace(ECDSAMethods.delimeter, ''),\n      };\n      const getBitgoAandBShare = await MPC.signConvertStep1({\n        xShare: bitgoXShare,\n        yShare: bitgoSigningKey.yShares[1], // corresponds to the user\n        kShare: userSignShare.kShare,\n      });\n      const bitgoAshare = getBitgoAandBShare.aShare;\n      aShare = bitgoAshare;\n      const aShareBitgoResponse = ECDSAMethods.convertAShare(bitgoAshare).share.replace(ECDSAMethods.delimeter, '');\n      const signatureShareOneFromBitgo: SignatureShareRecord = {\n        from: SignatureShareType.BITGO,\n        to: SignatureShareType.USER,\n        share: aShareBitgoResponse,\n      };\n      await nockSendSignatureShareWithResponse({\n        walletId: wallet.id(),\n        txRequestId: txRequest.txRequestId,\n        signatureShare: signatureShareOneFromUser,\n        response: signatureShareOneFromBitgo,\n        tssType: 'ecdsa',\n      });\n      /**  END STEP ONE */\n\n      /**\n       * START STEP TWO\n       * 1) Using the aShare got from bitgo and wShare from previous step,\n       * user creates gShare and muShare and sends muShare to bitgo\n       * 2) Bitgo using the signConvert step using bShare from previous step\n       * and muShare from user generates its gShare.\n       * 3) Using the signCombine operation using gShare, Bitgo generates oShare\n       * which it saves and dShare which is send back to the user.\n       */\n      const userGammaAndMuShares = await ECDSAMethods.createUserGammaAndMuShare(userSignShare.wShare, bitgoAshare);\n      const signatureShareTwoFromUser: SignatureShareRecord = {\n        from: SignatureShareType.USER,\n        to: SignatureShareType.BITGO,\n        share: ECDSAMethods.convertMuShare(userGammaAndMuShares.muShare!).share.replace(ECDSAMethods.delimeter, ''),\n      };\n      const getBitGoGShareAndSignerIndexes = await MPC.signConvertStep3({\n        bShare: getBitgoAandBShare.bShare,\n        muShare: userGammaAndMuShares.muShare,\n      });\n\n      const getBitgoOShareAndDShares = MPC.signCombine({\n        gShare: getBitGoGShareAndSignerIndexes.gShare as ECDSA.GShare,\n        signIndex: {\n          i: 1,\n          j: 3,\n        },\n      });\n      const bitgoDshare = getBitgoOShareAndDShares.dShare as ECDSA.DShare;\n      dShare = bitgoDshare;\n      const dShareBitgoResponse = (bitgoDshare.delta as string) + (bitgoDshare.Gamma as string);\n      const signatureShareTwoFromBitgo: SignatureShareRecord = {\n        from: SignatureShareType.BITGO,\n        to: SignatureShareType.USER,\n        share: dShareBitgoResponse,\n      };\n      await nockSendSignatureShareWithResponse({\n        walletId: wallet.id(),\n        txRequestId: txRequest.txRequestId,\n        signatureShare: signatureShareTwoFromUser,\n        response: signatureShareTwoFromBitgo,\n        tssType: 'ecdsa',\n      });\n      /**  END STEP TWO */\n\n      /**\n       * START STEP THREE\n       * 1) User creates its oShare and  dShare using the  private gShare\n       * from step two\n       * 2) User uses the private oShare and dShare from bitgo from step\n       * two to generate its signature share which it sends back along with dShare that\n       * user generated from the above step\n       * 3) Bitgo using its private oShare from step two and dShare from bitgo creates\n       * its signature share. Using the Signature Share received from user from the above\n       * step, bitgo constructs the final signature and is returned to the user\n       */\n      const userOmicronAndDeltaShare = await ECDSAMethods.createUserOmicronAndDeltaShare(\n        userGammaAndMuShares.gShare as ECDSA.GShare\n      );\n      oShare = userOmicronAndDeltaShare.oShare;\n      const signablePayload = Buffer.from(txRequest.unsignedTxs[0].signableHex, 'hex');\n      const userSShare = await ECDSAMethods.createUserSignatureShare(\n        userOmicronAndDeltaShare.oShare,\n        bitgoDshare,\n        signablePayload\n      );\n      const signatureShareThreeFromUser: SignatureShareRecord = {\n        from: SignatureShareType.USER,\n        to: SignatureShareType.BITGO,\n        share:\n          userSShare.R +\n          userSShare.s +\n          userSShare.y +\n          userOmicronAndDeltaShare.dShare.delta +\n          userOmicronAndDeltaShare.dShare.Gamma,\n      };\n      const getBitGoSShare = MPC.sign(\n        signablePayload,\n        getBitgoOShareAndDShares.oShare,\n        userOmicronAndDeltaShare.dShare,\n        createKeccakHash('keccak256') as Hash\n      );\n      const getBitGoFinalSignature = MPC.constructSignature([getBitGoSShare, userSShare]);\n      const finalSigantureBitgoResponse =\n        getBitGoFinalSignature.r + getBitGoFinalSignature.s + getBitGoFinalSignature.y;\n      const signatureShareThreeFromBitgo: SignatureShareRecord = {\n        from: SignatureShareType.BITGO,\n        to: SignatureShareType.USER,\n        share: finalSigantureBitgoResponse,\n      };\n      await nockSendSignatureShareWithResponse({\n        walletId: wallet.id(),\n        txRequestId: txRequest.txRequestId,\n        signatureShare: signatureShareThreeFromUser,\n        response: signatureShareThreeFromBitgo,\n        tssType: 'ecdsa',\n      });\n      /* END STEP THREE */\n      const signature = MPC.constructSignature([userSShare, getBitGoSShare]);\n      MPC.verify(signablePayload, signature, createKeccakHash('keccak256') as Hash).should.be.true;\n    });\n\n    afterEach(async () => {\n      sinon.restore();\n    });\n\n    it('signTxRequest should fail if wallet is in pendingEcdsaTssInitialization', async function () {\n      sandbox.stub(wallet, 'coinSpecific').returns({\n        customChangeWalletId: '',\n        pendingEcdsaTssInitialization: true,\n      });\n      await tssUtils\n        .signTxRequest({\n          txRequest,\n          prv: JSON.stringify({\n            pShare: userKeyShare.pShare,\n            bitgoNShare: bitgoKeyShare.nShares[1],\n            backupNShare: backupKeyShare.nShares[1],\n          }),\n          reqId,\n        })\n        .should.be.rejectedWith(\n          'Wallet is not ready for TSS ECDSA signing. Please contact your enterprise admin to finish the enterprise TSS initialization.'\n        );\n    });\n\n    it('signTxRequest should succeed with txRequest object as input', async function () {\n      const sendShareSpy = sinon.spy(ECDSAMethods, 'sendShareToBitgo' as any);\n      await setupSignTxRequestNocks(false, userSignShare, aShare, dShare, enterpriseData);\n      const signedTxRequest = await tssUtils.signTxRequest({\n        txRequest,\n        prv: JSON.stringify({\n          pShare: userKeyShare.pShare,\n          bitgoNShare: bitgoKeyShare.nShares[1],\n          backupNShare: backupKeyShare.nShares[1],\n        }),\n        reqId,\n      });\n      signedTxRequest.unsignedTxs.should.deepEqual(txRequest.unsignedTxs);\n      const userGpgActual = sendShareSpy.getCalls()[0].args[10] as string;\n      userGpgActual.should.startWith('-----BEGIN PGP PUBLIC KEY BLOCK-----');\n    });\n\n    it('signTxRequest should succeed with txRequest id as input', async function () {\n      const sendShareSpy = sinon.spy(ECDSAMethods, 'sendShareToBitgo' as any);\n      await setupSignTxRequestNocks(true, userSignShare, aShare, dShare, enterpriseData);\n      const signedTxRequest = await tssUtils.signTxRequest({\n        txRequest: txRequestId,\n        prv: JSON.stringify({\n          pShare: userKeyShare.pShare,\n          bitgoNShare: bitgoKeyShare.nShares[1],\n          backupNShare: backupKeyShare.nShares[1],\n        }),\n        reqId,\n      });\n      signedTxRequest.unsignedTxs.should.deepEqual(txRequest.unsignedTxs);\n      const userGpgActual = sendShareSpy.getCalls()[0].args[10] as string;\n      userGpgActual.should.startWith('-----BEGIN PGP PUBLIC KEY BLOCK-----');\n    });\n\n    it('signTxRequest should fail with wrong recipient', async function () {\n      await setupSignTxRequestNocks(true, userSignShare, aShare, dShare, enterpriseData);\n      await tssUtils\n        .signTxRequest({\n          txRequest: txRequestId,\n          prv: JSON.stringify({\n            pShare: userKeyShare.pShare,\n            bitgoNShare: bitgoKeyShare.nShares[1],\n            backupNShare: backupKeyShare.nShares[1],\n          }),\n          reqId,\n          txParams: { recipients: [{ address: '0x1234', amount: '100000000000000' }], type: 'transfer' },\n        })\n        .should.be.rejectedWith('destination address does not match with the recipient address');\n    });\n\n    it('signTxRequest should fail with incorrect value', async function () {\n      await setupSignTxRequestNocks(true, userSignShare, aShare, dShare, enterpriseData);\n      await tssUtils\n        .signTxRequest({\n          txRequest: txRequestId,\n          prv: JSON.stringify({\n            pShare: userKeyShare.pShare,\n            bitgoNShare: bitgoKeyShare.nShares[1],\n            backupNShare: backupKeyShare.nShares[1],\n          }),\n          reqId,\n          txParams: {\n            recipients: [{ address: '0xa1cfb9d51c0af191ff21c5f0f01723e056f7dc12', amount: '1' }],\n            type: 'transfer',\n          },\n        })\n        .should.be.rejectedWith('the transaction amount in txPrebuild does not match the value given by client');\n    });\n\n    it('signTxRequest should fail with incorrect value for token txn', async function () {\n      const signableHex =\n        '02f86d8242681083122c9e83122cae8301e04994ebe8b46a42f05072b723b00013ff822b2af1b5cb80b844a9059cbb0000000000000000000000002b0d6cb2f8c388757f4d7ad857fccab18290dbc900000000000000000000000000000000000000000000000000000000000186a0c0';\n      const serializedTxHex =\n        '02f8708242681083122c9e83122cae8301e04994ebe8b46a42f05072b723b00013ff822b2af1b5cb80b844a9059cbb0000000000000000000000002b0d6cb2f8c388757f4d7ad857fccab18290dbc900000000000000000000000000000000000000000000000000000000000186a0c0808080';\n      await setupSignTxRequestNocks(true, userSignShare, aShare, dShare, enterpriseData, {\n        signableHex,\n        serializedTxHex,\n        apiVersion: 'full',\n      });\n      await tssUtils\n        .signTxRequest({\n          txRequest: txRequestId,\n          prv: JSON.stringify({\n            pShare: userKeyShare.pShare,\n            bitgoNShare: bitgoKeyShare.nShares[1],\n            backupNShare: backupKeyShare.nShares[1],\n          }),\n          reqId,\n          txParams: {\n            recipients: [{ address: '0x2b0d6cb2f8c388757f4d7ad857fccab18290dbc9', amount: '707' }],\n            type: 'transfer',\n          },\n        })\n        .should.be.rejectedWith('the transaction amount in txPrebuild does not match the value given by client');\n    });\n\n    it('getOfflineSignerPaillierModulus should succeed', async function () {\n      const paillierModulus = tssUtils.getOfflineSignerPaillierModulus({\n        prv: JSON.stringify({\n          pShare: userKeyShare.pShare,\n          bitgoNShare: bitgoKeyShare.nShares[1],\n          backupNShare: backupKeyShare.nShares[1],\n        }),\n      });\n      paillierModulus.userPaillierModulus.should.equal(userKeyShare.pShare.n);\n    });\n\n    it('createOfflineKShare should succeed', async function () {\n      const mockPassword = 'password';\n      const step1SigningMaterial = await tssUtils.createOfflineKShare({\n        tssParams: {\n          txRequest,\n          reqId: reqId,\n        },\n        challenges: {\n          enterpriseChallenge: enterpriseChallenges,\n          bitgoChallenge: bitgoChallenges,\n        },\n        prv: JSON.stringify({\n          pShare: userKeyShare.pShare,\n          bitgoNShare: bitgoKeyShare.nShares[1],\n          backupNShare: backupKeyShare.nShares[1],\n        }),\n        requestType: RequestType.tx,\n        walletPassphrase: mockPassword,\n      });\n      step1SigningMaterial.privateShareProof.should.startWith('-----BEGIN PGP PUBLIC KEY BLOCK-----');\n      step1SigningMaterial.vssProof?.length.should.equal(userKeyShare.nShares[3].v?.length);\n      step1SigningMaterial.publicShare.length.should.equal(\n        userKeyShare.nShares[3].y.length + userKeyShare.nShares[3].chaincode.length\n      );\n      step1SigningMaterial.encryptedSignerOffsetShare.should.startWith('-----BEGIN PGP MESSAGE-----');\n      step1SigningMaterial.userPublicGpgKey.should.startWith('-----BEGIN PGP PUBLIC KEY BLOCK-----');\n      step1SigningMaterial.kShare.n.should.equal(userKeyShare.pShare.n);\n      step1SigningMaterial.wShare.should.startWith('{\"iv\":');\n    });\n\n    it('createOfflineKShare should fail with txId passed', async function () {\n      const mockPassword = 'password';\n      await tssUtils\n        .createOfflineKShare({\n          tssParams: {\n            txRequest: txRequest.txRequestId,\n            reqId: reqId,\n          },\n          challenges: {\n            enterpriseChallenge: enterpriseChallenges,\n            bitgoChallenge: bitgoChallenges,\n          },\n          prv: JSON.stringify({\n            pShare: userKeyShare.pShare,\n            bitgoNShare: bitgoKeyShare.nShares[1],\n            backupNShare: backupKeyShare.nShares[1],\n          }),\n          requestType: RequestType.tx,\n          walletPassphrase: mockPassword,\n        })\n        .should.be.rejectedWith('Invalid txRequest type');\n    });\n\n    // Seems to be flaky on CI, failed here: https://github.com/BitGo/BitGoJS/actions/runs/5902489990/job/16010623888?pr=3822\n    xit('createOfflineMuDeltaShare should succeed', async function () {\n      const mockPassword = 'password';\n      const alphaLength = 1536;\n      const deltaLength = 64;\n      const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n      const step2SigningMaterial = await tssUtils.createOfflineMuDeltaShare({\n        aShareFromBitgo: aShare,\n        bitgoChallenge: bitgoChallenges,\n        encryptedWShare: bitgo.encrypt({ input: JSON.stringify(wShare), password: mockPassword }),\n        walletPassphrase: mockPassword,\n      });\n      step2SigningMaterial.muDShare.muShare.alpha.length.should.equal(alphaLength);\n      step2SigningMaterial.muDShare.dShare.delta.length.should.equal(deltaLength);\n      step2SigningMaterial.oShare.should.startWith('{\"iv\":');\n    });\n\n    it('createOfflineMuDeltaShare should fail with incorrect password', async function () {\n      const mockPassword = 'password';\n      const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n      await tssUtils\n        .createOfflineMuDeltaShare({\n          aShareFromBitgo: aShare,\n          bitgoChallenge: bitgoChallenges,\n          encryptedWShare: bitgo.encrypt({ input: JSON.stringify(wShare), password: mockPassword }),\n          walletPassphrase: 'password1',\n        })\n        .should.be.rejectedWith(\"password error - ccm: tag doesn't match\");\n    });\n\n    it('createOfflineSShare should succeed', async function () {\n      const mockPassword = 'password';\n      const pubKeyLength = 66;\n      const privKeyLength = 64;\n      const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n      const step3SigningMaterial = await tssUtils.createOfflineSShare({\n        tssParams: {\n          txRequest: txRequest,\n          reqId: reqId,\n        },\n        dShareFromBitgo: dShare,\n        encryptedOShare: bitgo.encrypt({ input: JSON.stringify(oShare), password: mockPassword }),\n        walletPassphrase: mockPassword,\n        requestType: RequestType.tx,\n      });\n      step3SigningMaterial.R.length.should.equal(pubKeyLength);\n      step3SigningMaterial.y.length.should.equal(pubKeyLength);\n      step3SigningMaterial.s.length.should.equal(privKeyLength);\n    });\n\n    it('createOfflineSShare should fail with txId passed', async function () {\n      const mockPassword = 'password';\n      const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n      await tssUtils\n        .createOfflineSShare({\n          tssParams: {\n            txRequest: txRequest.txRequestId,\n            reqId: reqId,\n          },\n          dShareFromBitgo: dShare,\n          encryptedOShare: bitgo.encrypt({ input: JSON.stringify(oShare), password: mockPassword }),\n          walletPassphrase: mockPassword,\n          requestType: RequestType.tx,\n        })\n        .should.be.rejectedWith('Invalid txRequest type');\n    });\n\n    it('signTxRequest should fail with invalid user prv', async function () {\n      const invalidUserKey = { ...userKeyShare, pShare: { ...userKeyShare.pShare, i: 2 } };\n      await tssUtils\n        .signTxRequest({\n          txRequest: txRequestId,\n          prv: JSON.stringify({\n            pShare: invalidUserKey.pShare,\n            bitgoNShare: bitgoKeyShare.nShares[1],\n            backupNShare: backupKeyShare.nShares[1],\n          }),\n          reqId,\n        })\n        .should.be.rejectedWith('Invalid user key');\n    });\n\n    it('signTxRequest should fail with no backupNShares', async function () {\n      const getTxRequest = sandbox.stub(tssUtils, 'getTxRequest');\n      getTxRequest.resolves(txRequest);\n      getTxRequest.calledWith(txRequestId);\n      setupSignTxRequestNocks(false, userSignShare, aShare, dShare, enterpriseData);\n      await tssUtils\n        .signTxRequest({\n          txRequest: txRequestId,\n          prv: JSON.stringify({\n            pShare: userKeyShare.pShare,\n            bitgoNShare: bitgoKeyShare.nShares[1],\n          }),\n          reqId,\n        })\n        .should.be.rejectedWith('Invalid user key - missing backupNShare');\n    });\n\n    async function setupSignTxRequestNocks(\n      isTxRequest = true,\n      userSignShare: ECDSA.SignShareRT,\n      aShare: ECDSA.AShare,\n      dShare: ECDSA.DShare,\n      enterpriseData: EnterpriseData,\n      {\n        signableHex,\n        serializedTxHex,\n        apiVersion,\n      }: { signableHex?: string; serializedTxHex?: string; apiVersion?: 'full' | 'lite' } = {}\n    ) {\n      if (enterpriseData) {\n        await nockGetEnterprise({ enterpriseId: enterpriseData.id, response: enterpriseData, times: 1 });\n      }\n      const derivationPath = '';\n      sinon.stub(ECDSAMethods, 'createUserSignShare').resolves(userSignShare);\n      let response = {\n        txRequests: [\n          {\n            ...txRequest,\n            transactions: [\n              {\n                ...txRequest,\n                unsignedTx: {\n                  signableHex: signableHex ?? txRequest.unsignedTxs[0].signableHex,\n                  serializedTxHex: serializedTxHex ?? txRequest.unsignedTxs[0].serializedTxHex,\n                  derivationPath,\n                },\n              },\n            ],\n            apiVersion: apiVersion,\n          },\n        ],\n      };\n      if (isTxRequest) {\n        await nockGetTxRequest({ walletId: wallet.id(), txRequestId: txRequest.txRequestId, response: response });\n      }\n      const aRecord = ECDSAMethods.convertAShare(aShare);\n      const signatureShares = [aRecord];\n      txRequest.signatureShares = signatureShares;\n      response = {\n        txRequests: [\n          {\n            ...txRequest,\n            transactions: [\n              {\n                ...txRequest,\n                unsignedTx: {\n                  signableHex: txRequest.unsignedTxs[0].signableHex,\n                  serializedTxHex: txRequest.unsignedTxs[0].serializedTxHex,\n                  derivationPath,\n                },\n              },\n            ],\n            apiVersion: apiVersion,\n          },\n        ],\n      };\n      await nockGetTxRequest({ walletId: wallet.id(), txRequestId: txRequest.txRequestId, response: response });\n      const dRecord = ECDSAMethods.convertDShare(dShare);\n      signatureShares.push(dRecord);\n      response = {\n        txRequests: [\n          {\n            ...txRequest,\n            transactions: [\n              {\n                ...txRequest,\n                unsignedTx: {\n                  signableHex: txRequest.unsignedTxs[0].signableHex,\n                  serializedTxHex: txRequest.unsignedTxs[0].serializedTxHex,\n                  derivationPath,\n                },\n              },\n            ],\n            apiVersion: apiVersion,\n          },\n        ],\n      };\n      await nockGetTxRequest({ walletId: wallet.id(), txRequestId: txRequest.txRequestId, response: response });\n      await nockGetTxRequest({ walletId: wallet.id(), txRequestId: txRequest.txRequestId, response: response });\n    }\n  });\n\n  describe('getEcdsaSigningChallenges', function () {\n    const mockWalletPaillierKey = {\n      n: 'f47be4c2d8bc1e28f88c6c4da634da97d92a1c279a7b0fe7b87c337c36a27b32ce0ff0c45f16e4e15bbd20e4e640de12047eff9b1a2b98144f9a268d406bd000d192a35b6847a17e40fb85f55b314d001ff87393481cafe391807d0eb83eff9e38614b38e5f25fc4449cb01caed805584d026b5d866c723f3d4d4f1e462662f2113b1561eb2bf755b4b91d0308d8eacc439167da8b7d6e108524f226960360af00215d9614457414ebdbe8834999689e2e903208c8713ff5d9901f9eaba3aa81d705323cbbba61ba7fa9f3228f30853fb55da1b3d3ed7db1dfc6545bc96aa8d2eb848931c1b807fdfe8f65af72f68638a82fe9e22ac1f0f032e621066806a1f144b5719a5f091986867b384be6c34146c8241cbfbd781966ebbcd19e6caa27fab040e62e5a162888aa8624d046c8fe3b72244f04a7264c4a36b6366dbe7da98afb201d34be2c0d6dd11982af35bf7535582b263914725aaec280d52290527382d3ab297d746c41aacd8de98c09fcfb85a95e02de1b34d4933e51045e2f1ce8af',\n      lambda:\n        'f47be4c2d8bc1e28f88c6c4da634da97d92a1c279a7b0fe7b87c337c36a27b32ce0ff0c45f16e4e15bbd20e4e640de12047eff9b1a2b98144f9a268d406bd000d192a35b6847a17e40fb85f55b314d001ff87393481cafe391807d0eb83eff9e38614b38e5f25fc4449cb01caed805584d026b5d866c723f3d4d4f1e462662f2113b1561eb2bf755b4b91d0308d8eacc439167da8b7d6e108524f226960360af00215d9614457414ebdbe8834999689e2e903208c8713ff5d9901f9eaba3aa7fc3d0c0bcc5bff644156ab887146d51bcee1eef70f45c486147d687ee37def1f8a16bc945eff22dd4dca3614a99158823acd9492e347f7ec79a7771024205d07f27b30cd20340e330411da8fa2da209e5cc688da94d1dbef54bfd9c69b4e99cf06d67309a3420b82c78a0fe0dd0b9c31382eae38746cfdd27fa90022a50532246c8ae1339c93e183c03bf6fd7014be3658abc73baae1fa5b86dab94b9f125395a818e54dde6235c45d3dbc032b3078e9df1cad69d8ac19a7cb6405a558b7bfba8',\n    };\n    const mockBitgoPaillierKey = {\n      n: 'f010d294effceb8c4f96af1978ab367c4fbb272c2169317e41ae87220652cae2ce929696ee55ec6831aa6b4b3b931babc2bac9c1a20fddbca925cc99680791f7c3157b3d31256ee72c47d47db567e0f070dce121c3a4d9e003c1f1389073acb252c65d2b0723e86e3265f67a137cb1e23f4551544405644d0ae63d35f25f40becd2b693879f3bdbec3f7250791a3f3c975a5ac78a0e81dcd1a87eb2ca67010dff880b2338556275de23d9e88d21b77da0d524ddc2b394f8de00b1af0ce85f6eee2e05a184e05494d66d2c636045bf70ed15ebd0f41a8eea2920af85e6d68a0ce11fc2abbcb3cebcc3c23ec2e148c318683a5426e15b5207efd3b9b05cb919ec4340f74dff336986d0c923df10a789007b1da9daddf8edf3014e93989f30243f27f9a307d55d630cbfcd16cd6a95a41dee10c31acc293df6834ce0e3ea5b68f170bd7938ea0c2eeb788e16f30af57b3f0888fb44d3610e7eeba60e7fd8cc4a8f044718dfc6174bf4a380690dc1dc77472a48892eb3e81775540ea0acc9e89b639',\n      lambda:\n        'f010d294effceb8c4f96af1978ab367c4fbb272c2169317e41ae87220652cae2ce929696ee55ec6831aa6b4b3b931babc2bac9c1a20fddbca925cc99680791f7c3157b3d31256ee72c47d47db567e0f070dce121c3a4d9e003c1f1389073acb252c65d2b0723e86e3265f67a137cb1e23f4551544405644d0ae63d35f25f40becd2b693879f3bdbec3f7250791a3f3c975a5ac78a0e81dcd1a87eb2ca67010dff880b2338556275de23d9e88d21b77da0d524ddc2b394f8de00b1af0ce85f6ecdef8a4bc955a28ecef7d97cded079d390e77c80998d78ad9510cbabfeb8f0a157dbfc590b4d59ee8c0b088f9d89473b557320078a117478624f5d1df36e30f320b6722a4217dcb46b978cc6c8f1a21c8a6c74bce84d82c481402c99a69b798e3c05f23350b4aade4f79784b1c09692b6a33cfba7f145597d82b799cccef620c36f1fbbe2cb4ac0ea395c476e381bc475d41722320f541ae9bf56aa4a12dff3ea7ab11174fb5b8df7429c9f57d36f8fc51e1a8c647d5b8fa0189fb8acdbd0a780',\n    };\n    const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n    const txRequestId = 'fakeTxRequestId';\n    const rawEntChallengeWithProofs: EcdsaTypes.SerializedNtildeWithProofs = mockSerializedChallengeWithProofs;\n    let rawBitgoChallenge: EcdsaTypes.SerializedEcdsaChallenges & { n: string };\n    const adminEcdhKey = bitgo.keychains().create();\n    const fakeAdminEcdhKey = bitgo.keychains().create();\n    const derivationPath = 'm/0/0';\n    const mockedSigningKey = {\n      userId: 'id',\n      userEmail: 'user@bitgo.com',\n      derivedPubkey: bip32.fromBase58(adminEcdhKey.xpub).derivePath(derivationPath).publicKey.toString('hex'),\n      derivationPath: derivationPath,\n      ecdhKeychain: 'my keychain',\n    };\n\n    before(async function () {\n      const p = await EcdsaPaillierProof.generateP(hexToBigInt(mockWalletPaillierKey.n));\n      rawBitgoChallenge = {\n        ...EcdsaTypes.serializeNtilde(EcdsaTypes.deserializeNtilde(mockSerializedChallengeWithProofs2)),\n        p: EcdsaTypes.serializePaillierChallenge({ p }).p,\n        n: mockBitgoPaillierKey.n,\n      };\n    });\n\n    afterEach(function () {\n      sinon.restore();\n      nock.cleanAll();\n    });\n\n    it('should fetch static ent and bitgo challenges with the ent feature flag and verify them', async function () {\n      await nockGetChallenge({ walletId, txRequestId, addendum: '/transactions/0', response: rawBitgoChallenge });\n      await nockGetSigningKey({ enterpriseId, userId: mockedSigningKey.userId, response: mockedSigningKey, times: 1 });\n      const adminSignatureEntChallenge = ECDSAUtils.EcdsaUtils.signChallenge(\n        rawEntChallengeWithProofs,\n        adminEcdhKey.xprv,\n        derivationPath\n      );\n      const adminSignatureBitGoChallenge = ECDSAUtils.EcdsaUtils.signChallenge(\n        rawBitgoChallenge,\n        adminEcdhKey.xprv,\n        derivationPath\n      );\n      const mockChallengesResponse = {\n        enterpriseChallenge: {\n          ...rawEntChallengeWithProofs,\n          verifiers: {\n            adminSignature: adminSignatureEntChallenge.toString('hex'),\n          },\n        },\n        bitgoChallenge: {\n          ...rawBitgoChallenge,\n          verifiers: {\n            adminSignature: adminSignatureBitGoChallenge.toString('hex'),\n          },\n        },\n        createdBy: 'id',\n      };\n\n      await nockGetChallenges({ walletId: walletId, response: mockChallengesResponse });\n      const challenges = await tssUtils.getEcdsaSigningChallenges(txRequestId, 0, mockWalletPaillierKey.n, 0);\n      should.exist(challenges);\n      const expectedRangeProofChallenges = {\n        enterpriseChallenge: {\n          ntilde: challenges.enterpriseChallenge.ntilde,\n          h1: challenges.enterpriseChallenge.h1,\n          h2: challenges.enterpriseChallenge.h2,\n        },\n        bitgoChallenge: rawBitgoChallenge,\n      };\n      expectedRangeProofChallenges.should.deepEqual({\n        enterpriseChallenge: {\n          ntilde: rawEntChallengeWithProofs.ntilde,\n          h1: rawEntChallengeWithProofs.h1,\n          h2: rawEntChallengeWithProofs.h2,\n        },\n        bitgoChallenge: rawBitgoChallenge,\n      });\n    });\n\n    it('Fails if the enterprise challenge signature is different from the admin ecdh key', async function () {\n      await nockGetChallenge({ walletId, txRequestId, addendum: '/transactions/0', response: rawBitgoChallenge });\n      await nockGetEnterprise({\n        enterpriseId: enterpriseData.id,\n        response: {\n          ...enterpriseData,\n          featureFlags: ['useEnterpriseEcdsaTssChallenge'],\n        },\n        times: 1,\n      });\n      await nockGetSigningKey({ enterpriseId, userId: mockedSigningKey.userId, response: mockedSigningKey, times: 1 });\n      // Bad sign\n      const adminSignedEntChallenge = ECDSAUtils.EcdsaUtils.signChallenge(\n        rawEntChallengeWithProofs,\n        fakeAdminEcdhKey.xprv,\n        derivationPath\n      );\n      const adminSignedBitGoChallenge = ECDSAUtils.EcdsaUtils.signChallenge(\n        rawBitgoChallenge,\n        adminEcdhKey.xprv,\n        derivationPath\n      );\n      const mockChallengesResponse = {\n        enterpriseChallenge: {\n          ...rawEntChallengeWithProofs,\n          verifiers: {\n            adminSignature: adminSignedEntChallenge.toString('hex'),\n          },\n        },\n        bitgoChallenge: {\n          ...rawBitgoChallenge,\n          verifiers: {\n            adminSignature: adminSignedBitGoChallenge.toString('hex'),\n          },\n        },\n        createdBy: 'id',\n      };\n      await nockGetChallenges({ walletId: walletId, response: mockChallengesResponse });\n      await tssUtils\n        .getEcdsaSigningChallenges(txRequestId, 0, mockWalletPaillierKey.n)\n        .should.be.rejectedWith(\n          'Admin signature for enterprise challenge is not valid. Please contact your enterprise admin.'\n        );\n    });\n\n    it('Fails if the bitgo challenge signature is different from the admin ecdh key', async function () {\n      await nockGetChallenge({ walletId, txRequestId, addendum: '/transactions/0', response: rawBitgoChallenge });\n      await nockGetEnterprise({\n        enterpriseId: enterpriseData.id,\n        response: {\n          ...enterpriseData,\n          featureFlags: ['useEnterpriseEcdsaTssChallenge'],\n        },\n        times: 1,\n      });\n      await nockGetSigningKey({ enterpriseId, userId: mockedSigningKey.userId, response: mockedSigningKey, times: 1 });\n      const adminSignedEntChallenge = ECDSAUtils.EcdsaUtils.signChallenge(\n        rawEntChallengeWithProofs,\n        adminEcdhKey.xprv,\n        derivationPath\n      );\n      // Bad sign\n      const adminSignedBitGoChallenge = ECDSAUtils.EcdsaUtils.signChallenge(\n        rawBitgoChallenge,\n        fakeAdminEcdhKey.xprv,\n        derivationPath\n      );\n      const mockChallengesResponse = {\n        enterpriseChallenge: {\n          ...rawEntChallengeWithProofs,\n          verifiers: {\n            adminSignature: adminSignedEntChallenge.toString('hex'),\n          },\n        },\n        bitgoChallenge: {\n          ...rawBitgoChallenge,\n          verifiers: {\n            adminSignature: adminSignedBitGoChallenge.toString('hex'),\n          },\n        },\n        createdBy: 'id',\n      };\n      await nockGetChallenges({ walletId: walletId, response: mockChallengesResponse });\n      await tssUtils\n        .getEcdsaSigningChallenges(txRequestId, 0, mockWalletPaillierKey.n)\n        .should.be.rejectedWith(\n          \"Admin signature for BitGo's challenge is not valid. Please contact your enterprise admin.\"\n        );\n    });\n  });\n\n  describe('getVerifyAndSignBitGoChallenges', function () {\n    const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n    const adminEcdhKey = bitgo.keychains().create();\n    const derivationPath = 'm/0/0';\n    const bitgoInstChallenge = mockChallengeA;\n    const bitgoNitroChallenge = mockChallengeB;\n    const userPassword = 'password123';\n    const encryptedXprv = bitgo.encrypt({\n      password: userPassword,\n      input: adminEcdhKey.xprv,\n    });\n\n    beforeEach(async function () {\n      sinon.stub(bitgo, 'getSigningKeyForUser').resolves({\n        userId: 'id',\n        userEmail: 'user@bitgo.com',\n        derivedPubkey: bip32.fromBase58(adminEcdhKey.xpub).derivePath(derivationPath).publicKey.toString('hex'),\n        derivationPath: derivationPath,\n        ecdhKeychain: 'my keychain',\n      });\n      sinon.stub(bitgo, 'getECDHKeychain').resolves({\n        encryptedXprv: encryptedXprv,\n      });\n    });\n\n    afterEach(async function () {\n      sinon.restore();\n      nock.cleanAll();\n    });\n\n    function nockGetBitgoChallenges(response: unknown): nock.Scope {\n      return nock(bgUrl)\n        .get(`/api/v2/tss/ecdsa/challenges`)\n        .times(1)\n        .reply(() => [200, response]);\n    }\n\n    it('succeeds for valid bitgo proofs', async function () {\n      const nockGetBitgoChallengesApi = nockGetBitgoChallenges({\n        bitgoNitroHsm: bitgoNitroChallenge,\n        bitgoInstitutionalHsm: bitgoInstChallenge,\n      });\n\n      await ECDSAUtils.EcdsaUtils.getVerifyAndSignBitGoChallenges(\n        bitgo,\n        'ent_id',\n        userPassword\n      ).should.not.be.rejected();\n      nockGetBitgoChallengesApi.isDone().should.be.true();\n    });\n\n    it('Fails if bitgo challenge proofs are not present', async function () {\n      const nockGetBitgoChallengesApi = nockGetBitgoChallenges({\n        bitgoNitroHsm: {\n          ...bitgoNitroChallenge,\n          ntildeProof: undefined,\n        },\n        bitgoInstitutionalHsm: bitgoInstChallenge,\n      });\n      await ECDSAUtils.EcdsaUtils.getVerifyAndSignBitGoChallenges(bitgo, 'ent_id', userPassword).should.be.rejectedWith(\n        'Expected BitGo challenge proof to be present. Contact support@bitgo.com.'\n      );\n      nockGetBitgoChallengesApi.isDone().should.be.true();\n    });\n\n    it('Fails if the user password to decrypt the ecdhkeychain is wrong', async function () {\n      const nockGetBitgoChallengesApi = nockGetBitgoChallenges({\n        bitgoNitroHsm: bitgoNitroChallenge,\n        bitgoInstitutionalHsm: bitgoInstChallenge,\n      });\n      await ECDSAUtils.EcdsaUtils.getVerifyAndSignBitGoChallenges(bitgo, 'ent_id', 'bro').should.be.rejectedWith(\n        'Incorrect password. Please try again.'\n      );\n      nockGetBitgoChallengesApi.isDone().should.be.true();\n    });\n\n    it('Fails bitgo challenge proofs for faulty nitro h2WrtH1 proof', async function () {\n      const nockGetBitgoChallengesApi = nockGetBitgoChallenges({\n        bitgoNitroHsm: {\n          ...bitgoNitroChallenge,\n          ntildeProof: {\n            ...bitgoNitroChallenge.ntildeProof,\n            h2WrtH1: bitgoNitroChallenge.ntildeProof.h1WrtH2,\n          },\n        },\n        bitgoInstitutionalHsm: bitgoInstChallenge,\n      });\n      await ECDSAUtils.EcdsaUtils.getVerifyAndSignBitGoChallenges(bitgo, 'ent_id', userPassword).should.be.rejectedWith(\n        \"Failed to verify BitGo's challenge needed to enable ECDSA signing. Please contact support@bitgo.com\"\n      );\n      nockGetBitgoChallengesApi.isDone().should.be.true();\n    });\n\n    it('Fails bitgo challenge proofs for faulty nitro h1WrtH2 proof', async function () {\n      const nockGetBitgoChallengesApi = nockGetBitgoChallenges({\n        bitgoNitroHsm: {\n          ...bitgoNitroChallenge,\n          ntildeProof: {\n            ...bitgoNitroChallenge.ntildeProof,\n            h1WrtH2: bitgoNitroChallenge.ntildeProof.h2WrtH1,\n          },\n        },\n        bitgoInstitutionalHsm: bitgoInstChallenge,\n      });\n      await ECDSAUtils.EcdsaUtils.getVerifyAndSignBitGoChallenges(bitgo, 'ent_id', userPassword).should.be.rejectedWith(\n        \"Failed to verify BitGo's challenge needed to enable ECDSA signing. Please contact support@bitgo.com\"\n      );\n      nockGetBitgoChallengesApi.isDone().should.be.true();\n    });\n\n    it('Fails bitgo challenge proofs for faulty inst h2WrtH1 proof', async function () {\n      const nockGetBitgoChallengesApi = nock(bgUrl)\n        .get(`/api/v2/tss/ecdsa/challenges`)\n        .times(1)\n        .reply(200, {\n          bitgoNitroHsm: bitgoNitroChallenge,\n          bitgoInstitutionalHsm: {\n            ...bitgoInstChallenge,\n            ntildeProof: {\n              ...bitgoInstChallenge.ntildeProof,\n              h2WrtH1: bitgoInstChallenge.ntildeProof.h1WrtH2,\n            },\n          },\n        });\n      await ECDSAUtils.EcdsaUtils.getVerifyAndSignBitGoChallenges(bitgo, 'ent_id', userPassword).should.be.rejectedWith(\n        \"Failed to verify BitGo's challenge needed to enable ECDSA signing. Please contact support@bitgo.com\"\n      );\n      nockGetBitgoChallengesApi.isDone().should.be.true();\n    });\n\n    it('Fails bitgo challenge proofs for faulty inst h1WrtH2 proof', async function () {\n      const nockGetBitgoChallengesApi = nock(bgUrl)\n        .get(`/api/v2/tss/ecdsa/challenges`)\n        .times(1)\n        .reply(200, {\n          bitgoNitroHsm: bitgoNitroChallenge,\n          bitgoInstitutionalHsm: {\n            ...bitgoInstChallenge,\n            ntildeProof: {\n              ...bitgoInstChallenge.ntildeProof,\n              h1WrtH2: bitgoInstChallenge.ntildeProof.h2WrtH1,\n            },\n          },\n        });\n      await ECDSAUtils.EcdsaUtils.getVerifyAndSignBitGoChallenges(bitgo, 'ent_id', userPassword).should.be.rejectedWith(\n        \"Failed to verify BitGo's challenge needed to enable ECDSA signing. Please contact support@bitgo.com\"\n      );\n      nockGetBitgoChallengesApi.isDone().should.be.true();\n    });\n  });\n\n  describe('supportedTxRequestVersions', function () {\n    it('returns only full for hot wallets', function () {\n      const hotWallet = new Wallet(bitgo, baseCoin, { type: 'hot', multisigType: 'tss' });\n      const hotWalletTssUtils = new ECDSAUtils.EcdsaUtils(bitgo, baseCoin, hotWallet);\n      hotWalletTssUtils.supportedTxRequestVersions().should.deepEqual(['full']);\n    });\n    it('returns only full for cold wallets', function () {\n      const coldWallet = new Wallet(bitgo, baseCoin, {\n        type: 'cold',\n        multisigType: 'tss',\n      });\n      const coldWalletTssUtils = new ECDSAUtils.EcdsaUtils(bitgo, baseCoin, coldWallet);\n\n      coldWalletTssUtils.supportedTxRequestVersions().should.deepEqual(['full']);\n    });\n    it('returns only full for custodial wallets', function () {\n      const custodialWallet = new Wallet(bitgo, baseCoin, { type: 'custodial', multisigType: 'tss' });\n      const custodialWalletTssUtils = new ECDSAUtils.EcdsaUtils(bitgo, baseCoin, custodialWallet);\n      custodialWalletTssUtils.supportedTxRequestVersions().should.deepEqual(['full']);\n    });\n    it('returns empty for trading wallets', function () {\n      const tradingWallet = new Wallet(bitgo, baseCoin, { type: 'trading', multisigType: 'tss' });\n      const tradingWalletTssUtils = new ECDSAUtils.EcdsaUtils(bitgo, baseCoin, tradingWallet);\n      tradingWalletTssUtils.supportedTxRequestVersions().should.deepEqual([]);\n    });\n    it('returns empty for non-tss wallets', function () {\n      const nonTssWalletData = { coin: 'tbtc', multisigType: 'onchain' };\n      const btcCoin = bitgo.coin('tbtc');\n      const nonTssWallet = new Wallet(bitgo, btcCoin, nonTssWalletData);\n      const nonTssWalletTssUtils = new TssUtils(bitgo, btcCoin, nonTssWallet);\n      nonTssWalletTssUtils.supportedTxRequestVersions().should.deepEqual([]);\n    });\n  });\n\n  describe('initiateChallengesForEnterprise', function () {\n    const bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n    const adminEcdhKey = bitgo.keychains().create();\n    const derivationPath = 'm/0/0';\n    const bitgoInstChallenge = mockChallengeA;\n    const bitgoNitroChallenge = mockChallengeB;\n    const serializedEntChallenge = mockChallengeC;\n    const userPassword = 'password123';\n    const encryptedXprv = bitgo.encrypt({\n      password: userPassword,\n      input: adminEcdhKey.xprv,\n    });\n\n    beforeEach(async function () {\n      sinon.stub(bitgo, 'getSigningKeyForUser').resolves({\n        userId: 'id',\n        userEmail: 'user@bitgo.com',\n        derivedPubkey: bip32.fromBase58(adminEcdhKey.xpub).derivePath(derivationPath).publicKey.toString('hex'),\n        derivationPath: derivationPath,\n        ecdhKeychain: 'my keychain',\n      });\n\n      sinon.stub(bitgo, 'getECDHKeychain').resolves({\n        encryptedXprv: encryptedXprv,\n      });\n    });\n\n    afterEach(async function () {\n      sinon.restore();\n    });\n\n    it('should upload challenge without generating if passed in', async function () {\n      const stubUploadChallenge = sinon.stub(ECDSAUtils.EcdsaUtils, 'uploadChallengesToEnterprise');\n      const deserializedEntChallenge = EcdsaTypes.deserializeNtildeWithProofs(serializedEntChallenge);\n\n      const signedEntChallenge = ECDSAUtils.EcdsaUtils.signChallenge(\n        serializedEntChallenge,\n        adminEcdhKey.xprv,\n        derivationPath\n      );\n      const signedInstChallenge = ECDSAUtils.EcdsaUtils.signChallenge(\n        bitgoInstChallenge,\n        adminEcdhKey.xprv,\n        derivationPath\n      );\n      const signedNitroChallenge = ECDSAUtils.EcdsaUtils.signChallenge(\n        bitgoNitroChallenge,\n        adminEcdhKey.xprv,\n        derivationPath\n      );\n\n      await ECDSAUtils.EcdsaUtils.initiateChallengesForEnterprise(\n        bitgo,\n        'ent_id',\n        userPassword,\n        signedInstChallenge,\n        signedNitroChallenge,\n        openSSLBytes,\n        deserializedEntChallenge\n      ).should.not.be.rejected();\n      stubUploadChallenge.calledWith(\n        bitgo,\n        'ent_id',\n        serializedEntChallenge,\n        signedEntChallenge.toString('hex'),\n        signedInstChallenge.toString('hex'),\n        signedNitroChallenge.toString('hex')\n      );\n    });\n\n    it('should generate a challenge and if one is not provided', async function () {\n      const stubUploadChallenge = sinon.stub(ECDSAUtils.EcdsaUtils, 'uploadChallengesToEnterprise');\n      const deserializedEntChallenge = EcdsaTypes.deserializeNtildeWithProofs(serializedEntChallenge);\n      sinon.stub(EcdsaRangeProof, 'generateNtilde').resolves(deserializedEntChallenge);\n\n      const signedEntChallenge = ECDSAUtils.EcdsaUtils.signChallenge(\n        serializedEntChallenge,\n        adminEcdhKey.xprv,\n        derivationPath\n      );\n      const signedInstChallenge = ECDSAUtils.EcdsaUtils.signChallenge(\n        bitgoInstChallenge,\n        adminEcdhKey.xprv,\n        derivationPath\n      );\n      const signedNitroChallenge = ECDSAUtils.EcdsaUtils.signChallenge(\n        bitgoNitroChallenge,\n        adminEcdhKey.xprv,\n        derivationPath\n      );\n\n      await ECDSAUtils.EcdsaUtils.initiateChallengesForEnterprise(\n        bitgo,\n        'ent_id',\n        userPassword,\n        signedInstChallenge,\n        signedNitroChallenge,\n        openSSLBytes\n      ).should.not.be.rejected();\n      stubUploadChallenge.calledWith(\n        bitgo,\n        'ent_id',\n        serializedEntChallenge,\n        signedEntChallenge.toString('hex'),\n        signedInstChallenge.toString('hex'),\n        signedNitroChallenge.toString('hex')\n      );\n    });\n  });\n\n  it('getMessageToSignFromChallenge concatenates the challenge values only', function () {\n    const challenge = mockChallengeA;\n    const expectedMessageToSign = challenge.ntilde.concat(challenge.h1).concat(challenge.h2);\n    const message = ECDSAUtils.EcdsaUtils.getMessageToSignFromChallenge(challenge);\n    message.should.equal(expectedMessageToSign);\n  });\n  describe('validateCommonKeychainPublicKey', function () {\n    it('validateCommonKeychainPublicKey returns correct public key', function () {\n      const commonKeychain =\n        '03f40c70545b519bb7bbc7195fd4b7d5bbfc873bfd38b18596e4b47a05b6a88d552e2e8319cb31e279b99dbe54115a983d35e86679af96d81b7478d1df368f76a8';\n      const expectedPubKeyResult = `f40c70545b519bb7bbc7195fd4b7d5bbfc873bfd38b18596e4b47a05b6a88d556a10d6ab8055dc0b3a9af9dc4e42f4f9773c590afcc298d017c1b1ce29a88041`;\n      const actualPubKey = ECDSAUtils.EcdsaUtils.validateCommonKeychainPublicKey(commonKeychain);\n      actualPubKey.should.equal(expectedPubKeyResult);\n    });\n    it('validateCommonKeychainPublicKey throws correctly with invalid length', function () {\n      const commonKeychain = '03f40c70548';\n      should(() => ECDSAUtils.EcdsaUtils.validateCommonKeychainPublicKey(commonKeychain)).throwError(\n        'Invalid commonKeychain length, expected 130, got 11'\n      );\n    });\n    it('validateCommonKeychainPublicKey throws correctly with invalid commonKeychain', function () {\n      const commonKeychainWithInvalidCharacters =\n        '!@#$^0c70545b519bb7bbc7195fd4b7d5bfc873bfd38b18596e4b47a05b6a88d552e2e8319cb31e279b99dbe54115a983d35e86679af96d81b7478d1df368f76a8'; // 129 chars\n      should(() =>\n        ECDSAUtils.EcdsaUtils.validateCommonKeychainPublicKey(commonKeychainWithInvalidCharacters)\n      ).throwError(/^Invalid commonKeychain, error:/);\n    });\n  });\n\n  // #region Nock helpers\n  async function createIncompleteBitgoHeldBackupKeyShare(\n    userGpgKey: openpgp.SerializedKeyPair<string>,\n    backupKeyShare: KeyShare,\n    bitgoGpgKey: openpgp.SerializedKeyPair<string>\n  ): Promise<BitgoHeldBackupKeyShare> {\n    const nSharePromises = [\n      encryptNShare(backupKeyShare, 1, userGpgKey.publicKey, userGpgKey, false),\n      encryptNShare(backupKeyShare, 3, bitgoGpgKey.publicKey, userGpgKey, false),\n    ];\n\n    const backupToUserPublicShare = Buffer.concat([\n      Buffer.from(backupKeyShare.nShares[1].y, 'hex'),\n      Buffer.from(backupKeyShare.nShares[1].chaincode, 'hex'),\n    ]).toString('hex');\n\n    const backupToBitgoPublicShare = Buffer.concat([\n      Buffer.from(backupKeyShare.nShares[3].y, 'hex'),\n      Buffer.from(backupKeyShare.nShares[3].chaincode, 'hex'),\n    ]).toString('hex');\n\n    return {\n      id: '4711',\n      keyShares: [\n        {\n          from: 'backup',\n          to: 'user',\n          publicShare: backupToUserPublicShare,\n          privateShare: (await nSharePromises[0]).encryptedPrivateShare,\n        },\n        {\n          from: 'backup',\n          to: 'bitgo',\n          publicShare: backupToBitgoPublicShare,\n          privateShare: (await nSharePromises[1]).encryptedPrivateShare,\n        },\n      ],\n    };\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      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 nockCreateBitgoHeldBackupKeyShare(\n    coin: string,\n    enterpriseId: string,\n    userGpgKey: openpgp.SerializedKeyPair<string>,\n    backupKeyShare: KeyShare,\n    bitgoGpgKey: openpgp.SerializedKeyPair<string>\n  ): Promise<BitgoHeldBackupKeyShare> {\n    const keyShare = await createIncompleteBitgoHeldBackupKeyShare(userGpgKey, backupKeyShare, bitgoGpgKey);\n\n    nock(bgUrl)\n      .post(\n        `/api/v2/${coin}/krs/backupkeys`,\n        _.matches({ enterprise: enterpriseId, userGPGPublicKey: userGpgKey.publicKey })\n      )\n      .reply(201, keyShare);\n\n    return keyShare;\n  }\n\n  async function nockFinalizeBitgoHeldBackupKeyShare(\n    coin: string,\n    originalKeyShare: BitgoHeldBackupKeyShare,\n    commonKeychain: string,\n    userKeyShare: KeyShare,\n    userLocalBackupGpgKey: openpgp.SerializedKeyPair<string>,\n    bitgoKeychain: Keychain\n  ): Promise<BitgoHeldBackupKeyShare> {\n    const encryptedUserToBackupKeyShare = await encryptNShare(\n      userKeyShare,\n      2,\n      userLocalBackupGpgKey.publicKey,\n      userGpgKey,\n      false\n    );\n\n    assert(bitgoKeychain.keyShares);\n    const bitgoToBackupKeyShare = bitgoKeychain.keyShares.find(\n      (keyShare) => keyShare.from === 'bitgo' && keyShare.to === 'backup'\n    );\n    assert(bitgoToBackupKeyShare);\n\n    const userPublicShare = Buffer.concat([\n      Buffer.from(userKeyShare.nShares[2].y, 'hex'),\n      Buffer.from(userKeyShare.nShares[2].chaincode, 'hex'),\n    ]).toString('hex');\n\n    const expectedKeyShares = [\n      {\n        from: 'user',\n        to: 'backup',\n        publicShare: userPublicShare,\n        // Omitting the private share, the actual encryption happens inside the function where we make the matching call\n        // to this nock. We cannot recreate the same encrypted value here because gpg encryption is not deterministic\n      },\n      bitgoToBackupKeyShare,\n    ];\n\n    const updatedKeyShare: BitgoHeldBackupKeyShare = {\n      id: originalKeyShare.id,\n      commonKeychain,\n      keyShares: [\n        ...originalKeyShare.keyShares,\n        {\n          from: 'user',\n          to: 'backup',\n          publicShare: userPublicShare,\n          privateShare: encryptedUserToBackupKeyShare.encryptedPrivateShare,\n        },\n        bitgoToBackupKeyShare,\n      ],\n    };\n\n    nock(bgUrl)\n      .put(\n        `/api/v2/${coin}/krs/backupkeys/${originalKeyShare.id}`,\n        _.matches({ commonKeychain, keyShares: expectedKeyShares })\n      )\n      .reply(200, updatedKeyShare);\n\n    return updatedKeyShare;\n  }\n\n  /**\n   * Helper function to generate a bitgo keychain given the full set of keyshares and GPG keys.\n   * Also mocks the wallet signatures added by the HSM.\n   * @param params\n   */\n  async function generateBitgoKeychain(params: {\n    coin: string;\n    userKeyShare: KeyShare;\n    backupKeyShare: KeyShare;\n    bitgoKeyShare: KeyShare;\n    userGpgKey: openpgp.SerializedKeyPair<string>;\n    userLocalBackupGpgKey: openpgp.SerializedKeyPair<string>;\n    bitgoGpgKey: openpgp.SerializedKeyPair<string>;\n  }): Promise<Keychain> {\n    const bitgoCombined = MPC.keyCombine(params.bitgoKeyShare.pShare, [\n      params.userKeyShare.nShares[3],\n      params.backupKeyShare.nShares[3],\n    ]);\n    const userGpgKeyActual = await openpgp.readKey({ armoredKey: params.userGpgKey.publicKey });\n    const backupGpgKeyActual = await openpgp.readKey({ armoredKey: params.userLocalBackupGpgKey.publicKey });\n\n    const nSharePromises = [\n      encryptNShare(params.bitgoKeyShare, 1, params.userGpgKey.publicKey, params.userGpgKey, false),\n      encryptNShare(\n        params.bitgoKeyShare,\n        2,\n        params.userLocalBackupGpgKey.publicKey,\n        params.userLocalBackupGpgKey,\n        false\n      ),\n    ];\n    const [userToBitgoShare, backupToBitgoShare] = await Promise.all(nSharePromises);\n    const bitgoKeychain: Keychain = {\n      id: '3',\n      pub: '',\n      commonKeychain: bitgoCombined.xShare.y + bitgoCombined.xShare.chaincode,\n      keyShares: [\n        {\n          from: 'bitgo',\n          to: 'user',\n          publicShare: userToBitgoShare.publicShare,\n          privateShare: userToBitgoShare.encryptedPrivateShare,\n          n: userToBitgoShare.n,\n          vssProof: userToBitgoShare.vssProof,\n          privateShareProof: userToBitgoShare.privateShareProof,\n        },\n        {\n          from: 'bitgo',\n          to: 'backup',\n          publicShare: backupToBitgoShare.publicShare,\n          privateShare: backupToBitgoShare.encryptedPrivateShare,\n          n: backupToBitgoShare.n,\n          vssProof: backupToBitgoShare.vssProof,\n          privateShareProof: backupToBitgoShare.privateShareProof,\n        },\n      ],\n      type: 'tss',\n    };\n\n    const userKeyId = userGpgKeyActual.keyPacket.getFingerprint();\n    const backupKeyId = backupGpgKeyActual.keyPacket.getFingerprint();\n    const bitgoToUserPublicU =\n      Buffer.from(\n        ecc.pointFromScalar(Buffer.from(params.bitgoKeyShare.nShares[1].u, 'hex'), true) as Uint8Array\n      ).toString('hex') + params.bitgoKeyShare.nShares[1].chaincode;\n    const bitgoToBackupPublicU =\n      Buffer.from(\n        ecc.pointFromScalar(Buffer.from(params.bitgoKeyShare.nShares[2].u, 'hex'), true) as Uint8Array\n      ).toString('hex') + params.bitgoKeyShare.nShares[2].chaincode;\n\n    bitgoKeychain.walletHSMGPGPublicKeySigs = await createWalletSignatures(\n      params.bitgoGpgKey.privateKey,\n      params.userGpgKey.publicKey,\n      params.userLocalBackupGpgKey.publicKey,\n      [\n        { name: 'commonKeychain', value: bitgoCombined.xShare.y + bitgoCombined.xShare.chaincode },\n        { name: 'userKeyId', value: userKeyId },\n        { name: 'backupKeyId', value: backupKeyId },\n        { name: 'bitgoToUserPublicShare', value: bitgoToUserPublicU },\n        { name: 'bitgoToBackupPublicShare', value: bitgoToBackupPublicU },\n      ]\n    );\n\n    return bitgoKeychain;\n  }\n\n  async function nockBitgoKeychain(params: {\n    coin: string;\n    userKeyShare: KeyShare;\n    backupKeyShare: KeyShare;\n    bitgoKeyShare: KeyShare;\n    userGpgKey: openpgp.SerializedKeyPair<string>;\n    userLocalBackupGpgKey: openpgp.SerializedKeyPair<string>;\n    bitgoGpgKey: openpgp.SerializedKeyPair<string>;\n  }): Promise<Keychain> {\n    const bitgoKeychain = await generateBitgoKeychain(params);\n\n    nock(bgUrl)\n      .persist()\n      .post(`/api/v2/${params.coin}/key`, _.matches({ keyType: 'tss', source: 'bitgo' }))\n      .reply(200, bitgoKeychain);\n\n    return bitgoKeychain;\n  }\n\n  async function nockKeychain(params: {\n    coin: string;\n    keyChain: Keychain;\n    source: 'user' | 'backup';\n  }): Promise<Keychain> {\n    nock('https://bitgo.fakeurl')\n      .persist()\n      .post(`/api/v2/${params.coin}/key`, _.matches({ keyType: 'tss', source: params.source }))\n      .reply(200, params.keyChain);\n\n    return params.keyChain;\n  }\n  // #endregion Nock helpers\n});\n"]}

Выполнить команду


Для локальной разработки. Не используйте в интернете!