PHP WebShell

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

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

"use strict";
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
    if (k2 === undefined) k2 = k;
    var desc = Object.getOwnPropertyDescriptor(m, k);
    if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
      desc = { enumerable: true, get: function() { return m[k]; } };
    }
    Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
    if (k2 === undefined) k2 = k;
    o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
    Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
    o["default"] = v;
});
var __importStar = (this && this.__importStar) || (function () {
    var ownKeys = function(o) {
        ownKeys = Object.getOwnPropertyNames || function (o) {
            var ar = [];
            for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
            return ar;
        };
        return ownKeys(o);
    };
    return function (mod) {
        if (mod && mod.__esModule) return mod;
        var result = {};
        if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
        __setModuleDefault(result, mod);
        return result;
    };
})();
Object.defineProperty(exports, "__esModule", { value: true });
exports.TransactionBuilder = void 0;
const sdk_core_1 = require("@bitgo/sdk-core");
const transaction_1 = require("./transaction");
const keyPair_1 = require("./keyPair");
const utils_1 = __importStar(require("./utils"));
const CardanoWasm = __importStar(require("@emurgo/cardano-serialization-lib-nodejs"));
const cardano_serialization_lib_nodejs_1 = require("@emurgo/cardano-serialization-lib-nodejs");
class TransactionBuilder extends sdk_core_1.BaseTransactionBuilder {
    constructor(_coinConfig) {
        super(_coinConfig);
        this._signers = [];
        this._transactionInputs = [];
        this._transactionOutputs = [];
        this._initSignatures = [];
        this._signatures = [];
        this._ttl = 0;
        this._certs = [];
        this._withdrawals = [];
        this._multiAssets = [];
        this.transaction = new transaction_1.Transaction(_coinConfig);
        this._fee = cardano_serialization_lib_nodejs_1.BigNum.zero();
    }
    input(i) {
        this._transactionInputs.push(i);
        return this;
    }
    output(o) {
        this._transactionOutputs.push(o);
        return this;
    }
    assets(a) {
        this._multiAssets.push(a);
        return this;
    }
    ttl(t) {
        this._ttl = t;
        return this;
    }
    changeAddress(addr, totalInputBalance) {
        this._changeAddress = addr;
        this._senderBalance = totalInputBalance;
        return this;
    }
    fee(fee) {
        this._fee = cardano_serialization_lib_nodejs_1.BigNum.from_str(fee);
        return this;
    }
    /**
     * Initialize the transaction builder fields using the decoded transaction data
     *
     * @param {Transaction} tx the transaction data
     */
    initBuilder(tx) {
        this._transaction = tx;
        const txnBody = tx.transaction.body();
        for (let i = 0; i < txnBody.inputs().len(); i++) {
            const input = txnBody.inputs().get(i);
            this.input({
                transaction_id: Buffer.from(input.transaction_id().to_bytes()).toString('hex'),
                transaction_index: input.index(),
            });
        }
        for (let i = 0; i < txnBody.outputs().len(); i++) {
            const output = txnBody.outputs().get(i);
            this.output({
                address: output.address().to_bech32(),
                amount: output.amount().coin().to_str(),
                multiAssets: output.amount().multiasset() || undefined,
            });
        }
        if (txnBody.certs() !== undefined) {
            const certs = txnBody.certs();
            for (let i = 0; i < certs.len(); i++) {
                this._certs.push(certs.get(i));
            }
        }
        if (txnBody.withdrawals() !== undefined) {
            const withdrawals = txnBody.withdrawals();
            const keys = withdrawals.keys();
            for (let i = 0; i < keys.len(); i++) {
                const rewardAddress = keys.get(i);
                const reward = withdrawals.get(rewardAddress);
                this._withdrawals.push({ stakeAddress: rewardAddress.to_address().to_bech32(), value: reward.to_str() });
            }
        }
        this._ttl = tx.transaction.body().ttl();
        this._fee = tx.transaction.body().fee();
        if (tx.transaction.witness_set().vkeys()) {
            const vkeys = tx.transaction.witness_set().vkeys();
            for (let i = 0; i < vkeys.len(); i++) {
                const vkey = vkeys.get(i);
                this._initSignatures.push({
                    publicKey: { pub: vkey.vkey().public_key().to_hex() },
                    signature: Buffer.from(vkey.signature().to_hex(), 'hex'),
                });
            }
        }
    }
    /** @inheritdoc */
    fromImplementation(rawTransaction) {
        this.validateRawTransaction(rawTransaction);
        this.buildImplementation();
        return this.transaction;
    }
    /** @inheritdoc */
    async buildImplementation() {
        const inputs = CardanoWasm.TransactionInputs.new();
        this._transactionInputs.forEach((input) => {
            inputs.add(CardanoWasm.TransactionInput.new(CardanoWasm.TransactionHash.from_bytes(Buffer.from(input.transaction_id, 'hex')), input.transaction_index));
        });
        let outputs = CardanoWasm.TransactionOutputs.new();
        let totalAmountToSend = CardanoWasm.BigNum.zero();
        this._transactionOutputs.forEach((output) => {
            const amount = CardanoWasm.BigNum.from_str(output.amount);
            outputs.add(CardanoWasm.TransactionOutput.new(utils_1.default.getWalletAddress(output.address), CardanoWasm.Value.new(amount)));
            totalAmountToSend = totalAmountToSend.checked_add(amount);
        });
        if (this._fee.is_zero()) {
            // estimate fee
            // add extra output for the change
            if (this._changeAddress && this._senderBalance) {
                const changeAddress = utils_1.default.getWalletAddress(this._changeAddress);
                const utxoBalance = CardanoWasm.BigNum.from_str(this._senderBalance);
                const adjustment = cardano_serialization_lib_nodejs_1.BigNum.from_str('2000000');
                let change = utxoBalance.checked_sub(this._fee).checked_sub(totalAmountToSend);
                if (this._type === sdk_core_1.TransactionType.StakingActivate) {
                    change = change.checked_sub(adjustment);
                }
                else if (this._type === sdk_core_1.TransactionType.StakingDeactivate) {
                    change = change.checked_add(adjustment);
                }
                else if (this._type === sdk_core_1.TransactionType.StakingWithdraw || this._type === sdk_core_1.TransactionType.StakingClaim) {
                    this._withdrawals.forEach((withdrawal) => {
                        change = change.checked_add(CardanoWasm.BigNum.from_str(withdrawal.value));
                    });
                }
                // If totalAmountToSend is 0, its consolidation
                if (totalAmountToSend.to_str() == '0') {
                    // support for multi-asset consolidation
                    if (this._multiAssets !== undefined) {
                        const totalNumberOfAssets = CardanoWasm.BigNum.from_str(this._multiAssets.length.toString());
                        const minAmountNeededForOneAssetOutput = CardanoWasm.BigNum.from_str(utils_1.MIN_ADA_FOR_ONE_ASSET);
                        const minAmountNeededForTotalAssetOutputs = minAmountNeededForOneAssetOutput.checked_mul(totalNumberOfAssets);
                        if (!change.less_than(minAmountNeededForTotalAssetOutputs)) {
                            this._multiAssets.forEach((asset) => {
                                let txOutputBuilder = CardanoWasm.TransactionOutputBuilder.new();
                                // changeAddress is the root address, which is where we want the tokens assets to be sent to
                                const toAddress = utils_1.default.getWalletAddress(this._changeAddress);
                                txOutputBuilder = txOutputBuilder.with_address(toAddress);
                                let txOutputAmountBuilder = txOutputBuilder.next();
                                const assetName = CardanoWasm.AssetName.new(Buffer.from(asset.asset_name, 'hex'));
                                const policyId = CardanoWasm.ScriptHash.from_bytes(Buffer.from(asset.policy_id, 'hex'));
                                const multiAsset = CardanoWasm.MultiAsset.new();
                                const assets = CardanoWasm.Assets.new();
                                assets.insert(assetName, CardanoWasm.BigNum.from_str(asset.quantity));
                                multiAsset.insert(policyId, assets);
                                txOutputAmountBuilder = txOutputAmountBuilder.with_coin_and_asset(minAmountNeededForOneAssetOutput, multiAsset);
                                const txOutput = txOutputAmountBuilder.build();
                                outputs.add(txOutput);
                            });
                            // finally send the remaining ADA in its own output
                            const remainingOutputAmount = change.checked_sub(minAmountNeededForTotalAssetOutputs);
                            const changeOutput = CardanoWasm.TransactionOutput.new(changeAddress, CardanoWasm.Value.new(remainingOutputAmount));
                            outputs.add(changeOutput);
                        }
                    }
                    else {
                        // If there are no tokens to consolidate, you only have 1 output which is ADA alone
                        const changeOutput = CardanoWasm.TransactionOutput.new(changeAddress, CardanoWasm.Value.new(change));
                        outputs.add(changeOutput);
                    }
                }
                else {
                    // If this isn't a consolidate request, whatever change that needs to be sent back to the rootaddress is added as a separate output here
                    const changeOutput = CardanoWasm.TransactionOutput.new(changeAddress, CardanoWasm.Value.new(change));
                    outputs.add(changeOutput);
                }
            }
            const txBody = CardanoWasm.TransactionBody.new_tx_body(inputs, outputs, this._fee);
            txBody.set_ttl(CardanoWasm.BigNum.from_str(this._ttl.toString()));
            const txHash = CardanoWasm.hash_transaction(txBody);
            // we add witnesses once so that we can get the appropriate amount of signers for calculating the fee
            const witnessSet = CardanoWasm.TransactionWitnessSet.new();
            const vkeyWitnesses = CardanoWasm.Vkeywitnesses.new();
            this._signers.forEach((keyPair) => {
                const prv = keyPair.getKeys().prv;
                const vkeyWitness = CardanoWasm.make_vkey_witness(txHash, CardanoWasm.PrivateKey.from_normal_bytes(Buffer.from(prv, 'hex')));
                vkeyWitnesses.add(vkeyWitness);
            });
            this.getAllSignatures().forEach((signature) => {
                const vkey = CardanoWasm.Vkey.new(CardanoWasm.PublicKey.from_bytes(Buffer.from(signature.publicKey.pub, 'hex')));
                const ed255Sig = CardanoWasm.Ed25519Signature.from_bytes(signature.signature);
                vkeyWitnesses.add(CardanoWasm.Vkeywitness.new(vkey, ed255Sig));
            });
            if (vkeyWitnesses.len() === 0) {
                const prv = CardanoWasm.PrivateKey.generate_ed25519();
                const vkeyWitness = CardanoWasm.make_vkey_witness(txHash, prv);
                vkeyWitnesses.add(vkeyWitness);
                if (this._type !== sdk_core_1.TransactionType.Send) {
                    vkeyWitnesses.add(vkeyWitness);
                }
            }
            witnessSet.set_vkeys(vkeyWitnesses);
            // add in withdrawal if this is a withdrawal tx
            if (this._withdrawals.length > 0) {
                const withdrawals = CardanoWasm.Withdrawals.new();
                this._withdrawals.forEach((withdrawal) => {
                    const rewardAddress = CardanoWasm.RewardAddress.from_address(CardanoWasm.Address.from_bech32(withdrawal.stakeAddress));
                    withdrawals.insert(rewardAddress, CardanoWasm.BigNum.from_str(withdrawal.value));
                });
                txBody.set_withdrawals(withdrawals);
            }
            // add in certificates to get mock size
            const draftCerts = CardanoWasm.Certificates.new();
            for (const cert of this._certs) {
                draftCerts.add(cert);
            }
            txBody.set_certs(draftCerts);
            const txDraft = CardanoWasm.Transaction.new(txBody, witnessSet);
            const linearFee = CardanoWasm.LinearFee.new(CardanoWasm.BigNum.from_str('44'), CardanoWasm.BigNum.from_str('155381'));
            // calculate the fee based off our dummy transaction
            const fee = CardanoWasm.min_fee(txDraft, linearFee).checked_add(cardano_serialization_lib_nodejs_1.BigNum.from_str('440'));
            this._fee = fee;
        }
        this._transaction.fee(this._fee.to_str());
        // now calculate the change based off of <utxoBalance> - <fee> - <amountToSend>
        // reset the outputs collection because now our last output has changed
        outputs = CardanoWasm.TransactionOutputs.new();
        this._transactionOutputs.forEach((output) => {
            if (output.multiAssets) {
                const policyId = output.multiAssets.keys().get(0);
                const assets = output.multiAssets.get(policyId);
                const assetName = assets.keys().get(0);
                const quantity = assets.get(assetName);
                let txOutputBuilder = CardanoWasm.TransactionOutputBuilder.new();
                const outputAmount = CardanoWasm.BigNum.from_str(output.amount);
                const toAddress = utils_1.default.getWalletAddress(output.address);
                txOutputBuilder = txOutputBuilder.with_address(toAddress);
                let txOutputAmountBuilder = txOutputBuilder.next();
                const multiAsset = CardanoWasm.MultiAsset.new();
                const asset = CardanoWasm.Assets.new();
                asset.insert(assetName, quantity);
                multiAsset.insert(policyId, asset);
                txOutputAmountBuilder = txOutputAmountBuilder.with_coin_and_asset(outputAmount, multiAsset);
                const txOutput = txOutputAmountBuilder.build();
                outputs.add(txOutput);
            }
            else {
                outputs.add(CardanoWasm.TransactionOutput.new(utils_1.default.getWalletAddress(output.address), CardanoWasm.Value.new(CardanoWasm.BigNum.from_str(output.amount))));
            }
        });
        if (this._changeAddress && this._senderBalance) {
            const changeAddress = utils_1.default.getWalletAddress(this._changeAddress);
            const utxoBalance = CardanoWasm.BigNum.from_str(this._senderBalance);
            const adjustment = cardano_serialization_lib_nodejs_1.BigNum.from_str('2000000');
            let change = utxoBalance.checked_sub(this._fee).checked_sub(totalAmountToSend);
            if (this._type === sdk_core_1.TransactionType.StakingActivate) {
                change = change.checked_sub(adjustment);
            }
            else if (this._type === sdk_core_1.TransactionType.StakingDeactivate) {
                change = change.checked_add(adjustment);
            }
            else if (this._type === sdk_core_1.TransactionType.StakingWithdraw || this._type === sdk_core_1.TransactionType.StakingClaim) {
                this._withdrawals.forEach((withdrawal) => {
                    change = change.checked_add(CardanoWasm.BigNum.from_str(withdrawal.value));
                });
            }
            // If totalAmountToSend is 0, its consolidation
            if (totalAmountToSend.to_str() == '0') {
                // support for multi-asset consolidation
                if (this._multiAssets !== undefined) {
                    const totalNumberOfAssets = CardanoWasm.BigNum.from_str(this._multiAssets.length.toString());
                    const minAmountNeededForOneAssetOutput = CardanoWasm.BigNum.from_str('1500000');
                    const minAmountNeededForTotalAssetOutputs = minAmountNeededForOneAssetOutput.checked_mul(totalNumberOfAssets);
                    if (!change.less_than(minAmountNeededForTotalAssetOutputs)) {
                        this._multiAssets.forEach((asset) => {
                            let txOutputBuilder = CardanoWasm.TransactionOutputBuilder.new();
                            // changeAddress is the root address, which is where we want the tokens assets to be sent to
                            const toAddress = utils_1.default.getWalletAddress(this._changeAddress);
                            txOutputBuilder = txOutputBuilder.with_address(toAddress);
                            let txOutputAmountBuilder = txOutputBuilder.next();
                            const assetName = CardanoWasm.AssetName.new(Buffer.from(asset.asset_name, 'hex'));
                            const policyId = CardanoWasm.ScriptHash.from_bytes(Buffer.from(asset.policy_id, 'hex'));
                            const multiAsset = CardanoWasm.MultiAsset.new();
                            const assets = CardanoWasm.Assets.new();
                            assets.insert(assetName, CardanoWasm.BigNum.from_str(asset.quantity));
                            multiAsset.insert(policyId, assets);
                            txOutputAmountBuilder = txOutputAmountBuilder.with_coin_and_asset(minAmountNeededForOneAssetOutput, multiAsset);
                            const txOutput = txOutputAmountBuilder.build();
                            outputs.add(txOutput);
                        });
                        // finally send the remaining ADA in its own output
                        const remainingOutputAmount = change.checked_sub(minAmountNeededForTotalAssetOutputs);
                        const changeOutput = CardanoWasm.TransactionOutput.new(changeAddress, CardanoWasm.Value.new(remainingOutputAmount));
                        outputs.add(changeOutput);
                    }
                    else {
                        throw new sdk_core_1.BuildTransactionError('Insufficient funds: need a minimum of 1.5 ADA per output to construct token consolidation');
                    }
                }
                else {
                    // If there are no tokens to consolidate, you only have 1 output which is ADA alone
                    const changeOutput = CardanoWasm.TransactionOutput.new(changeAddress, CardanoWasm.Value.new(change));
                    outputs.add(changeOutput);
                }
            }
            else {
                // If this isn't a consolidate request, whatever change that needs to be sent back to the rootaddress is added as a separate output here
                const changeOutput = CardanoWasm.TransactionOutput.new(changeAddress, CardanoWasm.Value.new(change));
                outputs.add(changeOutput);
            }
        }
        const txRaw = CardanoWasm.TransactionBody.new_tx_body(inputs, outputs, this._fee);
        const certs = CardanoWasm.Certificates.new();
        for (const cert of this._certs) {
            certs.add(cert);
        }
        txRaw.set_certs(certs);
        // add in withdrawal if this is a withdrawal tx
        if (this._withdrawals.length > 0) {
            const withdrawals = CardanoWasm.Withdrawals.new();
            this._withdrawals.forEach((withdrawal) => {
                const rewardAddress = CardanoWasm.RewardAddress.from_address(CardanoWasm.Address.from_bech32(withdrawal.stakeAddress));
                withdrawals.insert(rewardAddress, CardanoWasm.BigNum.from_str(withdrawal.value));
            });
            txRaw.set_withdrawals(withdrawals);
        }
        txRaw.set_ttl(CardanoWasm.BigNum.from_str(this._ttl.toString()));
        const txRawHash = CardanoWasm.hash_transaction(txRaw);
        // now add the witnesses again this time for real. We need to do this again
        // because now that we've added our real fee and change output, we have a difference transaction hash
        const witnessSet = CardanoWasm.TransactionWitnessSet.new();
        const vkeyWitnesses = CardanoWasm.Vkeywitnesses.new();
        this._signers.forEach((keyPair) => {
            const prv = keyPair.getKeys().prv;
            const vkeyWitness = CardanoWasm.make_vkey_witness(txRawHash, CardanoWasm.PrivateKey.from_normal_bytes(Buffer.from(prv, 'hex')));
            vkeyWitnesses.add(vkeyWitness);
        });
        // Clear the cosmetic signature array in native txn wrapper to prevent duplicate when builder is inited from a partially witnessed txn
        this._transaction.signature.length = 0;
        this.getAllSignatures().forEach((signature) => {
            const vkey = CardanoWasm.Vkey.new(CardanoWasm.PublicKey.from_bytes(Buffer.from(signature.publicKey.pub, 'hex')));
            const ed255Sig = CardanoWasm.Ed25519Signature.from_bytes(signature.signature);
            vkeyWitnesses.add(CardanoWasm.Vkeywitness.new(vkey, ed255Sig));
            this._transaction.signature.push(signature.signature.toString('hex'));
        });
        witnessSet.set_vkeys(vkeyWitnesses);
        this._transaction.transaction = CardanoWasm.Transaction.new(txRaw, witnessSet);
        return this.transaction;
    }
    /** @inheritdoc */
    signImplementation(key) {
        this._signers.push(new keyPair_1.KeyPair({ prv: key.key }));
        return this._transaction;
    }
    /** @inheritdoc */
    get transaction() {
        return this._transaction;
    }
    /** @inheritdoc */
    set transaction(transaction) {
        this._transaction = transaction;
    }
    /** @inheritdoc */
    validateAddress(address, addressFormat) {
        if (!utils_1.default.isValidAddress(address.address)) {
            throw new sdk_core_1.UtilsError('invalid address ' + address.address);
        }
    }
    /** @inheritdoc */
    validateKey(key) {
        try {
            new keyPair_1.KeyPair({ prv: key.key });
        }
        catch {
            throw new sdk_core_1.BuildTransactionError(`Key validation failed`);
        }
    }
    /** @inheritdoc */
    validateRawTransaction(rawTransaction) {
        try {
            CardanoWasm.Transaction.from_bytes(rawTransaction);
        }
        catch {
            throw new sdk_core_1.BuildTransactionError('invalid raw transaction');
        }
    }
    /** @inheritdoc */
    validateTransaction(transaction) {
        if (!transaction.transaction) {
            return;
        }
    }
    /** @inheritdoc */
    validateValue(value) {
        if (value.isLessThan(0)) {
            throw new sdk_core_1.BuildTransactionError('Value cannot be less than zero');
        }
    }
    // endregion
    /** @inheritDoc */
    addSignature(publicKey, signature) {
        this._signatures.push({ publicKey, signature });
    }
    getAllSignatures() {
        return this._initSignatures.concat(this._signatures);
    }
}
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 BigNumber from 'bignumber.js';\nimport { BaseCoin as CoinConfig } from '@bitgo/statics';\nimport {\n  BaseAddress,\n  BaseKey,\n  BaseTransactionBuilder,\n  BuildTransactionError,\n  PublicKey as BasePublicKey,\n  Signature,\n  TransactionType,\n  UtilsError,\n} from '@bitgo/sdk-core';\nimport { Asset, Transaction, TransactionInput, TransactionOutput, Withdrawal } from './transaction';\nimport { KeyPair } from './keyPair';\nimport util, { MIN_ADA_FOR_ONE_ASSET } from './utils';\nimport * as CardanoWasm from '@emurgo/cardano-serialization-lib-nodejs';\nimport { BigNum } from '@emurgo/cardano-serialization-lib-nodejs';\n\nexport abstract class TransactionBuilder extends BaseTransactionBuilder {\n  protected _transaction!: Transaction;\n  protected _signers: KeyPair[] = [];\n  protected _transactionInputs: TransactionInput[] = [];\n  protected _transactionOutputs: TransactionOutput[] = [];\n  protected _initSignatures: Signature[] = [];\n  protected _signatures: Signature[] = [];\n  protected _changeAddress: string;\n  protected _senderBalance: string;\n  protected _ttl = 0;\n  protected _certs: CardanoWasm.Certificate[] = [];\n  protected _withdrawals: Withdrawal[] = [];\n  protected _type: TransactionType;\n  protected _multiAssets: Asset[] = [];\n  private _fee: BigNum;\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    this.transaction = new Transaction(_coinConfig);\n    this._fee = BigNum.zero();\n  }\n\n  input(i: TransactionInput): this {\n    this._transactionInputs.push(i);\n    return this;\n  }\n\n  output(o: TransactionOutput): this {\n    this._transactionOutputs.push(o);\n    return this;\n  }\n\n  assets(a: Asset): this {\n    this._multiAssets.push(a);\n    return this;\n  }\n\n  ttl(t: number): this {\n    this._ttl = t;\n    return this;\n  }\n\n  changeAddress(addr: string, totalInputBalance: string): this {\n    this._changeAddress = addr;\n    this._senderBalance = totalInputBalance;\n    return this;\n  }\n\n  fee(fee: string): this {\n    this._fee = BigNum.from_str(fee);\n    return this;\n  }\n\n  /**\n   * Initialize the transaction builder fields using the decoded transaction data\n   *\n   * @param {Transaction} tx the transaction data\n   */\n  initBuilder(tx: Transaction): void {\n    this._transaction = tx;\n    const txnBody = tx.transaction.body();\n    for (let i = 0; i < txnBody.inputs().len(); i++) {\n      const input = txnBody.inputs().get(i);\n      this.input({\n        transaction_id: Buffer.from(input.transaction_id().to_bytes()).toString('hex'),\n        transaction_index: input.index(),\n      });\n    }\n    for (let i = 0; i < txnBody.outputs().len(); i++) {\n      const output = txnBody.outputs().get(i);\n      this.output({\n        address: output.address().to_bech32(),\n        amount: output.amount().coin().to_str(),\n        multiAssets: output.amount().multiasset() || undefined,\n      });\n    }\n\n    if (txnBody.certs() !== undefined) {\n      const certs = txnBody.certs() as CardanoWasm.Certificates;\n      for (let i = 0; i < certs.len(); i++) {\n        this._certs.push(certs.get(i));\n      }\n    }\n\n    if (txnBody.withdrawals() !== undefined) {\n      const withdrawals = txnBody.withdrawals() as CardanoWasm.Withdrawals;\n      const keys = withdrawals.keys();\n      for (let i = 0; i < keys.len(); i++) {\n        const rewardAddress = keys.get(i) as CardanoWasm.RewardAddress;\n        const reward = withdrawals.get(rewardAddress) as CardanoWasm.BigNum;\n        this._withdrawals.push({ stakeAddress: rewardAddress.to_address().to_bech32(), value: reward.to_str() });\n      }\n    }\n\n    this._ttl = tx.transaction.body().ttl() as number;\n    this._fee = tx.transaction.body().fee();\n\n    if (tx.transaction.witness_set().vkeys()) {\n      const vkeys = tx.transaction.witness_set().vkeys()! as CardanoWasm.Vkeywitnesses;\n      for (let i = 0; i < vkeys.len(); i++) {\n        const vkey = vkeys.get(i);\n        this._initSignatures.push({\n          publicKey: { pub: vkey.vkey().public_key().to_hex() },\n          signature: Buffer.from(vkey.signature().to_hex(), 'hex'),\n        });\n      }\n    }\n  }\n\n  /** @inheritdoc */\n  protected fromImplementation(rawTransaction: string): Transaction {\n    this.validateRawTransaction(rawTransaction);\n    this.buildImplementation();\n    return this.transaction;\n  }\n\n  /** @inheritdoc */\n  protected async buildImplementation(): Promise<Transaction> {\n    const inputs = CardanoWasm.TransactionInputs.new();\n    this._transactionInputs.forEach((input) => {\n      inputs.add(\n        CardanoWasm.TransactionInput.new(\n          CardanoWasm.TransactionHash.from_bytes(Buffer.from(input.transaction_id, 'hex')),\n          input.transaction_index\n        )\n      );\n    });\n    let outputs = CardanoWasm.TransactionOutputs.new();\n    let totalAmountToSend = CardanoWasm.BigNum.zero();\n    this._transactionOutputs.forEach((output) => {\n      const amount = CardanoWasm.BigNum.from_str(output.amount);\n      outputs.add(\n        CardanoWasm.TransactionOutput.new(util.getWalletAddress(output.address), CardanoWasm.Value.new(amount))\n      );\n      totalAmountToSend = totalAmountToSend.checked_add(amount);\n    });\n\n    if (this._fee.is_zero()) {\n      // estimate fee\n      // add extra output for the change\n      if (this._changeAddress && this._senderBalance) {\n        const changeAddress = util.getWalletAddress(this._changeAddress);\n        const utxoBalance = CardanoWasm.BigNum.from_str(this._senderBalance);\n\n        const adjustment = BigNum.from_str('2000000');\n        let change = utxoBalance.checked_sub(this._fee).checked_sub(totalAmountToSend);\n        if (this._type === TransactionType.StakingActivate) {\n          change = change.checked_sub(adjustment);\n        } else if (this._type === TransactionType.StakingDeactivate) {\n          change = change.checked_add(adjustment);\n        } else if (this._type === TransactionType.StakingWithdraw || this._type === TransactionType.StakingClaim) {\n          this._withdrawals.forEach((withdrawal: Withdrawal) => {\n            change = change.checked_add(CardanoWasm.BigNum.from_str(withdrawal.value));\n          });\n        }\n\n        // If totalAmountToSend is 0, its consolidation\n        if (totalAmountToSend.to_str() == '0') {\n          // support for multi-asset consolidation\n          if (this._multiAssets !== undefined) {\n            const totalNumberOfAssets = CardanoWasm.BigNum.from_str(this._multiAssets.length.toString());\n            const minAmountNeededForOneAssetOutput = CardanoWasm.BigNum.from_str(MIN_ADA_FOR_ONE_ASSET);\n            const minAmountNeededForTotalAssetOutputs =\n              minAmountNeededForOneAssetOutput.checked_mul(totalNumberOfAssets);\n\n            if (!change.less_than(minAmountNeededForTotalAssetOutputs)) {\n              this._multiAssets.forEach((asset) => {\n                let txOutputBuilder = CardanoWasm.TransactionOutputBuilder.new();\n                // changeAddress is the root address, which is where we want the tokens assets to be sent to\n                const toAddress = util.getWalletAddress(this._changeAddress);\n                txOutputBuilder = txOutputBuilder.with_address(toAddress);\n                let txOutputAmountBuilder = txOutputBuilder.next();\n                const assetName = CardanoWasm.AssetName.new(Buffer.from(asset.asset_name, 'hex'));\n                const policyId = CardanoWasm.ScriptHash.from_bytes(Buffer.from(asset.policy_id, 'hex'));\n                const multiAsset = CardanoWasm.MultiAsset.new();\n                const assets = CardanoWasm.Assets.new();\n                assets.insert(assetName, CardanoWasm.BigNum.from_str(asset.quantity));\n                multiAsset.insert(policyId, assets);\n\n                txOutputAmountBuilder = txOutputAmountBuilder.with_coin_and_asset(\n                  minAmountNeededForOneAssetOutput,\n                  multiAsset\n                );\n\n                const txOutput = txOutputAmountBuilder.build();\n                outputs.add(txOutput);\n              });\n\n              // finally send the remaining ADA in its own output\n              const remainingOutputAmount = change.checked_sub(minAmountNeededForTotalAssetOutputs);\n              const changeOutput = CardanoWasm.TransactionOutput.new(\n                changeAddress,\n                CardanoWasm.Value.new(remainingOutputAmount)\n              );\n              outputs.add(changeOutput);\n            }\n          } else {\n            // If there are no tokens to consolidate, you only have 1 output which is ADA alone\n            const changeOutput = CardanoWasm.TransactionOutput.new(changeAddress, CardanoWasm.Value.new(change));\n            outputs.add(changeOutput);\n          }\n        } else {\n          // If this isn't a consolidate request, whatever change that needs to be sent back to the rootaddress is added as a separate output here\n          const changeOutput = CardanoWasm.TransactionOutput.new(changeAddress, CardanoWasm.Value.new(change));\n          outputs.add(changeOutput);\n        }\n      }\n\n      const txBody = CardanoWasm.TransactionBody.new_tx_body(inputs, outputs, this._fee);\n      txBody.set_ttl(CardanoWasm.BigNum.from_str(this._ttl.toString()));\n      const txHash = CardanoWasm.hash_transaction(txBody);\n\n      // we add witnesses once so that we can get the appropriate amount of signers for calculating the fee\n      const witnessSet = CardanoWasm.TransactionWitnessSet.new();\n      const vkeyWitnesses = CardanoWasm.Vkeywitnesses.new();\n      this._signers.forEach((keyPair) => {\n        const prv = keyPair.getKeys().prv as string;\n        const vkeyWitness = CardanoWasm.make_vkey_witness(\n          txHash,\n          CardanoWasm.PrivateKey.from_normal_bytes(Buffer.from(prv, 'hex'))\n        );\n        vkeyWitnesses.add(vkeyWitness);\n      });\n      this.getAllSignatures().forEach((signature) => {\n        const vkey = CardanoWasm.Vkey.new(\n          CardanoWasm.PublicKey.from_bytes(Buffer.from(signature.publicKey.pub, 'hex'))\n        );\n        const ed255Sig = CardanoWasm.Ed25519Signature.from_bytes(signature.signature);\n        vkeyWitnesses.add(CardanoWasm.Vkeywitness.new(vkey, ed255Sig));\n      });\n      if (vkeyWitnesses.len() === 0) {\n        const prv = CardanoWasm.PrivateKey.generate_ed25519();\n        const vkeyWitness = CardanoWasm.make_vkey_witness(txHash, prv);\n        vkeyWitnesses.add(vkeyWitness);\n        if (this._type !== TransactionType.Send) {\n          vkeyWitnesses.add(vkeyWitness);\n        }\n      }\n      witnessSet.set_vkeys(vkeyWitnesses);\n\n      // add in withdrawal if this is a withdrawal tx\n      if (this._withdrawals.length > 0) {\n        const withdrawals = CardanoWasm.Withdrawals.new();\n        this._withdrawals.forEach((withdrawal: Withdrawal) => {\n          const rewardAddress = CardanoWasm.RewardAddress.from_address(\n            CardanoWasm.Address.from_bech32(withdrawal.stakeAddress)\n          );\n          withdrawals.insert(rewardAddress!, CardanoWasm.BigNum.from_str(withdrawal.value));\n        });\n\n        txBody.set_withdrawals(withdrawals);\n      }\n\n      // add in certificates to get mock size\n      const draftCerts = CardanoWasm.Certificates.new();\n      for (const cert of this._certs) {\n        draftCerts.add(cert);\n      }\n      txBody.set_certs(draftCerts);\n\n      const txDraft = CardanoWasm.Transaction.new(txBody, witnessSet);\n      const linearFee = CardanoWasm.LinearFee.new(\n        CardanoWasm.BigNum.from_str('44'),\n        CardanoWasm.BigNum.from_str('155381')\n      );\n\n      // calculate the fee based off our dummy transaction\n      const fee = CardanoWasm.min_fee(txDraft, linearFee).checked_add(BigNum.from_str('440'));\n      this._fee = fee;\n    }\n    this._transaction.fee(this._fee.to_str());\n    // now calculate the change based off of <utxoBalance> - <fee> - <amountToSend>\n    // reset the outputs collection because now our last output has changed\n    outputs = CardanoWasm.TransactionOutputs.new();\n    this._transactionOutputs.forEach((output) => {\n      if (output.multiAssets) {\n        const policyId = output.multiAssets.keys().get(0);\n        const assets = output.multiAssets.get(policyId);\n        const assetName = assets!.keys().get(0);\n        const quantity = assets!.get(assetName);\n        let txOutputBuilder = CardanoWasm.TransactionOutputBuilder.new();\n        const outputAmount = CardanoWasm.BigNum.from_str(output.amount);\n        const toAddress = util.getWalletAddress(output.address);\n        txOutputBuilder = txOutputBuilder.with_address(toAddress);\n        let txOutputAmountBuilder = txOutputBuilder.next();\n        const multiAsset = CardanoWasm.MultiAsset.new();\n        const asset = CardanoWasm.Assets.new();\n        asset.insert(assetName, quantity!);\n        multiAsset.insert(policyId, asset);\n        txOutputAmountBuilder = txOutputAmountBuilder.with_coin_and_asset(outputAmount, multiAsset);\n        const txOutput = txOutputAmountBuilder.build();\n        outputs.add(txOutput);\n      } else {\n        outputs.add(\n          CardanoWasm.TransactionOutput.new(\n            util.getWalletAddress(output.address),\n            CardanoWasm.Value.new(CardanoWasm.BigNum.from_str(output.amount))\n          )\n        );\n      }\n    });\n    if (this._changeAddress && this._senderBalance) {\n      const changeAddress = util.getWalletAddress(this._changeAddress);\n      const utxoBalance = CardanoWasm.BigNum.from_str(this._senderBalance);\n\n      const adjustment = BigNum.from_str('2000000');\n      let change = utxoBalance.checked_sub(this._fee).checked_sub(totalAmountToSend);\n      if (this._type === TransactionType.StakingActivate) {\n        change = change.checked_sub(adjustment);\n      } else if (this._type === TransactionType.StakingDeactivate) {\n        change = change.checked_add(adjustment);\n      } else if (this._type === TransactionType.StakingWithdraw || this._type === TransactionType.StakingClaim) {\n        this._withdrawals.forEach((withdrawal: Withdrawal) => {\n          change = change.checked_add(CardanoWasm.BigNum.from_str(withdrawal.value));\n        });\n      }\n\n      // If totalAmountToSend is 0, its consolidation\n      if (totalAmountToSend.to_str() == '0') {\n        // support for multi-asset consolidation\n        if (this._multiAssets !== undefined) {\n          const totalNumberOfAssets = CardanoWasm.BigNum.from_str(this._multiAssets.length.toString());\n          const minAmountNeededForOneAssetOutput = CardanoWasm.BigNum.from_str('1500000');\n          const minAmountNeededForTotalAssetOutputs = minAmountNeededForOneAssetOutput.checked_mul(totalNumberOfAssets);\n\n          if (!change.less_than(minAmountNeededForTotalAssetOutputs)) {\n            this._multiAssets.forEach((asset) => {\n              let txOutputBuilder = CardanoWasm.TransactionOutputBuilder.new();\n              // changeAddress is the root address, which is where we want the tokens assets to be sent to\n              const toAddress = util.getWalletAddress(this._changeAddress);\n              txOutputBuilder = txOutputBuilder.with_address(toAddress);\n              let txOutputAmountBuilder = txOutputBuilder.next();\n              const assetName = CardanoWasm.AssetName.new(Buffer.from(asset.asset_name, 'hex'));\n              const policyId = CardanoWasm.ScriptHash.from_bytes(Buffer.from(asset.policy_id, 'hex'));\n              const multiAsset = CardanoWasm.MultiAsset.new();\n              const assets = CardanoWasm.Assets.new();\n              assets.insert(assetName, CardanoWasm.BigNum.from_str(asset.quantity));\n              multiAsset.insert(policyId, assets);\n\n              txOutputAmountBuilder = txOutputAmountBuilder.with_coin_and_asset(\n                minAmountNeededForOneAssetOutput,\n                multiAsset\n              );\n\n              const txOutput = txOutputAmountBuilder.build();\n              outputs.add(txOutput);\n            });\n\n            // finally send the remaining ADA in its own output\n            const remainingOutputAmount = change.checked_sub(minAmountNeededForTotalAssetOutputs);\n            const changeOutput = CardanoWasm.TransactionOutput.new(\n              changeAddress,\n              CardanoWasm.Value.new(remainingOutputAmount)\n            );\n            outputs.add(changeOutput);\n          } else {\n            throw new BuildTransactionError(\n              'Insufficient funds: need a minimum of 1.5 ADA per output to construct token consolidation'\n            );\n          }\n        } else {\n          // If there are no tokens to consolidate, you only have 1 output which is ADA alone\n          const changeOutput = CardanoWasm.TransactionOutput.new(changeAddress, CardanoWasm.Value.new(change));\n          outputs.add(changeOutput);\n        }\n      } else {\n        // If this isn't a consolidate request, whatever change that needs to be sent back to the rootaddress is added as a separate output here\n        const changeOutput = CardanoWasm.TransactionOutput.new(changeAddress, CardanoWasm.Value.new(change));\n        outputs.add(changeOutput);\n      }\n    }\n\n    const txRaw = CardanoWasm.TransactionBody.new_tx_body(inputs, outputs, this._fee);\n\n    const certs = CardanoWasm.Certificates.new();\n    for (const cert of this._certs) {\n      certs.add(cert);\n    }\n    txRaw.set_certs(certs);\n\n    // add in withdrawal if this is a withdrawal tx\n    if (this._withdrawals.length > 0) {\n      const withdrawals = CardanoWasm.Withdrawals.new();\n      this._withdrawals.forEach((withdrawal: Withdrawal) => {\n        const rewardAddress = CardanoWasm.RewardAddress.from_address(\n          CardanoWasm.Address.from_bech32(withdrawal.stakeAddress)\n        );\n        withdrawals.insert(rewardAddress!, CardanoWasm.BigNum.from_str(withdrawal.value));\n      });\n\n      txRaw.set_withdrawals(withdrawals);\n    }\n\n    txRaw.set_ttl(CardanoWasm.BigNum.from_str(this._ttl.toString()));\n    const txRawHash = CardanoWasm.hash_transaction(txRaw);\n\n    // now add the witnesses again this time for real. We need to do this again\n    // because now that we've added our real fee and change output, we have a difference transaction hash\n    const witnessSet = CardanoWasm.TransactionWitnessSet.new();\n    const vkeyWitnesses = CardanoWasm.Vkeywitnesses.new();\n    this._signers.forEach((keyPair) => {\n      const prv = keyPair.getKeys().prv as string;\n      const vkeyWitness = CardanoWasm.make_vkey_witness(\n        txRawHash,\n        CardanoWasm.PrivateKey.from_normal_bytes(Buffer.from(prv, 'hex'))\n      );\n      vkeyWitnesses.add(vkeyWitness);\n    });\n\n    // Clear the cosmetic signature array in native txn wrapper to prevent duplicate when builder is inited from a partially witnessed txn\n    this._transaction.signature.length = 0;\n    this.getAllSignatures().forEach((signature) => {\n      const vkey = CardanoWasm.Vkey.new(CardanoWasm.PublicKey.from_bytes(Buffer.from(signature.publicKey.pub, 'hex')));\n      const ed255Sig = CardanoWasm.Ed25519Signature.from_bytes(signature.signature);\n      vkeyWitnesses.add(CardanoWasm.Vkeywitness.new(vkey, ed255Sig));\n      this._transaction.signature.push(signature.signature.toString('hex'));\n    });\n    witnessSet.set_vkeys(vkeyWitnesses);\n    this._transaction.transaction = CardanoWasm.Transaction.new(txRaw, witnessSet);\n    return this.transaction;\n  }\n\n  /** @inheritdoc */\n  protected signImplementation(key: BaseKey): Transaction {\n    this._signers.push(new KeyPair({ prv: key.key }));\n    return this._transaction;\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  /** @inheritdoc */\n  validateAddress(address: BaseAddress, addressFormat?: string): void {\n    if (!util.isValidAddress(address.address)) {\n      throw new UtilsError('invalid address ' + address.address);\n    }\n  }\n\n  /** @inheritdoc */\n  validateKey(key: BaseKey): void {\n    try {\n      new KeyPair({ prv: key.key });\n    } catch {\n      throw new BuildTransactionError(`Key validation failed`);\n    }\n  }\n\n  /** @inheritdoc */\n  validateRawTransaction(rawTransaction: any): void {\n    try {\n      CardanoWasm.Transaction.from_bytes(rawTransaction);\n    } catch {\n      throw new BuildTransactionError('invalid raw transaction');\n    }\n  }\n\n  /** @inheritdoc */\n  validateTransaction(transaction: Transaction): void {\n    if (!transaction.transaction) {\n      return;\n    }\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  // endregion\n\n  /** @inheritDoc */\n  addSignature(publicKey: BasePublicKey, signature: Buffer): void {\n    this._signatures.push({ publicKey, signature });\n  }\n\n  private getAllSignatures(): Signature[] {\n    return this._initSignatures.concat(this._signatures);\n  }\n}\n"]}

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


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