@bitgo-beta/sdk-coin-trx 1.2.3-alpha.40 → 1.2.3-alpha.401
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/.mocharc.yml +1 -1
- package/CHANGELOG.md +988 -0
- package/dist/resources/protobuf/Contract.proto +32 -0
- package/dist/resources/protobuf/tron.d.ts +1478 -214
- package/dist/resources/protobuf/tron.js +6312 -2600
- package/dist/resources/protobuf/tron.proto +6 -0
- package/package.json +16 -14
- package/resources/protobuf/Contract.proto +32 -0
- package/resources/protobuf/tron.d.ts +1478 -214
- package/resources/protobuf/tron.js +6312 -2600
- package/resources/protobuf/tron.proto +6 -0
- package/dist/src/index.d.ts +0 -6
- package/dist/src/index.d.ts.map +0 -1
- package/dist/src/index.js +0 -18
- package/dist/src/lib/address.d.ts +0 -11
- package/dist/src/lib/address.d.ts.map +0 -1
- package/dist/src/lib/address.js +0 -13
- package/dist/src/lib/builder.d.ts +0 -3
- package/dist/src/lib/builder.d.ts.map +0 -1
- package/dist/src/lib/builder.js +0 -10
- package/dist/src/lib/contractCallBuilder.d.ts +0 -96
- package/dist/src/lib/contractCallBuilder.d.ts.map +0 -1
- package/dist/src/lib/contractCallBuilder.js +0 -281
- package/dist/src/lib/enum.d.ts +0 -23
- package/dist/src/lib/enum.d.ts.map +0 -1
- package/dist/src/lib/enum.js +0 -28
- package/dist/src/lib/iface.d.ts +0 -103
- package/dist/src/lib/iface.d.ts.map +0 -1
- package/dist/src/lib/iface.js +0 -3
- package/dist/src/lib/index.d.ts +0 -12
- package/dist/src/lib/index.d.ts.map +0 -1
- package/dist/src/lib/index.js +0 -39
- package/dist/src/lib/keyPair.d.ts +0 -44
- package/dist/src/lib/keyPair.d.ts.map +0 -1
- package/dist/src/lib/keyPair.js +0 -121
- package/dist/src/lib/tokenTransferBuilder.d.ts +0 -23
- package/dist/src/lib/tokenTransferBuilder.d.ts.map +0 -1
- package/dist/src/lib/tokenTransferBuilder.js +0 -42
- package/dist/src/lib/transaction.d.ts +0 -74
- package/dist/src/lib/transaction.d.ts.map +0 -1
- package/dist/src/lib/transaction.js +0 -226
- package/dist/src/lib/transactionBuilder.d.ts +0 -65
- package/dist/src/lib/transactionBuilder.d.ts.map +0 -1
- package/dist/src/lib/transactionBuilder.js +0 -215
- package/dist/src/lib/utils.d.ts +0 -144
- package/dist/src/lib/utils.d.ts.map +0 -1
- package/dist/src/lib/utils.js +0 -426
- package/dist/src/lib/wrappedBuilder.d.ts +0 -54
- package/dist/src/lib/wrappedBuilder.d.ts.map +0 -1
- package/dist/src/lib/wrappedBuilder.js +0 -108
- package/dist/src/register.d.ts +0 -3
- package/dist/src/register.d.ts.map +0 -1
- package/dist/src/register.js +0 -15
- package/dist/src/trx.d.ts +0 -208
- package/dist/src/trx.d.ts.map +0 -1
- package/dist/src/trx.js +0 -563
- package/dist/src/trxToken.d.ts +0 -38
- package/dist/src/trxToken.d.ts.map +0 -1
- package/dist/src/trxToken.js +0 -91
- package/dist/src/ttrx.d.ts +0 -13
- package/dist/src/ttrx.d.ts.map +0 -1
- package/dist/src/ttrx.js +0 -14
package/dist/src/trx.js
DELETED
|
@@ -1,563 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
-
if (k2 === undefined) k2 = k;
|
|
4
|
-
Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
|
|
5
|
-
}) : (function(o, m, k, k2) {
|
|
6
|
-
if (k2 === undefined) k2 = k;
|
|
7
|
-
o[k2] = m[k];
|
|
8
|
-
}));
|
|
9
|
-
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
10
|
-
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
11
|
-
}) : function(o, v) {
|
|
12
|
-
o["default"] = v;
|
|
13
|
-
});
|
|
14
|
-
var __importStar = (this && this.__importStar) || function (mod) {
|
|
15
|
-
if (mod && mod.__esModule) return mod;
|
|
16
|
-
var result = {};
|
|
17
|
-
if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
|
|
18
|
-
__setModuleDefault(result, mod);
|
|
19
|
-
return result;
|
|
20
|
-
};
|
|
21
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
22
|
-
exports.Trx = exports.NodeTypes = exports.MINIMUM_TRON_MSIG_TRANSACTION_FEE = void 0;
|
|
23
|
-
/**
|
|
24
|
-
* @prettier
|
|
25
|
-
*/
|
|
26
|
-
const secp256k1 = __importStar(require("secp256k1"));
|
|
27
|
-
const crypto_1 = require("crypto");
|
|
28
|
-
const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
|
|
29
|
-
const request = __importStar(require("superagent"));
|
|
30
|
-
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
31
|
-
const lib_1 = require("./lib");
|
|
32
|
-
const builder_1 = require("./lib/builder");
|
|
33
|
-
exports.MINIMUM_TRON_MSIG_TRANSACTION_FEE = 1e6;
|
|
34
|
-
var NodeTypes;
|
|
35
|
-
(function (NodeTypes) {
|
|
36
|
-
NodeTypes[NodeTypes["Full"] = 0] = "Full";
|
|
37
|
-
NodeTypes[NodeTypes["Solidity"] = 1] = "Solidity";
|
|
38
|
-
})(NodeTypes = exports.NodeTypes || (exports.NodeTypes = {}));
|
|
39
|
-
class Trx extends sdk_core_1.BaseCoin {
|
|
40
|
-
constructor(bitgo, staticsCoin) {
|
|
41
|
-
super(bitgo);
|
|
42
|
-
if (!staticsCoin) {
|
|
43
|
-
throw new Error('missing required constructor parameter staticsCoin');
|
|
44
|
-
}
|
|
45
|
-
this._staticsCoin = staticsCoin;
|
|
46
|
-
}
|
|
47
|
-
getChain() {
|
|
48
|
-
return this._staticsCoin.name;
|
|
49
|
-
}
|
|
50
|
-
getFamily() {
|
|
51
|
-
return this._staticsCoin.family;
|
|
52
|
-
}
|
|
53
|
-
getFullName() {
|
|
54
|
-
return this._staticsCoin.fullName;
|
|
55
|
-
}
|
|
56
|
-
getBaseFactor() {
|
|
57
|
-
return Math.pow(10, this._staticsCoin.decimalPlaces);
|
|
58
|
-
}
|
|
59
|
-
/** @inheritdoc */
|
|
60
|
-
transactionDataAllowed() {
|
|
61
|
-
return true;
|
|
62
|
-
}
|
|
63
|
-
static createInstance(bitgo, staticsCoin) {
|
|
64
|
-
return new Trx(bitgo, staticsCoin);
|
|
65
|
-
}
|
|
66
|
-
/**
|
|
67
|
-
* Flag for sending value of 0
|
|
68
|
-
* @returns {boolean} True if okay to send 0 value, false otherwise
|
|
69
|
-
*/
|
|
70
|
-
valuelessTransferAllowed() {
|
|
71
|
-
return true;
|
|
72
|
-
}
|
|
73
|
-
/** @inheritDoc */
|
|
74
|
-
allowsAccountConsolidations() {
|
|
75
|
-
return true;
|
|
76
|
-
}
|
|
77
|
-
/**
|
|
78
|
-
* Checks if this is a valid base58 or hex address
|
|
79
|
-
* @param address
|
|
80
|
-
*/
|
|
81
|
-
isValidAddress(address) {
|
|
82
|
-
if (!address) {
|
|
83
|
-
return false;
|
|
84
|
-
}
|
|
85
|
-
return this.isValidHexAddress(address) || lib_1.Utils.isBase58Address(address);
|
|
86
|
-
}
|
|
87
|
-
/**
|
|
88
|
-
* Checks if this is a valid hex address
|
|
89
|
-
* @param address hex address
|
|
90
|
-
*/
|
|
91
|
-
isValidHexAddress(address) {
|
|
92
|
-
return address.length === 42 && /^(0x)?([0-9a-f]{2})+$/i.test(address);
|
|
93
|
-
}
|
|
94
|
-
/**
|
|
95
|
-
* Generate ed25519 key pair
|
|
96
|
-
*
|
|
97
|
-
* @param seed
|
|
98
|
-
* @returns {Object} object with generated pub, prv
|
|
99
|
-
*/
|
|
100
|
-
generateKeyPair(seed) {
|
|
101
|
-
// TODO: move this and address creation logic to account-lib
|
|
102
|
-
if (!seed) {
|
|
103
|
-
// An extended private key has both a normal 256 bit private key and a 256 bit chain code, both of which must be
|
|
104
|
-
// random. 512 bits is therefore the maximum entropy and gives us maximum security against cracking.
|
|
105
|
-
seed = crypto_1.randomBytes(512 / 8);
|
|
106
|
-
}
|
|
107
|
-
const hd = utxo_lib_1.bip32.fromSeed(seed);
|
|
108
|
-
return {
|
|
109
|
-
pub: hd.neutered().toBase58(),
|
|
110
|
-
prv: hd.toBase58(),
|
|
111
|
-
};
|
|
112
|
-
}
|
|
113
|
-
isValidXpub(xpub) {
|
|
114
|
-
try {
|
|
115
|
-
return utxo_lib_1.bip32.fromBase58(xpub).isNeutered();
|
|
116
|
-
}
|
|
117
|
-
catch (e) {
|
|
118
|
-
return false;
|
|
119
|
-
}
|
|
120
|
-
}
|
|
121
|
-
isValidPub(pub) {
|
|
122
|
-
if (this.isValidXpub(pub)) {
|
|
123
|
-
// xpubs can be converted into regular pubs, so technically it is a valid pub
|
|
124
|
-
return true;
|
|
125
|
-
}
|
|
126
|
-
return new RegExp('^04[a-zA-Z0-9]{128}$').test(pub);
|
|
127
|
-
}
|
|
128
|
-
async parseTransaction(params) {
|
|
129
|
-
return {};
|
|
130
|
-
}
|
|
131
|
-
async isWalletAddress(params) {
|
|
132
|
-
throw new sdk_core_1.MethodNotImplementedError();
|
|
133
|
-
}
|
|
134
|
-
async verifyTransaction(params) {
|
|
135
|
-
return true;
|
|
136
|
-
}
|
|
137
|
-
/**
|
|
138
|
-
* Derive a user key using the chain path of the address
|
|
139
|
-
* @param key
|
|
140
|
-
* @param path
|
|
141
|
-
* @returns {string} derived private key
|
|
142
|
-
*/
|
|
143
|
-
deriveKeyWithPath({ key, path }) {
|
|
144
|
-
const keychain = utxo_lib_1.bip32.fromBase58(key);
|
|
145
|
-
const derivedKeyNode = keychain.derivePath(path);
|
|
146
|
-
return derivedKeyNode.toBase58();
|
|
147
|
-
}
|
|
148
|
-
/**
|
|
149
|
-
* Assemble keychain and half-sign prebuilt transaction
|
|
150
|
-
*
|
|
151
|
-
* @param params
|
|
152
|
-
* @param params.txPrebuild {Object} prebuild object returned by platform
|
|
153
|
-
* @param params.prv {String} user prv
|
|
154
|
-
* @param params.wallet.addressVersion {String} this is the version of the Algorand multisig address generation format
|
|
155
|
-
* @returns Bluebird<SignedTransaction>
|
|
156
|
-
*/
|
|
157
|
-
async signTransaction(params) {
|
|
158
|
-
var _a, _b;
|
|
159
|
-
const txBuilder = builder_1.getBuilder(this.getChain()).from(params.txPrebuild.txHex);
|
|
160
|
-
let key;
|
|
161
|
-
const { chain, index } = (_b = (_a = params.txPrebuild) === null || _a === void 0 ? void 0 : _a.addressInfo) !== null && _b !== void 0 ? _b : { chain: 0, index: 0 };
|
|
162
|
-
if (chain === 0 && index === 0) {
|
|
163
|
-
key = params.prv;
|
|
164
|
-
}
|
|
165
|
-
else {
|
|
166
|
-
const derivationPath = `0/0/${chain}/${index}`;
|
|
167
|
-
key = this.deriveKeyWithPath({ key: params.prv, path: derivationPath });
|
|
168
|
-
}
|
|
169
|
-
txBuilder.sign({ key });
|
|
170
|
-
const transaction = await txBuilder.build();
|
|
171
|
-
const response = {
|
|
172
|
-
txHex: JSON.stringify(transaction.toJson()),
|
|
173
|
-
};
|
|
174
|
-
if (transaction.toJson().signature.length >= 2) {
|
|
175
|
-
return response;
|
|
176
|
-
}
|
|
177
|
-
// Half signed transaction
|
|
178
|
-
return {
|
|
179
|
-
halfSigned: response,
|
|
180
|
-
};
|
|
181
|
-
}
|
|
182
|
-
/**
|
|
183
|
-
* Return boolean indicating whether input is valid seed for the coin
|
|
184
|
-
*
|
|
185
|
-
* @param prv - the prv to be checked
|
|
186
|
-
*/
|
|
187
|
-
isValidXprv(prv) {
|
|
188
|
-
try {
|
|
189
|
-
return !utxo_lib_1.bip32.fromBase58(prv).isNeutered();
|
|
190
|
-
}
|
|
191
|
-
catch {
|
|
192
|
-
return false;
|
|
193
|
-
}
|
|
194
|
-
}
|
|
195
|
-
/**
|
|
196
|
-
* Convert a message to string in hexadecimal format.
|
|
197
|
-
*
|
|
198
|
-
* @param message {Buffer|String} message to sign
|
|
199
|
-
* @return the message as a hexadecimal string
|
|
200
|
-
*/
|
|
201
|
-
toHexString(message) {
|
|
202
|
-
if (typeof message === 'string') {
|
|
203
|
-
return Buffer.from(message).toString('hex');
|
|
204
|
-
}
|
|
205
|
-
else if (Buffer.isBuffer(message)) {
|
|
206
|
-
return message.toString('hex');
|
|
207
|
-
}
|
|
208
|
-
else {
|
|
209
|
-
throw new Error('Invalid messaged passed to signMessage');
|
|
210
|
-
}
|
|
211
|
-
}
|
|
212
|
-
/**
|
|
213
|
-
* Sign message with private key
|
|
214
|
-
*
|
|
215
|
-
* @param key
|
|
216
|
-
* @param message
|
|
217
|
-
*/
|
|
218
|
-
async signMessage(key, message) {
|
|
219
|
-
var _a;
|
|
220
|
-
const toSign = this.toHexString(message);
|
|
221
|
-
let prv = key.prv;
|
|
222
|
-
if (this.isValidXprv(prv)) {
|
|
223
|
-
prv = (_a = utxo_lib_1.bip32.fromBase58(prv).privateKey) === null || _a === void 0 ? void 0 : _a.toString('hex');
|
|
224
|
-
}
|
|
225
|
-
if (!prv) {
|
|
226
|
-
throw new Error('no privateKey');
|
|
227
|
-
}
|
|
228
|
-
let sig = lib_1.Utils.signString(toSign, prv, true);
|
|
229
|
-
// remove the preceding 0x
|
|
230
|
-
sig = sig.replace(/^0x/, '');
|
|
231
|
-
return Buffer.from(sig, 'hex');
|
|
232
|
-
}
|
|
233
|
-
/**
|
|
234
|
-
* Converts an xpub to a uncompressed pub
|
|
235
|
-
* @param xpub
|
|
236
|
-
*/
|
|
237
|
-
xpubToUncompressedPub(xpub) {
|
|
238
|
-
if (!this.isValidXpub(xpub)) {
|
|
239
|
-
throw new Error('invalid xpub');
|
|
240
|
-
}
|
|
241
|
-
const publicKey = utxo_lib_1.bip32.fromBase58(xpub, utxo_lib_1.networks.bitcoin).publicKey;
|
|
242
|
-
return Buffer.from(secp256k1.publicKeyConvert(publicKey, false /* compressed */)).toString('hex');
|
|
243
|
-
}
|
|
244
|
-
/**
|
|
245
|
-
* Modify prebuild before sending it to the server.
|
|
246
|
-
* @param buildParams The whitelisted parameters for this prebuild
|
|
247
|
-
*/
|
|
248
|
-
async getExtraPrebuildParams(buildParams) {
|
|
249
|
-
if (buildParams.recipients[0].data && buildParams.feeLimit) {
|
|
250
|
-
buildParams.recipients[0].feeLimit = buildParams.feeLimit;
|
|
251
|
-
}
|
|
252
|
-
}
|
|
253
|
-
pubToHexAddress(pub) {
|
|
254
|
-
const byteArrayAddr = lib_1.Utils.getByteArrayFromHexAddress(pub);
|
|
255
|
-
const rawAddress = lib_1.Utils.getRawAddressFromPubKey(byteArrayAddr);
|
|
256
|
-
return lib_1.Utils.getHexAddressFromByteArray(rawAddress);
|
|
257
|
-
}
|
|
258
|
-
xprvToCompressedPrv(xprv) {
|
|
259
|
-
if (!this.isValidXprv(xprv)) {
|
|
260
|
-
throw new Error('invalid xprv');
|
|
261
|
-
}
|
|
262
|
-
const hdNode = utxo_lib_1.bip32.fromBase58(xprv, utxo_lib_1.networks.bitcoin);
|
|
263
|
-
if (!hdNode.privateKey) {
|
|
264
|
-
throw new Error('no privateKey');
|
|
265
|
-
}
|
|
266
|
-
return hdNode.privateKey.toString('hex');
|
|
267
|
-
}
|
|
268
|
-
/**
|
|
269
|
-
* Make a query to Trongrid for information such as balance, token balance, solidity calls
|
|
270
|
-
* @param query {Object} key-value pairs of parameters to append after /api
|
|
271
|
-
* @returns {Object} response from Trongrid
|
|
272
|
-
*/
|
|
273
|
-
async recoveryPost(query) {
|
|
274
|
-
let nodeUri = '';
|
|
275
|
-
switch (query.node) {
|
|
276
|
-
case NodeTypes.Full:
|
|
277
|
-
nodeUri = sdk_core_1.common.Environments[this.bitgo.getEnv()].tronNodes.full;
|
|
278
|
-
break;
|
|
279
|
-
case NodeTypes.Solidity:
|
|
280
|
-
nodeUri = sdk_core_1.common.Environments[this.bitgo.getEnv()].tronNodes.solidity;
|
|
281
|
-
break;
|
|
282
|
-
default:
|
|
283
|
-
throw new Error('node type not found');
|
|
284
|
-
}
|
|
285
|
-
const response = await request
|
|
286
|
-
.post(nodeUri + query.path)
|
|
287
|
-
.type('json')
|
|
288
|
-
.send(query.jsonObj);
|
|
289
|
-
if (!response.ok) {
|
|
290
|
-
throw new Error('could not reach Tron node');
|
|
291
|
-
}
|
|
292
|
-
// unfortunately, it doesn't look like most TRON nodes return valid json as body
|
|
293
|
-
return JSON.parse(response.text);
|
|
294
|
-
}
|
|
295
|
-
/**
|
|
296
|
-
* Make a query to Trongrid for information such as balance, token balance, solidity calls
|
|
297
|
-
* @param query {Object} key-value pairs of parameters to append after /api
|
|
298
|
-
* @returns {Object} response from Trongrid
|
|
299
|
-
*/
|
|
300
|
-
async recoveryGet(query) {
|
|
301
|
-
let nodeUri = '';
|
|
302
|
-
switch (query.node) {
|
|
303
|
-
case NodeTypes.Full:
|
|
304
|
-
nodeUri = sdk_core_1.common.Environments[this.bitgo.getEnv()].tronNodes.full;
|
|
305
|
-
break;
|
|
306
|
-
case NodeTypes.Solidity:
|
|
307
|
-
nodeUri = sdk_core_1.common.Environments[this.bitgo.getEnv()].tronNodes.solidity;
|
|
308
|
-
break;
|
|
309
|
-
default:
|
|
310
|
-
throw new Error('node type not found');
|
|
311
|
-
}
|
|
312
|
-
const response = await request
|
|
313
|
-
.get(nodeUri + query.path)
|
|
314
|
-
.type('json')
|
|
315
|
-
.send(query.jsonObj);
|
|
316
|
-
if (!response.ok) {
|
|
317
|
-
throw new Error('could not reach Tron node');
|
|
318
|
-
}
|
|
319
|
-
// unfortunately, it doesn't look like most TRON nodes return valid json as body
|
|
320
|
-
return JSON.parse(response.text);
|
|
321
|
-
}
|
|
322
|
-
/**
|
|
323
|
-
* Query our explorer for the balance of an address
|
|
324
|
-
* @param address {String} the address encoded in hex
|
|
325
|
-
* @returns {BigNumber} address balance
|
|
326
|
-
*/
|
|
327
|
-
async getAccountBalancesFromNode(address) {
|
|
328
|
-
return await this.recoveryGet({
|
|
329
|
-
path: '/v1/accounts/' + address,
|
|
330
|
-
jsonObj: {},
|
|
331
|
-
node: NodeTypes.Full,
|
|
332
|
-
});
|
|
333
|
-
}
|
|
334
|
-
/**
|
|
335
|
-
* Retrieves our build transaction from a node.
|
|
336
|
-
* @param toAddr hex-encoded address
|
|
337
|
-
* @param fromAddr hex-encoded address
|
|
338
|
-
* @param amount
|
|
339
|
-
*/
|
|
340
|
-
async getBuildTransaction(toAddr, fromAddr, amount) {
|
|
341
|
-
// our addresses should be base58, we'll have to encode to hex
|
|
342
|
-
return await this.recoveryPost({
|
|
343
|
-
path: '/wallet/createtransaction',
|
|
344
|
-
jsonObj: {
|
|
345
|
-
to_address: toAddr,
|
|
346
|
-
owner_address: fromAddr,
|
|
347
|
-
amount,
|
|
348
|
-
},
|
|
349
|
-
node: NodeTypes.Full,
|
|
350
|
-
});
|
|
351
|
-
}
|
|
352
|
-
/**
|
|
353
|
-
* Retrieves our build transaction from a node.
|
|
354
|
-
* @param toAddr hex-encoded address
|
|
355
|
-
* @param fromAddr hex-encoded address
|
|
356
|
-
* @param amount
|
|
357
|
-
*/
|
|
358
|
-
async getTriggerSmartContractTransaction(toAddr, fromAddr, amount, contractAddr) {
|
|
359
|
-
const functionSelector = 'transfer(address,uint256)';
|
|
360
|
-
const types = ['address', 'uint256'];
|
|
361
|
-
const values = [toAddr, amount];
|
|
362
|
-
const parameter = lib_1.Utils.encodeDataParams(types, values, '');
|
|
363
|
-
return await this.recoveryPost({
|
|
364
|
-
path: '/wallet/triggersmartcontract',
|
|
365
|
-
jsonObj: {
|
|
366
|
-
owner_address: fromAddr,
|
|
367
|
-
contract_address: contractAddr,
|
|
368
|
-
function_selector: functionSelector,
|
|
369
|
-
parameter: parameter,
|
|
370
|
-
fee_limit: 100000000,
|
|
371
|
-
},
|
|
372
|
-
node: NodeTypes.Full,
|
|
373
|
-
});
|
|
374
|
-
}
|
|
375
|
-
/**
|
|
376
|
-
* Throws an error if any keys in the ownerKeys collection don't match the keys array we pass
|
|
377
|
-
* @param ownerKeys
|
|
378
|
-
* @param keys
|
|
379
|
-
*/
|
|
380
|
-
checkPermissions(ownerKeys, keys) {
|
|
381
|
-
keys = keys.map((k) => k.toUpperCase());
|
|
382
|
-
ownerKeys.map((key) => {
|
|
383
|
-
const hexKey = key.address.toUpperCase();
|
|
384
|
-
if (!keys.includes(hexKey)) {
|
|
385
|
-
throw new Error(`pub address ${hexKey} not found in account`);
|
|
386
|
-
}
|
|
387
|
-
if (key.weight !== 1) {
|
|
388
|
-
throw new Error('owner permission is invalid for this structure');
|
|
389
|
-
}
|
|
390
|
-
});
|
|
391
|
-
}
|
|
392
|
-
/**
|
|
393
|
-
* Builds a funds recovery transaction without BitGo.
|
|
394
|
-
* We need to do three queries during this:
|
|
395
|
-
* 1) Node query - how much money is in the account
|
|
396
|
-
* 2) Build transaction - build our transaction for the amount
|
|
397
|
-
* 3) Send signed build - send our signed build to a public node
|
|
398
|
-
* @param params
|
|
399
|
-
*/
|
|
400
|
-
async recover(params) {
|
|
401
|
-
const isKrsRecovery = sdk_core_1.getIsKrsRecovery(params);
|
|
402
|
-
const isUnsignedSweep = sdk_core_1.getIsUnsignedSweep(params);
|
|
403
|
-
if (!this.isValidAddress(params.recoveryDestination)) {
|
|
404
|
-
throw new Error('Invalid destination address!');
|
|
405
|
-
}
|
|
406
|
-
// get our user, backup keys
|
|
407
|
-
const keys = sdk_core_1.getBip32Keys(this.bitgo, params, { requireBitGoXpub: false });
|
|
408
|
-
// we need to decode our bitgoKey to a base58 address
|
|
409
|
-
const bitgoHexAddr = this.pubToHexAddress(this.xpubToUncompressedPub(params.bitgoKey));
|
|
410
|
-
const recoveryAddressHex = lib_1.Utils.getHexAddressFromBase58Address(params.recoveryDestination);
|
|
411
|
-
// call the node to get our account balance
|
|
412
|
-
const account = await this.getAccountBalancesFromNode(lib_1.Utils.getBase58AddressFromHex(bitgoHexAddr));
|
|
413
|
-
const recoveryAmount = account.data[0].balance;
|
|
414
|
-
const userXPub = keys[0].neutered().toBase58();
|
|
415
|
-
const userXPrv = keys[0].toBase58();
|
|
416
|
-
const backupXPub = keys[1].neutered().toBase58();
|
|
417
|
-
// first construct token txns
|
|
418
|
-
const tokenTxns = [];
|
|
419
|
-
for (const token of account.data[0].trc20) {
|
|
420
|
-
// mainnet tokens
|
|
421
|
-
if (token.TEkxiTehnzSmSe2XqrBj4w32RUN966rdz8) {
|
|
422
|
-
const amount = token.TEkxiTehnzSmSe2XqrBj4w32RUN966rdz8;
|
|
423
|
-
const contractAddr = lib_1.Utils.getHexAddressFromBase58Address('TEkxiTehnzSmSe2XqrBj4w32RUN966rdz8');
|
|
424
|
-
tokenTxns.push((await this.getTriggerSmartContractTransaction(recoveryAddressHex, bitgoHexAddr, amount, contractAddr))
|
|
425
|
-
.transaction);
|
|
426
|
-
}
|
|
427
|
-
else if (token.TR7NHqjeKQxGTCi8q8ZY4pL8otSzgjLj6t) {
|
|
428
|
-
const amount = token.TR7NHqjeKQxGTCi8q8ZY4pL8otSzgjLj6t;
|
|
429
|
-
const contractAddr = lib_1.Utils.getHexAddressFromBase58Address('TR7NHqjeKQxGTCi8q8ZY4pL8otSzgjLj6t');
|
|
430
|
-
tokenTxns.push((await this.getTriggerSmartContractTransaction(recoveryAddressHex, bitgoHexAddr, amount, contractAddr))
|
|
431
|
-
.transaction);
|
|
432
|
-
// testnet tokens
|
|
433
|
-
}
|
|
434
|
-
else if (token.TSdZwNqpHofzP6BsBKGQUWdBeJphLmF6id) {
|
|
435
|
-
const amount = token.TSdZwNqpHofzP6BsBKGQUWdBeJphLmF6id;
|
|
436
|
-
const contractAddr = lib_1.Utils.getHexAddressFromBase58Address('TSdZwNqpHofzP6BsBKGQUWdBeJphLmF6id');
|
|
437
|
-
tokenTxns.push((await this.getTriggerSmartContractTransaction(recoveryAddressHex, bitgoHexAddr, amount, contractAddr))
|
|
438
|
-
.transaction);
|
|
439
|
-
}
|
|
440
|
-
else if (token.TG3XXyExBkPp9nzdajDZsozEu4BkaSJozs) {
|
|
441
|
-
const amount = token.TG3XXyExBkPp9nzdajDZsozEu4BkaSJozs;
|
|
442
|
-
const contractAddr = lib_1.Utils.getHexAddressFromBase58Address('TG3XXyExBkPp9nzdajDZsozEu4BkaSJozs');
|
|
443
|
-
tokenTxns.push((await this.getTriggerSmartContractTransaction(recoveryAddressHex, bitgoHexAddr, amount, contractAddr))
|
|
444
|
-
.transaction);
|
|
445
|
-
}
|
|
446
|
-
}
|
|
447
|
-
// construct the tx -
|
|
448
|
-
// there's an assumption here being made about fees: for a wallet that hasn't been used in awhile, the implication is
|
|
449
|
-
// it has maximum bandwidth. thus, a recovery should cost the minimum amount (1e6 sun or 1 Tron)
|
|
450
|
-
if (exports.MINIMUM_TRON_MSIG_TRANSACTION_FEE > recoveryAmount) {
|
|
451
|
-
throw new Error('Amount of funds to recover wouldnt be able to fund a send');
|
|
452
|
-
}
|
|
453
|
-
const keyHexAddresses = [
|
|
454
|
-
this.pubToHexAddress(this.xpubToUncompressedPub(userXPub)),
|
|
455
|
-
this.pubToHexAddress(this.xpubToUncompressedPub(backupXPub)),
|
|
456
|
-
bitgoHexAddr,
|
|
457
|
-
];
|
|
458
|
-
// run checks to ensure this is a valid tx - permissions match our signer keys
|
|
459
|
-
const ownerKeys = [];
|
|
460
|
-
for (const key of account.data[0].owner_permission.keys) {
|
|
461
|
-
const address = lib_1.Utils.getHexAddressFromBase58Address(key.address);
|
|
462
|
-
const weight = key.weight;
|
|
463
|
-
ownerKeys.push({ address, weight });
|
|
464
|
-
}
|
|
465
|
-
const activePermissionKeys = [];
|
|
466
|
-
for (const key of account.data[0].active_permission[0].keys) {
|
|
467
|
-
const address = lib_1.Utils.getHexAddressFromBase58Address(key.address);
|
|
468
|
-
const weight = key.weight;
|
|
469
|
-
activePermissionKeys.push({ address, weight });
|
|
470
|
-
}
|
|
471
|
-
this.checkPermissions(ownerKeys, keyHexAddresses);
|
|
472
|
-
this.checkPermissions(activePermissionKeys, keyHexAddresses);
|
|
473
|
-
// build and sign token txns
|
|
474
|
-
const finalTokenTxs = [];
|
|
475
|
-
for (const tokenTxn of tokenTxns) {
|
|
476
|
-
const txBuilder = builder_1.getBuilder(this.getChain()).from(tokenTxn);
|
|
477
|
-
// this tx should be enough to drop into a node
|
|
478
|
-
if (isUnsignedSweep) {
|
|
479
|
-
finalTokenTxs.push((await txBuilder.build()).toJson());
|
|
480
|
-
continue;
|
|
481
|
-
}
|
|
482
|
-
const userPrv = this.xprvToCompressedPrv(userXPrv);
|
|
483
|
-
txBuilder.sign({ key: userPrv });
|
|
484
|
-
// krs recoveries don't get signed
|
|
485
|
-
if (!isKrsRecovery) {
|
|
486
|
-
const backupXPrv = keys[1].toBase58();
|
|
487
|
-
const backupPrv = this.xprvToCompressedPrv(backupXPrv);
|
|
488
|
-
txBuilder.sign({ key: backupPrv });
|
|
489
|
-
}
|
|
490
|
-
finalTokenTxs.push((await txBuilder.build()).toJson());
|
|
491
|
-
}
|
|
492
|
-
// tokens must be recovered before the native asset, so that there is sufficient of the native asset to cover fees
|
|
493
|
-
if (finalTokenTxs.length > 0) {
|
|
494
|
-
return {
|
|
495
|
-
tokenTxs: finalTokenTxs,
|
|
496
|
-
};
|
|
497
|
-
}
|
|
498
|
-
const recoveryAmountMinusFees = recoveryAmount - exports.MINIMUM_TRON_MSIG_TRANSACTION_FEE;
|
|
499
|
-
const buildTx = await this.getBuildTransaction(recoveryAddressHex, bitgoHexAddr, recoveryAmountMinusFees);
|
|
500
|
-
// construct our tx
|
|
501
|
-
const txBuilder = builder_1.getBuilder(this.getChain()).from(buildTx);
|
|
502
|
-
// this tx should be enough to drop into a node
|
|
503
|
-
if (isUnsignedSweep) {
|
|
504
|
-
return {
|
|
505
|
-
tx: (await txBuilder.build()).toJson(),
|
|
506
|
-
recoveryAmount: recoveryAmountMinusFees,
|
|
507
|
-
};
|
|
508
|
-
}
|
|
509
|
-
const userPrv = this.xprvToCompressedPrv(userXPrv);
|
|
510
|
-
txBuilder.sign({ key: userPrv });
|
|
511
|
-
// krs recoveries don't get signed
|
|
512
|
-
if (!isKrsRecovery) {
|
|
513
|
-
const backupXPrv = keys[1].toBase58();
|
|
514
|
-
const backupPrv = this.xprvToCompressedPrv(backupXPrv);
|
|
515
|
-
txBuilder.sign({ key: backupPrv });
|
|
516
|
-
}
|
|
517
|
-
return {
|
|
518
|
-
tx: (await txBuilder.build()).toJson(),
|
|
519
|
-
recoveryAmount: recoveryAmountMinusFees,
|
|
520
|
-
};
|
|
521
|
-
}
|
|
522
|
-
/**
|
|
523
|
-
* Explain a Tron transaction from txHex
|
|
524
|
-
* @param params
|
|
525
|
-
*/
|
|
526
|
-
async explainTransaction(params) {
|
|
527
|
-
const txHex = params.txHex || (params.halfSigned && params.halfSigned.txHex);
|
|
528
|
-
if (!txHex || !params.feeInfo) {
|
|
529
|
-
throw new Error('missing explain tx parameters');
|
|
530
|
-
}
|
|
531
|
-
const txBuilder = builder_1.getBuilder(this.getChain()).from(txHex);
|
|
532
|
-
const tx = await txBuilder.build();
|
|
533
|
-
const outputs = [
|
|
534
|
-
{
|
|
535
|
-
amount: tx.outputs[0].value.toString(),
|
|
536
|
-
address: tx.outputs[0].address, // Should turn it into a readable format, aka base58
|
|
537
|
-
},
|
|
538
|
-
];
|
|
539
|
-
const displayOrder = [
|
|
540
|
-
'id',
|
|
541
|
-
'outputAmount',
|
|
542
|
-
'changeAmount',
|
|
543
|
-
'outputs',
|
|
544
|
-
'changeOutputs',
|
|
545
|
-
'fee',
|
|
546
|
-
'timestamp',
|
|
547
|
-
'expiration',
|
|
548
|
-
];
|
|
549
|
-
return {
|
|
550
|
-
displayOrder,
|
|
551
|
-
id: tx.id,
|
|
552
|
-
outputs,
|
|
553
|
-
outputAmount: outputs[0].amount,
|
|
554
|
-
changeOutputs: [],
|
|
555
|
-
changeAmount: '0',
|
|
556
|
-
fee: params.feeInfo,
|
|
557
|
-
timestamp: tx.validFrom,
|
|
558
|
-
expiration: tx.validTo,
|
|
559
|
-
};
|
|
560
|
-
}
|
|
561
|
-
}
|
|
562
|
-
exports.Trx = Trx;
|
|
563
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"trx.js","sourceRoot":"","sources":["../../src/trx.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport * as secp256k1 from 'secp256k1';\nimport { randomBytes } from 'crypto';\nimport { CoinFamily, BaseCoin as StaticsBaseCoin } from '@bitgo-beta/statics';\nimport { bip32, networks } from '@bitgo-beta/utxo-lib';\nimport * as request from 'superagent';\nimport {\n  BaseCoin,\n  BitGoBase,\n  common,\n  getBip32Keys,\n  getIsKrsRecovery,\n  getIsUnsignedSweep,\n  KeyPair,\n  MethodNotImplementedError,\n  ParsedTransaction,\n  ParseTransactionOptions,\n  SignedTransaction,\n  SignTransactionOptions,\n  TransactionExplanation,\n  TransactionFee,\n  TransactionPrebuild as BaseTransactionPrebuild,\n  TransactionRecipient as Recipient,\n  VerifyAddressOptions,\n  VerifyTransactionOptions,\n} from '@bitgo-beta/sdk-core';\nimport { Interface, Utils, WrappedBuilder } from './lib';\nimport { getBuilder } from './lib/builder';\nimport { TransactionReceipt } from './lib/iface';\n\nexport const MINIMUM_TRON_MSIG_TRANSACTION_FEE = 1e6;\n\nexport interface TronSignTransactionOptions extends SignTransactionOptions {\n  txPrebuild: TransactionPrebuild;\n  prv: string;\n}\n\nexport interface TxInfo {\n  recipients: Recipient[];\n  from: string;\n  txid: string;\n}\n\nexport interface AddressInfo {\n  address: string;\n  chain: number;\n  index: number;\n}\n\nexport interface TronTransactionExplanation extends TransactionExplanation {\n  expiration: number;\n  timestamp: number;\n}\n\nexport interface TransactionPrebuild extends BaseTransactionPrebuild {\n  txHex: string;\n  txInfo: TxInfo;\n  addressInfo?: AddressInfo;\n  feeInfo: TransactionFee;\n}\n\nexport interface ExplainTransactionOptions {\n  txHex?: string; // txHex is poorly named here; it is just a wrapped JSON object\n  halfSigned?: {\n    txHex: string; // txHex is poorly named here; it is just a wrapped JSON object\n  };\n  feeInfo: TransactionFee;\n}\n\nexport interface RecoveryOptions {\n  userKey: string; // Box A\n  backupKey: string; // Box B\n  bitgoKey: string; // Box C - this is bitgo's xpub and will be used to derive their root address\n  recoveryDestination: string; // base58 address\n  krsProvider?: string;\n  walletPassphrase?: string;\n}\n\nexport interface RecoveryTransaction {\n  tx?: TransactionPrebuild;\n  recoveryAmount?: number;\n  tokenTxs?: TransactionReceipt[];\n}\n\nexport enum NodeTypes {\n  Full,\n  Solidity,\n}\n\n/**\n * This structure is not a complete model of the AccountResponse from a node.\n */\nexport interface AccountResponse {\n  data: [Interface.AccountInfo];\n}\n\nexport class Trx extends BaseCoin {\n  protected readonly _staticsCoin: Readonly<StaticsBaseCoin>;\n\n  constructor(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>) {\n    super(bitgo);\n\n    if (!staticsCoin) {\n      throw new Error('missing required constructor parameter staticsCoin');\n    }\n\n    this._staticsCoin = staticsCoin;\n  }\n\n  getChain() {\n    return this._staticsCoin.name;\n  }\n\n  getFamily(): CoinFamily {\n    return this._staticsCoin.family;\n  }\n\n  getFullName() {\n    return this._staticsCoin.fullName;\n  }\n\n  getBaseFactor() {\n    return Math.pow(10, this._staticsCoin.decimalPlaces);\n  }\n\n  /** @inheritdoc */\n  transactionDataAllowed() {\n    return true;\n  }\n\n  static createInstance(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>): BaseCoin {\n    return new Trx(bitgo, staticsCoin);\n  }\n\n  /**\n   * Flag for sending value of 0\n   * @returns {boolean} True if okay to send 0 value, false otherwise\n   */\n  valuelessTransferAllowed(): boolean {\n    return true;\n  }\n\n  /** @inheritDoc */\n  allowsAccountConsolidations(): boolean {\n    return true;\n  }\n\n  /**\n   * Checks if this is a valid base58 or hex address\n   * @param address\n   */\n  isValidAddress(address: string): boolean {\n    if (!address) {\n      return false;\n    }\n    return this.isValidHexAddress(address) || Utils.isBase58Address(address);\n  }\n\n  /**\n   * Checks if this is a valid hex address\n   * @param address hex address\n   */\n  isValidHexAddress(address: string): boolean {\n    return address.length === 42 && /^(0x)?([0-9a-f]{2})+$/i.test(address);\n  }\n\n  /**\n   * Generate ed25519 key pair\n   *\n   * @param seed\n   * @returns {Object} object with generated pub, prv\n   */\n  generateKeyPair(seed?: Buffer): KeyPair {\n    // TODO: move this and address creation logic to account-lib\n    if (!seed) {\n      // An extended private key has both a normal 256 bit private key and a 256 bit chain code, both of which must be\n      // random. 512 bits is therefore the maximum entropy and gives us maximum security against cracking.\n      seed = randomBytes(512 / 8);\n    }\n    const hd = bip32.fromSeed(seed);\n    return {\n      pub: hd.neutered().toBase58(),\n      prv: hd.toBase58(),\n    };\n  }\n\n  isValidXpub(xpub: string): boolean {\n    try {\n      return bip32.fromBase58(xpub).isNeutered();\n    } catch (e) {\n      return false;\n    }\n  }\n\n  isValidPub(pub: string): boolean {\n    if (this.isValidXpub(pub)) {\n      // xpubs can be converted into regular pubs, so technically it is a valid pub\n      return true;\n    }\n    return new RegExp('^04[a-zA-Z0-9]{128}$').test(pub);\n  }\n\n  async parseTransaction(params: ParseTransactionOptions): Promise<ParsedTransaction> {\n    return {};\n  }\n\n  async isWalletAddress(params: VerifyAddressOptions): Promise<boolean> {\n    throw new MethodNotImplementedError();\n  }\n\n  async verifyTransaction(params: VerifyTransactionOptions): Promise<boolean> {\n    return true;\n  }\n\n  /**\n   * Derive a user key using the chain path of the address\n   * @param key\n   * @param path\n   * @returns {string} derived private key\n   */\n  deriveKeyWithPath({ key, path }: { key: string; path: string }): string {\n    const keychain = bip32.fromBase58(key);\n    const derivedKeyNode = keychain.derivePath(path);\n    return derivedKeyNode.toBase58();\n  }\n\n  /**\n   * Assemble keychain and half-sign prebuilt transaction\n   *\n   * @param params\n   * @param params.txPrebuild {Object} prebuild object returned by platform\n   * @param params.prv {String} user prv\n   * @param params.wallet.addressVersion {String} this is the version of the Algorand multisig address generation format\n   * @returns Bluebird<SignedTransaction>\n   */\n  async signTransaction(params: TronSignTransactionOptions): Promise<SignedTransaction> {\n    const txBuilder = getBuilder(this.getChain()).from(params.txPrebuild.txHex);\n\n    let key;\n    const { chain, index } = params.txPrebuild?.addressInfo ?? { chain: 0, index: 0 };\n    if (chain === 0 && index === 0) {\n      key = params.prv;\n    } else {\n      const derivationPath = `0/0/${chain}/${index}`;\n      key = this.deriveKeyWithPath({ key: params.prv, path: derivationPath });\n    }\n    txBuilder.sign({ key });\n\n    const transaction = await txBuilder.build();\n    const response = {\n      txHex: JSON.stringify(transaction.toJson()),\n    };\n    if (transaction.toJson().signature.length >= 2) {\n      return response;\n    }\n    // Half signed transaction\n    return {\n      halfSigned: response,\n    };\n  }\n\n  /**\n   * Return boolean indicating whether input is valid seed for the coin\n   *\n   * @param prv - the prv to be checked\n   */\n  isValidXprv(prv: string): boolean {\n    try {\n      return !bip32.fromBase58(prv).isNeutered();\n    } catch {\n      return false;\n    }\n  }\n\n  /**\n   * Convert a message to string in hexadecimal format.\n   *\n   * @param message {Buffer|String} message to sign\n   * @return the message as a hexadecimal string\n   */\n  toHexString(message: string | Buffer): string {\n    if (typeof message === 'string') {\n      return Buffer.from(message).toString('hex');\n    } else if (Buffer.isBuffer(message)) {\n      return message.toString('hex');\n    } else {\n      throw new Error('Invalid messaged passed to signMessage');\n    }\n  }\n\n  /**\n   * Sign message with private key\n   *\n   * @param key\n   * @param message\n   */\n  async signMessage(key: KeyPair, message: string | Buffer): Promise<Buffer> {\n    const toSign = this.toHexString(message);\n\n    let prv: string | undefined = key.prv;\n    if (this.isValidXprv(prv)) {\n      prv = bip32.fromBase58(prv).privateKey?.toString('hex');\n    }\n\n    if (!prv) {\n      throw new Error('no privateKey');\n    }\n    let sig = Utils.signString(toSign, prv, true);\n\n    // remove the preceding 0x\n    sig = sig.replace(/^0x/, '');\n\n    return Buffer.from(sig, 'hex');\n  }\n\n  /**\n   * Converts an xpub to a uncompressed pub\n   * @param xpub\n   */\n  xpubToUncompressedPub(xpub: string): string {\n    if (!this.isValidXpub(xpub)) {\n      throw new Error('invalid xpub');\n    }\n\n    const publicKey = bip32.fromBase58(xpub, networks.bitcoin).publicKey;\n    return Buffer.from(secp256k1.publicKeyConvert(publicKey, false /* compressed */)).toString('hex');\n  }\n\n  /**\n   * Modify prebuild before sending it to the server.\n   * @param buildParams The whitelisted parameters for this prebuild\n   */\n  async getExtraPrebuildParams(buildParams: any): Promise<any> {\n    if (buildParams.recipients[0].data && buildParams.feeLimit) {\n      buildParams.recipients[0].feeLimit = buildParams.feeLimit;\n    }\n  }\n\n  pubToHexAddress(pub: string): string {\n    const byteArrayAddr = Utils.getByteArrayFromHexAddress(pub);\n    const rawAddress = Utils.getRawAddressFromPubKey(byteArrayAddr);\n    return Utils.getHexAddressFromByteArray(rawAddress);\n  }\n\n  xprvToCompressedPrv(xprv: string): string {\n    if (!this.isValidXprv(xprv)) {\n      throw new Error('invalid xprv');\n    }\n\n    const hdNode = bip32.fromBase58(xprv, networks.bitcoin);\n    if (!hdNode.privateKey) {\n      throw new Error('no privateKey');\n    }\n    return hdNode.privateKey.toString('hex');\n  }\n\n  /**\n   * Make a query to Trongrid for information such as balance, token balance, solidity calls\n   * @param query {Object} key-value pairs of parameters to append after /api\n   * @returns {Object} response from Trongrid\n   */\n  private async recoveryPost(query: { path: string; jsonObj: any; node: NodeTypes }): Promise<any> {\n    let nodeUri = '';\n    switch (query.node) {\n      case NodeTypes.Full:\n        nodeUri = common.Environments[this.bitgo.getEnv()].tronNodes.full;\n        break;\n      case NodeTypes.Solidity:\n        nodeUri = common.Environments[this.bitgo.getEnv()].tronNodes.solidity;\n        break;\n      default:\n        throw new Error('node type not found');\n    }\n\n    const response = await request\n      .post(nodeUri + query.path)\n      .type('json')\n      .send(query.jsonObj);\n\n    if (!response.ok) {\n      throw new Error('could not reach Tron node');\n    }\n\n    // unfortunately, it doesn't look like most TRON nodes return valid json as body\n    return JSON.parse(response.text);\n  }\n\n  /**\n   * Make a query to Trongrid for information such as balance, token balance, solidity calls\n   * @param query {Object} key-value pairs of parameters to append after /api\n   * @returns {Object} response from Trongrid\n   */\n  private async recoveryGet(query: { path: string; jsonObj: any; node: NodeTypes }): Promise<any> {\n    let nodeUri = '';\n    switch (query.node) {\n      case NodeTypes.Full:\n        nodeUri = common.Environments[this.bitgo.getEnv()].tronNodes.full;\n        break;\n      case NodeTypes.Solidity:\n        nodeUri = common.Environments[this.bitgo.getEnv()].tronNodes.solidity;\n        break;\n      default:\n        throw new Error('node type not found');\n    }\n\n    const response = await request\n      .get(nodeUri + query.path)\n      .type('json')\n      .send(query.jsonObj);\n\n    if (!response.ok) {\n      throw new Error('could not reach Tron node');\n    }\n\n    // unfortunately, it doesn't look like most TRON nodes return valid json as body\n    return JSON.parse(response.text);\n  }\n\n  /**\n   * Query our explorer for the balance of an address\n   * @param address {String} the address encoded in hex\n   * @returns {BigNumber} address balance\n   */\n  private async getAccountBalancesFromNode(address: string): Promise<AccountResponse> {\n    return await this.recoveryGet({\n      path: '/v1/accounts/' + address,\n      jsonObj: {},\n      node: NodeTypes.Full,\n    });\n  }\n\n  /**\n   * Retrieves our build transaction from a node.\n   * @param toAddr hex-encoded address\n   * @param fromAddr hex-encoded address\n   * @param amount\n   */\n  private async getBuildTransaction(\n    toAddr: string,\n    fromAddr: string,\n    amount: number\n  ): Promise<Interface.TransactionReceipt> {\n    // our addresses should be base58, we'll have to encode to hex\n    return await this.recoveryPost({\n      path: '/wallet/createtransaction',\n      jsonObj: {\n        to_address: toAddr,\n        owner_address: fromAddr,\n        amount,\n      },\n      node: NodeTypes.Full,\n    });\n  }\n\n  /**\n   * Retrieves our build transaction from a node.\n   * @param toAddr hex-encoded address\n   * @param fromAddr hex-encoded address\n   * @param amount\n   */\n  private async getTriggerSmartContractTransaction(\n    toAddr: string,\n    fromAddr: string,\n    amount: string,\n    contractAddr: string\n  ): Promise<{ transaction: Interface.TransactionReceipt }> {\n    const functionSelector = 'transfer(address,uint256)';\n    const types = ['address', 'uint256'];\n    const values = [toAddr, amount];\n    const parameter = Utils.encodeDataParams(types, values, '');\n    return await this.recoveryPost({\n      path: '/wallet/triggersmartcontract',\n      jsonObj: {\n        owner_address: fromAddr,\n        contract_address: contractAddr,\n        function_selector: functionSelector,\n        parameter: parameter,\n        fee_limit: 100000000,\n      },\n      node: NodeTypes.Full,\n    });\n  }\n\n  /**\n   * Throws an error if any keys in the ownerKeys collection don't match the keys array we pass\n   * @param ownerKeys\n   * @param keys\n   */\n  checkPermissions(ownerKeys: { address: string; weight: number }[], keys: string[]) {\n    keys = keys.map((k) => k.toUpperCase());\n\n    ownerKeys.map((key) => {\n      const hexKey = key.address.toUpperCase();\n      if (!keys.includes(hexKey)) {\n        throw new Error(`pub address ${hexKey} not found in account`);\n      }\n\n      if (key.weight !== 1) {\n        throw new Error('owner permission is invalid for this structure');\n      }\n    });\n  }\n\n  /**\n   * Builds a funds recovery transaction without BitGo.\n   * We need to do three queries during this:\n   * 1) Node query - how much money is in the account\n   * 2) Build transaction - build our transaction for the amount\n   * 3) Send signed build - send our signed build to a public node\n   * @param params\n   */\n  async recover(params: RecoveryOptions): Promise<RecoveryTransaction> {\n    const isKrsRecovery = getIsKrsRecovery(params);\n    const isUnsignedSweep = getIsUnsignedSweep(params);\n\n    if (!this.isValidAddress(params.recoveryDestination)) {\n      throw new Error('Invalid destination address!');\n    }\n\n    // get our user, backup keys\n    const keys = getBip32Keys(this.bitgo, params, { requireBitGoXpub: false });\n\n    // we need to decode our bitgoKey to a base58 address\n    const bitgoHexAddr = this.pubToHexAddress(this.xpubToUncompressedPub(params.bitgoKey));\n    const recoveryAddressHex = Utils.getHexAddressFromBase58Address(params.recoveryDestination);\n\n    // call the node to get our account balance\n    const account = await this.getAccountBalancesFromNode(Utils.getBase58AddressFromHex(bitgoHexAddr));\n    const recoveryAmount = account.data[0].balance;\n\n    const userXPub = keys[0].neutered().toBase58();\n    const userXPrv = keys[0].toBase58();\n    const backupXPub = keys[1].neutered().toBase58();\n\n    // first construct token txns\n    const tokenTxns: any = [];\n    for (const token of account.data[0].trc20) {\n      // mainnet tokens\n      if (token.TEkxiTehnzSmSe2XqrBj4w32RUN966rdz8) {\n        const amount = token.TEkxiTehnzSmSe2XqrBj4w32RUN966rdz8;\n        const contractAddr = Utils.getHexAddressFromBase58Address('TEkxiTehnzSmSe2XqrBj4w32RUN966rdz8');\n        tokenTxns.push(\n          (await this.getTriggerSmartContractTransaction(recoveryAddressHex, bitgoHexAddr, amount, contractAddr))\n            .transaction\n        );\n      } else if (token.TR7NHqjeKQxGTCi8q8ZY4pL8otSzgjLj6t) {\n        const amount = token.TR7NHqjeKQxGTCi8q8ZY4pL8otSzgjLj6t;\n        const contractAddr = Utils.getHexAddressFromBase58Address('TR7NHqjeKQxGTCi8q8ZY4pL8otSzgjLj6t');\n        tokenTxns.push(\n          (await this.getTriggerSmartContractTransaction(recoveryAddressHex, bitgoHexAddr, amount, contractAddr))\n            .transaction\n        );\n\n        // testnet tokens\n      } else if (token.TSdZwNqpHofzP6BsBKGQUWdBeJphLmF6id) {\n        const amount = token.TSdZwNqpHofzP6BsBKGQUWdBeJphLmF6id;\n        const contractAddr = Utils.getHexAddressFromBase58Address('TSdZwNqpHofzP6BsBKGQUWdBeJphLmF6id');\n        tokenTxns.push(\n          (await this.getTriggerSmartContractTransaction(recoveryAddressHex, bitgoHexAddr, amount, contractAddr))\n            .transaction\n        );\n      } else if (token.TG3XXyExBkPp9nzdajDZsozEu4BkaSJozs) {\n        const amount = token.TG3XXyExBkPp9nzdajDZsozEu4BkaSJozs;\n        const contractAddr = Utils.getHexAddressFromBase58Address('TG3XXyExBkPp9nzdajDZsozEu4BkaSJozs');\n        tokenTxns.push(\n          (await this.getTriggerSmartContractTransaction(recoveryAddressHex, bitgoHexAddr, amount, contractAddr))\n            .transaction\n        );\n      }\n    }\n    // construct the tx -\n    // there's an assumption here being made about fees: for a wallet that hasn't been used in awhile, the implication is\n    // it has maximum bandwidth. thus, a recovery should cost the minimum amount (1e6 sun or 1 Tron)\n    if (MINIMUM_TRON_MSIG_TRANSACTION_FEE > recoveryAmount) {\n      throw new Error('Amount of funds to recover wouldnt be able to fund a send');\n    }\n\n    const keyHexAddresses = [\n      this.pubToHexAddress(this.xpubToUncompressedPub(userXPub)),\n      this.pubToHexAddress(this.xpubToUncompressedPub(backupXPub)),\n      bitgoHexAddr,\n    ];\n\n    // run checks to ensure this is a valid tx - permissions match our signer keys\n    const ownerKeys: { address: string; weight: number }[] = [];\n    for (const key of account.data[0].owner_permission.keys) {\n      const address = Utils.getHexAddressFromBase58Address(key.address);\n      const weight = key.weight;\n      ownerKeys.push({ address, weight });\n    }\n    const activePermissionKeys: { address: string; weight: number }[] = [];\n    for (const key of account.data[0].active_permission[0].keys) {\n      const address = Utils.getHexAddressFromBase58Address(key.address);\n      const weight = key.weight;\n      activePermissionKeys.push({ address, weight });\n    }\n    this.checkPermissions(ownerKeys, keyHexAddresses);\n    this.checkPermissions(activePermissionKeys, keyHexAddresses);\n\n    // build and sign token txns\n    const finalTokenTxs: any = [];\n    for (const tokenTxn of tokenTxns) {\n      const txBuilder = getBuilder(this.getChain()).from(tokenTxn);\n\n      // this tx should be enough to drop into a node\n      if (isUnsignedSweep) {\n        finalTokenTxs.push((await txBuilder.build()).toJson());\n        continue;\n      }\n\n      const userPrv = this.xprvToCompressedPrv(userXPrv);\n\n      txBuilder.sign({ key: userPrv });\n\n      // krs recoveries don't get signed\n      if (!isKrsRecovery) {\n        const backupXPrv = keys[1].toBase58();\n        const backupPrv = this.xprvToCompressedPrv(backupXPrv);\n\n        txBuilder.sign({ key: backupPrv });\n      }\n      finalTokenTxs.push((await txBuilder.build()).toJson());\n    }\n\n    // tokens must be recovered before the native asset, so that there is sufficient of the native asset to cover fees\n    if (finalTokenTxs.length > 0) {\n      return {\n        tokenTxs: finalTokenTxs,\n      };\n    }\n\n    const recoveryAmountMinusFees = recoveryAmount - MINIMUM_TRON_MSIG_TRANSACTION_FEE;\n    const buildTx = await this.getBuildTransaction(recoveryAddressHex, bitgoHexAddr, recoveryAmountMinusFees);\n\n    // construct our tx\n    const txBuilder = (getBuilder(this.getChain()) as WrappedBuilder).from(buildTx);\n\n    // this tx should be enough to drop into a node\n    if (isUnsignedSweep) {\n      return {\n        tx: (await txBuilder.build()).toJson(),\n        recoveryAmount: recoveryAmountMinusFees,\n      };\n    }\n\n    const userPrv = this.xprvToCompressedPrv(userXPrv);\n\n    txBuilder.sign({ key: userPrv });\n\n    // krs recoveries don't get signed\n    if (!isKrsRecovery) {\n      const backupXPrv = keys[1].toBase58();\n      const backupPrv = this.xprvToCompressedPrv(backupXPrv);\n\n      txBuilder.sign({ key: backupPrv });\n    }\n\n    return {\n      tx: (await txBuilder.build()).toJson(),\n      recoveryAmount: recoveryAmountMinusFees,\n    };\n  }\n\n  /**\n   * Explain a Tron transaction from txHex\n   * @param params\n   */\n  async explainTransaction(params: ExplainTransactionOptions): Promise<TronTransactionExplanation> {\n    const txHex = params.txHex || (params.halfSigned && params.halfSigned.txHex);\n    if (!txHex || !params.feeInfo) {\n      throw new Error('missing explain tx parameters');\n    }\n    const txBuilder = getBuilder(this.getChain()).from(txHex);\n    const tx = await txBuilder.build();\n    const outputs = [\n      {\n        amount: tx.outputs[0].value.toString(),\n        address: tx.outputs[0].address, // Should turn it into a readable format, aka base58\n      },\n    ];\n\n    const displayOrder = [\n      'id',\n      'outputAmount',\n      'changeAmount',\n      'outputs',\n      'changeOutputs',\n      'fee',\n      'timestamp',\n      'expiration',\n    ];\n\n    return {\n      displayOrder,\n      id: tx.id,\n      outputs,\n      outputAmount: outputs[0].amount,\n      changeOutputs: [], // account based does not use change outputs\n      changeAmount: '0', // account base does not make change\n      fee: params.feeInfo,\n      timestamp: tx.validFrom,\n      expiration: tx.validTo,\n    };\n  }\n}\n"]}
|