PHP WebShell

Текущая директория: /opt/BitGoJS/modules/sdk-coin-avaxp/dist/src/lib

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

"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {
    return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.DelegatorTxBuilder = void 0;
const sdk_core_1 = require("@bitgo/sdk-core");
const deprecatedTransactionBuilder_1 = require("./deprecatedTransactionBuilder");
const platformvm_1 = require("avalanche/dist/apis/platformvm");
const avalanche_1 = require("avalanche");
const iface_1 = require("./iface");
const utils_1 = __importDefault(require("./utils"));
const utxoEngine_1 = require("./utxoEngine");
class DelegatorTxBuilder extends deprecatedTransactionBuilder_1.DeprecatedTransactionBuilder {
    /**
     *
     * @param coinConfig
     */
    constructor(coinConfig) {
        super(coinConfig);
        const network = coinConfig.network;
        this._stakeAmount = new avalanche_1.BN(network.minStake);
    }
    /**
     * get transaction type
     * @protected
     */
    get transactionType() {
        return sdk_core_1.TransactionType.AddDelegator;
    }
    /**
     * Addresses where reward should be deposit
     * @param {string | string[]} address - single address or array of addresses to receive rewards
     */
    rewardAddresses(address) {
        const rewardAddresses = address instanceof Array ? address : [address];
        this.transaction._rewardAddresses = rewardAddresses.map(utils_1.default.parseAddress);
        return this;
    }
    /**
     *
     * @param nodeID
     */
    nodeID(value) {
        this.validateNodeID(value);
        this._nodeID = value;
        return this;
    }
    /**
     * start time of staking period
     * @param value
     */
    startTime(value) {
        this._startTime = new avalanche_1.BN(value);
        return this;
    }
    /**
     * end time of staking period
     * @param value
     */
    endTime(value) {
        this._endTime = new avalanche_1.BN(value);
        return this;
    }
    /**
     *
     * @param value
     */
    stakeAmount(value) {
        const valueBN = avalanche_1.BN.isBN(value) ? value : new avalanche_1.BN(value);
        this.validateStakeAmount(valueBN);
        this._stakeAmount = valueBN;
        return this;
    }
    // region Validators
    /**
     * validates a correct NodeID is used
     * @param nodeID
     */
    validateNodeID(nodeID) {
        if (!nodeID) {
            throw new sdk_core_1.BuildTransactionError('Invalid transaction: missing nodeID');
        }
        if (nodeID.slice(0, 6) !== 'NodeID') {
            throw new sdk_core_1.BuildTransactionError('Invalid transaction: invalid NodeID tag');
        }
        const bintools = avalanche_1.BinTools.getInstance();
        if (!(bintools.b58ToBuffer(nodeID.slice(7)).length === 24)) {
            throw new sdk_core_1.BuildTransactionError('Invalid transaction: NodeID is not in cb58 format');
        }
    }
    /**
     *
     *   protected _startTime: Date;
     *   protected _endTime: Date;
     *   2 weeks = 1209600
     *   1 year = 31556926
     *   unix time stamp based off seconds
     */
    validateStakeDuration(startTime, endTime) {
        const oneDayLater = new avalanche_1.BN(Date.now()).add(new avalanche_1.BN(86400));
        if (!startTime.gt(oneDayLater)) {
            throw new sdk_core_1.BuildTransactionError('Start time needs to be one day greater than current time');
        }
        if (endTime < startTime) {
            throw new sdk_core_1.BuildTransactionError('End date cannot be less than start date');
        }
        if (startTime.add(new avalanche_1.BN(this.transaction._network.minStakeDuration)).gt(endTime)) {
            throw new sdk_core_1.BuildTransactionError('End date must be greater than or equal to two weeks');
        }
        if (endTime.gt(startTime.add(new avalanche_1.BN(this.transaction._network.maxStakeDuration)))) {
            throw new sdk_core_1.BuildTransactionError('End date must be less than or equal to one year');
        }
    }
    /**
     *
     * @param amount
     */
    validateStakeAmount(amount) {
        const minStake = new avalanche_1.BN(this.transaction._network.minStake);
        if (amount.lt(minStake)) {
            throw new sdk_core_1.BuildTransactionError('Minimum staking amount is ' + Number(minStake) / 1000000000 + ' AVAX.');
        }
    }
    // endregion
    /** @inheritdoc */
    initBuilder(tx) {
        super.initBuilder(tx);
        const baseTx = tx.getUnsignedTx().getTransaction();
        if (!this.verifyTxType(baseTx)) {
            throw new sdk_core_1.NotSupported('Transaction cannot be parsed or has an unsupported transaction type');
        }
        // The StakeOuts is a {@link stakeTransferOut} result.
        // It's expected to have only one outputs with the addresses of the sender.
        const outputs = baseTx.getStakeOuts();
        if (outputs.length != 1) {
            throw new sdk_core_1.BuildTransactionError('Transaction can have one external output');
        }
        const output = outputs[0];
        if (!output.getAssetID().equals(this.transaction._assetId)) {
            throw new Error('The Asset ID of the output does not match the transaction');
        }
        const secpOut = output.getOutput();
        this.transaction._locktime = secpOut.getLocktime();
        this.transaction._threshold = secpOut.getThreshold();
        // output addresses are the sender addresses
        this.transaction._fromAddresses = secpOut.getAddresses();
        this._nodeID = baseTx.getNodeIDString();
        this._startTime = baseTx.getStartTime();
        this._endTime = baseTx.getEndTime();
        this._stakeAmount = baseTx.getStakeAmount();
        this.transaction._utxos = (0, utxoEngine_1.deprecatedRecoverUtxos)(baseTx.getIns());
        return this;
    }
    static verifyTxType(baseTx) {
        return baseTx.getTypeID() === platformvm_1.PlatformVMConstants.ADDVALIDATORTX;
    }
    verifyTxType(baseTx) {
        return DelegatorTxBuilder.verifyTxType(baseTx);
    }
    /**
     *
     * @protected
     */
    buildAvaxTransaction() {
        this.validateStakeDuration(this._startTime, this._endTime);
        const { inputs, outputs, credentials } = this.createInputOutput();
        this.transaction.setTransaction(new platformvm_1.Tx(new platformvm_1.UnsignedTx(new platformvm_1.AddDelegatorTx(this.transaction._networkID, this.transaction._blockchainID, outputs, inputs, undefined, utils_1.default.NodeIDStringToBuffer(this._nodeID), this._startTime, this._endTime, this._stakeAmount, [this.stakeTransferOut()], this.rewardOwnersOutput())), credentials));
    }
    /**
     * Create the StakeOut where the recipient address are the sender.
     * @protected
     *
     */
    stakeTransferOut() {
        return new platformvm_1.TransferableOutput(this.transaction._assetId, new platformvm_1.SECPTransferOutput(this._stakeAmount, this.transaction._fromAddresses, this.transaction._locktime, this.transaction._threshold));
    }
    rewardOwnersOutput() {
        // if there are no reward addresses, the sender gets the rewards
        if (!this.transaction._rewardAddresses || this.transaction._rewardAddresses.length === 0) {
            this.transaction._rewardAddresses = this.transaction._fromAddresses;
        }
        return new platformvm_1.ParseableOutput(new platformvm_1.SECPOwnerOutput(this.transaction._rewardAddresses, this.transaction._locktime, this.transaction._threshold));
    }
    /**
     * Threshold must be 2 and since output always get reordered we want to make sure we can always add signatures in the correct location
     * To find the correct location for the signature, we use the ouput's addresses to create the signatureIdx in the order that we desire
     * 0: user key, 1: hsm key, 2: recovery key
     * @protected
     */
    createInputOutput() {
        const inputs = [];
        const outputs = [];
        // amount spent so far
        let currentTotal = new avalanche_1.BN(0);
        // delegating and validating have no fees
        const totalTarget = this._stakeAmount.clone();
        const credentials = [];
        // convert fromAddresses to string
        // fromAddresses = bitgo order if we are in WP
        // fromAddresses = onchain order if we are in from
        const bitgoAddresses = this.transaction._fromAddresses.map((b) => utils_1.default.addressToString(this.transaction._network.hrp, this.transaction._network.alias, b));
        /*
        A = user key
        B = hsm key
        C = backup key
        bitgoAddresses = bitgo addresses [ A, B, C ]
        utxo.addresses = IMS addresses [ B, C, A ]
        utxo.addressesIndex = [ 2, 0, 1 ]
        we pick 0, 1 for non-recovery
        we pick 1, 2 for recovery
        */
        this.transaction._utxos.forEach((utxo) => {
            // in WP, output.addressesIndex is empty, so fill it
            if (!utxo.addressesIndex || utxo.addressesIndex.length === 0) {
                utxo.addressesIndex = bitgoAddresses.map((a) => utxo.addresses.indexOf(a));
            }
            // in OVC, output.addressesIndex is defined correctly from the previous iteration
        });
        // validate the utxos
        this.transaction._utxos.forEach((utxo) => {
            if (!utxo) {
                throw new sdk_core_1.BuildTransactionError('Utxo is undefined');
            }
            // addressesIndex should neve have a mismatch
            if (utxo.addressesIndex?.includes(-1)) {
                throw new sdk_core_1.BuildTransactionError('Addresses are inconsistent');
            }
            if (utxo.threshold !== this.transaction._threshold) {
                throw new sdk_core_1.BuildTransactionError('Threshold is inconsistent');
            }
        });
        // if we are in OVC, none of the utxos will have addresses since they come from
        // deserialized inputs (which don't have addresses), not the IMS
        const buildOutputs = this.transaction._utxos[0].addresses.length !== 0;
        this.transaction._utxos.forEach((utxo, i) => {
            if (utxo.outputID === iface_1.SECP256K1_Transfer_Output) {
                const txidBuf = utils_1.default.cb58Decode(utxo.txid);
                const amt = new avalanche_1.BN(utxo.amount);
                const outputidx = utils_1.default.outputidxNumberToBuffer(utxo.outputidx);
                const addressesIndex = utxo.addressesIndex ?? [];
                // either user (0) or recovery (2)
                const firstIndex = this.recoverSigner ? 2 : 0;
                const bitgoIndex = 1;
                currentTotal = currentTotal.add(amt);
                const secpTransferInput = new platformvm_1.SECPTransferInput(amt);
                if (!buildOutputs) {
                    addressesIndex.forEach((i) => secpTransferInput.addSignatureIdx(i, this.transaction._fromAddresses[i]));
                }
                else {
                    // if user/backup > bitgo
                    if (addressesIndex[bitgoIndex] < addressesIndex[firstIndex]) {
                        secpTransferInput.addSignatureIdx(addressesIndex[bitgoIndex], this.transaction._fromAddresses[bitgoIndex]);
                        secpTransferInput.addSignatureIdx(addressesIndex[firstIndex], this.transaction._fromAddresses[firstIndex]);
                        credentials.push((0, platformvm_1.SelectCredentialClass)(secpTransferInput.getCredentialID(), // 9
                        ['', this.transaction._fromAddresses[firstIndex].toString('hex')].map(utils_1.default.createSig)));
                    }
                    else {
                        secpTransferInput.addSignatureIdx(addressesIndex[firstIndex], this.transaction._fromAddresses[firstIndex]);
                        secpTransferInput.addSignatureIdx(addressesIndex[bitgoIndex], this.transaction._fromAddresses[bitgoIndex]);
                        credentials.push((0, platformvm_1.SelectCredentialClass)(secpTransferInput.getCredentialID(), [this.transaction._fromAddresses[firstIndex].toString('hex'), ''].map(utils_1.default.createSig)));
                    }
                }
                const input = new platformvm_1.TransferableInput(txidBuf, outputidx, this.transaction._assetId, secpTransferInput);
                inputs.push(input);
            }
        });
        if (buildOutputs) {
            if (currentTotal.lt(totalTarget)) {
                throw new sdk_core_1.BuildTransactionError(`Utxo outputs get ${currentTotal.toString()} and ${totalTarget.toString()} is required`);
            }
            else if (currentTotal.gt(totalTarget)) {
                outputs.push(new platformvm_1.TransferableOutput(this.transaction._assetId, new platformvm_1.SECPTransferOutput(currentTotal.sub(totalTarget), this.transaction._fromAddresses, this.transaction._locktime, this.transaction._threshold)));
            }
        }
        // get outputs and credentials from the deserialized transaction if we are in OVC
        return {
            inputs,
            outputs: !buildOutputs ? this.transaction.avaxPTransaction.getOuts() : outputs,
            credentials: credentials.length === 0 ? this.transaction.credentials : credentials,
        };
    }
}
exports.DelegatorTxBuilder = DelegatorTxBuilder;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"delegatorTxBuilder.js","sourceRoot":"","sources":["../../../src/lib/delegatorTxBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import { BuildTransactionError, NotSupported, TransactionType } from '@bitgo/sdk-core';\nimport { AvalancheNetwork, BaseCoin as CoinConfig } from '@bitgo/statics';\nimport { DeprecatedTransactionBuilder } from './deprecatedTransactionBuilder';\nimport {\n  AddDelegatorTx,\n  BaseTx as PVMBaseTx,\n  ParseableOutput,\n  PlatformVMConstants,\n  SECPOwnerOutput,\n  SECPTransferInput,\n  SECPTransferOutput,\n  SelectCredentialClass,\n  TransferableInput,\n  TransferableOutput,\n  Tx as PVMTx,\n  UnsignedTx,\n} from 'avalanche/dist/apis/platformvm';\nimport { BinTools, BN } from 'avalanche';\nimport { SECP256K1_Transfer_Output, DeprecatedTx, DeprecatedBaseTx } from './iface';\nimport utils from './utils';\nimport { Credential } from 'avalanche/dist/common';\nimport { deprecatedRecoverUtxos } from './utxoEngine';\n\nexport class DelegatorTxBuilder extends DeprecatedTransactionBuilder {\n  protected _nodeID: string;\n  protected _startTime: BN;\n  protected _endTime: BN;\n  protected _stakeAmount: BN;\n\n  /**\n   *\n   * @param coinConfig\n   */\n  constructor(coinConfig: Readonly<CoinConfig>) {\n    super(coinConfig);\n    const network = coinConfig.network as AvalancheNetwork;\n    this._stakeAmount = new BN(network.minStake);\n  }\n\n  /**\n   * get transaction type\n   * @protected\n   */\n  protected get transactionType(): TransactionType {\n    return TransactionType.AddDelegator;\n  }\n\n  /**\n   * Addresses where reward should be deposit\n   * @param {string | string[]} address - single address or array of addresses to receive rewards\n   */\n  rewardAddresses(address: string | string[]): this {\n    const rewardAddresses = address instanceof Array ? address : [address];\n    this.transaction._rewardAddresses = rewardAddresses.map(utils.parseAddress);\n    return this;\n  }\n\n  /**\n   *\n   * @param nodeID\n   */\n  nodeID(value: string): this {\n    this.validateNodeID(value);\n    this._nodeID = value;\n    return this;\n  }\n\n  /**\n   * start time of staking period\n   * @param value\n   */\n  startTime(value: string | number): this {\n    this._startTime = new BN(value);\n    return this;\n  }\n\n  /**\n   * end time of staking period\n   * @param value\n   */\n  endTime(value: string | number): this {\n    this._endTime = new BN(value);\n    return this;\n  }\n\n  /**\n   *\n   * @param value\n   */\n  stakeAmount(value: BN | string): this {\n    const valueBN = BN.isBN(value) ? value : new BN(value);\n    this.validateStakeAmount(valueBN);\n    this._stakeAmount = valueBN;\n    return this;\n  }\n\n  // region Validators\n  /**\n   * validates a correct NodeID is used\n   * @param nodeID\n   */\n  validateNodeID(nodeID: string): void {\n    if (!nodeID) {\n      throw new BuildTransactionError('Invalid transaction: missing nodeID');\n    }\n    if (nodeID.slice(0, 6) !== 'NodeID') {\n      throw new BuildTransactionError('Invalid transaction: invalid NodeID tag');\n    }\n    const bintools = BinTools.getInstance();\n    if (!(bintools.b58ToBuffer(nodeID.slice(7)).length === 24)) {\n      throw new BuildTransactionError('Invalid transaction: NodeID is not in cb58 format');\n    }\n  }\n  /**\n   *\n   *   protected _startTime: Date;\n   *   protected _endTime: Date;\n   *   2 weeks = 1209600\n   *   1 year = 31556926\n   *   unix time stamp based off seconds\n   */\n  validateStakeDuration(startTime: BN, endTime: BN): void {\n    const oneDayLater = new BN(Date.now()).add(new BN(86400));\n    if (!startTime.gt(oneDayLater)) {\n      throw new BuildTransactionError('Start time needs to be one day greater than current time');\n    }\n    if (endTime < startTime) {\n      throw new BuildTransactionError('End date cannot be less than start date');\n    }\n    if (startTime.add(new BN(this.transaction._network.minStakeDuration)).gt(endTime)) {\n      throw new BuildTransactionError('End date must be greater than or equal to two weeks');\n    }\n    if (endTime.gt(startTime.add(new BN(this.transaction._network.maxStakeDuration)))) {\n      throw new BuildTransactionError('End date must be less than or equal to one year');\n    }\n  }\n\n  /**\n   *\n   * @param amount\n   */\n  validateStakeAmount(amount: BN): void {\n    const minStake = new BN(this.transaction._network.minStake);\n    if (amount.lt(minStake)) {\n      throw new BuildTransactionError('Minimum staking amount is ' + Number(minStake) / 1000000000 + ' AVAX.');\n    }\n  }\n\n  // endregion\n\n  /** @inheritdoc */\n  initBuilder(tx: DeprecatedTx): this {\n    super.initBuilder(tx);\n    const baseTx: DeprecatedBaseTx = tx.getUnsignedTx().getTransaction();\n    if (!this.verifyTxType(baseTx)) {\n      throw new NotSupported('Transaction cannot be parsed or has an unsupported transaction type');\n    }\n    // The StakeOuts is a {@link stakeTransferOut} result.\n    // It's expected to have only one outputs with the addresses of the sender.\n    const outputs = baseTx.getStakeOuts();\n    if (outputs.length != 1) {\n      throw new BuildTransactionError('Transaction can have one external output');\n    }\n    const output = outputs[0];\n    if (!output.getAssetID().equals(this.transaction._assetId)) {\n      throw new Error('The Asset ID of the output does not match the transaction');\n    }\n    const secpOut = output.getOutput();\n    this.transaction._locktime = secpOut.getLocktime();\n    this.transaction._threshold = secpOut.getThreshold();\n    // output addresses are the sender addresses\n    this.transaction._fromAddresses = secpOut.getAddresses();\n    this._nodeID = baseTx.getNodeIDString();\n    this._startTime = baseTx.getStartTime();\n    this._endTime = baseTx.getEndTime();\n    this._stakeAmount = baseTx.getStakeAmount();\n    this.transaction._utxos = deprecatedRecoverUtxos(baseTx.getIns());\n    return this;\n  }\n\n  static verifyTxType(baseTx: DeprecatedBaseTx): baseTx is AddDelegatorTx {\n    return baseTx.getTypeID() === PlatformVMConstants.ADDVALIDATORTX;\n  }\n\n  verifyTxType(baseTx: DeprecatedBaseTx): baseTx is AddDelegatorTx {\n    return DelegatorTxBuilder.verifyTxType(baseTx);\n  }\n\n  /**\n   *\n   * @protected\n   */\n  protected buildAvaxTransaction(): void {\n    this.validateStakeDuration(this._startTime, this._endTime);\n    const { inputs, outputs, credentials } = this.createInputOutput();\n    this.transaction.setTransaction(\n      new PVMTx(\n        new UnsignedTx(\n          new AddDelegatorTx(\n            this.transaction._networkID,\n            this.transaction._blockchainID,\n            outputs,\n            inputs,\n            undefined,\n            utils.NodeIDStringToBuffer(this._nodeID),\n            this._startTime,\n            this._endTime,\n            this._stakeAmount,\n            [this.stakeTransferOut()],\n            this.rewardOwnersOutput()\n          )\n        ),\n        credentials\n      )\n    );\n  }\n\n  /**\n   * Create the StakeOut where the recipient address are the sender.\n   * @protected\n   *\n   */\n  protected stakeTransferOut(): TransferableOutput {\n    return new TransferableOutput(\n      this.transaction._assetId,\n      new SECPTransferOutput(\n        this._stakeAmount,\n        this.transaction._fromAddresses,\n        this.transaction._locktime,\n        this.transaction._threshold\n      )\n    );\n  }\n\n  protected rewardOwnersOutput(): ParseableOutput {\n    // if there are no reward addresses, the sender gets the rewards\n    if (!this.transaction._rewardAddresses || this.transaction._rewardAddresses.length === 0) {\n      this.transaction._rewardAddresses = this.transaction._fromAddresses;\n    }\n\n    return new ParseableOutput(\n      new SECPOwnerOutput(this.transaction._rewardAddresses, this.transaction._locktime, this.transaction._threshold)\n    );\n  }\n\n  /**\n   * Threshold must be 2 and since output always get reordered we want to make sure we can always add signatures in the correct location\n   * To find the correct location for the signature, we use the ouput's addresses to create the signatureIdx in the order that we desire\n   * 0: user key, 1: hsm key, 2: recovery key\n   * @protected\n   */\n  protected createInputOutput(): {\n    inputs: TransferableInput[];\n    outputs: TransferableOutput[];\n    credentials: Credential[];\n  } {\n    const inputs: TransferableInput[] = [];\n    const outputs: TransferableOutput[] = [];\n\n    // amount spent so far\n    let currentTotal: BN = new BN(0);\n\n    // delegating and validating have no fees\n    const totalTarget = this._stakeAmount.clone();\n\n    const credentials: Credential[] = [];\n\n    // convert fromAddresses to string\n    // fromAddresses = bitgo order if we are in WP\n    // fromAddresses = onchain order if we are in from\n    const bitgoAddresses = this.transaction._fromAddresses.map((b) =>\n      utils.addressToString(this.transaction._network.hrp, this.transaction._network.alias, b)\n    );\n\n    /*\n    A = user key\n    B = hsm key\n    C = backup key\n    bitgoAddresses = bitgo addresses [ A, B, C ]\n    utxo.addresses = IMS addresses [ B, C, A ]\n    utxo.addressesIndex = [ 2, 0, 1 ]\n    we pick 0, 1 for non-recovery\n    we pick 1, 2 for recovery\n    */\n    this.transaction._utxos.forEach((utxo) => {\n      // in WP, output.addressesIndex is empty, so fill it\n      if (!utxo.addressesIndex || utxo.addressesIndex.length === 0) {\n        utxo.addressesIndex = bitgoAddresses.map((a) => utxo.addresses.indexOf(a));\n      }\n      // in OVC, output.addressesIndex is defined correctly from the previous iteration\n    });\n\n    // validate the utxos\n    this.transaction._utxos.forEach((utxo) => {\n      if (!utxo) {\n        throw new BuildTransactionError('Utxo is undefined');\n      }\n      // addressesIndex should neve have a mismatch\n      if (utxo.addressesIndex?.includes(-1)) {\n        throw new BuildTransactionError('Addresses are inconsistent');\n      }\n      if (utxo.threshold !== this.transaction._threshold) {\n        throw new BuildTransactionError('Threshold is inconsistent');\n      }\n    });\n\n    // if we are in OVC, none of the utxos will have addresses since they come from\n    // deserialized inputs (which don't have addresses), not the IMS\n    const buildOutputs = this.transaction._utxos[0].addresses.length !== 0;\n\n    this.transaction._utxos.forEach((utxo, i) => {\n      if (utxo.outputID === SECP256K1_Transfer_Output) {\n        const txidBuf = utils.cb58Decode(utxo.txid);\n        const amt: BN = new BN(utxo.amount);\n        const outputidx = utils.outputidxNumberToBuffer(utxo.outputidx);\n        const addressesIndex = utxo.addressesIndex ?? [];\n\n        // either user (0) or recovery (2)\n        const firstIndex = this.recoverSigner ? 2 : 0;\n        const bitgoIndex = 1;\n        currentTotal = currentTotal.add(amt);\n\n        const secpTransferInput = new SECPTransferInput(amt);\n\n        if (!buildOutputs) {\n          addressesIndex.forEach((i) => secpTransferInput.addSignatureIdx(i, this.transaction._fromAddresses[i]));\n        } else {\n          // if user/backup > bitgo\n          if (addressesIndex[bitgoIndex] < addressesIndex[firstIndex]) {\n            secpTransferInput.addSignatureIdx(addressesIndex[bitgoIndex], this.transaction._fromAddresses[bitgoIndex]);\n            secpTransferInput.addSignatureIdx(addressesIndex[firstIndex], this.transaction._fromAddresses[firstIndex]);\n            credentials.push(\n              SelectCredentialClass(\n                secpTransferInput.getCredentialID(), // 9\n                ['', this.transaction._fromAddresses[firstIndex].toString('hex')].map(utils.createSig)\n              )\n            );\n          } else {\n            secpTransferInput.addSignatureIdx(addressesIndex[firstIndex], this.transaction._fromAddresses[firstIndex]);\n            secpTransferInput.addSignatureIdx(addressesIndex[bitgoIndex], this.transaction._fromAddresses[bitgoIndex]);\n            credentials.push(\n              SelectCredentialClass(\n                secpTransferInput.getCredentialID(),\n                [this.transaction._fromAddresses[firstIndex].toString('hex'), ''].map(utils.createSig)\n              )\n            );\n          }\n        }\n\n        const input: TransferableInput = new TransferableInput(\n          txidBuf,\n          outputidx,\n          this.transaction._assetId,\n          secpTransferInput\n        );\n        inputs.push(input);\n      }\n    });\n\n    if (buildOutputs) {\n      if (currentTotal.lt(totalTarget)) {\n        throw new BuildTransactionError(\n          `Utxo outputs get ${currentTotal.toString()} and ${totalTarget.toString()} is required`\n        );\n      } else if (currentTotal.gt(totalTarget)) {\n        outputs.push(\n          new TransferableOutput(\n            this.transaction._assetId,\n            new SECPTransferOutput(\n              currentTotal.sub(totalTarget),\n              this.transaction._fromAddresses,\n              this.transaction._locktime,\n              this.transaction._threshold\n            )\n          )\n        );\n      }\n    }\n    // get outputs and credentials from the deserialized transaction if we are in OVC\n    return {\n      inputs,\n      outputs: !buildOutputs ? (this.transaction.avaxPTransaction as PVMBaseTx).getOuts() : outputs,\n      credentials: credentials.length === 0 ? this.transaction.credentials : credentials,\n    };\n  }\n}\n"]}

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


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