@bitgo-beta/sdk-coin-polygon 1.3.3-alpha.45 → 1.3.3-alpha.450
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/dist/src/index.js +6 -2
- package/dist/src/lib/index.d.ts +2 -1
- package/dist/src/lib/index.d.ts.map +1 -1
- package/dist/src/lib/index.js +30 -12
- package/dist/src/lib/resources.d.ts.map +1 -1
- package/dist/src/lib/resources.js +8 -4
- package/dist/src/lib/transactionBuilder.d.ts +11 -8
- package/dist/src/lib/transactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilder.js +21 -21
- package/dist/src/lib/transferBuilder.d.ts +1 -1
- package/dist/src/lib/transferBuilder.js +3 -3
- package/dist/src/lib/utils.d.ts +1 -1
- package/dist/src/lib/utils.js +6 -6
- package/dist/src/polygon.d.ts +23 -63
- package/dist/src/polygon.d.ts.map +1 -1
- package/dist/src/polygon.js +26 -392
- package/dist/src/polygonToken.d.ts +15 -24
- package/dist/src/polygonToken.d.ts.map +1 -1
- package/dist/src/polygonToken.js +31 -59
- package/dist/src/tpolygon.d.ts +0 -1
- package/dist/src/tpolygon.d.ts.map +1 -1
- package/dist/src/tpolygon.js +1 -4
- package/dist/test/fixtures/polygon.d.ts +151 -0
- package/dist/test/fixtures/polygon.d.ts.map +1 -0
- package/dist/test/fixtures/polygon.js +233 -0
- package/dist/test/getBuilder.d.ts +3 -0
- package/dist/test/getBuilder.d.ts.map +1 -0
- package/dist/test/getBuilder.js +9 -0
- package/dist/test/resources.d.ts +27 -0
- package/dist/test/resources.d.ts.map +1 -0
- package/dist/test/resources.js +41 -0
- package/dist/test/unit/polygon.d.ts +2 -0
- package/dist/test/unit/polygon.d.ts.map +1 -0
- package/dist/test/unit/polygon.js +1202 -0
- package/dist/test/unit/polygonToken.d.ts +2 -0
- package/dist/test/unit/polygonToken.d.ts.map +1 -0
- package/dist/test/unit/polygonToken.js +62 -0
- package/dist/test/unit/transaction.d.ts +2 -0
- package/dist/test/unit/transaction.d.ts.map +1 -0
- package/dist/test/unit/transaction.js +92 -0
- package/dist/test/unit/transactionBuilder/addressInitialization.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/addressInitialization.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/addressInitialization.js +104 -0
- package/dist/test/unit/transactionBuilder/flushTokens.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/flushTokens.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/flushTokens.js +9 -0
- package/dist/test/unit/transactionBuilder/send.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/send.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/send.js +114 -0
- package/dist/test/unit/transactionBuilder/walletInitialization.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/walletInitialization.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/walletInitialization.js +169 -0
- package/dist/tsconfig.tsbuildinfo +1 -0
- package/package.json +20 -19
- package/.eslintignore +0 -5
- package/.mocharc.yml +0 -8
- package/CHANGELOG.md +0 -102
|
@@ -0,0 +1,1202 @@
|
|
|
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 bignumber_js_1 = __importDefault(require("bignumber.js"));
|
|
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 secp256k1 = __importStar(require("secp256k1"));
|
|
46
|
+
const should = __importStar(require("should"));
|
|
47
|
+
const src_1 = require("../../src");
|
|
48
|
+
const abstract_eth_1 = require("@bitgo-beta/abstract-eth");
|
|
49
|
+
const getBuilder_1 = require("../getBuilder");
|
|
50
|
+
const mockData = __importStar(require("../fixtures/polygon"));
|
|
51
|
+
const sjcl = __importStar(require("@bitgo-beta/sjcl"));
|
|
52
|
+
const assert_1 = __importDefault(require("assert"));
|
|
53
|
+
nock_1.default.enableNetConnect();
|
|
54
|
+
describe('Polygon', function () {
|
|
55
|
+
let bitgo;
|
|
56
|
+
let basecoin;
|
|
57
|
+
let hopTxBitgoSignature;
|
|
58
|
+
const address1 = '0x174cfd823af8ce27ed0afee3fcf3c3ba259116be';
|
|
59
|
+
const address2 = '0x7e85bdc27c050e3905ebf4b8e634d9ad6edd0de6';
|
|
60
|
+
const hopContractAddress = '0x47ce7cc86efefef19f8fb516b11735d183da8635';
|
|
61
|
+
const hopDestinationAddress = '0x9c7e8ce6825bD48278B3Ab59228EE26f8BE7925b';
|
|
62
|
+
const hopTx = '0xf86b808504a817c8ff8252ff949c7e8ce6825bd48278b3ab59228ee26f8be7925b87038d7ea4c68000801ca011bc22c664570133dfca4f08a0b8d02339cf467046d6a4152f04f368d0eaf99ea01d6dc5cf0c897c8d4c3e1df53d0d042784c424536a4cc5b802552b7d64fee8b5';
|
|
63
|
+
const hopTxid = '0x4af65143bc77da2b50f35b3d13cacb4db18f026bf84bc0743550bc57b9b53351';
|
|
64
|
+
const userReqSig = '0x404db307f6147f0d8cd338c34c13906ef46a6faa7e0e119d5194ef05aec16e6f3d710f9b7901460f97e924066b62efd74443bd34402c6d40b49c203a559ff2c8';
|
|
65
|
+
before(function () {
|
|
66
|
+
const bitgoKeyXprv = 'xprv9s21ZrQH143K3tpWBHWe31sLoXNRQ9AvRYJgitkKxQ4ATFQMwvr7hHNqYRUnS7PsjzB7aK1VxqHLuNQjj1sckJ2Jwo2qxmsvejwECSpFMfC';
|
|
67
|
+
const bitgoKey = secp256k1_1.bip32.fromBase58(bitgoKeyXprv);
|
|
68
|
+
if (!bitgoKey.privateKey) {
|
|
69
|
+
throw new Error('no privateKey');
|
|
70
|
+
}
|
|
71
|
+
const bitgoXpub = bitgoKey.neutered().toBase58();
|
|
72
|
+
hopTxBitgoSignature =
|
|
73
|
+
'0xaa' +
|
|
74
|
+
Buffer.from(secp256k1.ecdsaSign(Buffer.from(hopTxid.slice(2), 'hex'), bitgoKey.privateKey).signature).toString('hex');
|
|
75
|
+
const env = 'test';
|
|
76
|
+
bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env });
|
|
77
|
+
sdk_core_1.common.Environments[env].hsmXpub = bitgoXpub;
|
|
78
|
+
bitgo.safeRegister('polygon', src_1.Polygon.createInstance);
|
|
79
|
+
bitgo.safeRegister('tpolygon', src_1.Tpolygon.createInstance);
|
|
80
|
+
bitgo.initializeTestVars();
|
|
81
|
+
basecoin = bitgo.coin('tpolygon');
|
|
82
|
+
});
|
|
83
|
+
after(function () {
|
|
84
|
+
nock_1.default.cleanAll();
|
|
85
|
+
});
|
|
86
|
+
/**
|
|
87
|
+
* Build an unsigned account-lib multi-signature send transactino
|
|
88
|
+
* @param destination The destination address of the transaction
|
|
89
|
+
* @param contractAddress The address of the smart contract processing the transaction
|
|
90
|
+
* @param contractSequenceId The sequence id of the contract
|
|
91
|
+
* @param nonce The nonce of the sending address
|
|
92
|
+
* @param expireTime The expire time of the transaction
|
|
93
|
+
* @param amount The amount to send to the recipient
|
|
94
|
+
* @param gasPrice The gas price of the transaction
|
|
95
|
+
* @param gasLimit The gas limit of the transaction
|
|
96
|
+
*/
|
|
97
|
+
const buildUnsignedTransaction = async function ({ destination, contractAddress, contractSequenceId = 1, nonce = 0, expireTime = Math.floor(new Date().getTime() / 1000), amount = '100000', gasPrice = '10000', gasLimit = '20000', }) {
|
|
98
|
+
const txBuilder = (0, getBuilder_1.getBuilder)('tpolygon');
|
|
99
|
+
txBuilder.type(sdk_core_1.TransactionType.Send);
|
|
100
|
+
txBuilder.fee({
|
|
101
|
+
fee: gasPrice,
|
|
102
|
+
gasLimit: gasLimit,
|
|
103
|
+
});
|
|
104
|
+
txBuilder.counter(nonce);
|
|
105
|
+
txBuilder.contract(contractAddress);
|
|
106
|
+
const transferBuilder = txBuilder.transfer();
|
|
107
|
+
transferBuilder
|
|
108
|
+
.coin('tpolygon')
|
|
109
|
+
.expirationTime(expireTime)
|
|
110
|
+
.amount(amount)
|
|
111
|
+
.to(destination)
|
|
112
|
+
.contractSequenceId(contractSequenceId);
|
|
113
|
+
return await txBuilder.build();
|
|
114
|
+
};
|
|
115
|
+
describe('Instantiate', () => {
|
|
116
|
+
it('should instantiate the coin', function () {
|
|
117
|
+
let localBasecoin = bitgo.coin('tpolygon');
|
|
118
|
+
localBasecoin.should.be.an.instanceof(src_1.Tpolygon);
|
|
119
|
+
localBasecoin = bitgo.coin('polygon');
|
|
120
|
+
localBasecoin.should.be.an.instanceof(src_1.Polygon);
|
|
121
|
+
});
|
|
122
|
+
});
|
|
123
|
+
describe('Explain transaction:', () => {
|
|
124
|
+
it('should fail if the options object is missing parameters', async function () {
|
|
125
|
+
const explainParams = {
|
|
126
|
+
feeInfo: { fee: 1 },
|
|
127
|
+
txHex: null,
|
|
128
|
+
};
|
|
129
|
+
await basecoin.explainTransaction(explainParams).should.be.rejectedWith('missing explain tx parameters');
|
|
130
|
+
});
|
|
131
|
+
it('explain a transfer transaction', async function () {
|
|
132
|
+
const destination = '0xfaa8f14f46a99eb439c50e0c3b835cc21dad51b4';
|
|
133
|
+
const contractAddress = '0x9e2c5712ab4caf402a98c4bf58c79a0dfe718ad1';
|
|
134
|
+
const unsignedTransaction = await buildUnsignedTransaction({
|
|
135
|
+
destination,
|
|
136
|
+
contractAddress,
|
|
137
|
+
});
|
|
138
|
+
const explainParams = {
|
|
139
|
+
halfSigned: {
|
|
140
|
+
txHex: unsignedTransaction.toBroadcastFormat(),
|
|
141
|
+
},
|
|
142
|
+
feeInfo: { fee: 1 },
|
|
143
|
+
};
|
|
144
|
+
const explanation = await basecoin.explainTransaction(explainParams);
|
|
145
|
+
should.exist(explanation.id);
|
|
146
|
+
});
|
|
147
|
+
});
|
|
148
|
+
describe('Sign Transaction', () => {
|
|
149
|
+
const account_1 = {
|
|
150
|
+
address: '0x8Ce59c2d1702844F8EdED451AA103961bC37B4e8',
|
|
151
|
+
owner_1: '4ee089aceabf3ddbf748db79b1066c33b7d3ea1ab3eb7e325121bba2bff2f5ca',
|
|
152
|
+
owner_2: '5c7e4efff7304d4dfff6d5f1591844ec6f2adfa6a47e9fece6a3c1a4d755f1e3',
|
|
153
|
+
owner_3: '4421ab25dd91e1a3180d03d57c323a7886dcc313d3b3a4b4256a5791572bf597',
|
|
154
|
+
};
|
|
155
|
+
const account_2 = {
|
|
156
|
+
address: '0xeeaf0F05f37891ab4a21208B105A0687d12c5aF7',
|
|
157
|
+
owner_1: '4ee089aceabf3ddbf748db79b1066c33b7d3ea1ab3eb7e325121bba2bff2f5ca',
|
|
158
|
+
owner_2: '5ca116d25aec5f765465432cc421ff25ef9ffdc330b10bb3d9ad61e3baad88d7',
|
|
159
|
+
owner_3: '1fae946cc84af8bd74d610a88537e24e19c3349d478d86fc5bb59ba4c88fb9cc',
|
|
160
|
+
};
|
|
161
|
+
it('should sign an unsigned test tx', async function () {
|
|
162
|
+
const builder = (0, getBuilder_1.getBuilder)('tpolygon');
|
|
163
|
+
builder.fee({
|
|
164
|
+
fee: '280000000000',
|
|
165
|
+
gasLimit: '7000000',
|
|
166
|
+
});
|
|
167
|
+
builder.counter(1);
|
|
168
|
+
builder.type(sdk_core_1.TransactionType.Send);
|
|
169
|
+
builder.contract(account_1.address);
|
|
170
|
+
builder.transfer().amount('1').to(account_2.address).expirationTime(10000).contractSequenceId(1);
|
|
171
|
+
const unsignedTx = await builder.build();
|
|
172
|
+
const unsignedTxForBroadcasting = unsignedTx.toBroadcastFormat();
|
|
173
|
+
const halfSignedRawTx = await basecoin.signTransaction({
|
|
174
|
+
txPrebuild: {
|
|
175
|
+
txHex: unsignedTxForBroadcasting,
|
|
176
|
+
},
|
|
177
|
+
prv: account_1.owner_2,
|
|
178
|
+
});
|
|
179
|
+
builder.transfer().key(account_1.owner_2);
|
|
180
|
+
const halfSignedTx = await builder.build();
|
|
181
|
+
const halfSignedTxForBroadcasting = halfSignedTx.toBroadcastFormat();
|
|
182
|
+
halfSignedRawTx.halfSigned.txHex.should.equals(halfSignedTxForBroadcasting);
|
|
183
|
+
halfSignedRawTx.halfSigned.recipients.length.should.equals(1);
|
|
184
|
+
halfSignedRawTx.halfSigned.recipients[0].address.toLowerCase().should.equals(account_2.address.toLowerCase());
|
|
185
|
+
halfSignedRawTx.halfSigned.recipients[0].amount.toLowerCase().should.equals('1');
|
|
186
|
+
});
|
|
187
|
+
it('should sign an unsigned test tx with eip1559', async function () {
|
|
188
|
+
const builder = (0, getBuilder_1.getBuilder)('tpolygon');
|
|
189
|
+
builder.fee({
|
|
190
|
+
fee: '280000000000',
|
|
191
|
+
gasLimit: '7000000',
|
|
192
|
+
eip1559: {
|
|
193
|
+
maxFeePerGas: '7593123',
|
|
194
|
+
maxPriorityFeePerGas: '150',
|
|
195
|
+
},
|
|
196
|
+
});
|
|
197
|
+
builder.counter(1);
|
|
198
|
+
builder.type(sdk_core_1.TransactionType.Send);
|
|
199
|
+
builder.contract(account_1.address);
|
|
200
|
+
builder.transfer().amount('1').to(account_2.address).expirationTime(10000).contractSequenceId(1);
|
|
201
|
+
const unsignedTx = await builder.build();
|
|
202
|
+
const unsignedTxForBroadcasting = unsignedTx.toBroadcastFormat();
|
|
203
|
+
const halfSignedRawTx = await basecoin.signTransaction({
|
|
204
|
+
txPrebuild: {
|
|
205
|
+
txHex: unsignedTxForBroadcasting,
|
|
206
|
+
eip1559: {
|
|
207
|
+
maxFeePerGas: '7593123',
|
|
208
|
+
maxPriorityFeePerGas: '150',
|
|
209
|
+
},
|
|
210
|
+
},
|
|
211
|
+
prv: account_1.owner_2,
|
|
212
|
+
});
|
|
213
|
+
builder.transfer().key(account_1.owner_2);
|
|
214
|
+
const halfSignedTx = await builder.build();
|
|
215
|
+
const halfSignedTxForBroadcasting = halfSignedTx.toBroadcastFormat();
|
|
216
|
+
halfSignedRawTx.halfSigned.txHex.should.equals(halfSignedTxForBroadcasting);
|
|
217
|
+
halfSignedRawTx.halfSigned.recipients.length.should.equals(1);
|
|
218
|
+
halfSignedRawTx.halfSigned.recipients[0].address.toLowerCase().should.equals(account_2.address.toLowerCase());
|
|
219
|
+
halfSignedRawTx.halfSigned.recipients[0].amount.toLowerCase().should.equals('1');
|
|
220
|
+
halfSignedRawTx.halfSigned.eip1559.maxFeePerGas.should.equal('7593123');
|
|
221
|
+
halfSignedRawTx.halfSigned.eip1559.maxPriorityFeePerGas.should.equal('150');
|
|
222
|
+
});
|
|
223
|
+
it('should include isBatch field in halfSigned when signing a native batch transaction', async function () {
|
|
224
|
+
const batcherContractAddress = '0xb1b7e7cc1ecafbfd0771a5eb5454ab5b0356980d';
|
|
225
|
+
const recipients = [
|
|
226
|
+
{
|
|
227
|
+
address: account_2.address,
|
|
228
|
+
amount: '500000000000',
|
|
229
|
+
},
|
|
230
|
+
{
|
|
231
|
+
address: '0x7e85bdc27c050e3905ebf4b8e634d9ad6edd0de6',
|
|
232
|
+
amount: '500000000000',
|
|
233
|
+
},
|
|
234
|
+
];
|
|
235
|
+
const totalAmount = recipients.reduce((sum, recipient) => new bignumber_js_1.default(sum).plus(recipient.amount).toString(), '0');
|
|
236
|
+
const BATCH_METHOD_NAME = 'batch';
|
|
237
|
+
const BATCH_METHOD_TYPES = ['address[]', 'uint256[]'];
|
|
238
|
+
const addresses = recipients.map((r) => r.address);
|
|
239
|
+
const amounts = recipients.map((r) => r.amount);
|
|
240
|
+
const batchExecutionInfo = {
|
|
241
|
+
values: [addresses, amounts],
|
|
242
|
+
totalAmount: totalAmount,
|
|
243
|
+
};
|
|
244
|
+
const ethAbi = require('ethereumjs-abi');
|
|
245
|
+
const ethUtil = require('ethereumjs-util');
|
|
246
|
+
const getMethodCallData = (functionName, types, values) => {
|
|
247
|
+
return Buffer.concat([
|
|
248
|
+
// function signature
|
|
249
|
+
ethAbi.methodID(functionName, types),
|
|
250
|
+
// function arguments
|
|
251
|
+
ethAbi.rawEncode(types, values),
|
|
252
|
+
]);
|
|
253
|
+
};
|
|
254
|
+
const batchData = ethUtil.addHexPrefix(getMethodCallData(BATCH_METHOD_NAME, BATCH_METHOD_TYPES, batchExecutionInfo.values).toString('hex'));
|
|
255
|
+
const txBuilder = (0, getBuilder_1.getBuilder)('tpolygon');
|
|
256
|
+
txBuilder.fee({
|
|
257
|
+
fee: '280000000000',
|
|
258
|
+
gasLimit: '7000000',
|
|
259
|
+
});
|
|
260
|
+
txBuilder.counter(1);
|
|
261
|
+
txBuilder.type(sdk_core_1.TransactionType.Send);
|
|
262
|
+
txBuilder.contract(account_1.address);
|
|
263
|
+
txBuilder
|
|
264
|
+
.transfer()
|
|
265
|
+
.amount(totalAmount)
|
|
266
|
+
.to(batcherContractAddress)
|
|
267
|
+
.data(batchData)
|
|
268
|
+
.expirationTime(10000)
|
|
269
|
+
.contractSequenceId(1);
|
|
270
|
+
const unsignedTx = await txBuilder.build();
|
|
271
|
+
const unsignedTxHex = unsignedTx.toBroadcastFormat();
|
|
272
|
+
const txPrebuild = {
|
|
273
|
+
txHex: unsignedTxHex,
|
|
274
|
+
isBatch: true,
|
|
275
|
+
recipients: [
|
|
276
|
+
{
|
|
277
|
+
amount: totalAmount,
|
|
278
|
+
address: batcherContractAddress,
|
|
279
|
+
},
|
|
280
|
+
],
|
|
281
|
+
nextContractSequenceId: 1,
|
|
282
|
+
gasPrice: 280000000000,
|
|
283
|
+
gasLimit: 7000000,
|
|
284
|
+
coin: 'tpolygon',
|
|
285
|
+
walletId: 'fakeWalletId',
|
|
286
|
+
walletContractAddress: account_1.address,
|
|
287
|
+
};
|
|
288
|
+
const signedTx = await basecoin.signTransaction({
|
|
289
|
+
txPrebuild,
|
|
290
|
+
prv: account_1.owner_2,
|
|
291
|
+
});
|
|
292
|
+
should.exist(signedTx);
|
|
293
|
+
should.exist(signedTx.halfSigned);
|
|
294
|
+
signedTx.halfSigned.should.have.property('isBatch', true);
|
|
295
|
+
});
|
|
296
|
+
});
|
|
297
|
+
describe('Transaction Verification', function () {
|
|
298
|
+
it('should verify a normal txPrebuild from the bitgo server that matches the client txParams', async function () {
|
|
299
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
300
|
+
const txParams = {
|
|
301
|
+
recipients: [{ amount: '1000000000000', address: address1 }],
|
|
302
|
+
wallet: wallet,
|
|
303
|
+
walletPassphrase: 'fakeWalletPassphrase',
|
|
304
|
+
};
|
|
305
|
+
const txPrebuild = {
|
|
306
|
+
recipients: [{ amount: '1000000000000', address: address1 }],
|
|
307
|
+
nextContractSequenceId: 0,
|
|
308
|
+
gasPrice: 20000000000,
|
|
309
|
+
gasLimit: 500000,
|
|
310
|
+
isBatch: false,
|
|
311
|
+
coin: 'tpolygon',
|
|
312
|
+
walletId: 'fakeWalletId',
|
|
313
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
314
|
+
};
|
|
315
|
+
const verification = {};
|
|
316
|
+
const isTransactionVerified = await basecoin.verifyTransaction({ txParams, txPrebuild, wallet, verification });
|
|
317
|
+
isTransactionVerified.should.equal(true);
|
|
318
|
+
});
|
|
319
|
+
it('should verify a hop txPrebuild from the bitgo server that matches the client txParams', async function () {
|
|
320
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
321
|
+
const txParams = {
|
|
322
|
+
recipients: [{ amount: 1000000000000000, address: hopDestinationAddress }],
|
|
323
|
+
wallet: wallet,
|
|
324
|
+
walletPassphrase: 'fakeWalletPassphrase',
|
|
325
|
+
hop: true,
|
|
326
|
+
};
|
|
327
|
+
const txPrebuild = {
|
|
328
|
+
recipients: [{ amount: '5000000000000000', address: hopContractAddress }],
|
|
329
|
+
nextContractSequenceId: 0,
|
|
330
|
+
gasPrice: 20000000000,
|
|
331
|
+
gasLimit: 500000,
|
|
332
|
+
isBatch: false,
|
|
333
|
+
coin: 'tpolygon',
|
|
334
|
+
walletId: 'fakeWalletId',
|
|
335
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
336
|
+
hopTransaction: {
|
|
337
|
+
tx: hopTx,
|
|
338
|
+
id: hopTxid,
|
|
339
|
+
signature: hopTxBitgoSignature,
|
|
340
|
+
paymentId: '2773928196',
|
|
341
|
+
gasPrice: 20000000000,
|
|
342
|
+
gasLimit: 500000,
|
|
343
|
+
amount: '1000000000000000',
|
|
344
|
+
recipient: hopDestinationAddress,
|
|
345
|
+
nonce: 0,
|
|
346
|
+
userReqSig: userReqSig,
|
|
347
|
+
gasPriceMax: 500000000000,
|
|
348
|
+
},
|
|
349
|
+
};
|
|
350
|
+
const verification = {};
|
|
351
|
+
const isTransactionVerified = await basecoin.verifyTransaction({ txParams, txPrebuild, wallet, verification });
|
|
352
|
+
isTransactionVerified.should.equal(true);
|
|
353
|
+
});
|
|
354
|
+
it('should reject when client txParams are missing', async function () {
|
|
355
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
356
|
+
const txParams = null;
|
|
357
|
+
const txPrebuild = {
|
|
358
|
+
recipients: [{ amount: '1000000000000', address: address1 }],
|
|
359
|
+
nextContractSequenceId: 0,
|
|
360
|
+
gasPrice: 20000000000,
|
|
361
|
+
gasLimit: 500000,
|
|
362
|
+
isBatch: false,
|
|
363
|
+
coin: 'tpolygon',
|
|
364
|
+
walletId: 'fakeWalletId',
|
|
365
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
366
|
+
};
|
|
367
|
+
const verification = {};
|
|
368
|
+
await basecoin
|
|
369
|
+
.verifyTransaction({ txParams, txPrebuild, wallet, verification })
|
|
370
|
+
.should.be.rejectedWith('missing params');
|
|
371
|
+
});
|
|
372
|
+
it('should reject txPrebuild that is both batch and hop', async function () {
|
|
373
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
374
|
+
const txParams = {
|
|
375
|
+
recipients: [
|
|
376
|
+
{ amount: '1000000000000', address: address1 },
|
|
377
|
+
{ amount: '2500000000000', address: address2 },
|
|
378
|
+
],
|
|
379
|
+
wallet: wallet,
|
|
380
|
+
walletPassphrase: 'fakeWalletPassphrase',
|
|
381
|
+
hop: true,
|
|
382
|
+
};
|
|
383
|
+
const txPrebuild = {
|
|
384
|
+
recipients: [{ amount: '3500000000000', address: address1 }],
|
|
385
|
+
nextContractSequenceId: 0,
|
|
386
|
+
gasPrice: 20000000000,
|
|
387
|
+
gasLimit: 500000,
|
|
388
|
+
isBatch: true,
|
|
389
|
+
coin: 'tpolygon',
|
|
390
|
+
walletId: 'fakeWalletId',
|
|
391
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
392
|
+
hopTransaction: {
|
|
393
|
+
tx: hopTx,
|
|
394
|
+
id: hopTxid,
|
|
395
|
+
signature: hopTxBitgoSignature,
|
|
396
|
+
paymentId: '2773928196',
|
|
397
|
+
gasPrice: 20000000000,
|
|
398
|
+
gasLimit: 500000,
|
|
399
|
+
amount: '1000000000000000',
|
|
400
|
+
recipient: hopDestinationAddress,
|
|
401
|
+
nonce: 0,
|
|
402
|
+
userReqSig: userReqSig,
|
|
403
|
+
gasPriceMax: 500000000000,
|
|
404
|
+
},
|
|
405
|
+
};
|
|
406
|
+
const verification = {};
|
|
407
|
+
await basecoin
|
|
408
|
+
.verifyTransaction({ txParams, txPrebuild, wallet, verification })
|
|
409
|
+
.should.be.rejectedWith('tx cannot be both a batch and hop transaction');
|
|
410
|
+
});
|
|
411
|
+
it('should reject a txPrebuild with more than one recipient', async function () {
|
|
412
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
413
|
+
const txParams = {
|
|
414
|
+
recipients: [
|
|
415
|
+
{ amount: '1000000000000', address: address1 },
|
|
416
|
+
{ amount: '2500000000000', address: address2 },
|
|
417
|
+
],
|
|
418
|
+
wallet: wallet,
|
|
419
|
+
walletPassphrase: 'fakeWalletPassphrase',
|
|
420
|
+
};
|
|
421
|
+
const txPrebuild = {
|
|
422
|
+
recipients: [
|
|
423
|
+
{ amount: '1000000000000', address: address1 },
|
|
424
|
+
{ amount: '2500000000000', address: address2 },
|
|
425
|
+
],
|
|
426
|
+
nextContractSequenceId: 0,
|
|
427
|
+
gasPrice: 20000000000,
|
|
428
|
+
gasLimit: 500000,
|
|
429
|
+
isBatch: true,
|
|
430
|
+
coin: 'tpolygon',
|
|
431
|
+
walletId: 'fakeWalletId',
|
|
432
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
433
|
+
};
|
|
434
|
+
const verification = {};
|
|
435
|
+
await basecoin
|
|
436
|
+
.verifyTransaction({ txParams, txPrebuild, wallet, verification })
|
|
437
|
+
.should.be.rejectedWith(`tpolygon doesn't support sending to more than 1 destination address within a single transaction. Try again, using only a single recipient.`);
|
|
438
|
+
});
|
|
439
|
+
it('should reject a hop txPrebuild that does not send to its hop address', async function () {
|
|
440
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
441
|
+
const txParams = {
|
|
442
|
+
recipients: [{ amount: '1000000000000000', address: hopDestinationAddress }],
|
|
443
|
+
wallet: wallet,
|
|
444
|
+
walletPassphrase: 'fakeWalletPassphrase',
|
|
445
|
+
hop: true,
|
|
446
|
+
};
|
|
447
|
+
const txPrebuild = {
|
|
448
|
+
recipients: [{ amount: '5000000000000000', address: address1 }],
|
|
449
|
+
nextContractSequenceId: 0,
|
|
450
|
+
gasPrice: 20000000000,
|
|
451
|
+
gasLimit: 500000,
|
|
452
|
+
isBatch: false,
|
|
453
|
+
coin: 'tpolygon',
|
|
454
|
+
walletId: 'fakeWalletId',
|
|
455
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
456
|
+
hopTransaction: {
|
|
457
|
+
tx: hopTx,
|
|
458
|
+
id: hopTxid,
|
|
459
|
+
signature: hopTxBitgoSignature,
|
|
460
|
+
paymentId: '0',
|
|
461
|
+
gasPrice: 20000000000,
|
|
462
|
+
gasLimit: 500000,
|
|
463
|
+
amount: '1000000000000000',
|
|
464
|
+
recipient: hopDestinationAddress,
|
|
465
|
+
nonce: 0,
|
|
466
|
+
userReqSig: userReqSig,
|
|
467
|
+
gasPriceMax: 500000000000,
|
|
468
|
+
},
|
|
469
|
+
};
|
|
470
|
+
const verification = {};
|
|
471
|
+
await basecoin
|
|
472
|
+
.verifyTransaction({ txParams, txPrebuild, wallet, verification })
|
|
473
|
+
.should.be.rejectedWith('recipient address of txPrebuild does not match hop address');
|
|
474
|
+
});
|
|
475
|
+
it('should reject a normal txPrebuild from the bitgo server with the wrong amount', async function () {
|
|
476
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
477
|
+
const txParams = {
|
|
478
|
+
recipients: [{ amount: '1000000000000', address: address1 }],
|
|
479
|
+
wallet: wallet,
|
|
480
|
+
walletPassphrase: 'fakeWalletPassphrase',
|
|
481
|
+
};
|
|
482
|
+
const txPrebuild = {
|
|
483
|
+
recipients: [{ amount: '2000000000000', address: address1 }],
|
|
484
|
+
nextContractSequenceId: 0,
|
|
485
|
+
gasPrice: 20000000000,
|
|
486
|
+
gasLimit: 500000,
|
|
487
|
+
isBatch: false,
|
|
488
|
+
coin: 'tpolygon',
|
|
489
|
+
walletId: 'fakeWalletId',
|
|
490
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
491
|
+
};
|
|
492
|
+
const verification = {};
|
|
493
|
+
await basecoin
|
|
494
|
+
.verifyTransaction({ txParams, txPrebuild, wallet, verification })
|
|
495
|
+
.should.be.rejectedWith('normal transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client');
|
|
496
|
+
});
|
|
497
|
+
it('should reject a normal txPrebuild from the bitgo server with the wrong recipient', async function () {
|
|
498
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
499
|
+
const txParams = {
|
|
500
|
+
recipients: [{ amount: '1000000000000', address: address1 }],
|
|
501
|
+
wallet: wallet,
|
|
502
|
+
walletPassphrase: 'fakeWalletPassphrase',
|
|
503
|
+
};
|
|
504
|
+
const txPrebuild = {
|
|
505
|
+
recipients: [{ amount: '1000000000000', address: address2 }],
|
|
506
|
+
nextContractSequenceId: 0,
|
|
507
|
+
gasPrice: 20000000000,
|
|
508
|
+
gasLimit: 500000,
|
|
509
|
+
isBatch: false,
|
|
510
|
+
coin: 'tpolygon',
|
|
511
|
+
walletId: 'fakeWalletId',
|
|
512
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
513
|
+
};
|
|
514
|
+
const verification = {};
|
|
515
|
+
await basecoin
|
|
516
|
+
.verifyTransaction({ txParams, txPrebuild, wallet, verification })
|
|
517
|
+
.should.be.rejectedWith('destination address in normal txPrebuild does not match that in txParams supplied by client');
|
|
518
|
+
});
|
|
519
|
+
it('should reject a txPrebuild from the bitgo server with the wrong coin', async function () {
|
|
520
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
521
|
+
const txParams = {
|
|
522
|
+
recipients: [{ amount: '1000000000000', address: address1 }],
|
|
523
|
+
wallet: wallet,
|
|
524
|
+
walletPassphrase: 'fakeWalletPassphrase',
|
|
525
|
+
};
|
|
526
|
+
const txPrebuild = {
|
|
527
|
+
recipients: [{ amount: '1000000000000', address: address1 }],
|
|
528
|
+
nextContractSequenceId: 0,
|
|
529
|
+
gasPrice: 20000000000,
|
|
530
|
+
gasLimit: 500000,
|
|
531
|
+
isBatch: false,
|
|
532
|
+
coin: 'btc',
|
|
533
|
+
walletId: 'fakeWalletId',
|
|
534
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
535
|
+
};
|
|
536
|
+
const verification = {};
|
|
537
|
+
await basecoin
|
|
538
|
+
.verifyTransaction({ txParams, txPrebuild, wallet, verification })
|
|
539
|
+
.should.be.rejectedWith('coin in txPrebuild did not match that in txParams supplied by client');
|
|
540
|
+
});
|
|
541
|
+
});
|
|
542
|
+
describe('Recover transaction:', function () {
|
|
543
|
+
const baseUrl = sdk_core_1.common.Environments.test.polygonscanBaseUrl;
|
|
544
|
+
const userXpub = 'xpub661MyMwAqRbcEeTc8789MK5PUGEYiPG4F4V17n2Rd2LoTATA1XoCnJT5FAYAShQxSxtFjpo5NHmcWwTp2LiWGBMwpUcAA3HywhxivgYfq7q';
|
|
545
|
+
const userXprv = 'xprv9s21ZrQH143K2AP925b8zB8evEQ4JvYCsqZQKPcp4gopaN81TzUxEW8bPtVyDgjmddGhRRETn8xi1cVAB9bf1Bx9kGRRFgTZXxJayZLnag1';
|
|
546
|
+
const backupXpub = 'xpub661MyMwAqRbcFZX15xpZf4ERCGHiVSJm8r5C4yh1yXV2GrdZCUPYo4WQr6tN9oUywKXsgSHo7Risf9r22GH5joVD2hEEEhqnSCvK8qy11wW';
|
|
547
|
+
const backupXprv = 'xprv9s21ZrQH143K35SXywHZHvHgeETE5yaumd9bGbHQRBx3Q4JQew5JFGBvzqiZjCUkBdBUZnfuMDTGURRayN1hFSWxEJQsCEAMm1D3pk1h7Jj';
|
|
548
|
+
it('should generate an unsigned sweep', async function () {
|
|
549
|
+
const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
|
|
550
|
+
const backupKeyAddress = '0x4f2c4830cc37f2785c646f89ded8a919219fa0e9';
|
|
551
|
+
(0, nock_1.default)(baseUrl)
|
|
552
|
+
.get('/api')
|
|
553
|
+
.twice()
|
|
554
|
+
.query(mockData.getTxListRequest(backupKeyAddress))
|
|
555
|
+
.reply(200, mockData.getTxListResponse);
|
|
556
|
+
(0, nock_1.default)(baseUrl)
|
|
557
|
+
.get('/api')
|
|
558
|
+
.query(mockData.getBalanceRequest(walletContractAddress))
|
|
559
|
+
.reply(200, mockData.getBalanceResponse);
|
|
560
|
+
(0, nock_1.default)(baseUrl)
|
|
561
|
+
.get('/api')
|
|
562
|
+
.query(mockData.getBalanceRequest(backupKeyAddress))
|
|
563
|
+
.reply(200, mockData.getBalanceResponse);
|
|
564
|
+
(0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
|
|
565
|
+
const basecoin = bitgo.coin('tpolygon');
|
|
566
|
+
const transaction = (await basecoin.recover({
|
|
567
|
+
userKey: userXpub,
|
|
568
|
+
backupKey: backupXpub,
|
|
569
|
+
walletContractAddress: walletContractAddress,
|
|
570
|
+
recoveryDestination: sdk_test_1.TestBitGo.V2.TEST_ERC20_TOKEN_RECIPIENT,
|
|
571
|
+
eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
|
|
572
|
+
gasLimit: 500000,
|
|
573
|
+
}));
|
|
574
|
+
should.exist(transaction);
|
|
575
|
+
transaction.should.have.property('txHex');
|
|
576
|
+
transaction.should.have.property('contractSequenceId');
|
|
577
|
+
transaction.should.have.property('expireTime');
|
|
578
|
+
transaction.should.have.property('gasLimit');
|
|
579
|
+
transaction.gasLimit.should.equal('500000');
|
|
580
|
+
transaction.should.have.property('walletContractAddress');
|
|
581
|
+
transaction.walletContractAddress.should.equal(sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);
|
|
582
|
+
transaction.should.have.property('recipients');
|
|
583
|
+
const recipient = transaction.recipients[0];
|
|
584
|
+
recipient.should.have.property('address');
|
|
585
|
+
recipient.address.should.equal(sdk_test_1.TestBitGo.V2.TEST_ERC20_TOKEN_RECIPIENT);
|
|
586
|
+
recipient.should.have.property('amount');
|
|
587
|
+
recipient.amount.should.equal('9999999999999999928');
|
|
588
|
+
});
|
|
589
|
+
it('should construct a recovery transaction without BitGo', async function () {
|
|
590
|
+
const backupKeyAddress = '0x6d22efdd634996248170c948e5726007fc251bb3';
|
|
591
|
+
const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
|
|
592
|
+
(0, nock_1.default)(baseUrl)
|
|
593
|
+
.get('/api')
|
|
594
|
+
.query(mockData.getTxListRequest(backupKeyAddress))
|
|
595
|
+
.reply(200, mockData.getTxListResponse);
|
|
596
|
+
(0, nock_1.default)(baseUrl)
|
|
597
|
+
.get('/api')
|
|
598
|
+
.query(mockData.getBalanceRequest(walletContractAddress))
|
|
599
|
+
.reply(200, mockData.getBalanceResponse);
|
|
600
|
+
(0, nock_1.default)(baseUrl)
|
|
601
|
+
.get('/api')
|
|
602
|
+
.query(mockData.getBalanceRequest(backupKeyAddress))
|
|
603
|
+
.reply(200, mockData.getBalanceResponse);
|
|
604
|
+
(0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
|
|
605
|
+
const basecoin = bitgo.coin('tpolygon');
|
|
606
|
+
const transaction = (await basecoin.recover({
|
|
607
|
+
userKey: '{"iv":"VFZ3jvXhxo1Z+Yaf2MtZnA==","v":1,"iter":10000,"ks":256,"ts":64,"mode"\n' +
|
|
608
|
+
':"ccm","adata":"","cipher":"aes","salt":"p+fkHuLa/8k=","ct":"hYG7pvljLIgCjZ\n' +
|
|
609
|
+
'53PBlCde5KZRmlUKKHLtDMk+HJfuU46hW+x+C9WsIAO4gFPnTCvFVmQ8x7czCtcNFub5AO2otOG\n' +
|
|
610
|
+
'OsX4GE2gXOEmCl1TpWwwNhm7yMUjGJUpgW6ZZgXSXdDitSKi4V/hk78SGSzjFOBSPYRa6I="}\n',
|
|
611
|
+
backupKey: '{"iv":"AbsCtv1qwPIhOgyrCpNagA==","v":1,"iter":10000,"ks":256,"ts":64,"mode"\n' +
|
|
612
|
+
':"ccm","adata":"","cipher":"aes","salt":"5vpUDBUlzm8=","ct":"PapYYCjBXRLUKA\n' +
|
|
613
|
+
'JbOsB/EJ9B8fUmVQTxMPjUnQyAky12me9K66GiMEAxTD7kd6bYAQJuuTkATXKU7Bnf7vK9JxNOw\n' +
|
|
614
|
+
'oji7HF9eFH0aD4/hX5SWFfHF2Qfi+TnXv6hVsMAoisDZs3/F67/ZUaDYR0ZsdrQ4Q/cLD0="}\n',
|
|
615
|
+
walletContractAddress: walletContractAddress,
|
|
616
|
+
walletPassphrase: sdk_test_1.TestBitGo.V2.TEST_RECOVERY_PASSCODE,
|
|
617
|
+
recoveryDestination: sdk_test_1.TestBitGo.V2.TEST_ERC20_TOKEN_RECIPIENT,
|
|
618
|
+
gasLimit: 500000,
|
|
619
|
+
}));
|
|
620
|
+
should.exist(transaction);
|
|
621
|
+
transaction.should.have.property('tx');
|
|
622
|
+
transaction.should.have.property('id');
|
|
623
|
+
const decodedTx = abstract_eth_1.optionalDeps.EthTx.Transaction.fromSerializedTx(abstract_eth_1.optionalDeps.ethUtil.toBuffer(transaction.tx));
|
|
624
|
+
decodedTx.should.have.property('gasPrice');
|
|
625
|
+
decodedTx.should.have.property('nonce');
|
|
626
|
+
decodedTx.should.have.property('to');
|
|
627
|
+
});
|
|
628
|
+
xit('should construct a recovery tx with TSS', async function () {
|
|
629
|
+
const backupKeyAddress = '0xe7406dc43d13f698fb41a345c7783d39a4c2d191';
|
|
630
|
+
(0, nock_1.default)(baseUrl)
|
|
631
|
+
.get('/api')
|
|
632
|
+
.query(mockData.getTxListRequest(backupKeyAddress))
|
|
633
|
+
.reply(200, mockData.getTxListResponse);
|
|
634
|
+
(0, nock_1.default)(baseUrl)
|
|
635
|
+
.get('/api')
|
|
636
|
+
.query(mockData.getBalanceRequest(backupKeyAddress))
|
|
637
|
+
.reply(200, mockData.getBalanceResponse);
|
|
638
|
+
const basecoin = bitgo.coin('tpolygon');
|
|
639
|
+
const userKey = mockData.keyShares.userKeyShare;
|
|
640
|
+
const backupKey = mockData.keyShares.backupKeyShare;
|
|
641
|
+
const bitgoKey = mockData.keyShares.bitgoKeyShare;
|
|
642
|
+
const userSigningMaterial = {
|
|
643
|
+
pShare: userKey.pShare,
|
|
644
|
+
backupNShare: backupKey.nShares[1],
|
|
645
|
+
bitgoNShare: bitgoKey.nShares[1],
|
|
646
|
+
};
|
|
647
|
+
const backupSigningMaterial = {
|
|
648
|
+
pShare: backupKey.pShare,
|
|
649
|
+
userNShare: userKey.nShares[2],
|
|
650
|
+
bitgoNShare: bitgoKey.nShares[2],
|
|
651
|
+
};
|
|
652
|
+
const encryptedBackupSigningMaterial = sjcl.encrypt(sdk_test_1.TestBitGo.V2.TEST_RECOVERY_PASSCODE, JSON.stringify(backupSigningMaterial));
|
|
653
|
+
const encryptedUserSigningMaterial = sjcl.encrypt(sdk_test_1.TestBitGo.V2.TEST_RECOVERY_PASSCODE, JSON.stringify(userSigningMaterial));
|
|
654
|
+
const recoveryParams = {
|
|
655
|
+
userKey: encryptedUserSigningMaterial,
|
|
656
|
+
backupKey: encryptedBackupSigningMaterial,
|
|
657
|
+
walletContractAddress: '0xe7406dc43d13f698fb41a345c7783d39a4c2d191',
|
|
658
|
+
recoveryDestination: '0xac05da78464520aa7c9d4c19bd7a440b111b3054',
|
|
659
|
+
walletPassphrase: sdk_test_1.TestBitGo.V2.TEST_RECOVERY_PASSCODE,
|
|
660
|
+
isTss: true,
|
|
661
|
+
};
|
|
662
|
+
const recovery = await basecoin.recover(recoveryParams);
|
|
663
|
+
// id and tx will always be different because of expireTime
|
|
664
|
+
should.exist(recovery);
|
|
665
|
+
recovery.should.have.property('id');
|
|
666
|
+
recovery.should.have.property('tx');
|
|
667
|
+
});
|
|
668
|
+
it('should be able to second sign', async function () {
|
|
669
|
+
const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
|
|
670
|
+
const backupKeyAddress = '0x4f2c4830cc37f2785c646f89ded8a919219fa0e9';
|
|
671
|
+
(0, nock_1.default)(baseUrl)
|
|
672
|
+
.get('/api')
|
|
673
|
+
.twice()
|
|
674
|
+
.query(mockData.getTxListRequest(backupKeyAddress))
|
|
675
|
+
.reply(200, mockData.getTxListResponse);
|
|
676
|
+
(0, nock_1.default)(baseUrl)
|
|
677
|
+
.get('/api')
|
|
678
|
+
.query(mockData.getBalanceRequest(walletContractAddress))
|
|
679
|
+
.reply(200, mockData.getBalanceResponse);
|
|
680
|
+
(0, nock_1.default)(baseUrl)
|
|
681
|
+
.get('/api')
|
|
682
|
+
.query(mockData.getBalanceRequest(backupKeyAddress))
|
|
683
|
+
.reply(200, mockData.getBalanceResponse);
|
|
684
|
+
(0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
|
|
685
|
+
const basecoin = bitgo.coin('tpolygon');
|
|
686
|
+
const transaction = (await basecoin.recover({
|
|
687
|
+
userKey: userXpub,
|
|
688
|
+
backupKey: backupXpub,
|
|
689
|
+
walletContractAddress: walletContractAddress,
|
|
690
|
+
recoveryDestination: sdk_test_1.TestBitGo.V2.TEST_ERC20_TOKEN_RECIPIENT,
|
|
691
|
+
eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
|
|
692
|
+
replayProtectionOptions: { chain: 80002, hardfork: 'london' },
|
|
693
|
+
gasLimit: 500000,
|
|
694
|
+
}));
|
|
695
|
+
const txPrebuild = {
|
|
696
|
+
txHex: transaction.txHex,
|
|
697
|
+
};
|
|
698
|
+
const params = {
|
|
699
|
+
txPrebuild,
|
|
700
|
+
prv: userXprv,
|
|
701
|
+
};
|
|
702
|
+
// sign transaction once
|
|
703
|
+
const halfSigned = await basecoin.signTransaction(params);
|
|
704
|
+
const halfSignedParams = {
|
|
705
|
+
txPrebuild: halfSigned,
|
|
706
|
+
isLastSignature: true,
|
|
707
|
+
walletContractAddress: walletContractAddress,
|
|
708
|
+
prv: backupXprv,
|
|
709
|
+
};
|
|
710
|
+
const finalSigned = (await basecoin.signTransaction(halfSignedParams));
|
|
711
|
+
finalSigned.should.have.property('txHex');
|
|
712
|
+
const txBuilder = (0, getBuilder_1.getBuilder)('tpolygon');
|
|
713
|
+
txBuilder.from(finalSigned.txHex);
|
|
714
|
+
const rebuiltTx = await txBuilder.build();
|
|
715
|
+
rebuiltTx.signature.length.should.equal(2);
|
|
716
|
+
rebuiltTx.outputs.length.should.equal(1);
|
|
717
|
+
});
|
|
718
|
+
});
|
|
719
|
+
describe('Evm Based Cross Chain Recovery transaction:', function () {
|
|
720
|
+
const baseUrl = sdk_core_1.common.Environments.test.polygonscanBaseUrl;
|
|
721
|
+
const userXpub = 'xpub661MyMwAqRbcEeTc8789MK5PUGEYiPG4F4V17n2Rd2LoTATA1XoCnJT5FAYAShQxSxtFjpo5NHmcWwTp2LiWGBMwpUcAA3HywhxivgYfq7q';
|
|
722
|
+
const bitgoFeeAddress = '0x33a42faea3c6e87021347e51700b48aaf49aa1e7';
|
|
723
|
+
const destinationAddress = '0xd5adde17fed8baed3f32b84af05b8f2816f7b560';
|
|
724
|
+
const bitgoDestinationAddress = '0xe5986ce4490deb67d2950562ceb930ddf9be7a14';
|
|
725
|
+
it('should generate an unsigned recovery txn for cold wallet', async function () {
|
|
726
|
+
const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
|
|
727
|
+
const basecoin = bitgo.coin('tpolygon');
|
|
728
|
+
(0, nock_1.default)(baseUrl)
|
|
729
|
+
.get('/api')
|
|
730
|
+
.query(mockData.getTxListRequest(bitgoFeeAddress))
|
|
731
|
+
.reply(200, mockData.getTxListResponse);
|
|
732
|
+
(0, nock_1.default)(baseUrl)
|
|
733
|
+
.get('/api')
|
|
734
|
+
.query(mockData.getBalanceRequest(bitgoFeeAddress))
|
|
735
|
+
.reply(200, mockData.getBalanceResponse);
|
|
736
|
+
(0, nock_1.default)(baseUrl)
|
|
737
|
+
.get('/api')
|
|
738
|
+
.query(mockData.getBalanceRequest(walletContractAddress))
|
|
739
|
+
.reply(200, mockData.getBalanceResponse);
|
|
740
|
+
(0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
|
|
741
|
+
const transaction = (await basecoin.recover({
|
|
742
|
+
userKey: userXpub,
|
|
743
|
+
backupKey: '',
|
|
744
|
+
walletContractAddress: walletContractAddress,
|
|
745
|
+
bitgoFeeAddress: bitgoFeeAddress,
|
|
746
|
+
recoveryDestination: destinationAddress,
|
|
747
|
+
eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
|
|
748
|
+
gasLimit: 500000,
|
|
749
|
+
bitgoDestinationAddress: bitgoDestinationAddress,
|
|
750
|
+
}));
|
|
751
|
+
should.exist(transaction);
|
|
752
|
+
transaction.should.have.property('txHex');
|
|
753
|
+
transaction.should.have.property('userKey');
|
|
754
|
+
transaction.should.have.property('coin');
|
|
755
|
+
transaction.should.have.property('contractSequenceId');
|
|
756
|
+
transaction.should.have.property('expireTime');
|
|
757
|
+
transaction.should.have.property('gasLimit');
|
|
758
|
+
transaction.gasLimit.should.equal('500000');
|
|
759
|
+
transaction.should.have.property('isEvmBasedCrossChainRecovery');
|
|
760
|
+
transaction.isEvmBasedCrossChainRecovery?.should.equal(true);
|
|
761
|
+
transaction.should.have.property('walletContractAddress');
|
|
762
|
+
transaction.walletContractAddress.should.equal(sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);
|
|
763
|
+
transaction.should.have.property('recipients');
|
|
764
|
+
const recipient = transaction.recipients[0];
|
|
765
|
+
recipient.should.have.property('address');
|
|
766
|
+
recipient.address.should.equal(destinationAddress);
|
|
767
|
+
recipient.should.have.property('amount');
|
|
768
|
+
recipient.amount.should.equal('9999999999999999928');
|
|
769
|
+
});
|
|
770
|
+
it('should generate an unsigned recovery txn for custody wallet', async function () {
|
|
771
|
+
const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
|
|
772
|
+
const basecoin = bitgo.coin('tpolygon');
|
|
773
|
+
(0, nock_1.default)(baseUrl)
|
|
774
|
+
.get('/api')
|
|
775
|
+
.query(mockData.getTxListRequest(bitgoFeeAddress))
|
|
776
|
+
.reply(200, mockData.getTxListResponse);
|
|
777
|
+
(0, nock_1.default)(baseUrl)
|
|
778
|
+
.get('/api')
|
|
779
|
+
.query(mockData.getBalanceRequest(bitgoFeeAddress))
|
|
780
|
+
.reply(200, mockData.getBalanceResponse);
|
|
781
|
+
(0, nock_1.default)(baseUrl)
|
|
782
|
+
.get('/api')
|
|
783
|
+
.query(mockData.getBalanceRequest(walletContractAddress))
|
|
784
|
+
.reply(200, mockData.getBalanceResponse);
|
|
785
|
+
(0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
|
|
786
|
+
const transaction = (await basecoin.recover({
|
|
787
|
+
userKey: '',
|
|
788
|
+
backupKey: '',
|
|
789
|
+
walletContractAddress: walletContractAddress,
|
|
790
|
+
bitgoFeeAddress: bitgoFeeAddress,
|
|
791
|
+
recoveryDestination: destinationAddress,
|
|
792
|
+
eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
|
|
793
|
+
gasLimit: 500000,
|
|
794
|
+
bitgoDestinationAddress: bitgoDestinationAddress,
|
|
795
|
+
}));
|
|
796
|
+
should.exist(transaction);
|
|
797
|
+
transaction.should.have.property('txHex');
|
|
798
|
+
transaction.should.have.property('coin');
|
|
799
|
+
transaction.should.have.property('contractSequenceId');
|
|
800
|
+
transaction.should.have.property('expireTime');
|
|
801
|
+
transaction.should.have.property('gasLimit');
|
|
802
|
+
transaction.gasLimit.should.equal('500000');
|
|
803
|
+
transaction.should.have.property('isEvmBasedCrossChainRecovery');
|
|
804
|
+
transaction.isEvmBasedCrossChainRecovery?.should.equal(true);
|
|
805
|
+
transaction.should.have.property('walletContractAddress');
|
|
806
|
+
transaction.walletContractAddress.should.equal(sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);
|
|
807
|
+
transaction.should.have.property('recipients');
|
|
808
|
+
const recipient = transaction.recipients[0];
|
|
809
|
+
recipient.should.have.property('address');
|
|
810
|
+
recipient.address.should.equal(destinationAddress);
|
|
811
|
+
recipient.should.have.property('amount');
|
|
812
|
+
recipient.amount.should.equal('9999999999999999928');
|
|
813
|
+
});
|
|
814
|
+
it('should generate an unsigned recovery txn for hot wallet', async function () {
|
|
815
|
+
const userKey = '{"iv":"VFZ3jvXhxo1Z+Yaf2MtZnA==","v":1,"iter":10000,"ks":256,"ts":64,"mode"\n' +
|
|
816
|
+
':"ccm","adata":"","cipher":"aes","salt":"p+fkHuLa/8k=","ct":"hYG7pvljLIgCjZ\n' +
|
|
817
|
+
'53PBlCde5KZRmlUKKHLtDMk+HJfuU46hW+x+C9WsIAO4gFPnTCvFVmQ8x7czCtcNFub5AO2otOG\n' +
|
|
818
|
+
'OsX4GE2gXOEmCl1TpWwwNhm7yMUjGJUpgW6ZZgXSXdDitSKi4V/hk78SGSzjFOBSPYRa6I="}\n';
|
|
819
|
+
const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
|
|
820
|
+
const walletPassphrase = sdk_test_1.TestBitGo.V2.TEST_RECOVERY_PASSCODE;
|
|
821
|
+
const basecoin = bitgo.coin('tpolygon');
|
|
822
|
+
(0, nock_1.default)(baseUrl)
|
|
823
|
+
.get('/api')
|
|
824
|
+
.query(mockData.getTxListRequest(bitgoFeeAddress))
|
|
825
|
+
.reply(200, mockData.getTxListResponse);
|
|
826
|
+
(0, nock_1.default)(baseUrl)
|
|
827
|
+
.get('/api')
|
|
828
|
+
.query(mockData.getBalanceRequest(bitgoFeeAddress))
|
|
829
|
+
.reply(200, mockData.getBalanceResponse);
|
|
830
|
+
(0, nock_1.default)(baseUrl)
|
|
831
|
+
.get('/api')
|
|
832
|
+
.query(mockData.getBalanceRequest(walletContractAddress))
|
|
833
|
+
.reply(200, mockData.getBalanceResponse);
|
|
834
|
+
(0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
|
|
835
|
+
const transaction = (await basecoin.recover({
|
|
836
|
+
userKey: userKey,
|
|
837
|
+
backupKey: '',
|
|
838
|
+
walletPassphrase: walletPassphrase,
|
|
839
|
+
walletContractAddress: walletContractAddress,
|
|
840
|
+
bitgoFeeAddress: bitgoFeeAddress,
|
|
841
|
+
recoveryDestination: destinationAddress,
|
|
842
|
+
eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
|
|
843
|
+
gasLimit: 500000,
|
|
844
|
+
bitgoDestinationAddress: bitgoDestinationAddress,
|
|
845
|
+
}));
|
|
846
|
+
should.exist(transaction);
|
|
847
|
+
transaction.should.have.property('txHex');
|
|
848
|
+
transaction.should.have.property('coin');
|
|
849
|
+
transaction.should.have.property('contractSequenceId');
|
|
850
|
+
transaction.should.have.property('expireTime');
|
|
851
|
+
transaction.should.have.property('gasLimit');
|
|
852
|
+
transaction.gasLimit.should.equal('500000');
|
|
853
|
+
transaction.should.have.property('isEvmBasedCrossChainRecovery');
|
|
854
|
+
transaction.isEvmBasedCrossChainRecovery?.should.equal(true);
|
|
855
|
+
transaction.should.have.property('walletContractAddress');
|
|
856
|
+
transaction.walletContractAddress.should.equal(sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);
|
|
857
|
+
transaction.should.have.property('recipients');
|
|
858
|
+
const recipient = transaction.recipients[0];
|
|
859
|
+
recipient.should.have.property('address');
|
|
860
|
+
recipient.address.should.equal(destinationAddress);
|
|
861
|
+
recipient.should.have.property('amount');
|
|
862
|
+
recipient.amount.should.equal('9999999999999999928');
|
|
863
|
+
transaction.should.have.property('feesUsed');
|
|
864
|
+
transaction.feesUsed?.gasLimit.should.equal('500000');
|
|
865
|
+
transaction.should.have.property('halfSigned');
|
|
866
|
+
transaction.halfSigned?.should.have.property('txHex');
|
|
867
|
+
transaction.halfSigned?.should.have.property('recipients');
|
|
868
|
+
});
|
|
869
|
+
it('should generate an unsigned recovery txn of a token for cold wallet ', async function () {
|
|
870
|
+
const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
|
|
871
|
+
const tokenContractAddress = '0x326c977e6efc84e512bb9c30f76e30c160ed06fb'; // tpolygon-link contract token address
|
|
872
|
+
const basecoin = bitgo.coin('tpolygon');
|
|
873
|
+
(0, nock_1.default)(baseUrl)
|
|
874
|
+
.get('/api')
|
|
875
|
+
.query(mockData.getTxListRequest(bitgoFeeAddress))
|
|
876
|
+
.reply(200, mockData.getTxListResponse);
|
|
877
|
+
(0, nock_1.default)(baseUrl)
|
|
878
|
+
.get('/api')
|
|
879
|
+
.query(mockData.getBalanceRequest(bitgoFeeAddress))
|
|
880
|
+
.reply(200, mockData.getBalanceResponse);
|
|
881
|
+
(0, nock_1.default)(baseUrl)
|
|
882
|
+
.get('/api')
|
|
883
|
+
.query(mockData.getTokenBalanceRequest(tokenContractAddress, walletContractAddress))
|
|
884
|
+
.reply(200, mockData.getBalanceResponse);
|
|
885
|
+
(0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
|
|
886
|
+
const transaction = (await basecoin.recover({
|
|
887
|
+
userKey: userXpub,
|
|
888
|
+
backupKey: '',
|
|
889
|
+
walletContractAddress: walletContractAddress,
|
|
890
|
+
bitgoFeeAddress: bitgoFeeAddress,
|
|
891
|
+
recoveryDestination: destinationAddress,
|
|
892
|
+
eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
|
|
893
|
+
gasLimit: 500000,
|
|
894
|
+
bitgoDestinationAddress: bitgoDestinationAddress,
|
|
895
|
+
tokenContractAddress: tokenContractAddress,
|
|
896
|
+
}));
|
|
897
|
+
should.exist(transaction);
|
|
898
|
+
transaction.should.have.property('txHex');
|
|
899
|
+
const txBuilder = (0, getBuilder_1.getBuilder)('tpolygon');
|
|
900
|
+
txBuilder.from(transaction.txHex);
|
|
901
|
+
const rebuiltTx = await txBuilder.build();
|
|
902
|
+
const rebuiltTxJson = rebuiltTx.toJson();
|
|
903
|
+
rebuiltTxJson.should.have.property('data');
|
|
904
|
+
rebuiltTxJson.data.should.startWith('0x0dcd7a6c'); // sendMultiSigToken func
|
|
905
|
+
transaction.should.have.property('userKey');
|
|
906
|
+
transaction.should.have.property('coin');
|
|
907
|
+
transaction.coin.should.equal('tpolygon:link');
|
|
908
|
+
transaction.should.have.property('contractSequenceId');
|
|
909
|
+
transaction.should.have.property('expireTime');
|
|
910
|
+
transaction.should.have.property('gasLimit');
|
|
911
|
+
transaction.gasLimit.should.equal('500000');
|
|
912
|
+
transaction.should.have.property('isEvmBasedCrossChainRecovery');
|
|
913
|
+
transaction.isEvmBasedCrossChainRecovery?.should.equal(true);
|
|
914
|
+
transaction.should.have.property('walletContractAddress');
|
|
915
|
+
transaction.walletContractAddress.should.equal(sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);
|
|
916
|
+
transaction.should.have.property('recipients');
|
|
917
|
+
const recipient = transaction.recipients[0];
|
|
918
|
+
recipient.should.have.property('address');
|
|
919
|
+
recipient.address.should.equal(destinationAddress);
|
|
920
|
+
recipient.should.have.property('amount');
|
|
921
|
+
recipient.amount.should.equal('9999999999999999928');
|
|
922
|
+
});
|
|
923
|
+
it('should generate an unsigned recovery txn of a token for custody wallet', async function () {
|
|
924
|
+
const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
|
|
925
|
+
const tokenContractAddress = '0x326c977e6efc84e512bb9c30f76e30c160ed06fb'; // tpolygon-link contract token address
|
|
926
|
+
const basecoin = bitgo.coin('tpolygon');
|
|
927
|
+
(0, nock_1.default)(baseUrl)
|
|
928
|
+
.get('/api')
|
|
929
|
+
.query(mockData.getTxListRequest(bitgoFeeAddress))
|
|
930
|
+
.reply(200, mockData.getTxListResponse);
|
|
931
|
+
(0, nock_1.default)(baseUrl)
|
|
932
|
+
.get('/api')
|
|
933
|
+
.query(mockData.getBalanceRequest(bitgoFeeAddress))
|
|
934
|
+
.reply(200, mockData.getBalanceResponse);
|
|
935
|
+
(0, nock_1.default)(baseUrl)
|
|
936
|
+
.get('/api')
|
|
937
|
+
.query(mockData.getTokenBalanceRequest(tokenContractAddress, walletContractAddress))
|
|
938
|
+
.reply(200, mockData.getBalanceResponse);
|
|
939
|
+
(0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
|
|
940
|
+
const transaction = (await basecoin.recover({
|
|
941
|
+
userKey: '',
|
|
942
|
+
backupKey: '',
|
|
943
|
+
walletContractAddress: walletContractAddress,
|
|
944
|
+
bitgoFeeAddress: bitgoFeeAddress,
|
|
945
|
+
recoveryDestination: destinationAddress,
|
|
946
|
+
eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
|
|
947
|
+
gasLimit: 500000,
|
|
948
|
+
bitgoDestinationAddress: bitgoDestinationAddress,
|
|
949
|
+
tokenContractAddress: tokenContractAddress,
|
|
950
|
+
}));
|
|
951
|
+
should.exist(transaction);
|
|
952
|
+
transaction.should.have.property('txHex');
|
|
953
|
+
const txBuilder = (0, getBuilder_1.getBuilder)('tpolygon');
|
|
954
|
+
txBuilder.from(transaction.txHex);
|
|
955
|
+
const rebuiltTx = await txBuilder.build();
|
|
956
|
+
const rebuiltTxJson = rebuiltTx.toJson();
|
|
957
|
+
rebuiltTxJson.should.have.property('data');
|
|
958
|
+
rebuiltTxJson.data.should.startWith('0x0dcd7a6c'); // sendMultiSigToken func
|
|
959
|
+
transaction.should.have.property('coin');
|
|
960
|
+
transaction.coin.should.equal('tpolygon:link');
|
|
961
|
+
transaction.should.have.property('contractSequenceId');
|
|
962
|
+
transaction.should.have.property('expireTime');
|
|
963
|
+
transaction.should.have.property('gasLimit');
|
|
964
|
+
transaction.gasLimit.should.equal('500000');
|
|
965
|
+
transaction.should.have.property('isEvmBasedCrossChainRecovery');
|
|
966
|
+
transaction.isEvmBasedCrossChainRecovery?.should.equal(true);
|
|
967
|
+
transaction.should.have.property('walletContractAddress');
|
|
968
|
+
transaction.walletContractAddress.should.equal(sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);
|
|
969
|
+
transaction.should.have.property('recipients');
|
|
970
|
+
const recipient = transaction.recipients[0];
|
|
971
|
+
recipient.should.have.property('address');
|
|
972
|
+
recipient.address.should.equal(destinationAddress);
|
|
973
|
+
recipient.should.have.property('amount');
|
|
974
|
+
recipient.amount.should.equal('9999999999999999928');
|
|
975
|
+
});
|
|
976
|
+
it('should generate an unsigned recovery txn of a token for hot wallet', async function () {
|
|
977
|
+
const userKey = '{"iv":"VFZ3jvXhxo1Z+Yaf2MtZnA==","v":1,"iter":10000,"ks":256,"ts":64,"mode"\n' +
|
|
978
|
+
':"ccm","adata":"","cipher":"aes","salt":"p+fkHuLa/8k=","ct":"hYG7pvljLIgCjZ\n' +
|
|
979
|
+
'53PBlCde5KZRmlUKKHLtDMk+HJfuU46hW+x+C9WsIAO4gFPnTCvFVmQ8x7czCtcNFub5AO2otOG\n' +
|
|
980
|
+
'OsX4GE2gXOEmCl1TpWwwNhm7yMUjGJUpgW6ZZgXSXdDitSKi4V/hk78SGSzjFOBSPYRa6I="}\n';
|
|
981
|
+
const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
|
|
982
|
+
const walletPassphrase = sdk_test_1.TestBitGo.V2.TEST_RECOVERY_PASSCODE;
|
|
983
|
+
const tokenContractAddress = '0x326c977e6efc84e512bb9c30f76e30c160ed06fb'; // tpolygon-link contract token address
|
|
984
|
+
const basecoin = bitgo.coin('tpolygon');
|
|
985
|
+
(0, nock_1.default)(baseUrl)
|
|
986
|
+
.get('/api')
|
|
987
|
+
.query(mockData.getTxListRequest(bitgoFeeAddress))
|
|
988
|
+
.reply(200, mockData.getTxListResponse);
|
|
989
|
+
(0, nock_1.default)(baseUrl)
|
|
990
|
+
.get('/api')
|
|
991
|
+
.query(mockData.getBalanceRequest(bitgoFeeAddress))
|
|
992
|
+
.reply(200, mockData.getBalanceResponse);
|
|
993
|
+
(0, nock_1.default)(baseUrl)
|
|
994
|
+
.get('/api')
|
|
995
|
+
.query(mockData.getTokenBalanceRequest(tokenContractAddress, walletContractAddress))
|
|
996
|
+
.reply(200, mockData.getBalanceResponse);
|
|
997
|
+
(0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
|
|
998
|
+
const transaction = (await basecoin.recover({
|
|
999
|
+
userKey: userKey,
|
|
1000
|
+
backupKey: '',
|
|
1001
|
+
walletPassphrase: walletPassphrase,
|
|
1002
|
+
walletContractAddress: walletContractAddress,
|
|
1003
|
+
bitgoFeeAddress: bitgoFeeAddress,
|
|
1004
|
+
recoveryDestination: destinationAddress,
|
|
1005
|
+
eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
|
|
1006
|
+
gasLimit: 500000,
|
|
1007
|
+
bitgoDestinationAddress: bitgoDestinationAddress,
|
|
1008
|
+
tokenContractAddress: tokenContractAddress,
|
|
1009
|
+
}));
|
|
1010
|
+
should.exist(transaction);
|
|
1011
|
+
transaction.should.have.property('txHex');
|
|
1012
|
+
const txBuilder = (0, getBuilder_1.getBuilder)('tpolygon');
|
|
1013
|
+
txBuilder.from(transaction.txHex);
|
|
1014
|
+
const rebuiltTx = await txBuilder.build();
|
|
1015
|
+
const rebuiltTxJson = rebuiltTx.toJson();
|
|
1016
|
+
rebuiltTxJson.should.have.property('data');
|
|
1017
|
+
rebuiltTxJson.data.should.startWith('0x0dcd7a6c'); // sendMultiSigToken func
|
|
1018
|
+
transaction.should.have.property('coin');
|
|
1019
|
+
transaction.coin.should.equal('tpolygon:link');
|
|
1020
|
+
transaction.should.have.property('contractSequenceId');
|
|
1021
|
+
transaction.should.have.property('expireTime');
|
|
1022
|
+
transaction.should.have.property('gasLimit');
|
|
1023
|
+
transaction.gasLimit.should.equal('500000');
|
|
1024
|
+
transaction.should.have.property('isEvmBasedCrossChainRecovery');
|
|
1025
|
+
transaction.isEvmBasedCrossChainRecovery?.should.equal(true);
|
|
1026
|
+
transaction.should.have.property('walletContractAddress');
|
|
1027
|
+
transaction.walletContractAddress.should.equal(sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);
|
|
1028
|
+
transaction.should.have.property('recipients');
|
|
1029
|
+
const recipient = transaction.recipients[0];
|
|
1030
|
+
recipient.should.have.property('address');
|
|
1031
|
+
recipient.address.should.equal(destinationAddress);
|
|
1032
|
+
recipient.should.have.property('amount');
|
|
1033
|
+
recipient.amount.should.equal('9999999999999999928');
|
|
1034
|
+
transaction.should.have.property('feesUsed');
|
|
1035
|
+
transaction.feesUsed?.gasLimit.should.equal('500000');
|
|
1036
|
+
transaction.should.have.property('halfSigned');
|
|
1037
|
+
transaction.halfSigned?.should.have.property('txHex');
|
|
1038
|
+
transaction.halfSigned?.should.have.property('recipients');
|
|
1039
|
+
});
|
|
1040
|
+
});
|
|
1041
|
+
describe('Non-BitGo Recovery for Hot Wallets (MPCv2)', function () {
|
|
1042
|
+
const baseUrl = sdk_core_1.common.Environments.test.polygonscanBaseUrl;
|
|
1043
|
+
it('should build a recovery transaction for MPCv2 kind of hot wallets', async function () {
|
|
1044
|
+
(0, nock_1.default)(baseUrl)
|
|
1045
|
+
.get('/api')
|
|
1046
|
+
.query(mockData.getTxListRequest(mockData.getNonBitGoRecoveryForHotWalletsMPCv2().bitgoFeeAddress))
|
|
1047
|
+
.reply(200, mockData.getTxListResponse);
|
|
1048
|
+
(0, nock_1.default)(baseUrl)
|
|
1049
|
+
.get('/api')
|
|
1050
|
+
.query(mockData.getBalanceRequest(mockData.getNonBitGoRecoveryForHotWalletsMPCv2().bitgoFeeAddress))
|
|
1051
|
+
.reply(200, mockData.getBalanceResponse);
|
|
1052
|
+
(0, nock_1.default)(baseUrl)
|
|
1053
|
+
.get('/api')
|
|
1054
|
+
.query(mockData.getBalanceRequest(mockData.getNonBitGoRecoveryForHotWalletsMPCv2().walletContractAddress))
|
|
1055
|
+
.reply(200, mockData.getBalanceResponse);
|
|
1056
|
+
const params = mockData.getNonBitGoRecoveryForHotWalletsMPCv2();
|
|
1057
|
+
const transaction = await basecoin.recover({
|
|
1058
|
+
userKey: params.userKey,
|
|
1059
|
+
backupKey: params.backupKey,
|
|
1060
|
+
walletPassphrase: params.walletPassphrase,
|
|
1061
|
+
walletContractAddress: params.walletContractAddress,
|
|
1062
|
+
bitgoFeeAddress: params.bitgoFeeAddress,
|
|
1063
|
+
recoveryDestination: params.recoveryDestination,
|
|
1064
|
+
eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
|
|
1065
|
+
gasLimit: 500000,
|
|
1066
|
+
isTss: true,
|
|
1067
|
+
bitgoDestinationAddress: params.bitgoDestinationAddress,
|
|
1068
|
+
replayProtectionOptions: { chain: 80002, hardfork: 'london' },
|
|
1069
|
+
intendedChain: params.intendedChain,
|
|
1070
|
+
});
|
|
1071
|
+
should.exist(transaction);
|
|
1072
|
+
transaction.should.have.property('tx');
|
|
1073
|
+
});
|
|
1074
|
+
it('should throw an error for invalid user key', async function () {
|
|
1075
|
+
const params = mockData.getInvalidNonBitGoRecoveryParams();
|
|
1076
|
+
await assert_1.default.rejects(async () => {
|
|
1077
|
+
await basecoin.recover({
|
|
1078
|
+
userKey: params.userKey,
|
|
1079
|
+
backupKey: params.backupKey,
|
|
1080
|
+
walletPassphrase: params.walletPassphrase,
|
|
1081
|
+
walletContractAddress: params.walletContractAddress,
|
|
1082
|
+
bitgoFeeAddress: params.bitgoFeeAddress,
|
|
1083
|
+
recoveryDestination: params.recoveryDestination,
|
|
1084
|
+
eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
|
|
1085
|
+
gasLimit: 500000,
|
|
1086
|
+
bitgoDestinationAddress: params.bitgoDestinationAddress,
|
|
1087
|
+
intendedChain: params.intendedChain,
|
|
1088
|
+
});
|
|
1089
|
+
}, Error, 'user key is invalid');
|
|
1090
|
+
});
|
|
1091
|
+
});
|
|
1092
|
+
describe('Build Unsigned Sweep for Self-Custody Cold Wallets (MPCv2)', function () {
|
|
1093
|
+
const baseUrl = sdk_core_1.common.Environments.test.polygonscanBaseUrl;
|
|
1094
|
+
it('should generate an unsigned sweep without derivation seed', async function () {
|
|
1095
|
+
(0, nock_1.default)(baseUrl)
|
|
1096
|
+
.get('/api')
|
|
1097
|
+
.query(mockData.getTxListRequest(mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2().address))
|
|
1098
|
+
.reply(200, mockData.getTxListResponse);
|
|
1099
|
+
(0, nock_1.default)(baseUrl)
|
|
1100
|
+
.get('/api')
|
|
1101
|
+
.query(mockData.getBalanceRequest(mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2().address))
|
|
1102
|
+
.reply(200, mockData.getBalanceResponse);
|
|
1103
|
+
(0, nock_1.default)(baseUrl)
|
|
1104
|
+
.get('/api')
|
|
1105
|
+
.query(mockData.getBalanceRequest(mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2().walletContractAddress))
|
|
1106
|
+
.reply(200, mockData.getBalanceResponse);
|
|
1107
|
+
const params = mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2();
|
|
1108
|
+
const sweepResult = await basecoin.recover({
|
|
1109
|
+
userKey: params.commonKeyChain,
|
|
1110
|
+
backupKey: params.commonKeyChain,
|
|
1111
|
+
recoveryDestination: params.recoveryDestination,
|
|
1112
|
+
gasLimit: 200000,
|
|
1113
|
+
eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
|
|
1114
|
+
walletContractAddress: params.walletContractAddress,
|
|
1115
|
+
isTss: true,
|
|
1116
|
+
replayProtectionOptions: {
|
|
1117
|
+
chain: '137',
|
|
1118
|
+
hardfork: 'london',
|
|
1119
|
+
},
|
|
1120
|
+
});
|
|
1121
|
+
should.exist(sweepResult);
|
|
1122
|
+
const output = sweepResult;
|
|
1123
|
+
output.should.have.property('txRequests');
|
|
1124
|
+
output.txRequests.should.have.length(1);
|
|
1125
|
+
output.txRequests[0].should.have.property('transactions');
|
|
1126
|
+
output.txRequests[0].transactions.should.have.length(1);
|
|
1127
|
+
output.txRequests[0].should.have.property('walletCoin');
|
|
1128
|
+
output.txRequests[0].transactions[0].should.have.property('unsignedTx');
|
|
1129
|
+
output.txRequests[0].transactions[0].unsignedTx.should.have.property('serializedTxHex');
|
|
1130
|
+
output.txRequests[0].transactions[0].unsignedTx.should.have.property('signableHex');
|
|
1131
|
+
output.txRequests[0].transactions[0].unsignedTx.should.have.property('derivationPath');
|
|
1132
|
+
output.txRequests[0].transactions[0].unsignedTx.should.have.property('feeInfo');
|
|
1133
|
+
output.txRequests[0].transactions[0].unsignedTx.should.have.property('parsedTx');
|
|
1134
|
+
const parsedTx = output.txRequests[0].transactions[0].unsignedTx.parsedTx;
|
|
1135
|
+
parsedTx.should.have.property('spendAmount');
|
|
1136
|
+
output.txRequests[0].transactions[0].unsignedTx.parsedTx.should.have.property('outputs');
|
|
1137
|
+
});
|
|
1138
|
+
it('should throw an error for invalid address', async function () {
|
|
1139
|
+
const params = mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2();
|
|
1140
|
+
params.recoveryDestination = 'invalidAddress';
|
|
1141
|
+
params.userKey =
|
|
1142
|
+
'0234eb39b22fed523ece7c78da29ba1f1de5b64a6e48013e0914de793bc1df0570e779de04758732734d97e54b782c8b336283811af6a2c57bd81438798e1c2446';
|
|
1143
|
+
params.backupKey =
|
|
1144
|
+
'0234eb39b22fed523ece7c78da29ba1f1de5b64a6e48013e0914de793bc1df0570e779de04758732734d97e54b782c8b336283811af6a2c57bd81438798e1c2446';
|
|
1145
|
+
await assert_1.default.rejects(async () => {
|
|
1146
|
+
await basecoin.recover({
|
|
1147
|
+
recoveryDestination: params.recoveryDestination,
|
|
1148
|
+
gasLimit: 2000,
|
|
1149
|
+
eip1559: { maxFeePerGas: 200, maxPriorityFeePerGas: 10000 },
|
|
1150
|
+
userKey: params.userKey,
|
|
1151
|
+
backupKey: params.backupKey,
|
|
1152
|
+
walletContractAddress: params.walletContractAddress,
|
|
1153
|
+
isTss: true,
|
|
1154
|
+
replayProtectionOptions: {
|
|
1155
|
+
chain: '137',
|
|
1156
|
+
hardfork: 'london',
|
|
1157
|
+
},
|
|
1158
|
+
});
|
|
1159
|
+
}, Error, 'Error: invalid address');
|
|
1160
|
+
});
|
|
1161
|
+
});
|
|
1162
|
+
describe('Test isWalletAddress', function () {
|
|
1163
|
+
it('verify address for tpolygon', async function () {
|
|
1164
|
+
const keychains = [
|
|
1165
|
+
{
|
|
1166
|
+
id: '6920c05e0b195abd8ed0bb0a5df32cdc',
|
|
1167
|
+
source: 'user',
|
|
1168
|
+
type: 'tss',
|
|
1169
|
+
commonKeychain: '02f929f92c25eaccb0e3ebaa0f5ed900c5ad30ba94f8444dc89c94e12f01aa5371522d810b8918ecc9e41eb901352df1c7977420fbaf9b8617f61b780b32b2ccad',
|
|
1170
|
+
},
|
|
1171
|
+
{
|
|
1172
|
+
id: '6920c05e93c3b1c9e5006bf00a3cf016',
|
|
1173
|
+
source: 'backup',
|
|
1174
|
+
type: 'tss',
|
|
1175
|
+
commonKeychain: '02f929f92c25eaccb0e3ebaa0f5ed900c5ad30ba94f8444dc89c94e12f01aa5371522d810b8918ecc9e41eb901352df1c7977420fbaf9b8617f61b780b32b2ccad',
|
|
1176
|
+
},
|
|
1177
|
+
{
|
|
1178
|
+
id: '6920c05d9ebee0100ec4a8aa7e300c02',
|
|
1179
|
+
source: 'bitgo',
|
|
1180
|
+
type: 'tss',
|
|
1181
|
+
commonKeychain: '02f929f92c25eaccb0e3ebaa0f5ed900c5ad30ba94f8444dc89c94e12f01aa5371522d810b8918ecc9e41eb901352df1c7977420fbaf9b8617f61b780b32b2ccad',
|
|
1182
|
+
},
|
|
1183
|
+
];
|
|
1184
|
+
const params = {
|
|
1185
|
+
address: '0x4e9fc44697f4135455157396485f6fe8f909752f',
|
|
1186
|
+
baseAddress: '0x8cf5ebd51585d159c4a1ca36178f9ad0fd7a594c',
|
|
1187
|
+
coinSpecific: {
|
|
1188
|
+
salt: '0xd',
|
|
1189
|
+
forwarderVersion: 4,
|
|
1190
|
+
feeAddress: '0x44dcb3504e323a3d70142036a99e2d4bba3f2270',
|
|
1191
|
+
},
|
|
1192
|
+
keychains,
|
|
1193
|
+
index: 13,
|
|
1194
|
+
walletVersion: 5,
|
|
1195
|
+
};
|
|
1196
|
+
const coin = bitgo.coin('tpolygon');
|
|
1197
|
+
const isWalletAddr = await coin.isWalletAddress(params);
|
|
1198
|
+
isWalletAddr.should.equal(true);
|
|
1199
|
+
});
|
|
1200
|
+
});
|
|
1201
|
+
});
|
|
1202
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"polygon.js","sourceRoot":"","sources":["../../../test/unit/polygon.ts"],"names":[],"mappings":"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","sourcesContent":["import BigNumber from 'bignumber.js';\nimport { BitGoAPI } from '@bitgo-beta/sdk-api';\nimport { common, ECDSAMethodTypes, FullySignedTransaction, Recipient, TransactionType, Wallet } from '@bitgo-beta/sdk-core';\nimport { TestBitGo, TestBitGoAPI } from '@bitgo-beta/sdk-test';\nimport { bip32 } from '@bitgo-beta/secp256k1';\nimport nock from 'nock';\nimport * as secp256k1 from 'secp256k1';\nimport * as should from 'should';\nimport { Polygon, Tpolygon, TransactionBuilder, TransferBuilder } from '../../src';\nimport { AbstractEthLikeNewCoins, UnsignedSweepTxMPCv2, OfflineVaultTxInfo, optionalDeps } from '@bitgo-beta/abstract-eth';\nimport { getBuilder } from '../getBuilder';\nimport * as mockData from '../fixtures/polygon';\nimport * as sjcl from '@bitgo-beta/sjcl';\nimport assert from 'assert';\n\nnock.enableNetConnect();\n\ndescribe('Polygon', function () {\n  let bitgo: TestBitGoAPI;\n  let basecoin;\n  let hopTxBitgoSignature;\n\n  const address1 = '0x174cfd823af8ce27ed0afee3fcf3c3ba259116be';\n  const address2 = '0x7e85bdc27c050e3905ebf4b8e634d9ad6edd0de6';\n  const hopContractAddress = '0x47ce7cc86efefef19f8fb516b11735d183da8635';\n  const hopDestinationAddress = '0x9c7e8ce6825bD48278B3Ab59228EE26f8BE7925b';\n  const hopTx =\n    '0xf86b808504a817c8ff8252ff949c7e8ce6825bd48278b3ab59228ee26f8be7925b87038d7ea4c68000801ca011bc22c664570133dfca4f08a0b8d02339cf467046d6a4152f04f368d0eaf99ea01d6dc5cf0c897c8d4c3e1df53d0d042784c424536a4cc5b802552b7d64fee8b5';\n  const hopTxid = '0x4af65143bc77da2b50f35b3d13cacb4db18f026bf84bc0743550bc57b9b53351';\n  const userReqSig =\n    '0x404db307f6147f0d8cd338c34c13906ef46a6faa7e0e119d5194ef05aec16e6f3d710f9b7901460f97e924066b62efd74443bd34402c6d40b49c203a559ff2c8';\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    hopTxBitgoSignature =\n      '0xaa' +\n      Buffer.from(secp256k1.ecdsaSign(Buffer.from(hopTxid.slice(2), 'hex'), bitgoKey.privateKey).signature).toString(\n        'hex'\n      );\n\n    const env = 'test';\n    bitgo = TestBitGo.decorate(BitGoAPI, { env });\n    common.Environments[env].hsmXpub = bitgoXpub;\n    bitgo.safeRegister('polygon', Polygon.createInstance);\n    bitgo.safeRegister('tpolygon', Tpolygon.createInstance);\n    bitgo.initializeTestVars();\n    basecoin = bitgo.coin('tpolygon');\n  });\n\n  after(function () {\n    nock.cleanAll();\n  });\n\n  /**\n   * Build an unsigned account-lib multi-signature send transactino\n   * @param destination The destination address of the transaction\n   * @param contractAddress The address of the smart contract processing the transaction\n   * @param contractSequenceId The sequence id of the contract\n   * @param nonce The nonce of the sending address\n   * @param expireTime The expire time of the transaction\n   * @param amount The amount to send to the recipient\n   * @param gasPrice The gas price of the transaction\n   * @param gasLimit The gas limit of the transaction\n   */\n  const buildUnsignedTransaction = async function ({\n    destination,\n    contractAddress,\n    contractSequenceId = 1,\n    nonce = 0,\n    expireTime = Math.floor(new Date().getTime() / 1000),\n    amount = '100000',\n    gasPrice = '10000',\n    gasLimit = '20000',\n  }) {\n    const txBuilder: TransactionBuilder = getBuilder('tpolygon') as TransactionBuilder;\n    txBuilder.type(TransactionType.Send);\n    txBuilder.fee({\n      fee: gasPrice,\n      gasLimit: gasLimit,\n    });\n    txBuilder.counter(nonce);\n    txBuilder.contract(contractAddress);\n    const transferBuilder = txBuilder.transfer() as TransferBuilder;\n\n    transferBuilder\n      .coin('tpolygon')\n      .expirationTime(expireTime)\n      .amount(amount)\n      .to(destination)\n      .contractSequenceId(contractSequenceId);\n\n    return await txBuilder.build();\n  };\n\n  describe('Instantiate', () => {\n    it('should instantiate the coin', function () {\n      let localBasecoin = bitgo.coin('tpolygon');\n      localBasecoin.should.be.an.instanceof(Tpolygon);\n\n      localBasecoin = bitgo.coin('polygon');\n      localBasecoin.should.be.an.instanceof(Polygon);\n    });\n  });\n\n  describe('Explain transaction:', () => {\n    it('should fail if the options object is missing parameters', async function () {\n      const explainParams = {\n        feeInfo: { fee: 1 },\n        txHex: null,\n      };\n      await basecoin.explainTransaction(explainParams).should.be.rejectedWith('missing explain tx parameters');\n    });\n\n    it('explain a transfer transaction', async function () {\n      const destination = '0xfaa8f14f46a99eb439c50e0c3b835cc21dad51b4';\n      const contractAddress = '0x9e2c5712ab4caf402a98c4bf58c79a0dfe718ad1';\n\n      const unsignedTransaction = await buildUnsignedTransaction({\n        destination,\n        contractAddress,\n      });\n\n      const explainParams = {\n        halfSigned: {\n          txHex: unsignedTransaction.toBroadcastFormat(),\n        },\n        feeInfo: { fee: 1 },\n      };\n      const explanation = await basecoin.explainTransaction(explainParams);\n      should.exist(explanation.id);\n    });\n  });\n\n  describe('Sign Transaction', () => {\n    const account_1 = {\n      address: '0x8Ce59c2d1702844F8EdED451AA103961bC37B4e8',\n      owner_1: '4ee089aceabf3ddbf748db79b1066c33b7d3ea1ab3eb7e325121bba2bff2f5ca',\n      owner_2: '5c7e4efff7304d4dfff6d5f1591844ec6f2adfa6a47e9fece6a3c1a4d755f1e3',\n      owner_3: '4421ab25dd91e1a3180d03d57c323a7886dcc313d3b3a4b4256a5791572bf597',\n    };\n\n    const account_2 = {\n      address: '0xeeaf0F05f37891ab4a21208B105A0687d12c5aF7',\n      owner_1: '4ee089aceabf3ddbf748db79b1066c33b7d3ea1ab3eb7e325121bba2bff2f5ca',\n      owner_2: '5ca116d25aec5f765465432cc421ff25ef9ffdc330b10bb3d9ad61e3baad88d7',\n      owner_3: '1fae946cc84af8bd74d610a88537e24e19c3349d478d86fc5bb59ba4c88fb9cc',\n    };\n\n    it('should sign an unsigned test tx', async function () {\n      const builder = getBuilder('tpolygon') as TransactionBuilder;\n      builder.fee({\n        fee: '280000000000',\n        gasLimit: '7000000',\n      });\n      builder.counter(1);\n      builder.type(TransactionType.Send);\n      builder.contract(account_1.address);\n      builder.transfer().amount('1').to(account_2.address).expirationTime(10000).contractSequenceId(1);\n\n      const unsignedTx = await builder.build();\n      const unsignedTxForBroadcasting = unsignedTx.toBroadcastFormat();\n\n      const halfSignedRawTx = await basecoin.signTransaction({\n        txPrebuild: {\n          txHex: unsignedTxForBroadcasting,\n        },\n        prv: account_1.owner_2,\n      });\n\n      builder.transfer().key(account_1.owner_2);\n      const halfSignedTx = await builder.build();\n      const halfSignedTxForBroadcasting = halfSignedTx.toBroadcastFormat();\n\n      halfSignedRawTx.halfSigned.txHex.should.equals(halfSignedTxForBroadcasting);\n      halfSignedRawTx.halfSigned.recipients.length.should.equals(1);\n      halfSignedRawTx.halfSigned.recipients[0].address.toLowerCase().should.equals(account_2.address.toLowerCase());\n      halfSignedRawTx.halfSigned.recipients[0].amount.toLowerCase().should.equals('1');\n    });\n\n    it('should sign an unsigned test tx with eip1559', async function () {\n      const builder = getBuilder('tpolygon') as TransactionBuilder;\n      builder.fee({\n        fee: '280000000000',\n        gasLimit: '7000000',\n        eip1559: {\n          maxFeePerGas: '7593123',\n          maxPriorityFeePerGas: '150',\n        },\n      });\n      builder.counter(1);\n      builder.type(TransactionType.Send);\n      builder.contract(account_1.address);\n      builder.transfer().amount('1').to(account_2.address).expirationTime(10000).contractSequenceId(1);\n\n      const unsignedTx = await builder.build();\n      const unsignedTxForBroadcasting = unsignedTx.toBroadcastFormat();\n\n      const halfSignedRawTx = await basecoin.signTransaction({\n        txPrebuild: {\n          txHex: unsignedTxForBroadcasting,\n          eip1559: {\n            maxFeePerGas: '7593123',\n            maxPriorityFeePerGas: '150',\n          },\n        },\n        prv: account_1.owner_2,\n      });\n\n      builder.transfer().key(account_1.owner_2);\n      const halfSignedTx = await builder.build();\n      const halfSignedTxForBroadcasting = halfSignedTx.toBroadcastFormat();\n\n      halfSignedRawTx.halfSigned.txHex.should.equals(halfSignedTxForBroadcasting);\n      halfSignedRawTx.halfSigned.recipients.length.should.equals(1);\n      halfSignedRawTx.halfSigned.recipients[0].address.toLowerCase().should.equals(account_2.address.toLowerCase());\n      halfSignedRawTx.halfSigned.recipients[0].amount.toLowerCase().should.equals('1');\n      halfSignedRawTx.halfSigned.eip1559.maxFeePerGas.should.equal('7593123');\n      halfSignedRawTx.halfSigned.eip1559.maxPriorityFeePerGas.should.equal('150');\n    });\n\n    it('should include isBatch field in halfSigned when signing a native batch transaction', async function () {\n      const batcherContractAddress = '0xb1b7e7cc1ecafbfd0771a5eb5454ab5b0356980d';\n      const recipients: Recipient[] = [\n        {\n          address: account_2.address,\n          amount: '500000000000',\n        },\n        {\n          address: '0x7e85bdc27c050e3905ebf4b8e634d9ad6edd0de6',\n          amount: '500000000000',\n        },\n      ];\n      const totalAmount = recipients.reduce(\n        (sum, recipient) => new BigNumber(sum).plus(recipient.amount).toString(),\n        '0'\n      );\n      const BATCH_METHOD_NAME = 'batch';\n      const BATCH_METHOD_TYPES = ['address[]', 'uint256[]'];\n\n      const addresses = recipients.map((r) => r.address);\n      const amounts = recipients.map((r) => r.amount);\n      const batchExecutionInfo = {\n        values: [addresses, amounts],\n        totalAmount: totalAmount,\n      };\n\n      const ethAbi = require('ethereumjs-abi');\n      const ethUtil = require('ethereumjs-util');\n\n      const getMethodCallData = (functionName, types, values) => {\n        return Buffer.concat([\n          // function signature\n          ethAbi.methodID(functionName, types),\n          // function arguments\n          ethAbi.rawEncode(types, values),\n        ]);\n      };\n\n      const batchData = ethUtil.addHexPrefix(\n        getMethodCallData(BATCH_METHOD_NAME, BATCH_METHOD_TYPES, batchExecutionInfo.values).toString('hex')\n      );\n\n      const txBuilder = getBuilder('tpolygon') as TransactionBuilder;\n      txBuilder.fee({\n        fee: '280000000000',\n        gasLimit: '7000000',\n      });\n      txBuilder.counter(1);\n      txBuilder.type(TransactionType.Send);\n      txBuilder.contract(account_1.address);\n      txBuilder\n        .transfer()\n        .amount(totalAmount)\n        .to(batcherContractAddress)\n        .data(batchData)\n        .expirationTime(10000)\n        .contractSequenceId(1);\n\n      const unsignedTx = await txBuilder.build();\n      const unsignedTxHex = unsignedTx.toBroadcastFormat();\n\n      const txPrebuild = {\n        txHex: unsignedTxHex,\n        isBatch: true,\n        recipients: [\n          {\n            amount: totalAmount,\n            address: batcherContractAddress,\n          },\n        ],\n        nextContractSequenceId: 1,\n        gasPrice: 280000000000,\n        gasLimit: 7000000,\n        coin: 'tpolygon',\n        walletId: 'fakeWalletId',\n        walletContractAddress: account_1.address,\n      };\n\n      const signedTx = await basecoin.signTransaction({\n        txPrebuild,\n        prv: account_1.owner_2,\n      });\n\n      should.exist(signedTx);\n      should.exist(signedTx.halfSigned);\n      signedTx.halfSigned.should.have.property('isBatch', true);\n    });\n  });\n\n  describe('Transaction Verification', function () {\n    it('should verify a normal txPrebuild from the bitgo server that matches the client txParams', async function () {\n      const wallet = new Wallet(bitgo, basecoin, {});\n\n      const txParams = {\n        recipients: [{ amount: '1000000000000', address: address1 }],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '1000000000000', address: address1 }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: false,\n        coin: 'tpolygon',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      const isTransactionVerified = await basecoin.verifyTransaction({ txParams, txPrebuild, wallet, verification });\n      isTransactionVerified.should.equal(true);\n    });\n\n    it('should verify a hop txPrebuild from the bitgo server that matches the client txParams', async function () {\n      const wallet = new Wallet(bitgo, basecoin, {});\n\n      const txParams = {\n        recipients: [{ amount: 1000000000000000, address: hopDestinationAddress }],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n        hop: true,\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '5000000000000000', address: hopContractAddress }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: false,\n        coin: 'tpolygon',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n        hopTransaction: {\n          tx: hopTx,\n          id: hopTxid,\n          signature: hopTxBitgoSignature,\n          paymentId: '2773928196',\n          gasPrice: 20000000000,\n          gasLimit: 500000,\n          amount: '1000000000000000',\n          recipient: hopDestinationAddress,\n          nonce: 0,\n          userReqSig: userReqSig,\n          gasPriceMax: 500000000000,\n        },\n      };\n\n      const verification = {};\n\n      const isTransactionVerified = await basecoin.verifyTransaction({ txParams, txPrebuild, wallet, verification });\n      isTransactionVerified.should.equal(true);\n    });\n\n    it('should reject when client txParams are missing', async function () {\n      const wallet = new Wallet(bitgo, basecoin, {});\n\n      const txParams = null;\n\n      const txPrebuild = {\n        recipients: [{ amount: '1000000000000', address: address1 }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: false,\n        coin: 'tpolygon',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      await basecoin\n        .verifyTransaction({ txParams, txPrebuild, wallet, verification })\n        .should.be.rejectedWith('missing params');\n    });\n\n    it('should reject txPrebuild that is both batch and hop', async function () {\n      const wallet = new Wallet(bitgo, basecoin, {});\n\n      const txParams = {\n        recipients: [\n          { amount: '1000000000000', address: address1 },\n          { amount: '2500000000000', address: address2 },\n        ],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n        hop: true,\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '3500000000000', address: address1 }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: true,\n        coin: 'tpolygon',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n        hopTransaction: {\n          tx: hopTx,\n          id: hopTxid,\n          signature: hopTxBitgoSignature,\n          paymentId: '2773928196',\n          gasPrice: 20000000000,\n          gasLimit: 500000,\n          amount: '1000000000000000',\n          recipient: hopDestinationAddress,\n          nonce: 0,\n          userReqSig: userReqSig,\n          gasPriceMax: 500000000000,\n        },\n      };\n\n      const verification = {};\n\n      await basecoin\n        .verifyTransaction({ txParams, txPrebuild, wallet, verification })\n        .should.be.rejectedWith('tx cannot be both a batch and hop transaction');\n    });\n\n    it('should reject a txPrebuild with more than one recipient', async function () {\n      const wallet = new Wallet(bitgo, basecoin, {});\n\n      const txParams = {\n        recipients: [\n          { amount: '1000000000000', address: address1 },\n          { amount: '2500000000000', address: address2 },\n        ],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n      };\n\n      const txPrebuild = {\n        recipients: [\n          { amount: '1000000000000', address: address1 },\n          { amount: '2500000000000', address: address2 },\n        ],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: true,\n        coin: 'tpolygon',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      await basecoin\n        .verifyTransaction({ txParams, txPrebuild, wallet, verification })\n        .should.be.rejectedWith(\n          `tpolygon doesn't support sending to more than 1 destination address within a single transaction. Try again, using only a single recipient.`\n        );\n    });\n    it('should reject a hop txPrebuild that does not send to its hop address', async function () {\n      const wallet = new Wallet(bitgo, basecoin, {});\n\n      const txParams = {\n        recipients: [{ amount: '1000000000000000', address: hopDestinationAddress }],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n        hop: true,\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '5000000000000000', address: address1 }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: false,\n        coin: 'tpolygon',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n        hopTransaction: {\n          tx: hopTx,\n          id: hopTxid,\n          signature: hopTxBitgoSignature,\n          paymentId: '0',\n          gasPrice: 20000000000,\n          gasLimit: 500000,\n          amount: '1000000000000000',\n          recipient: hopDestinationAddress,\n          nonce: 0,\n          userReqSig: userReqSig,\n          gasPriceMax: 500000000000,\n        },\n      };\n\n      const verification = {};\n\n      await basecoin\n        .verifyTransaction({ txParams, txPrebuild, wallet, verification })\n        .should.be.rejectedWith('recipient address of txPrebuild does not match hop address');\n    });\n\n    it('should reject a normal txPrebuild from the bitgo server with the wrong amount', async function () {\n      const wallet = new Wallet(bitgo, basecoin, {});\n\n      const txParams = {\n        recipients: [{ amount: '1000000000000', address: address1 }],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '2000000000000', address: address1 }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: false,\n        coin: 'tpolygon',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      await basecoin\n        .verifyTransaction({ txParams, txPrebuild, wallet, verification })\n        .should.be.rejectedWith(\n          'normal transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client'\n        );\n    });\n\n    it('should reject a normal txPrebuild from the bitgo server with the wrong recipient', async function () {\n      const wallet = new Wallet(bitgo, basecoin, {});\n\n      const txParams = {\n        recipients: [{ amount: '1000000000000', address: address1 }],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '1000000000000', address: address2 }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: false,\n        coin: 'tpolygon',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      await basecoin\n        .verifyTransaction({ txParams, txPrebuild, wallet, verification })\n        .should.be.rejectedWith(\n          'destination address in normal txPrebuild does not match that in txParams supplied by client'\n        );\n    });\n\n    it('should reject a txPrebuild from the bitgo server with the wrong coin', async function () {\n      const wallet = new Wallet(bitgo, basecoin, {});\n\n      const txParams = {\n        recipients: [{ amount: '1000000000000', address: address1 }],\n        wallet: wallet,\n        walletPassphrase: 'fakeWalletPassphrase',\n      };\n\n      const txPrebuild = {\n        recipients: [{ amount: '1000000000000', address: address1 }],\n        nextContractSequenceId: 0,\n        gasPrice: 20000000000,\n        gasLimit: 500000,\n        isBatch: false,\n        coin: 'btc',\n        walletId: 'fakeWalletId',\n        walletContractAddress: 'fakeWalletContractAddress',\n      };\n\n      const verification = {};\n\n      await basecoin\n        .verifyTransaction({ txParams, txPrebuild, wallet, verification })\n        .should.be.rejectedWith('coin in txPrebuild did not match that in txParams supplied by client');\n    });\n  });\n\n  describe('Recover transaction:', function () {\n    const baseUrl = common.Environments.test.polygonscanBaseUrl as string;\n    const userXpub =\n      'xpub661MyMwAqRbcEeTc8789MK5PUGEYiPG4F4V17n2Rd2LoTATA1XoCnJT5FAYAShQxSxtFjpo5NHmcWwTp2LiWGBMwpUcAA3HywhxivgYfq7q';\n    const userXprv =\n      'xprv9s21ZrQH143K2AP925b8zB8evEQ4JvYCsqZQKPcp4gopaN81TzUxEW8bPtVyDgjmddGhRRETn8xi1cVAB9bf1Bx9kGRRFgTZXxJayZLnag1';\n    const backupXpub =\n      'xpub661MyMwAqRbcFZX15xpZf4ERCGHiVSJm8r5C4yh1yXV2GrdZCUPYo4WQr6tN9oUywKXsgSHo7Risf9r22GH5joVD2hEEEhqnSCvK8qy11wW';\n    const backupXprv =\n      'xprv9s21ZrQH143K35SXywHZHvHgeETE5yaumd9bGbHQRBx3Q4JQew5JFGBvzqiZjCUkBdBUZnfuMDTGURRayN1hFSWxEJQsCEAMm1D3pk1h7Jj';\n\n    it('should generate an unsigned sweep', 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      nock(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);\n      const basecoin = bitgo.coin('tpolygon') as Polygon;\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      })) as OfflineVaultTxInfo;\n      should.exist(transaction);\n      transaction.should.have.property('txHex');\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('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(TestBitGo.V2.TEST_ERC20_TOKEN_RECIPIENT);\n      recipient.should.have.property('amount');\n      recipient.amount.should.equal('9999999999999999928');\n    });\n\n    it('should construct a recovery transaction without BitGo', async function () {\n      const backupKeyAddress = '0x6d22efdd634996248170c948e5726007fc251bb3';\n      const walletContractAddress = TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS as string;\n      nock(baseUrl)\n        .get('/api')\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      nock(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);\n      const basecoin = bitgo.coin('tpolygon') as Polygon;\n      const transaction = (await basecoin.recover({\n        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        backupKey:\n          '{\"iv\":\"AbsCtv1qwPIhOgyrCpNagA==\",\"v\":1,\"iter\":10000,\"ks\":256,\"ts\":64,\"mode\"\\n' +\n          ':\"ccm\",\"adata\":\"\",\"cipher\":\"aes\",\"salt\":\"5vpUDBUlzm8=\",\"ct\":\"PapYYCjBXRLUKA\\n' +\n          'JbOsB/EJ9B8fUmVQTxMPjUnQyAky12me9K66GiMEAxTD7kd6bYAQJuuTkATXKU7Bnf7vK9JxNOw\\n' +\n          'oji7HF9eFH0aD4/hX5SWFfHF2Qfi+TnXv6hVsMAoisDZs3/F67/ZUaDYR0ZsdrQ4Q/cLD0=\"}\\n',\n\n        walletContractAddress: walletContractAddress,\n        walletPassphrase: TestBitGo.V2.TEST_RECOVERY_PASSCODE,\n        recoveryDestination: TestBitGo.V2.TEST_ERC20_TOKEN_RECIPIENT as string,\n        gasLimit: 500000,\n      })) as OfflineVaultTxInfo;\n      should.exist(transaction);\n      transaction.should.have.property('tx');\n      transaction.should.have.property('id');\n      const decodedTx = optionalDeps.EthTx.Transaction.fromSerializedTx(optionalDeps.ethUtil.toBuffer(transaction.tx));\n      decodedTx.should.have.property('gasPrice');\n      decodedTx.should.have.property('nonce');\n      decodedTx.should.have.property('to');\n    });\n\n    xit('should construct a recovery tx with TSS', async function () {\n      const backupKeyAddress = '0xe7406dc43d13f698fb41a345c7783d39a4c2d191';\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getTxListRequest(backupKeyAddress))\n        .reply(200, mockData.getTxListResponse);\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(backupKeyAddress))\n        .reply(200, mockData.getBalanceResponse);\n\n      const basecoin = bitgo.coin('tpolygon') as Polygon;\n      const userKey = mockData.keyShares.userKeyShare;\n      const backupKey = mockData.keyShares.backupKeyShare;\n      const bitgoKey = mockData.keyShares.bitgoKeyShare;\n\n      const userSigningMaterial: ECDSAMethodTypes.SigningMaterial = {\n        pShare: userKey.pShare,\n        backupNShare: backupKey.nShares[1],\n        bitgoNShare: bitgoKey.nShares[1],\n      };\n\n      const backupSigningMaterial: ECDSAMethodTypes.SigningMaterial = {\n        pShare: backupKey.pShare,\n        userNShare: userKey.nShares[2],\n        bitgoNShare: bitgoKey.nShares[2],\n      };\n\n      const encryptedBackupSigningMaterial = sjcl.encrypt(\n        TestBitGo.V2.TEST_RECOVERY_PASSCODE,\n        JSON.stringify(backupSigningMaterial)\n      );\n      const encryptedUserSigningMaterial = sjcl.encrypt(\n        TestBitGo.V2.TEST_RECOVERY_PASSCODE,\n        JSON.stringify(userSigningMaterial)\n      );\n\n      const recoveryParams = {\n        userKey: encryptedUserSigningMaterial,\n        backupKey: encryptedBackupSigningMaterial,\n        walletContractAddress: '0xe7406dc43d13f698fb41a345c7783d39a4c2d191',\n        recoveryDestination: '0xac05da78464520aa7c9d4c19bd7a440b111b3054',\n        walletPassphrase: TestBitGo.V2.TEST_RECOVERY_PASSCODE,\n        isTss: true,\n      };\n\n      const recovery = await basecoin.recover(recoveryParams);\n      // id and tx will always be different because of expireTime\n      should.exist(recovery);\n      recovery.should.have.property('id');\n      recovery.should.have.property('tx');\n    });\n\n    it('should be able to second sign', 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      nock(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);\n      const basecoin = bitgo.coin('tpolygon') as Polygon;\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        replayProtectionOptions: { chain: 80002, hardfork: 'london' },\n        gasLimit: 500000,\n      })) as OfflineVaultTxInfo;\n\n      const txPrebuild = {\n        txHex: transaction.txHex,\n      };\n\n      const params = {\n        txPrebuild,\n        prv: userXprv,\n      };\n      // sign transaction once\n      const halfSigned = await basecoin.signTransaction(params as any);\n      const halfSignedParams = {\n        txPrebuild: halfSigned,\n        isLastSignature: true,\n        walletContractAddress: walletContractAddress,\n        prv: backupXprv,\n      };\n\n      const finalSigned = (await basecoin.signTransaction(halfSignedParams as any)) as FullySignedTransaction;\n      finalSigned.should.have.property('txHex');\n      const txBuilder = getBuilder('tpolygon') as TransactionBuilder;\n      txBuilder.from(finalSigned.txHex);\n      const rebuiltTx = await txBuilder.build();\n      rebuiltTx.signature.length.should.equal(2);\n      rebuiltTx.outputs.length.should.equal(1);\n    });\n  });\n\n  describe('Evm Based Cross Chain Recovery transaction:', function () {\n    const baseUrl = common.Environments.test.polygonscanBaseUrl as string;\n    const userXpub =\n      'xpub661MyMwAqRbcEeTc8789MK5PUGEYiPG4F4V17n2Rd2LoTATA1XoCnJT5FAYAShQxSxtFjpo5NHmcWwTp2LiWGBMwpUcAA3HywhxivgYfq7q';\n    const bitgoFeeAddress = '0x33a42faea3c6e87021347e51700b48aaf49aa1e7';\n    const destinationAddress = '0xd5adde17fed8baed3f32b84af05b8f2816f7b560';\n    const bitgoDestinationAddress = '0xe5986ce4490deb67d2950562ceb930ddf9be7a14';\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\n      const basecoin = bitgo.coin('tpolygon') as Polygon;\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      })) as OfflineVaultTxInfo;\n\n      should.exist(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\n      const basecoin = bitgo.coin('tpolygon') as Polygon;\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      })) as OfflineVaultTxInfo;\n\n      should.exist(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 walletPassphrase = TestBitGo.V2.TEST_RECOVERY_PASSCODE as string;\n\n      const basecoin = bitgo.coin('tpolygon') as Polygon;\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      })) as OfflineVaultTxInfo;\n\n      should.exist(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 tokenContractAddress = '0x326c977e6efc84e512bb9c30f76e30c160ed06fb'; // tpolygon-link contract token address\n\n      const basecoin = bitgo.coin('tpolygon') as Polygon;\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      })) as OfflineVaultTxInfo;\n\n      should.exist(transaction);\n      transaction.should.have.property('txHex');\n\n      const txBuilder = getBuilder('tpolygon') as TransactionBuilder;\n      txBuilder.from(transaction.txHex);\n      const rebuiltTx = await txBuilder.build();\n      const rebuiltTxJson = rebuiltTx.toJson();\n      rebuiltTxJson.should.have.property('data');\n      rebuiltTxJson.data.should.startWith('0x0dcd7a6c'); // sendMultiSigToken func\n\n      transaction.should.have.property('userKey');\n      transaction.should.have.property('coin');\n      transaction.coin.should.equal('tpolygon:link');\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 tokenContractAddress = '0x326c977e6efc84e512bb9c30f76e30c160ed06fb'; // tpolygon-link contract token address\n\n      const basecoin = bitgo.coin('tpolygon') as Polygon;\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      })) as OfflineVaultTxInfo;\n\n      should.exist(transaction);\n      transaction.should.have.property('txHex');\n\n      const txBuilder = getBuilder('tpolygon') as TransactionBuilder;\n      txBuilder.from(transaction.txHex);\n      const rebuiltTx = await txBuilder.build();\n      const rebuiltTxJson = rebuiltTx.toJson();\n      rebuiltTxJson.should.have.property('data');\n      rebuiltTxJson.data.should.startWith('0x0dcd7a6c'); // sendMultiSigToken func\n\n      transaction.should.have.property('coin');\n      transaction.coin.should.equal('tpolygon:link');\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 walletPassphrase = TestBitGo.V2.TEST_RECOVERY_PASSCODE as string;\n      const tokenContractAddress = '0x326c977e6efc84e512bb9c30f76e30c160ed06fb'; // tpolygon-link contract token address\n\n      const basecoin = bitgo.coin('tpolygon') as Polygon;\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      })) as OfflineVaultTxInfo;\n\n      should.exist(transaction);\n      transaction.should.have.property('txHex');\n\n      const txBuilder = getBuilder('tpolygon') as TransactionBuilder;\n      txBuilder.from(transaction.txHex);\n      const rebuiltTx = await txBuilder.build();\n      const rebuiltTxJson = rebuiltTx.toJson();\n      rebuiltTxJson.should.have.property('data');\n      rebuiltTxJson.data.should.startWith('0x0dcd7a6c'); // sendMultiSigToken func\n\n      transaction.should.have.property('coin');\n      transaction.coin.should.equal('tpolygon:link');\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('Non-BitGo Recovery for Hot Wallets (MPCv2)', function () {\n    const baseUrl = common.Environments.test.polygonscanBaseUrl as string;\n\n    it('should build a recovery transaction for MPCv2 kind of hot wallets', async function () {\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getTxListRequest(mockData.getNonBitGoRecoveryForHotWalletsMPCv2().bitgoFeeAddress))\n        .reply(200, mockData.getTxListResponse);\n\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(mockData.getNonBitGoRecoveryForHotWalletsMPCv2().bitgoFeeAddress))\n        .reply(200, mockData.getBalanceResponse);\n\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(mockData.getNonBitGoRecoveryForHotWalletsMPCv2().walletContractAddress))\n        .reply(200, mockData.getBalanceResponse);\n\n      const params = mockData.getNonBitGoRecoveryForHotWalletsMPCv2();\n\n      const transaction = await (basecoin as AbstractEthLikeNewCoins).recover({\n        userKey: params.userKey,\n        backupKey: params.backupKey,\n        walletPassphrase: params.walletPassphrase,\n        walletContractAddress: params.walletContractAddress,\n        bitgoFeeAddress: params.bitgoFeeAddress,\n        recoveryDestination: params.recoveryDestination,\n        eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },\n        gasLimit: 500000,\n        isTss: true,\n        bitgoDestinationAddress: params.bitgoDestinationAddress,\n        replayProtectionOptions: { chain: 80002, hardfork: 'london' },\n        intendedChain: params.intendedChain,\n      });\n      should.exist(transaction);\n      transaction.should.have.property('tx');\n    });\n    it('should throw an error for invalid user key', async function () {\n      const params = mockData.getInvalidNonBitGoRecoveryParams();\n\n      await assert.rejects(\n        async () => {\n          await (basecoin as AbstractEthLikeNewCoins).recover({\n            userKey: params.userKey,\n            backupKey: params.backupKey,\n            walletPassphrase: params.walletPassphrase,\n            walletContractAddress: params.walletContractAddress,\n            bitgoFeeAddress: params.bitgoFeeAddress,\n            recoveryDestination: params.recoveryDestination,\n            eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },\n            gasLimit: 500000,\n            bitgoDestinationAddress: params.bitgoDestinationAddress,\n            intendedChain: params.intendedChain,\n          });\n        },\n        Error,\n        'user key is invalid'\n      );\n    });\n  });\n\n  describe('Build Unsigned Sweep for Self-Custody Cold Wallets (MPCv2)', function () {\n    const baseUrl = common.Environments.test.polygonscanBaseUrl as string;\n\n    it('should generate an unsigned sweep without derivation seed', async function () {\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getTxListRequest(mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2().address))\n        .reply(200, mockData.getTxListResponse);\n\n      nock(baseUrl)\n        .get('/api')\n        .query(mockData.getBalanceRequest(mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2().address))\n        .reply(200, mockData.getBalanceResponse);\n\n      nock(baseUrl)\n        .get('/api')\n        .query(\n          mockData.getBalanceRequest(\n            mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2().walletContractAddress\n          )\n        )\n        .reply(200, mockData.getBalanceResponse);\n\n      const params = mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2();\n      const sweepResult = await (basecoin as AbstractEthLikeNewCoins).recover({\n        userKey: params.commonKeyChain,\n        backupKey: params.commonKeyChain,\n        recoveryDestination: params.recoveryDestination,\n        gasLimit: 200000,\n        eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },\n        walletContractAddress: params.walletContractAddress,\n        isTss: true,\n        replayProtectionOptions: {\n          chain: '137',\n          hardfork: 'london',\n        },\n      });\n\n      should.exist(sweepResult);\n      const output = sweepResult as UnsignedSweepTxMPCv2;\n      output.should.have.property('txRequests');\n      output.txRequests.should.have.length(1);\n      output.txRequests[0].should.have.property('transactions');\n      output.txRequests[0].transactions.should.have.length(1);\n      output.txRequests[0].should.have.property('walletCoin');\n      output.txRequests[0].transactions[0].should.have.property('unsignedTx');\n      output.txRequests[0].transactions[0].unsignedTx.should.have.property('serializedTxHex');\n      output.txRequests[0].transactions[0].unsignedTx.should.have.property('signableHex');\n      output.txRequests[0].transactions[0].unsignedTx.should.have.property('derivationPath');\n      output.txRequests[0].transactions[0].unsignedTx.should.have.property('feeInfo');\n      output.txRequests[0].transactions[0].unsignedTx.should.have.property('parsedTx');\n      const parsedTx = output.txRequests[0].transactions[0].unsignedTx.parsedTx as { spendAmount: string };\n      parsedTx.should.have.property('spendAmount');\n      (output.txRequests[0].transactions[0].unsignedTx.parsedTx as { outputs: any[] }).should.have.property('outputs');\n    });\n\n    it('should throw an error for invalid address', async function () {\n      const params = mockData.getBuildUnsignedSweepForSelfCustodyColdWalletsMPCv2();\n      params.recoveryDestination = 'invalidAddress';\n\n      params.userKey =\n        '0234eb39b22fed523ece7c78da29ba1f1de5b64a6e48013e0914de793bc1df0570e779de04758732734d97e54b782c8b336283811af6a2c57bd81438798e1c2446';\n      params.backupKey =\n        '0234eb39b22fed523ece7c78da29ba1f1de5b64a6e48013e0914de793bc1df0570e779de04758732734d97e54b782c8b336283811af6a2c57bd81438798e1c2446';\n\n      await assert.rejects(\n        async () => {\n          await (basecoin as AbstractEthLikeNewCoins).recover({\n            recoveryDestination: params.recoveryDestination,\n            gasLimit: 2000,\n            eip1559: { maxFeePerGas: 200, maxPriorityFeePerGas: 10000 },\n            userKey: params.userKey,\n            backupKey: params.backupKey,\n            walletContractAddress: params.walletContractAddress,\n            isTss: true,\n            replayProtectionOptions: {\n              chain: '137',\n              hardfork: 'london',\n            },\n          });\n        },\n        Error,\n        'Error: invalid address'\n      );\n    });\n  });\n\n  describe('Test isWalletAddress', function () {\n    it('verify address for tpolygon', async function () {\n      const keychains = [\n        {\n          id: '6920c05e0b195abd8ed0bb0a5df32cdc',\n          source: 'user',\n          type: 'tss',\n          commonKeychain:\n            '02f929f92c25eaccb0e3ebaa0f5ed900c5ad30ba94f8444dc89c94e12f01aa5371522d810b8918ecc9e41eb901352df1c7977420fbaf9b8617f61b780b32b2ccad',\n        },\n        {\n          id: '6920c05e93c3b1c9e5006bf00a3cf016',\n          source: 'backup',\n          type: 'tss',\n          commonKeychain:\n            '02f929f92c25eaccb0e3ebaa0f5ed900c5ad30ba94f8444dc89c94e12f01aa5371522d810b8918ecc9e41eb901352df1c7977420fbaf9b8617f61b780b32b2ccad',\n        },\n        {\n          id: '6920c05d9ebee0100ec4a8aa7e300c02',\n          source: 'bitgo',\n          type: 'tss',\n          commonKeychain:\n            '02f929f92c25eaccb0e3ebaa0f5ed900c5ad30ba94f8444dc89c94e12f01aa5371522d810b8918ecc9e41eb901352df1c7977420fbaf9b8617f61b780b32b2ccad',\n        },\n      ];\n\n      const params = {\n        address: '0x4e9fc44697f4135455157396485f6fe8f909752f',\n        baseAddress: '0x8cf5ebd51585d159c4a1ca36178f9ad0fd7a594c',\n        coinSpecific: {\n          salt: '0xd',\n          forwarderVersion: 4,\n          feeAddress: '0x44dcb3504e323a3d70142036a99e2d4bba3f2270',\n        },\n        keychains,\n        index: 13,\n        walletVersion: 5,\n      };\n\n      const coin = bitgo.coin('tpolygon');\n      const isWalletAddr = await coin.isWalletAddress(params);\n      isWalletAddr.should.equal(true);\n    });\n  });\n});\n"]}
|