PHP WebShell

Текущая директория: /opt/BitGoJS/modules/express/dist/test/unit/clientRoutes

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

"use strict";
/**
 * @prettier
 */
Object.defineProperty(exports, "__esModule", { value: true });
const sdk_core_1 = require("@bitgo/sdk-core");
const sdk_test_1 = require("@bitgo/sdk-test");
const should = require("should");
const sinon = require("sinon");
const sdk_lib_mpc_1 = require("@bitgo/sdk-lib-mpc");
const assert = require("assert");
const nock = require("nock");
const fs = require("fs");
require("should-http");
require("should-sinon");
require("../../lib/asserts");
const clientRoutes_1 = require("../../../src/clientRoutes");
const fetchEncryptedPrivKeys_1 = require("../../../src/fetchEncryptedPrivKeys");
const ecdsaNtilde_1 = require("./mocks/ecdsaNtilde");
const bitgo_1 = require("bitgo");
const keyShares_1 = require("./mocks/keyShares");
const gpgKeys_1 = require("./mocks/gpgKeys");
const createKeccakHash = require('keccak');
nock.restore();
describe('External signer', () => {
    let bitgo;
    let bgUrl;
    let MPC;
    let mpcEcdsa;
    let hdTree;
    const walletId = '61f039aad587c2000745c687373e0fa9';
    const walletPassword = 'wDX058%c4plL1@pP';
    const secret = 'xprv9s21ZrQH143K3EuPWCBuqnWxydaQV6et9htQige4EswvcHKEzNmkVmwTwKoadyHzJYppuADB7Us7AbaNLToNvoFoSxuWqndQRYtnNy5DUY2';
    const validPrv = '{"61f039aad587c2000745c687373e0fa9":"{\\"iv\\":\\"+1u1Y9cvsYuRMeyH2slnXQ==\\",\\"v\\":1,\\"iter\\":10000,\\"ks\\":256,\\"ts\\":64,\\"mode\\":\\"ccm\\",\\"adata\\":\\"\\",\\"cipher\\":\\"aes\\",\\"salt\\":\\"54kOXTqJ9mc=\\",\\"ct\\":\\"JF5wQ82wa1dYyFxFlbHCvK4a+A6MTHdhOqc5uXsz2icWhkY2Lin/3Ab8ZwvwDaR1JYKmC/g1gXIGwVZEOl1M/bRHY420h7sDtmTS6Ebse5NWbF0ItfUJlk6HVATGa+C6mkbaVxJ4kQW/ehnT3riqzU069ATPz8E=\\"}"}';
    before(async function () {
        if (!nock.isActive()) {
            nock.activate();
        }
        bitgo = sdk_test_1.TestBitGo.decorate(bitgo_1.BitGo, { env: 'test' });
        bitgo.initializeTestVars();
        bgUrl = sdk_core_1.common.Environments[bitgo.getEnv()].uri;
        hdTree = await sdk_core_1.Ed25519BIP32.initialize();
        MPC = await sdk_core_1.Eddsa.initialize(hdTree);
        mpcEcdsa = new sdk_core_1.Ecdsa();
        const constants = {
            mpc: {
                bitgoPublicKey: gpgKeys_1.bitgoGpgKey.public,
                bitgoMPCv2PublicKey: gpgKeys_1.bitgoGpgKey.public,
            },
        };
        nock(bgUrl).get('/api/v1/client/constants').times(10).reply(200, { ttl: 3600, constants });
    });
    after(() => {
        if (nock.isActive()) {
            nock.restore();
        }
    });
    it('should read an encrypted prv from signerFileSystemPath and pass it to coin.signTransaction', async () => {
        const readFileStub = sinon.stub(fs.promises, 'readFile').resolves(validPrv);
        const envStub = sinon
            .stub(process, 'env')
            .value({ WALLET_61f039aad587c2000745c687373e0fa9_PASSPHRASE: walletPassword });
        const signTransactionStub = sinon
            .stub(bitgo_1.Coin.Btc.prototype, 'signTransaction')
            .resolves({ txHex: 'signedTx', txRequestId: '' });
        const req = {
            bitgo: bitgo,
            body: {
                txPrebuild: {
                    walletId: walletId,
                },
            },
            params: {
                coin: 'tbtc',
            },
            config: {
                signerFileSystemPath: 'signerFileSystemPath',
            },
        };
        await (0, clientRoutes_1.handleV2Sign)(req);
        readFileStub.calledOnceWith('signerFileSystemPath').should.be.true();
        signTransactionStub
            .calledOnceWith(sinon.match({
            prv: secret,
        }))
            .should.be.true();
        readFileStub.restore();
        signTransactionStub.restore();
        envStub.restore();
    });
    it('should read an encrypted prv from signerFileSystemPath and pass it to PaillierModulus, K, MuDelta, and S share generators', async () => {
        const walletID = '62fe536a6b4cf70007acb48c0e7bb0b0';
        const user = keyShares_1.keyShareOneEcdsa; // await mpcEcdsa.keyShare(1, 2, 3);
        const backup = keyShares_1.keyShareTwoEcdsa; // await mpcEcdsa.keyShare(2, 2, 3);
        const bitgo = keyShares_1.keyShareThreeEcdsa; // await mpcEcdsa.keyShare(3, 2, 3);
        const bitgoCombinedKey = await mpcEcdsa.keyCombine(bitgo.pShare, [backup.nShares[3], user.nShares[3]]);
        const userChallenge = ecdsaNtilde_1.mockChallengeA;
        const bitgoChallenge = ecdsaNtilde_1.mockChallengeB;
        const userSigningMaterial = {
            pShare: user.pShare,
            bitgoNShare: bitgo.nShares[1],
            backupNShare: backup.nShares[1],
        };
        const bg = new bitgo_1.BitGo({ env: 'test' });
        const walletPassphrase = 'testPass';
        const validPrv = bg.encrypt({ input: JSON.stringify(userSigningMaterial), password: walletPassphrase });
        const output = {};
        output[walletID] = validPrv;
        const readFileStub = sinon.stub(fs.promises, 'readFile').resolves(JSON.stringify(output));
        const envStub = sinon
            .stub(process, 'env')
            .value({ WALLET_62fe536a6b4cf70007acb48c0e7bb0b0_PASSPHRASE: walletPassphrase });
        const tMessage = 'testMessage';
        const bgTest = new bitgo_1.BitGo({ env: 'test' });
        const derivationPath = '';
        const reqPaillierModulus = {
            bitgo: bgTest,
            body: {
                txRequest: {
                    apiVersion: 'full',
                    walletId: walletID,
                    transactions: [
                        {
                            unsignedTx: {
                                derivationPath,
                                signableHex: tMessage,
                            },
                        },
                    ],
                },
            },
            params: {
                coin: 'tbsc',
                sharetype: 'PaillierModulus',
            },
            config: {
                signerFileSystemPath: 'signerFileSystemPath',
            },
        };
        const paillierResult = await (0, clientRoutes_1.handleV2GenerateShareTSS)(reqPaillierModulus);
        paillierResult.should.have.property('userPaillierModulus');
        const userPaillierModulus = paillierResult.userPaillierModulus;
        const [bitgoToUserPaillierChallenge, userToBitgoPaillierChallenge] = await Promise.all([
            sdk_lib_mpc_1.EcdsaPaillierProof.generateP((0, sdk_lib_mpc_1.hexToBigInt)(userPaillierModulus)),
            sdk_lib_mpc_1.EcdsaPaillierProof.generateP((0, sdk_lib_mpc_1.hexToBigInt)(bitgoCombinedKey.yShares[1].n)),
        ]);
        const reqK = {
            bitgo: bgTest,
            body: {
                tssParams: {
                    txRequest: {
                        apiVersion: 'full',
                        walletId: walletID,
                        transactions: [
                            {
                                unsignedTx: {
                                    derivationPath,
                                    signableHex: tMessage,
                                },
                            },
                        ],
                    },
                },
                challenges: {
                    enterpriseChallenge: {
                        ntilde: userChallenge.ntilde,
                        h1: userChallenge.h1,
                        h2: userChallenge.h2,
                        p: sdk_lib_mpc_1.EcdsaTypes.serializePaillierChallenge({ p: userToBitgoPaillierChallenge }).p,
                    },
                    bitgoChallenge: {
                        ntilde: bitgoChallenge.ntilde,
                        h1: bitgoChallenge.h1,
                        h2: bitgoChallenge.h2,
                        p: sdk_lib_mpc_1.EcdsaTypes.serializePaillierChallenge({ p: bitgoToUserPaillierChallenge }).p,
                        n: bitgo.pShare.n,
                    },
                },
                requestType: 0,
            },
            params: {
                coin: 'tbsc',
                sharetype: 'K',
            },
            config: {
                signerFileSystemPath: 'signerFileSystemPath',
            },
        };
        const kResult = await (0, clientRoutes_1.handleV2GenerateShareTSS)(reqK);
        kResult.should.have.property('kShare');
        kResult.should.have.property('wShare');
        const aShareFromBitgo = await mpcEcdsa.signConvertStep1({
            xShare: mpcEcdsa.appendChallenge(bitgoCombinedKey.xShare, bitgoChallenge, sdk_lib_mpc_1.EcdsaTypes.serializePaillierChallenge({ p: bitgoToUserPaillierChallenge })),
            yShare: bitgoCombinedKey.yShares[1],
            kShare: kResult.kShare,
        });
        const reqMuDelta = {
            bitgo: bgTest,
            body: {
                txRequest: {
                    apiVersion: 'full',
                    walletId: walletID,
                    transactions: [
                        {
                            unsignedTx: {
                                derivationPath,
                                signableHex: tMessage,
                            },
                        },
                    ],
                },
                aShareFromBitgo: aShareFromBitgo.aShare,
                bitgoChallenge: {
                    ntilde: bitgoChallenge.ntilde,
                    h1: bitgoChallenge.h1,
                    h2: bitgoChallenge.h2,
                    p: sdk_lib_mpc_1.EcdsaTypes.serializePaillierChallenge({ p: bitgoToUserPaillierChallenge }).p,
                    n: bitgo.pShare.n,
                },
                encryptedWShare: kResult.wShare,
            },
            params: {
                coin: 'tbsc',
                sharetype: 'MuDelta',
            },
            config: {
                signerFileSystemPath: 'signerFileSystemPath',
            },
        };
        const muDeltaResult = await (0, clientRoutes_1.handleV2GenerateShareTSS)(reqMuDelta);
        muDeltaResult.should.have.property('muDShare');
        muDeltaResult.should.have.property('oShare');
        const bitgoGShare = await mpcEcdsa.signConvertStep3({
            bShare: aShareFromBitgo.bShare,
            muShare: muDeltaResult.muDShare.muShare,
        });
        const bitgoDShare = mpcEcdsa.signCombine({
            gShare: bitgoGShare.gShare,
            signIndex: {
                i: 1,
                j: 3,
            },
        });
        const reqS = {
            bitgo: bgTest,
            body: {
                tssParams: {
                    txRequest: {
                        apiVersion: 'full',
                        walletId: walletID,
                        transactions: [
                            {
                                unsignedTx: {
                                    derivationPath,
                                    signableHex: tMessage,
                                },
                            },
                        ],
                    },
                },
                dShareFromBitgo: bitgoDShare.dShare,
                requestType: 0,
                encryptedOShare: muDeltaResult.oShare,
            },
            params: {
                coin: 'tbsc',
                sharetype: 'S',
            },
            config: {
                signerFileSystemPath: 'signerFileSystemPath',
            },
        };
        const sResult = await (0, clientRoutes_1.handleV2GenerateShareTSS)(reqS);
        sResult.should.have.property('R');
        sResult.should.have.property('s');
        sResult.should.have.property('y');
        const bitGoSShare = mpcEcdsa.sign(Buffer.from(tMessage, 'hex'), bitgoDShare.oShare, muDeltaResult.muDShare.dShare, createKeccakHash('keccak256'));
        const signature = mpcEcdsa.constructSignature([bitGoSShare, sResult]);
        mpcEcdsa.verify(Buffer.from(tMessage, 'hex'), signature, createKeccakHash('keccak256')).should.be.true;
        readFileStub.restore();
        envStub.restore();
    });
    it('should read an encrypted prv from signerFileSystemPath and pass it to commitment, R and G share generators', async () => {
        const walletID = '62fe536a6b4cf70007acb48c0e7bb0b0';
        const user = MPC.keyShare(1, 2, 3);
        const backup = MPC.keyShare(2, 2, 3);
        const bitgo = MPC.keyShare(3, 2, 3);
        const userSigningMaterial = {
            uShare: user.uShare,
            bitgoYShare: bitgo.yShares[1],
            backupYShare: backup.yShares[1],
        };
        const bg = new bitgo_1.BitGo({ env: 'test' });
        const walletPassphrase = 'testPass';
        const validPrv = bg.encrypt({ input: JSON.stringify(userSigningMaterial), password: walletPassphrase });
        const output = {};
        output[walletID] = validPrv;
        const readFileStub = sinon.stub(fs.promises, 'readFile').resolves(JSON.stringify(output));
        const envStub = sinon
            .stub(process, 'env')
            .value({ WALLET_62fe536a6b4cf70007acb48c0e7bb0b0_PASSPHRASE: walletPassphrase });
        const tMessage = 'testMessage';
        const bgTest = new bitgo_1.BitGo({ env: 'test' });
        const derivationPath = 'm/0';
        const reqCommitment = {
            bitgo: bgTest,
            body: {
                bitgoGpgPubKey: gpgKeys_1.bitgoGpgKey.public,
                txRequest: {
                    apiVersion: 'full',
                    walletId: walletID,
                    transactions: [
                        {
                            unsignedTx: {
                                derivationPath,
                                signableHex: tMessage,
                            },
                        },
                    ],
                },
            },
            params: {
                coin: 'tsol',
                sharetype: 'commitment',
            },
            config: {
                signerFileSystemPath: 'signerFileSystemPath',
            },
        };
        const cResult = await (0, clientRoutes_1.handleV2GenerateShareTSS)(reqCommitment);
        cResult.should.have.property('userToBitgoCommitment');
        cResult.should.have.property('encryptedSignerShare');
        cResult.should.have.property('encryptedUserToBitgoRShare');
        const encryptedUserToBitgoRShare = cResult.encryptedUserToBitgoRShare;
        const reqR = {
            bitgo: bgTest,
            body: {
                txRequest: {
                    apiVersion: 'full',
                    walletId: walletID,
                    transactions: [
                        {
                            unsignedTx: {
                                derivationPath,
                                signableHex: tMessage,
                            },
                        },
                    ],
                },
                encryptedUserToBitgoRShare,
            },
            params: {
                coin: 'tsol',
                sharetype: 'R',
            },
            config: {
                signerFileSystemPath: 'signerFileSystemPath',
            },
        };
        const rResult = await (0, clientRoutes_1.handleV2GenerateShareTSS)(reqR);
        rResult.should.have.property('rShare');
        const signingKey = MPC.keyDerive(userSigningMaterial.uShare, [userSigningMaterial.bitgoYShare, userSigningMaterial.backupYShare], derivationPath);
        const bitgoCombine = MPC.keyCombine(bitgo.uShare, [signingKey.yShares[3], backup.yShares[3]]);
        const bitgoSignShare = await MPC.signShare(Buffer.from(tMessage, 'hex'), bitgoCombine.pShare, [
            bitgoCombine.jShares[1],
        ]);
        const signatureShareRec = {
            from: sdk_core_1.SignatureShareType.BITGO,
            to: sdk_core_1.SignatureShareType.USER,
            share: bitgoSignShare.rShares[1].r + bitgoSignShare.rShares[1].R,
        };
        const bitgoToUserCommitmentShare = {
            from: sdk_core_1.SignatureShareType.BITGO,
            to: sdk_core_1.SignatureShareType.USER,
            share: bitgoSignShare.rShares[1].commitment,
            type: 'commitment',
        };
        const reqG = {
            bitgo: bgTest,
            body: {
                txRequest: {
                    apiVersion: 'full',
                    walletId: walletID,
                    transactions: [
                        {
                            unsignedTx: {
                                derivationPath,
                                signableHex: tMessage,
                            },
                        },
                    ],
                },
                userToBitgoRShare: rResult.rShare,
                bitgoToUserRShare: signatureShareRec,
                bitgoToUserCommitment: bitgoToUserCommitmentShare,
            },
            params: {
                coin: 'tsol',
                sharetype: 'G',
            },
            config: {
                signerFileSystemPath: 'signerFileSystemPath',
            },
        };
        const userGShare = await (0, clientRoutes_1.handleV2GenerateShareTSS)(reqG);
        userGShare.should.have.property('i');
        userGShare.should.have.property('y');
        userGShare.should.have.property('gamma');
        userGShare.should.have.property('R');
        const userToBitgoRShare = {
            i: sdk_core_1.ShareKeyPosition.BITGO,
            j: sdk_core_1.ShareKeyPosition.USER,
            u: signingKey.yShares[3].u,
            v: rResult.rShare.rShares[3].v,
            r: rResult.rShare.rShares[3].r,
            R: rResult.rShare.rShares[3].R,
            commitment: rResult.rShare.rShares[3].commitment,
        };
        const bitgoGShare = MPC.sign(Buffer.from(tMessage, 'hex'), bitgoSignShare.xShare, [userToBitgoRShare], [backup.yShares[3]]);
        const signature = MPC.signCombine([userGShare, bitgoGShare]);
        const veriResult = MPC.verify(Buffer.from(tMessage, 'hex'), signature);
        veriResult.should.be.true();
        readFileStub.restore();
        envStub.restore();
    });
    it('should read an encrypted prv from signerFileSystemPath and pass it to MPCv2Round1, MPCv2Round2 and MPCv2Round3 share generators', async () => {
        const walletID = '62fe536a6b4cf70007acb48c0e7bb0b0';
        const tMessage = 'testMessage';
        const derivationPath = 'm/0';
        const walletPassphrase = 'testPass';
        const [userShare, backupShare, bitgoShare] = await sdk_lib_mpc_1.DklsUtils.generateDKGKeyShares();
        assert(backupShare, 'backupShare is not defined');
        const bgTest = new bitgo_1.BitGo({ env: 'test' });
        const userKeyShare = userShare.getKeyShare().toString('base64');
        const validPrv = bgTest.encrypt({ input: userKeyShare, password: walletPassphrase });
        const readFileStub = sinon.stub(fs.promises, 'readFile').resolves(JSON.stringify({ [walletID]: validPrv }));
        const envStub = sinon.stub(process, 'env').value({ ['WALLET_' + walletID + '_PASSPHRASE']: walletPassphrase });
        // round 1
        const reqMPCv2Round1 = {
            bitgo: bgTest,
            body: {
                txRequest: {
                    txRequestId: '123456',
                    apiVersion: 'full',
                    walletId: walletID,
                    transactions: [
                        {
                            unsignedTx: {
                                derivationPath,
                                signableHex: tMessage,
                            },
                            signatureShares: [],
                        },
                    ],
                },
            },
            params: {
                coin: 'hteth',
                sharetype: 'MPCv2Round1',
            },
            config: {
                signerFileSystemPath: 'signerFileSystemPath',
            },
        };
        const round1Result = await (0, clientRoutes_1.handleV2GenerateShareTSS)(reqMPCv2Round1);
        round1Result.should.have.property('signatureShareRound1');
        round1Result.should.have.property('userGpgPubKey');
        round1Result.should.have.property('encryptedRound1Session');
        round1Result.should.have.property('encryptedUserGpgPrvKey');
        const hashFn = createKeccakHash('keccak256');
        const hashBuffer = hashFn.update(Buffer.from(tMessage, 'hex')).digest();
        const bitgoSession = new sdk_lib_mpc_1.DklsDsg.Dsg(bitgoShare.getKeyShare(), 2, derivationPath, hashBuffer);
        const txRequestRound1 = await signBitgoMPCv2Round1(bitgoSession, reqMPCv2Round1.body.txRequest, round1Result.signatureShareRound1, round1Result.userGpgPubKey);
        assert(txRequestRound1.transactions &&
            txRequestRound1.transactions.length === 1 &&
            txRequestRound1.transactions[0].signatureShares.length === 2, 'txRequestRound2.transactions is not an array of length 1 with 2 signatureShares');
        // round 2
        const reqMPCv2Round2 = {
            bitgo: bgTest,
            body: {
                txRequest: txRequestRound1,
                encryptedRound1Session: round1Result.encryptedRound1Session,
                encryptedUserGpgPrvKey: round1Result.encryptedUserGpgPrvKey,
                bitgoPublicGpgKey: gpgKeys_1.bitgoGpgKey.public,
            },
            params: {
                coin: 'hteth',
                sharetype: 'MPCv2Round2',
            },
            config: {
                signerFileSystemPath: 'signerFileSystemPath',
            },
        };
        const round2Result = await (0, clientRoutes_1.handleV2GenerateShareTSS)(reqMPCv2Round2);
        round2Result.should.have.property('signatureShareRound2');
        round2Result.should.have.property('encryptedRound2Session');
        const { txRequest: txRequestRound2, bitgoMsg4 } = await signBitgoMPCv2Round2(bitgoSession, reqMPCv2Round2.body.txRequest, round2Result.signatureShareRound2, round1Result.userGpgPubKey);
        assert(txRequestRound2.transactions &&
            txRequestRound2.transactions.length === 1 &&
            txRequestRound2.transactions[0].signatureShares.length === 4, 'txRequestRound2.transactions is not an array of length 1 with 4 signatureShares');
        // round 3
        const reqMPCv2Round3 = {
            bitgo: bgTest,
            body: {
                txRequest: txRequestRound2,
                encryptedRound2Session: round2Result.encryptedRound2Session,
                encryptedUserGpgPrvKey: round1Result.encryptedUserGpgPrvKey,
                bitgoPublicGpgKey: gpgKeys_1.bitgoGpgKey.public,
            },
            params: {
                coin: 'hteth',
                sharetype: 'MPCv2Round3',
            },
            config: {
                signerFileSystemPath: 'signerFileSystemPath',
            },
        };
        const round3Result = await (0, clientRoutes_1.handleV2GenerateShareTSS)(reqMPCv2Round3);
        round3Result.should.have.property('signatureShareRound3');
        const { userMsg4 } = await signBitgoMPCv2Round3(bitgoSession, round3Result.signatureShareRound3, round1Result.userGpgPubKey);
        // signature generation and validation
        assert(userMsg4.data.msg4.signatureR === bitgoMsg4.signatureR, 'User and BitGo signaturesR do not match');
        const deserializedBitgoMsg4 = sdk_lib_mpc_1.DklsTypes.deserializeMessages({
            p2pMessages: [],
            broadcastMessages: [bitgoMsg4],
        });
        const deserializedUserMsg4 = sdk_lib_mpc_1.DklsTypes.deserializeMessages({
            p2pMessages: [],
            broadcastMessages: [
                {
                    from: userMsg4.data.msg4.from,
                    payload: userMsg4.data.msg4.message,
                },
            ],
        });
        const combinedSigUsingUtil = sdk_lib_mpc_1.DklsUtils.combinePartialSignatures([deserializedUserMsg4.broadcastMessages[0].payload, deserializedBitgoMsg4.broadcastMessages[0].payload], Buffer.from(userMsg4.data.msg4.signatureR, 'base64').toString('hex'));
        const convertedSignature = sdk_lib_mpc_1.DklsUtils.verifyAndConvertDklsSignature(Buffer.from(tMessage, 'hex'), combinedSigUsingUtil, sdk_lib_mpc_1.DklsTypes.getCommonKeychain(userShare.getKeyShare()), derivationPath, createKeccakHash('keccak256'));
        assert(convertedSignature, 'Signature is not valid');
        assert(convertedSignature.split(':').length === 4, 'Signature is not valid');
        readFileStub.restore();
        envStub.restore();
    });
    it('should accept a local secret and password for a wallet', async () => {
        const accessToken = '';
        const walletIds = {
            tbtc: [
                {
                    walletId,
                    walletPassword,
                    secret,
                },
            ],
        };
        const walletResult = {
            walletId,
            keys: [walletId, walletId, walletId],
        };
        const keyResult = {
            walletId,
        };
        nock(bgUrl).get(`/api/v2/tbtc/wallet/${walletId}`).reply(200, walletResult);
        nock(bgUrl).get(`/api/v2/tbtc/key/${walletId}`).reply(200, keyResult);
        const data = await (0, fetchEncryptedPrivKeys_1.fetchKeys)(walletIds, accessToken);
        should.exist(data[walletId]);
        data[walletId].should.startWith('{"iv":"');
    });
});
// #region MPCv2 utils
function getBitGoPartyGpgKeyPrv(bitgoPrvKey) {
    return {
        partyId: 2,
        gpgKey: bitgoPrvKey,
    };
}
function getUserPartyGpgKeyPublic(userPubKey) {
    return {
        partyId: 0,
        gpgKey: userPubKey,
    };
}
async function signBitgoMPCv2Round1(bitgoSession, txRequest, userShare, userGPGPubKey) {
    assert(txRequest.transactions && txRequest.transactions.length === 1, 'txRequest.transactions is not an array of length 1');
    txRequest.transactions[0].signatureShares.push(userShare);
    // Do the actual signing on BitGo's side based on User's messages
    const signatureShare = JSON.parse(userShare.share);
    const deserializedMessages = sdk_lib_mpc_1.DklsTypes.deserializeMessages({
        p2pMessages: [],
        broadcastMessages: [
            {
                from: signatureShare.data.msg1.from,
                payload: signatureShare.data.msg1.message,
            },
        ],
    });
    const bitgoToUserRound1BroadcastMsg = await bitgoSession.init();
    const bitgoToUserRound2Msg = bitgoSession.handleIncomingMessages({
        p2pMessages: [],
        broadcastMessages: deserializedMessages.broadcastMessages,
    });
    const serializedBitGoToUserRound1And2Msgs = sdk_lib_mpc_1.DklsTypes.serializeMessages({
        p2pMessages: bitgoToUserRound2Msg.p2pMessages,
        broadcastMessages: [bitgoToUserRound1BroadcastMsg],
    });
    const authEncMessages = await sdk_lib_mpc_1.DklsComms.encryptAndAuthOutgoingMessages(serializedBitGoToUserRound1And2Msgs, [getUserPartyGpgKeyPublic(userGPGPubKey)], [getBitGoPartyGpgKeyPrv(gpgKeys_1.bitgoGpgKey.private)]);
    const bitgoToUserSignatureShare = {
        type: 'round1Output',
        data: {
            msg1: {
                from: authEncMessages.broadcastMessages[0].from,
                signature: authEncMessages.broadcastMessages[0].payload.signature,
                message: authEncMessages.broadcastMessages[0].payload.message,
            },
            msg2: {
                from: authEncMessages.p2pMessages[0].from,
                to: authEncMessages.p2pMessages[0].to,
                encryptedMessage: authEncMessages.p2pMessages[0].payload.encryptedMessage,
                signature: authEncMessages.p2pMessages[0].payload.signature,
            },
        },
    };
    txRequest.transactions[0].signatureShares.push({
        from: sdk_core_1.SignatureShareType.BITGO,
        to: sdk_core_1.SignatureShareType.USER,
        share: JSON.stringify(bitgoToUserSignatureShare),
    });
    return txRequest;
}
async function signBitgoMPCv2Round2(bitgoSession, txRequest, userShare, userGPGPubKey) {
    assert(txRequest.transactions && txRequest.transactions.length === 1, 'txRequest.transactions is not an array of length 1');
    txRequest.transactions[0].signatureShares.push(userShare);
    // Do the actual signing on BitGo's side based on User's messages
    const parsedSignatureShare = JSON.parse(userShare.share);
    const serializedMessages = await sdk_lib_mpc_1.DklsComms.decryptAndVerifyIncomingMessages({
        p2pMessages: [
            {
                from: parsedSignatureShare.data.msg2.from,
                to: parsedSignatureShare.data.msg2.to,
                payload: {
                    encryptedMessage: parsedSignatureShare.data.msg2.encryptedMessage,
                    signature: parsedSignatureShare.data.msg2.signature,
                },
            },
            {
                from: parsedSignatureShare.data.msg3.from,
                to: parsedSignatureShare.data.msg3.to,
                payload: {
                    encryptedMessage: parsedSignatureShare.data.msg3.encryptedMessage,
                    signature: parsedSignatureShare.data.msg3.signature,
                },
            },
        ],
        broadcastMessages: [],
    }, [getUserPartyGpgKeyPublic(userGPGPubKey)], [getBitGoPartyGpgKeyPrv(gpgKeys_1.bitgoGpgKey.private)]);
    const deserializedMessages2 = sdk_lib_mpc_1.DklsTypes.deserializeMessages({
        p2pMessages: [serializedMessages.p2pMessages[0]],
        broadcastMessages: [],
    });
    const bitgoToUserRound3Msg = bitgoSession.handleIncomingMessages(deserializedMessages2);
    const serializedBitGoToUserRound3Msgs = sdk_lib_mpc_1.DklsTypes.serializeMessages(bitgoToUserRound3Msg);
    const authEncMessages = await sdk_lib_mpc_1.DklsComms.encryptAndAuthOutgoingMessages(serializedBitGoToUserRound3Msgs, [getUserPartyGpgKeyPublic(userGPGPubKey)], [getBitGoPartyGpgKeyPrv(gpgKeys_1.bitgoGpgKey.private)]);
    const bitgoToUserSignatureShare = {
        type: 'round2Output',
        data: {
            msg3: {
                from: authEncMessages.p2pMessages[0].from,
                to: authEncMessages.p2pMessages[0].to,
                encryptedMessage: authEncMessages.p2pMessages[0].payload.encryptedMessage,
                signature: authEncMessages.p2pMessages[0].payload.signature,
            },
        },
    };
    // handling user msg3 but not returning bitgo msg4 since its stored on bitgo side only
    const deserializedMessages3 = sdk_lib_mpc_1.DklsTypes.deserializeMessages({
        p2pMessages: [serializedMessages.p2pMessages[1]],
        broadcastMessages: [],
    });
    const deserializedBitgoMsg4 = bitgoSession.handleIncomingMessages(deserializedMessages3);
    const serializedBitGoToUserRound4Msgs = sdk_lib_mpc_1.DklsTypes.serializeMessages(deserializedBitgoMsg4);
    txRequest.transactions[0].signatureShares.push({
        from: sdk_core_1.SignatureShareType.BITGO,
        to: sdk_core_1.SignatureShareType.USER,
        share: JSON.stringify(bitgoToUserSignatureShare),
    });
    return { txRequest, bitgoMsg4: serializedBitGoToUserRound4Msgs.broadcastMessages[0] };
}
async function signBitgoMPCv2Round3(bitgoSession, userShare, userGPGPubKey) {
    const parsedSignatureShare = JSON.parse(userShare.share);
    const serializedMessages = await sdk_lib_mpc_1.DklsComms.decryptAndVerifyIncomingMessages({
        p2pMessages: [],
        broadcastMessages: [
            {
                from: parsedSignatureShare.data.msg4.from,
                payload: {
                    message: parsedSignatureShare.data.msg4.message,
                    signature: parsedSignatureShare.data.msg4.signature,
                },
            },
        ],
    }, [getUserPartyGpgKeyPublic(userGPGPubKey)], [getBitGoPartyGpgKeyPrv(gpgKeys_1.bitgoGpgKey.private)]);
    const deserializedMessages = sdk_lib_mpc_1.DklsTypes.deserializeMessages({
        p2pMessages: [],
        broadcastMessages: [serializedMessages.broadcastMessages[0]],
    });
    bitgoSession.handleIncomingMessages(deserializedMessages);
    return {
        userMsg4: parsedSignatureShare,
    };
}
// #endregion
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"externalSign.js","sourceRoot":"","sources":["../../../../test/unit/clientRoutes/externalSign.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\n\nimport {\n  common,\n  Ed25519BIP32,\n  Eddsa,\n  Ecdsa,\n  HDTree,\n  SignatureShareType,\n  ShareKeyPosition,\n  TxRequest,\n  SignatureShareRecord,\n} from '@bitgo/sdk-core';\nimport { Hash } from 'crypto';\nimport { TestBitGo, TestBitGoAPI } from '@bitgo/sdk-test';\nimport * as should from 'should';\nimport * as sinon from 'sinon';\nimport {\n  EcdsaPaillierProof,\n  EcdsaTypes,\n  hexToBigInt,\n  DklsUtils,\n  DklsTypes,\n  DklsComms,\n  DklsDsg,\n} from '@bitgo/sdk-lib-mpc';\nimport {\n  MPCv2PartyFromStringOrNumber,\n  MPCv2SignatureShareRound1Input,\n  MPCv2SignatureShareRound1Output,\n  MPCv2SignatureShareRound2Input,\n  MPCv2SignatureShareRound2Output,\n  MPCv2SignatureShareRound3Input,\n} from '@bitgo/public-types';\nimport * as assert from 'assert';\nimport * as nock from 'nock';\nimport * as fs from 'fs';\nimport * as express from 'express';\n\nimport 'should-http';\nimport 'should-sinon';\nimport '../../lib/asserts';\n\nimport { handleV2GenerateShareTSS, handleV2Sign } from '../../../src/clientRoutes';\nimport { fetchKeys } from '../../../src/fetchEncryptedPrivKeys';\nimport { mockChallengeA, mockChallengeB } from './mocks/ecdsaNtilde';\nimport { Coin, BitGo, SignedTransaction } from 'bitgo';\nimport { keyShareOneEcdsa, keyShareTwoEcdsa, keyShareThreeEcdsa } from './mocks/keyShares';\nimport { bitgoGpgKey } from './mocks/gpgKeys';\n\nconst createKeccakHash = require('keccak');\nnock.restore();\n\ntype Output = {\n  [key: string]: string;\n};\n\ndescribe('External signer', () => {\n  let bitgo: TestBitGoAPI;\n  let bgUrl;\n  let MPC: Eddsa;\n  let mpcEcdsa: Ecdsa;\n  let hdTree: HDTree;\n\n  const walletId = '61f039aad587c2000745c687373e0fa9';\n  const walletPassword = 'wDX058%c4plL1@pP';\n  const secret =\n    'xprv9s21ZrQH143K3EuPWCBuqnWxydaQV6et9htQige4EswvcHKEzNmkVmwTwKoadyHzJYppuADB7Us7AbaNLToNvoFoSxuWqndQRYtnNy5DUY2';\n  const validPrv =\n    '{\"61f039aad587c2000745c687373e0fa9\":\"{\\\\\"iv\\\\\":\\\\\"+1u1Y9cvsYuRMeyH2slnXQ==\\\\\",\\\\\"v\\\\\":1,\\\\\"iter\\\\\":10000,\\\\\"ks\\\\\":256,\\\\\"ts\\\\\":64,\\\\\"mode\\\\\":\\\\\"ccm\\\\\",\\\\\"adata\\\\\":\\\\\"\\\\\",\\\\\"cipher\\\\\":\\\\\"aes\\\\\",\\\\\"salt\\\\\":\\\\\"54kOXTqJ9mc=\\\\\",\\\\\"ct\\\\\":\\\\\"JF5wQ82wa1dYyFxFlbHCvK4a+A6MTHdhOqc5uXsz2icWhkY2Lin/3Ab8ZwvwDaR1JYKmC/g1gXIGwVZEOl1M/bRHY420h7sDtmTS6Ebse5NWbF0ItfUJlk6HVATGa+C6mkbaVxJ4kQW/ehnT3riqzU069ATPz8E=\\\\\"}\"}';\n\n  before(async function () {\n    if (!nock.isActive()) {\n      nock.activate();\n    }\n\n    bitgo = TestBitGo.decorate(BitGo, { env: 'test' });\n    bitgo.initializeTestVars();\n\n    bgUrl = common.Environments[bitgo.getEnv()].uri;\n    hdTree = await Ed25519BIP32.initialize();\n    MPC = await Eddsa.initialize(hdTree);\n    mpcEcdsa = new Ecdsa();\n\n    const constants = {\n      mpc: {\n        bitgoPublicKey: bitgoGpgKey.public,\n        bitgoMPCv2PublicKey: bitgoGpgKey.public,\n      },\n    };\n\n    nock(bgUrl).get('/api/v1/client/constants').times(10).reply(200, { ttl: 3600, constants });\n  });\n\n  after(() => {\n    if (nock.isActive()) {\n      nock.restore();\n    }\n  });\n\n  it('should read an encrypted prv from signerFileSystemPath and pass it to coin.signTransaction', async () => {\n    const readFileStub = sinon.stub(fs.promises, 'readFile').resolves(validPrv);\n    const envStub = sinon\n      .stub(process, 'env')\n      .value({ WALLET_61f039aad587c2000745c687373e0fa9_PASSPHRASE: walletPassword });\n    const signTransactionStub = sinon\n      .stub(Coin.Btc.prototype, 'signTransaction')\n      .resolves({ txHex: 'signedTx', txRequestId: '' } as SignedTransaction);\n\n    const req = {\n      bitgo: bitgo,\n      body: {\n        txPrebuild: {\n          walletId: walletId,\n        },\n      },\n      params: {\n        coin: 'tbtc',\n      },\n      config: {\n        signerFileSystemPath: 'signerFileSystemPath',\n      },\n    } as unknown as express.Request;\n\n    await handleV2Sign(req);\n\n    readFileStub.calledOnceWith('signerFileSystemPath').should.be.true();\n    signTransactionStub\n      .calledOnceWith(\n        sinon.match({\n          prv: secret,\n        })\n      )\n      .should.be.true();\n    readFileStub.restore();\n    signTransactionStub.restore();\n    envStub.restore();\n  });\n\n  it('should read an encrypted prv from signerFileSystemPath and pass it to PaillierModulus, K, MuDelta, and S share generators', async () => {\n    const walletID = '62fe536a6b4cf70007acb48c0e7bb0b0';\n    const user = keyShareOneEcdsa; // await mpcEcdsa.keyShare(1, 2, 3);\n    const backup = keyShareTwoEcdsa; // await mpcEcdsa.keyShare(2, 2, 3);\n    const bitgo = keyShareThreeEcdsa; // await mpcEcdsa.keyShare(3, 2, 3);\n    const bitgoCombinedKey = await mpcEcdsa.keyCombine(bitgo.pShare, [backup.nShares[3], user.nShares[3]]);\n    const userChallenge = mockChallengeA;\n    const bitgoChallenge = mockChallengeB;\n    const userSigningMaterial = {\n      pShare: user.pShare,\n      bitgoNShare: bitgo.nShares[1],\n      backupNShare: backup.nShares[1],\n    };\n    const bg = new BitGo({ env: 'test' });\n    const walletPassphrase = 'testPass';\n    const validPrv = bg.encrypt({ input: JSON.stringify(userSigningMaterial), password: walletPassphrase });\n    const output: Output = {};\n    output[walletID] = validPrv;\n    const readFileStub = sinon.stub(fs.promises, 'readFile').resolves(JSON.stringify(output));\n    const envStub = sinon\n      .stub(process, 'env')\n      .value({ WALLET_62fe536a6b4cf70007acb48c0e7bb0b0_PASSPHRASE: walletPassphrase });\n    const tMessage = 'testMessage';\n    const bgTest = new BitGo({ env: 'test' });\n    const derivationPath = '';\n    const reqPaillierModulus = {\n      bitgo: bgTest,\n      body: {\n        txRequest: {\n          apiVersion: 'full',\n          walletId: walletID,\n          transactions: [\n            {\n              unsignedTx: {\n                derivationPath,\n                signableHex: tMessage,\n              },\n            },\n          ],\n        },\n      },\n      params: {\n        coin: 'tbsc',\n        sharetype: 'PaillierModulus',\n      },\n      config: {\n        signerFileSystemPath: 'signerFileSystemPath',\n      },\n    } as unknown as express.Request;\n    const paillierResult = await handleV2GenerateShareTSS(reqPaillierModulus);\n    paillierResult.should.have.property('userPaillierModulus');\n    const userPaillierModulus = paillierResult.userPaillierModulus;\n    const [bitgoToUserPaillierChallenge, userToBitgoPaillierChallenge] = await Promise.all([\n      EcdsaPaillierProof.generateP(hexToBigInt(userPaillierModulus)),\n      EcdsaPaillierProof.generateP(hexToBigInt(bitgoCombinedKey.yShares[1].n)),\n    ]);\n    const reqK = {\n      bitgo: bgTest,\n      body: {\n        tssParams: {\n          txRequest: {\n            apiVersion: 'full',\n            walletId: walletID,\n            transactions: [\n              {\n                unsignedTx: {\n                  derivationPath,\n                  signableHex: tMessage,\n                },\n              },\n            ],\n          },\n        },\n        challenges: {\n          enterpriseChallenge: {\n            ntilde: userChallenge.ntilde,\n            h1: userChallenge.h1,\n            h2: userChallenge.h2,\n            p: EcdsaTypes.serializePaillierChallenge({ p: userToBitgoPaillierChallenge }).p,\n          },\n          bitgoChallenge: {\n            ntilde: bitgoChallenge.ntilde,\n            h1: bitgoChallenge.h1,\n            h2: bitgoChallenge.h2,\n            p: EcdsaTypes.serializePaillierChallenge({ p: bitgoToUserPaillierChallenge }).p,\n            n: bitgo.pShare.n,\n          },\n        },\n        requestType: 0,\n      },\n      params: {\n        coin: 'tbsc',\n        sharetype: 'K',\n      },\n      config: {\n        signerFileSystemPath: 'signerFileSystemPath',\n      },\n    } as unknown as express.Request;\n    const kResult = await handleV2GenerateShareTSS(reqK);\n    kResult.should.have.property('kShare');\n    kResult.should.have.property('wShare');\n    const aShareFromBitgo = await mpcEcdsa.signConvertStep1({\n      xShare: mpcEcdsa.appendChallenge(\n        bitgoCombinedKey.xShare,\n        bitgoChallenge,\n        EcdsaTypes.serializePaillierChallenge({ p: bitgoToUserPaillierChallenge })\n      ),\n      yShare: bitgoCombinedKey.yShares[1],\n      kShare: kResult.kShare,\n    });\n    const reqMuDelta = {\n      bitgo: bgTest,\n      body: {\n        txRequest: {\n          apiVersion: 'full',\n          walletId: walletID,\n          transactions: [\n            {\n              unsignedTx: {\n                derivationPath,\n                signableHex: tMessage,\n              },\n            },\n          ],\n        },\n        aShareFromBitgo: aShareFromBitgo.aShare,\n        bitgoChallenge: {\n          ntilde: bitgoChallenge.ntilde,\n          h1: bitgoChallenge.h1,\n          h2: bitgoChallenge.h2,\n          p: EcdsaTypes.serializePaillierChallenge({ p: bitgoToUserPaillierChallenge }).p,\n          n: bitgo.pShare.n,\n        },\n        encryptedWShare: kResult.wShare,\n      },\n      params: {\n        coin: 'tbsc',\n        sharetype: 'MuDelta',\n      },\n      config: {\n        signerFileSystemPath: 'signerFileSystemPath',\n      },\n    } as unknown as express.Request;\n    const muDeltaResult = await handleV2GenerateShareTSS(reqMuDelta);\n    muDeltaResult.should.have.property('muDShare');\n    muDeltaResult.should.have.property('oShare');\n    const bitgoGShare = await mpcEcdsa.signConvertStep3({\n      bShare: aShareFromBitgo.bShare,\n      muShare: muDeltaResult.muDShare.muShare,\n    });\n    const bitgoDShare = mpcEcdsa.signCombine({\n      gShare: bitgoGShare.gShare,\n      signIndex: {\n        i: 1,\n        j: 3,\n      },\n    });\n    const reqS = {\n      bitgo: bgTest,\n      body: {\n        tssParams: {\n          txRequest: {\n            apiVersion: 'full',\n            walletId: walletID,\n            transactions: [\n              {\n                unsignedTx: {\n                  derivationPath,\n                  signableHex: tMessage,\n                },\n              },\n            ],\n          },\n        },\n        dShareFromBitgo: bitgoDShare.dShare,\n        requestType: 0,\n        encryptedOShare: muDeltaResult.oShare,\n      },\n      params: {\n        coin: 'tbsc',\n        sharetype: 'S',\n      },\n      config: {\n        signerFileSystemPath: 'signerFileSystemPath',\n      },\n    } as unknown as express.Request;\n    const sResult = await handleV2GenerateShareTSS(reqS);\n    sResult.should.have.property('R');\n    sResult.should.have.property('s');\n    sResult.should.have.property('y');\n    const bitGoSShare = mpcEcdsa.sign(\n      Buffer.from(tMessage, 'hex'),\n      bitgoDShare.oShare,\n      muDeltaResult.muDShare.dShare,\n      createKeccakHash('keccak256') as Hash\n    );\n    const signature = mpcEcdsa.constructSignature([bitGoSShare, sResult]);\n    mpcEcdsa.verify(Buffer.from(tMessage, 'hex'), signature, createKeccakHash('keccak256') as Hash).should.be.true;\n    readFileStub.restore();\n    envStub.restore();\n  });\n\n  it('should read an encrypted prv from signerFileSystemPath and pass it to commitment, R and G share generators', async () => {\n    const walletID = '62fe536a6b4cf70007acb48c0e7bb0b0';\n    const user = MPC.keyShare(1, 2, 3);\n    const backup = MPC.keyShare(2, 2, 3);\n    const bitgo = MPC.keyShare(3, 2, 3);\n    const userSigningMaterial = {\n      uShare: user.uShare,\n      bitgoYShare: bitgo.yShares[1],\n      backupYShare: backup.yShares[1],\n    };\n    const bg = new BitGo({ env: 'test' });\n    const walletPassphrase = 'testPass';\n    const validPrv = bg.encrypt({ input: JSON.stringify(userSigningMaterial), password: walletPassphrase });\n    const output: Output = {};\n    output[walletID] = validPrv;\n    const readFileStub = sinon.stub(fs.promises, 'readFile').resolves(JSON.stringify(output));\n    const envStub = sinon\n      .stub(process, 'env')\n      .value({ WALLET_62fe536a6b4cf70007acb48c0e7bb0b0_PASSPHRASE: walletPassphrase });\n    const tMessage = 'testMessage';\n    const bgTest = new BitGo({ env: 'test' });\n    const derivationPath = 'm/0';\n\n    const reqCommitment = {\n      bitgo: bgTest,\n      body: {\n        bitgoGpgPubKey: bitgoGpgKey.public,\n        txRequest: {\n          apiVersion: 'full',\n          walletId: walletID,\n          transactions: [\n            {\n              unsignedTx: {\n                derivationPath,\n                signableHex: tMessage,\n              },\n            },\n          ],\n        },\n      },\n      params: {\n        coin: 'tsol',\n        sharetype: 'commitment',\n      },\n      config: {\n        signerFileSystemPath: 'signerFileSystemPath',\n      },\n    } as unknown as express.Request;\n    const cResult = await handleV2GenerateShareTSS(reqCommitment);\n    cResult.should.have.property('userToBitgoCommitment');\n    cResult.should.have.property('encryptedSignerShare');\n    cResult.should.have.property('encryptedUserToBitgoRShare');\n    const encryptedUserToBitgoRShare = cResult.encryptedUserToBitgoRShare;\n    const reqR = {\n      bitgo: bgTest,\n      body: {\n        txRequest: {\n          apiVersion: 'full',\n          walletId: walletID,\n          transactions: [\n            {\n              unsignedTx: {\n                derivationPath,\n                signableHex: tMessage,\n              },\n            },\n          ],\n        },\n        encryptedUserToBitgoRShare,\n      },\n      params: {\n        coin: 'tsol',\n        sharetype: 'R',\n      },\n      config: {\n        signerFileSystemPath: 'signerFileSystemPath',\n      },\n    } as unknown as express.Request;\n    const rResult = await handleV2GenerateShareTSS(reqR);\n    rResult.should.have.property('rShare');\n\n    const signingKey = MPC.keyDerive(\n      userSigningMaterial.uShare,\n      [userSigningMaterial.bitgoYShare, userSigningMaterial.backupYShare],\n      derivationPath\n    );\n\n    const bitgoCombine = MPC.keyCombine(bitgo.uShare, [signingKey.yShares[3], backup.yShares[3]]);\n    const bitgoSignShare = await MPC.signShare(Buffer.from(tMessage, 'hex'), bitgoCombine.pShare, [\n      bitgoCombine.jShares[1],\n    ]);\n    const signatureShareRec = {\n      from: SignatureShareType.BITGO,\n      to: SignatureShareType.USER,\n      share: bitgoSignShare.rShares[1].r + bitgoSignShare.rShares[1].R,\n    };\n    const bitgoToUserCommitmentShare = {\n      from: SignatureShareType.BITGO,\n      to: SignatureShareType.USER,\n      share: bitgoSignShare.rShares[1].commitment,\n      type: 'commitment',\n    };\n    const reqG = {\n      bitgo: bgTest,\n      body: {\n        txRequest: {\n          apiVersion: 'full',\n          walletId: walletID,\n          transactions: [\n            {\n              unsignedTx: {\n                derivationPath,\n                signableHex: tMessage,\n              },\n            },\n          ],\n        },\n        userToBitgoRShare: rResult.rShare,\n        bitgoToUserRShare: signatureShareRec,\n        bitgoToUserCommitment: bitgoToUserCommitmentShare,\n      },\n      params: {\n        coin: 'tsol',\n        sharetype: 'G',\n      },\n      config: {\n        signerFileSystemPath: 'signerFileSystemPath',\n      },\n    } as unknown as express.Request;\n    const userGShare = await handleV2GenerateShareTSS(reqG);\n    userGShare.should.have.property('i');\n    userGShare.should.have.property('y');\n    userGShare.should.have.property('gamma');\n    userGShare.should.have.property('R');\n    const userToBitgoRShare = {\n      i: ShareKeyPosition.BITGO,\n      j: ShareKeyPosition.USER,\n      u: signingKey.yShares[3].u,\n      v: rResult.rShare.rShares[3].v,\n      r: rResult.rShare.rShares[3].r,\n      R: rResult.rShare.rShares[3].R,\n      commitment: rResult.rShare.rShares[3].commitment,\n    };\n    const bitgoGShare = MPC.sign(\n      Buffer.from(tMessage, 'hex'),\n      bitgoSignShare.xShare,\n      [userToBitgoRShare],\n      [backup.yShares[3]]\n    );\n    const signature = MPC.signCombine([userGShare, bitgoGShare]);\n    const veriResult = MPC.verify(Buffer.from(tMessage, 'hex'), signature);\n    veriResult.should.be.true();\n    readFileStub.restore();\n    envStub.restore();\n  });\n\n  it('should read an encrypted prv from signerFileSystemPath and pass it to MPCv2Round1, MPCv2Round2 and MPCv2Round3 share generators', async () => {\n    const walletID = '62fe536a6b4cf70007acb48c0e7bb0b0';\n    const tMessage = 'testMessage';\n    const derivationPath = 'm/0';\n    const walletPassphrase = 'testPass';\n\n    const [userShare, backupShare, bitgoShare] = await DklsUtils.generateDKGKeyShares();\n    assert(backupShare, 'backupShare is not defined');\n\n    const bgTest = new BitGo({ env: 'test' });\n    const userKeyShare = userShare.getKeyShare().toString('base64');\n    const validPrv = bgTest.encrypt({ input: userKeyShare, password: walletPassphrase });\n    const readFileStub = sinon.stub(fs.promises, 'readFile').resolves(JSON.stringify({ [walletID]: validPrv }));\n    const envStub = sinon.stub(process, 'env').value({ ['WALLET_' + walletID + '_PASSPHRASE']: walletPassphrase });\n\n    // round 1\n    const reqMPCv2Round1 = {\n      bitgo: bgTest,\n      body: {\n        txRequest: {\n          txRequestId: '123456',\n          apiVersion: 'full',\n          walletId: walletID,\n          transactions: [\n            {\n              unsignedTx: {\n                derivationPath,\n                signableHex: tMessage,\n              },\n              signatureShares: [],\n            },\n          ],\n        },\n      },\n      params: {\n        coin: 'hteth',\n        sharetype: 'MPCv2Round1',\n      },\n      config: {\n        signerFileSystemPath: 'signerFileSystemPath',\n      },\n    } as unknown as express.Request;\n    const round1Result = await handleV2GenerateShareTSS(reqMPCv2Round1);\n    round1Result.should.have.property('signatureShareRound1');\n    round1Result.should.have.property('userGpgPubKey');\n    round1Result.should.have.property('encryptedRound1Session');\n    round1Result.should.have.property('encryptedUserGpgPrvKey');\n\n    const hashFn = createKeccakHash('keccak256') as Hash;\n    const hashBuffer = hashFn.update(Buffer.from(tMessage, 'hex')).digest();\n    const bitgoSession = new DklsDsg.Dsg(bitgoShare.getKeyShare(), 2, derivationPath, hashBuffer);\n\n    const txRequestRound1 = await signBitgoMPCv2Round1(\n      bitgoSession,\n      reqMPCv2Round1.body.txRequest,\n      round1Result.signatureShareRound1,\n      round1Result.userGpgPubKey\n    );\n    assert(\n      txRequestRound1.transactions &&\n        txRequestRound1.transactions.length === 1 &&\n        txRequestRound1.transactions[0].signatureShares.length === 2,\n      'txRequestRound2.transactions is not an array of length 1 with 2 signatureShares'\n    );\n\n    // round 2\n    const reqMPCv2Round2 = {\n      bitgo: bgTest,\n      body: {\n        txRequest: txRequestRound1,\n        encryptedRound1Session: round1Result.encryptedRound1Session,\n        encryptedUserGpgPrvKey: round1Result.encryptedUserGpgPrvKey,\n        bitgoPublicGpgKey: bitgoGpgKey.public,\n      },\n      params: {\n        coin: 'hteth',\n        sharetype: 'MPCv2Round2',\n      },\n      config: {\n        signerFileSystemPath: 'signerFileSystemPath',\n      },\n    } as unknown as express.Request;\n    const round2Result = await handleV2GenerateShareTSS(reqMPCv2Round2);\n    round2Result.should.have.property('signatureShareRound2');\n    round2Result.should.have.property('encryptedRound2Session');\n\n    const { txRequest: txRequestRound2, bitgoMsg4 } = await signBitgoMPCv2Round2(\n      bitgoSession,\n      reqMPCv2Round2.body.txRequest,\n      round2Result.signatureShareRound2,\n      round1Result.userGpgPubKey\n    );\n    assert(\n      txRequestRound2.transactions &&\n        txRequestRound2.transactions.length === 1 &&\n        txRequestRound2.transactions[0].signatureShares.length === 4,\n      'txRequestRound2.transactions is not an array of length 1 with 4 signatureShares'\n    );\n\n    // round 3\n    const reqMPCv2Round3 = {\n      bitgo: bgTest,\n      body: {\n        txRequest: txRequestRound2,\n        encryptedRound2Session: round2Result.encryptedRound2Session,\n        encryptedUserGpgPrvKey: round1Result.encryptedUserGpgPrvKey,\n        bitgoPublicGpgKey: bitgoGpgKey.public,\n      },\n      params: {\n        coin: 'hteth',\n        sharetype: 'MPCv2Round3',\n      },\n      config: {\n        signerFileSystemPath: 'signerFileSystemPath',\n      },\n    } as unknown as express.Request;\n    const round3Result = await handleV2GenerateShareTSS(reqMPCv2Round3);\n    round3Result.should.have.property('signatureShareRound3');\n\n    const { userMsg4 } = await signBitgoMPCv2Round3(\n      bitgoSession,\n      round3Result.signatureShareRound3,\n      round1Result.userGpgPubKey\n    );\n\n    // signature generation and validation\n    assert(userMsg4.data.msg4.signatureR === bitgoMsg4.signatureR, 'User and BitGo signaturesR do not match');\n\n    const deserializedBitgoMsg4 = DklsTypes.deserializeMessages({\n      p2pMessages: [],\n      broadcastMessages: [bitgoMsg4],\n    });\n\n    const deserializedUserMsg4 = DklsTypes.deserializeMessages({\n      p2pMessages: [],\n      broadcastMessages: [\n        {\n          from: userMsg4.data.msg4.from,\n          payload: userMsg4.data.msg4.message,\n        },\n      ],\n    });\n\n    const combinedSigUsingUtil = DklsUtils.combinePartialSignatures(\n      [deserializedUserMsg4.broadcastMessages[0].payload, deserializedBitgoMsg4.broadcastMessages[0].payload],\n      Buffer.from(userMsg4.data.msg4.signatureR, 'base64').toString('hex')\n    );\n\n    const convertedSignature = DklsUtils.verifyAndConvertDklsSignature(\n      Buffer.from(tMessage, 'hex'),\n      combinedSigUsingUtil,\n      DklsTypes.getCommonKeychain(userShare.getKeyShare()),\n      derivationPath,\n      createKeccakHash('keccak256') as Hash\n    );\n    assert(convertedSignature, 'Signature is not valid');\n    assert(convertedSignature.split(':').length === 4, 'Signature is not valid');\n    readFileStub.restore();\n    envStub.restore();\n  });\n\n  it('should accept a local secret and password for a wallet', async () => {\n    const accessToken = '';\n    const walletIds = {\n      tbtc: [\n        {\n          walletId,\n          walletPassword,\n          secret,\n        },\n      ],\n    };\n\n    const walletResult = {\n      walletId,\n      keys: [walletId, walletId, walletId],\n    };\n\n    const keyResult = {\n      walletId,\n    };\n\n    nock(bgUrl).get(`/api/v2/tbtc/wallet/${walletId}`).reply(200, walletResult);\n    nock(bgUrl).get(`/api/v2/tbtc/key/${walletId}`).reply(200, keyResult);\n\n    const data = await fetchKeys(walletIds, accessToken);\n\n    should.exist(data[walletId]);\n    data[walletId].should.startWith('{\"iv\":\"');\n  });\n});\n\n// #region MPCv2 utils\nfunction getBitGoPartyGpgKeyPrv(bitgoPrvKey: string): DklsTypes.PartyGpgKey {\n  return {\n    partyId: 2,\n    gpgKey: bitgoPrvKey,\n  };\n}\n\nfunction getUserPartyGpgKeyPublic(userPubKey: string): DklsTypes.PartyGpgKey {\n  return {\n    partyId: 0,\n    gpgKey: userPubKey,\n  };\n}\n\nasync function signBitgoMPCv2Round1(\n  bitgoSession: DklsDsg.Dsg,\n  txRequest: TxRequest,\n  userShare: SignatureShareRecord,\n  userGPGPubKey: string\n): Promise<TxRequest> {\n  assert(\n    txRequest.transactions && txRequest.transactions.length === 1,\n    'txRequest.transactions is not an array of length 1'\n  );\n  txRequest.transactions[0].signatureShares.push(userShare);\n  // Do the actual signing on BitGo's side based on User's messages\n  const signatureShare = JSON.parse(userShare.share) as MPCv2SignatureShareRound1Input;\n  const deserializedMessages = DklsTypes.deserializeMessages({\n    p2pMessages: [],\n    broadcastMessages: [\n      {\n        from: signatureShare.data.msg1.from,\n        payload: signatureShare.data.msg1.message,\n      },\n    ],\n  });\n  const bitgoToUserRound1BroadcastMsg = await bitgoSession.init();\n  const bitgoToUserRound2Msg = bitgoSession.handleIncomingMessages({\n    p2pMessages: [],\n    broadcastMessages: deserializedMessages.broadcastMessages,\n  });\n  const serializedBitGoToUserRound1And2Msgs = DklsTypes.serializeMessages({\n    p2pMessages: bitgoToUserRound2Msg.p2pMessages,\n    broadcastMessages: [bitgoToUserRound1BroadcastMsg],\n  });\n\n  const authEncMessages = await DklsComms.encryptAndAuthOutgoingMessages(\n    serializedBitGoToUserRound1And2Msgs,\n    [getUserPartyGpgKeyPublic(userGPGPubKey)],\n    [getBitGoPartyGpgKeyPrv(bitgoGpgKey.private)]\n  );\n\n  const bitgoToUserSignatureShare: MPCv2SignatureShareRound1Output = {\n    type: 'round1Output',\n    data: {\n      msg1: {\n        from: authEncMessages.broadcastMessages[0].from as MPCv2PartyFromStringOrNumber,\n        signature: authEncMessages.broadcastMessages[0].payload.signature,\n        message: authEncMessages.broadcastMessages[0].payload.message,\n      },\n      msg2: {\n        from: authEncMessages.p2pMessages[0].from as MPCv2PartyFromStringOrNumber,\n        to: authEncMessages.p2pMessages[0].to as MPCv2PartyFromStringOrNumber,\n        encryptedMessage: authEncMessages.p2pMessages[0].payload.encryptedMessage,\n        signature: authEncMessages.p2pMessages[0].payload.signature,\n      },\n    },\n  };\n  txRequest.transactions[0].signatureShares.push({\n    from: SignatureShareType.BITGO,\n    to: SignatureShareType.USER,\n    share: JSON.stringify(bitgoToUserSignatureShare),\n  });\n  return txRequest;\n}\n\nasync function signBitgoMPCv2Round2(\n  bitgoSession: DklsDsg.Dsg,\n  txRequest: TxRequest,\n  userShare: SignatureShareRecord,\n  userGPGPubKey: string\n): Promise<{ txRequest: TxRequest; bitgoMsg4: DklsTypes.SerializedBroadcastMessage }> {\n  assert(\n    txRequest.transactions && txRequest.transactions.length === 1,\n    'txRequest.transactions is not an array of length 1'\n  );\n  txRequest.transactions[0].signatureShares.push(userShare);\n\n  // Do the actual signing on BitGo's side based on User's messages\n  const parsedSignatureShare = JSON.parse(userShare.share) as MPCv2SignatureShareRound2Input;\n  const serializedMessages = await DklsComms.decryptAndVerifyIncomingMessages(\n    {\n      p2pMessages: [\n        {\n          from: parsedSignatureShare.data.msg2.from,\n          to: parsedSignatureShare.data.msg2.to,\n          payload: {\n            encryptedMessage: parsedSignatureShare.data.msg2.encryptedMessage,\n            signature: parsedSignatureShare.data.msg2.signature,\n          },\n        },\n        {\n          from: parsedSignatureShare.data.msg3.from,\n          to: parsedSignatureShare.data.msg3.to,\n          payload: {\n            encryptedMessage: parsedSignatureShare.data.msg3.encryptedMessage,\n            signature: parsedSignatureShare.data.msg3.signature,\n          },\n        },\n      ],\n      broadcastMessages: [],\n    },\n    [getUserPartyGpgKeyPublic(userGPGPubKey)],\n    [getBitGoPartyGpgKeyPrv(bitgoGpgKey.private)]\n  );\n  const deserializedMessages2 = DklsTypes.deserializeMessages({\n    p2pMessages: [serializedMessages.p2pMessages[0]],\n    broadcastMessages: [],\n  });\n\n  const bitgoToUserRound3Msg = bitgoSession.handleIncomingMessages(deserializedMessages2);\n  const serializedBitGoToUserRound3Msgs = DklsTypes.serializeMessages(bitgoToUserRound3Msg);\n\n  const authEncMessages = await DklsComms.encryptAndAuthOutgoingMessages(\n    serializedBitGoToUserRound3Msgs,\n    [getUserPartyGpgKeyPublic(userGPGPubKey)],\n    [getBitGoPartyGpgKeyPrv(bitgoGpgKey.private)]\n  );\n\n  const bitgoToUserSignatureShare: MPCv2SignatureShareRound2Output = {\n    type: 'round2Output',\n    data: {\n      msg3: {\n        from: authEncMessages.p2pMessages[0].from as MPCv2PartyFromStringOrNumber,\n        to: authEncMessages.p2pMessages[0].to as MPCv2PartyFromStringOrNumber,\n        encryptedMessage: authEncMessages.p2pMessages[0].payload.encryptedMessage,\n        signature: authEncMessages.p2pMessages[0].payload.signature,\n      },\n    },\n  };\n\n  // handling user msg3 but not returning bitgo msg4 since its stored on bitgo side only\n  const deserializedMessages3 = DklsTypes.deserializeMessages({\n    p2pMessages: [serializedMessages.p2pMessages[1]],\n    broadcastMessages: [],\n  });\n  const deserializedBitgoMsg4 = bitgoSession.handleIncomingMessages(deserializedMessages3);\n  const serializedBitGoToUserRound4Msgs = DklsTypes.serializeMessages(deserializedBitgoMsg4);\n\n  txRequest.transactions[0].signatureShares.push({\n    from: SignatureShareType.BITGO,\n    to: SignatureShareType.USER,\n    share: JSON.stringify(bitgoToUserSignatureShare),\n  });\n  return { txRequest, bitgoMsg4: serializedBitGoToUserRound4Msgs.broadcastMessages[0] };\n}\n\nasync function signBitgoMPCv2Round3(\n  bitgoSession: DklsDsg.Dsg,\n  userShare: SignatureShareRecord,\n  userGPGPubKey: string\n): Promise<{ userMsg4: MPCv2SignatureShareRound3Input }> {\n  const parsedSignatureShare = JSON.parse(userShare.share) as MPCv2SignatureShareRound3Input;\n  const serializedMessages = await DklsComms.decryptAndVerifyIncomingMessages(\n    {\n      p2pMessages: [],\n      broadcastMessages: [\n        {\n          from: parsedSignatureShare.data.msg4.from,\n          payload: {\n            message: parsedSignatureShare.data.msg4.message,\n            signature: parsedSignatureShare.data.msg4.signature,\n          },\n        },\n      ],\n    },\n    [getUserPartyGpgKeyPublic(userGPGPubKey)],\n    [getBitGoPartyGpgKeyPrv(bitgoGpgKey.private)]\n  );\n  const deserializedMessages = DklsTypes.deserializeMessages({\n    p2pMessages: [],\n    broadcastMessages: [serializedMessages.broadcastMessages[0]],\n  });\n  bitgoSession.handleIncomingMessages(deserializedMessages);\n\n  return {\n    userMsg4: parsedSignatureShare,\n  };\n}\n// #endregion\n"]}

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


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