PHP WebShell
Текущая директория: /opt/BitGoJS/modules/sdk-coin-sui/dist/src/lib
Просмотр файла: stakingBuilder.js
"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.StakingBuilder = void 0;
const sdk_core_1 = require("@bitgo/sdk-core");
const iface_1 = require("./iface");
const transactionBuilder_1 = require("./transactionBuilder");
const utils_1 = __importDefault(require("./utils"));
const assert_1 = __importDefault(require("assert"));
const stakingTransaction_1 = require("./stakingTransaction");
const builder_1 = require("./mystenlab/builder");
const framework_1 = require("./mystenlab/framework");
const bcs_1 = require("@mysten/bcs");
const constants_1 = require("./constants");
class StakingBuilder extends transactionBuilder_1.TransactionBuilder {
constructor(_coinConfig) {
super(_coinConfig);
this._transaction = new stakingTransaction_1.StakingTransaction(_coinConfig);
}
/**
* Build a MoveCall transaction ready to be signed and executed.
*
* @returns {BitGoSuiTransaction} an unsigned Sui transaction
*/
buildStakeTransaction() {
return {
type: iface_1.SuiTransactionType.AddStake,
sender: this._sender,
tx: {
inputs: [],
transactions: [],
},
gasData: this._gasData,
};
}
/**
* Get staking transaction type
*
* @return {TransactionType}
* @protected
*/
get transactionType() {
return sdk_core_1.TransactionType.StakingAdd;
}
/** @inheritdoc */
validateTransaction(transaction) {
if (!transaction.suiTransaction) {
return;
}
this.validateTransactionFields();
}
/** @inheritdoc */
sign(key) {
this.transaction.setSuiTransaction(this.buildSuiTransaction());
super.sign(key);
}
/**
* Create a new transaction for staking coins ready to be signed and executed.
*
* @param {RequestAddStake[]} request: a list of staking request
*/
stake(request) {
request.forEach((req) => {
utils_1.default.validateAddress(req.validatorAddress, 'validatorAddress');
(0, assert_1.default)(utils_1.default.isValidAmount(req.amount), 'Invalid recipient amount');
if (this._sender === req.validatorAddress) {
throw new sdk_core_1.BuildTransactionError('Sender address cannot be the same as the Staking address');
}
});
this._addStakeTx = request;
return this;
}
/**
* Create a new transaction for withdrawing coins ready to be signed
*
* @param {RequestWithdrawStakedSui} request
*/
unstake(request) {
this.validateSuiObjectRef(request.stakedSui, 'stakedSui');
this._withdrawDelegation = request;
return this;
}
/** @inheritdoc */
fromImplementation(rawTransaction) {
const tx = new stakingTransaction_1.StakingTransaction(this._coinConfig);
this.validateRawTransaction(rawTransaction);
tx.fromRawTransaction(rawTransaction);
this.initBuilder(tx);
return this.transaction;
}
/** @inheritdoc */
async buildImplementation() {
this.transaction.setSuiTransaction(this.buildSuiTransaction());
this.transaction.transactionType(this.transactionType);
if (this._signer) {
this.transaction.sign(this._signer);
}
this._signatures.forEach((signature) => {
this.transaction.addSignature(signature.publicKey, signature.signature);
});
this.transaction.loadInputsAndOutputs();
return this.transaction;
}
/**
* Initialize the transaction builder fields using the decoded transaction data
*
* @param {StakingTransaction} tx the transaction data
*/
initBuilder(tx) {
this._transaction = tx;
if (tx.signature && tx.signature.length > 0) {
this._signatures = [tx.suiSignature];
}
const txData = tx.toJson();
this.type(iface_1.SuiTransactionType.AddStake);
this.sender(txData.sender);
this.gasData(txData.gasData);
const requests = utils_1.default.getStakeRequests(tx.suiTransaction.tx);
this.stake(requests);
}
/**
* Validates all fields are defined
*/
validateTransactionFields() {
(0, assert_1.default)(this._type, new sdk_core_1.BuildTransactionError('type is required before building'));
(0, assert_1.default)(this._sender, new sdk_core_1.BuildTransactionError('sender is required before building'));
this._addStakeTx.forEach((req) => {
(0, assert_1.default)(req.validatorAddress, new sdk_core_1.BuildTransactionError('validator address is required before building'));
(0, assert_1.default)(req.amount, new sdk_core_1.BuildTransactionError('staking amount is required before building'));
});
(0, assert_1.default)(this._gasData, new sdk_core_1.BuildTransactionError('gasData is required before building'));
this.validateGasData(this._gasData);
}
/**
* Build SuiTransaction
*
* @return {BitGoSuiTransaction<MoveCallTx>}
* @protected
*/
buildSuiTransaction() {
this.validateTransactionFields();
const programmableTxBuilder = new builder_1.TransactionBlock();
switch (this._type) {
case iface_1.SuiTransactionType.AddStake:
// number of objects passed as gas payment should be strictly less than `MAX_GAS_OBJECTS`. When the transaction
// requires a larger number of inputs we use the merge command to merge the rest of the objects into the gasCoin
if (this._gasData.payment.length >= constants_1.MAX_GAS_OBJECTS) {
const gasPaymentObjects = this._gasData.payment
.slice(constants_1.MAX_GAS_OBJECTS - 1)
.map((object) => builder_1.Inputs.ObjectRef(object));
// limit for total number of `args: CallArg[]` for a single command is MAX_COMMAND_ARGS so the max length of
// `sources[]` for a `mergeCoins(destination, sources[])` command is MAX_COMMAND_ARGS - 1 (1 used up for
// `destination`). We need to create a total of `gasPaymentObjects/(MAX_COMMAND_ARGS - 1)` merge commands to
// merge all the objects
while (gasPaymentObjects.length > 0) {
programmableTxBuilder.mergeCoins(programmableTxBuilder.gas, gasPaymentObjects.splice(0, constants_1.MAX_COMMAND_ARGS - 1).map((object) => programmableTxBuilder.object(object)));
}
}
// Create a new coin with staking balance, based on the coins used as gas payment.
this._addStakeTx.forEach((req) => {
const coin = programmableTxBuilder.splitCoins(programmableTxBuilder.gas, [
programmableTxBuilder.pure(req.amount),
]);
// Stake the split coin to a specific validator address.
programmableTxBuilder.moveCall({
target: `${framework_1.SUI_SYSTEM_ADDRESS}::${framework_1.SUI_SYSTEM_MODULE_NAME}::${framework_1.ADD_STAKE_FUN_NAME}`,
arguments: [
programmableTxBuilder.object(builder_1.Inputs.SharedObjectRef(framework_1.SUI_SYSTEM_STATE_OBJECT)),
coin,
programmableTxBuilder.pure(builder_1.Inputs.Pure(req.validatorAddress, bcs_1.BCS.ADDRESS)),
],
});
});
break;
case iface_1.SuiTransactionType.WithdrawStake:
// Unstake staked object.
programmableTxBuilder.moveCall({
target: `${framework_1.SUI_SYSTEM_ADDRESS}::${framework_1.SUI_SYSTEM_MODULE_NAME}::${framework_1.WITHDRAW_STAKE_FUN_NAME}`,
arguments: [
programmableTxBuilder.object(builder_1.Inputs.SharedObjectRef(framework_1.SUI_SYSTEM_STATE_OBJECT)),
programmableTxBuilder.pure(builder_1.Inputs.ObjectRef(this._withdrawDelegation.stakedSui)),
],
});
break;
default:
throw new sdk_core_1.InvalidTransactionError(`unsupported target method`);
}
const txData = programmableTxBuilder.blockData;
return {
type: this._type,
sender: this._sender,
tx: {
inputs: [...txData.inputs],
transactions: [...txData.transactions],
},
gasData: {
...this._gasData,
payment: this._gasData.payment.slice(0, constants_1.MAX_GAS_OBJECTS - 1),
},
};
}
}
exports.StakingBuilder = StakingBuilder;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stakingBuilder.js","sourceRoot":"","sources":["../../../src/lib/stakingBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import { BaseCoin as CoinConfig } from '@bitgo/statics';\nimport { BaseKey, BuildTransactionError, InvalidTransactionError, TransactionType } from '@bitgo/sdk-core';\nimport {\n  SuiTransaction,\n  RequestAddStake,\n  RequestWithdrawStakedSui,\n  SuiTransactionType,\n  StakingProgrammableTransaction,\n} from './iface';\nimport { TransactionBuilder } from './transactionBuilder';\nimport { Transaction } from './transaction';\nimport utils from './utils';\nimport assert from 'assert';\nimport { TransferTransaction } from './transferTransaction';\nimport { StakingTransaction } from './stakingTransaction';\nimport {\n  TransactionBlock as ProgrammingTransactionBlockBuilder,\n  MoveCallTransaction,\n  Inputs,\n} from './mystenlab/builder';\nimport {\n  ADD_STAKE_FUN_NAME,\n  SUI_SYSTEM_ADDRESS,\n  SUI_SYSTEM_MODULE_NAME,\n  SUI_SYSTEM_STATE_OBJECT,\n  WITHDRAW_STAKE_FUN_NAME,\n} from './mystenlab/framework';\nimport { BCS } from '@mysten/bcs';\nimport { MAX_COMMAND_ARGS, MAX_GAS_OBJECTS } from './constants';\n\nexport class StakingBuilder extends TransactionBuilder<StakingProgrammableTransaction> {\n  protected _addStakeTx: RequestAddStake[];\n  protected _withdrawDelegation: RequestWithdrawStakedSui;\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    this._transaction = new StakingTransaction(_coinConfig);\n  }\n\n  /**\n   * Build a MoveCall transaction ready to be signed and executed.\n   *\n   * @returns {BitGoSuiTransaction} an unsigned Sui transaction\n   */\n  protected buildStakeTransaction(): SuiTransaction<StakingProgrammableTransaction> {\n    return {\n      type: SuiTransactionType.AddStake,\n      sender: this._sender,\n      tx: {\n        inputs: [],\n        transactions: [],\n      },\n      gasData: this._gasData,\n    };\n  }\n\n  /**\n   * Get staking transaction type\n   *\n   * @return {TransactionType}\n   * @protected\n   */\n  protected get transactionType(): TransactionType {\n    return TransactionType.StakingAdd;\n  }\n\n  /** @inheritdoc */\n  validateTransaction(transaction: TransferTransaction): void {\n    if (!transaction.suiTransaction) {\n      return;\n    }\n    this.validateTransactionFields();\n  }\n\n  /** @inheritdoc */\n  sign(key: BaseKey) {\n    this.transaction.setSuiTransaction(this.buildSuiTransaction());\n    super.sign(key);\n  }\n\n  /**\n   * Create a new transaction for staking coins ready to be signed and executed.\n   *\n   * @param {RequestAddStake[]} request: a list of staking request\n   */\n  stake(request: RequestAddStake[]): this {\n    request.forEach((req) => {\n      utils.validateAddress(req.validatorAddress, 'validatorAddress');\n      assert(utils.isValidAmount(req.amount), 'Invalid recipient amount');\n\n      if (this._sender === req.validatorAddress) {\n        throw new BuildTransactionError('Sender address cannot be the same as the Staking address');\n      }\n    });\n\n    this._addStakeTx = request;\n    return this;\n  }\n\n  /**\n   * Create a new transaction for withdrawing coins ready to be signed\n   *\n   * @param {RequestWithdrawStakedSui} request\n   */\n  unstake(request: RequestWithdrawStakedSui): this {\n    this.validateSuiObjectRef(request.stakedSui, 'stakedSui');\n    this._withdrawDelegation = request;\n    return this;\n  }\n\n  /** @inheritdoc */\n  protected fromImplementation(rawTransaction: string): Transaction<StakingProgrammableTransaction> {\n    const tx = new StakingTransaction(this._coinConfig);\n    this.validateRawTransaction(rawTransaction);\n    tx.fromRawTransaction(rawTransaction);\n    this.initBuilder(tx);\n    return this.transaction;\n  }\n\n  /** @inheritdoc */\n  protected async buildImplementation(): Promise<Transaction<StakingProgrammableTransaction>> {\n    this.transaction.setSuiTransaction(this.buildSuiTransaction());\n    this.transaction.transactionType(this.transactionType);\n\n    if (this._signer) {\n      this.transaction.sign(this._signer);\n    }\n\n    this._signatures.forEach((signature) => {\n      this.transaction.addSignature(signature.publicKey, signature.signature);\n    });\n\n    this.transaction.loadInputsAndOutputs();\n    return this.transaction;\n  }\n\n  /**\n   * Initialize the transaction builder fields using the decoded transaction data\n   *\n   * @param {StakingTransaction} tx the transaction data\n   */\n  initBuilder(tx: Transaction<StakingProgrammableTransaction>): void {\n    this._transaction = tx;\n\n    if (tx.signature && tx.signature.length > 0) {\n      this._signatures = [tx.suiSignature];\n    }\n\n    const txData = tx.toJson();\n    this.type(SuiTransactionType.AddStake);\n    this.sender(txData.sender);\n    this.gasData(txData.gasData);\n\n    const requests = utils.getStakeRequests(tx.suiTransaction.tx);\n    this.stake(requests);\n  }\n\n  /**\n   * Validates all fields are defined\n   */\n  private validateTransactionFields(): void {\n    assert(this._type, new BuildTransactionError('type is required before building'));\n    assert(this._sender, new BuildTransactionError('sender is required before building'));\n    this._addStakeTx.forEach((req) => {\n      assert(req.validatorAddress, new BuildTransactionError('validator address is required before building'));\n      assert(req.amount, new BuildTransactionError('staking amount is required before building'));\n    });\n    assert(this._gasData, new BuildTransactionError('gasData is required before building'));\n    this.validateGasData(this._gasData);\n  }\n\n  /**\n   * Build SuiTransaction\n   *\n   * @return {BitGoSuiTransaction<MoveCallTx>}\n   * @protected\n   */\n  protected buildSuiTransaction(): SuiTransaction<StakingProgrammableTransaction> {\n    this.validateTransactionFields();\n\n    const programmableTxBuilder = new ProgrammingTransactionBlockBuilder();\n    switch (this._type) {\n      case SuiTransactionType.AddStake:\n        // number of objects passed as gas payment should be strictly less than `MAX_GAS_OBJECTS`. When the transaction\n        // requires a larger number of inputs we use the merge command to merge the rest of the objects into the gasCoin\n        if (this._gasData.payment.length >= MAX_GAS_OBJECTS) {\n          const gasPaymentObjects = this._gasData.payment\n            .slice(MAX_GAS_OBJECTS - 1)\n            .map((object) => Inputs.ObjectRef(object));\n\n          // limit for total number of `args: CallArg[]` for a single command is MAX_COMMAND_ARGS so the max length of\n          // `sources[]` for a `mergeCoins(destination, sources[])` command is MAX_COMMAND_ARGS - 1 (1 used up for\n          // `destination`). We need to create a total of `gasPaymentObjects/(MAX_COMMAND_ARGS - 1)` merge commands to\n          // merge all the objects\n          while (gasPaymentObjects.length > 0) {\n            programmableTxBuilder.mergeCoins(\n              programmableTxBuilder.gas,\n              gasPaymentObjects.splice(0, MAX_COMMAND_ARGS - 1).map((object) => programmableTxBuilder.object(object))\n            );\n          }\n        }\n\n        // Create a new coin with staking balance, based on the coins used as gas payment.\n        this._addStakeTx.forEach((req) => {\n          const coin = programmableTxBuilder.splitCoins(programmableTxBuilder.gas, [\n            programmableTxBuilder.pure(req.amount),\n          ]);\n          // Stake the split coin to a specific validator address.\n          programmableTxBuilder.moveCall({\n            target: `${SUI_SYSTEM_ADDRESS}::${SUI_SYSTEM_MODULE_NAME}::${ADD_STAKE_FUN_NAME}`,\n            arguments: [\n              programmableTxBuilder.object(Inputs.SharedObjectRef(SUI_SYSTEM_STATE_OBJECT)),\n              coin,\n              programmableTxBuilder.pure(Inputs.Pure(req.validatorAddress, BCS.ADDRESS)),\n            ],\n          } as unknown as MoveCallTransaction);\n        });\n        break;\n      case SuiTransactionType.WithdrawStake:\n        // Unstake staked object.\n        programmableTxBuilder.moveCall({\n          target: `${SUI_SYSTEM_ADDRESS}::${SUI_SYSTEM_MODULE_NAME}::${WITHDRAW_STAKE_FUN_NAME}`,\n          arguments: [\n            programmableTxBuilder.object(Inputs.SharedObjectRef(SUI_SYSTEM_STATE_OBJECT)),\n            programmableTxBuilder.pure(Inputs.ObjectRef(this._withdrawDelegation.stakedSui)),\n          ],\n        } as unknown as MoveCallTransaction);\n        break;\n      default:\n        throw new InvalidTransactionError(`unsupported target method`);\n    }\n\n    const txData = programmableTxBuilder.blockData;\n    return {\n      type: this._type,\n      sender: this._sender,\n      tx: {\n        inputs: [...txData.inputs],\n        transactions: [...txData.transactions],\n      },\n      gasData: {\n        ...this._gasData,\n        payment: this._gasData.payment.slice(0, MAX_GAS_OBJECTS - 1),\n      },\n    };\n  }\n}\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!