PHP WebShell

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

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

"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {
    return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.BatchStakingBuilder = void 0;
const baseBuilder_1 = require("./baseBuilder");
const txwrapper_polkadot_1 = require("@substrate/txwrapper-polkadot");
const sdk_core_1 = require("@bitgo/sdk-core");
const txnSchema_1 = require("./txnSchema");
const utils_1 = __importDefault(require("./utils"));
const bignumber_js_1 = __importDefault(require("bignumber.js"));
class BatchStakingBuilder extends baseBuilder_1.PolyxBaseBuilder {
    constructor(_coinConfig) {
        super(_coinConfig);
        // For nominate operation
        this._validators = [];
        this.material(utils_1.default.getMaterial(_coinConfig.network.type));
    }
    get transactionType() {
        return sdk_core_1.TransactionType.Batch;
    }
    /**
     * Build a batch transaction that combines bond and nominate operations
     * Both operations are required and always atomic (using batchAll)
     */
    buildTransaction() {
        // Ensure both bond and nominate operations are included
        if (!this._amount || this._validators.length === 0) {
            throw new sdk_core_1.InvalidTransactionError('Batch transaction must include both bond and nominate operations');
        }
        const baseTxInfo = this.createBaseTxInfo();
        // Create the individual calls
        const calls = [];
        // Add bond call
        const bondCall = txwrapper_polkadot_1.methods.staking.bond({
            controller: this._controller || this._sender,
            value: this._amount,
            payee: this._payee || 'Staked',
        }, baseTxInfo.baseTxInfo, baseTxInfo.options);
        calls.push(bondCall.method);
        // Add nominate call
        const nominateCall = txwrapper_polkadot_1.methods.staking.nominate({
            targets: this._validators,
        }, baseTxInfo.baseTxInfo, baseTxInfo.options);
        calls.push(nominateCall.method);
        // Always use batchAll (atomic)
        return txwrapper_polkadot_1.methods.utility.batchAll({
            calls,
        }, baseTxInfo.baseTxInfo, baseTxInfo.options);
    }
    /**
     * Set the staking amount for bond
     */
    amount(amount) {
        this.validateValue(new bignumber_js_1.default(amount));
        this._amount = amount;
        return this;
    }
    /**
     * Get the staking amount
     */
    getAmount() {
        return this._amount;
    }
    /**
     * Set the controller account for bond
     */
    controller(controller) {
        this.validateAddress(controller);
        this._controller = controller.address;
        return this;
    }
    /**
     * Get the controller address
     */
    getController() {
        return this._controller;
    }
    /**
     * Set the rewards destination for bond ('Staked', 'Stash','Controller', or { Account: string })
     */
    payee(payee) {
        if (typeof payee === 'object' && payee.Account) {
            this._payee = payee;
        }
        else {
            this._payee = payee;
        }
        return this;
    }
    /**
     * Get the payee
     */
    getPayee() {
        return this._payee;
    }
    /**
     * Set the validators to nominate
     */
    validators(validators) {
        for (const address of validators) {
            this.validateAddress({ address });
        }
        this._validators = validators;
        return this;
    }
    /**
     * Get the validators to nominate
     */
    getValidators() {
        return this._validators;
    }
    /** @inheritdoc */
    validateDecodedTransaction(decodedTxn) {
        const methodName = decodedTxn.method?.name;
        // batch bond and nominate
        if (methodName === 'batchAll') {
            const txMethod = decodedTxn.method.args;
            const calls = txMethod.calls;
            if (calls.length !== 2) {
                throw new sdk_core_1.InvalidTransactionError(`Invalid batch staking transaction: expected 2 calls but got ${calls.length}`);
            }
            // Check that first call is bond
            const firstCallMethod = utils_1.default.decodeMethodName(calls[0], this._registry);
            if (firstCallMethod !== 'bond') {
                throw new sdk_core_1.InvalidTransactionError(`Invalid batch staking transaction: first call should be bond but got ${firstCallMethod}`);
            }
            // Check that second call is nominate
            const secondCallMethod = utils_1.default.decodeMethodName(calls[1], this._registry);
            if (secondCallMethod !== 'nominate') {
                throw new sdk_core_1.InvalidTransactionError(`Invalid batch staking transaction: second call should be nominate but got ${secondCallMethod}`);
            }
            // Validate bond arguments
            const bondArgs = calls[0].args;
            this.validateBondArgs(bondArgs);
            // Validate nominate arguments
            const nominateArgs = calls[1].args;
            this.validateNominateArgs(nominateArgs);
        }
        else {
            throw new sdk_core_1.InvalidTransactionError(`Invalid transaction type: ${methodName}`);
        }
    }
    /**
     * Validate bond arguments
     */
    validateBondArgs(args) {
        // Handle both string and object formats for controller
        const controllerValue = args.controller;
        const controllerAddress = typeof controllerValue === 'string' ? controllerValue : controllerValue.id;
        if (!utils_1.default.isValidAddress(controllerAddress)) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid bond args: controller address ${controllerAddress} is not a well-formed address`);
        }
        // Handle both string and number formats for value
        const amountValue = args.value;
        const valueString = typeof amountValue === 'string' ? amountValue : amountValue.toString();
        // Handle different payee formats
        const payeeValue = args.payee;
        let normalizedPayee = payeeValue;
        if (typeof payeeValue === 'object' && payeeValue !== null) {
            const decodedPayee = payeeValue;
            if (decodedPayee.staked !== undefined) {
                normalizedPayee = 'Staked';
            }
            else if (decodedPayee.stash !== undefined) {
                normalizedPayee = 'Stash';
            }
            else if (decodedPayee.controller !== undefined) {
                normalizedPayee = 'Controller';
            }
            else if (decodedPayee.account) {
                normalizedPayee = { Account: decodedPayee.account };
            }
        }
        const validationResult = txnSchema_1.BatchTransactionSchema.validateBond({
            value: valueString,
            controller: controllerAddress,
            payee: normalizedPayee,
        });
        if (validationResult.error) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid bond args: ${validationResult.error.message}`);
        }
    }
    /**
     * Validate nominate arguments
     */
    validateNominateArgs(args) {
        // Handle both string and object formats for targets
        const targetAddresses = args.targets.map((target) => {
            if (typeof target === 'string') {
                return target;
            }
            else if (target && typeof target === 'object' && 'id' in target) {
                return target.id;
            }
            throw new sdk_core_1.InvalidTransactionError(`Invalid target format: ${JSON.stringify(target)}`);
        });
        const validationResult = txnSchema_1.BatchTransactionSchema.validateNominate({
            validators: targetAddresses,
        });
        if (validationResult.error) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid nominate args: ${validationResult.error.message}`);
        }
    }
    /** @inheritdoc */
    fromImplementation(rawTransaction) {
        const tx = super.fromImplementation(rawTransaction);
        // Check if the transaction is a batch transaction
        if (this._method?.name !== 'batchAll') {
            throw new sdk_core_1.InvalidTransactionError(`Invalid Transaction Type: ${this._method?.name}. Expected batchAll`);
        }
        if (this._method) {
            const txMethod = this._method.args;
            for (const call of txMethod.calls) {
                const callMethod = utils_1.default.decodeMethodName(call, this._registry);
                if (callMethod === 'bond') {
                    const bondArgs = call.args;
                    // Handle both string and number formats for value
                    const amountValue = bondArgs.value;
                    const valueString = typeof amountValue === 'string' ? amountValue : amountValue.toString();
                    this.amount(valueString);
                    // Handle both string and object formats for controller
                    const controllerValue = bondArgs.controller;
                    const controllerAddress = typeof controllerValue === 'string' ? controllerValue : controllerValue.id;
                    this.controller({ address: controllerAddress });
                    // Handle different payee formats
                    const payeeValue = bondArgs.payee;
                    let normalizedPayee = payeeValue;
                    if (typeof payeeValue === 'object' && payeeValue !== null) {
                        const decodedPayee = payeeValue;
                        if (decodedPayee.staked !== undefined) {
                            normalizedPayee = 'Staked';
                        }
                        else if (decodedPayee.stash !== undefined) {
                            normalizedPayee = 'Stash';
                        }
                        else if (decodedPayee.controller !== undefined) {
                            normalizedPayee = 'Controller';
                        }
                        else if (decodedPayee.account) {
                            normalizedPayee = { Account: decodedPayee.account };
                        }
                    }
                    this.payee(normalizedPayee);
                }
                else if (callMethod === 'nominate') {
                    const nominateArgs = call.args;
                    // Handle both string and object formats for targets
                    const targetAddresses = nominateArgs.targets.map((target) => {
                        if (typeof target === 'string') {
                            return target;
                        }
                        else if (target && typeof target === 'object' && 'id' in target) {
                            return target.id;
                        }
                        throw new sdk_core_1.InvalidTransactionError(`Invalid target format: ${JSON.stringify(target)}`);
                    });
                    this.validators(targetAddresses);
                }
            }
        }
        return tx;
    }
    /** @inheritdoc */
    validateTransaction(tx) {
        super.validateTransaction(tx);
        this.validateFields();
    }
    /**
     * Validate the builder fields
     */
    validateFields() {
        // Ensure both bond and nominate operations are included
        if (!this._amount || this._validators.length === 0) {
            throw new sdk_core_1.InvalidTransactionError('Batch transaction must include both bond and nominate operations');
        }
        const validationResult = txnSchema_1.BatchTransactionSchema.validate({
            amount: this._amount,
            controller: this._controller,
            payee: this._payee,
            validators: this._validators,
        });
        if (validationResult.error) {
            throw new sdk_core_1.InvalidTransactionError(`Invalid transaction: ${validationResult.error.message}`);
        }
    }
    testValidateFields() {
        this.validateFields();
    }
    testValidateBondArgs(args) {
        return this.validateBondArgs(args);
    }
    testValidateNominateArgs(args) {
        return this.validateNominateArgs(args);
    }
}
exports.BatchStakingBuilder = BatchStakingBuilder;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"batchStakingBuilder.js","sourceRoot":"","sources":["../../../src/lib/batchStakingBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import { Transaction } from './transaction';\nimport { PolyxBaseBuilder } from './baseBuilder';\nimport { DecodedSignedTx, DecodedSigningPayload, UnsignedTransaction } from '@substrate/txwrapper-core';\nimport { methods } from '@substrate/txwrapper-polkadot';\nimport { BaseCoin as CoinConfig } from '@bitgo/statics';\nimport { BaseAddress, InvalidTransactionError, TransactionType } from '@bitgo/sdk-core';\nimport { BatchTransactionSchema } from './txnSchema';\nimport utils from './utils';\nimport { BatchArgs, BondArgs, NominateArgs } from './iface';\nimport BigNumber from 'bignumber.js';\n\n// Type definitions for decoded transaction formats\ninterface DecodedController {\n  id: string;\n}\n\ninterface DecodedPayee {\n  staked?: null;\n  stash?: null;\n  controller?: null;\n  account?: string;\n}\n\ntype ControllerValue = string | DecodedController;\ntype PayeeValue = string | DecodedPayee;\ntype AmountValue = string | number;\n\nexport class BatchStakingBuilder extends PolyxBaseBuilder {\n  // For bond operation\n  protected _amount: string;\n  protected _controller: string;\n  protected _payee: string | { Account: string };\n\n  // For nominate operation\n  protected _validators: string[] = [];\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    this.material(utils.getMaterial(_coinConfig.network.type));\n  }\n\n  protected get transactionType(): TransactionType {\n    return TransactionType.Batch;\n  }\n\n  /**\n   * Build a batch transaction that combines bond and nominate operations\n   * Both operations are required and always atomic (using batchAll)\n   */\n  protected buildTransaction(): UnsignedTransaction {\n    // Ensure both bond and nominate operations are included\n    if (!this._amount || this._validators.length === 0) {\n      throw new InvalidTransactionError('Batch transaction must include both bond and nominate operations');\n    }\n\n    const baseTxInfo = this.createBaseTxInfo();\n\n    // Create the individual calls\n    const calls: string[] = [];\n\n    // Add bond call\n    const bondCall = methods.staking.bond(\n      {\n        controller: this._controller || this._sender,\n        value: this._amount,\n        payee: this._payee || 'Staked',\n      },\n      baseTxInfo.baseTxInfo,\n      baseTxInfo.options\n    );\n    calls.push(bondCall.method);\n\n    // Add nominate call\n    const nominateCall = methods.staking.nominate(\n      {\n        targets: this._validators,\n      },\n      baseTxInfo.baseTxInfo,\n      baseTxInfo.options\n    );\n    calls.push(nominateCall.method);\n\n    // Always use batchAll (atomic)\n    return methods.utility.batchAll(\n      {\n        calls,\n      },\n      baseTxInfo.baseTxInfo,\n      baseTxInfo.options\n    );\n  }\n\n  /**\n   * Set the staking amount for bond\n   */\n  amount(amount: string): this {\n    this.validateValue(new BigNumber(amount));\n    this._amount = amount;\n    return this;\n  }\n\n  /**\n   * Get the staking amount\n   */\n  getAmount(): string {\n    return this._amount;\n  }\n\n  /**\n   * Set the controller account for bond\n   */\n  controller(controller: BaseAddress): this {\n    this.validateAddress(controller);\n    this._controller = controller.address;\n    return this;\n  }\n\n  /**\n   * Get the controller address\n   */\n  getController(): string {\n    return this._controller;\n  }\n\n  /**\n   * Set the rewards destination for bond ('Staked', 'Stash','Controller', or { Account: string })\n   */\n  payee(payee: string | { Account: string }): this {\n    if (typeof payee === 'object' && payee.Account) {\n      this._payee = payee;\n    } else {\n      this._payee = payee;\n    }\n    return this;\n  }\n\n  /**\n   * Get the payee\n   */\n  getPayee(): string | { Account: string } {\n    return this._payee;\n  }\n\n  /**\n   * Set the validators to nominate\n   */\n  validators(validators: string[]): this {\n    for (const address of validators) {\n      this.validateAddress({ address });\n    }\n    this._validators = validators;\n    return this;\n  }\n\n  /**\n   * Get the validators to nominate\n   */\n  getValidators(): string[] {\n    return this._validators;\n  }\n\n  /** @inheritdoc */\n  validateDecodedTransaction(decodedTxn: DecodedSigningPayload | DecodedSignedTx): void {\n    const methodName = decodedTxn.method?.name as string;\n\n    // batch bond and nominate\n    if (methodName === 'batchAll') {\n      const txMethod = decodedTxn.method.args as unknown as BatchArgs;\n      const calls = txMethod.calls;\n\n      if (calls.length !== 2) {\n        throw new InvalidTransactionError(\n          `Invalid batch staking transaction: expected 2 calls but got ${calls.length}`\n        );\n      }\n\n      // Check that first call is bond\n      const firstCallMethod = utils.decodeMethodName(calls[0], this._registry);\n      if (firstCallMethod !== 'bond') {\n        throw new InvalidTransactionError(\n          `Invalid batch staking transaction: first call should be bond but got ${firstCallMethod}`\n        );\n      }\n\n      // Check that second call is nominate\n      const secondCallMethod = utils.decodeMethodName(calls[1], this._registry);\n      if (secondCallMethod !== 'nominate') {\n        throw new InvalidTransactionError(\n          `Invalid batch staking transaction: second call should be nominate but got ${secondCallMethod}`\n        );\n      }\n\n      // Validate bond arguments\n      const bondArgs = calls[0].args as unknown as BondArgs;\n      this.validateBondArgs(bondArgs);\n\n      // Validate nominate arguments\n      const nominateArgs = calls[1].args as unknown as NominateArgs;\n      this.validateNominateArgs(nominateArgs);\n    } else {\n      throw new InvalidTransactionError(`Invalid transaction type: ${methodName}`);\n    }\n  }\n\n  /**\n   * Validate bond arguments\n   */\n  private validateBondArgs(args: BondArgs): void {\n    // Handle both string and object formats for controller\n    const controllerValue = args.controller as ControllerValue;\n    const controllerAddress = typeof controllerValue === 'string' ? controllerValue : controllerValue.id;\n\n    if (!utils.isValidAddress(controllerAddress)) {\n      throw new InvalidTransactionError(\n        `Invalid bond args: controller address ${controllerAddress} is not a well-formed address`\n      );\n    }\n\n    // Handle both string and number formats for value\n    const amountValue = args.value as AmountValue;\n    const valueString = typeof amountValue === 'string' ? amountValue : amountValue.toString();\n\n    // Handle different payee formats\n    const payeeValue = args.payee as PayeeValue;\n    let normalizedPayee: string | { Account: string } = payeeValue as string;\n    if (typeof payeeValue === 'object' && payeeValue !== null) {\n      const decodedPayee = payeeValue as DecodedPayee;\n      if (decodedPayee.staked !== undefined) {\n        normalizedPayee = 'Staked';\n      } else if (decodedPayee.stash !== undefined) {\n        normalizedPayee = 'Stash';\n      } else if (decodedPayee.controller !== undefined) {\n        normalizedPayee = 'Controller';\n      } else if (decodedPayee.account) {\n        normalizedPayee = { Account: decodedPayee.account };\n      }\n    }\n\n    const validationResult = BatchTransactionSchema.validateBond({\n      value: valueString,\n      controller: controllerAddress,\n      payee: normalizedPayee,\n    });\n\n    if (validationResult.error) {\n      throw new InvalidTransactionError(`Invalid bond args: ${validationResult.error.message}`);\n    }\n  }\n\n  /**\n   * Validate nominate arguments\n   */\n  private validateNominateArgs(args: NominateArgs): void {\n    // Handle both string and object formats for targets\n    const targetAddresses = args.targets.map((target) => {\n      if (typeof target === 'string') {\n        return target;\n      } else if (target && typeof target === 'object' && 'id' in target) {\n        return (target as { id: string }).id;\n      }\n      throw new InvalidTransactionError(`Invalid target format: ${JSON.stringify(target)}`);\n    });\n\n    const validationResult = BatchTransactionSchema.validateNominate({\n      validators: targetAddresses,\n    });\n\n    if (validationResult.error) {\n      throw new InvalidTransactionError(`Invalid nominate args: ${validationResult.error.message}`);\n    }\n  }\n\n  /** @inheritdoc */\n  protected fromImplementation(rawTransaction: string): Transaction {\n    const tx = super.fromImplementation(rawTransaction);\n\n    // Check if the transaction is a batch transaction\n    if ((this._method?.name as string) !== 'batchAll') {\n      throw new InvalidTransactionError(`Invalid Transaction Type: ${this._method?.name}. Expected batchAll`);\n    }\n\n    if (this._method) {\n      const txMethod = this._method.args as unknown as BatchArgs;\n\n      for (const call of txMethod.calls) {\n        const callMethod = utils.decodeMethodName(call, this._registry);\n        if (callMethod === 'bond') {\n          const bondArgs = call.args as unknown as BondArgs;\n          // Handle both string and number formats for value\n          const amountValue = bondArgs.value as AmountValue;\n          const valueString = typeof amountValue === 'string' ? amountValue : amountValue.toString();\n          this.amount(valueString);\n\n          // Handle both string and object formats for controller\n          const controllerValue = bondArgs.controller as ControllerValue;\n          const controllerAddress = typeof controllerValue === 'string' ? controllerValue : controllerValue.id;\n          this.controller({ address: controllerAddress });\n\n          // Handle different payee formats\n          const payeeValue = bondArgs.payee as PayeeValue;\n          let normalizedPayee: string | { Account: string } = payeeValue as string;\n          if (typeof payeeValue === 'object' && payeeValue !== null) {\n            const decodedPayee = payeeValue as DecodedPayee;\n            if (decodedPayee.staked !== undefined) {\n              normalizedPayee = 'Staked';\n            } else if (decodedPayee.stash !== undefined) {\n              normalizedPayee = 'Stash';\n            } else if (decodedPayee.controller !== undefined) {\n              normalizedPayee = 'Controller';\n            } else if (decodedPayee.account) {\n              normalizedPayee = { Account: decodedPayee.account };\n            }\n          }\n          this.payee(normalizedPayee);\n        } else if (callMethod === 'nominate') {\n          const nominateArgs = call.args as unknown as NominateArgs;\n\n          // Handle both string and object formats for targets\n          const targetAddresses = nominateArgs.targets.map((target) => {\n            if (typeof target === 'string') {\n              return target;\n            } else if (target && typeof target === 'object' && 'id' in target) {\n              return (target as { id: string }).id;\n            }\n            throw new InvalidTransactionError(`Invalid target format: ${JSON.stringify(target)}`);\n          });\n          this.validators(targetAddresses);\n        }\n      }\n    }\n\n    return tx;\n  }\n\n  /** @inheritdoc */\n  validateTransaction(tx: Transaction): void {\n    super.validateTransaction(tx);\n    this.validateFields();\n  }\n\n  /**\n   * Validate the builder fields\n   */\n  private validateFields(): void {\n    // Ensure both bond and nominate operations are included\n    if (!this._amount || this._validators.length === 0) {\n      throw new InvalidTransactionError('Batch transaction must include both bond and nominate operations');\n    }\n\n    const validationResult = BatchTransactionSchema.validate({\n      amount: this._amount,\n      controller: this._controller,\n      payee: this._payee,\n      validators: this._validators,\n    });\n\n    if (validationResult.error) {\n      throw new InvalidTransactionError(`Invalid transaction: ${validationResult.error.message}`);\n    }\n  }\n\n  testValidateFields(): void {\n    this.validateFields();\n  }\n\n  public testValidateBondArgs(args: BondArgs): void {\n    return this.validateBondArgs(args);\n  }\n\n  public testValidateNominateArgs(args: NominateArgs): void {\n    return this.validateNominateArgs(args);\n  }\n}\n"]}

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


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