PHP WebShell
Текущая директория: /usr/lib/node_modules/bitgo/node_modules/@bitgo/abstract-eth/dist/src/lib
Просмотр файла: transferBuilder.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;
};
})();
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.TransferBuilder = void 0;
const assert_1 = __importDefault(require("assert"));
const ethUtil = __importStar(require("ethereumjs-util"));
const ethereumjs_abi_1 = __importDefault(require("ethereumjs-abi"));
const bn_js_1 = __importDefault(require("bn.js"));
const statics_1 = require("@bitgo/statics");
const sdk_core_1 = require("@bitgo/sdk-core");
const utils_1 = require("./utils");
const utils_2 = require("ethers/lib/utils");
const walletUtil_1 = require("./walletUtil");
/** ETH transfer builder */
class TransferBuilder {
constructor(serializedData, isFirstSigner) {
this._EMPTY_HEX_VALUE = '0x';
this._isFirstSigner = isFirstSigner;
if (serializedData) {
this.decodeTransferData(serializedData);
}
else {
// initialize with default values for non mandatory fields
this._expirationTime = this.getExpirationTime();
this._data = this._EMPTY_HEX_VALUE;
this._signature = this._EMPTY_HEX_VALUE;
}
}
/**
* A method to set the native coin or ERC20 token to be transferred.
* This ERC20 token may not be compatible with the network.
*
* @param {string} coin - the native coin or ERC20 token to be set
* @returns {TransferBuilder} the transfer builder instance modified
*/
coin(coin) {
this._coin = statics_1.coins.get(coin);
if (this._coin instanceof statics_1.ContractAddressDefinedToken) {
this._tokenContractAddress = this._coin.contractAddress.toString();
}
return this;
}
getIsFirstSigner() {
return this._isFirstSigner ? this._isFirstSigner : false;
}
walletVersion(version) {
this._walletVersion = version;
return this;
}
data(additionalData) {
this._signature = this._EMPTY_HEX_VALUE;
this._data = additionalData;
return this;
}
amount(amount) {
if (!(0, utils_1.isValidAmount)(amount)) {
throw new sdk_core_1.InvalidParameterValueError('Invalid amount');
}
this._signature = this._EMPTY_HEX_VALUE;
this._amount = amount;
return this;
}
to(address) {
if ((0, utils_1.isValidEthAddress)(address)) {
this._signature = this._EMPTY_HEX_VALUE;
this._toAddress = address;
return this;
}
throw new sdk_core_1.InvalidParameterValueError('Invalid address');
}
contractSequenceId(counter) {
if (counter >= 0) {
this._signature = this._EMPTY_HEX_VALUE;
this._sequenceId = counter;
return this;
}
throw new sdk_core_1.InvalidParameterValueError('Invalid contract sequence id');
}
key(signKey) {
this._signKey = signKey;
return this;
}
expirationTime(date) {
if (date > 0) {
this._signature = this._EMPTY_HEX_VALUE;
this._expirationTime = date;
return this;
}
throw new sdk_core_1.InvalidParameterValueError('Invalid expiration time');
}
isFirstSigner(isFirstSigner) {
this._isFirstSigner = isFirstSigner;
return this;
}
tokenContractAddress(tokenContractAddress) {
this._tokenContractAddress = tokenContractAddress;
return this;
}
setCoinUsesNonPackedEncodingForTxData(isCoinUsesNonPackedEncodingForTxData) {
this._coinUsesNonPackedEncodingForTxData = isCoinUsesNonPackedEncodingForTxData;
return this;
}
setSignature(signature) {
this._signKey = null;
this._signature = ethUtil.addHexPrefix(signature);
return this;
}
signAndBuild(chainId, coinUsesNonPackedEncodingForTxData) {
this._chainId = chainId;
// If the coin uses non-packed encoding for tx data, the operation hash is calculated differently
// This new encoding type is applicable only for native coins and not tokens
this._coinUsesNonPackedEncodingForTxData =
coinUsesNonPackedEncodingForTxData && this._tokenContractAddress === undefined;
if (this.hasMandatoryFields()) {
if (this._isFirstSigner) {
// First signer signs different data than the second signer in multisig evm contracts.
return ethUtil.addHexPrefix(this.getSignatureData().toString('hex'));
}
else {
if (this._tokenContractAddress !== undefined) {
return (0, utils_1.sendMultiSigTokenData)(this._toAddress, this._amount, this._tokenContractAddress, this._expirationTime, this._sequenceId, this.getSignature());
}
else {
return (0, utils_1.sendMultiSigData)(this._toAddress, this._amount, this._data, this._expirationTime, this._sequenceId, this.getSignature());
}
}
}
throw new sdk_core_1.BuildTransactionError('Missing transfer mandatory fields. Amount, destination (to) address and sequenceID are mandatory');
}
hasMandatoryFields() {
return this._amount !== undefined && this._toAddress !== undefined && this._sequenceId !== undefined;
}
/**
* Obtains the proper operation hash to sign either a sendMultiSig data
* or a sendMultiSigToken data
*
* @returns {string} the operation hash
*/
getOperationHash() {
const operationData = this.getOperationData();
let operationHash;
if (this._coinUsesNonPackedEncodingForTxData) {
const types = operationData[0];
const values = operationData[1].map((item) => typeof item === 'string' || typeof item === 'number' ? item : '0x' + item.toString('hex'));
operationHash = (0, utils_2.keccak256)(utils_2.defaultAbiCoder.encode(types, values));
}
else {
// If the coin uses packed encoding for tx data or it is a token, the operation hash is calculated using the Ethereum ABI
operationHash = ethUtil.bufferToHex(ethereumjs_abi_1.default.soliditySHA3(...operationData));
}
return operationHash;
}
getOperationData() {
let operationData;
const prefix = this.getOperationHashPrefix();
if (this._tokenContractAddress !== undefined) {
operationData = [
['string', 'address', 'uint', 'address', 'uint', 'uint'],
[
prefix,
new bn_js_1.default(ethUtil.stripHexPrefix(this._toAddress), 16),
this._amount,
new bn_js_1.default(ethUtil.stripHexPrefix(this._tokenContractAddress), 16),
this._expirationTime,
this._sequenceId,
],
];
}
else {
const toAddress = this._coinUsesNonPackedEncodingForTxData
? this._toAddress
: new bn_js_1.default(ethUtil.stripHexPrefix(this._toAddress), 16);
operationData = [
['string', 'address', 'uint', 'bytes', 'uint', 'uint'],
[
prefix,
toAddress,
this._amount,
Buffer.from(ethUtil.padToEven(ethUtil.stripHexPrefix(this._data)) || '', 'hex'),
this._expirationTime,
this._sequenceId,
],
];
}
return operationData;
}
getOperationHashPrefix() {
if (this._walletVersion === 4) {
return this._tokenContractAddress ? `${this._chainId}-ERC20` : `${this._chainId}`;
}
return this._tokenContractAddress ? this.getTokenOperationHashPrefix() : this.getNativeOperationHashPrefix();
}
/**
* Get the prefix used in generating an operation hash for sending tokens
*
* @returns the string prefix
*/
getTokenOperationHashPrefix() {
return this._coin?.network?.tokenOperationHashPrefix ?? `${this._chainId}-ERC20` ?? 'ERC20';
}
/**
* Get the prefix used in generating an operation hash for sending native coins
*
* @returns the string prefix
*/
getNativeOperationHashPrefix() {
return this._coin?.network?.nativeCoinOperationHashPrefix ?? `${this._chainId}` ?? 'ETHER';
}
/** Return an expiration time, in seconds, set to one hour from now
*
* @returns {number} expiration time
*/
getExpirationTime() {
const currentDate = new Date();
currentDate.setHours(currentDate.getHours() + 1);
return currentDate.getTime() / 1000;
}
/**
* If a signing key is set for this builder, recalculates the signature
*
* @returns {string} the signature value
*/
getSignature() {
if (this._signKey) {
this._signature = this.ethSignMsgHash();
}
return this._signature;
}
ethSignMsgHash() {
const data = this.getOperationHash();
(0, assert_1.default)(this._signKey);
const keyBuffer = Buffer.from(ethUtil.padToEven(this._signKey), 'hex');
if (keyBuffer.length !== 32) {
throw new Error('private key length is invalid');
}
const signatureInParts = ethUtil.ecsign(Buffer.from(ethUtil.padToEven(ethUtil.stripHexPrefix(data)), 'hex'), keyBuffer);
// Assemble strings from r, s and v
const r = ethUtil.setLengthLeft(signatureInParts.r, 32).toString('hex');
const s = ethUtil.setLengthLeft(signatureInParts.s, 32).toString('hex');
const v = ethUtil.stripHexPrefix(ethUtil.intToHex(signatureInParts.v));
// Concatenate the r, s and v parts to make the signature string
return ethUtil.addHexPrefix(r.concat(s, v));
}
decodeTransferData(data) {
const transferData = (0, utils_1.decodeTransferData)(data, this._isFirstSigner);
this._toAddress = transferData.to;
this._amount = transferData.amount;
this._expirationTime = transferData.expireTime;
this._sequenceId = transferData.sequenceId;
this._signature = transferData.signature;
if (transferData.data) {
this._data = transferData.data;
}
if (transferData.tokenContractAddress) {
this._tokenContractAddress = transferData.tokenContractAddress;
}
}
getSignatureData() {
const method = this._tokenContractAddress
? ethereumjs_abi_1.default.methodID('sendMultiSigToken', walletUtil_1.sendMultiSigTokenTypes)
: ethereumjs_abi_1.default.methodID('sendMultiSig', walletUtil_1.sendMultiSigTypes);
const operationData = this.getOperationData();
const rawEncodedOperationData = ethereumjs_abi_1.default.rawEncode(...operationData);
return Buffer.concat([
method,
rawEncodedOperationData,
Buffer.from([this._coinUsesNonPackedEncodingForTxData ? 1 : 0]),
]);
}
}
exports.TransferBuilder = TransferBuilder;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transferBuilder.js","sourceRoot":"","sources":["../../../src/lib/transferBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\nimport * as ethUtil from 'ethereumjs-util';\nimport EthereumAbi from 'ethereumjs-abi';\nimport BN from 'bn.js';\nimport { coins, BaseCoin, ContractAddressDefinedToken, EthereumNetwork as EthLikeNetwork } from '@bitgo/statics';\nimport { BuildTransactionError, InvalidParameterValueError } from '@bitgo/sdk-core';\nimport { decodeTransferData, sendMultiSigData, sendMultiSigTokenData, isValidEthAddress, isValidAmount } from './utils';\nimport { defaultAbiCoder, keccak256 } from 'ethers/lib/utils';\nimport { sendMultiSigTokenTypes, sendMultiSigTypes } from './walletUtil';\n\n/** ETH transfer builder */\nexport class TransferBuilder {\n  private readonly _EMPTY_HEX_VALUE = '0x';\n  protected _amount: string;\n  protected _toAddress: string;\n  protected _sequenceId: number;\n  protected _signKey: string | null;\n  protected _expirationTime: number;\n  protected _signature: string;\n  protected _isFirstSigner: boolean | undefined;\n  private _data: string;\n  private _tokenContractAddress?: string;\n  private _coin: Readonly<BaseCoin>;\n  private _chainId?: string;\n  private _coinUsesNonPackedEncodingForTxData?: boolean;\n  private _walletVersion?: number;\n\n  constructor(serializedData?: string, isFirstSigner?: boolean) {\n    this._isFirstSigner = isFirstSigner;\n    if (serializedData) {\n      this.decodeTransferData(serializedData);\n    } else {\n      // initialize with default values for non mandatory fields\n      this._expirationTime = this.getExpirationTime();\n      this._data = this._EMPTY_HEX_VALUE;\n      this._signature = this._EMPTY_HEX_VALUE;\n    }\n  }\n\n  /**\n   * A method to set the native coin or ERC20 token to be transferred.\n   * This ERC20 token may not be compatible with the network.\n   *\n   * @param {string} coin - the native coin or ERC20 token to be set\n   * @returns {TransferBuilder} the transfer builder instance modified\n   */\n  coin(coin: string): TransferBuilder {\n    this._coin = coins.get(coin);\n    if (this._coin instanceof ContractAddressDefinedToken) {\n      this._tokenContractAddress = this._coin.contractAddress.toString();\n    }\n\n    return this;\n  }\n\n  getIsFirstSigner(): boolean {\n    return this._isFirstSigner ? this._isFirstSigner : false;\n  }\n\n  walletVersion(version: number): TransferBuilder {\n    this._walletVersion = version;\n    return this;\n  }\n\n  data(additionalData: string): TransferBuilder {\n    this._signature = this._EMPTY_HEX_VALUE;\n    this._data = additionalData;\n    return this;\n  }\n\n  amount(amount: string): this {\n    if (!isValidAmount(amount)) {\n      throw new InvalidParameterValueError('Invalid amount');\n    }\n    this._signature = this._EMPTY_HEX_VALUE;\n    this._amount = amount;\n    return this;\n  }\n\n  to(address: string): TransferBuilder {\n    if (isValidEthAddress(address)) {\n      this._signature = this._EMPTY_HEX_VALUE;\n      this._toAddress = address;\n      return this;\n    }\n    throw new InvalidParameterValueError('Invalid address');\n  }\n\n  contractSequenceId(counter: number): TransferBuilder {\n    if (counter >= 0) {\n      this._signature = this._EMPTY_HEX_VALUE;\n      this._sequenceId = counter;\n      return this;\n    }\n    throw new InvalidParameterValueError('Invalid contract sequence id');\n  }\n\n  key(signKey: string): TransferBuilder {\n    this._signKey = signKey;\n    return this;\n  }\n\n  expirationTime(date: number): TransferBuilder {\n    if (date > 0) {\n      this._signature = this._EMPTY_HEX_VALUE;\n      this._expirationTime = date;\n      return this;\n    }\n    throw new InvalidParameterValueError('Invalid expiration time');\n  }\n\n  isFirstSigner(isFirstSigner: boolean): TransferBuilder {\n    this._isFirstSigner = isFirstSigner;\n    return this;\n  }\n\n  tokenContractAddress(tokenContractAddress: string): TransferBuilder {\n    this._tokenContractAddress = tokenContractAddress;\n    return this;\n  }\n\n  setCoinUsesNonPackedEncodingForTxData(isCoinUsesNonPackedEncodingForTxData: boolean): TransferBuilder {\n    this._coinUsesNonPackedEncodingForTxData = isCoinUsesNonPackedEncodingForTxData;\n    return this;\n  }\n\n  setSignature(signature: string): TransferBuilder {\n    this._signKey = null;\n    this._signature = ethUtil.addHexPrefix(signature);\n    return this;\n  }\n\n  signAndBuild(chainId: string, coinUsesNonPackedEncodingForTxData?: boolean): string {\n    this._chainId = chainId;\n\n    // If the coin uses non-packed encoding for tx data, the operation hash is calculated differently\n    // This new encoding type is applicable only for native coins and not tokens\n    this._coinUsesNonPackedEncodingForTxData =\n      coinUsesNonPackedEncodingForTxData && this._tokenContractAddress === undefined;\n    if (this.hasMandatoryFields()) {\n      if (this._isFirstSigner) {\n        // First signer signs different data than the second signer in multisig evm contracts.\n        return ethUtil.addHexPrefix(this.getSignatureData().toString('hex'));\n      } else {\n        if (this._tokenContractAddress !== undefined) {\n          return sendMultiSigTokenData(\n            this._toAddress,\n            this._amount,\n            this._tokenContractAddress,\n            this._expirationTime,\n            this._sequenceId,\n            this.getSignature()\n          );\n        } else {\n          return sendMultiSigData(\n            this._toAddress,\n            this._amount,\n            this._data,\n            this._expirationTime,\n            this._sequenceId,\n            this.getSignature()\n          );\n        }\n      }\n    }\n    throw new BuildTransactionError(\n      'Missing transfer mandatory fields. Amount, destination (to) address and sequenceID are mandatory'\n    );\n  }\n\n  private hasMandatoryFields(): boolean {\n    return this._amount !== undefined && this._toAddress !== undefined && this._sequenceId !== undefined;\n  }\n\n  /**\n   * Obtains the proper operation hash to sign either a sendMultiSig data\n   * or a sendMultiSigToken data\n   *\n   * @returns {string} the operation hash\n   */\n  public getOperationHash(): string {\n    const operationData = this.getOperationData();\n    let operationHash: string;\n\n    if (this._coinUsesNonPackedEncodingForTxData) {\n      const types: string[] = operationData[0] as string[];\n      const values: (string | number)[] = operationData[1].map((item) =>\n        typeof item === 'string' || typeof item === 'number' ? item : '0x' + item.toString('hex')\n      );\n      operationHash = keccak256(defaultAbiCoder.encode(types, values));\n    } else {\n      // If the coin uses packed encoding for tx data or it is a token, the operation hash is calculated using the Ethereum ABI\n      operationHash = ethUtil.bufferToHex(EthereumAbi.soliditySHA3(...operationData));\n    }\n    return operationHash;\n  }\n\n  protected getOperationData(): (string | number | Buffer)[][] {\n    let operationData;\n    const prefix = this.getOperationHashPrefix();\n    if (this._tokenContractAddress !== undefined) {\n      operationData = [\n        ['string', 'address', 'uint', 'address', 'uint', 'uint'],\n        [\n          prefix,\n          new BN(ethUtil.stripHexPrefix(this._toAddress), 16),\n          this._amount,\n          new BN(ethUtil.stripHexPrefix(this._tokenContractAddress), 16),\n          this._expirationTime,\n          this._sequenceId,\n        ],\n      ];\n    } else {\n      const toAddress = this._coinUsesNonPackedEncodingForTxData\n        ? this._toAddress\n        : new BN(ethUtil.stripHexPrefix(this._toAddress), 16);\n      operationData = [\n        ['string', 'address', 'uint', 'bytes', 'uint', 'uint'],\n        [\n          prefix,\n          toAddress,\n          this._amount,\n          Buffer.from(ethUtil.padToEven(ethUtil.stripHexPrefix(this._data)) || '', 'hex'),\n          this._expirationTime,\n          this._sequenceId,\n        ],\n      ];\n    }\n    return operationData;\n  }\n\n  private getOperationHashPrefix(): string {\n    if (this._walletVersion === 4) {\n      return this._tokenContractAddress ? `${this._chainId}-ERC20` : `${this._chainId}`;\n    }\n    return this._tokenContractAddress ? this.getTokenOperationHashPrefix() : this.getNativeOperationHashPrefix();\n  }\n\n  /**\n   * Get the prefix used in generating an operation hash for sending tokens\n   *\n   * @returns the string prefix\n   */\n  protected getTokenOperationHashPrefix(): string {\n    return (this._coin?.network as EthLikeNetwork)?.tokenOperationHashPrefix ?? `${this._chainId}-ERC20` ?? 'ERC20';\n  }\n\n  /**\n   * Get the prefix used in generating an operation hash for sending native coins\n   *\n   * @returns the string prefix\n   */\n  protected getNativeOperationHashPrefix(): string {\n    return (this._coin?.network as EthLikeNetwork)?.nativeCoinOperationHashPrefix ?? `${this._chainId}` ?? 'ETHER';\n  }\n\n  /** Return an expiration time, in seconds, set to one hour from now\n   *\n   * @returns {number} expiration time\n   */\n  private getExpirationTime(): number {\n    const currentDate = new Date();\n    currentDate.setHours(currentDate.getHours() + 1);\n    return currentDate.getTime() / 1000;\n  }\n\n  /**\n   * If a signing key is set for this builder, recalculates the signature\n   *\n   * @returns {string} the signature value\n   */\n  protected getSignature(): string {\n    if (this._signKey) {\n      this._signature = this.ethSignMsgHash();\n    }\n    return this._signature!;\n  }\n\n  protected ethSignMsgHash(): string {\n    const data = this.getOperationHash();\n    assert(this._signKey);\n    const keyBuffer = Buffer.from(ethUtil.padToEven(this._signKey), 'hex');\n    if (keyBuffer.length !== 32) {\n      throw new Error('private key length is invalid');\n    }\n    const signatureInParts = ethUtil.ecsign(\n      Buffer.from(ethUtil.padToEven(ethUtil.stripHexPrefix(data)), 'hex'),\n      keyBuffer\n    );\n\n    // Assemble strings from r, s and v\n    const r = ethUtil.setLengthLeft(signatureInParts.r, 32).toString('hex');\n    const s = ethUtil.setLengthLeft(signatureInParts.s, 32).toString('hex');\n    const v = ethUtil.stripHexPrefix(ethUtil.intToHex(signatureInParts.v));\n\n    // Concatenate the r, s and v parts to make the signature string\n    return ethUtil.addHexPrefix(r.concat(s, v));\n  }\n\n  private decodeTransferData(data: string): void {\n    const transferData = decodeTransferData(data, this._isFirstSigner);\n\n    this._toAddress = transferData.to;\n    this._amount = transferData.amount;\n    this._expirationTime = transferData.expireTime;\n    this._sequenceId = transferData.sequenceId;\n    this._signature = transferData.signature;\n\n    if (transferData.data) {\n      this._data = transferData.data;\n    }\n\n    if (transferData.tokenContractAddress) {\n      this._tokenContractAddress = transferData.tokenContractAddress;\n    }\n  }\n\n  public getSignatureData(): Buffer<ArrayBuffer> {\n    const method = this._tokenContractAddress\n      ? EthereumAbi.methodID('sendMultiSigToken', sendMultiSigTokenTypes)\n      : EthereumAbi.methodID('sendMultiSig', sendMultiSigTypes);\n    const operationData = this.getOperationData();\n    const rawEncodedOperationData = EthereumAbi.rawEncode(...operationData);\n    return Buffer.concat([\n      method,\n      rawEncodedOperationData,\n      Buffer.from([this._coinUsesNonPackedEncodingForTxData ? 1 : 0]),\n    ]);\n  }\n}\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!