PHP WebShell
Текущая директория: /opt/BitGoJS/modules/sdk-core/dist/src/bitgo/utils
Просмотр файла: mpcUtils.js
"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.MpcUtils = void 0;
/**
* @prettier
*/
const assert_1 = __importDefault(require("assert"));
const openpgp_1 = require("openpgp");
const opengpgUtils_1 = require("./opengpgUtils");
const bitgoPubKeys_1 = require("../tss/bitgoPubKeys");
const statics_1 = require("@bitgo/statics");
class MpcUtils {
constructor(bitgo, baseCoin) {
this.bitgo = bitgo;
this.baseCoin = baseCoin;
}
async decryptPrivateShare(privateShare, userGpgKey) {
const privateShareMessage = await (0, openpgp_1.readMessage)({
armoredMessage: privateShare,
});
const userGpgPrivateKey = await (0, openpgp_1.readPrivateKey)({ armoredKey: userGpgKey.privateKey });
const decryptedPrivateShare = (await (0, openpgp_1.decrypt)({
message: privateShareMessage,
decryptionKeys: [userGpgPrivateKey],
format: 'utf8',
})).data;
return decryptedPrivateShare;
}
async createBitgoKeychainInWP(userGpgKey, backupGpgKey, userKeyShare, backupKeyShare, keyType, enterprise) {
const bitgoKey = (await (0, opengpgUtils_1.getBitgoGpgPubKey)(this.bitgo)).mpcV1;
if ((0, bitgoPubKeys_1.envRequiresBitgoPubGpgKeyConfig)(this.bitgo.getEnv())) {
// Ensure the public key is one of the expected BitGo public keys when in test or prod.
(0, assert_1.default)((0, bitgoPubKeys_1.isBitgoMpcPubKey)(bitgoKey.armor(), 'mpcv1'), 'Invalid BitGo GPG public key');
}
const encUserToBitGoMessage = await (0, opengpgUtils_1.encryptText)(userKeyShare.privateShare, bitgoKey);
const encBackupToBitGoMessage = await (0, opengpgUtils_1.encryptText)(backupKeyShare.privateShare, bitgoKey);
const createBitGoMPCParams = {
keyType,
source: 'bitgo',
keyShares: [
{
from: 'user',
to: 'bitgo',
publicShare: userKeyShare.publicShare,
privateShare: encUserToBitGoMessage,
privateShareProof: userKeyShare.privateShareProof,
vssProof: userKeyShare.vssProof,
},
{
from: 'backup',
to: 'bitgo',
publicShare: backupKeyShare.publicShare,
privateShare: encBackupToBitGoMessage,
privateShareProof: backupKeyShare.privateShareProof,
vssProof: backupKeyShare.vssProof,
},
],
userGPGPublicKey: userGpgKey.publicKey,
backupGPGPublicKey: backupGpgKey.publicKey,
enterprise: enterprise,
};
return await this.baseCoin.keychains().add(createBitGoMPCParams);
}
/**
* This function would be responsible for populating intents
* based on the type of coin / sig scheme the coin uses
* @param {IBaseCoin} baseCoin
* @param {PrebuildTransactionWithIntentOptions} params
* @returns {Record<string, unknown>}
*/
populateIntent(baseCoin, params) {
const chain = this.baseCoin.getChain();
if (!['acceleration', 'fillNonce', 'transferToken'].includes(params.intentType)) {
(0, assert_1.default)(params.recipients, `'recipients' is a required parameter for ${params.intentType} intent`);
}
const intentRecipients = params.recipients?.map((recipient) => {
const formattedRecipient = {
address: { address: recipient.address },
amount: { value: `${recipient.amount}`, symbol: recipient.tokenName ? recipient.tokenName : chain },
};
if (recipient.data) {
formattedRecipient.data = recipient.data;
}
const { tokenData } = recipient;
if (tokenData && (tokenData.tokenContractAddress || tokenData.tokenName)) {
// token related recipient data gets validated in WP
if (!(tokenData.tokenType && tokenData.tokenQuantity)) {
throw new Error('token type and quantity is required to request a transaction with intent to transfer a token');
}
tokenData.tokenName = this.getTokenName(baseCoin, tokenData);
if (tokenData.tokenName) {
formattedRecipient.amount.symbol = tokenData.tokenName;
}
formattedRecipient.tokenData = tokenData;
}
return formattedRecipient;
});
const baseIntent = {
intentType: params.intentType,
sequenceId: params.sequenceId,
comment: params.comment,
nonce: params.nonce,
recipients: intentRecipients,
};
if (baseCoin.getFamily() === 'eth' || baseCoin.getFamily() === 'polygon' || baseCoin.getFamily() === 'bsc') {
switch (params.intentType) {
case 'payment':
case 'transferToken':
case 'fillNonce':
return {
...baseIntent,
selfSend: params.selfSend,
feeOptions: params.feeOptions,
hopParams: params.hopParams,
isTss: params.isTss,
nonce: params.nonce,
custodianTransactionId: params.custodianTransactionId,
receiveAddress: params.receiveAddress,
};
case 'acceleration':
return {
...baseIntent,
txid: params.lowFeeTxid,
receiveAddress: params.receiveAddress,
feeOptions: params.feeOptions,
};
default:
throw new Error(`Unsupported intent type ${params.intentType}`);
}
}
this.baseCoin.setCoinSpecificFieldsInIntent(baseIntent, params);
if (params.feeOptions !== undefined) {
return {
...baseIntent,
memo: params.memo?.value,
token: params.tokenName,
enableTokens: params.enableTokens,
feeOptions: params.feeOptions,
};
}
return {
...baseIntent,
memo: params.memo?.value,
token: params.tokenName,
enableTokens: params.enableTokens,
};
}
getTokenName(baseCoin, tokenData) {
if (tokenData.tokenName) {
return tokenData.tokenName;
}
const networkPrefix = baseCoin.getConfig().network.type === statics_1.NetworkType.TESTNET ? 't' : '';
const tokenStaticsKey = `${networkPrefix}${baseCoin.getFamily()}:${tokenData.tokenContractAddress}`;
if (statics_1.coins.has(tokenStaticsKey)) {
const tokenStatics = statics_1.coins.get(tokenStaticsKey);
tokenData.tokenName = tokenStatics.name;
}
return tokenData.tokenName;
}
}
exports.MpcUtils = MpcUtils;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mpcUtils.js","sourceRoot":"","sources":["../../../../src/bitgo/utils/mpcUtils.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport assert from 'assert';\nimport { decrypt, readMessage, readPrivateKey, SerializedKeyPair } from 'openpgp';\nimport { IBaseCoin, KeychainsTriplet } from '../baseCoin';\nimport { BitGoBase } from '../bitgoBase';\nimport { AddKeychainOptions, Keychain, KeyType } from '../keychain';\nimport { encryptText, getBitgoGpgPubKey } from './opengpgUtils';\nimport {\n  IntentRecipient,\n  PopulatedIntent,\n  PrebuildTransactionWithIntentOptions,\n  TokenTransferRecipientParams,\n} from './tss/baseTypes';\nimport { envRequiresBitgoPubGpgKeyConfig, isBitgoMpcPubKey } from '../tss/bitgoPubKeys';\nimport { coins, NetworkType } from '@bitgo/statics';\n\nexport interface MpcKeyShare {\n  publicShare: string;\n  privateShare: string;\n  privateShareProof?: string;\n  vssProof?: string;\n}\n\nexport abstract class MpcUtils {\n  protected bitgo: BitGoBase;\n  protected baseCoin: IBaseCoin;\n\n  constructor(bitgo: BitGoBase, baseCoin: IBaseCoin) {\n    this.bitgo = bitgo;\n    this.baseCoin = baseCoin;\n  }\n\n  protected async decryptPrivateShare(privateShare: string, userGpgKey: SerializedKeyPair<string>): Promise<string> {\n    const privateShareMessage = await readMessage({\n      armoredMessage: privateShare,\n    });\n    const userGpgPrivateKey = await readPrivateKey({ armoredKey: userGpgKey.privateKey });\n\n    const decryptedPrivateShare = (\n      await decrypt({\n        message: privateShareMessage,\n        decryptionKeys: [userGpgPrivateKey],\n        format: 'utf8',\n      })\n    ).data;\n\n    return decryptedPrivateShare;\n  }\n\n  protected async createBitgoKeychainInWP(\n    userGpgKey: SerializedKeyPair<string>,\n    backupGpgKey: SerializedKeyPair<string>,\n    userKeyShare: MpcKeyShare,\n    backupKeyShare: MpcKeyShare,\n    keyType: KeyType,\n    enterprise?: string\n  ): Promise<Keychain> {\n    const bitgoKey = (await getBitgoGpgPubKey(this.bitgo)).mpcV1;\n    if (envRequiresBitgoPubGpgKeyConfig(this.bitgo.getEnv())) {\n      // Ensure the public key is one of the expected BitGo public keys when in test or prod.\n      assert(isBitgoMpcPubKey(bitgoKey.armor(), 'mpcv1'), 'Invalid BitGo GPG public key');\n    }\n    const encUserToBitGoMessage = await encryptText(userKeyShare.privateShare, bitgoKey);\n    const encBackupToBitGoMessage = await encryptText(backupKeyShare.privateShare, bitgoKey);\n\n    const createBitGoMPCParams: AddKeychainOptions = {\n      keyType,\n      source: 'bitgo',\n      keyShares: [\n        {\n          from: 'user',\n          to: 'bitgo',\n          publicShare: userKeyShare.publicShare,\n          privateShare: encUserToBitGoMessage,\n          privateShareProof: userKeyShare.privateShareProof,\n          vssProof: userKeyShare.vssProof,\n        },\n        {\n          from: 'backup',\n          to: 'bitgo',\n          publicShare: backupKeyShare.publicShare,\n          privateShare: encBackupToBitGoMessage,\n          privateShareProof: backupKeyShare.privateShareProof,\n          vssProof: backupKeyShare.vssProof,\n        },\n      ],\n      userGPGPublicKey: userGpgKey.publicKey,\n      backupGPGPublicKey: backupGpgKey.publicKey,\n      enterprise: enterprise,\n    };\n\n    return await this.baseCoin.keychains().add(createBitGoMPCParams);\n  }\n\n  /**\n   * Creates User, Backup, and BitGo MPC Keychains.\n   *\n   * @param params.passphrase - passphrase used to encrypt signing materials created for User and Backup\n   * @param params.enterprise - optional enterprise id that will be attached to the BitGo Keychain\n   * @param params.originalPasscodeEncryptionCode - optional encryption code used to reset the user's password, if absent, password recovery will not work\n   */\n  abstract createKeychains(params: {\n    passphrase: string;\n    enterprise?: string;\n    originalPasscodeEncryptionCode?: string;\n  }): Promise<KeychainsTriplet>;\n\n  /**\n   * This function would be responsible for populating intents\n   * based on the type of coin / sig scheme the coin uses\n   * @param {IBaseCoin} baseCoin\n   * @param {PrebuildTransactionWithIntentOptions} params\n   * @returns {Record<string, unknown>}\n   */\n  populateIntent(baseCoin: IBaseCoin, params: PrebuildTransactionWithIntentOptions): PopulatedIntent {\n    const chain = this.baseCoin.getChain();\n\n    if (!['acceleration', 'fillNonce', 'transferToken'].includes(params.intentType)) {\n      assert(params.recipients, `'recipients' is a required parameter for ${params.intentType} intent`);\n    }\n    const intentRecipients = params.recipients?.map((recipient) => {\n      const formattedRecipient: IntentRecipient = {\n        address: { address: recipient.address },\n        amount: { value: `${recipient.amount}`, symbol: recipient.tokenName ? recipient.tokenName : chain },\n      };\n\n      if (recipient.data) {\n        formattedRecipient.data = recipient.data;\n      }\n\n      const { tokenData } = recipient;\n      if (tokenData && (tokenData.tokenContractAddress || tokenData.tokenName)) {\n        // token related recipient data gets validated in WP\n        if (!(tokenData.tokenType && tokenData.tokenQuantity)) {\n          throw new Error(\n            'token type and quantity is required to request a transaction with intent to transfer a token'\n          );\n        }\n        tokenData.tokenName = this.getTokenName(baseCoin, tokenData);\n        if (tokenData.tokenName) {\n          formattedRecipient.amount.symbol = tokenData.tokenName;\n        }\n        formattedRecipient.tokenData = tokenData;\n      }\n      return formattedRecipient;\n    });\n\n    const baseIntent: PopulatedIntent = {\n      intentType: params.intentType,\n      sequenceId: params.sequenceId,\n      comment: params.comment,\n      nonce: params.nonce,\n      recipients: intentRecipients,\n    };\n\n    if (baseCoin.getFamily() === 'eth' || baseCoin.getFamily() === 'polygon' || baseCoin.getFamily() === 'bsc') {\n      switch (params.intentType) {\n        case 'payment':\n        case 'transferToken':\n        case 'fillNonce':\n          return {\n            ...baseIntent,\n            selfSend: params.selfSend,\n            feeOptions: params.feeOptions,\n            hopParams: params.hopParams,\n            isTss: params.isTss,\n            nonce: params.nonce,\n            custodianTransactionId: params.custodianTransactionId,\n            receiveAddress: params.receiveAddress,\n          };\n        case 'acceleration':\n          return {\n            ...baseIntent,\n            txid: params.lowFeeTxid,\n            receiveAddress: params.receiveAddress,\n            feeOptions: params.feeOptions,\n          };\n        default:\n          throw new Error(`Unsupported intent type ${params.intentType}`);\n      }\n    }\n\n    this.baseCoin.setCoinSpecificFieldsInIntent(baseIntent, params);\n\n    if (params.feeOptions !== undefined) {\n      return {\n        ...baseIntent,\n        memo: params.memo?.value,\n        token: params.tokenName,\n        enableTokens: params.enableTokens,\n        feeOptions: params.feeOptions,\n      };\n    }\n\n    return {\n      ...baseIntent,\n      memo: params.memo?.value,\n      token: params.tokenName,\n      enableTokens: params.enableTokens,\n    };\n  }\n\n  getTokenName(baseCoin: IBaseCoin, tokenData: TokenTransferRecipientParams): string | undefined {\n    if (tokenData.tokenName) {\n      return tokenData.tokenName;\n    }\n    const networkPrefix = baseCoin.getConfig().network.type === NetworkType.TESTNET ? 't' : '';\n    const tokenStaticsKey = `${networkPrefix}${baseCoin.getFamily()}:${tokenData.tokenContractAddress}`;\n    if (coins.has(tokenStaticsKey)) {\n      const tokenStatics = coins.get(tokenStaticsKey);\n      tokenData.tokenName = tokenStatics.name;\n    }\n    return tokenData.tokenName;\n  }\n}\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!