PHP WebShell
Текущая директория: /opt/BitGoJS/modules/sdk-coin-algo/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 = exports.BETANET_GENESIS_HASH = exports.BETANET_GENESIS_ID = exports.TESTNET_GENESIS_HASH = exports.TESTNET_GENESIS_ID = exports.MAINNET_GENESIS_HASH = exports.MAINNET_GENESIS_ID = void 0;
const bignumber_js_1 = __importDefault(require("bignumber.js"));
const algosdk_1 = __importDefault(require("algosdk"));
const transaction_1 = require("./transaction");
const errors_1 = require("./errors");
const keyPair_1 = require("./keyPair");
const txnSchema_1 = require("./txnSchema");
const utils_1 = __importDefault(require("./utils"));
const sdk_core_1 = require("@bitgo/sdk-core");
const _1 = require(".");
const MIN_FEE = 1000; // in microalgos
exports.MAINNET_GENESIS_ID = 'mainnet-v1.0';
exports.MAINNET_GENESIS_HASH = 'wGHE2Pwdvd7S12BL5FaOP20EGYesN73ktiC1qzkkit8=';
exports.TESTNET_GENESIS_ID = 'testnet-v1.0';
exports.TESTNET_GENESIS_HASH = 'SGO1GKSzyE7IEPItTxCByw9x8FmnrCDexi9/cOUJOiI=';
exports.BETANET_GENESIS_ID = 'betanet-v1.0';
exports.BETANET_GENESIS_HASH = 'mFgazF+2uRS1tMiL9dsj01hJGySEmPN28B/TjjvpVW0=';
class TransactionBuilder extends sdk_core_1.BaseTransactionBuilder {
constructor(coinConfig) {
super(coinConfig);
this._transaction = new transaction_1.Transaction(coinConfig);
this._keyPairs = [];
}
/**
* Sets the fee.
*
* The minimum fee is 1000 microalgos.
*
* @param {BaseFee} feeObj The amount to pay to the fee sink denoted in microalgos
* @returns {TransactionBuilder} This transaction builder.
*
* @see https://developer.algorand.org/docs/reference/transactions/
*/
fee(feeObj) {
this._isFlatFee = true;
const fee = new bignumber_js_1.default(feeObj.fee).toNumber();
if (this._isFlatFee && fee < MIN_FEE) {
throw new errors_1.InsufficientFeeError(fee, MIN_FEE);
}
this._fee = fee;
return this;
}
/**
* Sets whether the fee is a flat fee.
*
* A flat fee is the fee for the entire transaction whereas a normal fee
* is a fee for every byte in the transaction.
*
* @param {boolean} isFlatFee Whether the fee should be specified as a flat fee.
* @returns {TransactionBuilder} This transaction builder.
*/
isFlatFee(isFlatFee) {
this._isFlatFee = isFlatFee;
return this;
}
/**
* Sets the transaction sender.
*
* @param {BaseAddress} sender The sender account
* @returns {TransactionBuilder} This transaction builder.
*
* @see https://developer.algorand.org/docs/reference/transactions/
*/
sender(sender) {
this.validateAddress(sender);
this._sender = sender.address;
this._transaction.sender(sender.address);
return this;
}
/**
* Sets the genesis id.
*
* @param {string} genesisId The genesis id.
* @example "mainnet-v1.0"
*
* @returns {TransactionBuilder} This transaction builder.
*
* @see https://developer.algorand.org/docs/reference/transactions/
*/
genesisId(genesisId) {
this._genesisId = genesisId;
return this;
}
/**
* Sets the genesis hash.
*
* The genesis hash must be base64 encoded.
*
* @param {string} genesisHash The genesis hash.
* @example "wGHE2Pwdvd7S12BL5FaOP20EGYesN73ktiC1qzkkit8="
*
* @returns {TransactionBuilder} This transaction builder.
*
* @see https://developer.algorand.org/docs/reference/transactions/
*/
genesisHash(genesisHash) {
this._genesisHash = genesisHash;
return this;
}
/**
* Sets the genesis id and hash to mainnet.
*
* @returns {TransactionBuilder} This transaction builder.
*
* @see https://developer.algorand.org/docs/reference/algorand-networks/mainnet/#genesis-id
* @see https://developer.algorand.org/docs/reference/algorand-networks/mainnet/#genesis-hash
*/
mainnet() {
this.genesisId(exports.MAINNET_GENESIS_ID);
this.genesisHash(exports.MAINNET_GENESIS_HASH);
return this;
}
/**
* Sets the genesis id and hash to testnet.
*
* @returns {TransactionBuilder} This transaction builder.
*
* @see https://developer.algorand.org/docs/reference/algorand-networks/testnet/#genesis-id
* @see https://developer.algorand.org/docs/reference/algorand-networks/testnet/#genesis-hash
*/
testnet() {
this.genesisId(exports.TESTNET_GENESIS_ID);
this.genesisHash(exports.TESTNET_GENESIS_HASH);
return this;
}
/**
* Sets the genesis id and hash to betanet.
*
* @returns {TransactionBuilder} This transaction builder.
*
* @see https://developer.algorand.org/docs/reference/algorand-networks/betanet/#genesis-id
* @see https://developer.algorand.org/docs/reference/algorand-networks/betanet/#genesis-hash
*/
betanet() {
this.genesisId(exports.BETANET_GENESIS_ID);
this.genesisHash(exports.BETANET_GENESIS_HASH);
return this;
}
/**
* Sets the first round.
*
* @param {number} round The first protocol round on which this txn is valid.
* @returns {TransactionBuilder} This transaction builder.
*
* @see https://developer.algorand.org/docs/reference/transactions/
*/
firstRound(round) {
this.validateValue(new bignumber_js_1.default(round));
this._firstRound = round;
return this;
}
/**
* Sets the last round.
*
* @param {number} round The first protocol round on which this txn is valid.
* @returns {TransactionBuilder} This transaction builder.
*
* @see https://developer.algorand.org/docs/reference/transactions/
*/
lastRound(round) {
this.validateValue(new bignumber_js_1.default(round));
this._lastRound = round;
return this;
}
/**
* Sets the lease on the transaction.
*
* A lease is a mutex on the transaction that prevents any other transaction
* from being sent with the same lease until the prior transaction's last
* round has passed.
*
* @param {Uint8Array} lease The lease to put the transaction.
* @returns {TransactionBuilder} This transaction builder.
*
* @see https://developer.algorand.org/docs/reference/transactions/
*/
lease(lease) {
this._lease = lease;
return this;
}
/**
* Sets the note for the transaction.
*
* @param {Uint8Array} note Arbitrary data for sender to store.
* @returns {TransactionBuilder} This transaction builder.
*
* @see https://developer.algorand.org/docs/reference/transactions/
*/
note(note) {
this._note = note;
return this;
}
/**
* Sets the authorized address.
*
* The authorized asset will be used to authorize all future transactions.
*
* @param {BaseAddress} authorizer The address to delegate authorization authority to.
* @returns {TransactionBuilder} This transaction builder.
*
* @see https://developer.algorand.org/docs/reference/transactions/
*/
reKeyTo(authorizer) {
this.validateAddress(authorizer);
this._reKeyTo = authorizer.address;
return this;
}
/** @inheritdoc */
validateAddress({ address }) {
if (!algosdk_1.default.isValidAddress(address)) {
throw new errors_1.AddressValidationError(address);
}
}
/** @inheritdoc */
async buildImplementation() {
this.transaction.setAlgoTransaction(this.buildAlgoTxn());
this.transaction.setTransactionType(this.transactionType);
this.transaction.sign(this._keyPairs);
this._transaction.loadInputsAndOutputs();
return this._transaction;
}
/** @inheritdoc */
fromImplementation(rawTransaction) {
const decodedTxn = utils_1.default.decodeAlgoTxn(rawTransaction);
const algosdkTxn = decodedTxn.txn;
if (decodedTxn.signed) {
this._transaction.signedTransaction = decodedTxn.rawTransaction;
if (decodedTxn.signers) {
this.setSigners(decodedTxn.signers);
}
if (decodedTxn.signedBy) {
this._transaction.signedBy = decodedTxn.signedBy;
}
}
this.sender({ address: algosdk_1.default.encodeAddress(algosdkTxn.from.publicKey) });
this._isFlatFee = true;
this._fee = algosdkTxn.fee;
this._genesisHash = algosdkTxn.genesisHash.toString('base64');
this._genesisId = algosdkTxn.genesisID;
this._firstRound = algosdkTxn.firstRound;
this._lastRound = algosdkTxn.lastRound;
this._lease = algosdkTxn.lease;
this._note = algosdkTxn.note;
this._reKeyTo = algosdkTxn.reKeyTo ? algosdk_1.default.encodeAddress(algosdkTxn.reKeyTo.publicKey) : undefined;
this._transaction.setAlgoTransaction(algosdkTxn);
return this._transaction;
}
/** @inheritdoc */
signImplementation({ key }) {
try {
const buffKey = utils_1.default.decodeSeed(key);
const keypair = new keyPair_1.KeyPair({ prv: Buffer.from(buffKey.seed).toString('hex') });
this._keyPairs.push(keypair);
}
catch (e) {
if (_1.algoUtils.isValidPrivateKey(key)) {
const keypair = new keyPair_1.KeyPair({ prv: key });
this._keyPairs.push(keypair);
}
else {
throw e;
}
}
return this._transaction;
}
numberOfSigners(num) {
this._transaction.setNumberOfRequiredSigners(num);
return this;
}
setSigners(addrs) {
const signers = addrs instanceof Array ? addrs : [addrs];
signers.forEach((address) => this.validateAddress({ address: address }));
this._transaction.signers = signers;
return this;
}
/**
* Sets the number of signers required to sign the transaction.
*
* The number of signers cannot be set to a negative value.
*
* @param {number} n The number of signers.
* @returns {TransactionBuilder} This transaction builder.
*/
numberOfRequiredSigners(n) {
if (n < 0) {
throw new sdk_core_1.BuildTransactionError(`Number of signers: '${n}' cannot be negative`);
}
this._transaction.setNumberOfRequiredSigners(n);
return this;
}
/**
* @inheritdoc
* @see https://developer.algorand.org/docs/features/accounts/#transformation-private-key-to-base64-private-key
*/
validateKey({ key }) {
let isValidPrivateKeyFromBytes;
const isValidPrivateKeyFromHex = (0, sdk_core_1.isValidEd25519Seed)(key);
const isValidPrivateKeyFromBase64 = (0, sdk_core_1.isValidEd25519Seed)(Buffer.from(key, 'base64').toString('hex'));
const isValidRootPrvKey = (0, sdk_core_1.isValidEd25519SecretKey)(key);
try {
const decodedSeed = utils_1.default.decodeSeed(key);
isValidPrivateKeyFromBytes = (0, sdk_core_1.isValidEd25519Seed)(Buffer.from(decodedSeed.seed).toString('hex'));
}
catch (err) {
isValidPrivateKeyFromBytes = false;
}
if (!isValidPrivateKeyFromBytes &&
!isValidPrivateKeyFromHex &&
!isValidPrivateKeyFromBase64 &&
!isValidRootPrvKey) {
throw new sdk_core_1.BuildTransactionError(`Key validation failed`);
}
}
/** @inheritdoc */
validateRawTransaction(rawTransaction) {
const decodedTxn = utils_1.default.decodeAlgoTxn(rawTransaction);
const algoTxn = decodedTxn.txn;
const validationResult = txnSchema_1.BaseTransactionSchema.validate({
fee: algoTxn?.fee,
firstRound: algoTxn?.firstRound,
genesisHash: algoTxn?.genesisHash.toString('base64'),
lastRound: algoTxn?.lastRound,
sender: algoTxn ? algosdk_1.default.encodeAddress(algoTxn.from.publicKey) : undefined,
genesisId: algoTxn?.genesisID,
lease: algoTxn?.lease,
note: algoTxn?.note,
reKeyTo: algoTxn?.reKeyTo ? algosdk_1.default.encodeAddress(algoTxn.reKeyTo.publicKey) : undefined,
});
if (validationResult.error) {
throw new sdk_core_1.InvalidTransactionError(`Transaction validation failed: ${validationResult.error.message}`);
}
}
/** @inheritdoc */
validateTransaction(_) {
this.validateBaseFields(this._fee, this._firstRound, this._genesisHash, this._lastRound, this._sender, this._genesisId, this._lease, this._note, this._reKeyTo);
}
validateBaseFields(fee, firstRound, genesisHash, lastRound, sender, genesisId, lease, note, reKeyTo) {
const validationResult = txnSchema_1.BaseTransactionSchema.validate({
fee,
firstRound,
genesisHash,
lastRound,
sender,
genesisId,
lease,
note,
reKeyTo,
});
if (validationResult.error) {
throw new sdk_core_1.InvalidTransactionError(`Transaction validation failed: ${validationResult.error.message}`);
}
}
/** @inheritdoc */
validateValue(value) {
if (value.isLessThan(0)) {
throw new sdk_core_1.BuildTransactionError('Value cannot be less than zero');
}
}
/** @inheritdoc */
get transaction() {
return this._transaction;
}
/** @inheritdoc */
set transaction(transaction) {
this._transaction = transaction;
}
/**
* Convenience method to retrieve the algosdk suggested parameters.
*
* @returns {algosdk.SuggestedParams} The algosdk suggested parameters.
*/
get suggestedParams() {
return {
flatFee: this._isFlatFee,
fee: this._fee,
firstRound: this._firstRound,
lastRound: this._lastRound,
genesisID: this._genesisId,
genesisHash: this._genesisHash,
};
}
}
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 { BaseCoin as CoinConfig } from '@bitgo/statics';\nimport BigNumber from 'bignumber.js';\nimport algosdk from 'algosdk';\n\nimport { Transaction } from './transaction';\nimport { AddressValidationError, InsufficientFeeError } from './errors';\nimport { KeyPair } from './keyPair';\nimport { BaseTransactionSchema } from './txnSchema';\nimport Utils from './utils';\nimport {\n  BaseAddress,\n  BaseFee,\n  BaseKey,\n  BaseTransactionBuilder,\n  BuildTransactionError,\n  InvalidTransactionError,\n  isValidEd25519SecretKey,\n  isValidEd25519Seed,\n  TransactionType,\n} from '@bitgo/sdk-core';\nimport { algoUtils } from '.';\n\nconst MIN_FEE = 1000; // in microalgos\n\nexport const MAINNET_GENESIS_ID = 'mainnet-v1.0';\nexport const MAINNET_GENESIS_HASH = 'wGHE2Pwdvd7S12BL5FaOP20EGYesN73ktiC1qzkkit8=';\nexport const TESTNET_GENESIS_ID = 'testnet-v1.0';\nexport const TESTNET_GENESIS_HASH = 'SGO1GKSzyE7IEPItTxCByw9x8FmnrCDexi9/cOUJOiI=';\nexport const BETANET_GENESIS_ID = 'betanet-v1.0';\nexport const BETANET_GENESIS_HASH = 'mFgazF+2uRS1tMiL9dsj01hJGySEmPN28B/TjjvpVW0=';\n\nexport abstract class TransactionBuilder extends BaseTransactionBuilder {\n  protected _transaction: Transaction;\n  protected _keyPairs: KeyPair[];\n\n  // the fee is specified as a number here instead of a big number because\n  // the algosdk also specifies it as a number.\n  protected _fee: number;\n  protected _isFlatFee: boolean;\n\n  protected _sender: string;\n  protected _genesisHash: string;\n  protected _genesisId: string;\n  protected _firstRound: number;\n  protected _lastRound: number;\n  protected _lease?: Uint8Array;\n  protected _note?: Uint8Array;\n  protected _reKeyTo?: string;\n  protected _suggestedParams: algosdk.SuggestedParams;\n\n  constructor(coinConfig: Readonly<CoinConfig>) {\n    super(coinConfig);\n\n    this._transaction = new Transaction(coinConfig);\n    this._keyPairs = [];\n  }\n\n  /**\n   * Sets the fee.\n   *\n   * The minimum fee is 1000 microalgos.\n   *\n   * @param {BaseFee} feeObj The amount to pay to the fee sink denoted in microalgos\n   * @returns {TransactionBuilder} This transaction builder.\n   *\n   * @see https://developer.algorand.org/docs/reference/transactions/\n   */\n  fee(feeObj: BaseFee): this {\n    this._isFlatFee = true;\n    const fee = new BigNumber(feeObj.fee).toNumber();\n    if (this._isFlatFee && fee < MIN_FEE) {\n      throw new InsufficientFeeError(fee, MIN_FEE);\n    }\n\n    this._fee = fee;\n\n    return this;\n  }\n\n  /**\n   * Sets whether the fee is a flat fee.\n   *\n   * A flat fee is the fee for the entire transaction whereas a normal fee\n   * is a fee for every byte in the transaction.\n   *\n   * @param {boolean} isFlatFee Whether the fee should be specified as a flat fee.\n   * @returns {TransactionBuilder} This transaction builder.\n   */\n  isFlatFee(isFlatFee: boolean): this {\n    this._isFlatFee = isFlatFee;\n\n    return this;\n  }\n\n  /**\n   * Sets the transaction sender.\n   *\n   * @param {BaseAddress} sender The sender account\n   * @returns {TransactionBuilder} This transaction builder.\n   *\n   * @see https://developer.algorand.org/docs/reference/transactions/\n   */\n  sender(sender: BaseAddress): this {\n    this.validateAddress(sender);\n    this._sender = sender.address;\n    this._transaction.sender(sender.address);\n\n    return this;\n  }\n\n  /**\n   * Sets the genesis id.\n   *\n   * @param {string} genesisId The genesis id.\n   * @example \"mainnet-v1.0\"\n   *\n   * @returns {TransactionBuilder} This transaction builder.\n   *\n   * @see https://developer.algorand.org/docs/reference/transactions/\n   */\n  genesisId(genesisId: string): this {\n    this._genesisId = genesisId;\n\n    return this;\n  }\n\n  /**\n   * Sets the genesis hash.\n   *\n   * The genesis hash must be base64 encoded.\n   *\n   * @param {string} genesisHash The genesis hash.\n   * @example \"wGHE2Pwdvd7S12BL5FaOP20EGYesN73ktiC1qzkkit8=\"\n   *\n   * @returns {TransactionBuilder} This transaction builder.\n   *\n   * @see https://developer.algorand.org/docs/reference/transactions/\n   */\n  genesisHash(genesisHash: string): this {\n    this._genesisHash = genesisHash;\n\n    return this;\n  }\n\n  /**\n   * Sets the genesis id and hash to mainnet.\n   *\n   * @returns {TransactionBuilder} This transaction builder.\n   *\n   * @see https://developer.algorand.org/docs/reference/algorand-networks/mainnet/#genesis-id\n   * @see https://developer.algorand.org/docs/reference/algorand-networks/mainnet/#genesis-hash\n   */\n  mainnet(): this {\n    this.genesisId(MAINNET_GENESIS_ID);\n    this.genesisHash(MAINNET_GENESIS_HASH);\n\n    return this;\n  }\n\n  /**\n   * Sets the genesis id and hash to testnet.\n   *\n   * @returns {TransactionBuilder} This transaction builder.\n   *\n   * @see https://developer.algorand.org/docs/reference/algorand-networks/testnet/#genesis-id\n   * @see https://developer.algorand.org/docs/reference/algorand-networks/testnet/#genesis-hash\n   */\n  testnet(): this {\n    this.genesisId(TESTNET_GENESIS_ID);\n    this.genesisHash(TESTNET_GENESIS_HASH);\n\n    return this;\n  }\n\n  /**\n   * Sets the genesis id and hash to betanet.\n   *\n   * @returns {TransactionBuilder} This transaction builder.\n   *\n   * @see https://developer.algorand.org/docs/reference/algorand-networks/betanet/#genesis-id\n   * @see https://developer.algorand.org/docs/reference/algorand-networks/betanet/#genesis-hash\n   */\n  betanet(): this {\n    this.genesisId(BETANET_GENESIS_ID);\n    this.genesisHash(BETANET_GENESIS_HASH);\n\n    return this;\n  }\n\n  /**\n   * Sets the first round.\n   *\n   * @param {number} round The first protocol round on which this txn is valid.\n   * @returns {TransactionBuilder} This transaction builder.\n   *\n   * @see https://developer.algorand.org/docs/reference/transactions/\n   */\n  firstRound(round: number): this {\n    this.validateValue(new BigNumber(round));\n\n    this._firstRound = round;\n\n    return this;\n  }\n\n  /**\n   * Sets the last round.\n   *\n   * @param {number} round The first protocol round on which this txn is valid.\n   * @returns {TransactionBuilder} This transaction builder.\n   *\n   * @see https://developer.algorand.org/docs/reference/transactions/\n   */\n  lastRound(round: number): this {\n    this.validateValue(new BigNumber(round));\n\n    this._lastRound = round;\n\n    return this;\n  }\n\n  /**\n   * Sets the lease on the transaction.\n   *\n   * A lease is a mutex on the transaction that prevents any other transaction\n   * from being sent with the same lease until the prior transaction's last\n   * round has passed.\n   *\n   * @param {Uint8Array} lease The lease to put the transaction.\n   * @returns {TransactionBuilder} This transaction builder.\n   *\n   * @see https://developer.algorand.org/docs/reference/transactions/\n   */\n  lease(lease: Uint8Array): this {\n    this._lease = lease;\n\n    return this;\n  }\n\n  /**\n   * Sets the note for the transaction.\n   *\n   * @param {Uint8Array} note Arbitrary data for sender to store.\n   * @returns {TransactionBuilder} This transaction builder.\n   *\n   * @see https://developer.algorand.org/docs/reference/transactions/\n   */\n  note(note: Uint8Array): this {\n    this._note = note;\n\n    return this;\n  }\n\n  /**\n   * Sets the authorized address.\n   *\n   * The authorized asset will be used to authorize all future transactions.\n   *\n   * @param {BaseAddress} authorizer The address to delegate authorization authority to.\n   * @returns {TransactionBuilder} This transaction builder.\n   *\n   * @see https://developer.algorand.org/docs/reference/transactions/\n   */\n  reKeyTo(authorizer: BaseAddress): this {\n    this.validateAddress(authorizer);\n    this._reKeyTo = authorizer.address;\n\n    return this;\n  }\n\n  /** @inheritdoc */\n  validateAddress({ address }: BaseAddress): void {\n    if (!algosdk.isValidAddress(address)) {\n      throw new AddressValidationError(address);\n    }\n  }\n\n  /** @inheritdoc */\n  protected async buildImplementation(): Promise<Transaction> {\n    this.transaction.setAlgoTransaction(this.buildAlgoTxn());\n    this.transaction.setTransactionType(this.transactionType);\n\n    this.transaction.sign(this._keyPairs);\n    this._transaction.loadInputsAndOutputs();\n    return this._transaction;\n  }\n\n  /**\n   * Builds the algorand transaction.\n   */\n  protected abstract buildAlgoTxn(): algosdk.Transaction;\n\n  /**\n   * The transaction type.\n   */\n  protected abstract get transactionType(): TransactionType;\n\n  /** @inheritdoc */\n  protected fromImplementation(rawTransaction: Uint8Array | string): Transaction {\n    const decodedTxn = Utils.decodeAlgoTxn(rawTransaction);\n    const algosdkTxn = decodedTxn.txn;\n\n    if (decodedTxn.signed) {\n      this._transaction.signedTransaction = decodedTxn.rawTransaction;\n      if (decodedTxn.signers) {\n        this.setSigners(decodedTxn.signers);\n      }\n      if (decodedTxn.signedBy) {\n        this._transaction.signedBy = decodedTxn.signedBy;\n      }\n    }\n    this.sender({ address: algosdk.encodeAddress(algosdkTxn.from.publicKey) });\n    this._isFlatFee = true;\n    this._fee = algosdkTxn.fee;\n    this._genesisHash = algosdkTxn.genesisHash.toString('base64');\n    this._genesisId = algosdkTxn.genesisID;\n    this._firstRound = algosdkTxn.firstRound;\n    this._lastRound = algosdkTxn.lastRound;\n    this._lease = algosdkTxn.lease;\n    this._note = algosdkTxn.note;\n    this._reKeyTo = algosdkTxn.reKeyTo ? algosdk.encodeAddress(algosdkTxn.reKeyTo.publicKey) : undefined;\n\n    this._transaction.setAlgoTransaction(algosdkTxn);\n\n    return this._transaction;\n  }\n\n  /** @inheritdoc */\n  protected signImplementation({ key }: BaseKey): Transaction {\n    try {\n      const buffKey = Utils.decodeSeed(key);\n      const keypair = new KeyPair({ prv: Buffer.from(buffKey.seed).toString('hex') });\n      this._keyPairs.push(keypair);\n    } catch (e) {\n      if (algoUtils.isValidPrivateKey(key)) {\n        const keypair = new KeyPair({ prv: key });\n        this._keyPairs.push(keypair);\n      } else {\n        throw e;\n      }\n    }\n\n    return this._transaction;\n  }\n\n  numberOfSigners(num: number): this {\n    this._transaction.setNumberOfRequiredSigners(num);\n\n    return this;\n  }\n\n  setSigners(addrs: string | string[]): this {\n    const signers = addrs instanceof Array ? addrs : [addrs];\n    signers.forEach((address) => this.validateAddress({ address: address }));\n    this._transaction.signers = signers;\n    return this;\n  }\n\n  /**\n   * Sets the number of signers required to sign the transaction.\n   *\n   * The number of signers cannot be set to a negative value.\n   *\n   * @param {number} n The number of signers.\n   * @returns {TransactionBuilder} This transaction builder.\n   */\n  numberOfRequiredSigners(n: number): this {\n    if (n < 0) {\n      throw new BuildTransactionError(`Number of signers: '${n}' cannot be negative`);\n    }\n\n    this._transaction.setNumberOfRequiredSigners(n);\n\n    return this;\n  }\n\n  /**\n   * @inheritdoc\n   * @see https://developer.algorand.org/docs/features/accounts/#transformation-private-key-to-base64-private-key\n   */\n  validateKey({ key }: BaseKey): void {\n    let isValidPrivateKeyFromBytes;\n    const isValidPrivateKeyFromHex = isValidEd25519Seed(key);\n    const isValidPrivateKeyFromBase64 = isValidEd25519Seed(Buffer.from(key, 'base64').toString('hex'));\n    const isValidRootPrvKey = isValidEd25519SecretKey(key);\n    try {\n      const decodedSeed = Utils.decodeSeed(key);\n      isValidPrivateKeyFromBytes = isValidEd25519Seed(Buffer.from(decodedSeed.seed).toString('hex'));\n    } catch (err) {\n      isValidPrivateKeyFromBytes = false;\n    }\n\n    if (\n      !isValidPrivateKeyFromBytes &&\n      !isValidPrivateKeyFromHex &&\n      !isValidPrivateKeyFromBase64 &&\n      !isValidRootPrvKey\n    ) {\n      throw new BuildTransactionError(`Key validation failed`);\n    }\n  }\n\n  /** @inheritdoc */\n  validateRawTransaction(rawTransaction: Uint8Array | string): void {\n    const decodedTxn = Utils.decodeAlgoTxn(rawTransaction);\n    const algoTxn = decodedTxn.txn;\n\n    const validationResult = BaseTransactionSchema.validate({\n      fee: algoTxn?.fee,\n      firstRound: algoTxn?.firstRound,\n      genesisHash: algoTxn?.genesisHash.toString('base64'),\n      lastRound: algoTxn?.lastRound,\n      sender: algoTxn ? algosdk.encodeAddress(algoTxn.from.publicKey) : undefined,\n      genesisId: algoTxn?.genesisID,\n      lease: algoTxn?.lease,\n      note: algoTxn?.note,\n      reKeyTo: algoTxn?.reKeyTo ? algosdk.encodeAddress(algoTxn.reKeyTo.publicKey) : undefined,\n    });\n\n    if (validationResult.error) {\n      throw new InvalidTransactionError(`Transaction validation failed: ${validationResult.error.message}`);\n    }\n  }\n\n  /** @inheritdoc */\n  validateTransaction(_: Transaction): void {\n    this.validateBaseFields(\n      this._fee,\n      this._firstRound,\n      this._genesisHash,\n      this._lastRound,\n      this._sender,\n      this._genesisId,\n      this._lease,\n      this._note,\n      this._reKeyTo\n    );\n  }\n\n  private validateBaseFields(\n    fee: number,\n    firstRound: number,\n    genesisHash: string,\n    lastRound: number,\n    sender: string,\n    genesisId: string,\n    lease: Uint8Array | undefined,\n    note: Uint8Array | undefined,\n    reKeyTo: string | undefined\n  ): void {\n    const validationResult = BaseTransactionSchema.validate({\n      fee,\n      firstRound,\n      genesisHash,\n      lastRound,\n      sender,\n      genesisId,\n      lease,\n      note,\n      reKeyTo,\n    });\n\n    if (validationResult.error) {\n      throw new InvalidTransactionError(`Transaction validation failed: ${validationResult.error.message}`);\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  /** @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  /**\n   * Convenience method to retrieve the algosdk suggested parameters.\n   *\n   * @returns {algosdk.SuggestedParams} The algosdk suggested parameters.\n   */\n  protected get suggestedParams(): algosdk.SuggestedParams {\n    return {\n      flatFee: this._isFlatFee,\n      fee: this._fee,\n      firstRound: this._firstRound,\n      lastRound: this._lastRound,\n      genesisID: this._genesisId,\n      genesisHash: this._genesisHash,\n    };\n  }\n}\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!