@bitgo-beta/sdk-coin-ethlike 1.0.1-beta.97 → 1.0.1-beta.970

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.
@@ -0,0 +1,716 @@
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
+ const assert_1 = __importDefault(require("assert"));
40
+ const sdk_api_1 = require("@bitgo-beta/sdk-api");
41
+ const sdk_core_1 = require("@bitgo-beta/sdk-core");
42
+ const sdk_test_1 = require("@bitgo-beta/sdk-test");
43
+ const secp256k1_1 = require("@bitgo-beta/secp256k1");
44
+ const nock_1 = __importDefault(require("nock"));
45
+ const src_1 = require("../../src");
46
+ const getBuilder_1 = require("../getBuilder");
47
+ const resources_1 = require("../resources");
48
+ const mockData = __importStar(require("../fixtures/ethlikeCoin"));
49
+ nock_1.default.enableNetConnect();
50
+ const coins = [
51
+ {
52
+ name: 'hteth',
53
+ common: (0, resources_1.getCommon)('hteth'),
54
+ },
55
+ {
56
+ name: 'tarbeth',
57
+ common: (0, resources_1.getCommon)('tarbeth'),
58
+ },
59
+ ];
60
+ describe('EthLike coin tests', function () {
61
+ let bitgo;
62
+ let basecoin;
63
+ coins.forEach((coin) => {
64
+ describe(coin.name, function () {
65
+ before(function () {
66
+ const env = 'test';
67
+ bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env });
68
+ bitgo.safeRegister(coin.name, src_1.TethLikeCoin.createInstance);
69
+ bitgo.initializeTestVars();
70
+ basecoin = bitgo.coin(coin.name);
71
+ });
72
+ after(function () {
73
+ nock_1.default.cleanAll();
74
+ });
75
+ it('should instantiate a coin', function () {
76
+ basecoin.should.be.an.instanceof(src_1.TethLikeCoin);
77
+ });
78
+ it('should reject for missing encryptedPrv for hot wallet', async function () {
79
+ const recoveryId = '0x1234567890abcdef';
80
+ (0, nock_1.default)(bitgo.microservicesUrl(`/api/recovery/v1/crosschain`)).get(`/${recoveryId}/buildtx`).reply(200, {
81
+ txHex: mockData.ccr[coin.name].txHex,
82
+ });
83
+ const walletPassphrase = sdk_test_1.TestBitGo.V2.TEST_RECOVERY_PASSCODE;
84
+ const params = {
85
+ recoveryId,
86
+ walletPassphrase,
87
+ common: coin.common,
88
+ };
89
+ await basecoin
90
+ .sendCrossChainRecoveryTransaction({ ...params, walletType: 'hot' })
91
+ .should.be.rejectedWith('missing encryptedPrv');
92
+ });
93
+ it('should send cross chain recovery transaction for hot wallet', async function () {
94
+ const recoveryId = '0x1234567890abcdef';
95
+ (0, nock_1.default)(bitgo.microservicesUrl(`/api/recovery/v1/crosschain`)).get(`/${recoveryId}/buildtx`).reply(200, {
96
+ txHex: mockData.ccr[coin.name].txHex,
97
+ });
98
+ (0, nock_1.default)(bitgo.microservicesUrl(`/api/recovery/v1/crosschain`)).post(`/${recoveryId}/sign`).reply(200, {
99
+ coin: coin.name,
100
+ txid: mockData.ccr[coin.name].txid,
101
+ });
102
+ const walletPassphrase = sdk_test_1.TestBitGo.V2.TEST_RECOVERY_PASSCODE;
103
+ const params = {
104
+ recoveryId,
105
+ walletPassphrase,
106
+ encryptedPrv: mockData.encryptedUserKey,
107
+ common: coin.common,
108
+ };
109
+ const result = await basecoin.sendCrossChainRecoveryTransaction({ ...params, walletType: 'hot' });
110
+ result.coin.should.equal(coin.name);
111
+ result.txid.should.equal(mockData.ccr[coin.name].txid);
112
+ });
113
+ it('should build txn for cross chain recovery for cold wallet', async function () {
114
+ const recoveryId = '0x1234567890abcdef';
115
+ (0, nock_1.default)(bitgo.microservicesUrl(`/api/recovery/v1/crosschain`)).get(`/${recoveryId}/buildtx`).reply(200, {
116
+ txHex: mockData.ccr[coin.name].txHex,
117
+ });
118
+ const params = {
119
+ recoveryId,
120
+ common: coin.common,
121
+ };
122
+ const result = await basecoin.sendCrossChainRecoveryTransaction({ ...params, walletType: 'cold' });
123
+ (0, assert_1.default)(result.txHex);
124
+ result.txHex.should.equal(mockData.ccr[coin.name].txHex);
125
+ });
126
+ it('should build cross chain recovery transaction and extract recipients', async function () {
127
+ const recoveryId = '0x1234567890abcdef';
128
+ const mockResponse = {
129
+ coin: coin.name,
130
+ txHex: mockData.ccr[coin.name].txHex,
131
+ txid: mockData.ccr[coin.name].txid,
132
+ walletVersion: 1,
133
+ };
134
+ (0, nock_1.default)(bitgo.microservicesUrl(`/api/recovery/v1/crosschain`))
135
+ .get(`/${recoveryId}/buildtx`)
136
+ .reply(200, mockResponse);
137
+ const result = await basecoin.buildCrossChainRecoveryTransaction(recoveryId);
138
+ result.should.have.property('coin');
139
+ result.coin.should.equal(coin.name);
140
+ result.should.have.property('txHex');
141
+ result.txHex.should.equal(mockData.ccr[coin.name].txHex);
142
+ result.should.have.property('txid');
143
+ result.txid.should.equal(mockData.ccr[coin.name].txid);
144
+ result.should.have.property('walletVersion');
145
+ result.should.have.property('recipients');
146
+ result.recipients.should.be.an.Array();
147
+ const recipient = result.recipients[0];
148
+ recipient.should.have.property('address');
149
+ recipient.should.have.property('amount');
150
+ recipient.address.should.be.a.String();
151
+ recipient.amount.should.be.a.String();
152
+ });
153
+ it('should generate signature data for custodial hot wallet and sign using hsm signature', async function () {
154
+ const baseAddress = '0x702cf81e03aa310ec9481d814e3d04a20b04b505';
155
+ const destinationAddress = '0xb9f62c71d5f6949cfb211a67fb13ccf079cc760b';
156
+ const tokenContractAddress = '0xe4ab69c077896252fafbd49efd26b5d171a32410';
157
+ const txBuilder = (0, getBuilder_1.getBuilder)(coin.name, coin.common);
158
+ txBuilder.contract(baseAddress);
159
+ txBuilder.contractCounter(0);
160
+ txBuilder.fee({
161
+ fee: '100000',
162
+ gasLimit: '21000',
163
+ });
164
+ const transferBuilder = txBuilder.transfer();
165
+ transferBuilder
166
+ .coin(coin.name)
167
+ .amount('100000000')
168
+ .contractSequenceId(100)
169
+ .expirationTime(1744049633)
170
+ .to(destinationAddress)
171
+ .tokenContractAddress(tokenContractAddress);
172
+ const signatureData = transferBuilder.getSignatureData();
173
+ assert_1.default.strictEqual(signatureData.toString('hex'), mockData.custodialHot[coin.name].signatureData);
174
+ // Set HSM Signature
175
+ transferBuilder.setSignature(mockData.custodialHot[coin.name].signature);
176
+ const tx = await txBuilder.build();
177
+ const txHex = tx.toBroadcastFormat();
178
+ assert_1.default.strictEqual(txHex, mockData.custodialHot[coin.name].signedTxHex);
179
+ });
180
+ });
181
+ });
182
+ });
183
+ describe('EthLikeCoin', function () {
184
+ let bitgo;
185
+ const coinName = 'tbaseeth';
186
+ let basecoin;
187
+ before(function () {
188
+ const bitgoKeyXprv = 'xprv9s21ZrQH143K3tpWBHWe31sLoXNRQ9AvRYJgitkKxQ4ATFQMwvr7hHNqYRUnS7PsjzB7aK1VxqHLuNQjj1sckJ2Jwo2qxmsvejwECSpFMfC';
189
+ const bitgoKey = secp256k1_1.bip32.fromBase58(bitgoKeyXprv);
190
+ if (!bitgoKey.privateKey) {
191
+ throw new Error('no privateKey');
192
+ }
193
+ const bitgoXpub = bitgoKey.neutered().toBase58();
194
+ const env = 'test';
195
+ bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env });
196
+ sdk_core_1.common.Environments[env].hsmXpub = bitgoXpub;
197
+ bitgo.safeRegister('baseeth', src_1.EthLikeCoin.createInstance);
198
+ bitgo.safeRegister('tbaseeth', src_1.TethLikeCoin.createInstance);
199
+ bitgo.initializeTestVars();
200
+ basecoin = bitgo.coin('tbaseeth');
201
+ });
202
+ after(function () {
203
+ nock_1.default.cleanAll();
204
+ });
205
+ it('should instantiate a coin', function () {
206
+ let coin = bitgo.coin('tbaseeth');
207
+ coin.should.be.an.instanceof(src_1.TethLikeCoin);
208
+ coin = bitgo.coin('baseeth');
209
+ coin.should.be.an.instanceof(src_1.EthLikeCoin);
210
+ });
211
+ it('should build unsigned transaction', async function () {
212
+ const expireTime = Math.floor(new Date().getTime() / 1000);
213
+ const txBuilder = (0, getBuilder_1.getBuilder)(coinName, resources_1.baseChainCommon);
214
+ txBuilder.type(sdk_core_1.TransactionType.Send);
215
+ txBuilder.fee({
216
+ fee: '1000000000',
217
+ gasLimit: '100000',
218
+ });
219
+ txBuilder.counter(1);
220
+ txBuilder.contract('0x2c2b9c9a4a25e24b174f26114e8926a9f2128fe4');
221
+ const transferBuilder = txBuilder.transfer();
222
+ transferBuilder
223
+ .coin(coinName)
224
+ .expirationTime(expireTime)
225
+ .amount('1000000000000000000')
226
+ .to('0x2c2b9c9a4a25e24b174f26114e8926a9f2128fe4')
227
+ .contractSequenceId(1);
228
+ const tx = await txBuilder.build();
229
+ const txJson = tx.toJson();
230
+ txJson.gasLimit.should.equal('100000');
231
+ txJson.gasPrice.should.equal('1000000000');
232
+ txJson.chainId.should.equal('0x14a34');
233
+ });
234
+ it('should sign a transaction', async function () {
235
+ const account_1 = {
236
+ address: '0x8Ce59c2d1702844F8EdED451AA103961bC37B4e8',
237
+ owner_1: '4ee089aceabf3ddbf748db79b1066c33b7d3ea1ab3eb7e325121bba2bff2f5ca',
238
+ owner_2: '5c7e4efff7304d4dfff6d5f1591844ec6f2adfa6a47e9fece6a3c1a4d755f1e3',
239
+ owner_3: '4421ab25dd91e1a3180d03d57c323a7886dcc313d3b3a4b4256a5791572bf597',
240
+ };
241
+ const expireTime = Math.floor(new Date().getTime() / 1000);
242
+ const txBuilder = (0, getBuilder_1.getBuilder)(coinName, resources_1.baseChainCommon);
243
+ txBuilder.type(sdk_core_1.TransactionType.Send);
244
+ txBuilder.fee({
245
+ fee: '1000000000',
246
+ gasLimit: '100000',
247
+ });
248
+ txBuilder.counter(1);
249
+ txBuilder.contract(account_1.address);
250
+ const transferBuilder = txBuilder.transfer();
251
+ transferBuilder
252
+ .coin(coinName)
253
+ .expirationTime(expireTime)
254
+ .amount('1000000000000000000')
255
+ .to('0x2c2b9c9a4a25e24b174f26114e8926a9f2128fe4')
256
+ .contractSequenceId(1);
257
+ const unsignedTx = await txBuilder.build();
258
+ const unsignedTxHex = unsignedTx.toBroadcastFormat();
259
+ const halfSignedTx = (await basecoin.signTransaction({
260
+ txPrebuild: {
261
+ txHex: unsignedTxHex,
262
+ },
263
+ prv: account_1.owner_1,
264
+ common: resources_1.baseChainCommon,
265
+ }));
266
+ transferBuilder.key(account_1.owner_1);
267
+ const halfSignedTxBuilder = await txBuilder.build();
268
+ const halfSignedTxHexBuilder = halfSignedTxBuilder.toBroadcastFormat();
269
+ halfSignedTxHexBuilder.should.equal(halfSignedTx.halfSigned.txHex);
270
+ // Sign with the second key
271
+ const fullSignedTxn = (await basecoin.signTransaction({
272
+ txPrebuild: {
273
+ halfSigned: {
274
+ txHex: halfSignedTxHexBuilder,
275
+ expireTime: expireTime,
276
+ contractSequenceId: 1,
277
+ signature: '',
278
+ },
279
+ },
280
+ prv: account_1.owner_2,
281
+ common: resources_1.baseChainCommon,
282
+ isLastSignature: true,
283
+ }));
284
+ (0, assert_1.default)(fullSignedTxn.txHex);
285
+ });
286
+ describe('explainTransaction', function () {
287
+ const txHex = mockData.ccr[coinName].txHex;
288
+ const feeInfo = {
289
+ fee: '1000000000',
290
+ gasLimit: '100000',
291
+ };
292
+ it('should explain transaction when common is provided', async function () {
293
+ const explanation = await basecoin.explainTransaction({
294
+ txHex,
295
+ feeInfo,
296
+ common: resources_1.baseChainCommon,
297
+ });
298
+ explanation.should.have.property('id');
299
+ explanation.should.have.property('outputs');
300
+ explanation.should.have.property('outputAmount');
301
+ explanation.should.have.property('changeOutputs');
302
+ explanation.should.have.property('changeAmount');
303
+ explanation.should.have.property('fee');
304
+ explanation.fee.should.equal(feeInfo);
305
+ explanation.outputs.should.be.an.Array();
306
+ });
307
+ it('should fail to explain transaction when common is not provided', async function () {
308
+ await basecoin
309
+ .explainTransaction({
310
+ txHex,
311
+ feeInfo,
312
+ })
313
+ .should.be.rejectedWith('Common must be provided for EthLikeTransactionBuilder');
314
+ });
315
+ });
316
+ describe('Recovery', function () {
317
+ const baseUrl = 'https://api-sepolia.basescan.org/';
318
+ const userXpub = 'xpub661MyMwAqRbcEeTc8789MK5PUGEYiPG4F4V17n2Rd2LoTATA1XoCnJT5FAYAShQxSxtFjpo5NHmcWwTp2LiWGBMwpUcAA3HywhxivgYfq7q';
319
+ const backupXpub = 'xpub661MyMwAqRbcFZX15xpZf4ERCGHiVSJm8r5C4yh1yXV2GrdZCUPYo4WQr6tN9oUywKXsgSHo7Risf9r22GH5joVD2hEEEhqnSCvK8qy11wW';
320
+ it('should generate an unsigned sweep transaction', async function () {
321
+ const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
322
+ const backupKeyAddress = '0x4f2c4830cc37f2785c646f89ded8a919219fa0e9';
323
+ (0, nock_1.default)(baseUrl)
324
+ .get('/api')
325
+ .twice()
326
+ .query(mockData.getTxListRequest(backupKeyAddress))
327
+ .reply(200, mockData.getTxListResponse);
328
+ (0, nock_1.default)(baseUrl)
329
+ .get('/api')
330
+ .query(mockData.getBalanceRequest(walletContractAddress))
331
+ .reply(200, mockData.getBalanceResponse);
332
+ (0, nock_1.default)(baseUrl)
333
+ .get('/api')
334
+ .query(mockData.getBalanceRequest(backupKeyAddress))
335
+ .reply(200, mockData.getBalanceResponse);
336
+ (0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
337
+ const baseCoin = bitgo.coin('tbaseeth');
338
+ const transaction = (await baseCoin.recover({
339
+ userKey: userXpub,
340
+ backupKey: backupXpub,
341
+ walletContractAddress: walletContractAddress,
342
+ recoveryDestination: sdk_test_1.TestBitGo.V2.TEST_ERC20_TOKEN_RECIPIENT,
343
+ eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
344
+ gasLimit: 500000,
345
+ common: resources_1.baseChainCommon,
346
+ }));
347
+ (0, assert_1.default)(transaction.txHex);
348
+ (0, assert_1.default)(transaction.contractSequenceId);
349
+ assert_1.default.strictEqual(transaction.gasLimit, '500000');
350
+ });
351
+ });
352
+ describe('Evm Based Cross Chain Recovery transaction:', function () {
353
+ const baseUrl = 'https://api-sepolia.basescan.org/';
354
+ const userXpub = 'xpub661MyMwAqRbcEeTc8789MK5PUGEYiPG4F4V17n2Rd2LoTATA1XoCnJT5FAYAShQxSxtFjpo5NHmcWwTp2LiWGBMwpUcAA3HywhxivgYfq7q';
355
+ it('should generate an unsigned recovery txn for cold wallet', async function () {
356
+ const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
357
+ const bitgoFeeAddress = '0x33a42faea3c6e87021347e51700b48aaf49aa1e7';
358
+ const destinationAddress = '0xd5adde17fed8baed3f32b84af05b8f2816f7b560';
359
+ const bitgoDestinationAddress = '0xE5986CE4490Deb67d2950562Ceb930Ddf9be7a14';
360
+ const basecoin = bitgo.coin('tbaseeth');
361
+ (0, nock_1.default)(baseUrl)
362
+ .get('/api')
363
+ .query(mockData.getTxListRequest(bitgoFeeAddress))
364
+ .reply(200, mockData.getTxListResponse);
365
+ (0, nock_1.default)(baseUrl)
366
+ .get('/api')
367
+ .query(mockData.getBalanceRequest(bitgoFeeAddress))
368
+ .reply(200, mockData.getBalanceResponse);
369
+ (0, nock_1.default)(baseUrl)
370
+ .get('/api')
371
+ .query(mockData.getBalanceRequest(walletContractAddress))
372
+ .reply(200, mockData.getBalanceResponse);
373
+ (0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
374
+ const transaction = (await basecoin.recover({
375
+ userKey: userXpub,
376
+ backupKey: '',
377
+ walletContractAddress: walletContractAddress,
378
+ bitgoFeeAddress: bitgoFeeAddress,
379
+ recoveryDestination: destinationAddress,
380
+ eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
381
+ gasLimit: 500000,
382
+ bitgoDestinationAddress: bitgoDestinationAddress,
383
+ common: resources_1.baseChainCommon,
384
+ }));
385
+ (0, assert_1.default)(transaction);
386
+ transaction.should.have.property('txHex');
387
+ transaction.should.have.property('userKey');
388
+ transaction.should.have.property('coin');
389
+ transaction.should.have.property('contractSequenceId');
390
+ transaction.should.have.property('expireTime');
391
+ transaction.should.have.property('gasLimit');
392
+ transaction.gasLimit.should.equal('500000');
393
+ transaction.should.have.property('isEvmBasedCrossChainRecovery');
394
+ transaction.isEvmBasedCrossChainRecovery?.should.equal(true);
395
+ transaction.should.have.property('walletContractAddress');
396
+ transaction.walletContractAddress.should.equal(sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);
397
+ transaction.should.have.property('recipients');
398
+ const recipient = transaction.recipients[0];
399
+ recipient.should.have.property('address');
400
+ recipient.address.should.equal(destinationAddress);
401
+ recipient.should.have.property('amount');
402
+ recipient.amount.should.equal('9999999999999999928');
403
+ });
404
+ it('should generate an unsigned recovery txn for custody wallet', async function () {
405
+ const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
406
+ const bitgoFeeAddress = '0x33a42faea3c6e87021347e51700b48aaf49aa1e7';
407
+ const destinationAddress = '0xd5adde17fed8baed3f32b84af05b8f2816f7b560';
408
+ const bitgoDestinationAddress = '0xE5986CE4490Deb67d2950562Ceb930Ddf9be7a14';
409
+ const basecoin = bitgo.coin('tbaseeth');
410
+ (0, nock_1.default)(baseUrl)
411
+ .get('/api')
412
+ .query(mockData.getTxListRequest(bitgoFeeAddress))
413
+ .reply(200, mockData.getTxListResponse);
414
+ (0, nock_1.default)(baseUrl)
415
+ .get('/api')
416
+ .query(mockData.getBalanceRequest(bitgoFeeAddress))
417
+ .reply(200, mockData.getBalanceResponse);
418
+ (0, nock_1.default)(baseUrl)
419
+ .get('/api')
420
+ .query(mockData.getBalanceRequest(walletContractAddress))
421
+ .reply(200, mockData.getBalanceResponse);
422
+ (0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
423
+ const transaction = (await basecoin.recover({
424
+ userKey: '',
425
+ backupKey: '',
426
+ walletContractAddress: walletContractAddress,
427
+ bitgoFeeAddress: bitgoFeeAddress,
428
+ recoveryDestination: destinationAddress,
429
+ eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
430
+ gasLimit: 500000,
431
+ bitgoDestinationAddress: bitgoDestinationAddress,
432
+ common: resources_1.baseChainCommon,
433
+ }));
434
+ (0, assert_1.default)(transaction);
435
+ transaction.should.have.property('txHex');
436
+ transaction.should.have.property('coin');
437
+ transaction.should.have.property('contractSequenceId');
438
+ transaction.should.have.property('expireTime');
439
+ transaction.should.have.property('gasLimit');
440
+ transaction.gasLimit.should.equal('500000');
441
+ transaction.should.have.property('isEvmBasedCrossChainRecovery');
442
+ transaction.isEvmBasedCrossChainRecovery?.should.equal(true);
443
+ transaction.should.have.property('walletContractAddress');
444
+ transaction.walletContractAddress.should.equal(sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);
445
+ transaction.should.have.property('recipients');
446
+ const recipient = transaction.recipients[0];
447
+ recipient.should.have.property('address');
448
+ recipient.address.should.equal(destinationAddress);
449
+ recipient.should.have.property('amount');
450
+ recipient.amount.should.equal('9999999999999999928');
451
+ });
452
+ it('should generate an unsigned recovery txn for hot wallet', async function () {
453
+ const userKey = '{"iv":"VFZ3jvXhxo1Z+Yaf2MtZnA==","v":1,"iter":10000,"ks":256,"ts":64,"mode"\n' +
454
+ ':"ccm","adata":"","cipher":"aes","salt":"p+fkHuLa/8k=","ct":"hYG7pvljLIgCjZ\n' +
455
+ '53PBlCde5KZRmlUKKHLtDMk+HJfuU46hW+x+C9WsIAO4gFPnTCvFVmQ8x7czCtcNFub5AO2otOG\n' +
456
+ 'OsX4GE2gXOEmCl1TpWwwNhm7yMUjGJUpgW6ZZgXSXdDitSKi4V/hk78SGSzjFOBSPYRa6I="}\n';
457
+ const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
458
+ const bitgoFeeAddress = '0x33a42faea3c6e87021347e51700b48aaf49aa1e7';
459
+ const destinationAddress = '0xd5adde17fed8baed3f32b84af05b8f2816f7b560';
460
+ const bitgoDestinationAddress = '0xE5986CE4490Deb67d2950562Ceb930Ddf9be7a14';
461
+ const walletPassphrase = sdk_test_1.TestBitGo.V2.TEST_RECOVERY_PASSCODE;
462
+ const basecoin = bitgo.coin('tbaseeth');
463
+ (0, nock_1.default)(baseUrl)
464
+ .get('/api')
465
+ .query(mockData.getTxListRequest(bitgoFeeAddress))
466
+ .reply(200, mockData.getTxListResponse);
467
+ (0, nock_1.default)(baseUrl)
468
+ .get('/api')
469
+ .query(mockData.getBalanceRequest(bitgoFeeAddress))
470
+ .reply(200, mockData.getBalanceResponse);
471
+ (0, nock_1.default)(baseUrl)
472
+ .get('/api')
473
+ .query(mockData.getBalanceRequest(walletContractAddress))
474
+ .reply(200, mockData.getBalanceResponse);
475
+ (0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
476
+ const transaction = (await basecoin.recover({
477
+ userKey: userKey,
478
+ backupKey: '',
479
+ walletPassphrase: walletPassphrase,
480
+ walletContractAddress: walletContractAddress,
481
+ bitgoFeeAddress: bitgoFeeAddress,
482
+ recoveryDestination: destinationAddress,
483
+ eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
484
+ gasLimit: 500000,
485
+ bitgoDestinationAddress: bitgoDestinationAddress,
486
+ common: resources_1.baseChainCommon,
487
+ }));
488
+ (0, assert_1.default)(transaction);
489
+ transaction.should.have.property('txHex');
490
+ transaction.should.have.property('coin');
491
+ transaction.should.have.property('contractSequenceId');
492
+ transaction.should.have.property('expireTime');
493
+ transaction.should.have.property('gasLimit');
494
+ transaction.gasLimit.should.equal('500000');
495
+ transaction.should.have.property('isEvmBasedCrossChainRecovery');
496
+ transaction.isEvmBasedCrossChainRecovery?.should.equal(true);
497
+ transaction.should.have.property('walletContractAddress');
498
+ transaction.walletContractAddress.should.equal(sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);
499
+ transaction.should.have.property('recipients');
500
+ const recipient = transaction.recipients[0];
501
+ recipient.should.have.property('address');
502
+ recipient.address.should.equal(destinationAddress);
503
+ recipient.should.have.property('amount');
504
+ recipient.amount.should.equal('9999999999999999928');
505
+ transaction.should.have.property('feesUsed');
506
+ transaction.feesUsed?.gasLimit.should.equal('500000');
507
+ transaction.should.have.property('halfSigned');
508
+ transaction.halfSigned?.should.have.property('txHex');
509
+ transaction.halfSigned?.should.have.property('recipients');
510
+ });
511
+ it('should generate an unsigned recovery txn of a token for cold wallet ', async function () {
512
+ const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
513
+ const bitgoFeeAddress = '0x33a42faea3c6e87021347e51700b48aaf49aa1e7';
514
+ const destinationAddress = '0xd5adde17fed8baed3f32b84af05b8f2816f7b560';
515
+ const bitgoDestinationAddress = '0xE5986CE4490Deb67d2950562Ceb930Ddf9be7a14';
516
+ const tokenContractAddress = '0x326c977e6efc84e512bb9c30f76e30c160ed06fb';
517
+ const basecoin = bitgo.coin('tbaseeth');
518
+ (0, nock_1.default)(baseUrl)
519
+ .get('/api')
520
+ .query(mockData.getTxListRequest(bitgoFeeAddress))
521
+ .reply(200, mockData.getTxListResponse);
522
+ (0, nock_1.default)(baseUrl)
523
+ .get('/api')
524
+ .query(mockData.getBalanceRequest(bitgoFeeAddress))
525
+ .reply(200, mockData.getBalanceResponse);
526
+ (0, nock_1.default)(baseUrl)
527
+ .get('/api')
528
+ .query(mockData.getTokenBalanceRequest(tokenContractAddress, walletContractAddress))
529
+ .reply(200, mockData.getBalanceResponse);
530
+ (0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
531
+ const transaction = (await basecoin.recover({
532
+ userKey: userXpub,
533
+ backupKey: '',
534
+ walletContractAddress: walletContractAddress,
535
+ bitgoFeeAddress: bitgoFeeAddress,
536
+ recoveryDestination: destinationAddress,
537
+ eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
538
+ gasLimit: 500000,
539
+ bitgoDestinationAddress: bitgoDestinationAddress,
540
+ tokenContractAddress: tokenContractAddress,
541
+ common: resources_1.baseChainCommon,
542
+ }));
543
+ (0, assert_1.default)(transaction);
544
+ transaction.should.have.property('txHex');
545
+ transaction.should.have.property('userKey');
546
+ transaction.should.have.property('coin');
547
+ transaction.coin.should.equal('tbaseeth');
548
+ transaction.should.have.property('contractSequenceId');
549
+ transaction.should.have.property('expireTime');
550
+ transaction.should.have.property('gasLimit');
551
+ transaction.gasLimit.should.equal('500000');
552
+ transaction.should.have.property('isEvmBasedCrossChainRecovery');
553
+ transaction.isEvmBasedCrossChainRecovery?.should.equal(true);
554
+ transaction.should.have.property('walletContractAddress');
555
+ transaction.walletContractAddress.should.equal(sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);
556
+ transaction.should.have.property('recipients');
557
+ const recipient = transaction.recipients[0];
558
+ recipient.should.have.property('address');
559
+ recipient.address.should.equal(destinationAddress);
560
+ recipient.should.have.property('amount');
561
+ recipient.amount.should.equal('9999999999999999928');
562
+ });
563
+ it('should generate an unsigned recovery txn of a token for custody wallet', async function () {
564
+ const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
565
+ const bitgoFeeAddress = '0x33a42faea3c6e87021347e51700b48aaf49aa1e7';
566
+ const destinationAddress = '0xd5adde17fed8baed3f32b84af05b8f2816f7b560';
567
+ const bitgoDestinationAddress = '0xE5986CE4490Deb67d2950562Ceb930Ddf9be7a14';
568
+ const tokenContractAddress = '0x326c977e6efc84e512bb9c30f76e30c160ed06fb'; // unsupported token contract address
569
+ const basecoin = bitgo.coin('tbaseeth');
570
+ (0, nock_1.default)(baseUrl)
571
+ .get('/api')
572
+ .query(mockData.getTxListRequest(bitgoFeeAddress))
573
+ .reply(200, mockData.getTxListResponse);
574
+ (0, nock_1.default)(baseUrl)
575
+ .get('/api')
576
+ .query(mockData.getBalanceRequest(bitgoFeeAddress))
577
+ .reply(200, mockData.getBalanceResponse);
578
+ (0, nock_1.default)(baseUrl)
579
+ .get('/api')
580
+ .query(mockData.getTokenBalanceRequest(tokenContractAddress, walletContractAddress))
581
+ .reply(200, mockData.getBalanceResponse);
582
+ (0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
583
+ const transaction = (await basecoin.recover({
584
+ userKey: '',
585
+ backupKey: '',
586
+ walletContractAddress: walletContractAddress,
587
+ bitgoFeeAddress: bitgoFeeAddress,
588
+ recoveryDestination: destinationAddress,
589
+ eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
590
+ gasLimit: 500000,
591
+ bitgoDestinationAddress: bitgoDestinationAddress,
592
+ tokenContractAddress: tokenContractAddress,
593
+ common: resources_1.baseChainCommon,
594
+ }));
595
+ (0, assert_1.default)(transaction);
596
+ transaction.should.have.property('txHex');
597
+ transaction.should.have.property('coin');
598
+ transaction.coin.should.equal('tbaseeth');
599
+ transaction.should.have.property('contractSequenceId');
600
+ transaction.should.have.property('expireTime');
601
+ transaction.should.have.property('gasLimit');
602
+ transaction.gasLimit.should.equal('500000');
603
+ transaction.should.have.property('isEvmBasedCrossChainRecovery');
604
+ transaction.isEvmBasedCrossChainRecovery?.should.equal(true);
605
+ transaction.should.have.property('walletContractAddress');
606
+ transaction.walletContractAddress.should.equal(sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);
607
+ transaction.should.have.property('recipients');
608
+ const recipient = transaction.recipients[0];
609
+ recipient.should.have.property('address');
610
+ recipient.address.should.equal(destinationAddress);
611
+ recipient.should.have.property('amount');
612
+ recipient.amount.should.equal('9999999999999999928');
613
+ });
614
+ it('should generate an unsigned recovery txn of a token for hot wallet', async function () {
615
+ const userKey = '{"iv":"VFZ3jvXhxo1Z+Yaf2MtZnA==","v":1,"iter":10000,"ks":256,"ts":64,"mode"\n' +
616
+ ':"ccm","adata":"","cipher":"aes","salt":"p+fkHuLa/8k=","ct":"hYG7pvljLIgCjZ\n' +
617
+ '53PBlCde5KZRmlUKKHLtDMk+HJfuU46hW+x+C9WsIAO4gFPnTCvFVmQ8x7czCtcNFub5AO2otOG\n' +
618
+ 'OsX4GE2gXOEmCl1TpWwwNhm7yMUjGJUpgW6ZZgXSXdDitSKi4V/hk78SGSzjFOBSPYRa6I="}\n';
619
+ const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
620
+ const bitgoFeeAddress = '0x33a42faea3c6e87021347e51700b48aaf49aa1e7';
621
+ const destinationAddress = '0xd5adde17fed8baed3f32b84af05b8f2816f7b560';
622
+ const bitgoDestinationAddress = '0xE5986CE4490Deb67d2950562Ceb930Ddf9be7a14';
623
+ const walletPassphrase = sdk_test_1.TestBitGo.V2.TEST_RECOVERY_PASSCODE;
624
+ const tokenContractAddress = '0x326c977e6efc84e512bb9c30f76e30c160ed06fb'; // unsupported contract token address
625
+ const basecoin = bitgo.coin('tbaseeth');
626
+ (0, nock_1.default)(baseUrl)
627
+ .get('/api')
628
+ .query(mockData.getTxListRequest(bitgoFeeAddress))
629
+ .reply(200, mockData.getTxListResponse);
630
+ (0, nock_1.default)(baseUrl)
631
+ .get('/api')
632
+ .query(mockData.getBalanceRequest(bitgoFeeAddress))
633
+ .reply(200, mockData.getBalanceResponse);
634
+ (0, nock_1.default)(baseUrl)
635
+ .get('/api')
636
+ .query(mockData.getTokenBalanceRequest(tokenContractAddress, walletContractAddress))
637
+ .reply(200, mockData.getBalanceResponse);
638
+ (0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
639
+ const transaction = (await basecoin.recover({
640
+ userKey: userKey,
641
+ backupKey: '',
642
+ walletPassphrase: walletPassphrase,
643
+ walletContractAddress: walletContractAddress,
644
+ bitgoFeeAddress: bitgoFeeAddress,
645
+ recoveryDestination: destinationAddress,
646
+ eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
647
+ gasLimit: 500000,
648
+ bitgoDestinationAddress: bitgoDestinationAddress,
649
+ tokenContractAddress: tokenContractAddress,
650
+ common: resources_1.baseChainCommon,
651
+ }));
652
+ (0, assert_1.default)(transaction);
653
+ transaction.should.have.property('txHex');
654
+ transaction.should.have.property('coin');
655
+ transaction.coin.should.equal('tbaseeth');
656
+ transaction.should.have.property('contractSequenceId');
657
+ transaction.should.have.property('expireTime');
658
+ transaction.should.have.property('gasLimit');
659
+ transaction.gasLimit.should.equal('500000');
660
+ transaction.should.have.property('isEvmBasedCrossChainRecovery');
661
+ transaction.isEvmBasedCrossChainRecovery?.should.equal(true);
662
+ transaction.should.have.property('walletContractAddress');
663
+ transaction.walletContractAddress.should.equal(sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);
664
+ transaction.should.have.property('recipients');
665
+ const recipient = transaction.recipients[0];
666
+ recipient.should.have.property('address');
667
+ recipient.address.should.equal(destinationAddress);
668
+ recipient.should.have.property('amount');
669
+ recipient.amount.should.equal('9999999999999999928');
670
+ transaction.should.have.property('feesUsed');
671
+ transaction.feesUsed?.gasLimit.should.equal('500000');
672
+ transaction.should.have.property('halfSigned');
673
+ transaction.halfSigned?.should.have.property('txHex');
674
+ transaction.halfSigned?.should.have.property('recipients');
675
+ });
676
+ });
677
+ describe('Common parameter handling', function () {
678
+ it('should handle plain object common parameter for BASE chain', function () {
679
+ const baseCoin = bitgo.coin('baseeth');
680
+ const plainCommon = { chain: 8453, hardfork: 'london' };
681
+ const txBuilder = baseCoin.getTransactionBuilder(plainCommon);
682
+ txBuilder.should.be.an.instanceof(src_1.EthLikeTransactionBuilder);
683
+ });
684
+ it('should handle plain object common parameter with chainId property', function () {
685
+ const baseCoin = bitgo.coin('baseeth');
686
+ const plainCommon = { chainId: 8453, hardfork: 'london' };
687
+ const txBuilder = baseCoin.getTransactionBuilder(plainCommon);
688
+ txBuilder.should.be.an.instanceof(src_1.EthLikeTransactionBuilder);
689
+ });
690
+ it('should handle EthereumCommon instance', function () {
691
+ const baseCoin = bitgo.coin('baseeth');
692
+ const txBuilder = baseCoin.getTransactionBuilder(resources_1.baseChainCommon);
693
+ txBuilder.should.be.an.instanceof(src_1.EthLikeTransactionBuilder);
694
+ });
695
+ it('should require common parameter and throw error when undefined', function () {
696
+ const baseCoin = bitgo.coin('baseeth');
697
+ (() => {
698
+ baseCoin.getTransactionBuilder(undefined);
699
+ }).should.throw('Common must be provided for EthLikeTransactionBuilder');
700
+ });
701
+ it('should convert plain object common to EthereumCommon with working gteHardfork method', function () {
702
+ const baseCoin = bitgo.coin('baseeth');
703
+ const plainCommon = { chain: 8453, hardfork: 'london' };
704
+ const txBuilder = baseCoin.getTransactionBuilder(plainCommon);
705
+ txBuilder.should.be.an.instanceof(src_1.EthLikeTransactionBuilder);
706
+ const common = txBuilder._common;
707
+ common.should.not.be.undefined;
708
+ common.should.have.property('gteHardfork');
709
+ (typeof common.gteHardfork).should.equal('function');
710
+ const result = common.gteHardfork('homestead');
711
+ (typeof result).should.equal('boolean');
712
+ result.should.equal(true);
713
+ });
714
+ });
715
+ });
716
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ethlikeCoin.js","sourceRoot":"","sources":["../../../test/unit/ethlikeCoin.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\nimport { BitGoAPI } from '@bitgo-beta/sdk-api';\nimport { common, FullySignedTransaction, HalfSignedTransaction, TransactionType } from '@bitgo-beta/sdk-core';\nimport { OfflineVaultTxInfo, TransferBuilder } from '@bitgo-beta/abstract-eth';\nimport { TestBitGo, TestBitGoAPI } from '@bitgo-beta/sdk-test';\nimport { bip32 } from '@bitgo-beta/secp256k1';\nimport nock from 'nock';\n\nimport { EthLikeCoin, TethLikeCoin, EthLikeTransactionBuilder } from '../../src';\nimport { getBuilder } from '../getBuilder';\nimport { baseChainCommon, getCommon } from '../resources';\nimport * as mockData from '../fixtures/ethlikeCoin';\n\nnock.enableNetConnect();\n\nconst coins = [\n  {\n    name: 'hteth',\n    common: getCommon('hteth'),\n  },\n  {\n    name: 'tarbeth',\n    common: getCommon('tarbeth'),\n  },\n];\n\ndescribe('EthLike coin tests', function () {\n  let bitgo: TestBitGoAPI;\n  let basecoin: TethLikeCoin;\n  coins.forEach((coin) => {\n    describe(coin.name, function () {\n      before(function () {\n        const env = 'test';\n        bitgo = TestBitGo.decorate(BitGoAPI, { env });\n        bitgo.safeRegister(coin.name, TethLikeCoin.createInstance);\n        bitgo.initializeTestVars();\n        basecoin = bitgo.coin(coin.name) as TethLikeCoin;\n      });\n\n      after(function () {\n        nock.cleanAll();\n      });\n\n      it('should instantiate a coin', function () {\n        basecoin.should.be.an.instanceof(TethLikeCoin);\n      });\n      it('should reject for missing encryptedPrv for hot wallet', async function () {\n        const recoveryId = '0x1234567890abcdef';\n        nock(bitgo.microservicesUrl(`/api/recovery/v1/crosschain`)).get(`/${recoveryId}/buildtx`).reply(200, {\n          txHex: mockData.ccr[coin.name].txHex,\n        });\n        const walletPassphrase = TestBitGo.V2.TEST_RECOVERY_PASSCODE as string;\n        const params = {\n          recoveryId,\n          walletPassphrase,\n          common: coin.common,\n        };\n        await basecoin\n          .sendCrossChainRecoveryTransaction({ ...params, walletType: 'hot' })\n          .should.be.rejectedWith('missing encryptedPrv');\n      });\n      it('should send cross chain recovery transaction for hot wallet', async function () {\n        const recoveryId = '0x1234567890abcdef';\n        nock(bitgo.microservicesUrl(`/api/recovery/v1/crosschain`)).get(`/${recoveryId}/buildtx`).reply(200, {\n          txHex: mockData.ccr[coin.name].txHex,\n        });\n        nock(bitgo.microservicesUrl(`/api/recovery/v1/crosschain`)).post(`/${recoveryId}/sign`).reply(200, {\n          coin: coin.name,\n          txid: mockData.ccr[coin.name].txid,\n        });\n        const walletPassphrase = TestBitGo.V2.TEST_RECOVERY_PASSCODE as string;\n        const params = {\n          recoveryId,\n          walletPassphrase,\n          encryptedPrv: mockData.encryptedUserKey,\n          common: coin.common,\n        };\n        const result = await basecoin.sendCrossChainRecoveryTransaction({ ...params, walletType: 'hot' });\n        result.coin.should.equal(coin.name);\n        result.txid.should.equal(mockData.ccr[coin.name].txid);\n      });\n\n      it('should build txn for cross chain recovery for cold wallet', async function () {\n        const recoveryId = '0x1234567890abcdef';\n        nock(bitgo.microservicesUrl(`/api/recovery/v1/crosschain`)).get(`/${recoveryId}/buildtx`).reply(200, {\n          txHex: mockData.ccr[coin.name].txHex,\n        });\n        const params = {\n          recoveryId,\n          common: coin.common,\n        };\n        const result = await basecoin.sendCrossChainRecoveryTransaction({ ...params, walletType: 'cold' });\n        assert(result.txHex);\n        result.txHex.should.equal(mockData.ccr[coin.name].txHex);\n      });\n\n      it('should build cross chain recovery transaction and extract recipients', async function () {\n        const recoveryId = '0x1234567890abcdef';\n        const mockResponse = {\n          coin: coin.name,\n          txHex: mockData.ccr[coin.name].txHex,\n          txid: mockData.ccr[coin.name].txid,\n          walletVersion: 1,\n        };\n\n        nock(bitgo.microservicesUrl(`/api/recovery/v1/crosschain`))\n          .get(`/${recoveryId}/buildtx`)\n          .reply(200, mockResponse);\n\n        const result = await basecoin.buildCrossChainRecoveryTransaction(recoveryId);\n\n        result.should.have.property('coin');\n        result.coin.should.equal(coin.name);\n        result.should.have.property('txHex');\n        result.txHex.should.equal(mockData.ccr[coin.name].txHex);\n        result.should.have.property('txid');\n        result.txid.should.equal(mockData.ccr[coin.name].txid);\n        result.should.have.property('walletVersion');\n        result.should.have.property('recipients');\n        result.recipients.should.be.an.Array();\n        const recipient = result.recipients[0];\n        recipient.should.have.property('address');\n        recipient.should.have.property('amount');\n        recipient.address.should.be.a.String();\n        recipient.amount.should.be.a.String();\n      });\n\n      it('should generate signature data for custodial hot wallet and sign using hsm signature', async function () {\n        const baseAddress = '0x702cf81e03aa310ec9481d814e3d04a20b04b505';\n        const destinationAddress = '0xb9f62c71d5f6949cfb211a67fb13ccf079cc760b';\n        const tokenContractAddress = '0xe4ab69c077896252fafbd49efd26b5d171a32410';\n        const txBuilder = getBuilder(coin.name, coin.common) as EthLikeTransactionBuilder;\n\n        txBuilder.contract(baseAddress);\n        txBuilder.contractCounter(0);\n        txBuilder.fee({\n          fee: '100000',\n          gasLimit: '21000',\n        });\n\n        const transferBuilder = txBuilder.transfer() as TransferBuilder;\n        transferBuilder\n          .coin(coin.name)\n          .amount('100000000')\n          .contractSequenceId(100)\n          .expirationTime(1744049633)\n          .to(destinationAddress)\n          .tokenContractAddress(tokenContractAddress);\n        const signatureData = transferBuilder.getSignatureData();\n        assert.strictEqual(signatureData.toString('hex'), mockData.custodialHot[coin.name].signatureData);\n\n        // Set HSM Signature\n        transferBuilder.setSignature(mockData.custodialHot[coin.name].signature);\n        const tx = await txBuilder.build();\n        const txHex = tx.toBroadcastFormat();\n        assert.strictEqual(txHex, mockData.custodialHot[coin.name].signedTxHex);\n      });\n    });\n  });\n});\ndescribe('EthLikeCoin', function () {\n  let bitgo: TestBitGoAPI;\n  const coinName = 'tbaseeth';\n  let basecoin: TethLikeCoin;\n\n  before(function () {\n    const bitgoKeyXprv =\n      'xprv9s21ZrQH143K3tpWBHWe31sLoXNRQ9AvRYJgitkKxQ4ATFQMwvr7hHNqYRUnS7PsjzB7aK1VxqHLuNQjj1sckJ2Jwo2qxmsvejwECSpFMfC';\n    const bitgoKey = bip32.fromBase58(bitgoKeyXprv);\n    if (!bitgoKey.privateKey) {\n      throw new Error('no privateKey');\n    }\n    const bitgoXpub = bitgoKey.neutered().toBase58();\n    const env = 'test';\n    bitgo = TestBitGo.decorate(BitGoAPI, { env });\n    common.Environments[env].hsmXpub = bitgoXpub;\n    bitgo.safeRegister('baseeth', EthLikeCoin.createInstance);\n    bitgo.safeRegister('tbaseeth', TethLikeCoin.createInstance);\n    bitgo.initializeTestVars();\n    basecoin = bitgo.coin('tbaseeth') as TethLikeCoin;\n  });\n\n  after(function () {\n    nock.cleanAll();\n  });\n\n  it('should instantiate a coin', function () {\n    let coin = bitgo.coin('tbaseeth');\n    coin.should.be.an.instanceof(TethLikeCoin);\n    coin = bitgo.coin('baseeth');\n    coin.should.be.an.instanceof(EthLikeCoin);\n  });\n\n  it('should build unsigned transaction', async function () {\n    const expireTime = Math.floor(new Date().getTime() / 1000);\n    const txBuilder = getBuilder(coinName, baseChainCommon) as EthLikeTransactionBuilder;\n    txBuilder.type(TransactionType.Send);\n    txBuilder.fee({\n      fee: '1000000000',\n      gasLimit: '100000',\n    });\n    txBuilder.counter(1);\n    txBuilder.contract('0x2c2b9c9a4a25e24b174f26114e8926a9f2128fe4');\n\n    const transferBuilder = txBuilder.transfer();\n    transferBuilder\n      .coin(coinName)\n      .expirationTime(expireTime)\n      .amount('1000000000000000000')\n      .to('0x2c2b9c9a4a25e24b174f26114e8926a9f2128fe4')\n      .contractSequenceId(1);\n\n    const tx = await txBuilder.build();\n    const txJson = tx.toJson();\n    txJson.gasLimit.should.equal('100000');\n    txJson.gasPrice.should.equal('1000000000');\n    txJson.chainId.should.equal('0x14a34');\n  });\n\n  it('should sign a transaction', async function () {\n    const account_1 = {\n      address: '0x8Ce59c2d1702844F8EdED451AA103961bC37B4e8',\n      owner_1: '4ee089aceabf3ddbf748db79b1066c33b7d3ea1ab3eb7e325121bba2bff2f5ca',\n      owner_2: '5c7e4efff7304d4dfff6d5f1591844ec6f2adfa6a47e9fece6a3c1a4d755f1e3',\n      owner_3: '4421ab25dd91e1a3180d03d57c323a7886dcc313d3b3a4b4256a5791572bf597',\n    };\n    const expireTime = Math.floor(new Date().getTime() / 1000);\n    const txBuilder = getBuilder(coinName, baseChainCommon) as EthLikeTransactionBuilder;\n    txBuilder.type(TransactionType.Send);\n    txBuilder.fee({\n      fee: '1000000000',\n      gasLimit: '100000',\n    });\n    txBuilder.counter(1);\n    txBuilder.contract(account_1.address);\n\n    const transferBuilder = txBuilder.transfer();\n    transferBuilder\n      .coin(coinName)\n      .expirationTime(expireTime)\n      .amount('1000000000000000000')\n      .to('0x2c2b9c9a4a25e24b174f26114e8926a9f2128fe4')\n      .contractSequenceId(1);\n\n    const unsignedTx = await txBuilder.build();\n    const unsignedTxHex = unsignedTx.toBroadcastFormat();\n    const halfSignedTx = (await basecoin.signTransaction({\n      txPrebuild: {\n        txHex: unsignedTxHex,\n      },\n      prv: account_1.owner_1,\n      common: baseChainCommon,\n    })) as HalfSignedTransaction;\n    transferBuilder.key(account_1.owner_1);\n    const halfSignedTxBuilder = await txBuilder.build();\n    const halfSignedTxHexBuilder = halfSignedTxBuilder.toBroadcastFormat();\n    halfSignedTxHexBuilder.should.equal(halfSignedTx.halfSigned.txHex);\n\n    // Sign with the second key\n\n    const fullSignedTxn = (await basecoin.signTransaction({\n      txPrebuild: {\n        halfSigned: {\n          txHex: halfSignedTxHexBuilder,\n          expireTime: expireTime,\n          contractSequenceId: 1,\n          signature: '',\n        },\n      },\n      prv: account_1.owner_2,\n      common: baseChainCommon,\n      isLastSignature: true,\n    })) as FullySignedTransaction;\n\n    assert(fullSignedTxn.txHex);\n  });\n\n  describe('explainTransaction', function () {\n    const txHex = mockData.ccr[coinName].txHex;\n    const feeInfo = {\n      fee: '1000000000',\n      gasLimit: '100000',\n    };\n\n    it('should explain transaction when common is provided', async function () {\n      const explanation = await basecoin.explainTransaction({\n        txHex,\n        feeInfo,\n        common: baseChainCommon,\n      });\n\n      explanation.should.have.property('id');\n      explanation.should.have.property('outputs');\n      explanation.should.have.property('outputAmount');\n      explanation.should.have.property('changeOutputs');\n      explanation.should.have.property('changeAmount');\n      explanation.should.have.property('fee');\n      explanation.fee.should.equal(feeInfo);\n      explanation.outputs.should.be.an.Array();\n    });\n\n    it('should fail to explain transaction when common is not provided', async function () {\n      await basecoin\n        .explainTransaction({\n          txHex,\n          feeInfo,\n        })\n        .should.be.rejectedWith('Common must be provided for EthLikeTransactionBuilder');\n    });\n  });\n\n  describe('Recovery', function () {\n    const baseUrl = 'https://api-sepolia.basescan.org/';\n    const userXpub =\n      'xpub661MyMwAqRbcEeTc8789MK5PUGEYiPG4F4V17n2Rd2LoTATA1XoCnJT5FAYAShQxSxtFjpo5NHmcWwTp2LiWGBMwpUcAA3HywhxivgYfq7q';\n    const backupXpub =\n      'xpub661MyMwAqRbcFZX15xpZf4ERCGHiVSJm8r5C4yh1yXV2GrdZCUPYo4WQr6tN9oUywKXsgSHo7Risf9r22GH5joVD2hEEEhqnSCvK8qy11wW';\n\n    it('should generate an unsigned sweep transaction', async function () {\n      const walletContractAddress = TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS as string;\n      const backupKeyAddress = '0x4f2c4830cc37f2785c646f89ded8a919219fa0e9';\n      nock(baseUrl)\n        .get('/api')\n        .twice()\n        .query(mockData.getTxListRequest(backupKeyAddress))\n        .reply(200, mockData.getTxListResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(walletContractAddress))\n        .reply(200, mockData.getBalanceResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(backupKeyAddress))\n        .reply(200, mockData.getBalanceResponse);\n\n      nock(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);\n\n      const baseCoin = bitgo.coin('tbaseeth') as TethLikeCoin;\n      const transaction = (await baseCoin.recover({\n        userKey: userXpub,\n        backupKey: backupXpub,\n        walletContractAddress: walletContractAddress,\n        recoveryDestination: TestBitGo.V2.TEST_ERC20_TOKEN_RECIPIENT as string,\n        eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },\n        gasLimit: 500000,\n        common: baseChainCommon,\n      })) as OfflineVaultTxInfo;\n      assert(transaction.txHex);\n      assert(transaction.contractSequenceId);\n      assert.strictEqual(transaction.gasLimit, '500000');\n    });\n  });\n\n  describe('Evm Based Cross Chain Recovery transaction:', function () {\n    const baseUrl = 'https://api-sepolia.basescan.org/';\n    const userXpub =\n      'xpub661MyMwAqRbcEeTc8789MK5PUGEYiPG4F4V17n2Rd2LoTATA1XoCnJT5FAYAShQxSxtFjpo5NHmcWwTp2LiWGBMwpUcAA3HywhxivgYfq7q';\n\n    it('should generate an unsigned recovery txn for cold wallet', async function () {\n      const walletContractAddress = TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS as string;\n      const bitgoFeeAddress = '0x33a42faea3c6e87021347e51700b48aaf49aa1e7';\n      const destinationAddress = '0xd5adde17fed8baed3f32b84af05b8f2816f7b560';\n      const bitgoDestinationAddress = '0xE5986CE4490Deb67d2950562Ceb930Ddf9be7a14';\n\n      const basecoin = bitgo.coin('tbaseeth') as TethLikeCoin;\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getTxListRequest(bitgoFeeAddress))\n        .reply(200, mockData.getTxListResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(bitgoFeeAddress))\n        .reply(200, mockData.getBalanceResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(walletContractAddress))\n        .reply(200, mockData.getBalanceResponse);\n      nock(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);\n\n      const transaction = (await basecoin.recover({\n        userKey: userXpub,\n        backupKey: '',\n        walletContractAddress: walletContractAddress,\n        bitgoFeeAddress: bitgoFeeAddress,\n        recoveryDestination: destinationAddress,\n        eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },\n        gasLimit: 500000,\n        bitgoDestinationAddress: bitgoDestinationAddress,\n        common: baseChainCommon,\n      })) as OfflineVaultTxInfo;\n\n      assert(transaction);\n      transaction.should.have.property('txHex');\n      transaction.should.have.property('userKey');\n      transaction.should.have.property('coin');\n      transaction.should.have.property('contractSequenceId');\n      transaction.should.have.property('expireTime');\n      transaction.should.have.property('gasLimit');\n      transaction.gasLimit.should.equal('500000');\n      transaction.should.have.property('isEvmBasedCrossChainRecovery');\n      transaction.isEvmBasedCrossChainRecovery?.should.equal(true);\n      transaction.should.have.property('walletContractAddress');\n      transaction.walletContractAddress.should.equal(TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);\n      transaction.should.have.property('recipients');\n      const recipient = transaction.recipients[0];\n      recipient.should.have.property('address');\n      recipient.address.should.equal(destinationAddress);\n      recipient.should.have.property('amount');\n      recipient.amount.should.equal('9999999999999999928');\n    });\n\n    it('should generate an unsigned recovery txn for custody wallet', async function () {\n      const walletContractAddress = TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS as string;\n      const bitgoFeeAddress = '0x33a42faea3c6e87021347e51700b48aaf49aa1e7';\n      const destinationAddress = '0xd5adde17fed8baed3f32b84af05b8f2816f7b560';\n      const bitgoDestinationAddress = '0xE5986CE4490Deb67d2950562Ceb930Ddf9be7a14';\n\n      const basecoin = bitgo.coin('tbaseeth') as TethLikeCoin;\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getTxListRequest(bitgoFeeAddress))\n        .reply(200, mockData.getTxListResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(bitgoFeeAddress))\n        .reply(200, mockData.getBalanceResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(walletContractAddress))\n        .reply(200, mockData.getBalanceResponse);\n      nock(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);\n\n      const transaction = (await basecoin.recover({\n        userKey: '',\n        backupKey: '',\n        walletContractAddress: walletContractAddress,\n        bitgoFeeAddress: bitgoFeeAddress,\n        recoveryDestination: destinationAddress,\n        eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },\n        gasLimit: 500000,\n        bitgoDestinationAddress: bitgoDestinationAddress,\n        common: baseChainCommon,\n      })) as OfflineVaultTxInfo;\n\n      assert(transaction);\n      transaction.should.have.property('txHex');\n      transaction.should.have.property('coin');\n      transaction.should.have.property('contractSequenceId');\n      transaction.should.have.property('expireTime');\n      transaction.should.have.property('gasLimit');\n      transaction.gasLimit.should.equal('500000');\n      transaction.should.have.property('isEvmBasedCrossChainRecovery');\n      transaction.isEvmBasedCrossChainRecovery?.should.equal(true);\n      transaction.should.have.property('walletContractAddress');\n      transaction.walletContractAddress.should.equal(TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);\n      transaction.should.have.property('recipients');\n      const recipient = transaction.recipients[0];\n      recipient.should.have.property('address');\n      recipient.address.should.equal(destinationAddress);\n      recipient.should.have.property('amount');\n      recipient.amount.should.equal('9999999999999999928');\n    });\n\n    it('should generate an unsigned recovery txn for hot wallet', async function () {\n      const userKey =\n        '{\"iv\":\"VFZ3jvXhxo1Z+Yaf2MtZnA==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\"\\n' +\n        ':\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"p+fkHuLa/8k=\",\"ct\":\"hYG7pvljLIgCjZ\\n' +\n        '53PBlCde5KZRmlUKKHLtDMk+HJfuU46hW+x+C9WsIAO4gFPnTCvFVmQ8x7czCtcNFub5AO2otOG\\n' +\n        'OsX4GE2gXOEmCl1TpWwwNhm7yMUjGJUpgW6ZZgXSXdDitSKi4V/hk78SGSzjFOBSPYRa6I=\"}\\n';\n      const walletContractAddress = TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS as string;\n      const bitgoFeeAddress = '0x33a42faea3c6e87021347e51700b48aaf49aa1e7';\n      const destinationAddress = '0xd5adde17fed8baed3f32b84af05b8f2816f7b560';\n      const bitgoDestinationAddress = '0xE5986CE4490Deb67d2950562Ceb930Ddf9be7a14';\n      const walletPassphrase = TestBitGo.V2.TEST_RECOVERY_PASSCODE as string;\n\n      const basecoin = bitgo.coin('tbaseeth') as TethLikeCoin;\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getTxListRequest(bitgoFeeAddress))\n        .reply(200, mockData.getTxListResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(bitgoFeeAddress))\n        .reply(200, mockData.getBalanceResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(walletContractAddress))\n        .reply(200, mockData.getBalanceResponse);\n      nock(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);\n\n      const transaction = (await basecoin.recover({\n        userKey: userKey,\n        backupKey: '',\n        walletPassphrase: walletPassphrase,\n        walletContractAddress: walletContractAddress,\n        bitgoFeeAddress: bitgoFeeAddress,\n        recoveryDestination: destinationAddress,\n        eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },\n        gasLimit: 500000,\n        bitgoDestinationAddress: bitgoDestinationAddress,\n        common: baseChainCommon,\n      })) as OfflineVaultTxInfo;\n\n      assert(transaction);\n      transaction.should.have.property('txHex');\n      transaction.should.have.property('coin');\n      transaction.should.have.property('contractSequenceId');\n      transaction.should.have.property('expireTime');\n      transaction.should.have.property('gasLimit');\n      transaction.gasLimit.should.equal('500000');\n      transaction.should.have.property('isEvmBasedCrossChainRecovery');\n      transaction.isEvmBasedCrossChainRecovery?.should.equal(true);\n      transaction.should.have.property('walletContractAddress');\n      transaction.walletContractAddress.should.equal(TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);\n      transaction.should.have.property('recipients');\n      const recipient = transaction.recipients[0];\n      recipient.should.have.property('address');\n      recipient.address.should.equal(destinationAddress);\n      recipient.should.have.property('amount');\n      recipient.amount.should.equal('9999999999999999928');\n      transaction.should.have.property('feesUsed');\n      transaction.feesUsed?.gasLimit.should.equal('500000');\n      transaction.should.have.property('halfSigned');\n      transaction.halfSigned?.should.have.property('txHex');\n      transaction.halfSigned?.should.have.property('recipients');\n    });\n\n    it('should generate an unsigned recovery txn of a token for cold wallet ', async function () {\n      const walletContractAddress = TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS as string;\n      const bitgoFeeAddress = '0x33a42faea3c6e87021347e51700b48aaf49aa1e7';\n      const destinationAddress = '0xd5adde17fed8baed3f32b84af05b8f2816f7b560';\n      const bitgoDestinationAddress = '0xE5986CE4490Deb67d2950562Ceb930Ddf9be7a14';\n      const tokenContractAddress = '0x326c977e6efc84e512bb9c30f76e30c160ed06fb';\n\n      const basecoin = bitgo.coin('tbaseeth') as TethLikeCoin;\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getTxListRequest(bitgoFeeAddress))\n        .reply(200, mockData.getTxListResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(bitgoFeeAddress))\n        .reply(200, mockData.getBalanceResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getTokenBalanceRequest(tokenContractAddress, walletContractAddress))\n        .reply(200, mockData.getBalanceResponse);\n      nock(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);\n\n      const transaction = (await basecoin.recover({\n        userKey: userXpub,\n        backupKey: '',\n        walletContractAddress: walletContractAddress,\n        bitgoFeeAddress: bitgoFeeAddress,\n        recoveryDestination: destinationAddress,\n        eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },\n        gasLimit: 500000,\n        bitgoDestinationAddress: bitgoDestinationAddress,\n        tokenContractAddress: tokenContractAddress,\n        common: baseChainCommon,\n      })) as OfflineVaultTxInfo;\n\n      assert(transaction);\n      transaction.should.have.property('txHex');\n      transaction.should.have.property('userKey');\n      transaction.should.have.property('coin');\n      transaction.coin.should.equal('tbaseeth');\n      transaction.should.have.property('contractSequenceId');\n      transaction.should.have.property('expireTime');\n      transaction.should.have.property('gasLimit');\n      transaction.gasLimit.should.equal('500000');\n      transaction.should.have.property('isEvmBasedCrossChainRecovery');\n      transaction.isEvmBasedCrossChainRecovery?.should.equal(true);\n      transaction.should.have.property('walletContractAddress');\n      transaction.walletContractAddress.should.equal(TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);\n      transaction.should.have.property('recipients');\n      const recipient = transaction.recipients[0];\n      recipient.should.have.property('address');\n      recipient.address.should.equal(destinationAddress);\n      recipient.should.have.property('amount');\n      recipient.amount.should.equal('9999999999999999928');\n    });\n\n    it('should generate an unsigned recovery txn of a token for custody wallet', async function () {\n      const walletContractAddress = TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS as string;\n      const bitgoFeeAddress = '0x33a42faea3c6e87021347e51700b48aaf49aa1e7';\n      const destinationAddress = '0xd5adde17fed8baed3f32b84af05b8f2816f7b560';\n      const bitgoDestinationAddress = '0xE5986CE4490Deb67d2950562Ceb930Ddf9be7a14';\n      const tokenContractAddress = '0x326c977e6efc84e512bb9c30f76e30c160ed06fb'; // unsupported token contract address\n\n      const basecoin = bitgo.coin('tbaseeth') as TethLikeCoin;\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getTxListRequest(bitgoFeeAddress))\n        .reply(200, mockData.getTxListResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(bitgoFeeAddress))\n        .reply(200, mockData.getBalanceResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getTokenBalanceRequest(tokenContractAddress, walletContractAddress))\n        .reply(200, mockData.getBalanceResponse);\n      nock(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);\n\n      const transaction = (await basecoin.recover({\n        userKey: '',\n        backupKey: '',\n        walletContractAddress: walletContractAddress,\n        bitgoFeeAddress: bitgoFeeAddress,\n        recoveryDestination: destinationAddress,\n        eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },\n        gasLimit: 500000,\n        bitgoDestinationAddress: bitgoDestinationAddress,\n        tokenContractAddress: tokenContractAddress,\n        common: baseChainCommon,\n      })) as OfflineVaultTxInfo;\n\n      assert(transaction);\n      transaction.should.have.property('txHex');\n      transaction.should.have.property('coin');\n      transaction.coin.should.equal('tbaseeth');\n      transaction.should.have.property('contractSequenceId');\n      transaction.should.have.property('expireTime');\n      transaction.should.have.property('gasLimit');\n      transaction.gasLimit.should.equal('500000');\n      transaction.should.have.property('isEvmBasedCrossChainRecovery');\n      transaction.isEvmBasedCrossChainRecovery?.should.equal(true);\n      transaction.should.have.property('walletContractAddress');\n      transaction.walletContractAddress.should.equal(TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);\n      transaction.should.have.property('recipients');\n      const recipient = transaction.recipients[0];\n      recipient.should.have.property('address');\n      recipient.address.should.equal(destinationAddress);\n      recipient.should.have.property('amount');\n      recipient.amount.should.equal('9999999999999999928');\n    });\n\n    it('should generate an unsigned recovery txn of a token for hot wallet', async function () {\n      const userKey =\n        '{\"iv\":\"VFZ3jvXhxo1Z+Yaf2MtZnA==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\"\\n' +\n        ':\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"p+fkHuLa/8k=\",\"ct\":\"hYG7pvljLIgCjZ\\n' +\n        '53PBlCde5KZRmlUKKHLtDMk+HJfuU46hW+x+C9WsIAO4gFPnTCvFVmQ8x7czCtcNFub5AO2otOG\\n' +\n        'OsX4GE2gXOEmCl1TpWwwNhm7yMUjGJUpgW6ZZgXSXdDitSKi4V/hk78SGSzjFOBSPYRa6I=\"}\\n';\n      const walletContractAddress = TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS as string;\n      const bitgoFeeAddress = '0x33a42faea3c6e87021347e51700b48aaf49aa1e7';\n      const destinationAddress = '0xd5adde17fed8baed3f32b84af05b8f2816f7b560';\n      const bitgoDestinationAddress = '0xE5986CE4490Deb67d2950562Ceb930Ddf9be7a14';\n      const walletPassphrase = TestBitGo.V2.TEST_RECOVERY_PASSCODE as string;\n      const tokenContractAddress = '0x326c977e6efc84e512bb9c30f76e30c160ed06fb'; // unsupported contract token address\n\n      const basecoin = bitgo.coin('tbaseeth') as TethLikeCoin;\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getTxListRequest(bitgoFeeAddress))\n        .reply(200, mockData.getTxListResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(bitgoFeeAddress))\n        .reply(200, mockData.getBalanceResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getTokenBalanceRequest(tokenContractAddress, walletContractAddress))\n        .reply(200, mockData.getBalanceResponse);\n      nock(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);\n\n      const transaction = (await basecoin.recover({\n        userKey: userKey,\n        backupKey: '',\n        walletPassphrase: walletPassphrase,\n        walletContractAddress: walletContractAddress,\n        bitgoFeeAddress: bitgoFeeAddress,\n        recoveryDestination: destinationAddress,\n        eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },\n        gasLimit: 500000,\n        bitgoDestinationAddress: bitgoDestinationAddress,\n        tokenContractAddress: tokenContractAddress,\n        common: baseChainCommon,\n      })) as OfflineVaultTxInfo;\n\n      assert(transaction);\n      transaction.should.have.property('txHex');\n      transaction.should.have.property('coin');\n      transaction.coin.should.equal('tbaseeth');\n      transaction.should.have.property('contractSequenceId');\n      transaction.should.have.property('expireTime');\n      transaction.should.have.property('gasLimit');\n      transaction.gasLimit.should.equal('500000');\n      transaction.should.have.property('isEvmBasedCrossChainRecovery');\n      transaction.isEvmBasedCrossChainRecovery?.should.equal(true);\n      transaction.should.have.property('walletContractAddress');\n      transaction.walletContractAddress.should.equal(TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);\n      transaction.should.have.property('recipients');\n      const recipient = transaction.recipients[0];\n      recipient.should.have.property('address');\n      recipient.address.should.equal(destinationAddress);\n      recipient.should.have.property('amount');\n      recipient.amount.should.equal('9999999999999999928');\n      transaction.should.have.property('feesUsed');\n      transaction.feesUsed?.gasLimit.should.equal('500000');\n      transaction.should.have.property('halfSigned');\n      transaction.halfSigned?.should.have.property('txHex');\n      transaction.halfSigned?.should.have.property('recipients');\n    });\n  });\n\n  describe('Common parameter handling', function () {\n    it('should handle plain object common parameter for BASE chain', function () {\n      const baseCoin = bitgo.coin('baseeth') as EthLikeCoin;\n\n      const plainCommon = { chain: 8453, hardfork: 'london' };\n\n      const txBuilder = (baseCoin as any).getTransactionBuilder(plainCommon);\n      txBuilder.should.be.an.instanceof(EthLikeTransactionBuilder);\n    });\n\n    it('should handle plain object common parameter with chainId property', function () {\n      const baseCoin = bitgo.coin('baseeth') as EthLikeCoin;\n\n      const plainCommon = { chainId: 8453, hardfork: 'london' };\n\n      const txBuilder = (baseCoin as any).getTransactionBuilder(plainCommon);\n      txBuilder.should.be.an.instanceof(EthLikeTransactionBuilder);\n    });\n\n    it('should handle EthereumCommon instance', function () {\n      const baseCoin = bitgo.coin('baseeth') as EthLikeCoin;\n\n      const txBuilder = (baseCoin as any).getTransactionBuilder(baseChainCommon);\n      txBuilder.should.be.an.instanceof(EthLikeTransactionBuilder);\n    });\n\n    it('should require common parameter and throw error when undefined', function () {\n      const baseCoin = bitgo.coin('baseeth') as EthLikeCoin;\n\n      (() => {\n        (baseCoin as any).getTransactionBuilder(undefined);\n      }).should.throw('Common must be provided for EthLikeTransactionBuilder');\n    });\n\n    it('should convert plain object common to EthereumCommon with working gteHardfork method', function () {\n      const baseCoin = bitgo.coin('baseeth') as EthLikeCoin;\n\n      const plainCommon = { chain: 8453, hardfork: 'london' };\n\n      const txBuilder = (baseCoin as any).getTransactionBuilder(plainCommon);\n      txBuilder.should.be.an.instanceof(EthLikeTransactionBuilder);\n\n      const common = (txBuilder as any)._common;\n      common.should.not.be.undefined;\n\n      common.should.have.property('gteHardfork');\n      (typeof common.gteHardfork).should.equal('function');\n\n      const result = common.gteHardfork('homestead');\n      (typeof result).should.equal('boolean');\n      result.should.equal(true);\n    });\n  });\n});\n"]}