PHP WebShell

Текущая директория: /usr/lib/node_modules/bitgo/node_modules/@bitgo/sdk-core/dist/src/bitgo/staking

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

"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {
    return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.StakingWallet = void 0;
/**
 * @prettier
 */
const statics_1 = require("@bitgo/statics");
const utils_1 = require("../utils");
const assert_1 = __importDefault(require("assert"));
const transactionUtils_1 = require("../utils/transactionUtils");
const debug_1 = __importDefault(require("debug"));
class StakingWallet {
    constructor(wallet, isEthTss) {
        this.wallet = wallet;
        this.bitgo = wallet.bitgo;
        this.tssUtil = new utils_1.TssUtils(this.bitgo, this.wallet.baseCoin, this.wallet);
        this.isEthTss = isEthTss;
    }
    get walletId() {
        return this.wallet.id();
    }
    get coin() {
        return this.wallet.baseCoin.tokenConfig ? this.wallet.baseCoin.tokenConfig.type : this.wallet.coin();
    }
    /**
     * Stake coins
     * @param options - stake options
     * @return StakingRequest
     */
    async stake(options) {
        return await this.createStakingRequest(options, 'STAKE');
    }
    /**
     * Unstake coins
     * @param options - unstake options
     * @return StakingRequest
     */
    async unstake(options) {
        return await this.createStakingRequest(options, 'UNSTAKE');
    }
    /**
     * Submit a request to switch the validator used for a specific delegation
     * This will create a new delegation with the new validator address and mark the old delegation as inactive
     * @param options - switch validator options
     * @return StakingRequest
     */
    async switchValidator(options) {
        return await this.createStakingRequest(options, 'SWITCH_VALIDATOR');
    }
    /**
     * Submit a request to claim rewards for a specific delegation
     * @param options - claim rewards options
     * @return StakingRequest
     */
    async claimRewards(options) {
        return await this.createStakingRequest(options, 'CLAIM_REWARDS');
    }
    /**
     * Cancel staking request
     * @param stakingRequestId - id of the staking request to cancel
     * @return StakingRequest
     */
    async cancelStakingRequest(stakingRequestId) {
        return await this.bitgo.del(this.bitgo.microservicesUrl(this.stakingRequestUrl(stakingRequestId))).result();
    }
    /**
     * Fetch delegations for a specific wallet
     * @param options - unstake options
     * @return StakingRequest
     */
    async delegations(options) {
        return await this.getDelegations(options);
    }
    /**
     * Get a staking request by ID
     * @param stakingRequestId - id of the staking request to retrieve
     * @return StakingRequest
     */
    async getStakingRequest(stakingRequestId) {
        return await this.bitgo.get(this.bitgo.microservicesUrl(this.stakingRequestUrl(stakingRequestId))).result();
    }
    /**
     * Get transactions ready to sign
     * @param stakingRequestId
     * @return TransactionsReadyToSign
     */
    async getTransactionsReadyToSign(stakingRequestId) {
        const stakingRequest = await this.getStakingRequest(stakingRequestId);
        const readyToSign = stakingRequest.transactions.filter((transaction) => transaction.status === `READY`);
        const newTransactions = stakingRequest.transactions.filter((transaction) => transaction.status === `NEW`);
        return Promise.resolve({
            allSigningComplete: stakingRequest.transactions.length > 0 && newTransactions.length === 0 && readyToSign.length === 0,
            transactions: readyToSign,
        });
    }
    /**
     * Build the staking transaction
     * If TSS delete signature shares, else expand build params and then build
     * @param transaction - staking transaction to build
     */
    async build(transaction) {
        if ((this.wallet.multisigType() === 'tss' && this.wallet.baseCoin.getFamily() !== 'eth') || this.isEthTss) {
            if (!transaction.txRequestId) {
                throw new Error('txRequestId is required to sign and send');
            }
            // delete signature shares before signing for transaction request API
            await this.tssUtil.deleteSignatureShares(transaction.txRequestId);
            return {
                transaction: transaction,
                result: {
                    walletId: this.walletId,
                    txRequestId: transaction.txRequestId,
                },
            };
        }
        else {
            transaction = await this.expandBuildParams(transaction);
            if (!transaction.buildParams) {
                throw Error(`Staking transaction ${transaction.id} build params not expanded`);
            }
            const isBtcUndelegate = this.wallet.baseCoin.getFamily() === 'btc' &&
                transaction.transactionType.toLowerCase() === 'undelegate_withdraw';
            const wallet = isBtcUndelegate
                ? await this.getDescriptorWallet(transaction)
                : await this.getWalletForBuildingAndSigning();
            return {
                transaction: transaction,
                result: await wallet.prebuildTransaction(transaction.buildParams),
            };
        }
    }
    /**
     * Sign the staking transaction
     * @param signOptions
     * @param stakingPrebuildTransaction
     */
    async sign(signOptions, stakingPrebuildTransaction) {
        const reqId = new utils_1.RequestTracer();
        const isBtcUndelegate = this.wallet.baseCoin.getFamily() === 'btc' &&
            stakingPrebuildTransaction.transaction.transactionType.toLowerCase() === 'undelegate_withdraw';
        const wallet = isBtcUndelegate
            ? await this.getDescriptorWallet(stakingPrebuildTransaction.transaction)
            : await this.getWalletForBuildingAndSigning();
        const keychain = await wallet.baseCoin.keychains().getKeysForSigning({
            wallet: this.wallet,
            reqId: reqId,
        });
        return {
            transaction: stakingPrebuildTransaction.transaction,
            signed: await wallet.signTransaction({
                txPrebuild: stakingPrebuildTransaction.result,
                walletPassphrase: signOptions.walletPassphrase,
                keychain: keychain[0],
            }),
        };
    }
    /**
     * Send the signed staking transaction if required. Send call is not required if api version is full
     * and this method will return the staking transaction from the incoming object.
     * @param signedTransaction
     */
    async send(signedTransaction) {
        if (this.isSendCallRequired()) {
            return await this.bitgo
                .post(this.bitgo.microservicesUrl(this.stakingTransactionURL(signedTransaction.transaction)))
                .send(signedTransaction.signed)
                .result();
        }
        return signedTransaction.transaction;
    }
    /**
     * @Deprecated use buildAndSign
     * Build, sign and send the transaction.
     * @param signOptions
     * @param transaction
     */
    async buildSignAndSend(signOptions, transaction) {
        return await this.buildAndSign(signOptions, transaction).then((result) => {
            return this.send(result);
        });
    }
    /**
     * Create prebuilt staking transaction.
     *
     * for transactions with tx request id (TSS transactions), we need to delete signature shares before creating prebuild transaction
     * we only need to get transaction build params if they exist to pre build
     *
     * @param transaction
     */
    async prebuildSelfManagedStakingTransaction(transaction) {
        if (transaction.txRequestId) {
            await this.tssUtil.deleteSignatureShares(transaction.txRequestId);
        }
        const buildParams = (await this.expandBuildParams(transaction)).buildParams;
        const formattedParams = {
            ...buildParams,
            coin: this.coin,
            walletId: this.walletId,
            walletType: this.wallet.type(),
            preview: true,
        };
        return await (await this.getWalletForBuildingAndSigning()).prebuildTransaction(formattedParams);
    }
    /**
     * Build and sign the transaction.
     * @param signOptions
     * @param transaction
     */
    async buildAndSign(signOptions, transaction) {
        const builtTx = await this.build(transaction);
        // default to verifying a transaction unless explicitly skipped
        const skipVerification = signOptions.transactionVerificationOptions?.skipTransactionVerification ?? false;
        if (!isStakingTxRequestPrebuildResult(builtTx.result) && !skipVerification) {
            await this.validateBuiltStakingTransaction(transaction, builtTx);
        }
        return await this.sign(signOptions, builtTx);
    }
    async expandBuildParams(stakingTransaction) {
        return await this.bitgo
            .get(this.bitgo.microservicesUrl(this.stakingTransactionURL(stakingTransaction)))
            .query({ expandBuildParams: true })
            .result();
    }
    async createStakingRequest(options, type) {
        return await this.bitgo
            .post(this.bitgo.microservicesUrl(this.stakingRequestsURL()))
            .send({
            ...options,
            type: type,
        })
            .result();
    }
    stakingRequestsURL() {
        return `/api/staking/v1/${this.coin}/wallets/${this.walletId}/requests`;
    }
    async getDelegations(options) {
        return await this.bitgo.get(this.bitgo.microservicesUrl(this.stakingDelegationsURL())).query(options).result();
    }
    stakingDelegationsURL() {
        return `/api/staking/v1/${this.coin}/wallets/${this.walletId}/delegations`;
    }
    stakingRequestUrl(stakingRequestId) {
        return `${this.stakingRequestsURL()}/${stakingRequestId}`;
    }
    stakingTransactionURL(stakingTransaction) {
        return `${this.stakingRequestUrl(stakingTransaction.stakingRequestId)}/transactions/${stakingTransaction.id}`;
    }
    async getWalletForBuildingAndSigning() {
        if (this.wallet.baseCoin.tokenConfig) {
            if (!this.tokenParentWallet) {
                this.tokenParentWallet = await this.bitgo
                    .coin(this.wallet.baseCoin.tokenConfig.coin)
                    .wallets()
                    .get({ id: this.wallet.id() });
            }
            return this.tokenParentWallet;
        }
        else {
            return Promise.resolve(this.wallet);
        }
    }
    /**
     * Send API call is only required for TSS TxRequest api version lite or multi-sig transactions.
     * For Full api version, sign transaction moves the transaction to delivered state.
     * @returns true if send API call to staking service is required else false
     */
    isSendCallRequired() {
        if (this.wallet.baseCoin.getFamily() === statics_1.CoinFamily.ETH) {
            return !this.isEthTss;
        }
        else if (this.wallet.multisigType() === 'tss') {
            return this.wallet.baseCoin.getMPCAlgorithm() !== 'ecdsa';
        }
        else {
            return true;
        }
    }
    async getDescriptorWallet(transaction) {
        (0, assert_1.default)(transaction.buildParams?.senderWalletId, 'senderWalletId is required for btc undelegate transaction');
        return await this.wallet.baseCoin.wallets().get({ id: transaction.buildParams.senderWalletId });
    }
    async validateBuiltStakingTransaction(transaction, prebuiltStakingTransaction) {
        const { buildParams } = transaction;
        const { result } = prebuiltStakingTransaction;
        const coin = this.wallet.baseCoin;
        (0, debug_1.default)(`Validating staking transaction ${transaction.stakingRequestId} with prebuilt transaction`);
        if (!('txHex' in result) || !result.txHex) {
            (0, debug_1.default)(`Skipping validation for staking transaction ${transaction.stakingRequestId} - txHex is undefined`);
            return;
        }
        const explainedTransaction = await coin.explainTransaction(result);
        if (buildParams?.recipients) {
            const userRecipientMap = new Map(buildParams.recipients.map((recipient) => [recipient.address.toLowerCase(), recipient]));
            const platformRecipientMap = new Map((explainedTransaction?.outputs ?? []).map((recipient) => [recipient.address.toLowerCase(), recipient]));
            const mismatchErrors = [];
            for (const [recipientAddress, recipientInfo] of platformRecipientMap) {
                if (userRecipientMap.has(recipientAddress)) {
                    const userRecipient = userRecipientMap.get(recipientAddress);
                    if (!userRecipient) {
                        console.error('Unable to determine recipient address');
                        return;
                    }
                    const matchResult = (0, transactionUtils_1.transactionRecipientsMatch)(userRecipient, recipientInfo);
                    if (!matchResult.exactMatch) {
                        if (!matchResult.tokenMatch) {
                            mismatchErrors.push(`Invalid token ${recipientInfo.tokenName} transfer with amount ${recipientInfo.amount} to ${recipientInfo.address} found in built transaction, specified ${userRecipient.tokenName}`);
                        }
                        if (!matchResult.amountMatch) {
                            mismatchErrors.push(`Invalid recipient amount for ${recipientInfo.address}, specified ${userRecipient.amount} got ${recipientInfo.amount}`);
                        }
                    }
                }
                else {
                    mismatchErrors.push(`Invalid recipient address: ${recipientAddress}`);
                }
            }
            const missingRecipientAddresses = Array.from(userRecipientMap.keys()).filter((address) => !platformRecipientMap.has(address));
            if (missingRecipientAddresses.length > 0) {
                mismatchErrors.push(`Missing recipient address(es): ${missingRecipientAddresses.join(', ')}`);
            }
            if (mismatchErrors.length > 0) {
                console.error(mismatchErrors.join(', '));
                return;
            }
        }
        else {
            (0, debug_1.default)(`Cannot validate staking transaction ${transaction.stakingRequestId} without specified build params`);
        }
    }
}
exports.StakingWallet = StakingWallet;
function isStakingTxRequestPrebuildResult(tx) {
    return tx.txRequestId !== undefined;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stakingWallet.js","sourceRoot":"","sources":["../../../../src/bitgo/staking/stakingWallet.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport { CoinFamily } from '@bitgo/statics';\n\nimport {\n  DelegationOptions,\n  DelegationResults,\n  IStakingWallet,\n  StakeOptions,\n  StakingPrebuildTransactionResult,\n  StakingRequest,\n  StakingSignedTransaction,\n  StakingSignOptions,\n  StakingTransaction,\n  SwitchValidatorOptions,\n  TransactionsReadyToSign,\n  UnstakeOptions,\n  EthUnstakeOptions,\n  ClaimRewardsOptions,\n  StakingTxRequestPrebuildTransactionResult,\n} from './iStakingWallet';\nimport { BitGoBase } from '../bitgoBase';\nimport { IWallet, PrebuildTransactionResult } from '../wallet';\nimport { ITssUtils, RequestTracer, TssUtils } from '../utils';\nimport assert from 'assert';\nimport { transactionRecipientsMatch } from '../utils/transactionUtils';\nimport debug from 'debug';\n\nexport class StakingWallet implements IStakingWallet {\n  private readonly bitgo: BitGoBase;\n  private tokenParentWallet?: IWallet;\n  private readonly isEthTss: boolean;\n\n  public wallet: IWallet;\n  public tssUtil: ITssUtils;\n\n  constructor(wallet: IWallet, isEthTss: boolean) {\n    this.wallet = wallet;\n    this.bitgo = wallet.bitgo;\n    this.tssUtil = new TssUtils(this.bitgo, this.wallet.baseCoin, this.wallet);\n    this.isEthTss = isEthTss;\n  }\n\n  get walletId(): string {\n    return this.wallet.id();\n  }\n\n  get coin(): string {\n    return this.wallet.baseCoin.tokenConfig ? this.wallet.baseCoin.tokenConfig.type : this.wallet.coin();\n  }\n\n  /**\n   * Stake coins\n   * @param options - stake options\n   * @return StakingRequest\n   */\n  async stake(options: StakeOptions): Promise<StakingRequest> {\n    return await this.createStakingRequest(options, 'STAKE');\n  }\n\n  /**\n   * Unstake coins\n   * @param options - unstake options\n   * @return StakingRequest\n   */\n  async unstake(options: UnstakeOptions | EthUnstakeOptions): Promise<StakingRequest> {\n    return await this.createStakingRequest(options, 'UNSTAKE');\n  }\n\n  /**\n   * Submit a request to switch the validator used for a specific delegation\n   * This will create a new delegation with the new validator address and mark the old delegation as inactive\n   * @param options - switch validator options\n   * @return StakingRequest\n   */\n  async switchValidator(options: SwitchValidatorOptions): Promise<StakingRequest> {\n    return await this.createStakingRequest(options, 'SWITCH_VALIDATOR');\n  }\n\n  /**\n   * Submit a request to claim rewards for a specific delegation\n   * @param options - claim rewards options\n   * @return StakingRequest\n   */\n  async claimRewards(options: ClaimRewardsOptions): Promise<StakingRequest> {\n    return await this.createStakingRequest(options, 'CLAIM_REWARDS');\n  }\n\n  /**\n   * Cancel staking request\n   * @param stakingRequestId - id of the staking request to cancel\n   * @return StakingRequest\n   */\n  async cancelStakingRequest(stakingRequestId: string): Promise<StakingRequest> {\n    return await this.bitgo.del(this.bitgo.microservicesUrl(this.stakingRequestUrl(stakingRequestId))).result();\n  }\n\n  /**\n   * Fetch delegations for a specific wallet\n   * @param options - unstake options\n   * @return StakingRequest\n   */\n  async delegations(options: DelegationOptions): Promise<DelegationResults> {\n    return await this.getDelegations(options);\n  }\n\n  /**\n   * Get a staking request by ID\n   * @param stakingRequestId - id of the staking request to retrieve\n   * @return StakingRequest\n   */\n  async getStakingRequest(stakingRequestId: string): Promise<StakingRequest> {\n    return await this.bitgo.get(this.bitgo.microservicesUrl(this.stakingRequestUrl(stakingRequestId))).result();\n  }\n\n  /**\n   * Get transactions ready to sign\n   * @param stakingRequestId\n   * @return TransactionsReadyToSign\n   */\n  async getTransactionsReadyToSign(stakingRequestId: string): Promise<TransactionsReadyToSign> {\n    const stakingRequest: StakingRequest = await this.getStakingRequest(stakingRequestId);\n    const readyToSign: StakingTransaction[] = stakingRequest.transactions.filter(\n      (transaction: StakingTransaction) => transaction.status === `READY`\n    );\n    const newTransactions: StakingTransaction[] = stakingRequest.transactions.filter(\n      (transaction: StakingTransaction) => transaction.status === `NEW`\n    );\n\n    return Promise.resolve({\n      allSigningComplete:\n        stakingRequest.transactions.length > 0 && newTransactions.length === 0 && readyToSign.length === 0,\n      transactions: readyToSign,\n    });\n  }\n\n  /**\n   * Build the staking transaction\n   * If TSS delete signature shares, else expand build params and then build\n   * @param transaction - staking transaction to build\n   */\n  async build(transaction: StakingTransaction): Promise<StakingPrebuildTransactionResult> {\n    if ((this.wallet.multisigType() === 'tss' && this.wallet.baseCoin.getFamily() !== 'eth') || this.isEthTss) {\n      if (!transaction.txRequestId) {\n        throw new Error('txRequestId is required to sign and send');\n      }\n      // delete signature shares before signing for transaction request API\n      await this.tssUtil.deleteSignatureShares(transaction.txRequestId);\n      return {\n        transaction: transaction,\n        result: {\n          walletId: this.walletId,\n          txRequestId: transaction.txRequestId,\n        },\n      };\n    } else {\n      transaction = await this.expandBuildParams(transaction);\n      if (!transaction.buildParams) {\n        throw Error(`Staking transaction ${transaction.id} build params not expanded`);\n      }\n      const isBtcUndelegate =\n        this.wallet.baseCoin.getFamily() === 'btc' &&\n        transaction.transactionType.toLowerCase() === 'undelegate_withdraw';\n      const wallet = isBtcUndelegate\n        ? await this.getDescriptorWallet(transaction)\n        : await this.getWalletForBuildingAndSigning();\n\n      return {\n        transaction: transaction,\n        result: await wallet.prebuildTransaction(transaction.buildParams),\n      };\n    }\n  }\n\n  /**\n   * Sign the staking transaction\n   * @param signOptions\n   * @param stakingPrebuildTransaction\n   */\n  async sign(\n    signOptions: StakingSignOptions,\n    stakingPrebuildTransaction: StakingPrebuildTransactionResult\n  ): Promise<StakingSignedTransaction> {\n    const reqId = new RequestTracer();\n    const isBtcUndelegate =\n      this.wallet.baseCoin.getFamily() === 'btc' &&\n      stakingPrebuildTransaction.transaction.transactionType.toLowerCase() === 'undelegate_withdraw';\n    const wallet = isBtcUndelegate\n      ? await this.getDescriptorWallet(stakingPrebuildTransaction.transaction)\n      : await this.getWalletForBuildingAndSigning();\n\n    const keychain = await wallet.baseCoin.keychains().getKeysForSigning({\n      wallet: this.wallet,\n      reqId: reqId,\n    });\n    return {\n      transaction: stakingPrebuildTransaction.transaction,\n      signed: await wallet.signTransaction({\n        txPrebuild: stakingPrebuildTransaction.result,\n        walletPassphrase: signOptions.walletPassphrase,\n        keychain: keychain[0],\n      }),\n    };\n  }\n\n  /**\n   * Send the signed staking transaction if required. Send call is not required if api version is full\n   * and this method will return the staking transaction from the incoming object.\n   * @param signedTransaction\n   */\n  async send(signedTransaction: StakingSignedTransaction): Promise<StakingTransaction> {\n    if (this.isSendCallRequired()) {\n      return await this.bitgo\n        .post(this.bitgo.microservicesUrl(this.stakingTransactionURL(signedTransaction.transaction)))\n        .send(signedTransaction.signed)\n        .result();\n    }\n    return signedTransaction.transaction;\n  }\n\n  /**\n   * @Deprecated use buildAndSign\n   * Build, sign and send the transaction.\n   * @param signOptions\n   * @param transaction\n   */\n  async buildSignAndSend(\n    signOptions: StakingSignOptions,\n    transaction: StakingTransaction\n  ): Promise<StakingTransaction> {\n    return await this.buildAndSign(signOptions, transaction).then((result: StakingSignedTransaction) => {\n      return this.send(result);\n    });\n  }\n\n  /**\n   * Create prebuilt staking transaction.\n   *\n   * for transactions with tx request id (TSS transactions), we need to delete signature shares before creating prebuild transaction\n   * we only need to get transaction build params if they exist to pre build\n   *\n   * @param transaction\n   */\n  async prebuildSelfManagedStakingTransaction(transaction: StakingTransaction): Promise<PrebuildTransactionResult> {\n    if (transaction.txRequestId) {\n      await this.tssUtil.deleteSignatureShares(transaction.txRequestId);\n    }\n    const buildParams = (await this.expandBuildParams(transaction)).buildParams;\n    const formattedParams = {\n      ...buildParams,\n      coin: this.coin,\n      walletId: this.walletId,\n      walletType: this.wallet.type(),\n      preview: true,\n    };\n    return await (await this.getWalletForBuildingAndSigning()).prebuildTransaction(formattedParams);\n  }\n\n  /**\n   * Build and sign the transaction.\n   * @param signOptions\n   * @param transaction\n   */\n  async buildAndSign(\n    signOptions: StakingSignOptions,\n    transaction: StakingTransaction\n  ): Promise<StakingSignedTransaction> {\n    const builtTx = await this.build(transaction);\n    // default to verifying a transaction unless explicitly skipped\n    const skipVerification = signOptions.transactionVerificationOptions?.skipTransactionVerification ?? false;\n    if (!isStakingTxRequestPrebuildResult(builtTx.result) && !skipVerification) {\n      await this.validateBuiltStakingTransaction(transaction, builtTx);\n    }\n    return await this.sign(signOptions, builtTx);\n  }\n\n  private async expandBuildParams(stakingTransaction: StakingTransaction): Promise<StakingTransaction> {\n    return await this.bitgo\n      .get(this.bitgo.microservicesUrl(this.stakingTransactionURL(stakingTransaction)))\n      .query({ expandBuildParams: true })\n      .result();\n  }\n\n  private async createStakingRequest(\n    options: StakeOptions | UnstakeOptions | EthUnstakeOptions | SwitchValidatorOptions | ClaimRewardsOptions,\n    type: string\n  ): Promise<StakingRequest> {\n    return await this.bitgo\n      .post(this.bitgo.microservicesUrl(this.stakingRequestsURL()))\n      .send({\n        ...options,\n        type: type,\n      })\n      .result();\n  }\n\n  private stakingRequestsURL() {\n    return `/api/staking/v1/${this.coin}/wallets/${this.walletId}/requests`;\n  }\n\n  private async getDelegations(options: DelegationOptions): Promise<DelegationResults> {\n    return await this.bitgo.get(this.bitgo.microservicesUrl(this.stakingDelegationsURL())).query(options).result();\n  }\n\n  private stakingDelegationsURL() {\n    return `/api/staking/v1/${this.coin}/wallets/${this.walletId}/delegations`;\n  }\n\n  private stakingRequestUrl(stakingRequestId: string): string {\n    return `${this.stakingRequestsURL()}/${stakingRequestId}`;\n  }\n\n  private stakingTransactionURL(stakingTransaction: StakingTransaction): string {\n    return `${this.stakingRequestUrl(stakingTransaction.stakingRequestId)}/transactions/${stakingTransaction.id}`;\n  }\n\n  private async getWalletForBuildingAndSigning(): Promise<IWallet> {\n    if (this.wallet.baseCoin.tokenConfig) {\n      if (!this.tokenParentWallet) {\n        this.tokenParentWallet = await this.bitgo\n          .coin(this.wallet.baseCoin.tokenConfig.coin)\n          .wallets()\n          .get({ id: this.wallet.id() });\n      }\n      return this.tokenParentWallet;\n    } else {\n      return Promise.resolve(this.wallet);\n    }\n  }\n\n  /**\n   * Send API call is only required for TSS TxRequest api version lite or multi-sig transactions.\n   * For Full api version, sign transaction moves the transaction to delivered state.\n   * @returns true if send API call to staking service is required else false\n   */\n  private isSendCallRequired(): boolean {\n    if (this.wallet.baseCoin.getFamily() === CoinFamily.ETH) {\n      return !this.isEthTss;\n    } else if (this.wallet.multisigType() === 'tss') {\n      return this.wallet.baseCoin.getMPCAlgorithm() !== 'ecdsa';\n    } else {\n      return true;\n    }\n  }\n\n  private async getDescriptorWallet(transaction: StakingTransaction): Promise<IWallet> {\n    assert(transaction.buildParams?.senderWalletId, 'senderWalletId is required for btc undelegate transaction');\n    return await this.wallet.baseCoin.wallets().get({ id: transaction.buildParams.senderWalletId });\n  }\n\n  private async validateBuiltStakingTransaction(\n    transaction: StakingTransaction,\n    prebuiltStakingTransaction: StakingPrebuildTransactionResult\n  ) {\n    const { buildParams } = transaction;\n    const { result } = prebuiltStakingTransaction;\n    const coin = this.wallet.baseCoin;\n    debug(`Validating staking transaction ${transaction.stakingRequestId} with prebuilt transaction`);\n\n    if (!('txHex' in result) || !result.txHex) {\n      debug(`Skipping validation for staking transaction ${transaction.stakingRequestId} - txHex is undefined`);\n      return;\n    }\n\n    const explainedTransaction = await coin.explainTransaction(result);\n\n    if (buildParams?.recipients) {\n      const userRecipientMap = new Map(\n        buildParams.recipients.map((recipient) => [recipient.address.toLowerCase(), recipient])\n      );\n      const platformRecipientMap = new Map(\n        (explainedTransaction?.outputs ?? []).map((recipient) => [recipient.address.toLowerCase(), recipient])\n      );\n\n      const mismatchErrors: string[] = [];\n\n      for (const [recipientAddress, recipientInfo] of platformRecipientMap) {\n        if (userRecipientMap.has(recipientAddress)) {\n          const userRecipient = userRecipientMap.get(recipientAddress);\n          if (!userRecipient) {\n            console.error('Unable to determine recipient address');\n            return;\n          }\n          const matchResult = transactionRecipientsMatch(userRecipient, recipientInfo);\n          if (!matchResult.exactMatch) {\n            if (!matchResult.tokenMatch) {\n              mismatchErrors.push(\n                `Invalid token ${recipientInfo.tokenName} transfer with amount ${recipientInfo.amount} to ${recipientInfo.address} found in built transaction, specified ${userRecipient.tokenName}`\n              );\n            }\n            if (!matchResult.amountMatch) {\n              mismatchErrors.push(\n                `Invalid recipient amount for ${recipientInfo.address}, specified ${userRecipient.amount} got ${recipientInfo.amount}`\n              );\n            }\n          }\n        } else {\n          mismatchErrors.push(`Invalid recipient address: ${recipientAddress}`);\n        }\n      }\n\n      const missingRecipientAddresses = Array.from(userRecipientMap.keys()).filter(\n        (address) => !platformRecipientMap.has(address)\n      );\n\n      if (missingRecipientAddresses.length > 0) {\n        mismatchErrors.push(`Missing recipient address(es): ${missingRecipientAddresses.join(', ')}`);\n      }\n      if (mismatchErrors.length > 0) {\n        console.error(mismatchErrors.join(', '));\n        return;\n      }\n    } else {\n      debug(`Cannot validate staking transaction ${transaction.stakingRequestId} without specified build params`);\n    }\n  }\n}\n\nfunction isStakingTxRequestPrebuildResult(\n  tx: StakingPrebuildTransactionResult['result']\n): tx is StakingTxRequestPrebuildTransactionResult {\n  return (tx as StakingTxRequestPrebuildTransactionResult).txRequestId !== undefined;\n}\n"]}

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


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