@bitgo/sdk-coin-eth 25.3.2 → 25.3.3
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/test/fixtures/eth.d.ts +39 -0
- package/dist/test/fixtures/eth.d.ts.map +1 -0
- package/dist/test/fixtures/eth.js +262 -0
- package/dist/test/resources/erc1155Abi.d.ts +40 -0
- package/dist/test/resources/erc1155Abi.d.ts.map +1 -0
- package/dist/test/resources/erc1155Abi.js +329 -0
- package/dist/test/resources/erc721Abi.d.ts +12 -0
- package/dist/test/resources/erc721Abi.d.ts.map +1 -0
- package/dist/test/resources/erc721Abi.js +34 -0
- package/dist/test/resources/eth.d.ts +51 -0
- package/dist/test/resources/eth.d.ts.map +1 -0
- package/dist/test/resources/eth.js +72 -0
- package/dist/test/resources/nftEncodings.d.ts +4 -0
- package/dist/test/resources/nftEncodings.d.ts.map +1 -0
- package/dist/test/resources/nftEncodings.js +7 -0
- package/dist/test/resources/walletSimpleAbi.d.ts +12 -0
- package/dist/test/resources/walletSimpleAbi.d.ts.map +1 -0
- package/dist/test/resources/walletSimpleAbi.js +44 -0
- package/dist/test/unit/eth.d.ts +2 -0
- package/dist/test/unit/eth.d.ts.map +1 -0
- package/dist/test/unit/eth.js +1171 -0
- package/dist/test/unit/ethWallet.d.ts +2 -0
- package/dist/test/unit/ethWallet.d.ts.map +1 -0
- package/dist/test/unit/ethWallet.js +472 -0
- package/dist/test/unit/fixtures/ethMultiSigBackupKey.d.ts +4 -0
- package/dist/test/unit/fixtures/ethMultiSigBackupKey.d.ts.map +1 -0
- package/dist/test/unit/fixtures/ethMultiSigBackupKey.js +10 -0
- package/dist/test/unit/fixtures/ethTssBackupKey.d.ts +5 -0
- package/dist/test/unit/fixtures/ethTssBackupKey.d.ts.map +1 -0
- package/dist/test/unit/fixtures/ethTssBackupKey.js +25 -0
- package/dist/test/unit/getBuilder.d.ts +3 -0
- package/dist/test/unit/getBuilder.d.ts.map +1 -0
- package/dist/test/unit/getBuilder.js +10 -0
- package/dist/test/unit/keyPair.d.ts +2 -0
- package/dist/test/unit/keyPair.d.ts.map +1 -0
- package/dist/test/unit/keyPair.js +175 -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 +96 -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 +101 -0
- package/dist/test/unit/transactionBuilder/contractCall.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/contractCall.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/contractCall.js +101 -0
- package/dist/test/unit/transactionBuilder/flushCoins.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/flushCoins.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/flushCoins.js +133 -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 +281 -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 +331 -0
- package/dist/test/unit/transactionBuilder/sendNFT.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/sendNFT.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/sendNFT.js +250 -0
- package/dist/test/unit/transactionBuilder/singleSigSend.d.ts +2 -0
- package/dist/test/unit/transactionBuilder/singleSigSend.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder/singleSigSend.js +196 -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 +423 -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 +124 -0
- package/dist/test/unit/utils.d.ts +2 -0
- package/dist/test/unit/utils.d.ts.map +1 -0
- package/dist/test/unit/utils.js +74 -0
- package/dist/tsconfig.tsbuildinfo +1 -0
- package/package.json +10 -7
- package/.eslintignore +0 -5
- package/.mocharc.yml +0 -8
- package/CHANGELOG.md +0 -1858
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ethWallet.d.ts","sourceRoot":"","sources":["../../../test/unit/ethWallet.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,472 @@
|
|
|
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 should_1 = __importDefault(require("should"));
|
|
40
|
+
const secp256k1_1 = require("@bitgo/secp256k1");
|
|
41
|
+
const secp256k1 = __importStar(require("secp256k1"));
|
|
42
|
+
const nock_1 = __importDefault(require("nock"));
|
|
43
|
+
const sinon_1 = __importDefault(require("sinon"));
|
|
44
|
+
const sdk_core_1 = require("@bitgo/sdk-core");
|
|
45
|
+
const sdk_test_1 = require("@bitgo/sdk-test");
|
|
46
|
+
const tx_1 = require("@ethereumjs/tx");
|
|
47
|
+
const fixtures = require('../fixtures/eth');
|
|
48
|
+
const sdk_api_1 = require("@bitgo/sdk-api");
|
|
49
|
+
const src_1 = require("../../src");
|
|
50
|
+
const getBuilder_1 = require("./getBuilder");
|
|
51
|
+
describe('Sign ETH Transaction', async function () {
|
|
52
|
+
let bitgo;
|
|
53
|
+
let ethWallet;
|
|
54
|
+
let recipients;
|
|
55
|
+
let tx;
|
|
56
|
+
before(function () {
|
|
57
|
+
bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env: 'test' });
|
|
58
|
+
bitgo.initializeTestVars();
|
|
59
|
+
bitgo.safeRegister('teth', src_1.Teth.createInstance);
|
|
60
|
+
const coin = bitgo.coin('teth');
|
|
61
|
+
ethWallet = coin.newWalletObject({});
|
|
62
|
+
recipients = [
|
|
63
|
+
{
|
|
64
|
+
address: '0xe59dfe5c67114b39a5662cc856be536c614124c0',
|
|
65
|
+
amount: '100000',
|
|
66
|
+
},
|
|
67
|
+
];
|
|
68
|
+
tx = { recipients, nextContractSequenceId: 0 };
|
|
69
|
+
});
|
|
70
|
+
it('should read transaction recipients from txPrebuild even if none are specified as top-level params', async function () {
|
|
71
|
+
sinon_1.default.stub(sdk_core_1.Util, 'xprvToEthPrivateKey');
|
|
72
|
+
sinon_1.default.stub(sdk_core_1.Util, 'ethSignMsgHash');
|
|
73
|
+
sinon_1.default.stub(ethWallet.getOperationSha3ForExecuteAndConfirm);
|
|
74
|
+
const { halfSigned } = (await ethWallet.signTransaction({ txPrebuild: tx, prv: 'my_user_prv' }));
|
|
75
|
+
halfSigned.should.have.property('recipients', recipients);
|
|
76
|
+
sinon_1.default.restore();
|
|
77
|
+
});
|
|
78
|
+
it('should throw an error if no recipients are in the txPrebuild and none are specified as params', async function () {
|
|
79
|
+
await ethWallet
|
|
80
|
+
.signTransaction({ txPrebuild: {}, prv: 'my_user_prv' })
|
|
81
|
+
.should.be.rejectedWith('recipients missing or not array');
|
|
82
|
+
});
|
|
83
|
+
it('should throw an error if the recipients param is not an array', async function () {
|
|
84
|
+
await ethWallet
|
|
85
|
+
.signTransaction({ txPrebuild: { recipients: 'not-array' }, prv: 'my_user_prv' })
|
|
86
|
+
.should.be.rejectedWith('recipients missing or not array');
|
|
87
|
+
});
|
|
88
|
+
it('should set isBatch to false if single recipient', async function () {
|
|
89
|
+
sinon_1.default.stub(sdk_core_1.Util, 'xprvToEthPrivateKey');
|
|
90
|
+
sinon_1.default.stub(sdk_core_1.Util, 'ethSignMsgHash');
|
|
91
|
+
sinon_1.default.stub(ethWallet.getOperationSha3ForExecuteAndConfirm);
|
|
92
|
+
const singleRecipientsTx = { recipients: recipients, nextContractSequenceId: 0, isBatch: false };
|
|
93
|
+
const { halfSigned } = (await ethWallet.signTransaction({
|
|
94
|
+
txPrebuild: singleRecipientsTx,
|
|
95
|
+
prv: 'my_user_prv',
|
|
96
|
+
}));
|
|
97
|
+
halfSigned.should.have.property('recipients', recipients);
|
|
98
|
+
halfSigned.should.have.property('isBatch', false);
|
|
99
|
+
sinon_1.default.restore();
|
|
100
|
+
});
|
|
101
|
+
it('should set isBatch to true if multiple recipients', async function () {
|
|
102
|
+
const multipleRecipients = [
|
|
103
|
+
{
|
|
104
|
+
address: '0x0c7f3bc5d2b2c0dbee1b45536b82569f41b54331',
|
|
105
|
+
amount: '200',
|
|
106
|
+
data: '0xcf4c58e2000000000000000000000000000000000000000000000000000000000000004000000000000000000000000000000000000000000000000000000000000000a00000000000000000000000000000000000000000000000000000000000000002000000000000000000000000431745b89e73230b3bc8a19e019194efb4b99efd000000000000000000000000431745b89e73230b3bc8a19e019194efb4b99efd000000000000000000000000000000000000000000000000000000000000000200000000000000000000000000000000000000000000000000000000000000640000000000000000000000000000000000000000000000000000000000000064',
|
|
107
|
+
},
|
|
108
|
+
];
|
|
109
|
+
const multipleRecipientsTx = { recipients: multipleRecipients, nextContractSequenceId: 0, isBatch: true };
|
|
110
|
+
sinon_1.default.stub(sdk_core_1.Util, 'xprvToEthPrivateKey');
|
|
111
|
+
sinon_1.default.stub(sdk_core_1.Util, 'ethSignMsgHash');
|
|
112
|
+
sinon_1.default.stub(ethWallet.getOperationSha3ForExecuteAndConfirm);
|
|
113
|
+
const { halfSigned } = (await ethWallet.signTransaction({
|
|
114
|
+
txPrebuild: multipleRecipientsTx,
|
|
115
|
+
prv: 'my_user_prv',
|
|
116
|
+
}));
|
|
117
|
+
halfSigned.should.have.property('isBatch', true);
|
|
118
|
+
sinon_1.default.restore();
|
|
119
|
+
});
|
|
120
|
+
});
|
|
121
|
+
describe('Ethereum Hop Transactions', function () {
|
|
122
|
+
let bitgo;
|
|
123
|
+
let ethWallet;
|
|
124
|
+
let tx;
|
|
125
|
+
let txid;
|
|
126
|
+
let bitgoSignature;
|
|
127
|
+
let bitgoKeyXprv;
|
|
128
|
+
let bgUrl;
|
|
129
|
+
let env;
|
|
130
|
+
const userKeypair = {
|
|
131
|
+
xprv: 'xprv9s21ZrQH143K2fJ91S4BRsupcYrE6mmY96fcX5HkhoTrrwmwjd16Cn87cWinJjByrfpojjx7ezsJLx7TAKLT8m8hM5Kax9YcoxnBeJZ3t2k',
|
|
132
|
+
xpub: 'xpub661MyMwAqRbcF9Nc7TbBo1rZAagiWEVPWKbDKThNG8zqjk76HAKLkaSbTn6dK2dQPfuD7xjicxCZVWvj67fP5nQ9W7QURmoMVAX8m6jZsGp',
|
|
133
|
+
rawPub: '02c103ac74481874b5ef0f385d12725e4f14aedc9e00bc814ce96f47f62ce7adf2',
|
|
134
|
+
rawPrv: '936c5af3f8af81f75cdad1b08f29e7d9c01e598e2db2d7be18b9e5a8646e87c6',
|
|
135
|
+
path: 'm',
|
|
136
|
+
walletSubPath: '/0/0',
|
|
137
|
+
};
|
|
138
|
+
before(function () {
|
|
139
|
+
tx =
|
|
140
|
+
'0xf86c82015285012a05f200825208945208d8e80c6d1aef9be37b4bd19a9cf75ed93dc886b5e620f480008026a00e13f9e0e11337b2b0227e3412211d3625e43f1083fda399cc361dd4bf89083ba06c801a761e0aa3bc8db0ac2568d575b0fb306a1f04f4d5ba82ba3cc0ea0a83bd';
|
|
141
|
+
txid = '0x0ac669c5fef8294443c75a31e32c44b97bbc9e43a18ea8beabcc2a3b45eb6ffa';
|
|
142
|
+
bitgoKeyXprv =
|
|
143
|
+
'xprv9s21ZrQH143K3tpWBHWe31sLoXNRQ9AvRYJgitkKxQ4ATFQMwvr7hHNqYRUnS7PsjzB7aK1VxqHLuNQjj1sckJ2Jwo2qxmsvejwECSpFMfC';
|
|
144
|
+
const bitgoKey = secp256k1_1.bip32.fromBase58(bitgoKeyXprv);
|
|
145
|
+
if (!bitgoKey.privateKey) {
|
|
146
|
+
throw new Error('no privateKey');
|
|
147
|
+
}
|
|
148
|
+
const bitgoXpub = bitgoKey.neutered().toBase58();
|
|
149
|
+
bitgoSignature =
|
|
150
|
+
'0xaa' +
|
|
151
|
+
Buffer.from(secp256k1.ecdsaSign(Buffer.from(txid.slice(2), 'hex'), bitgoKey.privateKey).signature).toString('hex');
|
|
152
|
+
env = 'test';
|
|
153
|
+
bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env });
|
|
154
|
+
bitgo.safeRegister('teth', src_1.Teth.createInstance);
|
|
155
|
+
sdk_core_1.common.Environments[env].hsmXpub = bitgoXpub;
|
|
156
|
+
bitgo.initializeTestVars();
|
|
157
|
+
bgUrl = sdk_core_1.common.Environments[bitgo.getEnv()].uri;
|
|
158
|
+
const coin = bitgo.coin('teth');
|
|
159
|
+
ethWallet = coin.newWalletObject({ keys: ['user', 'backup', 'bitgo'] });
|
|
160
|
+
});
|
|
161
|
+
describe('Verify HSM Hop prebuild', function () {
|
|
162
|
+
let prebuild;
|
|
163
|
+
let buildParams;
|
|
164
|
+
let finalRecipient;
|
|
165
|
+
let sendAmount;
|
|
166
|
+
before(function () {
|
|
167
|
+
finalRecipient = '0x5208d8e80c6d1aef9be37b4bd19a9cf75ed93dc8';
|
|
168
|
+
sendAmount = '200000000000000';
|
|
169
|
+
prebuild = {
|
|
170
|
+
tx,
|
|
171
|
+
id: txid,
|
|
172
|
+
signature: bitgoSignature,
|
|
173
|
+
};
|
|
174
|
+
buildParams = {
|
|
175
|
+
recipients: [
|
|
176
|
+
{
|
|
177
|
+
address: finalRecipient,
|
|
178
|
+
amount: sendAmount,
|
|
179
|
+
},
|
|
180
|
+
],
|
|
181
|
+
};
|
|
182
|
+
});
|
|
183
|
+
it('should accept a valid hop prebuild', async function () {
|
|
184
|
+
await ethWallet.baseCoin.validateHopPrebuild(ethWallet, prebuild, buildParams).should.be.resolved();
|
|
185
|
+
});
|
|
186
|
+
it('should fail if the HSM prebuild recipient is wrong', async function () {
|
|
187
|
+
const badBuildParams = JSON.parse(JSON.stringify(buildParams));
|
|
188
|
+
badBuildParams.recipients[0].address = '0x54bf1609aeed804aa231f08c53dbb18f7d374615';
|
|
189
|
+
await ethWallet.baseCoin
|
|
190
|
+
.validateHopPrebuild(ethWallet, prebuild, badBuildParams)
|
|
191
|
+
.should.be.rejectedWith(/does not equal original recipient/);
|
|
192
|
+
});
|
|
193
|
+
it('should fail if the HSM prebuild tx amount is wrong', async function () {
|
|
194
|
+
const badBuildParams = JSON.parse(JSON.stringify(buildParams));
|
|
195
|
+
badBuildParams.recipients[0].amount = '50000000';
|
|
196
|
+
await ethWallet.baseCoin
|
|
197
|
+
.validateHopPrebuild(ethWallet, prebuild, badBuildParams)
|
|
198
|
+
.should.be.rejectedWith(/does not equal original amount/);
|
|
199
|
+
});
|
|
200
|
+
it('should fail if the HSM signature is invalid', async function () {
|
|
201
|
+
// Mocking a different BitGo key means the signing key should be wrong (it maps to a different address than this xpub)
|
|
202
|
+
const goodXpub = sdk_core_1.common.Environments[env].hsmXpub;
|
|
203
|
+
sdk_core_1.common.Environments[env].hsmXpub =
|
|
204
|
+
'xpub661MyMwAqRbcErFqVXGiUFv9YeoPbhN72UiNCUdj9nj3T6M8h7iKNmbCYpMVWVZP7LA2ma3HWcPngz1gRTm4FPdtm9mHfrNvU93MCoszsGL';
|
|
205
|
+
await ethWallet.baseCoin
|
|
206
|
+
.validateHopPrebuild(ethWallet, prebuild, buildParams)
|
|
207
|
+
.should.be.rejectedWith(/Hop txid signature invalid/);
|
|
208
|
+
sdk_core_1.common.Environments[env].hsmXpub = goodXpub;
|
|
209
|
+
});
|
|
210
|
+
it('should fail if the HSM signature signed the wrong HSM commitment digest', async function () {
|
|
211
|
+
const badTxid = '0xb4b3827a529c9166786e796528017889ac5027255b65b3fa2a3d3ad91244a12b';
|
|
212
|
+
const badTxidBuffer = Buffer.from(badTxid.slice(2), 'hex');
|
|
213
|
+
const xprvNode = secp256k1_1.bip32.fromBase58(bitgoKeyXprv);
|
|
214
|
+
if (!xprvNode.privateKey) {
|
|
215
|
+
throw new Error('no privateKey');
|
|
216
|
+
}
|
|
217
|
+
const badSignature = '0xaa' + Buffer.from(secp256k1.ecdsaSign(badTxidBuffer, xprvNode.privateKey).signature).toString('hex');
|
|
218
|
+
const badPrebuild = JSON.parse(JSON.stringify(prebuild));
|
|
219
|
+
badPrebuild.signature = badSignature;
|
|
220
|
+
await ethWallet.baseCoin
|
|
221
|
+
.validateHopPrebuild(ethWallet, badPrebuild, buildParams)
|
|
222
|
+
.should.be.rejectedWith(/Hop txid signature invalid/);
|
|
223
|
+
});
|
|
224
|
+
});
|
|
225
|
+
describe('Prebuild hop transaction', function () {
|
|
226
|
+
let prebuild;
|
|
227
|
+
let buildParams;
|
|
228
|
+
let finalRecipient;
|
|
229
|
+
let sendAmount;
|
|
230
|
+
let gasLimitEstimate;
|
|
231
|
+
let gasPrice;
|
|
232
|
+
const nockUserKey = function () {
|
|
233
|
+
(0, nock_1.default)(bgUrl)
|
|
234
|
+
.get(`/api/v2/teth/key/user`)
|
|
235
|
+
.reply(200, {
|
|
236
|
+
encryptedPrv: bitgo.encrypt({ input: userKeypair.xprv, password: sdk_test_1.TestBitGo.TEST_WALLET1_PASSCODE }),
|
|
237
|
+
path: userKeypair.path + userKeypair.walletSubPath,
|
|
238
|
+
});
|
|
239
|
+
};
|
|
240
|
+
const nockFees = function () {
|
|
241
|
+
const scope = (0, nock_1.default)(bgUrl)
|
|
242
|
+
.get('/api/v2/teth/tx/fee')
|
|
243
|
+
.query(true)
|
|
244
|
+
.reply(200, {
|
|
245
|
+
gasLimitEstimate: gasLimitEstimate,
|
|
246
|
+
feeEstimate: gasLimitEstimate * gasPrice,
|
|
247
|
+
});
|
|
248
|
+
return scope;
|
|
249
|
+
};
|
|
250
|
+
const nockBuild = function (walletId) {
|
|
251
|
+
(0, nock_1.default)(bgUrl)
|
|
252
|
+
.post('/api/v2/teth/wallet/' + walletId + '/tx/build')
|
|
253
|
+
.reply(200, { hopTransaction: prebuild, buildParams });
|
|
254
|
+
};
|
|
255
|
+
before(function () {
|
|
256
|
+
gasLimitEstimate = 100000;
|
|
257
|
+
gasPrice = 50000000;
|
|
258
|
+
finalRecipient = '0x5208d8e80c6d1aef9be37b4bd19a9cf75ed93dc8';
|
|
259
|
+
sendAmount = '200000000000000';
|
|
260
|
+
prebuild = {
|
|
261
|
+
tx,
|
|
262
|
+
id: txid,
|
|
263
|
+
signature: bitgoSignature,
|
|
264
|
+
};
|
|
265
|
+
buildParams = {
|
|
266
|
+
recipients: [
|
|
267
|
+
{
|
|
268
|
+
address: finalRecipient,
|
|
269
|
+
amount: sendAmount,
|
|
270
|
+
},
|
|
271
|
+
],
|
|
272
|
+
hop: true,
|
|
273
|
+
walletPassphrase: sdk_test_1.TestBitGo.TEST_WALLET1_PASSCODE,
|
|
274
|
+
};
|
|
275
|
+
});
|
|
276
|
+
it('should prebuild a hop transaction if given the correct args', async function () {
|
|
277
|
+
nockUserKey();
|
|
278
|
+
const feeScope = nockFees();
|
|
279
|
+
nockBuild(ethWallet.id());
|
|
280
|
+
const res = (await ethWallet.prebuildTransaction(buildParams));
|
|
281
|
+
should_1.default.exist(res.hopTransaction);
|
|
282
|
+
should_1.default.exist(res.hopTransaction.tx);
|
|
283
|
+
should_1.default.exist(res.hopTransaction.tx);
|
|
284
|
+
should_1.default.exist(res.hopTransaction.id);
|
|
285
|
+
should_1.default.exist(res.hopTransaction.signature);
|
|
286
|
+
should_1.default.not.exist(res.wallet);
|
|
287
|
+
should_1.default.not.exist(res.buildParams);
|
|
288
|
+
feeScope.isDone().should.equal(true);
|
|
289
|
+
const feeReq = feeScope.interceptors[0].req;
|
|
290
|
+
feeReq.path.should.containEql('hop=true');
|
|
291
|
+
feeReq.path.should.containEql('recipient=' + finalRecipient);
|
|
292
|
+
feeReq.path.should.containEql('amount=' + sendAmount);
|
|
293
|
+
});
|
|
294
|
+
});
|
|
295
|
+
});
|
|
296
|
+
describe('Add final signature to ETH tx from offline vault', function () {
|
|
297
|
+
let paramsFromVault, expectedResult, bitgo, coin;
|
|
298
|
+
before(function () {
|
|
299
|
+
const vals = fixtures.getHalfSignedTethFromVault();
|
|
300
|
+
paramsFromVault = vals.paramsFromVault;
|
|
301
|
+
expectedResult = vals.expectedResult;
|
|
302
|
+
bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env: 'test' });
|
|
303
|
+
coin = bitgo.coin('teth');
|
|
304
|
+
});
|
|
305
|
+
it('should successfully fully sign a half-signed transaction from the offline vault', async function () {
|
|
306
|
+
const response = (await coin.signTransaction(paramsFromVault));
|
|
307
|
+
const expectedTx = tx_1.Transaction.fromSerializedTx(Buffer.from(expectedResult.txHex, 'hex'));
|
|
308
|
+
const actualTx = tx_1.Transaction.fromSerializedTx(Buffer.from(response.txHex, 'hex'));
|
|
309
|
+
actualTx.nonce.toString().should.deepEqual(expectedTx.nonce.toString());
|
|
310
|
+
should_1.default.exist(actualTx.to);
|
|
311
|
+
actualTx.to?.should.deepEqual(expectedTx.to);
|
|
312
|
+
actualTx.value.should.deepEqual(expectedTx.value);
|
|
313
|
+
actualTx.data.should.deepEqual(expectedTx.data);
|
|
314
|
+
actualTx.isSigned().should.equal(true);
|
|
315
|
+
actualTx.supports(tx_1.Capability.EIP155ReplayProtection).should.equal(false);
|
|
316
|
+
actualTx.verifySignature().should.equal(true);
|
|
317
|
+
should_1.default.exist(actualTx.v);
|
|
318
|
+
actualTx?.v?.toString().should.deepEqual(expectedTx?.v?.toString());
|
|
319
|
+
actualTx?.r?.toString().should.deepEqual(expectedTx?.r?.toString());
|
|
320
|
+
actualTx?.s?.toString().should.deepEqual(expectedTx?.s?.toString());
|
|
321
|
+
actualTx.gasPrice.toString().should.deepEqual(expectedTx.gasPrice.toString());
|
|
322
|
+
actualTx.gasLimit.toString().should.deepEqual(expectedTx.gasLimit.toString());
|
|
323
|
+
response.txHex.toString().should.equal(expectedResult.txHex.toString());
|
|
324
|
+
});
|
|
325
|
+
});
|
|
326
|
+
describe('Add signature to EIP1559 tx from offline vault', function () {
|
|
327
|
+
let bitgo;
|
|
328
|
+
let paramsFromVault, expectedResult, coin;
|
|
329
|
+
before(function () {
|
|
330
|
+
const vals = fixtures.getUnsignedEip1559TethFromVault();
|
|
331
|
+
paramsFromVault = vals.paramsFromVault;
|
|
332
|
+
expectedResult = vals.expectedResult;
|
|
333
|
+
bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env: 'test' });
|
|
334
|
+
bitgo.safeRegister('teth', src_1.Teth.createInstance);
|
|
335
|
+
coin = bitgo.coin('teth');
|
|
336
|
+
});
|
|
337
|
+
it('should successfully sign an unsigned transaction from the offline vault', async function* () {
|
|
338
|
+
const response = await coin.signTransaction(paramsFromVault);
|
|
339
|
+
should_1.default.exist(response.halfSigned);
|
|
340
|
+
response.halfSigned.eip1559.should.deepEqual(expectedResult.halfSigned.eip1559);
|
|
341
|
+
response.halfSigned.recipients.should.deepEqual(expectedResult.halfSigned.recipients);
|
|
342
|
+
});
|
|
343
|
+
});
|
|
344
|
+
describe('prebuildTransaction', function () {
|
|
345
|
+
let bitgo;
|
|
346
|
+
let ethWallet;
|
|
347
|
+
let recipients;
|
|
348
|
+
let bgUrl;
|
|
349
|
+
let gasLimit;
|
|
350
|
+
before(function () {
|
|
351
|
+
bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env: 'test' });
|
|
352
|
+
src_1.Erc20Token.createTokenConstructors().forEach(({ name, coinConstructor }) => {
|
|
353
|
+
bitgo.safeRegister(name, coinConstructor);
|
|
354
|
+
});
|
|
355
|
+
bitgo.safeRegister('teth', src_1.Teth.createInstance);
|
|
356
|
+
bitgo.initializeTestVars();
|
|
357
|
+
const coin = bitgo.coin('teth');
|
|
358
|
+
ethWallet = coin.newWalletObject({});
|
|
359
|
+
gasLimit = 2100000;
|
|
360
|
+
recipients = [
|
|
361
|
+
{
|
|
362
|
+
address: '0xe59dfe5c67114b39a5662cc856be536c614124c0',
|
|
363
|
+
amount: '100000',
|
|
364
|
+
},
|
|
365
|
+
];
|
|
366
|
+
bgUrl = sdk_core_1.common.Environments[bitgo.getEnv()].uri;
|
|
367
|
+
});
|
|
368
|
+
it('should successfully accept gasLimit as a param', async function () {
|
|
369
|
+
const scope = (0, nock_1.default)(bgUrl)
|
|
370
|
+
.post('/api/v2/teth/wallet/' + ethWallet.id() + '/tx/build', {
|
|
371
|
+
recipients,
|
|
372
|
+
gasLimit,
|
|
373
|
+
})
|
|
374
|
+
.reply(200, { success: true });
|
|
375
|
+
const prebuild = await ethWallet.prebuildTransaction({ recipients, gasLimit });
|
|
376
|
+
scope.isDone().should.equal(true);
|
|
377
|
+
prebuild.success.should.equal(true);
|
|
378
|
+
});
|
|
379
|
+
it('should reject hop param for an erc20 token build', async function () {
|
|
380
|
+
const token = bitgo.coin('terc');
|
|
381
|
+
const tokenWallet = token.newWalletObject({});
|
|
382
|
+
recipients = [
|
|
383
|
+
{
|
|
384
|
+
address: '0xe59dfe5c67114b39a5662cc856be536c614124c0',
|
|
385
|
+
amount: '100',
|
|
386
|
+
},
|
|
387
|
+
];
|
|
388
|
+
await tokenWallet
|
|
389
|
+
.prebuildTransaction({ recipients, hop: true, walletPassphrase: 'hi' })
|
|
390
|
+
.should.be.rejectedWith(`Hop transactions are not enabled for ERC-20 tokens, nor are they necessary. Please remove the 'hop' parameter and try again.`);
|
|
391
|
+
});
|
|
392
|
+
});
|
|
393
|
+
describe('final-sign transaction from WRW', function () {
|
|
394
|
+
it('should add a second signature to unsigned sweep for teth', async function () {
|
|
395
|
+
const bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env: 'test' });
|
|
396
|
+
const basecoin = bitgo.coin('teth');
|
|
397
|
+
const gasPrice = 200000000000;
|
|
398
|
+
const gasLimit = 500000;
|
|
399
|
+
const prv = 'xprv9s21ZrQH143K3D8TXfvAJgHVfTEeQNW5Ys9wZtnUZkqPzFzSjbEJrWC1vZ4GnXCvR7rQL2UFX3RSuYeU9MrERm1XBvACow7c36vnz5iYyj2'; // placeholder test prv
|
|
400
|
+
const tx = {
|
|
401
|
+
txPrebuild: fixtures.WRWUnsignedSweepETHTx,
|
|
402
|
+
prv,
|
|
403
|
+
};
|
|
404
|
+
// sign transaction once
|
|
405
|
+
const halfSigned = await basecoin.signTransaction(tx);
|
|
406
|
+
const wrapper = {};
|
|
407
|
+
wrapper.txPrebuild = halfSigned;
|
|
408
|
+
wrapper.txPrebuild.recipients = halfSigned.halfSigned.recipients;
|
|
409
|
+
wrapper.txPrebuild.gasPrice = gasPrice.toString();
|
|
410
|
+
wrapper.txPrebuild.gasLimit = gasLimit.toString();
|
|
411
|
+
wrapper.isLastSignature = true;
|
|
412
|
+
wrapper.walletContractAddress = fixtures.WRWUnsignedSweepETHTx.walletContractAddress;
|
|
413
|
+
wrapper.prv = prv;
|
|
414
|
+
// sign transaction twice with the "isLastSignature" flag
|
|
415
|
+
const finalSignedTx = await basecoin.signTransaction(wrapper);
|
|
416
|
+
finalSignedTx.should.have.property('txHex');
|
|
417
|
+
const txBuilder = (0, getBuilder_1.getBuilder)('eth');
|
|
418
|
+
txBuilder.from('0x' + finalSignedTx.txHex); // add a 0x in front of this txhex
|
|
419
|
+
const rebuiltTx = await txBuilder.build();
|
|
420
|
+
const outputs = rebuiltTx.outputs.map((output) => {
|
|
421
|
+
return {
|
|
422
|
+
address: output.address,
|
|
423
|
+
amount: output.value,
|
|
424
|
+
};
|
|
425
|
+
});
|
|
426
|
+
rebuiltTx.signature.length.should.equal(2);
|
|
427
|
+
outputs.length.should.equal(1);
|
|
428
|
+
outputs[0].address.should.equal(fixtures.WRWUnsignedSweepETHTx.recipient.address);
|
|
429
|
+
outputs[0].amount.should.equal(fixtures.WRWUnsignedSweepETHTx.recipient.amount);
|
|
430
|
+
});
|
|
431
|
+
it('should add a second signature to unsigned sweep for erc20 token', async function () {
|
|
432
|
+
const bitgo = sdk_test_1.TestBitGo.decorate(sdk_api_1.BitGoAPI, { env: 'test' });
|
|
433
|
+
src_1.Erc20Token.createTokenConstructors().forEach(({ name, coinConstructor }) => {
|
|
434
|
+
bitgo.safeRegister(name, coinConstructor);
|
|
435
|
+
});
|
|
436
|
+
const basecoin = bitgo.coin('tdai');
|
|
437
|
+
const gasPrice = 200000000000;
|
|
438
|
+
const gasLimit = 500000;
|
|
439
|
+
const prv = 'xprv9s21ZrQH143K3399QBVvbmhs4RB5QzXD8XiW3NwtaeTem93QGd5VNjukUnwJQ94nUgugHSVzSVVe3RP16Urv1ZyijpYdyDamsxf2Shbq4w1'; // placeholder test prv
|
|
440
|
+
const tx = {
|
|
441
|
+
txPrebuild: fixtures.WRWUnsignedSweepERC20Tx,
|
|
442
|
+
prv,
|
|
443
|
+
};
|
|
444
|
+
// sign transaction once
|
|
445
|
+
const halfSigned = await basecoin.signTransaction(tx);
|
|
446
|
+
const wrapper = {};
|
|
447
|
+
wrapper.txPrebuild = halfSigned;
|
|
448
|
+
wrapper.txPrebuild.recipients = halfSigned.halfSigned.recipients;
|
|
449
|
+
wrapper.txPrebuild.gasPrice = gasPrice.toString();
|
|
450
|
+
wrapper.txPrebuild.gasLimit = gasLimit.toString();
|
|
451
|
+
wrapper.isLastSignature = true;
|
|
452
|
+
wrapper.walletContractAddress = fixtures.WRWUnsignedSweepERC20Tx.walletContractAddress;
|
|
453
|
+
wrapper.prv = prv;
|
|
454
|
+
// sign transaction twice with the "isLastSignature" flag
|
|
455
|
+
const finalSignedTx = await basecoin.signTransaction(wrapper);
|
|
456
|
+
finalSignedTx.should.have.property('txHex');
|
|
457
|
+
const txBuilder = (0, getBuilder_1.getBuilder)('eth');
|
|
458
|
+
txBuilder.from('0x' + finalSignedTx.txHex); // add a 0x in front of this txhex
|
|
459
|
+
const rebuiltTx = await txBuilder.build();
|
|
460
|
+
const outputs = rebuiltTx.outputs.map((output) => {
|
|
461
|
+
return {
|
|
462
|
+
address: output.address,
|
|
463
|
+
amount: output.value,
|
|
464
|
+
};
|
|
465
|
+
});
|
|
466
|
+
rebuiltTx.signature.length.should.equal(2);
|
|
467
|
+
outputs.length.should.equal(1);
|
|
468
|
+
outputs[0].address.should.equal(fixtures.WRWUnsignedSweepERC20Tx.recipient.address);
|
|
469
|
+
outputs[0].amount.should.equal(fixtures.WRWUnsignedSweepERC20Tx.recipient.amount);
|
|
470
|
+
});
|
|
471
|
+
});
|
|
472
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ethWallet.js","sourceRoot":"","sources":["../../../test/unit/ethWallet.ts"],"names":[],"mappings":"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","sourcesContent":["import should from 'should';\nimport { bip32 } from '@bitgo/secp256k1';\nimport * as secp256k1 from 'secp256k1';\nimport nock from 'nock';\nimport sinon from 'sinon';\nimport { common, Util } from '@bitgo/sdk-core';\n\nimport { TestBitGo, TestBitGoAPI } from '@bitgo/sdk-test';\n\nimport { Capability, Transaction as EthTx } from '@ethereumjs/tx';\nconst fixtures = require('../fixtures/eth');\n\nimport { BitGoAPI } from '@bitgo/sdk-api';\nimport { Erc20Token, SignTransactionOptions, Teth } from '../../src';\nimport { getBuilder } from './getBuilder';\n\ndescribe('Sign ETH Transaction', async function () {\n  let bitgo: TestBitGoAPI;\n  let ethWallet;\n  let recipients;\n  let tx;\n\n  before(function () {\n    bitgo = TestBitGo.decorate(BitGoAPI, { env: 'test' });\n    bitgo.initializeTestVars();\n    bitgo.safeRegister('teth', Teth.createInstance);\n    const coin = bitgo.coin('teth');\n    ethWallet = coin.newWalletObject({});\n    recipients = [\n      {\n        address: '0xe59dfe5c67114b39a5662cc856be536c614124c0',\n        amount: '100000',\n      },\n    ];\n    tx = { recipients, nextContractSequenceId: 0 };\n  });\n\n  it('should read transaction recipients from txPrebuild even if none are specified as top-level params', async function () {\n    sinon.stub(Util, 'xprvToEthPrivateKey');\n    sinon.stub(Util, 'ethSignMsgHash');\n    sinon.stub(ethWallet.getOperationSha3ForExecuteAndConfirm);\n\n    const { halfSigned } = (await ethWallet.signTransaction({ txPrebuild: tx, prv: 'my_user_prv' })) as any;\n    halfSigned.should.have.property('recipients', recipients);\n    sinon.restore();\n  });\n\n  it('should throw an error if no recipients are in the txPrebuild and none are specified as params', async function () {\n    await ethWallet\n      .signTransaction({ txPrebuild: {}, prv: 'my_user_prv' })\n      .should.be.rejectedWith('recipients missing or not array');\n  });\n\n  it('should throw an error if the recipients param is not an array', async function () {\n    await ethWallet\n      .signTransaction({ txPrebuild: { recipients: 'not-array' }, prv: 'my_user_prv' })\n      .should.be.rejectedWith('recipients missing or not array');\n  });\n\n  it('should set isBatch to false if single recipient', async function () {\n    sinon.stub(Util, 'xprvToEthPrivateKey');\n    sinon.stub(Util, 'ethSignMsgHash');\n    sinon.stub(ethWallet.getOperationSha3ForExecuteAndConfirm);\n    const singleRecipientsTx = { recipients: recipients, nextContractSequenceId: 0, isBatch: false };\n    const { halfSigned } = (await ethWallet.signTransaction({\n      txPrebuild: singleRecipientsTx,\n      prv: 'my_user_prv',\n    })) as any;\n    halfSigned.should.have.property('recipients', recipients);\n    halfSigned.should.have.property('isBatch', false);\n    sinon.restore();\n  });\n\n  it('should set isBatch to true if multiple recipients', async function () {\n    const multipleRecipients = [\n      {\n        address: '0x0c7f3bc5d2b2c0dbee1b45536b82569f41b54331',\n        amount: '200',\n        data: '0xcf4c58e2000000000000000000000000000000000000000000000000000000000000004000000000000000000000000000000000000000000000000000000000000000a00000000000000000000000000000000000000000000000000000000000000002000000000000000000000000431745b89e73230b3bc8a19e019194efb4b99efd000000000000000000000000431745b89e73230b3bc8a19e019194efb4b99efd000000000000000000000000000000000000000000000000000000000000000200000000000000000000000000000000000000000000000000000000000000640000000000000000000000000000000000000000000000000000000000000064',\n      },\n    ];\n\n    const multipleRecipientsTx = { recipients: multipleRecipients, nextContractSequenceId: 0, isBatch: true };\n\n    sinon.stub(Util, 'xprvToEthPrivateKey');\n    sinon.stub(Util, 'ethSignMsgHash');\n    sinon.stub(ethWallet.getOperationSha3ForExecuteAndConfirm);\n    const { halfSigned } = (await ethWallet.signTransaction({\n      txPrebuild: multipleRecipientsTx,\n      prv: 'my_user_prv',\n    })) as any;\n    halfSigned.should.have.property('isBatch', true);\n    sinon.restore();\n  });\n});\n\ndescribe('Ethereum Hop Transactions', function () {\n  let bitgo: TestBitGoAPI;\n  let ethWallet;\n  let tx;\n  let txid;\n  let bitgoSignature;\n  let bitgoKeyXprv;\n  let bgUrl;\n  let env;\n\n  const userKeypair = {\n    xprv: 'xprv9s21ZrQH143K2fJ91S4BRsupcYrE6mmY96fcX5HkhoTrrwmwjd16Cn87cWinJjByrfpojjx7ezsJLx7TAKLT8m8hM5Kax9YcoxnBeJZ3t2k',\n    xpub: 'xpub661MyMwAqRbcF9Nc7TbBo1rZAagiWEVPWKbDKThNG8zqjk76HAKLkaSbTn6dK2dQPfuD7xjicxCZVWvj67fP5nQ9W7QURmoMVAX8m6jZsGp',\n    rawPub: '02c103ac74481874b5ef0f385d12725e4f14aedc9e00bc814ce96f47f62ce7adf2',\n    rawPrv: '936c5af3f8af81f75cdad1b08f29e7d9c01e598e2db2d7be18b9e5a8646e87c6',\n    path: 'm',\n    walletSubPath: '/0/0',\n  };\n\n  before(function () {\n    tx =\n      '0xf86c82015285012a05f200825208945208d8e80c6d1aef9be37b4bd19a9cf75ed93dc886b5e620f480008026a00e13f9e0e11337b2b0227e3412211d3625e43f1083fda399cc361dd4bf89083ba06c801a761e0aa3bc8db0ac2568d575b0fb306a1f04f4d5ba82ba3cc0ea0a83bd';\n    txid = '0x0ac669c5fef8294443c75a31e32c44b97bbc9e43a18ea8beabcc2a3b45eb6ffa';\n    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    bitgoSignature =\n      '0xaa' +\n      Buffer.from(secp256k1.ecdsaSign(Buffer.from(txid.slice(2), 'hex'), bitgoKey.privateKey).signature).toString(\n        'hex'\n      );\n\n    env = 'test';\n    bitgo = TestBitGo.decorate(BitGoAPI, { env });\n    bitgo.safeRegister('teth', Teth.createInstance);\n    common.Environments[env].hsmXpub = bitgoXpub;\n    bitgo.initializeTestVars();\n    bgUrl = common.Environments[bitgo.getEnv()].uri;\n    const coin = bitgo.coin('teth');\n    ethWallet = coin.newWalletObject({ keys: ['user', 'backup', 'bitgo'] });\n  });\n\n  describe('Verify HSM Hop prebuild', function () {\n    let prebuild;\n    let buildParams;\n    let finalRecipient;\n    let sendAmount;\n\n    before(function () {\n      finalRecipient = '0x5208d8e80c6d1aef9be37b4bd19a9cf75ed93dc8';\n      sendAmount = '200000000000000';\n      prebuild = {\n        tx,\n        id: txid,\n        signature: bitgoSignature,\n      };\n      buildParams = {\n        recipients: [\n          {\n            address: finalRecipient,\n            amount: sendAmount,\n          },\n        ],\n      };\n    });\n\n    it('should accept a valid hop prebuild', async function () {\n      await ethWallet.baseCoin.validateHopPrebuild(ethWallet, prebuild, buildParams).should.be.resolved();\n    });\n\n    it('should fail if the HSM prebuild recipient is wrong', async function () {\n      const badBuildParams = JSON.parse(JSON.stringify(buildParams));\n      badBuildParams.recipients[0].address = '0x54bf1609aeed804aa231f08c53dbb18f7d374615';\n\n      await ethWallet.baseCoin\n        .validateHopPrebuild(ethWallet, prebuild, badBuildParams)\n        .should.be.rejectedWith(/does not equal original recipient/);\n    });\n\n    it('should fail if the HSM prebuild tx amount is wrong', async function () {\n      const badBuildParams = JSON.parse(JSON.stringify(buildParams));\n      badBuildParams.recipients[0].amount = '50000000';\n\n      await ethWallet.baseCoin\n        .validateHopPrebuild(ethWallet, prebuild, badBuildParams)\n        .should.be.rejectedWith(/does not equal original amount/);\n    });\n\n    it('should fail if the HSM signature is invalid', async function () {\n      // Mocking a different BitGo key means the signing key should be wrong (it maps to a different address than this xpub)\n      const goodXpub = common.Environments[env].hsmXpub;\n      common.Environments[env].hsmXpub =\n        'xpub661MyMwAqRbcErFqVXGiUFv9YeoPbhN72UiNCUdj9nj3T6M8h7iKNmbCYpMVWVZP7LA2ma3HWcPngz1gRTm4FPdtm9mHfrNvU93MCoszsGL';\n\n      await ethWallet.baseCoin\n        .validateHopPrebuild(ethWallet, prebuild, buildParams)\n        .should.be.rejectedWith(/Hop txid signature invalid/);\n      common.Environments[env].hsmXpub = goodXpub;\n    });\n\n    it('should fail if the HSM signature signed the wrong HSM commitment digest', async function () {\n      const badTxid = '0xb4b3827a529c9166786e796528017889ac5027255b65b3fa2a3d3ad91244a12b';\n      const badTxidBuffer = Buffer.from(badTxid.slice(2), 'hex');\n      const xprvNode = bip32.fromBase58(bitgoKeyXprv);\n      if (!xprvNode.privateKey) {\n        throw new Error('no privateKey');\n      }\n      const badSignature =\n        '0xaa' + Buffer.from(secp256k1.ecdsaSign(badTxidBuffer, xprvNode.privateKey).signature).toString('hex');\n      const badPrebuild = JSON.parse(JSON.stringify(prebuild));\n      badPrebuild.signature = badSignature;\n\n      await ethWallet.baseCoin\n        .validateHopPrebuild(ethWallet, badPrebuild, buildParams)\n        .should.be.rejectedWith(/Hop txid signature invalid/);\n    });\n  });\n\n  describe('Prebuild hop transaction', function () {\n    let prebuild;\n    let buildParams;\n    let finalRecipient;\n    let sendAmount;\n    let gasLimitEstimate;\n    let gasPrice;\n\n    const nockUserKey = function () {\n      nock(bgUrl)\n        .get(`/api/v2/teth/key/user`)\n        .reply(200, {\n          encryptedPrv: bitgo.encrypt({ input: userKeypair.xprv, password: TestBitGo.TEST_WALLET1_PASSCODE }),\n          path: userKeypair.path + userKeypair.walletSubPath,\n        });\n    };\n    const nockFees = function () {\n      const scope = nock(bgUrl)\n        .get('/api/v2/teth/tx/fee')\n        .query(true)\n        .reply(200, {\n          gasLimitEstimate: gasLimitEstimate,\n          feeEstimate: gasLimitEstimate * gasPrice,\n        });\n      return scope;\n    };\n\n    const nockBuild = function (walletId) {\n      nock(bgUrl)\n        .post('/api/v2/teth/wallet/' + walletId + '/tx/build')\n        .reply(200, { hopTransaction: prebuild, buildParams });\n    };\n\n    before(function () {\n      gasLimitEstimate = 100000;\n      gasPrice = 50000000;\n      finalRecipient = '0x5208d8e80c6d1aef9be37b4bd19a9cf75ed93dc8';\n      sendAmount = '200000000000000';\n      prebuild = {\n        tx,\n        id: txid,\n        signature: bitgoSignature,\n      };\n      buildParams = {\n        recipients: [\n          {\n            address: finalRecipient,\n            amount: sendAmount,\n          },\n        ],\n        hop: true,\n        walletPassphrase: TestBitGo.TEST_WALLET1_PASSCODE,\n      };\n    });\n\n    it('should prebuild a hop transaction if given the correct args', async function () {\n      nockUserKey();\n      const feeScope = nockFees();\n      nockBuild(ethWallet.id());\n      const res = (await ethWallet.prebuildTransaction(buildParams)) as any;\n      should.exist(res.hopTransaction);\n      should.exist(res.hopTransaction.tx);\n      should.exist(res.hopTransaction.tx);\n      should.exist(res.hopTransaction.id);\n      should.exist(res.hopTransaction.signature);\n      should.not.exist(res.wallet);\n      should.not.exist(res.buildParams);\n      feeScope.isDone().should.equal(true);\n      const feeReq = (feeScope as any).interceptors[0].req;\n      feeReq.path.should.containEql('hop=true');\n      feeReq.path.should.containEql('recipient=' + finalRecipient);\n      feeReq.path.should.containEql('amount=' + sendAmount);\n    });\n  });\n});\n\ndescribe('Add final signature to ETH tx from offline vault', function () {\n  let paramsFromVault, expectedResult, bitgo, coin;\n  before(function () {\n    const vals = fixtures.getHalfSignedTethFromVault();\n    paramsFromVault = vals.paramsFromVault;\n    expectedResult = vals.expectedResult;\n    bitgo = TestBitGo.decorate(BitGoAPI, { env: 'test' });\n    coin = bitgo.coin('teth');\n  });\n\n  it('should successfully fully sign a half-signed transaction from the offline vault', async function () {\n    const response = (await coin.signTransaction(paramsFromVault)) as any;\n    const expectedTx = EthTx.fromSerializedTx(Buffer.from(expectedResult.txHex, 'hex'));\n    const actualTx = EthTx.fromSerializedTx(Buffer.from(response.txHex, 'hex'));\n    actualTx.nonce.toString().should.deepEqual(expectedTx.nonce.toString());\n    should.exist(actualTx.to);\n    actualTx.to?.should.deepEqual(expectedTx.to);\n    actualTx.value.should.deepEqual(expectedTx.value);\n    actualTx.data.should.deepEqual(expectedTx.data);\n    actualTx.isSigned().should.equal(true);\n    actualTx.supports(Capability.EIP155ReplayProtection).should.equal(false);\n    actualTx.verifySignature().should.equal(true);\n    should.exist(actualTx.v);\n    actualTx?.v?.toString().should.deepEqual(expectedTx?.v?.toString());\n    actualTx?.r?.toString().should.deepEqual(expectedTx?.r?.toString());\n    actualTx?.s?.toString().should.deepEqual(expectedTx?.s?.toString());\n    actualTx.gasPrice.toString().should.deepEqual(expectedTx.gasPrice.toString());\n    actualTx.gasLimit.toString().should.deepEqual(expectedTx.gasLimit.toString());\n    response.txHex.toString().should.equal(expectedResult.txHex.toString());\n  });\n});\n\ndescribe('Add signature to EIP1559 tx from offline vault', function () {\n  let bitgo: TestBitGoAPI;\n  let paramsFromVault, expectedResult, coin;\n  before(function () {\n    const vals = fixtures.getUnsignedEip1559TethFromVault();\n    paramsFromVault = vals.paramsFromVault;\n    expectedResult = vals.expectedResult;\n    bitgo = TestBitGo.decorate(BitGoAPI, { env: 'test' });\n    bitgo.safeRegister('teth', Teth.createInstance);\n    coin = bitgo.coin('teth');\n  });\n\n  it('should successfully sign an unsigned transaction from the offline vault', async function* () {\n    const response = await coin.signTransaction(paramsFromVault);\n    should.exist(response.halfSigned);\n    response.halfSigned.eip1559.should.deepEqual(expectedResult.halfSigned.eip1559);\n    response.halfSigned.recipients.should.deepEqual(expectedResult.halfSigned.recipients);\n  });\n});\n\ndescribe('prebuildTransaction', function () {\n  let bitgo: TestBitGoAPI;\n  let ethWallet;\n  let recipients;\n  let bgUrl;\n  let gasLimit;\n\n  before(function () {\n    bitgo = TestBitGo.decorate(BitGoAPI, { env: 'test' });\n    Erc20Token.createTokenConstructors().forEach(({ name, coinConstructor }) => {\n      bitgo.safeRegister(name, coinConstructor);\n    });\n    bitgo.safeRegister('teth', Teth.createInstance);\n    bitgo.initializeTestVars();\n    const coin = bitgo.coin('teth');\n    ethWallet = coin.newWalletObject({});\n    gasLimit = 2100000;\n    recipients = [\n      {\n        address: '0xe59dfe5c67114b39a5662cc856be536c614124c0',\n        amount: '100000',\n      },\n    ];\n    bgUrl = common.Environments[bitgo.getEnv()].uri;\n  });\n\n  it('should successfully accept gasLimit as a param', async function () {\n    const scope = nock(bgUrl)\n      .post('/api/v2/teth/wallet/' + ethWallet.id() + '/tx/build', {\n        recipients,\n        gasLimit,\n      })\n      .reply(200, { success: true });\n    const prebuild = await ethWallet.prebuildTransaction({ recipients, gasLimit });\n    scope.isDone().should.equal(true);\n    prebuild.success.should.equal(true);\n  });\n\n  it('should reject hop param for an erc20 token build', async function () {\n    const token = bitgo.coin('terc');\n    const tokenWallet = token.newWalletObject({});\n    recipients = [\n      {\n        address: '0xe59dfe5c67114b39a5662cc856be536c614124c0',\n        amount: '100',\n      },\n    ];\n    await tokenWallet\n      .prebuildTransaction({ recipients, hop: true, walletPassphrase: 'hi' })\n      .should.be.rejectedWith(\n        `Hop transactions are not enabled for ERC-20 tokens, nor are they necessary. Please remove the 'hop' parameter and try again.`\n      );\n  });\n});\n\ndescribe('final-sign transaction from WRW', function () {\n  it('should add a second signature to unsigned sweep for teth', async function () {\n    const bitgo = TestBitGo.decorate(BitGoAPI, { env: 'test' });\n\n    const basecoin: any = bitgo.coin('teth');\n    const gasPrice = 200000000000;\n    const gasLimit = 500000;\n    const prv =\n      'xprv9s21ZrQH143K3D8TXfvAJgHVfTEeQNW5Ys9wZtnUZkqPzFzSjbEJrWC1vZ4GnXCvR7rQL2UFX3RSuYeU9MrERm1XBvACow7c36vnz5iYyj2'; // placeholder test prv\n    const tx = {\n      txPrebuild: fixtures.WRWUnsignedSweepETHTx,\n      prv,\n    };\n    // sign transaction once\n    const halfSigned = await basecoin.signTransaction(tx);\n\n    const wrapper = {} as SignTransactionOptions;\n    wrapper.txPrebuild = halfSigned;\n    wrapper.txPrebuild.recipients = halfSigned.halfSigned.recipients;\n    wrapper.txPrebuild.gasPrice = gasPrice.toString();\n    wrapper.txPrebuild.gasLimit = gasLimit.toString();\n    wrapper.isLastSignature = true;\n    wrapper.walletContractAddress = fixtures.WRWUnsignedSweepETHTx.walletContractAddress;\n    wrapper.prv = prv;\n\n    // sign transaction twice with the \"isLastSignature\" flag\n    const finalSignedTx = await basecoin.signTransaction(wrapper);\n    finalSignedTx.should.have.property('txHex');\n    const txBuilder = getBuilder('eth');\n    txBuilder.from('0x' + finalSignedTx.txHex); // add a 0x in front of this txhex\n    const rebuiltTx = await txBuilder.build();\n    const outputs = rebuiltTx.outputs.map((output) => {\n      return {\n        address: output.address,\n        amount: output.value,\n      };\n    });\n    rebuiltTx.signature.length.should.equal(2);\n    outputs.length.should.equal(1);\n    outputs[0].address.should.equal(fixtures.WRWUnsignedSweepETHTx.recipient.address);\n    outputs[0].amount.should.equal(fixtures.WRWUnsignedSweepETHTx.recipient.amount);\n  });\n\n  it('should add a second signature to unsigned sweep for erc20 token', async function () {\n    const bitgo = TestBitGo.decorate(BitGoAPI, { env: 'test' });\n    Erc20Token.createTokenConstructors().forEach(({ name, coinConstructor }) => {\n      bitgo.safeRegister(name, coinConstructor);\n    });\n    const basecoin: any = bitgo.coin('tdai');\n    const gasPrice = 200000000000;\n    const gasLimit = 500000;\n    const prv =\n      'xprv9s21ZrQH143K3399QBVvbmhs4RB5QzXD8XiW3NwtaeTem93QGd5VNjukUnwJQ94nUgugHSVzSVVe3RP16Urv1ZyijpYdyDamsxf2Shbq4w1'; // placeholder test prv\n    const tx = {\n      txPrebuild: fixtures.WRWUnsignedSweepERC20Tx,\n      prv,\n    };\n    // sign transaction once\n    const halfSigned = await basecoin.signTransaction(tx);\n\n    const wrapper = {} as SignTransactionOptions;\n    wrapper.txPrebuild = halfSigned;\n    wrapper.txPrebuild.recipients = halfSigned.halfSigned.recipients;\n    wrapper.txPrebuild.gasPrice = gasPrice.toString();\n    wrapper.txPrebuild.gasLimit = gasLimit.toString();\n    wrapper.isLastSignature = true;\n    wrapper.walletContractAddress = fixtures.WRWUnsignedSweepERC20Tx.walletContractAddress;\n    wrapper.prv = prv;\n\n    // sign transaction twice with the \"isLastSignature\" flag\n    const finalSignedTx = await basecoin.signTransaction(wrapper);\n    finalSignedTx.should.have.property('txHex');\n    const txBuilder = getBuilder('eth');\n    txBuilder.from('0x' + finalSignedTx.txHex); // add a 0x in front of this txhex\n    const rebuiltTx = await txBuilder.build();\n    const outputs = rebuiltTx.outputs.map((output) => {\n      return {\n        address: output.address,\n        amount: output.value,\n      };\n    });\n    rebuiltTx.signature.length.should.equal(2);\n    outputs.length.should.equal(1);\n    outputs[0].address.should.equal(fixtures.WRWUnsignedSweepERC20Tx.recipient.address);\n    outputs[0].amount.should.equal(fixtures.WRWUnsignedSweepERC20Tx.recipient.amount);\n  });\n});\n"]}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ethMultiSigBackupKey.d.ts","sourceRoot":"","sources":["../../../../test/unit/fixtures/ethMultiSigBackupKey.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,oBAAoB;;CAMhC,CAAC"}
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.ethMultiSigBackupKey = void 0;
|
|
4
|
+
exports.ethMultiSigBackupKey = {
|
|
5
|
+
key: '{"iv":"aT3q3wZuRF1KZmAwBgbTAg==","v":1,"iter":10000,"ks":256,"ts":64,"mode"\n' +
|
|
6
|
+
':"ccm","adata":"","cipher":"aes","salt":"CcVKl6CewXI=","ct":"C2vf/Cnf2H8UW9\n' +
|
|
7
|
+
'RsMnlUUOlXa8bzVUhCLGDL0Oulv1RDHl6iM8otzokdmz8IPMENjJf9z0qua1jnbiwPjW7FP14FV\n' +
|
|
8
|
+
'M1ku0v8lUFtYPGj/rzA5aSm2pPqAKe1uutC1tvG0aEWg+V5h02EXxvleFUfCsrVMqOez/0="}',
|
|
9
|
+
};
|
|
10
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZXRoTXVsdGlTaWdCYWNrdXBLZXkuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi90ZXN0L3VuaXQvZml4dHVyZXMvZXRoTXVsdGlTaWdCYWNrdXBLZXkudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBQWEsUUFBQSxvQkFBb0IsR0FBRztJQUNsQyxHQUFHLEVBQ0QsK0VBQStFO1FBQy9FLCtFQUErRTtRQUMvRSwrRUFBK0U7UUFDL0UsMkVBQTJFO0NBQzlFLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgZXRoTXVsdGlTaWdCYWNrdXBLZXkgPSB7XG4gIGtleTpcbiAgICAne1wiaXZcIjpcImFUM3Ezd1p1UkYxS1ptQXdCZ2JUQWc9PVwiLFwidlwiOjEsXCJpdGVyXCI6MTAwMDAsXCJrc1wiOjI1NixcInRzXCI6NjQsXCJtb2RlXCJcXG4nICtcbiAgICAnOlwiY2NtXCIsXCJhZGF0YVwiOlwiXCIsXCJjaXBoZXJcIjpcImFlc1wiLFwic2FsdFwiOlwiQ2NWS2w2Q2V3WEk9XCIsXCJjdFwiOlwiQzJ2Zi9DbmYySDhVVzlcXG4nICtcbiAgICAnUnNNbmxVVU9sWGE4YnpWVWhDTEdETDBPdWx2MVJESGw2aU04b3R6b2tkbXo4SVBNRU5qSmY5ejBxdWExam5iaXdQalc3RlAxNEZWXFxuJyArXG4gICAgJ00xa3UwdjhsVUZ0WVBHai9yekE1YVNtMnBQcUFLZTF1dXRDMXR2RzBhRVdnK1Y1aDAyRVh4dmxlRlVmQ3NyVk1xT2V6LzA9XCJ9Jyxcbn07XG4iXX0=
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"ethTssBackupKey.d.ts","sourceRoot":"","sources":["../../../../test/unit/fixtures/ethTssBackupKey.ts"],"names":[],"mappings":"AAAA,eAAO,MAAM,eAAe;;;CAsB3B,CAAC"}
|