PHP WebShell

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

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

"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {
    return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.PermissionlessValidatorTxBuilder = void 0;
const avalanchejs_1 = require("@bitgo-forks/avalanchejs");
const sdk_core_1 = require("@bitgo/sdk-core");
const avalanche_1 = require("avalanche");
const iface_1 = require("./iface");
const keyPair_1 = require("./keyPair");
const transactionBuilder_1 = require("./transactionBuilder");
const utils_1 = __importDefault(require("./utils"));
const utxoEngine_1 = require("./utxoEngine");
class PermissionlessValidatorTxBuilder extends transactionBuilder_1.TransactionBuilder {
    /**
     *
     * @param coinConfig
     */
    constructor(coinConfig) {
        super(coinConfig);
        this._signer = [];
        this.recoverSigner = false;
        this.transaction._fee.fee = this.transaction._network.txFee;
    }
    /**
     * get transaction type
     * @protected
     */
    get transactionType() {
        return sdk_core_1.TransactionType.AddPermissionlessValidator;
    }
    // region Validators
    /**
     * Validates locktime
     * @param locktime
     */
    validateLocktime(locktime) {
        if (locktime < BigInt(0)) {
            throw new sdk_core_1.BuildTransactionError('Invalid transaction: locktime must be 0 or higher');
        }
    }
    /**
     * Validate that the delegation fee is at least the minDelegationFee
     * @param delegationFeeRate number
     */
    validateDelegationFeeRate(delegationFeeRate) {
        if (delegationFeeRate < Number(this.transaction._network.minDelegationFee)) {
            throw new sdk_core_1.BuildTransactionError(`Delegation fee cannot be less than ${this.transaction._network.minDelegationFee}`);
        }
    }
    /**
     * Check the UTXO has expected fields.
     * @param UTXO
     */
    validateUtxo(value) {
        ['outputID', 'amount', 'txid', 'outputidx'].forEach((field) => {
            if (!value.hasOwnProperty(field))
                throw new sdk_core_1.BuildTransactionError(`Utxos required ${field}`);
        });
    }
    // endregion
    /**
     * 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;
    }
    /** @inheritdoc */
    fromImplementation(rawTransaction) {
        const manager = avalanchejs_1.utils.getManagerForVM('PVM');
        const [codec, rest] = manager.getCodecFromBuffer(avalanchejs_1.utils.hexToBuffer(rawTransaction));
        const tx = codec.UnpackPrefix(rest)[0];
        this.initBuilder(tx);
        return this.transaction;
    }
    /** @inheritdoc */
    async buildImplementation() {
        this.buildAvaxTransaction();
        this.transaction.setTransactionType(this.transactionType);
        if (this.hasSigner()) {
            for (const keyPair of this._signer) {
                await this.transaction.sign(keyPair);
            }
        }
        return this.transaction;
    }
    /**
     *
     * @param nodeID
     */
    nodeID(nodeID) {
        this.validateNodeID(nodeID);
        this._nodeID = nodeID;
        return this;
    }
    /**
     *
     * @param blsPublicKey
     */
    blsPublicKey(blsPublicKey) {
        (0, sdk_core_1.isValidBLSPublicKey)(blsPublicKey);
        this._blsPublicKey = blsPublicKey;
        return this;
    }
    /**
     *
     * @param blsSignature
     */
    blsSignature(blsSignature) {
        (0, sdk_core_1.isValidBLSSignature)(blsSignature);
        this._blsSignature = blsSignature;
        return this;
    }
    /**
     * Locktime is a long that contains the unix timestamp that this output can be spent after.
     * The unix timestamp is specific to the second.
     * @param value
     */
    locktime(value) {
        this.validateLocktime(BigInt(value));
        this._transaction._locktime = BigInt(value);
        return this;
    }
    /**
     * set the delegationFeeRate
     * @param value number
     */
    delegationFeeRate(value) {
        this.validateDelegationFeeRate(value);
        this._delegationFeeRate = value;
        return this;
    }
    /**
     * start time of staking period
     * @param value
     */
    startTime(value) {
        this._startTime = BigInt(value);
        return this;
    }
    /**
     * end time of staking period
     * @param value
     */
    endTime(value) {
        this._endTime = BigInt(value);
        return this;
    }
    /**
     *
     * @param value
     */
    stakeAmount(value) {
        const valueBigInt = typeof value === 'bigint' ? value : BigInt(value);
        this.validateStakeAmount(valueBigInt);
        this._stakeAmount = valueBigInt;
        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');
        }
        if (!(avalanchejs_1.utils.base58.decode(nodeID.slice(7)).length === 24)) {
            throw new sdk_core_1.BuildTransactionError('Invalid transaction: NodeID is not in cb58 format');
        }
    }
    /**
     * Validate stake duration
     * @param startTime
     * @param endTime
     */
    validateStakeDuration(startTime, endTime) {
        if (endTime < startTime) {
            throw new sdk_core_1.BuildTransactionError('End date cannot be less than start date');
        }
    }
    /**
     * Validate stake amount
     * @param amount
     */
    validateStakeAmount(amount) {
        const minStake = BigInt(this.transaction._network.minStake);
        if (amount < minStake) {
            throw new sdk_core_1.BuildTransactionError('Minimum staking amount is ' + Number(minStake) / 1000000000 + ' AVAX.');
        }
        return;
    }
    // endregion
    /** @inheritdoc */
    initBuilder(tx) {
        super.initBuilder(tx);
        const permissionlessValidatorTx = tx.tx;
        if (!this.verifyTxType(permissionlessValidatorTx)) {
            throw new sdk_core_1.NotSupported('Transaction cannot be parsed or has an unsupported transaction type');
        }
        const outputs = permissionlessValidatorTx.baseTx.outputs;
        if (outputs.length !== 1) {
            throw new sdk_core_1.BuildTransactionError('Transaction can have one external output');
        }
        const output = outputs[0].output;
        if (outputs[0].getAssetId() !== this.transaction._assetId) {
            throw new Error('The Asset ID of the output does not match the transaction');
        }
        this.transaction._blsPublicKey = avalanchejs_1.utils.bufferToHex(permissionlessValidatorTx.signer.proof.publicKey);
        this._blsPublicKey = this.transaction._blsPublicKey;
        this.transaction._blsSignature = avalanchejs_1.utils.bufferToHex(permissionlessValidatorTx.signer.proof.signature);
        this._blsSignature = this.transaction._blsSignature;
        this.transaction._locktime = output.outputOwners.locktime.value();
        this.transaction._threshold = output.outputOwners.threshold.value();
        this.transaction._nodeID = permissionlessValidatorTx.subnetValidator.validator.nodeId.toString();
        this._nodeID = this.transaction._nodeID;
        this.transaction._startTime = permissionlessValidatorTx.subnetValidator.validator.startTime.value();
        this._startTime = this.transaction._startTime;
        this.transaction._endTime = permissionlessValidatorTx.subnetValidator.validator.endTime.value();
        this._endTime = this.transaction._endTime;
        this.transaction._fromAddresses = output.outputOwners.addrs.map((a) => a.toBytes());
        this.transaction._stakeAmount = permissionlessValidatorTx.stake[0].output.amount();
        this.stakeAmount(this.transaction._stakeAmount);
        this.transaction._utxos = (0, utxoEngine_1.recoverUtxos)(permissionlessValidatorTx.getInputs());
        // TODO(CR-1073): remove log
        console.log('utxos: ', this.transaction._utxos);
        console.log('fromAddresses: ', this.transaction.fromAddresses);
        return this;
    }
    static verifyTxType(type) {
        return type === avalanchejs_1.TypeSymbols.AddPermissionlessValidatorTx;
    }
    verifyTxType(tx) {
        return PermissionlessValidatorTxBuilder.verifyTxType(tx._type);
    }
    /**
     * Since addresses in outputs get reordered, we need to make sure signatures
     * are added in the correct position
     * To find the position, we use the output's addresses to create the
     * signatureIdx in the order needed (i.e. [user, bitgo, backup])
     * @protected
     */
    calculateUtxos() {
        const inputs = [];
        const stakeOutputs = [];
        const changeOutputs = [];
        const utxos = [];
        let currentTotal = BigInt(0);
        // staking tx requires fees after ACP-103
        // https://github.com/avalanche-foundation/ACPs/blob/main/ACPs/103-dynamic-fees/README.md
        const totalTarget = this._stakeAmount + BigInt(this.transaction._network.txFee);
        const credentials = this.transaction.credentials ?? [];
        // Convert fromAddresses to string
        // The order of fromAddresses is determined by the source of the data
        // When building from params, the order is [user, bitgo, backup]
        // The order from tx hex is [bitgo, backup, user]
        const bitgoAddresses = this.transaction._fromAddresses.map((b) => avalanchejs_1.utils.format(this.transaction._network.alias, this.transaction._network.hrp, b));
        // TODO(CR-1073): remove log
        console.log(`bitgoAddress: ${bitgoAddresses}`);
        // 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[0].addressesIndex?.length !== 0;
        const assetId = avalanchejs_1.Id.fromString(this.transaction._assetId);
        this.transaction._utxos.forEach((utxo, index) => {
            // validate the utxos
            if (!utxo) {
                throw new sdk_core_1.BuildTransactionError('Utxo is undefined');
            }
            // addressesIndex should never 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');
            }
            const bitgoIndexToOnChainIndex = new Map();
            // 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));
            }
            // utxo.addresses is null when build from raw
            // but utxo.addressesIndex has only 2 elements when build from raw
            // so the bitgoIndexToOnChainIndex map will be empty
            utxo.addresses.forEach((a) => {
                bitgoIndexToOnChainIndex.set(bitgoAddresses.indexOf(a), utxo.addresses.indexOf(a));
            });
            // TODO(CR-1073): remove log
            console.log(`utxo.addresses: ${utxo.addresses}`);
            console.log(`bitgoIndexToOnChainIndex: ${Array.from(bitgoIndexToOnChainIndex)}`);
            // in OVC, output.addressesIndex is defined correctly from the previous iteration
            if (utxo.outputID === iface_1.SECP256K1_Transfer_Output) {
                const utxoAmount = BigInt(utxo.amount);
                // either user (0) or recovery (2)
                // On regular mode: [user, bitgo] (i.e. [0, 1])
                // On recovery mode: [backup, bitgo] (i.e. [2, 1])
                const userOrBackupIndex = this.recoverSigner ? 2 : 0;
                const bitgoIndex = 1;
                currentTotal = currentTotal + utxoAmount;
                const utxoId = avalanchejs_1.avaxSerial.UTXOID.fromNative(utxo.txid, Number(utxo.outputidx));
                let addressesIndex = [];
                if (utxo.addressesIndex && bitgoIndexToOnChainIndex.size === 0) {
                    addressesIndex = [...utxo.addressesIndex];
                }
                else {
                    addressesIndex.push(bitgoIndexToOnChainIndex.get(userOrBackupIndex));
                    addressesIndex.push(bitgoIndexToOnChainIndex.get(bitgoIndex));
                }
                const transferInputs = new avalanchejs_1.TransferInput(new avalanchejs_1.BigIntPr(utxoAmount), new avalanchejs_1.Input([...addressesIndex].sort().map((num) => new avalanchejs_1.Int(num))));
                // TODO(CR-1073): remove log
                console.log(`using addressesIndex sorted: ${[...addressesIndex].sort()}`);
                const input = new avalanchejs_1.avaxSerial.TransferableInput(utxoId, assetId, transferInputs);
                utxos.push(new avalanchejs_1.Utxo(utxoId, assetId, transferInputs));
                inputs.push(input);
                if (!this.transaction.credentials || this.transaction.credentials.length == 0) {
                    if (buildOutputs) {
                        // For the bitgo signature we create an empty signature
                        // For the user/backup signature we store the address that matches the key
                        // if bitgo address comes before  < user/backup address
                        // TODO(CR-1073): remove log
                        console.log(`bitgo index on chain: ${utxo.addressesIndex[bitgoIndex]}`);
                        console.log(`user Or Backup Index: ${utxo.addressesIndex[userOrBackupIndex]}`);
                        if (utxo.addressesIndex[bitgoIndex] < utxo.addressesIndex[userOrBackupIndex]) {
                            // TODO(CR-1073): remove log
                            console.log(`user or backup credentials after bitgo`);
                            credentials.push(new avalanchejs_1.Credential([
                                utils_1.default.createNewSig(avalanche_1.Buffer.from('').toString('hex')),
                                utils_1.default.createNewSig(avalanche_1.Buffer.from(this.transaction._fromAddresses[userOrBackupIndex]).toString('hex')),
                            ]));
                        }
                        else {
                            // TODO(CR-1073): remove log
                            console.log(`user or backup credentials before bitgo`);
                            credentials.push(new avalanchejs_1.Credential([
                                utils_1.default.createNewSig(avalanche_1.Buffer.from(this.transaction._fromAddresses[userOrBackupIndex]).toString('hex')),
                                utils_1.default.createNewSig(avalanche_1.Buffer.from('').toString('hex')),
                            ]));
                        }
                    }
                    else {
                        // TODO(CR-1073): verify this else case for OVC
                        credentials.push(new avalanchejs_1.Credential(addressesIndex.map((i) => utils_1.default.createNewSig(avalanche_1.Buffer.from(this.transaction._fromAddresses[i]).toString('hex')))));
                    }
                }
                else {
                    // TODO(CR-1073): remove log
                    console.log(`reusing credentials from transaction`);
                }
            }
        });
        if (buildOutputs) {
            if (currentTotal < totalTarget) {
                throw new sdk_core_1.BuildTransactionError(`Utxo outputs get ${currentTotal.toString()} and ${totalTarget.toString()} is required`);
            }
            else if (currentTotal >= totalTarget) {
                const stakeOutput = new avalanchejs_1.avaxSerial.TransferableOutput(assetId, new avalanchejs_1.TransferOutput(new avalanchejs_1.BigIntPr(this._stakeAmount), new avalanchejs_1.OutputOwners(new avalanchejs_1.BigIntPr(this.transaction._locktime), new avalanchejs_1.Int(this.transaction._threshold), [...this.transaction._fromAddresses]
                    .sort((a, b) => avalanchejs_1.utils.bytesCompare(a, b))
                    .map((a) => avalanchejs_1.Address.fromBytes(a)[0]))));
                stakeOutputs.push(stakeOutput);
                if (currentTotal > totalTarget) {
                    const changeOutput = new avalanchejs_1.avaxSerial.TransferableOutput(assetId, new avalanchejs_1.TransferOutput(new avalanchejs_1.BigIntPr(currentTotal - totalTarget), new avalanchejs_1.OutputOwners(new avalanchejs_1.BigIntPr(this.transaction._locktime), new avalanchejs_1.Int(this.transaction._threshold), [...this.transaction._fromAddresses]
                        .sort((a, b) => avalanchejs_1.utils.bytesCompare(a, b))
                        .map((a) => avalanchejs_1.Address.fromBytes(a)[0]))));
                    changeOutputs.push(changeOutput);
                }
            }
        }
        inputs.sort((a, b) => {
            if (avalanchejs_1.utils.bytesEqual(a.utxoID.txID.toBytes(), b.utxoID.txID.toBytes())) {
                return a.utxoID.outputIdx.value() - b.utxoID.outputIdx.value();
            }
            return avalanchejs_1.utils.bytesCompare(a.utxoID.txID.toBytes(), b.utxoID.txID.toBytes());
        });
        return { inputs, stakeOutputs, changeOutputs, utxos, credentials };
    }
    /**
     * Build the add validator transaction
     * @protected
     */
    buildAvaxTransaction() {
        this.validateStakeDuration(this.transaction._startTime, this.transaction._endTime);
        const { inputs, stakeOutputs, changeOutputs, utxos, credentials } = this.calculateUtxos();
        const baseTx = avalanchejs_1.avaxSerial.BaseTx.fromNative(this.transaction._networkID, this.transaction._blockchainID, changeOutputs, inputs, new Uint8Array() // default empty memo
        );
        const subnetValidator = avalanchejs_1.pvmSerial.SubnetValidator.fromNative(this._nodeID, this._startTime, this._endTime, this._stakeAmount, avalanchejs_1.networkIDs.PrimaryNetworkID);
        const signer = new avalanchejs_1.pvmSerial.Signer(new avalanchejs_1.pvmSerial.ProofOfPossession(avalanchejs_1.utils.hexToBuffer(this._blsPublicKey), avalanchejs_1.utils.hexToBuffer(this._blsSignature)));
        const outputOwners = new avalanchejs_1.OutputOwners(new avalanchejs_1.BigIntPr(this.transaction._locktime), new avalanchejs_1.Int(this.transaction._threshold), [...this.transaction._fromAddresses]
            .sort((a, b) => avalanchejs_1.utils.bytesCompare(a, b))
            .map((a) => avalanchejs_1.Address.fromBytes(a)[0]));
        // TODO(CR-1073): check this value
        //  Shares 10,000 times percentage of reward taken from delegators
        //  https://docs.avax.network/reference/avalanchego/p-chain/txn-format#unsigned-add-validator-tx
        const shares = new avalanchejs_1.Int(1e4 * 2);
        const addressMaps = [...this.transaction._fromAddresses]
            .sort((a, b) => avalanchejs_1.utils.bytesCompare(a, b))
            .map((address) => new avalanchejs_1.utils.AddressMap([[new avalanchejs_1.Address(address), 0]]));
        this.transaction.setTransaction(new avalanchejs_1.UnsignedTx(new avalanchejs_1.pvmSerial.AddPermissionlessValidatorTx(baseTx, subnetValidator, signer, stakeOutputs, outputOwners, outputOwners, shares), utxos, new avalanchejs_1.utils.AddressMaps(addressMaps), credentials));
    }
    /** @inheritdoc */
    signImplementation({ key }) {
        this._signer.push(new keyPair_1.KeyPair({ prv: key }));
        return this.transaction;
    }
    /** @inheritdoc */
    validateAddress(address, addressFormat) {
        if (!utils_1.default.isValidAddress(address.address)) {
            throw new sdk_core_1.BuildTransactionError('Invalid address');
        }
    }
    /** @inheritdoc */
    get transaction() {
        return this._transaction;
    }
    set transaction(transaction) {
        this._transaction = transaction;
    }
    hasSigner() {
        return this._signer !== undefined && this._signer.length > 0;
    }
    /** @inheritdoc */
    validateKey({ key }) {
        if (!new keyPair_1.KeyPair({ prv: key })) {
            throw new sdk_core_1.BuildTransactionError('Invalid key');
        }
    }
    /**
     * Check the raw transaction has a valid format in the blockchain context, throw otherwise.
     *
     * @param rawTransaction Transaction in any format
     */
    validateRawTransaction(rawTransaction) {
        utils_1.default.validateRawTransaction(rawTransaction);
    }
    /** @inheritdoc */
    validateTransaction(transaction) {
        // throw new NotImplementedError('validateTransaction not implemented');
    }
    /** @inheritdoc */
    validateValue(value) {
        if (value.isLessThan(0)) {
            throw new sdk_core_1.BuildTransactionError('Value cannot be less than zero');
        }
    }
}
exports.PermissionlessValidatorTxBuilder = PermissionlessValidatorTxBuilder;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"permissionlessValidatorTxBuilder.js","sourceRoot":"","sources":["../../../src/lib/permissionlessValidatorTxBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Address,\n  avaxSerial,\n  utils as AvaxUtils,\n  BigIntPr,\n  Credential,\n  Id,\n  Input,\n  Int,\n  networkIDs,\n  OutputOwners,\n  pvmSerial,\n  TransferInput,\n  TransferOutput,\n  TypeSymbols,\n  UnsignedTx,\n  Utxo,\n  utils as avaxUtils,\n} from '@bitgo-forks/avalanchejs';\nimport {\n  BaseAddress,\n  BaseKey,\n  BuildTransactionError,\n  isValidBLSPublicKey,\n  isValidBLSSignature,\n  NotSupported,\n  TransactionType,\n} from '@bitgo/sdk-core';\n\nimport { BaseCoin as CoinConfig } from '@bitgo/statics';\nimport { Buffer as BufferAvax } from 'avalanche';\nimport BigNumber from 'bignumber.js';\nimport { DecodedUtxoObj, SECP256K1_Transfer_Output, Tx } from './iface';\nimport { KeyPair } from './keyPair';\nimport { Transaction } from './transaction';\nimport { TransactionBuilder } from './transactionBuilder';\nimport utils from './utils';\nimport { recoverUtxos } from './utxoEngine';\n\nexport class PermissionlessValidatorTxBuilder extends TransactionBuilder {\n  public _signer: KeyPair[] = [];\n  protected _nodeID: string;\n  protected _blsPublicKey: string;\n  protected _blsSignature: string;\n  protected _startTime: bigint;\n  protected _endTime: bigint;\n  protected _stakeAmount: bigint;\n  protected recoverSigner = false;\n  protected _delegationFeeRate: number;\n\n  /**\n   *\n   * @param coinConfig\n   */\n  constructor(coinConfig: Readonly<CoinConfig>) {\n    super(coinConfig);\n    this.transaction._fee.fee = this.transaction._network.txFee;\n  }\n\n  /**\n   * get transaction type\n   * @protected\n   */\n  protected get transactionType(): TransactionType {\n    return TransactionType.AddPermissionlessValidator;\n  }\n\n  // region Validators\n  /**\n   * Validates locktime\n   * @param locktime\n   */\n  validateLocktime(locktime: bigint): void {\n    if (locktime < BigInt(0)) {\n      throw new BuildTransactionError('Invalid transaction: locktime must be 0 or higher');\n    }\n  }\n\n  /**\n   * Validate that the delegation fee is at least the minDelegationFee\n   * @param delegationFeeRate number\n   */\n  validateDelegationFeeRate(delegationFeeRate: number): void {\n    if (delegationFeeRate < Number(this.transaction._network.minDelegationFee)) {\n      throw new BuildTransactionError(\n        `Delegation fee cannot be less than ${this.transaction._network.minDelegationFee}`\n      );\n    }\n  }\n\n  /**\n   * Check the UTXO has expected fields.\n   * @param UTXO\n   */\n  validateUtxo(value: DecodedUtxoObj): void {\n    ['outputID', 'amount', 'txid', 'outputidx'].forEach((field) => {\n      if (!value.hasOwnProperty(field)) throw new BuildTransactionError(`Utxos required ${field}`);\n    });\n  }\n  // endregion\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  /** @inheritdoc */\n  protected fromImplementation(rawTransaction: string): Transaction {\n    const manager = AvaxUtils.getManagerForVM('PVM');\n    const [codec, rest] = manager.getCodecFromBuffer(AvaxUtils.hexToBuffer(rawTransaction));\n    const tx = codec.UnpackPrefix<pvmSerial.AddPermissionlessValidatorTx>(rest)[0];\n    this.initBuilder(tx);\n    return this.transaction;\n  }\n\n  /** @inheritdoc */\n  protected async buildImplementation(): Promise<Transaction> {\n    this.buildAvaxTransaction();\n    this.transaction.setTransactionType(this.transactionType);\n    if (this.hasSigner()) {\n      for (const keyPair of this._signer) {\n        await this.transaction.sign(keyPair);\n      }\n    }\n    return this.transaction;\n  }\n\n  /**\n   *\n   * @param nodeID\n   */\n  nodeID(nodeID: string): this {\n    this.validateNodeID(nodeID);\n    this._nodeID = nodeID;\n    return this;\n  }\n\n  /**\n   *\n   * @param blsPublicKey\n   */\n  blsPublicKey(blsPublicKey: string): this {\n    isValidBLSPublicKey(blsPublicKey);\n    this._blsPublicKey = blsPublicKey;\n    return this;\n  }\n\n  /**\n   *\n   * @param blsSignature\n   */\n  blsSignature(blsSignature: string): this {\n    isValidBLSSignature(blsSignature);\n    this._blsSignature = blsSignature;\n    return this;\n  }\n\n  /**\n   * Locktime is a long that contains the unix timestamp that this output can be spent after.\n   * The unix timestamp is specific to the second.\n   * @param value\n   */\n  locktime(value: string | number): this {\n    this.validateLocktime(BigInt(value));\n    this._transaction._locktime = BigInt(value);\n    return this;\n  }\n\n  /**\n   * set the delegationFeeRate\n   * @param value number\n   */\n  delegationFeeRate(value: number): this {\n    this.validateDelegationFeeRate(value);\n    this._delegationFeeRate = 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 = BigInt(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 = BigInt(value);\n    return this;\n  }\n\n  /**\n   *\n   * @param value\n   */\n  stakeAmount(value: bigint | string): this {\n    const valueBigInt = typeof value === 'bigint' ? value : BigInt(value);\n    this.validateStakeAmount(valueBigInt);\n    this._stakeAmount = valueBigInt;\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    if (!(AvaxUtils.base58.decode(nodeID.slice(7)).length === 24)) {\n      throw new BuildTransactionError('Invalid transaction: NodeID is not in cb58 format');\n    }\n  }\n\n  /**\n   * Validate stake duration\n   * @param startTime\n   * @param endTime\n   */\n  validateStakeDuration(startTime: bigint, endTime: bigint): void {\n    if (endTime < startTime) {\n      throw new BuildTransactionError('End date cannot be less than start date');\n    }\n  }\n\n  /**\n   * Validate stake amount\n   * @param amount\n   */\n  validateStakeAmount(amount: bigint): void {\n    const minStake = BigInt(this.transaction._network.minStake);\n    if (amount < minStake) {\n      throw new BuildTransactionError('Minimum staking amount is ' + Number(minStake) / 1000000000 + ' AVAX.');\n    }\n    return;\n  }\n\n  // endregion\n\n  /** @inheritdoc */\n  initBuilder(tx: Tx): this {\n    super.initBuilder(tx);\n    const permissionlessValidatorTx = (tx as UnsignedTx).tx as pvmSerial.AddPermissionlessValidatorTx;\n    if (!this.verifyTxType(permissionlessValidatorTx)) {\n      throw new NotSupported('Transaction cannot be parsed or has an unsupported transaction type');\n    }\n\n    const outputs = permissionlessValidatorTx.baseTx.outputs;\n    if (outputs.length !== 1) {\n      throw new BuildTransactionError('Transaction can have one external output');\n    }\n\n    const output = outputs[0].output as TransferOutput;\n    if (outputs[0].getAssetId() !== this.transaction._assetId) {\n      throw new Error('The Asset ID of the output does not match the transaction');\n    }\n\n    this.transaction._blsPublicKey = AvaxUtils.bufferToHex(\n      (permissionlessValidatorTx.signer as pvmSerial.Signer).proof.publicKey\n    );\n    this._blsPublicKey = this.transaction._blsPublicKey;\n    this.transaction._blsSignature = AvaxUtils.bufferToHex(\n      (permissionlessValidatorTx.signer as pvmSerial.Signer).proof.signature\n    );\n    this._blsSignature = this.transaction._blsSignature;\n\n    this.transaction._locktime = output.outputOwners.locktime.value();\n    this.transaction._threshold = output.outputOwners.threshold.value();\n    this.transaction._nodeID = permissionlessValidatorTx.subnetValidator.validator.nodeId.toString();\n    this._nodeID = this.transaction._nodeID;\n    this.transaction._startTime = permissionlessValidatorTx.subnetValidator.validator.startTime.value();\n    this._startTime = this.transaction._startTime;\n    this.transaction._endTime = permissionlessValidatorTx.subnetValidator.validator.endTime.value();\n    this._endTime = this.transaction._endTime;\n    this.transaction._fromAddresses = output.outputOwners.addrs.map((a) => a.toBytes());\n    this.transaction._stakeAmount = permissionlessValidatorTx.stake[0].output.amount();\n    this.stakeAmount(this.transaction._stakeAmount);\n    this.transaction._utxos = recoverUtxos(permissionlessValidatorTx.getInputs());\n    // TODO(CR-1073): remove log\n    console.log('utxos: ', this.transaction._utxos);\n    console.log('fromAddresses: ', this.transaction.fromAddresses);\n    return this;\n  }\n\n  static verifyTxType(type: TypeSymbols): boolean {\n    return type === TypeSymbols.AddPermissionlessValidatorTx;\n  }\n\n  verifyTxType(tx: Tx): tx is pvmSerial.AddPermissionlessValidatorTx {\n    return PermissionlessValidatorTxBuilder.verifyTxType((tx as pvmSerial.AddPermissionlessValidatorTx)._type);\n  }\n\n  /**\n   * Since addresses in outputs get reordered, we need to make sure signatures\n   * are added in the correct position\n   * To find the position, we use the output's addresses to create the\n   * signatureIdx in the order needed (i.e. [user, bitgo, backup])\n   * @protected\n   */\n  protected calculateUtxos(): {\n    inputs: avaxSerial.TransferableInput[];\n    stakeOutputs: avaxSerial.TransferableOutput[];\n    changeOutputs: avaxSerial.TransferableOutput[];\n    utxos: Utxo[];\n    credentials: Credential[];\n  } {\n    const inputs: avaxSerial.TransferableInput[] = [];\n    const stakeOutputs: avaxSerial.TransferableOutput[] = [];\n    const changeOutputs: avaxSerial.TransferableOutput[] = [];\n    const utxos: Utxo[] = [];\n\n    let currentTotal = BigInt(0);\n\n    // staking tx requires fees after ACP-103\n    // https://github.com/avalanche-foundation/ACPs/blob/main/ACPs/103-dynamic-fees/README.md\n    const totalTarget = this._stakeAmount + BigInt(this.transaction._network.txFee);\n\n    const credentials: Credential[] = this.transaction.credentials ?? [];\n    // Convert fromAddresses to string\n    // The order of fromAddresses is determined by the source of the data\n    // When building from params, the order is [user, bitgo, backup]\n    // The order from tx hex is [bitgo, backup, user]\n    const bitgoAddresses = this.transaction._fromAddresses.map((b) =>\n      avaxUtils.format(this.transaction._network.alias, this.transaction._network.hrp, b)\n    );\n    // TODO(CR-1073): remove log\n    console.log(`bitgoAddress: ${bitgoAddresses}`);\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 =\n      this.transaction._utxos[0].addresses.length !== 0 || this.transaction._utxos[0].addressesIndex?.length !== 0;\n\n    const assetId = Id.fromString(this.transaction._assetId);\n    this.transaction._utxos.forEach((utxo, index) => {\n      // validate the utxos\n      if (!utxo) {\n        throw new BuildTransactionError('Utxo is undefined');\n      }\n      // addressesIndex should never 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      const bitgoIndexToOnChainIndex = new Map();\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      // utxo.addresses is null when build from raw\n      // but utxo.addressesIndex has only 2 elements when build from raw\n      // so the bitgoIndexToOnChainIndex map will be empty\n      utxo.addresses.forEach((a) => {\n        bitgoIndexToOnChainIndex.set(bitgoAddresses.indexOf(a), utxo.addresses.indexOf(a));\n      });\n      // TODO(CR-1073): remove log\n      console.log(`utxo.addresses: ${utxo.addresses}`);\n      console.log(`bitgoIndexToOnChainIndex: ${Array.from(bitgoIndexToOnChainIndex)}`);\n      // in OVC, output.addressesIndex is defined correctly from the previous iteration\n\n      if (utxo.outputID === SECP256K1_Transfer_Output) {\n        const utxoAmount = BigInt(utxo.amount);\n        // either user (0) or recovery (2)\n        // On regular mode: [user, bitgo] (i.e. [0, 1])\n        // On recovery mode: [backup, bitgo] (i.e. [2, 1])\n        const userOrBackupIndex = this.recoverSigner ? 2 : 0;\n        const bitgoIndex = 1;\n\n        currentTotal = currentTotal + utxoAmount;\n\n        const utxoId = avaxSerial.UTXOID.fromNative(utxo.txid, Number(utxo.outputidx));\n\n        let addressesIndex: number[] = [];\n        if (utxo.addressesIndex && bitgoIndexToOnChainIndex.size === 0) {\n          addressesIndex = [...utxo.addressesIndex];\n        } else {\n          addressesIndex.push(bitgoIndexToOnChainIndex.get(userOrBackupIndex));\n          addressesIndex.push(bitgoIndexToOnChainIndex.get(bitgoIndex));\n        }\n\n        const transferInputs = new TransferInput(\n          new BigIntPr(utxoAmount),\n          new Input([...addressesIndex].sort().map((num) => new Int(num)))\n        );\n        // TODO(CR-1073): remove log\n        console.log(`using addressesIndex sorted: ${[...addressesIndex].sort()}`);\n\n        const input = new avaxSerial.TransferableInput(utxoId, assetId, transferInputs);\n        utxos.push(new Utxo(utxoId, assetId, transferInputs));\n\n        inputs.push(input);\n        if (!this.transaction.credentials || this.transaction.credentials.length == 0) {\n          if (buildOutputs) {\n            // For the bitgo signature we create an empty signature\n            // For the user/backup signature we store the address that matches the key\n            // if bitgo address comes before  < user/backup address\n\n            // TODO(CR-1073): remove log\n            console.log(`bitgo index on chain: ${utxo.addressesIndex[bitgoIndex]}`);\n            console.log(`user Or Backup Index: ${utxo.addressesIndex[userOrBackupIndex]}`);\n            if (utxo.addressesIndex[bitgoIndex] < utxo.addressesIndex[userOrBackupIndex]) {\n              // TODO(CR-1073): remove log\n              console.log(`user or backup credentials after bitgo`);\n              credentials.push(\n                new Credential([\n                  utils.createNewSig(BufferAvax.from('').toString('hex')),\n                  utils.createNewSig(\n                    BufferAvax.from(this.transaction._fromAddresses[userOrBackupIndex]).toString('hex')\n                  ),\n                ])\n              );\n            } else {\n              // TODO(CR-1073): remove log\n              console.log(`user or backup credentials before bitgo`);\n              credentials.push(\n                new Credential([\n                  utils.createNewSig(\n                    BufferAvax.from(this.transaction._fromAddresses[userOrBackupIndex]).toString('hex')\n                  ),\n                  utils.createNewSig(BufferAvax.from('').toString('hex')),\n                ])\n              );\n            }\n          } else {\n            // TODO(CR-1073): verify this else case for OVC\n            credentials.push(\n              new Credential(\n                addressesIndex.map((i) =>\n                  utils.createNewSig(BufferAvax.from(this.transaction._fromAddresses[i]).toString('hex'))\n                )\n              )\n            );\n          }\n        } else {\n          // TODO(CR-1073): remove log\n          console.log(`reusing credentials from transaction`);\n        }\n      }\n    });\n\n    if (buildOutputs) {\n      if (currentTotal < totalTarget) {\n        throw new BuildTransactionError(\n          `Utxo outputs get ${currentTotal.toString()} and ${totalTarget.toString()} is required`\n        );\n      } else if (currentTotal >= totalTarget) {\n        const stakeOutput = new avaxSerial.TransferableOutput(\n          assetId,\n          new TransferOutput(\n            new BigIntPr(this._stakeAmount),\n            new OutputOwners(\n              new BigIntPr(this.transaction._locktime),\n              new Int(this.transaction._threshold),\n              [...this.transaction._fromAddresses]\n                .sort((a, b) => avaxUtils.bytesCompare(a, b))\n                .map((a) => Address.fromBytes(a)[0])\n            )\n          )\n        );\n        stakeOutputs.push(stakeOutput);\n\n        if (currentTotal > totalTarget) {\n          const changeOutput = new avaxSerial.TransferableOutput(\n            assetId,\n            new TransferOutput(\n              new BigIntPr(currentTotal - totalTarget),\n              new OutputOwners(\n                new BigIntPr(this.transaction._locktime),\n                new Int(this.transaction._threshold),\n                [...this.transaction._fromAddresses]\n                  .sort((a, b) => avaxUtils.bytesCompare(a, b))\n                  .map((a) => Address.fromBytes(a)[0])\n              )\n            )\n          );\n          changeOutputs.push(changeOutput);\n        }\n      }\n    }\n    inputs.sort((a, b) => {\n      if (avaxUtils.bytesEqual(a.utxoID.txID.toBytes(), b.utxoID.txID.toBytes())) {\n        return a.utxoID.outputIdx.value() - b.utxoID.outputIdx.value();\n      }\n      return avaxUtils.bytesCompare(a.utxoID.txID.toBytes(), b.utxoID.txID.toBytes());\n    });\n    return { inputs, stakeOutputs, changeOutputs, utxos, credentials };\n  }\n\n  /**\n   * Build the add validator transaction\n   * @protected\n   */\n  protected buildAvaxTransaction(): void {\n    this.validateStakeDuration(this.transaction._startTime, this.transaction._endTime);\n    const { inputs, stakeOutputs, changeOutputs, utxos, credentials } = this.calculateUtxos();\n    const baseTx = avaxSerial.BaseTx.fromNative(\n      this.transaction._networkID,\n      this.transaction._blockchainID,\n      changeOutputs,\n      inputs,\n      new Uint8Array() // default empty memo\n    );\n\n    const subnetValidator = pvmSerial.SubnetValidator.fromNative(\n      this._nodeID,\n      this._startTime,\n      this._endTime,\n      this._stakeAmount,\n      networkIDs.PrimaryNetworkID\n    );\n\n    const signer = new pvmSerial.Signer(\n      new pvmSerial.ProofOfPossession(\n        AvaxUtils.hexToBuffer(this._blsPublicKey),\n        AvaxUtils.hexToBuffer(this._blsSignature)\n      )\n    );\n\n    const outputOwners = new OutputOwners(\n      new BigIntPr(this.transaction._locktime),\n      new Int(this.transaction._threshold),\n      [...this.transaction._fromAddresses]\n        .sort((a, b) => avaxUtils.bytesCompare(a, b))\n        .map((a) => Address.fromBytes(a)[0])\n    );\n\n    // TODO(CR-1073): check this value\n    //  Shares 10,000 times percentage of reward taken from delegators\n    //  https://docs.avax.network/reference/avalanchego/p-chain/txn-format#unsigned-add-validator-tx\n    const shares = new Int(1e4 * 2);\n\n    const addressMaps = [...this.transaction._fromAddresses]\n      .sort((a, b) => avaxUtils.bytesCompare(a, b))\n      .map((address) => new AvaxUtils.AddressMap([[new Address(address), 0]]));\n\n    this.transaction.setTransaction(\n      new UnsignedTx(\n        new pvmSerial.AddPermissionlessValidatorTx(\n          baseTx,\n          subnetValidator,\n          signer,\n          stakeOutputs,\n          outputOwners,\n          outputOwners,\n          shares\n        ),\n        utxos,\n        new AvaxUtils.AddressMaps(addressMaps),\n        credentials\n      )\n    );\n  }\n\n  /** @inheritdoc */\n  protected signImplementation({ key }: BaseKey): Transaction {\n    this._signer.push(new KeyPair({ prv: key }));\n    return this.transaction;\n  }\n\n  /** @inheritdoc */\n  validateAddress(address: BaseAddress, addressFormat?: string): void {\n    if (!utils.isValidAddress(address.address)) {\n      throw new BuildTransactionError('Invalid address');\n    }\n  }\n\n  /** @inheritdoc */\n  protected get transaction(): Transaction {\n    return this._transaction;\n  }\n\n  protected set transaction(transaction: Transaction) {\n    this._transaction = transaction;\n  }\n\n  hasSigner(): boolean {\n    return this._signer !== undefined && this._signer.length > 0;\n  }\n\n  /** @inheritdoc */\n  validateKey({ key }: BaseKey): void {\n    if (!new KeyPair({ prv: key })) {\n      throw new BuildTransactionError('Invalid key');\n    }\n  }\n\n  /**\n   * Check the raw transaction has a valid format in the blockchain context, throw otherwise.\n   *\n   * @param rawTransaction Transaction in any format\n   */\n  validateRawTransaction(rawTransaction: string): void {\n    utils.validateRawTransaction(rawTransaction);\n  }\n\n  /** @inheritdoc */\n  validateTransaction(transaction?: Transaction): void {\n    // throw new NotImplementedError('validateTransaction not implemented');\n  }\n\n  /** @inheritdoc */\n  validateValue(value: BigNumber): void {\n    if (value.isLessThan(0)) {\n      throw new BuildTransactionError('Value cannot be less than zero');\n    }\n  }\n}\n"]}

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


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