PHP WebShell
Текущая директория: /usr/lib/node_modules/bitgo/node_modules/@bitgo/sdk-coin-stx/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 bignumber_js_1 = __importDefault(require("bignumber.js"));
const bn_js_1 = __importDefault(require("bn.js"));
const statics_1 = require("@bitgo/statics");
const transactions_1 = require("@stacks/transactions");
const network_1 = require("@stacks/network");
const sdk_core_1 = require("@bitgo/sdk-core");
const transaction_1 = require("./transaction");
const keyPair_1 = require("./keyPair");
const utils_1 = require("./utils");
const constants_1 = require("./constants");
class TransactionBuilder extends sdk_core_1.BaseTransactionBuilder {
constructor(_coinConfig) {
super(_coinConfig);
this.getSignature = (_, index) => this._signatures.find((s) => s.index === index);
this.getPrivateKey = (pubKey, _) => this._multiSignerKeyPairs.find((kp) => kp.getKeys(true).pub === pubKey || kp.getKeys().pub === pubKey);
this._anchorMode = constants_1.ANCHOR_MODE;
this._multiSignerKeyPairs = [];
this._fromPubKeys = [];
this._signatures = [];
this._numberSignatures = constants_1.DEFAULT_MULTISIG_SIG_NUMBER;
this._network = _coinConfig.network.type === statics_1.NetworkType.MAINNET ? new network_1.StacksMainnet() : new network_1.StacksTestnet();
this._transaction = new transaction_1.Transaction(_coinConfig);
}
/**
* Initialize the transaction builder fields using the decoded transaction data
*
* @param {Transaction} tx the transaction data
*/
initBuilder(tx) {
this.transaction = tx;
// check if it is signed or unsigned tx
if (tx.stxTransaction.auth.spendingCondition === undefined) {
throw new sdk_core_1.InvalidTransactionError('spending condition cannot be undefined');
}
const txData = tx.toJson();
this.fee({ fee: txData.fee.toString() });
this.nonce(txData.nonce);
let sigHash = tx.stxTransaction.verifyBegin();
const authType = tx.stxTransaction.auth.authType ? tx.stxTransaction.auth.authType : transactions_1.AuthType.Standard;
if ((0, transactions_1.isSingleSig)(tx.stxTransaction.auth.spendingCondition)) {
this._numberSignatures = 1;
if (tx.stxTransaction.auth.spendingCondition.signature.data !== (0, transactions_1.emptyMessageSignature)().data) {
const signature = tx.stxTransaction.auth.spendingCondition.signature;
sigHash = (0, transactions_1.makeSigHashPreSign)(sigHash, authType, new bn_js_1.default(this._fee.fee), new bn_js_1.default(this._nonce));
this._signatures.push({ ...signature, index: 0, sigHash });
this._fromPubKeys = [(0, transactions_1.publicKeyFromSignature)(sigHash, signature)];
}
}
else {
this._numberSignatures = tx.stxTransaction.auth.spendingCondition.signaturesRequired;
tx.stxTransaction.auth.spendingCondition.fields.forEach((field, index) => {
if (field.contents.type === transactions_1.StacksMessageType.MessageSignature) {
const signature = field.contents;
const nextVerify = (0, transactions_1.nextVerification)(sigHash, authType, new bn_js_1.default(this._fee.fee), new bn_js_1.default(this._nonce), transactions_1.PubKeyEncoding.Compressed, // useless param as Compressed is hardcoded in stacks lib
signature);
sigHash = nextVerify.nextSigHash;
this._signatures.push({ ...signature, index, sigHash });
this._fromPubKeys.push(nextVerify.pubKey.data.toString('hex'));
}
else {
this._fromPubKeys.push(field.contents.data.toString('hex'));
}
});
}
}
/** @inheritdoc */
fromImplementation(rawTransaction) {
const tx = new transaction_1.Transaction(this._coinConfig);
this.validateRawTransaction(rawTransaction);
const stackstransaction = (0, transactions_1.deserializeTransaction)(transactions_1.BufferReader.fromBuffer(Buffer.from((0, utils_1.removeHexPrefix)(rawTransaction), 'hex')));
tx.stxTransaction = stackstransaction;
this.initBuilder(tx);
return this.transaction;
}
// region Base Builder
/** @inheritdoc */
async buildImplementation() {
const isMultiSig = this._fromPubKeys.length > 1;
this._transaction.stxTransaction.setFee(new bn_js_1.default(this._fee.fee));
this._transaction.stxTransaction.setNonce(new bn_js_1.default(this._nonce));
for (let index = 0; index < this._fromPubKeys.length; index++) {
const pubKey = this._fromPubKeys[index];
const signature = this.getSignature(pubKey, index);
if (signature) {
await this.transaction.signWithSignatures(signature, isMultiSig);
}
else {
const prvKey = this.getPrivateKey(pubKey, index);
if (prvKey) {
await this.transaction.sign(prvKey);
}
else if (isMultiSig) {
await this.transaction.appendOrigin(pubKey);
}
}
}
this._transaction.loadInputsAndOutputs();
return this._transaction;
}
/** @inheritdoc */
signImplementation(key) {
this.checkDuplicatedKeys(key);
let prv = key.key;
if (prv.startsWith('xprv')) {
const rawPrv = (0, sdk_core_1.xprvToRawPrv)(prv);
prv = new keyPair_1.KeyPair({ prv: rawPrv }).getKeys(true).prv;
}
const signer = new keyPair_1.KeyPair({ prv: prv });
// Signing the transaction is an operation that relies on all the data being set,
// so we set the source here and leave the actual signing for the build step
this._multiSignerKeyPairs.push(signer);
const publicKey = signer.getKeys(signer.getCompressed()).pub;
if (!this._fromPubKeys.includes(publicKey)) {
this._fromPubKeys.push(publicKey);
}
return this.transaction;
}
/** @inheritdoc */
get transaction() {
return this._transaction;
}
/** @inheritdoc */
set transaction(transaction) {
this._transaction = transaction;
}
/**
* Validates that the given key is not already in this._multiSignerKeyPairs
*
* @param {BaseKey} key - The key to check
*/
checkDuplicatedKeys(key) {
this._multiSignerKeyPairs.forEach((_sourceKeyPair) => {
if (_sourceKeyPair.getKeys().prv === key.key) {
throw new sdk_core_1.SigningError('Repeated sign: ' + key.key);
}
});
}
/**
* Set the transaction fees
*
* @param {BaseFee} fee The maximum gas to pay
* @returns {TransactionBuilder} This transaction builder
*/
fee(fee) {
this.validateValue(new bignumber_js_1.default(fee.fee));
this._fee = fee;
return this;
}
nonce(n) {
this._nonce = n;
return this;
}
fromPubKey(senderPubKey) {
const pubKeys = senderPubKey instanceof Array ? senderPubKey : [senderPubKey];
this._fromPubKeys = [];
pubKeys.forEach((key) => {
if ((0, utils_1.isValidPublicKey)(key)) {
this._fromPubKeys.push(key);
}
else {
throw new sdk_core_1.InvalidParameterValueError('Invalid public key');
}
});
return this;
}
/**
* Set the memo
*
* @param {string} memo
* @returns {TransactionBuilder} This transaction builder
*/
memo(memo) {
if (!(0, utils_1.isValidMemo)(memo)) {
throw new sdk_core_1.BuildTransactionError('Memo is too long');
}
this._memo = memo;
return this;
}
/**
* Set the number of signatures for multi-sig
*
* @param {number} numSigns
* @returns {TransactionBuilder} This transaction builder
*/
numberSignatures(numSigns) {
this.validateValue(new bignumber_js_1.default(numSigns));
this._numberSignatures = numSigns;
return this;
}
// region Validators
/** @inheritdoc */
validateAddress(address, addressFormat) {
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) {
if (!rawTransaction) {
throw new sdk_core_1.InvalidTransactionError('Raw transaction is empty');
}
try {
(0, transactions_1.deserializeTransaction)(transactions_1.BufferReader.fromBuffer(Buffer.from((0, utils_1.removeHexPrefix)(rawTransaction), 'hex')));
}
catch (e) {
throw new sdk_core_1.ParseTransactionError('There was an error parsing the raw transaction');
}
}
/** @inheritdoc */
validateTransaction(transaction) {
this.validateFee();
this.validateNonce();
}
/** @inheritdoc */
validateValue(value) {
if (value.isLessThan(0)) {
throw new sdk_core_1.BuildTransactionError('Value cannot be less than zero');
}
}
/**
* Validates that the fee field is defined
*/
validateFee() {
if (this._fee === undefined) {
throw new sdk_core_1.BuildTransactionError('Invalid transaction: missing fee');
}
try {
this.validateValue(new bignumber_js_1.default(this._fee.fee));
}
catch (e) {
throw new sdk_core_1.BuildTransactionError('Invalid fee');
}
}
/**
* Validates that nonce is defined
*/
validateNonce() {
if (this._nonce === undefined) {
throw new sdk_core_1.BuildTransactionError('Invalid transaction: missing nonce');
}
try {
this.validateValue(new bignumber_js_1.default(this._nonce));
}
catch (e) {
throw new sdk_core_1.BuildTransactionError(`Invalid nonce ${this._nonce}`);
}
}
}
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 BigNum from 'bn.js';\nimport { BaseCoin as CoinConfig, NetworkType } from '@bitgo/statics';\nimport {\n  AuthType,\n  BufferReader,\n  deserializeTransaction,\n  emptyMessageSignature,\n  isSingleSig,\n  makeSigHashPreSign,\n  nextVerification,\n  publicKeyFromSignature,\n  StacksMessageType,\n  PubKeyEncoding,\n} from '@stacks/transactions';\nimport { StacksNetwork, StacksTestnet, StacksMainnet } from '@stacks/network';\nimport {\n  BaseAddress,\n  BaseFee,\n  BaseKey,\n  BaseTransactionBuilder,\n  BuildTransactionError,\n  InvalidParameterValueError,\n  InvalidTransactionError,\n  ParseTransactionError,\n  SigningError,\n  xprvToRawPrv,\n} from '@bitgo/sdk-core';\nimport { Transaction } from './transaction';\nimport { KeyPair } from './keyPair';\nimport { SignatureData } from './iface';\nimport { isValidAddress, removeHexPrefix, isValidMemo, isValidPublicKey } from './utils';\nimport { ANCHOR_MODE, DEFAULT_MULTISIG_SIG_NUMBER } from './constants';\n\nexport abstract class TransactionBuilder extends BaseTransactionBuilder {\n  private _transaction: Transaction;\n  protected _anchorMode: number;\n  protected _fee: BaseFee;\n  protected _nonce: number;\n  protected _memo: string;\n  protected _numberSignatures: number;\n  protected _multiSignerKeyPairs: KeyPair[];\n  protected _signatures: SignatureData[];\n  protected _network: StacksNetwork;\n  protected _fromPubKeys: string[];\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    this._anchorMode = ANCHOR_MODE;\n    this._multiSignerKeyPairs = [];\n    this._fromPubKeys = [];\n    this._signatures = [];\n    this._numberSignatures = DEFAULT_MULTISIG_SIG_NUMBER;\n    this._network = _coinConfig.network.type === NetworkType.MAINNET ? new StacksMainnet() : new StacksTestnet();\n    this._transaction = new Transaction(_coinConfig);\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    // check if it is signed or unsigned tx\n    if (tx.stxTransaction.auth.spendingCondition === undefined) {\n      throw new InvalidTransactionError('spending condition cannot be undefined');\n    }\n    const txData = tx.toJson();\n    this.fee({ fee: txData.fee.toString() });\n    this.nonce(txData.nonce);\n    let sigHash = tx.stxTransaction.verifyBegin();\n\n    const authType = tx.stxTransaction.auth.authType ? tx.stxTransaction.auth.authType : AuthType.Standard;\n    if (isSingleSig(tx.stxTransaction.auth.spendingCondition)) {\n      this._numberSignatures = 1;\n      if (tx.stxTransaction.auth.spendingCondition.signature.data !== emptyMessageSignature().data) {\n        const signature = tx.stxTransaction.auth.spendingCondition.signature;\n        sigHash = makeSigHashPreSign(sigHash, authType, new BigNum(this._fee.fee), new BigNum(this._nonce));\n        this._signatures.push({ ...signature, index: 0, sigHash });\n        this._fromPubKeys = [publicKeyFromSignature(sigHash, signature)];\n      }\n    } else {\n      this._numberSignatures = tx.stxTransaction.auth.spendingCondition.signaturesRequired;\n      tx.stxTransaction.auth.spendingCondition.fields.forEach((field, index) => {\n        if (field.contents.type === StacksMessageType.MessageSignature) {\n          const signature = field.contents;\n          const nextVerify = nextVerification(\n            sigHash,\n            authType,\n            new BigNum(this._fee.fee),\n            new BigNum(this._nonce),\n            PubKeyEncoding.Compressed, // useless param as Compressed is hardcoded in stacks lib\n            signature\n          );\n          sigHash = nextVerify.nextSigHash;\n          this._signatures.push({ ...signature, index, sigHash });\n          this._fromPubKeys.push(nextVerify.pubKey.data.toString('hex'));\n        } else {\n          this._fromPubKeys.push(field.contents.data.toString('hex'));\n        }\n      });\n    }\n  }\n\n  /** @inheritdoc */\n  protected fromImplementation(rawTransaction: string): Transaction {\n    const tx = new Transaction(this._coinConfig);\n    this.validateRawTransaction(rawTransaction);\n    const stackstransaction = deserializeTransaction(\n      BufferReader.fromBuffer(Buffer.from(removeHexPrefix(rawTransaction), 'hex'))\n    );\n    tx.stxTransaction = stackstransaction;\n    this.initBuilder(tx);\n    return this.transaction;\n  }\n\n  // region Base Builder\n  /** @inheritdoc */\n  protected async buildImplementation(): Promise<Transaction> {\n    const isMultiSig: boolean = this._fromPubKeys.length > 1;\n    this._transaction.stxTransaction.setFee(new BigNum(this._fee.fee));\n    this._transaction.stxTransaction.setNonce(new BigNum(this._nonce));\n\n    for (let index = 0; index < this._fromPubKeys.length; index++) {\n      const pubKey = this._fromPubKeys[index];\n      const signature = this.getSignature(pubKey, index);\n      if (signature) {\n        await this.transaction.signWithSignatures(signature, isMultiSig);\n      } else {\n        const prvKey = this.getPrivateKey(pubKey, index);\n        if (prvKey) {\n          await this.transaction.sign(prvKey);\n        } else if (isMultiSig) {\n          await this.transaction.appendOrigin(pubKey);\n        }\n      }\n    }\n\n    this._transaction.loadInputsAndOutputs();\n    return this._transaction;\n  }\n\n  private getSignature = (_: string, index: number): SignatureData | undefined =>\n    this._signatures.find((s) => s.index === index);\n  private getPrivateKey = (pubKey: string, _: number): KeyPair | undefined =>\n    this._multiSignerKeyPairs.find((kp) => kp.getKeys(true).pub === pubKey || kp.getKeys().pub === pubKey);\n\n  /** @inheritdoc */\n  protected signImplementation(key: BaseKey): Transaction {\n    this.checkDuplicatedKeys(key);\n    let prv = key.key;\n    if (prv.startsWith('xprv')) {\n      const rawPrv = xprvToRawPrv(prv);\n      prv = new KeyPair({ prv: rawPrv }).getKeys(true).prv;\n    }\n    const signer = new KeyPair({ prv: prv });\n\n    // Signing the transaction is an operation that relies on all the data being set,\n    // so we set the source here and leave the actual signing for the build step\n    this._multiSignerKeyPairs.push(signer);\n    const publicKey = signer.getKeys(signer.getCompressed()).pub;\n    if (!this._fromPubKeys.includes(publicKey)) {\n      this._fromPubKeys.push(publicKey);\n    }\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  /**\n   * Validates that the given key is not already in this._multiSignerKeyPairs\n   *\n   * @param {BaseKey} key - The key to check\n   */\n  private checkDuplicatedKeys(key: BaseKey) {\n    this._multiSignerKeyPairs.forEach((_sourceKeyPair) => {\n      if (_sourceKeyPair.getKeys().prv === key.key) {\n        throw new SigningError('Repeated sign: ' + key.key);\n      }\n    });\n  }\n\n  /**\n   * Set the transaction fees\n   *\n   * @param {BaseFee} fee The maximum gas to pay\n   * @returns {TransactionBuilder} This transaction builder\n   */\n  fee(fee: BaseFee): this {\n    this.validateValue(new BigNumber(fee.fee));\n    this._fee = fee;\n    return this;\n  }\n\n  nonce(n: number): this {\n    this._nonce = n;\n    return this;\n  }\n\n  fromPubKey(senderPubKey: string | string[]): this {\n    const pubKeys = senderPubKey instanceof Array ? senderPubKey : [senderPubKey];\n    this._fromPubKeys = [];\n    pubKeys.forEach((key) => {\n      if (isValidPublicKey(key)) {\n        this._fromPubKeys.push(key);\n      } else {\n        throw new InvalidParameterValueError('Invalid public key');\n      }\n    });\n    return this;\n  }\n\n  /**\n   *  Set the memo\n   *\n   * @param {string} memo\n   * @returns {TransactionBuilder} This transaction builder\n   */\n  memo(memo: string): this {\n    if (!isValidMemo(memo)) {\n      throw new BuildTransactionError('Memo is too long');\n    }\n    this._memo = memo;\n    return this;\n  }\n\n  /**\n   *  Set the number of signatures for multi-sig\n   *\n   * @param {number} numSigns\n   * @returns {TransactionBuilder} This transaction builder\n   */\n  numberSignatures(numSigns: number): this {\n    this.validateValue(new BigNumber(numSigns));\n    this._numberSignatures = numSigns;\n    return this;\n  }\n\n  // region Validators\n  /** @inheritdoc */\n  validateAddress(address: BaseAddress, addressFormat?: string): 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: string): void {\n    if (!rawTransaction) {\n      throw new InvalidTransactionError('Raw transaction is empty');\n    }\n    try {\n      deserializeTransaction(BufferReader.fromBuffer(Buffer.from(removeHexPrefix(rawTransaction), 'hex')));\n    } catch (e) {\n      throw new ParseTransactionError('There was an error parsing the raw transaction');\n    }\n  }\n\n  /** @inheritdoc */\n  validateTransaction(transaction?: Transaction): void {\n    this.validateFee();\n    this.validateNonce();\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  /**\n   * Validates that the fee field is defined\n   */\n  private validateFee(): void {\n    if (this._fee === undefined) {\n      throw new BuildTransactionError('Invalid transaction: missing fee');\n    }\n    try {\n      this.validateValue(new BigNumber(this._fee.fee));\n    } catch (e) {\n      throw new BuildTransactionError('Invalid fee');\n    }\n  }\n\n  /**\n   * Validates that nonce is defined\n   */\n  private validateNonce(): void {\n    if (this._nonce === undefined) {\n      throw new BuildTransactionError('Invalid transaction: missing nonce');\n    }\n    try {\n      this.validateValue(new BigNumber(this._nonce));\n    } catch (e) {\n      throw new BuildTransactionError(`Invalid nonce ${this._nonce}`);\n    }\n  }\n}\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!