PHP WebShell
Текущая директория: /opt/BitGoJS/modules/bitgo/dist/test/v2/unit/internal/tssUtils/ecdsaMPCv2
Просмотр файла: signTxRequest.js
"use strict";
Object.defineProperty(exports, "__esModule", { value: true });
exports.getBitGoPartyGpgKeyPrv = getBitGoPartyGpgKeyPrv;
exports.getUserPartyGpgKeyPublic = getUserPartyGpgKeyPublic;
const sdk_core_1 = require("@bitgo/sdk-core");
const sdk_lib_mpc_1 = require("@bitgo/sdk-lib-mpc");
const fs = require("fs");
const common_1 = require("../common");
const openpgp = require("openpgp");
const nock = require("nock");
const sdk_test_1 = require("@bitgo/sdk-test");
const src_1 = require("../../../../../../src");
const createKeccakHash = require('keccak');
describe('signTxRequest:', function () {
let tssUtils;
let wallet;
let bitgo;
let baseCoin;
let bitgoGpgKey;
const coinName = 'hteth';
const reqId = new sdk_core_1.RequestTracer();
const txRequestId = 'randomTxReqId';
const signableHex = 'e27aecaea559fbedc9ae8a22b0ab6654c2d686403c2aeb434b302545c94eed3b';
const txRequest = {
txRequestId,
enterpriseId: '4517abfb-f567-4b7a-9f91-407509d29403',
transactions: [
{
unsignedTx: {
serializedTxHex: 'TOO MANY SECRETS',
signableHex,
derivationPath: 'm/0', // Needs this when key derivation is supported
},
state: 'pendingSignature',
signatureShares: [],
},
],
unsignedTxs: [],
date: new Date().toISOString(),
intent: {
intentType: 'payment',
},
latest: true,
state: 'pendingUserSignature',
walletType: 'hot',
walletId: 'walletId',
policiesChecked: true,
version: 1,
userId: 'userId',
apiVersion: 'full',
};
const txRequestForMessageSigning = {
txRequestId,
enterpriseId: '4517abfb-f567-4b7a-9f91-407509d29403',
messages: [
{
messageRaw: 'TOO MANY SECRETS',
derivationPath: 'm/0',
state: 'pendingSignature',
signatureShares: [],
},
],
unsignedTxs: [],
date: new Date().toISOString(),
intent: {
intentType: 'payment',
},
latest: true,
state: 'pendingUserSignature',
walletType: 'hot',
walletId: 'walletId',
policiesChecked: true,
version: 1,
userId: 'userId',
apiVersion: 'full',
};
const vector = {
party1: 0,
party2: 2,
party3: 1,
};
// To generate the fixtures, run DKG as in the dklsDkg.ts tests and save the resulting party.getKeyShare in a file by doing fs.writeSync(party.getKeyShare()).
const shareFiles = [
`${__dirname}/fixtures/userShare`,
`${__dirname}/fixtures/backupShare`,
`${__dirname}/fixtures/bitgoShare`,
];
let bitgoParty;
before(async () => {
bitgo = sdk_test_1.TestBitGo.decorate(src_1.BitGo, { env: 'mock' });
bitgo.initializeTestVars();
const bgUrl = sdk_core_1.common.Environments[bitgo.getEnv()].uri;
bitgoGpgKey = await openpgp.generateKey({
userIDs: [
{
name: 'bitgo',
},
],
curve: 'secp256k1',
config: {
rejectCurves: new Set(),
},
});
const constants = {
mpc: {
bitgoPublicKey: bitgoGpgKey.publicKey,
bitgoMPCv2PublicKey: bitgoGpgKey.publicKey,
},
};
nock(bgUrl).get('/api/v1/client/constants').times(20).reply(200, { ttl: 3600, constants });
baseCoin = bitgo.coin(coinName);
let hashFn;
try {
hashFn = baseCoin.getHashFunction();
}
catch (err) {
hashFn = createKeccakHash('keccak256');
}
const hashBuffer = hashFn.update(Buffer.from(signableHex, 'hex')).digest();
// Nock out both the user and bitgo side responses to create valid signatures
bitgoParty = new sdk_lib_mpc_1.DklsDsg.Dsg(fs.readFileSync(shareFiles[vector.party2]), vector.party2, txRequest.transactions[0].unsignedTx.derivationPath, hashBuffer);
// // Round 1 ////
const walletData = {
id: txRequest.walletId,
enterprise: txRequest.enterpriseId,
coin: coinName,
coinSpecific: {},
multisigType: 'tss',
multisigTypeVersion: 'MPCv2',
};
wallet = new sdk_core_1.Wallet(bitgo, baseCoin, walletData);
tssUtils = new sdk_core_1.ECDSAUtils.EcdsaMPCv2Utils(bitgo, baseCoin, wallet);
});
beforeEach(async function () {
await nockGetBitgoPublicKeyBasedOnFeatureFlags(coinName, txRequest.enterpriseId, bitgoGpgKey);
});
after(function () {
nock.cleanAll();
});
afterEach(function () {
bitgoParty.endSession();
nock.cleanAll();
});
it('successfully signs a txRequest with user key for a dkls hot wallet with WP', async function () {
const nockPromises = [
await nockTxRequestResponseSignatureShareRoundOne(bitgoParty, txRequest, bitgoGpgKey),
await nockTxRequestResponseSignatureShareRoundTwo(bitgoParty, txRequest, bitgoGpgKey),
await nockTxRequestResponseSignatureShareRoundThree(txRequest),
await nockSendTxRequest(txRequest),
];
await Promise.all(nockPromises);
const userShare = fs.readFileSync(shareFiles[vector.party1]);
const userPrvBase64 = Buffer.from(userShare).toString('base64');
await tssUtils.signTxRequest({
txRequest,
prv: userPrvBase64,
reqId,
});
nockPromises[0].isDone().should.be.true();
nockPromises[1].isDone().should.be.true();
nockPromises[2].isDone().should.be.true();
});
it('successfully signs a txRequest with backup key for a dkls hot wallet with WP', async function () {
const nockPromises = [
await nockTxRequestResponseSignatureShareRoundOne(bitgoParty, txRequest, bitgoGpgKey, 1),
await nockTxRequestResponseSignatureShareRoundTwo(bitgoParty, txRequest, bitgoGpgKey, 1),
await nockTxRequestResponseSignatureShareRoundThree(txRequest),
await nockSendTxRequest(txRequest),
];
await Promise.all(nockPromises);
const backupShare = fs.readFileSync(shareFiles[vector.party3]);
const backupPrvBase64 = Buffer.from(backupShare).toString('base64');
await tssUtils.signTxRequest({
txRequest,
prv: backupPrvBase64,
mpcv2PartyId: 1,
reqId,
});
nockPromises[0].isDone().should.be.true();
nockPromises[1].isDone().should.be.true();
nockPromises[2].isDone().should.be.true();
});
it('successfully signs a txRequest with a message for a dkls hot wallet with WP', async function () {
const nockPromises = [
await nockTxRequestResponseSignatureShareRoundOne(bitgoParty, txRequestForMessageSigning, bitgoGpgKey),
await nockTxRequestResponseSignatureShareRoundTwo(bitgoParty, txRequestForMessageSigning, bitgoGpgKey),
await nockTxRequestResponseSignatureShareRoundThree(txRequestForMessageSigning),
await nockSendTxRequest(txRequestForMessageSigning),
];
await Promise.all(nockPromises);
const userShare = fs.readFileSync(shareFiles[vector.party1]);
const userPrvBase64 = Buffer.from(userShare).toString('base64');
await tssUtils.signTxRequest({
txRequest,
prv: userPrvBase64,
reqId,
});
nockPromises[0].isDone().should.be.true();
nockPromises[1].isDone().should.be.true();
nockPromises[2].isDone().should.be.true();
});
it('successfully signs a txRequest for a dkls hot wallet after receiving multiple 429 errors', async function () {
const nockPromises = [
await nockTxRequestResponseSignatureShareRoundOne(bitgoParty, txRequest, bitgoGpgKey),
await nockTxRequestResponseSignatureShareRoundTwo(bitgoParty, txRequest, bitgoGpgKey, 0, 3),
await nockTxRequestResponseSignatureShareRoundThree(txRequest),
await nockSendTxRequest(txRequest),
];
await Promise.all(nockPromises);
const userShare = fs.readFileSync(shareFiles[vector.party1]);
const userPrvBase64 = Buffer.from(userShare).toString('base64');
await tssUtils.signTxRequest({
txRequest,
prv: userPrvBase64,
reqId,
});
nockPromises[0].isDone().should.be.true();
nockPromises[1].isDone().should.be.true();
nockPromises[2].isDone().should.be.true();
});
it('fails to signs a txRequest for a dkls hot wallet after receiving over 3 429 errors', async function () {
const nockPromises = [
await nockTxRequestResponseSignatureShareRoundOne(bitgoParty, txRequest, bitgoGpgKey),
await nockTxRequestResponseSignatureShareRoundTwo(bitgoParty, txRequest, bitgoGpgKey, 0, 4),
];
await Promise.all(nockPromises);
const userShare = fs.readFileSync(shareFiles[vector.party1]);
const userPrvBase64 = Buffer.from(userShare).toString('base64');
await tssUtils
.signTxRequest({
txRequest,
prv: userPrvBase64,
reqId,
})
.should.be.rejectedWith('Too many requests, slow down!');
nockPromises[0].isDone().should.be.true();
nockPromises[1].isDone().should.be.false();
});
});
function getBitGoPartyGpgKeyPrv(key) {
return {
partyId: 2,
gpgKey: key.privateKey,
};
}
function getUserPartyGpgKeyPublic(userPubKey, partyId = 0) {
return {
partyId: partyId,
gpgKey: userPubKey,
};
}
async function nockTxRequestResponseSignatureShareRoundOne(bitgoSession, txRequest, bitgoGpgKey, partyId = 0) {
const transactions = (0, common_1.getRoute)('ecdsa');
return nock('https://bitgo.fakeurl')
.persist(true)
.post(`/api/v2/wallet/${txRequest.walletId}/txrequests/${txRequest.txRequestId + transactions}/sign`, (body) => JSON.parse(body.signatureShares[0].share).type === 'round1Input')
.times(1)
.reply(200, async (uri, body) => {
// Do the actual signing on BitGo's side based on User's messages
const signatureShare = JSON.parse(body.signatureShares[0].share);
const deserializedMessages = sdk_lib_mpc_1.DklsTypes.deserializeMessages({
p2pMessages: [],
broadcastMessages: [
{
from: signatureShare.data.msg1.from,
payload: signatureShare.data.msg1.message,
},
],
});
if (signatureShare.type === 'round1Input') {
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(body.signerGpgPublicKey, partyId)], [getBitGoPartyGpgKeyPrv(bitgoGpgKey)]);
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,
},
},
};
return {
txRequestId: txRequest.txRequestId,
transactions: [
{
signatureShares: [
{
from: sdk_core_1.SignatureShareType.BITGO,
to: partyId === 0 ? sdk_core_1.SignatureShareType.USER : sdk_core_1.SignatureShareType.BACKUP,
share: JSON.stringify(bitgoToUserSignatureShare),
},
],
},
],
};
}
});
}
async function nockTxRequestResponseSignatureShareRoundTwo(bitgoSession, txRequest, bitgoGpgKey, partyId = 0, rateLimitErrorCount = 0) {
const transactions = (0, common_1.getRoute)('ecdsa');
const scope = nock('https://bitgo.fakeurl');
if (rateLimitErrorCount > 0) {
scope
.post(`/api/v2/wallet/${txRequest.walletId}/txrequests/${txRequest.txRequestId + transactions}/sign`, (body) => JSON.parse(body.signatureShares[0].share).type === 'round2Input')
.times(rateLimitErrorCount)
.reply(429, {
error: 'Too many requests, slow down!',
name: 'TooManyRequests',
requestId: 'cm5qx01lh0013b2ek2sxl4w00',
context: {},
});
}
return scope
.post(`/api/v2/wallet/${txRequest.walletId}/txrequests/${txRequest.txRequestId + transactions}/sign`, (body) => JSON.parse(body.signatureShares[0].share).type === 'round2Input')
.times(1)
.reply(200, async (uri, body) => {
// Do the actual signing on BitGo's side based on User's messages
const parsedSignatureShare = JSON.parse(body.signatureShares[0].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(body.signerGpgPublicKey, partyId)], [getBitGoPartyGpgKeyPrv(bitgoGpgKey)]);
const deserializedMessages = sdk_lib_mpc_1.DklsTypes.deserializeMessages({
p2pMessages: [serializedMessages.p2pMessages[0]],
broadcastMessages: [],
});
if (parsedSignatureShare.type === 'round2Input') {
const bitgoToUserRound3Msg = bitgoSession.handleIncomingMessages(deserializedMessages);
const serializedBitGoToUserRound3Msgs = sdk_lib_mpc_1.DklsTypes.serializeMessages(bitgoToUserRound3Msg);
const authEncMessages = await sdk_lib_mpc_1.DklsComms.encryptAndAuthOutgoingMessages(serializedBitGoToUserRound3Msgs, [getUserPartyGpgKeyPublic(body.signerGpgPublicKey, partyId)], [getBitGoPartyGpgKeyPrv(bitgoGpgKey)]);
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,
},
},
};
return {
txRequestId: txRequest.txRequestId,
transactions: [
{
signatureShares: [
{
from: partyId === 0 ? sdk_core_1.SignatureShareType.USER : sdk_core_1.SignatureShareType.BACKUP,
to: sdk_core_1.SignatureShareType.BITGO,
share: 'some old share we dont care about',
},
{
from: sdk_core_1.SignatureShareType.BITGO,
to: partyId === 0 ? sdk_core_1.SignatureShareType.USER : sdk_core_1.SignatureShareType.BACKUP,
share: JSON.stringify(bitgoToUserSignatureShare),
},
],
},
],
};
}
});
}
async function nockTxRequestResponseSignatureShareRoundThree(txRequest) {
const transactions = (0, common_1.getRoute)('ecdsa');
return nock('https://bitgo.fakeurl')
.post(`/api/v2/wallet/${txRequest.walletId}/txrequests/${txRequest.txRequestId + transactions}/sign`, (body) => JSON.parse(body.signatureShares[0].share).type === 'round3Input')
.times(1)
.reply(200, async (uri, body) => {
// Do the actual signing on BitGo's side based on User's messages
return {
txRequestId: txRequest.txRequestId,
};
});
}
async function nockSendTxRequest(txRequest) {
const transactions = (0, common_1.getRoute)('ecdsa');
return nock('https://bitgo.fakeurl')
.post(`/api/v2/wallet/${txRequest.walletId}/txrequests/${txRequest.txRequestId + transactions}/send`)
.times(1)
.reply(200, {
txRequestId: txRequest.txRequestId,
});
}
async function nockGetBitgoPublicKeyBasedOnFeatureFlags(coin, enterpriseId, bitgoGpgKeyPair) {
const bitgoGPGPublicKeyResponse = {
name: 'irrelevant',
publicKey: bitgoGpgKeyPair.publicKey,
mpcv2PublicKey: bitgoGpgKeyPair.publicKey,
enterpriseId,
};
nock('https://bitgo.fakeurl')
.get(`/api/v2/${coin}/tss/pubkey`)
.times(4)
.query({ enterpriseId })
.reply(200, bitgoGPGPublicKeyResponse);
return bitgoGPGPublicKeyResponse;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"signTxRequest.js","sourceRoot":"","sources":["../../../../../../../test/v2/unit/internal/tssUtils/ecdsaMPCv2/signTxRequest.ts"],"names":[],"mappings":"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","sourcesContent":["import { Hash } from 'crypto';\nimport {\n  BaseCoin,\n  BitgoGPGPublicKey,\n  common,\n  ECDSAUtils,\n  RequestTracer,\n  SignatureShareRecord,\n  SignatureShareType,\n  TxRequest,\n  Wallet,\n} from '@bitgo/sdk-core';\nimport { DklsDsg, DklsTypes, DklsComms } from '@bitgo/sdk-lib-mpc';\nimport * as fs from 'fs';\nimport { getRoute } from '../common';\nimport {\n  MPCv2SignatureShareRound1Output,\n  MPCv2SignatureShareRound1Input,\n  MPCv2SignatureShareRound2Input,\n  MPCv2SignatureShareRound2Output,\n  MPCv2SignatureShareRound3Input,\n  MPCv2PartyFromStringOrNumber,\n} from '@bitgo/public-types';\nimport * as openpgp from 'openpgp';\nimport * as nock from 'nock';\nimport { TestableBG, TestBitGo } from '@bitgo/sdk-test';\nimport { BitGo } from '../../../../../../src';\nconst createKeccakHash = require('keccak');\n\ninterface SignatureShareApiBody {\n  signatureShares: SignatureShareRecord[];\n  signerGpgPublicKey: string;\n}\n\ndescribe('signTxRequest:', function () {\n  let tssUtils: ECDSAUtils.EcdsaMPCv2Utils;\n  let wallet: Wallet;\n  let bitgo: TestableBG & BitGo;\n  let baseCoin: BaseCoin;\n  let bitgoGpgKey: openpgp.SerializedKeyPair<string>;\n  const coinName = 'hteth';\n\n  const reqId = new RequestTracer();\n  const txRequestId = 'randomTxReqId';\n  const signableHex = 'e27aecaea559fbedc9ae8a22b0ab6654c2d686403c2aeb434b302545c94eed3b';\n  const txRequest: TxRequest = {\n    txRequestId,\n    enterpriseId: '4517abfb-f567-4b7a-9f91-407509d29403',\n    transactions: [\n      {\n        unsignedTx: {\n          serializedTxHex: 'TOO MANY SECRETS',\n          signableHex,\n          derivationPath: 'm/0', // Needs this when key derivation is supported\n        },\n        state: 'pendingSignature',\n        signatureShares: [],\n      },\n    ],\n    unsignedTxs: [],\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    apiVersion: 'full',\n  };\n\n  const txRequestForMessageSigning: TxRequest = {\n    txRequestId,\n    enterpriseId: '4517abfb-f567-4b7a-9f91-407509d29403',\n    messages: [\n      {\n        messageRaw: 'TOO MANY SECRETS',\n        derivationPath: 'm/0',\n        state: 'pendingSignature',\n        signatureShares: [],\n      },\n    ],\n    unsignedTxs: [],\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    apiVersion: 'full',\n  };\n\n  const vector = {\n    party1: 0,\n    party2: 2,\n    party3: 1,\n  };\n  // To generate the fixtures, run DKG as in the dklsDkg.ts tests and save the resulting party.getKeyShare in a file by doing fs.writeSync(party.getKeyShare()).\n  const shareFiles = [\n    `${__dirname}/fixtures/userShare`,\n    `${__dirname}/fixtures/backupShare`,\n    `${__dirname}/fixtures/bitgoShare`,\n  ];\n\n  let bitgoParty: DklsDsg.Dsg;\n\n  before(async () => {\n    bitgo = TestBitGo.decorate(BitGo, { env: 'mock' });\n    bitgo.initializeTestVars();\n    const bgUrl = common.Environments[bitgo.getEnv()].uri;\n    bitgoGpgKey = await openpgp.generateKey({\n      userIDs: [\n        {\n          name: 'bitgo',\n        },\n      ],\n      curve: 'secp256k1',\n      config: {\n        rejectCurves: new Set(),\n      },\n    });\n    const constants = {\n      mpc: {\n        bitgoPublicKey: bitgoGpgKey.publicKey,\n        bitgoMPCv2PublicKey: bitgoGpgKey.publicKey,\n      },\n    };\n    nock(bgUrl).get('/api/v1/client/constants').times(20).reply(200, { ttl: 3600, constants });\n\n    baseCoin = bitgo.coin(coinName);\n\n    let hashFn: Hash;\n    try {\n      hashFn = baseCoin.getHashFunction();\n    } catch (err) {\n      hashFn = createKeccakHash('keccak256') as Hash;\n    }\n    const hashBuffer = hashFn.update(Buffer.from(signableHex, 'hex')).digest();\n\n    // Nock out both the user and bitgo side responses to create valid signatures\n    bitgoParty = new DklsDsg.Dsg(\n      fs.readFileSync(shareFiles[vector.party2]),\n      vector.party2,\n      txRequest.transactions![0].unsignedTx.derivationPath,\n      hashBuffer\n    );\n    // // Round 1 ////\n    const walletData = {\n      id: txRequest.walletId,\n      enterprise: txRequest.enterpriseId,\n      coin: coinName,\n      coinSpecific: {},\n      multisigType: 'tss',\n      multisigTypeVersion: 'MPCv2',\n    };\n    wallet = new Wallet(bitgo, baseCoin, walletData);\n    tssUtils = new ECDSAUtils.EcdsaMPCv2Utils(bitgo, baseCoin, wallet);\n  });\n\n  beforeEach(async function () {\n    await nockGetBitgoPublicKeyBasedOnFeatureFlags(coinName, txRequest.enterpriseId!, bitgoGpgKey);\n  });\n\n  after(function () {\n    nock.cleanAll();\n  });\n\n  afterEach(function () {\n    bitgoParty.endSession();\n    nock.cleanAll();\n  });\n\n  it('successfully signs a txRequest with user key for a dkls hot wallet with WP', async function () {\n    const nockPromises = [\n      await nockTxRequestResponseSignatureShareRoundOne(bitgoParty, txRequest, bitgoGpgKey),\n      await nockTxRequestResponseSignatureShareRoundTwo(bitgoParty, txRequest, bitgoGpgKey),\n      await nockTxRequestResponseSignatureShareRoundThree(txRequest),\n      await nockSendTxRequest(txRequest),\n    ];\n    await Promise.all(nockPromises);\n\n    const userShare = fs.readFileSync(shareFiles[vector.party1]);\n    const userPrvBase64 = Buffer.from(userShare).toString('base64');\n    await tssUtils.signTxRequest({\n      txRequest,\n      prv: userPrvBase64,\n      reqId,\n    });\n    nockPromises[0].isDone().should.be.true();\n    nockPromises[1].isDone().should.be.true();\n    nockPromises[2].isDone().should.be.true();\n  });\n\n  it('successfully signs a txRequest with backup key for a dkls hot wallet with WP', async function () {\n    const nockPromises = [\n      await nockTxRequestResponseSignatureShareRoundOne(bitgoParty, txRequest, bitgoGpgKey, 1),\n      await nockTxRequestResponseSignatureShareRoundTwo(bitgoParty, txRequest, bitgoGpgKey, 1),\n      await nockTxRequestResponseSignatureShareRoundThree(txRequest),\n      await nockSendTxRequest(txRequest),\n    ];\n    await Promise.all(nockPromises);\n\n    const backupShare = fs.readFileSync(shareFiles[vector.party3]);\n    const backupPrvBase64 = Buffer.from(backupShare).toString('base64');\n    await tssUtils.signTxRequest({\n      txRequest,\n      prv: backupPrvBase64,\n      mpcv2PartyId: 1,\n      reqId,\n    });\n    nockPromises[0].isDone().should.be.true();\n    nockPromises[1].isDone().should.be.true();\n    nockPromises[2].isDone().should.be.true();\n  });\n\n  it('successfully signs a txRequest with a message for a dkls hot wallet with WP', async function () {\n    const nockPromises = [\n      await nockTxRequestResponseSignatureShareRoundOne(bitgoParty, txRequestForMessageSigning, bitgoGpgKey),\n      await nockTxRequestResponseSignatureShareRoundTwo(bitgoParty, txRequestForMessageSigning, bitgoGpgKey),\n      await nockTxRequestResponseSignatureShareRoundThree(txRequestForMessageSigning),\n      await nockSendTxRequest(txRequestForMessageSigning),\n    ];\n    await Promise.all(nockPromises);\n\n    const userShare = fs.readFileSync(shareFiles[vector.party1]);\n    const userPrvBase64 = Buffer.from(userShare).toString('base64');\n    await tssUtils.signTxRequest({\n      txRequest,\n      prv: userPrvBase64,\n      reqId,\n    });\n    nockPromises[0].isDone().should.be.true();\n    nockPromises[1].isDone().should.be.true();\n    nockPromises[2].isDone().should.be.true();\n  });\n\n  it('successfully signs a txRequest for a dkls hot wallet after receiving multiple 429 errors', async function () {\n    const nockPromises = [\n      await nockTxRequestResponseSignatureShareRoundOne(bitgoParty, txRequest, bitgoGpgKey),\n      await nockTxRequestResponseSignatureShareRoundTwo(bitgoParty, txRequest, bitgoGpgKey, 0, 3),\n      await nockTxRequestResponseSignatureShareRoundThree(txRequest),\n      await nockSendTxRequest(txRequest),\n    ];\n    await Promise.all(nockPromises);\n\n    const userShare = fs.readFileSync(shareFiles[vector.party1]);\n    const userPrvBase64 = Buffer.from(userShare).toString('base64');\n    await tssUtils.signTxRequest({\n      txRequest,\n      prv: userPrvBase64,\n      reqId,\n    });\n    nockPromises[0].isDone().should.be.true();\n    nockPromises[1].isDone().should.be.true();\n    nockPromises[2].isDone().should.be.true();\n  });\n\n  it('fails to signs a txRequest for a dkls hot wallet after receiving over 3 429 errors', async function () {\n    const nockPromises = [\n      await nockTxRequestResponseSignatureShareRoundOne(bitgoParty, txRequest, bitgoGpgKey),\n      await nockTxRequestResponseSignatureShareRoundTwo(bitgoParty, txRequest, bitgoGpgKey, 0, 4),\n    ];\n    await Promise.all(nockPromises);\n\n    const userShare = fs.readFileSync(shareFiles[vector.party1]);\n    const userPrvBase64 = Buffer.from(userShare).toString('base64');\n    await tssUtils\n      .signTxRequest({\n        txRequest,\n        prv: userPrvBase64,\n        reqId,\n      })\n      .should.be.rejectedWith('Too many requests, slow down!');\n    nockPromises[0].isDone().should.be.true();\n    nockPromises[1].isDone().should.be.false();\n  });\n});\n\nexport function getBitGoPartyGpgKeyPrv(key: openpgp.SerializedKeyPair<string>): DklsTypes.PartyGpgKey {\n  return {\n    partyId: 2,\n    gpgKey: key.privateKey,\n  };\n}\n\nexport function getUserPartyGpgKeyPublic(userPubKey: string, partyId: 0 | 1 = 0): DklsTypes.PartyGpgKey {\n  return {\n    partyId: partyId,\n    gpgKey: userPubKey,\n  };\n}\n\nasync function nockTxRequestResponseSignatureShareRoundOne(\n  bitgoSession: DklsDsg.Dsg,\n  txRequest: TxRequest,\n  bitgoGpgKey: openpgp.SerializedKeyPair<string>,\n  partyId: 0 | 1 = 0\n): Promise<nock.Scope> {\n  const transactions = getRoute('ecdsa');\n  return nock('https://bitgo.fakeurl')\n    .persist(true)\n    .post(\n      `/api/v2/wallet/${txRequest.walletId}/txrequests/${txRequest.txRequestId + transactions}/sign`,\n      (body) => (JSON.parse(body.signatureShares[0].share) as MPCv2SignatureShareRound1Input).type === 'round1Input'\n    )\n    .times(1)\n    .reply(200, async (uri, body: SignatureShareApiBody) => {\n      // Do the actual signing on BitGo's side based on User's messages\n      const signatureShare = JSON.parse(body.signatureShares[0].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      if (signatureShare.type === 'round1Input') {\n        const bitgoToUserRound1BroadcastMsg = await bitgoSession.init();\n\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(body.signerGpgPublicKey, partyId)],\n          [getBitGoPartyGpgKeyPrv(bitgoGpgKey)]\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        return {\n          txRequestId: txRequest.txRequestId,\n          transactions: [\n            {\n              signatureShares: [\n                {\n                  from: SignatureShareType.BITGO,\n                  to: partyId === 0 ? SignatureShareType.USER : SignatureShareType.BACKUP,\n                  share: JSON.stringify(bitgoToUserSignatureShare),\n                },\n              ],\n            },\n          ],\n        };\n      }\n    });\n}\n\nasync function nockTxRequestResponseSignatureShareRoundTwo(\n  bitgoSession: DklsDsg.Dsg,\n  txRequest: TxRequest,\n  bitgoGpgKey: openpgp.SerializedKeyPair<string>,\n  partyId: 0 | 1 = 0,\n  rateLimitErrorCount = 0\n): Promise<nock.Scope> {\n  const transactions = getRoute('ecdsa');\n  const scope = nock('https://bitgo.fakeurl');\n\n  if (rateLimitErrorCount > 0) {\n    scope\n      .post(\n        `/api/v2/wallet/${txRequest.walletId}/txrequests/${txRequest.txRequestId + transactions}/sign`,\n        (body) => (JSON.parse(body.signatureShares[0].share) as MPCv2SignatureShareRound2Input).type === 'round2Input'\n      )\n      .times(rateLimitErrorCount)\n      .reply(429, {\n        error: 'Too many requests, slow down!',\n        name: 'TooManyRequests',\n        requestId: 'cm5qx01lh0013b2ek2sxl4w00',\n        context: {},\n      });\n  }\n  return scope\n    .post(\n      `/api/v2/wallet/${txRequest.walletId}/txrequests/${txRequest.txRequestId + transactions}/sign`,\n      (body) => (JSON.parse(body.signatureShares[0].share) as MPCv2SignatureShareRound2Input).type === 'round2Input'\n    )\n    .times(1)\n    .reply(200, async (uri, body: SignatureShareApiBody) => {\n      // Do the actual signing on BitGo's side based on User's messages\n      const parsedSignatureShare = JSON.parse(body.signatureShares[0].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(body.signerGpgPublicKey, partyId)],\n        [getBitGoPartyGpgKeyPrv(bitgoGpgKey)]\n      );\n      const deserializedMessages = DklsTypes.deserializeMessages({\n        p2pMessages: [serializedMessages.p2pMessages[0]],\n        broadcastMessages: [],\n      });\n      if (parsedSignatureShare.type === 'round2Input') {\n        const bitgoToUserRound3Msg = bitgoSession.handleIncomingMessages(deserializedMessages);\n        const serializedBitGoToUserRound3Msgs = DklsTypes.serializeMessages(bitgoToUserRound3Msg);\n\n        const authEncMessages = await DklsComms.encryptAndAuthOutgoingMessages(\n          serializedBitGoToUserRound3Msgs,\n          [getUserPartyGpgKeyPublic(body.signerGpgPublicKey, partyId)],\n          [getBitGoPartyGpgKeyPrv(bitgoGpgKey)]\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        return {\n          txRequestId: txRequest.txRequestId,\n          transactions: [\n            {\n              signatureShares: [\n                {\n                  from: partyId === 0 ? SignatureShareType.USER : SignatureShareType.BACKUP,\n                  to: SignatureShareType.BITGO,\n                  share: 'some old share we dont care about',\n                },\n                {\n                  from: SignatureShareType.BITGO,\n                  to: partyId === 0 ? SignatureShareType.USER : SignatureShareType.BACKUP,\n                  share: JSON.stringify(bitgoToUserSignatureShare),\n                },\n              ],\n            },\n          ],\n        };\n      }\n    });\n}\n\nasync function nockTxRequestResponseSignatureShareRoundThree(txRequest: TxRequest): Promise<nock.Scope> {\n  const transactions = getRoute('ecdsa');\n  return nock('https://bitgo.fakeurl')\n    .post(\n      `/api/v2/wallet/${txRequest.walletId}/txrequests/${txRequest.txRequestId + transactions}/sign`,\n      (body: SignatureShareApiBody) =>\n        (JSON.parse(body.signatureShares[0].share) as MPCv2SignatureShareRound3Input).type === 'round3Input'\n    )\n    .times(1)\n    .reply(200, async (uri, body) => {\n      // Do the actual signing on BitGo's side based on User's messages\n\n      return {\n        txRequestId: txRequest.txRequestId,\n      };\n    });\n}\n\nasync function nockSendTxRequest(txRequest: TxRequest): Promise<nock.Scope> {\n  const transactions = getRoute('ecdsa');\n  return nock('https://bitgo.fakeurl')\n    .post(`/api/v2/wallet/${txRequest.walletId}/txrequests/${txRequest.txRequestId + transactions}/send`)\n    .times(1)\n    .reply(200, {\n      txRequestId: txRequest.txRequestId,\n    });\n}\n\nasync function nockGetBitgoPublicKeyBasedOnFeatureFlags(\n  coin: string,\n  enterpriseId: string,\n  bitgoGpgKeyPair: openpgp.SerializedKeyPair<string>\n): Promise<BitgoGPGPublicKey> {\n  const bitgoGPGPublicKeyResponse: BitgoGPGPublicKey = {\n    name: 'irrelevant',\n    publicKey: bitgoGpgKeyPair.publicKey,\n    mpcv2PublicKey: bitgoGpgKeyPair.publicKey,\n    enterpriseId,\n  };\n  nock('https://bitgo.fakeurl')\n    .get(`/api/v2/${coin}/tss/pubkey`)\n    .times(4)\n    .query({ enterpriseId })\n    .reply(200, bitgoGPGPublicKeyResponse);\n  return bitgoGPGPublicKeyResponse;\n}\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!