@bitgo-beta/sdk-coin-xrp 1.3.3-alpha.40 → 1.3.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/CHANGELOG.md +724 -0
- package/dist/tsconfig.tsbuildinfo +1 -8709
- package/package.json +11 -11
- package/.mocharc.yml +0 -8
- package/dist/src/index.d.ts +0 -4
- package/dist/src/index.d.ts.map +0 -1
- package/dist/src/index.js +0 -16
- package/dist/src/register.d.ts +0 -3
- package/dist/src/register.d.ts.map +0 -1
- package/dist/src/register.js +0 -11
- package/dist/src/ripple.d.ts +0 -7
- package/dist/src/ripple.d.ts.map +0 -1
- package/dist/src/ripple.js +0 -90
- package/dist/src/txrp.d.ts +0 -24
- package/dist/src/txrp.d.ts.map +0 -1
- package/dist/src/txrp.js +0 -32
- package/dist/src/xrp.d.ts +0 -152
- package/dist/src/xrp.d.ts.map +0 -1
- package/dist/src/xrp.js +0 -502
package/dist/src/xrp.js
DELETED
|
@@ -1,502 +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.Xrp = void 0;
|
|
23
|
-
/**
|
|
24
|
-
* @prettier
|
|
25
|
-
*/
|
|
26
|
-
const bignumber_js_1 = require("bignumber.js");
|
|
27
|
-
const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
|
|
28
|
-
const crypto_1 = require("crypto");
|
|
29
|
-
const _ = __importStar(require("lodash"));
|
|
30
|
-
const url = __importStar(require("url"));
|
|
31
|
-
const querystring = __importStar(require("querystring"));
|
|
32
|
-
const rippleAddressCodec = __importStar(require("ripple-address-codec"));
|
|
33
|
-
const rippleBinaryCodec = __importStar(require("ripple-binary-codec"));
|
|
34
|
-
const hashes_1 = require("ripple-lib/dist/npm/common/hashes");
|
|
35
|
-
const rippleKeypairs = __importStar(require("ripple-keypairs"));
|
|
36
|
-
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
37
|
-
const ripple = require('./ripple');
|
|
38
|
-
class Xrp extends sdk_core_1.BaseCoin {
|
|
39
|
-
constructor(bitgo) {
|
|
40
|
-
super(bitgo);
|
|
41
|
-
}
|
|
42
|
-
static createInstance(bitgo) {
|
|
43
|
-
return new Xrp(bitgo);
|
|
44
|
-
}
|
|
45
|
-
/**
|
|
46
|
-
* Factor between the coin's base unit and its smallest subdivison
|
|
47
|
-
*/
|
|
48
|
-
getBaseFactor() {
|
|
49
|
-
return 1e6;
|
|
50
|
-
}
|
|
51
|
-
/**
|
|
52
|
-
* Identifier for the blockchain which supports this coin
|
|
53
|
-
*/
|
|
54
|
-
getChain() {
|
|
55
|
-
return 'xrp';
|
|
56
|
-
}
|
|
57
|
-
/**
|
|
58
|
-
* Identifier for the coin family
|
|
59
|
-
*/
|
|
60
|
-
getFamily() {
|
|
61
|
-
return 'xrp';
|
|
62
|
-
}
|
|
63
|
-
/**
|
|
64
|
-
* Complete human-readable name of this coin
|
|
65
|
-
*/
|
|
66
|
-
getFullName() {
|
|
67
|
-
return 'Ripple';
|
|
68
|
-
}
|
|
69
|
-
/**
|
|
70
|
-
* Parse an address string into address and destination tag
|
|
71
|
-
*/
|
|
72
|
-
getAddressDetails(address) {
|
|
73
|
-
const destinationDetails = url.parse(address);
|
|
74
|
-
const destinationAddress = destinationDetails.pathname;
|
|
75
|
-
if (!destinationAddress || !rippleAddressCodec.isValidClassicAddress(destinationAddress)) {
|
|
76
|
-
throw new sdk_core_1.InvalidAddressError(`destination address "${destinationAddress}" is not valid`);
|
|
77
|
-
}
|
|
78
|
-
// there are no other properties like destination tags
|
|
79
|
-
if (destinationDetails.pathname === address) {
|
|
80
|
-
return {
|
|
81
|
-
address: address,
|
|
82
|
-
destinationTag: undefined,
|
|
83
|
-
};
|
|
84
|
-
}
|
|
85
|
-
if (!destinationDetails.query) {
|
|
86
|
-
throw new sdk_core_1.InvalidAddressError('no query params present');
|
|
87
|
-
}
|
|
88
|
-
const queryDetails = querystring.parse(destinationDetails.query);
|
|
89
|
-
if (!queryDetails.dt) {
|
|
90
|
-
// if there are more properties, the query details need to contain the destination tag property.
|
|
91
|
-
throw new sdk_core_1.InvalidAddressError('destination tag missing');
|
|
92
|
-
}
|
|
93
|
-
if (Array.isArray(queryDetails.dt)) {
|
|
94
|
-
// if queryDetails.dt is an array, that means dt was given multiple times, which is not valid
|
|
95
|
-
throw new sdk_core_1.InvalidAddressError(`destination tag can appear at most once, but ${queryDetails.dt.length} destination tags were found`);
|
|
96
|
-
}
|
|
97
|
-
const parsedTag = parseInt(queryDetails.dt, 10);
|
|
98
|
-
if (!Number.isSafeInteger(parsedTag)) {
|
|
99
|
-
throw new sdk_core_1.InvalidAddressError('invalid destination tag');
|
|
100
|
-
}
|
|
101
|
-
if (parsedTag > 0xffffffff || parsedTag < 0) {
|
|
102
|
-
throw new sdk_core_1.InvalidAddressError('destination tag out of range');
|
|
103
|
-
}
|
|
104
|
-
return {
|
|
105
|
-
address: destinationAddress,
|
|
106
|
-
destinationTag: parsedTag,
|
|
107
|
-
};
|
|
108
|
-
}
|
|
109
|
-
/**
|
|
110
|
-
* Construct a full, normalized address from an address and destination tag
|
|
111
|
-
*/
|
|
112
|
-
normalizeAddress({ address, destinationTag }) {
|
|
113
|
-
if (!_.isString(address)) {
|
|
114
|
-
throw new sdk_core_1.InvalidAddressError('invalid address details');
|
|
115
|
-
}
|
|
116
|
-
if (_.isInteger(destinationTag)) {
|
|
117
|
-
return `${address}?dt=${destinationTag}`;
|
|
118
|
-
}
|
|
119
|
-
return address;
|
|
120
|
-
}
|
|
121
|
-
/**
|
|
122
|
-
* Evaluates whether an address string is valid for this coin
|
|
123
|
-
* @param address
|
|
124
|
-
*/
|
|
125
|
-
isValidAddress(address) {
|
|
126
|
-
try {
|
|
127
|
-
const addressDetails = this.getAddressDetails(address);
|
|
128
|
-
return address === this.normalizeAddress(addressDetails);
|
|
129
|
-
}
|
|
130
|
-
catch (e) {
|
|
131
|
-
return false;
|
|
132
|
-
}
|
|
133
|
-
}
|
|
134
|
-
/**
|
|
135
|
-
* Return boolean indicating whether input is valid public key for the coin.
|
|
136
|
-
*
|
|
137
|
-
* @param {String} pub the pub to be checked
|
|
138
|
-
* @returns {Boolean} is it valid?
|
|
139
|
-
*/
|
|
140
|
-
isValidPub(pub) {
|
|
141
|
-
try {
|
|
142
|
-
return utxo_lib_1.bip32.fromBase58(pub).isNeutered();
|
|
143
|
-
}
|
|
144
|
-
catch (e) {
|
|
145
|
-
return false;
|
|
146
|
-
}
|
|
147
|
-
}
|
|
148
|
-
/**
|
|
149
|
-
* Get fee info from server
|
|
150
|
-
*/
|
|
151
|
-
async getFeeInfo() {
|
|
152
|
-
return this.bitgo.get(this.url('/public/feeinfo')).result();
|
|
153
|
-
}
|
|
154
|
-
/**
|
|
155
|
-
* Assemble keychain and half-sign prebuilt transaction
|
|
156
|
-
* @param params
|
|
157
|
-
* - txPrebuild
|
|
158
|
-
* - prv
|
|
159
|
-
* @returns Bluebird<HalfSignedTransaction>
|
|
160
|
-
*/
|
|
161
|
-
async signTransaction({ txPrebuild, prv }) {
|
|
162
|
-
if (_.isUndefined(txPrebuild) || !_.isObject(txPrebuild)) {
|
|
163
|
-
if (!_.isUndefined(txPrebuild) && !_.isObject(txPrebuild)) {
|
|
164
|
-
throw new Error(`txPrebuild must be an object, got type ${typeof txPrebuild}`);
|
|
165
|
-
}
|
|
166
|
-
throw new Error('missing txPrebuild parameter');
|
|
167
|
-
}
|
|
168
|
-
if (_.isUndefined(prv) || !_.isString(prv)) {
|
|
169
|
-
if (!_.isUndefined(prv) && !_.isString(prv)) {
|
|
170
|
-
throw new Error(`prv must be a string, got type ${typeof prv}`);
|
|
171
|
-
}
|
|
172
|
-
throw new Error('missing prv parameter to sign transaction');
|
|
173
|
-
}
|
|
174
|
-
const userKey = utxo_lib_1.bip32.fromBase58(prv);
|
|
175
|
-
const userPrivateKey = userKey.privateKey;
|
|
176
|
-
if (!userPrivateKey) {
|
|
177
|
-
throw new Error(`no privateKey`);
|
|
178
|
-
}
|
|
179
|
-
const userAddress = rippleKeypairs.deriveAddress(userKey.publicKey.toString('hex'));
|
|
180
|
-
const rippleLib = ripple();
|
|
181
|
-
const halfSigned = rippleLib.signWithPrivateKey(txPrebuild.txHex, userPrivateKey.toString('hex'), {
|
|
182
|
-
signAs: userAddress,
|
|
183
|
-
});
|
|
184
|
-
return { halfSigned: { txHex: halfSigned.signedTransaction } };
|
|
185
|
-
}
|
|
186
|
-
/**
|
|
187
|
-
* Ripple requires additional parameters for wallet generation to be sent to the server. The additional parameters are
|
|
188
|
-
* the root public key, which is the basis of the root address, two signed, and one half-signed initialization txs
|
|
189
|
-
* @param walletParams
|
|
190
|
-
* - rootPrivateKey: optional hex-encoded Ripple private key
|
|
191
|
-
*/
|
|
192
|
-
async supplementGenerateWallet(walletParams) {
|
|
193
|
-
if (walletParams.rootPrivateKey) {
|
|
194
|
-
if (walletParams.rootPrivateKey.length !== 64) {
|
|
195
|
-
throw new Error('rootPrivateKey needs to be a hexadecimal private key string');
|
|
196
|
-
}
|
|
197
|
-
}
|
|
198
|
-
else {
|
|
199
|
-
const keyPair = utxo_lib_1.ECPair.makeRandom();
|
|
200
|
-
if (!keyPair.privateKey) {
|
|
201
|
-
throw new Error('no privateKey');
|
|
202
|
-
}
|
|
203
|
-
walletParams.rootPrivateKey = keyPair.privateKey.toString('hex');
|
|
204
|
-
}
|
|
205
|
-
return walletParams;
|
|
206
|
-
}
|
|
207
|
-
/**
|
|
208
|
-
* Explain/parse transaction
|
|
209
|
-
* @param params
|
|
210
|
-
*/
|
|
211
|
-
async explainTransaction(params = {}) {
|
|
212
|
-
if (!params.txHex) {
|
|
213
|
-
throw new Error('missing required param txHex');
|
|
214
|
-
}
|
|
215
|
-
let transaction;
|
|
216
|
-
let txHex;
|
|
217
|
-
try {
|
|
218
|
-
transaction = rippleBinaryCodec.decode(params.txHex);
|
|
219
|
-
txHex = params.txHex;
|
|
220
|
-
}
|
|
221
|
-
catch (e) {
|
|
222
|
-
try {
|
|
223
|
-
transaction = JSON.parse(params.txHex);
|
|
224
|
-
txHex = rippleBinaryCodec.encode(transaction);
|
|
225
|
-
}
|
|
226
|
-
catch (e) {
|
|
227
|
-
throw new Error('txHex needs to be either hex or JSON string for XRP');
|
|
228
|
-
}
|
|
229
|
-
}
|
|
230
|
-
const id = hashes_1.computeBinaryTransactionHash(txHex);
|
|
231
|
-
if (transaction.TransactionType == 'AccountSet') {
|
|
232
|
-
return {
|
|
233
|
-
displayOrder: ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee', 'accountSet'],
|
|
234
|
-
id: id,
|
|
235
|
-
changeOutputs: [],
|
|
236
|
-
outputAmount: 0,
|
|
237
|
-
changeAmount: 0,
|
|
238
|
-
outputs: [],
|
|
239
|
-
fee: {
|
|
240
|
-
fee: transaction.Fee,
|
|
241
|
-
feeRate: null,
|
|
242
|
-
size: txHex.length / 2,
|
|
243
|
-
},
|
|
244
|
-
accountSet: {
|
|
245
|
-
messageKey: transaction.MessageKey,
|
|
246
|
-
},
|
|
247
|
-
};
|
|
248
|
-
}
|
|
249
|
-
const address = transaction.Destination + (transaction.DestinationTag >= 0 ? '?dt=' + transaction.DestinationTag : '');
|
|
250
|
-
return {
|
|
251
|
-
displayOrder: ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee'],
|
|
252
|
-
id: id,
|
|
253
|
-
changeOutputs: [],
|
|
254
|
-
outputAmount: transaction.Amount,
|
|
255
|
-
changeAmount: 0,
|
|
256
|
-
outputs: [
|
|
257
|
-
{
|
|
258
|
-
address,
|
|
259
|
-
amount: transaction.Amount,
|
|
260
|
-
},
|
|
261
|
-
],
|
|
262
|
-
fee: {
|
|
263
|
-
fee: transaction.Fee,
|
|
264
|
-
feeRate: null,
|
|
265
|
-
size: txHex.length / 2,
|
|
266
|
-
},
|
|
267
|
-
};
|
|
268
|
-
}
|
|
269
|
-
/**
|
|
270
|
-
* Verify that a transaction prebuild complies with the original intention
|
|
271
|
-
* @param txParams params object passed to send
|
|
272
|
-
* @param txPrebuild prebuild object returned by server
|
|
273
|
-
* @param wallet
|
|
274
|
-
* @returns {boolean}
|
|
275
|
-
*/
|
|
276
|
-
async verifyTransaction({ txParams, txPrebuild }) {
|
|
277
|
-
const explanation = await this.explainTransaction({
|
|
278
|
-
txHex: txPrebuild.txHex,
|
|
279
|
-
});
|
|
280
|
-
const output = [...explanation.outputs, ...explanation.changeOutputs][0];
|
|
281
|
-
const expectedOutput = txParams.recipients && txParams.recipients[0];
|
|
282
|
-
const comparator = (recipient1, recipient2) => {
|
|
283
|
-
if (recipient1.address !== recipient2.address) {
|
|
284
|
-
return false;
|
|
285
|
-
}
|
|
286
|
-
const amount1 = new bignumber_js_1.BigNumber(recipient1.amount);
|
|
287
|
-
const amount2 = new bignumber_js_1.BigNumber(recipient2.amount);
|
|
288
|
-
return amount1.toFixed() === amount2.toFixed();
|
|
289
|
-
};
|
|
290
|
-
if (!comparator(output, expectedOutput)) {
|
|
291
|
-
throw new Error('transaction prebuild does not match expected output');
|
|
292
|
-
}
|
|
293
|
-
return true;
|
|
294
|
-
}
|
|
295
|
-
/**
|
|
296
|
-
* Check if address is a valid XRP address, and then make sure the root addresses match.
|
|
297
|
-
* This prevents attacks where an attack may switch out the new address for one of their own
|
|
298
|
-
* @param address {String} the address to verify
|
|
299
|
-
* @param rootAddress {String} the wallet's root address
|
|
300
|
-
* @return true iff address is a wallet address (based on rootAddress)
|
|
301
|
-
*/
|
|
302
|
-
async isWalletAddress({ address, rootAddress }) {
|
|
303
|
-
if (!this.isValidAddress(address)) {
|
|
304
|
-
throw new sdk_core_1.InvalidAddressError(`address verification failure: address "${address}" is not valid`);
|
|
305
|
-
}
|
|
306
|
-
const addressDetails = this.getAddressDetails(address);
|
|
307
|
-
const rootAddressDetails = this.getAddressDetails(rootAddress);
|
|
308
|
-
if (addressDetails.address !== rootAddressDetails.address) {
|
|
309
|
-
throw new sdk_core_1.UnexpectedAddressError(`address validation failure: ${addressDetails.address} vs. ${rootAddressDetails.address}`);
|
|
310
|
-
}
|
|
311
|
-
return true;
|
|
312
|
-
}
|
|
313
|
-
/**
|
|
314
|
-
* URL of a well-known, public facing (non-bitgo) rippled instance which can be used for recovery
|
|
315
|
-
*/
|
|
316
|
-
getRippledUrl() {
|
|
317
|
-
return 'https://s1.ripple.com:51234';
|
|
318
|
-
}
|
|
319
|
-
/**
|
|
320
|
-
* Builds a funds recovery transaction without BitGo
|
|
321
|
-
* @param params
|
|
322
|
-
* - rootAddress: root XRP wallet address to recover funds from
|
|
323
|
-
* - userKey: [encrypted] xprv
|
|
324
|
-
* - backupKey: [encrypted] xprv, or xpub if the xprv is held by a KRS provider
|
|
325
|
-
* - walletPassphrase: necessary if one of the xprvs is encrypted
|
|
326
|
-
* - bitgoKey: xpub
|
|
327
|
-
* - krsProvider: necessary if backup key is held by KRS
|
|
328
|
-
* - recoveryDestination: target address to send recovered funds to
|
|
329
|
-
*/
|
|
330
|
-
async recover(params) {
|
|
331
|
-
const rippledUrl = this.getRippledUrl();
|
|
332
|
-
const isKrsRecovery = params.backupKey.startsWith('xpub') && !params.userKey.startsWith('xpub');
|
|
333
|
-
const isUnsignedSweep = params.backupKey.startsWith('xpub') && params.userKey.startsWith('xpub');
|
|
334
|
-
const accountInfoParams = {
|
|
335
|
-
method: 'account_info',
|
|
336
|
-
params: [
|
|
337
|
-
{
|
|
338
|
-
account: params.rootAddress,
|
|
339
|
-
strict: true,
|
|
340
|
-
ledger_index: 'current',
|
|
341
|
-
queue: true,
|
|
342
|
-
signer_lists: true,
|
|
343
|
-
},
|
|
344
|
-
],
|
|
345
|
-
};
|
|
346
|
-
if (isKrsRecovery) {
|
|
347
|
-
sdk_core_1.checkKrsProvider(this, params.krsProvider);
|
|
348
|
-
}
|
|
349
|
-
// Validate the destination address
|
|
350
|
-
if (!this.isValidAddress(params.recoveryDestination)) {
|
|
351
|
-
throw new Error('Invalid destination address!');
|
|
352
|
-
}
|
|
353
|
-
const keys = sdk_core_1.getBip32Keys(this.bitgo, params, { requireBitGoXpub: false });
|
|
354
|
-
const { addressDetails, feeDetails, serverDetails } = await sdk_core_1.promiseProps({
|
|
355
|
-
addressDetails: this.bitgo.post(rippledUrl).send(accountInfoParams),
|
|
356
|
-
feeDetails: this.bitgo.post(rippledUrl).send({ method: 'fee' }),
|
|
357
|
-
serverDetails: this.bitgo.post(rippledUrl).send({ method: 'server_info' }),
|
|
358
|
-
});
|
|
359
|
-
const openLedgerFee = new bignumber_js_1.BigNumber(feeDetails.body.result.drops.open_ledger_fee);
|
|
360
|
-
const baseReserve = new bignumber_js_1.BigNumber(serverDetails.body.result.info.validated_ledger.reserve_base_xrp).times(this.getBaseFactor());
|
|
361
|
-
const reserveDelta = new bignumber_js_1.BigNumber(serverDetails.body.result.info.validated_ledger.reserve_inc_xrp).times(this.getBaseFactor());
|
|
362
|
-
const currentLedger = serverDetails.body.result.info.validated_ledger.seq;
|
|
363
|
-
const sequenceId = addressDetails.body.result.account_data.Sequence;
|
|
364
|
-
const balance = new bignumber_js_1.BigNumber(addressDetails.body.result.account_data.Balance);
|
|
365
|
-
const signerLists = addressDetails.body.result.account_data.signer_lists;
|
|
366
|
-
const accountFlags = addressDetails.body.result.account_data.Flags;
|
|
367
|
-
// make sure there is only one signer list set
|
|
368
|
-
if (signerLists.length !== 1) {
|
|
369
|
-
throw new Error('unexpected set of signer lists');
|
|
370
|
-
}
|
|
371
|
-
// make sure the signers are user, backup, bitgo
|
|
372
|
-
const userAddress = rippleKeypairs.deriveAddress(keys[0].publicKey.toString('hex'));
|
|
373
|
-
const backupAddress = rippleKeypairs.deriveAddress(keys[1].publicKey.toString('hex'));
|
|
374
|
-
const signerList = signerLists[0];
|
|
375
|
-
if (signerList.SignerQuorum !== 2) {
|
|
376
|
-
throw new Error('invalid minimum signature count');
|
|
377
|
-
}
|
|
378
|
-
const foundAddresses = {};
|
|
379
|
-
const signerEntries = signerList.SignerEntries;
|
|
380
|
-
if (signerEntries.length !== 3) {
|
|
381
|
-
throw new Error('invalid signer list length');
|
|
382
|
-
}
|
|
383
|
-
for (const { SignerEntry } of signerEntries) {
|
|
384
|
-
const weight = SignerEntry.SignerWeight;
|
|
385
|
-
const address = SignerEntry.Account;
|
|
386
|
-
if (weight !== 1) {
|
|
387
|
-
throw new Error('invalid signer weight');
|
|
388
|
-
}
|
|
389
|
-
// if it's a dupe of an address we already know, block
|
|
390
|
-
if (foundAddresses[address] >= 1) {
|
|
391
|
-
throw new Error('duplicate signer address');
|
|
392
|
-
}
|
|
393
|
-
foundAddresses[address] = (foundAddresses[address] || 0) + 1;
|
|
394
|
-
}
|
|
395
|
-
if (foundAddresses[userAddress] !== 1) {
|
|
396
|
-
throw new Error('unexpected incidence frequency of user signer address');
|
|
397
|
-
}
|
|
398
|
-
if (foundAddresses[backupAddress] !== 1) {
|
|
399
|
-
throw new Error('unexpected incidence frequency of user signer address');
|
|
400
|
-
}
|
|
401
|
-
// make sure the flags disable the master key and enforce destination tags
|
|
402
|
-
const USER_KEY_SETTING_FLAG = 65536;
|
|
403
|
-
const MASTER_KEY_DEACTIVATION_FLAG = 1048576;
|
|
404
|
-
const REQUIRE_DESTINATION_TAG_FLAG = 131072;
|
|
405
|
-
if ((accountFlags & USER_KEY_SETTING_FLAG) !== 0) {
|
|
406
|
-
throw new Error('a custom user key has been set');
|
|
407
|
-
}
|
|
408
|
-
if ((accountFlags & MASTER_KEY_DEACTIVATION_FLAG) !== MASTER_KEY_DEACTIVATION_FLAG) {
|
|
409
|
-
throw new Error('the master key has not been deactivated');
|
|
410
|
-
}
|
|
411
|
-
if ((accountFlags & REQUIRE_DESTINATION_TAG_FLAG) !== REQUIRE_DESTINATION_TAG_FLAG) {
|
|
412
|
-
throw new Error('the destination flag requirement has not been activated');
|
|
413
|
-
}
|
|
414
|
-
// recover the funds
|
|
415
|
-
const reserve = baseReserve.plus(reserveDelta.times(5));
|
|
416
|
-
const recoverableBalance = balance.minus(reserve);
|
|
417
|
-
const rawDestination = params.recoveryDestination;
|
|
418
|
-
const destinationDetails = url.parse(rawDestination);
|
|
419
|
-
const destinationAddress = destinationDetails.pathname;
|
|
420
|
-
// parse destination tag from query
|
|
421
|
-
let destinationTag;
|
|
422
|
-
if (destinationDetails.query) {
|
|
423
|
-
const queryDetails = querystring.parse(destinationDetails.query);
|
|
424
|
-
if (Array.isArray(queryDetails.dt)) {
|
|
425
|
-
// if queryDetails.dt is an array, that means dt was given multiple times, which is not valid
|
|
426
|
-
throw new sdk_core_1.InvalidAddressError(`destination tag can appear at most once, but ${queryDetails.dt.length} destination tags were found`);
|
|
427
|
-
}
|
|
428
|
-
const parsedTag = parseInt(queryDetails.dt, 10);
|
|
429
|
-
if (Number.isInteger(parsedTag)) {
|
|
430
|
-
destinationTag = parsedTag;
|
|
431
|
-
}
|
|
432
|
-
}
|
|
433
|
-
const transaction = {
|
|
434
|
-
TransactionType: 'Payment',
|
|
435
|
-
Account: params.rootAddress,
|
|
436
|
-
Destination: destinationAddress,
|
|
437
|
-
DestinationTag: destinationTag,
|
|
438
|
-
Amount: recoverableBalance.toFixed(0),
|
|
439
|
-
Flags: 2147483648,
|
|
440
|
-
LastLedgerSequence: currentLedger + 1000000,
|
|
441
|
-
Fee: openLedgerFee.times(3).toFixed(0),
|
|
442
|
-
Sequence: sequenceId,
|
|
443
|
-
};
|
|
444
|
-
const txJSON = JSON.stringify(transaction);
|
|
445
|
-
if (isUnsignedSweep) {
|
|
446
|
-
return txJSON;
|
|
447
|
-
}
|
|
448
|
-
const rippleLib = ripple();
|
|
449
|
-
if (!keys[0].privateKey) {
|
|
450
|
-
throw new Error(`userKey is not a private key`);
|
|
451
|
-
}
|
|
452
|
-
const userKey = keys[0].privateKey.toString('hex');
|
|
453
|
-
const userSignature = rippleLib.signWithPrivateKey(txJSON, userKey, { signAs: userAddress });
|
|
454
|
-
let signedTransaction;
|
|
455
|
-
if (isKrsRecovery) {
|
|
456
|
-
signedTransaction = userSignature;
|
|
457
|
-
}
|
|
458
|
-
else {
|
|
459
|
-
if (!keys[1].privateKey) {
|
|
460
|
-
throw new Error(`backupKey is not a private key`);
|
|
461
|
-
}
|
|
462
|
-
const backupKey = keys[1].privateKey.toString('hex');
|
|
463
|
-
const backupSignature = rippleLib.signWithPrivateKey(txJSON, backupKey, { signAs: backupAddress });
|
|
464
|
-
signedTransaction = rippleLib.combine([userSignature.signedTransaction, backupSignature.signedTransaction]);
|
|
465
|
-
}
|
|
466
|
-
const transactionExplanation = (await this.explainTransaction({
|
|
467
|
-
txHex: signedTransaction.signedTransaction,
|
|
468
|
-
}));
|
|
469
|
-
transactionExplanation.txHex = signedTransaction.signedTransaction;
|
|
470
|
-
if (isKrsRecovery) {
|
|
471
|
-
transactionExplanation.backupKey = params.backupKey;
|
|
472
|
-
transactionExplanation.coin = this.getChain();
|
|
473
|
-
}
|
|
474
|
-
return transactionExplanation;
|
|
475
|
-
}
|
|
476
|
-
initiateRecovery(params) {
|
|
477
|
-
throw new Error('deprecated method');
|
|
478
|
-
}
|
|
479
|
-
/**
|
|
480
|
-
* Generate a new keypair for this coin.
|
|
481
|
-
* @param seed Seed from which the new keypair should be generated, otherwise a random seed is used
|
|
482
|
-
*/
|
|
483
|
-
generateKeyPair(seed) {
|
|
484
|
-
if (!seed) {
|
|
485
|
-
// An extended private key has both a normal 256 bit private key and a 256
|
|
486
|
-
// bit chain code, both of which must be random. 512 bits is therefore the
|
|
487
|
-
// maximum entropy and gives us maximum security against cracking.
|
|
488
|
-
seed = crypto_1.randomBytes(512 / 8);
|
|
489
|
-
}
|
|
490
|
-
const extendedKey = utxo_lib_1.bip32.fromSeed(seed);
|
|
491
|
-
const xpub = extendedKey.neutered().toBase58();
|
|
492
|
-
return {
|
|
493
|
-
pub: xpub,
|
|
494
|
-
prv: extendedKey.toBase58(),
|
|
495
|
-
};
|
|
496
|
-
}
|
|
497
|
-
async parseTransaction(params) {
|
|
498
|
-
return {};
|
|
499
|
-
}
|
|
500
|
-
}
|
|
501
|
-
exports.Xrp = Xrp;
|
|
502
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"xrp.js","sourceRoot":"","sources":["../../src/xrp.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @prettier\n */\nimport { BigNumber } from 'bignumber.js';\nimport { bip32, ECPair } from '@bitgo-beta/utxo-lib';\nimport { randomBytes } from 'crypto';\nimport * as _ from 'lodash';\nimport * as url from 'url';\nimport * as querystring from 'querystring';\n\nimport * as rippleAddressCodec from 'ripple-address-codec';\nimport * as rippleBinaryCodec from 'ripple-binary-codec';\nimport { computeBinaryTransactionHash } from 'ripple-lib/dist/npm/common/hashes';\nimport * as rippleKeypairs from 'ripple-keypairs';\nimport {\n  BaseCoin,\n  BitGoBase,\n  checkKrsProvider,\n  getBip32Keys,\n  InitiateRecoveryOptions as BaseInitiateRecoveryOptions,\n  InvalidAddressError,\n  KeyPair,\n  ParsedTransaction,\n  ParseTransactionOptions,\n  promiseProps,\n  SignTransactionOptions as BaseSignTransactionOptions,\n  TransactionExplanation,\n  TransactionPrebuild,\n  UnexpectedAddressError,\n  VerifyAddressOptions as BaseVerifyAddressOptions,\n  VerifyTransactionOptions,\n} from '@bitgo-beta/sdk-core';\n\nconst ripple = require('./ripple');\n\ninterface Address {\n  address: string;\n  destinationTag?: number;\n}\n\ninterface FeeInfo {\n  date: string;\n  height: number;\n  baseReserve: string;\n  baseFee: string;\n}\n\ninterface SignTransactionOptions extends BaseSignTransactionOptions {\n  txPrebuild: TransactionPrebuild;\n  prv: string;\n}\n\ninterface ExplainTransactionOptions {\n  txHex?: string;\n}\n\ninterface VerifyAddressOptions extends BaseVerifyAddressOptions {\n  rootAddress: string;\n}\n\ninterface RecoveryInfo extends TransactionExplanation {\n  txHex: string;\n  backupKey?: string;\n  coin?: string;\n}\n\nexport interface InitiateRecoveryOptions extends BaseInitiateRecoveryOptions {\n  krsProvider?: string;\n}\n\nexport interface RecoveryOptions {\n  backupKey: string;\n  userKey: string;\n  rootAddress: string;\n  recoveryDestination: string;\n  bitgoKey?: string;\n  walletPassphrase: string;\n  krsProvider?: string;\n}\n\ninterface HalfSignedTransaction {\n  halfSigned: {\n    txHex: string;\n  };\n}\n\ninterface SupplementGenerateWalletOptions {\n  rootPrivateKey?: string;\n}\n\nexport class Xrp extends BaseCoin {\n  protected constructor(bitgo: BitGoBase) {\n    super(bitgo);\n  }\n\n  static createInstance(bitgo: BitGoBase): BaseCoin {\n    return new Xrp(bitgo);\n  }\n\n  /**\n   * Factor between the coin's base unit and its smallest subdivison\n   */\n  public getBaseFactor(): number {\n    return 1e6;\n  }\n\n  /**\n   * Identifier for the blockchain which supports this coin\n   */\n  public getChain(): string {\n    return 'xrp';\n  }\n\n  /**\n   * Identifier for the coin family\n   */\n  public getFamily(): string {\n    return 'xrp';\n  }\n\n  /**\n   * Complete human-readable name of this coin\n   */\n  public getFullName(): string {\n    return 'Ripple';\n  }\n\n  /**\n   * Parse an address string into address and destination tag\n   */\n  public getAddressDetails(address: string): Address {\n    const destinationDetails = url.parse(address);\n    const destinationAddress = destinationDetails.pathname;\n    if (!destinationAddress || !rippleAddressCodec.isValidClassicAddress(destinationAddress)) {\n      throw new InvalidAddressError(`destination address \"${destinationAddress}\" is not valid`);\n    }\n    // there are no other properties like destination tags\n    if (destinationDetails.pathname === address) {\n      return {\n        address: address,\n        destinationTag: undefined,\n      };\n    }\n\n    if (!destinationDetails.query) {\n      throw new InvalidAddressError('no query params present');\n    }\n\n    const queryDetails = querystring.parse(destinationDetails.query);\n    if (!queryDetails.dt) {\n      // if there are more properties, the query details need to contain the destination tag property.\n      throw new InvalidAddressError('destination tag missing');\n    }\n\n    if (Array.isArray(queryDetails.dt)) {\n      // if queryDetails.dt is an array, that means dt was given multiple times, which is not valid\n      throw new InvalidAddressError(\n        `destination tag can appear at most once, but ${queryDetails.dt.length} destination tags were found`\n      );\n    }\n\n    const parsedTag = parseInt(queryDetails.dt, 10);\n    if (!Number.isSafeInteger(parsedTag)) {\n      throw new InvalidAddressError('invalid destination tag');\n    }\n\n    if (parsedTag > 0xffffffff || parsedTag < 0) {\n      throw new InvalidAddressError('destination tag out of range');\n    }\n\n    return {\n      address: destinationAddress,\n      destinationTag: parsedTag,\n    };\n  }\n\n  /**\n   * Construct a full, normalized address from an address and destination tag\n   */\n  public normalizeAddress({ address, destinationTag }: Address): string {\n    if (!_.isString(address)) {\n      throw new InvalidAddressError('invalid address details');\n    }\n    if (_.isInteger(destinationTag)) {\n      return `${address}?dt=${destinationTag}`;\n    }\n    return address;\n  }\n\n  /**\n   * Evaluates whether an address string is valid for this coin\n   * @param address\n   */\n  public isValidAddress(address: string): boolean {\n    try {\n      const addressDetails = this.getAddressDetails(address);\n      return address === this.normalizeAddress(addressDetails);\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Return boolean indicating whether input is valid public key for the coin.\n   *\n   * @param {String} pub the pub to be checked\n   * @returns {Boolean} is it valid?\n   */\n  public isValidPub(pub: string): boolean {\n    try {\n      return bip32.fromBase58(pub).isNeutered();\n    } catch (e) {\n      return false;\n    }\n  }\n\n  /**\n   * Get fee info from server\n   */\n  public async getFeeInfo(): Promise<FeeInfo> {\n    return this.bitgo.get(this.url('/public/feeinfo')).result();\n  }\n\n  /**\n   * Assemble keychain and half-sign prebuilt transaction\n   * @param params\n   * - txPrebuild\n   * - prv\n   * @returns Bluebird<HalfSignedTransaction>\n   */\n  public async signTransaction({ txPrebuild, prv }: SignTransactionOptions): Promise<HalfSignedTransaction> {\n    if (_.isUndefined(txPrebuild) || !_.isObject(txPrebuild)) {\n      if (!_.isUndefined(txPrebuild) && !_.isObject(txPrebuild)) {\n        throw new Error(`txPrebuild must be an object, got type ${typeof txPrebuild}`);\n      }\n      throw new Error('missing txPrebuild parameter');\n    }\n\n    if (_.isUndefined(prv) || !_.isString(prv)) {\n      if (!_.isUndefined(prv) && !_.isString(prv)) {\n        throw new Error(`prv must be a string, got type ${typeof prv}`);\n      }\n      throw new Error('missing prv parameter to sign transaction');\n    }\n\n    const userKey = bip32.fromBase58(prv);\n    const userPrivateKey = userKey.privateKey;\n    if (!userPrivateKey) {\n      throw new Error(`no privateKey`);\n    }\n    const userAddress = rippleKeypairs.deriveAddress(userKey.publicKey.toString('hex'));\n\n    const rippleLib = ripple();\n    const halfSigned = rippleLib.signWithPrivateKey(txPrebuild.txHex, userPrivateKey.toString('hex'), {\n      signAs: userAddress,\n    });\n    return { halfSigned: { txHex: halfSigned.signedTransaction } };\n  }\n\n  /**\n   * Ripple requires additional parameters for wallet generation to be sent to the server. The additional parameters are\n   * the root public key, which is the basis of the root address, two signed, and one half-signed initialization txs\n   * @param walletParams\n   * - rootPrivateKey: optional hex-encoded Ripple private key\n   */\n  async supplementGenerateWallet(\n    walletParams: SupplementGenerateWalletOptions\n  ): Promise<SupplementGenerateWalletOptions> {\n    if (walletParams.rootPrivateKey) {\n      if (walletParams.rootPrivateKey.length !== 64) {\n        throw new Error('rootPrivateKey needs to be a hexadecimal private key string');\n      }\n    } else {\n      const keyPair = ECPair.makeRandom();\n      if (!keyPair.privateKey) {\n        throw new Error('no privateKey');\n      }\n      walletParams.rootPrivateKey = keyPair.privateKey.toString('hex');\n    }\n    return walletParams;\n  }\n\n  /**\n   * Explain/parse transaction\n   * @param params\n   */\n  async explainTransaction(params: ExplainTransactionOptions = {}): Promise<TransactionExplanation> {\n    if (!params.txHex) {\n      throw new Error('missing required param txHex');\n    }\n    let transaction;\n    let txHex;\n    try {\n      transaction = rippleBinaryCodec.decode(params.txHex);\n      txHex = params.txHex;\n    } catch (e) {\n      try {\n        transaction = JSON.parse(params.txHex);\n        txHex = rippleBinaryCodec.encode(transaction);\n      } catch (e) {\n        throw new Error('txHex needs to be either hex or JSON string for XRP');\n      }\n    }\n    const id = computeBinaryTransactionHash(txHex);\n\n    if (transaction.TransactionType == 'AccountSet') {\n      return {\n        displayOrder: ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee', 'accountSet'],\n        id: id,\n        changeOutputs: [],\n        outputAmount: 0,\n        changeAmount: 0,\n        outputs: [],\n        fee: {\n          fee: transaction.Fee,\n          feeRate: null,\n          size: txHex.length / 2,\n        },\n        accountSet: {\n          messageKey: transaction.MessageKey,\n        },\n      } as any;\n    }\n\n    const address =\n      transaction.Destination + (transaction.DestinationTag >= 0 ? '?dt=' + transaction.DestinationTag : '');\n    return {\n      displayOrder: ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee'],\n      id: id,\n      changeOutputs: [],\n      outputAmount: transaction.Amount,\n      changeAmount: 0,\n      outputs: [\n        {\n          address,\n          amount: transaction.Amount,\n        },\n      ],\n      fee: {\n        fee: transaction.Fee,\n        feeRate: null,\n        size: txHex.length / 2,\n      },\n    } as any;\n  }\n\n  /**\n   * Verify that a transaction prebuild complies with the original intention\n   * @param txParams params object passed to send\n   * @param txPrebuild prebuild object returned by server\n   * @param wallet\n   * @returns {boolean}\n   */\n  public async verifyTransaction({ txParams, txPrebuild }: VerifyTransactionOptions): Promise<boolean> {\n    const explanation = await this.explainTransaction({\n      txHex: txPrebuild.txHex,\n    });\n\n    const output = [...explanation.outputs, ...explanation.changeOutputs][0];\n    const expectedOutput = txParams.recipients && txParams.recipients[0];\n\n    const comparator = (recipient1, recipient2) => {\n      if (recipient1.address !== recipient2.address) {\n        return false;\n      }\n      const amount1 = new BigNumber(recipient1.amount);\n      const amount2 = new BigNumber(recipient2.amount);\n      return amount1.toFixed() === amount2.toFixed();\n    };\n\n    if (!comparator(output, expectedOutput)) {\n      throw new Error('transaction prebuild does not match expected output');\n    }\n\n    return true;\n  }\n\n  /**\n   * Check if address is a valid XRP address, and then make sure the root addresses match.\n   * This prevents attacks where an attack may switch out the new address for one of their own\n   * @param address {String} the address to verify\n   * @param rootAddress {String} the wallet's root address\n   * @return true iff address is a wallet address (based on rootAddress)\n   */\n  public async isWalletAddress({ address, rootAddress }: VerifyAddressOptions): Promise<boolean> {\n    if (!this.isValidAddress(address)) {\n      throw new InvalidAddressError(`address verification failure: address \"${address}\" is not valid`);\n    }\n\n    const addressDetails = this.getAddressDetails(address);\n    const rootAddressDetails = this.getAddressDetails(rootAddress);\n\n    if (addressDetails.address !== rootAddressDetails.address) {\n      throw new UnexpectedAddressError(\n        `address validation failure: ${addressDetails.address} vs. ${rootAddressDetails.address}`\n      );\n    }\n\n    return true;\n  }\n\n  /**\n   * URL of a well-known, public facing (non-bitgo) rippled instance which can be used for recovery\n   */\n  public getRippledUrl(): string {\n    return 'https://s1.ripple.com:51234';\n  }\n\n  /**\n   * Builds a funds recovery transaction without BitGo\n   * @param params\n   * - rootAddress: root XRP wallet address to recover funds from\n   * - userKey: [encrypted] xprv\n   * - backupKey: [encrypted] xprv, or xpub if the xprv is held by a KRS provider\n   * - walletPassphrase: necessary if one of the xprvs is encrypted\n   * - bitgoKey: xpub\n   * - krsProvider: necessary if backup key is held by KRS\n   * - recoveryDestination: target address to send recovered funds to\n   */\n  public async recover(params: RecoveryOptions): Promise<RecoveryInfo | string> {\n    const rippledUrl = this.getRippledUrl();\n    const isKrsRecovery = params.backupKey.startsWith('xpub') && !params.userKey.startsWith('xpub');\n    const isUnsignedSweep = params.backupKey.startsWith('xpub') && params.userKey.startsWith('xpub');\n\n    const accountInfoParams = {\n      method: 'account_info',\n      params: [\n        {\n          account: params.rootAddress,\n          strict: true,\n          ledger_index: 'current',\n          queue: true,\n          signer_lists: true,\n        },\n      ],\n    };\n\n    if (isKrsRecovery) {\n      checkKrsProvider(this, params.krsProvider);\n    }\n\n    // Validate the destination address\n    if (!this.isValidAddress(params.recoveryDestination)) {\n      throw new Error('Invalid destination address!');\n    }\n\n    const keys = getBip32Keys(this.bitgo, params, { requireBitGoXpub: false });\n\n    const { addressDetails, feeDetails, serverDetails } = await promiseProps({\n      addressDetails: this.bitgo.post(rippledUrl).send(accountInfoParams),\n      feeDetails: this.bitgo.post(rippledUrl).send({ method: 'fee' }),\n      serverDetails: this.bitgo.post(rippledUrl).send({ method: 'server_info' }),\n    });\n\n    const openLedgerFee = new BigNumber(feeDetails.body.result.drops.open_ledger_fee);\n    const baseReserve = new BigNumber(serverDetails.body.result.info.validated_ledger.reserve_base_xrp).times(\n      this.getBaseFactor()\n    );\n    const reserveDelta = new BigNumber(serverDetails.body.result.info.validated_ledger.reserve_inc_xrp).times(\n      this.getBaseFactor()\n    );\n    const currentLedger = serverDetails.body.result.info.validated_ledger.seq;\n    const sequenceId = addressDetails.body.result.account_data.Sequence;\n    const balance = new BigNumber(addressDetails.body.result.account_data.Balance);\n    const signerLists = addressDetails.body.result.account_data.signer_lists;\n    const accountFlags = addressDetails.body.result.account_data.Flags;\n\n    // make sure there is only one signer list set\n    if (signerLists.length !== 1) {\n      throw new Error('unexpected set of signer lists');\n    }\n\n    // make sure the signers are user, backup, bitgo\n    const userAddress = rippleKeypairs.deriveAddress(keys[0].publicKey.toString('hex'));\n    const backupAddress = rippleKeypairs.deriveAddress(keys[1].publicKey.toString('hex'));\n\n    const signerList = signerLists[0];\n    if (signerList.SignerQuorum !== 2) {\n      throw new Error('invalid minimum signature count');\n    }\n    const foundAddresses = {};\n\n    const signerEntries = signerList.SignerEntries;\n    if (signerEntries.length !== 3) {\n      throw new Error('invalid signer list length');\n    }\n    for (const { SignerEntry } of signerEntries) {\n      const weight = SignerEntry.SignerWeight;\n      const address = SignerEntry.Account;\n      if (weight !== 1) {\n        throw new Error('invalid signer weight');\n      }\n\n      // if it's a dupe of an address we already know, block\n      if (foundAddresses[address] >= 1) {\n        throw new Error('duplicate signer address');\n      }\n      foundAddresses[address] = (foundAddresses[address] || 0) + 1;\n    }\n\n    if (foundAddresses[userAddress] !== 1) {\n      throw new Error('unexpected incidence frequency of user signer address');\n    }\n    if (foundAddresses[backupAddress] !== 1) {\n      throw new Error('unexpected incidence frequency of user signer address');\n    }\n\n    // make sure the flags disable the master key and enforce destination tags\n    const USER_KEY_SETTING_FLAG = 65536;\n    const MASTER_KEY_DEACTIVATION_FLAG = 1048576;\n    const REQUIRE_DESTINATION_TAG_FLAG = 131072;\n    if ((accountFlags & USER_KEY_SETTING_FLAG) !== 0) {\n      throw new Error('a custom user key has been set');\n    }\n    if ((accountFlags & MASTER_KEY_DEACTIVATION_FLAG) !== MASTER_KEY_DEACTIVATION_FLAG) {\n      throw new Error('the master key has not been deactivated');\n    }\n    if ((accountFlags & REQUIRE_DESTINATION_TAG_FLAG) !== REQUIRE_DESTINATION_TAG_FLAG) {\n      throw new Error('the destination flag requirement has not been activated');\n    }\n\n    // recover the funds\n    const reserve = baseReserve.plus(reserveDelta.times(5));\n    const recoverableBalance = balance.minus(reserve);\n\n    const rawDestination = params.recoveryDestination;\n    const destinationDetails = url.parse(rawDestination);\n    const destinationAddress = destinationDetails.pathname;\n\n    // parse destination tag from query\n    let destinationTag: number | undefined;\n    if (destinationDetails.query) {\n      const queryDetails = querystring.parse(destinationDetails.query);\n      if (Array.isArray(queryDetails.dt)) {\n        // if queryDetails.dt is an array, that means dt was given multiple times, which is not valid\n        throw new InvalidAddressError(\n          `destination tag can appear at most once, but ${queryDetails.dt.length} destination tags were found`\n        );\n      }\n\n      const parsedTag = parseInt(queryDetails.dt as string, 10);\n      if (Number.isInteger(parsedTag)) {\n        destinationTag = parsedTag;\n      }\n    }\n\n    const transaction = {\n      TransactionType: 'Payment',\n      Account: params.rootAddress, // source address\n      Destination: destinationAddress,\n      DestinationTag: destinationTag,\n      Amount: recoverableBalance.toFixed(0),\n      Flags: 2147483648,\n      LastLedgerSequence: currentLedger + 1000000, // give it 1 million ledgers' time (~1 month, suitable for KRS)\n      Fee: openLedgerFee.times(3).toFixed(0), // the factor three is for the multisigning\n      Sequence: sequenceId,\n    };\n    const txJSON: string = JSON.stringify(transaction);\n\n    if (isUnsignedSweep) {\n      return txJSON;\n    }\n    const rippleLib = ripple();\n    if (!keys[0].privateKey) {\n      throw new Error(`userKey is not a private key`);\n    }\n    const userKey = keys[0].privateKey.toString('hex');\n    const userSignature = rippleLib.signWithPrivateKey(txJSON, userKey, { signAs: userAddress });\n\n    let signedTransaction;\n\n    if (isKrsRecovery) {\n      signedTransaction = userSignature;\n    } else {\n      if (!keys[1].privateKey) {\n        throw new Error(`backupKey is not a private key`);\n      }\n      const backupKey = keys[1].privateKey.toString('hex');\n      const backupSignature = rippleLib.signWithPrivateKey(txJSON, backupKey, { signAs: backupAddress });\n      signedTransaction = rippleLib.combine([userSignature.signedTransaction, backupSignature.signedTransaction]);\n    }\n\n    const transactionExplanation: RecoveryInfo = (await this.explainTransaction({\n      txHex: signedTransaction.signedTransaction,\n    })) as any;\n    transactionExplanation.txHex = signedTransaction.signedTransaction;\n\n    if (isKrsRecovery) {\n      transactionExplanation.backupKey = params.backupKey;\n      transactionExplanation.coin = this.getChain();\n    }\n    return transactionExplanation;\n  }\n\n  initiateRecovery(params: InitiateRecoveryOptions): never {\n    throw new Error('deprecated method');\n  }\n\n  /**\n   * Generate a new keypair for this coin.\n   * @param seed Seed from which the new keypair should be generated, otherwise a random seed is used\n   */\n  public generateKeyPair(seed?: Buffer): KeyPair {\n    if (!seed) {\n      // An extended private key has both a normal 256 bit private key and a 256\n      // bit chain code, both of which must be random. 512 bits is therefore the\n      // maximum entropy and gives us maximum security against cracking.\n      seed = randomBytes(512 / 8);\n    }\n    const extendedKey = bip32.fromSeed(seed);\n    const xpub = extendedKey.neutered().toBase58();\n    return {\n      pub: xpub,\n      prv: extendedKey.toBase58(),\n    };\n  }\n\n  async parseTransaction(params: ParseTransactionOptions): Promise<ParsedTransaction> {\n    return {};\n  }\n}\n"]}
|