@bitgo-beta/sdk-coin-etc 1.2.3-alpha.171 → 1.2.3-alpha.172

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/CHANGELOG.md CHANGED
@@ -3,10 +3,6 @@
3
3
  All notable changes to this project will be documented in this file.
4
4
  See [Conventional Commits](https://conventionalcommits.org) for commit guidelines.
5
5
 
6
- ## [2.0.29](https://github.com/BitGo/BitGoJS/compare/@bitgo/sdk-coin-etc@2.0.28...@bitgo/sdk-coin-etc@2.0.29) (2024-08-07)
7
-
8
- **Note:** Version bump only for package @bitgo/sdk-coin-etc
9
-
10
6
  ## [2.0.28](https://github.com/BitGo/BitGoJS/compare/@bitgo/sdk-coin-etc@2.0.27...@bitgo/sdk-coin-etc@2.0.28) (2024-07-30)
11
7
 
12
8
  **Note:** Version bump only for package @bitgo/sdk-coin-etc
package/dist/src/etc.d.ts CHANGED
@@ -1,14 +1,94 @@
1
+ /// <reference types="node" />
1
2
  /**
2
3
  * @prettier
3
4
  */
4
- import { AbstractEthLikeCoin } from '@bitgo-beta/abstract-eth';
5
- import { BaseCoin, BitGoBase } from '@bitgo-beta/sdk-core';
6
- import { BaseCoin as StaticsBaseCoin } from '@bitgo-beta/statics';
5
+ import { AbstractEthLikeCoin, OfflineVaultTxInfo, RecoverOptions, RecoveryInfo } from '@bitgo-beta/abstract-eth';
6
+ import { BaseCoin, BitGoBase, Recipient } from '@bitgo-beta/sdk-core';
7
+ import { BaseCoin as StaticsBaseCoin, EthereumNetwork as EthLikeNetwork } from '@bitgo-beta/statics';
7
8
  import { TransactionBuilder } from './lib';
8
9
  export declare class Etc extends AbstractEthLikeCoin {
10
+ readonly staticsCoin?: Readonly<StaticsBaseCoin>;
11
+ protected readonly sendMethodName: 'sendMultiSig' | 'sendMultiSigToken';
9
12
  protected constructor(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>);
10
13
  static createInstance(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>): BaseCoin;
11
14
  isValidPub(pub: string): boolean;
15
+ /**
16
+ * Builds a funds recovery transaction without BitGo
17
+ * @param params
18
+ * @param {string} params.userKey - [encrypted] xprv
19
+ * @param {string} params.backupKey - [encrypted] xprv or xpub if the xprv is held by a KRS provider
20
+ * @param {string} params.walletPassphrase - used to decrypt userKey and backupKey
21
+ * @param {string} params.walletContractAddress - the ETH address of the wallet contract
22
+ * @param {string} params.krsProvider - necessary if backup key is held by KRS
23
+ * @param {string} params.recoveryDestination - target address to send recovered funds to
24
+ * @param {string} params.bitgoFeeAddress - wrong chain wallet fee address for evm based cross chain recovery txn
25
+ * @param {string} params.bitgoDestinationAddress - target bitgo address where fee will be sent for evm based cross chain recovery txn
26
+ */
27
+ recover(params: RecoverOptions): Promise<RecoveryInfo | OfflineVaultTxInfo>;
12
28
  protected getTransactionBuilder(): TransactionBuilder;
29
+ /**
30
+ * Query explorer for the balance of an address
31
+ * @param {String} address - the ETHLike address
32
+ * @returns {BigNumber} address balance
33
+ */
34
+ queryAddressBalance(address: string): Promise<any>;
35
+ /**
36
+ * Make a query to Arbiscan for information such as balance, token balance, solidity calls
37
+ * @param {Object} query key-value pairs of parameters to append after /api
38
+ * @returns {Promise<Object>} response from Arbiscan
39
+ */
40
+ recoveryBlockchainExplorerQuery(query: Record<string, string>): Promise<Record<string, unknown>>;
41
+ /**
42
+ * Method to validate recovery params
43
+ * @param {RecoverOptions} params
44
+ * @returns {void}
45
+ */
46
+ validateRecoveryParams(params: RecoverOptions): void;
47
+ /**
48
+ * Queries public block explorer to get the next ETHLike coin's nonce that should be used for the given ETH address
49
+ * @param {string} address
50
+ * @returns {Promise<number>}
51
+ */
52
+ getAddressNonce(address: string): Promise<number>;
53
+ /**
54
+ * Queries the contract (via explorer API) for the next sequence ID
55
+ * @param {String} address - address of the contract
56
+ * @returns {Promise<Number>} sequence ID
57
+ */
58
+ querySequenceId(address: string): Promise<number>;
59
+ /**
60
+ * Check whether the gas price passed in by user are within our max and min bounds
61
+ * If they are not set, set them to the defaults
62
+ * @param {number} userGasPrice - user defined gas price
63
+ * @returns {number} the gas price to use for this transaction
64
+ */
65
+ setGasPrice(userGasPrice?: number): number;
66
+ /**
67
+ * Check whether gas limit passed in by user are within our max and min bounds
68
+ * If they are not set, set them to the defaults
69
+ * @param {number} userGasLimit user defined gas limit
70
+ * @returns {number} the gas limit to use for this transaction
71
+ */
72
+ setGasLimit(userGasLimit?: number): number;
73
+ /**
74
+ * @param {Recipient[]} recipients - the recipients of the transaction
75
+ * @param {number} expireTime - the expire time of the transaction
76
+ * @param {number} contractSequenceId - the contract sequence id of the transaction
77
+ * @returns {string}
78
+ */
79
+ getOperationSha3ForExecuteAndConfirm(recipients: Recipient[], expireTime: number, contractSequenceId: number): string;
80
+ /**
81
+ * Get transfer operation for coin
82
+ * @param {Recipient} recipient - recipient info
83
+ * @param {number} expireTime - expiry time
84
+ * @param {number} contractSequenceId - sequence id
85
+ * @returns {Array} operation array
86
+ */
87
+ getOperation(recipient: Recipient, expireTime: number, contractSequenceId: number): (string | Buffer)[][];
88
+ /**
89
+ * Method to return the coin's network object
90
+ * @returns {EthLikeNetwork | undefined}
91
+ */
92
+ getNetwork(): EthLikeNetwork | undefined;
13
93
  }
14
94
  //# sourceMappingURL=etc.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"etc.d.ts","sourceRoot":"","sources":["../../src/etc.ts"],"names":[],"mappings":"AAAA;;GAEG;AACH,OAAO,EAAE,mBAAmB,EAAE,MAAM,0BAA0B,CAAC;AAC/D,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,MAAM,sBAAsB,CAAC;AAC3D,OAAO,EAAE,QAAQ,IAAI,eAAe,EAAS,MAAM,qBAAqB,CAAC;AACzE,OAAO,EAAW,kBAAkB,EAAE,MAAM,OAAO,CAAC;AAEpD,qBAAa,GAAI,SAAQ,mBAAmB;IAC1C,SAAS,aAAa,KAAK,EAAE,SAAS,EAAE,WAAW,CAAC,EAAE,QAAQ,CAAC,eAAe,CAAC;IAI/E,MAAM,CAAC,cAAc,CAAC,KAAK,EAAE,SAAS,EAAE,WAAW,CAAC,EAAE,QAAQ,CAAC,eAAe,CAAC,GAAG,QAAQ;IAI1F,UAAU,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO;IAUhC,SAAS,CAAC,qBAAqB,IAAI,kBAAkB;CAGtD"}
1
+ {"version":3,"file":"etc.d.ts","sourceRoot":"","sources":["../../src/etc.ts"],"names":[],"mappings":";AAAA;;GAEG;AACH,OAAO,EACL,mBAAmB,EAEnB,kBAAkB,EAElB,cAAc,EAEd,YAAY,EACb,MAAM,0BAA0B,CAAC;AAClC,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAoC,SAAS,EAAE,MAAM,sBAAsB,CAAC;AACxG,OAAO,EAAE,QAAQ,IAAI,eAAe,EAAS,eAAe,IAAI,cAAc,EAAiB,MAAM,qBAAqB,CAAC;AAC3H,OAAO,EAAE,kBAAkB,EAA0C,MAAM,OAAO,CAAC;AAKnF,qBAAa,GAAI,SAAQ,mBAAmB;IAC1C,QAAQ,CAAC,WAAW,CAAC,EAAE,QAAQ,CAAC,eAAe,CAAC,CAAC;IACjD,SAAS,CAAC,QAAQ,CAAC,cAAc,EAAE,cAAc,GAAG,mBAAmB,CAAC;IAExE,SAAS,aAAa,KAAK,EAAE,SAAS,EAAE,WAAW,CAAC,EAAE,QAAQ,CAAC,eAAe,CAAC;IAU/E,MAAM,CAAC,cAAc,CAAC,KAAK,EAAE,SAAS,EAAE,WAAW,CAAC,EAAE,QAAQ,CAAC,eAAe,CAAC,GAAG,QAAQ;IAI1F,UAAU,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO;IAUhC;;;;;;;;;;;OAWG;IACG,OAAO,CAAC,MAAM,EAAE,cAAc,GAAG,OAAO,CAAC,YAAY,GAAG,kBAAkB,CAAC;IAsKjF,SAAS,CAAC,qBAAqB,IAAI,kBAAkB;IAIrD;;;;OAIG;IACG,mBAAmB,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAAC,GAAG,CAAC;IAYxD;;;;OAIG;IACG,+BAA+B,CAAC,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,GAAG,OAAO,CAAC,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IAMtG;;;;OAIG;IACH,sBAAsB,CAAC,MAAM,EAAE,cAAc,GAAG,IAAI;IAsBpD;;;;OAIG;IACG,eAAe,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;IAqBvD;;;;OAIG;IACG,eAAe,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;IAuBvD;;;;;OAKG;IACH,WAAW,CAAC,YAAY,CAAC,EAAE,MAAM,GAAG,MAAM;IAY1C;;;;;OAKG;IACH,WAAW,CAAC,YAAY,CAAC,EAAE,MAAM,GAAG,MAAM;IAY1C;;;;;OAKG;IACH,oCAAoC,CAClC,UAAU,EAAE,SAAS,EAAE,EACvB,UAAU,EAAE,MAAM,EAClB,kBAAkB,EAAE,MAAM,GACzB,MAAM;IA+CT;;;;;;OAMG;IACH,YAAY,CAAC,SAAS,EAAE,SAAS,EAAE,UAAU,EAAE,MAAM,EAAE,kBAAkB,EAAE,MAAM,GAAG,CAAC,MAAM,GAAG,MAAM,CAAC,EAAE,EAAE;IAezG;;;OAGG;IACH,UAAU,IAAI,cAAc,GAAG,SAAS;CAGzC"}
package/dist/src/etc.js CHANGED
@@ -1,15 +1,47 @@
1
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 (mod) {
19
+ if (mod && mod.__esModule) return mod;
20
+ var result = {};
21
+ if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
22
+ __setModuleDefault(result, mod);
23
+ return result;
24
+ };
2
25
  Object.defineProperty(exports, "__esModule", { value: true });
3
26
  exports.Etc = void 0;
4
27
  /**
5
28
  * @prettier
6
29
  */
7
30
  const abstract_eth_1 = require("@bitgo-beta/abstract-eth");
31
+ const sdk_core_1 = require("@bitgo-beta/sdk-core");
8
32
  const statics_1 = require("@bitgo-beta/statics");
9
33
  const lib_1 = require("./lib");
34
+ const _ = __importStar(require("lodash"));
35
+ const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
36
+ const bignumber_js_1 = require("bignumber.js");
10
37
  class Etc extends abstract_eth_1.AbstractEthLikeCoin {
11
38
  constructor(bitgo, staticsCoin) {
12
39
  super(bitgo, staticsCoin);
40
+ if (!staticsCoin) {
41
+ throw new Error('missing required constructor parameter staticsCoin');
42
+ }
43
+ this.staticsCoin = staticsCoin;
44
+ this.sendMethodName = 'sendMultiSig';
13
45
  }
14
46
  static createInstance(bitgo, staticsCoin) {
15
47
  return new Etc(bitgo, staticsCoin);
@@ -24,9 +56,378 @@ class Etc extends abstract_eth_1.AbstractEthLikeCoin {
24
56
  }
25
57
  return valid;
26
58
  }
59
+ /**
60
+ * Builds a funds recovery transaction without BitGo
61
+ * @param params
62
+ * @param {string} params.userKey - [encrypted] xprv
63
+ * @param {string} params.backupKey - [encrypted] xprv or xpub if the xprv is held by a KRS provider
64
+ * @param {string} params.walletPassphrase - used to decrypt userKey and backupKey
65
+ * @param {string} params.walletContractAddress - the ETH address of the wallet contract
66
+ * @param {string} params.krsProvider - necessary if backup key is held by KRS
67
+ * @param {string} params.recoveryDestination - target address to send recovered funds to
68
+ * @param {string} params.bitgoFeeAddress - wrong chain wallet fee address for evm based cross chain recovery txn
69
+ * @param {string} params.bitgoDestinationAddress - target bitgo address where fee will be sent for evm based cross chain recovery txn
70
+ */
71
+ async recover(params) {
72
+ var _a, _b;
73
+ this.validateRecoveryParams(params);
74
+ const isUnsignedSweep = (0, sdk_core_1.getIsUnsignedSweep)(params);
75
+ // Clean up whitespace from entered values
76
+ let userKey = params.userKey.replace(/\s/g, '');
77
+ const backupKey = params.backupKey.replace(/\s/g, '');
78
+ const gasLimit = new abstract_eth_1.optionalDeps.ethUtil.BN(this.setGasLimit(params.gasLimit));
79
+ const gasPrice = params.eip1559
80
+ ? new abstract_eth_1.optionalDeps.ethUtil.BN(params.eip1559.maxFeePerGas)
81
+ : new abstract_eth_1.optionalDeps.ethUtil.BN(this.setGasPrice(params.gasPrice));
82
+ if (!userKey.startsWith('xpub') && !userKey.startsWith('xprv')) {
83
+ try {
84
+ userKey = this.bitgo.decrypt({
85
+ input: userKey,
86
+ password: params.walletPassphrase,
87
+ });
88
+ }
89
+ catch (e) {
90
+ throw new Error(`Error decrypting user keychain: ${e.message}`);
91
+ }
92
+ }
93
+ let backupKeyAddress;
94
+ let backupSigningKey;
95
+ if (isUnsignedSweep) {
96
+ const backupHDNode = utxo_lib_1.bip32.fromBase58(backupKey);
97
+ backupSigningKey = backupHDNode.publicKey;
98
+ backupKeyAddress = `0x${abstract_eth_1.optionalDeps.ethUtil.publicToAddress(backupSigningKey, true).toString('hex')}`;
99
+ }
100
+ else {
101
+ // Decrypt backup private key and get address
102
+ let backupPrv;
103
+ try {
104
+ backupPrv = this.bitgo.decrypt({
105
+ input: backupKey,
106
+ password: params.walletPassphrase,
107
+ });
108
+ }
109
+ catch (e) {
110
+ throw new Error(`Error decrypting backup keychain: ${e.message}`);
111
+ }
112
+ const keyPair = new lib_1.KeyPair({ prv: backupPrv });
113
+ backupSigningKey = keyPair.getKeys().prv;
114
+ if (!backupSigningKey) {
115
+ throw new Error('no private key');
116
+ }
117
+ backupKeyAddress = keyPair.getAddress();
118
+ }
119
+ const backupKeyNonce = await this.getAddressNonce(backupKeyAddress);
120
+ // get balance of backupKey to ensure funds are available to pay fees
121
+ const backupKeyBalance = await this.queryAddressBalance(backupKeyAddress);
122
+ const totalGasNeeded = gasPrice.mul(gasLimit);
123
+ const weiToGwei = 10 ** 9;
124
+ if (backupKeyBalance.lt(totalGasNeeded)) {
125
+ throw new Error(`Backup key address ${backupKeyAddress} has balance ${(backupKeyBalance / weiToGwei).toString()} Gwei.` +
126
+ `This address must have a balance of at least ${(totalGasNeeded / weiToGwei).toString()}` +
127
+ ` Gwei to perform recoveries. Try sending some funds to this address then retry.`);
128
+ }
129
+ // get balance of wallet
130
+ const txAmount = await this.queryAddressBalance(params.walletContractAddress);
131
+ if (new bignumber_js_1.BigNumber(txAmount).isLessThanOrEqualTo(0)) {
132
+ throw new Error('Wallet does not have enough funds to recover');
133
+ }
134
+ // build recipients object
135
+ const recipients = [
136
+ {
137
+ address: params.recoveryDestination,
138
+ amount: txAmount.toString(10),
139
+ },
140
+ ];
141
+ // Get sequence ID using contract call
142
+ // we need to wait between making two explorer api calls to avoid getting banned
143
+ await new Promise((resolve) => setTimeout(resolve, 1000));
144
+ const sequenceId = await this.querySequenceId(params.walletContractAddress);
145
+ let operationHash, signature;
146
+ // Get operation hash and sign it
147
+ if (!isUnsignedSweep) {
148
+ operationHash = this.getOperationSha3ForExecuteAndConfirm(recipients, (0, abstract_eth_1.getDefaultExpireTime)(), sequenceId);
149
+ signature = sdk_core_1.Util.ethSignMsgHash(operationHash, sdk_core_1.Util.xprvToEthPrivateKey(userKey));
150
+ try {
151
+ sdk_core_1.Util.ecRecoverEthAddress(operationHash, signature);
152
+ }
153
+ catch (e) {
154
+ throw new Error('Invalid signature');
155
+ }
156
+ }
157
+ // Build unsigned transaction
158
+ const txInfo = {
159
+ recipient: recipients[0],
160
+ expireTime: (0, abstract_eth_1.getDefaultExpireTime)(),
161
+ contractSequenceId: sequenceId,
162
+ operationHash: operationHash,
163
+ signature: signature,
164
+ gasLimit: gasLimit.toString(10),
165
+ };
166
+ const txBuilder = this.getTransactionBuilder();
167
+ txBuilder.counter(backupKeyNonce);
168
+ txBuilder.contract(params.walletContractAddress);
169
+ let txFee;
170
+ if (params.eip1559) {
171
+ txFee = {
172
+ eip1559: {
173
+ maxPriorityFeePerGas: params.eip1559.maxPriorityFeePerGas,
174
+ maxFeePerGas: params.eip1559.maxFeePerGas,
175
+ },
176
+ };
177
+ }
178
+ else {
179
+ txFee = { fee: gasPrice.toString() };
180
+ }
181
+ txBuilder.fee({
182
+ ...txFee,
183
+ gasLimit: gasLimit.toString(),
184
+ });
185
+ const transferBuilder = txBuilder.transfer();
186
+ transferBuilder
187
+ .coin((_a = this.staticsCoin) === null || _a === void 0 ? void 0 : _a.name)
188
+ .amount(recipients[0].amount)
189
+ .contractSequenceId(sequenceId)
190
+ .expirationTime((0, abstract_eth_1.getDefaultExpireTime)())
191
+ .to(params.recoveryDestination);
192
+ const tx = await txBuilder.build();
193
+ if (isUnsignedSweep) {
194
+ const response = {
195
+ txHex: tx.toBroadcastFormat(),
196
+ userKey,
197
+ backupKey,
198
+ coin: this.getChain(),
199
+ gasPrice: abstract_eth_1.optionalDeps.ethUtil.bufferToInt(gasPrice).toFixed(),
200
+ gasLimit,
201
+ recipients: [txInfo.recipient],
202
+ walletContractAddress: tx.toJson().to,
203
+ amount: txInfo.recipient.amount,
204
+ backupKeyNonce,
205
+ eip1559: params.eip1559,
206
+ };
207
+ _.extend(response, txInfo);
208
+ response.nextContractSequenceId = response.contractSequenceId;
209
+ return response;
210
+ }
211
+ // sign the transaction
212
+ txBuilder
213
+ .transfer()
214
+ .coin((_b = this.staticsCoin) === null || _b === void 0 ? void 0 : _b.name)
215
+ .key(new lib_1.KeyPair({ prv: userKey }).getKeys().prv);
216
+ txBuilder.sign({ key: backupSigningKey });
217
+ const signedTx = await txBuilder.build();
218
+ return {
219
+ id: signedTx.toJson().id,
220
+ tx: signedTx.toBroadcastFormat(),
221
+ };
222
+ }
27
223
  getTransactionBuilder() {
28
224
  return new lib_1.TransactionBuilder(statics_1.coins.get(this.getBaseChain()));
29
225
  }
226
+ /**
227
+ * Query explorer for the balance of an address
228
+ * @param {String} address - the ETHLike address
229
+ * @returns {BigNumber} address balance
230
+ */
231
+ async queryAddressBalance(address) {
232
+ const result = await this.recoveryBlockchainExplorerQuery({
233
+ module: 'account',
234
+ action: 'balance',
235
+ address: address,
236
+ });
237
+ // throw if the result does not exist or the result is not a valid number
238
+ if (!result || !result.result || (typeof result.result === 'number' && isNaN(result.result))) {
239
+ throw new Error(`Could not obtain address balance for ${address} from the explorer, got: ${result.result}`);
240
+ }
241
+ return new abstract_eth_1.optionalDeps.ethUtil.BN(result.result, 10);
242
+ }
243
+ /**
244
+ * Make a query to Arbiscan for information such as balance, token balance, solidity calls
245
+ * @param {Object} query key-value pairs of parameters to append after /api
246
+ * @returns {Promise<Object>} response from Arbiscan
247
+ */
248
+ async recoveryBlockchainExplorerQuery(query) {
249
+ // const apiToken = common.Environments[this.bitgo.getEnv()].arbiscanApiToken;
250
+ const explorerUrl = sdk_core_1.common.Environments[this.bitgo.getEnv()].etcNodeUrl;
251
+ return await (0, abstract_eth_1.recoveryBlockchainExplorerQuery)(query, explorerUrl);
252
+ }
253
+ /**
254
+ * Method to validate recovery params
255
+ * @param {RecoverOptions} params
256
+ * @returns {void}
257
+ */
258
+ validateRecoveryParams(params) {
259
+ if (_.isUndefined(params.userKey)) {
260
+ throw new Error('missing userKey');
261
+ }
262
+ if (_.isUndefined(params.backupKey)) {
263
+ throw new Error('missing backupKey');
264
+ }
265
+ if (_.isUndefined(params.walletPassphrase) && !params.userKey.startsWith('xpub') && !params.isTss) {
266
+ throw new Error('missing wallet passphrase');
267
+ }
268
+ if (_.isUndefined(params.walletContractAddress) || !this.isValidAddress(params.walletContractAddress)) {
269
+ throw new Error('invalid walletContractAddress');
270
+ }
271
+ if (_.isUndefined(params.recoveryDestination) || !this.isValidAddress(params.recoveryDestination)) {
272
+ throw new Error('invalid recoveryDestination');
273
+ }
274
+ }
275
+ /**
276
+ * Queries public block explorer to get the next ETHLike coin's nonce that should be used for the given ETH address
277
+ * @param {string} address
278
+ * @returns {Promise<number>}
279
+ */
280
+ async getAddressNonce(address) {
281
+ // Get nonce for backup key (should be 0)
282
+ let nonce = 0;
283
+ const result = await this.recoveryBlockchainExplorerQuery({
284
+ module: 'account',
285
+ action: 'txlist',
286
+ address,
287
+ });
288
+ if (!result || !Array.isArray(result.result)) {
289
+ throw new Error('Unable to find next nonce from Etherscan, got: ' + JSON.stringify(result));
290
+ }
291
+ const backupKeyTxList = result.result;
292
+ if (backupKeyTxList.length > 0) {
293
+ // Calculate last nonce used
294
+ const outgoingTxs = backupKeyTxList.filter((tx) => tx.from === address);
295
+ nonce = outgoingTxs.length;
296
+ }
297
+ return nonce;
298
+ }
299
+ /**
300
+ * Queries the contract (via explorer API) for the next sequence ID
301
+ * @param {String} address - address of the contract
302
+ * @returns {Promise<Number>} sequence ID
303
+ */
304
+ async querySequenceId(address) {
305
+ // Get sequence ID using contract call
306
+ const sequenceIdMethodSignature = abstract_eth_1.optionalDeps.ethAbi.methodID('getNextSequenceId', []);
307
+ const sequenceIdArgs = abstract_eth_1.optionalDeps.ethAbi.rawEncode([], []);
308
+ const sequenceIdData = Buffer.concat([sequenceIdMethodSignature, sequenceIdArgs]).toString('hex');
309
+ const result = await this.recoveryBlockchainExplorerQuery({
310
+ module: 'proxy',
311
+ action: 'eth_call',
312
+ to: address,
313
+ data: sequenceIdData,
314
+ tag: 'latest',
315
+ });
316
+ if (!result || !result.result) {
317
+ throw new Error('Could not obtain sequence ID from explorer, got: ' + result.result);
318
+ }
319
+ const sequenceIdHex = result.result;
320
+ if (typeof sequenceIdHex === 'string') {
321
+ return new abstract_eth_1.optionalDeps.ethUtil.BN(sequenceIdHex.slice(2), 16).toNumber();
322
+ }
323
+ else {
324
+ throw new Error('Expected sequenceIdHex to be a string');
325
+ }
326
+ }
327
+ /**
328
+ * Check whether the gas price passed in by user are within our max and min bounds
329
+ * If they are not set, set them to the defaults
330
+ * @param {number} userGasPrice - user defined gas price
331
+ * @returns {number} the gas price to use for this transaction
332
+ */
333
+ setGasPrice(userGasPrice) {
334
+ if (!userGasPrice) {
335
+ return statics_1.ethGasConfigs.defaultGasPrice;
336
+ }
337
+ const gasPriceMax = statics_1.ethGasConfigs.maximumGasPrice;
338
+ const gasPriceMin = statics_1.ethGasConfigs.minimumGasPrice;
339
+ if (userGasPrice < gasPriceMin || userGasPrice > gasPriceMax) {
340
+ throw new Error(`Gas price must be between ${gasPriceMin} and ${gasPriceMax}`);
341
+ }
342
+ return userGasPrice;
343
+ }
344
+ /**
345
+ * Check whether gas limit passed in by user are within our max and min bounds
346
+ * If they are not set, set them to the defaults
347
+ * @param {number} userGasLimit user defined gas limit
348
+ * @returns {number} the gas limit to use for this transaction
349
+ */
350
+ setGasLimit(userGasLimit) {
351
+ if (!userGasLimit) {
352
+ return statics_1.ethGasConfigs.defaultGasLimit;
353
+ }
354
+ const gasLimitMax = statics_1.ethGasConfigs.maximumGasLimit;
355
+ const gasLimitMin = statics_1.ethGasConfigs.minimumGasLimit;
356
+ if (userGasLimit < gasLimitMin || userGasLimit > gasLimitMax) {
357
+ throw new Error(`Gas limit must be between ${gasLimitMin} and ${gasLimitMax}`);
358
+ }
359
+ return userGasLimit;
360
+ }
361
+ /**
362
+ * @param {Recipient[]} recipients - the recipients of the transaction
363
+ * @param {number} expireTime - the expire time of the transaction
364
+ * @param {number} contractSequenceId - the contract sequence id of the transaction
365
+ * @returns {string}
366
+ */
367
+ getOperationSha3ForExecuteAndConfirm(recipients, expireTime, contractSequenceId) {
368
+ if (!recipients || !Array.isArray(recipients)) {
369
+ throw new Error('expecting array of recipients');
370
+ }
371
+ // Right now we only support 1 recipient
372
+ if (recipients.length !== 1) {
373
+ throw new Error('must send to exactly 1 recipient');
374
+ }
375
+ if (!_.isNumber(expireTime)) {
376
+ throw new Error('expireTime must be number of seconds since epoch');
377
+ }
378
+ if (!_.isNumber(contractSequenceId)) {
379
+ throw new Error('contractSequenceId must be number');
380
+ }
381
+ // Check inputs
382
+ recipients.forEach(function (recipient) {
383
+ if (!_.isString(recipient.address) ||
384
+ !abstract_eth_1.optionalDeps.ethUtil.isValidAddress(abstract_eth_1.optionalDeps.ethUtil.addHexPrefix(recipient.address))) {
385
+ throw new Error('Invalid address: ' + recipient.address);
386
+ }
387
+ let amount;
388
+ try {
389
+ amount = new bignumber_js_1.BigNumber(recipient.amount);
390
+ }
391
+ catch (e) {
392
+ throw new Error('Invalid amount for: ' + recipient.address + ' - should be numeric');
393
+ }
394
+ recipient.amount = amount.toFixed(0);
395
+ if (recipient.data && !_.isString(recipient.data)) {
396
+ throw new Error('Data for recipient ' + recipient.address + ' - should be of type hex string');
397
+ }
398
+ });
399
+ const recipient = recipients[0];
400
+ return abstract_eth_1.optionalDeps.ethUtil.bufferToHex(abstract_eth_1.optionalDeps.ethAbi.soliditySHA3(...this.getOperation(recipient, expireTime, contractSequenceId)));
401
+ }
402
+ /**
403
+ * Get transfer operation for coin
404
+ * @param {Recipient} recipient - recipient info
405
+ * @param {number} expireTime - expiry time
406
+ * @param {number} contractSequenceId - sequence id
407
+ * @returns {Array} operation array
408
+ */
409
+ getOperation(recipient, expireTime, contractSequenceId) {
410
+ const network = this.getNetwork();
411
+ return [
412
+ ['string', 'address', 'uint', 'bytes', 'uint', 'uint'],
413
+ [
414
+ network.nativeCoinOperationHashPrefix,
415
+ new abstract_eth_1.optionalDeps.ethUtil.BN(abstract_eth_1.optionalDeps.ethUtil.stripHexPrefix(recipient.address), 16),
416
+ recipient.amount,
417
+ Buffer.from(abstract_eth_1.optionalDeps.ethUtil.stripHexPrefix(abstract_eth_1.optionalDeps.ethUtil.padToEven(recipient.data || '')), 'hex'),
418
+ expireTime,
419
+ contractSequenceId,
420
+ ],
421
+ ];
422
+ }
423
+ /**
424
+ * Method to return the coin's network object
425
+ * @returns {EthLikeNetwork | undefined}
426
+ */
427
+ getNetwork() {
428
+ var _a;
429
+ return (_a = this.staticsCoin) === null || _a === void 0 ? void 0 : _a.network;
430
+ }
30
431
  }
31
432
  exports.Etc = Etc;
32
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXRjLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2V0Yy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7QUFBQTs7R0FFRztBQUNILDJEQUErRDtBQUUvRCxpREFBeUU7QUFDekUsK0JBQW9EO0FBRXBELE1BQWEsR0FBSSxTQUFRLGtDQUFtQjtJQUMxQyxZQUFzQixLQUFnQixFQUFFLFdBQXVDO1FBQzdFLEtBQUssQ0FBQyxLQUFLLEVBQUUsV0FBVyxDQUFDLENBQUM7SUFDNUIsQ0FBQztJQUVELE1BQU0sQ0FBQyxjQUFjLENBQUMsS0FBZ0IsRUFBRSxXQUF1QztRQUM3RSxPQUFPLElBQUksR0FBRyxDQUFDLEtBQUssRUFBRSxXQUFXLENBQUMsQ0FBQztJQUNyQyxDQUFDO0lBRUQsVUFBVSxDQUFDLEdBQVc7UUFDcEIsSUFBSSxLQUFLLEdBQUcsSUFBSSxDQUFDO1FBQ2pCLElBQUk7WUFDRixJQUFJLGFBQU8sQ0FBQyxFQUFFLEdBQUcsRUFBRSxDQUFDLENBQUM7U0FDdEI7UUFBQyxPQUFPLENBQUMsRUFBRTtZQUNWLEtBQUssR0FBRyxLQUFLLENBQUM7U0FDZjtRQUNELE9BQU8sS0FBSyxDQUFDO0lBQ2YsQ0FBQztJQUVTLHFCQUFxQjtRQUM3QixPQUFPLElBQUksd0JBQWtCLENBQUMsZUFBSyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsWUFBWSxFQUFFLENBQUMsQ0FBQyxDQUFDO0lBQ2hFLENBQUM7Q0FDRjtBQXRCRCxrQkFzQkMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBwcmV0dGllclxuICovXG5pbXBvcnQgeyBBYnN0cmFjdEV0aExpa2VDb2luIH0gZnJvbSAnQGJpdGdvLWJldGEvYWJzdHJhY3QtZXRoJztcbmltcG9ydCB7IEJhc2VDb2luLCBCaXRHb0Jhc2UgfSBmcm9tICdAYml0Z28tYmV0YS9zZGstY29yZSc7XG5pbXBvcnQgeyBCYXNlQ29pbiBhcyBTdGF0aWNzQmFzZUNvaW4sIGNvaW5zIH0gZnJvbSAnQGJpdGdvLWJldGEvc3RhdGljcyc7XG5pbXBvcnQgeyBLZXlQYWlyLCBUcmFuc2FjdGlvbkJ1aWxkZXIgfSBmcm9tICcuL2xpYic7XG5cbmV4cG9ydCBjbGFzcyBFdGMgZXh0ZW5kcyBBYnN0cmFjdEV0aExpa2VDb2luIHtcbiAgcHJvdGVjdGVkIGNvbnN0cnVjdG9yKGJpdGdvOiBCaXRHb0Jhc2UsIHN0YXRpY3NDb2luPzogUmVhZG9ubHk8U3RhdGljc0Jhc2VDb2luPikge1xuICAgIHN1cGVyKGJpdGdvLCBzdGF0aWNzQ29pbik7XG4gIH1cblxuICBzdGF0aWMgY3JlYXRlSW5zdGFuY2UoYml0Z286IEJpdEdvQmFzZSwgc3RhdGljc0NvaW4/OiBSZWFkb25seTxTdGF0aWNzQmFzZUNvaW4+KTogQmFzZUNvaW4ge1xuICAgIHJldHVybiBuZXcgRXRjKGJpdGdvLCBzdGF0aWNzQ29pbik7XG4gIH1cblxuICBpc1ZhbGlkUHViKHB1Yjogc3RyaW5nKTogYm9vbGVhbiB7XG4gICAgbGV0IHZhbGlkID0gdHJ1ZTtcbiAgICB0cnkge1xuICAgICAgbmV3IEtleVBhaXIoeyBwdWIgfSk7XG4gICAgfSBjYXRjaCAoZSkge1xuICAgICAgdmFsaWQgPSBmYWxzZTtcbiAgICB9XG4gICAgcmV0dXJuIHZhbGlkO1xuICB9XG5cbiAgcHJvdGVjdGVkIGdldFRyYW5zYWN0aW9uQnVpbGRlcigpOiBUcmFuc2FjdGlvbkJ1aWxkZXIge1xuICAgIHJldHVybiBuZXcgVHJhbnNhY3Rpb25CdWlsZGVyKGNvaW5zLmdldCh0aGlzLmdldEJhc2VDaGFpbigpKSk7XG4gIH1cbn1cbiJdfQ==
433
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"etc.js","sourceRoot":"","sources":["../../src/etc.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport {\n  AbstractEthLikeCoin,\n  getDefaultExpireTime,\n  OfflineVaultTxInfo,\n  optionalDeps,\n  RecoverOptions,\n  recoveryBlockchainExplorerQuery,\n  RecoveryInfo,\n} from '@bitgo-beta/abstract-eth';\nimport { BaseCoin, BitGoBase, common, getIsUnsignedSweep, Util, Recipient } from '@bitgo-beta/sdk-core';\nimport { BaseCoin as StaticsBaseCoin, coins, EthereumNetwork as EthLikeNetwork, ethGasConfigs } from '@bitgo-beta/statics';\nimport { TransactionBuilder, KeyPair as KeyPairLib, TransferBuilder } from './lib';\nimport * as _ from 'lodash';\nimport { bip32 } from '@bitgo-beta/utxo-lib';\nimport { BigNumber } from 'bignumber.js';\n\nexport class Etc extends AbstractEthLikeCoin {\n  readonly staticsCoin?: Readonly<StaticsBaseCoin>;\n  protected readonly sendMethodName: 'sendMultiSig' | 'sendMultiSigToken';\n\n  protected constructor(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>) {\n    super(bitgo, staticsCoin);\n    if (!staticsCoin) {\n      throw new Error('missing required constructor parameter staticsCoin');\n    }\n\n    this.staticsCoin = staticsCoin;\n    this.sendMethodName = 'sendMultiSig';\n  }\n\n  static createInstance(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>): BaseCoin {\n    return new Etc(bitgo, staticsCoin);\n  }\n\n  isValidPub(pub: string): boolean {\n    let valid = true;\n    try {\n      new KeyPairLib({ pub });\n    } catch (e) {\n      valid = false;\n    }\n    return valid;\n  }\n\n  /**\n   * Builds a funds recovery transaction without BitGo\n   * @param params\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 ETH address of the wallet contract\n   * @param {string} params.krsProvider - necessary if backup key is held by KRS\n   * @param {string} params.recoveryDestination - target address to send recovered funds to\n   * @param {string} params.bitgoFeeAddress - wrong chain wallet fee address for evm based cross chain recovery txn\n   * @param {string} params.bitgoDestinationAddress - target bitgo address where fee will be sent for evm based cross chain recovery txn\n   */\n  async recover(params: RecoverOptions): Promise<RecoveryInfo | OfflineVaultTxInfo> {\n    this.validateRecoveryParams(params);\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    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\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    let backupKeyAddress;\n    let backupSigningKey;\n    if (isUnsignedSweep) {\n      const backupHDNode = bip32.fromBase58(backupKey);\n      backupSigningKey = backupHDNode.publicKey;\n      backupKeyAddress = `0x${optionalDeps.ethUtil.publicToAddress(backupSigningKey, true).toString('hex')}`;\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 KeyPairLib({ prv: backupPrv });\n      backupSigningKey = keyPair.getKeys().prv;\n      if (!backupSigningKey) {\n        throw new Error('no private key');\n      }\n      backupKeyAddress = keyPair.getAddress();\n    }\n\n    const backupKeyNonce = await this.getAddressNonce(backupKeyAddress);\n    // get balance of backupKey to ensure funds are available to pay fees\n    const backupKeyBalance = await this.queryAddressBalance(backupKeyAddress);\n    const totalGasNeeded = gasPrice.mul(gasLimit);\n\n    const weiToGwei = 10 ** 9;\n    if (backupKeyBalance.lt(totalGasNeeded)) {\n      throw new Error(\n        `Backup key address ${backupKeyAddress} has balance ${(backupKeyBalance / weiToGwei).toString()} Gwei.` +\n          `This address must have a balance of at least ${(totalGasNeeded / weiToGwei).toString()}` +\n          ` Gwei to perform recoveries. Try sending some funds to this address then retry.`\n      );\n    }\n\n    // get balance of wallet\n    const txAmount = await this.queryAddressBalance(params.walletContractAddress);\n    if (new BigNumber(txAmount).isLessThanOrEqualTo(0)) {\n      throw new Error('Wallet does not have enough funds to recover');\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 explorer api 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, 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    // Build unsigned transaction\n    const txInfo = {\n      recipient: recipients[0],\n      expireTime: getDefaultExpireTime(),\n      contractSequenceId: sequenceId,\n      operationHash: operationHash,\n      signature: signature,\n      gasLimit: gasLimit.toString(10),\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    const transferBuilder = txBuilder.transfer() as TransferBuilder;\n    transferBuilder\n      .coin(this.staticsCoin?.name as string)\n      .amount(recipients[0].amount)\n      .contractSequenceId(sequenceId)\n      .expirationTime(getDefaultExpireTime())\n      .to(params.recoveryDestination);\n\n    const tx = await txBuilder.build();\n    if (isUnsignedSweep) {\n      const response: OfflineVaultTxInfo = {\n        txHex: tx.toBroadcastFormat(),\n        userKey,\n        backupKey,\n        coin: this.getChain(),\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    // sign the transaction\n    txBuilder\n      .transfer()\n      .coin(this.staticsCoin?.name as string)\n      .key(new KeyPairLib({ prv: userKey }).getKeys().prv as string);\n    txBuilder.sign({ key: backupSigningKey });\n\n    const signedTx = await txBuilder.build();\n\n    return {\n      id: signedTx.toJson().id,\n      tx: signedTx.toBroadcastFormat(),\n    };\n  }\n\n  protected getTransactionBuilder(): TransactionBuilder {\n    return new TransactionBuilder(coins.get(this.getBaseChain()));\n  }\n\n  /**\n   * Query explorer for the balance of an address\n   * @param {String} address - the ETHLike address\n   * @returns {BigNumber} address balance\n   */\n  async queryAddressBalance(address: string): Promise<any> {\n    const result = await this.recoveryBlockchainExplorerQuery({\n      module: 'account',\n      action: 'balance',\n      address: address,\n    });\n    // throw if the result does not exist or the result is not a valid number\n    if (!result || !result.result || (typeof result.result === 'number' && isNaN(result.result))) {\n      throw new Error(`Could not obtain address balance for ${address} from the explorer, got: ${result.result}`);\n    }\n    return new optionalDeps.ethUtil.BN(result.result, 10);\n  }\n  /**\n   * Make a query to Arbiscan for information such as balance, token balance, solidity calls\n   * @param {Object} query key-value pairs of parameters to append after /api\n   * @returns {Promise<Object>} response from Arbiscan\n   */\n  async recoveryBlockchainExplorerQuery(query: Record<string, string>): Promise<Record<string, unknown>> {\n    // const apiToken = common.Environments[this.bitgo.getEnv()].arbiscanApiToken;\n    const explorerUrl = common.Environments[this.bitgo.getEnv()].etcNodeUrl;\n    return await recoveryBlockchainExplorerQuery(query, explorerUrl as string);\n  }\n\n  /**\n   * Method to validate recovery params\n   * @param {RecoverOptions} params\n   * @returns {void}\n   */\n  validateRecoveryParams(params: RecoverOptions): void {\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') && !params.isTss) {\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    if (_.isUndefined(params.recoveryDestination) || !this.isValidAddress(params.recoveryDestination)) {\n      throw new Error('invalid recoveryDestination');\n    }\n  }\n\n  /**\n   * Queries public block explorer to get the next ETHLike coin's nonce that should be used for the given ETH address\n   * @param {string} address\n   * @returns {Promise<number>}\n   */\n  async getAddressNonce(address: string): Promise<number> {\n    // Get nonce for backup key (should be 0)\n    let nonce = 0;\n\n    const result = await this.recoveryBlockchainExplorerQuery({\n      module: 'account',\n      action: 'txlist',\n      address,\n    });\n    if (!result || !Array.isArray(result.result)) {\n      throw new Error('Unable to find next nonce from Etherscan, got: ' + JSON.stringify(result));\n    }\n    const backupKeyTxList = result.result;\n    if (backupKeyTxList.length > 0) {\n      // Calculate last nonce used\n      const outgoingTxs = backupKeyTxList.filter((tx) => tx.from === address);\n      nonce = outgoingTxs.length;\n    }\n    return nonce;\n  }\n\n  /**\n   * Queries the contract (via explorer API) 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 result = await this.recoveryBlockchainExplorerQuery({\n      module: 'proxy',\n      action: 'eth_call',\n      to: address,\n      data: sequenceIdData,\n      tag: 'latest',\n    });\n    if (!result || !result.result) {\n      throw new Error('Could not obtain sequence ID from explorer, got: ' + result.result);\n    }\n    const sequenceIdHex = result.result;\n    if (typeof sequenceIdHex === 'string') {\n      return new optionalDeps.ethUtil.BN(sequenceIdHex.slice(2), 16).toNumber();\n    } else {\n      throw new Error('Expected sequenceIdHex to be a string');\n    }\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 {number} 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   * 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   * @param {Recipient[]} recipients - the recipients of the transaction\n   * @param {number} expireTime - the expire time of the transaction\n   * @param {number} contractSequenceId - the contract sequence id of the transaction\n   * @returns {string}\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: BigNumber;\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   * Get transfer operation for coin\n   * @param {Recipient} recipient - recipient info\n   * @param {number} expireTime - expiry time\n   * @param {number} contractSequenceId - sequence id\n   * @returns {Array} operation array\n   */\n  getOperation(recipient: Recipient, expireTime: number, contractSequenceId: number): (string | Buffer)[][] {\n    const network = this.getNetwork() as EthLikeNetwork;\n    return [\n      ['string', 'address', 'uint', 'bytes', 'uint', 'uint'],\n      [\n        network.nativeCoinOperationHashPrefix,\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   * Method to return the coin's network object\n   * @returns {EthLikeNetwork | undefined}\n   */\n  getNetwork(): EthLikeNetwork | undefined {\n    return this.staticsCoin?.network as EthLikeNetwork;\n  }\n}\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@bitgo-beta/sdk-coin-etc",
3
- "version": "1.2.3-alpha.171",
3
+ "version": "1.2.3-alpha.172",
4
4
  "description": "BitGo SDK coin library for Ethereum classic",
5
5
  "main": "./dist/src/index.js",
6
6
  "types": "./dist/src/index.d.ts",
@@ -40,16 +40,16 @@
40
40
  ]
41
41
  },
