PHP WebShell
Текущая директория: /opt/BitGoJS/modules/sdk-core/dist/src/bitgo/tss/eddsa
Просмотр файла: eddsa.js
"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.sendSignatureShare = exports.getTxRequest = void 0;
exports.createCombinedKey = createCombinedKey;
exports.createUserSignShare = createUserSignShare;
exports.createUserToBitGoGShare = createUserToBitGoGShare;
exports.offerUserToBitgoRShare = offerUserToBitgoRShare;
exports.getBitgoToUserRShare = getBitgoToUserRShare;
exports.sendUserToBitgoGShare = sendUserToBitgoGShare;
exports.encryptYShare = encryptYShare;
exports.getInitializedMpcInstance = getInitializedMpcInstance;
exports.getTSSSignature = getTSSSignature;
exports.verifyWalletSignature = verifyWalletSignature;
const assert_1 = __importDefault(require("assert"));
const libsodium_wrappers_sumo_1 = __importDefault(require("libsodium-wrappers-sumo"));
const tss_1 = __importDefault(require("./../../../account-lib/mpc/tss"));
const types_1 = require("../types");
const utils_1 = require("../../utils");
const sdk_lib_mpc_1 = require("@bitgo/sdk-lib-mpc");
const _ = require("lodash");
const common_1 = require("../common");
Object.defineProperty(exports, "getTxRequest", { enumerable: true, get: function () { return common_1.getTxRequest; } });
Object.defineProperty(exports, "sendSignatureShare", { enumerable: true, get: function () { return common_1.sendSignatureShare; } });
/**
* Combines YShares to combine the final TSS key
* This can only be used to create the User or Backup key since it requires the common keychain from BitGo first
*
* @param params.keyShare - TSS key share
* @param params.encryptedYShares - encrypted YShares with information on how to decrypt
* @param params.commonKeychain - expected common keychain of the combined key
* @returns {CombinedKey} combined TSS key
*/
async function createCombinedKey(params) {
await tss_1.default.initialize();
const MPC = new tss_1.default();
const { keyShare, encryptedYShares, commonKeychain } = params;
const yShares = [];
let bitgoYShare;
let userYShare;
let backupYShare;
for (const encryptedYShare of encryptedYShares) {
const privateShare = await (0, utils_1.readSignedMessage)(encryptedYShare.yShare.encryptedPrivateShare, encryptedYShare.senderPublicArmor, encryptedYShare.recipientPrivateArmor);
const yShare = {
i: encryptedYShare.yShare.i,
j: encryptedYShare.yShare.j,
y: encryptedYShare.yShare.publicShare.slice(0, 64),
v: encryptedYShare.yShare.publicShare.slice(64, 128),
u: privateShare.slice(0, 64),
chaincode: privateShare.slice(64),
};
switch (encryptedYShare.yShare.j) {
case 1:
userYShare = yShare;
break;
case 2:
backupYShare = yShare;
break;
case 3:
bitgoYShare = yShare;
break;
default:
throw new Error('Invalid YShare index');
}
yShares.push(yShare);
}
const combinedKey = MPC.keyCombine(keyShare.uShare, yShares);
if (combinedKey.pShare.y + combinedKey.pShare.chaincode !== commonKeychain) {
throw new Error('Common keychains do not match');
}
if (!bitgoYShare) {
throw new Error('Missing BitGo Y Share');
}
const signingMaterial = {
uShare: keyShare.uShare,
bitgoYShare,
backupYShare,
userYShare,
};
return {
signingMaterial,
commonKeychain,
};
}
/**
* Creates the User Sign Share containing the User XShare , the User to Bitgo RShare and User to Bitgo commitment
*
* @param {Buffer} signablePayload - the signablePayload as a buffer
* @param {PShare} pShare - User's signing material
* @returns {Promise<SignShare>} - User Sign Share
*/
async function createUserSignShare(signablePayload, pShare) {
const MPC = await tss_1.default.initialize();
if (pShare.i !== types_1.ShareKeyPosition.USER) {
throw new Error('Invalid PShare, PShare doesnt belong to the User');
}
const jShare = { i: types_1.ShareKeyPosition.BITGO, j: types_1.ShareKeyPosition.USER };
return MPC.signShare(signablePayload, pShare, [jShare]);
}
/**
* Creates the User to Bitgo GShare
*
* @param {SignShare} userSignShare - the User Sign Share
* @param {SignatureShareRecord} bitgoToUserRShare - the Bitgo to User RShare
* @param {YShare} backupToUserYShare - the backup key Y share received during wallet creation
* @param {Buffer} signablePayload - the signable payload from a tx
* @param {CommitmentShareRecord} [bitgoToUserCommitment] - the Bitgo to User Commitment
* @returns {Promise<GShare>} - the User to Bitgo GShare
*/
async function createUserToBitGoGShare(userSignShare, bitgoToUserRShare, backupToUserYShare, bitgoToUserYShare, signablePayload, bitgoToUserCommitment) {
if (userSignShare.xShare.i !== types_1.ShareKeyPosition.USER) {
throw new Error('Invalid XShare, doesnt belong to the User');
}
if (bitgoToUserRShare.from !== utils_1.SignatureShareType.BITGO || bitgoToUserRShare.to !== utils_1.SignatureShareType.USER) {
throw new Error('Invalid RShare, is not from Bitgo to User');
}
if (backupToUserYShare.i !== types_1.ShareKeyPosition.USER) {
throw new Error('Invalid YShare, doesnt belong to the User');
}
if (backupToUserYShare.j !== types_1.ShareKeyPosition.BACKUP) {
throw new Error('Invalid YShare, is not backup key');
}
if (bitgoToUserCommitment.from !== utils_1.SignatureShareType.BITGO || bitgoToUserCommitment.to !== utils_1.SignatureShareType.USER) {
throw new Error('Invalid Commitment, is not from Bitgo to User');
}
if (bitgoToUserCommitment.type !== utils_1.CommitmentType.COMMITMENT) {
throw new Error('Invalid Commitment type, got: ' + bitgoToUserCommitment.type + ' expected: commitment');
}
let v, r, R;
if (bitgoToUserRShare.share.length > 128) {
v = bitgoToUserRShare.share.substring(0, 64);
r = bitgoToUserRShare.share.substring(64, 128);
R = bitgoToUserRShare.share.substring(128, 192);
}
else {
r = bitgoToUserRShare.share.substring(0, 64);
R = bitgoToUserRShare.share.substring(64, 128);
}
const MPC = await tss_1.default.initialize();
const updatedBitgoToUserRShare = {
i: types_1.ShareKeyPosition.USER,
j: types_1.ShareKeyPosition.BITGO,
u: bitgoToUserYShare.u,
v,
r,
R,
commitment: bitgoToUserCommitment.share,
};
return MPC.sign(signablePayload, userSignShare.xShare, [updatedBitgoToUserRShare], [backupToUserYShare]);
}
/**
* Sends the User to Bitgo RShare to Bitgo
* @param {BitGoBase} bitgo - the bitgo instance
* @param {String} walletId - the wallet id
* @param {String} txRequestId - the txRequest Id
* @param {SignShare} userSignShare - the user Sign Share
* @param {String} encryptedSignerShare - signer share encrypted to bitgo key
* @returns {Promise<void>}
* @param {IRequestTracer} reqId - the request tracer request id
*/
async function offerUserToBitgoRShare(bitgo, walletId, txRequestId, userSignShare, encryptedSignerShare, apiMode = 'lite', reqId) {
const rShare = userSignShare.rShares[types_1.ShareKeyPosition.BITGO];
if (_.isNil(rShare)) {
throw new Error('userToBitgo RShare not found');
}
if (rShare.i !== types_1.ShareKeyPosition.BITGO || rShare.j !== types_1.ShareKeyPosition.USER) {
throw new Error('Invalid RShare, is not from User to Bitgo');
}
const signatureShare = {
from: utils_1.SignatureShareType.USER,
to: utils_1.SignatureShareType.BITGO,
share: rShare.r + rShare.R,
};
// TODO (BG-57944): implement message signing for EDDSA
await (0, common_1.sendSignatureShare)(bitgo, walletId, txRequestId, signatureShare, utils_1.RequestType.tx, encryptedSignerShare, 'eddsa', apiMode, undefined, reqId);
}
/**
* Gets the Bitgo to User RShare from Bitgo
*
* @param {BitGoBase} bitgo - the bitgo instance
* @param {String} walletId - the wallet id
* @param {String} txRequestId - the txRequest Id
* @param {IRequestTracer} reqId - the request tracer request id
* @returns {Promise<SignatureShareRecord>} - a Signature Share
*/
async function getBitgoToUserRShare(bitgo, walletId, txRequestId, reqId) {
const txRequest = await (0, common_1.getTxRequest)(bitgo, walletId, txRequestId, reqId);
let signatureShares;
if (txRequest.apiVersion === 'full') {
(0, assert_1.default)(txRequest.transactions, 'transactions required as part of txRequest');
signatureShares = txRequest.transactions[0].signatureShares;
}
else {
signatureShares = txRequest.signatureShares;
}
if (_.isNil(signatureShares) || _.isEmpty(signatureShares)) {
throw new Error(`No signatures shares found for id: ${txRequestId}`);
}
// at this point we expect the only share to be the RShare
const bitgoToUserRShare = signatureShares.find((sigShare) => sigShare.from === utils_1.SignatureShareType.BITGO && sigShare.to === utils_1.SignatureShareType.USER);
if (_.isNil(bitgoToUserRShare)) {
throw new Error(`Bitgo to User RShare not found for id: ${txRequestId}`);
}
return bitgoToUserRShare;
}
/**
* Sends the User to Bitgo GShare to Bitgo
*
* @param {BitGoBase} bitgo - the bitgo instance
* @param {String} walletId - the wallet id
* @param {String} txRequestId - the txRequest Id
* @param {GShare} userToBitgoGShare - the User to Bitgo GShare
* @param {IRequestTracer} reqId - the request tracer request id
* @returns {Promise<void>}
*/
async function sendUserToBitgoGShare(bitgo, walletId, txRequestId, userToBitgoGShare, apiMode = 'lite', reqId) {
if (userToBitgoGShare.i !== types_1.ShareKeyPosition.USER) {
throw new Error('Invalid GShare, doesnt belong to the User');
}
const signatureShare = {
from: utils_1.SignatureShareType.USER,
to: utils_1.SignatureShareType.BITGO,
share: userToBitgoGShare.R + userToBitgoGShare.gamma,
};
// TODO (BG-57944): implement message signing for EDDSA
await (0, common_1.sendSignatureShare)(bitgo, walletId, txRequestId, signatureShare, utils_1.RequestType.tx, undefined, 'eddsa', apiMode, undefined, reqId);
}
/**
* Prepares a YShare to be exchanged with other key holders.
* Output is in a format that is usable within BitGo's ecosystem.
*
* @param params.keyShare - TSS key share of the party preparing exchange materials
* @param params.recipientIndex - index of the recipient (1, 2, or 3)
* @param params.recipientGpgPublicArmor - recipient's public gpg key in armor format
* @param params.senderGpgPrivateArmor - sender's private gpg key in armor format
* @returns { EncryptedYShare } encrypted Y Share
*/
async function encryptYShare(params) {
const { keyShare, recipientIndex, recipientGpgPublicArmor, senderGpgPrivateArmor } = params;
const yShare = keyShare.yShares[recipientIndex];
if (!yShare) {
throw new Error('Invalid recipient');
}
const publicShare = Buffer.concat([
Buffer.from(keyShare.uShare.y, 'hex'),
Buffer.from(yShare.v, 'hex'),
Buffer.from(keyShare.uShare.chaincode, 'hex'),
]).toString('hex');
const privateShare = Buffer.concat([Buffer.from(yShare.u, 'hex'), Buffer.from(yShare.chaincode, 'hex')]).toString('hex');
const encryptedPrivateShare = await (0, utils_1.encryptAndSignText)(privateShare, recipientGpgPublicArmor, senderGpgPrivateArmor);
return {
i: yShare.i,
j: yShare.j,
publicShare,
encryptedPrivateShare,
};
}
/**
*
* Initializes Eddsa instance
*
* @returns {Promise<Eddsa>} the Eddsa instance
*/
async function getInitializedMpcInstance() {
const hdTree = await sdk_lib_mpc_1.Ed25519Bip32HdTree.initialize();
return await tss_1.default.initialize(hdTree);
}
/**
*
* Generates a TSS signature using the user and backup key
*
* @param {UserSigningMaterial} userSigningMaterial decrypted user TSS key
* @param {BackupSigningMaterial} backupSigningMaterial decrypted backup TSS key
* @param {string} path bip32 derivation path
* @param {BaseTransaction} transaction the transaction to sign
* @returns {Buffer} the signature
*/
async function getTSSSignature(userSigningMaterial, backupSigningMaterial, path = 'm/0', transaction) {
const MPC = await getInitializedMpcInstance();
const userCombine = MPC.keyCombine(userSigningMaterial.uShare, [
userSigningMaterial.bitgoYShare,
userSigningMaterial.backupYShare,
]);
const backupCombine = MPC.keyCombine(backupSigningMaterial.uShare, [
backupSigningMaterial.bitgoYShare,
backupSigningMaterial.userYShare,
]);
const userSubkey = MPC.keyDerive(userSigningMaterial.uShare, [userSigningMaterial.bitgoYShare, userSigningMaterial.backupYShare], path);
const backupSubkey = MPC.keyCombine(backupSigningMaterial.uShare, [
userSubkey.yShares[2],
backupSigningMaterial.bitgoYShare,
]);
const messageBuffer = transaction.signablePayload;
const userSignShare = MPC.signShare(messageBuffer, userSubkey.pShare, [userCombine.jShares[2]]);
const backupSignShare = MPC.signShare(messageBuffer, backupSubkey.pShare, [backupCombine.jShares[1]]);
const userSign = MPC.sign(messageBuffer, userSignShare.xShare, [backupSignShare.rShares[1]], [userSigningMaterial.bitgoYShare]);
const backupSign = MPC.sign(messageBuffer, backupSignShare.xShare, [userSignShare.rShares[2]], [backupSigningMaterial.bitgoYShare]);
const signature = MPC.signCombine([userSign, backupSign]);
const result = MPC.verify(messageBuffer, signature);
if (!result) {
throw new Error('Invalid signature');
}
const rawSignature = Buffer.concat([Buffer.from(signature.R, 'hex'), Buffer.from(signature.sigma, 'hex')]);
return rawSignature;
}
/**
* Verifies that a TSS wallet signature was produced with the expected key and that the signed data contains the
* expected common keychain, the expected user and backup key ids as well as the public share that is generated from the
* private share that was passed in.
*/
async function verifyWalletSignature(params) {
const rawNotations = await (0, common_1.commonVerifyWalletSignature)(params);
const { decryptedShare, verifierIndex } = params;
const publicShare = Buffer.from(await libsodium_wrappers_sumo_1.default.crypto_scalarmult_ed25519_base_noclamp(Buffer.from(decryptedShare.slice(0, 64), 'hex'))).toString('hex') + decryptedShare.slice(64);
const publicShareRawNotationIndex = 2 + verifierIndex;
(0, assert_1.default)(publicShare === Buffer.from(rawNotations[publicShareRawNotationIndex].value).toString(), 'bitgo share mismatch');
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"eddsa.js","sourceRoot":"","sources":["../../../../../src/bitgo/tss/eddsa/eddsa.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\nimport openpgp from 'openpgp';\nimport sodium from 'libsodium-wrappers-sumo';\nimport Eddsa, { GShare, JShare, KeyShare, PShare, RShare, SignShare, YShare } from './../../../account-lib/mpc/tss';\nimport { BitGoBase } from '../../bitgoBase';\nimport {\n  DecryptableYShare,\n  CombinedKey,\n  SigningMaterial,\n  EncryptedYShare,\n  UserSigningMaterial,\n  BackupSigningMaterial,\n} from './types';\nimport { ShareKeyPosition } from '../types';\nimport {\n  encryptAndSignText,\n  readSignedMessage,\n  SignatureShareRecord,\n  SignatureShareType,\n  RequestType,\n  CommitmentShareRecord,\n  CommitmentType,\n} from '../../utils';\nimport { BaseTransaction } from '../../../account-lib';\nimport { Ed25519Bip32HdTree } from '@bitgo/sdk-lib-mpc';\nimport _ = require('lodash');\nimport { commonVerifyWalletSignature, getTxRequest, sendSignatureShare } from '../common';\nimport { IRequestTracer } from '../../../api';\n\nexport { getTxRequest, sendSignatureShare };\n\n/**\n * Combines YShares to combine the final TSS key\n * This can only be used to create the User or Backup key since it requires the common keychain from BitGo first\n *\n * @param params.keyShare - TSS key share\n * @param params.encryptedYShares - encrypted YShares with information on how to decrypt\n * @param params.commonKeychain - expected common keychain of the combined key\n * @returns {CombinedKey} combined TSS key\n */\nexport async function createCombinedKey(params: {\n  keyShare: KeyShare;\n  encryptedYShares: DecryptableYShare[];\n  commonKeychain: string;\n}): Promise<CombinedKey> {\n  await Eddsa.initialize();\n  const MPC = new Eddsa();\n\n  const { keyShare, encryptedYShares, commonKeychain } = params;\n  const yShares: YShare[] = [];\n\n  let bitgoYShare: YShare | undefined;\n  let userYShare: YShare | undefined;\n  let backupYShare: YShare | undefined;\n\n  for (const encryptedYShare of encryptedYShares) {\n    const privateShare = await readSignedMessage(\n      encryptedYShare.yShare.encryptedPrivateShare,\n      encryptedYShare.senderPublicArmor,\n      encryptedYShare.recipientPrivateArmor\n    );\n\n    const yShare: YShare = {\n      i: encryptedYShare.yShare.i,\n      j: encryptedYShare.yShare.j,\n      y: encryptedYShare.yShare.publicShare.slice(0, 64),\n      v: encryptedYShare.yShare.publicShare.slice(64, 128),\n      u: privateShare.slice(0, 64),\n      chaincode: privateShare.slice(64),\n    };\n\n    switch (encryptedYShare.yShare.j) {\n      case 1:\n        userYShare = yShare;\n        break;\n      case 2:\n        backupYShare = yShare;\n        break;\n      case 3:\n        bitgoYShare = yShare;\n        break;\n      default:\n        throw new Error('Invalid YShare index');\n    }\n\n    yShares.push(yShare);\n  }\n\n  const combinedKey = MPC.keyCombine(keyShare.uShare, yShares);\n  if (combinedKey.pShare.y + combinedKey.pShare.chaincode !== commonKeychain) {\n    throw new Error('Common keychains do not match');\n  }\n  if (!bitgoYShare) {\n    throw new Error('Missing BitGo Y Share');\n  }\n\n  const signingMaterial: SigningMaterial = {\n    uShare: keyShare.uShare,\n    bitgoYShare,\n    backupYShare,\n    userYShare,\n  };\n\n  return {\n    signingMaterial,\n    commonKeychain,\n  };\n}\n\n/**\n * Creates the User Sign Share containing the User XShare , the User to Bitgo RShare and User to Bitgo commitment\n *\n * @param {Buffer} signablePayload - the signablePayload as a buffer\n * @param {PShare} pShare - User's signing material\n * @returns {Promise<SignShare>} - User Sign Share\n */\nexport async function createUserSignShare(signablePayload: Buffer, pShare: PShare): Promise<SignShare> {\n  const MPC = await Eddsa.initialize();\n\n  if (pShare.i !== ShareKeyPosition.USER) {\n    throw new Error('Invalid PShare, PShare doesnt belong to the User');\n  }\n  const jShare: JShare = { i: ShareKeyPosition.BITGO, j: ShareKeyPosition.USER };\n  return MPC.signShare(signablePayload, pShare, [jShare]);\n}\n\n/**\n * Creates the User to Bitgo GShare\n *\n * @param {SignShare} userSignShare - the User Sign Share\n * @param {SignatureShareRecord} bitgoToUserRShare - the Bitgo to User RShare\n * @param {YShare} backupToUserYShare - the backup key Y share received during wallet creation\n * @param {Buffer} signablePayload - the signable payload from a tx\n * @param {CommitmentShareRecord} [bitgoToUserCommitment] - the Bitgo to User Commitment\n * @returns {Promise<GShare>} - the User to Bitgo GShare\n */\nexport async function createUserToBitGoGShare(\n  userSignShare: SignShare,\n  bitgoToUserRShare: SignatureShareRecord,\n  backupToUserYShare: YShare,\n  bitgoToUserYShare: YShare,\n  signablePayload: Buffer,\n  bitgoToUserCommitment: CommitmentShareRecord\n): Promise<GShare> {\n  if (userSignShare.xShare.i !== ShareKeyPosition.USER) {\n    throw new Error('Invalid XShare, doesnt belong to the User');\n  }\n  if (bitgoToUserRShare.from !== SignatureShareType.BITGO || bitgoToUserRShare.to !== SignatureShareType.USER) {\n    throw new Error('Invalid RShare, is not from Bitgo to User');\n  }\n  if (backupToUserYShare.i !== ShareKeyPosition.USER) {\n    throw new Error('Invalid YShare, doesnt belong to the User');\n  }\n  if (backupToUserYShare.j !== ShareKeyPosition.BACKUP) {\n    throw new Error('Invalid YShare, is not backup key');\n  }\n  if (bitgoToUserCommitment.from !== SignatureShareType.BITGO || bitgoToUserCommitment.to !== SignatureShareType.USER) {\n    throw new Error('Invalid Commitment, is not from Bitgo to User');\n  }\n  if (bitgoToUserCommitment.type !== CommitmentType.COMMITMENT) {\n    throw new Error('Invalid Commitment type, got: ' + bitgoToUserCommitment.type + ' expected: commitment');\n  }\n\n  let v, r, R;\n  if (bitgoToUserRShare.share.length > 128) {\n    v = bitgoToUserRShare.share.substring(0, 64);\n    r = bitgoToUserRShare.share.substring(64, 128);\n    R = bitgoToUserRShare.share.substring(128, 192);\n  } else {\n    r = bitgoToUserRShare.share.substring(0, 64);\n    R = bitgoToUserRShare.share.substring(64, 128);\n  }\n\n  const MPC = await Eddsa.initialize();\n\n  const updatedBitgoToUserRShare: RShare = {\n    i: ShareKeyPosition.USER,\n    j: ShareKeyPosition.BITGO,\n    u: bitgoToUserYShare.u,\n    v,\n    r,\n    R,\n    commitment: bitgoToUserCommitment.share,\n  };\n\n  return MPC.sign(signablePayload, userSignShare.xShare, [updatedBitgoToUserRShare], [backupToUserYShare]);\n}\n\n/**\n * Sends the User to Bitgo RShare to Bitgo\n * @param {BitGoBase} bitgo - the bitgo instance\n * @param {String} walletId - the wallet id\n * @param {String} txRequestId - the txRequest Id\n * @param {SignShare} userSignShare - the user Sign Share\n * @param {String} encryptedSignerShare - signer share encrypted to bitgo key\n * @returns {Promise<void>}\n * @param {IRequestTracer} reqId - the request tracer request id\n */\nexport async function offerUserToBitgoRShare(\n  bitgo: BitGoBase,\n  walletId: string,\n  txRequestId: string,\n  userSignShare: SignShare,\n  encryptedSignerShare: string,\n  apiMode: 'full' | 'lite' = 'lite',\n  reqId?: IRequestTracer\n): Promise<void> {\n  const rShare: RShare = userSignShare.rShares[ShareKeyPosition.BITGO];\n  if (_.isNil(rShare)) {\n    throw new Error('userToBitgo RShare not found');\n  }\n  if (rShare.i !== ShareKeyPosition.BITGO || rShare.j !== ShareKeyPosition.USER) {\n    throw new Error('Invalid RShare, is not from User to Bitgo');\n  }\n  const signatureShare: SignatureShareRecord = {\n    from: SignatureShareType.USER,\n    to: SignatureShareType.BITGO,\n    share: rShare.r + rShare.R,\n  };\n\n  // TODO (BG-57944): implement message signing for EDDSA\n  await sendSignatureShare(\n    bitgo,\n    walletId,\n    txRequestId,\n    signatureShare,\n    RequestType.tx,\n    encryptedSignerShare,\n    'eddsa',\n    apiMode,\n    undefined,\n    reqId\n  );\n}\n\n/**\n * Gets the Bitgo to User RShare from Bitgo\n *\n * @param {BitGoBase} bitgo - the bitgo instance\n * @param {String} walletId - the wallet id\n * @param {String} txRequestId - the txRequest Id\n * @param {IRequestTracer} reqId - the request tracer request id\n * @returns {Promise<SignatureShareRecord>} - a Signature Share\n */\nexport async function getBitgoToUserRShare(\n  bitgo: BitGoBase,\n  walletId: string,\n  txRequestId: string,\n  reqId?: IRequestTracer\n): Promise<SignatureShareRecord> {\n  const txRequest = await getTxRequest(bitgo, walletId, txRequestId, reqId);\n  let signatureShares;\n  if (txRequest.apiVersion === 'full') {\n    assert(txRequest.transactions, 'transactions required as part of txRequest');\n    signatureShares = txRequest.transactions[0].signatureShares;\n  } else {\n    signatureShares = txRequest.signatureShares;\n  }\n  if (_.isNil(signatureShares) || _.isEmpty(signatureShares)) {\n    throw new Error(`No signatures shares found for id: ${txRequestId}`);\n  }\n  // at this point we expect the only share to be the RShare\n  const bitgoToUserRShare = signatureShares.find(\n    (sigShare) => sigShare.from === SignatureShareType.BITGO && sigShare.to === SignatureShareType.USER\n  );\n  if (_.isNil(bitgoToUserRShare)) {\n    throw new Error(`Bitgo to User RShare not found for id: ${txRequestId}`);\n  }\n  return bitgoToUserRShare;\n}\n\n/**\n * Sends the User to Bitgo GShare to Bitgo\n *\n * @param {BitGoBase} bitgo - the bitgo instance\n * @param {String} walletId - the wallet id\n * @param {String} txRequestId - the txRequest Id\n * @param {GShare} userToBitgoGShare - the User to Bitgo GShare\n * @param {IRequestTracer} reqId - the request tracer request id\n * @returns {Promise<void>}\n */\nexport async function sendUserToBitgoGShare(\n  bitgo: BitGoBase,\n  walletId: string,\n  txRequestId: string,\n  userToBitgoGShare: GShare,\n  apiMode: 'full' | 'lite' = 'lite',\n  reqId?: IRequestTracer\n): Promise<void> {\n  if (userToBitgoGShare.i !== ShareKeyPosition.USER) {\n    throw new Error('Invalid GShare, doesnt belong to the User');\n  }\n  const signatureShare: SignatureShareRecord = {\n    from: SignatureShareType.USER,\n    to: SignatureShareType.BITGO,\n    share: userToBitgoGShare.R + userToBitgoGShare.gamma,\n  };\n\n  // TODO (BG-57944): implement message signing for EDDSA\n  await sendSignatureShare(\n    bitgo,\n    walletId,\n    txRequestId,\n    signatureShare,\n    RequestType.tx,\n    undefined,\n    'eddsa',\n    apiMode,\n    undefined,\n    reqId\n  );\n}\n\n/**\n * Prepares a YShare to be exchanged with other key holders.\n * Output is in a format that is usable within BitGo's ecosystem.\n *\n * @param params.keyShare - TSS key share of the party preparing exchange materials\n * @param params.recipientIndex - index of the recipient (1, 2, or 3)\n * @param params.recipientGpgPublicArmor - recipient's public gpg key in armor format\n * @param params.senderGpgPrivateArmor - sender's private gpg key in armor format\n * @returns { EncryptedYShare } encrypted Y Share\n */\nexport async function encryptYShare(params: {\n  keyShare: KeyShare;\n  recipientIndex: number;\n  recipientGpgPublicArmor: string;\n  senderGpgPrivateArmor: string;\n}): Promise<EncryptedYShare> {\n  const { keyShare, recipientIndex, recipientGpgPublicArmor, senderGpgPrivateArmor } = params;\n\n  const yShare = keyShare.yShares[recipientIndex];\n  if (!yShare) {\n    throw new Error('Invalid recipient');\n  }\n\n  const publicShare = Buffer.concat([\n    Buffer.from(keyShare.uShare.y, 'hex'),\n    Buffer.from(yShare.v!, 'hex'),\n    Buffer.from(keyShare.uShare.chaincode, 'hex'),\n  ]).toString('hex');\n\n  const privateShare = Buffer.concat([Buffer.from(yShare.u, 'hex'), Buffer.from(yShare.chaincode, 'hex')]).toString(\n    'hex'\n  );\n\n  const encryptedPrivateShare = await encryptAndSignText(privateShare, recipientGpgPublicArmor, senderGpgPrivateArmor);\n\n  return {\n    i: yShare.i,\n    j: yShare.j,\n    publicShare,\n    encryptedPrivateShare,\n  };\n}\n\n/**\n *\n * Initializes Eddsa instance\n *\n * @returns {Promise<Eddsa>} the Eddsa instance\n */\nexport async function getInitializedMpcInstance() {\n  const hdTree = await Ed25519Bip32HdTree.initialize();\n  return await Eddsa.initialize(hdTree);\n}\n\n/**\n *\n * Generates a TSS signature using the user and backup key\n *\n * @param {UserSigningMaterial} userSigningMaterial decrypted user TSS key\n * @param {BackupSigningMaterial} backupSigningMaterial decrypted backup TSS key\n * @param {string} path bip32 derivation path\n * @param {BaseTransaction} transaction the transaction to sign\n * @returns {Buffer} the signature\n */\nexport async function getTSSSignature(\n  userSigningMaterial: UserSigningMaterial,\n  backupSigningMaterial: BackupSigningMaterial,\n  path = 'm/0',\n  transaction: BaseTransaction\n): Promise<Buffer> {\n  const MPC = await getInitializedMpcInstance();\n\n  const userCombine = MPC.keyCombine(userSigningMaterial.uShare, [\n    userSigningMaterial.bitgoYShare,\n    userSigningMaterial.backupYShare,\n  ]);\n  const backupCombine = MPC.keyCombine(backupSigningMaterial.uShare, [\n    backupSigningMaterial.bitgoYShare,\n    backupSigningMaterial.userYShare,\n  ]);\n\n  const userSubkey = MPC.keyDerive(\n    userSigningMaterial.uShare,\n    [userSigningMaterial.bitgoYShare, userSigningMaterial.backupYShare],\n    path\n  );\n\n  const backupSubkey = MPC.keyCombine(backupSigningMaterial.uShare, [\n    userSubkey.yShares[2],\n    backupSigningMaterial.bitgoYShare,\n  ]);\n\n  const messageBuffer = transaction.signablePayload;\n  const userSignShare = MPC.signShare(messageBuffer, userSubkey.pShare, [userCombine.jShares[2]]);\n  const backupSignShare = MPC.signShare(messageBuffer, backupSubkey.pShare, [backupCombine.jShares[1]]);\n  const userSign = MPC.sign(\n    messageBuffer,\n    userSignShare.xShare,\n    [backupSignShare.rShares[1]],\n    [userSigningMaterial.bitgoYShare]\n  );\n  const backupSign = MPC.sign(\n    messageBuffer,\n    backupSignShare.xShare,\n    [userSignShare.rShares[2]],\n    [backupSigningMaterial.bitgoYShare]\n  );\n  const signature = MPC.signCombine([userSign, backupSign]);\n  const result = MPC.verify(messageBuffer, signature);\n  if (!result) {\n    throw new Error('Invalid signature');\n  }\n  const rawSignature = Buffer.concat([Buffer.from(signature.R, 'hex'), Buffer.from(signature.sigma, 'hex')]);\n  return rawSignature;\n}\n\n/**\n * Verifies that a TSS wallet signature was produced with the expected key and that the signed data contains the\n * expected common keychain, the expected user and backup key ids as well as the public share that is generated from the\n * private share that was passed in.\n */\nexport async function verifyWalletSignature(params: {\n  walletSignature: openpgp.Key;\n  bitgoPub: openpgp.Key;\n  commonKeychain: string;\n  userKeyId: string;\n  backupKeyId: string;\n  decryptedShare: string;\n  verifierIndex: 1 | 2; // the index of the verifier, 1 means user, 2 means backup\n}): Promise<void> {\n  const rawNotations = await commonVerifyWalletSignature(params);\n\n  const { decryptedShare, verifierIndex } = params;\n\n  const publicShare =\n    Buffer.from(\n      await sodium.crypto_scalarmult_ed25519_base_noclamp(Buffer.from(decryptedShare.slice(0, 64), 'hex'))\n    ).toString('hex') + decryptedShare.slice(64);\n  const publicShareRawNotationIndex = 2 + verifierIndex;\n\n  assert(\n    publicShare === Buffer.from(rawNotations[publicShareRawNotationIndex].value).toString(),\n    'bitgo share mismatch'\n  );\n}\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!