@bitgo/sdk-coin-avaxc 6.2.5 → 6.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.mocharc.yml +1 -1
- package/CHANGELOG.md +6 -0
- package/LICENSE +191 -0
- package/dist/src/avaxc.d.ts +263 -0
- package/dist/src/avaxc.d.ts.map +1 -0
- package/dist/src/avaxc.js +1105 -0
- package/dist/src/avaxcToken.d.ts +37 -0
- package/dist/src/avaxcToken.d.ts.map +1 -0
- package/dist/src/avaxcToken.js +79 -0
- package/dist/src/iface.d.ts +148 -0
- package/dist/src/iface.d.ts.map +1 -0
- package/dist/src/iface.js +3 -0
- package/dist/src/index.d.ts +6 -0
- package/dist/src/index.d.ts.map +1 -0
- package/dist/src/index.js +22 -0
- package/dist/src/lib/index.d.ts +7 -0
- package/dist/src/lib/index.d.ts.map +1 -0
- package/dist/src/lib/index.js +48 -0
- package/dist/src/lib/keyPair.d.ts +26 -0
- package/dist/src/lib/keyPair.d.ts.map +1 -0
- package/dist/src/lib/keyPair.js +63 -0
- package/dist/src/lib/resources.d.ts +12 -0
- package/dist/src/lib/resources.d.ts.map +1 -0
- package/dist/src/lib/resources.js +27 -0
- package/dist/src/lib/transactionBuilder.d.ts +17 -0
- package/dist/src/lib/transactionBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder.js +44 -0
- package/dist/src/lib/transferBuilder.d.ts +17 -0
- package/dist/src/lib/transferBuilder.d.ts.map +1 -0
- package/dist/src/lib/transferBuilder.js +25 -0
- package/dist/src/lib/utils.d.ts +39 -0
- package/dist/src/lib/utils.d.ts.map +1 -0
- package/dist/src/lib/utils.js +97 -0
- package/dist/src/lib/walletUtil.d.ts +22 -0
- package/dist/src/lib/walletUtil.d.ts.map +1 -0
- package/dist/src/lib/walletUtil.js +62 -0
- package/dist/src/register.d.ts +3 -0
- package/dist/src/register.d.ts.map +1 -0
- package/dist/src/register.js +15 -0
- package/dist/src/tavaxc.d.ts +11 -0
- package/dist/src/tavaxc.d.ts.map +1 -0
- package/dist/src/tavaxc.js +14 -0
- package/package.json +10 -10
|
@@ -0,0 +1,1105 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
19
|
+
var ownKeys = function(o) {
|
|
20
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
21
|
+
var ar = [];
|
|
22
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
23
|
+
return ar;
|
|
24
|
+
};
|
|
25
|
+
return ownKeys(o);
|
|
26
|
+
};
|
|
27
|
+
return function (mod) {
|
|
28
|
+
if (mod && mod.__esModule) return mod;
|
|
29
|
+
var result = {};
|
|
30
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
31
|
+
__setModuleDefault(result, mod);
|
|
32
|
+
return result;
|
|
33
|
+
};
|
|
34
|
+
})();
|
|
35
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
36
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
37
|
+
};
|
|
38
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
39
|
+
exports.AvaxC = void 0;
|
|
40
|
+
/**
|
|
41
|
+
* @prettier
|
|
42
|
+
*/
|
|
43
|
+
const bignumber_js_1 = require("bignumber.js");
|
|
44
|
+
const secp256k1_1 = require("@bitgo/secp256k1");
|
|
45
|
+
const keccak_1 = __importDefault(require("keccak"));
|
|
46
|
+
const secp256k1 = __importStar(require("secp256k1"));
|
|
47
|
+
const _ = __importStar(require("lodash"));
|
|
48
|
+
const statics_1 = require("@bitgo/statics");
|
|
49
|
+
const sdk_core_1 = require("@bitgo/sdk-core");
|
|
50
|
+
const sdk_coin_eth_1 = require("@bitgo/sdk-coin-eth");
|
|
51
|
+
const utils_1 = require("./lib/utils");
|
|
52
|
+
const lib_1 = require("./lib");
|
|
53
|
+
const superagent_1 = __importDefault(require("superagent"));
|
|
54
|
+
const ethereumjs_util_1 = require("ethereumjs-util");
|
|
55
|
+
const buffer_1 = require("buffer");
|
|
56
|
+
const sdk_coin_avaxp_1 = require("@bitgo/sdk-coin-avaxp");
|
|
57
|
+
/** COIN-1708 : Avaxc is added for CCR in WRW,
|
|
58
|
+
* hence adding the feature for AbstractEthLikeNewCoins
|
|
59
|
+
* Super class changed from BaseCoin to AbstractEthLikeNewCoins
|
|
60
|
+
* @since Sept 2024
|
|
61
|
+
*/
|
|
62
|
+
class AvaxC extends sdk_coin_eth_1.AbstractEthLikeNewCoins {
|
|
63
|
+
constructor(bitgo, staticsCoin) {
|
|
64
|
+
super(bitgo, staticsCoin);
|
|
65
|
+
if (!staticsCoin) {
|
|
66
|
+
throw new Error('missing required constructor parameter staticsCoin');
|
|
67
|
+
}
|
|
68
|
+
this._staticsCoin = staticsCoin;
|
|
69
|
+
}
|
|
70
|
+
static createInstance(bitgo, staticsCoin) {
|
|
71
|
+
return new AvaxC(bitgo, staticsCoin);
|
|
72
|
+
}
|
|
73
|
+
getBaseFactor() {
|
|
74
|
+
return Math.pow(10, this._staticsCoin.decimalPlaces);
|
|
75
|
+
}
|
|
76
|
+
getChain() {
|
|
77
|
+
return this._staticsCoin.name;
|
|
78
|
+
}
|
|
79
|
+
/**
|
|
80
|
+
* Method to return the coin's network object
|
|
81
|
+
* @returns {BaseNetwork}
|
|
82
|
+
*/
|
|
83
|
+
getNetwork() {
|
|
84
|
+
return this._staticsCoin.network;
|
|
85
|
+
}
|
|
86
|
+
/**
|
|
87
|
+
* Get the base chain that the coin exists on.
|
|
88
|
+
*/
|
|
89
|
+
getBaseChain() {
|
|
90
|
+
return this.getChain();
|
|
91
|
+
}
|
|
92
|
+
getFamily() {
|
|
93
|
+
return this._staticsCoin.family;
|
|
94
|
+
}
|
|
95
|
+
getFullName() {
|
|
96
|
+
return this._staticsCoin.fullName;
|
|
97
|
+
}
|
|
98
|
+
valuelessTransferAllowed() {
|
|
99
|
+
return true;
|
|
100
|
+
}
|
|
101
|
+
isValidAddress(address) {
|
|
102
|
+
// also validate p-chain address for cross-chain txs
|
|
103
|
+
return !!address && ((0, utils_1.isValidEthAddress)(address) || sdk_coin_avaxp_1.AvaxpLib.Utils.isValidAddress(address));
|
|
104
|
+
}
|
|
105
|
+
isToken() {
|
|
106
|
+
return false;
|
|
107
|
+
}
|
|
108
|
+
/** inherited doc */
|
|
109
|
+
getDefaultMultisigType() {
|
|
110
|
+
return sdk_core_1.multisigTypes.onchain;
|
|
111
|
+
}
|
|
112
|
+
generateKeyPair(seed) {
|
|
113
|
+
const avaxKeyPair = seed ? new lib_1.KeyPair({ seed }) : new lib_1.KeyPair();
|
|
114
|
+
const extendedKeys = avaxKeyPair.getExtendedKeys();
|
|
115
|
+
return {
|
|
116
|
+
pub: extendedKeys.xpub,
|
|
117
|
+
prv: extendedKeys.xprv,
|
|
118
|
+
};
|
|
119
|
+
}
|
|
120
|
+
async parseTransaction(params) {
|
|
121
|
+
return {};
|
|
122
|
+
}
|
|
123
|
+
async verifyAddress({ address }) {
|
|
124
|
+
if (!this.isValidAddress(address)) {
|
|
125
|
+
throw new sdk_core_1.InvalidAddressError(`invalid address: ${address}`);
|
|
126
|
+
}
|
|
127
|
+
return true;
|
|
128
|
+
}
|
|
129
|
+
/**
|
|
130
|
+
* Verify that a transaction prebuild complies with the original intention
|
|
131
|
+
*
|
|
132
|
+
* @param params
|
|
133
|
+
* @param params.txParams params object passed to send
|
|
134
|
+
* @param params.txPrebuild prebuild object returned by server
|
|
135
|
+
* @param params.wallet Wallet object to obtain keys to verify against
|
|
136
|
+
* @returns {boolean}
|
|
137
|
+
*/
|
|
138
|
+
async verifyTransaction(params) {
|
|
139
|
+
const { txParams, txPrebuild, wallet } = params;
|
|
140
|
+
if (!txParams?.recipients || !txPrebuild?.recipients || !wallet) {
|
|
141
|
+
throw new Error(`missing params`);
|
|
142
|
+
}
|
|
143
|
+
if (txParams.hop && txParams.recipients.length > 1) {
|
|
144
|
+
throw new Error(`tx cannot be both a batch and hop transaction`);
|
|
145
|
+
}
|
|
146
|
+
if (txPrebuild.recipients.length > 1) {
|
|
147
|
+
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.`);
|
|
148
|
+
}
|
|
149
|
+
if (txParams.hop && txPrebuild.hopTransaction) {
|
|
150
|
+
// Check recipient amount for hop transaction
|
|
151
|
+
if (txParams.recipients.length !== 1) {
|
|
152
|
+
throw new Error(`hop transaction only supports 1 recipient but ${txParams.recipients.length} found`);
|
|
153
|
+
}
|
|
154
|
+
// Check tx sends to hop address
|
|
155
|
+
let expectedHopAddress;
|
|
156
|
+
if (txPrebuild.hopTransaction.type === 'Export') {
|
|
157
|
+
const decodedHopTx = await this.explainAtomicTransaction(txPrebuild.hopTransaction.tx);
|
|
158
|
+
expectedHopAddress = sdk_coin_eth_1.optionalDeps.ethUtil.stripHexPrefix(decodedHopTx.inputs[0].address);
|
|
159
|
+
}
|
|
160
|
+
else {
|
|
161
|
+
const decodedHopTx = sdk_coin_eth_1.optionalDeps.EthTx.TransactionFactory.fromSerializedData(sdk_coin_eth_1.optionalDeps.ethUtil.toBuffer(txPrebuild.hopTransaction.tx));
|
|
162
|
+
expectedHopAddress = sdk_coin_eth_1.optionalDeps.ethUtil.stripHexPrefix(decodedHopTx.getSenderAddress().toString());
|
|
163
|
+
}
|
|
164
|
+
const actualHopAddress = sdk_coin_eth_1.optionalDeps.ethUtil.stripHexPrefix(txPrebuild.recipients[0].address);
|
|
165
|
+
if (expectedHopAddress.toLowerCase() !== actualHopAddress.toLowerCase()) {
|
|
166
|
+
throw new Error('recipient address of txPrebuild does not match hop address');
|
|
167
|
+
}
|
|
168
|
+
// Convert TransactionRecipient array to Recipient array
|
|
169
|
+
const recipients = txParams.recipients.map((r) => {
|
|
170
|
+
return {
|
|
171
|
+
address: r.address,
|
|
172
|
+
amount: typeof r.amount === 'number' ? r.amount.toString() : r.amount,
|
|
173
|
+
};
|
|
174
|
+
});
|
|
175
|
+
// Check destination address and amount
|
|
176
|
+
await this.validateHopPrebuild(wallet, txPrebuild.hopTransaction, { recipients });
|
|
177
|
+
}
|
|
178
|
+
else if (txParams.recipients.length > 1) {
|
|
179
|
+
// Check total amount for batch transaction
|
|
180
|
+
let expectedTotalAmount = new bignumber_js_1.BigNumber(0);
|
|
181
|
+
for (let i = 0; i < txParams.recipients.length; i++) {
|
|
182
|
+
expectedTotalAmount = expectedTotalAmount.plus(txParams.recipients[i].amount);
|
|
183
|
+
}
|
|
184
|
+
if (!expectedTotalAmount.isEqualTo(txPrebuild.recipients[0].amount)) {
|
|
185
|
+
throw new Error('batch transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client');
|
|
186
|
+
}
|
|
187
|
+
}
|
|
188
|
+
else {
|
|
189
|
+
// Check recipient address and amount for normal transaction
|
|
190
|
+
if (txParams.recipients.length !== 1) {
|
|
191
|
+
throw new Error(`normal transaction only supports 1 recipient but ${txParams.recipients.length} found`);
|
|
192
|
+
}
|
|
193
|
+
const expectedAmount = new bignumber_js_1.BigNumber(txParams.recipients[0].amount);
|
|
194
|
+
if (!expectedAmount.isEqualTo(txPrebuild.recipients[0].amount)) {
|
|
195
|
+
throw new Error('normal transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client');
|
|
196
|
+
}
|
|
197
|
+
if (AvaxC.isAVAXCAddress(txParams.recipients[0].address) &&
|
|
198
|
+
txParams.recipients[0].address !== txPrebuild.recipients[0].address) {
|
|
199
|
+
throw new Error('destination address in normal txPrebuild does not match that in txParams supplied by client');
|
|
200
|
+
}
|
|
201
|
+
}
|
|
202
|
+
// Check coin is correct for all transaction types
|
|
203
|
+
if (!this.verifyCoin(txPrebuild)) {
|
|
204
|
+
throw new Error(`coin in txPrebuild did not match that in txParams supplied by client`);
|
|
205
|
+
}
|
|
206
|
+
return true;
|
|
207
|
+
}
|
|
208
|
+
static isAVAXCAddress(address) {
|
|
209
|
+
return !!address.match(/0x[a-fA-F0-9]{40}/);
|
|
210
|
+
}
|
|
211
|
+
verifyCoin(txPrebuild) {
|
|
212
|
+
return txPrebuild.coin === this.getChain();
|
|
213
|
+
}
|
|
214
|
+
isValidPub(pub) {
|
|
215
|
+
let valid = true;
|
|
216
|
+
try {
|
|
217
|
+
new lib_1.KeyPair({ pub });
|
|
218
|
+
}
|
|
219
|
+
catch (e) {
|
|
220
|
+
valid = false;
|
|
221
|
+
}
|
|
222
|
+
return valid;
|
|
223
|
+
}
|
|
224
|
+
/**
|
|
225
|
+
* Check whether gas limit passed in by user are within our max and min bounds
|
|
226
|
+
* If they are not set, set them to the defaults
|
|
227
|
+
* @param {number} userGasLimit - user defined gas limit
|
|
228
|
+
* @returns {number} the gas limit to use for this transaction
|
|
229
|
+
*/
|
|
230
|
+
setGasLimit(userGasLimit) {
|
|
231
|
+
if (!userGasLimit) {
|
|
232
|
+
return statics_1.ethGasConfigs.defaultGasLimit;
|
|
233
|
+
}
|
|
234
|
+
const gasLimitMax = statics_1.ethGasConfigs.maximumGasLimit;
|
|
235
|
+
const gasLimitMin = statics_1.ethGasConfigs.minimumGasLimit;
|
|
236
|
+
if (userGasLimit < gasLimitMin || userGasLimit > gasLimitMax) {
|
|
237
|
+
throw new Error(`Gas limit must be between ${gasLimitMin} and ${gasLimitMax}`);
|
|
238
|
+
}
|
|
239
|
+
return userGasLimit;
|
|
240
|
+
}
|
|
241
|
+
/**
|
|
242
|
+
* Check whether the gas price passed in by user are within our max and min bounds
|
|
243
|
+
* If they are not set, set them to the defaults
|
|
244
|
+
* @param {number} userGasPrice - user defined gas price
|
|
245
|
+
* @returns the gas price to use for this transaction
|
|
246
|
+
*/
|
|
247
|
+
setGasPrice(userGasPrice) {
|
|
248
|
+
if (!userGasPrice) {
|
|
249
|
+
return statics_1.ethGasConfigs.defaultGasPrice;
|
|
250
|
+
}
|
|
251
|
+
const gasPriceMax = statics_1.ethGasConfigs.maximumGasPrice;
|
|
252
|
+
const gasPriceMin = statics_1.ethGasConfigs.minimumGasPrice;
|
|
253
|
+
if (userGasPrice < gasPriceMin || userGasPrice > gasPriceMax) {
|
|
254
|
+
throw new Error(`Gas price must be between ${gasPriceMin} and ${gasPriceMax}`);
|
|
255
|
+
}
|
|
256
|
+
return userGasPrice;
|
|
257
|
+
}
|
|
258
|
+
/**
|
|
259
|
+
* Make a query to avax.network for information such as balance, token balance, solidity calls
|
|
260
|
+
* @param {Object} query — key-value pairs of parameters to append after /api
|
|
261
|
+
* @param {string} apiKey - optional API key to use instead of the one from the environment
|
|
262
|
+
* @returns {Promise<Object>} response from avax.network
|
|
263
|
+
*/
|
|
264
|
+
async recoveryBlockchainExplorerQuery(query, apiKey) {
|
|
265
|
+
const response = await superagent_1.default
|
|
266
|
+
.post(sdk_core_1.common.Environments[this.bitgo.getEnv()].avaxcNetworkBaseUrl + '/ext/bc/C/rpc')
|
|
267
|
+
.send(query);
|
|
268
|
+
if (!response.ok) {
|
|
269
|
+
throw new Error('could not reach avax.network');
|
|
270
|
+
}
|
|
271
|
+
if (response.body.status === '0' && response.body.message === 'NOTOK') {
|
|
272
|
+
throw new Error('avax.network rate limit reached');
|
|
273
|
+
}
|
|
274
|
+
return response.body;
|
|
275
|
+
}
|
|
276
|
+
/**
|
|
277
|
+
* Queries public block explorer to get the next nonce that should be used for
|
|
278
|
+
* the given AVAXC address
|
|
279
|
+
* @param {string} address — address to fetch for
|
|
280
|
+
* @returns {number} address nonce
|
|
281
|
+
*/
|
|
282
|
+
async getAddressNonce(address) {
|
|
283
|
+
// Get nonce for backup key (should be 0)
|
|
284
|
+
const result = await this.recoveryBlockchainExplorerQuery({
|
|
285
|
+
jsonrpc: '2.0',
|
|
286
|
+
method: 'eth_getTransactionCount',
|
|
287
|
+
params: [address, 'latest'],
|
|
288
|
+
id: 1,
|
|
289
|
+
});
|
|
290
|
+
if (!result || isNaN(result.result)) {
|
|
291
|
+
throw new Error('Unable to find next nonce from avax.network, got: ' + JSON.stringify(result));
|
|
292
|
+
}
|
|
293
|
+
const nonceHex = result.result;
|
|
294
|
+
return new sdk_coin_eth_1.optionalDeps.ethUtil.BN(nonceHex.slice(2), 16).toNumber();
|
|
295
|
+
}
|
|
296
|
+
/**
|
|
297
|
+
* Queries avax.network for the balance of an address
|
|
298
|
+
* @param {string} address - the AVAXC address
|
|
299
|
+
* @returns {Promise<BigNumber>} address balance
|
|
300
|
+
*/
|
|
301
|
+
async queryAddressBalance(address) {
|
|
302
|
+
const result = await this.recoveryBlockchainExplorerQuery({
|
|
303
|
+
jsonrpc: '2.0',
|
|
304
|
+
method: 'eth_getBalance',
|
|
305
|
+
params: [address, 'latest'],
|
|
306
|
+
id: 1,
|
|
307
|
+
});
|
|
308
|
+
// throw if the result does not exist or the result is not a valid number
|
|
309
|
+
if (!result || !result.result || isNaN(result.result)) {
|
|
310
|
+
throw new Error(`Could not obtain address balance for ${address} from avax.network, got: ${result.result}`);
|
|
311
|
+
}
|
|
312
|
+
const nativeBalanceHex = result.result;
|
|
313
|
+
return new sdk_coin_eth_1.optionalDeps.ethUtil.BN(nativeBalanceHex.slice(2), 16);
|
|
314
|
+
}
|
|
315
|
+
/**
|
|
316
|
+
* Queries avax.network for the token balance of an address
|
|
317
|
+
* @param {string} walletContractAddress - the AVAXC address
|
|
318
|
+
* @param {string} tokenContractAddress - the Token contract address
|
|
319
|
+
* @returns {Promise<BigNumber>} address balance
|
|
320
|
+
*/
|
|
321
|
+
async queryAddressTokenBalance(tokenContractAddress, walletContractAddress) {
|
|
322
|
+
// get token balance using contract call
|
|
323
|
+
const tokenBalanceData = sdk_coin_eth_1.optionalDeps.ethAbi
|
|
324
|
+
.simpleEncode('balanceOf(address)', walletContractAddress)
|
|
325
|
+
.toString('hex');
|
|
326
|
+
const tokenBalanceDataHex = sdk_coin_eth_1.optionalDeps.ethUtil.addHexPrefix(tokenBalanceData);
|
|
327
|
+
const result = await this.recoveryBlockchainExplorerQuery({
|
|
328
|
+
jsonrpc: '2.0',
|
|
329
|
+
method: 'eth_call',
|
|
330
|
+
params: [
|
|
331
|
+
{
|
|
332
|
+
to: tokenContractAddress,
|
|
333
|
+
data: tokenBalanceDataHex,
|
|
334
|
+
},
|
|
335
|
+
'latest',
|
|
336
|
+
],
|
|
337
|
+
id: 1,
|
|
338
|
+
});
|
|
339
|
+
// throw if the result does not exist or the result is not a valid number
|
|
340
|
+
if (!result || !result.result || isNaN(result.result)) {
|
|
341
|
+
throw new Error(`Could not obtain address token balance for ${walletContractAddress} from avax.network, got: ${result.result}`);
|
|
342
|
+
}
|
|
343
|
+
const tokenBalanceHex = result.result;
|
|
344
|
+
return new sdk_coin_eth_1.optionalDeps.ethUtil.BN(tokenBalanceHex.slice(2), 16);
|
|
345
|
+
}
|
|
346
|
+
/**
|
|
347
|
+
* Queries the contract (via avax.network) for the next sequence ID
|
|
348
|
+
* @param {string} address - address of the contract
|
|
349
|
+
* @returns {Promise<number>} sequence ID
|
|
350
|
+
*/
|
|
351
|
+
async querySequenceId(address) {
|
|
352
|
+
// Get sequence ID using contract call
|
|
353
|
+
const sequenceIdMethodSignature = sdk_coin_eth_1.optionalDeps.ethAbi.methodID('getNextSequenceId', []);
|
|
354
|
+
const sequenceIdArgs = sdk_coin_eth_1.optionalDeps.ethAbi.rawEncode([], []);
|
|
355
|
+
const sequenceIdData = buffer_1.Buffer.concat([sequenceIdMethodSignature, sequenceIdArgs]).toString('hex');
|
|
356
|
+
const sequenceIdDataHex = sdk_coin_eth_1.optionalDeps.ethUtil.addHexPrefix(sequenceIdData);
|
|
357
|
+
const result = await this.recoveryBlockchainExplorerQuery({
|
|
358
|
+
jsonrpc: '2.0',
|
|
359
|
+
method: 'eth_call',
|
|
360
|
+
params: [{ to: address, data: sequenceIdDataHex }, 'latest'],
|
|
361
|
+
id: 1,
|
|
362
|
+
});
|
|
363
|
+
if (!result || !result.result) {
|
|
364
|
+
throw new Error('Could not obtain sequence ID from avax.network, got: ' + result.result);
|
|
365
|
+
}
|
|
366
|
+
const sequenceIdHex = result.result;
|
|
367
|
+
return new sdk_coin_eth_1.optionalDeps.ethUtil.BN(sequenceIdHex.slice(2), 16).toNumber();
|
|
368
|
+
}
|
|
369
|
+
/**
|
|
370
|
+
* @param {Object} recipient - recipient info
|
|
371
|
+
* @param {number} expireTime - expiry time
|
|
372
|
+
* @param {number} contractSequenceId - sequence id
|
|
373
|
+
* @returns {(string|Array)} operation array
|
|
374
|
+
*/
|
|
375
|
+
getOperation(recipient, expireTime, contractSequenceId) {
|
|
376
|
+
return [
|
|
377
|
+
['string', 'address', 'uint', 'bytes', 'uint', 'uint'],
|
|
378
|
+
[
|
|
379
|
+
'ETHER',
|
|
380
|
+
new sdk_coin_eth_1.optionalDeps.ethUtil.BN(sdk_coin_eth_1.optionalDeps.ethUtil.stripHexPrefix(recipient.address), 16),
|
|
381
|
+
recipient.amount,
|
|
382
|
+
buffer_1.Buffer.from(sdk_coin_eth_1.optionalDeps.ethUtil.stripHexPrefix(sdk_coin_eth_1.optionalDeps.ethUtil.padToEven(recipient.data || '')), 'hex'),
|
|
383
|
+
expireTime,
|
|
384
|
+
contractSequenceId,
|
|
385
|
+
],
|
|
386
|
+
];
|
|
387
|
+
}
|
|
388
|
+
/**
|
|
389
|
+
* Calculate the operation hash in the same way solidity would
|
|
390
|
+
* @param {Recipient[]} recipients - tx recipients
|
|
391
|
+
* @param {number} expireTime - expiration time
|
|
392
|
+
* @param {number} contractSequenceId - contract sequence id
|
|
393
|
+
* @returns {string} operation hash
|
|
394
|
+
*/
|
|
395
|
+
getOperationSha3ForExecuteAndConfirm(recipients, expireTime, contractSequenceId) {
|
|
396
|
+
if (!recipients || !Array.isArray(recipients)) {
|
|
397
|
+
throw new Error('expecting array of recipients');
|
|
398
|
+
}
|
|
399
|
+
// Right now we only support 1 recipient
|
|
400
|
+
if (recipients.length !== 1) {
|
|
401
|
+
throw new Error('must send to exactly 1 recipient');
|
|
402
|
+
}
|
|
403
|
+
if (!_.isNumber(expireTime)) {
|
|
404
|
+
throw new Error('expireTime must be number of seconds since epoch');
|
|
405
|
+
}
|
|
406
|
+
if (!_.isNumber(contractSequenceId)) {
|
|
407
|
+
throw new Error('contractSequenceId must be number');
|
|
408
|
+
}
|
|
409
|
+
// Check inputs
|
|
410
|
+
recipients.forEach(function (recipient) {
|
|
411
|
+
if (!_.isString(recipient.address) ||
|
|
412
|
+
!sdk_coin_eth_1.optionalDeps.ethUtil.isValidAddress(sdk_coin_eth_1.optionalDeps.ethUtil.addHexPrefix(recipient.address))) {
|
|
413
|
+
throw new Error('Invalid address: ' + recipient.address);
|
|
414
|
+
}
|
|
415
|
+
let amount;
|
|
416
|
+
try {
|
|
417
|
+
amount = new bignumber_js_1.BigNumber(recipient.amount);
|
|
418
|
+
}
|
|
419
|
+
catch (e) {
|
|
420
|
+
throw new Error('Invalid amount for: ' + recipient.address + ' - should be numeric');
|
|
421
|
+
}
|
|
422
|
+
recipient.amount = amount.toFixed(0);
|
|
423
|
+
if (recipient.data && !_.isString(recipient.data)) {
|
|
424
|
+
throw new Error('Data for recipient ' + recipient.address + ' - should be of type hex string');
|
|
425
|
+
}
|
|
426
|
+
});
|
|
427
|
+
const recipient = recipients[0];
|
|
428
|
+
return sdk_coin_eth_1.optionalDeps.ethUtil.bufferToHex(sdk_coin_eth_1.optionalDeps.ethAbi.soliditySHA3(...this.getOperation(recipient, expireTime, contractSequenceId)));
|
|
429
|
+
}
|
|
430
|
+
/**
|
|
431
|
+
* Default expire time for a contract call (1 week)
|
|
432
|
+
* @returns {number} Time in seconds
|
|
433
|
+
*/
|
|
434
|
+
getDefaultExpireTime() {
|
|
435
|
+
return Math.floor(new Date().getTime() / 1000) + 60 * 60 * 24 * 7;
|
|
436
|
+
}
|
|
437
|
+
/**
|
|
438
|
+
* Build arguments to call the send method on the wallet contract
|
|
439
|
+
* @param {Object} txInfo - data for send method args
|
|
440
|
+
* @returns {SendMethodArgs[]}
|
|
441
|
+
*/
|
|
442
|
+
getSendMethodArgs(txInfo) {
|
|
443
|
+
// Method signature is
|
|
444
|
+
// sendMultiSig(address toAddress, uint value, bytes data, uint expireTime, uint sequenceId, bytes signature)
|
|
445
|
+
return [
|
|
446
|
+
{
|
|
447
|
+
name: 'toAddress',
|
|
448
|
+
type: 'address',
|
|
449
|
+
value: txInfo.recipient.address,
|
|
450
|
+
},
|
|
451
|
+
{
|
|
452
|
+
name: 'value',
|
|
453
|
+
type: 'uint',
|
|
454
|
+
value: txInfo.recipient.amount,
|
|
455
|
+
},
|
|
456
|
+
{
|
|
457
|
+
name: 'data',
|
|
458
|
+
type: 'bytes',
|
|
459
|
+
value: sdk_coin_eth_1.optionalDeps.ethUtil.toBuffer(sdk_coin_eth_1.optionalDeps.ethUtil.addHexPrefix(txInfo.recipient.data || '')),
|
|
460
|
+
},
|
|
461
|
+
{
|
|
462
|
+
name: 'expireTime',
|
|
463
|
+
type: 'uint',
|
|
464
|
+
value: txInfo.expireTime,
|
|
465
|
+
},
|
|
466
|
+
{
|
|
467
|
+
name: 'sequenceId',
|
|
468
|
+
type: 'uint',
|
|
469
|
+
value: txInfo.contractSequenceId,
|
|
470
|
+
},
|
|
471
|
+
{
|
|
472
|
+
name: 'signature',
|
|
473
|
+
type: 'bytes',
|
|
474
|
+
value: sdk_coin_eth_1.optionalDeps.ethUtil.toBuffer(sdk_coin_eth_1.optionalDeps.ethUtil.addHexPrefix(txInfo.signature)),
|
|
475
|
+
},
|
|
476
|
+
];
|
|
477
|
+
}
|
|
478
|
+
/**
|
|
479
|
+
* Builds a funds recovery transaction without BitGo
|
|
480
|
+
* Steps:
|
|
481
|
+
* 1) Node query - how much money is in the account
|
|
482
|
+
* 2) Build transaction - build our transaction for the amount
|
|
483
|
+
* 3) Send signed build - send our signed build to a public node
|
|
484
|
+
* @param {Object} params The options with which to recover
|
|
485
|
+
* @param {string} params.userKey - [encrypted] xprv
|
|
486
|
+
* @param {string} params.backupKey - [encrypted] xprv or xpub if the xprv is held by a KRS provider
|
|
487
|
+
* @param {string} params.walletPassphrase - used to decrypt userKey and backupKey
|
|
488
|
+
* @param {string} params.walletContractAddress - the AVAXC address of the wallet contract
|
|
489
|
+
* @param {string} params.recoveryDestination - target address to send recovered funds to
|
|
490
|
+
* @returns {Promise<RecoveryInfo>} - recovery tx info
|
|
491
|
+
*/
|
|
492
|
+
async recover(params) {
|
|
493
|
+
if (params.bitgoFeeAddress) {
|
|
494
|
+
return (await this.recoverEthLikeforEvmBasedRecovery(params));
|
|
495
|
+
}
|
|
496
|
+
if (_.isUndefined(params.userKey)) {
|
|
497
|
+
throw new Error('missing userKey');
|
|
498
|
+
}
|
|
499
|
+
if (_.isUndefined(params.backupKey)) {
|
|
500
|
+
throw new Error('missing backupKey');
|
|
501
|
+
}
|
|
502
|
+
if (_.isUndefined(params.walletPassphrase) && !params.userKey.startsWith('xpub')) {
|
|
503
|
+
throw new Error('missing wallet passphrase');
|
|
504
|
+
}
|
|
505
|
+
if (_.isUndefined(params.walletContractAddress) || !this.isValidAddress(params.walletContractAddress)) {
|
|
506
|
+
throw new Error('invalid walletContractAddress');
|
|
507
|
+
}
|
|
508
|
+
let tokenName;
|
|
509
|
+
if (params.tokenContractAddress) {
|
|
510
|
+
if (!this.isValidAddress(params.tokenContractAddress)) {
|
|
511
|
+
throw new Error('invalid tokenContractAddress');
|
|
512
|
+
}
|
|
513
|
+
const network = this.getNetwork();
|
|
514
|
+
const token = (0, utils_1.getToken)(params.tokenContractAddress, network);
|
|
515
|
+
if (_.isUndefined(token)) {
|
|
516
|
+
throw new Error('token not supported');
|
|
517
|
+
}
|
|
518
|
+
tokenName = token.name;
|
|
519
|
+
}
|
|
520
|
+
if (_.isUndefined(params.recoveryDestination) || !this.isValidAddress(params.recoveryDestination)) {
|
|
521
|
+
throw new Error('invalid recoveryDestination');
|
|
522
|
+
}
|
|
523
|
+
// TODO (BG-56531): add support for krs
|
|
524
|
+
const isUnsignedSweep = (0, sdk_core_1.getIsUnsignedSweep)(params);
|
|
525
|
+
// Clean up whitespace from entered values
|
|
526
|
+
let userKey = params.userKey.replace(/\s/g, '');
|
|
527
|
+
const backupKey = params.backupKey.replace(/\s/g, '');
|
|
528
|
+
// Set new tx fees (using default config values from platform)
|
|
529
|
+
const gasLimit = new sdk_coin_eth_1.optionalDeps.ethUtil.BN(this.setGasLimit(params.gasLimit));
|
|
530
|
+
const gasPrice = params.eip1559
|
|
531
|
+
? new sdk_coin_eth_1.optionalDeps.ethUtil.BN(params.eip1559.maxFeePerGas)
|
|
532
|
+
: new sdk_coin_eth_1.optionalDeps.ethUtil.BN(this.setGasPrice(params.gasPrice));
|
|
533
|
+
if (!userKey.startsWith('xpub') && !userKey.startsWith('xprv')) {
|
|
534
|
+
try {
|
|
535
|
+
userKey = this.bitgo.decrypt({
|
|
536
|
+
input: userKey,
|
|
537
|
+
password: params.walletPassphrase,
|
|
538
|
+
});
|
|
539
|
+
}
|
|
540
|
+
catch (e) {
|
|
541
|
+
throw new Error(`Error decrypting user keychain: ${e.message}`);
|
|
542
|
+
}
|
|
543
|
+
}
|
|
544
|
+
let backupKeyAddress;
|
|
545
|
+
let backupSigningKey;
|
|
546
|
+
if (isUnsignedSweep) {
|
|
547
|
+
const backupKeyPair = new lib_1.KeyPair({ pub: backupKey });
|
|
548
|
+
backupKeyAddress = backupKeyPair.getAddress();
|
|
549
|
+
}
|
|
550
|
+
else {
|
|
551
|
+
// Decrypt backup private key and get address
|
|
552
|
+
let backupPrv;
|
|
553
|
+
try {
|
|
554
|
+
backupPrv = this.bitgo.decrypt({
|
|
555
|
+
input: backupKey,
|
|
556
|
+
password: params.walletPassphrase,
|
|
557
|
+
});
|
|
558
|
+
}
|
|
559
|
+
catch (e) {
|
|
560
|
+
throw new Error(`Error decrypting backup keychain: ${e.message}`);
|
|
561
|
+
}
|
|
562
|
+
const keyPair = new lib_1.KeyPair({ prv: backupPrv });
|
|
563
|
+
backupSigningKey = keyPair.getKeys().prv;
|
|
564
|
+
if (!backupSigningKey) {
|
|
565
|
+
throw new Error('no private key');
|
|
566
|
+
}
|
|
567
|
+
backupKeyAddress = keyPair.getAddress();
|
|
568
|
+
}
|
|
569
|
+
const backupKeyNonce = await this.getAddressNonce(backupKeyAddress);
|
|
570
|
+
// get balance of backupKey to ensure funds are available to pay fees
|
|
571
|
+
const backupKeyBalance = await this.queryAddressBalance(backupKeyAddress);
|
|
572
|
+
const totalGasNeeded = gasPrice.mul(gasLimit);
|
|
573
|
+
const weiToGwei = 10 ** 9;
|
|
574
|
+
if (backupKeyBalance.lt(totalGasNeeded)) {
|
|
575
|
+
throw new Error(`Backup key address ${backupKeyAddress} has balance ${backupKeyBalance
|
|
576
|
+
.div(new ethereumjs_util_1.BN(weiToGwei))
|
|
577
|
+
.toString()} Gwei.` +
|
|
578
|
+
`This address must have a balance of at least ${(totalGasNeeded / weiToGwei).toString()}` +
|
|
579
|
+
` Gwei to perform recoveries. Try sending some AVAX to this address then retry.`);
|
|
580
|
+
}
|
|
581
|
+
let txAmount;
|
|
582
|
+
if (params.tokenContractAddress) {
|
|
583
|
+
// get token balance of wallet
|
|
584
|
+
txAmount = await this.queryAddressTokenBalance(params.tokenContractAddress, params.walletContractAddress);
|
|
585
|
+
}
|
|
586
|
+
else {
|
|
587
|
+
// get balance of wallet and deduct fees to get transaction amount
|
|
588
|
+
txAmount = await this.queryAddressBalance(params.walletContractAddress);
|
|
589
|
+
}
|
|
590
|
+
// build recipients object
|
|
591
|
+
const recipients = [
|
|
592
|
+
{
|
|
593
|
+
address: params.recoveryDestination,
|
|
594
|
+
amount: txAmount.toString(10),
|
|
595
|
+
},
|
|
596
|
+
];
|
|
597
|
+
// Get sequence ID using contract call
|
|
598
|
+
// we need to wait between making two avax.network calls to avoid getting banned
|
|
599
|
+
await new Promise((resolve) => setTimeout(resolve, 1000));
|
|
600
|
+
const sequenceId = await this.querySequenceId(params.walletContractAddress);
|
|
601
|
+
let operationHash, signature;
|
|
602
|
+
// Get operation hash and sign it
|
|
603
|
+
if (!isUnsignedSweep) {
|
|
604
|
+
operationHash = this.getOperationSha3ForExecuteAndConfirm(recipients, this.getDefaultExpireTime(), sequenceId);
|
|
605
|
+
signature = sdk_core_1.Util.ethSignMsgHash(operationHash, sdk_core_1.Util.xprvToEthPrivateKey(userKey));
|
|
606
|
+
try {
|
|
607
|
+
sdk_core_1.Util.ecRecoverEthAddress(operationHash, signature);
|
|
608
|
+
}
|
|
609
|
+
catch (e) {
|
|
610
|
+
throw new Error('Invalid signature');
|
|
611
|
+
}
|
|
612
|
+
}
|
|
613
|
+
const txInfo = {
|
|
614
|
+
recipient: recipients[0],
|
|
615
|
+
expireTime: this.getDefaultExpireTime(),
|
|
616
|
+
contractSequenceId: sequenceId,
|
|
617
|
+
operationHash,
|
|
618
|
+
signature,
|
|
619
|
+
gasLimit: gasLimit.toString(10),
|
|
620
|
+
tokenContractAddress: params.tokenContractAddress,
|
|
621
|
+
};
|
|
622
|
+
const txBuilder = this.getTransactionBuilder();
|
|
623
|
+
txBuilder.counter(backupKeyNonce);
|
|
624
|
+
txBuilder.contract(params.walletContractAddress);
|
|
625
|
+
let txFee;
|
|
626
|
+
if (params.eip1559) {
|
|
627
|
+
txFee = {
|
|
628
|
+
eip1559: {
|
|
629
|
+
maxPriorityFeePerGas: params.eip1559.maxPriorityFeePerGas,
|
|
630
|
+
maxFeePerGas: params.eip1559.maxFeePerGas,
|
|
631
|
+
},
|
|
632
|
+
};
|
|
633
|
+
}
|
|
634
|
+
else {
|
|
635
|
+
txFee = { fee: gasPrice.toString() };
|
|
636
|
+
}
|
|
637
|
+
txBuilder.fee({
|
|
638
|
+
...txFee,
|
|
639
|
+
gasLimit: gasLimit.toString(),
|
|
640
|
+
});
|
|
641
|
+
if (params.tokenContractAddress) {
|
|
642
|
+
txBuilder
|
|
643
|
+
.transfer()
|
|
644
|
+
.coin(tokenName)
|
|
645
|
+
.amount(recipients[0].amount)
|
|
646
|
+
.contractSequenceId(sequenceId)
|
|
647
|
+
.expirationTime(this.getDefaultExpireTime())
|
|
648
|
+
.to(params.recoveryDestination);
|
|
649
|
+
}
|
|
650
|
+
else {
|
|
651
|
+
txBuilder
|
|
652
|
+
.transfer()
|
|
653
|
+
.coin(this.getChain())
|
|
654
|
+
.amount(recipients[0].amount)
|
|
655
|
+
.contractSequenceId(sequenceId)
|
|
656
|
+
.expirationTime(this.getDefaultExpireTime())
|
|
657
|
+
.to(params.recoveryDestination);
|
|
658
|
+
}
|
|
659
|
+
if (isUnsignedSweep) {
|
|
660
|
+
const tx = await txBuilder.build();
|
|
661
|
+
const response = {
|
|
662
|
+
txHex: tx.toBroadcastFormat(),
|
|
663
|
+
userKey,
|
|
664
|
+
backupKey,
|
|
665
|
+
coin: this.getChain(),
|
|
666
|
+
token: tokenName,
|
|
667
|
+
gasPrice: sdk_coin_eth_1.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
|
|
668
|
+
gasLimit,
|
|
669
|
+
recipients: [txInfo.recipient],
|
|
670
|
+
walletContractAddress: tx.toJson().to,
|
|
671
|
+
amount: txInfo.recipient.amount,
|
|
672
|
+
backupKeyNonce,
|
|
673
|
+
eip1559: params.eip1559,
|
|
674
|
+
};
|
|
675
|
+
_.extend(response, txInfo);
|
|
676
|
+
response.nextContractSequenceId = response.contractSequenceId;
|
|
677
|
+
return response;
|
|
678
|
+
}
|
|
679
|
+
const userKeyPair = new lib_1.KeyPair({ prv: userKey });
|
|
680
|
+
txBuilder.transfer().key(userKeyPair.getKeys().prv);
|
|
681
|
+
txBuilder.sign({ key: backupSigningKey });
|
|
682
|
+
const signedTx = await txBuilder.build();
|
|
683
|
+
return {
|
|
684
|
+
id: signedTx.toJson().id,
|
|
685
|
+
tx: signedTx.toBroadcastFormat(),
|
|
686
|
+
};
|
|
687
|
+
}
|
|
688
|
+
/**
|
|
689
|
+
* Create a new transaction builder for the current chain
|
|
690
|
+
* @return a new transaction builder
|
|
691
|
+
*/
|
|
692
|
+
getTransactionBuilder() {
|
|
693
|
+
return new lib_1.TransactionBuilder(statics_1.coins.get(this.getBaseChain()));
|
|
694
|
+
}
|
|
695
|
+
getAtomicBuilder() {
|
|
696
|
+
return new sdk_coin_avaxp_1.AvaxpLib.TransactionBuilderFactory(statics_1.coins.get(this.getAvaxP()));
|
|
697
|
+
}
|
|
698
|
+
/**
|
|
699
|
+
* Explain a transaction from txHex, overriding BaseCoins
|
|
700
|
+
* transaction can be either atomic or eth txn.
|
|
701
|
+
* @param params The options with which to explain the transaction
|
|
702
|
+
*/
|
|
703
|
+
async explainTransaction(params) {
|
|
704
|
+
const txHex = params.txHex || (params.halfSigned && params.halfSigned.txHex);
|
|
705
|
+
if (!txHex) {
|
|
706
|
+
throw new Error('missing txHex in explain tx parameters');
|
|
707
|
+
}
|
|
708
|
+
if (params.crossChainType) {
|
|
709
|
+
return this.explainAtomicTransaction(txHex);
|
|
710
|
+
}
|
|
711
|
+
if (!params.feeInfo) {
|
|
712
|
+
throw new Error('missing feeInfo in explain tx parameters');
|
|
713
|
+
}
|
|
714
|
+
const txBuilder = this.getTransactionBuilder();
|
|
715
|
+
txBuilder.from(txHex);
|
|
716
|
+
const tx = await txBuilder.build();
|
|
717
|
+
return Object.assign(this.explainEVMTransaction(tx), { fee: params.feeInfo });
|
|
718
|
+
}
|
|
719
|
+
/**
|
|
720
|
+
* Explains an atomic transaction using atomic builder.
|
|
721
|
+
* @param txHex
|
|
722
|
+
* @private
|
|
723
|
+
*/
|
|
724
|
+
async explainAtomicTransaction(txHex) {
|
|
725
|
+
const txBuilder = this.getAtomicBuilder().from(txHex);
|
|
726
|
+
const tx = await txBuilder.build();
|
|
727
|
+
return tx.explainTransaction();
|
|
728
|
+
}
|
|
729
|
+
/**
|
|
730
|
+
* Verify signature for an atomic transaction using atomic builder.
|
|
731
|
+
* @param txHex
|
|
732
|
+
* @return true if signature is from the input address
|
|
733
|
+
* @private
|
|
734
|
+
*/
|
|
735
|
+
async verifySignatureForAtomicTransaction(txHex) {
|
|
736
|
+
const txBuilder = this.getAtomicBuilder().from(txHex);
|
|
737
|
+
const tx = await txBuilder.build();
|
|
738
|
+
const payload = tx.signablePayload;
|
|
739
|
+
const signatures = tx.signature.map((s) => buffer_1.Buffer.from(s, 'hex'));
|
|
740
|
+
const network = _.get(tx, '_network');
|
|
741
|
+
const recoverPubky = signatures.map((s) => sdk_coin_avaxp_1.AvaxpLib.Utils.recoverySignature(network, payload, s));
|
|
742
|
+
const expectedSenders = recoverPubky.map((r) => (0, ethereumjs_util_1.pubToAddress)(r, true));
|
|
743
|
+
const senders = tx.inputs.map((i) => sdk_coin_avaxp_1.AvaxpLib.Utils.parseAddress(i.address));
|
|
744
|
+
return expectedSenders.every((e) => senders.some((sender) => e.equals(sender)));
|
|
745
|
+
}
|
|
746
|
+
/**
|
|
747
|
+
* Explains an EVM transaction using regular eth txn builder
|
|
748
|
+
* @param tx
|
|
749
|
+
* @private
|
|
750
|
+
*/
|
|
751
|
+
explainEVMTransaction(tx) {
|
|
752
|
+
const outputs = tx.outputs.map((output) => {
|
|
753
|
+
return {
|
|
754
|
+
address: output.address,
|
|
755
|
+
amount: output.value,
|
|
756
|
+
};
|
|
757
|
+
});
|
|
758
|
+
const displayOrder = ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee'];
|
|
759
|
+
return {
|
|
760
|
+
displayOrder,
|
|
761
|
+
id: tx.id,
|
|
762
|
+
outputs: outputs,
|
|
763
|
+
outputAmount: outputs
|
|
764
|
+
.reduce((accumulator, output) => accumulator.plus(output.amount), new bignumber_js_1.BigNumber('0'))
|
|
765
|
+
.toFixed(0),
|
|
766
|
+
changeOutputs: [], // account based does not use change outputs
|
|
767
|
+
changeAmount: '0', // account base does not make change
|
|
768
|
+
};
|
|
769
|
+
}
|
|
770
|
+
/**
|
|
771
|
+
* Above is standard BaseCoins functions
|
|
772
|
+
* ================================================================================================================
|
|
773
|
+
* ================================================================================================================
|
|
774
|
+
* Below is transaction functions
|
|
775
|
+
*/
|
|
776
|
+
/**
|
|
777
|
+
* Coin-specific things done before signing a transaction, i.e. verification
|
|
778
|
+
* @param params
|
|
779
|
+
*/
|
|
780
|
+
async presignTransaction(params) {
|
|
781
|
+
if (!_.isUndefined(params.hopTransaction) && !_.isUndefined(params.wallet) && !_.isUndefined(params.buildParams)) {
|
|
782
|
+
await this.validateHopPrebuild(params.wallet, params.hopTransaction);
|
|
783
|
+
}
|
|
784
|
+
return params;
|
|
785
|
+
}
|
|
786
|
+
/**
|
|
787
|
+
* Modify prebuild after receiving it from the server. Add things like nlocktime
|
|
788
|
+
*/
|
|
789
|
+
async postProcessPrebuild(params) {
|
|
790
|
+
if (!_.isUndefined(params.hopTransaction) && !_.isUndefined(params.wallet) && !_.isUndefined(params.buildParams)) {
|
|
791
|
+
await this.validateHopPrebuild(params.wallet, params.hopTransaction, params.buildParams);
|
|
792
|
+
}
|
|
793
|
+
return params;
|
|
794
|
+
}
|
|
795
|
+
/**
|
|
796
|
+
* Validates that the hop prebuild from the HSM is valid and correct
|
|
797
|
+
* @param wallet The wallet that the prebuild is for
|
|
798
|
+
* @param hopPrebuild The prebuild to validate
|
|
799
|
+
* @param originalParams The original parameters passed to prebuildTransaction
|
|
800
|
+
* @returns void
|
|
801
|
+
* @throws Error if The prebuild is invalid
|
|
802
|
+
*/
|
|
803
|
+
async validateHopPrebuild(wallet, hopPrebuild, originalParams) {
|
|
804
|
+
const { tx, id, signature } = hopPrebuild;
|
|
805
|
+
// first, validate the HSM signature
|
|
806
|
+
const serverXpub = sdk_core_1.common.Environments[this.bitgo.getEnv()].hsmXpub;
|
|
807
|
+
const serverPubkeyBuffer = secp256k1_1.bip32.fromBase58(serverXpub).publicKey;
|
|
808
|
+
const signatureBuffer = buffer_1.Buffer.from(sdk_coin_eth_1.optionalDeps.ethUtil.stripHexPrefix(signature), 'hex');
|
|
809
|
+
const messageBuffer = hopPrebuild.type === 'Export' ? AvaxC.getTxHash(tx) : buffer_1.Buffer.from(sdk_coin_eth_1.optionalDeps.ethUtil.stripHexPrefix(id), 'hex');
|
|
810
|
+
const sig = new Uint8Array(signatureBuffer.length === 64 ? signatureBuffer : signatureBuffer.slice(1));
|
|
811
|
+
const isValidSignature = secp256k1.ecdsaVerify(sig, messageBuffer, serverPubkeyBuffer);
|
|
812
|
+
if (!isValidSignature) {
|
|
813
|
+
throw new Error(`Hop txid signature invalid`);
|
|
814
|
+
}
|
|
815
|
+
if (hopPrebuild.type === 'Export') {
|
|
816
|
+
const explainHopExportTx = await this.explainAtomicTransaction(tx);
|
|
817
|
+
// If original params are given, we can check them against the transaction prebuild params
|
|
818
|
+
if (!_.isNil(originalParams)) {
|
|
819
|
+
const { recipients } = originalParams;
|
|
820
|
+
// Then validate that the tx params actually equal the requested params to nano avax plus import tx fee.
|
|
821
|
+
const originalAmount = new bignumber_js_1.BigNumber(recipients[0].amount).div(1e9).plus(1e6).toFixed(0);
|
|
822
|
+
const originalDestination = recipients[0].address;
|
|
823
|
+
const hopAmount = explainHopExportTx.outputAmount;
|
|
824
|
+
const hopDestination = explainHopExportTx.outputs[0].address;
|
|
825
|
+
if (originalAmount !== hopAmount) {
|
|
826
|
+
throw new Error(`Hop amount: ${hopAmount} does not equal original amount: ${originalAmount}`);
|
|
827
|
+
}
|
|
828
|
+
if (originalDestination && hopDestination.toLowerCase() !== originalDestination.toLowerCase()) {
|
|
829
|
+
throw new Error(`Hop destination: ${hopDestination} does not equal original recipient: ${originalDestination}`);
|
|
830
|
+
}
|
|
831
|
+
}
|
|
832
|
+
if (!(await this.verifySignatureForAtomicTransaction(tx))) {
|
|
833
|
+
throw new Error(`Invalid hop transaction signature, txid: ${id}`);
|
|
834
|
+
}
|
|
835
|
+
}
|
|
836
|
+
else {
|
|
837
|
+
const builtHopTx = sdk_coin_eth_1.optionalDeps.EthTx.TransactionFactory.fromSerializedData(sdk_coin_eth_1.optionalDeps.ethUtil.toBuffer(tx));
|
|
838
|
+
// If original params are given, we can check them against the transaction prebuild params
|
|
839
|
+
if (!_.isNil(originalParams)) {
|
|
840
|
+
const { recipients } = originalParams;
|
|
841
|
+
// Then validate that the tx params actually equal the requested params
|
|
842
|
+
const originalAmount = new bignumber_js_1.BigNumber(recipients[0].amount);
|
|
843
|
+
const originalDestination = recipients[0].address;
|
|
844
|
+
const hopAmount = new bignumber_js_1.BigNumber(sdk_coin_eth_1.optionalDeps.ethUtil.bufferToHex(builtHopTx.value));
|
|
845
|
+
if (!builtHopTx.to) {
|
|
846
|
+
throw new Error(`Transaction does not have a destination address`);
|
|
847
|
+
}
|
|
848
|
+
const hopDestination = builtHopTx.to.toString();
|
|
849
|
+
if (!hopAmount.eq(originalAmount)) {
|
|
850
|
+
throw new Error(`Hop amount: ${hopAmount} does not equal original amount: ${originalAmount}`);
|
|
851
|
+
}
|
|
852
|
+
if (hopDestination.toLowerCase() !== originalDestination.toLowerCase()) {
|
|
853
|
+
throw new Error(`Hop destination: ${hopDestination} does not equal original recipient: ${hopDestination}`);
|
|
854
|
+
}
|
|
855
|
+
}
|
|
856
|
+
if (!builtHopTx.verifySignature()) {
|
|
857
|
+
// We dont want to continue at all in this case, at risk of AVAX being stuck on the hop address
|
|
858
|
+
throw new Error(`Invalid hop transaction signature, txid: ${id}`);
|
|
859
|
+
}
|
|
860
|
+
if (sdk_coin_eth_1.optionalDeps.ethUtil.addHexPrefix(builtHopTx.hash().toString('hex')) !== id) {
|
|
861
|
+
throw new Error(`Signed hop txid does not equal actual txid`);
|
|
862
|
+
}
|
|
863
|
+
}
|
|
864
|
+
}
|
|
865
|
+
/**
|
|
866
|
+
* Helper function for signTransaction for the rare case that SDK is doing the second signature
|
|
867
|
+
* Note: we are expecting this to be called from the offline vault
|
|
868
|
+
* @param params.txPrebuild
|
|
869
|
+
* @param params.prv
|
|
870
|
+
* @returns {{txHex: string}}
|
|
871
|
+
*/
|
|
872
|
+
async signFinal(params) {
|
|
873
|
+
const keyPair = new lib_1.KeyPair({ prv: params.prv });
|
|
874
|
+
const signingKey = keyPair.getKeys().prv;
|
|
875
|
+
if (_.isUndefined(signingKey)) {
|
|
876
|
+
throw new Error('missing private key');
|
|
877
|
+
}
|
|
878
|
+
const txBuilder = this.getTransactionBuilder();
|
|
879
|
+
try {
|
|
880
|
+
txBuilder.from(params.txPrebuild.halfSigned.txHex);
|
|
881
|
+
}
|
|
882
|
+
catch (e) {
|
|
883
|
+
throw new Error('invalid half-signed transaction');
|
|
884
|
+
}
|
|
885
|
+
txBuilder.sign({ key: signingKey });
|
|
886
|
+
const tx = await txBuilder.build();
|
|
887
|
+
return {
|
|
888
|
+
txHex: tx.toBroadcastFormat(),
|
|
889
|
+
};
|
|
890
|
+
}
|
|
891
|
+
/**
|
|
892
|
+
* Assemble half-sign prebuilt transaction
|
|
893
|
+
* @param params
|
|
894
|
+
*/
|
|
895
|
+
async signTransaction(params) {
|
|
896
|
+
// Normally the SDK provides the first signature for an AVAXC tx,
|
|
897
|
+
// but for unsigned sweep recoveries it can provide the second and final one.
|
|
898
|
+
if (params.isLastSignature) {
|
|
899
|
+
// In this case when we're doing the second (final) signature, the logic is different.
|
|
900
|
+
return await this.signFinal(params);
|
|
901
|
+
}
|
|
902
|
+
const txBuilder = this.getTransactionBuilder();
|
|
903
|
+
txBuilder.from(params.txPrebuild.txHex);
|
|
904
|
+
txBuilder.transfer().key(new lib_1.KeyPair({ prv: params.prv }).getKeys().prv);
|
|
905
|
+
if (params.walletVersion) {
|
|
906
|
+
txBuilder.walletVersion(params.walletVersion);
|
|
907
|
+
}
|
|
908
|
+
const transaction = await txBuilder.build();
|
|
909
|
+
// we need to preserve the calldata of the recipients specified in the request for custodial transactions
|
|
910
|
+
let recipients = params.txPrebuild.recipients || params.recipients;
|
|
911
|
+
if (recipients === undefined) {
|
|
912
|
+
recipients = transaction.outputs.map((output) => ({ address: output.address, amount: output.value }));
|
|
913
|
+
}
|
|
914
|
+
const txParams = {
|
|
915
|
+
eip1559: params.txPrebuild.eip1559,
|
|
916
|
+
txHex: transaction.toBroadcastFormat(),
|
|
917
|
+
recipients: recipients,
|
|
918
|
+
expireTime: params.txPrebuild.expireTime,
|
|
919
|
+
hopTransaction: params.txPrebuild.hopTransaction,
|
|
920
|
+
custodianTransactionId: params.custodianTransactionId,
|
|
921
|
+
};
|
|
922
|
+
return { halfSigned: txParams };
|
|
923
|
+
}
|
|
924
|
+
/**
|
|
925
|
+
* Modify prebuild before sending it to the server. Add things like hop transaction params
|
|
926
|
+
* @param buildParams The whitelisted parameters for this prebuild
|
|
927
|
+
* @param buildParams.hop True if this should prebuild a hop tx, else false
|
|
928
|
+
* @param buildParams.recipients The recipients array of this transaction
|
|
929
|
+
* @param buildParams.wallet The wallet sending this tx
|
|
930
|
+
* @param buildParams.walletPassphrase the passphrase for this wallet
|
|
931
|
+
*/
|
|
932
|
+
async getExtraPrebuildParams(buildParams) {
|
|
933
|
+
if (!_.isUndefined(buildParams.hop) &&
|
|
934
|
+
buildParams.hop &&
|
|
935
|
+
!_.isUndefined(buildParams.wallet) &&
|
|
936
|
+
!_.isUndefined(buildParams.recipients)) {
|
|
937
|
+
if (this.isToken()) {
|
|
938
|
+
throw new Error(`Hop transactions are not enabled for AVAXC tokens, nor are they necessary. Please remove the 'hop' parameter and try again.`);
|
|
939
|
+
}
|
|
940
|
+
return (await this.createHopTransactionParams({
|
|
941
|
+
recipients: buildParams.recipients,
|
|
942
|
+
type: buildParams.type,
|
|
943
|
+
}));
|
|
944
|
+
}
|
|
945
|
+
return {};
|
|
946
|
+
}
|
|
947
|
+
/**
|
|
948
|
+
* Creates the extra parameters needed to build a hop transaction
|
|
949
|
+
* @param {HopTransactionBuildOptions} The original build parameters
|
|
950
|
+
* @returns extra parameters object to merge with the original build parameters object and send to the platform
|
|
951
|
+
*/
|
|
952
|
+
async createHopTransactionParams({ recipients, type }) {
|
|
953
|
+
if (!recipients || !Array.isArray(recipients)) {
|
|
954
|
+
throw new Error('expecting array of recipients');
|
|
955
|
+
}
|
|
956
|
+
// Right now we only support 1 recipient
|
|
957
|
+
if (recipients.length !== 1) {
|
|
958
|
+
throw new Error('must send to exactly 1 recipient');
|
|
959
|
+
}
|
|
960
|
+
const recipientAddress = recipients[0].address;
|
|
961
|
+
const recipientAmount = recipients[0].amount;
|
|
962
|
+
const feeEstimateParams = {
|
|
963
|
+
recipient: recipientAddress,
|
|
964
|
+
amount: recipientAmount,
|
|
965
|
+
hop: true,
|
|
966
|
+
type,
|
|
967
|
+
};
|
|
968
|
+
const feeEstimate = await this.feeEstimate(feeEstimateParams);
|
|
969
|
+
const gasLimit = feeEstimate.gasLimitEstimate;
|
|
970
|
+
const gasPrice = Math.round(feeEstimate.feeEstimate / gasLimit);
|
|
971
|
+
const gasPriceMax = gasPrice * 5;
|
|
972
|
+
// Payment id a random number so its different for every tx
|
|
973
|
+
const paymentId = Math.floor(Math.random() * 10000000000).toString();
|
|
974
|
+
// TODO(BG-62671): after completed [Wallet-platform] Remove use of userReqSig for avaxc hop transaction
|
|
975
|
+
const userReqSig = '0x';
|
|
976
|
+
return {
|
|
977
|
+
hopParams: {
|
|
978
|
+
userReqSig,
|
|
979
|
+
gasPriceMax,
|
|
980
|
+
paymentId,
|
|
981
|
+
gasLimit,
|
|
982
|
+
},
|
|
983
|
+
};
|
|
984
|
+
}
|
|
985
|
+
/**
|
|
986
|
+
* Fetch fee estimate information from the server
|
|
987
|
+
* @param {Object} params The params passed into the function
|
|
988
|
+
* @param {Boolean} [params.hop] True if we should estimate fee for a hop transaction
|
|
989
|
+
* @param {String} [params.recipient] The recipient of the transaction to estimate a send to
|
|
990
|
+
* @param {String} [params.data] The ETH tx data to estimate a send for
|
|
991
|
+
* @returns {Object} The fee info returned from the server
|
|
992
|
+
*/
|
|
993
|
+
async feeEstimate(params) {
|
|
994
|
+
const query = {};
|
|
995
|
+
if (params && params.hop) {
|
|
996
|
+
query.hop = params.hop;
|
|
997
|
+
}
|
|
998
|
+
if (params && params.recipient) {
|
|
999
|
+
query.recipient = params.recipient;
|
|
1000
|
+
}
|
|
1001
|
+
if (params && params.data) {
|
|
1002
|
+
query.data = params.data;
|
|
1003
|
+
}
|
|
1004
|
+
if (params && params.amount) {
|
|
1005
|
+
query.amount = params.amount;
|
|
1006
|
+
}
|
|
1007
|
+
if (params && params.type) {
|
|
1008
|
+
query.type = params.type;
|
|
1009
|
+
}
|
|
1010
|
+
return await this.bitgo.get(this.url('/tx/fee')).query(query).result();
|
|
1011
|
+
}
|
|
1012
|
+
/**
|
|
1013
|
+
* Calculate tx hash like evm from tx hex.
|
|
1014
|
+
* @param {string} tx
|
|
1015
|
+
* @returns {Buffer} tx hash
|
|
1016
|
+
*/
|
|
1017
|
+
static getTxHash(tx) {
|
|
1018
|
+
const hash = (0, keccak_1.default)('keccak256');
|
|
1019
|
+
hash.update(sdk_coin_eth_1.optionalDeps.ethUtil.stripHexPrefix(tx), 'hex');
|
|
1020
|
+
return hash.digest();
|
|
1021
|
+
}
|
|
1022
|
+
async isWalletAddress(params) {
|
|
1023
|
+
// TODO: Fix this later
|
|
1024
|
+
return true;
|
|
1025
|
+
}
|
|
1026
|
+
/**
|
|
1027
|
+
* Ensure either enterprise or newFeeAddress is passed, to know whether to create new key or use enterprise key
|
|
1028
|
+
* @param params
|
|
1029
|
+
* @param params.enterprise {String} the enterprise id to associate with this key
|
|
1030
|
+
* @param params.newFeeAddress {Boolean} create a new fee address (enterprise not needed in this case)
|
|
1031
|
+
*/
|
|
1032
|
+
preCreateBitGo(params) {
|
|
1033
|
+
// We always need params object, since either enterprise or newFeeAddress is required
|
|
1034
|
+
if (!_.isObject(params)) {
|
|
1035
|
+
throw new Error(`preCreateBitGo must be passed a params object. Got ${params} (type ${typeof params})`);
|
|
1036
|
+
}
|
|
1037
|
+
if (_.isUndefined(params.enterprise) && _.isUndefined(params.newFeeAddress)) {
|
|
1038
|
+
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.');
|
|
1039
|
+
}
|
|
1040
|
+
// Check whether key should be an enterprise key or a BitGo key for a new fee address
|
|
1041
|
+
if (!_.isUndefined(params.enterprise) && !_.isUndefined(params.newFeeAddress)) {
|
|
1042
|
+
throw new Error(`Incompatible arguments - cannot pass both enterprise and newFeeAddress parameter.`);
|
|
1043
|
+
}
|
|
1044
|
+
if (!_.isUndefined(params.enterprise) && !_.isString(params.enterprise)) {
|
|
1045
|
+
throw new Error(`enterprise should be a string - got ${params.enterprise} (type ${typeof params.enterprise})`);
|
|
1046
|
+
}
|
|
1047
|
+
if (!_.isUndefined(params.newFeeAddress) && !_.isBoolean(params.newFeeAddress)) {
|
|
1048
|
+
throw new Error(`newFeeAddress should be a boolean - got ${params.newFeeAddress} (type ${typeof params.newFeeAddress})`);
|
|
1049
|
+
}
|
|
1050
|
+
}
|
|
1051
|
+
getAvaxP() {
|
|
1052
|
+
return this.getChain().toString() === 'avaxc' ? 'avaxp' : 'tavaxp';
|
|
1053
|
+
}
|
|
1054
|
+
/**
|
|
1055
|
+
* Fetch the gas price from the explorer
|
|
1056
|
+
*/
|
|
1057
|
+
async getGasPriceFromExternalAPI() {
|
|
1058
|
+
try {
|
|
1059
|
+
const res = await this.recoveryBlockchainExplorerQuery({
|
|
1060
|
+
jsonrpc: '2.0',
|
|
1061
|
+
method: 'eth_gasPrice',
|
|
1062
|
+
id: 1,
|
|
1063
|
+
});
|
|
1064
|
+
const gasPrice = new ethereumjs_util_1.BN(res.result.slice(2), 16);
|
|
1065
|
+
console.log(` Got gas price: ${gasPrice}`);
|
|
1066
|
+
return gasPrice;
|
|
1067
|
+
}
|
|
1068
|
+
catch (e) {
|
|
1069
|
+
throw new Error('Failed to get gas price');
|
|
1070
|
+
}
|
|
1071
|
+
}
|
|
1072
|
+
/**
|
|
1073
|
+
* Fetch the gas limit from the explorer
|
|
1074
|
+
* @param intendedChain
|
|
1075
|
+
* @param from
|
|
1076
|
+
* @param to
|
|
1077
|
+
* @param data
|
|
1078
|
+
*/
|
|
1079
|
+
async getGasLimitFromExternalAPI(intendedChain, from, to, data) {
|
|
1080
|
+
try {
|
|
1081
|
+
const res = await this.recoveryBlockchainExplorerQuery({
|
|
1082
|
+
jsonrpc: '2.0',
|
|
1083
|
+
method: 'eth_estimateGas',
|
|
1084
|
+
params: [
|
|
1085
|
+
{
|
|
1086
|
+
from,
|
|
1087
|
+
to,
|
|
1088
|
+
data,
|
|
1089
|
+
},
|
|
1090
|
+
'latest',
|
|
1091
|
+
],
|
|
1092
|
+
id: 1,
|
|
1093
|
+
});
|
|
1094
|
+
const gasLimit = new ethereumjs_util_1.BN(res.result.slice(2), 16);
|
|
1095
|
+
console.log(`Got gas limit: ${gasLimit}`);
|
|
1096
|
+
return gasLimit;
|
|
1097
|
+
}
|
|
1098
|
+
catch (e) {
|
|
1099
|
+
throw new Error(`Failed to get gas limit. Please make sure to use the privateKey aka userKey of ${intendedChain} wallet ${to}`);
|
|
1100
|
+
}
|
|
1101
|
+
}
|
|
1102
|
+
}
|
|
1103
|
+
exports.AvaxC = AvaxC;
|
|
1104
|
+
AvaxC.hopTransactionSalt = 'bitgoHopAddressRequestSalt';
|
|
1105
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"avaxc.js","sourceRoot":"","sources":["../../src/avaxc.ts"],"names":[],"mappings":"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","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"]}
|