@bitgo-beta/abstract-eth 1.2.3-alpha.21 → 1.2.3-alpha.210
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +1429 -0
- package/dist/src/abstractEthLikeCoin.d.ts +13 -7
- package/dist/src/abstractEthLikeCoin.d.ts.map +1 -1
- package/dist/src/abstractEthLikeCoin.js +14 -11
- package/dist/src/abstractEthLikeNewCoins.d.ts +662 -0
- package/dist/src/abstractEthLikeNewCoins.d.ts.map +1 -0
- package/dist/src/abstractEthLikeNewCoins.js +1960 -0
- package/dist/src/ethLikeToken.d.ts +35 -5
- package/dist/src/ethLikeToken.d.ts.map +1 -1
- package/dist/src/ethLikeToken.js +281 -7
- package/dist/src/index.d.ts +2 -0
- package/dist/src/index.d.ts.map +1 -1
- package/dist/src/index.js +8 -2
- package/dist/src/lib/contractCall.d.ts +8 -0
- package/dist/src/lib/contractCall.d.ts.map +1 -0
- package/dist/src/lib/contractCall.js +17 -0
- package/dist/src/lib/iface.d.ts +132 -0
- package/dist/src/lib/iface.d.ts.map +1 -0
- package/dist/src/lib/iface.js +8 -0
- package/dist/src/lib/index.d.ts +15 -0
- package/dist/src/lib/index.d.ts.map +1 -0
- package/dist/src/lib/index.js +46 -0
- package/dist/src/lib/keyPair.d.ts +26 -0
- package/dist/src/lib/keyPair.d.ts.map +1 -0
- package/dist/src/lib/keyPair.js +66 -0
- package/dist/src/lib/transaction.d.ts +64 -0
- package/dist/src/lib/transaction.d.ts.map +1 -0
- package/dist/src/lib/transaction.js +137 -0
- package/dist/src/lib/transactionBuilder.d.ts +249 -0
- package/dist/src/lib/transactionBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder.js +732 -0
- package/dist/src/lib/transferBuilder.d.ts +71 -0
- package/dist/src/lib/transferBuilder.d.ts.map +1 -0
- package/dist/src/lib/transferBuilder.js +261 -0
- package/dist/src/lib/transferBuilders/baseNFTTransferBuilder.d.ts +54 -0
- package/dist/src/lib/transferBuilders/baseNFTTransferBuilder.d.ts.map +1 -0
- package/dist/src/lib/transferBuilders/baseNFTTransferBuilder.js +121 -0
- package/dist/src/lib/transferBuilders/index.d.ts +4 -0
- package/dist/src/lib/transferBuilders/index.d.ts.map +1 -0
- package/dist/src/lib/transferBuilders/index.js +20 -0
- package/dist/src/lib/transferBuilders/transferBuilderERC1155.d.ts +16 -0
- package/dist/src/lib/transferBuilders/transferBuilderERC1155.d.ts.map +1 -0
- package/dist/src/lib/transferBuilders/transferBuilderERC1155.js +93 -0
- package/dist/src/lib/transferBuilders/transferBuilderERC721.d.ts +15 -0
- package/dist/src/lib/transferBuilders/transferBuilderERC721.d.ts.map +1 -0
- package/dist/src/lib/transferBuilders/transferBuilderERC721.js +78 -0
- package/dist/src/lib/types.d.ts +39 -0
- package/dist/src/lib/types.d.ts.map +1 -0
- package/dist/src/lib/types.js +137 -0
- package/dist/src/lib/utils.d.ts +268 -0
- package/dist/src/lib/utils.d.ts.map +1 -0
- package/dist/src/lib/utils.js +689 -0
- package/dist/src/lib/walletUtil.d.ts +30 -0
- package/dist/src/lib/walletUtil.d.ts.map +1 -0
- package/dist/src/lib/walletUtil.js +33 -0
- package/dist/test/index.d.ts +2 -0
- package/dist/test/index.d.ts.map +1 -0
- package/dist/test/index.js +18 -0
- package/dist/test/unit/coin.d.ts +8 -0
- package/dist/test/unit/coin.d.ts.map +1 -0
- package/dist/test/unit/coin.js +568 -0
- package/dist/test/unit/index.d.ts +5 -0
- package/dist/test/unit/index.d.ts.map +1 -0
- package/dist/test/unit/index.js +21 -0
- package/dist/test/unit/token.d.ts +2 -0
- package/dist/test/unit/token.d.ts.map +1 -0
- package/dist/test/unit/token.js +38 -0
- package/dist/test/unit/transaction.d.ts +3 -0
- package/dist/test/unit/transaction.d.ts.map +1 -0
- package/dist/test/unit/transaction.js +61 -0
- package/dist/test/unit/transactionBuilder/addressInitialization.d.ts +8 -0
- package/dist/test/unit/transactionBuilder/addressInitialization.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/addressInitialization.js +96 -0
- package/dist/test/unit/transactionBuilder/index.d.ts +4 -0
- package/dist/test/unit/transactionBuilder/index.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/index.js +20 -0
- package/dist/test/unit/transactionBuilder/send.d.ts +3 -0
- package/dist/test/unit/transactionBuilder/send.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/send.js +188 -0
- package/dist/test/unit/transactionBuilder/walletInitialization.d.ts +10 -0
- package/dist/test/unit/transactionBuilder/walletInitialization.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/walletInitialization.js +125 -0
- package/dist/test/unit/transferBuilder.d.ts +2 -0
- package/dist/test/unit/transferBuilder.d.ts.map +1 -0
- package/dist/test/unit/transferBuilder.js +76 -0
- package/dist/tsconfig.tsbuildinfo +1 -8020
- package/index.ts +2 -0
- package/package.json +29 -9
|
@@ -0,0 +1,568 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
+
if (k2 === undefined) k2 = k;
|
|
4
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
+
}
|
|
8
|
+
Object.defineProperty(o, k2, desc);
|
|
9
|
+
}) : (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
o[k2] = m[k];
|
|
12
|
+
}));
|
|
13
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
14
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
15
|
+
}) : function(o, v) {
|
|
16
|
+
o["default"] = v;
|
|
17
|
+
});
|
|
18
|
+
var __importStar = (this && this.__importStar) || function (mod) {
|
|
19
|
+
if (mod && mod.__esModule) return mod;
|
|
20
|
+
var result = {};
|
|
21
|
+
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
|
|
22
|
+
__setModuleDefault(result, mod);
|
|
23
|
+
return result;
|
|
24
|
+
};
|
|
25
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
26
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
27
|
+
};
|
|
28
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
29
|
+
exports.runRecoveryTransactionTests = exports.runTransactionVerificationTests = exports.runSignTransactionTests = exports.runExplainTransactionTests = exports.runBasicCoinInfoTests = void 0;
|
|
30
|
+
const sdk_test_1 = require("@bitgo-beta/sdk-test");
|
|
31
|
+
const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
|
|
32
|
+
const secp256k1 = __importStar(require("secp256k1"));
|
|
33
|
+
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
34
|
+
const nock_1 = __importDefault(require("nock"));
|
|
35
|
+
const should = __importStar(require("should"));
|
|
36
|
+
const src_1 = require("../../src");
|
|
37
|
+
nock_1.default.enableNetConnect();
|
|
38
|
+
function runBasicCoinInfoTests(coinName, bitgo, MainCoin, TestCoin, testData) {
|
|
39
|
+
describe(`${coinName} basic info test`, () => {
|
|
40
|
+
const coinTest = testData.COIN;
|
|
41
|
+
const coinMain = coinTest.slice(1);
|
|
42
|
+
it(`should return the right info for ${coinMain}`, () => {
|
|
43
|
+
const coin = bitgo.coin(coinMain);
|
|
44
|
+
coin.should.be.an.instanceof(MainCoin);
|
|
45
|
+
coin.getChain().should.equal(coinMain);
|
|
46
|
+
coin.getFamily().should.equal(coinMain);
|
|
47
|
+
coin.getFullName().should.equal(testData.CHAIN_FULL_NAME);
|
|
48
|
+
coin.getBaseFactor().should.equal(1e18);
|
|
49
|
+
});
|
|
50
|
+
it(`should return the right info for ${coinTest}`, () => {
|
|
51
|
+
const coin = bitgo.coin(coinTest);
|
|
52
|
+
coin.should.be.an.instanceof(TestCoin);
|
|
53
|
+
coin.getChain().should.equal(coinTest);
|
|
54
|
+
coin.getFamily().should.equal(coinMain);
|
|
55
|
+
coin.getFullName().should.equal(`Testnet ${testData.CHAIN_FULL_NAME}`);
|
|
56
|
+
coin.getBaseFactor().should.equal(1e18);
|
|
57
|
+
});
|
|
58
|
+
});
|
|
59
|
+
}
|
|
60
|
+
exports.runBasicCoinInfoTests = runBasicCoinInfoTests;
|
|
61
|
+
function runExplainTransactionTests(coinName, txBuilder, basecoin, testData) {
|
|
62
|
+
describe(`${coinName} explain transaction`, () => {
|
|
63
|
+
const coinTest = testData.COIN;
|
|
64
|
+
/**
|
|
65
|
+
* Build an unsigned account-lib multi-signature send transactino
|
|
66
|
+
* @param destination The destination address of the transaction
|
|
67
|
+
* @param contractAddress The address of the smart contract processing the transaction
|
|
68
|
+
* @param contractSequenceId The sequence id of the contract
|
|
69
|
+
* @param nonce The nonce of the sending address
|
|
70
|
+
* @param expireTime The expire time of the transaction
|
|
71
|
+
* @param amount The amount to send to the recipient
|
|
72
|
+
* @param gasPrice The gas price of the transaction
|
|
73
|
+
* @param gasLimit The gas limit of the transaction
|
|
74
|
+
*/
|
|
75
|
+
const buildUnsignedTransaction = async function ({ destination, contractAddress, contractSequenceId = 1, nonce = 0, expireTime = Math.floor(new Date().getTime() / 1000), amount = '100000', gasPrice = '10000', gasLimit = '20000', }) {
|
|
76
|
+
txBuilder.type(sdk_core_1.TransactionType.Send);
|
|
77
|
+
txBuilder.fee({
|
|
78
|
+
fee: gasPrice,
|
|
79
|
+
gasLimit: gasLimit,
|
|
80
|
+
});
|
|
81
|
+
txBuilder.counter(nonce);
|
|
82
|
+
txBuilder.contract(contractAddress);
|
|
83
|
+
const transferBuilder = txBuilder.transfer();
|
|
84
|
+
transferBuilder
|
|
85
|
+
.coin(coinTest)
|
|
86
|
+
.expirationTime(expireTime)
|
|
87
|
+
.amount(amount)
|
|
88
|
+
.to(destination)
|
|
89
|
+
.contractSequenceId(contractSequenceId);
|
|
90
|
+
return await txBuilder.build();
|
|
91
|
+
};
|
|
92
|
+
it('should fail if the options object is missing parameters', async () => {
|
|
93
|
+
const explainParams = {
|
|
94
|
+
feeInfo: { fee: 1 },
|
|
95
|
+
txHex: null,
|
|
96
|
+
};
|
|
97
|
+
await basecoin.explainTransaction(explainParams).should.be.rejectedWith('missing explain tx parameters');
|
|
98
|
+
});
|
|
99
|
+
it('explain a transfer transaction', async () => {
|
|
100
|
+
const destination = '0xfaa8f14f46a99eb439c50e0c3b835cc21dad51b4';
|
|
101
|
+
const contractAddress = '0x9e2c5712ab4caf402a98c4bf58c79a0dfe718ad1';
|
|
102
|
+
const unsignedTransaction = await buildUnsignedTransaction({
|
|
103
|
+
destination,
|
|
104
|
+
contractAddress,
|
|
105
|
+
});
|
|
106
|
+
const explainParams = {
|
|
107
|
+
halfSigned: {
|
|
108
|
+
txHex: unsignedTransaction.toBroadcastFormat(),
|
|
109
|
+
},
|
|
110
|
+
feeInfo: { fee: 1 },
|
|
111
|
+
};
|
|
112
|
+
const explanation = await basecoin.explainTransaction(explainParams);
|
|
113
|
+
should.exist(explanation.id);
|
|
114
|
+
});
|
|
115
|
+
});
|
|
116
|
+
}
|
|
117
|
+
exports.runExplainTransactionTests = runExplainTransactionTests;
|
|
118
|
+
function runSignTransactionTests(coinName, builder, basecoin, testData) {
|
|
119
|
+
describe(`${coinName} sign transaction tests`, async () => {
|
|
120
|
+
const account_1 = {
|
|
121
|
+
address: '0x8Ce59c2d1702844F8EdED451AA103961bC37B4e8',
|
|
122
|
+
owner_1: '4ee089aceabf3ddbf748db79b1066c33b7d3ea1ab3eb7e325121bba2bff2f5ca',
|
|
123
|
+
owner_2: '5c7e4efff7304d4dfff6d5f1591844ec6f2adfa6a47e9fece6a3c1a4d755f1e3',
|
|
124
|
+
owner_3: '4421ab25dd91e1a3180d03d57c323a7886dcc313d3b3a4b4256a5791572bf597',
|
|
125
|
+
};
|
|
126
|
+
const account_2 = {
|
|
127
|
+
address: '0xeeaf0F05f37891ab4a21208B105A0687d12c5aF7',
|
|
128
|
+
owner_1: '4ee089aceabf3ddbf748db79b1066c33b7d3ea1ab3eb7e325121bba2bff2f5ca',
|
|
129
|
+
owner_2: '5ca116d25aec5f765465432cc421ff25ef9ffdc330b10bb3d9ad61e3baad88d7',
|
|
130
|
+
owner_3: '1fae946cc84af8bd74d610a88537e24e19c3349d478d86fc5bb59ba4c88fb9cc',
|
|
131
|
+
};
|
|
132
|
+
const coinTest = testData.COIN;
|
|
133
|
+
it('should sign an unsigned test tx', async () => {
|
|
134
|
+
builder.fee({
|
|
135
|
+
fee: '280000000000',
|
|
136
|
+
gasLimit: '7000000',
|
|
137
|
+
});
|
|
138
|
+
builder.counter(1);
|
|
139
|
+
builder.type(sdk_core_1.TransactionType.Send);
|
|
140
|
+
builder.contract(account_1.address);
|
|
141
|
+
const transferBuilder = builder.transfer();
|
|
142
|
+
transferBuilder.coin(coinTest).amount('1').to(account_2.address).expirationTime(10000).contractSequenceId(1);
|
|
143
|
+
const unsignedTx = await builder.build();
|
|
144
|
+
const unsignedTxForBroadcasting = unsignedTx.toBroadcastFormat();
|
|
145
|
+
const halfSignedRawTx = await basecoin.signTransaction({
|
|
146
|
+
txPrebuild: {
|
|
147
|
+
txHex: unsignedTxForBroadcasting,
|
|
148
|
+
},
|
|
149
|
+
prv: account_1.owner_2,
|
|
150
|
+
});
|
|
151
|
+
builder.transfer().key(account_1.owner_2);
|
|
152
|
+
const halfSignedTx = await builder.build();
|
|
153
|
+
const halfSignedTxForBroadcasting = halfSignedTx.toBroadcastFormat();
|
|
154
|
+
halfSignedRawTx.halfSigned.txHex.should.equals(halfSignedTxForBroadcasting);
|
|
155
|
+
halfSignedRawTx.halfSigned.recipients.length.should.equals(1);
|
|
156
|
+
halfSignedRawTx.halfSigned.recipients[0].address.toLowerCase().should.equals(account_2.address.toLowerCase());
|
|
157
|
+
halfSignedRawTx.halfSigned.recipients[0].amount.toLowerCase().should.equals('1');
|
|
158
|
+
});
|
|
159
|
+
it('should sign an unsigned test tx with eip1559', async () => {
|
|
160
|
+
builder.fee({
|
|
161
|
+
fee: '280000000000',
|
|
162
|
+
gasLimit: '7000000',
|
|
163
|
+
eip1559: {
|
|
164
|
+
maxFeePerGas: '7593123',
|
|
165
|
+
maxPriorityFeePerGas: '150',
|
|
166
|
+
},
|
|
167
|
+
});
|
|
168
|
+
builder.counter(1);
|
|
169
|
+
builder.type(sdk_core_1.TransactionType.Send);
|
|
170
|
+
builder.contract(account_1.address);
|
|
171
|
+
const transferBuilder = builder.transfer();
|
|
172
|
+
transferBuilder.coin(coinTest).amount('1').to(account_2.address).expirationTime(10000).contractSequenceId(1);
|
|
173
|
+
const unsignedTx = await builder.build();
|
|
174
|
+
const unsignedTxForBroadcasting = unsignedTx.toBroadcastFormat();
|
|
175
|
+
const halfSignedRawTx = await basecoin.signTransaction({
|
|
176
|
+
txPrebuild: {
|
|
177
|
+
txHex: unsignedTxForBroadcasting,
|
|
178
|
+
eip1559: {
|
|
179
|
+
maxFeePerGas: '7593123',
|
|
180
|
+
maxPriorityFeePerGas: '150',
|
|
181
|
+
},
|
|
182
|
+
},
|
|
183
|
+
prv: account_1.owner_2,
|
|
184
|
+
});
|
|
185
|
+
builder.transfer().key(account_1.owner_2);
|
|
186
|
+
const halfSignedTx = await builder.build();
|
|
187
|
+
const halfSignedTxForBroadcasting = halfSignedTx.toBroadcastFormat();
|
|
188
|
+
halfSignedRawTx.halfSigned.txHex.should.equals(halfSignedTxForBroadcasting);
|
|
189
|
+
halfSignedRawTx.halfSigned.recipients.length.should.equals(1);
|
|
190
|
+
halfSignedRawTx.halfSigned.recipients[0].address.toLowerCase().should.equals(account_2.address.toLowerCase());
|
|
191
|
+
halfSignedRawTx.halfSigned.recipients[0].amount.toLowerCase().should.equals('1');
|
|
192
|
+
halfSignedRawTx.halfSigned.eip1559.maxFeePerGas.should.equal('7593123');
|
|
193
|
+
halfSignedRawTx.halfSigned.eip1559.maxPriorityFeePerGas.should.equal('150');
|
|
194
|
+
});
|
|
195
|
+
});
|
|
196
|
+
}
|
|
197
|
+
exports.runSignTransactionTests = runSignTransactionTests;
|
|
198
|
+
function runTransactionVerificationTests(coinName, bitgo, basecoin, testData) {
|
|
199
|
+
describe(`${coinName} Transaction Verification`, () => {
|
|
200
|
+
const coinTest = testData.COIN;
|
|
201
|
+
const address1 = '0x174cfd823af8ce27ed0afee3fcf3c3ba259116be';
|
|
202
|
+
const address2 = '0x7e85bdc27c050e3905ebf4b8e634d9ad6edd0de6';
|
|
203
|
+
const hopDestinationAddress = '0x9c7e8ce6825bD48278B3Ab59228EE26f8BE7925b';
|
|
204
|
+
const hopTx = '0xf86b808504a817c8ff8252ff949c7e8ce6825bd48278b3ab59228ee26f8be7925b87038d7ea4c68000801ca011bc22c664570133dfca4f08a0b8d02339cf467046d6a4152f04f368d0eaf99ea01d6dc5cf0c897c8d4c3e1df53d0d042784c424536a4cc5b802552b7d64fee8b5';
|
|
205
|
+
const hopTxid = '0x4af65143bc77da2b50f35b3d13cacb4db18f026bf84bc0743550bc57b9b53351';
|
|
206
|
+
const userReqSig = '0x404db307f6147f0d8cd338c34c13906ef46a6faa7e0e119d5194ef05aec16e6f3d710f9b7901460f97e924066b62efd74443bd34402c6d40b49c203a559ff2c8';
|
|
207
|
+
const bitgoKeyXprv = 'xprv9s21ZrQH143K3tpWBHWe31sLoXNRQ9AvRYJgitkKxQ4ATFQMwvr7hHNqYRUnS7PsjzB7aK1VxqHLuNQjj1sckJ2Jwo2qxmsvejwECSpFMfC';
|
|
208
|
+
const bitgoKey = utxo_lib_1.bip32.fromBase58(bitgoKeyXprv);
|
|
209
|
+
if (!bitgoKey.privateKey) {
|
|
210
|
+
throw new Error('no privateKey');
|
|
211
|
+
}
|
|
212
|
+
const hopTxBitgoSignature = '0xaa' +
|
|
213
|
+
Buffer.from(secp256k1.ecdsaSign(Buffer.from(hopTxid.slice(2), 'hex'), bitgoKey.privateKey).signature).toString('hex');
|
|
214
|
+
it('should verify a normal txPrebuild from the bitgo server that matches the client txParams', async () => {
|
|
215
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
216
|
+
const txParams = {
|
|
217
|
+
recipients: [{ amount: '1000000000000', address: address1 }],
|
|
218
|
+
wallet: wallet,
|
|
219
|
+
walletPassphrase: 'fakeWalletPassphrase',
|
|
220
|
+
};
|
|
221
|
+
const txPrebuild = {
|
|
222
|
+
recipients: [{ amount: '1000000000000', address: address1 }],
|
|
223
|
+
nextContractSequenceId: 0,
|
|
224
|
+
gasPrice: 20000000000,
|
|
225
|
+
gasLimit: 500000,
|
|
226
|
+
isBatch: false,
|
|
227
|
+
coin: coinTest,
|
|
228
|
+
walletId: 'fakeWalletId',
|
|
229
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
230
|
+
};
|
|
231
|
+
const verification = {};
|
|
232
|
+
const isTransactionVerified = await basecoin.verifyTransaction({ txParams, txPrebuild, wallet, verification });
|
|
233
|
+
isTransactionVerified.should.equal(true);
|
|
234
|
+
});
|
|
235
|
+
it('should reject when client txParams are missing', async () => {
|
|
236
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
237
|
+
const txParams = null;
|
|
238
|
+
const txPrebuild = {
|
|
239
|
+
recipients: [{ amount: '1000000000000', address: address1 }],
|
|
240
|
+
nextContractSequenceId: 0,
|
|
241
|
+
gasPrice: 20000000000,
|
|
242
|
+
gasLimit: 500000,
|
|
243
|
+
isBatch: false,
|
|
244
|
+
coin: coinTest,
|
|
245
|
+
walletId: 'fakeWalletId',
|
|
246
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
247
|
+
};
|
|
248
|
+
const verification = {};
|
|
249
|
+
await basecoin
|
|
250
|
+
.verifyTransaction({ txParams, txPrebuild, wallet, verification })
|
|
251
|
+
.should.be.rejectedWith('missing params');
|
|
252
|
+
});
|
|
253
|
+
it('should reject txPrebuild that is both batch and hop', async () => {
|
|
254
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
255
|
+
const txParams = {
|
|
256
|
+
recipients: [
|
|
257
|
+
{ amount: '1000000000000', address: address1 },
|
|
258
|
+
{ amount: '2500000000000', address: address2 },
|
|
259
|
+
],
|
|
260
|
+
wallet: wallet,
|
|
261
|
+
walletPassphrase: 'fakeWalletPassphrase',
|
|
262
|
+
hop: true,
|
|
263
|
+
};
|
|
264
|
+
const txPrebuild = {
|
|
265
|
+
recipients: [{ amount: '3500000000000', address: address1 }],
|
|
266
|
+
nextContractSequenceId: 0,
|
|
267
|
+
gasPrice: 20000000000,
|
|
268
|
+
gasLimit: 500000,
|
|
269
|
+
isBatch: true,
|
|
270
|
+
coin: coinTest,
|
|
271
|
+
walletId: 'fakeWalletId',
|
|
272
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
273
|
+
hopTransaction: {
|
|
274
|
+
tx: hopTx,
|
|
275
|
+
id: hopTxid,
|
|
276
|
+
signature: hopTxBitgoSignature,
|
|
277
|
+
paymentId: '2773928196',
|
|
278
|
+
gasPrice: 20000000000,
|
|
279
|
+
gasLimit: 500000,
|
|
280
|
+
amount: '1000000000000000',
|
|
281
|
+
recipient: hopDestinationAddress,
|
|
282
|
+
nonce: 0,
|
|
283
|
+
userReqSig: userReqSig,
|
|
284
|
+
gasPriceMax: 500000000000,
|
|
285
|
+
},
|
|
286
|
+
};
|
|
287
|
+
const verification = {};
|
|
288
|
+
await basecoin
|
|
289
|
+
.verifyTransaction({ txParams, txPrebuild, wallet, verification })
|
|
290
|
+
.should.be.rejectedWith('tx cannot be both a batch and hop transaction');
|
|
291
|
+
});
|
|
292
|
+
it('should reject a txPrebuild with more than one recipient', async () => {
|
|
293
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
294
|
+
const txParams = {
|
|
295
|
+
recipients: [
|
|
296
|
+
{ amount: '1000000000000', address: address1 },
|
|
297
|
+
{ amount: '2500000000000', address: address2 },
|
|
298
|
+
],
|
|
299
|
+
wallet: wallet,
|
|
300
|
+
walletPassphrase: 'fakeWalletPassphrase',
|
|
301
|
+
};
|
|
302
|
+
const txPrebuild = {
|
|
303
|
+
recipients: [
|
|
304
|
+
{ amount: '1000000000000', address: address1 },
|
|
305
|
+
{ amount: '2500000000000', address: address2 },
|
|
306
|
+
],
|
|
307
|
+
nextContractSequenceId: 0,
|
|
308
|
+
gasPrice: 20000000000,
|
|
309
|
+
gasLimit: 500000,
|
|
310
|
+
isBatch: true,
|
|
311
|
+
coin: coinTest,
|
|
312
|
+
walletId: 'fakeWalletId',
|
|
313
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
314
|
+
};
|
|
315
|
+
const verification = {};
|
|
316
|
+
await basecoin
|
|
317
|
+
.verifyTransaction({ txParams, txPrebuild, wallet, verification })
|
|
318
|
+
.should.be.rejectedWith('txPrebuild should only have 1 recipient but 2 found');
|
|
319
|
+
});
|
|
320
|
+
it('should reject a hop txPrebuild that does not send to its hop address', async () => {
|
|
321
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
322
|
+
const txParams = {
|
|
323
|
+
recipients: [{ amount: '1000000000000000', address: hopDestinationAddress }],
|
|
324
|
+
wallet: wallet,
|
|
325
|
+
walletPassphrase: 'fakeWalletPassphrase',
|
|
326
|
+
hop: true,
|
|
327
|
+
};
|
|
328
|
+
const txPrebuild = {
|
|
329
|
+
recipients: [{ amount: '5000000000000000', address: address1 }],
|
|
330
|
+
nextContractSequenceId: 0,
|
|
331
|
+
gasPrice: 20000000000,
|
|
332
|
+
gasLimit: 500000,
|
|
333
|
+
isBatch: false,
|
|
334
|
+
coin: coinTest,
|
|
335
|
+
walletId: 'fakeWalletId',
|
|
336
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
337
|
+
hopTransaction: {
|
|
338
|
+
tx: hopTx,
|
|
339
|
+
id: hopTxid,
|
|
340
|
+
signature: hopTxBitgoSignature,
|
|
341
|
+
paymentId: '0',
|
|
342
|
+
gasPrice: 20000000000,
|
|
343
|
+
gasLimit: 500000,
|
|
344
|
+
amount: '1000000000000000',
|
|
345
|
+
recipient: hopDestinationAddress,
|
|
346
|
+
nonce: 0,
|
|
347
|
+
userReqSig: userReqSig,
|
|
348
|
+
gasPriceMax: 500000000000,
|
|
349
|
+
},
|
|
350
|
+
};
|
|
351
|
+
const verification = {};
|
|
352
|
+
await basecoin
|
|
353
|
+
.verifyTransaction({ txParams, txPrebuild, wallet, verification })
|
|
354
|
+
.should.be.rejectedWith('recipient address of txPrebuild does not match hop address');
|
|
355
|
+
});
|
|
356
|
+
it('should reject a normal txPrebuild from the bitgo server with the wrong amount', async () => {
|
|
357
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
358
|
+
const txParams = {
|
|
359
|
+
recipients: [{ amount: '1000000000000', address: address1 }],
|
|
360
|
+
wallet: wallet,
|
|
361
|
+
walletPassphrase: 'fakeWalletPassphrase',
|
|
362
|
+
};
|
|
363
|
+
const txPrebuild = {
|
|
364
|
+
recipients: [{ amount: '2000000000000', address: address1 }],
|
|
365
|
+
nextContractSequenceId: 0,
|
|
366
|
+
gasPrice: 20000000000,
|
|
367
|
+
gasLimit: 500000,
|
|
368
|
+
isBatch: false,
|
|
369
|
+
coin: coinTest,
|
|
370
|
+
walletId: 'fakeWalletId',
|
|
371
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
372
|
+
};
|
|
373
|
+
const verification = {};
|
|
374
|
+
await basecoin
|
|
375
|
+
.verifyTransaction({ txParams, txPrebuild, wallet, verification })
|
|
376
|
+
.should.be.rejectedWith('normal transaction amount in txPrebuild received from BitGo servers does not match txParams supplied by client');
|
|
377
|
+
});
|
|
378
|
+
it('should reject a normal txPrebuild from the bitgo server with the wrong recipient', async () => {
|
|
379
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
380
|
+
const txParams = {
|
|
381
|
+
recipients: [{ amount: '1000000000000', address: address1 }],
|
|
382
|
+
wallet: wallet,
|
|
383
|
+
walletPassphrase: 'fakeWalletPassphrase',
|
|
384
|
+
};
|
|
385
|
+
const txPrebuild = {
|
|
386
|
+
recipients: [{ amount: '1000000000000', address: address2 }],
|
|
387
|
+
nextContractSequenceId: 0,
|
|
388
|
+
gasPrice: 20000000000,
|
|
389
|
+
gasLimit: 500000,
|
|
390
|
+
isBatch: false,
|
|
391
|
+
coin: coinTest,
|
|
392
|
+
walletId: 'fakeWalletId',
|
|
393
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
394
|
+
};
|
|
395
|
+
const verification = {};
|
|
396
|
+
await basecoin
|
|
397
|
+
.verifyTransaction({ txParams, txPrebuild, wallet, verification })
|
|
398
|
+
.should.be.rejectedWith('destination address in normal txPrebuild does not match that in txParams supplied by client');
|
|
399
|
+
});
|
|
400
|
+
it('should reject a txPrebuild from the bitgo server with the wrong coin', async () => {
|
|
401
|
+
const wallet = new sdk_core_1.Wallet(bitgo, basecoin, {});
|
|
402
|
+
const txParams = {
|
|
403
|
+
recipients: [{ amount: '1000000000000', address: address1 }],
|
|
404
|
+
wallet: wallet,
|
|
405
|
+
walletPassphrase: 'fakeWalletPassphrase',
|
|
406
|
+
};
|
|
407
|
+
const txPrebuild = {
|
|
408
|
+
recipients: [{ amount: '1000000000000', address: address1 }],
|
|
409
|
+
nextContractSequenceId: 0,
|
|
410
|
+
gasPrice: 20000000000,
|
|
411
|
+
gasLimit: 500000,
|
|
412
|
+
isBatch: false,
|
|
413
|
+
coin: 'btc',
|
|
414
|
+
walletId: 'fakeWalletId',
|
|
415
|
+
walletContractAddress: 'fakeWalletContractAddress',
|
|
416
|
+
};
|
|
417
|
+
const verification = {};
|
|
418
|
+
await basecoin
|
|
419
|
+
.verifyTransaction({ txParams, txPrebuild, wallet, verification })
|
|
420
|
+
.should.be.rejectedWith('coin in txPrebuild did not match that in txParams supplied by client');
|
|
421
|
+
});
|
|
422
|
+
});
|
|
423
|
+
}
|
|
424
|
+
exports.runTransactionVerificationTests = runTransactionVerificationTests;
|
|
425
|
+
function runRecoveryTransactionTests(coinName, txBuilder, bitgo, testData, mockData) {
|
|
426
|
+
describe(`${coinName} Recover transaction:`, function () {
|
|
427
|
+
const baseUrl = testData.BASE_URL;
|
|
428
|
+
const userXpub = 'xpub661MyMwAqRbcEeTc8789MK5PUGEYiPG4F4V17n2Rd2LoTATA1XoCnJT5FAYAShQxSxtFjpo5NHmcWwTp2LiWGBMwpUcAA3HywhxivgYfq7q';
|
|
429
|
+
const userXprv = 'xprv9s21ZrQH143K2AP925b8zB8evEQ4JvYCsqZQKPcp4gopaN81TzUxEW8bPtVyDgjmddGhRRETn8xi1cVAB9bf1Bx9kGRRFgTZXxJayZLnag1';
|
|
430
|
+
const backupXpub = 'xpub661MyMwAqRbcFZX15xpZf4ERCGHiVSJm8r5C4yh1yXV2GrdZCUPYo4WQr6tN9oUywKXsgSHo7Risf9r22GH5joVD2hEEEhqnSCvK8qy11wW';
|
|
431
|
+
const backupXprv = 'xprv9s21ZrQH143K35SXywHZHvHgeETE5yaumd9bGbHQRBx3Q4JQew5JFGBvzqiZjCUkBdBUZnfuMDTGURRayN1hFSWxEJQsCEAMm1D3pk1h7Jj';
|
|
432
|
+
const coin = testData.COIN;
|
|
433
|
+
after(function () {
|
|
434
|
+
nock_1.default.cleanAll();
|
|
435
|
+
});
|
|
436
|
+
it('should generate an unsigned sweep', async () => {
|
|
437
|
+
const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
|
|
438
|
+
const backupKeyAddress = '0x4f2c4830cc37f2785c646f89ded8a919219fa0e9';
|
|
439
|
+
(0, nock_1.default)(baseUrl)
|
|
440
|
+
.get('/api')
|
|
441
|
+
.twice()
|
|
442
|
+
.query(mockData.getTxListRequest(backupKeyAddress))
|
|
443
|
+
.reply(200, mockData.getTxListResponse);
|
|
444
|
+
(0, nock_1.default)(baseUrl)
|
|
445
|
+
.get('/api')
|
|
446
|
+
.query(mockData.getBalanceRequest(walletContractAddress))
|
|
447
|
+
.reply(200, mockData.getBalanceResponse);
|
|
448
|
+
(0, nock_1.default)(baseUrl)
|
|
449
|
+
.get('/api')
|
|
450
|
+
.query(mockData.getBalanceRequest(backupKeyAddress))
|
|
451
|
+
.reply(200, mockData.getBalanceResponse);
|
|
452
|
+
(0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
|
|
453
|
+
const basecoin = bitgo.coin(coin);
|
|
454
|
+
const transaction = (await basecoin.recover({
|
|
455
|
+
userKey: userXpub,
|
|
456
|
+
backupKey: backupXpub,
|
|
457
|
+
walletContractAddress: walletContractAddress,
|
|
458
|
+
recoveryDestination: sdk_test_1.TestBitGo.V2.TEST_ERC20_TOKEN_RECIPIENT,
|
|
459
|
+
eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
|
|
460
|
+
gasLimit: 500000,
|
|
461
|
+
}));
|
|
462
|
+
should.exist(transaction);
|
|
463
|
+
transaction.should.have.property('txHex');
|
|
464
|
+
transaction.should.have.property('contractSequenceId');
|
|
465
|
+
transaction.should.have.property('expireTime');
|
|
466
|
+
transaction.should.have.property('gasLimit');
|
|
467
|
+
transaction.gasLimit.should.equal('500000');
|
|
468
|
+
transaction.should.have.property('walletContractAddress');
|
|
469
|
+
transaction.walletContractAddress.should.equal(sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS);
|
|
470
|
+
transaction.should.have.property('recipients');
|
|
471
|
+
const recipient = transaction.recipients[0];
|
|
472
|
+
recipient.should.have.property('address');
|
|
473
|
+
recipient.address.should.equal(sdk_test_1.TestBitGo.V2.TEST_ERC20_TOKEN_RECIPIENT);
|
|
474
|
+
recipient.should.have.property('amount');
|
|
475
|
+
recipient.amount.should.equal('9999999999999999928');
|
|
476
|
+
});
|
|
477
|
+
it('should construct a recovery transaction without BitGo', async () => {
|
|
478
|
+
const backupKeyAddress = '0x6d22efdd634996248170c948e5726007fc251bb3';
|
|
479
|
+
const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
|
|
480
|
+
(0, nock_1.default)(baseUrl)
|
|
481
|
+
.get('/api')
|
|
482
|
+
.query(mockData.getTxListRequest(backupKeyAddress))
|
|
483
|
+
.reply(200, mockData.getTxListResponse);
|
|
484
|
+
(0, nock_1.default)(baseUrl)
|
|
485
|
+
.get('/api')
|
|
486
|
+
.query(mockData.getBalanceRequest(walletContractAddress))
|
|
487
|
+
.reply(200, mockData.getBalanceResponse);
|
|
488
|
+
(0, nock_1.default)(baseUrl)
|
|
489
|
+
.get('/api')
|
|
490
|
+
.query(mockData.getBalanceRequest(backupKeyAddress))
|
|
491
|
+
.reply(200, mockData.getBalanceResponse);
|
|
492
|
+
(0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
|
|
493
|
+
const basecoin = bitgo.coin(coin);
|
|
494
|
+
const transaction = (await basecoin.recover({
|
|
495
|
+
userKey: '{"iv":"VFZ3jvXhxo1Z+Yaf2MtZnA==","v":1,"iter":10000,"ks":256,"ts":64,"mode"\n' +
|
|
496
|
+
':"ccm","adata":"","cipher":"aes","salt":"p+fkHuLa/8k=","ct":"hYG7pvljLIgCjZ\n' +
|
|
497
|
+
'53PBlCde5KZRmlUKKHLtDMk+HJfuU46hW+x+C9WsIAO4gFPnTCvFVmQ8x7czCtcNFub5AO2otOG\n' +
|
|
498
|
+
'OsX4GE2gXOEmCl1TpWwwNhm7yMUjGJUpgW6ZZgXSXdDitSKi4V/hk78SGSzjFOBSPYRa6I="}\n',
|
|
499
|
+
backupKey: '{"iv":"AbsCtv1qwPIhOgyrCpNagA==","v":1,"iter":10000,"ks":256,"ts":64,"mode"\n' +
|
|
500
|
+
':"ccm","adata":"","cipher":"aes","salt":"5vpUDBUlzm8=","ct":"PapYYCjBXRLUKA\n' +
|
|
501
|
+
'JbOsB/EJ9B8fUmVQTxMPjUnQyAky12me9K66GiMEAxTD7kd6bYAQJuuTkATXKU7Bnf7vK9JxNOw\n' +
|
|
502
|
+
'oji7HF9eFH0aD4/hX5SWFfHF2Qfi+TnXv6hVsMAoisDZs3/F67/ZUaDYR0ZsdrQ4Q/cLD0="}\n',
|
|
503
|
+
walletContractAddress: walletContractAddress,
|
|
504
|
+
walletPassphrase: sdk_test_1.TestBitGo.V2.TEST_RECOVERY_PASSCODE,
|
|
505
|
+
recoveryDestination: sdk_test_1.TestBitGo.V2.TEST_ERC20_TOKEN_RECIPIENT,
|
|
506
|
+
gasLimit: 500000,
|
|
507
|
+
}));
|
|
508
|
+
should.exist(transaction);
|
|
509
|
+
transaction.should.have.property('tx');
|
|
510
|
+
transaction.should.have.property('id');
|
|
511
|
+
const decodedTx = src_1.optionalDeps.EthTx.Transaction.fromSerializedTx(src_1.optionalDeps.ethUtil.toBuffer(transaction.tx));
|
|
512
|
+
decodedTx.should.have.property('gasPrice');
|
|
513
|
+
decodedTx.should.have.property('nonce');
|
|
514
|
+
decodedTx.should.have.property('to');
|
|
515
|
+
});
|
|
516
|
+
it('should be able to second sign', async () => {
|
|
517
|
+
const walletContractAddress = sdk_test_1.TestBitGo.V2.TEST_ETH_WALLET_FIRST_ADDRESS;
|
|
518
|
+
const backupKeyAddress = '0x4f2c4830cc37f2785c646f89ded8a919219fa0e9';
|
|
519
|
+
(0, nock_1.default)(baseUrl)
|
|
520
|
+
.get('/api')
|
|
521
|
+
.twice()
|
|
522
|
+
.query(mockData.getTxListRequest(backupKeyAddress))
|
|
523
|
+
.reply(200, mockData.getTxListResponse);
|
|
524
|
+
(0, nock_1.default)(baseUrl)
|
|
525
|
+
.get('/api')
|
|
526
|
+
.query(mockData.getBalanceRequest(walletContractAddress))
|
|
527
|
+
.reply(200, mockData.getBalanceResponse);
|
|
528
|
+
(0, nock_1.default)(baseUrl)
|
|
529
|
+
.get('/api')
|
|
530
|
+
.query(mockData.getBalanceRequest(backupKeyAddress))
|
|
531
|
+
.reply(200, mockData.getBalanceResponse);
|
|
532
|
+
(0, nock_1.default)(baseUrl).get('/api').query(mockData.getContractCallRequest).reply(200, mockData.getContractCallResponse);
|
|
533
|
+
const basecoin = bitgo.coin(coin);
|
|
534
|
+
const transaction = (await basecoin.recover({
|
|
535
|
+
userKey: userXpub,
|
|
536
|
+
backupKey: backupXpub,
|
|
537
|
+
walletContractAddress: walletContractAddress,
|
|
538
|
+
recoveryDestination: sdk_test_1.TestBitGo.V2.TEST_ERC20_TOKEN_RECIPIENT,
|
|
539
|
+
eip1559: { maxFeePerGas: 20000000000, maxPriorityFeePerGas: 10000000000 },
|
|
540
|
+
replayProtectionOptions: { chain: 80001, hardfork: 'london' },
|
|
541
|
+
gasLimit: 500000,
|
|
542
|
+
}));
|
|
543
|
+
const txPrebuild = {
|
|
544
|
+
txHex: transaction.txHex,
|
|
545
|
+
};
|
|
546
|
+
const params = {
|
|
547
|
+
txPrebuild,
|
|
548
|
+
prv: userXprv,
|
|
549
|
+
};
|
|
550
|
+
// sign transaction once
|
|
551
|
+
const halfSigned = await basecoin.signTransaction(params);
|
|
552
|
+
const halfSignedParams = {
|
|
553
|
+
txPrebuild: halfSigned,
|
|
554
|
+
isLastSignature: true,
|
|
555
|
+
walletContractAddress: walletContractAddress,
|
|
556
|
+
prv: backupXprv,
|
|
557
|
+
};
|
|
558
|
+
const finalSigned = (await basecoin.signTransaction(halfSignedParams));
|
|
559
|
+
finalSigned.should.have.property('txHex');
|
|
560
|
+
txBuilder.from(finalSigned.txHex);
|
|
561
|
+
const rebuiltTx = await txBuilder.build();
|
|
562
|
+
rebuiltTx.signature.length.should.equal(2);
|
|
563
|
+
rebuiltTx.outputs.length.should.equal(1);
|
|
564
|
+
});
|
|
565
|
+
});
|
|
566
|
+
}
|
|
567
|
+
exports.runRecoveryTransactionTests = runRecoveryTransactionTests;
|
|
568
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"coin.js","sourceRoot":"","sources":["../../../test/unit/coin.ts"],"names":[],"mappings":"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","sourcesContent":["import { TestBitGo, TestBitGoAPI } from '@bitgo-beta/sdk-test';\nimport { bip32 } from '@bitgo-beta/utxo-lib';\nimport * as secp256k1 from 'secp256k1';\nimport { FullySignedTransaction, TransactionType, Wallet } from '@bitgo-beta/sdk-core';\nimport nock from 'nock';\nimport * as should from 'should';\nimport {\n  OfflineVaultTxInfo,\n  optionalDeps,\n  SignTransactionOptions,\n  TransferBuilder,\n  TransactionBuilder,\n} from '../../src';\n\nnock.enableNetConnect();\n\nexport function runBasicCoinInfoTests(coinName: string, bitgo: TestBitGoAPI, MainCoin, TestCoin, testData: any) {\n  describe(`${coinName} basic info test`, () => {\n    const coinTest = testData.COIN;\n    const coinMain = coinTest.slice(1);\n    it(`should return the right info for ${coinMain}`, () => {\n      const coin = bitgo.coin(coinMain);\n\n      coin.should.be.an.instanceof(MainCoin);\n      coin.getChain().should.equal(coinMain);\n      coin.getFamily().should.equal(coinMain);\n      coin.getFullName().should.equal(testData.CHAIN_FULL_NAME);\n      coin.getBaseFactor().should.equal(1e18);\n    });\n\n    it(`should return the right info for ${coinTest}`, () => {\n      const coin = bitgo.coin(coinTest);\n\n      coin.should.be.an.instanceof(TestCoin);\n      coin.getChain().should.equal(coinTest);\n      coin.getFamily().should.equal(coinMain);\n      coin.getFullName().should.equal(`Testnet ${testData.CHAIN_FULL_NAME}`);\n      coin.getBaseFactor().should.equal(1e18);\n    });\n  });\n}\n\nexport function runExplainTransactionTests(coinName: string, txBuilder: TransactionBuilder, basecoin, testData: any) {\n  describe(`${coinName} explain transaction`, () => {\n    const coinTest = testData.COIN;\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      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(coinTest)\n        .expirationTime(expireTime)\n        .amount(amount)\n        .to(destination)\n        .contractSequenceId(contractSequenceId);\n\n      return await txBuilder.build();\n    };\n    it('should fail if the options object is missing parameters', async () => {\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 () => {\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\nexport function runSignTransactionTests(coinName: string, builder: TransactionBuilder, basecoin, testData: any) {\n  describe(`${coinName} sign transaction tests`, async () => {\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    const coinTest = testData.COIN;\n\n    it('should sign an unsigned test tx', async () => {\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      const transferBuilder = builder.transfer() as TransferBuilder;\n      transferBuilder.coin(coinTest).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 () => {\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      const transferBuilder = builder.transfer() as TransferBuilder;\n      transferBuilder.coin(coinTest).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}\n\nexport function runTransactionVerificationTests(coinName: string, bitgo: TestBitGoAPI, basecoin, testData: any) {\n  describe(`${coinName} Transaction Verification`, () => {\n    const coinTest = testData.COIN;\n    const address1 = '0x174cfd823af8ce27ed0afee3fcf3c3ba259116be';\n    const address2 = '0x7e85bdc27c050e3905ebf4b8e634d9ad6edd0de6';\n    const hopDestinationAddress = '0x9c7e8ce6825bD48278B3Ab59228EE26f8BE7925b';\n    const hopTx =\n      '0xf86b808504a817c8ff8252ff949c7e8ce6825bd48278b3ab59228ee26f8be7925b87038d7ea4c68000801ca011bc22c664570133dfca4f08a0b8d02339cf467046d6a4152f04f368d0eaf99ea01d6dc5cf0c897c8d4c3e1df53d0d042784c424536a4cc5b802552b7d64fee8b5';\n    const hopTxid = '0x4af65143bc77da2b50f35b3d13cacb4db18f026bf84bc0743550bc57b9b53351';\n    const userReqSig =\n      '0x404db307f6147f0d8cd338c34c13906ef46a6faa7e0e119d5194ef05aec16e6f3d710f9b7901460f97e924066b62efd74443bd34402c6d40b49c203a559ff2c8';\n    const bitgoKeyXprv =\n      'xprv9s21ZrQH143K3tpWBHWe31sLoXNRQ9AvRYJgitkKxQ4ATFQMwvr7hHNqYRUnS7PsjzB7aK1VxqHLuNQjj1sckJ2Jwo2qxmsvejwECSpFMfC';\n    const bitgoKey = bip32.fromBase58(bitgoKeyXprv);\n    if (!bitgoKey.privateKey) {\n      throw new Error('no privateKey');\n    }\n    const hopTxBitgoSignature =\n      '0xaa' +\n      Buffer.from(secp256k1.ecdsaSign(Buffer.from(hopTxid.slice(2), 'hex'), bitgoKey.privateKey).signature).toString(\n        'hex'\n      );\n    it('should verify a normal txPrebuild from the bitgo server that matches the client txParams', async () => {\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: coinTest,\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 reject when client txParams are missing', async () => {\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: coinTest,\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 () => {\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: coinTest,\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 () => {\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: coinTest,\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('txPrebuild should only have 1 recipient but 2 found');\n    });\n\n    it('should reject a hop txPrebuild that does not send to its hop address', async () => {\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: coinTest,\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 () => {\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: coinTest,\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 () => {\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: coinTest,\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 () => {\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\nexport function runRecoveryTransactionTests(\n  coinName: string,\n  txBuilder: TransactionBuilder,\n  bitgo: TestBitGoAPI,\n  testData: any,\n  mockData: any\n) {\n  describe(`${coinName} Recover transaction:`, function () {\n    const baseUrl = testData.BASE_URL;\n    const userXpub =\n      'xpub661MyMwAqRbcEeTc8789MK5PUGEYiPG4F4V17n2Rd2LoTATA1XoCnJT5FAYAShQxSxtFjpo5NHmcWwTp2LiWGBMwpUcAA3HywhxivgYfq7q';\n    const userXprv =\n      'xprv9s21ZrQH143K2AP925b8zB8evEQ4JvYCsqZQKPcp4gopaN81TzUxEW8bPtVyDgjmddGhRRETn8xi1cVAB9bf1Bx9kGRRFgTZXxJayZLnag1';\n    const backupXpub =\n      'xpub661MyMwAqRbcFZX15xpZf4ERCGHiVSJm8r5C4yh1yXV2GrdZCUPYo4WQr6tN9oUywKXsgSHo7Risf9r22GH5joVD2hEEEhqnSCvK8qy11wW';\n    const backupXprv =\n      'xprv9s21ZrQH143K35SXywHZHvHgeETE5yaumd9bGbHQRBx3Q4JQew5JFGBvzqiZjCUkBdBUZnfuMDTGURRayN1hFSWxEJQsCEAMm1D3pk1h7Jj';\n    const coin = testData.COIN;\n\n    after(function () {\n      nock.cleanAll();\n    });\n\n    it('should generate an unsigned sweep', async () => {\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: any = bitgo.coin(coin);\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 () => {\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: any = bitgo.coin(coin);\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    it('should be able to second sign', async () => {\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: any = bitgo.coin(coin);\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: 80001, 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 SignTransactionOptions);\n      const halfSignedParams = {\n        txPrebuild: halfSigned,\n        isLastSignature: true,\n        walletContractAddress: walletContractAddress,\n        prv: backupXprv,\n      };\n\n      const finalSigned = (await basecoin.signTransaction(\n        halfSignedParams as SignTransactionOptions\n      )) as FullySignedTransaction;\n      finalSigned.should.have.property('txHex');\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"]}
|