PHP WebShell

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

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

"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {
    return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.TransactionBuilder = void 0;
const sdk_core_1 = require("@bitgo/sdk-core");
const bignumber_js_1 = __importDefault(require("bignumber.js"));
const keyPair_1 = require("./keyPair");
const multisigUtils_1 = require("./multisigUtils");
const transaction_1 = require("./transaction");
const transferBuilder_1 = require("./transferBuilder");
const utils_1 = require("./utils");
const DEFAULT_M = 3;
/**
 * Tezos transaction builder.
 */
class TransactionBuilder extends sdk_core_1.BaseTransactionBuilder {
    /**
     * Public constructor.
     *
     * @param {CoinConfig} _coinConfig - coin configuration
     */
    constructor(_coinConfig) {
        super(_coinConfig);
        this._type = sdk_core_1.TransactionType.Send;
        this._counter = new bignumber_js_1.default(0);
        this._transfers = [];
        this._walletOwnerPublicKeys = [];
        this._multisigSignerKeyPairs = [];
        this._dataToSignOverride = [];
        this.transaction = new transaction_1.Transaction(_coinConfig);
    }
    // region Base Builder
    /** @inheritdoc */
    fromImplementation(rawTransaction) {
        // Decoding the transaction is an async operation, so save it and leave the decoding for the
        // build step
        this._serializedTransaction = rawTransaction;
        return new transaction_1.Transaction(this._coinConfig);
    }
    /** @inheritdoc */
    signImplementation(key) {
        const signer = new keyPair_1.KeyPair({ prv: key.key });
        // Currently public key revelation is the only type of account update tx supported in Tezos
        if (this._type === sdk_core_1.TransactionType.AccountUpdate && !this._publicKeyToReveal) {
            throw new sdk_core_1.SigningError('Cannot sign a public key revelation transaction without public key');
        }
        if (this._type === sdk_core_1.TransactionType.WalletInitialization && this._walletOwnerPublicKeys.length === 0) {
            throw new sdk_core_1.SigningError('Cannot sign an wallet initialization transaction without owners');
        }
        if (this._type === sdk_core_1.TransactionType.Send &&
            this._transfers.length === 0 &&
            this._serializedTransaction === undefined) {
            throw new sdk_core_1.SigningError('Cannot sign an empty send transaction');
        }
        if (this._type === sdk_core_1.TransactionType.Send && (!this._sourceAddress || this._sourceAddress !== signer.getAddress())) {
            // If the signer is not the source and it is a send transaction, add it to the list of
            // multisig wallet signers
            // TODO: support a combination of keys with and without custom index
            if (key.index && key.index >= DEFAULT_M) {
                throw new sdk_core_1.BuildTransactionError('Custom index cannot be greater than the wallet total number of signers (owners)');
            }
            // Make sure either all keys passed have a custom index or none of them have
            const shouldHaveCustomIndex = key.hasOwnProperty('index');
            for (let i = 0; i < this._multisigSignerKeyPairs.length; i++) {
                if (shouldHaveCustomIndex !== (this._multisigSignerKeyPairs[i].index !== undefined)) {
                    throw new sdk_core_1.BuildTransactionError('Custom index has to be set for all multisig contract signing keys or none');
                }
            }
            const multisigSignerKey = shouldHaveCustomIndex ? { key: signer, index: key.index } : { key: signer };
            this._multisigSignerKeyPairs.push(multisigSignerKey);
        }
        else {
            if (this._sourceKeyPair) {
                throw new sdk_core_1.SigningError('Cannot sign multiple times a non send-type transaction');
            }
            this._sourceKeyPair = signer;
        }
        // Signing the transaction is an async operation, so save the source and leave the actual
        // signing for the build step
        return this.transaction;
    }
    /** @inheritdoc */
    async buildImplementation() {
        // If the from() method was called, use the serialized transaction as a base
        if (this._serializedTransaction) {
            await this.transaction.initFromSerializedTransaction(this._serializedTransaction);
            for (let i = 0; i < this._dataToSignOverride.length; i++) {
                const signatures = await this.getSignatures(this._dataToSignOverride[i].dataToSign);
                await this.transaction.addTransferSignature(signatures, this._dataToSignOverride[i].index || i);
            }
            // TODO: make changes to the transaction if any extra parameter has been set then sign it
        }
        else {
            let contents = [];
            switch (this._type) {
                case sdk_core_1.TransactionType.AccountUpdate:
                    if (this._publicKeyToReveal) {
                        contents.push(this.buildPublicKeyRevelationOperation());
                    }
                    break;
                case sdk_core_1.TransactionType.WalletInitialization:
                    if (this._publicKeyToReveal) {
                        contents.push(this.buildPublicKeyRevelationOperation());
                    }
                    contents.push(this.buildWalletInitializationOperations());
                    break;
                case sdk_core_1.TransactionType.Send:
                    if (this._publicKeyToReveal) {
                        contents.push(this.buildPublicKeyRevelationOperation());
                    }
                    contents = contents.concat(await this.buildSendTransactionContent());
                    break;
                case sdk_core_1.TransactionType.AddressInitialization:
                    if (this._publicKeyToReveal) {
                        contents.push(this.buildPublicKeyRevelationOperation());
                    }
                    contents = contents.concat(this.buildForwarderDeploymentContent());
                    break;
                case sdk_core_1.TransactionType.SingleSigSend:
                    // No support for revelation txns as primary use case is to send from fee address
                    contents = contents.concat(await this.buildSendTransactionContent());
                    break;
                default:
                    throw new sdk_core_1.BuildTransactionError('Unsupported transaction type');
            }
            if (contents.length === 0) {
                throw new sdk_core_1.BuildTransactionError('Empty transaction');
            }
            const parsedTransaction = {
                branch: this._blockHeader,
                contents,
            };
            this.transaction = new transaction_1.Transaction(this._coinConfig);
            // Build and sign a new transaction based on the latest changes
            await this.transaction.initFromParsedTransaction(parsedTransaction);
        }
        if (this._sourceKeyPair && this._sourceKeyPair.getKeys().prv) {
            // TODO: check if there are more signers than needed for a singlesig or multisig transaction
            await this.transaction.sign(this._sourceKeyPair);
        }
        return this.transaction;
    }
    // endregion
    // region Common builder methods
    /**
     * Set the transaction branch id.
     *
     * @param {string} blockId A block hash to use as branch reference
     */
    branch(blockId) {
        if (!(0, utils_1.isValidBlockHash)(blockId)) {
            throw new sdk_core_1.BuildTransactionError('Invalid block hash ' + blockId);
        }
        this._blockHeader = blockId;
    }
    /**
     * The type of transaction being built.
     *
     * @param {TransactionType} type - type of the transaction
     */
    type(type) {
        if (type === sdk_core_1.TransactionType.Send && this._walletOwnerPublicKeys.length > 0) {
            throw new sdk_core_1.BuildTransactionError('Transaction cannot be labeled as Send when owners have already been set');
        }
        if (type !== sdk_core_1.TransactionType.Send && this._transfers.length > 0) {
            throw new sdk_core_1.BuildTransactionError('Transaction contains transfers and can only be labeled as Send');
        }
        this._type = type;
    }
    /**
     * Set the transaction fees. Low fees may get a transaction rejected or never picked up by bakers.
     *
     * @param {Fee} fee Baker fees. May also include the maximum gas and storage fees to pay
     */
    fee(fee) {
        this.validateValue(new bignumber_js_1.default(fee.fee));
        if (fee.gasLimit) {
            this.validateValue(new bignumber_js_1.default(fee.gasLimit));
        }
        if (fee.storageLimit) {
            this.validateValue(new bignumber_js_1.default(fee.storageLimit));
        }
        this._fee = fee;
    }
    /**
     * Set the transaction initiator. This account will pay for the transaction fees, but it will not
     * be added as an owner of a wallet in a init transaction, unless manually set as one of the
     * owners.
     *
     * @param {string} source A Tezos address
     */
    source(source) {
        this.validateAddress({ address: source });
        this._sourceAddress = source;
    }
    /**
     * Set an amount of mutez to transfer in this transaction this transaction. This is different than
     * the amount to transfer from a multisig wallet.
     *
     * @param {string} amount Amount in mutez (1/1000000 Tezies)
     */
    initialBalance(amount) {
        if (this._type !== sdk_core_1.TransactionType.WalletInitialization) {
            throw new sdk_core_1.BuildTransactionError('Initial balance can only be set for wallet initialization transactions');
        }
        this.validateValue(new bignumber_js_1.default(amount));
        this._initialBalance = amount;
    }
    /**
     * Set the transaction counter to prevent submitting repeated transactions.
     *
     * @param {string} counter The counter to use
     */
    counter(counter) {
        this._counter = new bignumber_js_1.default(counter);
    }
    /**
     * Set the destination address of a forwarder contract
     * Used in forwarder contract deployment as destination address
     *
     * @param {string} contractAddress - contract address to use
     */
    forwarderDestination(contractAddress) {
        if (this._type !== sdk_core_1.TransactionType.AddressInitialization) {
            throw new sdk_core_1.BuildTransactionError('Forwarder destination can only be set for address initialization transactions');
        }
        if (!(0, utils_1.isValidOriginatedAddress)(contractAddress)) {
            throw new sdk_core_1.BuildTransactionError('Forwarder destination can only be an originated address');
        }
        this._forwarderDestination = contractAddress;
    }
    // endregion
    // region PublicKeyRevelation builder methods
    /**
     * The public key to reveal.
     *
     * @param {string} publicKey A Tezos public key
     */
    publicKeyToReveal(publicKey) {
        if (this._publicKeyToReveal) {
            throw new sdk_core_1.BuildTransactionError('Public key to reveal already set: ' + this._publicKeyToReveal);
        }
        const keyPair = new keyPair_1.KeyPair({ pub: publicKey });
        if (keyPair.getAddress() !== this._sourceAddress) {
            throw new sdk_core_1.BuildTransactionError('Public key does not match the source address: ' + this._sourceAddress);
        }
        this._publicKeyToReveal = keyPair.getKeys().pub;
    }
    /**
     * Build a reveal operation for the source account with default fees.
     *
     * @returns {RevealOp} A Tezos reveal operation
     */
    buildPublicKeyRevelationOperation() {
        const operation = (0, multisigUtils_1.revealOperation)(this._counter.toString(), this._sourceAddress, this._publicKeyToReveal);
        this._counter = this._counter.plus(1);
        return operation;
    }
    // endregion
    // region WalletInitialization builder methods
    /**
     * Set one of the owners of the multisig wallet.
     *
     * @param {string} publicKey A Tezos public key
     */
    owner(publicKey) {
        if (this._type !== sdk_core_1.TransactionType.WalletInitialization) {
            throw new sdk_core_1.BuildTransactionError('Multisig wallet owner can only be set for initialization transactions');
        }
        if (this._walletOwnerPublicKeys.length >= DEFAULT_M) {
            throw new sdk_core_1.BuildTransactionError('A maximum of ' + DEFAULT_M + ' owners can be set for a multisig wallet');
        }
        if (!(0, utils_1.isValidPublicKey)(publicKey)) {
            throw new sdk_core_1.BuildTransactionError('Invalid public key: ' + publicKey);
        }
        if (this._walletOwnerPublicKeys.includes(publicKey)) {
            throw new sdk_core_1.BuildTransactionError('Repeated owner public key: ' + publicKey);
        }
        this._walletOwnerPublicKeys.push(publicKey);
    }
    /**
     * Set an initial delegate to initialize this wallet to. This is different than the delegation to
     * set while doing a separate delegation transaction.
     *
     * @param {string} delegate The address to delegate the funds to
     */
    initialDelegate(delegate) {
        if (this._type !== sdk_core_1.TransactionType.WalletInitialization) {
            throw new sdk_core_1.BuildTransactionError('Initial delegation can only be set for wallet initialization transactions');
        }
        this.validateAddress({ address: delegate });
        this._initialDelegate = delegate;
    }
    /**
     * Build an origination operation for a generic multisig contract.
     *
     * @returns {Operation} A Tezos origination operation
     */
    buildWalletInitializationOperations() {
        const originationOp = (0, multisigUtils_1.genericMultisigOriginationOperation)(this._counter.toString(), this._sourceAddress, this._fee.fee, this._fee.gasLimit || utils_1.DEFAULT_GAS_LIMIT.ORIGINATION.toString(), this._fee.storageLimit || utils_1.DEFAULT_STORAGE_LIMIT.ORIGINATION.toString(), this._initialBalance || '0', this._walletOwnerPublicKeys, this._initialDelegate);
        this._counter = this._counter.plus(1);
        return originationOp;
    }
    // endregion
    // region Send builder methods
    /**
     * Initialize a new TransferBuilder to for a singlesig or multisig transaction.
     *
     * @param {string} amount Amount in mutez to be transferred
     * @returns {TransferBuilder} A transfer builder
     */
    transfer(amount) {
        if (this._type !== sdk_core_1.TransactionType.Send && this._type !== sdk_core_1.TransactionType.SingleSigSend) {
            throw new sdk_core_1.BuildTransactionError('Transfers can only be set for send transactions');
        }
        let transferBuilder = new transferBuilder_1.TransferBuilder();
        // If source was set, use it as default for
        if (this._sourceAddress) {
            transferBuilder = transferBuilder.from(this._sourceAddress);
        }
        if (this._fee) {
            transferBuilder = transferBuilder.fee(this._fee.fee);
            transferBuilder = this._fee.gasLimit ? transferBuilder.gasLimit(this._fee.gasLimit) : transferBuilder;
            transferBuilder = this._fee.storageLimit ? transferBuilder.storageLimit(this._fee.storageLimit) : transferBuilder;
        }
        this._transfers.push(transferBuilder);
        return transferBuilder.amount(amount);
    }
    /**
     * Calculate the signatures for the multisig transaction.
     *
     * @param {string} packedData The string in hexadecimal to sign
     * @returns {Promise<string[]>} List of signatures for packedData
     */
    async getSignatures(packedData) {
        const signatures = [];
        // Generate the multisig contract signatures
        for (let i = 0; i < this._multisigSignerKeyPairs.length; i++) {
            const signature = await (0, utils_1.sign)(this._multisigSignerKeyPairs[i].key, packedData, new Uint8Array(0));
            const index = this._multisigSignerKeyPairs[i].index;
            signatures.push(index ? { signature: signature.sig, index } : { signature: signature.sig });
        }
        return signatures;
    }
    /**
     * Override the data to sign for a specific transfer. Used for offline signing to pass the
     * respective dataToSign for transfer at a particular index.
     *
     * @param {DataToSignOverride} data - data to override
     */
    overrideDataToSign(data) {
        if (!data.index) {
            data.index = this._dataToSignOverride.length;
        }
        this._dataToSignOverride.push(data);
    }
    /**
     * Build a transaction operation for a generic multisig contract.
     *
     * @returns {Promise<TransactionOp[]>} A Tezos transaction operation
     */
    async buildSendTransactionContent() {
        const contents = [];
        for (let i = 0; i < this._transfers.length; i++) {
            const transfer = this._transfers[i].build();
            let transactionOp;
            if ((0, utils_1.isValidOriginatedAddress)(transfer.from)) {
                // Offline transactions may not have the data to sign
                const signatures = transfer.dataToSign ? await this.getSignatures(transfer.dataToSign) : [];
                transactionOp = (0, multisigUtils_1.multisigTransactionOperation)(this._counter.toString(), this._sourceAddress, transfer.amount, transfer.from, transfer.counter || '0', transfer.to, signatures, transfer.fee.fee, transfer.fee.gasLimit, transfer.fee.storageLimit);
            }
            else {
                transactionOp = (0, multisigUtils_1.singlesigTransactionOperation)(this._counter.toString(), this._sourceAddress, transfer.amount, transfer.to, transfer.fee.fee, transfer.fee.gasLimit, transfer.fee.storageLimit);
            }
            contents.push(transactionOp);
            this._counter = this._counter.plus(1);
        }
        return contents;
    }
    // endregion
    // region ForwarderAddressDeployment
    /**
     * Build a transaction operation for a forwarder contract
     *
     * @returns {OriginationOp} a Tezos transaction operation
     */
    buildForwarderDeploymentContent() {
        const operation = (0, multisigUtils_1.forwarderOriginationOperation)(this._forwarderDestination, this._counter.toString(), this._sourceAddress, this._fee.fee, this._fee.gasLimit || utils_1.DEFAULT_GAS_LIMIT.ORIGINATION.toString(), this._fee.storageLimit || utils_1.DEFAULT_STORAGE_LIMIT.ORIGINATION.toString(), this._initialBalance || '0');
        this._counter = this._counter.plus(1);
        return operation;
    }
    // endregion
    // region Validators
    /** @inheritdoc */
    validateValue(value) {
        if (value.isLessThan(0)) {
            throw new sdk_core_1.BuildTransactionError('Value cannot be below less than zero');
        }
        // TODO: validate the amount is not bigger than the max amount in Tezos
    }
    /** @inheritdoc */
    validateAddress(address) {
        if (!(0, utils_1.isValidAddress)(address.address)) {
            throw new sdk_core_1.BuildTransactionError('Invalid address ' + address.address);
        }
    }
    /** @inheritdoc */
    validateKey(key) {
        const keyPair = new keyPair_1.KeyPair({ prv: key.key });
        if (!keyPair.getKeys().prv) {
            throw new sdk_core_1.BuildTransactionError('Invalid key');
        }
    }
    /** @inheritdoc */
    validateRawTransaction(rawTransaction) {
        // TODO: validate the transaction is either a JSON or a hex
    }
    /** @inheritdoc */
    validateTransaction(transaction) {
        // TODO: validate all required fields are present in the builder before buildImplementation
        switch (this._type) {
            case sdk_core_1.TransactionType.AccountUpdate:
                break;
            case sdk_core_1.TransactionType.WalletInitialization:
                break;
            case sdk_core_1.TransactionType.Send:
                break;
            case sdk_core_1.TransactionType.AddressInitialization:
                break;
            case sdk_core_1.TransactionType.SingleSigSend:
                break;
            default:
                throw new sdk_core_1.BuildTransactionError('Transaction type not supported');
        }
    }
    // endregion
    /** @inheritdoc */
    displayName() {
        return this._coinConfig.fullName;
    }
    /** @inheritdoc */
    get transaction() {
        return this._transaction;
    }
    /** @inheritdoc */
    set transaction(transaction) {
        this._transaction = transaction;
    }
}
exports.TransactionBuilder = TransactionBuilder;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transactionBuilder.js","sourceRoot":"","sources":["../../../src/lib/transactionBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import { BaseKey, BuildTransactionError, SigningError, BaseTransactionBuilder, TransactionType } from '@bitgo/sdk-core';\nimport { BaseCoin as CoinConfig } from '@bitgo/statics';\nimport BigNumber from 'bignumber.js';\nimport { Address } from './address';\nimport { Fee, IndexedData, IndexedSignature, Key, Operation, OriginationOp, RevealOp, TransactionOp } from './iface';\nimport { KeyPair } from './keyPair';\nimport {\n  forwarderOriginationOperation,\n  genericMultisigOriginationOperation,\n  multisigTransactionOperation,\n  revealOperation,\n  singlesigTransactionOperation,\n} from './multisigUtils';\nimport { Transaction } from './transaction';\nimport { TransferBuilder } from './transferBuilder';\nimport {\n  DEFAULT_GAS_LIMIT,\n  DEFAULT_STORAGE_LIMIT,\n  isValidAddress,\n  isValidBlockHash,\n  isValidOriginatedAddress,\n  isValidPublicKey,\n  sign,\n} from './utils';\n\nconst DEFAULT_M = 3;\n\ninterface DataToSignOverride extends IndexedData {\n  dataToSign: string;\n}\n\ninterface IndexedKeyPair extends IndexedData {\n  key: KeyPair;\n}\n\n/**\n * Tezos transaction builder.\n */\nexport class TransactionBuilder extends BaseTransactionBuilder {\n  private _serializedTransaction: string;\n  private _transaction: Transaction;\n  private _type: TransactionType;\n  private _blockHeader: string;\n  private _counter: BigNumber;\n  private _fee: Fee;\n  private _sourceAddress: string;\n  private _sourceKeyPair?: KeyPair;\n\n  // Public key revelation transaction parameters\n  private _publicKeyToReveal: string;\n\n  // Wallet initialization transaction parameters\n  private _initialBalance: string;\n  private _initialDelegate: string;\n  private _walletOwnerPublicKeys: string[];\n\n  // Send transaction parameters\n  private _multisigSignerKeyPairs: IndexedKeyPair[];\n  private _dataToSignOverride: DataToSignOverride[];\n  private _transfers: TransferBuilder[];\n\n  // Address initialization parameters\n  private _forwarderDestination: string;\n\n  /**\n   * Public constructor.\n   *\n   * @param {CoinConfig} _coinConfig - coin configuration\n   */\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    this._type = TransactionType.Send;\n    this._counter = new BigNumber(0);\n    this._transfers = [];\n    this._walletOwnerPublicKeys = [];\n    this._multisigSignerKeyPairs = [];\n    this._dataToSignOverride = [];\n    this.transaction = new Transaction(_coinConfig);\n  }\n\n  // region Base Builder\n  /** @inheritdoc */\n  protected fromImplementation(rawTransaction: string): Transaction {\n    // Decoding the transaction is an async operation, so save it and leave the decoding for the\n    // build step\n    this._serializedTransaction = rawTransaction;\n    return new Transaction(this._coinConfig);\n  }\n\n  /** @inheritdoc */\n  protected signImplementation(key: Key): Transaction {\n    const signer = new KeyPair({ prv: key.key });\n    // Currently public key revelation is the only type of account update tx supported in Tezos\n    if (this._type === TransactionType.AccountUpdate && !this._publicKeyToReveal) {\n      throw new SigningError('Cannot sign a public key revelation transaction without public key');\n    }\n\n    if (this._type === TransactionType.WalletInitialization && this._walletOwnerPublicKeys.length === 0) {\n      throw new SigningError('Cannot sign an wallet initialization transaction without owners');\n    }\n\n    if (\n      this._type === TransactionType.Send &&\n      this._transfers.length === 0 &&\n      this._serializedTransaction === undefined\n    ) {\n      throw new SigningError('Cannot sign an empty send transaction');\n    }\n\n    if (this._type === TransactionType.Send && (!this._sourceAddress || this._sourceAddress !== signer.getAddress())) {\n      // If the signer is not the source and it is a send transaction, add it to the list of\n      // multisig wallet signers\n\n      // TODO: support a combination of keys with and without custom index\n      if (key.index && key.index >= DEFAULT_M) {\n        throw new BuildTransactionError(\n          'Custom index cannot be greater than the wallet total number of signers (owners)'\n        );\n      }\n      // Make sure either all keys passed have a custom index or none of them have\n      const shouldHaveCustomIndex = key.hasOwnProperty('index');\n      for (let i = 0; i < this._multisigSignerKeyPairs.length; i++) {\n        if (shouldHaveCustomIndex !== (this._multisigSignerKeyPairs[i].index !== undefined)) {\n          throw new BuildTransactionError('Custom index has to be set for all multisig contract signing keys or none');\n        }\n      }\n      const multisigSignerKey = shouldHaveCustomIndex ? { key: signer, index: key.index } : { key: signer };\n      this._multisigSignerKeyPairs.push(multisigSignerKey);\n    } else {\n      if (this._sourceKeyPair) {\n        throw new SigningError('Cannot sign multiple times a non send-type transaction');\n      }\n      this._sourceKeyPair = signer;\n    }\n\n    // Signing the transaction is an async operation, so save the source and leave the actual\n    // signing for the build step\n    return this.transaction;\n  }\n\n  /** @inheritdoc */\n  protected async buildImplementation(): Promise<Transaction> {\n    // If the from() method was called, use the serialized transaction as a base\n    if (this._serializedTransaction) {\n      await this.transaction.initFromSerializedTransaction(this._serializedTransaction);\n      for (let i = 0; i < this._dataToSignOverride.length; i++) {\n        const signatures = await this.getSignatures(this._dataToSignOverride[i].dataToSign);\n        await this.transaction.addTransferSignature(signatures, this._dataToSignOverride[i].index || i);\n      }\n      // TODO: make changes to the transaction if any extra parameter has been set then sign it\n    } else {\n      let contents: Operation[] = [];\n      switch (this._type) {\n        case TransactionType.AccountUpdate:\n          if (this._publicKeyToReveal) {\n            contents.push(this.buildPublicKeyRevelationOperation());\n          }\n          break;\n        case TransactionType.WalletInitialization:\n          if (this._publicKeyToReveal) {\n            contents.push(this.buildPublicKeyRevelationOperation());\n          }\n          contents.push(this.buildWalletInitializationOperations());\n          break;\n        case TransactionType.Send:\n          if (this._publicKeyToReveal) {\n            contents.push(this.buildPublicKeyRevelationOperation());\n          }\n          contents = contents.concat(await this.buildSendTransactionContent());\n          break;\n        case TransactionType.AddressInitialization:\n          if (this._publicKeyToReveal) {\n            contents.push(this.buildPublicKeyRevelationOperation());\n          }\n          contents = contents.concat(this.buildForwarderDeploymentContent());\n          break;\n        case TransactionType.SingleSigSend:\n          // No support for revelation txns as primary use case is to send from fee address\n          contents = contents.concat(await this.buildSendTransactionContent());\n          break;\n        default:\n          throw new BuildTransactionError('Unsupported transaction type');\n      }\n      if (contents.length === 0) {\n        throw new BuildTransactionError('Empty transaction');\n      }\n      const parsedTransaction = {\n        branch: this._blockHeader,\n        contents,\n      };\n\n      this.transaction = new Transaction(this._coinConfig);\n      // Build and sign a new transaction based on the latest changes\n      await this.transaction.initFromParsedTransaction(parsedTransaction);\n    }\n\n    if (this._sourceKeyPair && this._sourceKeyPair.getKeys().prv) {\n      // TODO: check if there are more signers than needed for a singlesig or multisig transaction\n      await this.transaction.sign(this._sourceKeyPair);\n    }\n    return this.transaction;\n  }\n  // endregion\n\n  // region Common builder methods\n  /**\n   * Set the transaction branch id.\n   *\n   * @param {string} blockId A block hash to use as branch reference\n   */\n  branch(blockId: string): void {\n    if (!isValidBlockHash(blockId)) {\n      throw new BuildTransactionError('Invalid block hash ' + blockId);\n    }\n    this._blockHeader = blockId;\n  }\n\n  /**\n   * The type of transaction being built.\n   *\n   * @param {TransactionType} type - type of the transaction\n   */\n  type(type: TransactionType): void {\n    if (type === TransactionType.Send && this._walletOwnerPublicKeys.length > 0) {\n      throw new BuildTransactionError('Transaction cannot be labeled as Send when owners have already been set');\n    }\n    if (type !== TransactionType.Send && this._transfers.length > 0) {\n      throw new BuildTransactionError('Transaction contains transfers and can only be labeled as Send');\n    }\n    this._type = type;\n  }\n\n  /**\n   * Set the transaction fees. Low fees may get a transaction rejected or never picked up by bakers.\n   *\n   * @param {Fee} fee Baker fees. May also include the maximum gas and storage fees to pay\n   */\n  fee(fee: Fee): void {\n    this.validateValue(new BigNumber(fee.fee));\n    if (fee.gasLimit) {\n      this.validateValue(new BigNumber(fee.gasLimit));\n    }\n    if (fee.storageLimit) {\n      this.validateValue(new BigNumber(fee.storageLimit));\n    }\n    this._fee = fee;\n  }\n\n  /**\n   * Set the transaction initiator. This account will pay for the transaction fees, but it will not\n   * be added as an owner of a wallet in a init transaction, unless manually set as one of the\n   * owners.\n   *\n   * @param {string} source A Tezos address\n   */\n  source(source: string): void {\n    this.validateAddress({ address: source });\n    this._sourceAddress = source;\n  }\n\n  /**\n   * Set an amount of mutez to transfer in this transaction this transaction. This is different than\n   * the amount to transfer from a multisig wallet.\n   *\n   * @param {string} amount Amount in mutez (1/1000000 Tezies)\n   */\n  initialBalance(amount: string): void {\n    if (this._type !== TransactionType.WalletInitialization) {\n      throw new BuildTransactionError('Initial balance can only be set for wallet initialization transactions');\n    }\n    this.validateValue(new BigNumber(amount));\n    this._initialBalance = amount;\n  }\n\n  /**\n   * Set the transaction counter to prevent submitting repeated transactions.\n   *\n   * @param {string} counter The counter to use\n   */\n  counter(counter: string): void {\n    this._counter = new BigNumber(counter);\n  }\n\n  /**\n   * Set the destination address of a forwarder contract\n   * Used in forwarder contract deployment as destination address\n   *\n   * @param {string} contractAddress - contract address to use\n   */\n  forwarderDestination(contractAddress: string): void {\n    if (this._type !== TransactionType.AddressInitialization) {\n      throw new BuildTransactionError('Forwarder destination can only be set for address initialization transactions');\n    }\n    if (!isValidOriginatedAddress(contractAddress)) {\n      throw new BuildTransactionError('Forwarder destination can only be an originated address');\n    }\n    this._forwarderDestination = contractAddress;\n  }\n\n  // endregion\n\n  // region PublicKeyRevelation builder methods\n  /**\n   * The public key to reveal.\n   *\n   * @param {string} publicKey A Tezos public key\n   */\n  publicKeyToReveal(publicKey: string): void {\n    if (this._publicKeyToReveal) {\n      throw new BuildTransactionError('Public key to reveal already set: ' + this._publicKeyToReveal);\n    }\n\n    const keyPair = new KeyPair({ pub: publicKey });\n    if (keyPair.getAddress() !== this._sourceAddress) {\n      throw new BuildTransactionError('Public key does not match the source address: ' + this._sourceAddress);\n    }\n    this._publicKeyToReveal = keyPair.getKeys().pub;\n  }\n\n  /**\n   * Build a reveal operation for the source account with default fees.\n   *\n   * @returns {RevealOp} A Tezos reveal operation\n   */\n  private buildPublicKeyRevelationOperation(): RevealOp {\n    const operation = revealOperation(this._counter.toString(), this._sourceAddress, this._publicKeyToReveal);\n    this._counter = this._counter.plus(1);\n    return operation;\n  }\n  // endregion\n\n  // region WalletInitialization builder methods\n  /**\n   * Set one of the owners of the multisig wallet.\n   *\n   * @param {string} publicKey A Tezos public key\n   */\n  owner(publicKey: string): void {\n    if (this._type !== TransactionType.WalletInitialization) {\n      throw new BuildTransactionError('Multisig wallet owner can only be set for initialization transactions');\n    }\n    if (this._walletOwnerPublicKeys.length >= DEFAULT_M) {\n      throw new BuildTransactionError('A maximum of ' + DEFAULT_M + ' owners can be set for a multisig wallet');\n    }\n    if (!isValidPublicKey(publicKey)) {\n      throw new BuildTransactionError('Invalid public key: ' + publicKey);\n    }\n    if (this._walletOwnerPublicKeys.includes(publicKey)) {\n      throw new BuildTransactionError('Repeated owner public key: ' + publicKey);\n    }\n    this._walletOwnerPublicKeys.push(publicKey);\n  }\n\n  /**\n   * Set an initial delegate to initialize this wallet to. This is different than the delegation to\n   * set while doing a separate delegation transaction.\n   *\n   * @param {string} delegate The address to delegate the funds to\n   */\n  initialDelegate(delegate: string): void {\n    if (this._type !== TransactionType.WalletInitialization) {\n      throw new BuildTransactionError('Initial delegation can only be set for wallet initialization transactions');\n    }\n    this.validateAddress({ address: delegate });\n    this._initialDelegate = delegate;\n  }\n\n  /**\n   * Build an origination operation for a generic multisig contract.\n   *\n   * @returns {Operation} A Tezos origination operation\n   */\n  private buildWalletInitializationOperations(): OriginationOp {\n    const originationOp = genericMultisigOriginationOperation(\n      this._counter.toString(),\n      this._sourceAddress,\n      this._fee.fee,\n      this._fee.gasLimit || DEFAULT_GAS_LIMIT.ORIGINATION.toString(),\n      this._fee.storageLimit || DEFAULT_STORAGE_LIMIT.ORIGINATION.toString(),\n      this._initialBalance || '0',\n      this._walletOwnerPublicKeys,\n      this._initialDelegate\n    );\n    this._counter = this._counter.plus(1);\n    return originationOp;\n  }\n  // endregion\n\n  // region Send builder methods\n  /**\n   * Initialize a new TransferBuilder to for a singlesig or multisig transaction.\n   *\n   * @param {string} amount Amount in mutez to be transferred\n   * @returns {TransferBuilder} A transfer builder\n   */\n  transfer(amount: string): TransferBuilder {\n    if (this._type !== TransactionType.Send && this._type !== TransactionType.SingleSigSend) {\n      throw new BuildTransactionError('Transfers can only be set for send transactions');\n    }\n    let transferBuilder = new TransferBuilder();\n    // If source was set, use it as default for\n    if (this._sourceAddress) {\n      transferBuilder = transferBuilder.from(this._sourceAddress);\n    }\n    if (this._fee) {\n      transferBuilder = transferBuilder.fee(this._fee.fee);\n      transferBuilder = this._fee.gasLimit ? transferBuilder.gasLimit(this._fee.gasLimit) : transferBuilder;\n      transferBuilder = this._fee.storageLimit ? transferBuilder.storageLimit(this._fee.storageLimit) : transferBuilder;\n    }\n    this._transfers.push(transferBuilder);\n    return transferBuilder.amount(amount);\n  }\n\n  /**\n   * Calculate the signatures for the multisig transaction.\n   *\n   * @param {string} packedData The string in hexadecimal to sign\n   * @returns {Promise<string[]>} List of signatures for packedData\n   */\n  private async getSignatures(packedData: string): Promise<IndexedSignature[]> {\n    const signatures: IndexedSignature[] = [];\n    // Generate the multisig contract signatures\n    for (let i = 0; i < this._multisigSignerKeyPairs.length; i++) {\n      const signature = await sign(this._multisigSignerKeyPairs[i].key, packedData, new Uint8Array(0));\n      const index = this._multisigSignerKeyPairs[i].index;\n      signatures.push(index ? { signature: signature.sig, index } : { signature: signature.sig });\n    }\n    return signatures;\n  }\n\n  /**\n   * Override the data to sign for a specific transfer. Used for offline signing to pass the\n   * respective dataToSign for transfer at a particular index.\n   *\n   * @param {DataToSignOverride} data - data to override\n   */\n  overrideDataToSign(data: DataToSignOverride): void {\n    if (!data.index) {\n      data.index = this._dataToSignOverride.length;\n    }\n    this._dataToSignOverride.push(data);\n  }\n\n  /**\n   * Build a transaction operation for a generic multisig contract.\n   *\n   * @returns {Promise<TransactionOp[]>} A Tezos transaction operation\n   */\n  private async buildSendTransactionContent(): Promise<TransactionOp[]> {\n    const contents: TransactionOp[] = [];\n    for (let i = 0; i < this._transfers.length; i++) {\n      const transfer = this._transfers[i].build();\n      let transactionOp;\n      if (isValidOriginatedAddress(transfer.from)) {\n        // Offline transactions may not have the data to sign\n        const signatures = transfer.dataToSign ? await this.getSignatures(transfer.dataToSign) : [];\n        transactionOp = multisigTransactionOperation(\n          this._counter.toString(),\n          this._sourceAddress,\n          transfer.amount,\n          transfer.from,\n          transfer.counter || '0',\n          transfer.to,\n          signatures,\n          transfer.fee.fee,\n          transfer.fee.gasLimit,\n          transfer.fee.storageLimit\n        );\n      } else {\n        transactionOp = singlesigTransactionOperation(\n          this._counter.toString(),\n          this._sourceAddress,\n          transfer.amount,\n          transfer.to,\n          transfer.fee.fee,\n          transfer.fee.gasLimit,\n          transfer.fee.storageLimit\n        );\n      }\n      contents.push(transactionOp);\n      this._counter = this._counter.plus(1);\n    }\n    return contents;\n  }\n  // endregion\n\n  // region ForwarderAddressDeployment\n  /**\n   * Build a transaction operation for a forwarder contract\n   *\n   * @returns {OriginationOp} a Tezos transaction operation\n   */\n  private buildForwarderDeploymentContent(): OriginationOp {\n    const operation = forwarderOriginationOperation(\n      this._forwarderDestination,\n      this._counter.toString(),\n      this._sourceAddress,\n      this._fee.fee,\n      this._fee.gasLimit || DEFAULT_GAS_LIMIT.ORIGINATION.toString(),\n      this._fee.storageLimit || DEFAULT_STORAGE_LIMIT.ORIGINATION.toString(),\n      this._initialBalance || '0'\n    );\n    this._counter = this._counter.plus(1);\n    return operation;\n  }\n  // endregion\n\n  // region Validators\n  /** @inheritdoc */\n  validateValue(value: BigNumber): void {\n    if (value.isLessThan(0)) {\n      throw new BuildTransactionError('Value cannot be below less than zero');\n    }\n    // TODO: validate the amount is not bigger than the max amount in Tezos\n  }\n\n  /** @inheritdoc */\n  validateAddress(address: Address): void {\n    if (!isValidAddress(address.address)) {\n      throw new BuildTransactionError('Invalid address ' + address.address);\n    }\n  }\n\n  /** @inheritdoc */\n  validateKey(key: BaseKey): void {\n    const keyPair = new KeyPair({ prv: key.key });\n    if (!keyPair.getKeys().prv) {\n      throw new BuildTransactionError('Invalid key');\n    }\n  }\n\n  /** @inheritdoc */\n  validateRawTransaction(rawTransaction: any): void {\n    // TODO: validate the transaction is either a JSON or a hex\n  }\n\n  /** @inheritdoc */\n  validateTransaction(transaction: Transaction): void {\n    // TODO: validate all required fields are present in the builder before buildImplementation\n    switch (this._type) {\n      case TransactionType.AccountUpdate:\n        break;\n      case TransactionType.WalletInitialization:\n        break;\n      case TransactionType.Send:\n        break;\n      case TransactionType.AddressInitialization:\n        break;\n      case TransactionType.SingleSigSend:\n        break;\n      default:\n        throw new BuildTransactionError('Transaction type not supported');\n    }\n  }\n  // endregion\n\n  /** @inheritdoc */\n  displayName(): string {\n    return this._coinConfig.fullName;\n  }\n\n  /** @inheritdoc */\n  protected get transaction(): Transaction {\n    return this._transaction;\n  }\n\n  /** @inheritdoc */\n  protected set transaction(transaction: Transaction) {\n    this._transaction = transaction;\n  }\n}\n"]}

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


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