@bitgo-beta/sdk-coin-xrp 1.3.3-alpha.24 → 1.3.3-alpha.241

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.
Files changed (60) hide show
  1. package/CHANGELOG.md +499 -0
  2. package/dist/src/index.d.ts +4 -2
  3. package/dist/src/index.d.ts.map +1 -1
  4. package/dist/src/index.js +10 -4
  5. package/dist/src/lib/accountSetBuilder.d.ts +18 -0
  6. package/dist/src/lib/accountSetBuilder.d.ts.map +1 -0
  7. package/dist/src/lib/accountSetBuilder.js +63 -0
  8. package/dist/src/lib/constants.d.ts +8 -0
  9. package/dist/src/lib/constants.d.ts.map +1 -0
  10. package/dist/src/lib/constants.js +30 -0
  11. package/dist/src/lib/iface.d.ts +109 -0
  12. package/dist/src/lib/iface.d.ts.map +1 -0
  13. package/dist/src/lib/iface.js +11 -0
  14. package/dist/src/lib/index.d.ts +14 -0
  15. package/dist/src/lib/index.d.ts.map +1 -0
  16. package/dist/src/lib/index.js +43 -0
  17. package/dist/src/lib/keyPair.d.ts +33 -0
  18. package/dist/src/lib/keyPair.d.ts.map +1 -0
  19. package/dist/src/lib/keyPair.js +118 -0
  20. package/dist/src/lib/tokenTransferBuilder.d.ts +29 -0
  21. package/dist/src/lib/tokenTransferBuilder.d.ts.map +1 -0
  22. package/dist/src/lib/tokenTransferBuilder.js +91 -0
  23. package/dist/src/lib/transaction.d.ts +62 -0
  24. package/dist/src/lib/transaction.d.ts.map +1 -0
  25. package/dist/src/lib/transaction.js +381 -0
  26. package/dist/src/lib/transactionBuilder.d.ts +72 -0
  27. package/dist/src/lib/transactionBuilder.d.ts.map +1 -0
  28. package/dist/src/lib/transactionBuilder.js +263 -0
  29. package/dist/src/lib/transactionBuilderFactory.d.ts +39 -0
  30. package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -0
  31. package/dist/src/lib/transactionBuilderFactory.js +97 -0
  32. package/dist/src/lib/transferBuilder.d.ts +28 -0
  33. package/dist/src/lib/transferBuilder.d.ts.map +1 -0
  34. package/dist/src/lib/transferBuilder.js +82 -0
  35. package/dist/src/lib/trustsetBuilder.d.ts +21 -0
  36. package/dist/src/lib/trustsetBuilder.d.ts.map +1 -0
  37. package/dist/src/lib/trustsetBuilder.js +72 -0
  38. package/dist/src/lib/utils.d.ts +78 -0
  39. package/dist/src/lib/utils.d.ts.map +1 -0
  40. package/dist/src/lib/utils.js +304 -0
  41. package/dist/src/lib/walletInitializationBuilder.d.ts +19 -0
  42. package/dist/src/lib/walletInitializationBuilder.d.ts.map +1 -0
  43. package/dist/src/lib/walletInitializationBuilder.js +76 -0
  44. package/dist/src/register.d.ts.map +1 -1
  45. package/dist/src/register.js +5 -1
  46. package/dist/src/ripple.d.ts +112 -2
  47. package/dist/src/ripple.d.ts.map +1 -1
  48. package/dist/src/ripple.js +34 -21
  49. package/dist/src/txrp.d.ts +3 -2
  50. package/dist/src/txrp.d.ts.map +1 -1
  51. package/dist/src/txrp.js +5 -5
  52. package/dist/src/xrp.d.ts +13 -61
  53. package/dist/src/xrp.d.ts.map +1 -1
  54. package/dist/src/xrp.js +280 -157
  55. package/dist/src/xrpToken.d.ts +21 -0
  56. package/dist/src/xrpToken.d.ts.map +1 -0
  57. package/dist/src/xrpToken.js +58 -0
  58. package/dist/tsconfig.tsbuildinfo +1 -8595
  59. package/package.json +11 -11
  60. package/.mocharc.yml +0 -8
package/dist/src/xrp.js CHANGED
@@ -1,7 +1,11 @@
1
1
  "use strict";
2
2
  var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
3
  if (k2 === undefined) k2 = k;
4
- Object.defineProperty(o, k2, { enumerable: true, get: function() { return m[k]; } });
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
5
9
  }) : (function(o, m, k, k2) {
6
10
  if (k2 === undefined) k2 = k;
7
11
  o[k2] = m[k];
@@ -11,12 +15,25 @@ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (
11
15
  }) : function(o, v) {
12
16
  o["default"] = v;
13
17
  });
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;
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ var __importDefault = (this && this.__importDefault) || function (mod) {
36
+ return (mod && mod.__esModule) ? mod : { "default": mod };
20
37
  };
21
38
  Object.defineProperty(exports, "__esModule", { value: true });
22
39
  exports.Xrp = void 0;
@@ -24,112 +41,59 @@ exports.Xrp = void 0;
24
41
  * @prettier
25
42
  */
26
43
  const bignumber_js_1 = require("bignumber.js");
27
- const utxo_lib_1 = require("@bitgo-beta/utxo-lib");
28
- const crypto_1 = require("crypto");
29
44
  const _ = __importStar(require("lodash"));
30
- const url = __importStar(require("url"));
31
45
  const querystring = __importStar(require("querystring"));
32
- const rippleAddressCodec = __importStar(require("ripple-address-codec"));
46
+ const url = __importStar(require("url"));
47
+ const sdk_core_1 = require("@bitgo-beta/sdk-core");
48
+ const statics_1 = require("@bitgo-beta/statics");
33
49
  const rippleBinaryCodec = __importStar(require("ripple-binary-codec"));
34
- const hashes_1 = require("ripple-lib/dist/npm/common/hashes");
35
50
  const rippleKeypairs = __importStar(require("ripple-keypairs"));
