PHP WebShell
Текущая директория: /opt/BitGoJS/modules/sdk-core/dist/src/bitgo/utils/tss/ecdsa
Просмотр файла: ecdsa.js
"use strict";
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
var desc = Object.getOwnPropertyDescriptor(m, k);
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
desc = { enumerable: true, get: function() { return m[k]; } };
}
Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
if (k2 === undefined) k2 = k;
o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
o["default"] = v;
});
var __importStar = (this && this.__importStar) || (function () {
var ownKeys = function(o) {
ownKeys = Object.getOwnPropertyNames || function (o) {
var ar = [];
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
return ar;
};
return ownKeys(o);
};
return function (mod) {
if (mod && mod.__esModule) return mod;
var result = {};
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
__setModuleDefault(result, mod);
return result;
};
})();
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.EcdsaUtils = void 0;
const assert_1 = __importDefault(require("assert"));
const buffer_1 = require("buffer");
const openpgp = __importStar(require("openpgp"));
const sdk_lib_mpc_1 = require("@bitgo/sdk-lib-mpc");
const utxo_lib_1 = require("@bitgo/utxo-lib");
const tss_1 = require("../../../../account-lib/mpc/tss");
const ecdsa_1 = __importDefault(require("../../../tss/ecdsa"));
const baseTypes_1 = require("../baseTypes");
const tss_2 = require("../../../tss");
const types_1 = require("../../../tss/ecdsa/types");
const opengpgUtils_1 = require("../../opengpgUtils");
const ecdsa_2 = require("../../../tss/ecdsa/ecdsa");
const ecdh_1 = require("../../../ecdh");
const common_1 = require("../../../tss/common");
const types_2 = require("../../../tss/types");
const base_1 = require("./base");
const encryptNShare = ecdsa_1.default.encryptNShare;
/** @inheritdoc */
class EcdsaUtils extends base_1.BaseEcdsaUtils {
async finalizeBitgoHeldBackupKeyShare(keyId, commonKeychain, userKeyShare, bitgoKeychain, userGpgKey, thirdPartyBackupPublicGpgKey) {
const encryptedUserToBackupShare = await encryptNShare(userKeyShare, 2, thirdPartyBackupPublicGpgKey.armor(), userGpgKey);
const bitgoToBackupKeyShare = bitgoKeychain.keyShares?.find((keyShare) => keyShare.from === 'bitgo' && keyShare.to === 'backup');
const userPublicShare = buffer_1.Buffer.concat([
buffer_1.Buffer.from(userKeyShare.nShares[2].y, 'hex'),
buffer_1.Buffer.from(userKeyShare.nShares[2].chaincode, 'hex'),
]).toString('hex');
(0, assert_1.default)(bitgoToBackupKeyShare);
const keyResponse = await this.bitgo
.put(this.baseCoin.url(`/krs/backupkeys/${keyId}`))
.send({
commonKeychain,
keyShares: [
{
from: 'user',
to: 'backup',
publicShare: userPublicShare,
privateShare: encryptedUserToBackupShare.encryptedPrivateShare,
privateShareProof: encryptedUserToBackupShare.privateShareProof,
vssProof: encryptedUserToBackupShare.vssProof,
},
bitgoToBackupKeyShare,
],
})
.result();
if (!keyResponse || !keyResponse.commonKeychain) {
throw new Error('Failed backup key verification.');
}
return {
id: keyResponse.id,
keyShares: keyResponse.keyShares,
commonKeychain: keyResponse.commonKeychain,
};
}
/** @inheritdoc */
async createKeychains(params) {
const MPC = new tss_1.Ecdsa();
const m = 2;
const n = 3;
const userKeyShare = await MPC.keyShare(1, m, n);
const userGpgKey = await (0, opengpgUtils_1.generateGPGKeyPair)('secp256k1');
const backupKeyShare = await this.createBackupKeyShares();
const backupGpgKey = await this.getBackupGpgPubKey();
// Get the BitGo public key based on user/enterprise feature flags
// If it doesn't work, use the default public key from the constants
const bitgoPublicGpgKey = (await this.getBitgoGpgPubkeyBasedOnFeatureFlags(params.enterprise)) ?? this.bitgoPublicGpgKey;
const bitgoKeychain = await this.createBitgoKeychain({
userGpgKey,
backupGpgKey,
bitgoPublicGpgKey,
userKeyShare,
backupKeyShare,
enterprise: params.enterprise,
});
const userKeychainPromise = this.createUserKeychain({
userGpgKey,
backupGpgKey,
bitgoPublicGpgKey,
userKeyShare,
backupKeyShare,
bitgoKeychain,
passphrase: params.passphrase,
originalPasscodeEncryptionCode: params.originalPasscodeEncryptionCode,
});
const backupKeychainPromise = this.createBackupKeychain({
userGpgKey,
backupGpgKey,
bitgoPublicGpgKey,
userKeyShare,
backupKeyShare,
bitgoKeychain,
passphrase: params.passphrase,
});
const [userKeychain, backupKeychain] = await Promise.all([userKeychainPromise, backupKeychainPromise]);
return {
userKeychain,
backupKeychain,
bitgoKeychain,
};
}
async createBackupKeyShares() {
const MPC = new tss_1.Ecdsa();
const m = 2;
const n = 3;
const backupKeyShare = {
userHeldKeyShare: await MPC.keyShare(2, m, n),
};
return backupKeyShare;
}
createUserKeychain({ userGpgKey, backupGpgKey, bitgoPublicGpgKey, userKeyShare, backupKeyShare, bitgoKeychain, passphrase, originalPasscodeEncryptionCode, }) {
if (!passphrase) {
throw new Error('Please provide a wallet passphrase');
}
(0, assert_1.default)(backupKeyShare.userHeldKeyShare);
return this.createParticipantKeychain(userGpgKey, backupGpgKey, bitgoPublicGpgKey, 1, userKeyShare, backupKeyShare.userHeldKeyShare, bitgoKeychain, passphrase, originalPasscodeEncryptionCode);
}
async createBackupKeychain({ userGpgKey, userKeyShare, backupGpgKey, backupKeyShare, bitgoKeychain, bitgoPublicGpgKey, passphrase, }) {
(0, assert_1.default)(backupKeyShare.userHeldKeyShare);
(0, assert_1.default)(passphrase);
return this.createParticipantKeychain(userGpgKey, backupGpgKey, bitgoPublicGpgKey, 2, userKeyShare, backupKeyShare.userHeldKeyShare, bitgoKeychain, passphrase);
}
/** @inheritdoc */
async createBitgoKeychain({ userGpgKey, backupGpgKey, userKeyShare, backupKeyShare, enterprise, bitgoPublicGpgKey, }) {
const recipientIndex = 3;
const userToBitgoShare = await encryptNShare(userKeyShare, recipientIndex, bitgoPublicGpgKey.armor(), userGpgKey);
const backupToBitgoShare = await this.getBackupEncryptedNShare(backupKeyShare, recipientIndex, bitgoPublicGpgKey.armor(), backupGpgKey);
const createBitGoMPCParams = {
keyType: 'tss',
source: 'bitgo',
keyShares: [
{
from: 'user',
to: 'bitgo',
publicShare: userToBitgoShare.publicShare,
privateShare: userToBitgoShare.encryptedPrivateShare,
n: userToBitgoShare.n,
vssProof: userToBitgoShare.vssProof,
privateShareProof: userToBitgoShare.privateShareProof,
},
{
from: 'backup',
to: 'bitgo',
publicShare: backupToBitgoShare.publicShare,
privateShare: backupToBitgoShare.encryptedPrivateShare,
n: backupToBitgoShare.n,
vssProof: backupToBitgoShare.vssProof,
privateShareProof: backupToBitgoShare.privateShareProof,
},
],
userGPGPublicKey: userGpgKey.publicKey,
backupGPGPublicKey: backupGpgKey.publicKey,
enterprise: enterprise,
algoUsed: 'ecdsa',
};
return await this.baseCoin.keychains().add(createBitGoMPCParams);
}
/**
* This builds the relevant backup encryptedNShare based on whether the
* backup key is user or third party generated
* @param backupShare can either have key shares from the user or third party
* @param recipientIndex index of the party receiving the backup shares
* @param recipientGpgPublicArmor gpg armor of the party receiving the backup shares
* @param backupGpgKey backup gpg key
* @param isThirdPartyBackup whether the backup is generated by third party
*/
async getBackupEncryptedNShare(backupShare, recipientIndex, recipientGpgPublicArmor, backupGpgKey) {
(0, assert_1.default)(backupShare.userHeldKeyShare);
const backupToRecipientShare = await encryptNShare(backupShare.userHeldKeyShare, recipientIndex, recipientGpgPublicArmor, backupGpgKey);
return backupToRecipientShare;
}
/** @inheritdoc */
async createParticipantKeychain(userGpgKey, userLocalBackupGpgKey, bitgoPublicGpgKey, recipientIndex, userKeyShare, backupKeyShare, bitgoKeychain, passphrase, originalPasscodeEncryptionCode) {
const bitgoKeyShares = bitgoKeychain.keyShares;
if (!bitgoKeyShares) {
throw new Error('Missing BitGo key shares');
}
if (!bitgoKeychain.commonKeychain) {
throw new Error(`Missing common key chain: ${bitgoKeychain.commonKeychain}`);
}
let recipient;
let keyShare;
let otherShare;
let recipientGpgKey;
let senderGpgKey;
if (recipientIndex === 1) {
keyShare = userKeyShare;
otherShare = backupKeyShare;
recipient = 'user';
recipientGpgKey = userGpgKey;
senderGpgKey = userLocalBackupGpgKey;
}
else if (recipientIndex === 2) {
keyShare = backupKeyShare;
otherShare = userKeyShare;
recipient = 'backup';
recipientGpgKey = userLocalBackupGpgKey;
senderGpgKey = userGpgKey;
}
else {
throw new Error('Invalid user index');
}
const bitGoToRecipientShare = bitgoKeyShares.find((keyShare) => keyShare.from === 'bitgo' && keyShare.to === recipient);
if (!bitGoToRecipientShare) {
throw new Error(`Missing BitGo to ${recipient} key share`);
}
const decryptedShare = await this.decryptPrivateShare(bitGoToRecipientShare.privateShare, recipientGpgKey);
await this.verifyWalletSignatures(userGpgKey.publicKey, userLocalBackupGpgKey.publicKey, bitgoKeychain, decryptedShare, recipientIndex);
const senderToRecipientShare = await encryptNShare(otherShare, recipientIndex, recipientGpgKey.publicKey, senderGpgKey);
const encryptedNShares = [
{
// userToBackup or backupToUser
nShare: senderToRecipientShare,
recipientPrivateArmor: recipientGpgKey.privateKey,
senderPublicArmor: senderGpgKey.publicKey,
},
{
// bitgoToRecipient
nShare: {
i: recipientIndex,
j: 3,
publicShare: bitGoToRecipientShare.publicShare,
encryptedPrivateShare: bitGoToRecipientShare.privateShare,
n: bitGoToRecipientShare.n,
vssProof: bitGoToRecipientShare.vssProof,
privateShareProof: bitGoToRecipientShare.privateShareProof,
},
recipientPrivateArmor: recipientGpgKey.privateKey,
senderPublicArmor: bitgoPublicGpgKey.armor(),
isbs58Encoded: false,
},
];
const recipientCombinedKey = await ecdsa_1.default.createCombinedKey(keyShare, encryptedNShares, bitgoKeychain.commonKeychain);
const prv = JSON.stringify(recipientCombinedKey.signingMaterial);
const recipientKeychainParams = {
source: recipient,
keyType: 'tss',
commonKeychain: bitgoKeychain.commonKeychain,
prv: prv,
encryptedPrv: this.bitgo.encrypt({
input: prv,
password: passphrase,
}),
originalPasscodeEncryptionCode,
};
const keychains = this.baseCoin.keychains();
return recipientIndex === 1
? await keychains.add(recipientKeychainParams)
: await keychains.createBackup(recipientKeychainParams);
}
async createTssEcdsaStep1SigningMaterial(params) {
const { challenges, derivationPath, prv } = params;
const userSigningMaterial = JSON.parse(prv);
if (userSigningMaterial.pShare.i !== 1) {
throw new Error('Invalid user key');
}
if (!userSigningMaterial.backupNShare) {
throw new Error('Invalid user key - missing backupNShare');
}
const MPC = new tss_1.Ecdsa();
const signingKey = MPC.keyDerive(userSigningMaterial.pShare, [userSigningMaterial.bitgoNShare, userSigningMaterial.backupNShare], derivationPath);
const bitgoIndex = types_2.ShareKeyPosition.BITGO;
const userIndex = userSigningMaterial.pShare.i;
const { ntilde: ntildea, h1: h1a, h2: h2a, p: pa } = challenges.enterpriseChallenge;
const { ntilde: ntildeb, h1: h1b, h2: h2b, p: pb, n: nb } = challenges.bitgoChallenge;
const userXShare = MPC.appendChallenge(signingKey.xShare, { ntilde: ntildea, h1: h1a, h2: h2a }, { p: pa });
const bitgoYShare = MPC.appendChallenge({
i: userIndex,
j: bitgoIndex,
n: nb,
}, { ntilde: ntildeb, h1: h1b, h2: h2b }, { p: pb });
const userSignShare = await ecdsa_1.default.createUserSignShare(userXShare, bitgoYShare);
const u = signingKey.nShares[bitgoIndex].u;
let chaincode = userSigningMaterial.bitgoNShare.chaincode;
while (chaincode.length < 64) {
chaincode = '0' + chaincode;
}
const signerShare = utxo_lib_1.bip32.fromPrivateKey(buffer_1.Buffer.from(u, 'hex'), buffer_1.Buffer.from(chaincode, 'hex')).toBase58();
const bitgoGpgKey = (await (0, opengpgUtils_1.getBitgoGpgPubKey)(this.bitgo)).mpcV1;
const encryptedSignerShare = (await openpgp.encrypt({
message: await openpgp.createMessage({
text: signerShare,
}),
config: {
rejectCurves: new Set(),
},
encryptionKeys: [bitgoGpgKey],
}));
const userGpgKey = await (0, opengpgUtils_1.generateGPGKeyPair)('secp256k1');
const privateShareProof = await (0, opengpgUtils_1.createShareProof)(userGpgKey.privateKey, signingKey.nShares[bitgoIndex].u, 'ecdsa');
const vssProof = signingKey.nShares[bitgoIndex].v;
const userPublicGpgKey = userGpgKey.publicKey;
const publicShare = signingKey.nShares[bitgoIndex].y + signingKey.nShares[bitgoIndex].chaincode;
return {
privateShareProof: privateShareProof,
vssProof: vssProof,
publicShare: publicShare,
encryptedSignerOffsetShare: encryptedSignerShare,
userPublicGpgKey: userPublicGpgKey,
kShare: userSignShare.kShare,
wShare: params.walletPassphrase
? this.bitgo.encrypt({ input: JSON.stringify(userSignShare.wShare), password: params.walletPassphrase })
: userSignShare.wShare,
};
}
async createTssEcdsaStep2SigningMaterial(params) {
// Append the BitGo challenge to the Ashare to be used in subsequent proofs
const bitgoToUserAShareWithNtilde = {
...params.aShareFromBitgo,
...params.bitgoChallenge,
};
const userGammaAndMuShares = await ecdsa_1.default.createUserGammaAndMuShare(params.wShare, bitgoToUserAShareWithNtilde);
const userOmicronAndDeltaShare = await ecdsa_1.default.createUserOmicronAndDeltaShare(userGammaAndMuShares.gShare);
return {
muDShare: {
muShare: userGammaAndMuShares.muShare,
dShare: userOmicronAndDeltaShare.dShare,
i: userGammaAndMuShares.muShare.i,
},
oShare: params.walletPassphrase
? this.bitgo.encrypt({
input: JSON.stringify(userOmicronAndDeltaShare.oShare),
password: params.walletPassphrase,
})
: userOmicronAndDeltaShare.oShare,
};
}
getOfflineSignerPaillierModulus(params) {
(0, assert_1.default)(params.prv, 'Params to get paillier modulus are missing prv.');
const userSigningMaterial = JSON.parse(params.prv);
return { userPaillierModulus: userSigningMaterial.pShare.n };
}
async createOfflineKShare(params) {
const { tssParams, prv, requestType, challenges } = params;
(0, assert_1.default)(typeof tssParams.txRequest !== 'string', 'Invalid txRequest type');
const txRequest = tssParams.txRequest;
let derivationPath;
if (requestType === baseTypes_1.RequestType.tx) {
(0, assert_1.default)(txRequest.transactions || txRequest.unsignedTxs, 'Unable to find transactions in txRequest');
const unsignedTx = txRequest.apiVersion === 'full' ? txRequest.transactions[0].unsignedTx : txRequest.unsignedTxs[0];
derivationPath = unsignedTx.derivationPath;
}
else if (requestType === baseTypes_1.RequestType.message) {
// TODO BG-67299 Message signing with derivation path
derivationPath = '';
}
return this.createTssEcdsaStep1SigningMaterial({
prv: prv,
challenges: challenges,
derivationPath: derivationPath,
walletPassphrase: params.walletPassphrase,
});
}
async createOfflineMuDeltaShare(params) {
const decryptedWShare = this.bitgo.decrypt({ input: params.encryptedWShare, password: params.walletPassphrase });
return await this.createTssEcdsaStep2SigningMaterial({
aShareFromBitgo: params.aShareFromBitgo,
bitgoChallenge: params.bitgoChallenge,
wShare: JSON.parse(decryptedWShare),
walletPassphrase: params.walletPassphrase,
});
}
async createOfflineSShare(params) {
const { tssParams, requestType, dShareFromBitgo, encryptedOShare, walletPassphrase } = params;
(0, assert_1.default)(typeof tssParams.txRequest !== 'string', 'Invalid txRequest type');
const txRequest = tssParams.txRequest;
let signablePayload;
if (requestType === baseTypes_1.RequestType.tx) {
(0, assert_1.default)(txRequest.transactions || txRequest.unsignedTxs, 'Unable to find transactions in txRequest');
const unsignedTx = txRequest.apiVersion === 'full' ? txRequest.transactions[0].unsignedTx : txRequest.unsignedTxs[0];
signablePayload = buffer_1.Buffer.from(unsignedTx.signableHex, 'hex');
}
else if (requestType === baseTypes_1.RequestType.message) {
signablePayload = params.tssParams.bufferToSign;
}
let hash;
try {
hash = this.baseCoin.getHashFunction();
}
catch (err) {
hash = undefined;
}
const decryptedOShare = this.bitgo.decrypt({ input: encryptedOShare, password: walletPassphrase });
const { i, R, s, y } = await ecdsa_1.default.createUserSignatureShare(JSON.parse(decryptedOShare), dShareFromBitgo, signablePayload, hash);
// return only required SShare without bigints from VAShare
return {
i,
R,
s,
y,
};
}
async signEcdsaTssUsingExternalSigner(params, requestType, externalSignerPaillierModulusGetter, externalSignerKShareGenerator, externalSignerMuDeltaShareGenerator, externalSignerSShareGenerator) {
const { txRequest } = params;
const pendingEcdsaTssInitialization = this.wallet.coinSpecific()?.pendingEcdsaTssInitialization;
if (pendingEcdsaTssInitialization) {
throw new Error('Wallet is not ready for TSS ECDSA signing. Please contact your enterprise admin to finish the enterprise TSS initialization.');
}
const txRequestObj = await (0, tss_2.getTxRequest)(this.bitgo, this.wallet.id(), txRequest, params.reqId);
const { userPaillierModulus } = await externalSignerPaillierModulusGetter({ txRequest: txRequestObj });
const { enterpriseChallenge, bitgoChallenge } = await this.getEcdsaSigningChallenges(txRequest, requestType, userPaillierModulus, 0, params.reqId);
const step1SigningMaterial = await externalSignerKShareGenerator({
tssParams: {
...params,
txRequest: txRequestObj,
},
challenges: { enterpriseChallenge, bitgoChallenge },
requestType: requestType,
});
// signing stage one with K share send to bitgo and receives A share
const bitgoToUserAShare = (await ecdsa_1.default.sendShareToBitgo(this.bitgo, this.wallet.id(), txRequestObj.txRequestId, requestType, types_1.SendShareType.KShare, step1SigningMaterial.kShare, step1SigningMaterial.encryptedSignerOffsetShare, step1SigningMaterial.vssProof, step1SigningMaterial.privateShareProof, step1SigningMaterial.publicShare, step1SigningMaterial.userPublicGpgKey, params.reqId)); // WP/HSM does not return the initial challenge
const step2Return = await externalSignerMuDeltaShareGenerator({
txRequest: txRequestObj,
aShareFromBitgo: bitgoToUserAShare,
bitgoChallenge: bitgoChallenge,
encryptedWShare: step1SigningMaterial.wShare,
});
// signing stage two with muShare and dShare send to bitgo and receives D share
const bitgoToUserDShare = (await ecdsa_1.default.sendShareToBitgo(this.bitgo, this.wallet.id(), txRequestObj.txRequestId, requestType, types_1.SendShareType.MUShare, step2Return.muDShare, undefined, undefined, undefined, undefined, undefined, params.reqId));
const userSShare = await externalSignerSShareGenerator({
tssParams: {
...params,
txRequest: txRequestObj,
},
dShareFromBitgo: bitgoToUserDShare,
requestType: requestType,
encryptedOShare: step2Return.oShare,
});
// signing stage three with SShare send to bitgo and receives SShare
await ecdsa_1.default.sendShareToBitgo(this.bitgo, this.wallet.id(), txRequestObj.txRequestId, requestType, types_1.SendShareType.SShare, userSShare, undefined, undefined, undefined, undefined, undefined, params.reqId);
return await (0, tss_2.getTxRequest)(this.bitgo, this.wallet.id(), txRequestObj.txRequestId, params.reqId);
}
/**
* Gets signing key, txRequestResolved and txRequestId
* @param {string | TxRequest} params.txRequest - transaction request object or id
* @param {string} params.prv - decrypted private key
* @param { string} params.reqId - request id
* @returns {Promise<TxRequest>}
*/
async signRequestBase(params, requestType) {
const pendingEcdsaTssInitialization = this.wallet.coinSpecific()?.pendingEcdsaTssInitialization;
if (pendingEcdsaTssInitialization) {
throw new Error('Wallet is not ready for TSS ECDSA signing. Please contact your enterprise admin to finish the enterprise TSS initialization.');
}
const userSigningMaterial = JSON.parse(params.prv);
if (userSigningMaterial.pShare.i !== 1) {
throw new Error('Invalid user key');
}
if (!userSigningMaterial.backupNShare) {
throw new Error('Invalid user key - missing backupNShare');
}
const txRequest = typeof params.txRequest === 'string'
? await (0, tss_2.getTxRequest)(this.bitgo, this.wallet.id(), params.txRequest, params.reqId)
: params.txRequest;
let signablePayload = new buffer_1.Buffer('');
let derivationPath = '';
if (requestType === baseTypes_1.RequestType.tx) {
(0, assert_1.default)(txRequest.transactions || txRequest.unsignedTxs, 'Unable to find transactions in txRequest');
const unsignedTx = txRequest.apiVersion === 'full' ? txRequest.transactions[0].unsignedTx : txRequest.unsignedTxs[0];
// For ICP transactions, the HSM signs the serializedTxHex, while the user signs the signableHex separately.
// Verification cannot be performed directly on the signableHex alone. However, we can parse the serializedTxHex
// to regenerate the signableHex and compare it against the provided value for verification.
// In contrast, for other coin families, verification is typically done using just the signableHex.
if (this.baseCoin.getConfig().family === 'icp') {
await this.baseCoin.verifyTransaction({
txPrebuild: { txHex: unsignedTx.serializedTxHex, txInfo: unsignedTx.signableHex },
txParams: params.txParams || { recipients: [] },
wallet: this.wallet,
walletType: this.wallet.multisigType(),
});
}
else {
await this.baseCoin.verifyTransaction({
txPrebuild: { txHex: unsignedTx.signableHex },
txParams: params.txParams || { recipients: [] },
wallet: this.wallet,
walletType: this.wallet.multisigType(),
});
}
signablePayload = buffer_1.Buffer.from(unsignedTx.signableHex, 'hex');
derivationPath = unsignedTx.derivationPath;
}
else if (requestType === baseTypes_1.RequestType.message) {
signablePayload = params.bufferToSign;
// TODO BG-67299 Message signing with derivation path
}
const paillierModulus = this.getOfflineSignerPaillierModulus({ prv: params.prv });
const challenges = await this.getEcdsaSigningChallenges(txRequest.txRequestId, requestType, paillierModulus.userPaillierModulus, 0, params.reqId);
const step1Return = await this.createTssEcdsaStep1SigningMaterial({
prv: params.prv,
challenges: challenges,
derivationPath: derivationPath,
});
// signing stage one with K share send to bitgo and receives A share
const bitgoToUserAShare = (await ecdsa_1.default.sendShareToBitgo(this.bitgo, this.wallet.id(), txRequest.txRequestId, requestType, types_1.SendShareType.KShare, step1Return.kShare, step1Return.encryptedSignerOffsetShare, step1Return.vssProof, step1Return.privateShareProof, step1Return.publicShare, step1Return.userPublicGpgKey, params.reqId)); // WP/HSM does not return the initial challenge
const step2Return = await this.createTssEcdsaStep2SigningMaterial({
aShareFromBitgo: bitgoToUserAShare,
bitgoChallenge: challenges.bitgoChallenge,
wShare: step1Return.wShare,
});
// signing stage two with muShare and dShare send to bitgo and receives D share
const bitgoToUserDShare = (await ecdsa_1.default.sendShareToBitgo(this.bitgo, this.wallet.id(), txRequest.txRequestId, requestType, types_1.SendShareType.MUShare, step2Return.muDShare, undefined, undefined, undefined, undefined, undefined, params.reqId));
// If only the getHashFunction() is defined for the coin use it otherwise
// pass undefined hash, default hash will be used in that case.
let hash;
try {
hash = this.baseCoin.getHashFunction();
}
catch (err) {
hash = undefined;
}
const userSShare = await ecdsa_1.default.createUserSignatureShare(step2Return.oShare, bitgoToUserDShare, signablePayload, hash);
// signing stage three with SShare send to bitgo and receives SShare
await ecdsa_1.default.sendShareToBitgo(this.bitgo, this.wallet.id(), txRequest.txRequestId, requestType, types_1.SendShareType.SShare, userSShare, undefined, undefined, undefined, undefined, undefined, params.reqId);
return await (0, tss_2.getTxRequest)(this.bitgo, this.wallet.id(), txRequest.txRequestId, params.reqId);
}
/**
* Signs the transaction associated to the transaction request.
* @param {string | TxRequest} params.txRequest - transaction request object or id
* @param {string} params.prv - decrypted private key
* @param {string} params.reqId - request id
* @returns {Promise<TxRequest>} fully signed TxRequest object
*/
async signTxRequest(params) {
this.bitgo.setRequestTracer(params.reqId);
return this.signRequestBase(params, baseTypes_1.RequestType.tx);
}
/**
* Signs the message associated to the transaction request.
* @param {string | TxRequest} params.txRequest - transaction request object or id
* @param {string} params.prv - decrypted private key
* @param {string} params.reqId - request id
* @returns {Promise<TxRequest>} fully signed TxRequest object
*/
async signTxRequestForMessage(params) {
if (!params.messageRaw) {
throw new Error('Raw message required to sign message');
}
return this.signRequestBase(params, baseTypes_1.RequestType.message);
}
/**
* Get the challenge values for enterprise and BitGo in ECDSA signing
* Only returns the challenges if they are verified by the user's enterprise admin's ecdh key
* @param {string} txRequestId - transaction request id
* @param {RequestType} requestType - (0 for tx, 1 for message)
* @param {string} walletPaillierModulus - paillier pubkey $n$
* @param {number} index - index of the requestType
* @param {IRequestTracer} reqId - request tracer request id
*/
async getEcdsaSigningChallenges(txRequestId, requestType, walletPaillierModulus, index = 0, reqId) {
const enterpriseId = this.wallet.toJSON().enterprise;
if (!enterpriseId) {
throw new Error('Wallet must be an enterprise wallet.');
}
// create BitGo range proof and paillier proof challenge
const createBitgoChallengeResponse = await (0, common_1.getTxRequestChallenge)(this.bitgo, this.wallet.id(), txRequestId, index.toString(), requestType, walletPaillierModulus, reqId);
const bitgoToEnterprisePaillierChallenge = { p: createBitgoChallengeResponse.p };
const enterpriseToBitgoPaillierChallenge = sdk_lib_mpc_1.EcdsaTypes.serializePaillierChallenge({
p: await sdk_lib_mpc_1.EcdsaPaillierProof.generateP((0, sdk_lib_mpc_1.hexToBigInt)(createBitgoChallengeResponse.n)),
});
// TODO(BG-78764): once the paillier proofs are complete, reduce challenge creation to one API call
const walletChallenges = await this.wallet.getChallengesForEcdsaSigning();
const challengeVerifierUserId = walletChallenges.createdBy;
const adminSigningKeyResponse = await this.bitgo.getSigningKeyForUser(enterpriseId, challengeVerifierUserId);
const pubkeyOfAdminEcdhKeyHex = adminSigningKeyResponse.derivedPubkey;
// Verify enterprise's challenge is signed by the respective admins ecdh keychain
const enterpriseRawChallenge = {
ntilde: walletChallenges.enterpriseChallenge.ntilde,
h1: walletChallenges.enterpriseChallenge.h1,
h2: walletChallenges.enterpriseChallenge.h2,
};
const adminSignatureOnEntChallenge = walletChallenges.enterpriseChallenge.verifiers.adminSignature;
if (!(0, ecdh_1.verifyEcdhSignature)(EcdsaUtils.getMessageToSignFromChallenge(enterpriseRawChallenge), adminSignatureOnEntChallenge, buffer_1.Buffer.from(pubkeyOfAdminEcdhKeyHex, 'hex'))) {
throw new Error(`Admin signature for enterprise challenge is not valid. Please contact your enterprise admin.`);
}
// Verify that the BitGo challenge's ZK proofs have been verified by the admin
const bitgoChallenge = {
ntilde: walletChallenges.bitgoChallenge.ntilde,
h1: walletChallenges.bitgoChallenge.h1,
h2: walletChallenges.bitgoChallenge.h2,
p: bitgoToEnterprisePaillierChallenge.p,
n: createBitgoChallengeResponse.n,
};
const adminVerificationSignatureForBitGoChallenge = walletChallenges.bitgoChallenge.verifiers.adminSignature;
if (!(0, ecdh_1.verifyEcdhSignature)(EcdsaUtils.getMessageToSignFromChallenge(bitgoChallenge), adminVerificationSignatureForBitGoChallenge, buffer_1.Buffer.from(pubkeyOfAdminEcdhKeyHex, 'hex'))) {
throw new Error(`Admin signature for BitGo's challenge is not valid. Please contact your enterprise admin.`);
}
return {
enterpriseChallenge: {
...enterpriseRawChallenge,
p: enterpriseToBitgoPaillierChallenge.p,
},
bitgoChallenge,
};
}
/**
* Verifies the u-value proofs and GPG keys used in generating a TSS ECDSA wallet.
* @param userGpgPub The user's public GPG key for encryption between user/server
* @param backupGpgPub The backup's public GPG key for encryption between backup/server
* @param bitgoKeychain previously created BitGo keychain; must be compatible with user and backup key shares
* @param decryptedShare The decrypted bitgo-to-user/backup private share retrieved from the keychain
* @param verifierIndex The index of the party to verify: 1 = user, 2 = backup
*/
async verifyWalletSignatures(userGpgPub, backupGpgPub, bitgoKeychain, decryptedShare, verifierIndex) {
(0, assert_1.default)(bitgoKeychain.commonKeychain);
(0, assert_1.default)(bitgoKeychain.walletHSMGPGPublicKeySigs);
const bitgoGpgKey = (await (0, opengpgUtils_1.getBitgoGpgPubKey)(this.bitgo)).mpcV1;
const userKeyPub = await openpgp.readKey({ armoredKey: userGpgPub });
const userKeyId = userKeyPub.keyPacket.getFingerprint();
const backupKeyPub = await openpgp.readKey({ armoredKey: backupGpgPub });
const backupKeyId = backupKeyPub.keyPacket.getFingerprint();
const walletSignatures = await openpgp.readKeys({ armoredKeys: bitgoKeychain.walletHSMGPGPublicKeySigs });
if (walletSignatures.length !== 2) {
throw new Error('Invalid wallet signatures');
}
if (userKeyId !== walletSignatures[0].keyPacket.getFingerprint()) {
throw new Error(`first wallet signature's fingerprint does not match passed user gpg key's fingerprint`);
}
if (backupKeyId !== walletSignatures[1].keyPacket.getFingerprint()) {
throw new Error(`second wallet signature's fingerprint does not match passed backup gpg key's fingerprint`);
}
await (0, ecdsa_2.verifyWalletSignature)({
walletSignature: walletSignatures[0],
commonKeychain: bitgoKeychain.commonKeychain,
userKeyId,
backupKeyId,
bitgoPub: bitgoGpgKey,
decryptedShare,
verifierIndex,
});
await (0, ecdsa_2.verifyWalletSignature)({
walletSignature: walletSignatures[1],
commonKeychain: bitgoKeychain.commonKeychain,
userKeyId,
backupKeyId,
bitgoPub: bitgoGpgKey,
decryptedShare,
verifierIndex,
});
}
/**
* Signs a challenge with the provided v1 ecdh key at a derived path
* @param challenge challenge to sign
* @param ecdhXprv xprv of the ecdh key
* @param derivationPath the derived path at which the ecdh key will sign
*/
static signChallenge(challenge, ecdhXprv, derivationPath) {
const messageToSign = this.getMessageToSignFromChallenge(challenge);
return (0, ecdh_1.signMessageWithDerivedEcdhKey)(messageToSign, ecdhXprv, derivationPath);
}
/**
* Converts challenge to a common message format which can be signed.
* @param challenge
*/
static getMessageToSignFromChallenge(challenge) {
return challenge.ntilde.concat(challenge.h1).concat(challenge.h2);
}
/**
Verifies ZK proofs of BitGo's challenges for both nitro and institutional HSMs
which are fetched from the WP API.
*/
static async verifyBitGoChallenges(bitgoChallenges) {
// Verify institutional hsm challenge proof
const instChallengeVerified = await this.verifyBitGoChallenge({
ntilde: bitgoChallenges.bitgoInstitutionalHsm.ntilde,
h1: bitgoChallenges.bitgoInstitutionalHsm.h1,
h2: bitgoChallenges.bitgoInstitutionalHsm.h2,
ntildeProof: bitgoChallenges.bitgoInstitutionalHsm.ntildeProof,
});
// Verify nitro hsm challenge proof
const nitroChallengeVerified = await this.verifyBitGoChallenge({
ntilde: bitgoChallenges.bitgoNitroHsm.ntilde,
h1: bitgoChallenges.bitgoNitroHsm.h1,
h2: bitgoChallenges.bitgoNitroHsm.h2,
ntildeProof: bitgoChallenges.bitgoNitroHsm.ntildeProof,
});
return instChallengeVerified && nitroChallengeVerified;
}
/**
* Verifies ZK proof for a single BitGo challenge
* @param bitgoChallenge
*/
static async verifyBitGoChallenge(bitgoChallenge) {
const deserializedInstChallenge = sdk_lib_mpc_1.EcdsaTypes.deserializeNtildeWithProofs(bitgoChallenge);
const ntildeProofH1WrtH2Verified = await sdk_lib_mpc_1.EcdsaRangeProof.verifyNtildeProof({
ntilde: deserializedInstChallenge.ntilde,
h1: deserializedInstChallenge.h1,
h2: deserializedInstChallenge.h2,
}, deserializedInstChallenge.ntildeProof.h1WrtH2);
const ntildeProofH2WrtH1Verified = await sdk_lib_mpc_1.EcdsaRangeProof.verifyNtildeProof({
ntilde: deserializedInstChallenge.ntilde,
h1: deserializedInstChallenge.h2,
h2: deserializedInstChallenge.h1,
}, deserializedInstChallenge.ntildeProof.h2WrtH1);
return ntildeProofH1WrtH2Verified && ntildeProofH2WrtH1Verified;
}
/**
* Gets the bitgo challenges for both nitro and institutional HSMs from WP API.
* @param bitgo
*/
static async getBitGoChallenges(bitgo) {
const res = await bitgo.get(bitgo.url('/tss/ecdsa/challenges', 2)).send().result();
if (!res.bitgoNitroHsm ||
!res.bitgoNitroHsm.ntilde ||
!res.bitgoNitroHsm.h1 ||
!res.bitgoNitroHsm.h2 ||
!res.bitgoNitroHsm.ntildeProof ||
!res.bitgoInstitutionalHsm ||
!res.bitgoInstitutionalHsm.ntilde ||
!res.bitgoInstitutionalHsm.h1 ||
!res.bitgoInstitutionalHsm.h2 ||
!res.bitgoInstitutionalHsm.ntildeProof) {
throw new Error('Expected BitGo challenge proof to be present. Contact support@bitgo.com.');
}
return res;
}
/**
* Gets BitGo's proofs from API and signs them if the proofs are valid.
* @param bitgo
* @param enterpriseId
* @param userPassword
*/
static async getVerifyAndSignBitGoChallenges(bitgo, enterpriseId, userPassword) {
// Fetch BitGo's challenge and verify
const bitgoChallengesWithProofs = await EcdsaUtils.getBitGoChallenges(bitgo);
if (!(await EcdsaUtils.verifyBitGoChallenges(bitgoChallengesWithProofs))) {
throw new Error(`Failed to verify BitGo's challenge needed to enable ECDSA signing. Please contact support@bitgo.com`);
}
return await EcdsaUtils.signBitgoChallenges(bitgo, enterpriseId, userPassword, bitgoChallengesWithProofs);
}
/**
* Sign Bitgo's proofs, verification of proofs is left to the caller
* @param bitgo
* @param enterpriseId
* @param userPassword
* @param bitgoChallengesWithProofs Optionally provide Bitgo Challaenge & Proofs instead of fetching from API
*/
static async signBitgoChallenges(bitgo, enterpriseId, userPassword, bitgoChallengesWithProofs) {
// fetch challenge & proof if none are provided
const challengesWithProofs = bitgoChallengesWithProofs
? bitgoChallengesWithProofs
: await EcdsaUtils.getBitGoChallenges(bitgo);
// Fetch user's ecdh public keychain needed for signing the challenges
const ecdhKeypair = await bitgo.getEcdhKeypairPrivate(userPassword, enterpriseId);
const signedBitGoInstChallenge = EcdsaUtils.signChallenge(challengesWithProofs.bitgoInstitutionalHsm, ecdhKeypair.xprv, ecdhKeypair.derivationPath);
const signedBitGoNitroChallenge = EcdsaUtils.signChallenge(challengesWithProofs.bitgoNitroHsm, ecdhKeypair.xprv, ecdhKeypair.derivationPath);
return {
bitgoInstHsmAdminSignature: signedBitGoInstChallenge,
bitgoNitroHsmAdminSignature: signedBitGoNitroChallenge,
};
}
/**
* This is needed to enable ecdsa signing on the enterprise.
* It receives the enterprise challenge and signatures of verified bitgo proofs
* and uploads them on the enterprise.
* @param bitgo
* @param entId - enterprise id to enable ecdsa signing on
* @param userPassword - enterprise admin's login pw
* @param bitgoInstChallengeProofSignature - signature on bitgo's institutional HSM challenge after verification
* @param bitgoNitroChallengeProofSignature - signature on bitgo's nitro HSM challenge after verification
* @param challenge - optionally use the challenge for enterprise challenge
*/
static async initiateChallengesForEnterprise(bitgo, entId, userPassword, bitgoInstChallengeProofSignature, bitgoNitroChallengeProofSignature, openSSLBytes, challenge) {
// Fetch user's ecdh public keychain needed for signing the challenges
const ecdhKeypair = await bitgo.getEcdhKeypairPrivate(userPassword, entId);
// Generate and sign enterprise challenge
const entChallengeWithProof = challenge ?? (await sdk_lib_mpc_1.EcdsaRangeProof.generateNtilde(openSSLBytes, sdk_lib_mpc_1.minModulusBitLength));
const serializedEntChallengeWithProof = sdk_lib_mpc_1.EcdsaTypes.serializeNtildeWithProofs(entChallengeWithProof);
const signedEnterpriseChallenge = EcdsaUtils.signChallenge(serializedEntChallengeWithProof, ecdhKeypair.xprv, ecdhKeypair.derivationPath);
await this.uploadChallengesToEnterprise(bitgo, entId, serializedEntChallengeWithProof, signedEnterpriseChallenge.toString('hex'), bitgoInstChallengeProofSignature.toString('hex'), bitgoNitroChallengeProofSignature.toString('hex'));
}
/**
* Uploads the signed challenges and their proofs on the enterprise.
* This initiates ecdsa signing for the enterprise users.
* @param bitgo
* @param entId - enterprise to enable ecdsa signing on
* @param entChallenge - client side generated ent challenge with ZK proofs
* @param entChallengeSignature - signature on enterprise challenge
* @param bitgoIntChallengeSignature - signature on BitGo's institutional HSM challenge
* @param bitgoNitroChallengeSignature - signature on BitGo's nitro HSM challenge
*/
static async uploadChallengesToEnterprise(bitgo, entId, entChallenge, entChallengeSignature, bitgoIntChallengeSignature, bitgoNitroChallengeSignature) {
const body = {
enterprise: {
ntilde: entChallenge.ntilde,
h1: entChallenge.h1,
h2: entChallenge.h2,
verifiers: {
adminSignature: entChallengeSignature,
},
},
bitgoInstitutionalHsm: {
verifiers: {
adminSignature: bitgoIntChallengeSignature,
},
},
bitgoNitroHsm: {
verifiers: {
adminSignature: bitgoNitroChallengeSignature,
},
},
};
if ('ntildeProof' in entChallenge) {
body.enterprise['ntildeProof'] = entChallenge.ntildeProof;
}
await bitgo
.put(bitgo.url(`/enterprise/${entId}/tssconfig/ecdsa/challenge`, 2))
.send(body)
.result();
}
}
exports.EcdsaUtils = EcdsaUtils;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ecdsa.js","sourceRoot":"","sources":["../../../../../../src/bitgo/utils/tss/ecdsa/ecdsa.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\nimport { Buffer } from 'buffer';\nimport * as openpgp from 'openpgp';\nimport { Key, SerializedKeyPair } from 'openpgp';\nimport { Hash } from 'crypto';\nimport { EcdsaPaillierProof, EcdsaRangeProof, EcdsaTypes, hexToBigInt, minModulusBitLength } from '@bitgo/sdk-lib-mpc';\nimport { bip32 } from '@bitgo/utxo-lib';\n\nimport { ECDSA, Ecdsa } from '../../../../account-lib/mpc/tss';\nimport { AddKeychainOptions, Keychain, KeyType } from '../../../keychain';\nimport ECDSAMethods, { ECDSAMethodTypes } from '../../../tss/ecdsa';\nimport { KeychainsTriplet } from '../../../baseCoin';\nimport {\n  BitGoProofSignatures,\n  CreateEcdsaBitGoKeychainParams,\n  CreateEcdsaKeychainParams,\n  DecryptableNShare,\n  GetBitGoChallengesApi,\n  KeyShare,\n} from './types';\nimport {\n  BackupKeyShare,\n  BitgoHeldBackupKeyShare,\n  CustomKShareGeneratingFunction,\n  CustomMuDeltaShareGeneratingFunction,\n  CustomPaillierModulusGetterFunction,\n  CustomSShareGeneratingFunction,\n  RequestType,\n  TSSParams,\n  TSSParamsForMessage,\n  TSSParamsForMessageWithPrv,\n  TSSParamsWithPrv,\n  TxRequest,\n} from '../baseTypes';\nimport { getTxRequest } from '../../../tss';\nimport { AShare, DShare, EncryptedNShare, SendShareType, SShare, WShare, OShare } from '../../../tss/ecdsa/types';\nimport { createShareProof, generateGPGKeyPair, getBitgoGpgPubKey } from '../../opengpgUtils';\nimport { BitGoBase } from '../../../bitgoBase';\nimport { verifyWalletSignature } from '../../../tss/ecdsa/ecdsa';\nimport { signMessageWithDerivedEcdhKey, verifyEcdhSignature } from '../../../ecdh';\nimport { getTxRequestChallenge } from '../../../tss/common';\nimport {\n  ShareKeyPosition,\n  TssEcdsaStep1ReturnMessage,\n  TssEcdsaStep2ReturnMessage,\n  TxRequestChallengeResponse,\n} from '../../../tss/types';\nimport { BaseEcdsaUtils } from './base';\nimport { IRequestTracer } from '../../../../api';\n\nconst encryptNShare = ECDSAMethods.encryptNShare;\n\n/** @inheritdoc */\nexport class EcdsaUtils extends BaseEcdsaUtils {\n  async finalizeBitgoHeldBackupKeyShare(\n    keyId: string,\n    commonKeychain: string,\n    userKeyShare: KeyShare,\n    bitgoKeychain: Keychain,\n    userGpgKey: SerializedKeyPair<string>,\n    thirdPartyBackupPublicGpgKey: Key\n  ): Promise<BitgoHeldBackupKeyShare> {\n    const encryptedUserToBackupShare = await encryptNShare(\n      userKeyShare,\n      2,\n      thirdPartyBackupPublicGpgKey.armor(),\n      userGpgKey\n    );\n    const bitgoToBackupKeyShare = bitgoKeychain.keyShares?.find(\n      (keyShare) => keyShare.from === 'bitgo' && keyShare.to === 'backup'\n    );\n    const userPublicShare = Buffer.concat([\n      Buffer.from(userKeyShare.nShares[2].y, 'hex'),\n      Buffer.from(userKeyShare.nShares[2].chaincode, 'hex'),\n    ]).toString('hex');\n    assert(bitgoToBackupKeyShare);\n    const keyResponse = await this.bitgo\n      .put(this.baseCoin.url(`/krs/backupkeys/${keyId}`))\n      .send({\n        commonKeychain,\n        keyShares: [\n          {\n            from: 'user',\n            to: 'backup',\n            publicShare: userPublicShare,\n            privateShare: encryptedUserToBackupShare.encryptedPrivateShare,\n            privateShareProof: encryptedUserToBackupShare.privateShareProof,\n            vssProof: encryptedUserToBackupShare.vssProof,\n          },\n          bitgoToBackupKeyShare,\n        ],\n      })\n      .result();\n    if (!keyResponse || !keyResponse.commonKeychain) {\n      throw new Error('Failed backup key verification.');\n    }\n    return {\n      id: keyResponse.id,\n      keyShares: keyResponse.keyShares,\n      commonKeychain: keyResponse.commonKeychain,\n    };\n  }\n\n  /** @inheritdoc */\n  async createKeychains(params: {\n    passphrase: string;\n    enterprise?: string | undefined;\n    originalPasscodeEncryptionCode?: string | undefined;\n  }): Promise<KeychainsTriplet> {\n    const MPC = new Ecdsa();\n    const m = 2;\n    const n = 3;\n\n    const userKeyShare = await MPC.keyShare(1, m, n);\n    const userGpgKey = await generateGPGKeyPair('secp256k1');\n    const backupKeyShare = await this.createBackupKeyShares();\n    const backupGpgKey = await this.getBackupGpgPubKey();\n\n    // Get the BitGo public key based on user/enterprise feature flags\n    // If it doesn't work, use the default public key from the constants\n    const bitgoPublicGpgKey =\n      (await this.getBitgoGpgPubkeyBasedOnFeatureFlags(params.enterprise)) ?? this.bitgoPublicGpgKey;\n\n    const bitgoKeychain = await this.createBitgoKeychain({\n      userGpgKey,\n      backupGpgKey,\n      bitgoPublicGpgKey,\n      userKeyShare,\n      backupKeyShare,\n      enterprise: params.enterprise,\n    });\n    const userKeychainPromise = this.createUserKeychain({\n      userGpgKey,\n      backupGpgKey,\n      bitgoPublicGpgKey,\n      userKeyShare,\n      backupKeyShare,\n      bitgoKeychain,\n      passphrase: params.passphrase,\n      originalPasscodeEncryptionCode: params.originalPasscodeEncryptionCode,\n    });\n    const backupKeychainPromise = this.createBackupKeychain({\n      userGpgKey,\n      backupGpgKey,\n      bitgoPublicGpgKey,\n      userKeyShare,\n      backupKeyShare,\n      bitgoKeychain,\n      passphrase: params.passphrase,\n    });\n\n    const [userKeychain, backupKeychain] = await Promise.all([userKeychainPromise, backupKeychainPromise]);\n\n    return {\n      userKeychain,\n      backupKeychain,\n      bitgoKeychain,\n    };\n  }\n\n  async createBackupKeyShares(): Promise<BackupKeyShare> {\n    const MPC = new Ecdsa();\n    const m = 2;\n    const n = 3;\n    const backupKeyShare = {\n      userHeldKeyShare: await MPC.keyShare(2, m, n),\n    };\n    return backupKeyShare;\n  }\n\n  createUserKeychain({\n    userGpgKey,\n    backupGpgKey,\n    bitgoPublicGpgKey,\n    userKeyShare,\n    backupKeyShare,\n    bitgoKeychain,\n    passphrase,\n    originalPasscodeEncryptionCode,\n  }: CreateEcdsaKeychainParams): Promise<Keychain> {\n    if (!passphrase) {\n      throw new Error('Please provide a wallet passphrase');\n    }\n    assert(backupKeyShare.userHeldKeyShare);\n    return this.createParticipantKeychain(\n      userGpgKey,\n      backupGpgKey as SerializedKeyPair<string>,\n      bitgoPublicGpgKey,\n      1,\n      userKeyShare,\n      backupKeyShare.userHeldKeyShare,\n      bitgoKeychain,\n      passphrase,\n      originalPasscodeEncryptionCode\n    );\n  }\n\n  async createBackupKeychain({\n    userGpgKey,\n    userKeyShare,\n    backupGpgKey,\n    backupKeyShare,\n    bitgoKeychain,\n    bitgoPublicGpgKey,\n    passphrase,\n  }: CreateEcdsaKeychainParams): Promise<Keychain> {\n    assert(backupKeyShare.userHeldKeyShare);\n    assert(passphrase);\n    return this.createParticipantKeychain(\n      userGpgKey,\n      backupGpgKey as SerializedKeyPair<string>,\n      bitgoPublicGpgKey,\n      2,\n      userKeyShare,\n      backupKeyShare.userHeldKeyShare,\n      bitgoKeychain,\n      passphrase\n    );\n  }\n\n  /** @inheritdoc */\n  async createBitgoKeychain({\n    userGpgKey,\n    backupGpgKey,\n    userKeyShare,\n    backupKeyShare,\n    enterprise,\n    bitgoPublicGpgKey,\n  }: CreateEcdsaBitGoKeychainParams): Promise<Keychain> {\n    const recipientIndex = 3;\n    const userToBitgoShare = await encryptNShare(userKeyShare, recipientIndex, bitgoPublicGpgKey.armor(), userGpgKey);\n\n    const backupToBitgoShare = await this.getBackupEncryptedNShare(\n      backupKeyShare,\n      recipientIndex,\n      bitgoPublicGpgKey.armor(),\n      backupGpgKey as SerializedKeyPair<string>\n    );\n\n    const createBitGoMPCParams: AddKeychainOptions = {\n      keyType: 'tss' as KeyType,\n      source: 'bitgo',\n      keyShares: [\n        {\n          from: 'user',\n          to: 'bitgo',\n          publicShare: userToBitgoShare.publicShare,\n          privateShare: userToBitgoShare.encryptedPrivateShare,\n          n: userToBitgoShare.n,\n          vssProof: userToBitgoShare.vssProof,\n          privateShareProof: userToBitgoShare.privateShareProof,\n        },\n        {\n          from: 'backup',\n          to: 'bitgo',\n          publicShare: backupToBitgoShare.publicShare,\n          privateShare: backupToBitgoShare.encryptedPrivateShare,\n          n: backupToBitgoShare.n,\n          vssProof: backupToBitgoShare.vssProof,\n          privateShareProof: backupToBitgoShare.privateShareProof,\n        },\n      ],\n      userGPGPublicKey: userGpgKey.publicKey,\n      backupGPGPublicKey: (backupGpgKey as SerializedKeyPair<string>).publicKey,\n      enterprise: enterprise,\n      algoUsed: 'ecdsa',\n    };\n\n    return await this.baseCoin.keychains().add(createBitGoMPCParams);\n  }\n\n  /**\n   * This builds the relevant backup encryptedNShare based on whether the\n   * backup key is user or third party generated\n   * @param backupShare can either have key shares from the user or third party\n   * @param recipientIndex index of the party receiving the backup shares\n   * @param recipientGpgPublicArmor gpg armor of the party receiving the backup shares\n   * @param backupGpgKey backup gpg key\n   * @param isThirdPartyBackup whether the backup is generated by third party\n   */\n  async getBackupEncryptedNShare(\n    backupShare: BackupKeyShare,\n    recipientIndex: number,\n    recipientGpgPublicArmor: string,\n    backupGpgKey: SerializedKeyPair<string>\n  ): Promise<EncryptedNShare> {\n    assert(backupShare.userHeldKeyShare);\n    const backupToRecipientShare = await encryptNShare(\n      backupShare.userHeldKeyShare,\n      recipientIndex,\n      recipientGpgPublicArmor,\n      backupGpgKey\n    );\n    return backupToRecipientShare;\n  }\n\n  /** @inheritdoc */\n  async createParticipantKeychain(\n    userGpgKey: openpgp.SerializedKeyPair<string>,\n    userLocalBackupGpgKey: openpgp.SerializedKeyPair<string>,\n    bitgoPublicGpgKey: Key,\n    recipientIndex: number,\n    userKeyShare: KeyShare,\n    backupKeyShare: KeyShare,\n    bitgoKeychain: Keychain,\n    passphrase: string,\n    originalPasscodeEncryptionCode?: string\n  ): Promise<Keychain> {\n    const bitgoKeyShares = bitgoKeychain.keyShares;\n    if (!bitgoKeyShares) {\n      throw new Error('Missing BitGo key shares');\n    }\n    if (!bitgoKeychain.commonKeychain) {\n      throw new Error(`Missing common key chain: ${bitgoKeychain.commonKeychain}`);\n    }\n\n    let recipient: string;\n    let keyShare: KeyShare;\n    let otherShare: KeyShare;\n    let recipientGpgKey: openpgp.SerializedKeyPair<string>;\n    let senderGpgKey: openpgp.SerializedKeyPair<string>;\n    if (recipientIndex === 1) {\n      keyShare = userKeyShare;\n      otherShare = backupKeyShare;\n      recipient = 'user';\n      recipientGpgKey = userGpgKey;\n      senderGpgKey = userLocalBackupGpgKey;\n    } else if (recipientIndex === 2) {\n      keyShare = backupKeyShare;\n      otherShare = userKeyShare;\n      recipient = 'backup';\n      recipientGpgKey = userLocalBackupGpgKey;\n      senderGpgKey = userGpgKey;\n    } else {\n      throw new Error('Invalid user index');\n    }\n\n    const bitGoToRecipientShare = bitgoKeyShares.find(\n      (keyShare) => keyShare.from === 'bitgo' && keyShare.to === recipient\n    );\n    if (!bitGoToRecipientShare) {\n      throw new Error(`Missing BitGo to ${recipient} key share`);\n    }\n\n    const decryptedShare = await this.decryptPrivateShare(bitGoToRecipientShare.privateShare, recipientGpgKey);\n\n    await this.verifyWalletSignatures(\n      userGpgKey.publicKey,\n      userLocalBackupGpgKey.publicKey,\n      bitgoKeychain,\n      decryptedShare,\n      recipientIndex\n    );\n\n    const senderToRecipientShare = await encryptNShare(\n      otherShare,\n      recipientIndex,\n      recipientGpgKey.publicKey,\n      senderGpgKey\n    );\n    const encryptedNShares: DecryptableNShare[] = [\n      {\n        // userToBackup or backupToUser\n        nShare: senderToRecipientShare,\n        recipientPrivateArmor: recipientGpgKey.privateKey,\n        senderPublicArmor: senderGpgKey.publicKey,\n      },\n      {\n        // bitgoToRecipient\n        nShare: {\n          i: recipientIndex,\n          j: 3,\n          publicShare: bitGoToRecipientShare.publicShare,\n          encryptedPrivateShare: bitGoToRecipientShare.privateShare,\n          n: bitGoToRecipientShare.n!,\n          vssProof: bitGoToRecipientShare.vssProof,\n          privateShareProof: bitGoToRecipientShare.privateShareProof,\n        },\n        recipientPrivateArmor: recipientGpgKey.privateKey,\n        senderPublicArmor: bitgoPublicGpgKey.armor(),\n        isbs58Encoded: false,\n      },\n    ];\n\n    const recipientCombinedKey = await ECDSAMethods.createCombinedKey(\n      keyShare,\n      encryptedNShares,\n      bitgoKeychain.commonKeychain\n    );\n\n    const prv = JSON.stringify(recipientCombinedKey.signingMaterial);\n    const recipientKeychainParams = {\n      source: recipient,\n      keyType: 'tss' as KeyType,\n      commonKeychain: bitgoKeychain.commonKeychain,\n      prv: prv,\n      encryptedPrv: this.bitgo.encrypt({\n        input: prv,\n        password: passphrase,\n      }),\n      originalPasscodeEncryptionCode,\n    };\n\n    const keychains = this.baseCoin.keychains();\n    return recipientIndex === 1\n      ? await keychains.add(recipientKeychainParams)\n      : await keychains.createBackup(recipientKeychainParams);\n  }\n\n  private async createTssEcdsaStep1SigningMaterial(params: {\n    challenges: {\n      enterpriseChallenge: EcdsaTypes.SerializedEcdsaChallenges;\n      bitgoChallenge: TxRequestChallengeResponse;\n    };\n    prv: string;\n    derivationPath: string;\n    walletPassphrase?: string;\n  }): Promise<TssEcdsaStep1ReturnMessage> {\n    const { challenges, derivationPath, prv } = params;\n    const userSigningMaterial: ECDSAMethodTypes.SigningMaterial = JSON.parse(prv);\n    if (userSigningMaterial.pShare.i !== 1) {\n      throw new Error('Invalid user key');\n    }\n    if (!userSigningMaterial.backupNShare) {\n      throw new Error('Invalid user key - missing backupNShare');\n    }\n    const MPC = new Ecdsa();\n    const signingKey = MPC.keyDerive(\n      userSigningMaterial.pShare,\n      [userSigningMaterial.bitgoNShare, userSigningMaterial.backupNShare],\n      derivationPath\n    );\n\n    const bitgoIndex = ShareKeyPosition.BITGO;\n    const userIndex = userSigningMaterial.pShare.i;\n\n    const { ntilde: ntildea, h1: h1a, h2: h2a, p: pa } = challenges.enterpriseChallenge;\n    const { ntilde: ntildeb, h1: h1b, h2: h2b, p: pb, n: nb } = challenges.bitgoChallenge;\n    const userXShare = MPC.appendChallenge(signingKey.xShare, { ntilde: ntildea, h1: h1a, h2: h2a }, { p: pa });\n    const bitgoYShare = MPC.appendChallenge(\n      {\n        i: userIndex,\n        j: bitgoIndex,\n        n: nb,\n      },\n      { ntilde: ntildeb, h1: h1b, h2: h2b },\n      { p: pb }\n    );\n\n    const userSignShare = await ECDSAMethods.createUserSignShare(userXShare, bitgoYShare);\n    const u = signingKey.nShares[bitgoIndex].u;\n\n    let chaincode = userSigningMaterial.bitgoNShare.chaincode;\n    while (chaincode.length < 64) {\n      chaincode = '0' + chaincode;\n    }\n    const signerShare = bip32.fromPrivateKey(Buffer.from(u, 'hex'), Buffer.from(chaincode, 'hex')).toBase58();\n    const bitgoGpgKey = (await getBitgoGpgPubKey(this.bitgo)).mpcV1;\n    const encryptedSignerShare = (await openpgp.encrypt({\n      message: await openpgp.createMessage({\n        text: signerShare,\n      }),\n      config: {\n        rejectCurves: new Set(),\n      },\n      encryptionKeys: [bitgoGpgKey],\n    })) as string;\n    const userGpgKey = await generateGPGKeyPair('secp256k1');\n    const privateShareProof = await createShareProof(userGpgKey.privateKey, signingKey.nShares[bitgoIndex].u, 'ecdsa');\n    const vssProof = signingKey.nShares[bitgoIndex].v;\n    const userPublicGpgKey = userGpgKey.publicKey;\n    const publicShare = signingKey.nShares[bitgoIndex].y + signingKey.nShares[bitgoIndex].chaincode;\n    return {\n      privateShareProof: privateShareProof,\n      vssProof: vssProof,\n      publicShare: publicShare,\n      encryptedSignerOffsetShare: encryptedSignerShare,\n      userPublicGpgKey: userPublicGpgKey,\n      kShare: userSignShare.kShare,\n      wShare: params.walletPassphrase\n        ? this.bitgo.encrypt({ input: JSON.stringify(userSignShare.wShare), password: params.walletPassphrase })\n        : userSignShare.wShare,\n    };\n  }\n\n  private async createTssEcdsaStep2SigningMaterial(params: {\n    bitgoChallenge: TxRequestChallengeResponse;\n    wShare: WShare;\n    aShareFromBitgo: Omit<AShare, 'h1' | 'h2' | 'ntilde'>;\n    walletPassphrase?: string;\n  }): Promise<TssEcdsaStep2ReturnMessage> {\n    // Append the BitGo challenge to the Ashare to be used in subsequent proofs\n    const bitgoToUserAShareWithNtilde: AShare = {\n      ...params.aShareFromBitgo,\n      ...params.bitgoChallenge,\n    };\n    const userGammaAndMuShares = await ECDSAMethods.createUserGammaAndMuShare(\n      params.wShare,\n      bitgoToUserAShareWithNtilde\n    );\n    const userOmicronAndDeltaShare = await ECDSAMethods.createUserOmicronAndDeltaShare(\n      userGammaAndMuShares.gShare as ECDSA.GShare\n    );\n    return {\n      muDShare: {\n        muShare: userGammaAndMuShares.muShare,\n        dShare: userOmicronAndDeltaShare.dShare,\n        i: userGammaAndMuShares.muShare.i,\n      },\n      oShare: params.walletPassphrase\n        ? this.bitgo.encrypt({\n            input: JSON.stringify(userOmicronAndDeltaShare.oShare),\n            password: params.walletPassphrase,\n          })\n        : userOmicronAndDeltaShare.oShare,\n    };\n  }\n\n  getOfflineSignerPaillierModulus(params: { prv: string }): { userPaillierModulus: string } {\n    assert(params.prv, 'Params to get paillier modulus are missing prv.');\n    const userSigningMaterial: ECDSAMethodTypes.SigningMaterial = JSON.parse(params.prv);\n    return { userPaillierModulus: userSigningMaterial.pShare.n };\n  }\n\n  async createOfflineKShare(params: {\n    tssParams: TSSParams | TSSParamsForMessage;\n    challenges: {\n      enterpriseChallenge: EcdsaTypes.SerializedEcdsaChallenges;\n      bitgoChallenge: TxRequestChallengeResponse;\n    };\n    requestType: RequestType;\n    prv: string;\n    walletPassphrase: string;\n  }): Promise<TssEcdsaStep1ReturnMessage> {\n    const { tssParams, prv, requestType, challenges } = params;\n    assert(typeof tssParams.txRequest !== 'string', 'Invalid txRequest type');\n    const txRequest: TxRequest = tssParams.txRequest;\n    let derivationPath;\n\n    if (requestType === RequestType.tx) {\n      assert(\n        txRequest.transactions || (txRequest as TxRequest).unsignedTxs,\n        'Unable to find transactions in txRequest'\n      );\n      const unsignedTx =\n        txRequest.apiVersion === 'full' ? txRequest.transactions![0].unsignedTx : txRequest.unsignedTxs[0];\n      derivationPath = unsignedTx.derivationPath;\n    } else if (requestType === RequestType.message) {\n      // TODO BG-67299 Message signing with derivation path\n      derivationPath = '';\n    }\n    return this.createTssEcdsaStep1SigningMaterial({\n      prv: prv,\n      challenges: challenges,\n      derivationPath: derivationPath,\n      walletPassphrase: params.walletPassphrase,\n    });\n  }\n\n  async createOfflineMuDeltaShare(params: {\n    aShareFromBitgo: Omit<AShare, 'ntilde' | 'h1' | 'h2'>;\n    bitgoChallenge: TxRequestChallengeResponse;\n    encryptedWShare: string;\n    walletPassphrase: string;\n  }): Promise<TssEcdsaStep2ReturnMessage> {\n    const decryptedWShare = this.bitgo.decrypt({ input: params.encryptedWShare, password: params.walletPassphrase });\n    return await this.createTssEcdsaStep2SigningMaterial({\n      aShareFromBitgo: params.aShareFromBitgo,\n      bitgoChallenge: params.bitgoChallenge,\n      wShare: JSON.parse(decryptedWShare),\n      walletPassphrase: params.walletPassphrase,\n    });\n  }\n\n  async createOfflineSShare(params: {\n    tssParams: TSSParams | TSSParamsForMessage;\n    dShareFromBitgo: DShare;\n    requestType: RequestType;\n    encryptedOShare: string;\n    walletPassphrase: string;\n  }): Promise<SShare> {\n    const { tssParams, requestType, dShareFromBitgo, encryptedOShare, walletPassphrase } = params;\n    assert(typeof tssParams.txRequest !== 'string', 'Invalid txRequest type');\n    const txRequest: TxRequest = tssParams.txRequest;\n    let signablePayload;\n    if (requestType === RequestType.tx) {\n      assert(txRequest.transactions || txRequest.unsignedTxs, 'Unable to find transactions in txRequest');\n      const unsignedTx =\n        txRequest.apiVersion === 'full' ? txRequest.transactions![0].unsignedTx : txRequest.unsignedTxs[0];\n      signablePayload = Buffer.from(unsignedTx.signableHex, 'hex');\n    } else if (requestType === RequestType.message) {\n      signablePayload = (params.tssParams as TSSParamsForMessage).bufferToSign;\n    }\n    let hash: Hash | undefined;\n    try {\n      hash = this.baseCoin.getHashFunction();\n    } catch (err) {\n      hash = undefined;\n    }\n    const decryptedOShare = this.bitgo.decrypt({ input: encryptedOShare, password: walletPassphrase });\n    const { i, R, s, y } = await ECDSAMethods.createUserSignatureShare(\n      JSON.parse(decryptedOShare),\n      dShareFromBitgo,\n      signablePayload,\n      hash\n    );\n    // return only required SShare without bigints from VAShare\n    return {\n      i,\n      R,\n      s,\n      y,\n    };\n  }\n  async signEcdsaTssUsingExternalSigner(\n    params: TSSParams | TSSParamsForMessage,\n    requestType: RequestType,\n    externalSignerPaillierModulusGetter: CustomPaillierModulusGetterFunction,\n    externalSignerKShareGenerator: CustomKShareGeneratingFunction,\n    externalSignerMuDeltaShareGenerator: CustomMuDeltaShareGeneratingFunction,\n    externalSignerSShareGenerator: CustomSShareGeneratingFunction\n  ): Promise<TxRequest> {\n    const { txRequest } = params;\n    const pendingEcdsaTssInitialization = this.wallet.coinSpecific()?.pendingEcdsaTssInitialization;\n    if (pendingEcdsaTssInitialization) {\n      throw new Error(\n        'Wallet is not ready for TSS ECDSA signing. Please contact your enterprise admin to finish the enterprise TSS initialization.'\n      );\n    }\n    const txRequestObj: TxRequest = await getTxRequest(this.bitgo, this.wallet.id(), txRequest as string, params.reqId);\n    const { userPaillierModulus } = await externalSignerPaillierModulusGetter({ txRequest: txRequestObj });\n    const { enterpriseChallenge, bitgoChallenge } = await this.getEcdsaSigningChallenges(\n      txRequest as string,\n      requestType,\n      userPaillierModulus,\n      0,\n      params.reqId\n    );\n    const step1SigningMaterial = await externalSignerKShareGenerator({\n      tssParams: {\n        ...params,\n        txRequest: txRequestObj,\n      },\n      challenges: { enterpriseChallenge, bitgoChallenge },\n      requestType: requestType,\n    });\n    // signing stage one with K share send to bitgo and receives A share\n    const bitgoToUserAShare = (await ECDSAMethods.sendShareToBitgo(\n      this.bitgo,\n      this.wallet.id(),\n      txRequestObj.txRequestId,\n      requestType,\n      SendShareType.KShare,\n      step1SigningMaterial.kShare,\n      step1SigningMaterial.encryptedSignerOffsetShare,\n      step1SigningMaterial.vssProof,\n      step1SigningMaterial.privateShareProof,\n      step1SigningMaterial.publicShare,\n      step1SigningMaterial.userPublicGpgKey,\n      params.reqId\n    )) as Omit<AShare, 'ntilde' | 'h1' | 'h2'>; // WP/HSM does not return the initial challenge\n    const step2Return = await externalSignerMuDeltaShareGenerator({\n      txRequest: txRequestObj,\n      aShareFromBitgo: bitgoToUserAShare,\n      bitgoChallenge: bitgoChallenge,\n      encryptedWShare: step1SigningMaterial.wShare as string,\n    });\n    // signing stage two with muShare and dShare send to bitgo and receives D share\n    const bitgoToUserDShare = (await ECDSAMethods.sendShareToBitgo(\n      this.bitgo,\n      this.wallet.id(),\n      txRequestObj.txRequestId,\n      requestType,\n      SendShareType.MUShare,\n      step2Return.muDShare,\n      undefined,\n      undefined,\n      undefined,\n      undefined,\n      undefined,\n      params.reqId\n    )) as DShare;\n    const userSShare = await externalSignerSShareGenerator({\n      tssParams: {\n        ...params,\n        txRequest: txRequestObj,\n      },\n      dShareFromBitgo: bitgoToUserDShare,\n      requestType: requestType,\n      encryptedOShare: step2Return.oShare as string,\n    });\n    // signing stage three with SShare send to bitgo and receives SShare\n    await ECDSAMethods.sendShareToBitgo(\n      this.bitgo,\n      this.wallet.id(),\n      txRequestObj.txRequestId,\n      requestType,\n      SendShareType.SShare,\n      userSShare,\n      undefined,\n      undefined,\n      undefined,\n      undefined,\n      undefined,\n      params.reqId\n    );\n    return await getTxRequest(this.bitgo, this.wallet.id(), txRequestObj.txRequestId, params.reqId);\n  }\n\n  /**\n   * Gets signing key, txRequestResolved and txRequestId\n   * @param {string | TxRequest} params.txRequest - transaction request object or id\n   * @param {string} params.prv - decrypted private key\n   * @param { string} params.reqId - request id\n   * @returns {Promise<TxRequest>}\n   */\n  private async signRequestBase(\n    params: TSSParamsWithPrv | TSSParamsForMessageWithPrv,\n    requestType: RequestType\n  ): Promise<TxRequest> {\n    const pendingEcdsaTssInitialization = this.wallet.coinSpecific()?.pendingEcdsaTssInitialization;\n    if (pendingEcdsaTssInitialization) {\n      throw new Error(\n        'Wallet is not ready for TSS ECDSA signing. Please contact your enterprise admin to finish the enterprise TSS initialization.'\n      );\n    }\n    const userSigningMaterial: ECDSAMethodTypes.SigningMaterial = JSON.parse(params.prv);\n    if (userSigningMaterial.pShare.i !== 1) {\n      throw new Error('Invalid user key');\n    }\n    if (!userSigningMaterial.backupNShare) {\n      throw new Error('Invalid user key - missing backupNShare');\n    }\n\n    const txRequest: TxRequest =\n      typeof params.txRequest === 'string'\n        ? await getTxRequest(this.bitgo, this.wallet.id(), params.txRequest, params.reqId)\n        : params.txRequest;\n\n    let signablePayload = new Buffer('');\n    let derivationPath = '';\n\n    if (requestType === RequestType.tx) {\n      assert(txRequest.transactions || txRequest.unsignedTxs, 'Unable to find transactions in txRequest');\n      const unsignedTx =\n        txRequest.apiVersion === 'full' ? txRequest.transactions![0].unsignedTx : txRequest.unsignedTxs[0];\n\n      // For ICP transactions, the HSM signs the serializedTxHex, while the user signs the signableHex separately.\n      // Verification cannot be performed directly on the signableHex alone. However, we can parse the serializedTxHex\n      // to regenerate the signableHex and compare it against the provided value for verification.\n      // In contrast, for other coin families, verification is typically done using just the signableHex.\n      if (this.baseCoin.getConfig().family === 'icp') {\n        await this.baseCoin.verifyTransaction({\n          txPrebuild: { txHex: unsignedTx.serializedTxHex, txInfo: unsignedTx.signableHex },\n          txParams: params.txParams || { recipients: [] },\n          wallet: this.wallet,\n          walletType: this.wallet.multisigType(),\n        });\n      } else {\n        await this.baseCoin.verifyTransaction({\n          txPrebuild: { txHex: unsignedTx.signableHex },\n          txParams: params.txParams || { recipients: [] },\n          wallet: this.wallet,\n          walletType: this.wallet.multisigType(),\n        });\n      }\n      signablePayload = Buffer.from(unsignedTx.signableHex, 'hex');\n      derivationPath = unsignedTx.derivationPath;\n    } else if (requestType === RequestType.message) {\n      signablePayload = (params as TSSParamsForMessage).bufferToSign;\n      // TODO BG-67299 Message signing with derivation path\n    }\n    const paillierModulus = this.getOfflineSignerPaillierModulus({ prv: params.prv });\n    const challenges = await this.getEcdsaSigningChallenges(\n      txRequest.txRequestId,\n      requestType,\n      paillierModulus.userPaillierModulus,\n      0,\n      params.reqId\n    );\n\n    const step1Return = await this.createTssEcdsaStep1SigningMaterial({\n      prv: params.prv,\n      challenges: challenges,\n      derivationPath: derivationPath,\n    });\n\n    // signing stage one with K share send to bitgo and receives A share\n    const bitgoToUserAShare = (await ECDSAMethods.sendShareToBitgo(\n      this.bitgo,\n      this.wallet.id(),\n      txRequest.txRequestId,\n      requestType,\n      SendShareType.KShare,\n      step1Return.kShare,\n      step1Return.encryptedSignerOffsetShare,\n      step1Return.vssProof,\n      step1Return.privateShareProof,\n      step1Return.publicShare,\n      step1Return.userPublicGpgKey,\n      params.reqId\n    )) as Omit<AShare, 'ntilde' | 'h1' | 'h2'>; // WP/HSM does not return the initial challenge\n\n    const step2Return = await this.createTssEcdsaStep2SigningMaterial({\n      aShareFromBitgo: bitgoToUserAShare,\n      bitgoChallenge: challenges.bitgoChallenge,\n      wShare: step1Return.wShare as WShare,\n    });\n\n    // signing stage two with muShare and dShare send to bitgo and receives D share\n    const bitgoToUserDShare = (await ECDSAMethods.sendShareToBitgo(\n      this.bitgo,\n      this.wallet.id(),\n      txRequest.txRequestId,\n      requestType,\n      SendShareType.MUShare,\n      step2Return.muDShare,\n      undefined,\n      undefined,\n      undefined,\n      undefined,\n      undefined,\n      params.reqId\n    )) as DShare;\n\n    // If only the getHashFunction() is defined for the coin use it otherwise\n    // pass undefined hash, default hash will be used in that case.\n    let hash: Hash | undefined;\n    try {\n      hash = this.baseCoin.getHashFunction();\n    } catch (err) {\n      hash = undefined;\n    }\n\n    const userSShare = await ECDSAMethods.createUserSignatureShare(\n      step2Return.oShare as OShare,\n      bitgoToUserDShare,\n      signablePayload,\n      hash\n    );\n\n    // signing stage three with SShare send to bitgo and receives SShare\n    await ECDSAMethods.sendShareToBitgo(\n      this.bitgo,\n      this.wallet.id(),\n      txRequest.txRequestId,\n      requestType,\n      SendShareType.SShare,\n      userSShare,\n      undefined,\n      undefined,\n      undefined,\n      undefined,\n      undefined,\n      params.reqId\n    );\n    return await getTxRequest(this.bitgo, this.wallet.id(), txRequest.txRequestId, params.reqId);\n  }\n\n  /**\n   * Signs the transaction associated to the transaction request.\n   * @param {string | TxRequest} params.txRequest - transaction request object or id\n   * @param {string} params.prv - decrypted private key\n   * @param {string} params.reqId - request id\n   * @returns {Promise<TxRequest>} fully signed TxRequest object\n   */\n  async signTxRequest(params: TSSParamsWithPrv): Promise<TxRequest> {\n    this.bitgo.setRequestTracer(params.reqId);\n    return this.signRequestBase(params, RequestType.tx);\n  }\n\n  /**\n   * Signs the message associated to the transaction request.\n   * @param {string | TxRequest} params.txRequest - transaction request object or id\n   * @param {string} params.prv - decrypted private key\n   * @param {string} params.reqId - request id\n   * @returns {Promise<TxRequest>} fully signed TxRequest object\n   */\n  async signTxRequestForMessage(params: TSSParamsForMessageWithPrv): Promise<TxRequest> {\n    if (!params.messageRaw) {\n      throw new Error('Raw message required to sign message');\n    }\n    return this.signRequestBase(params, RequestType.message);\n  }\n\n  /**\n   * Get the challenge values for enterprise and BitGo in ECDSA signing\n   * Only returns the challenges if they are verified by the user's enterprise admin's ecdh key\n   * @param {string} txRequestId - transaction request id\n   * @param {RequestType} requestType -  (0 for tx, 1 for message)\n   * @param {string} walletPaillierModulus - paillier pubkey $n$\n   * @param {number} index - index of the requestType\n   * @param {IRequestTracer} reqId - request tracer request id\n   */\n  async getEcdsaSigningChallenges(\n    txRequestId: string,\n    requestType: RequestType,\n    walletPaillierModulus: string,\n    index = 0,\n    reqId?: IRequestTracer\n  ): Promise<{\n    enterpriseChallenge: EcdsaTypes.SerializedEcdsaChallenges;\n    bitgoChallenge: TxRequestChallengeResponse;\n  }> {\n    const enterpriseId = this.wallet.toJSON().enterprise;\n    if (!enterpriseId) {\n      throw new Error('Wallet must be an enterprise wallet.');\n    }\n\n    // create BitGo range proof and paillier proof challenge\n    const createBitgoChallengeResponse = await getTxRequestChallenge(\n      this.bitgo,\n      this.wallet.id(),\n      txRequestId,\n      index.toString(),\n      requestType,\n      walletPaillierModulus,\n      reqId\n    );\n\n    const bitgoToEnterprisePaillierChallenge = { p: createBitgoChallengeResponse.p };\n    const enterpriseToBitgoPaillierChallenge = EcdsaTypes.serializePaillierChallenge({\n      p: await EcdsaPaillierProof.generateP(hexToBigInt(createBitgoChallengeResponse.n)),\n    });\n\n    // TODO(BG-78764): once the paillier proofs are complete, reduce challenge creation to one API call\n    const walletChallenges = await this.wallet.getChallengesForEcdsaSigning();\n\n    const challengeVerifierUserId = walletChallenges.createdBy;\n    const adminSigningKeyResponse = await this.bitgo.getSigningKeyForUser(enterpriseId, challengeVerifierUserId);\n    const pubkeyOfAdminEcdhKeyHex = adminSigningKeyResponse.derivedPubkey;\n\n    // Verify enterprise's challenge is signed by the respective admins ecdh keychain\n    const enterpriseRawChallenge = {\n      ntilde: walletChallenges.enterpriseChallenge.ntilde,\n      h1: walletChallenges.enterpriseChallenge.h1,\n      h2: walletChallenges.enterpriseChallenge.h2,\n    };\n    const adminSignatureOnEntChallenge: string = walletChallenges.enterpriseChallenge.verifiers.adminSignature;\n    if (\n      !verifyEcdhSignature(\n        EcdsaUtils.getMessageToSignFromChallenge(enterpriseRawChallenge),\n        adminSignatureOnEntChallenge,\n        Buffer.from(pubkeyOfAdminEcdhKeyHex, 'hex')\n      )\n    ) {\n      throw new Error(`Admin signature for enterprise challenge is not valid. Please contact your enterprise admin.`);\n    }\n\n    // Verify that the BitGo challenge's ZK proofs have been verified by the admin\n    const bitgoChallenge: TxRequestChallengeResponse = {\n      ntilde: walletChallenges.bitgoChallenge.ntilde,\n      h1: walletChallenges.bitgoChallenge.h1,\n      h2: walletChallenges.bitgoChallenge.h2,\n      p: bitgoToEnterprisePaillierChallenge.p,\n      n: createBitgoChallengeResponse.n,\n    };\n    const adminVerificationSignatureForBitGoChallenge = walletChallenges.bitgoChallenge.verifiers.adminSignature;\n    if (\n      !verifyEcdhSignature(\n        EcdsaUtils.getMessageToSignFromChallenge(bitgoChallenge),\n        adminVerificationSignatureForBitGoChallenge,\n        Buffer.from(pubkeyOfAdminEcdhKeyHex, 'hex')\n      )\n    ) {\n      throw new Error(`Admin signature for BitGo's challenge is not valid. Please contact your enterprise admin.`);\n    }\n\n    return {\n      enterpriseChallenge: {\n        ...enterpriseRawChallenge,\n        p: enterpriseToBitgoPaillierChallenge.p,\n      },\n      bitgoChallenge,\n    };\n  }\n\n  /**\n   * Verifies the u-value proofs and GPG keys used in generating a TSS ECDSA wallet.\n   * @param userGpgPub The user's public GPG key for encryption between user/server\n   * @param backupGpgPub The backup's public GPG key for encryption between backup/server\n   * @param bitgoKeychain previously created BitGo keychain; must be compatible with user and backup key shares\n   * @param decryptedShare The decrypted bitgo-to-user/backup private share retrieved from the keychain\n   * @param verifierIndex The index of the party to verify: 1 = user, 2 = backup\n   */\n  async verifyWalletSignatures(\n    userGpgPub: string,\n    backupGpgPub: string,\n    bitgoKeychain: Keychain,\n    decryptedShare: string,\n    verifierIndex: 1 | 2\n  ): Promise<void> {\n    assert(bitgoKeychain.commonKeychain);\n    assert(bitgoKeychain.walletHSMGPGPublicKeySigs);\n\n    const bitgoGpgKey = (await getBitgoGpgPubKey(this.bitgo)).mpcV1;\n    const userKeyPub = await openpgp.readKey({ armoredKey: userGpgPub });\n    const userKeyId = userKeyPub.keyPacket.getFingerprint();\n    const backupKeyPub = await openpgp.readKey({ armoredKey: backupGpgPub });\n    const backupKeyId = backupKeyPub.keyPacket.getFingerprint();\n\n    const walletSignatures = await openpgp.readKeys({ armoredKeys: bitgoKeychain.walletHSMGPGPublicKeySigs });\n    if (walletSignatures.length !== 2) {\n      throw new Error('Invalid wallet signatures');\n    }\n    if (userKeyId !== walletSignatures[0].keyPacket.getFingerprint()) {\n      throw new Error(`first wallet signature's fingerprint does not match passed user gpg key's fingerprint`);\n    }\n    if (backupKeyId !== walletSignatures[1].keyPacket.getFingerprint()) {\n      throw new Error(`second wallet signature's fingerprint does not match passed backup gpg key's fingerprint`);\n    }\n\n    await verifyWalletSignature({\n      walletSignature: walletSignatures[0],\n      commonKeychain: bitgoKeychain.commonKeychain,\n      userKeyId,\n      backupKeyId,\n      bitgoPub: bitgoGpgKey,\n      decryptedShare,\n      verifierIndex,\n    });\n\n    await verifyWalletSignature({\n      walletSignature: walletSignatures[1],\n      commonKeychain: bitgoKeychain.commonKeychain,\n      userKeyId,\n      backupKeyId,\n      bitgoPub: bitgoGpgKey,\n      decryptedShare,\n      verifierIndex,\n    });\n  }\n\n  /**\n   * Signs a challenge with the provided v1 ecdh key at a derived path\n   * @param challenge challenge to sign\n   * @param ecdhXprv xprv of the ecdh key\n   * @param derivationPath the derived path at which the ecdh key will sign\n   */\n  static signChallenge(challenge: EcdsaTypes.SerializedNtilde, ecdhXprv: string, derivationPath: string): Buffer {\n    const messageToSign = this.getMessageToSignFromChallenge(challenge);\n    return signMessageWithDerivedEcdhKey(messageToSign, ecdhXprv, derivationPath);\n  }\n\n  /**\n   * Converts challenge to a common message format which can be signed.\n   * @param challenge\n   */\n  static getMessageToSignFromChallenge(challenge: EcdsaTypes.SerializedNtilde): string {\n    return challenge.ntilde.concat(challenge.h1).concat(challenge.h2);\n  }\n\n  /**\n   Verifies ZK proofs of BitGo's challenges for both nitro and institutional HSMs\n   which are fetched from the WP API.\n   */\n  static async verifyBitGoChallenges(bitgoChallenges: GetBitGoChallengesApi): Promise<boolean> {\n    // Verify institutional hsm challenge proof\n    const instChallengeVerified = await this.verifyBitGoChallenge({\n      ntilde: bitgoChallenges.bitgoInstitutionalHsm.ntilde,\n      h1: bitgoChallenges.bitgoInstitutionalHsm.h1,\n      h2: bitgoChallenges.bitgoInstitutionalHsm.h2,\n      ntildeProof: bitgoChallenges.bitgoInstitutionalHsm.ntildeProof,\n    });\n\n    // Verify nitro hsm challenge proof\n    const nitroChallengeVerified = await this.verifyBitGoChallenge({\n      ntilde: bitgoChallenges.bitgoNitroHsm.ntilde,\n      h1: bitgoChallenges.bitgoNitroHsm.h1,\n      h2: bitgoChallenges.bitgoNitroHsm.h2,\n      ntildeProof: bitgoChallenges.bitgoNitroHsm.ntildeProof,\n    });\n\n    return instChallengeVerified && nitroChallengeVerified;\n  }\n\n  /**\n   * Verifies ZK proof for a single BitGo challenge\n   * @param bitgoChallenge\n   */\n  static async verifyBitGoChallenge(bitgoChallenge: EcdsaTypes.SerializedNtildeWithProofs): Promise<boolean> {\n    const deserializedInstChallenge = EcdsaTypes.deserializeNtildeWithProofs(bitgoChallenge);\n    const ntildeProofH1WrtH2Verified = await EcdsaRangeProof.verifyNtildeProof(\n      {\n        ntilde: deserializedInstChallenge.ntilde,\n        h1: deserializedInstChallenge.h1,\n        h2: deserializedInstChallenge.h2,\n      },\n      deserializedInstChallenge.ntildeProof.h1WrtH2\n    );\n    const ntildeProofH2WrtH1Verified = await EcdsaRangeProof.verifyNtildeProof(\n      {\n        ntilde: deserializedInstChallenge.ntilde,\n        h1: deserializedInstChallenge.h2,\n        h2: deserializedInstChallenge.h1,\n      },\n      deserializedInstChallenge.ntildeProof.h2WrtH1\n    );\n    return ntildeProofH1WrtH2Verified && ntildeProofH2WrtH1Verified;\n  }\n\n  /**\n   * Gets the bitgo challenges for both nitro and institutional HSMs from WP API.\n   * @param bitgo\n   */\n  static async getBitGoChallenges(bitgo: BitGoBase): Promise<GetBitGoChallengesApi> {\n    const res = await bitgo.get(bitgo.url('/tss/ecdsa/challenges', 2)).send().result();\n    if (\n      !res.bitgoNitroHsm ||\n      !res.bitgoNitroHsm.ntilde ||\n      !res.bitgoNitroHsm.h1 ||\n      !res.bitgoNitroHsm.h2 ||\n      !res.bitgoNitroHsm.ntildeProof ||\n      !res.bitgoInstitutionalHsm ||\n      !res.bitgoInstitutionalHsm.ntilde ||\n      !res.bitgoInstitutionalHsm.h1 ||\n      !res.bitgoInstitutionalHsm.h2 ||\n      !res.bitgoInstitutionalHsm.ntildeProof\n    ) {\n      throw new Error('Expected BitGo challenge proof to be present. Contact support@bitgo.com.');\n    }\n    return res;\n  }\n\n  /**\n   * Gets BitGo's proofs from API and signs them if the proofs are valid.\n   * @param bitgo\n   * @param enterpriseId\n   * @param userPassword\n   */\n  static async getVerifyAndSignBitGoChallenges(\n    bitgo: BitGoBase,\n    enterpriseId: string,\n    userPassword: string\n  ): Promise<BitGoProofSignatures> {\n    // Fetch BitGo's challenge and verify\n    const bitgoChallengesWithProofs = await EcdsaUtils.getBitGoChallenges(bitgo);\n    if (!(await EcdsaUtils.verifyBitGoChallenges(bitgoChallengesWithProofs))) {\n      throw new Error(\n        `Failed to verify BitGo's challenge needed to enable ECDSA signing. Please contact support@bitgo.com`\n      );\n    }\n    return await EcdsaUtils.signBitgoChallenges(bitgo, enterpriseId, userPassword, bitgoChallengesWithProofs);\n  }\n\n  /**\n   * Sign Bitgo's proofs, verification of proofs is left to the caller\n   * @param bitgo\n   * @param enterpriseId\n   * @param userPassword\n   * @param bitgoChallengesWithProofs Optionally provide Bitgo Challaenge & Proofs instead of fetching from API\n   */\n  static async signBitgoChallenges(\n    bitgo: BitGoBase,\n    enterpriseId: string,\n    userPassword: string,\n    bitgoChallengesWithProofs?: GetBitGoChallengesApi\n  ): Promise<BitGoProofSignatures> {\n    // fetch challenge & proof if none are provided\n    const challengesWithProofs = bitgoChallengesWithProofs\n      ? bitgoChallengesWithProofs\n      : await EcdsaUtils.getBitGoChallenges(bitgo);\n\n    // Fetch user's ecdh public keychain needed for signing the challenges\n    const ecdhKeypair = await bitgo.getEcdhKeypairPrivate(userPassword, enterpriseId);\n\n    const signedBitGoInstChallenge = EcdsaUtils.signChallenge(\n      challengesWithProofs.bitgoInstitutionalHsm,\n      ecdhKeypair.xprv,\n      ecdhKeypair.derivationPath\n    );\n    const signedBitGoNitroChallenge = EcdsaUtils.signChallenge(\n      challengesWithProofs.bitgoNitroHsm,\n      ecdhKeypair.xprv,\n      ecdhKeypair.derivationPath\n    );\n    return {\n      bitgoInstHsmAdminSignature: signedBitGoInstChallenge,\n      bitgoNitroHsmAdminSignature: signedBitGoNitroChallenge,\n    };\n  }\n\n  /**\n   * This is needed to enable ecdsa signing on the enterprise.\n   * It receives the enterprise challenge and signatures of verified bitgo proofs\n   * and uploads them on the enterprise.\n   * @param bitgo\n   * @param entId - enterprise id to enable ecdsa signing on\n   * @param userPassword - enterprise admin's login pw\n   * @param bitgoInstChallengeProofSignature - signature on bitgo's institutional HSM challenge after verification\n   * @param bitgoNitroChallengeProofSignature - signature on bitgo's nitro HSM challenge after verification\n   * @param challenge - optionally use the challenge for enterprise challenge\n   */\n  static async initiateChallengesForEnterprise(\n    bitgo: BitGoBase,\n    entId: string,\n    userPassword: string,\n    bitgoInstChallengeProofSignature: Buffer,\n    bitgoNitroChallengeProofSignature: Buffer,\n    openSSLBytes: Uint8Array,\n    challenge?: EcdsaTypes.DeserializedNtildeWithProofs\n  ): Promise<void> {\n    // Fetch user's ecdh public keychain needed for signing the challenges\n    const ecdhKeypair = await bitgo.getEcdhKeypairPrivate(userPassword, entId);\n\n    // Generate and sign enterprise challenge\n    const entChallengeWithProof =\n      challenge ?? (await EcdsaRangeProof.generateNtilde(openSSLBytes, minModulusBitLength));\n    const serializedEntChallengeWithProof = EcdsaTypes.serializeNtildeWithProofs(entChallengeWithProof);\n    const signedEnterpriseChallenge = EcdsaUtils.signChallenge(\n      serializedEntChallengeWithProof,\n      ecdhKeypair.xprv,\n      ecdhKeypair.derivationPath\n    );\n\n    await this.uploadChallengesToEnterprise(\n      bitgo,\n      entId,\n      serializedEntChallengeWithProof,\n      signedEnterpriseChallenge.toString('hex'),\n      bitgoInstChallengeProofSignature.toString('hex'),\n      bitgoNitroChallengeProofSignature.toString('hex')\n    );\n  }\n\n  /**\n   * Uploads the signed challenges and their proofs on the enterprise.\n   * This initiates ecdsa signing for the enterprise users.\n   * @param bitgo\n   * @param entId - enterprise to enable ecdsa signing on\n   * @param entChallenge - client side generated ent challenge with ZK proofs\n   * @param entChallengeSignature - signature on enterprise challenge\n   * @param bitgoIntChallengeSignature - signature on BitGo's institutional HSM challenge\n   * @param bitgoNitroChallengeSignature - signature on BitGo's nitro HSM challenge\n   */\n  static async uploadChallengesToEnterprise(\n    bitgo: BitGoBase,\n    entId: string,\n    entChallenge: EcdsaTypes.SerializedNtilde | EcdsaTypes.SerializedNtildeWithProofs,\n    entChallengeSignature: string,\n    bitgoIntChallengeSignature: string,\n    bitgoNitroChallengeSignature: string\n  ): Promise<void> {\n    const body = {\n      enterprise: {\n        ntilde: entChallenge.ntilde,\n        h1: entChallenge.h1,\n        h2: entChallenge.h2,\n        verifiers: {\n          adminSignature: entChallengeSignature,\n        },\n      },\n      bitgoInstitutionalHsm: {\n        verifiers: {\n          adminSignature: bitgoIntChallengeSignature,\n        },\n      },\n      bitgoNitroHsm: {\n        verifiers: {\n          adminSignature: bitgoNitroChallengeSignature,\n        },\n      },\n    };\n    if ('ntildeProof' in entChallenge) {\n      body.enterprise['ntildeProof'] = entChallenge.ntildeProof;\n    }\n    await bitgo\n      .put(bitgo.url(`/enterprise/${entId}/tssconfig/ecdsa/challenge`, 2))\n      .send(body)\n      .result();\n  }\n}\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!