PHP WebShell
Текущая директория: /usr/lib/node_modules/bitgo/node_modules/@bitgo/sdk-coin-rune/dist/src
Просмотр файла: rune.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.Rune = void 0;
const abstract_cosmos_1 = require("@bitgo/abstract-cosmos");
const sdk_core_1 = require("@bitgo/sdk-core");
const statics_1 = require("@bitgo/statics");
const lib_1 = require("./lib");
const constants_1 = require("./lib/constants");
const utils_1 = require("./lib/utils");
const bignumber_js_1 = require("bignumber.js");
const bech32 = require('bech32-buffer');
const _ = __importStar(require("lodash"));
const crypto_1 = require("crypto");
class Rune extends abstract_cosmos_1.CosmosCoin {
constructor(bitgo, staticsCoin) {
super(bitgo, staticsCoin);
if (!staticsCoin) {
throw new Error('missing required constructor parameter staticsCoin');
}
this._staticsCoin = staticsCoin;
this._utils = new utils_1.RuneUtils();
}
static createInstance(bitgo, staticsCoin) {
return new Rune(bitgo, staticsCoin);
}
/** @inheritDoc **/
getBuilder() {
return new lib_1.TransactionBuilderFactory(statics_1.coins.get(this.getChain()));
}
/**
* Factor between the coin's base unit and its smallest subdivison
*/
getBaseFactor() {
return 1e8;
}
isValidAddress(address) {
return this._utils.isValidAddress(address) || this._utils.isValidValidatorAddress(address);
}
/** @inheritDoc **/
getPublicNodeUrl() {
return sdk_core_1.Environments[this.bitgo.getEnv()].runeNodeUrl;
}
/** @inheritDoc **/
getDenomination() {
return statics_1.BaseUnit.RUNE;
}
/** @inheritDoc **/
getGasAmountDetails() {
return {
gasAmount: constants_1.GAS_AMOUNT,
gasLimit: constants_1.GAS_LIMIT,
};
}
/** @inheritDoc **/
getKeyPair(publicKey) {
return new lib_1.KeyPair({ pub: publicKey });
}
/** @inheritDoc **/
getAddressFromPublicKey(publicKey) {
return new lib_1.KeyPair({ pub: publicKey }).getAddress();
}
async verifyTransaction(params) {
let totalAmount = new bignumber_js_1.BigNumber(0);
const { txPrebuild, txParams } = params;
const rawTx = txPrebuild.txHex;
if (!rawTx) {
throw new Error('missing required tx prebuild property txHex');
}
const transaction = await this.getBuilder().from(rawTx).build();
const explainedTx = transaction.explainTransaction();
if (txParams.recipients && txParams.recipients.length > 0) {
const filteredRecipients = txParams.recipients.map((recipient) => ({
address: this.getAddressDetails(recipient.address).address,
amount: recipient.amount,
}));
let filteredOutputs = explainedTx.outputs.map((output) => _.pick(output, ['address', 'amount']));
filteredOutputs = filteredOutputs.map((output) => {
const prefix = this._utils.getNetworkPrefix();
const convertedAddress = bech32.encode(prefix, output.address);
return {
...output,
address: convertedAddress,
};
});
if (!_.isEqual(filteredOutputs, filteredRecipients)) {
throw new Error('Tx outputs does not match with expected txParams recipients');
}
// WithdrawDelegatorRewards and ContractCall transaction don't have amount
if (transaction.type !== sdk_core_1.TransactionType.StakingWithdraw && transaction.type !== sdk_core_1.TransactionType.ContractCall) {
for (const recipients of txParams.recipients) {
totalAmount = totalAmount.plus(recipients.amount);
}
if (!totalAmount.isEqualTo(explainedTx.outputAmount)) {
throw new Error('Tx total amount does not match with expected total amount field');
}
}
}
return true;
}
getNativeRuneTxnFees() {
return constants_1.RUNE_FEES;
}
/**
* This function is overridden from CosmosCoin class' recover function due to the difference in fees handling in thorchain
* @param {RecoveryOptions} params parameters needed to construct and
* (maybe) sign the transaction
*
* @returns {CosmosLikeCoinRecoveryOutput} the serialized transaction hex string and index
* of the address being swept
*/
async recover(params) {
// Step 1: Check if params contains the required parameters
if (!params.recoveryDestination || !this.isValidAddress(params.recoveryDestination)) {
throw new Error('invalid recoveryDestination');
}
if (!params.userKey) {
throw new Error('missing userKey');
}
if (!params.backupKey) {
throw new Error('missing backupKey');
}
if (!params.walletPassphrase) {
throw new Error('missing wallet passphrase');
}
// Step 2: Fetch the bitgo key from params
const userKey = params.userKey.replace(/\s/g, '');
const backupKey = params.backupKey.replace(/\s/g, '');
const { userKeyShare, backupKeyShare, commonKeyChain } = await sdk_core_1.ECDSAUtils.getMpcV2RecoveryKeyShares(userKey, backupKey, params.walletPassphrase); // baseAddress is not extracted
// Step 3: Instantiate the ECDSA signer and fetch the address details
const MPC = new sdk_core_1.Ecdsa();
const chainId = await this.getChainId();
const publicKey = MPC.deriveUnhardened(commonKeyChain, constants_1.ROOT_PATH).slice(0, 66);
const senderAddress = this.getAddressFromPublicKey(publicKey);
// Step 4: Fetch account details such as accountNo, balance and check for sufficient funds once gasAmount has been deducted
const [accountNumber, sequenceNo] = await this.getAccountDetails(senderAddress);
const balances = await this.getAccountBalance(senderAddress);
const balance = new bignumber_js_1.BigNumber(balances[0].amount);
const gasBudget = {
amount: [{ denom: this.getDenomination(), amount: this.getGasAmountDetails().gasAmount }],
gasLimit: this.getGasAmountDetails().gasLimit,
};
const actualBalance = balance.minus(this.getNativeRuneTxnFees());
if (actualBalance.isLessThanOrEqualTo(0)) {
throw new Error('Did not have enough funds to recover');
}
// Step 5: Once sufficient funds are present, construct the recover tx messsage
const amount = [
{
denom: this.getDenomination(),
amount: actualBalance.toFixed(),
},
];
const sendMessage = [
{
fromAddress: senderAddress,
toAddress: params.recoveryDestination,
amount: amount,
},
];
// Step 6: Build the unsigned tx using the constructed message
const txnBuilder = this.getBuilder().getTransferBuilder();
txnBuilder
.messages(sendMessage)
.gasBudget(gasBudget)
.publicKey(publicKey)
.sequence(Number(sequenceNo))
.accountNumber(Number(accountNumber))
.chainId(chainId);
const unsignedTransaction = (await txnBuilder.build());
let serializedTx = unsignedTransaction.toBroadcastFormat();
const signableHex = unsignedTransaction.signablePayload.toString('hex');
// Step 7: Sign the tx
const message = unsignedTransaction.signablePayload;
const messageHash = (0, crypto_1.createHash)('sha256').update(message).digest();
const signature = await sdk_core_1.ECDSAUtils.signRecoveryMpcV2(messageHash, userKeyShare, backupKeyShare, commonKeyChain);
const signableBuffer = Buffer.from(signableHex, 'hex');
MPC.verify(signableBuffer, signature, this.getHashFunction());
const cosmosKeyPair = this.getKeyPair(publicKey);
txnBuilder.addSignature({ pub: cosmosKeyPair.getKeys().pub }, Buffer.from(signature.r + signature.s, 'hex'));
const signedTransaction = await txnBuilder.build();
serializedTx = signedTransaction.toBroadcastFormat();
return { serializedTx: serializedTx };
}
}
exports.Rune = Rune;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"rune.js","sourceRoot":"","sources":["../../src/rune.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  CosmosCoin,\n  CosmosKeyPair,\n  CosmosLikeCoinRecoveryOutput,\n  CosmosTransaction,\n  FeeData,\n  GasAmountDetails,\n  RecoveryOptions,\n  SendMessage,\n} from '@bitgo/abstract-cosmos';\nimport {\n  BaseCoin,\n  BitGoBase,\n  Ecdsa,\n  ECDSAUtils,\n  Environments,\n  TransactionType,\n  VerifyTransactionOptions,\n} from '@bitgo/sdk-core';\nimport { BaseCoin as StaticsBaseCoin, BaseUnit, coins } from '@bitgo/statics';\nimport { KeyPair, TransactionBuilderFactory } from './lib';\nimport { GAS_AMOUNT, GAS_LIMIT, RUNE_FEES, ROOT_PATH } from './lib/constants';\nimport { RuneUtils } from './lib/utils';\nimport { BigNumber } from 'bignumber.js';\nconst bech32 = require('bech32-buffer');\nimport * as _ from 'lodash';\nimport { Coin } from '@cosmjs/stargate';\nimport { createHash } from 'crypto';\n\nexport class Rune extends CosmosCoin {\n  protected readonly _utils: RuneUtils;\n  protected readonly _staticsCoin: Readonly<StaticsBaseCoin>;\n  protected constructor(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>) {\n    super(bitgo, staticsCoin);\n    if (!staticsCoin) {\n      throw new Error('missing required constructor parameter staticsCoin');\n    }\n    this._staticsCoin = staticsCoin;\n    this._utils = new RuneUtils();\n  }\n\n  static createInstance(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>): BaseCoin {\n    return new Rune(bitgo, staticsCoin);\n  }\n\n  /** @inheritDoc **/\n  getBuilder(): TransactionBuilderFactory {\n    return new TransactionBuilderFactory(coins.get(this.getChain()));\n  }\n\n  /**\n   * Factor between the coin's base unit and its smallest subdivison\n   */\n  public getBaseFactor(): number {\n    return 1e8;\n  }\n\n  isValidAddress(address: string): boolean {\n    return this._utils.isValidAddress(address) || this._utils.isValidValidatorAddress(address);\n  }\n\n  /** @inheritDoc **/\n  protected getPublicNodeUrl(): string {\n    return Environments[this.bitgo.getEnv()].runeNodeUrl;\n  }\n\n  /** @inheritDoc **/\n  getDenomination(): string {\n    return BaseUnit.RUNE;\n  }\n\n  /** @inheritDoc **/\n  getGasAmountDetails(): GasAmountDetails {\n    return {\n      gasAmount: GAS_AMOUNT,\n      gasLimit: GAS_LIMIT,\n    };\n  }\n\n  /** @inheritDoc **/\n  getKeyPair(publicKey: string): CosmosKeyPair {\n    return new KeyPair({ pub: publicKey });\n  }\n\n  /** @inheritDoc **/\n  getAddressFromPublicKey(publicKey: string): string {\n    return new KeyPair({ pub: publicKey }).getAddress();\n  }\n\n  async verifyTransaction(params: VerifyTransactionOptions): Promise<boolean> {\n    let totalAmount = new BigNumber(0);\n    const { txPrebuild, txParams } = params;\n    const rawTx = txPrebuild.txHex;\n    if (!rawTx) {\n      throw new Error('missing required tx prebuild property txHex');\n    }\n    const transaction = await this.getBuilder().from(rawTx).build();\n    const explainedTx = transaction.explainTransaction();\n\n    if (txParams.recipients && txParams.recipients.length > 0) {\n      const filteredRecipients = txParams.recipients.map((recipient) => ({\n        address: this.getAddressDetails(recipient.address).address,\n        amount: recipient.amount,\n      }));\n      let filteredOutputs = explainedTx.outputs.map((output) => _.pick(output, ['address', 'amount']));\n\n      filteredOutputs = filteredOutputs.map((output) => {\n        const prefix = this._utils.getNetworkPrefix();\n        const convertedAddress = bech32.encode(prefix, output.address);\n        return {\n          ...output,\n          address: convertedAddress,\n        };\n      });\n\n      if (!_.isEqual(filteredOutputs, filteredRecipients)) {\n        throw new Error('Tx outputs does not match with expected txParams recipients');\n      }\n      // WithdrawDelegatorRewards and ContractCall transaction don't have amount\n      if (transaction.type !== TransactionType.StakingWithdraw && transaction.type !== TransactionType.ContractCall) {\n        for (const recipients of txParams.recipients) {\n          totalAmount = totalAmount.plus(recipients.amount);\n        }\n        if (!totalAmount.isEqualTo(explainedTx.outputAmount)) {\n          throw new Error('Tx total amount does not match with expected total amount field');\n        }\n      }\n    }\n    return true;\n  }\n\n  getNativeRuneTxnFees(): string {\n    return RUNE_FEES;\n  }\n\n  /**\n   * This function is overridden from CosmosCoin class' recover function due to the difference in fees handling in thorchain\n   * @param {RecoveryOptions} params parameters needed to construct and\n   * (maybe) sign the transaction\n   *\n   * @returns {CosmosLikeCoinRecoveryOutput} the serialized transaction hex string and index\n   * of the address being swept\n   */\n  async recover(params: RecoveryOptions): Promise<CosmosLikeCoinRecoveryOutput> {\n    // Step 1: Check if params contains the required parameters\n\n    if (!params.recoveryDestination || !this.isValidAddress(params.recoveryDestination)) {\n      throw new Error('invalid recoveryDestination');\n    }\n\n    if (!params.userKey) {\n      throw new Error('missing userKey');\n    }\n\n    if (!params.backupKey) {\n      throw new Error('missing backupKey');\n    }\n\n    if (!params.walletPassphrase) {\n      throw new Error('missing wallet passphrase');\n    }\n\n    // Step 2: Fetch the bitgo key from params\n    const userKey = params.userKey.replace(/\\s/g, '');\n    const backupKey = params.backupKey.replace(/\\s/g, '');\n\n    const { userKeyShare, backupKeyShare, commonKeyChain } = await ECDSAUtils.getMpcV2RecoveryKeyShares(\n      userKey,\n      backupKey,\n      params.walletPassphrase\n    ); // baseAddress is not extracted\n    // Step 3: Instantiate the ECDSA signer and fetch the address details\n    const MPC = new Ecdsa();\n    const chainId = await this.getChainId();\n    const publicKey = MPC.deriveUnhardened(commonKeyChain, ROOT_PATH).slice(0, 66);\n    const senderAddress = this.getAddressFromPublicKey(publicKey);\n\n    // Step 4: Fetch account details such as accountNo, balance and check for sufficient funds once gasAmount has been deducted\n    const [accountNumber, sequenceNo] = await this.getAccountDetails(senderAddress);\n    const balances = await this.getAccountBalance(senderAddress);\n    const balance = new BigNumber(balances[0].amount);\n    const gasBudget: FeeData = {\n      amount: [{ denom: this.getDenomination(), amount: this.getGasAmountDetails().gasAmount }],\n      gasLimit: this.getGasAmountDetails().gasLimit,\n    };\n    const actualBalance = balance.minus(this.getNativeRuneTxnFees());\n\n    if (actualBalance.isLessThanOrEqualTo(0)) {\n      throw new Error('Did not have enough funds to recover');\n    }\n\n    // Step 5: Once sufficient funds are present, construct the recover tx messsage\n    const amount: Coin[] = [\n      {\n        denom: this.getDenomination(),\n        amount: actualBalance.toFixed(),\n      },\n    ];\n    const sendMessage: SendMessage[] = [\n      {\n        fromAddress: senderAddress,\n        toAddress: params.recoveryDestination,\n        amount: amount,\n      },\n    ];\n\n    // Step 6: Build the unsigned tx using the constructed message\n    const txnBuilder = this.getBuilder().getTransferBuilder();\n    txnBuilder\n      .messages(sendMessage)\n      .gasBudget(gasBudget)\n      .publicKey(publicKey)\n      .sequence(Number(sequenceNo))\n      .accountNumber(Number(accountNumber))\n      .chainId(chainId);\n    const unsignedTransaction = (await txnBuilder.build()) as CosmosTransaction;\n    let serializedTx = unsignedTransaction.toBroadcastFormat();\n    const signableHex = unsignedTransaction.signablePayload.toString('hex');\n\n    // Step 7: Sign the tx\n    const message = unsignedTransaction.signablePayload;\n    const messageHash = createHash('sha256').update(message).digest();\n\n    const signature = await ECDSAUtils.signRecoveryMpcV2(messageHash, userKeyShare, backupKeyShare, commonKeyChain);\n\n    const signableBuffer = Buffer.from(signableHex, 'hex');\n    MPC.verify(signableBuffer, signature, this.getHashFunction());\n    const cosmosKeyPair = this.getKeyPair(publicKey);\n    txnBuilder.addSignature({ pub: cosmosKeyPair.getKeys().pub }, Buffer.from(signature.r + signature.s, 'hex'));\n    const signedTransaction = await txnBuilder.build();\n    serializedTx = signedTransaction.toBroadcastFormat();\n\n    return { serializedTx: serializedTx };\n  }\n}\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!