36
- const sdk_core_1 = require("@bitgo-beta/sdk-core");
37
- const ripple = require('./ripple');
51
+ const xrpl = __importStar(require("xrpl"));
52
+ const keyPair_1 = require("./lib/keyPair");
53
+ const utils_1 = __importDefault(require("./lib/utils"));
54
+ const ripple_1 = __importDefault(require("./ripple"));
55
+ const lib_1 = require("./lib");
38
56
  class Xrp extends sdk_core_1.BaseCoin {
39
- constructor(bitgo) {
57
+ constructor(bitgo, staticsCoin) {
40
58
  super(bitgo);
59
+ if (!staticsCoin) {
60
+ throw new Error('missing required constructor parameter staticsCoin');
61
+ }
62
+ this._staticsCoin = staticsCoin;
41
63
  }
42
- static createInstance(bitgo) {
43
- return new Xrp(bitgo);
64
+ static createInstance(bitgo, staticsCoin) {
65
+ return new Xrp(bitgo, staticsCoin);
44
66
  }
45
67
  /**
46
68
  * Factor between the coin's base unit and its smallest subdivison
47
69
  */
48
70
  getBaseFactor() {
49
- return 1e6;
71
+ return Math.pow(10, this._staticsCoin.decimalPlaces);
50
72
  }
51
73
  /**
52
74
  * Identifier for the blockchain which supports this coin
53
75
  */
54
76
  getChain() {
55
- return 'xrp';
77
+ return this._staticsCoin.name;
56
78
  }
57
79
  /**
58
80
  * Identifier for the coin family
59
81
  */
60
82
  getFamily() {
61
- return 'xrp';
83
+ return this._staticsCoin.family;
62
84
  }
63
85
  /**
64
86
  * Complete human-readable name of this coin
65
87
  */
66
88
  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;
89
+ return this._staticsCoin.fullName;
120
90
  }
121
91
  /**
122
92
  * Evaluates whether an address string is valid for this coin
123
93
  * @param address
124
94
  */
125
95
  isValidAddress(address) {
126
- try {
127
- const addressDetails = this.getAddressDetails(address);
128
- return address === this.normalizeAddress(addressDetails);
129
- }
130
- catch (e) {
131
- return false;
132
- }
96
+ return utils_1.default.isValidAddress(address);
133
97
  }
134
98
  /**
135
99
  * Return boolean indicating whether input is valid public key for the coin.
@@ -138,12 +102,7 @@ class Xrp extends sdk_core_1.BaseCoin {
138
102
  * @returns {Boolean} is it valid?
139
103
  */
140
104
  isValidPub(pub) {
141
- try {
142
- return utxo_lib_1.bip32.fromBase58(pub).isNeutered();
143
- }
144
- catch (e) {
145
- return false;
146
- }
105
+ return utils_1.default.isValidPublicKey(pub);
147
106
  }
148
107
  /**
149
108
  * Get fee info from server
@@ -151,6 +110,12 @@ class Xrp extends sdk_core_1.BaseCoin {
151
110
  async getFeeInfo() {
152
111
  return this.bitgo.get(this.url('/public/feeinfo')).result();
153
112
  }
113
+ getTokenEnablementConfig() {
114
+ return {
115
+ requiresTokenEnablement: true,
116
+ supportsMultipleTokenEnablements: false,
117
+ };
118
+ }
154
119
  /**
155
120
  * Assemble keychain and half-sign prebuilt transaction
156
121
  * @param params
@@ -158,7 +123,7 @@ class Xrp extends sdk_core_1.BaseCoin {
158
123
  * - prv
159
124
  * @returns Bluebird<HalfSignedTransaction>
160
125
  */
161
- async signTransaction({ txPrebuild, prv }) {
126
+ async signTransaction({ txPrebuild, prv, isLastSignature, }) {
162
127
  if (_.isUndefined(txPrebuild) || !_.isObject(txPrebuild)) {
163
128
  if (!_.isUndefined(txPrebuild) && !_.isObject(txPrebuild)) {
164
129
  throw new Error(`txPrebuild must be an object, got type ${typeof txPrebuild}`);
@@ -171,17 +136,21 @@ class Xrp extends sdk_core_1.BaseCoin {
171
136
  }
172
137
  throw new Error('missing prv parameter to sign transaction');
173
138
  }
174
- const userKey = utxo_lib_1.bip32.fromBase58(prv);
175
- const userPrivateKey = userKey.privateKey;
176
- if (!userPrivateKey) {
177
- throw new Error(`no privateKey`);
139
+ if (!txPrebuild.txHex) {
140
+ throw new Error(`missing txHex in txPrebuild`);
178
141
  }
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,
142
+ const keyPair = new keyPair_1.KeyPair({ prv });
143
+ const address = keyPair.getAddress();
144
+ const privateKey = keyPair.getPrivateKey().toString('hex');
145
+ const tx = ripple_1.default.signWithPrivateKey(txPrebuild.txHex, privateKey, {
146
+ signAs: address,
183
147
  });
184
- return { halfSigned: { txHex: halfSigned.signedTransaction } };
148
+ // Normally the SDK provides the first signature for an XRP tx, but occasionally it provides the final one as well
149
+ // (recoveries)
150
+ if (isLastSignature) {
151
+ return { txHex: tx.signedTransaction };
152
+ }
153
+ return { halfSigned: { txHex: tx.signedTransaction } };
185
154
  }
186
155
  /**
187
156
  * Ripple requires additional parameters for wallet generation to be sent to the server. The additional parameters are
@@ -196,11 +165,11 @@ class Xrp extends sdk_core_1.BaseCoin {
196
165
  }
197
166
  }
198
167
  else {
199
- const keyPair = utxo_lib_1.ECPair.makeRandom();
200
- if (!keyPair.privateKey) {
168
+ const keyPair = new keyPair_1.KeyPair().getKeys();
169
+ if (!keyPair.prv) {
201
170
  throw new Error('no privateKey');
202
171
  }
203
- walletParams.rootPrivateKey = keyPair.privateKey.toString('hex');
172
+ walletParams.rootPrivateKey = keyPair.prv;
204
173
  }
205
174
  return walletParams;
206
175
  }
@@ -209,26 +178,33 @@ class Xrp extends sdk_core_1.BaseCoin {
209
178
  * @param params
210
179
  */
211
180
  async explainTransaction(params = {}) {
212
- if (!params.txHex) {
181
+ let transaction;
182
+ let txHex = params.txHex || (params.halfSigned && params.halfSigned.txHex);
183
+ if (!txHex) {
213
184
  throw new Error('missing required param txHex');
214
185
  }
215
- let transaction;
216
- let txHex;
217
186
  try {
218
- transaction = rippleBinaryCodec.decode(params.txHex);
219
- txHex = params.txHex;
187
+ transaction = rippleBinaryCodec.decode(txHex);
220
188
  }
221
189
  catch (e) {
222
190
  try {
223
- transaction = JSON.parse(params.txHex);
191
+ transaction = JSON.parse(txHex);
224
192
  txHex = rippleBinaryCodec.encode(transaction);
225
193
  }
226
194
  catch (e) {
227
195
  throw new Error('txHex needs to be either hex or JSON string for XRP');
228
196
  }
229
197
  }
230
- const id = hashes_1.computeBinaryTransactionHash(txHex);
231
- if (transaction.TransactionType == 'AccountSet') {
198
+ let id;
199
+ // hashes ids are different for signed and unsigned tx
200
+ // first we try to get the hash id as if it is signed, will throw if its not
201
+ try {
202
+ id = xrpl.hashes.hashSignedTx(txHex);
203
+ }
204
+ catch (e) {
205
+ id = xrpl.hashes.hashTx(txHex);
206
+ }
207
+ if (transaction.TransactionType === 'AccountSet') {
232
208
  return {
233
209
  displayOrder: ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee', 'accountSet'],
234
210
  id: id,
@@ -238,11 +214,42 @@ class Xrp extends sdk_core_1.BaseCoin {
238
214
  outputs: [],
239
215
  fee: {
240
216
  fee: transaction.Fee,
241
- feeRate: null,
217
+ feeRate: undefined,
242
218
  size: txHex.length / 2,
243
219
  },
244
220
  accountSet: {
245
221
  messageKey: transaction.MessageKey,
222
+ setFlag: transaction.SetFlag,
223
+ },
224
+ };
225
+ }
226
+ else if (transaction.TransactionType === 'TrustSet') {
227
+ return {
228
+ displayOrder: [
229
+ 'id',
230
+ 'outputAmount',
231
+ 'changeAmount',
232
+ 'outputs',
233
+ 'changeOutputs',
234
+ 'fee',
235
+ 'account',
236
+ 'limitAmount',
237
+ ],
238
+ id: id,
239
+ changeOutputs: [],
240
+ outputAmount: 0,
241
+ changeAmount: 0,
242
+ outputs: [],
243
+ fee: {
244
+ fee: transaction.Fee,
245
+ feeRate: undefined,
246
+ size: txHex.length / 2,
247
+ },
248
+ account: transaction.Account,
249
+ limitAmount: {
250
+ currency: transaction.LimitAmount.currency,
251
+ issuer: transaction.LimitAmount.issuer,
252
+ value: transaction.LimitAmount.value,
246
253
  },
247
254
  };
248
255
  }
@@ -261,7 +268,7 @@ class Xrp extends sdk_core_1.BaseCoin {
261
268
  ],
262
269
  fee: {
263
270
  fee: transaction.Fee,
264
- feeRate: null,
271
+ feeRate: undefined,
265
272
  size: txHex.length / 2,
266
273
  },
267
274
  };
@@ -274,6 +281,7 @@ class Xrp extends sdk_core_1.BaseCoin {
274
281
  * @returns {boolean}
275
282
  */
276
283
  async verifyTransaction({ txParams, txPrebuild }) {
284
+ const coinConfig = statics_1.coins.get(this.getChain());
277
285
  const explanation = await this.explainTransaction({
278
286
  txHex: txPrebuild.txHex,
279
287
  });
@@ -287,9 +295,34 @@ class Xrp extends sdk_core_1.BaseCoin {
287
295
  const amount2 = new bignumber_js_1.BigNumber(recipient2.amount);
288
296
  return amount1.toFixed() === amount2.toFixed();
289
297
  };
290
- if (!comparator(output, expectedOutput)) {
298
+ if ((txParams.type === undefined || txParams.type === 'payment') &&
299
+ typeof output.amount !== 'object' &&
300
+ !comparator(output, expectedOutput)) {
291
301
  throw new Error('transaction prebuild does not match expected output');
292
302
  }
303
+ if (txParams.type === 'enabletoken') {
304
+ if (txParams.recipients?.length !== 1) {
305
+ throw new Error(`${this.getChain()} doesn't support sending to more than 1 destination address within a single transaction. Try again, using only a single recipient.`);
306
+ }
307
+ const recipient = txParams.recipients[0];
308
+ if (!recipient.tokenName) {
309
+ throw new Error('Recipient must include a token name.');
310
+ }
311
+ const recipientCurrency = utils_1.default.getXrpCurrencyFromTokenName(recipient.tokenName).currency;
312
+ if (coinConfig.isToken) {
313
+ if (recipientCurrency !== coinConfig.currencyCode) {
314
+ throw new Error('Incorrect token name specified in recipients');
315
+ }
316
+ }
317
+ if (!('account' in explanation) || !('limitAmount' in explanation) || !explanation.limitAmount.currency) {
318
+ throw new Error('Explanation is missing required keys (account or limitAmount with currency)');
319
+ }
320
+ const baseAddress = explanation.account;
321
+ const currency = explanation.limitAmount.currency;
322
+ if (recipient.address !== baseAddress || recipientCurrency !== currency) {
323
+ throw new Error('Tx outputs does not match with expected txParams recipients');
324
+ }
325
+ }
293
326
  return true;
294
327
  }
295
328
  /**
@@ -303,8 +336,8 @@ class Xrp extends sdk_core_1.BaseCoin {
303
336
  if (!this.isValidAddress(address)) {
304
337
  throw new sdk_core_1.InvalidAddressError(`address verification failure: address "${address}" is not valid`);
305
338
  }
306
- const addressDetails = this.getAddressDetails(address);
307
- const rootAddressDetails = this.getAddressDetails(rootAddress);
339
+ const addressDetails = utils_1.default.getAddressDetails(address);
340
+ const rootAddressDetails = utils_1.default.getAddressDetails(rootAddress);
308
341
  if (addressDetails.address !== rootAddressDetails.address) {
309
342
  throw new sdk_core_1.UnexpectedAddressError(`address validation failure: ${addressDetails.address} vs. ${rootAddressDetails.address}`);
310
343
  }
@@ -336,25 +369,35 @@ class Xrp extends sdk_core_1.BaseCoin {
336
369
  params: [
337
370
  {
338
371
  account: params.rootAddress,
339
- strict: true,
340
372
  ledger_index: 'current',
341
373
  queue: true,
374
+ strict: true,
342
375
  signer_lists: true,
343
376
  },
344
377
  ],
345
378
  };
379
+ const accountLinesParams = {
380
+ method: 'account_lines',
381
+ params: [
382
+ {
383
+ account: params.rootAddress,
384
+ ledger_index: 'validated',
385
+ },
386
+ ],
387
+ };
346
388
  if (isKrsRecovery) {
347
- sdk_core_1.checkKrsProvider(this, params.krsProvider);
389
+ (0, sdk_core_1.checkKrsProvider)(this, params.krsProvider);
348
390
  }
349
391
  // Validate the destination address
350
392
  if (!this.isValidAddress(params.recoveryDestination)) {
351
393
  throw new Error('Invalid destination address!');
352
394
  }
353
- const keys = sdk_core_1.getBip32Keys(this.bitgo, params, { requireBitGoXpub: false });
354
- const { addressDetails, feeDetails, serverDetails } = await sdk_core_1.promiseProps({
395
+ const keys = (0, sdk_core_1.getBip32Keys)(this.bitgo, params, { requireBitGoXpub: false });
396
+ const { addressDetails, feeDetails, serverDetails, accountLines } = await (0, sdk_core_1.promiseProps)({
355
397
  addressDetails: this.bitgo.post(rippledUrl).send(accountInfoParams),
356
398
  feeDetails: this.bitgo.post(rippledUrl).send({ method: 'fee' }),
357
399
  serverDetails: this.bitgo.post(rippledUrl).send({ method: 'server_info' }),
400
+ accountLines: this.bitgo.post(rippledUrl).send(accountLinesParams),
358
401
  });
359
402
  const openLedgerFee = new bignumber_js_1.BigNumber(feeDetails.body.result.drops.open_ledger_fee);
360
403
  const baseReserve = new bignumber_js_1.BigNumber(serverDetails.body.result.info.validated_ledger.reserve_base_xrp).times(this.getBaseFactor());
@@ -364,6 +407,7 @@ class Xrp extends sdk_core_1.BaseCoin {
364
407
  const balance = new bignumber_js_1.BigNumber(addressDetails.body.result.account_data.Balance);
365
408
  const signerLists = addressDetails.body.result.account_data.signer_lists;
366
409
  const accountFlags = addressDetails.body.result.account_data.Flags;
410
+ const ownerCount = new bignumber_js_1.BigNumber(addressDetails.body.result.account_data.OwnerCount);
367
411
  // make sure there is only one signer list set
368
412
  if (signerLists.length !== 1) {
369
413
  throw new Error('unexpected set of signer lists');
@@ -412,86 +456,165 @@ class Xrp extends sdk_core_1.BaseCoin {
412
456
  throw new Error('the destination flag requirement has not been activated');
413
457
  }
414
458
  // recover the funds
415
- const reserve = baseReserve.plus(reserveDelta.times(5));
459
+ const totalReserveDelta = reserveDelta.times(ownerCount);
460
+ const reserve = baseReserve.plus(totalReserveDelta);
416
461
  const recoverableBalance = balance.minus(reserve);
417
462
  const rawDestination = params.recoveryDestination;
418
463
  const destinationDetails = url.parse(rawDestination);
419
- const destinationAddress = destinationDetails.pathname;
420
- // parse destination tag from query
421
- let destinationTag;
422
464
  if (destinationDetails.query) {
423
465
  const queryDetails = querystring.parse(destinationDetails.query);
424
466
  if (Array.isArray(queryDetails.dt)) {
425
467
  // if queryDetails.dt is an array, that means dt was given multiple times, which is not valid
426
468
  throw new sdk_core_1.InvalidAddressError(`destination tag can appear at most once, but ${queryDetails.dt.length} destination tags were found`);
427
469
  }
428
- const parsedTag = parseInt(queryDetails.dt, 10);
429
- if (Number.isInteger(parsedTag)) {
430
- destinationTag = parsedTag;
431
- }
432
470
  }
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);
471
+ if (recoverableBalance.toNumber() <= 0) {
472
+ throw new Error(`Quantity of XRP to recover must be greater than 0. Current balance: ${balance.toNumber()}, blockchain reserve: ${reserve.toNumber()}, spendable balance: ${recoverableBalance.toNumber()}`);
473
+ }
474
+ const issuer = params?.issuerAddress;
475
+ const currency = params?.currencyCode;
476
+ if (!!issuer && !!currency) {
477
+ const tokenParams = {
478
+ recoveryDestination: params.recoveryDestination,
479
+ recoverableBalance,
480
+ currentLedger,
481
+ openLedgerFee,
482
+ sequenceId,
483
+ accountLines,
484
+ keys,
485
+ isKrsRecovery,
486
+ isUnsignedSweep,
487
+ userAddress,
488
+ backupAddress,
489
+ issuer,
490
+ currency,
491
+ };
492
+ return this.recoverXrpToken(params, tokenParams);
493
+ }
494
+ const factory = new lib_1.TransactionBuilderFactory(statics_1.coins.get(this.getChain()));
495
+ const txBuilder = factory.getTransferBuilder();
496
+ txBuilder
497
+ .to(params.recoveryDestination)
498
+ .amount(recoverableBalance.toFixed(0))
499
+ .sender(params.rootAddress)
500
+ .flags(2147483648)
501
+ .lastLedgerSequence(currentLedger + 1000000) // give it 1 million ledgers' time (~1 month, suitable for KRS)
502
+ .fee(openLedgerFee.times(3).toFixed(0)) // the factor three is for the multisigning
503
+ .sequence(sequenceId);
504
+ const tx = await txBuilder.build();
505
+ const serializedTx = tx.toBroadcastFormat();
445
506
  if (isUnsignedSweep) {
446
- return txJSON;
507
+ return {
508
+ txHex: serializedTx,
509
+ coin: this.getChain(),
510
+ };
447
511
  }
448
- const rippleLib = ripple();
449
512
  if (!keys[0].privateKey) {
450
513
  throw new Error(`userKey is not a private key`);
451
514
  }
452
515
  const userKey = keys[0].privateKey.toString('hex');
453
- const userSignature = rippleLib.signWithPrivateKey(txJSON, userKey, { signAs: userAddress });
516
+ const userSignature = ripple_1.default.signWithPrivateKey(serializedTx, userKey, { signAs: userAddress });
454
517
  let signedTransaction;
455
518
  if (isKrsRecovery) {
456
- signedTransaction = userSignature;
519
+ signedTransaction = userSignature.signedTransaction;
457
520
  }
458
521
  else {
459
522
  if (!keys[1].privateKey) {
460
523
  throw new Error(`backupKey is not a private key`);
461
524
  }
462
525
  const backupKey = keys[1].privateKey.toString('hex');
463
- const backupSignature = rippleLib.signWithPrivateKey(txJSON, backupKey, { signAs: backupAddress });
464
- signedTransaction = rippleLib.combine([userSignature.signedTransaction, backupSignature.signedTransaction]);
526
+ const backupSignature = ripple_1.default.signWithPrivateKey(serializedTx, backupKey, { signAs: backupAddress });
527
+ signedTransaction = ripple_1.default.multisign([userSignature.signedTransaction, backupSignature.signedTransaction]);
465
528
  }
466
529
  const transactionExplanation = (await this.explainTransaction({
467
- txHex: signedTransaction.signedTransaction,
530
+ txHex: signedTransaction,
468
531
  }));
469
- transactionExplanation.txHex = signedTransaction.signedTransaction;
532
+ transactionExplanation.txHex = signedTransaction;
470
533
  if (isKrsRecovery) {
471
534
  transactionExplanation.backupKey = params.backupKey;
472
535
  transactionExplanation.coin = this.getChain();
473
536
  }
474
537
  return transactionExplanation;
475
538
  }
476
- initiateRecovery(params) {
477
- throw new Error('deprecated method');
539
+ async recoverXrpToken(params, tokenParams) {
540
+ const { currency, issuer } = tokenParams;
541
+ const tokenName = utils_1.default.getXrpToken(issuer, currency).name;
542
+ const lines = tokenParams.accountLines.body.result.lines;
543
+ let amount;
544
+ for (const line of lines) {
545
+ if (line.currency === currency && line.account === issuer) {
546
+ amount = line.balance;
547
+ break;
548
+ }
549
+ }
550
+ if (amount === undefined) {
551
+ throw new Error(`Does not have Trustline with ${issuer}`);
552
+ }
553
+ if (amount === '0') {
554
+ throw new Error(`Does not have funds to recover`);
555
+ }
556
+ const decimalPlaces = statics_1.coins.get(tokenName).decimalPlaces;
557
+ amount = new bignumber_js_1.BigNumber(amount).shiftedBy(decimalPlaces).toFixed();
558
+ const FLAG_VALUE = 2147483648;
559
+ const factory = new lib_1.TransactionBuilderFactory(statics_1.coins.get(tokenName));
560
+ const txBuilder = factory.getTokenTransferBuilder();
561
+ txBuilder
562
+ .to(tokenParams.recoveryDestination)
563
+ .amount(amount)
564
+ .sender(params.rootAddress)
565
+ .flags(FLAG_VALUE)
566
+ .lastLedgerSequence(tokenParams.currentLedger + 1000000) // give it 1 million ledgers' time (~1 month, suitable for KRS)
567
+ .fee(tokenParams.openLedgerFee.times(3).toFixed(0)) // the factor three is for the multisigning
568
+ .sequence(tokenParams.sequenceId);
569
+ const tx = await txBuilder.build();
570
+ const serializedTx = tx.toBroadcastFormat();
571
+ const { keys, isKrsRecovery, isUnsignedSweep, userAddress, backupAddress } = tokenParams;
572
+ if (isUnsignedSweep) {
573
+ return {
574
+ txHex: serializedTx,
575
+ coin: this.getChain(),
576
+ };
577
+ }
578
+ if (!keys[0].privateKey) {
579
+ throw new Error(`userKey is not a private key`);
580
+ }
581
+ const userKey = keys[0].privateKey.toString('hex');
582
+ const userSignature = ripple_1.default.signWithPrivateKey(serializedTx, userKey, { signAs: userAddress });
583
+ let signedTransaction;
584
+ if (isKrsRecovery) {
585
+ signedTransaction = userSignature.signedTransaction;
586
+ }
587
+ else {
588
+ if (!keys[1].privateKey) {
589
+ throw new Error(`backupKey is not a private key`);
590
+ }
591
+ const backupKey = keys[1].privateKey.toString('hex');
592
+ const backupSignature = ripple_1.default.signWithPrivateKey(serializedTx, backupKey, { signAs: backupAddress });
593
+ signedTransaction = ripple_1.default.multisign([userSignature.signedTransaction, backupSignature.signedTransaction]);
594
+ }
595
+ const transactionExplanation = (await this.explainTransaction({
596
+ txHex: signedTransaction,
597
+ }));
598
+ transactionExplanation.txHex = signedTransaction;
599
+ if (isKrsRecovery) {
600
+ transactionExplanation.backupKey = params.backupKey;
601
+ transactionExplanation.coin = this.getChain();
602
+ }
603
+ return transactionExplanation;
478
604
  }
479
605
  /**
480
606
  * Generate a new keypair for this coin.
481
607
  * @param seed Seed from which the new keypair should be generated, otherwise a random seed is used
482
608
  */
483
609
  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();
610
+ const keyPair = seed ? new keyPair_1.KeyPair({ seed }) : new keyPair_1.KeyPair();
611
+ const keys = keyPair.getExtendedKeys();
612
+ if (!keys.xprv) {
613
+ throw new Error('Missing prv in key generation.');
614
+ }
492
615
  return {
493
- pub: xpub,
494
- prv: extendedKey.toBase58(),
616
+ pub: keys.xpub,
617
+ prv: keys.xprv,
495
618
  };
496
619
  }
497
620
  async parseTransaction(params) {
@@ -499,4 +622,4 @@ class Xrp extends sdk_core_1.BaseCoin {
499
622
  }
500
623
  }
501
624
  exports.Xrp = Xrp;
502
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"xrp.js","sourceRoot":"","sources":["../../src/xrp.ts"],"names":[],"mappings":";;;;;;;;;;;;;;;;;;;;;;AAAA;;GAEG;AACH,+CAAyC;AACzC,mDAAqD;AACrD,mCAAqC;AACrC,0CAA4B;AAC5B,yCAA2B;AAC3B,yDAA2C;AAE3C,yEAA2D;AAC3D,uEAAyD;AACzD,8DAAiF;AACjF,gEAAkD;AAClD,mDAiB8B;AAE9B,MAAM,MAAM,GAAG,OAAO,CAAC,UAAU,CAAC,CAAC;AAyDnC,MAAa,GAAI,SAAQ,mBAAQ;IAC/B,YAAsB,KAAgB;QACpC,KAAK,CAAC,KAAK,CAAC,CAAC;IACf,CAAC;IAED,MAAM,CAAC,cAAc,CAAC,KAAgB;QACpC,OAAO,IAAI,GAAG,CAAC,KAAK,CAAC,CAAC;IACxB,CAAC;IAED;;OAEG;IACI,aAAa;QAClB,OAAO,GAAG,CAAC;IACb,CAAC;IAED;;OAEG;IACI,QAAQ;QACb,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;OAEG;IACI,SAAS;QACd,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;OAEG;IACI,WAAW;QAChB,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED;;OAEG;IACI,iBAAiB,CAAC,OAAe;QACtC,MAAM,kBAAkB,GAAG,GAAG,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;QAC9C,MAAM,kBAAkB,GAAG,kBAAkB,CAAC,QAAQ,CAAC;QACvD,IAAI,CAAC,kBAAkB,IAAI,CAAC,kBAAkB,CAAC,qBAAqB,CAAC,kBAAkB,CAAC,EAAE;YACxF,MAAM,IAAI,8BAAmB,CAAC,wBAAwB,kBAAkB,gBAAgB,CAAC,CAAC;SAC3F;QACD,sDAAsD;QACtD,IAAI,kBAAkB,CAAC,QAAQ,KAAK,OAAO,EAAE;YAC3C,OAAO;gBACL,OAAO,EAAE,OAAO;gBAChB,cAAc,EAAE,SAAS;aAC1B,CAAC;SACH;QAED,IAAI,CAAC,kBAAkB,CAAC,KAAK,EAAE;YAC7B,MAAM,IAAI,8BAAmB,CAAC,yBAAyB,CAAC,CAAC;SAC1D;QAED,MAAM,YAAY,GAAG,WAAW,CAAC,KAAK,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC;QACjE,IAAI,CAAC,YAAY,CAAC,EAAE,EAAE;YACpB,gGAAgG;YAChG,MAAM,IAAI,8BAAmB,CAAC,yBAAyB,CAAC,CAAC;SAC1D;QAED,IAAI,KAAK,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE,CAAC,EAAE;YAClC,6FAA6F;YAC7F,MAAM,IAAI,8BAAmB,CAC3B,gDAAgD,YAAY,CAAC,EAAE,CAAC,MAAM,8BAA8B,CACrG,CAAC;SACH;QAED,MAAM,SAAS,GAAG,QAAQ,CAAC,YAAY,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC;QAChD,IAAI,CAAC,MAAM,CAAC,aAAa,CAAC,SAAS,CAAC,EAAE;YACpC,MAAM,IAAI,8BAAmB,CAAC,yBAAyB,CAAC,CAAC;SAC1D;QAED,IAAI,SAAS,GAAG,UAAU,IAAI,SAAS,GAAG,CAAC,EAAE;YAC3C,MAAM,IAAI,8BAAmB,CAAC,8BAA8B,CAAC,CAAC;SAC/D;QAED,OAAO;YACL,OAAO,EAAE,kBAAkB;YAC3B,cAAc,EAAE,SAAS;SAC1B,CAAC;IACJ,CAAC;IAED;;OAEG;IACI,gBAAgB,CAAC,EAAE,OAAO,EAAE,cAAc,EAAW;QAC1D,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE;YACxB,MAAM,IAAI,8BAAmB,CAAC,yBAAyB,CAAC,CAAC;SAC1D;QACD,IAAI,CAAC,CAAC,SAAS,CAAC,cAAc,CAAC,EAAE;YAC/B,OAAO,GAAG,OAAO,OAAO,cAAc,EAAE,CAAC;SAC1C;QACD,OAAO,OAAO,CAAC;IACjB,CAAC;IAED;;;OAGG;IACI,cAAc,CAAC,OAAe;QACnC,IAAI;YACF,MAAM,cAAc,GAAG,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAAC,CAAC;YACvD,OAAO,OAAO,KAAK,IAAI,CAAC,gBAAgB,CAAC,cAAc,CAAC,CAAC;SAC1D;QAAC,OAAO,CAAC,EAAE;YACV,OAAO,KAAK,CAAC;SACd;IACH,CAAC;IAED;;;;;OAKG;IACI,UAAU,CAAC,GAAW;QAC3B,IAAI;YACF,OAAO,gBAAK,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,UAAU,EAAE,CAAC;SAC3C;QAAC,OAAO,CAAC,EAAE;YACV,OAAO,KAAK,CAAC;SACd;IACH,CAAC;IAED;;OAEG;IACI,KAAK,CAAC,UAAU;QACrB,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC,CAAC,MAAM,EAAE,CAAC;IAC9D,CAAC;IAED;;;;;;OAMG;IACI,KAAK,CAAC,eAAe,CAAC,EAAE,UAAU,EAAE,GAAG,EAA0B;QACtE,IAAI,CAAC,CAAC,WAAW,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE;YACxD,IAAI,CAAC,CAAC,CAAC,WAAW,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,UAAU,CAAC,EAAE;gBACzD,MAAM,IAAI,KAAK,CAAC,0CAA0C,OAAO,UAAU,EAAE,CAAC,CAAC;aAChF;YACD,MAAM,IAAI,KAAK,CAAC,8BAA8B,CAAC,CAAC;SACjD;QAED,IAAI,CAAC,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE;YAC1C,IAAI,CAAC,CAAC,CAAC,WAAW,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE;gBAC3C,MAAM,IAAI,KAAK,CAAC,kCAAkC,OAAO,GAAG,EAAE,CAAC,CAAC;aACjE;YACD,MAAM,IAAI,KAAK,CAAC,2CAA2C,CAAC,CAAC;SAC9D;QAED,MAAM,OAAO,GAAG,gBAAK,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC;QACtC,MAAM,cAAc,GAAG,OAAO,CAAC,UAAU,CAAC;QAC1C,IAAI,CAAC,cAAc,EAAE;YACnB,MAAM,IAAI,KAAK,CAAC,eAAe,CAAC,CAAC;SAClC;QACD,MAAM,WAAW,GAAG,cAAc,CAAC,aAAa,CAAC,OAAO,CAAC,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC;QAEpF,MAAM,SAAS,GAAG,MAAM,EAAE,CAAC;QAC3B,MAAM,UAAU,GAAG,SAAS,CAAC,kBAAkB,CAAC,UAAU,CAAC,KAAK,EAAE,cAAc,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE;YAChG,MAAM,EAAE,WAAW;SACpB,CAAC,CAAC;QACH,OAAO,EAAE,UAAU,EAAE,EAAE,KAAK,EAAE,UAAU,CAAC,iBAAiB,EAAE,EAAE,CAAC;IACjE,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,wBAAwB,CAC5B,YAA6C;QAE7C,IAAI,YAAY,CAAC,cAAc,EAAE;YAC/B,IAAI,YAAY,CAAC,cAAc,CAAC,MAAM,KAAK,EAAE,EAAE;gBAC7C,MAAM,IAAI,KAAK,CAAC,6DAA6D,CAAC,CAAC;aAChF;SACF;aAAM;YACL,MAAM,OAAO,GAAG,iBAAM,CAAC,UAAU,EAAE,CAAC;YACpC,IAAI,CAAC,OAAO,CAAC,UAAU,EAAE;gBACvB,MAAM,IAAI,KAAK,CAAC,eAAe,CAAC,CAAC;aAClC;YACD,YAAY,CAAC,cAAc,GAAG,OAAO,CAAC,UAAU,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;SAClE;QACD,OAAO,YAAY,CAAC;IACtB,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,kBAAkB,CAAC,SAAoC,EAAE;QAC7D,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE;YACjB,MAAM,IAAI,KAAK,CAAC,8BAA8B,CAAC,CAAC;SACjD;QACD,IAAI,WAAW,CAAC;QAChB,IAAI,KAAK,CAAC;QACV,IAAI;YACF,WAAW,GAAG,iBAAiB,CAAC,MAAM,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;YACrD,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC;SACtB;QAAC,OAAO,CAAC,EAAE;YACV,IAAI;gBACF,WAAW,GAAG,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;gBACvC,KAAK,GAAG,iBAAiB,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC;aAC/C;YAAC,OAAO,CAAC,EAAE;gBACV,MAAM,IAAI,KAAK,CAAC,qDAAqD,CAAC,CAAC;aACxE;SACF;QACD,MAAM,EAAE,GAAG,qCAA4B,CAAC,KAAK,CAAC,CAAC;QAE/C,IAAI,WAAW,CAAC,eAAe,IAAI,YAAY,EAAE;YAC/C,OAAO;gBACL,YAAY,EAAE,CAAC,IAAI,EAAE,cAAc,EAAE,cAAc,EAAE,SAAS,EAAE,eAAe,EAAE,KAAK,EAAE,YAAY,CAAC;gBACrG,EAAE,EAAE,EAAE;gBACN,aAAa,EAAE,EAAE;gBACjB,YAAY,EAAE,CAAC;gBACf,YAAY,EAAE,CAAC;gBACf,OAAO,EAAE,EAAE;gBACX,GAAG,EAAE;oBACH,GAAG,EAAE,WAAW,CAAC,GAAG;oBACpB,OAAO,EAAE,IAAI;oBACb,IAAI,EAAE,KAAK,CAAC,MAAM,GAAG,CAAC;iBACvB;gBACD,UAAU,EAAE;oBACV,UAAU,EAAE,WAAW,CAAC,UAAU;iBACnC;aACK,CAAC;SACV;QAED,MAAM,OAAO,GACX,WAAW,CAAC,WAAW,GAAG,CAAC,WAAW,CAAC,cAAc,IAAI,CAAC,CAAC,CAAC,CAAC,MAAM,GAAG,WAAW,CAAC,cAAc,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;QACzG,OAAO;YACL,YAAY,EAAE,CAAC,IAAI,EAAE,cAAc,EAAE,cAAc,EAAE,SAAS,EAAE,eAAe,EAAE,KAAK,CAAC;YACvF,EAAE,EAAE,EAAE;YACN,aAAa,EAAE,EAAE;YACjB,YAAY,EAAE,WAAW,CAAC,MAAM;YAChC,YAAY,EAAE,CAAC;YACf,OAAO,EAAE;gBACP;oBACE,OAAO;oBACP,MAAM,EAAE,WAAW,CAAC,MAAM;iBAC3B;aACF;YACD,GAAG,EAAE;gBACH,GAAG,EAAE,WAAW,CAAC,GAAG;gBACpB,OAAO,EAAE,IAAI;gBACb,IAAI,EAAE,KAAK,CAAC,MAAM,GAAG,CAAC;aACvB;SACK,CAAC;IACX,CAAC;IAED;;;;;;OAMG;IACI,KAAK,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,UAAU,EAA4B;QAC/E,MAAM,WAAW,GAAG,MAAM,IAAI,CAAC,kBAAkB,CAAC;YAChD,KAAK,EAAE,UAAU,CAAC,KAAK;SACxB,CAAC,CAAC;QAEH,MAAM,MAAM,GAAG,CAAC,GAAG,WAAW,CAAC,OAAO,EAAE,GAAG,WAAW,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC;QACzE,MAAM,cAAc,GAAG,QAAQ,CAAC,UAAU,IAAI,QAAQ,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC;QAErE,MAAM,UAAU,GAAG,CAAC,UAAU,EAAE,UAAU,EAAE,EAAE;YAC5C,IAAI,UAAU,CAAC,OAAO,KAAK,UAAU,CAAC,OAAO,EAAE;gBAC7C,OAAO,KAAK,CAAC;aACd;YACD,MAAM,OAAO,GAAG,IAAI,wBAAS,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;YACjD,MAAM,OAAO,GAAG,IAAI,wBAAS,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;YACjD,OAAO,OAAO,CAAC,OAAO,EAAE,KAAK,OAAO,CAAC,OAAO,EAAE,CAAC;QACjD,CAAC,CAAC;QAEF,IAAI,CAAC,UAAU,CAAC,MAAM,EAAE,cAAc,CAAC,EAAE;YACvC,MAAM,IAAI,KAAK,CAAC,qDAAqD,CAAC,CAAC;SACxE;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;;;OAMG;IACI,KAAK,CAAC,eAAe,CAAC,EAAE,OAAO,EAAE,WAAW,EAAwB;QACzE,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,EAAE;YACjC,MAAM,IAAI,8BAAmB,CAAC,0CAA0C,OAAO,gBAAgB,CAAC,CAAC;SAClG;QAED,MAAM,cAAc,GAAG,IAAI,CAAC,iBAAiB,CAAC,OAAO,CAAC,CAAC;QACvD,MAAM,kBAAkB,GAAG,IAAI,CAAC,iBAAiB,CAAC,WAAW,CAAC,CAAC;QAE/D,IAAI,cAAc,CAAC,OAAO,KAAK,kBAAkB,CAAC,OAAO,EAAE;YACzD,MAAM,IAAI,iCAAsB,CAC9B,+BAA+B,cAAc,CAAC,OAAO,QAAQ,kBAAkB,CAAC,OAAO,EAAE,CAC1F,CAAC;SACH;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;OAEG;IACI,aAAa;QAClB,OAAO,6BAA6B,CAAC;IACvC,CAAC;IAED;;;;;;;;;;OAUG;IACI,KAAK,CAAC,OAAO,CAAC,MAAuB;QAC1C,MAAM,UAAU,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;QACxC,MAAM,aAAa,GAAG,MAAM,CAAC,SAAS,CAAC,UAAU,CAAC,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;QAChG,MAAM,eAAe,GAAG,MAAM,CAAC,SAAS,CAAC,UAAU,CAAC,MAAM,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC;QAEjG,MAAM,iBAAiB,GAAG;YACxB,MAAM,EAAE,cAAc;YACtB,MAAM,EAAE;gBACN;oBACE,OAAO,EAAE,MAAM,CAAC,WAAW;oBAC3B,MAAM,EAAE,IAAI;oBACZ,YAAY,EAAE,SAAS;oBACvB,KAAK,EAAE,IAAI;oBACX,YAAY,EAAE,IAAI;iBACnB;aACF;SACF,CAAC;QAEF,IAAI,aAAa,EAAE;YACjB,2BAAgB,CAAC,IAAI,EAAE,MAAM,CAAC,WAAW,CAAC,CAAC;SAC5C;QAED,mCAAmC;QACnC,IAAI,CAAC,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,mBAAmB,CAAC,EAAE;YACpD,MAAM,IAAI,KAAK,CAAC,8BAA8B,CAAC,CAAC;SACjD;QAED,MAAM,IAAI,GAAG,uBAAY,CAAC,IAAI,CAAC,KAAK,EAAE,MAAM,EAAE,EAAE,gBAAgB,EAAE,KAAK,EAAE,CAAC,CAAC;QAE3E,MAAM,EAAE,cAAc,EAAE,UAAU,EAAE,aAAa,EAAE,GAAG,MAAM,uBAAY,CAAC;YACvE,cAAc,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC;YACnE,UAAU,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,KAAK,EAAE,CAAC;YAC/D,aAAa,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,aAAa,EAAE,CAAC;SAC3E,CAAC,CAAC;QAEH,MAAM,aAAa,GAAG,IAAI,wBAAS,CAAC,UAAU,CAAC,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,eAAe,CAAC,CAAC;QAClF,MAAM,WAAW,GAAG,IAAI,wBAAS,CAAC,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,gBAAgB,CAAC,gBAAgB,CAAC,CAAC,KAAK,CACvG,IAAI,CAAC,aAAa,EAAE,CACrB,CAAC;QACF,MAAM,YAAY,GAAG,IAAI,wBAAS,CAAC,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,gBAAgB,CAAC,eAAe,CAAC,CAAC,KAAK,CACvG,IAAI,CAAC,aAAa,EAAE,CACrB,CAAC;QACF,MAAM,aAAa,GAAG,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC;QAC1E,MAAM,UAAU,GAAG,cAAc,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,QAAQ,CAAC;QACpE,MAAM,OAAO,GAAG,IAAI,wBAAS,CAAC,cAAc,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC;QAC/E,MAAM,WAAW,GAAG,cAAc,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,YAAY,CAAC;QACzE,MAAM,YAAY,GAAG,cAAc,CAAC,IAAI,CAAC,MAAM,CAAC,YAAY,CAAC,KAAK,CAAC;QAEnE,8CAA8C;QAC9C,IAAI,WAAW,CAAC,MAAM,KAAK,CAAC,EAAE;YAC5B,MAAM,IAAI,KAAK,CAAC,gCAAgC,CAAC,CAAC;SACnD;QAED,gDAAgD;QAChD,MAAM,WAAW,GAAG,cAAc,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC;QACpF,MAAM,aAAa,GAAG,cAAc,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC;QAEtF,MAAM,UAAU,GAAG,WAAW,CAAC,CAAC,CAAC,CAAC;QAClC,IAAI,UAAU,CAAC,YAAY,KAAK,CAAC,EAAE;YACjC,MAAM,IAAI,KAAK,CAAC,iCAAiC,CAAC,CAAC;SACpD;QACD,MAAM,cAAc,GAAG,EAAE,CAAC;QAE1B,MAAM,aAAa,GAAG,UAAU,CAAC,aAAa,CAAC;QAC/C,IAAI,aAAa,CAAC,MAAM,KAAK,CAAC,EAAE;YAC9B,MAAM,IAAI,KAAK,CAAC,4BAA4B,CAAC,CAAC;SAC/C;QACD,KAAK,MAAM,EAAE,WAAW,EAAE,IAAI,aAAa,EAAE;YAC3C,MAAM,MAAM,GAAG,WAAW,CAAC,YAAY,CAAC;YACxC,MAAM,OAAO,GAAG,WAAW,CAAC,OAAO,CAAC;YACpC,IAAI,MAAM,KAAK,CAAC,EAAE;gBAChB,MAAM,IAAI,KAAK,CAAC,uBAAuB,CAAC,CAAC;aAC1C;YAED,sDAAsD;YACtD,IAAI,cAAc,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;gBAChC,MAAM,IAAI,KAAK,CAAC,0BAA0B,CAAC,CAAC;aAC7C;YACD,cAAc,CAAC,OAAO,CAAC,GAAG,CAAC,cAAc,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC;SAC9D;QAED,IAAI,cAAc,CAAC,WAAW,CAAC,KAAK,CAAC,EAAE;YACrC,MAAM,IAAI,KAAK,CAAC,uDAAuD,CAAC,CAAC;SAC1E;QACD,IAAI,cAAc,CAAC,aAAa,CAAC,KAAK,CAAC,EAAE;YACvC,MAAM,IAAI,KAAK,CAAC,uDAAuD,CAAC,CAAC;SAC1E;QAED,0EAA0E;QAC1E,MAAM,qBAAqB,GAAG,KAAK,CAAC;QACpC,MAAM,4BAA4B,GAAG,OAAO,CAAC;QAC7C,MAAM,4BAA4B,GAAG,MAAM,CAAC;QAC5C,IAAI,CAAC,YAAY,GAAG,qBAAqB,CAAC,KAAK,CAAC,EAAE;YAChD,MAAM,IAAI,KAAK,CAAC,gCAAgC,CAAC,CAAC;SACnD;QACD,IAAI,CAAC,YAAY,GAAG,4BAA4B,CAAC,KAAK,4BAA4B,EAAE;YAClF,MAAM,IAAI,KAAK,CAAC,yCAAyC,CAAC,CAAC;SAC5D;QACD,IAAI,CAAC,YAAY,GAAG,4BAA4B,CAAC,KAAK,4BAA4B,EAAE;YAClF,MAAM,IAAI,KAAK,CAAC,yDAAyD,CAAC,CAAC;SAC5E;QAED,oBAAoB;QACpB,MAAM,OAAO,GAAG,WAAW,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QACxD,MAAM,kBAAkB,GAAG,OAAO,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;QAElD,MAAM,cAAc,GAAG,MAAM,CAAC,mBAAmB,CAAC;QAClD,MAAM,kBAAkB,GAAG,GAAG,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;QACrD,MAAM,kBAAkB,GAAG,kBAAkB,CAAC,QAAQ,CAAC;QAEvD,mCAAmC;QACnC,IAAI,cAAkC,CAAC;QACvC,IAAI,kBAAkB,CAAC,KAAK,EAAE;YAC5B,MAAM,YAAY,GAAG,WAAW,CAAC,KAAK,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC;YACjE,IAAI,KAAK,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE,CAAC,EAAE;gBAClC,6FAA6F;gBAC7F,MAAM,IAAI,8BAAmB,CAC3B,gDAAgD,YAAY,CAAC,EAAE,CAAC,MAAM,8BAA8B,CACrG,CAAC;aACH;YAED,MAAM,SAAS,GAAG,QAAQ,CAAC,YAAY,CAAC,EAAY,EAAE,EAAE,CAAC,CAAC;YAC1D,IAAI,MAAM,CAAC,SAAS,CAAC,SAAS,CAAC,EAAE;gBAC/B,cAAc,GAAG,SAAS,CAAC;aAC5B;SACF;QAED,MAAM,WAAW,GAAG;YAClB,eAAe,EAAE,SAAS;YAC1B,OAAO,EAAE,MAAM,CAAC,WAAW;YAC3B,WAAW,EAAE,kBAAkB;YAC/B,cAAc,EAAE,cAAc;YAC9B,MAAM,EAAE,kBAAkB,CAAC,OAAO,CAAC,CAAC,CAAC;YACrC,KAAK,EAAE,UAAU;YACjB,kBAAkB,EAAE,aAAa,GAAG,OAAO;YAC3C,GAAG,EAAE,aAAa,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC;YACtC,QAAQ,EAAE,UAAU;SACrB,CAAC;QACF,MAAM,MAAM,GAAW,IAAI,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC;QAEnD,IAAI,eAAe,EAAE;YACnB,OAAO,MAAM,CAAC;SACf;QACD,MAAM,SAAS,GAAG,MAAM,EAAE,CAAC;QAC3B,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,UAAU,EAAE;YACvB,MAAM,IAAI,KAAK,CAAC,8BAA8B,CAAC,CAAC;SACjD;QACD,MAAM,OAAO,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QACnD,MAAM,aAAa,GAAG,SAAS,CAAC,kBAAkB,CAAC,MAAM,EAAE,OAAO,EAAE,EAAE,MAAM,EAAE,WAAW,EAAE,CAAC,CAAC;QAE7F,IAAI,iBAAiB,CAAC;QAEtB,IAAI,aAAa,EAAE;YACjB,iBAAiB,GAAG,aAAa,CAAC;SACnC;aAAM;YACL,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,UAAU,EAAE;gBACvB,MAAM,IAAI,KAAK,CAAC,gCAAgC,CAAC,CAAC;aACnD;YACD,MAAM,SAAS,GAAG,IAAI,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;YACrD,MAAM,eAAe,GAAG,SAAS,CAAC,kBAAkB,CAAC,MAAM,EAAE,SAAS,EAAE,EAAE,MAAM,EAAE,aAAa,EAAE,CAAC,CAAC;YACnG,iBAAiB,GAAG,SAAS,CAAC,OAAO,CAAC,CAAC,aAAa,CAAC,iBAAiB,EAAE,eAAe,CAAC,iBAAiB,CAAC,CAAC,CAAC;SAC7G;QAED,MAAM,sBAAsB,GAAiB,CAAC,MAAM,IAAI,CAAC,kBAAkB,CAAC;YAC1E,KAAK,EAAE,iBAAiB,CAAC,iBAAiB;SAC3C,CAAC,CAAQ,CAAC;QACX,sBAAsB,CAAC,KAAK,GAAG,iBAAiB,CAAC,iBAAiB,CAAC;QAEnE,IAAI,aAAa,EAAE;YACjB,sBAAsB,CAAC,SAAS,GAAG,MAAM,CAAC,SAAS,CAAC;YACpD,sBAAsB,CAAC,IAAI,GAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;SAC/C;QACD,OAAO,sBAAsB,CAAC;IAChC,CAAC;IAED,gBAAgB,CAAC,MAA+B;QAC9C,MAAM,IAAI,KAAK,CAAC,mBAAmB,CAAC,CAAC;IACvC,CAAC;IAED;;;OAGG;IACI,eAAe,CAAC,IAAa;QAClC,IAAI,CAAC,IAAI,EAAE;YACT,0EAA0E;YAC1E,0EAA0E;YAC1E,kEAAkE;YAClE,IAAI,GAAG,oBAAW,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC;SAC7B;QACD,MAAM,WAAW,GAAG,gBAAK,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;QACzC,MAAM,IAAI,GAAG,WAAW,CAAC,QAAQ,EAAE,CAAC,QAAQ,EAAE,CAAC;QAC/C,OAAO;YACL,GAAG,EAAE,IAAI;YACT,GAAG,EAAE,WAAW,CAAC,QAAQ,EAAE;SAC5B,CAAC;IACJ,CAAC;IAED,KAAK,CAAC,gBAAgB,CAAC,MAA+B;QACpD,OAAO,EAAE,CAAC;IACZ,CAAC;CACF;AAlhBD,kBAkhBC","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"]}
625
+ //# 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 * as _ from 'lodash';\nimport * as querystring from 'querystring';\nimport * as url from 'url';\n\nimport {\n  BaseCoin,\n  BitGoBase,\n  checkKrsProvider,\n  getBip32Keys,\n  InvalidAddressError,\n  KeyPair,\n  ParsedTransaction,\n  ParseTransactionOptions,\n  promiseProps,\n  TokenEnablementConfig,\n  UnexpectedAddressError,\n  VerifyTransactionOptions,\n} from '@bitgo-beta/sdk-core';\nimport { BaseCoin as StaticsBaseCoin, coins, XrpCoin } from '@bitgo-beta/statics';\nimport * as rippleBinaryCodec from 'ripple-binary-codec';\nimport * as rippleKeypairs from 'ripple-keypairs';\nimport * as xrpl from 'xrpl';\n\nimport {\n  ExplainTransactionOptions,\n  FeeInfo,\n  HalfSignedTransaction,\n  RecoveryInfo,\n  RecoveryOptions,\n  RecoveryTransaction,\n  SignTransactionOptions,\n  SupplementGenerateWalletOptions,\n  TransactionExplanation,\n  VerifyAddressOptions,\n} from './lib/iface';\nimport { KeyPair as XrpKeyPair } from './lib/keyPair';\nimport utils from './lib/utils';\nimport ripple from './ripple';\nimport { TokenTransferBuilder, TransactionBuilderFactory, TransferBuilder } from './lib';\n\nexport class Xrp extends BaseCoin {\n  protected _staticsCoin: Readonly<StaticsBaseCoin>;\n  protected constructor(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>) {\n    super(bitgo);\n    if (!staticsCoin) {\n      throw new Error('missing required constructor parameter staticsCoin');\n    }\n    this._staticsCoin = staticsCoin;\n  }\n\n  static createInstance(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>): BaseCoin {\n    return new Xrp(bitgo, staticsCoin);\n  }\n\n  /**\n   * Factor between the coin's base unit and its smallest subdivison\n   */\n  public getBaseFactor(): number {\n    return Math.pow(10, this._staticsCoin.decimalPlaces);\n  }\n\n  /**\n   * Identifier for the blockchain which supports this coin\n   */\n  public getChain(): string {\n    return this._staticsCoin.name;\n  }\n\n  /**\n   * Identifier for the coin family\n   */\n  public getFamily(): string {\n    return this._staticsCoin.family;\n  }\n\n  /**\n   * Complete human-readable name of this coin\n   */\n  public getFullName(): string {\n    return this._staticsCoin.fullName;\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    return utils.isValidAddress(address);\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    return utils.isValidPublicKey(pub);\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  public getTokenEnablementConfig(): TokenEnablementConfig {\n    return {\n      requiresTokenEnablement: true,\n      supportsMultipleTokenEnablements: false,\n    };\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({\n    txPrebuild,\n    prv,\n    isLastSignature,\n  }: SignTransactionOptions): Promise<HalfSignedTransaction | RecoveryTransaction> {\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    if (!txPrebuild.txHex) {\n      throw new Error(`missing txHex in txPrebuild`);\n    }\n    const keyPair = new XrpKeyPair({ prv });\n    const address = keyPair.getAddress();\n    const privateKey = (keyPair.getPrivateKey() as Buffer).toString('hex');\n\n    const tx = ripple.signWithPrivateKey(txPrebuild.txHex, privateKey, {\n      signAs: address,\n    });\n\n    // Normally the SDK provides the first signature for an XRP tx, but occasionally it provides the final one as well\n    // (recoveries)\n    if (isLastSignature) {\n      return { txHex: tx.signedTransaction };\n    }\n    return { halfSigned: { txHex: tx.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 = new XrpKeyPair().getKeys();\n      if (!keyPair.prv) {\n        throw new Error('no privateKey');\n      }\n      walletParams.rootPrivateKey = keyPair.prv;\n    }\n    return walletParams;\n  }\n\n  /**\n   * Explain/parse transaction\n   * @param params\n   */\n  async explainTransaction(params: ExplainTransactionOptions = {}): Promise<TransactionExplanation> {\n    let transaction;\n    let txHex: string = params.txHex || ((params.halfSigned && params.halfSigned.txHex) as string);\n    if (!txHex) {\n      throw new Error('missing required param txHex');\n    }\n    try {\n      transaction = rippleBinaryCodec.decode(txHex);\n    } catch (e) {\n      try {\n        transaction = JSON.parse(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    let id: string;\n    // hashes ids are different for signed and unsigned tx\n    // first we try to get the hash id as if it is signed, will throw if its not\n    try {\n      id = xrpl.hashes.hashSignedTx(txHex);\n    } catch (e) {\n      id = xrpl.hashes.hashTx(txHex);\n    }\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: undefined,\n          size: txHex.length / 2,\n        },\n        accountSet: {\n          messageKey: transaction.MessageKey,\n          setFlag: transaction.SetFlag,\n        },\n      };\n    } else if (transaction.TransactionType === 'TrustSet') {\n      return {\n        displayOrder: [\n          'id',\n          'outputAmount',\n          'changeAmount',\n          'outputs',\n          'changeOutputs',\n          'fee',\n          'account',\n          'limitAmount',\n        ],\n        id: id,\n        changeOutputs: [],\n        outputAmount: 0,\n        changeAmount: 0,\n        outputs: [],\n        fee: {\n          fee: transaction.Fee,\n          feeRate: undefined,\n          size: txHex.length / 2,\n        },\n        account: transaction.Account,\n        limitAmount: {\n          currency: transaction.LimitAmount.currency,\n          issuer: transaction.LimitAmount.issuer,\n          value: transaction.LimitAmount.value,\n        },\n      };\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: undefined,\n        size: txHex.length / 2,\n      },\n    };\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 coinConfig = coins.get(this.getChain()) as XrpCoin;\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 (\n      (txParams.type === undefined || txParams.type === 'payment') &&\n      typeof output.amount !== 'object' &&\n      !comparator(output, expectedOutput)\n    ) {\n      throw new Error('transaction prebuild does not match expected output');\n    }\n\n    if (txParams.type === 'enabletoken') {\n      if (txParams.recipients?.length !== 1) {\n        throw new Error(\n          `${this.getChain()} doesn't support sending to more than 1 destination address within a single transaction. Try again, using only a single recipient.`\n        );\n      }\n      const recipient = txParams.recipients[0];\n      if (!recipient.tokenName) {\n        throw new Error('Recipient must include a token name.');\n      }\n      const recipientCurrency = utils.getXrpCurrencyFromTokenName(recipient.tokenName).currency;\n      if (coinConfig.isToken) {\n        if (recipientCurrency !== coinConfig.currencyCode) {\n          throw new Error('Incorrect token name specified in recipients');\n        }\n      }\n      if (!('account' in explanation) || !('limitAmount' in explanation) || !explanation.limitAmount.currency) {\n        throw new Error('Explanation is missing required keys (account or limitAmount with currency)');\n      }\n      const baseAddress = explanation.account;\n      const currency = explanation.limitAmount.currency;\n\n      if (recipient.address !== baseAddress || recipientCurrency !== currency) {\n        throw new Error('Tx outputs does not match with expected txParams recipients');\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 = utils.getAddressDetails(address);\n    const rootAddressDetails = utils.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 | RecoveryTransaction> {\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          ledger_index: 'current',\n          queue: true,\n          strict: true,\n          signer_lists: true,\n        },\n      ],\n    };\n\n    const accountLinesParams = {\n      method: 'account_lines',\n      params: [\n        {\n          account: params.rootAddress,\n          ledger_index: 'validated',\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, accountLines } = 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      accountLines: this.bitgo.post(rippledUrl).send(accountLinesParams),\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    const ownerCount = new BigNumber(addressDetails.body.result.account_data.OwnerCount);\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 totalReserveDelta = reserveDelta.times(ownerCount);\n    const reserve = baseReserve.plus(totalReserveDelta);\n    const recoverableBalance = balance.minus(reserve);\n\n    const rawDestination = params.recoveryDestination;\n    const destinationDetails = url.parse(rawDestination);\n\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\n    if (recoverableBalance.toNumber() <= 0) {\n      throw new Error(\n        `Quantity of XRP to recover must be greater than 0. Current balance: ${balance.toNumber()}, blockchain reserve: ${reserve.toNumber()}, spendable balance: ${recoverableBalance.toNumber()}`\n      );\n    }\n\n    const issuer = params?.issuerAddress;\n    const currency = params?.currencyCode;\n    if (!!issuer && !!currency) {\n      const tokenParams = {\n        recoveryDestination: params.recoveryDestination,\n        recoverableBalance,\n        currentLedger,\n        openLedgerFee,\n        sequenceId,\n        accountLines,\n        keys,\n        isKrsRecovery,\n        isUnsignedSweep,\n        userAddress,\n        backupAddress,\n        issuer,\n        currency,\n      };\n\n      return this.recoverXrpToken(params, tokenParams);\n    }\n\n    const factory = new TransactionBuilderFactory(coins.get(this.getChain()));\n    const txBuilder = factory.getTransferBuilder() as TransferBuilder;\n    txBuilder\n      .to(params.recoveryDestination as string)\n      .amount(recoverableBalance.toFixed(0))\n      .sender(params.rootAddress)\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 tx = await txBuilder.build();\n    const serializedTx = tx.toBroadcastFormat();\n\n    if (isUnsignedSweep) {\n      return {\n        txHex: serializedTx,\n        coin: this.getChain(),\n      };\n    }\n\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 = ripple.signWithPrivateKey(serializedTx, userKey, { signAs: userAddress });\n\n    let signedTransaction: string;\n\n    if (isKrsRecovery) {\n      signedTransaction = userSignature.signedTransaction;\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 = ripple.signWithPrivateKey(serializedTx, backupKey, { signAs: backupAddress });\n      signedTransaction = ripple.multisign([userSignature.signedTransaction, backupSignature.signedTransaction]);\n    }\n\n    const transactionExplanation: RecoveryInfo = (await this.explainTransaction({\n      txHex: signedTransaction,\n    })) as RecoveryInfo;\n\n    transactionExplanation.txHex = signedTransaction;\n\n    if (isKrsRecovery) {\n      transactionExplanation.backupKey = params.backupKey;\n      transactionExplanation.coin = this.getChain();\n    }\n    return transactionExplanation;\n  }\n\n  public async recoverXrpToken(params, tokenParams) {\n    const { currency, issuer } = tokenParams;\n    const tokenName = (utils.getXrpToken(issuer, currency) as XrpCoin).name;\n    const lines = tokenParams.accountLines.body.result.lines;\n\n    let amount;\n    for (const line of lines) {\n      if (line.currency === currency && line.account === issuer) {\n        amount = line.balance;\n        break;\n      }\n    }\n\n    if (amount === undefined) {\n      throw new Error(`Does not have Trustline with ${issuer}`);\n    }\n    if (amount === '0') {\n      throw new Error(`Does not have funds to recover`);\n    }\n\n    const decimalPlaces = coins.get(tokenName).decimalPlaces;\n    amount = new BigNumber(amount).shiftedBy(decimalPlaces).toFixed();\n\n    const FLAG_VALUE = 2147483648;\n\n    const factory = new TransactionBuilderFactory(coins.get(tokenName));\n    const txBuilder = factory.getTokenTransferBuilder() as TokenTransferBuilder;\n    txBuilder\n      .to(tokenParams.recoveryDestination)\n      .amount(amount)\n      .sender(params.rootAddress)\n      .flags(FLAG_VALUE)\n      .lastLedgerSequence(tokenParams.currentLedger + 1000000) // give it 1 million ledgers' time (~1 month, suitable for KRS)\n      .fee(tokenParams.openLedgerFee.times(3).toFixed(0)) // the factor three is for the multisigning\n      .sequence(tokenParams.sequenceId);\n\n    const tx = await txBuilder.build();\n    const serializedTx = tx.toBroadcastFormat();\n\n    const { keys, isKrsRecovery, isUnsignedSweep, userAddress, backupAddress } = tokenParams;\n\n    if (isUnsignedSweep) {\n      return {\n        txHex: serializedTx,\n        coin: this.getChain(),\n      };\n    }\n\n    if (!keys[0].privateKey) {\n      throw new Error(`userKey is not a private key`);\n    }\n\n    const userKey = keys[0].privateKey.toString('hex');\n    const userSignature = ripple.signWithPrivateKey(serializedTx, userKey, { signAs: userAddress });\n\n    let signedTransaction: string;\n\n    if (isKrsRecovery) {\n      signedTransaction = userSignature.signedTransaction;\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 = ripple.signWithPrivateKey(serializedTx, backupKey, { signAs: backupAddress });\n      signedTransaction = ripple.multisign([userSignature.signedTransaction, backupSignature.signedTransaction]);\n    }\n\n    const transactionExplanation: RecoveryInfo = (await this.explainTransaction({\n      txHex: signedTransaction,\n    })) as RecoveryInfo;\n\n    transactionExplanation.txHex = signedTransaction;\n\n    if (isKrsRecovery) {\n      transactionExplanation.backupKey = params.backupKey;\n      transactionExplanation.coin = this.getChain();\n    }\n    return transactionExplanation;\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    const keyPair = seed ? new XrpKeyPair({ seed }) : new XrpKeyPair();\n    const keys = keyPair.getExtendedKeys();\n    if (!keys.xprv) {\n      throw new Error('Missing prv in key generation.');\n    }\n    return {\n      pub: keys.xpub,\n      prv: keys.xprv,\n    };\n  }\n\n  async parseTransaction(params: ParseTransactionOptions): Promise<ParsedTransaction> {\n    return {};\n  }\n}\n"]}