42
42
  "dependencies": {
43
- "@bitgo-beta/abstract-eth": "1.2.3-alpha.173",
44
- "@bitgo-beta/sdk-coin-eth": "2.4.1-alpha.173",
45
- "@bitgo-beta/sdk-core": "2.4.1-alpha.173",
46
- "@bitgo-beta/statics": "10.0.1-alpha.173",
43
+ "@bitgo-beta/abstract-eth": "1.2.3-alpha.174",
44
+ "@bitgo-beta/sdk-coin-eth": "2.4.1-alpha.174",
45
+ "@bitgo-beta/sdk-core": "2.4.1-alpha.174",
46
+ "@bitgo-beta/statics": "10.0.1-alpha.174",
47
47
  "@ethereumjs/common": "^2.6.5",
48
48
  "ethereumjs-abi": "^0.6.5"
49
49
  },
50
50
  "devDependencies": {
51
- "@bitgo-beta/sdk-api": "1.6.1-alpha.173",
52
- "@bitgo-beta/sdk-test": "^8.0.34"
51
+ "@bitgo-beta/sdk-api": "1.6.1-alpha.174",
52
+ "@bitgo-beta/sdk-test": "^8.0.33"
53
53
  },
54
- "gitHead": "6ec669fe7cec9f4fd489bed3dcecdffbf49a0a92"
54
+ "gitHead": "846fef1839ddb34490d7e22a421c14b5e1b995b1"
55
55
  }