PHP WebShell
Текущая директория: /usr/lib/node_modules/bitgo/node_modules/@bitgo/sdk-coin-avaxc/dist/src
Просмотр файла: avaxc.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.AvaxC = void 0;
/**
* @prettier
*/
const bignumber_js_1 = require("bignumber.js");
const secp256k1_1 = require("@bitgo/secp256k1");
const keccak_1 = __importDefault(require("keccak"));
const secp256k1 = __importStar(require("secp256k1"));
const _ = __importStar(require("lodash"));
const statics_1 = require("@bitgo/statics");
const sdk_core_1 = require("@bitgo/sdk-core");
const sdk_coin_eth_1 = require("@bitgo/sdk-coin-eth");
const utils_1 = require("./lib/utils");
const lib_1 = require("./lib");
const superagent_1 = __importDefault(require("superagent"));
const ethereumjs_util_1 = require("ethereumjs-util");
const buffer_1 = require("buffer");
const sdk_coin_avaxp_1 = require("@bitgo/sdk-coin-avaxp");
/** COIN-1708 : Avaxc is added for CCR in WRW,
* hence adding the feature for AbstractEthLikeNewCoins
* Super class changed from BaseCoin to AbstractEthLikeNewCoins
* @since Sept 2024
*/
class AvaxC extends sdk_coin_eth_1.AbstractEthLikeNewCoins {
constructor(bitgo, staticsCoin) {
super(bitgo, staticsCoin);
if (!staticsCoin) {
throw new Error('missing required constructor parameter staticsCoin');
}
this._staticsCoin = staticsCoin;
}
static createInstance(bitgo, staticsCoin) {
return new AvaxC(bitgo, staticsCoin);
}
getBaseFactor() {
return Math.pow(10, this._staticsCoin.decimalPlaces);
}
getChain() {
return this._staticsCoin.name;
}
/**
* Method to return the coin's network object
* @returns {BaseNetwork}
*/
getNetwork() {
return this._staticsCoin.network;
}
/**
* Get the base chain that the coin exists on.
*/
getBaseChain() {
return this.getChain();
}
getFamily() {
return this._staticsCoin.family;
}
getFullName() {
return this._staticsCoin.fullName;
}
valuelessTransferAllowed() {
return true;
}
isValidAddress(address) {
// also validate p-chain address for cross-chain txs
return !!address && ((0, utils_1.isValidEthAddress)(address) || sdk_coin_avaxp_1.AvaxpLib.Utils.isValidAddress(address));
}
isToken() {
return false;
}
/** inherited doc */
getDefaultMultisigType() {
return sdk_core_1.multisigTypes.onchain;
}
generateKeyPair(seed) {
const avaxKeyPair = seed ? new lib_1.KeyPair({ seed }) : new lib_1.KeyPair();
const extendedKeys = avaxKeyPair.getExtendedKeys();
return {
pub: extendedKeys.xpub,
prv: extendedKeys.xprv,
};
}
async parseTransaction(params) {
return {};
}
async verifyAddress({ address }) {
if (!this.isValidAddress(address)) {
throw new sdk_core_1.InvalidAddressError(`invalid address: ${address}`);
}
return true;
}
/**
* Verify that a transaction prebuild complies with the original intention
*
* @param params
* @param params.txParams params object passed to send
* @param params.txPrebuild prebuild object returned by server
* @param params.wallet Wallet object to obtain keys to verify against
* @returns {boolean}
*/
async verifyTransaction(params) {
const { txParams, txPrebuild, wallet } = params;
if (!txParams?.recipients || !txPrebuild?.recipients || !wallet) {
throw new Error(`missing params`);
}
if (txParams.hop && txParams.recipients.length > 1) {
throw new Error(`tx cannot be both a batch and hop transaction`);
}
if (txPrebuild.recipients.length > 1) {
throw new Error(`${this.getChain()} doesn't support sending to more than 1 destination address within a single transaction. Try again, using only a single recipient.`);
}
if (txParams.hop && txPrebuild.hopTransaction) {
// Check recipient amount for hop transaction
if (txParams.recipients.length !== 1) {
throw new Error(`hop transaction only supports 1 recipient but ${txParams.recipients.length} found`);
}
// Check tx sends to hop address
let expectedHopAddress;
if (txPrebuild.hopTransaction.type === 'Export') {
const decodedHopTx = await this.explainAtomicTransaction(txPrebuild.hopTransaction.tx);
expectedHopAddress = sdk_coin_eth_1.optionalDeps.ethUtil.stripHexPrefix(decodedHopTx.inputs[0].address);
}
else {
const decodedHopTx = sdk_coin_eth_1.optionalDeps.EthTx.TransactionFactory.fromSerializedData(sdk_coin_eth_1.optionalDeps.ethUtil.toBuffer(txPrebuild.hopTransaction.tx));
expectedHopAddress = sdk_coin_eth_1.optionalDeps.ethUtil.stripHexPrefix(decodedHopTx.getSenderAddress().toString());
}
const actualHopAddress = sdk_coin_eth_1.optionalDeps.ethUtil.stripHexPrefix(txPrebuild.recipients[0].address);
if (expectedHopAddress.toLowerCase() !== actualHopAddress.toLowerCase()) {
throw new Error('recipient address of txPrebuild does not match hop address');
}
// Convert TransactionRecipient array to Recipient array
const recipients = txParams.recipients.map((r) => {
return {
address: r.address,
amount: typeof r.amount === 'number' ? r.amount.toString() : r.amount,
};
});
// Check destination address and amount
await this.validateHopPrebuild(wallet, txPrebuild.hopTransaction, { recipients });
}
else if (txParams.recipients.length > 1) {
// Check total amount for batch transaction
let expectedTotalAmount = new bignumber_js_1.BigNumber(0);
for (let i = 0; i < txParams.recipients.length; i++) {
expectedTotalAmount = expectedTotalAmount.plus(txParams.recipients[i].amount);
}
if (!expectedTotalAmount.isEqualTo(txPrebuild.recipients[0].amount)) {
throw new Error('batch transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client');
}
}
else {
// Check recipient address and amount for normal transaction
if (txParams.recipients.length !== 1) {
throw new Error(`normal transaction only supports 1 recipient but ${txParams.recipients.length} found`);
}
const expectedAmount = new bignumber_js_1.BigNumber(txParams.recipients[0].amount);
if (!expectedAmount.isEqualTo(txPrebuild.recipients[0].amount)) {
throw new Error('normal transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client');
}
if (AvaxC.isAVAXCAddress(txParams.recipients[0].address) &&
txParams.recipients[0].address !== txPrebuild.recipients[0].address) {
throw new Error('destination address in normal txPrebuild does not match that in txParams supplied by client');
}
}
// Check coin is correct for all transaction types
if (!this.verifyCoin(txPrebuild)) {
throw new Error(`coin in txPrebuild did not match that in txParams supplied by client`);
}
return true;
}
static isAVAXCAddress(address) {
return !!address.match(/0x[a-fA-F0-9]{40}/);
}
verifyCoin(txPrebuild) {
return txPrebuild.coin === this.getChain();
}
isValidPub(pub) {
let valid = true;
try {
new lib_1.KeyPair({ pub });
}
catch (e) {
valid = false;
}
return valid;
}
/**
* Check whether gas limit passed in by user are within our max and min bounds
* If they are not set, set them to the defaults
* @param {number} userGasLimit - user defined gas limit
* @returns {number} the gas limit to use for this transaction
*/
setGasLimit(userGasLimit) {
if (!userGasLimit) {
return statics_1.ethGasConfigs.defaultGasLimit;
}
const gasLimitMax = statics_1.ethGasConfigs.maximumGasLimit;
const gasLimitMin = statics_1.ethGasConfigs.minimumGasLimit;
if (userGasLimit < gasLimitMin || userGasLimit > gasLimitMax) {
throw new Error(`Gas limit must be between ${gasLimitMin} and ${gasLimitMax}`);
}
return userGasLimit;
}
/**
* Check whether the gas price passed in by user are within our max and min bounds
* If they are not set, set them to the defaults
* @param {number} userGasPrice - user defined gas price
* @returns the gas price to use for this transaction
*/
setGasPrice(userGasPrice) {
if (!userGasPrice) {
return statics_1.ethGasConfigs.defaultGasPrice;
}
const gasPriceMax = statics_1.ethGasConfigs.maximumGasPrice;
const gasPriceMin = statics_1.ethGasConfigs.minimumGasPrice;
if (userGasPrice < gasPriceMin || userGasPrice > gasPriceMax) {
throw new Error(`Gas price must be between ${gasPriceMin} and ${gasPriceMax}`);
}
return userGasPrice;
}
/**
* Make a query to avax.network for information such as balance, token balance, solidity calls
* @param {Object} query — key-value pairs of parameters to append after /api
* @param {string} apiKey - optional API key to use instead of the one from the environment
* @returns {Promise<Object>} response from avax.network
*/
async recoveryBlockchainExplorerQuery(query, apiKey) {
const response = await superagent_1.default
.post(sdk_core_1.common.Environments[this.bitgo.getEnv()].avaxcNetworkBaseUrl + '/ext/bc/C/rpc')
.send(query);
if (!response.ok) {
throw new Error('could not reach avax.network');
}
if (response.body.status === '0' && response.body.message === 'NOTOK') {
throw new Error('avax.network rate limit reached');
}
return response.body;
}
/**
* Queries public block explorer to get the next nonce that should be used for
* the given AVAXC address
* @param {string} address — address to fetch for
* @returns {number} address nonce
*/
async getAddressNonce(address) {
// Get nonce for backup key (should be 0)
const result = await this.recoveryBlockchainExplorerQuery({
jsonrpc: '2.0',
method: 'eth_getTransactionCount',
params: [address, 'latest'],
id: 1,
});
if (!result || isNaN(result.result)) {
throw new Error('Unable to find next nonce from avax.network, got: ' + JSON.stringify(result));
}
const nonceHex = result.result;
return new sdk_coin_eth_1.optionalDeps.ethUtil.BN(nonceHex.slice(2), 16).toNumber();
}
/**
* Queries avax.network for the balance of an address
* @param {string} address - the AVAXC address
* @returns {Promise<BigNumber>} address balance
*/
async queryAddressBalance(address) {
const result = await this.recoveryBlockchainExplorerQuery({
jsonrpc: '2.0',
method: 'eth_getBalance',
params: [address, 'latest'],
id: 1,
});
// throw if the result does not exist or the result is not a valid number
if (!result || !result.result || isNaN(result.result)) {
throw new Error(`Could not obtain address balance for ${address} from avax.network, got: ${result.result}`);
}
const nativeBalanceHex = result.result;
return new sdk_coin_eth_1.optionalDeps.ethUtil.BN(nativeBalanceHex.slice(2), 16);
}
/**
* Queries avax.network for the token balance of an address
* @param {string} walletContractAddress - the AVAXC address
* @param {string} tokenContractAddress - the Token contract address
* @returns {Promise<BigNumber>} address balance
*/
async queryAddressTokenBalance(tokenContractAddress, walletContractAddress) {
// get token balance using contract call
const tokenBalanceData = sdk_coin_eth_1.optionalDeps.ethAbi
.simpleEncode('balanceOf(address)', walletContractAddress)
.toString('hex');
const tokenBalanceDataHex = sdk_coin_eth_1.optionalDeps.ethUtil.addHexPrefix(tokenBalanceData);
const result = await this.recoveryBlockchainExplorerQuery({
jsonrpc: '2.0',
method: 'eth_call',
params: [
{
to: tokenContractAddress,
data: tokenBalanceDataHex,
},
'latest',
],
id: 1,
});
// throw if the result does not exist or the result is not a valid number
if (!result || !result.result || isNaN(result.result)) {
throw new Error(`Could not obtain address token balance for ${walletContractAddress} from avax.network, got: ${result.result}`);
}
const tokenBalanceHex = result.result;
return new sdk_coin_eth_1.optionalDeps.ethUtil.BN(tokenBalanceHex.slice(2), 16);
}
/**
* Queries the contract (via avax.network) for the next sequence ID
* @param {string} address - address of the contract
* @returns {Promise<number>} sequence ID
*/
async querySequenceId(address) {
// Get sequence ID using contract call
const sequenceIdMethodSignature = sdk_coin_eth_1.optionalDeps.ethAbi.methodID('getNextSequenceId', []);
const sequenceIdArgs = sdk_coin_eth_1.optionalDeps.ethAbi.rawEncode([], []);
const sequenceIdData = buffer_1.Buffer.concat([sequenceIdMethodSignature, sequenceIdArgs]).toString('hex');
const sequenceIdDataHex = sdk_coin_eth_1.optionalDeps.ethUtil.addHexPrefix(sequenceIdData);
const result = await this.recoveryBlockchainExplorerQuery({
jsonrpc: '2.0',
method: 'eth_call',
params: [{ to: address, data: sequenceIdDataHex }, 'latest'],
id: 1,
});
if (!result || !result.result) {
throw new Error('Could not obtain sequence ID from avax.network, got: ' + result.result);
}
const sequenceIdHex = result.result;
return new sdk_coin_eth_1.optionalDeps.ethUtil.BN(sequenceIdHex.slice(2), 16).toNumber();
}
/**
* @param {Object} recipient - recipient info
* @param {number} expireTime - expiry time
* @param {number} contractSequenceId - sequence id
* @returns {(string|Array)} operation array
*/
getOperation(recipient, expireTime, contractSequenceId) {
return [
['string', 'address', 'uint', 'bytes', 'uint', 'uint'],
[
'ETHER',
new sdk_coin_eth_1.optionalDeps.ethUtil.BN(sdk_coin_eth_1.optionalDeps.ethUtil.stripHexPrefix(recipient.address), 16),
recipient.amount,
buffer_1.Buffer.from(sdk_coin_eth_1.optionalDeps.ethUtil.stripHexPrefix(sdk_coin_eth_1.optionalDeps.ethUtil.padToEven(recipient.data || '')), 'hex'),
expireTime,
contractSequenceId,
],
];
}
/**
* Calculate the operation hash in the same way solidity would
* @param {Recipient[]} recipients - tx recipients
* @param {number} expireTime - expiration time
* @param {number} contractSequenceId - contract sequence id
* @returns {string} operation hash
*/
getOperationSha3ForExecuteAndConfirm(recipients, expireTime, contractSequenceId) {
if (!recipients || !Array.isArray(recipients)) {
throw new Error('expecting array of recipients');
}
// Right now we only support 1 recipient
if (recipients.length !== 1) {
throw new Error('must send to exactly 1 recipient');
}
if (!_.isNumber(expireTime)) {
throw new Error('expireTime must be number of seconds since epoch');
}
if (!_.isNumber(contractSequenceId)) {
throw new Error('contractSequenceId must be number');
}
// Check inputs
recipients.forEach(function (recipient) {
if (!_.isString(recipient.address) ||
!sdk_coin_eth_1.optionalDeps.ethUtil.isValidAddress(sdk_coin_eth_1.optionalDeps.ethUtil.addHexPrefix(recipient.address))) {
throw new Error('Invalid address: ' + recipient.address);
}
let amount;
try {
amount = new bignumber_js_1.BigNumber(recipient.amount);
}
catch (e) {
throw new Error('Invalid amount for: ' + recipient.address + ' - should be numeric');
}
recipient.amount = amount.toFixed(0);
if (recipient.data && !_.isString(recipient.data)) {
throw new Error('Data for recipient ' + recipient.address + ' - should be of type hex string');
}
});
const recipient = recipients[0];
return sdk_coin_eth_1.optionalDeps.ethUtil.bufferToHex(sdk_coin_eth_1.optionalDeps.ethAbi.soliditySHA3(...this.getOperation(recipient, expireTime, contractSequenceId)));
}
/**
* Default expire time for a contract call (1 week)
* @returns {number} Time in seconds
*/
getDefaultExpireTime() {
return Math.floor(new Date().getTime() / 1000) + 60 * 60 * 24 * 7;
}
/**
* Build arguments to call the send method on the wallet contract
* @param {Object} txInfo - data for send method args
* @returns {SendMethodArgs[]}
*/
getSendMethodArgs(txInfo) {
// Method signature is
// sendMultiSig(address toAddress, uint value, bytes data, uint expireTime, uint sequenceId, bytes signature)
return [
{
name: 'toAddress',
type: 'address',
value: txInfo.recipient.address,
},
{
name: 'value',
type: 'uint',
value: txInfo.recipient.amount,
},
{
name: 'data',
type: 'bytes',
value: sdk_coin_eth_1.optionalDeps.ethUtil.toBuffer(sdk_coin_eth_1.optionalDeps.ethUtil.addHexPrefix(txInfo.recipient.data || '')),
},
{
name: 'expireTime',
type: 'uint',
value: txInfo.expireTime,
},
{
name: 'sequenceId',
type: 'uint',
value: txInfo.contractSequenceId,
},
{
name: 'signature',
type: 'bytes',
value: sdk_coin_eth_1.optionalDeps.ethUtil.toBuffer(sdk_coin_eth_1.optionalDeps.ethUtil.addHexPrefix(txInfo.signature)),
},
];
}
/**
* Builds a funds recovery transaction without BitGo
* Steps:
* 1) Node query - how much money is in the account
* 2) Build transaction - build our transaction for the amount
* 3) Send signed build - send our signed build to a public node
* @param {Object} params The options with which to recover
* @param {string} params.userKey - [encrypted] xprv
* @param {string} params.backupKey - [encrypted] xprv or xpub if the xprv is held by a KRS provider
* @param {string} params.walletPassphrase - used to decrypt userKey and backupKey
* @param {string} params.walletContractAddress - the AVAXC address of the wallet contract
* @param {string} params.recoveryDestination - target address to send recovered funds to
* @returns {Promise<RecoveryInfo>} - recovery tx info
*/
async recover(params) {
if (params.bitgoFeeAddress) {
return (await this.recoverEthLikeforEvmBasedRecovery(params));
}
if (_.isUndefined(params.userKey)) {
throw new Error('missing userKey');
}
if (_.isUndefined(params.backupKey)) {
throw new Error('missing backupKey');
}
if (_.isUndefined(params.walletPassphrase) && !params.userKey.startsWith('xpub')) {
throw new Error('missing wallet passphrase');
}
if (_.isUndefined(params.walletContractAddress) || !this.isValidAddress(params.walletContractAddress)) {
throw new Error('invalid walletContractAddress');
}
let tokenName;
if (params.tokenContractAddress) {
if (!this.isValidAddress(params.tokenContractAddress)) {
throw new Error('invalid tokenContractAddress');
}
const network = this.getNetwork();
const token = (0, utils_1.getToken)(params.tokenContractAddress, network);
if (_.isUndefined(token)) {
throw new Error('token not supported');
}
tokenName = token.name;
}
if (_.isUndefined(params.recoveryDestination) || !this.isValidAddress(params.recoveryDestination)) {
throw new Error('invalid recoveryDestination');
}
// TODO (BG-56531): add support for krs
const isUnsignedSweep = (0, sdk_core_1.getIsUnsignedSweep)(params);
// Clean up whitespace from entered values
let userKey = params.userKey.replace(/\s/g, '');
const backupKey = params.backupKey.replace(/\s/g, '');
// Set new tx fees (using default config values from platform)
const gasLimit = new sdk_coin_eth_1.optionalDeps.ethUtil.BN(this.setGasLimit(params.gasLimit));
const gasPrice = params.eip1559
? new sdk_coin_eth_1.optionalDeps.ethUtil.BN(params.eip1559.maxFeePerGas)
: new sdk_coin_eth_1.optionalDeps.ethUtil.BN(this.setGasPrice(params.gasPrice));
if (!userKey.startsWith('xpub') && !userKey.startsWith('xprv')) {
try {
userKey = this.bitgo.decrypt({
input: userKey,
password: params.walletPassphrase,
});
}
catch (e) {
throw new Error(`Error decrypting user keychain: ${e.message}`);
}
}
let backupKeyAddress;
let backupSigningKey;
if (isUnsignedSweep) {
const backupKeyPair = new lib_1.KeyPair({ pub: backupKey });
backupKeyAddress = backupKeyPair.getAddress();
}
else {
// Decrypt backup private key and get address
let backupPrv;
try {
backupPrv = this.bitgo.decrypt({
input: backupKey,
password: params.walletPassphrase,
});
}
catch (e) {
throw new Error(`Error decrypting backup keychain: ${e.message}`);
}
const keyPair = new lib_1.KeyPair({ prv: backupPrv });
backupSigningKey = keyPair.getKeys().prv;
if (!backupSigningKey) {
throw new Error('no private key');
}
backupKeyAddress = keyPair.getAddress();
}
const backupKeyNonce = await this.getAddressNonce(backupKeyAddress);
// get balance of backupKey to ensure funds are available to pay fees
const backupKeyBalance = await this.queryAddressBalance(backupKeyAddress);
const totalGasNeeded = gasPrice.mul(gasLimit);
const weiToGwei = 10 ** 9;
if (backupKeyBalance.lt(totalGasNeeded)) {
throw new Error(`Backup key address ${backupKeyAddress} has balance ${backupKeyBalance
.div(new ethereumjs_util_1.BN(weiToGwei))
.toString()} Gwei.` +
`This address must have a balance of at least ${(totalGasNeeded / weiToGwei).toString()}` +
` Gwei to perform recoveries. Try sending some AVAX to this address then retry.`);
}
let txAmount;
if (params.tokenContractAddress) {
// get token balance of wallet
txAmount = await this.queryAddressTokenBalance(params.tokenContractAddress, params.walletContractAddress);
}
else {
// get balance of wallet and deduct fees to get transaction amount
txAmount = await this.queryAddressBalance(params.walletContractAddress);
}
// build recipients object
const recipients = [
{
address: params.recoveryDestination,
amount: txAmount.toString(10),
},
];
// Get sequence ID using contract call
// we need to wait between making two avax.network calls to avoid getting banned
await new Promise((resolve) => setTimeout(resolve, 1000));
const sequenceId = await this.querySequenceId(params.walletContractAddress);
let operationHash, signature;
// Get operation hash and sign it
if (!isUnsignedSweep) {
operationHash = this.getOperationSha3ForExecuteAndConfirm(recipients, this.getDefaultExpireTime(), sequenceId);
signature = sdk_core_1.Util.ethSignMsgHash(operationHash, sdk_core_1.Util.xprvToEthPrivateKey(userKey));
try {
sdk_core_1.Util.ecRecoverEthAddress(operationHash, signature);
}
catch (e) {
throw new Error('Invalid signature');
}
}
const txInfo = {
recipient: recipients[0],
expireTime: this.getDefaultExpireTime(),
contractSequenceId: sequenceId,
operationHash,
signature,
gasLimit: gasLimit.toString(10),
tokenContractAddress: params.tokenContractAddress,
};
const txBuilder = this.getTransactionBuilder();
txBuilder.counter(backupKeyNonce);
txBuilder.contract(params.walletContractAddress);
let txFee;
if (params.eip1559) {
txFee = {
eip1559: {
maxPriorityFeePerGas: params.eip1559.maxPriorityFeePerGas,
maxFeePerGas: params.eip1559.maxFeePerGas,
},
};
}
else {
txFee = { fee: gasPrice.toString() };
}
txBuilder.fee({
...txFee,
gasLimit: gasLimit.toString(),
});
if (params.tokenContractAddress) {
txBuilder
.transfer()
.coin(tokenName)
.amount(recipients[0].amount)
.contractSequenceId(sequenceId)
.expirationTime(this.getDefaultExpireTime())
.to(params.recoveryDestination);
}
else {
txBuilder
.transfer()
.coin(this.getChain())
.amount(recipients[0].amount)
.contractSequenceId(sequenceId)
.expirationTime(this.getDefaultExpireTime())
.to(params.recoveryDestination);
}
if (isUnsignedSweep) {
const tx = await txBuilder.build();
const response = {
txHex: tx.toBroadcastFormat(),
userKey,
backupKey,
coin: this.getChain(),
token: tokenName,
gasPrice: sdk_coin_eth_1.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
gasLimit,
recipients: [txInfo.recipient],
walletContractAddress: tx.toJson().to,
amount: txInfo.recipient.amount,
backupKeyNonce,
eip1559: params.eip1559,
};
_.extend(response, txInfo);
response.nextContractSequenceId = response.contractSequenceId;
return response;
}
const userKeyPair = new lib_1.KeyPair({ prv: userKey });
txBuilder.transfer().key(userKeyPair.getKeys().prv);
txBuilder.sign({ key: backupSigningKey });
const signedTx = await txBuilder.build();
return {
id: signedTx.toJson().id,
tx: signedTx.toBroadcastFormat(),
};
}
/**
* Create a new transaction builder for the current chain
* @return a new transaction builder
*/
getTransactionBuilder() {
return new lib_1.TransactionBuilder(statics_1.coins.get(this.getBaseChain()));
}
getAtomicBuilder() {
return new sdk_coin_avaxp_1.AvaxpLib.TransactionBuilderFactory(statics_1.coins.get(this.getAvaxP()));
}
/**
* Explain a transaction from txHex, overriding BaseCoins
* transaction can be either atomic or eth txn.
* @param params The options with which to explain the transaction
*/
async explainTransaction(params) {
const txHex = params.txHex || (params.halfSigned && params.halfSigned.txHex);
if (!txHex) {
throw new Error('missing txHex in explain tx parameters');
}
if (params.crossChainType) {
return this.explainAtomicTransaction(txHex);
}
if (!params.feeInfo) {
throw new Error('missing feeInfo in explain tx parameters');
}
const txBuilder = this.getTransactionBuilder();
txBuilder.from(txHex);
const tx = await txBuilder.build();
return Object.assign(this.explainEVMTransaction(tx), { fee: params.feeInfo });
}
/**
* Explains an atomic transaction using atomic builder.
* @param txHex
* @private
*/
async explainAtomicTransaction(txHex) {
const txBuilder = this.getAtomicBuilder().from(txHex);
const tx = await txBuilder.build();
return tx.explainTransaction();
}
/**
* Verify signature for an atomic transaction using atomic builder.
* @param txHex
* @return true if signature is from the input address
* @private
*/
async verifySignatureForAtomicTransaction(txHex) {
const txBuilder = this.getAtomicBuilder().from(txHex);
const tx = await txBuilder.build();
const payload = tx.signablePayload;
const signatures = tx.signature.map((s) => buffer_1.Buffer.from(s, 'hex'));
const network = _.get(tx, '_network');
const recoverPubky = signatures.map((s) => sdk_coin_avaxp_1.AvaxpLib.Utils.recoverySignature(network, payload, s));
const expectedSenders = recoverPubky.map((r) => (0, ethereumjs_util_1.pubToAddress)(r, true));
const senders = tx.inputs.map((i) => sdk_coin_avaxp_1.AvaxpLib.Utils.parseAddress(i.address));
return expectedSenders.every((e) => senders.some((sender) => e.equals(sender)));
}
/**
* Explains an EVM transaction using regular eth txn builder
* @param tx
* @private
*/
explainEVMTransaction(tx) {
const outputs = tx.outputs.map((output) => {
return {
address: output.address,
amount: output.value,
};
});
const displayOrder = ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee'];
return {
displayOrder,
id: tx.id,
outputs: outputs,
outputAmount: outputs
.reduce((accumulator, output) => accumulator.plus(output.amount), new bignumber_js_1.BigNumber('0'))
.toFixed(0),
changeOutputs: [], // account based does not use change outputs
changeAmount: '0', // account base does not make change
};
}
/**
* Above is standard BaseCoins functions
* ================================================================================================================
* ================================================================================================================
* Below is transaction functions
*/
/**
* Coin-specific things done before signing a transaction, i.e. verification
* @param params
*/
async presignTransaction(params) {
if (!_.isUndefined(params.hopTransaction) && !_.isUndefined(params.wallet) && !_.isUndefined(params.buildParams)) {
await this.validateHopPrebuild(params.wallet, params.hopTransaction);
}
return params;
}
/**
* Modify prebuild after receiving it from the server. Add things like nlocktime
*/
async postProcessPrebuild(params) {
if (!_.isUndefined(params.hopTransaction) && !_.isUndefined(params.wallet) && !_.isUndefined(params.buildParams)) {
await this.validateHopPrebuild(params.wallet, params.hopTransaction, params.buildParams);
}
return params;
}
/**
* Validates that the hop prebuild from the HSM is valid and correct
* @param wallet The wallet that the prebuild is for
* @param hopPrebuild The prebuild to validate
* @param originalParams The original parameters passed to prebuildTransaction
* @returns void
* @throws Error if The prebuild is invalid
*/
async validateHopPrebuild(wallet, hopPrebuild, originalParams) {
const { tx, id, signature } = hopPrebuild;
// first, validate the HSM signature
const serverXpub = sdk_core_1.common.Environments[this.bitgo.getEnv()].hsmXpub;
const serverPubkeyBuffer = secp256k1_1.bip32.fromBase58(serverXpub).publicKey;
const signatureBuffer = buffer_1.Buffer.from(sdk_coin_eth_1.optionalDeps.ethUtil.stripHexPrefix(signature), 'hex');
const messageBuffer = hopPrebuild.type === 'Export' ? AvaxC.getTxHash(tx) : buffer_1.Buffer.from(sdk_coin_eth_1.optionalDeps.ethUtil.stripHexPrefix(id), 'hex');
const sig = new Uint8Array(signatureBuffer.length === 64 ? signatureBuffer : signatureBuffer.slice(1));
const isValidSignature = secp256k1.ecdsaVerify(sig, messageBuffer, serverPubkeyBuffer);
if (!isValidSignature) {
throw new Error(`Hop txid signature invalid`);
}
if (hopPrebuild.type === 'Export') {
const explainHopExportTx = await this.explainAtomicTransaction(tx);
// If original params are given, we can check them against the transaction prebuild params
if (!_.isNil(originalParams)) {
const { recipients } = originalParams;
// Then validate that the tx params actually equal the requested params to nano avax plus import tx fee.
const originalAmount = new bignumber_js_1.BigNumber(recipients[0].amount).div(1e9).plus(1e6).toFixed(0);
const originalDestination = recipients[0].address;
const hopAmount = explainHopExportTx.outputAmount;
const hopDestination = explainHopExportTx.outputs[0].address;
if (originalAmount !== hopAmount) {
throw new Error(`Hop amount: ${hopAmount} does not equal original amount: ${originalAmount}`);
}
if (originalDestination && hopDestination.toLowerCase() !== originalDestination.toLowerCase()) {
throw new Error(`Hop destination: ${hopDestination} does not equal original recipient: ${originalDestination}`);
}
}
if (!(await this.verifySignatureForAtomicTransaction(tx))) {
throw new Error(`Invalid hop transaction signature, txid: ${id}`);
}
}
else {
const builtHopTx = sdk_coin_eth_1.optionalDeps.EthTx.TransactionFactory.fromSerializedData(sdk_coin_eth_1.optionalDeps.ethUtil.toBuffer(tx));
// If original params are given, we can check them against the transaction prebuild params
if (!_.isNil(originalParams)) {
const { recipients } = originalParams;
// Then validate that the tx params actually equal the requested params
const originalAmount = new bignumber_js_1.BigNumber(recipients[0].amount);
const originalDestination = recipients[0].address;
const hopAmount = new bignumber_js_1.BigNumber(sdk_coin_eth_1.optionalDeps.ethUtil.bufferToHex(builtHopTx.value));
if (!builtHopTx.to) {
throw new Error(`Transaction does not have a destination address`);
}
const hopDestination = builtHopTx.to.toString();
if (!hopAmount.eq(originalAmount)) {
throw new Error(`Hop amount: ${hopAmount} does not equal original amount: ${originalAmount}`);
}
if (hopDestination.toLowerCase() !== originalDestination.toLowerCase()) {
throw new Error(`Hop destination: ${hopDestination} does not equal original recipient: ${hopDestination}`);
}
}
if (!builtHopTx.verifySignature()) {
// We dont want to continue at all in this case, at risk of AVAX being stuck on the hop address
throw new Error(`Invalid hop transaction signature, txid: ${id}`);
}
if (sdk_coin_eth_1.optionalDeps.ethUtil.addHexPrefix(builtHopTx.hash().toString('hex')) !== id) {
throw new Error(`Signed hop txid does not equal actual txid`);
}
}
}
/**
* Helper function for signTransaction for the rare case that SDK is doing the second signature
* Note: we are expecting this to be called from the offline vault
* @param params.txPrebuild
* @param params.prv
* @returns {{txHex: string}}
*/
async signFinal(params) {
const keyPair = new lib_1.KeyPair({ prv: params.prv });
const signingKey = keyPair.getKeys().prv;
if (_.isUndefined(signingKey)) {
throw new Error('missing private key');
}
const txBuilder = this.getTransactionBuilder();
try {
txBuilder.from(params.txPrebuild.halfSigned.txHex);
}
catch (e) {
throw new Error('invalid half-signed transaction');
}
txBuilder.sign({ key: signingKey });
const tx = await txBuilder.build();
return {
txHex: tx.toBroadcastFormat(),
};
}
/**
* Assemble half-sign prebuilt transaction
* @param params
*/
async signTransaction(params) {
// Normally the SDK provides the first signature for an AVAXC tx,
// but for unsigned sweep recoveries it can provide the second and final one.
if (params.isLastSignature) {
// In this case when we're doing the second (final) signature, the logic is different.
return await this.signFinal(params);
}
const txBuilder = this.getTransactionBuilder();
txBuilder.from(params.txPrebuild.txHex);
txBuilder.transfer().key(new lib_1.KeyPair({ prv: params.prv }).getKeys().prv);
if (params.walletVersion) {
txBuilder.walletVersion(params.walletVersion);
}
const transaction = await txBuilder.build();
// we need to preserve the calldata of the recipients specified in the request for custodial transactions
let recipients = params.txPrebuild.recipients || params.recipients;
if (recipients === undefined) {
recipients = transaction.outputs.map((output) => ({ address: output.address, amount: output.value }));
}
const txParams = {
eip1559: params.txPrebuild.eip1559,
txHex: transaction.toBroadcastFormat(),
recipients: recipients,
expireTime: params.txPrebuild.expireTime,
hopTransaction: params.txPrebuild.hopTransaction,
custodianTransactionId: params.custodianTransactionId,
};
return { halfSigned: txParams };
}
/**
* Modify prebuild before sending it to the server. Add things like hop transaction params
* @param buildParams The whitelisted parameters for this prebuild
* @param buildParams.hop True if this should prebuild a hop tx, else false
* @param buildParams.recipients The recipients array of this transaction
* @param buildParams.wallet The wallet sending this tx
* @param buildParams.walletPassphrase the passphrase for this wallet
*/
async getExtraPrebuildParams(buildParams) {
if (!_.isUndefined(buildParams.hop) &&
buildParams.hop &&
!_.isUndefined(buildParams.wallet) &&
!_.isUndefined(buildParams.recipients)) {
if (this.isToken()) {
throw new Error(`Hop transactions are not enabled for AVAXC tokens, nor are they necessary. Please remove the 'hop' parameter and try again.`);
}
return (await this.createHopTransactionParams({
recipients: buildParams.recipients,
type: buildParams.type,
}));
}
return {};
}
/**
* Creates the extra parameters needed to build a hop transaction
* @param {HopTransactionBuildOptions} The original build parameters
* @returns extra parameters object to merge with the original build parameters object and send to the platform
*/
async createHopTransactionParams({ recipients, type }) {
if (!recipients || !Array.isArray(recipients)) {
throw new Error('expecting array of recipients');
}
// Right now we only support 1 recipient
if (recipients.length !== 1) {
throw new Error('must send to exactly 1 recipient');
}
const recipientAddress = recipients[0].address;
const recipientAmount = recipients[0].amount;
const feeEstimateParams = {
recipient: recipientAddress,
amount: recipientAmount,
hop: true,
type,
};
const feeEstimate = await this.feeEstimate(feeEstimateParams);
const gasLimit = feeEstimate.gasLimitEstimate;
const gasPrice = Math.round(feeEstimate.feeEstimate / gasLimit);
const gasPriceMax = gasPrice * 5;
// Payment id a random number so its different for every tx
const paymentId = Math.floor(Math.random() * 10000000000).toString();
// TODO(BG-62671): after completed [Wallet-platform] Remove use of userReqSig for avaxc hop transaction
const userReqSig = '0x';
return {
hopParams: {
userReqSig,
gasPriceMax,
paymentId,
gasLimit,
},
};
}
/**
* Fetch fee estimate information from the server
* @param {Object} params The params passed into the function
* @param {Boolean} [params.hop] True if we should estimate fee for a hop transaction
* @param {String} [params.recipient] The recipient of the transaction to estimate a send to
* @param {String} [params.data] The ETH tx data to estimate a send for
* @returns {Object} The fee info returned from the server
*/
async feeEstimate(params) {
const query = {};
if (params && params.hop) {
query.hop = params.hop;
}
if (params && params.recipient) {
query.recipient = params.recipient;
}
if (params && params.data) {
query.data = params.data;
}
if (params && params.amount) {
query.amount = params.amount;
}
if (params && params.type) {
query.type = params.type;
}
return await this.bitgo.get(this.url('/tx/fee')).query(query).result();
}
/**
* Calculate tx hash like evm from tx hex.
* @param {string} tx
* @returns {Buffer} tx hash
*/
static getTxHash(tx) {
const hash = (0, keccak_1.default)('keccak256');
hash.update(sdk_coin_eth_1.optionalDeps.ethUtil.stripHexPrefix(tx), 'hex');
return hash.digest();
}
async isWalletAddress(params) {
// TODO: Fix this later
return true;
}
/**
* Ensure either enterprise or newFeeAddress is passed, to know whether to create new key or use enterprise key
* @param params
* @param params.enterprise {String} the enterprise id to associate with this key
* @param params.newFeeAddress {Boolean} create a new fee address (enterprise not needed in this case)
*/
preCreateBitGo(params) {
// We always need params object, since either enterprise or newFeeAddress is required
if (!_.isObject(params)) {
throw new Error(`preCreateBitGo must be passed a params object. Got ${params} (type ${typeof params})`);
}
if (_.isUndefined(params.enterprise) && _.isUndefined(params.newFeeAddress)) {
throw new Error('expecting enterprise when adding BitGo key. If you want to create a new AVAX bitgo key, set the newFeeAddress parameter to true.');
}
// Check whether key should be an enterprise key or a BitGo key for a new fee address
if (!_.isUndefined(params.enterprise) && !_.isUndefined(params.newFeeAddress)) {
throw new Error(`Incompatible arguments - cannot pass both enterprise and newFeeAddress parameter.`);
}
if (!_.isUndefined(params.enterprise) && !_.isString(params.enterprise)) {
throw new Error(`enterprise should be a string - got ${params.enterprise} (type ${typeof params.enterprise})`);
}
if (!_.isUndefined(params.newFeeAddress) && !_.isBoolean(params.newFeeAddress)) {
throw new Error(`newFeeAddress should be a boolean - got ${params.newFeeAddress} (type ${typeof params.newFeeAddress})`);
}
}
getAvaxP() {
return this.getChain().toString() === 'avaxc' ? 'avaxp' : 'tavaxp';
}
/**
* Fetch the gas price from the explorer
*/
async getGasPriceFromExternalAPI() {
try {
const res = await this.recoveryBlockchainExplorerQuery({
jsonrpc: '2.0',
method: 'eth_gasPrice',
id: 1,
});
const gasPrice = new ethereumjs_util_1.BN(res.result.slice(2), 16);
console.log(` Got gas price: ${gasPrice}`);
return gasPrice;
}
catch (e) {
throw new Error('Failed to get gas price');
}
}
/**
* Fetch the gas limit from the explorer
* @param intendedChain
* @param from
* @param to
* @param data
*/
async getGasLimitFromExternalAPI(intendedChain, from, to, data) {
try {
const res = await this.recoveryBlockchainExplorerQuery({
jsonrpc: '2.0',
method: 'eth_estimateGas',
params: [
{
from,
to,
data,
},
'latest',
],
id: 1,
});
const gasLimit = new ethereumjs_util_1.BN(res.result.slice(2), 16);
console.log(`Got gas limit: ${gasLimit}`);
return gasLimit;
}
catch (e) {
throw new Error(`Failed to get gas limit. Please make sure to use the privateKey aka userKey of ${intendedChain} wallet ${to}`);
}
}
}
exports.AvaxC = AvaxC;
AvaxC.hopTransactionSalt = 'bitgoHopAddressRequestSalt';
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"avaxc.js","sourceRoot":"","sources":["../../src/avaxc.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;;GAEG;AACH,+CAAyC;AACzC,gDAAyC;AACzC,oDAA4B;AAC5B,qDAAuC;AACvC,0CAA4B;AAC5B,4CAOwB;AACxB,8CAmByB;AACzB,sDAS6B;AAC7B,uCAA0D;AAC1D,+BAAoE;AACpE,4DAAiC;AACjC,qDAAmD;AACnD,mCAAgC;AAgBhC,0DAAiD;AAGjD;;;;GAIG;AACH,MAAa,KAAM,SAAQ,sCAAuB;IAKhD,YAAsB,KAAgB,EAAE,WAAuC;QAC7E,KAAK,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC;QAE1B,IAAI,CAAC,WAAW,EAAE,CAAC;YACjB,MAAM,IAAI,KAAK,CAAC,oDAAoD,CAAC,CAAC;QACxE,CAAC;QAED,IAAI,CAAC,YAAY,GAAG,WAAW,CAAC;IAClC,CAAC;IAED,MAAM,CAAC,cAAc,CAAC,KAAgB,EAAE,WAAuC;QAC7E,OAAO,IAAI,KAAK,CAAC,KAAK,EAAE,WAAW,CAAC,CAAC;IACvC,CAAC;IAED,aAAa;QACX,OAAO,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE,IAAI,CAAC,YAAY,CAAC,aAAa,CAAC,CAAC;IACvD,CAAC;IAED,QAAQ;QACN,OAAO,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC;IAChC,CAAC;IAED;;;OAGG;IACH,UAAU;QACR,OAAO,IAAI,CAAC,YAAY,CAAC,OAA0B,CAAC;IACtD,CAAC;IAED;;OAEG;IACH,YAAY;QACV,OAAO,IAAI,CAAC,QAAQ,EAAE,CAAC;IACzB,CAAC;IAED,SAAS;QACP,OAAO,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC;IAClC,CAAC;IAED,WAAW;QACT,OAAO,IAAI,CAAC,YAAY,CAAC,QAAQ,CAAC;IACpC,CAAC;IAED,wBAAwB;QACtB,OAAO,IAAI,CAAC;IACd,CAAC;IAED,cAAc,CAAC,OAAe;QAC5B,oDAAoD;QACpD,OAAO,CAAC,CAAC,OAAO,IAAI,CAAC,IAAA,yBAAiB,EAAC,OAAO,CAAC,IAAI,yBAAQ,CAAC,KAAK,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC,CAAC;IAC7F,CAAC;IAED,OAAO;QACL,OAAO,KAAK,CAAC;IACf,CAAC;IAED,oBAAoB;IACpB,sBAAsB;QACpB,OAAO,wBAAa,CAAC,OAAO,CAAC;IAC/B,CAAC;IAED,eAAe,CAAC,IAAa;QAC3B,MAAM,WAAW,GAAG,IAAI,CAAC,CAAC,CAAC,IAAI,aAAY,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC,IAAI,aAAY,EAAE,CAAC;QAC3E,MAAM,YAAY,GAAG,WAAW,CAAC,eAAe,EAAE,CAAC;QACnD,OAAO;YACL,GAAG,EAAE,YAAY,CAAC,IAAI;YACtB,GAAG,EAAE,YAAY,CAAC,IAAK;SACxB,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,gBAAgB,CAAC,MAA+B;QACpD,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,KAAK,CAAC,aAAa,CAAC,EAAE,OAAO,EAAwB;QACnD,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,EAAE,CAAC;YAClC,MAAM,IAAI,8BAAmB,CAAC,oBAAoB,OAAO,EAAE,CAAC,CAAC;QAC/D,CAAC;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;;;;;OAQG;IACH,KAAK,CAAC,iBAAiB,CAAC,MAAqC;QAC3D,MAAM,EAAE,QAAQ,EAAE,UAAU,EAAE,MAAM,EAAE,GAAG,MAAM,CAAC;QAChD,IAAI,CAAC,QAAQ,EAAE,UAAU,IAAI,CAAC,UAAU,EAAE,UAAU,IAAI,CAAC,MAAM,EAAE,CAAC;YAChE,MAAM,IAAI,KAAK,CAAC,gBAAgB,CAAC,CAAC;QACpC,CAAC;QACD,IAAI,QAAQ,CAAC,GAAG,IAAI,QAAQ,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACnD,MAAM,IAAI,KAAK,CAAC,+CAA+C,CAAC,CAAC;QACnE,CAAC;QACD,IAAI,UAAU,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACrC,MAAM,IAAI,KAAK,CACb,GAAG,IAAI,CAAC,QAAQ,EAAE,oIAAoI,CACvJ,CAAC;QACJ,CAAC;QACD,IAAI,QAAQ,CAAC,GAAG,IAAI,UAAU,CAAC,cAAc,EAAE,CAAC;YAC9C,6CAA6C;YAC7C,IAAI,QAAQ,CAAC,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBACrC,MAAM,IAAI,KAAK,CAAC,iDAAiD,QAAQ,CAAC,UAAU,CAAC,MAAM,QAAQ,CAAC,CAAC;YACvG,CAAC;YACD,gCAAgC;YAChC,IAAI,kBAAkB,CAAC;YACvB,IAAI,UAAU,CAAC,cAAc,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;gBAChD,MAAM,YAAY,GAAG,MAAM,IAAI,CAAC,wBAAwB,CAAC,UAAU,CAAC,cAAc,CAAC,EAAE,CAAC,CAAC;gBACvF,kBAAkB,GAAG,2BAAY,CAAC,OAAO,CAAC,cAAc,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC;YAC3F,CAAC;iBAAM,CAAC;gBACN,MAAM,YAAY,GAAG,2BAAY,CAAC,KAAK,CAAC,kBAAkB,CAAC,kBAAkB,CAC3E,2BAAY,CAAC,OAAO,CAAC,QAAQ,CAAC,UAAU,CAAC,cAAc,CAAC,EAAE,CAAC,CAC5D,CAAC;gBACF,kBAAkB,GAAG,2BAAY,CAAC,OAAO,CAAC,cAAc,CAAC,YAAY,CAAC,gBAAgB,EAAE,CAAC,QAAQ,EAAE,CAAC,CAAC;YACvG,CAAC;YACD,MAAM,gBAAgB,GAAG,2BAAY,CAAC,OAAO,CAAC,cAAc,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC;YAC/F,IAAI,kBAAkB,CAAC,WAAW,EAAE,KAAK,gBAAgB,CAAC,WAAW,EAAE,EAAE,CAAC;gBACxE,MAAM,IAAI,KAAK,CAAC,4DAA4D,CAAC,CAAC;YAChF,CAAC;YAED,wDAAwD;YACxD,MAAM,UAAU,GAAgB,QAAQ,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE;gBAC5D,OAAO;oBACL,OAAO,EAAE,CAAC,CAAC,OAAO;oBAClB,MAAM,EAAE,OAAO,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM;iBACtE,CAAC;YACJ,CAAC,CAAC,CAAC;YAEH,uCAAuC;YACvC,MAAM,IAAI,CAAC,mBAAmB,CAAC,MAAM,EAAE,UAAU,CAAC,cAAc,EAAE,EAAE,UAAU,EAAE,CAAC,CAAC;QACpF,CAAC;aAAM,IAAI,QAAQ,CAAC,UAAU,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC1C,2CAA2C;YAC3C,IAAI,mBAAmB,GAAG,IAAI,wBAAS,CAAC,CAAC,CAAC,CAAC;YAC3C,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,QAAQ,CAAC,UAAU,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;gBACpD,mBAAmB,GAAG,mBAAmB,CAAC,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;YAChF,CAAC;YACD,IAAI,CAAC,mBAAmB,CAAC,SAAS,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC;gBACpE,MAAM,IAAI,KAAK,CACb,+GAA+G,CAChH,CAAC;YACJ,CAAC;QACH,CAAC;aAAM,CAAC;YACN,4DAA4D;YAC5D,IAAI,QAAQ,CAAC,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBACrC,MAAM,IAAI,KAAK,CAAC,oDAAoD,QAAQ,CAAC,UAAU,CAAC,MAAM,QAAQ,CAAC,CAAC;YAC1G,CAAC;YACD,MAAM,cAAc,GAAG,IAAI,wBAAS,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;YACpE,IAAI,CAAC,cAAc,CAAC,SAAS,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC;gBAC/D,MAAM,IAAI,KAAK,CACb,gHAAgH,CACjH,CAAC;YACJ,CAAC;YACD,IACE,KAAK,CAAC,cAAc,CAAC,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC;gBACpD,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,OAAO,KAAK,UAAU,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,OAAO,EACnE,CAAC;gBACD,MAAM,IAAI,KAAK,CAAC,6FAA6F,CAAC,CAAC;YACjH,CAAC;QACH,CAAC;QACD,kDAAkD;QAClD,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC;YACjC,MAAM,IAAI,KAAK,CAAC,sEAAsE,CAAC,CAAC;QAC1F,CAAC;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAEO,MAAM,CAAC,cAAc,CAAC,OAAe;QAC3C,OAAO,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,mBAAmB,CAAC,CAAC;IAC9C,CAAC;IAED,UAAU,CAAC,UAA+B;QACxC,OAAO,UAAU,CAAC,IAAI,KAAK,IAAI,CAAC,QAAQ,EAAE,CAAC;IAC7C,CAAC;IAED,UAAU,CAAC,GAAW;QACpB,IAAI,KAAK,GAAG,IAAI,CAAC;QACjB,IAAI,CAAC;YACH,IAAI,aAAY,CAAC,EAAE,GAAG,EAAE,CAAC,CAAC;QAC5B,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,KAAK,GAAG,KAAK,CAAC;QAChB,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;;;;OAKG;IACH,WAAW,CAAC,YAAqB;QAC/B,IAAI,CAAC,YAAY,EAAE,CAAC;YAClB,OAAO,uBAAa,CAAC,eAAe,CAAC;QACvC,CAAC;QACD,MAAM,WAAW,GAAG,uBAAa,CAAC,eAAe,CAAC;QAClD,MAAM,WAAW,GAAG,uBAAa,CAAC,eAAe,CAAC;QAClD,IAAI,YAAY,GAAG,WAAW,IAAI,YAAY,GAAG,WAAW,EAAE,CAAC;YAC7D,MAAM,IAAI,KAAK,CAAC,6BAA6B,WAAW,QAAQ,WAAW,EAAE,CAAC,CAAC;QACjF,CAAC;QACD,OAAO,YAAY,CAAC;IACtB,CAAC;IAED;;;;;OAKG;IACH,WAAW,CAAC,YAAqB;QAC/B,IAAI,CAAC,YAAY,EAAE,CAAC;YAClB,OAAO,uBAAa,CAAC,eAAe,CAAC;QACvC,CAAC;QAED,MAAM,WAAW,GAAG,uBAAa,CAAC,eAAe,CAAC;QAClD,MAAM,WAAW,GAAG,uBAAa,CAAC,eAAe,CAAC;QAClD,IAAI,YAAY,GAAG,WAAW,IAAI,YAAY,GAAG,WAAW,EAAE,CAAC;YAC7D,MAAM,IAAI,KAAK,CAAC,6BAA6B,WAAW,QAAQ,WAAW,EAAE,CAAC,CAAC;QACjF,CAAC;QACD,OAAO,YAAY,CAAC;IACtB,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,+BAA+B,CAAC,KAA0B,EAAE,MAAe;QAC/E,MAAM,QAAQ,GAAG,MAAM,oBAAO;aAC3B,IAAI,CAAC,iBAAM,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,CAAC,mBAAmB,GAAG,eAAe,CAAC;aACpF,IAAI,CAAC,KAAK,CAAC,CAAC;QAEf,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC;YACjB,MAAM,IAAI,KAAK,CAAC,8BAA8B,CAAC,CAAC;QAClD,CAAC;QAED,IAAI,QAAQ,CAAC,IAAI,CAAC,MAAM,KAAK,GAAG,IAAI,QAAQ,CAAC,IAAI,CAAC,OAAO,KAAK,OAAO,EAAE,CAAC;YACtE,MAAM,IAAI,KAAK,CAAC,iCAAiC,CAAC,CAAC;QACrD,CAAC;QACD,OAAO,QAAQ,CAAC,IAAI,CAAC;IACvB,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,eAAe,CAAC,OAAe;QACnC,yCAAyC;QACzC,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,+BAA+B,CAAC;YACxD,OAAO,EAAE,KAAK;YACd,MAAM,EAAE,yBAAyB;YACjC,MAAM,EAAE,CAAC,OAAO,EAAE,QAAQ,CAAC;YAC3B,EAAE,EAAE,CAAC;SACN,CAAC,CAAC;QACH,IAAI,CAAC,MAAM,IAAI,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC;YACpC,MAAM,IAAI,KAAK,CAAC,oDAAoD,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC;QACjG,CAAC;QACD,MAAM,QAAQ,GAAG,MAAM,CAAC,MAAM,CAAC;QAC/B,OAAO,IAAI,2BAAY,CAAC,OAAO,CAAC,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,QAAQ,EAAE,CAAC;IACvE,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,mBAAmB,CAAC,OAAe;QACvC,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,+BAA+B,CAAC;YACxD,OAAO,EAAE,KAAK;YACd,MAAM,EAAE,gBAAgB;YACxB,MAAM,EAAE,CAAC,OAAO,EAAE,QAAQ,CAAC;YAC3B,EAAE,EAAE,CAAC;SACN,CAAC,CAAC;QACH,yEAAyE;QACzE,IAAI,CAAC,MAAM,IAAI,CAAC,MAAM,CAAC,MAAM,IAAI,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC;YACtD,MAAM,IAAI,KAAK,CAAC,wCAAwC,OAAO,4BAA4B,MAAM,CAAC,MAAM,EAAE,CAAC,CAAC;QAC9G,CAAC;QACD,MAAM,gBAAgB,GAAG,MAAM,CAAC,MAAM,CAAC;QACvC,OAAO,IAAI,2BAAY,CAAC,OAAO,CAAC,EAAE,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IACpE,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,wBAAwB,CAAC,oBAA4B,EAAE,qBAA6B;QACxF,wCAAwC;QACxC,MAAM,gBAAgB,GAAG,2BAAY,CAAC,MAAM;aACzC,YAAY,CAAC,oBAAoB,EAAE,qBAAqB,CAAC;aACzD,QAAQ,CAAC,KAAK,CAAC,CAAC;QACnB,MAAM,mBAAmB,GAAG,2BAAY,CAAC,OAAO,CAAC,YAAY,CAAC,gBAAgB,CAAC,CAAC;QAChF,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,+BAA+B,CAAC;YACxD,OAAO,EAAE,KAAK;YACd,MAAM,EAAE,UAAU;YAClB,MAAM,EAAE;gBACN;oBACE,EAAE,EAAE,oBAAoB;oBACxB,IAAI,EAAE,mBAAmB;iBAC1B;gBACD,QAAQ;aACT;YACD,EAAE,EAAE,CAAC;SACN,CAAC,CAAC;QACH,yEAAyE;QACzE,IAAI,CAAC,MAAM,IAAI,CAAC,MAAM,CAAC,MAAM,IAAI,KAAK,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,CAAC;YACtD,MAAM,IAAI,KAAK,CACb,8CAA8C,qBAAqB,4BAA4B,MAAM,CAAC,MAAM,EAAE,CAC/G,CAAC;QACJ,CAAC;QACD,MAAM,eAAe,GAAG,MAAM,CAAC,MAAM,CAAC;QACtC,OAAO,IAAI,2BAAY,CAAC,OAAO,CAAC,EAAE,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IACnE,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,eAAe,CAAC,OAAe;QACnC,sCAAsC;QACtC,MAAM,yBAAyB,GAAG,2BAAY,CAAC,MAAM,CAAC,QAAQ,CAAC,mBAAmB,EAAE,EAAE,CAAC,CAAC;QACxF,MAAM,cAAc,GAAG,2BAAY,CAAC,MAAM,CAAC,SAAS,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC;QAC7D,MAAM,cAAc,GAAG,eAAM,CAAC,MAAM,CAAC,CAAC,yBAAyB,EAAE,cAAc,CAAC,CAAC,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QAClG,MAAM,iBAAiB,GAAG,2BAAY,CAAC,OAAO,CAAC,YAAY,CAAC,cAAc,CAAC,CAAC;QAC5E,MAAM,MAAM,GAAG,MAAM,IAAI,CAAC,+BAA+B,CAAC;YACxD,OAAO,EAAE,KAAK;YACd,MAAM,EAAE,UAAU;YAClB,MAAM,EAAE,CAAC,EAAE,EAAE,EAAE,OAAO,EAAE,IAAI,EAAE,iBAAiB,EAAE,EAAE,QAAQ,CAAC;YAC5D,EAAE,EAAE,CAAC;SACN,CAAC,CAAC;QACH,IAAI,CAAC,MAAM,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC;YAC9B,MAAM,IAAI,KAAK,CAAC,uDAAuD,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC;QAC3F,CAAC;QACD,MAAM,aAAa,GAAG,MAAM,CAAC,MAAM,CAAC;QACpC,OAAO,IAAI,2BAAY,CAAC,OAAO,CAAC,EAAE,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,QAAQ,EAAE,CAAC;IAC5E,CAAC;IAED;;;;;OAKG;IACH,YAAY,CAAC,SAAoB,EAAE,UAAkB,EAAE,kBAA0B;QAC/E,OAAO;YACL,CAAC,QAAQ,EAAE,SAAS,EAAE,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,MAAM,CAAC;YACtD;gBACE,OAAO;gBACP,IAAI,2BAAY,CAAC,OAAO,CAAC,EAAE,CAAC,2BAAY,CAAC,OAAO,CAAC,cAAc,CAAC,SAAS,CAAC,OAAO,CAAC,EAAE,EAAE,CAAC;gBACvF,SAAS,CAAC,MAAM;gBAChB,eAAM,CAAC,IAAI,CAAC,2BAAY,CAAC,OAAO,CAAC,cAAc,CAAC,2BAAY,CAAC,OAAO,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,IAAI,EAAE,CAAC,CAAC,EAAE,KAAK,CAAC;gBAC7G,UAAU;gBACV,kBAAkB;aACnB;SACF,CAAC;IACJ,CAAC;IAED;;;;;;OAMG;IACH,oCAAoC,CAClC,UAAuB,EACvB,UAAkB,EAClB,kBAA0B;QAE1B,IAAI,CAAC,UAAU,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,UAAU,CAAC,EAAE,CAAC;YAC9C,MAAM,IAAI,KAAK,CAAC,+BAA+B,CAAC,CAAC;QACnD,CAAC;QAED,wCAAwC;QACxC,IAAI,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC5B,MAAM,IAAI,KAAK,CAAC,kCAAkC,CAAC,CAAC;QACtD,CAAC;QAED,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE,CAAC;YAC5B,MAAM,IAAI,KAAK,CAAC,kDAAkD,CAAC,CAAC;QACtE,CAAC;QAED,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,kBAAkB,CAAC,EAAE,CAAC;YACpC,MAAM,IAAI,KAAK,CAAC,mCAAmC,CAAC,CAAC;QACvD,CAAC;QAED,eAAe;QACf,UAAU,CAAC,OAAO,CAAC,UAAU,SAAS;YACpC,IACE,CAAC,CAAC,CAAC,QAAQ,CAAC,SAAS,CAAC,OAAO,CAAC;gBAC9B,CAAC,2BAAY,CAAC,OAAO,CAAC,cAAc,CAAC,2BAAY,CAAC,OAAO,CAAC,YAAY,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,EAC1F,CAAC;gBACD,MAAM,IAAI,KAAK,CAAC,mBAAmB,GAAG,SAAS,CAAC,OAAO,CAAC,CAAC;YAC3D,CAAC;YAED,IAAI,MAAM,CAAC;YACX,IAAI,CAAC;gBACH,MAAM,GAAG,IAAI,wBAAS,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;YAC3C,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,MAAM,IAAI,KAAK,CAAC,sBAAsB,GAAG,SAAS,CAAC,OAAO,GAAG,sBAAsB,CAAC,CAAC;YACvF,CAAC;YAED,SAAS,CAAC,MAAM,GAAG,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;YAErC,IAAI,SAAS,CAAC,IAAI,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,CAAC;gBAClD,MAAM,IAAI,KAAK,CAAC,qBAAqB,GAAG,SAAS,CAAC,OAAO,GAAG,iCAAiC,CAAC,CAAC;YACjG,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,MAAM,SAAS,GAAG,UAAU,CAAC,CAAC,CAAC,CAAC;QAChC,OAAO,2BAAY,CAAC,OAAO,CAAC,WAAW,CACrC,2BAAY,CAAC,MAAM,CAAC,YAAY,CAAC,GAAG,IAAI,CAAC,YAAY,CAAC,SAAS,EAAE,UAAU,EAAE,kBAAkB,CAAC,CAAC,CAClG,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,oBAAoB;QAClB,OAAO,IAAI,CAAC,KAAK,CAAC,IAAI,IAAI,EAAE,CAAC,OAAO,EAAE,GAAG,IAAI,CAAC,GAAG,EAAE,GAAG,EAAE,GAAG,EAAE,GAAG,CAAC,CAAC;IACpE,CAAC;IAED;;;;OAIG;IACH,iBAAiB,CAAC,MAAgC;QAChD,sBAAsB;QACtB,6GAA6G;QAC7G,OAAO;YACL;gBACE,IAAI,EAAE,WAAW;gBACjB,IAAI,EAAE,SAAS;gBACf,KAAK,EAAE,MAAM,CAAC,SAAS,CAAC,OAAO;aAChC;YACD;gBACE,IAAI,EAAE,OAAO;gBACb,IAAI,EAAE,MAAM;gBACZ,KAAK,EAAE,MAAM,CAAC,SAAS,CAAC,MAAM;aAC/B;YACD;gBACE,IAAI,EAAE,MAAM;gBACZ,IAAI,EAAE,OAAO;gBACb,KAAK,EAAE,2BAAY,CAAC,OAAO,CAAC,QAAQ,CAAC,2BAAY,CAAC,OAAO,CAAC,YAAY,CAAC,MAAM,CAAC,SAAS,CAAC,IAAI,IAAI,EAAE,CAAC,CAAC;aACrG;YACD;gBACE,IAAI,EAAE,YAAY;gBAClB,IAAI,EAAE,MAAM;gBACZ,KAAK,EAAE,MAAM,CAAC,UAAU;aACzB;YACD;gBACE,IAAI,EAAE,YAAY;gBAClB,IAAI,EAAE,MAAM;gBACZ,KAAK,EAAE,MAAM,CAAC,kBAAkB;aACjC;YACD;gBACE,IAAI,EAAE,WAAW;gBACjB,IAAI,EAAE,OAAO;gBACb,KAAK,EAAE,2BAAY,CAAC,OAAO,CAAC,QAAQ,CAAC,2BAAY,CAAC,OAAO,CAAC,YAAY,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;aAC1F;SACF,CAAC;IACJ,CAAC;IAED;;;;;;;;;;;;;OAaG;IACH,KAAK,CAAC,OAAO,CAAC,MAAsB;QAClC,IAAI,MAAM,CAAC,eAAe,EAAE,CAAC;YAC3B,OAAO,CAAC,MAAM,IAAI,CAAC,iCAAiC,CAAC,MAAM,CAAC,CAAsC,CAAC;QACrG,CAAC;QAED,IAAI,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE,CAAC;YAClC,MAAM,IAAI,KAAK,CAAC,iBAAiB,CAAC,CAAC;QACrC,CAAC;QAED,IAAI,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,SAAS,CAAC,EAAE,CAAC;YACpC,MAAM,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC;QACvC,CAAC;QAED,IAAI,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,gBAAgB,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;YACjF,MAAM,IAAI,KAAK,CAAC,2BAA2B,CAAC,CAAC;QAC/C,CAAC;QAED,IAAI,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,qBAAqB,CAAC,EAAE,CAAC;YACtG,MAAM,IAAI,KAAK,CAAC,+BAA+B,CAAC,CAAC;QACnD,CAAC;QAED,IAAI,SAAS,CAAC;QACd,IAAI,MAAM,CAAC,oBAAoB,EAAE,CAAC;YAChC,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,oBAAoB,CAAC,EAAE,CAAC;gBACtD,MAAM,IAAI,KAAK,CAAC,8BAA8B,CAAC,CAAC;YAClD,CAAC;YACD,MAAM,OAAO,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;YAClC,MAAM,KAAK,GAAG,IAAA,gBAAQ,EAAC,MAAM,CAAC,oBAAoB,EAAE,OAAO,CAAC,CAAC;YAC7D,IAAI,CAAC,CAAC,WAAW,CAAC,KAAK,CAAC,EAAE,CAAC;gBACzB,MAAM,IAAI,KAAK,CAAC,qBAAqB,CAAC,CAAC;YACzC,CAAC;YACD,SAAS,GAAG,KAAK,CAAC,IAAI,CAAC;QACzB,CAAC;QAED,IAAI,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,mBAAmB,CAAC,EAAE,CAAC;YAClG,MAAM,IAAI,KAAK,CAAC,6BAA6B,CAAC,CAAC;QACjD,CAAC;QAED,uCAAuC;QACvC,MAAM,eAAe,GAAG,IAAA,6BAAkB,EAAC,MAAM,CAAC,CAAC;QAEnD,0CAA0C;QAC1C,IAAI,OAAO,GAAG,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAChD,MAAM,SAAS,GAAG,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC;QAEtD,8DAA8D;QAC9D,MAAM,QAAQ,GAAG,IAAI,2BAAY,CAAC,OAAO,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC;QAChF,MAAM,QAAQ,GAAG,MAAM,CAAC,OAAO;YAC7B,CAAC,CAAC,IAAI,2BAAY,CAAC,OAAO,CAAC,EAAE,CAAC,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC;YAC1D,CAAC,CAAC,IAAI,2BAAY,CAAC,OAAO,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC;QACnE,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;YAC/D,IAAI,CAAC;gBACH,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC;oBAC3B,KAAK,EAAE,OAAO;oBACd,QAAQ,EAAE,MAAM,CAAC,gBAAgB;iBAClC,CAAC,CAAC;YACL,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,MAAM,IAAI,KAAK,CAAC,mCAAmC,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC;YAClE,CAAC;QACH,CAAC;QAED,IAAI,gBAAgB,CAAC;QACrB,IAAI,gBAAgB,CAAC;QACrB,IAAI,eAAe,EAAE,CAAC;YACpB,MAAM,aAAa,GAAG,IAAI,aAAY,CAAC,EAAE,GAAG,EAAE,SAAS,EAAE,CAAC,CAAC;YAC3D,gBAAgB,GAAG,aAAa,CAAC,UAAU,EAAE,CAAC;QAChD,CAAC;aAAM,CAAC;YACN,6CAA6C;YAC7C,IAAI,SAAS,CAAC;YAEd,IAAI,CAAC;gBACH,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC;oBAC7B,KAAK,EAAE,SAAS;oBAChB,QAAQ,EAAE,MAAM,CAAC,gBAAgB;iBAClC,CAAC,CAAC;YACL,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,MAAM,IAAI,KAAK,CAAC,qCAAqC,CAAC,CAAC,OAAO,EAAE,CAAC,CAAC;YACpE,CAAC;YAED,MAAM,OAAO,GAAG,IAAI,aAAY,CAAC,EAAE,GAAG,EAAE,SAAS,EAAE,CAAC,CAAC;YACrD,gBAAgB,GAAG,OAAO,CAAC,OAAO,EAAE,CAAC,GAAG,CAAC;YACzC,IAAI,CAAC,gBAAgB,EAAE,CAAC;gBACtB,MAAM,IAAI,KAAK,CAAC,gBAAgB,CAAC,CAAC;YACpC,CAAC;YACD,gBAAgB,GAAG,OAAO,CAAC,UAAU,EAAE,CAAC;QAC1C,CAAC;QACD,MAAM,cAAc,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,gBAAgB,CAAC,CAAC;QAEpE,qEAAqE;QACrE,MAAM,gBAAgB,GAAG,MAAM,IAAI,CAAC,mBAAmB,CAAC,gBAAgB,CAAC,CAAC;QAE1E,MAAM,cAAc,GAAG,QAAQ,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAC9C,MAAM,SAAS,GAAG,EAAE,IAAI,CAAC,CAAC;QAC1B,IAAI,gBAAgB,CAAC,EAAE,CAAC,cAAc,CAAC,EAAE,CAAC;YACxC,MAAM,IAAI,KAAK,CACb,sBAAsB,gBAAgB,gBAAgB,gBAAgB;iBACnE,GAAG,CAAC,IAAI,oBAAE,CAAC,SAAS,CAAC,CAAC;iBACtB,QAAQ,EAAE,QAAQ;gBACnB,gDAAgD,CAAC,cAAc,GAAG,SAAS,CAAC,CAAC,QAAQ,EAAE,EAAE;gBACzF,gFAAgF,CACnF,CAAC;QACJ,CAAC;QAED,IAAI,QAAQ,CAAC;QACb,IAAI,MAAM,CAAC,oBAAoB,EAAE,CAAC;YAChC,8BAA8B;YAC9B,QAAQ,GAAG,MAAM,IAAI,CAAC,wBAAwB,CAAC,MAAM,CAAC,oBAAoB,EAAE,MAAM,CAAC,qBAAqB,CAAC,CAAC;QAC5G,CAAC;aAAM,CAAC;YACN,kEAAkE;YAClE,QAAQ,GAAG,MAAM,IAAI,CAAC,mBAAmB,CAAC,MAAM,CAAC,qBAAqB,CAAC,CAAC;QAC1E,CAAC;QAED,0BAA0B;QAC1B,MAAM,UAAU,GAAG;YACjB;gBACE,OAAO,EAAE,MAAM,CAAC,mBAAmB;gBACnC,MAAM,EAAE,QAAQ,CAAC,QAAQ,CAAC,EAAE,CAAC;aAC9B;SACF,CAAC;QAEF,sCAAsC;QACtC,gFAAgF;QAChF,MAAM,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,EAAE,CAAC,UAAU,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC,CAAC;QAC1D,MAAM,UAAU,GAAG,MAAM,IAAI,CAAC,eAAe,CAAC,MAAM,CAAC,qBAAqB,CAAC,CAAC;QAE5E,IAAI,aAAa,EAAE,SAAS,CAAC;QAC7B,iCAAiC;QACjC,IAAI,CAAC,eAAe,EAAE,CAAC;YACrB,aAAa,GAAG,IAAI,CAAC,oCAAoC,CAAC,UAAU,EAAE,IAAI,CAAC,oBAAoB,EAAE,EAAE,UAAU,CAAC,CAAC;YAC/G,SAAS,GAAG,eAAI,CAAC,cAAc,CAAC,aAAa,EAAE,eAAI,CAAC,mBAAmB,CAAC,OAAO,CAAC,CAAC,CAAC;YAElF,IAAI,CAAC;gBACH,eAAI,CAAC,mBAAmB,CAAC,aAAa,EAAE,SAAS,CAAC,CAAC;YACrD,CAAC;YAAC,OAAO,CAAC,EAAE,CAAC;gBACX,MAAM,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC;YACvC,CAAC;QACH,CAAC;QAED,MAAM,MAAM,GAAG;YACb,SAAS,EAAE,UAAU,CAAC,CAAC,CAAC;YACxB,UAAU,EAAE,IAAI,CAAC,oBAAoB,EAAE;YACvC,kBAAkB,EAAE,UAAU;YAC9B,aAAa;YACb,SAAS;YACT,QAAQ,EAAE,QAAQ,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC/B,oBAAoB,EAAE,MAAM,CAAC,oBAAoB;SAClD,CAAC;QAEF,MAAM,SAAS,GAAG,IAAI,CAAC,qBAAqB,EAAwB,CAAC;QACrE,SAAS,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC;QAClC,SAAS,CAAC,QAAQ,CAAC,MAAM,CAAC,qBAAqB,CAAC,CAAC;QACjD,IAAI,KAAK,CAAC;QACV,IAAI,MAAM,CAAC,OAAO,EAAE,CAAC;YACnB,KAAK,GAAG;gBACN,OAAO,EAAE;oBACP,oBAAoB,EAAE,MAAM,CAAC,OAAO,CAAC,oBAAoB;oBACzD,YAAY,EAAE,MAAM,CAAC,OAAO,CAAC,YAAY;iBAC1C;aACF,CAAC;QACJ,CAAC;aAAM,CAAC;YACN,KAAK,GAAG,EAAE,GAAG,EAAE,QAAQ,CAAC,QAAQ,EAAE,EAAE,CAAC;QACvC,CAAC;QACD,SAAS,CAAC,GAAG,CAAC;YACZ,GAAG,KAAK;YACR,QAAQ,EAAE,QAAQ,CAAC,QAAQ,EAAE;SAC9B,CAAC,CAAC;QACH,IAAI,MAAM,CAAC,oBAAoB,EAAE,CAAC;YAChC,SAAS;iBACN,QAAQ,EAAE;iBACV,IAAI,CAAC,SAAS,CAAC;iBACf,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC;iBAC5B,kBAAkB,CAAC,UAAU,CAAC;iBAC9B,cAAc,CAAC,IAAI,CAAC,oBAAoB,EAAE,CAAC;iBAC3C,EAAE,CAAC,MAAM,CAAC,mBAAmB,CAAC,CAAC;QACpC,CAAC;aAAM,CAAC;YACN,SAAS;iBACN,QAAQ,EAAE;iBACV,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC;iBACrB,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC;iBAC5B,kBAAkB,CAAC,UAAU,CAAC;iBAC9B,cAAc,CAAC,IAAI,CAAC,oBAAoB,EAAE,CAAC;iBAC3C,EAAE,CAAC,MAAM,CAAC,mBAAmB,CAAC,CAAC;QACpC,CAAC;QAED,IAAI,eAAe,EAAE,CAAC;YACpB,MAAM,EAAE,GAAG,MAAM,SAAS,CAAC,KAAK,EAAE,CAAC;YACnC,MAAM,QAAQ,GAAuB;gBACnC,KAAK,EAAE,EAAE,CAAC,iBAAiB,EAAE;gBAC7B,OAAO;gBACP,SAAS;gBACT,IAAI,EAAE,IAAI,CAAC,QAAQ,EAAE;gBACrB,KAAK,EAAE,SAAS;gBAChB,QAAQ,EAAE,2BAAY,CAAC,OAAO,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC,OAAO,EAAE;gBAC9D,QAAQ;gBACR,UAAU,EAAE,CAAC,MAAM,CAAC,SAAS,CAAC;gBAC9B,qBAAqB,EAAE,EAAE,CAAC,MAAM,EAAE,CAAC,EAAE;gBACrC,MAAM,EAAE,MAAM,CAAC,SAAS,CAAC,MAAM;gBAC/B,cAAc;gBACd,OAAO,EAAE,MAAM,CAAC,OAAO;aACxB,CAAC;YACF,CAAC,CAAC,MAAM,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;YAC3B,QAAQ,CAAC,sBAAsB,GAAG,QAAQ,CAAC,kBAAkB,CAAC;YAC9D,OAAO,QAAQ,CAAC;QAClB,CAAC;QAED,MAAM,WAAW,GAAG,IAAI,aAAY,CAAC,EAAE,GAAG,EAAE,OAAO,EAAE,CAAC,CAAC;QACvD,SAAS,CAAC,QAAQ,EAAE,CAAC,GAAG,CAAC,WAAW,CAAC,OAAO,EAAE,CAAC,GAAI,CAAC,CAAC;QACrD,SAAS,CAAC,IAAI,CAAC,EAAE,GAAG,EAAE,gBAAgB,EAAE,CAAC,CAAC;QAC1C,MAAM,QAAQ,GAAG,MAAM,SAAS,CAAC,KAAK,EAAE,CAAC;QAEzC,OAAO;YACL,EAAE,EAAE,QAAQ,CAAC,MAAM,EAAE,CAAC,EAAE;YACxB,EAAE,EAAE,QAAQ,CAAC,iBAAiB,EAAE;SACjC,CAAC;IACJ,CAAC;IAED;;;OAGG;IACO,qBAAqB;QAC7B,OAAO,IAAI,wBAAkB,CAAC,eAAK,CAAC,GAAG,CAAC,IAAI,CAAC,YAAY,EAAE,CAAC,CAAC,CAAC;IAChE,CAAC;IAES,gBAAgB;QACxB,OAAO,IAAI,yBAAQ,CAAC,yBAAyB,CAAC,eAAK,CAAC,GAAG,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC;IAC5E,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,kBAAkB,CAAC,MAAiC;QACxD,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,IAAI,CAAC,MAAM,CAAC,UAAU,IAAI,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;QAC7E,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,MAAM,IAAI,KAAK,CAAC,wCAAwC,CAAC,CAAC;QAC5D,CAAC;QACD,IAAI,MAAM,CAAC,cAAc,EAAE,CAAC;YAC1B,OAAO,IAAI,CAAC,wBAAwB,CAAC,KAAK,CAAC,CAAC;QAC9C,CAAC;QACD,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;YACpB,MAAM,IAAI,KAAK,CAAC,0CAA0C,CAAC,CAAC;QAC9D,CAAC;QACD,MAAM,SAAS,GAAG,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAC/C,SAAS,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACtB,MAAM,EAAE,GAAG,MAAM,SAAS,CAAC,KAAK,EAAE,CAAC;QACnC,OAAO,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,qBAAqB,CAAC,EAAE,CAAC,EAAE,EAAE,GAAG,EAAE,MAAM,CAAC,OAAO,EAAE,CAAC,CAAC;IAChF,CAAC;IAED;;;;OAIG;IACK,KAAK,CAAC,wBAAwB,CAAC,KAAa;QAClD,MAAM,SAAS,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACtD,MAAM,EAAE,GAAG,MAAM,SAAS,CAAC,KAAK,EAAE,CAAC;QACnC,OAAO,EAAE,CAAC,kBAAkB,EAAE,CAAC;IACjC,CAAC;IAED;;;;;OAKG;IACK,KAAK,CAAC,mCAAmC,CAAC,KAAa;QAC7D,MAAM,SAAS,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACtD,MAAM,EAAE,GAAG,MAAM,SAAS,CAAC,KAAK,EAAE,CAAC;QACnC,MAAM,OAAO,GAAG,EAAE,CAAC,eAAe,CAAC;QACnC,MAAM,UAAU,GAAG,EAAE,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,eAAM,CAAC,IAAI,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC;QAClE,MAAM,OAAO,GAAG,CAAC,CAAC,GAAG,CAAC,EAAE,EAAE,UAAU,CAAC,CAAC;QACtC,MAAM,YAAY,GAAG,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CACxC,yBAAQ,CAAC,KAAK,CAAC,iBAAiB,CAAC,OAAsC,EAAE,OAAO,EAAE,CAAC,CAAC,CACrF,CAAC;QACF,MAAM,eAAe,GAAG,YAAY,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,IAAA,8BAAY,EAAC,CAAC,EAAE,IAAI,CAAC,CAAC,CAAC;QACvE,MAAM,OAAO,GAAG,EAAE,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,yBAAQ,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;QAC7E,OAAO,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC;IAClF,CAAC;IAED;;;;OAIG;IACK,qBAAqB,CAAC,EAAmB;QAC/C,MAAM,OAAO,GAAG,EAAE,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,MAAM,EAAE,EAAE;YACxC,OAAO;gBACL,OAAO,EAAE,MAAM,CAAC,OAAO;gBACvB,MAAM,EAAE,MAAM,CAAC,KAAK;aACrB,CAAC;QACJ,CAAC,CAAC,CAAC;QACH,MAAM,YAAY,GAAG,CAAC,IAAI,EAAE,cAAc,EAAE,cAAc,EAAE,SAAS,EAAE,eAAe,EAAE,KAAK,CAAC,CAAC;QAC/F,OAAO;YACL,YAAY;YACZ,EAAE,EAAE,EAAE,CAAC,EAAE;YACT,OAAO,EAAE,OAAO;YAChB,YAAY,EAAE,OAAO;iBAClB,MAAM,CAAC,CAAC,WAAW,EAAE,MAAM,EAAE,EAAE,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,IAAI,wBAAS,CAAC,GAAG,CAAC,CAAC;iBACpF,OAAO,CAAC,CAAC,CAAC;YACb,aAAa,EAAE,EAAE,EAAE,4CAA4C;YAC/D,YAAY,EAAE,GAAG,EAAE,oCAAoC;SACxD,CAAC;IACJ,CAAC;IAED;;;;;OAKG;IAEH;;;OAGG;IACH,KAAK,CAAC,kBAAkB,CAAC,MAAiC;QACxD,IAAI,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC;YACjH,MAAM,IAAI,CAAC,mBAAmB,CAAC,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,cAAc,CAAC,CAAC;QACvE,CAAC;QACD,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,mBAAmB,CAAC,MAA2B;QACnD,IAAI,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC;YACjH,MAAM,IAAI,CAAC,mBAAmB,CAAC,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,cAAc,EAAE,MAAM,CAAC,WAAW,CAAC,CAAC;QAC3F,CAAC;QACD,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,mBAAmB,CACvB,MAAe,EACf,WAAwB,EACxB,cAA4C;QAE5C,MAAM,EAAE,EAAE,EAAE,EAAE,EAAE,SAAS,EAAE,GAAG,WAAW,CAAC;QAE1C,oCAAoC;QACpC,MAAM,UAAU,GAAG,iBAAM,CAAC,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,CAAC,OAAO,CAAC;QACpE,MAAM,kBAAkB,GAAW,iBAAK,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC,SAAS,CAAC;QAC1E,MAAM,eAAe,GAAW,eAAM,CAAC,IAAI,CAAC,2BAAY,CAAC,OAAO,CAAC,cAAc,CAAC,SAAS,CAAC,EAAE,KAAK,CAAC,CAAC;QACnG,MAAM,aAAa,GACjB,WAAW,CAAC,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC,KAAK,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,eAAM,CAAC,IAAI,CAAC,2BAAY,CAAC,OAAO,CAAC,cAAc,CAAC,EAAE,CAAC,EAAE,KAAK,CAAC,CAAC;QAEpH,MAAM,GAAG,GAAG,IAAI,UAAU,CAAC,eAAe,CAAC,MAAM,KAAK,EAAE,CAAC,CAAC,CAAC,eAAe,CAAC,CAAC,CAAC,eAAe,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QACvG,MAAM,gBAAgB,GAAY,SAAS,CAAC,WAAW,CAAC,GAAG,EAAE,aAAa,EAAE,kBAAkB,CAAC,CAAC;QAChG,IAAI,CAAC,gBAAgB,EAAE,CAAC;YACtB,MAAM,IAAI,KAAK,CAAC,4BAA4B,CAAC,CAAC;QAChD,CAAC;QAED,IAAI,WAAW,CAAC,IAAI,KAAK,QAAQ,EAAE,CAAC;YAClC,MAAM,kBAAkB,GAAG,MAAM,IAAI,CAAC,wBAAwB,CAAC,EAAE,CAAC,CAAC;YACnE,0FAA0F;YAC1F,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,cAAc,CAAC,EAAE,CAAC;gBAC7B,MAAM,EAAE,UAAU,EAAE,GAAG,cAAc,CAAC;gBAEtC,wGAAwG;gBACxG,MAAM,cAAc,GAAG,IAAI,wBAAS,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;gBACzF,MAAM,mBAAmB,GAAuB,UAAU,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC;gBACtE,MAAM,SAAS,GAAG,kBAAkB,CAAC,YAAY,CAAC;gBAClD,MAAM,cAAc,GAAG,kBAAkB,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC;gBAC7D,IAAI,cAAc,KAAK,SAAS,EAAE,CAAC;oBACjC,MAAM,IAAI,KAAK,CAAC,eAAe,SAAS,oCAAoC,cAAc,EAAE,CAAC,CAAC;gBAChG,CAAC;gBACD,IAAI,mBAAmB,IAAI,cAAc,CAAC,WAAW,EAAE,KAAK,mBAAmB,CAAC,WAAW,EAAE,EAAE,CAAC;oBAC9F,MAAM,IAAI,KAAK,CACb,oBAAoB,cAAc,uCAAuC,mBAAmB,EAAE,CAC/F,CAAC;gBACJ,CAAC;YACH,CAAC;YACD,IAAI,CAAC,CAAC,MAAM,IAAI,CAAC,mCAAmC,CAAC,EAAE,CAAC,CAAC,EAAE,CAAC;gBAC1D,MAAM,IAAI,KAAK,CAAC,4CAA4C,EAAE,EAAE,CAAC,CAAC;YACpE,CAAC;QACH,CAAC;aAAM,CAAC;YACN,MAAM,UAAU,GAAG,2BAAY,CAAC,KAAK,CAAC,kBAAkB,CAAC,kBAAkB,CAAC,2BAAY,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,CAAC;YAC/G,0FAA0F;YAC1F,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,cAAc,CAAC,EAAE,CAAC;gBAC7B,MAAM,EAAE,UAAU,EAAE,GAAG,cAAc,CAAC;gBAEtC,uEAAuE;gBACvE,MAAM,cAAc,GAAG,IAAI,wBAAS,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC;gBAC3D,MAAM,mBAAmB,GAAW,UAAU,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC;gBAE1D,MAAM,SAAS,GAAG,IAAI,wBAAS,CAAC,2BAAY,CAAC,OAAO,CAAC,WAAW,CAAC,UAAU,CAAC,KAA0B,CAAC,CAAC,CAAC;gBACzG,IAAI,CAAC,UAAU,CAAC,EAAE,EAAE,CAAC;oBACnB,MAAM,IAAI,KAAK,CAAC,iDAAiD,CAAC,CAAC;gBACrE,CAAC;gBACD,MAAM,cAAc,GAAG,UAAU,CAAC,EAAE,CAAC,QAAQ,EAAE,CAAC;gBAChD,IAAI,CAAC,SAAS,CAAC,EAAE,CAAC,cAAc,CAAC,EAAE,CAAC;oBAClC,MAAM,IAAI,KAAK,CAAC,eAAe,SAAS,oCAAoC,cAAc,EAAE,CAAC,CAAC;gBAChG,CAAC;gBACD,IAAI,cAAc,CAAC,WAAW,EAAE,KAAK,mBAAmB,CAAC,WAAW,EAAE,EAAE,CAAC;oBACvE,MAAM,IAAI,KAAK,CAAC,oBAAoB,cAAc,uCAAuC,cAAc,EAAE,CAAC,CAAC;gBAC7G,CAAC;YACH,CAAC;YAED,IAAI,CAAC,UAAU,CAAC,eAAe,EAAE,EAAE,CAAC;gBAClC,+FAA+F;gBAC/F,MAAM,IAAI,KAAK,CAAC,4CAA4C,EAAE,EAAE,CAAC,CAAC;YACpE,CAAC;YACD,IAAI,2BAAY,CAAC,OAAO,CAAC,YAAY,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,KAAK,EAAE,EAAE,CAAC;gBAChF,MAAM,IAAI,KAAK,CAAC,4CAA4C,CAAC,CAAC;YAChE,CAAC;QACH,CAAC;IACH,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,SAAS,CAAC,MAAwB;QACtC,MAAM,OAAO,GAAG,IAAI,aAAY,CAAC,EAAE,GAAG,EAAE,MAAM,CAAC,GAAG,EAAE,CAAC,CAAC;QACtD,MAAM,UAAU,GAAG,OAAO,CAAC,OAAO,EAAE,CAAC,GAAG,CAAC;QACzC,IAAI,CAAC,CAAC,WAAW,CAAC,UAAU,CAAC,EAAE,CAAC;YAC9B,MAAM,IAAI,KAAK,CAAC,qBAAqB,CAAC,CAAC;QACzC,CAAC;QAED,MAAM,SAAS,GAAG,IAAI,CAAC,qBAAqB,EAAwB,CAAC;QACrE,IAAI,CAAC;YACH,SAAS,CAAC,IAAI,CAAC,MAAM,CAAC,UAAW,CAAC,UAAW,CAAC,KAAK,CAAC,CAAC;QACvD,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,MAAM,IAAI,KAAK,CAAC,iCAAiC,CAAC,CAAC;QACrD,CAAC;QAED,SAAS,CAAC,IAAI,CAAC,EAAE,GAAG,EAAE,UAAU,EAAE,CAAC,CAAC;QACpC,MAAM,EAAE,GAAG,MAAM,SAAS,CAAC,KAAK,EAAE,CAAC;QACnC,OAAO;YACL,KAAK,EAAE,EAAE,CAAC,iBAAiB,EAAE;SAC9B,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,eAAe,CAAC,MAA2D;QAC/E,iEAAiE;QACjE,6EAA6E;QAC7E,IAAI,MAAM,CAAC,eAAe,EAAE,CAAC;YAC3B,sFAAsF;YACtF,OAAO,MAAM,IAAI,CAAC,SAAS,CAAC,MAAqC,CAAC,CAAC;QACrE,CAAC;QAED,MAAM,SAAS,GAAG,IAAI,CAAC,qBAAqB,EAAwB,CAAC;QACrE,SAAS,CAAC,IAAI,CAAC,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;QACxC,SAAS,CAAC,QAAQ,EAAE,CAAC,GAAG,CAAC,IAAI,aAAY,CAAC,EAAE,GAAG,EAAE,MAAM,CAAC,GAAG,EAAE,CAAC,CAAC,OAAO,EAAE,CAAC,GAAI,CAAC,CAAC;QAC/E,IAAI,MAAM,CAAC,aAAa,EAAE,CAAC;YACzB,SAAS,CAAC,aAAa,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;QAChD,CAAC;QACD,MAAM,WAAW,GAAG,MAAM,SAAS,CAAC,KAAK,EAAE,CAAC;QAE5C,yGAAyG;QACzG,IAAI,UAAU,GAAG,MAAM,CAAC,UAAU,CAAC,UAAU,IAAK,MAAM,CAAC,UAAsC,CAAC;QAChG,IAAI,UAAU,KAAK,SAAS,EAAE,CAAC;YAC7B,UAAU,GAAG,WAAW,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC,EAAE,OAAO,EAAE,MAAM,CAAC,OAAO,EAAE,MAAM,EAAE,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;QACxG,CAAC;QAED,MAAM,QAAQ,GAAG;YACf,OAAO,EAAE,MAAM,CAAC,UAAU,CAAC,OAAO;YAClC,KAAK,EAAE,WAAW,CAAC,iBAAiB,EAAE;YACtC,UAAU,EAAE,UAAU;YACtB,UAAU,EAAE,MAAM,CAAC,UAAU,CAAC,UAAU;YACxC,cAAc,EAAE,MAAM,CAAC,UAAU,CAAC,cAAc;YAChD,sBAAsB,EAAE,MAAM,CAAC,sBAAsB;SACtD,CAAC;QAEF,OAAO,EAAE,UAAU,EAAE,QAAQ,EAAE,CAAC;IAClC,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,sBAAsB,CAAC,WAAyB;QACpD,IACE,CAAC,CAAC,CAAC,WAAW,CAAC,WAAW,CAAC,GAAG,CAAC;YAC/B,WAAW,CAAC,GAAG;YACf,CAAC,CAAC,CAAC,WAAW,CAAC,WAAW,CAAC,MAAM,CAAC;YAClC,CAAC,CAAC,CAAC,WAAW,CAAC,WAAW,CAAC,UAAU,CAAC,EACtC,CAAC;YACD,IAAI,IAAI,CAAC,OAAO,EAAE,EAAE,CAAC;gBACnB,MAAM,IAAI,KAAK,CACb,6HAA6H,CAC9H,CAAC;YACJ,CAAC;YACD,OAAO,CAAC,MAAM,IAAI,CAAC,0BAA0B,CAAC;gBAC5C,UAAU,EAAE,WAAW,CAAC,UAAU;gBAClC,IAAI,EAAE,WAAW,CAAC,IAAI;aACvB,CAAC,CAAQ,CAAC;QACb,CAAC;QACD,OAAO,EAAE,CAAC;IACZ,CAAC;IAED;;;;OAIG;IACH,KAAK,CAAC,0BAA0B,CAAC,EAAE,UAAU,EAAE,IAAI,EAA8B;QAC/E,IAAI,CAAC,UAAU,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,UAAU,CAAC,EAAE,CAAC;YAC9C,MAAM,IAAI,KAAK,CAAC,+BAA+B,CAAC,CAAC;QACnD,CAAC;QAED,wCAAwC;QACxC,IAAI,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC5B,MAAM,IAAI,KAAK,CAAC,kCAAkC,CAAC,CAAC;QACtD,CAAC;QACD,MAAM,gBAAgB,GAAG,UAAU,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC;QAC/C,MAAM,eAAe,GAAG,UAAU,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC;QAC7C,MAAM,iBAAiB,GAAG;YACxB,SAAS,EAAE,gBAAgB;YAC3B,MAAM,EAAE,eAAe;YACvB,GAAG,EAAE,IAAI;YACT,IAAI;SACL,CAAC;QACF,MAAM,WAAW,GAAgB,MAAM,IAAI,CAAC,WAAW,CAAC,iBAAiB,CAAC,CAAC;QAE3E,MAAM,QAAQ,GAAG,WAAW,CAAC,gBAAgB,CAAC;QAC9C,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,WAAW,GAAG,QAAQ,CAAC,CAAC;QAChE,MAAM,WAAW,GAAG,QAAQ,GAAG,CAAC,CAAC;QACjC,2DAA2D;QAC3D,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,EAAE,GAAG,WAAW,CAAC,CAAC,QAAQ,EAAE,CAAC;QAErE,uGAAuG;QACvG,MAAM,UAAU,GAAG,IAAI,CAAC;QAExB,OAAO;YACL,SAAS,EAAE;gBACT,UAAU;gBACV,WAAW;gBACX,SAAS;gBACT,QAAQ;aACT;SACF,CAAC;IACJ,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,WAAW,CAAC,MAA0B;QAC1C,MAAM,KAAK,GAAuB,EAAE,CAAC;QACrC,IAAI,MAAM,IAAI,MAAM,CAAC,GAAG,EAAE,CAAC;YACzB,KAAK,CAAC,GAAG,GAAG,MAAM,CAAC,GAAG,CAAC;QACzB,CAAC;QACD,IAAI,MAAM,IAAI,MAAM,CAAC,SAAS,EAAE,CAAC;YAC/B,KAAK,CAAC,SAAS,GAAG,MAAM,CAAC,SAAS,CAAC;QACrC,CAAC;QACD,IAAI,MAAM,IAAI,MAAM,CAAC,IAAI,EAAE,CAAC;YAC1B,KAAK,CAAC,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC;QAC3B,CAAC;QACD,IAAI,MAAM,IAAI,MAAM,CAAC,MAAM,EAAE,CAAC;YAC5B,KAAK,CAAC,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC;QAC/B,CAAC;QACD,IAAI,MAAM,IAAI,MAAM,CAAC,IAAI,EAAE,CAAC;YAC1B,KAAK,CAAC,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC;QAC3B,CAAC;QAED,OAAO,MAAM,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,MAAM,EAAE,CAAC;IACzE,CAAC;IAED;;;;OAIG;IACH,MAAM,CAAC,SAAS,CAAC,EAAU;QACzB,MAAM,IAAI,GAAG,IAAA,gBAAM,EAAC,WAAW,CAAC,CAAC;QACjC,IAAI,CAAC,MAAM,CAAC,2BAAY,CAAC,OAAO,CAAC,cAAc,CAAC,EAAE,CAAC,EAAE,KAAK,CAAC,CAAC;QAC5D,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC;IACvB,CAAC;IAED,KAAK,CAAC,eAAe,CAAC,MAA4B;QAChD,uBAAuB;QACvB,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;;OAKG;IACH,cAAc,CAAC,MAA6B;QAC1C,qFAAqF;QACrF,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;YACxB,MAAM,IAAI,KAAK,CAAC,sDAAsD,MAAM,UAAU,OAAO,MAAM,GAAG,CAAC,CAAC;QAC1G,CAAC;QAED,IAAI,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,aAAa,CAAC,EAAE,CAAC;YAC5E,MAAM,IAAI,KAAK,CACb,kIAAkI,CACnI,CAAC;QACJ,CAAC;QAED,qFAAqF;QACrF,IAAI,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,aAAa,CAAC,EAAE,CAAC;YAC9E,MAAM,IAAI,KAAK,CAAC,mFAAmF,CAAC,CAAC;QACvG,CAAC;QAED,IAAI,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,UAAU,CAAC,EAAE,CAAC;YACxE,MAAM,IAAI,KAAK,CAAC,uCAAuC,MAAM,CAAC,UAAU,UAAU,OAAO,MAAM,CAAC,UAAU,GAAG,CAAC,CAAC;QACjH,CAAC;QAED,IAAI,CAAC,CAAC,CAAC,WAAW,CAAC,MAAM,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS,CAAC,MAAM,CAAC,aAAa,CAAC,EAAE,CAAC;YAC/E,MAAM,IAAI,KAAK,CACb,2CAA2C,MAAM,CAAC,aAAa,UAAU,OAAO,MAAM,CAAC,aAAa,GAAG,CACxG,CAAC;QACJ,CAAC;IACH,CAAC;IAED,QAAQ;QACN,OAAO,IAAI,CAAC,QAAQ,EAAE,CAAC,QAAQ,EAAE,KAAK,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,QAAQ,CAAC;IACrE,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,0BAA0B;QAC9B,IAAI,CAAC;YACH,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,+BAA+B,CAAC;gBACrD,OAAO,EAAE,KAAK;gBACd,MAAM,EAAE,cAAc;gBACtB,EAAE,EAAE,CAAC;aACN,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,IAAI,oBAAE,CAAC,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YACjD,OAAO,CAAC,GAAG,CAAC,mBAAmB,QAAQ,EAAE,CAAC,CAAC;YAC3C,OAAO,QAAQ,CAAC;QAClB,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,MAAM,IAAI,KAAK,CAAC,yBAAyB,CAAC,CAAC;QAC7C,CAAC;IACH,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,0BAA0B,CAAC,aAAqB,EAAE,IAAY,EAAE,EAAU,EAAE,IAAY;QAC5F,IAAI,CAAC;YACH,MAAM,GAAG,GAAG,MAAM,IAAI,CAAC,+BAA+B,CAAC;gBACrD,OAAO,EAAE,KAAK;gBACd,MAAM,EAAE,iBAAiB;gBACzB,MAAM,EAAE;oBACN;wBACE,IAAI;wBACJ,EAAE;wBACF,IAAI;qBACL;oBACD,QAAQ;iBACT;gBACD,EAAE,EAAE,CAAC;aACN,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,IAAI,oBAAE,CAAC,GAAG,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;YACjD,OAAO,CAAC,GAAG,CAAC,kBAAkB,QAAQ,EAAE,CAAC,CAAC;YAC1C,OAAO,QAAQ,CAAC;QAClB,CAAC;QAAC,OAAO,CAAC,EAAE,CAAC;YACX,MAAM,IAAI,KAAK,CACb,kFAAkF,aAAa,WAAW,EAAE,EAAE,CAC/G,CAAC;QACJ,CAAC;IACH,CAAC;;AA5pCH,sBA6pCC;AA5pCQ,wBAAkB,GAAG,4BAA4B,CAAC","sourcesContent":["/**\n * @prettier\n */\nimport { BigNumber } from 'bignumber.js';\nimport { bip32 } from '@bitgo/secp256k1';\nimport Keccak from 'keccak';\nimport * as secp256k1 from 'secp256k1';\nimport * as _ from 'lodash';\nimport {\n  AvalancheNetwork,\n  BaseCoin as StaticsBaseCoin,\n  CoinFamily,\n  coins,\n  ethGasConfigs,\n  EthereumNetwork,\n} from '@bitgo/statics';\nimport {\n  BaseCoin,\n  BaseTransaction,\n  BitGoBase,\n  common,\n  FeeEstimateOptions,\n  FullySignedTransaction,\n  getIsUnsignedSweep,\n  InvalidAddressError,\n  IWallet,\n  KeyPair,\n  MultisigType,\n  multisigTypes,\n  ParsedTransaction,\n  ParseTransactionOptions,\n  Recipient,\n  TransactionExplanation,\n  Util,\n  VerifyAddressOptions,\n} from '@bitgo/sdk-core';\nimport {\n  AbstractEthLikeNewCoins,\n  GetSendMethodArgsOptions,\n  optionalDeps,\n  RecoverOptions,\n  RecoveryInfo,\n  SendMethodArgs,\n  TransactionBuilder as EthTransactionBuilder,\n  TransactionPrebuild,\n} from '@bitgo/sdk-coin-eth';\nimport { getToken, isValidEthAddress } from './lib/utils';\nimport { KeyPair as AvaxcKeyPair, TransactionBuilder } from './lib';\nimport request from 'superagent';\nimport { BN, pubToAddress } from 'ethereumjs-util';\nimport { Buffer } from 'buffer';\nimport {\n  AvaxSignTransactionOptions,\n  BuildOptions,\n  ExplainTransactionOptions,\n  FeeEstimate,\n  HopParams,\n  HopPrebuild,\n  HopTransactionBuildOptions,\n  OfflineVaultTxInfo,\n  PrecreateBitGoOptions,\n  PresignTransactionOptions,\n  SignedTransaction,\n  SignFinalOptions,\n  VerifyAvaxcTransactionOptions,\n} from './iface';\nimport { AvaxpLib } from '@bitgo/sdk-coin-avaxp';\nimport { SignTransactionOptions } from '@bitgo/abstract-eth';\n\n/** COIN-1708 : Avaxc is added for CCR in WRW,\n * hence adding the feature for AbstractEthLikeNewCoins\n * Super class changed from BaseCoin to AbstractEthLikeNewCoins\n * @since Sept 2024\n */\nexport class AvaxC extends AbstractEthLikeNewCoins {\n  static hopTransactionSalt = 'bitgoHopAddressRequestSalt';\n\n  protected readonly _staticsCoin: Readonly<StaticsBaseCoin>;\n\n  protected constructor(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>) {\n    super(bitgo, staticsCoin);\n\n    if (!staticsCoin) {\n      throw new Error('missing required constructor parameter staticsCoin');\n    }\n\n    this._staticsCoin = staticsCoin;\n  }\n\n  static createInstance(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>): BaseCoin {\n    return new AvaxC(bitgo, staticsCoin);\n  }\n\n  getBaseFactor(): number {\n    return Math.pow(10, this._staticsCoin.decimalPlaces);\n  }\n\n  getChain(): string {\n    return this._staticsCoin.name;\n  }\n\n  /**\n   * Method to return the coin's network object\n   * @returns {BaseNetwork}\n   */\n  getNetwork(): EthereumNetwork {\n    return this._staticsCoin.network as EthereumNetwork;\n  }\n\n  /**\n   * Get the base chain that the coin exists on.\n   */\n  getBaseChain(): string {\n    return this.getChain();\n  }\n\n  getFamily(): CoinFamily {\n    return this._staticsCoin.family;\n  }\n\n  getFullName(): string {\n    return this._staticsCoin.fullName;\n  }\n\n  valuelessTransferAllowed(): boolean {\n    return true;\n  }\n\n  isValidAddress(address: string): boolean {\n    // also validate p-chain address for cross-chain txs\n    return !!address && (isValidEthAddress(address) || AvaxpLib.Utils.isValidAddress(address));\n  }\n\n  isToken(): boolean {\n    return false;\n  }\n\n  /** inherited doc */\n  getDefaultMultisigType(): MultisigType {\n    return multisigTypes.onchain;\n  }\n\n  generateKeyPair(seed?: Buffer): KeyPair {\n    const avaxKeyPair = seed ? new AvaxcKeyPair({ seed }) : new AvaxcKeyPair();\n    const extendedKeys = avaxKeyPair.getExtendedKeys();\n    return {\n      pub: extendedKeys.xpub,\n      prv: extendedKeys.xprv!,\n    };\n  }\n\n  async parseTransaction(params: ParseTransactionOptions): Promise<ParsedTransaction> {\n    return {};\n  }\n\n  async verifyAddress({ address }: VerifyAddressOptions): Promise<boolean> {\n    if (!this.isValidAddress(address)) {\n      throw new InvalidAddressError(`invalid address: ${address}`);\n    }\n    return true;\n  }\n\n  /**\n   * Verify that a transaction prebuild complies with the original intention\n   *\n   * @param params\n   * @param params.txParams params object passed to send\n   * @param params.txPrebuild prebuild object returned by server\n   * @param params.wallet Wallet object to obtain keys to verify against\n   * @returns {boolean}\n   */\n  async verifyTransaction(params: VerifyAvaxcTransactionOptions): Promise<boolean> {\n    const { txParams, txPrebuild, wallet } = params;\n    if (!txParams?.recipients || !txPrebuild?.recipients || !wallet) {\n      throw new Error(`missing params`);\n    }\n    if (txParams.hop && txParams.recipients.length > 1) {\n      throw new Error(`tx cannot be both a batch and hop transaction`);\n    }\n    if (txPrebuild.recipients.length > 1) {\n      throw new Error(\n        `${this.getChain()} doesn't support sending to more than 1 destination address within a single transaction. Try again, using only a single recipient.`\n      );\n    }\n    if (txParams.hop && txPrebuild.hopTransaction) {\n      // Check recipient amount for hop transaction\n      if (txParams.recipients.length !== 1) {\n        throw new Error(`hop transaction only supports 1 recipient but ${txParams.recipients.length} found`);\n      }\n      // Check tx sends to hop address\n      let expectedHopAddress;\n      if (txPrebuild.hopTransaction.type === 'Export') {\n        const decodedHopTx = await this.explainAtomicTransaction(txPrebuild.hopTransaction.tx);\n        expectedHopAddress = optionalDeps.ethUtil.stripHexPrefix(decodedHopTx.inputs[0].address);\n      } else {\n        const decodedHopTx = optionalDeps.EthTx.TransactionFactory.fromSerializedData(\n          optionalDeps.ethUtil.toBuffer(txPrebuild.hopTransaction.tx)\n        );\n        expectedHopAddress = optionalDeps.ethUtil.stripHexPrefix(decodedHopTx.getSenderAddress().toString());\n      }\n      const actualHopAddress = optionalDeps.ethUtil.stripHexPrefix(txPrebuild.recipients[0].address);\n      if (expectedHopAddress.toLowerCase() !== actualHopAddress.toLowerCase()) {\n        throw new Error('recipient address of txPrebuild does not match hop address');\n      }\n\n      // Convert TransactionRecipient array to Recipient array\n      const recipients: Recipient[] = txParams.recipients.map((r) => {\n        return {\n          address: r.address,\n          amount: typeof r.amount === 'number' ? r.amount.toString() : r.amount,\n        };\n      });\n\n      // Check destination address and amount\n      await this.validateHopPrebuild(wallet, txPrebuild.hopTransaction, { recipients });\n    } else if (txParams.recipients.length > 1) {\n      // Check total amount for batch transaction\n      let expectedTotalAmount = new BigNumber(0);\n      for (let i = 0; i < txParams.recipients.length; i++) {\n        expectedTotalAmount = expectedTotalAmount.plus(txParams.recipients[i].amount);\n      }\n      if (!expectedTotalAmount.isEqualTo(txPrebuild.recipients[0].amount)) {\n        throw new Error(\n          'batch transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client'\n        );\n      }\n    } else {\n      // Check recipient address and amount for normal transaction\n      if (txParams.recipients.length !== 1) {\n        throw new Error(`normal transaction only supports 1 recipient but ${txParams.recipients.length} found`);\n      }\n      const expectedAmount = new BigNumber(txParams.recipients[0].amount);\n      if (!expectedAmount.isEqualTo(txPrebuild.recipients[0].amount)) {\n        throw new Error(\n          'normal transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client'\n        );\n      }\n      if (\n        AvaxC.isAVAXCAddress(txParams.recipients[0].address) &&\n        txParams.recipients[0].address !== txPrebuild.recipients[0].address\n      ) {\n        throw new Error('destination address in normal txPrebuild does not match that in txParams supplied by client');\n      }\n    }\n    // Check coin is correct for all transaction types\n    if (!this.verifyCoin(txPrebuild)) {\n      throw new Error(`coin in txPrebuild did not match that in txParams supplied by client`);\n    }\n    return true;\n  }\n\n  private static isAVAXCAddress(address: string): boolean {\n    return !!address.match(/0x[a-fA-F0-9]{40}/);\n  }\n\n  verifyCoin(txPrebuild: TransactionPrebuild): boolean {\n    return txPrebuild.coin === this.getChain();\n  }\n\n  isValidPub(pub: string): boolean {\n    let valid = true;\n    try {\n      new AvaxcKeyPair({ pub });\n    } catch (e) {\n      valid = false;\n    }\n    return valid;\n  }\n\n  /**\n   * Check whether gas limit passed in by user are within our max and min bounds\n   * If they are not set, set them to the defaults\n   * @param {number} userGasLimit - user defined gas limit\n   * @returns {number} the gas limit to use for this transaction\n   */\n  setGasLimit(userGasLimit?: number): number {\n    if (!userGasLimit) {\n      return ethGasConfigs.defaultGasLimit;\n    }\n    const gasLimitMax = ethGasConfigs.maximumGasLimit;\n    const gasLimitMin = ethGasConfigs.minimumGasLimit;\n    if (userGasLimit < gasLimitMin || userGasLimit > gasLimitMax) {\n      throw new Error(`Gas limit must be between ${gasLimitMin} and ${gasLimitMax}`);\n    }\n    return userGasLimit;\n  }\n\n  /**\n   * Check whether the gas price passed in by user are within our max and min bounds\n   * If they are not set, set them to the defaults\n   * @param {number} userGasPrice - user defined gas price\n   * @returns the gas price to use for this transaction\n   */\n  setGasPrice(userGasPrice?: number): number {\n    if (!userGasPrice) {\n      return ethGasConfigs.defaultGasPrice;\n    }\n\n    const gasPriceMax = ethGasConfigs.maximumGasPrice;\n    const gasPriceMin = ethGasConfigs.minimumGasPrice;\n    if (userGasPrice < gasPriceMin || userGasPrice > gasPriceMax) {\n      throw new Error(`Gas price must be between ${gasPriceMin} and ${gasPriceMax}`);\n    }\n    return userGasPrice;\n  }\n\n  /**\n   * Make a query to avax.network for information such as balance, token balance, solidity calls\n   * @param {Object} query — key-value pairs of parameters to append after /api\n   * @param {string} apiKey - optional API key to use instead of the one from the environment\n   * @returns {Promise<Object>} response from avax.network\n   */\n  async recoveryBlockchainExplorerQuery(query: Record<string, any>, apiKey?: string): Promise<any> {\n    const response = await request\n      .post(common.Environments[this.bitgo.getEnv()].avaxcNetworkBaseUrl + '/ext/bc/C/rpc')\n      .send(query);\n\n    if (!response.ok) {\n      throw new Error('could not reach avax.network');\n    }\n\n    if (response.body.status === '0' && response.body.message === 'NOTOK') {\n      throw new Error('avax.network rate limit reached');\n    }\n    return response.body;\n  }\n\n  /**\n   * Queries public block explorer to get the next nonce that should be used for\n   * the given AVAXC address\n   * @param {string} address — address to fetch for\n   * @returns {number} address nonce\n   */\n  async getAddressNonce(address: string): Promise<number> {\n    // Get nonce for backup key (should be 0)\n    const result = await this.recoveryBlockchainExplorerQuery({\n      jsonrpc: '2.0',\n      method: 'eth_getTransactionCount',\n      params: [address, 'latest'],\n      id: 1,\n    });\n    if (!result || isNaN(result.result)) {\n      throw new Error('Unable to find next nonce from avax.network, got: ' + JSON.stringify(result));\n    }\n    const nonceHex = result.result;\n    return new optionalDeps.ethUtil.BN(nonceHex.slice(2), 16).toNumber();\n  }\n\n  /**\n   * Queries avax.network for the balance of an address\n   * @param {string} address - the AVAXC address\n   * @returns {Promise<BigNumber>} address balance\n   */\n  async queryAddressBalance(address: string): Promise<BN> {\n    const result = await this.recoveryBlockchainExplorerQuery({\n      jsonrpc: '2.0',\n      method: 'eth_getBalance',\n      params: [address, 'latest'],\n      id: 1,\n    });\n    // throw if the result does not exist or the result is not a valid number\n    if (!result || !result.result || isNaN(result.result)) {\n      throw new Error(`Could not obtain address balance for ${address} from avax.network, got: ${result.result}`);\n    }\n    const nativeBalanceHex = result.result;\n    return new optionalDeps.ethUtil.BN(nativeBalanceHex.slice(2), 16);\n  }\n\n  /**\n   * Queries avax.network for the token balance of an address\n   * @param {string} walletContractAddress - the AVAXC address\n   * @param {string} tokenContractAddress - the Token contract address\n   * @returns {Promise<BigNumber>} address balance\n   */\n  async queryAddressTokenBalance(tokenContractAddress: string, walletContractAddress: string): Promise<BN> {\n    // get token balance using contract call\n    const tokenBalanceData = optionalDeps.ethAbi\n      .simpleEncode('balanceOf(address)', walletContractAddress)\n      .toString('hex');\n    const tokenBalanceDataHex = optionalDeps.ethUtil.addHexPrefix(tokenBalanceData);\n    const result = await this.recoveryBlockchainExplorerQuery({\n      jsonrpc: '2.0',\n      method: 'eth_call',\n      params: [\n        {\n          to: tokenContractAddress,\n          data: tokenBalanceDataHex,\n        },\n        'latest',\n      ],\n      id: 1,\n    });\n    // throw if the result does not exist or the result is not a valid number\n    if (!result || !result.result || isNaN(result.result)) {\n      throw new Error(\n        `Could not obtain address token balance for ${walletContractAddress} from avax.network, got: ${result.result}`\n      );\n    }\n    const tokenBalanceHex = result.result;\n    return new optionalDeps.ethUtil.BN(tokenBalanceHex.slice(2), 16);\n  }\n\n  /**\n   * Queries the contract (via avax.network) for the next sequence ID\n   * @param {string} address - address of the contract\n   * @returns {Promise<number>} sequence ID\n   */\n  async querySequenceId(address: string): Promise<number> {\n    // Get sequence ID using contract call\n    const sequenceIdMethodSignature = optionalDeps.ethAbi.methodID('getNextSequenceId', []);\n    const sequenceIdArgs = optionalDeps.ethAbi.rawEncode([], []);\n    const sequenceIdData = Buffer.concat([sequenceIdMethodSignature, sequenceIdArgs]).toString('hex');\n    const sequenceIdDataHex = optionalDeps.ethUtil.addHexPrefix(sequenceIdData);\n    const result = await this.recoveryBlockchainExplorerQuery({\n      jsonrpc: '2.0',\n      method: 'eth_call',\n      params: [{ to: address, data: sequenceIdDataHex }, 'latest'],\n      id: 1,\n    });\n    if (!result || !result.result) {\n      throw new Error('Could not obtain sequence ID from avax.network, got: ' + result.result);\n    }\n    const sequenceIdHex = result.result;\n    return new optionalDeps.ethUtil.BN(sequenceIdHex.slice(2), 16).toNumber();\n  }\n\n  /**\n   * @param {Object} recipient - recipient info\n   * @param {number} expireTime - expiry time\n   * @param {number} contractSequenceId - sequence id\n   * @returns {(string|Array)} operation array\n   */\n  getOperation(recipient: Recipient, expireTime: number, contractSequenceId: number): (string | Buffer)[][] {\n    return [\n      ['string', 'address', 'uint', 'bytes', 'uint', 'uint'],\n      [\n        'ETHER',\n        new optionalDeps.ethUtil.BN(optionalDeps.ethUtil.stripHexPrefix(recipient.address), 16),\n        recipient.amount,\n        Buffer.from(optionalDeps.ethUtil.stripHexPrefix(optionalDeps.ethUtil.padToEven(recipient.data || '')), 'hex'),\n        expireTime,\n        contractSequenceId,\n      ],\n    ];\n  }\n\n  /**\n   * Calculate the operation hash in the same way solidity would\n   * @param {Recipient[]} recipients - tx recipients\n   * @param {number} expireTime - expiration time\n   * @param {number} contractSequenceId - contract sequence id\n   * @returns {string} operation hash\n   */\n  getOperationSha3ForExecuteAndConfirm(\n    recipients: Recipient[],\n    expireTime: number,\n    contractSequenceId: number\n  ): string {\n    if (!recipients || !Array.isArray(recipients)) {\n      throw new Error('expecting array of recipients');\n    }\n\n    // Right now we only support 1 recipient\n    if (recipients.length !== 1) {\n      throw new Error('must send to exactly 1 recipient');\n    }\n\n    if (!_.isNumber(expireTime)) {\n      throw new Error('expireTime must be number of seconds since epoch');\n    }\n\n    if (!_.isNumber(contractSequenceId)) {\n      throw new Error('contractSequenceId must be number');\n    }\n\n    // Check inputs\n    recipients.forEach(function (recipient) {\n      if (\n        !_.isString(recipient.address) ||\n        !optionalDeps.ethUtil.isValidAddress(optionalDeps.ethUtil.addHexPrefix(recipient.address))\n      ) {\n        throw new Error('Invalid address: ' + recipient.address);\n      }\n\n      let amount;\n      try {\n        amount = new BigNumber(recipient.amount);\n      } catch (e) {\n        throw new Error('Invalid amount for: ' + recipient.address + ' - should be numeric');\n      }\n\n      recipient.amount = amount.toFixed(0);\n\n      if (recipient.data && !_.isString(recipient.data)) {\n        throw new Error('Data for recipient ' + recipient.address + ' - should be of type hex string');\n      }\n    });\n\n    const recipient = recipients[0];\n    return optionalDeps.ethUtil.bufferToHex(\n      optionalDeps.ethAbi.soliditySHA3(...this.getOperation(recipient, expireTime, contractSequenceId))\n    );\n  }\n\n  /**\n   * Default expire time for a contract call (1 week)\n   * @returns {number} Time in seconds\n   */\n  getDefaultExpireTime(): number {\n    return Math.floor(new Date().getTime() / 1000) + 60 * 60 * 24 * 7;\n  }\n\n  /**\n   * Build arguments to call the send method on the wallet contract\n   * @param {Object} txInfo - data for send method args\n   * @returns {SendMethodArgs[]}\n   */\n  getSendMethodArgs(txInfo: GetSendMethodArgsOptions): SendMethodArgs[] {\n    // Method signature is\n    // sendMultiSig(address toAddress, uint value, bytes data, uint expireTime, uint sequenceId, bytes signature)\n    return [\n      {\n        name: 'toAddress',\n        type: 'address',\n        value: txInfo.recipient.address,\n      },\n      {\n        name: 'value',\n        type: 'uint',\n        value: txInfo.recipient.amount,\n      },\n      {\n        name: 'data',\n        type: 'bytes',\n        value: optionalDeps.ethUtil.toBuffer(optionalDeps.ethUtil.addHexPrefix(txInfo.recipient.data || '')),\n      },\n      {\n        name: 'expireTime',\n        type: 'uint',\n        value: txInfo.expireTime,\n      },\n      {\n        name: 'sequenceId',\n        type: 'uint',\n        value: txInfo.contractSequenceId,\n      },\n      {\n        name: 'signature',\n        type: 'bytes',\n        value: optionalDeps.ethUtil.toBuffer(optionalDeps.ethUtil.addHexPrefix(txInfo.signature)),\n      },\n    ];\n  }\n\n  /**\n   * Builds a funds recovery transaction without BitGo\n   * Steps:\n   * 1) Node query - how much money is in the account\n   * 2) Build transaction - build our transaction for the amount\n   * 3) Send signed build - send our signed build to a public node\n   * @param {Object} params The options with which to recover\n   * @param {string} params.userKey - [encrypted] xprv\n   * @param {string} params.backupKey - [encrypted] xprv or xpub if the xprv is held by a KRS provider\n   * @param {string} params.walletPassphrase - used to decrypt userKey and backupKey\n   * @param {string} params.walletContractAddress - the AVAXC address of the wallet contract\n   * @param {string} params.recoveryDestination - target address to send recovered funds to\n   * @returns {Promise<RecoveryInfo>} - recovery tx info\n   */\n  async recover(params: RecoverOptions): Promise<RecoveryInfo | OfflineVaultTxInfo> {\n    if (params.bitgoFeeAddress) {\n      return (await this.recoverEthLikeforEvmBasedRecovery(params)) as RecoveryInfo | OfflineVaultTxInfo;\n    }\n\n    if (_.isUndefined(params.userKey)) {\n      throw new Error('missing userKey');\n    }\n\n    if (_.isUndefined(params.backupKey)) {\n      throw new Error('missing backupKey');\n    }\n\n    if (_.isUndefined(params.walletPassphrase) && !params.userKey.startsWith('xpub')) {\n      throw new Error('missing wallet passphrase');\n    }\n\n    if (_.isUndefined(params.walletContractAddress) || !this.isValidAddress(params.walletContractAddress)) {\n      throw new Error('invalid walletContractAddress');\n    }\n\n    let tokenName;\n    if (params.tokenContractAddress) {\n      if (!this.isValidAddress(params.tokenContractAddress)) {\n        throw new Error('invalid tokenContractAddress');\n      }\n      const network = this.getNetwork();\n      const token = getToken(params.tokenContractAddress, network);\n      if (_.isUndefined(token)) {\n        throw new Error('token not supported');\n      }\n      tokenName = token.name;\n    }\n\n    if (_.isUndefined(params.recoveryDestination) || !this.isValidAddress(params.recoveryDestination)) {\n      throw new Error('invalid recoveryDestination');\n    }\n\n    // TODO (BG-56531): add support for krs\n    const isUnsignedSweep = getIsUnsignedSweep(params);\n\n    // Clean up whitespace from entered values\n    let userKey = params.userKey.replace(/\\s/g, '');\n    const backupKey = params.backupKey.replace(/\\s/g, '');\n\n    // Set new tx fees (using default config values from platform)\n    const gasLimit = new optionalDeps.ethUtil.BN(this.setGasLimit(params.gasLimit));\n    const gasPrice = params.eip1559\n      ? new optionalDeps.ethUtil.BN(params.eip1559.maxFeePerGas)\n      : new optionalDeps.ethUtil.BN(this.setGasPrice(params.gasPrice));\n    if (!userKey.startsWith('xpub') && !userKey.startsWith('xprv')) {\n      try {\n        userKey = this.bitgo.decrypt({\n          input: userKey,\n          password: params.walletPassphrase,\n        });\n      } catch (e) {\n        throw new Error(`Error decrypting user keychain: ${e.message}`);\n      }\n    }\n\n    let backupKeyAddress;\n    let backupSigningKey;\n    if (isUnsignedSweep) {\n      const backupKeyPair = new AvaxcKeyPair({ pub: backupKey });\n      backupKeyAddress = backupKeyPair.getAddress();\n    } else {\n      // Decrypt backup private key and get address\n      let backupPrv;\n\n      try {\n        backupPrv = this.bitgo.decrypt({\n          input: backupKey,\n          password: params.walletPassphrase,\n        });\n      } catch (e) {\n        throw new Error(`Error decrypting backup keychain: ${e.message}`);\n      }\n\n      const keyPair = new AvaxcKeyPair({ prv: backupPrv });\n      backupSigningKey = keyPair.getKeys().prv;\n      if (!backupSigningKey) {\n        throw new Error('no private key');\n      }\n      backupKeyAddress = keyPair.getAddress();\n    }\n    const backupKeyNonce = await this.getAddressNonce(backupKeyAddress);\n\n    // get balance of backupKey to ensure funds are available to pay fees\n    const backupKeyBalance = await this.queryAddressBalance(backupKeyAddress);\n\n    const totalGasNeeded = gasPrice.mul(gasLimit);\n    const weiToGwei = 10 ** 9;\n    if (backupKeyBalance.lt(totalGasNeeded)) {\n      throw new Error(\n        `Backup key address ${backupKeyAddress} has balance ${backupKeyBalance\n          .div(new BN(weiToGwei))\n          .toString()} Gwei.` +\n          `This address must have a balance of at least ${(totalGasNeeded / weiToGwei).toString()}` +\n          ` Gwei to perform recoveries. Try sending some AVAX to this address then retry.`\n      );\n    }\n\n    let txAmount;\n    if (params.tokenContractAddress) {\n      // get token balance of wallet\n      txAmount = await this.queryAddressTokenBalance(params.tokenContractAddress, params.walletContractAddress);\n    } else {\n      // get balance of wallet and deduct fees to get transaction amount\n      txAmount = await this.queryAddressBalance(params.walletContractAddress);\n    }\n\n    // build recipients object\n    const recipients = [\n      {\n        address: params.recoveryDestination,\n        amount: txAmount.toString(10),\n      },\n    ];\n\n    // Get sequence ID using contract call\n    // we need to wait between making two avax.network calls to avoid getting banned\n    await new Promise((resolve) => setTimeout(resolve, 1000));\n    const sequenceId = await this.querySequenceId(params.walletContractAddress);\n\n    let operationHash, signature;\n    // Get operation hash and sign it\n    if (!isUnsignedSweep) {\n      operationHash = this.getOperationSha3ForExecuteAndConfirm(recipients, this.getDefaultExpireTime(), sequenceId);\n      signature = Util.ethSignMsgHash(operationHash, Util.xprvToEthPrivateKey(userKey));\n\n      try {\n        Util.ecRecoverEthAddress(operationHash, signature);\n      } catch (e) {\n        throw new Error('Invalid signature');\n      }\n    }\n\n    const txInfo = {\n      recipient: recipients[0],\n      expireTime: this.getDefaultExpireTime(),\n      contractSequenceId: sequenceId,\n      operationHash,\n      signature,\n      gasLimit: gasLimit.toString(10),\n      tokenContractAddress: params.tokenContractAddress,\n    };\n\n    const txBuilder = this.getTransactionBuilder() as TransactionBuilder;\n    txBuilder.counter(backupKeyNonce);\n    txBuilder.contract(params.walletContractAddress);\n    let txFee;\n    if (params.eip1559) {\n      txFee = {\n        eip1559: {\n          maxPriorityFeePerGas: params.eip1559.maxPriorityFeePerGas,\n          maxFeePerGas: params.eip1559.maxFeePerGas,\n        },\n      };\n    } else {\n      txFee = { fee: gasPrice.toString() };\n    }\n    txBuilder.fee({\n      ...txFee,\n      gasLimit: gasLimit.toString(),\n    });\n    if (params.tokenContractAddress) {\n      txBuilder\n        .transfer()\n        .coin(tokenName)\n        .amount(recipients[0].amount)\n        .contractSequenceId(sequenceId)\n        .expirationTime(this.getDefaultExpireTime())\n        .to(params.recoveryDestination);\n    } else {\n      txBuilder\n        .transfer()\n        .coin(this.getChain())\n        .amount(recipients[0].amount)\n        .contractSequenceId(sequenceId)\n        .expirationTime(this.getDefaultExpireTime())\n        .to(params.recoveryDestination);\n    }\n\n    if (isUnsignedSweep) {\n      const tx = await txBuilder.build();\n      const response: OfflineVaultTxInfo = {\n        txHex: tx.toBroadcastFormat(),\n        userKey,\n        backupKey,\n        coin: this.getChain(),\n        token: tokenName,\n        gasPrice: optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),\n        gasLimit,\n        recipients: [txInfo.recipient],\n        walletContractAddress: tx.toJson().to,\n        amount: txInfo.recipient.amount,\n        backupKeyNonce,\n        eip1559: params.eip1559,\n      };\n      _.extend(response, txInfo);\n      response.nextContractSequenceId = response.contractSequenceId;\n      return response;\n    }\n\n    const userKeyPair = new AvaxcKeyPair({ prv: userKey });\n    txBuilder.transfer().key(userKeyPair.getKeys().prv!);\n    txBuilder.sign({ key: backupSigningKey });\n    const signedTx = await txBuilder.build();\n\n    return {\n      id: signedTx.toJson().id,\n      tx: signedTx.toBroadcastFormat(),\n    };\n  }\n\n  /**\n   * Create a new transaction builder for the current chain\n   * @return a new transaction builder\n   */\n  protected getTransactionBuilder(): EthTransactionBuilder {\n    return new TransactionBuilder(coins.get(this.getBaseChain()));\n  }\n\n  protected getAtomicBuilder(): AvaxpLib.TransactionBuilderFactory {\n    return new AvaxpLib.TransactionBuilderFactory(coins.get(this.getAvaxP()));\n  }\n\n  /**\n   * Explain a transaction from txHex, overriding BaseCoins\n   * transaction can be either atomic or eth txn.\n   * @param params The options with which to explain the transaction\n   */\n  async explainTransaction(params: ExplainTransactionOptions): Promise<TransactionExplanation> {\n    const txHex = params.txHex || (params.halfSigned && params.halfSigned.txHex);\n    if (!txHex) {\n      throw new Error('missing txHex in explain tx parameters');\n    }\n    if (params.crossChainType) {\n      return this.explainAtomicTransaction(txHex);\n    }\n    if (!params.feeInfo) {\n      throw new Error('missing feeInfo in explain tx parameters');\n    }\n    const txBuilder = this.getTransactionBuilder();\n    txBuilder.from(txHex);\n    const tx = await txBuilder.build();\n    return Object.assign(this.explainEVMTransaction(tx), { fee: params.feeInfo });\n  }\n\n  /**\n   * Explains an atomic transaction using atomic builder.\n   * @param txHex\n   * @private\n   */\n  private async explainAtomicTransaction(txHex: string) {\n    const txBuilder = this.getAtomicBuilder().from(txHex);\n    const tx = await txBuilder.build();\n    return tx.explainTransaction();\n  }\n\n  /**\n   * Verify signature for an atomic transaction using atomic builder.\n   * @param txHex\n   * @return true if signature is from the input address\n   * @private\n   */\n  private async verifySignatureForAtomicTransaction(txHex: string): Promise<boolean> {\n    const txBuilder = this.getAtomicBuilder().from(txHex);\n    const tx = await txBuilder.build();\n    const payload = tx.signablePayload;\n    const signatures = tx.signature.map((s) => Buffer.from(s, 'hex'));\n    const network = _.get(tx, '_network');\n    const recoverPubky = signatures.map((s) =>\n      AvaxpLib.Utils.recoverySignature(network as unknown as AvalancheNetwork, payload, s)\n    );\n    const expectedSenders = recoverPubky.map((r) => pubToAddress(r, true));\n    const senders = tx.inputs.map((i) => AvaxpLib.Utils.parseAddress(i.address));\n    return expectedSenders.every((e) => senders.some((sender) => e.equals(sender)));\n  }\n\n  /**\n   * Explains an EVM transaction using regular eth txn builder\n   * @param tx\n   * @private\n   */\n  private explainEVMTransaction(tx: BaseTransaction) {\n    const outputs = tx.outputs.map((output) => {\n      return {\n        address: output.address,\n        amount: output.value,\n      };\n    });\n    const displayOrder = ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee'];\n    return {\n      displayOrder,\n      id: tx.id,\n      outputs: outputs,\n      outputAmount: outputs\n        .reduce((accumulator, output) => accumulator.plus(output.amount), new BigNumber('0'))\n        .toFixed(0),\n      changeOutputs: [], // account based does not use change outputs\n      changeAmount: '0', // account base does not make change\n    };\n  }\n\n  /**\n   * Above is standard BaseCoins functions\n   * ================================================================================================================\n   * ================================================================================================================\n   * Below is transaction functions\n   */\n\n  /**\n   * Coin-specific things done before signing a transaction, i.e. verification\n   * @param params\n   */\n  async presignTransaction(params: PresignTransactionOptions): Promise<PresignTransactionOptions> {\n    if (!_.isUndefined(params.hopTransaction) && !_.isUndefined(params.wallet) && !_.isUndefined(params.buildParams)) {\n      await this.validateHopPrebuild(params.wallet, params.hopTransaction);\n    }\n    return params;\n  }\n\n  /**\n   * Modify prebuild after receiving it from the server. Add things like nlocktime\n   */\n  async postProcessPrebuild(params: TransactionPrebuild): Promise<TransactionPrebuild> {\n    if (!_.isUndefined(params.hopTransaction) && !_.isUndefined(params.wallet) && !_.isUndefined(params.buildParams)) {\n      await this.validateHopPrebuild(params.wallet, params.hopTransaction, params.buildParams);\n    }\n    return params;\n  }\n\n  /**\n   * Validates that the hop prebuild from the HSM is valid and correct\n   * @param wallet The wallet that the prebuild is for\n   * @param hopPrebuild The prebuild to validate\n   * @param originalParams The original parameters passed to prebuildTransaction\n   * @returns void\n   * @throws Error if The prebuild is invalid\n   */\n  async validateHopPrebuild(\n    wallet: IWallet,\n    hopPrebuild: HopPrebuild,\n    originalParams?: { recipients: Recipient[] }\n  ): Promise<void> {\n    const { tx, id, signature } = hopPrebuild;\n\n    // first, validate the HSM signature\n    const serverXpub = common.Environments[this.bitgo.getEnv()].hsmXpub;\n    const serverPubkeyBuffer: Buffer = bip32.fromBase58(serverXpub).publicKey;\n    const signatureBuffer: Buffer = Buffer.from(optionalDeps.ethUtil.stripHexPrefix(signature), 'hex');\n    const messageBuffer: Buffer =\n      hopPrebuild.type === 'Export' ? AvaxC.getTxHash(tx) : Buffer.from(optionalDeps.ethUtil.stripHexPrefix(id), 'hex');\n\n    const sig = new Uint8Array(signatureBuffer.length === 64 ? signatureBuffer : signatureBuffer.slice(1));\n    const isValidSignature: boolean = secp256k1.ecdsaVerify(sig, messageBuffer, serverPubkeyBuffer);\n    if (!isValidSignature) {\n      throw new Error(`Hop txid signature invalid`);\n    }\n\n    if (hopPrebuild.type === 'Export') {\n      const explainHopExportTx = await this.explainAtomicTransaction(tx);\n      // If original params are given, we can check them against the transaction prebuild params\n      if (!_.isNil(originalParams)) {\n        const { recipients } = originalParams;\n\n        // Then validate that the tx params actually equal the requested params to nano avax plus import tx fee.\n        const originalAmount = new BigNumber(recipients[0].amount).div(1e9).plus(1e6).toFixed(0);\n        const originalDestination: string | undefined = recipients[0].address;\n        const hopAmount = explainHopExportTx.outputAmount;\n        const hopDestination = explainHopExportTx.outputs[0].address;\n        if (originalAmount !== hopAmount) {\n          throw new Error(`Hop amount: ${hopAmount} does not equal original amount: ${originalAmount}`);\n        }\n        if (originalDestination && hopDestination.toLowerCase() !== originalDestination.toLowerCase()) {\n          throw new Error(\n            `Hop destination: ${hopDestination} does not equal original recipient: ${originalDestination}`\n          );\n        }\n      }\n      if (!(await this.verifySignatureForAtomicTransaction(tx))) {\n        throw new Error(`Invalid hop transaction signature, txid: ${id}`);\n      }\n    } else {\n      const builtHopTx = optionalDeps.EthTx.TransactionFactory.fromSerializedData(optionalDeps.ethUtil.toBuffer(tx));\n      // If original params are given, we can check them against the transaction prebuild params\n      if (!_.isNil(originalParams)) {\n        const { recipients } = originalParams;\n\n        // Then validate that the tx params actually equal the requested params\n        const originalAmount = new BigNumber(recipients[0].amount);\n        const originalDestination: string = recipients[0].address;\n\n        const hopAmount = new BigNumber(optionalDeps.ethUtil.bufferToHex(builtHopTx.value as unknown as Buffer));\n        if (!builtHopTx.to) {\n          throw new Error(`Transaction does not have a destination address`);\n        }\n        const hopDestination = builtHopTx.to.toString();\n        if (!hopAmount.eq(originalAmount)) {\n          throw new Error(`Hop amount: ${hopAmount} does not equal original amount: ${originalAmount}`);\n        }\n        if (hopDestination.toLowerCase() !== originalDestination.toLowerCase()) {\n          throw new Error(`Hop destination: ${hopDestination} does not equal original recipient: ${hopDestination}`);\n        }\n      }\n\n      if (!builtHopTx.verifySignature()) {\n        // We dont want to continue at all in this case, at risk of AVAX being stuck on the hop address\n        throw new Error(`Invalid hop transaction signature, txid: ${id}`);\n      }\n      if (optionalDeps.ethUtil.addHexPrefix(builtHopTx.hash().toString('hex')) !== id) {\n        throw new Error(`Signed hop txid does not equal actual txid`);\n      }\n    }\n  }\n\n  /**\n   * Helper function for signTransaction for the rare case that SDK is doing the second signature\n   * Note: we are expecting this to be called from the offline vault\n   * @param params.txPrebuild\n   * @param params.prv\n   * @returns {{txHex: string}}\n   */\n  async signFinal(params: SignFinalOptions): Promise<FullySignedTransaction> {\n    const keyPair = new AvaxcKeyPair({ prv: params.prv });\n    const signingKey = keyPair.getKeys().prv;\n    if (_.isUndefined(signingKey)) {\n      throw new Error('missing private key');\n    }\n\n    const txBuilder = this.getTransactionBuilder() as TransactionBuilder;\n    try {\n      txBuilder.from(params.txPrebuild!.halfSigned!.txHex);\n    } catch (e) {\n      throw new Error('invalid half-signed transaction');\n    }\n\n    txBuilder.sign({ key: signingKey });\n    const tx = await txBuilder.build();\n    return {\n      txHex: tx.toBroadcastFormat(),\n    };\n  }\n\n  /**\n   * Assemble half-sign prebuilt transaction\n   * @param params\n   */\n  async signTransaction(params: AvaxSignTransactionOptions | SignTransactionOptions): Promise<SignedTransaction> {\n    // Normally the SDK provides the first signature for an AVAXC tx,\n    // but for unsigned sweep recoveries it can provide the second and final one.\n    if (params.isLastSignature) {\n      // In this case when we're doing the second (final) signature, the logic is different.\n      return await this.signFinal(params as unknown as SignFinalOptions);\n    }\n\n    const txBuilder = this.getTransactionBuilder() as TransactionBuilder;\n    txBuilder.from(params.txPrebuild.txHex);\n    txBuilder.transfer().key(new AvaxcKeyPair({ prv: params.prv }).getKeys().prv!);\n    if (params.walletVersion) {\n      txBuilder.walletVersion(params.walletVersion);\n    }\n    const transaction = await txBuilder.build();\n\n    // we need to preserve the calldata of the recipients specified in the request for custodial transactions\n    let recipients = params.txPrebuild.recipients || (params.recipients as Recipient[] | undefined);\n    if (recipients === undefined) {\n      recipients = transaction.outputs.map((output) => ({ address: output.address, amount: output.value }));\n    }\n\n    const txParams = {\n      eip1559: params.txPrebuild.eip1559,\n      txHex: transaction.toBroadcastFormat(),\n      recipients: recipients,\n      expireTime: params.txPrebuild.expireTime,\n      hopTransaction: params.txPrebuild.hopTransaction,\n      custodianTransactionId: params.custodianTransactionId,\n    };\n\n    return { halfSigned: txParams };\n  }\n\n  /**\n   * Modify prebuild before sending it to the server. Add things like hop transaction params\n   * @param buildParams The whitelisted parameters for this prebuild\n   * @param buildParams.hop True if this should prebuild a hop tx, else false\n   * @param buildParams.recipients The recipients array of this transaction\n   * @param buildParams.wallet The wallet sending this tx\n   * @param buildParams.walletPassphrase the passphrase for this wallet\n   */\n  async getExtraPrebuildParams(buildParams: BuildOptions): Promise<BuildOptions> {\n    if (\n      !_.isUndefined(buildParams.hop) &&\n      buildParams.hop &&\n      !_.isUndefined(buildParams.wallet) &&\n      !_.isUndefined(buildParams.recipients)\n    ) {\n      if (this.isToken()) {\n        throw new Error(\n          `Hop transactions are not enabled for AVAXC tokens, nor are they necessary. Please remove the 'hop' parameter and try again.`\n        );\n      }\n      return (await this.createHopTransactionParams({\n        recipients: buildParams.recipients,\n        type: buildParams.type,\n      })) as any;\n    }\n    return {};\n  }\n\n  /**\n   * Creates the extra parameters needed to build a hop transaction\n   * @param {HopTransactionBuildOptions} The original build parameters\n   * @returns extra parameters object to merge with the original build parameters object and send to the platform\n   */\n  async createHopTransactionParams({ recipients, type }: HopTransactionBuildOptions): Promise<HopParams> {\n    if (!recipients || !Array.isArray(recipients)) {\n      throw new Error('expecting array of recipients');\n    }\n\n    // Right now we only support 1 recipient\n    if (recipients.length !== 1) {\n      throw new Error('must send to exactly 1 recipient');\n    }\n    const recipientAddress = recipients[0].address;\n    const recipientAmount = recipients[0].amount;\n    const feeEstimateParams = {\n      recipient: recipientAddress,\n      amount: recipientAmount,\n      hop: true,\n      type,\n    };\n    const feeEstimate: FeeEstimate = await this.feeEstimate(feeEstimateParams);\n\n    const gasLimit = feeEstimate.gasLimitEstimate;\n    const gasPrice = Math.round(feeEstimate.feeEstimate / gasLimit);\n    const gasPriceMax = gasPrice * 5;\n    // Payment id a random number so its different for every tx\n    const paymentId = Math.floor(Math.random() * 10000000000).toString();\n\n    // TODO(BG-62671): after completed [Wallet-platform] Remove use of userReqSig for avaxc hop transaction\n    const userReqSig = '0x';\n\n    return {\n      hopParams: {\n        userReqSig,\n        gasPriceMax,\n        paymentId,\n        gasLimit,\n      },\n    };\n  }\n\n  /**\n   * Fetch fee estimate information from the server\n   * @param {Object} params The params passed into the function\n   * @param {Boolean} [params.hop] True if we should estimate fee for a hop transaction\n   * @param {String} [params.recipient] The recipient of the transaction to estimate a send to\n   * @param {String} [params.data] The ETH tx data to estimate a send for\n   * @returns {Object} The fee info returned from the server\n   */\n  async feeEstimate(params: FeeEstimateOptions): Promise<FeeEstimate> {\n    const query: FeeEstimateOptions = {};\n    if (params && params.hop) {\n      query.hop = params.hop;\n    }\n    if (params && params.recipient) {\n      query.recipient = params.recipient;\n    }\n    if (params && params.data) {\n      query.data = params.data;\n    }\n    if (params && params.amount) {\n      query.amount = params.amount;\n    }\n    if (params && params.type) {\n      query.type = params.type;\n    }\n\n    return await this.bitgo.get(this.url('/tx/fee')).query(query).result();\n  }\n\n  /**\n   * Calculate tx hash like evm from tx hex.\n   * @param {string} tx\n   * @returns {Buffer} tx hash\n   */\n  static getTxHash(tx: string): Buffer {\n    const hash = Keccak('keccak256');\n    hash.update(optionalDeps.ethUtil.stripHexPrefix(tx), 'hex');\n    return hash.digest();\n  }\n\n  async isWalletAddress(params: VerifyAddressOptions): Promise<boolean> {\n    // TODO: Fix this later\n    return true;\n  }\n\n  /**\n   * Ensure either enterprise or newFeeAddress is passed, to know whether to create new key or use enterprise key\n   * @param params\n   * @param params.enterprise {String} the enterprise id to associate with this key\n   * @param params.newFeeAddress {Boolean} create a new fee address (enterprise not needed in this case)\n   */\n  preCreateBitGo(params: PrecreateBitGoOptions): void {\n    // We always need params object, since either enterprise or newFeeAddress is required\n    if (!_.isObject(params)) {\n      throw new Error(`preCreateBitGo must be passed a params object. Got ${params} (type ${typeof params})`);\n    }\n\n    if (_.isUndefined(params.enterprise) && _.isUndefined(params.newFeeAddress)) {\n      throw new Error(\n        'expecting enterprise when adding BitGo key. If you want to create a new AVAX bitgo key, set the newFeeAddress parameter to true.'\n      );\n    }\n\n    // Check whether key should be an enterprise key or a BitGo key for a new fee address\n    if (!_.isUndefined(params.enterprise) && !_.isUndefined(params.newFeeAddress)) {\n      throw new Error(`Incompatible arguments - cannot pass both enterprise and newFeeAddress parameter.`);\n    }\n\n    if (!_.isUndefined(params.enterprise) && !_.isString(params.enterprise)) {\n      throw new Error(`enterprise should be a string - got ${params.enterprise} (type ${typeof params.enterprise})`);\n    }\n\n    if (!_.isUndefined(params.newFeeAddress) && !_.isBoolean(params.newFeeAddress)) {\n      throw new Error(\n        `newFeeAddress should be a boolean - got ${params.newFeeAddress} (type ${typeof params.newFeeAddress})`\n      );\n    }\n  }\n\n  getAvaxP(): string {\n    return this.getChain().toString() === 'avaxc' ? 'avaxp' : 'tavaxp';\n  }\n\n  /**\n   * Fetch the gas price from the explorer\n   */\n  async getGasPriceFromExternalAPI(): Promise<BN> {\n    try {\n      const res = await this.recoveryBlockchainExplorerQuery({\n        jsonrpc: '2.0',\n        method: 'eth_gasPrice',\n        id: 1,\n      });\n      const gasPrice = new BN(res.result.slice(2), 16);\n      console.log(` Got gas price: ${gasPrice}`);\n      return gasPrice;\n    } catch (e) {\n      throw new Error('Failed to get gas price');\n    }\n  }\n\n  /**\n   * Fetch the gas limit from the explorer\n   * @param intendedChain\n   * @param from\n   * @param to\n   * @param data\n   */\n  async getGasLimitFromExternalAPI(intendedChain: string, from: string, to: string, data: string): Promise<BN> {\n    try {\n      const res = await this.recoveryBlockchainExplorerQuery({\n        jsonrpc: '2.0',\n        method: 'eth_estimateGas',\n        params: [\n          {\n            from,\n            to,\n            data,\n          },\n          'latest',\n        ],\n        id: 1,\n      });\n      const gasLimit = new BN(res.result.slice(2), 16);\n      console.log(`Got gas limit: ${gasLimit}`);\n      return gasLimit;\n    } catch (e) {\n      throw new Error(\n        `Failed to get gas limit. Please make sure to use the privateKey aka userKey of ${intendedChain} wallet ${to}`\n      );\n    }\n  }\n}\n"]}Выполнить команду
Для локальной разработки. Не используйте в интернете!