@bitgo-beta/sdk-coin-xrp 1.3.3-alpha.43 → 1.3.3-alpha.430
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/src/index.d.ts +4 -2
- package/dist/src/index.d.ts.map +1 -1
- package/dist/src/index.js +10 -4
- package/dist/src/lib/accountSetBuilder.d.ts +18 -0
- package/dist/src/lib/accountSetBuilder.d.ts.map +1 -0
- package/dist/src/lib/accountSetBuilder.js +63 -0
- package/dist/src/lib/constants.d.ts +8 -0
- package/dist/src/lib/constants.d.ts.map +1 -0
- package/dist/src/lib/constants.js +30 -0
- package/dist/src/lib/iface.d.ts +109 -0
- package/dist/src/lib/iface.d.ts.map +1 -0
- package/dist/src/lib/iface.js +11 -0
- package/dist/src/lib/index.d.ts +14 -0
- package/dist/src/lib/index.d.ts.map +1 -0
- package/dist/src/lib/index.js +43 -0
- package/dist/src/lib/keyPair.d.ts +33 -0
- package/dist/src/lib/keyPair.d.ts.map +1 -0
- package/dist/src/lib/keyPair.js +118 -0
- package/dist/src/lib/tokenTransferBuilder.d.ts +29 -0
- package/dist/src/lib/tokenTransferBuilder.d.ts.map +1 -0
- package/dist/src/lib/tokenTransferBuilder.js +91 -0
- package/dist/src/lib/transaction.d.ts +62 -0
- package/dist/src/lib/transaction.d.ts.map +1 -0
- package/dist/src/lib/transaction.js +381 -0
- package/dist/src/lib/transactionBuilder.d.ts +72 -0
- package/dist/src/lib/transactionBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder.js +263 -0
- package/dist/src/lib/transactionBuilderFactory.d.ts +39 -0
- package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilderFactory.js +97 -0
- package/dist/src/lib/transferBuilder.d.ts +28 -0
- package/dist/src/lib/transferBuilder.d.ts.map +1 -0
- package/dist/src/lib/transferBuilder.js +82 -0
- package/dist/src/lib/trustsetBuilder.d.ts +21 -0
- package/dist/src/lib/trustsetBuilder.d.ts.map +1 -0
- package/dist/src/lib/trustsetBuilder.js +72 -0
- package/dist/src/lib/utils.d.ts +78 -0
- package/dist/src/lib/utils.d.ts.map +1 -0
- package/dist/src/lib/utils.js +304 -0
- package/dist/src/lib/walletInitializationBuilder.d.ts +19 -0
- package/dist/src/lib/walletInitializationBuilder.d.ts.map +1 -0
- package/dist/src/lib/walletInitializationBuilder.js +76 -0
- package/dist/src/register.d.ts.map +1 -1
- package/dist/src/register.js +5 -1
- package/dist/src/ripple.d.ts +112 -2
- package/dist/src/ripple.d.ts.map +1 -1
- package/dist/src/ripple.js +57 -22
- package/dist/src/txrp.d.ts +3 -2
- package/dist/src/txrp.d.ts.map +1 -1
- package/dist/src/txrp.js +5 -5
- package/dist/src/xrp.d.ts +25 -64
- package/dist/src/xrp.d.ts.map +1 -1
- package/dist/src/xrp.js +375 -159
- package/dist/src/xrpToken.d.ts +22 -0
- package/dist/src/xrpToken.d.ts.map +1 -0
- package/dist/src/xrpToken.js +61 -0
- package/dist/tsconfig.tsbuildinfo +1 -8867
- package/package.json +15 -12
- package/.eslintignore +0 -5
- package/.mocharc.yml +0 -8
- package/CHANGELOG.md +0 -200
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.
|
|
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 (
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
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,60 @@ 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
|
|
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
|
|
37
|
-
const
|
|
51
|
+
const xrpl = __importStar(require("xrpl"));
|
|
52
|
+
const lib_1 = require("./lib");
|
|
53
|
+
const iface_1 = require("./lib/iface");
|
|
54
|
+
const keyPair_1 = require("./lib/keyPair");
|
|
55
|
+
const utils_1 = __importDefault(require("./lib/utils"));
|
|
56
|
+
const ripple_1 = __importDefault(require("./ripple"));
|
|
38
57
|
class Xrp extends sdk_core_1.BaseCoin {
|
|
39
|
-
constructor(bitgo) {
|
|
58
|
+
constructor(bitgo, staticsCoin) {
|
|
40
59
|
super(bitgo);
|
|
60
|
+
if (!staticsCoin) {
|
|
61
|
+
throw new Error('missing required constructor parameter staticsCoin');
|
|
62
|
+
}
|
|
63
|
+
this._staticsCoin = staticsCoin;
|
|
41
64
|
}
|
|
42
|
-
static createInstance(bitgo) {
|
|
43
|
-
return new Xrp(bitgo);
|
|
65
|
+
static createInstance(bitgo, staticsCoin) {
|
|
66
|
+
return new Xrp(bitgo, staticsCoin);
|
|
44
67
|
}
|
|
45
68
|
/**
|
|
46
69
|
* Factor between the coin's base unit and its smallest subdivison
|
|
47
70
|
*/
|
|
48
71
|
getBaseFactor() {
|
|
49
|
-
return
|
|
72
|
+
return Math.pow(10, this._staticsCoin.decimalPlaces);
|
|
50
73
|
}
|
|
51
74
|
/**
|
|
52
75
|
* Identifier for the blockchain which supports this coin
|
|
53
76
|
*/
|
|
54
77
|
getChain() {
|
|
55
|
-
return
|
|
78
|
+
return this._staticsCoin.name;
|
|
56
79
|
}
|
|
57
80
|
/**
|
|
58
81
|
* Identifier for the coin family
|
|
59
82
|
*/
|
|
60
83
|
getFamily() {
|
|
61
|
-
return
|
|
84
|
+
return this._staticsCoin.family;
|
|
62
85
|
}
|
|
63
86
|
/**
|
|
64
87
|
* Complete human-readable name of this coin
|
|
65
88
|
*/
|
|
66
89
|
getFullName() {
|
|
67
|
-
return
|
|
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;
|
|
90
|
+
return this._staticsCoin.fullName;
|
|
120
91
|
}
|
|
121
92
|
/**
|
|
122
93
|
* Evaluates whether an address string is valid for this coin
|
|
123
94
|
* @param address
|
|
124
95
|
*/
|
|
125
96
|
isValidAddress(address) {
|
|
126
|
-
|
|
127
|
-
const addressDetails = this.getAddressDetails(address);
|
|
128
|
-
return address === this.normalizeAddress(addressDetails);
|
|
129
|
-
}
|
|
130
|
-
catch (e) {
|
|
131
|
-
return false;
|
|
132
|
-
}
|
|
97
|
+
return utils_1.default.isValidAddress(address);
|
|
133
98
|
}
|
|
134
99
|
/**
|
|
135
100
|
* Return boolean indicating whether input is valid public key for the coin.
|
|
@@ -138,12 +103,7 @@ class Xrp extends sdk_core_1.BaseCoin {
|
|
|
138
103
|
* @returns {Boolean} is it valid?
|
|
139
104
|
*/
|
|
140
105
|
isValidPub(pub) {
|
|
141
|
-
|
|
142
|
-
return utxo_lib_1.bip32.fromBase58(pub).isNeutered();
|
|
143
|
-
}
|
|
144
|
-
catch (e) {
|
|
145
|
-
return false;
|
|
146
|
-
}
|
|
106
|
+
return utils_1.default.isValidPublicKey(pub);
|
|
147
107
|
}
|
|
148
108
|
/**
|
|
149
109
|
* Get fee info from server
|
|
@@ -151,6 +111,20 @@ class Xrp extends sdk_core_1.BaseCoin {
|
|
|
151
111
|
async getFeeInfo() {
|
|
152
112
|
return this.bitgo.get(this.url('/public/feeinfo')).result();
|
|
153
113
|
}
|
|
114
|
+
/** @inheritdoc */
|
|
115
|
+
valuelessTransferAllowed() {
|
|
116
|
+
return true;
|
|
117
|
+
}
|
|
118
|
+
/** inherited doc */
|
|
119
|
+
getDefaultMultisigType() {
|
|
120
|
+
return sdk_core_1.multisigTypes.onchain;
|
|
121
|
+
}
|
|
122
|
+
getTokenEnablementConfig() {
|
|
123
|
+
return {
|
|
124
|
+
requiresTokenEnablement: true,
|
|
125
|
+
supportsMultipleTokenEnablements: false,
|
|
126
|
+
};
|
|
127
|
+
}
|
|
154
128
|
/**
|
|
155
129
|
* Assemble keychain and half-sign prebuilt transaction
|
|
156
130
|
* @param params
|
|
@@ -158,7 +132,7 @@ class Xrp extends sdk_core_1.BaseCoin {
|
|
|
158
132
|
* - prv
|
|
159
133
|
* @returns Bluebird<HalfSignedTransaction>
|
|
160
134
|
*/
|
|
161
|
-
async signTransaction({ txPrebuild, prv }) {
|
|
135
|
+
async signTransaction({ txPrebuild, prv, isLastSignature, }) {
|
|
162
136
|
if (_.isUndefined(txPrebuild) || !_.isObject(txPrebuild)) {
|
|
163
137
|
if (!_.isUndefined(txPrebuild) && !_.isObject(txPrebuild)) {
|
|
164
138
|
throw new Error(`txPrebuild must be an object, got type ${typeof txPrebuild}`);
|
|
@@ -171,17 +145,21 @@ class Xrp extends sdk_core_1.BaseCoin {
|
|
|
171
145
|
}
|
|
172
146
|
throw new Error('missing prv parameter to sign transaction');
|
|
173
147
|
}
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
if (!userPrivateKey) {
|
|
177
|
-
throw new Error(`no privateKey`);
|
|
148
|
+
if (!txPrebuild.txHex) {
|
|
149
|
+
throw new Error(`missing txHex in txPrebuild`);
|
|
178
150
|
}
|
|
179
|
-
const
|
|
180
|
-
const
|
|
181
|
-
const
|
|
182
|
-
|
|
151
|
+
const keyPair = new keyPair_1.KeyPair({ prv });
|
|
152
|
+
const address = keyPair.getAddress();
|
|
153
|
+
const privateKey = keyPair.getPrivateKey().toString('hex');
|
|
154
|
+
const tx = ripple_1.default.signWithPrivateKey(txPrebuild.txHex, privateKey, {
|
|
155
|
+
signAs: address,
|
|
183
156
|
});
|
|
184
|
-
|
|
157
|
+
// Normally the SDK provides the first signature for an XRP tx, but occasionally it provides the final one as well
|
|
158
|
+
// (recoveries)
|
|
159
|
+
if (isLastSignature) {
|
|
160
|
+
return { txHex: tx.signedTransaction };
|
|
161
|
+
}
|
|
162
|
+
return { halfSigned: { txHex: tx.signedTransaction } };
|
|
185
163
|
}
|
|
186
164
|
/**
|
|
187
165
|
* Ripple requires additional parameters for wallet generation to be sent to the server. The additional parameters are
|
|
@@ -196,11 +174,11 @@ class Xrp extends sdk_core_1.BaseCoin {
|
|
|
196
174
|
}
|
|
197
175
|
}
|
|
198
176
|
else {
|
|
199
|
-
const keyPair =
|
|
200
|
-
if (!keyPair.
|
|
177
|
+
const keyPair = new keyPair_1.KeyPair().getKeys();
|
|
178
|
+
if (!keyPair.prv) {
|
|
201
179
|
throw new Error('no privateKey');
|
|
202
180
|
}
|
|
203
|
-
walletParams.rootPrivateKey = keyPair.
|
|
181
|
+
walletParams.rootPrivateKey = keyPair.prv;
|
|
204
182
|
}
|
|
205
183
|
return walletParams;
|
|
206
184
|
}
|
|
@@ -209,26 +187,33 @@ class Xrp extends sdk_core_1.BaseCoin {
|
|
|
209
187
|
* @param params
|
|
210
188
|
*/
|
|
211
189
|
async explainTransaction(params = {}) {
|
|
212
|
-
|
|
190
|
+
let transaction;
|
|
191
|
+
let txHex = params.txHex || (params.halfSigned && params.halfSigned.txHex);
|
|
192
|
+
if (!txHex) {
|
|
213
193
|
throw new Error('missing required param txHex');
|
|
214
194
|
}
|
|
215
|
-
let transaction;
|
|
216
|
-
let txHex;
|
|
217
195
|
try {
|
|
218
|
-
transaction = rippleBinaryCodec.decode(
|
|
219
|
-
txHex = params.txHex;
|
|
196
|
+
transaction = rippleBinaryCodec.decode(txHex);
|
|
220
197
|
}
|
|
221
198
|
catch (e) {
|
|
222
199
|
try {
|
|
223
|
-
transaction = JSON.parse(
|
|
200
|
+
transaction = JSON.parse(txHex);
|
|
224
201
|
txHex = rippleBinaryCodec.encode(transaction);
|
|
225
202
|
}
|
|
226
203
|
catch (e) {
|
|
227
204
|
throw new Error('txHex needs to be either hex or JSON string for XRP');
|
|
228
205
|
}
|
|
229
206
|
}
|
|
230
|
-
|
|
231
|
-
|
|
207
|
+
let id;
|
|
208
|
+
// hashes ids are different for signed and unsigned tx
|
|
209
|
+
// first we try to get the hash id as if it is signed, will throw if its not
|
|
210
|
+
try {
|
|
211
|
+
id = xrpl.hashes.hashSignedTx(txHex);
|
|
212
|
+
}
|
|
213
|
+
catch (e) {
|
|
214
|
+
id = xrpl.hashes.hashTx(txHex);
|
|
215
|
+
}
|
|
216
|
+
if (transaction.TransactionType === 'AccountSet') {
|
|
232
217
|
return {
|
|
233
218
|
displayOrder: ['id', 'outputAmount', 'changeAmount', 'outputs', 'changeOutputs', 'fee', 'accountSet'],
|
|
234
219
|
id: id,
|
|
@@ -238,11 +223,42 @@ class Xrp extends sdk_core_1.BaseCoin {
|
|
|
238
223
|
outputs: [],
|
|
239
224
|
fee: {
|
|
240
225
|
fee: transaction.Fee,
|
|
241
|
-
feeRate:
|
|
226
|
+
feeRate: undefined,
|
|
242
227
|
size: txHex.length / 2,
|
|
243
228
|
},
|
|
244
229
|
accountSet: {
|
|
245
230
|
messageKey: transaction.MessageKey,
|
|
231
|
+
setFlag: transaction.SetFlag,
|
|
232
|
+
},
|
|
233
|
+
};
|
|
234
|
+
}
|
|
235
|
+
else if (transaction.TransactionType === 'TrustSet') {
|
|
236
|
+
return {
|
|
237
|
+
displayOrder: [
|
|
238
|
+
'id',
|
|
239
|
+
'outputAmount',
|
|
240
|
+
'changeAmount',
|
|
241
|
+
'outputs',
|
|
242
|
+
'changeOutputs',
|
|
243
|
+
'fee',
|
|
244
|
+
'account',
|
|
245
|
+
'limitAmount',
|
|
246
|
+
],
|
|
247
|
+
id: id,
|
|
248
|
+
changeOutputs: [],
|
|
249
|
+
outputAmount: 0,
|
|
250
|
+
changeAmount: 0,
|
|
251
|
+
outputs: [],
|
|
252
|
+
fee: {
|
|
253
|
+
fee: transaction.Fee,
|
|
254
|
+
feeRate: undefined,
|
|
255
|
+
size: txHex.length / 2,
|
|
256
|
+
},
|
|
257
|
+
account: transaction.Account,
|
|
258
|
+
limitAmount: {
|
|
259
|
+
currency: transaction.LimitAmount.currency,
|
|
260
|
+
issuer: transaction.LimitAmount.issuer,
|
|
261
|
+
value: transaction.LimitAmount.value,
|
|
246
262
|
},
|
|
247
263
|
};
|
|
248
264
|
}
|
|
@@ -261,11 +277,87 @@ class Xrp extends sdk_core_1.BaseCoin {
|
|
|
261
277
|
],
|
|
262
278
|
fee: {
|
|
263
279
|
fee: transaction.Fee,
|
|
264
|
-
feeRate:
|
|
280
|
+
feeRate: undefined,
|
|
265
281
|
size: txHex.length / 2,
|
|
266
282
|
},
|
|
267
283
|
};
|
|
268
284
|
}
|
|
285
|
+
getTransactionTypeRawTxHex(txHex) {
|
|
286
|
+
let transaction;
|
|
287
|
+
if (!txHex) {
|
|
288
|
+
throw new Error('missing required param txHex');
|
|
289
|
+
}
|
|
290
|
+
try {
|
|
291
|
+
transaction = rippleBinaryCodec.decode(txHex);
|
|
292
|
+
}
|
|
293
|
+
catch (e) {
|
|
294
|
+
try {
|
|
295
|
+
transaction = JSON.parse(txHex);
|
|
296
|
+
}
|
|
297
|
+
catch (e) {
|
|
298
|
+
throw new Error('txHex needs to be either hex or JSON string for XRP');
|
|
299
|
+
}
|
|
300
|
+
}
|
|
301
|
+
return transaction.TransactionType;
|
|
302
|
+
}
|
|
303
|
+
verifyTxType(txPrebuildDecoded, txHexPrebuild) {
|
|
304
|
+
if (!txHexPrebuild)
|
|
305
|
+
throw new Error('Missing txHexPrebuild to verify token type for enabletoken tx');
|
|
306
|
+
const transactionType = this.getTransactionTypeRawTxHex(txHexPrebuild);
|
|
307
|
+
if (transactionType === undefined)
|
|
308
|
+
throw new Error('Missing TransactionType on token enablement tx');
|
|
309
|
+
if (transactionType !== iface_1.XrpTransactionType.TrustSet)
|
|
310
|
+
throw new Error(`tx type ${transactionType} does not match expected type TrustSet`);
|
|
311
|
+
// decoded payload type could come as undefined or any of the enabletoken like types but never as something else like Send, etc
|
|
312
|
+
const actualTypeFromDecoded = 'type' in txPrebuildDecoded && typeof txPrebuildDecoded.type === 'string' ? txPrebuildDecoded.type : undefined;
|
|
313
|
+
if (!actualTypeFromDecoded ||
|
|
314
|
+
actualTypeFromDecoded === 'enabletoken' ||
|
|
315
|
+
actualTypeFromDecoded === 'AssociatedTokenAccountInitialization')
|
|
316
|
+
return;
|
|
317
|
+
throw new Error(`tx type ${actualTypeFromDecoded} does not match the expected type enabletoken`);
|
|
318
|
+
}
|
|
319
|
+
verifyTokenName(txParams, txPrebuildDecoded, txHexPrebuild, coinConfig) {
|
|
320
|
+
if (!txHexPrebuild)
|
|
321
|
+
throw new Error('Missing txHexPrebuild param required for token enablement.');
|
|
322
|
+
if (!txParams.recipients || txParams.recipients.length === 0)
|
|
323
|
+
throw new Error('Missing recipients param for token enablement.');
|
|
324
|
+
const fullTokenName = txParams.recipients[0].tokenName;
|
|
325
|
+
if (fullTokenName === undefined)
|
|
326
|
+
throw new Error('Param tokenName is required for token enablement. Recipient must include a token name.');
|
|
327
|
+
if (!('limitAmount' in txPrebuildDecoded))
|
|
328
|
+
throw new Error('Missing limitAmount param for token enablement.');
|
|
329
|
+
// we check currency on both the txHex but also the explained payload
|
|
330
|
+
const expectedCurrency = utils_1.default.getXrpCurrencyFromTokenName(fullTokenName).currency;
|
|
331
|
+
if (coinConfig.isToken && expectedCurrency !== txPrebuildDecoded.limitAmount.currency)
|
|
332
|
+
throw new Error('Invalid token issuer or currency on token enablement tx');
|
|
333
|
+
}
|
|
334
|
+
verifyActivationAddress(txParams, txPrebuildDecoded) {
|
|
335
|
+
if (txParams.recipients === undefined || txParams.recipients.length === 0)
|
|
336
|
+
throw new Error('Missing recipients param for token enablement.');
|
|
337
|
+
if (txParams.recipients?.length !== 1) {
|
|
338
|
+
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.`);
|
|
339
|
+
}
|
|
340
|
+
if (!('account' in txPrebuildDecoded))
|
|
341
|
+
throw new Error('missing account on token enablement tx');
|
|
342
|
+
const activationAddress = txParams.recipients[0].address;
|
|
343
|
+
const accountAddress = txPrebuildDecoded.account;
|
|
344
|
+
if (activationAddress !== accountAddress)
|
|
345
|
+
throw new Error("Account address doesn't match with activation address.");
|
|
346
|
+
}
|
|
347
|
+
verifyTokenIssuer(txParams, txPrebuildDecoded) {
|
|
348
|
+
if (txPrebuildDecoded === undefined || !('limitAmount' in txPrebuildDecoded))
|
|
349
|
+
throw new Error('missing token issuer on token enablement tx');
|
|
350
|
+
const { issuer, currency } = txPrebuildDecoded.limitAmount;
|
|
351
|
+
if (!utils_1.default.getXrpToken(issuer, currency))
|
|
352
|
+
throw new Error('Invalid token issuer or currency on token enablement tx');
|
|
353
|
+
}
|
|
354
|
+
verifyRequiredKeys(txParams, txPrebuildDecoded) {
|
|
355
|
+
if (!('account' in txPrebuildDecoded) ||
|
|
356
|
+
!('limitAmount' in txPrebuildDecoded) ||
|
|
357
|
+
!txPrebuildDecoded.limitAmount.currency) {
|
|
358
|
+
throw new Error('Explanation is missing required keys (account or limitAmount with currency)');
|
|
359
|
+
}
|
|
360
|
+
}
|
|
269
361
|
/**
|
|
270
362
|
* Verify that a transaction prebuild complies with the original intention
|
|
271
363
|
* @param txParams params object passed to send
|
|
@@ -273,21 +365,33 @@ class Xrp extends sdk_core_1.BaseCoin {
|
|
|
273
365
|
* @param wallet
|
|
274
366
|
* @returns {boolean}
|
|
275
367
|
*/
|
|
276
|
-
async verifyTransaction({ txParams, txPrebuild }) {
|
|
368
|
+
async verifyTransaction({ txParams, txPrebuild, verification }) {
|
|
369
|
+
const coinConfig = statics_1.coins.get(this.getChain());
|
|
277
370
|
const explanation = await this.explainTransaction({
|
|
278
371
|
txHex: txPrebuild.txHex,
|
|
279
372
|
});
|
|
373
|
+
// Explaining a tx strips out certain data, for extra measurement we're checking vs the explained tx
|
|
374
|
+
// but also vs the tx pre explained.
|
|
375
|
+
if (txParams.type === 'enabletoken' && verification?.verifyTokenEnablement) {
|
|
376
|
+
this.verifyTxType(explanation, txPrebuild.txHex);
|
|
377
|
+
this.verifyActivationAddress(txParams, explanation);
|
|
378
|
+
this.verifyTokenIssuer(txParams, explanation);
|
|
379
|
+
this.verifyTokenName(txParams, explanation, txPrebuild.txHex, coinConfig);
|
|
380
|
+
this.verifyRequiredKeys(txParams, explanation);
|
|
381
|
+
}
|
|
280
382
|
const output = [...explanation.outputs, ...explanation.changeOutputs][0];
|
|
281
383
|
const expectedOutput = txParams.recipients && txParams.recipients[0];
|
|
282
384
|
const comparator = (recipient1, recipient2) => {
|
|
283
|
-
if (recipient1.address !== recipient2.address) {
|
|
385
|
+
if (utils_1.default.getAddressDetails(recipient1.address).address !== utils_1.default.getAddressDetails(recipient2.address).address) {
|
|
284
386
|
return false;
|
|
285
387
|
}
|
|
286
388
|
const amount1 = new bignumber_js_1.BigNumber(recipient1.amount);
|
|
287
389
|
const amount2 = new bignumber_js_1.BigNumber(recipient2.amount);
|
|
288
390
|
return amount1.toFixed() === amount2.toFixed();
|
|
289
391
|
};
|
|
290
|
-
if (
|
|
392
|
+
if ((txParams.type === undefined || txParams.type === 'payment') &&
|
|
393
|
+
typeof output.amount !== 'object' &&
|
|
394
|
+
!comparator(output, expectedOutput)) {
|
|
291
395
|
throw new Error('transaction prebuild does not match expected output');
|
|
292
396
|
}
|
|
293
397
|
return true;
|
|
@@ -303,8 +407,28 @@ class Xrp extends sdk_core_1.BaseCoin {
|
|
|
303
407
|
if (!this.isValidAddress(address)) {
|
|
304
408
|
throw new sdk_core_1.InvalidAddressError(`address verification failure: address "${address}" is not valid`);
|
|
305
409
|
}
|
|
306
|
-
const
|
|
307
|
-
|
|
410
|
+
const accountInfoParams = {
|
|
411
|
+
method: 'account_info',
|
|
412
|
+
params: [
|
|
413
|
+
{
|
|
414
|
+
account: address,
|
|
415
|
+
ledger_index: 'current',
|
|
416
|
+
queue: true,
|
|
417
|
+
strict: true,
|
|
418
|
+
signer_lists: true,
|
|
419
|
+
},
|
|
420
|
+
],
|
|
421
|
+
};
|
|
422
|
+
const accountInfo = (await this.bitgo.post(this.getRippledUrl()).send(accountInfoParams)).body;
|
|
423
|
+
if (accountInfo?.result?.account_data?.Flags == null) {
|
|
424
|
+
throw new Error('Invalid account information: Flags field is missing.');
|
|
425
|
+
}
|
|
426
|
+
const flags = xrpl.parseAccountRootFlags(accountInfo.result.account_data.Flags);
|
|
427
|
+
const addressDetails = utils_1.default.getAddressDetails(address);
|
|
428
|
+
const rootAddressDetails = utils_1.default.getAddressDetails(rootAddress);
|
|
429
|
+
if (flags.lsfRequireDestTag && addressDetails.destinationTag == null) {
|
|
430
|
+
throw new sdk_core_1.InvalidAddressError(`Invalid Address: Destination Tag is required for address "${address}".`);
|
|
431
|
+
}
|
|
308
432
|
if (addressDetails.address !== rootAddressDetails.address) {
|
|
309
433
|
throw new sdk_core_1.UnexpectedAddressError(`address validation failure: ${addressDetails.address} vs. ${rootAddressDetails.address}`);
|
|
310
434
|
}
|
|
@@ -336,25 +460,35 @@ class Xrp extends sdk_core_1.BaseCoin {
|
|
|
336
460
|
params: [
|
|
337
461
|
{
|
|
338
462
|
account: params.rootAddress,
|
|
339
|
-
strict: true,
|
|
340
463
|
ledger_index: 'current',
|
|
341
464
|
queue: true,
|
|
465
|
+
strict: true,
|
|
342
466
|
signer_lists: true,
|
|
343
467
|
},
|
|
344
468
|
],
|
|
345
469
|
};
|
|
470
|
+
const accountLinesParams = {
|
|
471
|
+
method: 'account_lines',
|
|
472
|
+
params: [
|
|
473
|
+
{
|
|
474
|
+
account: params.rootAddress,
|
|
475
|
+
ledger_index: 'validated',
|
|
476
|
+
},
|
|
477
|
+
],
|
|
478
|
+
};
|
|
346
479
|
if (isKrsRecovery) {
|
|
347
|
-
sdk_core_1.checkKrsProvider(this, params.krsProvider);
|
|
480
|
+
(0, sdk_core_1.checkKrsProvider)(this, params.krsProvider);
|
|
348
481
|
}
|
|
349
482
|
// Validate the destination address
|
|
350
483
|
if (!this.isValidAddress(params.recoveryDestination)) {
|
|
351
484
|
throw new Error('Invalid destination address!');
|
|
352
485
|
}
|
|
353
|
-
const keys = sdk_core_1.getBip32Keys(this.bitgo, params, { requireBitGoXpub: false });
|
|
354
|
-
const { addressDetails, feeDetails, serverDetails } = await sdk_core_1.promiseProps({
|
|
486
|
+
const keys = (0, sdk_core_1.getBip32Keys)(this.bitgo, params, { requireBitGoXpub: false });
|
|
487
|
+
const { addressDetails, feeDetails, serverDetails, accountLines } = await (0, sdk_core_1.promiseProps)({
|
|
355
488
|
addressDetails: this.bitgo.post(rippledUrl).send(accountInfoParams),
|
|
356
489
|
feeDetails: this.bitgo.post(rippledUrl).send({ method: 'fee' }),
|
|
357
490
|
serverDetails: this.bitgo.post(rippledUrl).send({ method: 'server_info' }),
|
|
491
|
+
accountLines: this.bitgo.post(rippledUrl).send(accountLinesParams),
|
|
358
492
|
});
|
|
359
493
|
const openLedgerFee = new bignumber_js_1.BigNumber(feeDetails.body.result.drops.open_ledger_fee);
|
|
360
494
|
const baseReserve = new bignumber_js_1.BigNumber(serverDetails.body.result.info.validated_ledger.reserve_base_xrp).times(this.getBaseFactor());
|
|
@@ -364,6 +498,7 @@ class Xrp extends sdk_core_1.BaseCoin {
|
|
|
364
498
|
const balance = new bignumber_js_1.BigNumber(addressDetails.body.result.account_data.Balance);
|
|
365
499
|
const signerLists = addressDetails.body.result.account_data.signer_lists;
|
|
366
500
|
const accountFlags = addressDetails.body.result.account_data.Flags;
|
|
501
|
+
const ownerCount = new bignumber_js_1.BigNumber(addressDetails.body.result.account_data.OwnerCount);
|
|
367
502
|
// make sure there is only one signer list set
|
|
368
503
|
if (signerLists.length !== 1) {
|
|
369
504
|
throw new Error('unexpected set of signer lists');
|
|
@@ -412,93 +547,174 @@ class Xrp extends sdk_core_1.BaseCoin {
|
|
|
412
547
|
throw new Error('the destination flag requirement has not been activated');
|
|
413
548
|
}
|
|
414
549
|
// recover the funds
|
|
415
|
-
const
|
|
550
|
+
const totalReserveDelta = reserveDelta.times(ownerCount);
|
|
551
|
+
const reserve = baseReserve.plus(totalReserveDelta);
|
|
416
552
|
const recoverableBalance = balance.minus(reserve);
|
|
417
553
|
const rawDestination = params.recoveryDestination;
|
|
418
554
|
const destinationDetails = url.parse(rawDestination);
|
|
419
|
-
const destinationAddress = destinationDetails.pathname;
|
|
420
|
-
// parse destination tag from query
|
|
421
|
-
let destinationTag;
|
|
422
555
|
if (destinationDetails.query) {
|
|
423
556
|
const queryDetails = querystring.parse(destinationDetails.query);
|
|
424
557
|
if (Array.isArray(queryDetails.dt)) {
|
|
425
558
|
// if queryDetails.dt is an array, that means dt was given multiple times, which is not valid
|
|
426
559
|
throw new sdk_core_1.InvalidAddressError(`destination tag can appear at most once, but ${queryDetails.dt.length} destination tags were found`);
|
|
427
560
|
}
|
|
428
|
-
const parsedTag = parseInt(queryDetails.dt, 10);
|
|
429
|
-
if (Number.isInteger(parsedTag)) {
|
|
430
|
-
destinationTag = parsedTag;
|
|
431
|
-
}
|
|
432
561
|
}
|
|
433
|
-
|
|
434
|
-
|
|
435
|
-
|
|
436
|
-
|
|
437
|
-
|
|
438
|
-
|
|
439
|
-
|
|
440
|
-
|
|
441
|
-
|
|
442
|
-
|
|
443
|
-
|
|
444
|
-
|
|
562
|
+
if (recoverableBalance.toNumber() <= 0) {
|
|
563
|
+
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()}`);
|
|
564
|
+
}
|
|
565
|
+
const issuer = params?.issuerAddress;
|
|
566
|
+
const currency = params?.currencyCode;
|
|
567
|
+
if (!!issuer && !!currency) {
|
|
568
|
+
const tokenParams = {
|
|
569
|
+
recoveryDestination: params.recoveryDestination,
|
|
570
|
+
recoverableBalance,
|
|
571
|
+
currentLedger,
|
|
572
|
+
openLedgerFee,
|
|
573
|
+
sequenceId,
|
|
574
|
+
accountLines,
|
|
575
|
+
keys,
|
|
576
|
+
isKrsRecovery,
|
|
577
|
+
isUnsignedSweep,
|
|
578
|
+
userAddress,
|
|
579
|
+
backupAddress,
|
|
580
|
+
issuer,
|
|
581
|
+
currency,
|
|
582
|
+
};
|
|
583
|
+
return this.recoverXrpToken(params, tokenParams);
|
|
584
|
+
}
|
|
585
|
+
const factory = new lib_1.TransactionBuilderFactory(statics_1.coins.get(this.getChain()));
|
|
586
|
+
const txBuilder = factory.getTransferBuilder();
|
|
587
|
+
txBuilder
|
|
588
|
+
.to(params.recoveryDestination)
|
|
589
|
+
.amount(recoverableBalance.toFixed(0))
|
|
590
|
+
.sender(params.rootAddress)
|
|
591
|
+
.flags(2147483648)
|
|
592
|
+
.lastLedgerSequence(currentLedger + 1000000) // give it 1 million ledgers' time (~1 month, suitable for KRS)
|
|
593
|
+
.fee(openLedgerFee.times(3).toFixed(0)) // the factor three is for the multisigning
|
|
594
|
+
.sequence(sequenceId);
|
|
595
|
+
const tx = await txBuilder.build();
|
|
596
|
+
const serializedTx = tx.toBroadcastFormat();
|
|
445
597
|
if (isUnsignedSweep) {
|
|
446
598
|
return {
|
|
447
|
-
txHex:
|
|
599
|
+
txHex: serializedTx,
|
|
600
|
+
coin: this.getChain(),
|
|
448
601
|
};
|
|
449
602
|
}
|
|
450
|
-
const rippleLib = ripple();
|
|
451
603
|
if (!keys[0].privateKey) {
|
|
452
604
|
throw new Error(`userKey is not a private key`);
|
|
453
605
|
}
|
|
454
606
|
const userKey = keys[0].privateKey.toString('hex');
|
|
455
|
-
const userSignature =
|
|
607
|
+
const userSignature = ripple_1.default.signWithPrivateKey(serializedTx, userKey, { signAs: userAddress });
|
|
456
608
|
let signedTransaction;
|
|
457
609
|
if (isKrsRecovery) {
|
|
458
|
-
signedTransaction = userSignature;
|
|
610
|
+
signedTransaction = userSignature.signedTransaction;
|
|
459
611
|
}
|
|
460
612
|
else {
|
|
461
613
|
if (!keys[1].privateKey) {
|
|
462
614
|
throw new Error(`backupKey is not a private key`);
|
|
463
615
|
}
|
|
464
616
|
const backupKey = keys[1].privateKey.toString('hex');
|
|
465
|
-
const backupSignature =
|
|
466
|
-
signedTransaction =
|
|
617
|
+
const backupSignature = ripple_1.default.signWithPrivateKey(serializedTx, backupKey, { signAs: backupAddress });
|
|
618
|
+
signedTransaction = ripple_1.default.multisign([userSignature.signedTransaction, backupSignature.signedTransaction]);
|
|
467
619
|
}
|
|
468
620
|
const transactionExplanation = (await this.explainTransaction({
|
|
469
|
-
txHex: signedTransaction
|
|
621
|
+
txHex: signedTransaction,
|
|
470
622
|
}));
|
|
471
|
-
transactionExplanation.txHex = signedTransaction
|
|
623
|
+
transactionExplanation.txHex = signedTransaction;
|
|
472
624
|
if (isKrsRecovery) {
|
|
473
625
|
transactionExplanation.backupKey = params.backupKey;
|
|
474
626
|
transactionExplanation.coin = this.getChain();
|
|
475
627
|
}
|
|
476
628
|
return transactionExplanation;
|
|
477
629
|
}
|
|
478
|
-
|
|
479
|
-
|
|
630
|
+
async recoverXrpToken(params, tokenParams) {
|
|
631
|
+
const { currency, issuer } = tokenParams;
|
|
632
|
+
const tokenName = utils_1.default.getXrpToken(issuer, currency).name;
|
|
633
|
+
const lines = tokenParams.accountLines.body.result.lines;
|
|
634
|
+
let amount;
|
|
635
|
+
for (const line of lines) {
|
|
636
|
+
if (line.currency === currency && line.account === issuer) {
|
|
637
|
+
amount = line.balance;
|
|
638
|
+
break;
|
|
639
|
+
}
|
|
640
|
+
}
|
|
641
|
+
if (amount === undefined) {
|
|
642
|
+
throw new Error(`Does not have Trustline with ${issuer}`);
|
|
643
|
+
}
|
|
644
|
+
if (amount === '0') {
|
|
645
|
+
throw new Error(`Does not have funds to recover`);
|
|
646
|
+
}
|
|
647
|
+
const decimalPlaces = statics_1.coins.get(tokenName).decimalPlaces;
|
|
648
|
+
amount = new bignumber_js_1.BigNumber(amount).shiftedBy(decimalPlaces).toFixed();
|
|
649
|
+
const FLAG_VALUE = 2147483648;
|
|
650
|
+
const factory = new lib_1.TransactionBuilderFactory(statics_1.coins.get(tokenName));
|
|
651
|
+
const txBuilder = factory.getTokenTransferBuilder();
|
|
652
|
+
txBuilder
|
|
653
|
+
.to(tokenParams.recoveryDestination)
|
|
654
|
+
.amount(amount)
|
|
655
|
+
.sender(params.rootAddress)
|
|
656
|
+
.flags(FLAG_VALUE)
|
|
657
|
+
.lastLedgerSequence(tokenParams.currentLedger + 1000000) // give it 1 million ledgers' time (~1 month, suitable for KRS)
|
|
658
|
+
.fee(tokenParams.openLedgerFee.times(3).toFixed(0)) // the factor three is for the multisigning
|
|
659
|
+
.sequence(tokenParams.sequenceId);
|
|
660
|
+
const tx = await txBuilder.build();
|
|
661
|
+
const serializedTx = tx.toBroadcastFormat();
|
|
662
|
+
const { keys, isKrsRecovery, isUnsignedSweep, userAddress, backupAddress } = tokenParams;
|
|
663
|
+
if (isUnsignedSweep) {
|
|
664
|
+
return {
|
|
665
|
+
txHex: serializedTx,
|
|
666
|
+
coin: this.getChain(),
|
|
667
|
+
};
|
|
668
|
+
}
|
|
669
|
+
if (!keys[0].privateKey) {
|
|
670
|
+
throw new Error(`userKey is not a private key`);
|
|
671
|
+
}
|
|
672
|
+
const userKey = keys[0].privateKey.toString('hex');
|
|
673
|
+
const userSignature = ripple_1.default.signWithPrivateKey(serializedTx, userKey, { signAs: userAddress });
|
|
674
|
+
let signedTransaction;
|
|
675
|
+
if (isKrsRecovery) {
|
|
676
|
+
signedTransaction = userSignature.signedTransaction;
|
|
677
|
+
}
|
|
678
|
+
else {
|
|
679
|
+
if (!keys[1].privateKey) {
|
|
680
|
+
throw new Error(`backupKey is not a private key`);
|
|
681
|
+
}
|
|
682
|
+
const backupKey = keys[1].privateKey.toString('hex');
|
|
683
|
+
const backupSignature = ripple_1.default.signWithPrivateKey(serializedTx, backupKey, { signAs: backupAddress });
|
|
684
|
+
signedTransaction = ripple_1.default.multisign([userSignature.signedTransaction, backupSignature.signedTransaction]);
|
|
685
|
+
}
|
|
686
|
+
const transactionExplanation = (await this.explainTransaction({
|
|
687
|
+
txHex: signedTransaction,
|
|
688
|
+
}));
|
|
689
|
+
transactionExplanation.txHex = signedTransaction;
|
|
690
|
+
if (isKrsRecovery) {
|
|
691
|
+
transactionExplanation.backupKey = params.backupKey;
|
|
692
|
+
transactionExplanation.coin = this.getChain();
|
|
693
|
+
}
|
|
694
|
+
return transactionExplanation;
|
|
480
695
|
}
|
|
481
696
|
/**
|
|
482
697
|
* Generate a new keypair for this coin.
|
|
483
698
|
* @param seed Seed from which the new keypair should be generated, otherwise a random seed is used
|
|
484
699
|
*/
|
|
485
700
|
generateKeyPair(seed) {
|
|
486
|
-
|
|
487
|
-
|
|
488
|
-
|
|
489
|
-
|
|
490
|
-
|
|
491
|
-
}
|
|
492
|
-
const extendedKey = utxo_lib_1.bip32.fromSeed(seed);
|
|
493
|
-
const xpub = extendedKey.neutered().toBase58();
|
|
701
|
+
const keyPair = seed ? new keyPair_1.KeyPair({ seed }) : new keyPair_1.KeyPair();
|
|
702
|
+
const keys = keyPair.getExtendedKeys();
|
|
703
|
+
if (!keys.xprv) {
|
|
704
|
+
throw new Error('Missing prv in key generation.');
|
|
705
|
+
}
|
|
494
706
|
return {
|
|
495
|
-
pub: xpub,
|
|
496
|
-
prv:
|
|
707
|
+
pub: keys.xpub,
|
|
708
|
+
prv: keys.xprv,
|
|
497
709
|
};
|
|
498
710
|
}
|
|
499
711
|
async parseTransaction(params) {
|
|
500
712
|
return {};
|
|
501
713
|
}
|
|
714
|
+
/** @inheritDoc */
|
|
715
|
+
auditDecryptedKey(params) {
|
|
716
|
+
throw new sdk_core_1.MethodNotImplementedError();
|
|
717
|
+
}
|
|
502
718
|
}
|
|
503
719
|
exports.Xrp = Xrp;
|
|
504
|
-
//# 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 RecoveryTransaction {\n  txHex: 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 | 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          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 {\n        txHex: txJSON,\n      };\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"]}
|
|
720
|
+
//# 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  AuditDecryptedKeyParams,\n  BaseCoin,\n  BitGoBase,\n  checkKrsProvider,\n  getBip32Keys,\n  InvalidAddressError,\n  KeyPair,\n  MethodNotImplementedError,\n  MultisigType,\n  multisigTypes,\n  ParsedTransaction,\n  ParseTransactionOptions,\n  promiseProps,\n  TokenEnablementConfig,\n  TransactionParams,\n  UnexpectedAddressError,\n  VerifyTransactionOptions,\n} from '@bitgo-beta/sdk-core';\nimport { coins, BaseCoin as StaticsBaseCoin, 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 { TokenTransferBuilder, TransactionBuilderFactory, TransferBuilder } from './lib';\nimport {\n  ExplainTransactionOptions,\n  FeeInfo,\n  HalfSignedTransaction,\n  RecoveryInfo,\n  RecoveryOptions,\n  RecoveryTransaction,\n  SignTransactionOptions,\n  SupplementGenerateWalletOptions,\n  TransactionExplanation,\n  VerifyAddressOptions,\n  XrpTransactionType,\n} from './lib/iface';\nimport { KeyPair as XrpKeyPair } from './lib/keyPair';\nimport utils from './lib/utils';\nimport ripple from './ripple';\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  /** @inheritdoc */\n  valuelessTransferAllowed(): boolean {\n    return true;\n  }\n\n  /** inherited doc */\n  getDefaultMultisigType(): MultisigType {\n    return multisigTypes.onchain;\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  getTransactionTypeRawTxHex(txHex: string): XrpTransactionType | undefined {\n    let transaction;\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      } catch (e) {\n        throw new Error('txHex needs to be either hex or JSON string for XRP');\n      }\n    }\n\n    return transaction.TransactionType;\n  }\n\n  verifyTxType(txPrebuildDecoded: TransactionExplanation, txHexPrebuild: string | undefined): void {\n    if (!txHexPrebuild) throw new Error('Missing txHexPrebuild to verify token type for enabletoken tx');\n    const transactionType = this.getTransactionTypeRawTxHex(txHexPrebuild);\n    if (transactionType === undefined) throw new Error('Missing TransactionType on token enablement tx');\n    if (transactionType !== XrpTransactionType.TrustSet)\n      throw new Error(`tx type ${transactionType} does not match expected type TrustSet`);\n    // decoded payload type could come as undefined or any of the enabletoken like types but never as something else like Send, etc\n    const actualTypeFromDecoded =\n      'type' in txPrebuildDecoded && typeof txPrebuildDecoded.type === 'string' ? txPrebuildDecoded.type : undefined;\n    if (\n      !actualTypeFromDecoded ||\n      actualTypeFromDecoded === 'enabletoken' ||\n      actualTypeFromDecoded === 'AssociatedTokenAccountInitialization'\n    )\n      return;\n\n    throw new Error(`tx type ${actualTypeFromDecoded} does not match the expected type enabletoken`);\n  }\n\n  verifyTokenName(\n    txParams: TransactionParams,\n    txPrebuildDecoded: TransactionExplanation,\n    txHexPrebuild: string | undefined,\n    coinConfig: XrpCoin\n  ): void {\n    if (!txHexPrebuild) throw new Error('Missing txHexPrebuild param required for token enablement.');\n    if (!txParams.recipients || txParams.recipients.length === 0)\n      throw new Error('Missing recipients param for token enablement.');\n    const fullTokenName = txParams.recipients[0].tokenName;\n    if (fullTokenName === undefined)\n      throw new Error('Param tokenName is required for token enablement. Recipient must include a token name.');\n\n    if (!('limitAmount' in txPrebuildDecoded)) throw new Error('Missing limitAmount param for token enablement.');\n\n    // we check currency on both the txHex but also the explained payload\n    const expectedCurrency = utils.getXrpCurrencyFromTokenName(fullTokenName).currency;\n    if (coinConfig.isToken && expectedCurrency !== txPrebuildDecoded.limitAmount.currency)\n      throw new Error('Invalid token issuer or currency on token enablement tx');\n  }\n\n  verifyActivationAddress(txParams: TransactionParams, txPrebuildDecoded: TransactionExplanation): void {\n    if (txParams.recipients === undefined || txParams.recipients.length === 0)\n      throw new Error('Missing recipients param for token enablement.');\n\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    if (!('account' in txPrebuildDecoded)) throw new Error('missing account on token enablement tx');\n\n    const activationAddress = txParams.recipients[0].address;\n    const accountAddress = txPrebuildDecoded.account;\n    if (activationAddress !== accountAddress) throw new Error(\"Account address doesn't match with activation address.\");\n  }\n\n  verifyTokenIssuer(txParams: TransactionParams, txPrebuildDecoded: TransactionExplanation): void {\n    if (txPrebuildDecoded === undefined || !('limitAmount' in txPrebuildDecoded))\n      throw new Error('missing token issuer on token enablement tx');\n    const { issuer, currency } = txPrebuildDecoded.limitAmount;\n    if (!utils.getXrpToken(issuer, currency))\n      throw new Error('Invalid token issuer or currency on token enablement tx');\n  }\n\n  verifyRequiredKeys(txParams: TransactionParams, txPrebuildDecoded: TransactionExplanation): void {\n    if (\n      !('account' in txPrebuildDecoded) ||\n      !('limitAmount' in txPrebuildDecoded) ||\n      !txPrebuildDecoded.limitAmount.currency\n    ) {\n      throw new Error('Explanation is missing required keys (account or limitAmount with currency)');\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, verification }: 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    // Explaining a tx strips out certain data, for extra measurement we're checking vs the explained tx\n    // but also vs the tx pre explained.\n    if (txParams.type === 'enabletoken' && verification?.verifyTokenEnablement) {\n      this.verifyTxType(explanation, txPrebuild.txHex);\n      this.verifyActivationAddress(txParams, explanation);\n      this.verifyTokenIssuer(txParams, explanation);\n      this.verifyTokenName(txParams, explanation, txPrebuild.txHex, coinConfig);\n      this.verifyRequiredKeys(txParams, explanation);\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 (utils.getAddressDetails(recipient1.address).address !== utils.getAddressDetails(recipient2.address).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    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 accountInfoParams = {\n      method: 'account_info',\n      params: [\n        {\n          account: address,\n          ledger_index: 'current',\n          queue: true,\n          strict: true,\n          signer_lists: true,\n        },\n      ],\n    };\n\n    const accountInfo = (await this.bitgo.post(this.getRippledUrl()).send(accountInfoParams)).body;\n\n    if (accountInfo?.result?.account_data?.Flags == null) {\n      throw new Error('Invalid account information: Flags field is missing.');\n    }\n\n    const flags = xrpl.parseAccountRootFlags(accountInfo.result.account_data.Flags);\n\n    const addressDetails = utils.getAddressDetails(address);\n    const rootAddressDetails = utils.getAddressDetails(rootAddress);\n\n    if (flags.lsfRequireDestTag && addressDetails.destinationTag == null) {\n      throw new InvalidAddressError(`Invalid Address: Destination Tag is required for address \"${address}\".`);\n    }\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  /** @inheritDoc */\n  auditDecryptedKey(params: AuditDecryptedKeyParams) {\n    throw new MethodNotImplementedError();\n  }\n}\n"]}
|