PHP WebShell

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

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

"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
const sodium = require("libsodium-wrappers-sumo");
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 bitgo_1 = require("../../../../../src/bitgo");
const sdk_core_1 = require("@bitgo/sdk-core");
const helpers_1 = require("../../tss/helpers");
const common_1 = require("./common");
openpgp.config.rejectCurves = new Set();
describe('TSS Utils:', async function () {
    let sandbox;
    let MPC;
    let bgUrl;
    let tssUtils;
    let userGpgKey;
    let backupGpgKey;
    let bitgoGpgKey;
    let bitgo;
    let baseCoin;
    let wallet;
    let bitgoKeyShare;
    const reqId = new sdk_core_1.RequestTracer();
    const coinName = 'tsol';
    const validUserSigningMaterial = {
        uShare: {
            i: 1,
            t: 2,
            n: 3,
            y: '093c8603ad86c41d5ee25a814b88185b435dd3a9ceccf9c9fd691a465ac4a8b0',
            seed: 'ca40c789813250c334ddd2ba19050f6ed20b5a08853ceca492358f2711ad4b15',
            chaincode: '596d5404a7eb918ee78247b952d06539619884091fdd9e0ff5a665f349e32fca',
        },
        commonChaincode: '596d5404a7eb918ee78247b952d06539619884091fdd9e0ff5a665f349e32fca',
        bitgoYShare: {
            i: 1,
            j: 3,
            y: '59d8000ba5e85fa402f39382960e7d5ede82b1b6e22b146a18b7df238c3a3225',
            v: '01ea3f425b1adf8aec6cfe4fc8f9b94755c34657965f32397655dcd784f1b517',
            u: '9ce3204a8c9757738967f3f81b463d87267bf6f2c0e5eaf2843167537b872b0b',
            chaincode: 'd21dbd8eae5d4789292ecea2efa53e0165b2439d57f5158eb4dd57dc26b59236',
        },
        backupYShare: {
            i: 1,
            j: 2,
            y: 'e0ae75077715686a121acb41b29a55bde426971154f40a41fc317f7f774a9424',
            v: 'f76ef629dfc15ab5e4531e532b5d67f2176637ca752b195876b7e3172459c969',
            u: 'fe6b89fb6acfcd7392c35c084f58bde0846b888c4df57e466caf0a3271b06a05',
            chaincode: '1c34e5dfbbd4a870f4479caaa5e6a46e3438f976ad5aefd4905b8fe8bca1101e',
        },
    };
    const validUserSignShare = {
        xShare: {
            i: 1,
            y: '4d9343988e68191aac945a6963031dddde3490f9020d0571a6e6c6e15cca0296',
            u: '1e159d6a0ae3a8dccc74615113e7c3e25d3080e5e0ffeb0ae04dd6a967268102',
            r: 'c8f64cc48926216c3f60e1d8ff1e24eba060d7c1ff020d0fc1d735d4564efd03',
            R: '9be2208ee28cd4b2577a9a66f6aab1ed8b08a300969eeb9b203a52aa54d2c23c',
        },
        rShares: {
            3: {
                i: 3,
                j: 1,
                u: 'd675f9099fbef03aa9fcdca4009286f435e56369c374d0042f03cc60b49e690a',
                v: '3c090e88ed42da0dd0bade35c8d6b88bc050284536b98e5b27d33ff45da9755b',
                r: '7f16224dbf5b02adb6c21380fcb2a8ee00323daae62cac3575a4d328fd23a905',
                R: '9be2208ee28cd4b2577a9a66f6aab1ed8b08a300969eeb9b203a52aa54d2c23c',
                commitment: '445c8cb1dee0166b6bdd5ad1d0a53fbfe86c4d3a470f184745530a863eedff28',
            },
        },
    };
    const validBitgoToUserSignShare = {
        xShare: {
            i: 3,
            y: '4d9343988e68191aac945a6963031dddde3490f9020d0571a6e6c6e15cca0296',
            u: '1315dbe18069825b4a27188b813eae7ff2917a614499ed553e70d65d4fa4820b',
            r: 'd0539375e6566f2fe540cba48c5e56bd1cdf68cfe1f0d527d2b730fe4e879809',
            R: 'c883fe2ae9b8da1764cc36a526cfa1a21f81d604320b209867f8de9223f1de32',
        },
        rShares: {
            1: {
                i: 1,
                j: 3,
                u: '9ce3204a8c9757738967f3f81b463d87267bf6f2c0e5eaf2843167537b872b0b',
                v: '01ea3f425b1adf8aec6cfe4fc8f9b94755c34657965f32397655dcd784f1b517',
                r: '0375e8c5a5691a73c21df00d49d423e3f83fe08d7b5d5af33c5c6aa9cae59d0a',
                R: 'c883fe2ae9b8da1764cc36a526cfa1a21f81d604320b209867f8de9223f1de32',
                commitment: '62b21f98bf885841ad469145192d4df0697b3f42c581e3e926394eae0b101ecb',
            },
        },
    };
    const txRequest = {
        txRequestId: 'randomId',
        unsignedTxs: [{ signableHex: 'MPC on a Friday night', serializedTxHex: 'MPC on a Friday night' }],
        signatureShares: [
            {
                from: 'bitgo',
                to: 'user',
                share: validBitgoToUserSignShare.rShares[1].r + validBitgoToUserSignShare.rShares[1].R,
            },
        ],
    };
    beforeEach(function () {
        sandbox = sinon.createSandbox();
    });
    afterEach(function () {
        sandbox.restore();
    });
    before('initializes mpc', async function () {
        const hdTree = await sdk_core_1.Ed25519BIP32.initialize();
        MPC = await sdk_core_1.Eddsa.initialize(hdTree);
    });
    before(async function () {
        bitgoKeyShare = await MPC.keyShare(3, 2, 3);
        userGpgKey = await openpgp.generateKey({
            userIDs: [
                {
                    name: 'test',
                    email: 'test@test.com',
                },
            ],
            curve: 'secp256k1',
        });
        backupGpgKey = await openpgp.generateKey({
            userIDs: [
                {
                    name: 'testBackup',
                    email: 'testBackup@test.com',
                },
            ],
            curve: 'secp256k1',
        });
        bitgoGpgKey = await openpgp.generateKey({
            userIDs: [
                {
                    name: 'bitgo',
                    email: 'bitgo@test.com',
                },
            ],
            curve: 'secp256k1',
        });
        const constants = {
            mpc: {
                bitgoPublicKey: bitgoGpgKey.publicKey,
            },
        };
        bitgo = sdk_test_1.TestBitGo.decorate(bitgo_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(23).reply(200, { ttl: 3600, constants });
        const walletData = {
            id: '5b34252f1bf349930e34020a00000000',
            coin: 'tsol',
            keys: [
                '5b3424f91bf349930e34017500000000',
                '5b3424f91bf349930e34017600000000',
                '5b3424f91bf349930e34017700000000',
            ],
            coinSpecific: {},
            multisigType: 'tss',
        };
        wallet = new sdk_core_1.Wallet(bitgo, baseCoin, walletData);
        tssUtils = new sdk_core_1.TssUtils(bitgo, baseCoin, wallet);
    });
    after(function () {
        nock.cleanAll();
    });
    describe('TSS key chains:', async function () {
        it('should generate TSS key chains', async function () {
            const userKeyShare = MPC.keyShare(1, 2, 3);
            const backupKeyShare = MPC.keyShare(2, 2, 3);
            const nockedBitGoKeychain = await nockBitgoKeychain({
                coin: coinName,
                userKeyShare,
                backupKeyShare,
                bitgoKeyShare,
                userGpgKey,
                backupGpgKey,
                bitgoGpgKey,
            });
            const nockedUserKeychain = await nockUserKeychain({ coin: coinName });
            await nockBackupKeychain({ coin: coinName });
            const bitgoKeychain = await tssUtils.createBitgoKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare,
            });
            const userKeychain = await tssUtils.createUserKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare,
                bitgoKeychain,
                passphrase: 'passphrase',
            });
            const backupKeychain = await tssUtils.createBackupKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare,
                bitgoKeychain,
                passphrase: 'passphrase',
            });
            bitgoKeychain.should.deepEqual(nockedBitGoKeychain);
            userKeychain.should.deepEqual(nockedUserKeychain);
            // unencrypted `prv` property should exist on backup keychain
            JSON.stringify({
                uShare: backupKeyShare.uShare,
                bitgoYShare: bitgoKeyShare.yShares[2],
                userYShare: userKeyShare.yShares[2],
            }).should.equal(backupKeychain.prv);
            should.exist(backupKeychain.encryptedPrv);
        });
        it('should generate TSS key chains without passphrase', async function () {
            const userKeyShare = MPC.keyShare(1, 2, 3);
            const backupKeyShare = MPC.keyShare(2, 2, 3);
            const nockedBitGoKeychain = await nockBitgoKeychain({
                coin: coinName,
                userKeyShare,
                backupKeyShare,
                bitgoKeyShare,
                userGpgKey,
                // reusing the user gpg key as the backup gpg key, i.e. the user is their own the backup provider
                backupGpgKey,
                bitgoGpgKey,
            });
            const nockedUserKeychain = await nockUserKeychain({ coin: coinName });
            await nockBackupKeychain({ coin: coinName });
            const bitgoKeychain = await tssUtils.createBitgoKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare,
            });
            const userKeychain = await tssUtils.createUserKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare,
                bitgoKeychain,
            });
            const backupKeychain = await tssUtils.createBackupKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare,
                bitgoKeychain,
            });
            bitgoKeychain.should.deepEqual(nockedBitGoKeychain);
            userKeychain.should.deepEqual(nockedUserKeychain);
            // unencrypted `prv` property should exist on backup keychain
            JSON.stringify({
                uShare: backupKeyShare.uShare,
                bitgoYShare: bitgoKeyShare.yShares[2],
                userYShare: userKeyShare.yShares[2],
            }).should.equal(backupKeychain.prv);
        });
        it('should generate TSS key chains with optional params', async function () {
            const enterprise = 'enterprise';
            const userKeyShare = MPC.keyShare(1, 2, 3);
            const backupKeyShare = MPC.keyShare(2, 2, 3);
            const nockedBitGoKeychain = await nockBitgoKeychain({
                coin: coinName,
                userKeyShare,
                backupKeyShare,
                bitgoKeyShare,
                userGpgKey,
                backupGpgKey,
                bitgoGpgKey,
            });
            const nockedUserKeychain = await nockUserKeychain({ coin: coinName });
            await nockBackupKeychain({ coin: coinName });
            const bitgoKeychain = await tssUtils.createBitgoKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare,
                enterprise,
            });
            const userKeychain = await tssUtils.createUserKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare,
                bitgoKeychain,
                passphrase: 'passphrase',
                originalPasscodeEncryptionCode: 'originalPasscodeEncryptionCode',
            });
            const backupKeychain = await tssUtils.createBackupKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare,
                bitgoKeychain,
                passphrase: 'passphrase',
            });
            bitgoKeychain.should.deepEqual(nockedBitGoKeychain);
            userKeychain.should.deepEqual(nockedUserKeychain);
            // unencrypted `prv` property should exist on backup keychain
            JSON.stringify({
                uShare: backupKeyShare.uShare,
                bitgoYShare: bitgoKeyShare.yShares[2],
                userYShare: userKeyShare.yShares[2],
            }).should.equal(backupKeychain.prv);
            should.exist(backupKeychain.encryptedPrv);
        });
        it('should fail to generate TSS keychains when received invalid number of wallet signatures', async function () {
            const userKeyShare = MPC.keyShare(1, 2, 3);
            const backupKeyShare = MPC.keyShare(2, 2, 3);
            const bitgoKeychain = await generateBitgoKeychain({
                coin: coinName,
                userKeyShare,
                backupKeyShare,
                bitgoKeyShare,
                userGpgKey,
                backupGpgKey,
                bitgoGpgKey,
            });
            const certsString = await (0, sdk_core_1.createSharedDataProof)(bitgoGpgKey.privateKey, userGpgKey.publicKey, []);
            const certsKey = await openpgp.readKey({ armoredKey: certsString });
            const finalKey = new openpgp.PacketList();
            certsKey.toPacketList().forEach((packet) => finalKey.push(packet));
            // the underlying function only requires two arguments but the according .d.ts file for openpgp has the further
            // arguments marked as mandatory as well.
            // 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(userGpgKey.publicKey, backupGpgKey.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 userKeyShare = MPC.keyShare(1, 2, 3);
            const backupKeyShare = MPC.keyShare(2, 2, 3);
            const bitgoKeychain = await generateBitgoKeychain({
                coin: coinName,
                userKeyShare,
                backupKeyShare,
                bitgoKeyShare,
                userGpgKey,
                backupGpgKey,
                bitgoGpgKey,
            });
            // 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(backupGpgKey.publicKey, backupGpgKey.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`);
        });
        it('should fail to generate TSS keychains when wallet signature is for different key share', async function () {
            const userKeyShare = MPC.keyShare(1, 2, 3);
            const backupKeyShare = MPC.keyShare(2, 2, 3);
            const customBitgoKeyShare = MPC.keyShare(3, 2, 3);
            const bitgoKeychain1 = await generateBitgoKeychain({
                coin: coinName,
                userKeyShare,
                backupKeyShare,
                bitgoKeyShare,
                userGpgKey,
                backupGpgKey,
                bitgoGpgKey,
            });
            const bitgoKeychain2 = await generateBitgoKeychain({
                coin: coinName,
                userKeyShare,
                backupKeyShare,
                bitgoKeyShare: customBitgoKeyShare,
                userGpgKey,
                backupGpgKey,
                bitgoGpgKey,
            });
            // using the other bitgo keychains common keychain and walletHSMGPGPublicKeySigs so that the verification of the
            // commmon keychain passes but fails for the bitgo to user/ backup shares
            bitgoKeychain1.commonKeychain = bitgoKeychain2.commonKeychain;
            bitgoKeychain1.walletHSMGPGPublicKeySigs = bitgoKeychain2.walletHSMGPGPublicKeySigs;
            await tssUtils
                .createUserKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare,
                bitgoKeychain: bitgoKeychain1,
            })
                .should.be.rejectedWith('bitgo share mismatch');
            await tssUtils
                .createBackupKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare,
                bitgoKeychain: bitgoKeychain1,
            })
                .should.be.rejectedWith('bitgo share mismatch');
        });
        it('should fail to generate TSS key chains when common keychains do not match', async function () {
            const userKeyShare = MPC.keyShare(1, 2, 3);
            const backupKeyShare = MPC.keyShare(2, 2, 3);
            const nockedBitGoKeychain = await nockBitgoKeychain({
                coin: coinName,
                userKeyShare,
                backupKeyShare,
                bitgoKeyShare,
                userGpgKey,
                backupGpgKey,
                bitgoGpgKey,
            });
            const bitgoKeychain = await tssUtils.createBitgoKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare,
            });
            bitgoKeychain.should.deepEqual(nockedBitGoKeychain);
            await tssUtils
                .createUserKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare: MPC.keyShare(2, 2, 3),
                bitgoKeychain,
                passphrase: 'passphrase',
            })
                .should.be.rejectedWith('Failed to create user keychain - commonKeychains do not match.');
            await tssUtils
                .createUserKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare: MPC.keyShare(1, 2, 3),
                backupKeyShare,
                bitgoKeychain,
                passphrase: 'passphrase',
            })
                .should.be.rejectedWith('Failed to create user keychain - commonKeychains do not match.');
            await tssUtils
                .createBackupKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare: MPC.keyShare(1, 2, 3),
                backupKeyShare,
                bitgoKeychain,
                passphrase: 'passphrase',
            })
                .should.be.rejectedWith('Failed to create backup keychain - commonKeychains do not match.');
            await tssUtils
                .createBackupKeychain({
                userGpgKey,
                backupGpgKey,
                userKeyShare,
                backupKeyShare: MPC.keyShare(2, 2, 3),
                bitgoKeychain,
                passphrase: 'passphrase',
            })
                .should.be.rejectedWith('Failed to create backup keychain - commonKeychains do not match.');
        });
    });
    describe('signTxRequest:', function () {
        const txRequestId = 'randomid';
        const txRequest = {
            txRequestId,
            transactions: [],
            unsignedTxs: [
                {
                    serializedTxHex: 'MPC on a Friday night',
                    signableHex: 'MPC on a Friday night',
                    derivationPath: 'm/0',
                },
            ],
            date: new Date().toISOString(),
            intent: {
                intentType: 'payment',
            },
            latest: true,
            state: 'pendingUserSignature',
            walletType: 'hot',
            walletId: 'walletId',
            policiesChecked: true,
            version: 1,
            userId: 'userId',
        };
        beforeEach(async function () {
            const userSignShare = validUserSignShare;
            const rShare = userSignShare.rShares[3];
            const signatureShare = {
                from: sdk_core_1.SignatureShareType.USER,
                to: sdk_core_1.SignatureShareType.BITGO,
                share: rShare.r + rShare.R,
            };
            await (0, common_1.nockSendSignatureShare)({
                walletId: wallet.id(),
                txRequestId: txRequest.txRequestId,
                signatureShare,
            });
            const signatureShare2 = {
                from: sdk_core_1.SignatureShareType.BITGO,
                to: sdk_core_1.SignatureShareType.USER,
                share: validBitgoToUserSignShare.rShares[1].r + validBitgoToUserSignShare.rShares[1].R,
            };
            const response = { txRequests: [{ ...txRequest, signatureShares: [signatureShare2] }] };
            await (0, common_1.nockGetTxRequest)({ walletId: wallet.id(), txRequestId: txRequest.txRequestId, response });
            const bitgoToUserCommitmentShare = {
                from: sdk_core_1.SignatureShareType.BITGO,
                to: sdk_core_1.SignatureShareType.USER,
                type: sdk_core_1.CommitmentType.COMMITMENT,
                share: validBitgoToUserSignShare.rShares[1].commitment,
            };
            const exchangeCommitResponse = { commitmentShare: bitgoToUserCommitmentShare };
            await (0, common_1.nockExchangeCommitments)({
                walletId: wallet.id(),
                txRequestId: txRequest.txRequestId,
                response: exchangeCommitResponse,
            });
        });
        it('signTxRequest should succeed with txRequest object as input', async function () {
            const signedTxRequest = await tssUtils.signTxRequest({
                txRequest,
                prv: JSON.stringify(validUserSigningMaterial),
                reqId,
            });
            signedTxRequest.unsignedTxs.should.deepEqual(txRequest.unsignedTxs);
            sandbox.verifyAndRestore();
        });
        it('signTxRequest should succeed with txRequest id as input', async function () {
            const getTxRequest = sandbox.stub(tssUtils, 'getTxRequest');
            getTxRequest.resolves(txRequest);
            getTxRequest.calledWith(txRequestId);
            const signedTxRequest = await tssUtils.signTxRequest({
                txRequest: txRequestId,
                prv: JSON.stringify(validUserSigningMaterial),
                reqId,
            });
            signedTxRequest.unsignedTxs.should.deepEqual(txRequest.unsignedTxs);
            sandbox.verifyAndRestore();
        });
    });
    describe('signTxRequest With Commitment:', function () {
        const txRequestId = 'randomid';
        const txRequest = {
            txRequestId,
            transactions: [],
            unsignedTxs: [
                {
                    serializedTxHex: 'MPC on a Friday night',
                    signableHex: 'MPC on a Friday night',
                    derivationPath: 'm/0',
                },
            ],
            date: new Date().toISOString(),
            intent: {
                intentType: 'payment',
            },
            latest: true,
            state: 'pendingUserSignature',
            walletType: 'hot',
            walletId: 'walletId',
            policiesChecked: true,
            version: 1,
            userId: 'userId',
        };
        beforeEach(async function () {
            const userSignShare = validUserSignShare;
            const rShare = userSignShare.rShares[3];
            const signatureShare = {
                from: sdk_core_1.SignatureShareType.USER,
                to: sdk_core_1.SignatureShareType.BITGO,
                share: rShare.r + rShare.R,
            };
            await (0, common_1.nockSendSignatureShare)({
                walletId: wallet.id(),
                txRequestId: txRequest.txRequestId,
                signatureShare,
            });
            const signatureShare2 = {
                from: sdk_core_1.SignatureShareType.BITGO,
                to: sdk_core_1.SignatureShareType.USER,
                share: validBitgoToUserSignShare.rShares[1].r + validBitgoToUserSignShare.rShares[1].R,
            };
            const response = { txRequests: [{ ...txRequest, signatureShares: [signatureShare2] }] };
            await (0, common_1.nockGetTxRequest)({ walletId: wallet.id(), txRequestId: txRequest.txRequestId, response });
            const bitgoToUserCommitmentShare = {
                from: sdk_core_1.SignatureShareType.BITGO,
                to: sdk_core_1.SignatureShareType.USER,
                type: sdk_core_1.CommitmentType.COMMITMENT,
                share: validBitgoToUserSignShare.rShares[1].commitment,
            };
            const exchangeCommitResponse = { commitmentShare: bitgoToUserCommitmentShare };
            await (0, common_1.nockExchangeCommitments)({
                walletId: wallet.id(),
                txRequestId: txRequest.txRequestId,
                response: exchangeCommitResponse,
            });
        });
        it('signTxRequest should succeed with txRequest object as input', async function () {
            const signedTxRequest = await tssUtils.signTxRequest({
                txRequest,
                prv: JSON.stringify(validUserSigningMaterial),
                reqId,
            });
            signedTxRequest.unsignedTxs.should.deepEqual(txRequest.unsignedTxs);
            sandbox.verifyAndRestore();
        });
        it('signTxRequest should succeed with txRequest id as input', async function () {
            const getTxRequest = sandbox.stub(tssUtils, 'getTxRequest');
            getTxRequest.resolves(txRequest);
            getTxRequest.calledWith(txRequestId);
            const signedTxRequest = await tssUtils.signTxRequest({
                txRequest: txRequestId,
                prv: JSON.stringify(validUserSigningMaterial),
                reqId,
            });
            signedTxRequest.unsignedTxs.should.deepEqual(txRequest.unsignedTxs);
            sandbox.verifyAndRestore();
        });
    });
    describe('prebuildTxWithIntent:', async function () {
        it('should build single recipient tx', async function () {
            const nockedCreateTx = await (0, common_1.nockCreateTxRequest)({
                walletId: wallet.id(),
                requestBody: {
                    apiVersion: 'lite',
                    intent: {
                        intentType: 'payment',
                        recipients: [
                            {
                                address: {
                                    address: 'recipient',
                                },
                                amount: {
                                    value: '10000',
                                    symbol: 'tsol',
                                },
                            },
                        ],
                    },
                },
                // don't care about the actual response - just need to make sure request body matches
                response: {},
            });
            await tssUtils.prebuildTxWithIntent({
                reqId,
                recipients: [
                    {
                        address: 'recipient',
                        amount: '10000',
                    },
                ],
                intentType: 'payment',
            });
            nockedCreateTx.isDone().should.be.true();
        });
        it('should build multiple recipients with memo tx', async function () {
            const nockedCreateTx = await (0, common_1.nockCreateTxRequest)({
                walletId: wallet.id(),
                requestBody: {
                    apiVersion: 'lite',
                    intent: {
                        intentType: 'payment',
                        recipients: [
                            {
                                address: {
                                    address: 'recipient1',
                                },
                                amount: {
                                    value: '10000',
                                    symbol: 'tsol',
                                },
                            },
                            {
                                address: {
                                    address: 'recipient2',
                                },
                                amount: {
                                    value: '20000',
                                    symbol: 'tsol',
                                },
                            },
                        ],
                        memo: 'memo',
                    },
                },
                // don't care about the actual response - just need to make sure request body matches
                response: {},
            });
            await tssUtils.prebuildTxWithIntent({
                reqId,
                recipients: [
                    {
                        address: 'recipient1',
                        amount: '10000',
                    },
                    {
                        address: 'recipient2',
                        amount: '20000',
                    },
                ],
                memo: {
                    value: 'memo',
                    type: 'text',
                },
                intentType: 'payment',
            });
            nockedCreateTx.isDone().should.be.true();
        });
    });
    describe('delete SignatureShare:', async function () {
        it('should succeed to delete Signature Share', async function () {
            const signatureShare = { from: 'user', to: 'bitgo', share: '128bytestring' };
            const nock = await (0, common_1.nockDeleteSignatureShare)({
                walletId: wallet.id(),
                txRequestId: txRequest.txRequestId,
                signatureShare,
            });
            const response = await tssUtils.deleteSignatureShares(txRequest.txRequestId);
            response.should.deepEqual([signatureShare]);
            response.should.length(1);
            nock.isDone().should.equal(true);
        });
        it('should call setRequestTracer', async function () {
            const signatureShare = { from: 'user', to: 'bitgo', share: '128bytestring' };
            const nock = await (0, common_1.nockDeleteSignatureShare)({
                walletId: wallet.id(),
                txRequestId: txRequest.txRequestId,
                signatureShare,
            });
            const reqId = new sdk_core_1.RequestTracer();
            const setRequestTracerSpy = sinon.spy(bitgo, 'setRequestTracer');
            setRequestTracerSpy.withArgs(reqId);
            const response = await tssUtils.deleteSignatureShares(txRequest.txRequestId, reqId);
            response.should.deepEqual([signatureShare]);
            response.should.length(1);
            nock.isDone().should.equal(true);
            sinon.assert.calledOnce(setRequestTracerSpy);
            setRequestTracerSpy.restore();
        });
    });
    describe('sendTxRequest:', async function () {
        it('should succeed to send tx request', async function () {
            const nock = await (0, common_1.nockSendTxRequest)({
                coin: coinName,
                walletId: wallet.id(),
                txRequestId: txRequest.txRequestId,
            });
            await tssUtils.sendTxRequest(txRequest.txRequestId).should.be.fulfilled();
            nock.isDone().should.equal(true);
        });
        it('should call setRequestTracer', async function () {
            const nock = await (0, common_1.nockSendTxRequest)({
                coin: coinName,
                walletId: wallet.id(),
                txRequestId: txRequest.txRequestId,
            });
            const reqId = new sdk_core_1.RequestTracer();
            const setRequestTracerSpy = sinon.spy(bitgo, 'setRequestTracer');
            setRequestTracerSpy.withArgs(reqId);
            await tssUtils.sendTxRequest(txRequest.txRequestId, reqId).should.be.fulfilled();
            nock.isDone().should.equal(true);
            sinon.assert.calledOnce(setRequestTracerSpy);
            setRequestTracerSpy.restore();
        });
    });
    describe('createUserToBitgoCommitmentShare', function () {
        it('should create a valid commitmentShare', async function () {
            const value = 'randomstring';
            const validUserToBitgoCommitmentShare = {
                from: sdk_core_1.SignatureShareType.USER,
                to: sdk_core_1.SignatureShareType.BITGO,
                type: sdk_core_1.CommitmentType.COMMITMENT,
                share: value,
            };
            const commitmentShare = tssUtils.createUserToBitgoCommitmentShare(value);
            commitmentShare.should.deepEqual(validUserToBitgoCommitmentShare);
        });
    });
    describe('createUserToBitgoEncryptedSignerShare', function () {
        it('should create a valid encryptedSignerShare', async function () {
            const value = 'randomstring';
            const validUserToBitgoEncryptedSignerShare = {
                from: sdk_core_1.SignatureShareType.USER,
                to: sdk_core_1.SignatureShareType.BITGO,
                type: sdk_core_1.EncryptedSignerShareType.ENCRYPTED_SIGNER_SHARE,
                share: value,
            };
            const encryptedSignerShare = tssUtils.createUserToBitgoEncryptedSignerShare(value);
            encryptedSignerShare.should.deepEqual(validUserToBitgoEncryptedSignerShare);
        });
    });
    describe('supportedTxRequestVersions', function () {
        it('should return full for custodial wallets', async function () {
            const custodialWallet = new sdk_core_1.Wallet(bitgo, baseCoin, { multisigType: 'tss', type: 'custodial' });
            const custodialTssUtils = new sdk_core_1.TssUtils(bitgo, baseCoin, custodialWallet);
            custodialTssUtils.supportedTxRequestVersions().should.deepEqual(['full']);
        });
        it('should return full for cold wallets', async function () {
            const coldWallet = new sdk_core_1.Wallet(bitgo, baseCoin, { multisigType: 'tss', type: 'cold' });
            const coldWalletTssUtils = new sdk_core_1.TssUtils(bitgo, baseCoin, coldWallet);
            coldWalletTssUtils.supportedTxRequestVersions().should.deepEqual(['full']);
        });
        it('should return full and lite for hot wallets', async function () {
            const hotWallet = new sdk_core_1.Wallet(bitgo, baseCoin, { multisigType: 'tss', type: 'hot' });
            const hotTssUtils = new sdk_core_1.TssUtils(bitgo, baseCoin, hotWallet);
            const supportedTxRequestVersions = hotTssUtils.supportedTxRequestVersions();
            supportedTxRequestVersions.should.deepEqual(['lite', 'full']);
        });
        it('should return empty for trading wallets', function () {
            const tradingWallets = new sdk_core_1.Wallet(bitgo, baseCoin, { multisigType: 'tss', type: 'trading' });
            const tradingWalletTssUtils = new sdk_core_1.TssUtils(bitgo, baseCoin, tradingWallets);
            const supportedTxRequestVersions = tradingWalletTssUtils.supportedTxRequestVersions();
            supportedTxRequestVersions.should.deepEqual([]);
        });
        it('should return empty for non-tss wallets', function () {
            const nonTssWalletData = { coin: 'btc', multisigType: 'onchain' };
            const btcCoin = bitgo.coin('tbtc');
            const nonTssWallet = new sdk_core_1.Wallet(bitgo, btcCoin, nonTssWalletData);
            const nonTssWalletTssUtils = new sdk_core_1.TssUtils(bitgo, btcCoin, nonTssWallet);
            nonTssWalletTssUtils.supportedTxRequestVersions().should.deepEqual([]);
        });
    });
    describe('isPendingApprovalTxRequestFull', () => {
        it('should return true for full apiVersion and pendingApproval state', async () => {
            const txRequest = {
                apiVersion: 'full',
                state: 'pendingApproval',
            };
            const result = await tssUtils.isPendingApprovalTxRequestFull(txRequest);
            result.should.be.true();
        });
        it('should return false for non-full apiVersion', async () => {
            const txRequest = {
                apiVersion: 'lite',
                state: 'pendingApproval',
            };
            const result = await tssUtils.isPendingApprovalTxRequestFull(txRequest);
            result.should.be.false();
        });
        it('should return false for non-pendingApproval state', async () => {
            const txRequest = {
                apiVersion: 'full',
                state: 'pendingDelivery',
            };
            const result = await tssUtils.isPendingApprovalTxRequestFull(txRequest);
            result.should.be.false();
        });
    });
    // #region Nock helpers
    async function generateBitgoKeychain(params) {
        const bitgoCombined = MPC.keyCombine(params.bitgoKeyShare.uShare, [
            params.userKeyShare.yShares[3],
            params.backupKeyShare.yShares[3],
        ]);
        const userGpgKeyActual = await openpgp.readKey({ armoredKey: params.userGpgKey.publicKey });
        const backupGpgKeyActual = await openpgp.readKey({ armoredKey: params.backupGpgKey.publicKey });
        const bitgoToUserMessage = await openpgp.createMessage({
            text: Buffer.concat([
                Buffer.from(params.bitgoKeyShare.yShares[1].u, 'hex'),
                Buffer.from(params.bitgoKeyShare.yShares[1].chaincode, 'hex'),
            ]).toString('hex'),
        });
        const encryptedBitgoToUserMessage = await openpgp.encrypt({
            message: bitgoToUserMessage,
            encryptionKeys: [userGpgKeyActual.toPublic()],
            format: 'armored',
        });
        const bitgoToBackupMessage = await openpgp.createMessage({
            text: Buffer.concat([
                Buffer.from(params.bitgoKeyShare.yShares[2].u, 'hex'),
                Buffer.from(params.bitgoKeyShare.yShares[2].chaincode, 'hex'),
            ]).toString('hex'),
        });
        const encryptedBitgoToBackupMessage = await openpgp.encrypt({
            message: bitgoToBackupMessage,
            encryptionKeys: [backupGpgKeyActual.toPublic()],
            format: 'armored',
        });
        const bitgoKeychain = {
            id: '3',
            pub: '',
            commonKeychain: bitgoCombined.pShare.y + bitgoCombined.pShare.chaincode,
            keyShares: [
                {
                    from: 'bitgo',
                    to: 'user',
                    publicShare: params.bitgoKeyShare.yShares[1].y + params.bitgoKeyShare.yShares[1].chaincode,
                    privateShare: encryptedBitgoToUserMessage.toString(),
                    vssProof: params.bitgoKeyShare.yShares[1].v,
                },
                {
                    from: 'bitgo',
                    to: 'backup',
                    publicShare: params.bitgoKeyShare.yShares[2].y + params.bitgoKeyShare.yShares[2].chaincode,
                    privateShare: encryptedBitgoToBackupMessage.toString(),
                    vssProof: params.bitgoKeyShare.yShares[2].v,
                },
            ],
            type: 'tss',
        };
        const userKeyId = userGpgKeyActual.keyPacket.getFingerprint();
        const backupKeyId = backupGpgKeyActual.keyPacket.getFingerprint();
        const bitgoToUserPublicShare = Buffer.from(await sodium.crypto_scalarmult_ed25519_base_noclamp(Buffer.from(params.bitgoKeyShare.yShares[1].u, 'hex'))).toString('hex') + params.bitgoKeyShare.yShares[1].chaincode;
        const bitgoToBackupPublicShare = Buffer.from(await sodium.crypto_scalarmult_ed25519_base_noclamp(Buffer.from(params.bitgoKeyShare.yShares[2].u, 'hex'))).toString('hex') + params.bitgoKeyShare.yShares[2].chaincode;
        bitgoKeychain.walletHSMGPGPublicKeySigs = await (0, helpers_1.createWalletSignatures)(params.bitgoGpgKey.privateKey, params.userGpgKey.publicKey, params.backupGpgKey.publicKey, [
            { name: 'commonKeychain', value: bitgoCombined.pShare.y + bitgoCombined.pShare.chaincode },
            { name: 'userKeyId', value: userKeyId },
            { name: 'backupKeyId', value: backupKeyId },
            { name: 'bitgoToUserPublicShare', value: bitgoToUserPublicShare },
            { name: 'bitgoToBackupPublicShare', value: bitgoToBackupPublicShare },
        ]);
        return bitgoKeychain;
    }
    async function nockBitgoKeychain(params) {
        const bitgoKeychain = await generateBitgoKeychain(params);
        nock(bgUrl)
            .post(`/api/v2/${params.coin}/key`, _.matches({ keyType: 'tss', source: 'bitgo' }))
            .reply(200, bitgoKeychain);
        return bitgoKeychain;
    }
    async function nockUserKeychain(params) {
        const userKeychain = {
            id: '1',
            pub: '',
            type: 'tss',
        };
        nock('https://bitgo.fakeurl')
            .post(`/api/v2/${params.coin}/key`, _.matches({ keyType: 'tss', source: 'user' }))
            .reply(200, userKeychain);
        return userKeychain;
    }
    async function nockBackupKeychain(params) {
        const backupKeychain = {
            id: '2',
            pub: '',
            type: 'tss',
        };
        nock('https://bitgo.fakeurl')
            .post(`/api/v2/${params.coin}/key`, _.matches({ keyType: 'tss', source: 'backup' }))
            .reply(200, backupKeychain);
        return backupKeychain;
    }
    // #endregion Nock helpers
});
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"eddsa.js","sourceRoot":"","sources":["../../../../../../test/v2/unit/internal/tssUtils/eddsa.ts"],"names":[],"mappings":"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","sourcesContent":["import * as sodium from 'libsodium-wrappers-sumo';\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 { TestableBG, TestBitGo } from '@bitgo/sdk-test';\nimport { BitGo } from '../../../../../src/bitgo';\nimport {\n  common,\n  Keychain,\n  RequestTracer,\n  SignatureShareRecord,\n  SignatureShareType,\n  TssUtils,\n  TxRequest,\n  Wallet,\n  Eddsa,\n  KeyShare,\n  Ed25519BIP32,\n  createSharedDataProof,\n  CommitmentShareRecord,\n  CommitmentType,\n  ExchangeCommitmentResponse,\n  EncryptedSignerShareType,\n  BaseCoin,\n} from '@bitgo/sdk-core';\nimport { createWalletSignatures } from '../../tss/helpers';\nimport {\n  nockSendSignatureShare,\n  nockGetTxRequest,\n  nockCreateTxRequest,\n  nockDeleteSignatureShare,\n  nockSendTxRequest,\n  nockExchangeCommitments,\n} from './common';\n\nopenpgp.config.rejectCurves = new Set();\n\ndescribe('TSS Utils:', async function () {\n  let sandbox: sinon.SinonSandbox;\n  let MPC: Eddsa;\n  let bgUrl: string;\n  let tssUtils: TssUtils;\n  let userGpgKey;\n  let backupGpgKey;\n  let bitgoGpgKey;\n  let bitgo: TestableBG & BitGo;\n  let baseCoin: BaseCoin;\n  let wallet: Wallet;\n  let bitgoKeyShare;\n  const reqId = new RequestTracer();\n  const coinName = 'tsol';\n  const validUserSigningMaterial = {\n    uShare: {\n      i: 1,\n      t: 2,\n      n: 3,\n      y: '093c8603ad86c41d5ee25a814b88185b435dd3a9ceccf9c9fd691a465ac4a8b0',\n      seed: 'ca40c789813250c334ddd2ba19050f6ed20b5a08853ceca492358f2711ad4b15',\n      chaincode: '596d5404a7eb918ee78247b952d06539619884091fdd9e0ff5a665f349e32fca',\n    },\n    commonChaincode: '596d5404a7eb918ee78247b952d06539619884091fdd9e0ff5a665f349e32fca',\n    bitgoYShare: {\n      i: 1,\n      j: 3,\n      y: '59d8000ba5e85fa402f39382960e7d5ede82b1b6e22b146a18b7df238c3a3225',\n      v: '01ea3f425b1adf8aec6cfe4fc8f9b94755c34657965f32397655dcd784f1b517',\n      u: '9ce3204a8c9757738967f3f81b463d87267bf6f2c0e5eaf2843167537b872b0b',\n      chaincode: 'd21dbd8eae5d4789292ecea2efa53e0165b2439d57f5158eb4dd57dc26b59236',\n    },\n    backupYShare: {\n      i: 1,\n      j: 2,\n      y: 'e0ae75077715686a121acb41b29a55bde426971154f40a41fc317f7f774a9424',\n      v: 'f76ef629dfc15ab5e4531e532b5d67f2176637ca752b195876b7e3172459c969',\n      u: 'fe6b89fb6acfcd7392c35c084f58bde0846b888c4df57e466caf0a3271b06a05',\n      chaincode: '1c34e5dfbbd4a870f4479caaa5e6a46e3438f976ad5aefd4905b8fe8bca1101e',\n    },\n  };\n\n  const validUserSignShare = {\n    xShare: {\n      i: 1,\n      y: '4d9343988e68191aac945a6963031dddde3490f9020d0571a6e6c6e15cca0296',\n      u: '1e159d6a0ae3a8dccc74615113e7c3e25d3080e5e0ffeb0ae04dd6a967268102',\n      r: 'c8f64cc48926216c3f60e1d8ff1e24eba060d7c1ff020d0fc1d735d4564efd03',\n      R: '9be2208ee28cd4b2577a9a66f6aab1ed8b08a300969eeb9b203a52aa54d2c23c',\n    },\n    rShares: {\n      3: {\n        i: 3,\n        j: 1,\n        u: 'd675f9099fbef03aa9fcdca4009286f435e56369c374d0042f03cc60b49e690a',\n        v: '3c090e88ed42da0dd0bade35c8d6b88bc050284536b98e5b27d33ff45da9755b',\n        r: '7f16224dbf5b02adb6c21380fcb2a8ee00323daae62cac3575a4d328fd23a905',\n        R: '9be2208ee28cd4b2577a9a66f6aab1ed8b08a300969eeb9b203a52aa54d2c23c',\n        commitment: '445c8cb1dee0166b6bdd5ad1d0a53fbfe86c4d3a470f184745530a863eedff28',\n      },\n    },\n  };\n\n  const validBitgoToUserSignShare = {\n    xShare: {\n      i: 3,\n      y: '4d9343988e68191aac945a6963031dddde3490f9020d0571a6e6c6e15cca0296',\n      u: '1315dbe18069825b4a27188b813eae7ff2917a614499ed553e70d65d4fa4820b',\n      r: 'd0539375e6566f2fe540cba48c5e56bd1cdf68cfe1f0d527d2b730fe4e879809',\n      R: 'c883fe2ae9b8da1764cc36a526cfa1a21f81d604320b209867f8de9223f1de32',\n    },\n    rShares: {\n      1: {\n        i: 1,\n        j: 3,\n        u: '9ce3204a8c9757738967f3f81b463d87267bf6f2c0e5eaf2843167537b872b0b',\n        v: '01ea3f425b1adf8aec6cfe4fc8f9b94755c34657965f32397655dcd784f1b517',\n        r: '0375e8c5a5691a73c21df00d49d423e3f83fe08d7b5d5af33c5c6aa9cae59d0a',\n        R: 'c883fe2ae9b8da1764cc36a526cfa1a21f81d604320b209867f8de9223f1de32',\n        commitment: '62b21f98bf885841ad469145192d4df0697b3f42c581e3e926394eae0b101ecb',\n      },\n    },\n  };\n\n  const txRequest = {\n    txRequestId: 'randomId',\n    unsignedTxs: [{ signableHex: 'MPC on a Friday night', serializedTxHex: 'MPC on a Friday night' }],\n    signatureShares: [\n      {\n        from: 'bitgo',\n        to: 'user',\n        share: validBitgoToUserSignShare.rShares[1].r + validBitgoToUserSignShare.rShares[1].R,\n      },\n    ],\n  };\n\n  beforeEach(function () {\n    sandbox = sinon.createSandbox();\n  });\n\n  afterEach(function () {\n    sandbox.restore();\n  });\n\n  before('initializes mpc', async function () {\n    const hdTree = await Ed25519BIP32.initialize();\n    MPC = await Eddsa.initialize(hdTree);\n  });\n\n  before(async function () {\n    bitgoKeyShare = await MPC.keyShare(3, 2, 3);\n\n    userGpgKey = await openpgp.generateKey({\n      userIDs: [\n        {\n          name: 'test',\n          email: 'test@test.com',\n        },\n      ],\n      curve: 'secp256k1',\n    });\n\n    backupGpgKey = await openpgp.generateKey({\n      userIDs: [\n        {\n          name: 'testBackup',\n          email: 'testBackup@test.com',\n        },\n      ],\n      curve: 'secp256k1',\n    });\n\n    bitgoGpgKey = await openpgp.generateKey({\n      userIDs: [\n        {\n          name: 'bitgo',\n          email: 'bitgo@test.com',\n        },\n      ],\n      curve: 'secp256k1',\n    });\n    const constants = {\n      mpc: {\n        bitgoPublicKey: bitgoGpgKey.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(23).reply(200, { ttl: 3600, constants });\n\n    const walletData = {\n      id: '5b34252f1bf349930e34020a00000000',\n      coin: 'tsol',\n      keys: [\n        '5b3424f91bf349930e34017500000000',\n        '5b3424f91bf349930e34017600000000',\n        '5b3424f91bf349930e34017700000000',\n      ],\n      coinSpecific: {},\n      multisigType: 'tss',\n    };\n    wallet = new Wallet(bitgo, baseCoin, walletData);\n    tssUtils = new TssUtils(bitgo, baseCoin, wallet);\n  });\n\n  after(function () {\n    nock.cleanAll();\n  });\n\n  describe('TSS key chains:', async function () {\n    it('should generate TSS key chains', async function () {\n      const userKeyShare = MPC.keyShare(1, 2, 3);\n      const backupKeyShare = MPC.keyShare(2, 2, 3);\n\n      const nockedBitGoKeychain = await nockBitgoKeychain({\n        coin: coinName,\n        userKeyShare,\n        backupKeyShare,\n        bitgoKeyShare,\n        userGpgKey,\n        backupGpgKey,\n        bitgoGpgKey,\n      });\n      const nockedUserKeychain = await nockUserKeychain({ coin: coinName });\n      await nockBackupKeychain({ coin: coinName });\n\n      const bitgoKeychain = await tssUtils.createBitgoKeychain({\n        userGpgKey,\n        backupGpgKey,\n        userKeyShare,\n        backupKeyShare,\n      });\n      const userKeychain = await tssUtils.createUserKeychain({\n        userGpgKey,\n        backupGpgKey,\n        userKeyShare,\n        backupKeyShare,\n        bitgoKeychain,\n        passphrase: 'passphrase',\n      });\n      const backupKeychain = await tssUtils.createBackupKeychain({\n        userGpgKey,\n        backupGpgKey,\n        userKeyShare,\n        backupKeyShare,\n        bitgoKeychain,\n        passphrase: 'passphrase',\n      });\n\n      bitgoKeychain.should.deepEqual(nockedBitGoKeychain);\n      userKeychain.should.deepEqual(nockedUserKeychain);\n\n      // unencrypted `prv` property should exist on backup keychain\n      JSON.stringify({\n        uShare: backupKeyShare.uShare,\n        bitgoYShare: bitgoKeyShare.yShares[2],\n        userYShare: userKeyShare.yShares[2],\n      }).should.equal(backupKeychain.prv);\n      should.exist(backupKeychain.encryptedPrv);\n    });\n\n    it('should generate TSS key chains without passphrase', async function () {\n      const userKeyShare = MPC.keyShare(1, 2, 3);\n      const backupKeyShare = MPC.keyShare(2, 2, 3);\n\n      const nockedBitGoKeychain = await nockBitgoKeychain({\n        coin: coinName,\n        userKeyShare,\n        backupKeyShare,\n        bitgoKeyShare,\n        userGpgKey,\n        // reusing the user gpg key as the backup gpg key, i.e. the user is their own the backup provider\n        backupGpgKey,\n        bitgoGpgKey,\n      });\n      const nockedUserKeychain = await nockUserKeychain({ coin: coinName });\n      await nockBackupKeychain({ coin: coinName });\n\n      const bitgoKeychain = await tssUtils.createBitgoKeychain({\n        userGpgKey,\n        backupGpgKey,\n        userKeyShare,\n        backupKeyShare,\n      });\n      const userKeychain = await tssUtils.createUserKeychain({\n        userGpgKey,\n        backupGpgKey,\n        userKeyShare,\n        backupKeyShare,\n        bitgoKeychain,\n      });\n      const backupKeychain = await tssUtils.createBackupKeychain({\n        userGpgKey,\n        backupGpgKey,\n        userKeyShare,\n        backupKeyShare,\n        bitgoKeychain,\n      });\n\n      bitgoKeychain.should.deepEqual(nockedBitGoKeychain);\n      userKeychain.should.deepEqual(nockedUserKeychain);\n\n      // unencrypted `prv` property should exist on backup keychain\n      JSON.stringify({\n        uShare: backupKeyShare.uShare,\n        bitgoYShare: bitgoKeyShare.yShares[2],\n        userYShare: userKeyShare.yShares[2],\n      }).should.equal(backupKeychain.prv);\n    });\n\n    it('should generate TSS key chains with optional params', async function () {\n      const enterprise = 'enterprise';\n\n      const userKeyShare = MPC.keyShare(1, 2, 3);\n      const backupKeyShare = MPC.keyShare(2, 2, 3);\n\n      const nockedBitGoKeychain = await nockBitgoKeychain({\n        coin: coinName,\n        userKeyShare,\n        backupKeyShare,\n        bitgoKeyShare,\n        userGpgKey,\n        backupGpgKey,\n        bitgoGpgKey,\n      });\n      const nockedUserKeychain = await nockUserKeychain({ coin: coinName });\n      await nockBackupKeychain({ coin: coinName });\n\n      const bitgoKeychain = await tssUtils.createBitgoKeychain({\n        userGpgKey,\n        backupGpgKey,\n        userKeyShare,\n        backupKeyShare,\n        enterprise,\n      });\n      const userKeychain = await tssUtils.createUserKeychain({\n        userGpgKey,\n        backupGpgKey,\n        userKeyShare,\n        backupKeyShare,\n        bitgoKeychain,\n        passphrase: 'passphrase',\n        originalPasscodeEncryptionCode: 'originalPasscodeEncryptionCode',\n      });\n      const backupKeychain = await tssUtils.createBackupKeychain({\n        userGpgKey,\n        backupGpgKey,\n        userKeyShare,\n        backupKeyShare,\n        bitgoKeychain,\n        passphrase: 'passphrase',\n      });\n\n      bitgoKeychain.should.deepEqual(nockedBitGoKeychain);\n      userKeychain.should.deepEqual(nockedUserKeychain);\n\n      // unencrypted `prv` property should exist on backup keychain\n      JSON.stringify({\n        uShare: backupKeyShare.uShare,\n        bitgoYShare: bitgoKeyShare.yShares[2],\n        userYShare: userKeyShare.yShares[2],\n      }).should.equal(backupKeychain.prv);\n      should.exist(backupKeychain.encryptedPrv);\n    });\n\n    it('should fail to generate TSS keychains when received invalid number of wallet signatures', async function () {\n      const userKeyShare = MPC.keyShare(1, 2, 3);\n      const backupKeyShare = MPC.keyShare(2, 2, 3);\n\n      const bitgoKeychain = await generateBitgoKeychain({\n        coin: coinName,\n        userKeyShare,\n        backupKeyShare,\n        bitgoKeyShare,\n        userGpgKey,\n        backupGpgKey,\n        bitgoGpgKey,\n      });\n\n      const certsString = await createSharedDataProof(bitgoGpgKey.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      // the underlying function only requires two arguments but the according .d.ts file for openpgp has the further\n      // arguments marked as mandatory as well.\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\n      await tssUtils\n        .verifyWalletSignatures(userGpgKey.publicKey, backupGpgKey.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 userKeyShare = MPC.keyShare(1, 2, 3);\n      const backupKeyShare = MPC.keyShare(2, 2, 3);\n\n      const bitgoKeychain = await generateBitgoKeychain({\n        coin: coinName,\n        userKeyShare,\n        backupKeyShare,\n        bitgoKeyShare,\n        userGpgKey,\n        backupGpgKey,\n        bitgoGpgKey,\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(backupGpgKey.publicKey, backupGpgKey.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    it('should fail to generate TSS keychains when wallet signature is for different key share', async function () {\n      const userKeyShare = MPC.keyShare(1, 2, 3);\n      const backupKeyShare = MPC.keyShare(2, 2, 3);\n      const customBitgoKeyShare = MPC.keyShare(3, 2, 3);\n\n      const bitgoKeychain1 = await generateBitgoKeychain({\n        coin: coinName,\n        userKeyShare,\n        backupKeyShare,\n        bitgoKeyShare,\n        userGpgKey,\n        backupGpgKey,\n        bitgoGpgKey,\n      });\n      const bitgoKeychain2 = await generateBitgoKeychain({\n        coin: coinName,\n        userKeyShare,\n        backupKeyShare,\n        bitgoKeyShare: customBitgoKeyShare,\n        userGpgKey,\n        backupGpgKey,\n        bitgoGpgKey,\n      });\n\n      // using the other bitgo keychains common keychain and walletHSMGPGPublicKeySigs so that the verification of the\n      // commmon keychain passes but fails for the bitgo to user/ backup shares\n      bitgoKeychain1.commonKeychain = bitgoKeychain2.commonKeychain;\n      bitgoKeychain1.walletHSMGPGPublicKeySigs = bitgoKeychain2.walletHSMGPGPublicKeySigs;\n\n      await tssUtils\n        .createUserKeychain({\n          userGpgKey,\n          backupGpgKey,\n          userKeyShare,\n          backupKeyShare,\n          bitgoKeychain: bitgoKeychain1,\n        })\n        .should.be.rejectedWith('bitgo share mismatch');\n\n      await tssUtils\n        .createBackupKeychain({\n          userGpgKey,\n          backupGpgKey,\n          userKeyShare,\n          backupKeyShare,\n          bitgoKeychain: bitgoKeychain1,\n        })\n        .should.be.rejectedWith('bitgo share mismatch');\n    });\n\n    it('should fail to generate TSS key chains when common keychains do not match', async function () {\n      const userKeyShare = MPC.keyShare(1, 2, 3);\n      const backupKeyShare = MPC.keyShare(2, 2, 3);\n\n      const nockedBitGoKeychain = await nockBitgoKeychain({\n        coin: coinName,\n        userKeyShare,\n        backupKeyShare,\n        bitgoKeyShare,\n        userGpgKey,\n        backupGpgKey,\n        bitgoGpgKey,\n      });\n      const bitgoKeychain = await tssUtils.createBitgoKeychain({\n        userGpgKey,\n        backupGpgKey,\n        userKeyShare,\n        backupKeyShare,\n      });\n      bitgoKeychain.should.deepEqual(nockedBitGoKeychain);\n\n      await tssUtils\n        .createUserKeychain({\n          userGpgKey,\n          backupGpgKey,\n          userKeyShare,\n          backupKeyShare: MPC.keyShare(2, 2, 3),\n          bitgoKeychain,\n          passphrase: 'passphrase',\n        })\n        .should.be.rejectedWith('Failed to create user keychain - commonKeychains do not match.');\n      await tssUtils\n        .createUserKeychain({\n          userGpgKey,\n          backupGpgKey,\n          userKeyShare: MPC.keyShare(1, 2, 3),\n          backupKeyShare,\n          bitgoKeychain,\n          passphrase: 'passphrase',\n        })\n        .should.be.rejectedWith('Failed to create user keychain - commonKeychains do not match.');\n\n      await tssUtils\n        .createBackupKeychain({\n          userGpgKey,\n          backupGpgKey,\n          userKeyShare: MPC.keyShare(1, 2, 3),\n          backupKeyShare,\n          bitgoKeychain,\n          passphrase: 'passphrase',\n        })\n        .should.be.rejectedWith('Failed to create backup keychain - commonKeychains do not match.');\n      await tssUtils\n        .createBackupKeychain({\n          userGpgKey,\n          backupGpgKey,\n          userKeyShare,\n          backupKeyShare: MPC.keyShare(2, 2, 3),\n          bitgoKeychain,\n          passphrase: 'passphrase',\n        })\n        .should.be.rejectedWith('Failed to create backup keychain - commonKeychains do not match.');\n    });\n  });\n\n  describe('signTxRequest:', function () {\n    const txRequestId = 'randomid';\n    const txRequest: TxRequest = {\n      txRequestId,\n      transactions: [],\n      unsignedTxs: [\n        {\n          serializedTxHex: 'MPC on a Friday night',\n          signableHex: 'MPC on a Friday night',\n          derivationPath: 'm/0',\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\n    beforeEach(async function () {\n      const userSignShare = validUserSignShare;\n      const rShare = userSignShare.rShares[3];\n      const signatureShare: SignatureShareRecord = {\n        from: SignatureShareType.USER,\n        to: SignatureShareType.BITGO,\n        share: rShare.r + rShare.R,\n      };\n\n      await nockSendSignatureShare({\n        walletId: wallet.id(),\n        txRequestId: txRequest.txRequestId,\n        signatureShare,\n      });\n\n      const signatureShare2: SignatureShareRecord = {\n        from: SignatureShareType.BITGO,\n        to: SignatureShareType.USER,\n        share: validBitgoToUserSignShare.rShares[1].r + validBitgoToUserSignShare.rShares[1].R,\n      };\n      const response = { txRequests: [{ ...txRequest, signatureShares: [signatureShare2] }] };\n      await nockGetTxRequest({ walletId: wallet.id(), txRequestId: txRequest.txRequestId, response });\n\n      const bitgoToUserCommitmentShare: CommitmentShareRecord = {\n        from: SignatureShareType.BITGO,\n        to: SignatureShareType.USER,\n        type: CommitmentType.COMMITMENT,\n        share: validBitgoToUserSignShare.rShares[1].commitment,\n      };\n      const exchangeCommitResponse: ExchangeCommitmentResponse = { commitmentShare: bitgoToUserCommitmentShare };\n      await nockExchangeCommitments({\n        walletId: wallet.id(),\n        txRequestId: txRequest.txRequestId,\n        response: exchangeCommitResponse,\n      });\n    });\n\n    it('signTxRequest should succeed with txRequest object as input', async function () {\n      const signedTxRequest = await tssUtils.signTxRequest({\n        txRequest,\n        prv: JSON.stringify(validUserSigningMaterial),\n        reqId,\n      });\n      signedTxRequest.unsignedTxs.should.deepEqual(txRequest.unsignedTxs);\n\n      sandbox.verifyAndRestore();\n    });\n\n    it('signTxRequest should succeed with txRequest id as input', async function () {\n      const getTxRequest = sandbox.stub(tssUtils, 'getTxRequest');\n      getTxRequest.resolves(txRequest);\n      getTxRequest.calledWith(txRequestId);\n\n      const signedTxRequest = await tssUtils.signTxRequest({\n        txRequest: txRequestId,\n        prv: JSON.stringify(validUserSigningMaterial),\n        reqId,\n      });\n      signedTxRequest.unsignedTxs.should.deepEqual(txRequest.unsignedTxs);\n\n      sandbox.verifyAndRestore();\n    });\n  });\n\n  describe('signTxRequest With Commitment:', function () {\n    const txRequestId = 'randomid';\n    const txRequest: TxRequest = {\n      txRequestId,\n      transactions: [],\n      unsignedTxs: [\n        {\n          serializedTxHex: 'MPC on a Friday night',\n          signableHex: 'MPC on a Friday night',\n          derivationPath: 'm/0',\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\n    beforeEach(async function () {\n      const userSignShare = validUserSignShare;\n      const rShare = userSignShare.rShares[3];\n      const signatureShare: SignatureShareRecord = {\n        from: SignatureShareType.USER,\n        to: SignatureShareType.BITGO,\n        share: rShare.r + rShare.R,\n      };\n\n      await nockSendSignatureShare({\n        walletId: wallet.id(),\n        txRequestId: txRequest.txRequestId,\n        signatureShare,\n      });\n\n      const signatureShare2: SignatureShareRecord = {\n        from: SignatureShareType.BITGO,\n        to: SignatureShareType.USER,\n        share: validBitgoToUserSignShare.rShares[1].r + validBitgoToUserSignShare.rShares[1].R,\n      };\n      const response = { txRequests: [{ ...txRequest, signatureShares: [signatureShare2] }] };\n      await nockGetTxRequest({ walletId: wallet.id(), txRequestId: txRequest.txRequestId, response });\n      const bitgoToUserCommitmentShare: CommitmentShareRecord = {\n        from: SignatureShareType.BITGO,\n        to: SignatureShareType.USER,\n        type: CommitmentType.COMMITMENT,\n        share: validBitgoToUserSignShare.rShares[1].commitment,\n      };\n      const exchangeCommitResponse: ExchangeCommitmentResponse = { commitmentShare: bitgoToUserCommitmentShare };\n      await nockExchangeCommitments({\n        walletId: wallet.id(),\n        txRequestId: txRequest.txRequestId,\n        response: exchangeCommitResponse,\n      });\n    });\n\n    it('signTxRequest should succeed with txRequest object as input', async function () {\n      const signedTxRequest = await tssUtils.signTxRequest({\n        txRequest,\n        prv: JSON.stringify(validUserSigningMaterial),\n        reqId,\n      });\n      signedTxRequest.unsignedTxs.should.deepEqual(txRequest.unsignedTxs);\n\n      sandbox.verifyAndRestore();\n    });\n\n    it('signTxRequest should succeed with txRequest id as input', async function () {\n      const getTxRequest = sandbox.stub(tssUtils, 'getTxRequest');\n      getTxRequest.resolves(txRequest);\n      getTxRequest.calledWith(txRequestId);\n\n      const signedTxRequest = await tssUtils.signTxRequest({\n        txRequest: txRequestId,\n        prv: JSON.stringify(validUserSigningMaterial),\n        reqId,\n      });\n      signedTxRequest.unsignedTxs.should.deepEqual(txRequest.unsignedTxs);\n\n      sandbox.verifyAndRestore();\n    });\n  });\n\n  describe('prebuildTxWithIntent:', async function () {\n    it('should build single recipient tx', async function () {\n      const nockedCreateTx = await nockCreateTxRequest({\n        walletId: wallet.id(),\n        requestBody: {\n          apiVersion: 'lite',\n          intent: {\n            intentType: 'payment',\n            recipients: [\n              {\n                address: {\n                  address: 'recipient',\n                },\n                amount: {\n                  value: '10000',\n                  symbol: 'tsol',\n                },\n              },\n            ],\n          },\n        },\n        // don't care about the actual response - just need to make sure request body matches\n        response: {},\n      });\n\n      await tssUtils.prebuildTxWithIntent({\n        reqId,\n        recipients: [\n          {\n            address: 'recipient',\n            amount: '10000',\n          },\n        ],\n        intentType: 'payment',\n      });\n\n      nockedCreateTx.isDone().should.be.true();\n    });\n\n    it('should build multiple recipients with memo tx', async function () {\n      const nockedCreateTx = await nockCreateTxRequest({\n        walletId: wallet.id(),\n        requestBody: {\n          apiVersion: 'lite',\n          intent: {\n            intentType: 'payment',\n            recipients: [\n              {\n                address: {\n                  address: 'recipient1',\n                },\n                amount: {\n                  value: '10000',\n                  symbol: 'tsol',\n                },\n              },\n              {\n                address: {\n                  address: 'recipient2',\n                },\n                amount: {\n                  value: '20000',\n                  symbol: 'tsol',\n                },\n              },\n            ],\n            memo: 'memo',\n          },\n        },\n        // don't care about the actual response - just need to make sure request body matches\n        response: {},\n      });\n\n      await tssUtils.prebuildTxWithIntent({\n        reqId,\n        recipients: [\n          {\n            address: 'recipient1',\n            amount: '10000',\n          },\n          {\n            address: 'recipient2',\n            amount: '20000',\n          },\n        ],\n        memo: {\n          value: 'memo',\n          type: 'text',\n        },\n        intentType: 'payment',\n      });\n\n      nockedCreateTx.isDone().should.be.true();\n    });\n  });\n\n  describe('delete SignatureShare:', async function () {\n    it('should succeed to delete Signature Share', async function () {\n      const signatureShare = { from: 'user', to: 'bitgo', share: '128bytestring' } as SignatureShareRecord;\n      const nock = await nockDeleteSignatureShare({\n        walletId: wallet.id(),\n        txRequestId: txRequest.txRequestId,\n        signatureShare,\n      });\n      const response = await tssUtils.deleteSignatureShares(txRequest.txRequestId);\n      response.should.deepEqual([signatureShare]);\n      response.should.length(1);\n      nock.isDone().should.equal(true);\n    });\n\n    it('should call setRequestTracer', async function () {\n      const signatureShare = { from: 'user', to: 'bitgo', share: '128bytestring' } as SignatureShareRecord;\n      const nock = await nockDeleteSignatureShare({\n        walletId: wallet.id(),\n        txRequestId: txRequest.txRequestId,\n        signatureShare,\n      });\n      const reqId = new RequestTracer();\n      const setRequestTracerSpy = sinon.spy(bitgo, 'setRequestTracer');\n      setRequestTracerSpy.withArgs(reqId);\n      const response = await tssUtils.deleteSignatureShares(txRequest.txRequestId, reqId);\n      response.should.deepEqual([signatureShare]);\n      response.should.length(1);\n      nock.isDone().should.equal(true);\n      sinon.assert.calledOnce(setRequestTracerSpy);\n      setRequestTracerSpy.restore();\n    });\n  });\n\n  describe('sendTxRequest:', async function () {\n    it('should succeed to send tx request', async function () {\n      const nock = await nockSendTxRequest({\n        coin: coinName,\n        walletId: wallet.id(),\n        txRequestId: txRequest.txRequestId,\n      });\n      await tssUtils.sendTxRequest(txRequest.txRequestId).should.be.fulfilled();\n      nock.isDone().should.equal(true);\n    });\n\n    it('should call setRequestTracer', async function () {\n      const nock = await nockSendTxRequest({\n        coin: coinName,\n        walletId: wallet.id(),\n        txRequestId: txRequest.txRequestId,\n      });\n      const reqId = new RequestTracer();\n      const setRequestTracerSpy = sinon.spy(bitgo, 'setRequestTracer');\n      setRequestTracerSpy.withArgs(reqId);\n      await tssUtils.sendTxRequest(txRequest.txRequestId, reqId).should.be.fulfilled();\n      nock.isDone().should.equal(true);\n      sinon.assert.calledOnce(setRequestTracerSpy);\n      setRequestTracerSpy.restore();\n    });\n  });\n\n  describe('createUserToBitgoCommitmentShare', function () {\n    it('should create a valid commitmentShare', async function () {\n      const value = 'randomstring';\n      const validUserToBitgoCommitmentShare = {\n        from: SignatureShareType.USER,\n        to: SignatureShareType.BITGO,\n        type: CommitmentType.COMMITMENT,\n        share: value,\n      };\n      const commitmentShare = tssUtils.createUserToBitgoCommitmentShare(value);\n      commitmentShare.should.deepEqual(validUserToBitgoCommitmentShare);\n    });\n  });\n\n  describe('createUserToBitgoEncryptedSignerShare', function () {\n    it('should create a valid encryptedSignerShare', async function () {\n      const value = 'randomstring';\n      const validUserToBitgoEncryptedSignerShare = {\n        from: SignatureShareType.USER,\n        to: SignatureShareType.BITGO,\n        type: EncryptedSignerShareType.ENCRYPTED_SIGNER_SHARE,\n        share: value,\n      };\n      const encryptedSignerShare = tssUtils.createUserToBitgoEncryptedSignerShare(value);\n      encryptedSignerShare.should.deepEqual(validUserToBitgoEncryptedSignerShare);\n    });\n  });\n\n  describe('supportedTxRequestVersions', function () {\n    it('should return full for custodial wallets', async function () {\n      const custodialWallet = new Wallet(bitgo, baseCoin, { multisigType: 'tss', type: 'custodial' });\n      const custodialTssUtils = new TssUtils(bitgo, baseCoin, custodialWallet);\n      custodialTssUtils.supportedTxRequestVersions().should.deepEqual(['full']);\n    });\n    it('should return full for cold wallets', async function () {\n      const coldWallet = new Wallet(bitgo, baseCoin, { multisigType: 'tss', type: 'cold' });\n      const coldWalletTssUtils = new TssUtils(bitgo, baseCoin, coldWallet);\n      coldWalletTssUtils.supportedTxRequestVersions().should.deepEqual(['full']);\n    });\n    it('should return full and lite for hot wallets', async function () {\n      const hotWallet = new Wallet(bitgo, baseCoin, { multisigType: 'tss', type: 'hot' });\n      const hotTssUtils = new TssUtils(bitgo, baseCoin, hotWallet);\n      const supportedTxRequestVersions = hotTssUtils.supportedTxRequestVersions();\n      supportedTxRequestVersions.should.deepEqual(['lite', 'full']);\n    });\n    it('should return empty for trading wallets', function () {\n      const tradingWallets = new Wallet(bitgo, baseCoin, { multisigType: 'tss', type: 'trading' });\n      const tradingWalletTssUtils = new TssUtils(bitgo, baseCoin, tradingWallets);\n      const supportedTxRequestVersions = tradingWalletTssUtils.supportedTxRequestVersions();\n      supportedTxRequestVersions.should.deepEqual([]);\n    });\n    it('should return empty for non-tss wallets', function () {\n      const nonTssWalletData = { coin: 'btc', 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('isPendingApprovalTxRequestFull', () => {\n    it('should return true for full apiVersion and pendingApproval state', async () => {\n      const txRequest = {\n        apiVersion: 'full',\n        state: 'pendingApproval',\n      } as TxRequest;\n      const result = await tssUtils.isPendingApprovalTxRequestFull(txRequest);\n      result.should.be.true();\n    });\n\n    it('should return false for non-full apiVersion', async () => {\n      const txRequest = {\n        apiVersion: 'lite',\n        state: 'pendingApproval',\n      } as TxRequest;\n      const result = await tssUtils.isPendingApprovalTxRequestFull(txRequest);\n      result.should.be.false();\n    });\n\n    it('should return false for non-pendingApproval state', async () => {\n      const txRequest = {\n        apiVersion: 'full',\n        state: 'pendingDelivery',\n      } as TxRequest;\n      const result = await tssUtils.isPendingApprovalTxRequestFull(txRequest);\n      result.should.be.false();\n    });\n  });\n\n  // #region Nock helpers\n  async function generateBitgoKeychain(params: {\n    coin: string;\n    userKeyShare: KeyShare;\n    backupKeyShare: KeyShare;\n    bitgoKeyShare: KeyShare;\n    userGpgKey: openpgp.SerializedKeyPair<string>;\n    backupGpgKey: openpgp.SerializedKeyPair<string>;\n    bitgoGpgKey: openpgp.SerializedKeyPair<string>;\n  }): Promise<Keychain> {\n    const bitgoCombined = MPC.keyCombine(params.bitgoKeyShare.uShare, [\n      params.userKeyShare.yShares[3],\n      params.backupKeyShare.yShares[3],\n    ]);\n    const userGpgKeyActual = await openpgp.readKey({ armoredKey: params.userGpgKey.publicKey });\n    const backupGpgKeyActual = await openpgp.readKey({ armoredKey: params.backupGpgKey.publicKey });\n\n    const bitgoToUserMessage = await openpgp.createMessage({\n      text: Buffer.concat([\n        Buffer.from(params.bitgoKeyShare.yShares[1].u, 'hex'),\n        Buffer.from(params.bitgoKeyShare.yShares[1].chaincode, 'hex'),\n      ]).toString('hex'),\n    });\n    const encryptedBitgoToUserMessage = await openpgp.encrypt({\n      message: bitgoToUserMessage,\n      encryptionKeys: [userGpgKeyActual.toPublic()],\n      format: 'armored',\n    });\n\n    const bitgoToBackupMessage = await openpgp.createMessage({\n      text: Buffer.concat([\n        Buffer.from(params.bitgoKeyShare.yShares[2].u, 'hex'),\n        Buffer.from(params.bitgoKeyShare.yShares[2].chaincode, 'hex'),\n      ]).toString('hex'),\n    });\n    const encryptedBitgoToBackupMessage = await openpgp.encrypt({\n      message: bitgoToBackupMessage,\n      encryptionKeys: [backupGpgKeyActual.toPublic()],\n      format: 'armored',\n    });\n\n    const bitgoKeychain: Keychain = {\n      id: '3',\n      pub: '',\n      commonKeychain: bitgoCombined.pShare.y + bitgoCombined.pShare.chaincode,\n      keyShares: [\n        {\n          from: 'bitgo',\n          to: 'user',\n          publicShare: params.bitgoKeyShare.yShares[1].y + params.bitgoKeyShare.yShares[1].chaincode,\n          privateShare: encryptedBitgoToUserMessage.toString(),\n          vssProof: params.bitgoKeyShare.yShares[1].v,\n        },\n        {\n          from: 'bitgo',\n          to: 'backup',\n          publicShare: params.bitgoKeyShare.yShares[2].y + params.bitgoKeyShare.yShares[2].chaincode,\n          privateShare: encryptedBitgoToBackupMessage.toString(),\n          vssProof: params.bitgoKeyShare.yShares[2].v,\n        },\n      ],\n      type: 'tss',\n    };\n\n    const userKeyId = userGpgKeyActual.keyPacket.getFingerprint();\n    const backupKeyId = backupGpgKeyActual.keyPacket.getFingerprint();\n    const bitgoToUserPublicShare =\n      Buffer.from(\n        await sodium.crypto_scalarmult_ed25519_base_noclamp(Buffer.from(params.bitgoKeyShare.yShares[1].u, 'hex'))\n      ).toString('hex') + params.bitgoKeyShare.yShares[1].chaincode;\n    const bitgoToBackupPublicShare =\n      Buffer.from(\n        await sodium.crypto_scalarmult_ed25519_base_noclamp(Buffer.from(params.bitgoKeyShare.yShares[2].u, 'hex'))\n      ).toString('hex') + params.bitgoKeyShare.yShares[2].chaincode;\n\n    bitgoKeychain.walletHSMGPGPublicKeySigs = await createWalletSignatures(\n      params.bitgoGpgKey.privateKey,\n      params.userGpgKey.publicKey,\n      params.backupGpgKey.publicKey,\n      [\n        { name: 'commonKeychain', value: bitgoCombined.pShare.y + bitgoCombined.pShare.chaincode },\n        { name: 'userKeyId', value: userKeyId },\n        { name: 'backupKeyId', value: backupKeyId },\n        { name: 'bitgoToUserPublicShare', value: bitgoToUserPublicShare },\n        { name: 'bitgoToBackupPublicShare', value: bitgoToBackupPublicShare },\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    backupGpgKey: openpgp.SerializedKeyPair<string>;\n    bitgoGpgKey: openpgp.SerializedKeyPair<string>;\n  }): Promise<Keychain> {\n    const bitgoKeychain = await generateBitgoKeychain(params);\n\n    nock(bgUrl)\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 nockUserKeychain(params: { coin: string }): Promise<Keychain> {\n    const userKeychain: Keychain = {\n      id: '1',\n      pub: '',\n      type: 'tss',\n    };\n\n    nock('https://bitgo.fakeurl')\n      .post(`/api/v2/${params.coin}/key`, _.matches({ keyType: 'tss', source: 'user' }))\n      .reply(200, userKeychain);\n\n    return userKeychain;\n  }\n\n  async function nockBackupKeychain(params: { coin: string }): Promise<Keychain> {\n    const backupKeychain: Keychain = {\n      id: '2',\n      pub: '',\n      type: 'tss',\n    };\n\n    nock('https://bitgo.fakeurl')\n      .post(`/api/v2/${params.coin}/key`, _.matches({ keyType: 'tss', source: 'backup' }))\n      .reply(200, backupKeychain);\n\n    return backupKeychain;\n  }\n\n  // #endregion Nock helpers\n});\n"]}

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


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