@bitgo-beta/sdk-coin-sui 3.0.3-beta.75 → 3.0.3-beta.751
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +753 -0
- package/dist/src/index.d.ts +1 -0
- package/dist/src/index.d.ts.map +1 -1
- package/dist/src/index.js +7 -2
- package/dist/src/lib/compareTransactionBlocks.js +2 -3
- package/dist/src/lib/constants.d.ts +9 -2
- package/dist/src/lib/constants.d.ts.map +1 -1
- package/dist/src/lib/constants.js +18 -2
- package/dist/src/lib/customTransaction.d.ts +57 -0
- package/dist/src/lib/customTransaction.d.ts.map +1 -0
- package/dist/src/lib/customTransaction.js +159 -0
- package/dist/src/lib/customTransactionBuilder.d.ts +46 -0
- package/dist/src/lib/customTransactionBuilder.d.ts.map +1 -0
- package/dist/src/lib/customTransactionBuilder.js +117 -0
- package/dist/src/lib/iface.d.ts +76 -10
- package/dist/src/lib/iface.d.ts.map +1 -1
- package/dist/src/lib/iface.js +40 -5
- package/dist/src/lib/index.d.ts +8 -0
- package/dist/src/lib/index.d.ts.map +1 -1
- package/dist/src/lib/index.js +40 -10
- package/dist/src/lib/keyPair.js +24 -10
- package/dist/src/lib/mystenlab/builder/Inputs.d.ts +9 -9
- package/dist/src/lib/mystenlab/builder/Inputs.d.ts.map +1 -1
- package/dist/src/lib/mystenlab/builder/Inputs.js +18 -19
- package/dist/src/lib/mystenlab/builder/TransactionBlock.d.ts +40 -354
- package/dist/src/lib/mystenlab/builder/TransactionBlock.d.ts.map +1 -1
- package/dist/src/lib/mystenlab/builder/TransactionBlock.js +22 -25
- package/dist/src/lib/mystenlab/builder/TransactionDataBlock.d.ts +74 -74
- package/dist/src/lib/mystenlab/builder/TransactionDataBlock.d.ts.map +1 -1
- package/dist/src/lib/mystenlab/builder/TransactionDataBlock.js +41 -44
- package/dist/src/lib/mystenlab/builder/Transactions.d.ts +133 -188
- package/dist/src/lib/mystenlab/builder/Transactions.d.ts.map +1 -1
- package/dist/src/lib/mystenlab/builder/Transactions.js +52 -53
- package/dist/src/lib/mystenlab/builder/bcs.d.ts +1 -1
- package/dist/src/lib/mystenlab/builder/bcs.d.ts.map +1 -1
- package/dist/src/lib/mystenlab/builder/bcs.js +2 -2
- package/dist/src/lib/mystenlab/builder/index.js +6 -2
- package/dist/src/lib/mystenlab/builder/serializer.js +6 -8
- package/dist/src/lib/mystenlab/builder/utils.d.ts +1 -1
- package/dist/src/lib/mystenlab/builder/utils.d.ts.map +1 -1
- package/dist/src/lib/mystenlab/builder/utils.js +4 -4
- package/dist/src/lib/mystenlab/cryptography/hash.js +3 -4
- package/dist/src/lib/mystenlab/framework/framework.d.ts +6 -6
- package/dist/src/lib/mystenlab/framework/framework.d.ts.map +1 -1
- package/dist/src/lib/mystenlab/framework/framework.js +22 -25
- package/dist/src/lib/mystenlab/framework/index.js +6 -2
- package/dist/src/lib/mystenlab/framework/sui-system-state.js +2 -2
- package/dist/src/lib/mystenlab/txn-data-serializers/type-tag-serializer.js +2 -2
- package/dist/src/lib/mystenlab/types/coin.d.ts +10 -10
- package/dist/src/lib/mystenlab/types/coin.d.ts.map +1 -1
- package/dist/src/lib/mystenlab/types/coin.js +19 -19
- package/dist/src/lib/mystenlab/types/common.d.ts +8 -8
- package/dist/src/lib/mystenlab/types/common.d.ts.map +1 -1
- package/dist/src/lib/mystenlab/types/common.js +22 -22
- package/dist/src/lib/mystenlab/types/events.d.ts +14 -14
- package/dist/src/lib/mystenlab/types/events.d.ts.map +1 -1
- package/dist/src/lib/mystenlab/types/events.js +17 -17
- package/dist/src/lib/mystenlab/types/index.js +6 -2
- package/dist/src/lib/mystenlab/types/normalized.d.ts +21 -21
- package/dist/src/lib/mystenlab/types/normalized.d.ts.map +1 -1
- package/dist/src/lib/mystenlab/types/normalized.js +41 -41
- package/dist/src/lib/mystenlab/types/objects.d.ts +51 -51
- package/dist/src/lib/mystenlab/types/objects.d.ts.map +1 -1
- package/dist/src/lib/mystenlab/types/objects.js +96 -106
- package/dist/src/lib/mystenlab/types/option.d.ts +1 -1
- package/dist/src/lib/mystenlab/types/option.d.ts.map +1 -1
- package/dist/src/lib/mystenlab/types/option.js +2 -3
- package/dist/src/lib/mystenlab/types/sui-bcs.d.ts +8 -8
- package/dist/src/lib/mystenlab/types/sui-bcs.d.ts.map +1 -1
- package/dist/src/lib/mystenlab/types/sui-bcs.js +5 -5
- package/dist/src/lib/mystenlab/types/transactions.d.ts +625 -625
- package/dist/src/lib/mystenlab/types/transactions.d.ts.map +1 -1
- package/dist/src/lib/mystenlab/types/transactions.js +178 -194
- package/dist/src/lib/mystenlab/types/validator.d.ts +9 -9
- package/dist/src/lib/mystenlab/types/validator.d.ts.map +1 -1
- package/dist/src/lib/mystenlab/types/validator.js +124 -124
- package/dist/src/lib/resources/walrusConfig.d.ts +22 -0
- package/dist/src/lib/resources/walrusConfig.d.ts.map +1 -0
- package/dist/src/lib/resources/walrusConfig.js +37 -0
- package/dist/src/lib/rpcClient.d.ts +5 -0
- package/dist/src/lib/rpcClient.d.ts.map +1 -0
- package/dist/src/lib/rpcClient.js +74 -0
- package/dist/src/lib/stakingBuilder.d.ts.map +1 -1
- package/dist/src/lib/stakingBuilder.js +23 -7
- package/dist/src/lib/stakingTransaction.d.ts +1 -1
- package/dist/src/lib/stakingTransaction.d.ts.map +1 -1
- package/dist/src/lib/stakingTransaction.js +26 -15
- package/dist/src/lib/tokenTransferBuilder.d.ts +38 -0
- package/dist/src/lib/tokenTransferBuilder.d.ts.map +1 -0
- package/dist/src/lib/tokenTransferBuilder.js +132 -0
- package/dist/src/lib/tokenTransferTransaction.d.ts +57 -0
- package/dist/src/lib/tokenTransferTransaction.d.ts.map +1 -0
- package/dist/src/lib/tokenTransferTransaction.js +250 -0
- package/dist/src/lib/transaction.d.ts +12 -4
- package/dist/src/lib/transaction.d.ts.map +1 -1
- package/dist/src/lib/transaction.js +91 -18
- package/dist/src/lib/transactionBuilder.d.ts +2 -3
- package/dist/src/lib/transactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilder.js +4 -4
- package/dist/src/lib/transactionBuilderFactory.d.ts +14 -2
- package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilderFactory.js +42 -1
- package/dist/src/lib/transferBuilder.d.ts.map +1 -1
- package/dist/src/lib/transferBuilder.js +21 -5
- package/dist/src/lib/transferTransaction.d.ts +1 -1
- package/dist/src/lib/transferTransaction.d.ts.map +1 -1
- package/dist/src/lib/transferTransaction.js +31 -8
- package/dist/src/lib/unstakingBuilder.js +6 -6
- package/dist/src/lib/unstakingTransaction.d.ts +1 -1
- package/dist/src/lib/unstakingTransaction.d.ts.map +1 -1
- package/dist/src/lib/unstakingTransaction.js +31 -17
- package/dist/src/lib/utils.d.ts +15 -4
- package/dist/src/lib/utils.d.ts.map +1 -1
- package/dist/src/lib/utils.js +264 -29
- package/dist/src/lib/walrusStakingBuilder.d.ts +66 -0
- package/dist/src/lib/walrusStakingBuilder.d.ts.map +1 -0
- package/dist/src/lib/walrusStakingBuilder.js +199 -0
- package/dist/src/lib/walrusStakingTransaction.d.ts +52 -0
- package/dist/src/lib/walrusStakingTransaction.d.ts.map +1 -0
- package/dist/src/lib/walrusStakingTransaction.js +269 -0
- package/dist/src/lib/walrusWithdrawStakeBuilder.d.ts +36 -0
- package/dist/src/lib/walrusWithdrawStakeBuilder.d.ts.map +1 -0
- package/dist/src/lib/walrusWithdrawStakeBuilder.js +172 -0
- package/dist/src/lib/walrusWithdrawStakeTransaction.d.ts +21 -0
- package/dist/src/lib/walrusWithdrawStakeTransaction.d.ts.map +1 -0
- package/dist/src/lib/walrusWithdrawStakeTransaction.js +186 -0
- package/dist/src/register.d.ts.map +1 -1
- package/dist/src/register.js +5 -1
- package/dist/src/sui.d.ts +42 -8
- package/dist/src/sui.d.ts.map +1 -1
- package/dist/src/sui.js +468 -32
- package/dist/src/suiToken.d.ts +22 -0
- package/dist/src/suiToken.d.ts.map +1 -0
- package/dist/src/suiToken.js +61 -0
- package/dist/src/tsui.js +1 -1
- package/package.json +10 -8
package/dist/src/sui.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,13 +15,23 @@ 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
|
-
|
|
20
|
-
};
|
|
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
|
+
})();
|
|
21
35
|
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
22
36
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
23
37
|
};
|
|
@@ -29,7 +43,9 @@ const bignumber_js_1 = __importDefault(require("bignumber.js"));
|
|
|
29
43
|
const lib_1 = require("./lib");
|
|
30
44
|
const utils_1 = __importDefault(require("./lib/utils"));
|
|
31
45
|
const _ = __importStar(require("lodash"));
|
|
32
|
-
const
|
|
46
|
+
const iface_1 = require("./lib/iface");
|
|
47
|
+
const constants_1 = require("./lib/constants");
|
|
48
|
+
const sdk_lib_mpc_1 = require("@bitgo-beta/sdk-lib-mpc");
|
|
33
49
|
class Sui extends sdk_core_1.BaseCoin {
|
|
34
50
|
constructor(bitgo, staticsCoin) {
|
|
35
51
|
super(bitgo);
|
|
@@ -56,6 +72,9 @@ class Sui extends sdk_core_1.BaseCoin {
|
|
|
56
72
|
getFullName() {
|
|
57
73
|
return 'Sui';
|
|
58
74
|
}
|
|
75
|
+
getNetwork() {
|
|
76
|
+
return this._staticsCoin.network;
|
|
77
|
+
}
|
|
59
78
|
/** @inheritDoc */
|
|
60
79
|
supportsTss() {
|
|
61
80
|
return true;
|
|
@@ -67,11 +86,10 @@ class Sui extends sdk_core_1.BaseCoin {
|
|
|
67
86
|
return true;
|
|
68
87
|
}
|
|
69
88
|
async verifyTransaction(params) {
|
|
70
|
-
var _a;
|
|
71
89
|
let totalAmount = new bignumber_js_1.default(0);
|
|
72
90
|
const coinConfig = statics_1.coins.get(this.getChain());
|
|
73
91
|
const { txPrebuild: txPrebuild, txParams: txParams } = params;
|
|
74
|
-
const transaction = new
|
|
92
|
+
const transaction = new lib_1.TransferTransaction(coinConfig);
|
|
75
93
|
const rawTx = txPrebuild.txHex;
|
|
76
94
|
if (!rawTx) {
|
|
77
95
|
throw new Error('missing required tx prebuild property txHex');
|
|
@@ -79,7 +97,7 @@ class Sui extends sdk_core_1.BaseCoin {
|
|
|
79
97
|
transaction.fromRawTransaction(Buffer.from(rawTx, 'hex').toString('base64'));
|
|
80
98
|
const explainedTx = transaction.explainTransaction();
|
|
81
99
|
if (txParams.recipients && txParams.recipients.length > 0) {
|
|
82
|
-
const filteredRecipients =
|
|
100
|
+
const filteredRecipients = txParams.recipients?.map((recipient) => {
|
|
83
101
|
const filteredRecipient = _.pick(recipient, ['address', 'amount']);
|
|
84
102
|
filteredRecipient.amount = new bignumber_js_1.default(filteredRecipient.amount).toFixed();
|
|
85
103
|
return filteredRecipient;
|
|
@@ -102,23 +120,10 @@ class Sui extends sdk_core_1.BaseCoin {
|
|
|
102
120
|
return true;
|
|
103
121
|
}
|
|
104
122
|
async isWalletAddress(params) {
|
|
105
|
-
const {
|
|
123
|
+
const { address: newAddress } = params;
|
|
106
124
|
if (!this.isValidAddress(newAddress)) {
|
|
107
125
|
throw new sdk_core_1.InvalidAddressError(`invalid address: ${newAddress}`);
|
|
108
126
|
}
|
|
109
|
-
if (!keychains) {
|
|
110
|
-
throw new Error('missing required param keychains');
|
|
111
|
-
}
|
|
112
|
-
for (const keychain of keychains) {
|
|
113
|
-
const MPC = await sdk_core_1.EDDSAMethods.getInitializedMpcInstance();
|
|
114
|
-
const commonKeychain = keychain.commonKeychain;
|
|
115
|
-
const derivationPath = 'm/' + index;
|
|
116
|
-
const derivedPublicKey = MPC.deriveUnhardened(commonKeychain, derivationPath).slice(0, 64);
|
|
117
|
-
const expectedAddress = this.getAddressFromPublicKey(derivedPublicKey);
|
|
118
|
-
if (newAddress !== expectedAddress) {
|
|
119
|
-
return false;
|
|
120
|
-
}
|
|
121
|
-
}
|
|
122
127
|
return true;
|
|
123
128
|
}
|
|
124
129
|
async parseTransaction(params) {
|
|
@@ -126,20 +131,24 @@ class Sui extends sdk_core_1.BaseCoin {
|
|
|
126
131
|
if (!transactionExplanation) {
|
|
127
132
|
throw new Error('Invalid transaction');
|
|
128
133
|
}
|
|
134
|
+
let fee = new bignumber_js_1.default(0);
|
|
129
135
|
const suiTransaction = transactionExplanation;
|
|
130
136
|
if (suiTransaction.outputs.length <= 0) {
|
|
131
137
|
return {
|
|
132
138
|
inputs: [],
|
|
133
139
|
outputs: [],
|
|
140
|
+
fee,
|
|
134
141
|
};
|
|
135
142
|
}
|
|
136
143
|
const senderAddress = suiTransaction.outputs[0].address;
|
|
137
|
-
|
|
144
|
+
if (suiTransaction.fee.fee !== '') {
|
|
145
|
+
fee = new bignumber_js_1.default(suiTransaction.fee.fee);
|
|
146
|
+
}
|
|
138
147
|
// assume 1 sender, who is also the fee payer
|
|
139
148
|
const inputs = [
|
|
140
149
|
{
|
|
141
150
|
address: senderAddress,
|
|
142
|
-
amount: new bignumber_js_1.default(suiTransaction.outputAmount).plus(
|
|
151
|
+
amount: new bignumber_js_1.default(suiTransaction.outputAmount).plus(fee).toFixed(),
|
|
143
152
|
},
|
|
144
153
|
];
|
|
145
154
|
const outputs = suiTransaction.outputs.map((output) => {
|
|
@@ -151,6 +160,7 @@ class Sui extends sdk_core_1.BaseCoin {
|
|
|
151
160
|
return {
|
|
152
161
|
inputs,
|
|
153
162
|
outputs,
|
|
163
|
+
fee,
|
|
154
164
|
};
|
|
155
165
|
}
|
|
156
166
|
generateKeyPair(seed) {
|
|
@@ -164,16 +174,16 @@ class Sui extends sdk_core_1.BaseCoin {
|
|
|
164
174
|
prv: keys.prv,
|
|
165
175
|
};
|
|
166
176
|
}
|
|
167
|
-
isValidPub(
|
|
177
|
+
isValidPub(_) {
|
|
168
178
|
throw new Error('Method not implemented.');
|
|
169
179
|
}
|
|
170
|
-
isValidPrv(
|
|
180
|
+
isValidPrv(_) {
|
|
171
181
|
throw new Error('Method not implemented.');
|
|
172
182
|
}
|
|
173
183
|
isValidAddress(address) {
|
|
174
184
|
return utils_1.default.isValidAddress(address);
|
|
175
185
|
}
|
|
176
|
-
signTransaction(
|
|
186
|
+
signTransaction(_) {
|
|
177
187
|
throw new Error('Method not implemented.');
|
|
178
188
|
}
|
|
179
189
|
/**
|
|
@@ -205,6 +215,432 @@ class Sui extends sdk_core_1.BaseCoin {
|
|
|
205
215
|
const rebuiltTransaction = await factory.from(serializedTx).build();
|
|
206
216
|
return rebuiltTransaction.signablePayload;
|
|
207
217
|
}
|
|
218
|
+
getPublicNodeUrl() {
|
|
219
|
+
return sdk_core_1.Environments[this.bitgo.getEnv()].suiNodeUrl;
|
|
220
|
+
}
|
|
221
|
+
async getBalance(owner, coinType) {
|
|
222
|
+
const url = this.getPublicNodeUrl();
|
|
223
|
+
return await utils_1.default.getBalance(url, owner, coinType);
|
|
224
|
+
}
|
|
225
|
+
async getInputCoins(owner, coinType) {
|
|
226
|
+
const url = this.getPublicNodeUrl();
|
|
227
|
+
return await utils_1.default.getInputCoins(url, owner, coinType);
|
|
228
|
+
}
|
|
229
|
+
async getFeeEstimate(txHex) {
|
|
230
|
+
const url = this.getPublicNodeUrl();
|
|
231
|
+
return await utils_1.default.getFeeEstimate(url, txHex);
|
|
232
|
+
}
|
|
233
|
+
/**
|
|
234
|
+
* Builds funds recovery transaction(s) without BitGo
|
|
235
|
+
*
|
|
236
|
+
* @param {MPCRecoveryOptions} params parameters needed to construct and
|
|
237
|
+
* (maybe) sign the transaction
|
|
238
|
+
*
|
|
239
|
+
* @returns {MPCTx | MPCSweepTxs} array of the serialized transaction hex strings and indices
|
|
240
|
+
* of the addresses being swept
|
|
241
|
+
*/
|
|
242
|
+
async recover(params) {
|
|
243
|
+
if (!params.bitgoKey) {
|
|
244
|
+
throw new Error('missing bitgoKey');
|
|
245
|
+
}
|
|
246
|
+
if (!params.recoveryDestination || !this.isValidAddress(params.recoveryDestination)) {
|
|
247
|
+
throw new Error('invalid recoveryDestination');
|
|
248
|
+
}
|
|
249
|
+
const startIdx = utils_1.default.validateNonNegativeNumber(0, 'Invalid starting index to scan for addresses', params.startingScanIndex);
|
|
250
|
+
const numIterations = utils_1.default.validateNonNegativeNumber(20, 'Invalid scanning factor', params.scan);
|
|
251
|
+
const endIdx = startIdx + numIterations;
|
|
252
|
+
const bitgoKey = params.bitgoKey.replace(/\s/g, '');
|
|
253
|
+
const MPC = await sdk_core_1.EDDSAMethods.getInitializedMpcInstance();
|
|
254
|
+
for (let idx = startIdx; idx < endIdx; idx++) {
|
|
255
|
+
const derivationPath = (params.seed ? (0, sdk_lib_mpc_1.getDerivationPath)(params.seed) : 'm') + `/${idx}`;
|
|
256
|
+
const derivedPublicKey = MPC.deriveUnhardened(bitgoKey, derivationPath).slice(0, 64);
|
|
257
|
+
const senderAddress = this.getAddressFromPublicKey(derivedPublicKey);
|
|
258
|
+
let availableBalance = new bignumber_js_1.default(0);
|
|
259
|
+
try {
|
|
260
|
+
availableBalance = new bignumber_js_1.default(await this.getBalance(senderAddress));
|
|
261
|
+
}
|
|
262
|
+
catch (e) {
|
|
263
|
+
continue;
|
|
264
|
+
}
|
|
265
|
+
if (availableBalance.minus(constants_1.MAX_GAS_BUDGET).toNumber() <= 0) {
|
|
266
|
+
continue;
|
|
267
|
+
}
|
|
268
|
+
// check for possible token recovery, recover the token provide by user
|
|
269
|
+
if (params.tokenContractAddress) {
|
|
270
|
+
const token = utils_1.default.getSuiTokenFromAddress(params.tokenContractAddress, this.getNetwork());
|
|
271
|
+
if (!token) {
|
|
272
|
+
throw new Error(`Sui Token Package ID not supported.`);
|
|
273
|
+
}
|
|
274
|
+
const coinType = `${token.packageId}::${token.module}::${token.symbol}`;
|
|
275
|
+
try {
|
|
276
|
+
const availableTokenBalance = new bignumber_js_1.default(await this.getBalance(senderAddress, coinType));
|
|
277
|
+
if (availableTokenBalance.toNumber() <= 0) {
|
|
278
|
+
continue;
|
|
279
|
+
}
|
|
280
|
+
}
|
|
281
|
+
catch (e) {
|
|
282
|
+
continue;
|
|
283
|
+
}
|
|
284
|
+
return this.recoverSuiToken(params, token, senderAddress, derivationPath, derivedPublicKey, idx, bitgoKey);
|
|
285
|
+
}
|
|
286
|
+
let inputCoins = await this.getInputCoins(senderAddress);
|
|
287
|
+
inputCoins = inputCoins.sort((a, b) => {
|
|
288
|
+
return b.balance.minus(a.balance).toNumber();
|
|
289
|
+
});
|
|
290
|
+
if (inputCoins.length > constants_1.MAX_OBJECT_LIMIT) {
|
|
291
|
+
inputCoins = inputCoins.slice(0, constants_1.MAX_OBJECT_LIMIT);
|
|
292
|
+
}
|
|
293
|
+
let netAmount = inputCoins.reduce((acc, obj) => acc.plus(obj.balance), new bignumber_js_1.default(0));
|
|
294
|
+
netAmount = netAmount.minus(constants_1.MAX_GAS_BUDGET);
|
|
295
|
+
const recipients = [
|
|
296
|
+
{
|
|
297
|
+
address: params.recoveryDestination,
|
|
298
|
+
amount: netAmount.toString(),
|
|
299
|
+
},
|
|
300
|
+
];
|
|
301
|
+
// first build the unsigned txn
|
|
302
|
+
const factory = new lib_1.TransactionBuilderFactory(statics_1.coins.get(this.getChain()));
|
|
303
|
+
const txBuilder = factory
|
|
304
|
+
.getTransferBuilder()
|
|
305
|
+
.type(iface_1.SuiTransactionType.Transfer)
|
|
306
|
+
.sender(senderAddress)
|
|
307
|
+
.send(recipients)
|
|
308
|
+
.gasData({
|
|
309
|
+
owner: senderAddress,
|
|
310
|
+
price: constants_1.DEFAULT_GAS_PRICE,
|
|
311
|
+
budget: constants_1.MAX_GAS_BUDGET,
|
|
312
|
+
payment: inputCoins,
|
|
313
|
+
});
|
|
314
|
+
const tempTx = (await txBuilder.build());
|
|
315
|
+
const feeEstimate = await this.getFeeEstimate(tempTx.toBroadcastFormat());
|
|
316
|
+
const gasBudget = Math.trunc(feeEstimate.toNumber() * constants_1.DEFAULT_GAS_OVERHEAD);
|
|
317
|
+
netAmount = netAmount.plus(constants_1.MAX_GAS_BUDGET).minus(gasBudget);
|
|
318
|
+
recipients[0].amount = netAmount.toString();
|
|
319
|
+
txBuilder.send(recipients);
|
|
320
|
+
txBuilder.gasData({
|
|
321
|
+
owner: senderAddress,
|
|
322
|
+
price: constants_1.DEFAULT_GAS_PRICE,
|
|
323
|
+
budget: gasBudget,
|
|
324
|
+
payment: inputCoins,
|
|
325
|
+
});
|
|
326
|
+
const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;
|
|
327
|
+
if (isUnsignedSweep) {
|
|
328
|
+
return this.buildUnsignedSweepTransaction(txBuilder, senderAddress, bitgoKey, idx, derivationPath);
|
|
329
|
+
}
|
|
330
|
+
await this.signRecoveryTransaction(txBuilder, params, derivationPath, derivedPublicKey, false);
|
|
331
|
+
const tx = (await txBuilder.build());
|
|
332
|
+
return {
|
|
333
|
+
transactions: [
|
|
334
|
+
{
|
|
335
|
+
scanIndex: idx,
|
|
336
|
+
recoveryAmount: netAmount.toString(),
|
|
337
|
+
serializedTx: tx.toBroadcastFormat(),
|
|
338
|
+
signature: Buffer.from(tx.serializedSig).toString('base64'),
|
|
339
|
+
coin: this.getChain(),
|
|
340
|
+
},
|
|
341
|
+
],
|
|
342
|
+
lastScanIndex: idx,
|
|
343
|
+
};
|
|
344
|
+
}
|
|
345
|
+
throw new Error(`Did not find an address with sufficient funds to recover. Please start the next scan at address index ${endIdx}. If it is token transaction, please keep sufficient Sui balance in the address for the transaction fee.`);
|
|
346
|
+
}
|
|
347
|
+
async recoverSuiToken(params, token, senderAddress, derivationPath, derivedPublicKey, idx, bitgoKey) {
|
|
348
|
+
const coinType = `${token.packageId}::${token.module}::${token.symbol}`;
|
|
349
|
+
let tokenObjects = await this.getInputCoins(senderAddress, coinType);
|
|
350
|
+
tokenObjects = tokenObjects.sort((a, b) => {
|
|
351
|
+
return b.balance.minus(a.balance).toNumber();
|
|
352
|
+
});
|
|
353
|
+
if (tokenObjects.length > constants_1.TOKEN_OBJECT_LIMIT) {
|
|
354
|
+
tokenObjects = tokenObjects.slice(0, constants_1.TOKEN_OBJECT_LIMIT);
|
|
355
|
+
}
|
|
356
|
+
const netAmount = tokenObjects.reduce((acc, obj) => acc.plus(obj.balance), new bignumber_js_1.default(0));
|
|
357
|
+
const recipients = [
|
|
358
|
+
{
|
|
359
|
+
address: params.recoveryDestination,
|
|
360
|
+
amount: netAmount.toString(),
|
|
361
|
+
},
|
|
362
|
+
];
|
|
363
|
+
const gasAmount = new bignumber_js_1.default(constants_1.MAX_GAS_BUDGET);
|
|
364
|
+
let gasObjects = await this.getInputCoins(senderAddress);
|
|
365
|
+
gasObjects = utils_1.default.selectObjectsInDescOrderOfBalance(gasObjects, gasAmount);
|
|
366
|
+
if (gasObjects.length >= constants_1.MAX_GAS_OBJECTS) {
|
|
367
|
+
gasObjects = gasObjects.slice(0, constants_1.MAX_GAS_OBJECTS - 1);
|
|
368
|
+
}
|
|
369
|
+
// first build the unsigned txn
|
|
370
|
+
const factory = new lib_1.TransactionBuilderFactory(token);
|
|
371
|
+
const txBuilder = factory
|
|
372
|
+
.getTokenTransferBuilder()
|
|
373
|
+
.type(iface_1.SuiTransactionType.TokenTransfer)
|
|
374
|
+
.sender(senderAddress)
|
|
375
|
+
.send(recipients)
|
|
376
|
+
.inputObjects(tokenObjects)
|
|
377
|
+
.gasData({
|
|
378
|
+
owner: senderAddress,
|
|
379
|
+
price: constants_1.DEFAULT_GAS_PRICE,
|
|
380
|
+
budget: constants_1.MAX_GAS_BUDGET,
|
|
381
|
+
payment: gasObjects,
|
|
382
|
+
});
|
|
383
|
+
const tempTx = (await txBuilder.build());
|
|
384
|
+
const feeEstimate = await this.getFeeEstimate(tempTx.toBroadcastFormat());
|
|
385
|
+
const gasBudget = Math.trunc(feeEstimate.toNumber() * constants_1.DEFAULT_GAS_OVERHEAD);
|
|
386
|
+
txBuilder.gasData({
|
|
387
|
+
owner: senderAddress,
|
|
388
|
+
price: constants_1.DEFAULT_GAS_PRICE,
|
|
389
|
+
budget: gasBudget,
|
|
390
|
+
payment: gasObjects,
|
|
391
|
+
});
|
|
392
|
+
const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;
|
|
393
|
+
if (isUnsignedSweep) {
|
|
394
|
+
return this.buildUnsignedSweepTransaction(txBuilder, senderAddress, bitgoKey, idx, derivationPath, token);
|
|
395
|
+
}
|
|
396
|
+
await this.signRecoveryTransaction(txBuilder, params, derivationPath, derivedPublicKey, true);
|
|
397
|
+
const tx = (await txBuilder.build());
|
|
398
|
+
return {
|
|
399
|
+
transactions: [
|
|
400
|
+
{
|
|
401
|
+
scanIndex: idx,
|
|
402
|
+
recoveryAmount: netAmount.toString(),
|
|
403
|
+
serializedTx: tx.toBroadcastFormat(),
|
|
404
|
+
signature: Buffer.from(tx.serializedSig).toString('base64'),
|
|
405
|
+
coin: token.name,
|
|
406
|
+
},
|
|
407
|
+
],
|
|
408
|
+
lastScanIndex: idx,
|
|
409
|
+
};
|
|
410
|
+
}
|
|
411
|
+
async buildUnsignedSweepTransaction(txBuilder, senderAddress, bitgoKey, index, derivationPath, token) {
|
|
412
|
+
const isTokenTransaction = !!token;
|
|
413
|
+
const unsignedTransaction = isTokenTransaction
|
|
414
|
+
? (await txBuilder.build())
|
|
415
|
+
: (await txBuilder.build());
|
|
416
|
+
const serializedTx = unsignedTransaction.toBroadcastFormat();
|
|
417
|
+
const serializedTxHex = Buffer.from(serializedTx, 'base64').toString('hex');
|
|
418
|
+
const parsedTx = await this.parseTransaction({ txHex: serializedTxHex });
|
|
419
|
+
const walletCoin = isTokenTransaction ? token.name : this.getChain();
|
|
420
|
+
const output = parsedTx.outputs[0];
|
|
421
|
+
const inputs = [
|
|
422
|
+
{
|
|
423
|
+
address: senderAddress,
|
|
424
|
+
valueString: output.amount,
|
|
425
|
+
value: new bignumber_js_1.default(output.amount),
|
|
426
|
+
},
|
|
427
|
+
];
|
|
428
|
+
const outputs = [
|
|
429
|
+
{
|
|
430
|
+
address: output.address,
|
|
431
|
+
valueString: output.amount,
|
|
432
|
+
coinName: walletCoin,
|
|
433
|
+
},
|
|
434
|
+
];
|
|
435
|
+
const spendAmount = output.amount;
|
|
436
|
+
const completedParsedTx = {
|
|
437
|
+
inputs: inputs,
|
|
438
|
+
outputs: outputs,
|
|
439
|
+
spendAmount: spendAmount,
|
|
440
|
+
type: isTokenTransaction ? iface_1.SuiTransactionType.TokenTransfer : iface_1.SuiTransactionType.Transfer,
|
|
441
|
+
};
|
|
442
|
+
const fee = parsedTx.fee;
|
|
443
|
+
const feeInfo = { fee: fee.toNumber(), feeString: fee.toString() };
|
|
444
|
+
const coinSpecific = { commonKeychain: bitgoKey };
|
|
445
|
+
const transaction = {
|
|
446
|
+
serializedTx: serializedTxHex,
|
|
447
|
+
scanIndex: index,
|
|
448
|
+
coin: walletCoin,
|
|
449
|
+
signableHex: unsignedTransaction.signablePayload.toString('hex'),
|
|
450
|
+
derivationPath,
|
|
451
|
+
parsedTx: completedParsedTx,
|
|
452
|
+
feeInfo: feeInfo,
|
|
453
|
+
coinSpecific: coinSpecific,
|
|
454
|
+
};
|
|
455
|
+
const unsignedTx = { unsignedTx: transaction, signatureShares: [] };
|
|
456
|
+
const transactions = [unsignedTx];
|
|
457
|
+
const txRequest = {
|
|
458
|
+
transactions: transactions,
|
|
459
|
+
walletCoin: walletCoin,
|
|
460
|
+
};
|
|
461
|
+
return { txRequests: [txRequest] };
|
|
462
|
+
}
|
|
463
|
+
async signRecoveryTransaction(txBuilder, params, derivationPath, derivedPublicKey, isTokenTransaction) {
|
|
464
|
+
// TODO(BG-51092): This looks like a common part which can be extracted out too
|
|
465
|
+
const unsignedTx = isTokenTransaction
|
|
466
|
+
? (await txBuilder.build())
|
|
467
|
+
: (await txBuilder.build());
|
|
468
|
+
if (!params.userKey) {
|
|
469
|
+
throw new Error('missing userKey');
|
|
470
|
+
}
|
|
471
|
+
if (!params.backupKey) {
|
|
472
|
+
throw new Error('missing backupKey');
|
|
473
|
+
}
|
|
474
|
+
if (!params.walletPassphrase) {
|
|
475
|
+
throw new Error('missing wallet passphrase');
|
|
476
|
+
}
|
|
477
|
+
// Clean up whitespace from entered values
|
|
478
|
+
const userKey = params.userKey.replace(/\s/g, '');
|
|
479
|
+
const backupKey = params.backupKey.replace(/\s/g, '');
|
|
480
|
+
// Decrypt private keys from KeyCard values
|
|
481
|
+
let userPrv;
|
|
482
|
+
try {
|
|
483
|
+
userPrv = this.bitgo.decrypt({
|
|
484
|
+
input: userKey,
|
|
485
|
+
password: params.walletPassphrase,
|
|
486
|
+
});
|
|
487
|
+
}
|
|
488
|
+
catch (e) {
|
|
489
|
+
throw new Error(`Error decrypting user keychain: ${e.message}`);
|
|
490
|
+
}
|
|
491
|
+
/** TODO BG-52419 Implement Codec for parsing */
|
|
492
|
+
const userSigningMaterial = JSON.parse(userPrv);
|
|
493
|
+
let backupPrv;
|
|
494
|
+
try {
|
|
495
|
+
backupPrv = this.bitgo.decrypt({
|
|
496
|
+
input: backupKey,
|
|
497
|
+
password: params.walletPassphrase,
|
|
498
|
+
});
|
|
499
|
+
}
|
|
500
|
+
catch (e) {
|
|
501
|
+
throw new Error(`Error decrypting backup keychain: ${e.message}`);
|
|
502
|
+
}
|
|
503
|
+
const backupSigningMaterial = JSON.parse(backupPrv);
|
|
504
|
+
/* ********************** END ***********************************/
|
|
505
|
+
// add signature
|
|
506
|
+
const signatureHex = await sdk_core_1.EDDSAMethods.getTSSSignature(userSigningMaterial, backupSigningMaterial, derivationPath, unsignedTx);
|
|
507
|
+
txBuilder.addSignature({ pub: derivedPublicKey }, signatureHex);
|
|
508
|
+
}
|
|
509
|
+
async broadcastTransaction({ transactions, }) {
|
|
510
|
+
const txIds = [];
|
|
511
|
+
const url = this.getPublicNodeUrl();
|
|
512
|
+
let digest = '';
|
|
513
|
+
if (!!transactions) {
|
|
514
|
+
for (const txn of transactions) {
|
|
515
|
+
try {
|
|
516
|
+
digest = await utils_1.default.executeTransactionBlock(url, txn.serializedTx, [txn.signature]);
|
|
517
|
+
}
|
|
518
|
+
catch (e) {
|
|
519
|
+
throw new Error(`Failed to broadcast transaction, error: ${e.message}`);
|
|
520
|
+
}
|
|
521
|
+
txIds.push(digest);
|
|
522
|
+
}
|
|
523
|
+
}
|
|
524
|
+
return { txIds };
|
|
525
|
+
}
|
|
526
|
+
/** inherited doc */
|
|
527
|
+
async createBroadcastableSweepTransaction(params) {
|
|
528
|
+
const req = params.signatureShares;
|
|
529
|
+
const broadcastableTransactions = [];
|
|
530
|
+
let lastScanIndex = 0;
|
|
531
|
+
for (let i = 0; i < req.length; i++) {
|
|
532
|
+
const MPC = await sdk_core_1.EDDSAMethods.getInitializedMpcInstance();
|
|
533
|
+
const transaction = req[i].txRequest.transactions[0].unsignedTx;
|
|
534
|
+
if (!req[i].ovc || !req[i].ovc[0].eddsaSignature) {
|
|
535
|
+
throw new Error('Missing signature(s)');
|
|
536
|
+
}
|
|
537
|
+
const signature = req[i].ovc[0].eddsaSignature;
|
|
538
|
+
if (!transaction.signableHex) {
|
|
539
|
+
throw new Error('Missing signable hex');
|
|
540
|
+
}
|
|
541
|
+
const messageBuffer = Buffer.from(transaction.signableHex, 'hex');
|
|
542
|
+
const result = MPC.verify(messageBuffer, signature);
|
|
543
|
+
if (!result) {
|
|
544
|
+
throw new Error('Invalid signature');
|
|
545
|
+
}
|
|
546
|
+
const signatureHex = Buffer.concat([Buffer.from(signature.R, 'hex'), Buffer.from(signature.sigma, 'hex')]);
|
|
547
|
+
const serializedTxBase64 = Buffer.from(transaction.serializedTx, 'hex').toString('base64');
|
|
548
|
+
const txBuilder = this.getBuilder().from(serializedTxBase64);
|
|
549
|
+
if (!transaction.coinSpecific?.commonKeychain) {
|
|
550
|
+
throw new Error('Missing common keychain');
|
|
551
|
+
}
|
|
552
|
+
const commonKeychain = transaction.coinSpecific.commonKeychain;
|
|
553
|
+
if (!transaction.derivationPath) {
|
|
554
|
+
throw new Error('Missing derivation path');
|
|
555
|
+
}
|
|
556
|
+
const derivationPath = transaction.derivationPath;
|
|
557
|
+
const derivedPublicKey = MPC.deriveUnhardened(commonKeychain, derivationPath).slice(0, 64);
|
|
558
|
+
// add combined signature from ovc
|
|
559
|
+
txBuilder.addSignature({ pub: derivedPublicKey }, signatureHex);
|
|
560
|
+
const signedTransaction = (await txBuilder.build());
|
|
561
|
+
const serializedTx = signedTransaction.toBroadcastFormat();
|
|
562
|
+
const outputAmount = signedTransaction.explainTransaction().outputAmount;
|
|
563
|
+
broadcastableTransactions.push({
|
|
564
|
+
serializedTx: serializedTx,
|
|
565
|
+
scanIndex: transaction.scanIndex,
|
|
566
|
+
signature: Buffer.from(signedTransaction.serializedSig).toString('base64'),
|
|
567
|
+
recoveryAmount: outputAmount.toString(),
|
|
568
|
+
});
|
|
569
|
+
if (i === req.length - 1 && transaction.coinSpecific.lastScanIndex) {
|
|
570
|
+
lastScanIndex = transaction.coinSpecific.lastScanIndex;
|
|
571
|
+
}
|
|
572
|
+
}
|
|
573
|
+
return { transactions: broadcastableTransactions, lastScanIndex };
|
|
574
|
+
}
|
|
575
|
+
/**
|
|
576
|
+
* Builds native SUI recoveries of receive addresses in batch without BitGo.
|
|
577
|
+
* Funds will be recovered to base address first. You need to initiate another sweep txn after that.
|
|
578
|
+
*
|
|
579
|
+
* @param {MPCConsolidationRecoveryOptions} params - options for consolidation recovery.
|
|
580
|
+
* @param {string} [params.startingScanIndex] - receive address index to start scanning from. default to 1 (inclusive).
|
|
581
|
+
* @param {string} [params.endingScanIndex] - receive address index to end scanning at. default to startingScanIndex + 20 (exclusive).
|
|
582
|
+
*/
|
|
583
|
+
async recoverConsolidations(params) {
|
|
584
|
+
const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;
|
|
585
|
+
const startIdx = utils_1.default.validateNonNegativeNumber(1, 'Invalid starting index to scan for addresses', params.startingScanIndex);
|
|
586
|
+
const endIdx = utils_1.default.validateNonNegativeNumber(startIdx + constants_1.DEFAULT_SCAN_FACTOR, 'Invalid ending index to scan for addresses', params.endingScanIndex);
|
|
587
|
+
if (startIdx < 1 || endIdx <= startIdx || endIdx - startIdx > 10 * constants_1.DEFAULT_SCAN_FACTOR) {
|
|
588
|
+
throw new Error(`Invalid starting or ending index to scan for addresses. startingScanIndex: ${startIdx}, endingScanIndex: ${endIdx}.`);
|
|
589
|
+
}
|
|
590
|
+
const bitgoKey = params.bitgoKey.replace(/\s/g, '');
|
|
591
|
+
const MPC = await sdk_core_1.EDDSAMethods.getInitializedMpcInstance();
|
|
592
|
+
const derivationPath = (params.seed ? (0, sdk_lib_mpc_1.getDerivationPath)(params.seed) : 'm') + '/0';
|
|
593
|
+
const derivedPublicKey = MPC.deriveUnhardened(bitgoKey, derivationPath).slice(0, 64);
|
|
594
|
+
const baseAddress = this.getAddressFromPublicKey(derivedPublicKey);
|
|
595
|
+
const consolidationTransactions = [];
|
|
596
|
+
let lastScanIndex = startIdx;
|
|
597
|
+
for (let idx = startIdx; idx < endIdx; idx++) {
|
|
598
|
+
const recoverParams = {
|
|
599
|
+
userKey: params.userKey,
|
|
600
|
+
backupKey: params.backupKey,
|
|
601
|
+
bitgoKey: params.bitgoKey,
|
|
602
|
+
walletPassphrase: params.walletPassphrase,
|
|
603
|
+
seed: params.seed,
|
|
604
|
+
tokenContractAddress: params.tokenContractAddress,
|
|
605
|
+
recoveryDestination: baseAddress,
|
|
606
|
+
startingScanIndex: idx,
|
|
607
|
+
scan: 1,
|
|
608
|
+
};
|
|
609
|
+
let recoveryTransaction;
|
|
610
|
+
try {
|
|
611
|
+
recoveryTransaction = await this.recover(recoverParams);
|
|
612
|
+
}
|
|
613
|
+
catch (e) {
|
|
614
|
+
if (e.message.startsWith('Did not find an address with sufficient funds to recover.')) {
|
|
615
|
+
lastScanIndex = idx;
|
|
616
|
+
continue;
|
|
617
|
+
}
|
|
618
|
+
throw e;
|
|
619
|
+
}
|
|
620
|
+
if (isUnsignedSweep) {
|
|
621
|
+
consolidationTransactions.push(recoveryTransaction.txRequests[0]);
|
|
622
|
+
}
|
|
623
|
+
else {
|
|
624
|
+
consolidationTransactions.push(recoveryTransaction.transactions[0]);
|
|
625
|
+
}
|
|
626
|
+
lastScanIndex = idx;
|
|
627
|
+
}
|
|
628
|
+
if (consolidationTransactions.length === 0) {
|
|
629
|
+
throw new Error(`Did not find an address with sufficient funds to recover. Please start the next scan at address index ${lastScanIndex + 1}.`);
|
|
630
|
+
}
|
|
631
|
+
if (isUnsignedSweep) {
|
|
632
|
+
// lastScanIndex will be used to inform user the last address index scanned for available funds (so they can
|
|
633
|
+
// appropriately adjust the scan range on the next iteration of consolidation recoveries). In the case of unsigned
|
|
634
|
+
// sweep consolidations, this lastScanIndex will be provided in the coinSpecific of the last txn made.
|
|
635
|
+
consolidationTransactions[consolidationTransactions.length - 1].transactions[0].unsignedTx.coinSpecific.lastScanIndex = lastScanIndex;
|
|
636
|
+
return { txRequests: consolidationTransactions };
|
|
637
|
+
}
|
|
638
|
+
return { transactions: consolidationTransactions, lastScanIndex };
|
|
639
|
+
}
|
|
640
|
+
/** inherited doc */
|
|
641
|
+
setCoinSpecificFieldsInIntent(intent, params) {
|
|
642
|
+
intent.unspents = params.unspents;
|
|
643
|
+
}
|
|
208
644
|
}
|
|
209
645
|
exports.Sui = Sui;
|
|
210
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sui.js","sourceRoot":"","sources":["../../src/sui.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  BaseCoin,\n  BaseTransaction,\n  BitGoBase,\n  EDDSAMethods,\n  InvalidAddressError,\n  KeyPair,\n  MPCAlgorithm,\n  ParsedTransaction,\n  ParseTransactionOptions as BaseParseTransactionOptions,\n  SignedTransaction,\n  SignTransactionOptions,\n  TransactionExplanation,\n  TssVerifyAddressOptions,\n  VerifyTransactionOptions,\n} from '@bitgo-beta/sdk-core';\nimport { BaseCoin as StaticsBaseCoin, coins } from '@bitgo-beta/statics';\nimport BigNumber from 'bignumber.js';\nimport { TransactionBuilderFactory, KeyPair as SuiKeyPair } from './lib';\nimport utils from './lib/utils';\nimport * as _ from 'lodash';\nimport { TransferTransaction } from './lib/transferTransaction';\n\nexport interface ExplainTransactionOptions {\n  txHex: string;\n}\n\nexport interface SuiParseTransactionOptions extends BaseParseTransactionOptions {\n  txHex: string;\n}\n\ninterface TransactionOutput {\n  address: string;\n  amount: string;\n}\n\ntype TransactionInput = TransactionOutput;\n\nexport interface SuiParsedTransaction extends ParsedTransaction {\n  // total assets being moved, including fees\n  inputs: TransactionInput[];\n\n  // where assets are moved to\n  outputs: TransactionOutput[];\n}\n\nexport type SuiTransactionExplanation = TransactionExplanation;\n\nexport class Sui extends BaseCoin {\n  protected readonly _staticsCoin: Readonly<StaticsBaseCoin>;\n  protected constructor(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>) {\n    super(bitgo);\n\n    if (!staticsCoin) {\n      throw new Error('missing required constructor parameter staticsCoin');\n    }\n\n    this._staticsCoin = staticsCoin;\n  }\n\n  static createInstance(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>): BaseCoin {\n    return new Sui(bitgo, staticsCoin);\n  }\n\n  /**\n   * Factor between the coin's base unit and its smallest subdivison\n   */\n  public getBaseFactor(): number {\n    return 1e9;\n  }\n\n  public getChain(): string {\n    return 'sui';\n  }\n\n  public getFamily(): string {\n    return 'sui';\n  }\n\n  public getFullName(): string {\n    return 'Sui';\n  }\n\n  /** @inheritDoc */\n  supportsTss(): boolean {\n    return true;\n  }\n\n  getMPCAlgorithm(): MPCAlgorithm {\n    return 'eddsa';\n  }\n\n  allowsAccountConsolidations(): boolean {\n    return true;\n  }\n\n  async verifyTransaction(params: VerifyTransactionOptions): Promise<boolean> {\n    let totalAmount = new BigNumber(0);\n    const coinConfig = coins.get(this.getChain());\n    const { txPrebuild: txPrebuild, txParams: txParams } = params;\n    const transaction = new TransferTransaction(coinConfig);\n    const rawTx = txPrebuild.txHex;\n    if (!rawTx) {\n      throw new Error('missing required tx prebuild property txHex');\n    }\n\n    transaction.fromRawTransaction(Buffer.from(rawTx, 'hex').toString('base64'));\n    const explainedTx = transaction.explainTransaction();\n\n    if (txParams.recipients && txParams.recipients.length > 0) {\n      const filteredRecipients = txParams.recipients?.map((recipient) => {\n        const filteredRecipient = _.pick(recipient, ['address', 'amount']);\n        filteredRecipient.amount = new BigNumber(filteredRecipient.amount).toFixed();\n        return filteredRecipient;\n      });\n      const filteredOutputs = explainedTx.outputs.map((output) => {\n        const filteredOutput = _.pick(output, ['address', 'amount']);\n        filteredOutput.amount = new BigNumber(filteredOutput.amount).toFixed();\n        return filteredOutput;\n      });\n\n      if (!_.isEqual(filteredOutputs, filteredRecipients)) {\n        throw new Error('Tx outputs does not match with expected txParams recipients');\n      }\n      for (const recipients of txParams.recipients) {\n        totalAmount = totalAmount.plus(recipients.amount);\n      }\n      if (!totalAmount.isEqualTo(explainedTx.outputAmount)) {\n        throw new Error('Tx total amount does not match with expected total amount field');\n      }\n    }\n    return true;\n  }\n\n  async isWalletAddress(params: TssVerifyAddressOptions): Promise<boolean> {\n    const { keychains, address: newAddress, index } = params;\n\n    if (!this.isValidAddress(newAddress)) {\n      throw new InvalidAddressError(`invalid address: ${newAddress}`);\n    }\n\n    if (!keychains) {\n      throw new Error('missing required param keychains');\n    }\n\n    for (const keychain of keychains) {\n      const MPC = await EDDSAMethods.getInitializedMpcInstance();\n      const commonKeychain = keychain.commonKeychain as string;\n\n      const derivationPath = 'm/' + index;\n      const derivedPublicKey = MPC.deriveUnhardened(commonKeychain, derivationPath).slice(0, 64);\n      const expectedAddress = this.getAddressFromPublicKey(derivedPublicKey);\n\n      if (newAddress !== expectedAddress) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  async parseTransaction(params: SuiParseTransactionOptions): Promise<SuiParsedTransaction> {\n    const transactionExplanation = await this.explainTransaction({ txHex: params.txHex });\n\n    if (!transactionExplanation) {\n      throw new Error('Invalid transaction');\n    }\n\n    const suiTransaction = transactionExplanation as SuiTransactionExplanation;\n    if (suiTransaction.outputs.length <= 0) {\n      return {\n        inputs: [],\n        outputs: [],\n      };\n    }\n\n    const senderAddress = suiTransaction.outputs[0].address;\n    const feeAmount = new BigNumber(suiTransaction.fee.fee === '' ? '0' : suiTransaction.fee.fee);\n\n    // assume 1 sender, who is also the fee payer\n    const inputs = [\n      {\n        address: senderAddress,\n        amount: new BigNumber(suiTransaction.outputAmount).plus(feeAmount).toFixed(),\n      },\n    ];\n\n    const outputs: TransactionOutput[] = suiTransaction.outputs.map((output) => {\n      return {\n        address: output.address,\n        amount: new BigNumber(output.amount).toFixed(),\n      };\n    });\n\n    return {\n      inputs,\n      outputs,\n    };\n  }\n\n  generateKeyPair(seed?: Buffer): KeyPair {\n    const keyPair = seed ? new SuiKeyPair({ seed }) : new SuiKeyPair();\n    const keys = keyPair.getKeys();\n    if (!keys.prv) {\n      throw new Error('Missing prv in key generation.');\n    }\n    return {\n      pub: keys.pub,\n      prv: keys.prv,\n    };\n  }\n\n  isValidPub(pub: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n\n  isValidPrv(prv: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n\n  isValidAddress(address: string): boolean {\n    return utils.isValidAddress(address);\n  }\n\n  signTransaction(params: SignTransactionOptions): Promise<SignedTransaction> {\n    throw new Error('Method not implemented.');\n  }\n\n  /**\n   * Explain a Sui transaction\n   * @param params\n   */\n  async explainTransaction(params: ExplainTransactionOptions): Promise<SuiTransactionExplanation> {\n    const factory = this.getBuilder();\n    let rebuiltTransaction: BaseTransaction;\n\n    try {\n      const transactionBuilder = factory.from(Buffer.from(params.txHex, 'hex').toString('base64'));\n      rebuiltTransaction = await transactionBuilder.build();\n    } catch {\n      throw new Error('Invalid transaction');\n    }\n\n    return rebuiltTransaction.explainTransaction();\n  }\n\n  private getBuilder(): TransactionBuilderFactory {\n    return new TransactionBuilderFactory(coins.get(this.getChain()));\n  }\n\n  private getAddressFromPublicKey(derivedPublicKey: string) {\n    // TODO(BG-59016) replace with account lib implementation\n    return utils.getAddressFromPublicKey(derivedPublicKey);\n  }\n\n  /** @inheritDoc */\n  async getSignablePayload(serializedTx: string): Promise<Buffer> {\n    const factory = this.getBuilder();\n    const rebuiltTransaction = await factory.from(serializedTx).build();\n    return rebuiltTransaction.signablePayload;\n  }\n}\n"]}
|
|
646
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sui.js","sourceRoot":"","sources":["../../src/sui.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  BaseBroadcastTransactionOptions,\n  BaseBroadcastTransactionResult,\n  BaseCoin,\n  BaseTransaction,\n  BitGoBase,\n  EDDSAMethods,\n  EDDSAMethodTypes,\n  Environments,\n  InvalidAddressError,\n  KeyPair,\n  MPCAlgorithm,\n  MPCRecoveryOptions,\n  MPCConsolidationRecoveryOptions,\n  MPCSweepRecoveryOptions,\n  MPCSweepTxs,\n  MPCTx,\n  MPCTxs,\n  MPCUnsignedTx,\n  ParsedTransaction,\n  ParseTransactionOptions as BaseParseTransactionOptions,\n  RecoveryTxRequest,\n  SignedTransaction,\n  SignTransactionOptions,\n  TransactionExplanation,\n  TssVerifyAddressOptions,\n  VerifyTransactionOptions,\n  PopulatedIntent,\n  PrebuildTransactionWithIntentOptions,\n} from '@bitgo-beta/sdk-core';\nimport { BaseCoin as StaticsBaseCoin, BaseNetwork, coins, SuiCoin } from '@bitgo-beta/statics';\nimport BigNumber from 'bignumber.js';\nimport {\n  KeyPair as SuiKeyPair,\n  TokenTransferTransaction,\n  TransactionBuilder,\n  TransactionBuilderFactory,\n  TransferTransaction,\n} from './lib';\nimport utils from './lib/utils';\nimport * as _ from 'lodash';\nimport { SuiObjectInfo, SuiTransactionType } from './lib/iface';\nimport {\n  DEFAULT_GAS_OVERHEAD,\n  DEFAULT_GAS_PRICE,\n  DEFAULT_SCAN_FACTOR,\n  MAX_GAS_BUDGET,\n  MAX_GAS_OBJECTS,\n  MAX_OBJECT_LIMIT,\n  TOKEN_OBJECT_LIMIT,\n} from './lib/constants';\nimport { getDerivationPath } from '@bitgo-beta/sdk-lib-mpc';\n\nexport interface ExplainTransactionOptions {\n  txHex: string;\n}\n\nexport interface SuiParseTransactionOptions extends BaseParseTransactionOptions {\n  txHex: string;\n}\n\ninterface TransactionOutput {\n  address: string;\n  amount: string;\n}\n\ntype TransactionInput = TransactionOutput;\n\nexport interface SuiParsedTransaction extends ParsedTransaction {\n  // total assets being moved, including fees\n  inputs: TransactionInput[];\n\n  // where assets are moved to\n  outputs: TransactionOutput[];\n\n  fee: BigNumber;\n}\n\nexport type SuiTransactionExplanation = TransactionExplanation;\n\nexport class Sui extends BaseCoin {\n  protected readonly _staticsCoin: Readonly<StaticsBaseCoin>;\n  protected constructor(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>) {\n    super(bitgo);\n\n    if (!staticsCoin) {\n      throw new Error('missing required constructor parameter staticsCoin');\n    }\n\n    this._staticsCoin = staticsCoin;\n  }\n\n  static createInstance(bitgo: BitGoBase, staticsCoin?: Readonly<StaticsBaseCoin>): BaseCoin {\n    return new Sui(bitgo, staticsCoin);\n  }\n\n  /**\n   * Factor between the coin's base unit and its smallest subdivison\n   */\n  public getBaseFactor(): number {\n    return 1e9;\n  }\n\n  public getChain(): string {\n    return 'sui';\n  }\n\n  public getFamily(): string {\n    return 'sui';\n  }\n\n  public getFullName(): string {\n    return 'Sui';\n  }\n\n  getNetwork(): BaseNetwork {\n    return this._staticsCoin.network;\n  }\n\n  /** @inheritDoc */\n  supportsTss(): boolean {\n    return true;\n  }\n\n  getMPCAlgorithm(): MPCAlgorithm {\n    return 'eddsa';\n  }\n\n  allowsAccountConsolidations(): boolean {\n    return true;\n  }\n\n  async verifyTransaction(params: VerifyTransactionOptions): Promise<boolean> {\n    let totalAmount = new BigNumber(0);\n    const coinConfig = coins.get(this.getChain());\n    const { txPrebuild: txPrebuild, txParams: txParams } = params;\n    const transaction = new TransferTransaction(coinConfig);\n    const rawTx = txPrebuild.txHex;\n    if (!rawTx) {\n      throw new Error('missing required tx prebuild property txHex');\n    }\n\n    transaction.fromRawTransaction(Buffer.from(rawTx, 'hex').toString('base64'));\n    const explainedTx = transaction.explainTransaction();\n\n    if (txParams.recipients && txParams.recipients.length > 0) {\n      const filteredRecipients = txParams.recipients?.map((recipient) => {\n        const filteredRecipient = _.pick(recipient, ['address', 'amount']);\n        filteredRecipient.amount = new BigNumber(filteredRecipient.amount).toFixed();\n        return filteredRecipient;\n      });\n      const filteredOutputs = explainedTx.outputs.map((output) => {\n        const filteredOutput = _.pick(output, ['address', 'amount']);\n        filteredOutput.amount = new BigNumber(filteredOutput.amount).toFixed();\n        return filteredOutput;\n      });\n\n      if (!_.isEqual(filteredOutputs, filteredRecipients)) {\n        throw new Error('Tx outputs does not match with expected txParams recipients');\n      }\n      for (const recipients of txParams.recipients) {\n        totalAmount = totalAmount.plus(recipients.amount);\n      }\n      if (!totalAmount.isEqualTo(explainedTx.outputAmount)) {\n        throw new Error('Tx total amount does not match with expected total amount field');\n      }\n    }\n    return true;\n  }\n\n  async isWalletAddress(params: TssVerifyAddressOptions): Promise<boolean> {\n    const { address: newAddress } = params;\n\n    if (!this.isValidAddress(newAddress)) {\n      throw new InvalidAddressError(`invalid address: ${newAddress}`);\n    }\n    return true;\n  }\n\n  async parseTransaction(params: SuiParseTransactionOptions): Promise<SuiParsedTransaction> {\n    const transactionExplanation = await this.explainTransaction({ txHex: params.txHex });\n\n    if (!transactionExplanation) {\n      throw new Error('Invalid transaction');\n    }\n\n    let fee = new BigNumber(0);\n\n    const suiTransaction = transactionExplanation as SuiTransactionExplanation;\n    if (suiTransaction.outputs.length <= 0) {\n      return {\n        inputs: [],\n        outputs: [],\n        fee,\n      };\n    }\n\n    const senderAddress = suiTransaction.outputs[0].address;\n    if (suiTransaction.fee.fee !== '') {\n      fee = new BigNumber(suiTransaction.fee.fee);\n    }\n\n    // assume 1 sender, who is also the fee payer\n    const inputs = [\n      {\n        address: senderAddress,\n        amount: new BigNumber(suiTransaction.outputAmount).plus(fee).toFixed(),\n      },\n    ];\n\n    const outputs: TransactionOutput[] = suiTransaction.outputs.map((output) => {\n      return {\n        address: output.address,\n        amount: new BigNumber(output.amount).toFixed(),\n      };\n    });\n\n    return {\n      inputs,\n      outputs,\n      fee,\n    };\n  }\n\n  generateKeyPair(seed?: Buffer): KeyPair {\n    const keyPair = seed ? new SuiKeyPair({ seed }) : new SuiKeyPair();\n    const keys = keyPair.getKeys();\n    if (!keys.prv) {\n      throw new Error('Missing prv in key generation.');\n    }\n    return {\n      pub: keys.pub,\n      prv: keys.prv,\n    };\n  }\n\n  isValidPub(_: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n\n  isValidPrv(_: string): boolean {\n    throw new Error('Method not implemented.');\n  }\n\n  isValidAddress(address: string): boolean {\n    return utils.isValidAddress(address);\n  }\n\n  signTransaction(_: SignTransactionOptions): Promise<SignedTransaction> {\n    throw new Error('Method not implemented.');\n  }\n\n  /**\n   * Explain a Sui transaction\n   * @param params\n   */\n  async explainTransaction(params: ExplainTransactionOptions): Promise<SuiTransactionExplanation> {\n    const factory = this.getBuilder();\n    let rebuiltTransaction: BaseTransaction;\n\n    try {\n      const transactionBuilder = factory.from(Buffer.from(params.txHex, 'hex').toString('base64'));\n      rebuiltTransaction = await transactionBuilder.build();\n    } catch {\n      throw new Error('Invalid transaction');\n    }\n\n    return rebuiltTransaction.explainTransaction();\n  }\n\n  private getBuilder(): TransactionBuilderFactory {\n    return new TransactionBuilderFactory(coins.get(this.getChain()));\n  }\n\n  private getAddressFromPublicKey(derivedPublicKey: string) {\n    // TODO(BG-59016) replace with account lib implementation\n    return utils.getAddressFromPublicKey(derivedPublicKey);\n  }\n\n  /** @inheritDoc */\n  async getSignablePayload(serializedTx: string): Promise<Buffer> {\n    const factory = this.getBuilder();\n    const rebuiltTransaction = await factory.from(serializedTx).build();\n    return rebuiltTransaction.signablePayload;\n  }\n\n  protected getPublicNodeUrl(): string {\n    return Environments[this.bitgo.getEnv()].suiNodeUrl;\n  }\n\n  protected async getBalance(owner: string, coinType?: string): Promise<string> {\n    const url = this.getPublicNodeUrl();\n    return await utils.getBalance(url, owner, coinType);\n  }\n\n  protected async getInputCoins(owner: string, coinType?: string): Promise<SuiObjectInfo[]> {\n    const url = this.getPublicNodeUrl();\n    return await utils.getInputCoins(url, owner, coinType);\n  }\n\n  protected async getFeeEstimate(txHex: string): Promise<BigNumber> {\n    const url = this.getPublicNodeUrl();\n    return await utils.getFeeEstimate(url, txHex);\n  }\n\n  /**\n   * Builds funds recovery transaction(s) without BitGo\n   *\n   * @param {MPCRecoveryOptions} params parameters needed to construct and\n   * (maybe) sign the transaction\n   *\n   * @returns {MPCTx | MPCSweepTxs} array of the serialized transaction hex strings and indices\n   * of the addresses being swept\n   */\n  async recover(params: MPCRecoveryOptions): Promise<MPCTxs | MPCSweepTxs> {\n    if (!params.bitgoKey) {\n      throw new Error('missing bitgoKey');\n    }\n    if (!params.recoveryDestination || !this.isValidAddress(params.recoveryDestination)) {\n      throw new Error('invalid recoveryDestination');\n    }\n\n    const startIdx = utils.validateNonNegativeNumber(\n      0,\n      'Invalid starting index to scan for addresses',\n      params.startingScanIndex\n    );\n    const numIterations = utils.validateNonNegativeNumber(20, 'Invalid scanning factor', params.scan);\n    const endIdx = startIdx + numIterations;\n    const bitgoKey = params.bitgoKey.replace(/\\s/g, '');\n    const MPC = await EDDSAMethods.getInitializedMpcInstance();\n\n    for (let idx = startIdx; idx < endIdx; idx++) {\n      const derivationPath = (params.seed ? getDerivationPath(params.seed) : 'm') + `/${idx}`;\n      const derivedPublicKey = MPC.deriveUnhardened(bitgoKey, derivationPath).slice(0, 64);\n      const senderAddress = this.getAddressFromPublicKey(derivedPublicKey);\n      let availableBalance = new BigNumber(0);\n      try {\n        availableBalance = new BigNumber(await this.getBalance(senderAddress));\n      } catch (e) {\n        continue;\n      }\n      if (availableBalance.minus(MAX_GAS_BUDGET).toNumber() <= 0) {\n        continue;\n      }\n\n      // check for possible token recovery, recover the token provide by user\n      if (params.tokenContractAddress) {\n        const token = utils.getSuiTokenFromAddress(params.tokenContractAddress!, this.getNetwork()) as SuiCoin;\n        if (!token) {\n          throw new Error(`Sui Token Package ID not supported.`);\n        }\n        const coinType = `${token.packageId}::${token.module}::${token.symbol}`;\n        try {\n          const availableTokenBalance = new BigNumber(await this.getBalance(senderAddress, coinType));\n          if (availableTokenBalance.toNumber() <= 0) {\n            continue;\n          }\n        } catch (e) {\n          continue;\n        }\n        return this.recoverSuiToken(params, token, senderAddress, derivationPath, derivedPublicKey, idx, bitgoKey);\n      }\n\n      let inputCoins = await this.getInputCoins(senderAddress);\n      inputCoins = inputCoins.sort((a, b) => {\n        return b.balance.minus(a.balance).toNumber();\n      });\n      if (inputCoins.length > MAX_OBJECT_LIMIT) {\n        inputCoins = inputCoins.slice(0, MAX_OBJECT_LIMIT);\n      }\n      let netAmount = inputCoins.reduce((acc, obj) => acc.plus(obj.balance), new BigNumber(0));\n      netAmount = netAmount.minus(MAX_GAS_BUDGET);\n\n      const recipients = [\n        {\n          address: params.recoveryDestination,\n          amount: netAmount.toString(),\n        },\n      ];\n\n      // first build the unsigned txn\n      const factory = new TransactionBuilderFactory(coins.get(this.getChain()));\n      const txBuilder = factory\n        .getTransferBuilder()\n        .type(SuiTransactionType.Transfer)\n        .sender(senderAddress)\n        .send(recipients)\n        .gasData({\n          owner: senderAddress,\n          price: DEFAULT_GAS_PRICE,\n          budget: MAX_GAS_BUDGET,\n          payment: inputCoins,\n        });\n\n      const tempTx = (await txBuilder.build()) as TransferTransaction;\n      const feeEstimate = await this.getFeeEstimate(tempTx.toBroadcastFormat());\n      const gasBudget = Math.trunc(feeEstimate.toNumber() * DEFAULT_GAS_OVERHEAD);\n\n      netAmount = netAmount.plus(MAX_GAS_BUDGET).minus(gasBudget);\n      recipients[0].amount = netAmount.toString();\n      txBuilder.send(recipients);\n      txBuilder.gasData({\n        owner: senderAddress,\n        price: DEFAULT_GAS_PRICE,\n        budget: gasBudget,\n        payment: inputCoins,\n      });\n\n      const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;\n      if (isUnsignedSweep) {\n        return this.buildUnsignedSweepTransaction(txBuilder, senderAddress, bitgoKey, idx, derivationPath);\n      }\n\n      await this.signRecoveryTransaction(txBuilder, params, derivationPath, derivedPublicKey, false);\n      const tx = (await txBuilder.build()) as TransferTransaction;\n      return {\n        transactions: [\n          {\n            scanIndex: idx,\n            recoveryAmount: netAmount.toString(),\n            serializedTx: tx.toBroadcastFormat(),\n            signature: Buffer.from(tx.serializedSig).toString('base64'),\n            coin: this.getChain(),\n          },\n        ],\n        lastScanIndex: idx,\n      };\n    }\n\n    throw new Error(\n      `Did not find an address with sufficient funds to recover. Please start the next scan at address index ${endIdx}. If it is token transaction, please keep sufficient Sui balance in the address for the transaction fee.`\n    );\n  }\n\n  private async recoverSuiToken(\n    params: MPCRecoveryOptions,\n    token: SuiCoin,\n    senderAddress: string,\n    derivationPath: string,\n    derivedPublicKey: string,\n    idx: number,\n    bitgoKey: string\n  ): Promise<MPCTxs | MPCSweepTxs> {\n    const coinType = `${token.packageId}::${token.module}::${token.symbol}`;\n    let tokenObjects = await this.getInputCoins(senderAddress, coinType);\n    tokenObjects = tokenObjects.sort((a, b) => {\n      return b.balance.minus(a.balance).toNumber();\n    });\n    if (tokenObjects.length > TOKEN_OBJECT_LIMIT) {\n      tokenObjects = tokenObjects.slice(0, TOKEN_OBJECT_LIMIT);\n    }\n    const netAmount = tokenObjects.reduce((acc, obj) => acc.plus(obj.balance), new BigNumber(0));\n    const recipients = [\n      {\n        address: params.recoveryDestination,\n        amount: netAmount.toString(),\n      },\n    ];\n\n    const gasAmount = new BigNumber(MAX_GAS_BUDGET);\n    let gasObjects = await this.getInputCoins(senderAddress);\n    gasObjects = utils.selectObjectsInDescOrderOfBalance(gasObjects, gasAmount);\n    if (gasObjects.length >= MAX_GAS_OBJECTS) {\n      gasObjects = gasObjects.slice(0, MAX_GAS_OBJECTS - 1);\n    }\n\n    // first build the unsigned txn\n    const factory = new TransactionBuilderFactory(token);\n    const txBuilder = factory\n      .getTokenTransferBuilder()\n      .type(SuiTransactionType.TokenTransfer)\n      .sender(senderAddress)\n      .send(recipients)\n      .inputObjects(tokenObjects)\n      .gasData({\n        owner: senderAddress,\n        price: DEFAULT_GAS_PRICE,\n        budget: MAX_GAS_BUDGET,\n        payment: gasObjects,\n      });\n\n    const tempTx = (await txBuilder.build()) as TokenTransferTransaction;\n    const feeEstimate = await this.getFeeEstimate(tempTx.toBroadcastFormat());\n    const gasBudget = Math.trunc(feeEstimate.toNumber() * DEFAULT_GAS_OVERHEAD);\n\n    txBuilder.gasData({\n      owner: senderAddress,\n      price: DEFAULT_GAS_PRICE,\n      budget: gasBudget,\n      payment: gasObjects,\n    });\n\n    const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;\n    if (isUnsignedSweep) {\n      return this.buildUnsignedSweepTransaction(txBuilder, senderAddress, bitgoKey, idx, derivationPath, token);\n    }\n\n    await this.signRecoveryTransaction(txBuilder, params, derivationPath, derivedPublicKey, true);\n    const tx = (await txBuilder.build()) as TokenTransferTransaction;\n    return {\n      transactions: [\n        {\n          scanIndex: idx,\n          recoveryAmount: netAmount.toString(),\n          serializedTx: tx.toBroadcastFormat(),\n          signature: Buffer.from(tx.serializedSig).toString('base64'),\n          coin: token.name,\n        },\n      ],\n      lastScanIndex: idx,\n    };\n  }\n\n  private async buildUnsignedSweepTransaction(\n    txBuilder: TransactionBuilder,\n    senderAddress: string,\n    bitgoKey: string,\n    index: number,\n    derivationPath: string,\n    token?: SuiCoin\n  ): Promise<MPCSweepTxs> {\n    const isTokenTransaction = !!token;\n    const unsignedTransaction = isTokenTransaction\n      ? ((await txBuilder.build()) as TokenTransferTransaction)\n      : ((await txBuilder.build()) as TransferTransaction);\n    const serializedTx = unsignedTransaction.toBroadcastFormat();\n    const serializedTxHex = Buffer.from(serializedTx, 'base64').toString('hex');\n    const parsedTx = await this.parseTransaction({ txHex: serializedTxHex });\n    const walletCoin = isTokenTransaction ? token.name : this.getChain();\n    const output = parsedTx.outputs[0];\n    const inputs = [\n      {\n        address: senderAddress,\n        valueString: output.amount,\n        value: new BigNumber(output.amount),\n      },\n    ];\n    const outputs = [\n      {\n        address: output.address,\n        valueString: output.amount,\n        coinName: walletCoin,\n      },\n    ];\n    const spendAmount = output.amount;\n    const completedParsedTx = {\n      inputs: inputs,\n      outputs: outputs,\n      spendAmount: spendAmount,\n      type: isTokenTransaction ? SuiTransactionType.TokenTransfer : SuiTransactionType.Transfer,\n    };\n    const fee = parsedTx.fee;\n    const feeInfo = { fee: fee.toNumber(), feeString: fee.toString() };\n    const coinSpecific = { commonKeychain: bitgoKey };\n    const transaction: MPCTx = {\n      serializedTx: serializedTxHex,\n      scanIndex: index,\n      coin: walletCoin,\n      signableHex: unsignedTransaction.signablePayload.toString('hex'),\n      derivationPath,\n      parsedTx: completedParsedTx,\n      feeInfo: feeInfo,\n      coinSpecific: coinSpecific,\n    };\n    const unsignedTx: MPCUnsignedTx = { unsignedTx: transaction, signatureShares: [] };\n    const transactions: MPCUnsignedTx[] = [unsignedTx];\n    const txRequest: RecoveryTxRequest = {\n      transactions: transactions,\n      walletCoin: walletCoin,\n    };\n    return { txRequests: [txRequest] };\n  }\n\n  private async signRecoveryTransaction(\n    txBuilder: TransactionBuilder,\n    params: MPCRecoveryOptions,\n    derivationPath: string,\n    derivedPublicKey: string,\n    isTokenTransaction: boolean\n  ) {\n    // TODO(BG-51092): This looks like a common part which can be extracted out too\n    const unsignedTx = isTokenTransaction\n      ? ((await txBuilder.build()) as TokenTransferTransaction)\n      : ((await txBuilder.build()) as TransferTransaction);\n    if (!params.userKey) {\n      throw new Error('missing userKey');\n    }\n    if (!params.backupKey) {\n      throw new Error('missing backupKey');\n    }\n    if (!params.walletPassphrase) {\n      throw new Error('missing wallet passphrase');\n    }\n\n    // Clean up whitespace from entered values\n    const userKey = params.userKey.replace(/\\s/g, '');\n    const backupKey = params.backupKey.replace(/\\s/g, '');\n\n    // Decrypt private keys from KeyCard values\n    let userPrv: string;\n    try {\n      userPrv = this.bitgo.decrypt({\n        input: userKey,\n        password: params.walletPassphrase,\n      });\n    } catch (e) {\n      throw new Error(`Error decrypting user keychain: ${e.message}`);\n    }\n    /** TODO BG-52419 Implement Codec for parsing */\n    const userSigningMaterial = JSON.parse(userPrv) as EDDSAMethodTypes.UserSigningMaterial;\n\n    let backupPrv: string;\n    try {\n      backupPrv = this.bitgo.decrypt({\n        input: backupKey,\n        password: params.walletPassphrase,\n      });\n    } catch (e) {\n      throw new Error(`Error decrypting backup keychain: ${e.message}`);\n    }\n    const backupSigningMaterial = JSON.parse(backupPrv) as EDDSAMethodTypes.BackupSigningMaterial;\n    /* ********************** END ***********************************/\n\n    // add signature\n    const signatureHex = await EDDSAMethods.getTSSSignature(\n      userSigningMaterial,\n      backupSigningMaterial,\n      derivationPath,\n      unsignedTx\n    );\n    txBuilder.addSignature({ pub: derivedPublicKey }, signatureHex);\n  }\n\n  async broadcastTransaction({\n    transactions,\n  }: BaseBroadcastTransactionOptions): Promise<BaseBroadcastTransactionResult> {\n    const txIds: string[] = [];\n    const url = this.getPublicNodeUrl();\n    let digest = '';\n    if (!!transactions) {\n      for (const txn of transactions) {\n        try {\n          digest = await utils.executeTransactionBlock(url, txn.serializedTx, [txn.signature!]);\n        } catch (e) {\n          throw new Error(`Failed to broadcast transaction, error: ${e.message}`);\n        }\n        txIds.push(digest);\n      }\n    }\n    return { txIds };\n  }\n\n  /** inherited doc */\n  async createBroadcastableSweepTransaction(params: MPCSweepRecoveryOptions): Promise<MPCTxs> {\n    const req = params.signatureShares;\n    const broadcastableTransactions: MPCTx[] = [];\n    let lastScanIndex = 0;\n\n    for (let i = 0; i < req.length; i++) {\n      const MPC = await EDDSAMethods.getInitializedMpcInstance();\n      const transaction = req[i].txRequest.transactions[0].unsignedTx;\n      if (!req[i].ovc || !req[i].ovc[0].eddsaSignature) {\n        throw new Error('Missing signature(s)');\n      }\n      const signature = req[i].ovc[0].eddsaSignature;\n      if (!transaction.signableHex) {\n        throw new Error('Missing signable hex');\n      }\n      const messageBuffer = Buffer.from(transaction.signableHex!, 'hex');\n      const result = MPC.verify(messageBuffer, signature);\n      if (!result) {\n        throw new Error('Invalid signature');\n      }\n      const signatureHex = Buffer.concat([Buffer.from(signature.R, 'hex'), Buffer.from(signature.sigma, 'hex')]);\n      const serializedTxBase64 = Buffer.from(transaction.serializedTx, 'hex').toString('base64');\n      const txBuilder = this.getBuilder().from(serializedTxBase64);\n      if (!transaction.coinSpecific?.commonKeychain) {\n        throw new Error('Missing common keychain');\n      }\n      const commonKeychain = transaction.coinSpecific!.commonKeychain! as string;\n      if (!transaction.derivationPath) {\n        throw new Error('Missing derivation path');\n      }\n      const derivationPath = transaction.derivationPath as string;\n      const derivedPublicKey = MPC.deriveUnhardened(commonKeychain, derivationPath).slice(0, 64);\n\n      // add combined signature from ovc\n      txBuilder.addSignature({ pub: derivedPublicKey }, signatureHex);\n      const signedTransaction = (await txBuilder.build()) as TransferTransaction;\n      const serializedTx = signedTransaction.toBroadcastFormat();\n      const outputAmount = signedTransaction.explainTransaction().outputAmount;\n\n      broadcastableTransactions.push({\n        serializedTx: serializedTx,\n        scanIndex: transaction.scanIndex,\n        signature: Buffer.from(signedTransaction.serializedSig).toString('base64'),\n        recoveryAmount: outputAmount.toString(),\n      });\n\n      if (i === req.length - 1 && transaction.coinSpecific!.lastScanIndex) {\n        lastScanIndex = transaction.coinSpecific!.lastScanIndex as number;\n      }\n    }\n\n    return { transactions: broadcastableTransactions, lastScanIndex };\n  }\n\n  /**\n   * Builds native SUI recoveries of receive addresses in batch without BitGo.\n   * Funds will be recovered to base address first. You need to initiate another sweep txn after that.\n   *\n   * @param {MPCConsolidationRecoveryOptions} params - options for consolidation recovery.\n   * @param {string} [params.startingScanIndex] - receive address index to start scanning from. default to 1 (inclusive).\n   * @param {string} [params.endingScanIndex] - receive address index to end scanning at. default to startingScanIndex + 20 (exclusive).\n   */\n  async recoverConsolidations(params: MPCConsolidationRecoveryOptions): Promise<MPCTxs | MPCSweepTxs> {\n    const isUnsignedSweep = !params.userKey && !params.backupKey && !params.walletPassphrase;\n    const startIdx = utils.validateNonNegativeNumber(\n      1,\n      'Invalid starting index to scan for addresses',\n      params.startingScanIndex\n    );\n    const endIdx = utils.validateNonNegativeNumber(\n      startIdx + DEFAULT_SCAN_FACTOR,\n      'Invalid ending index to scan for addresses',\n      params.endingScanIndex\n    );\n\n    if (startIdx < 1 || endIdx <= startIdx || endIdx - startIdx > 10 * DEFAULT_SCAN_FACTOR) {\n      throw new Error(\n        `Invalid starting or ending index to scan for addresses. startingScanIndex: ${startIdx}, endingScanIndex: ${endIdx}.`\n      );\n    }\n\n    const bitgoKey = params.bitgoKey.replace(/\\s/g, '');\n    const MPC = await EDDSAMethods.getInitializedMpcInstance();\n    const derivationPath = (params.seed ? getDerivationPath(params.seed) : 'm') + '/0';\n    const derivedPublicKey = MPC.deriveUnhardened(bitgoKey, derivationPath).slice(0, 64);\n    const baseAddress = this.getAddressFromPublicKey(derivedPublicKey);\n\n    const consolidationTransactions: any[] = [];\n    let lastScanIndex = startIdx;\n    for (let idx = startIdx; idx < endIdx; idx++) {\n      const recoverParams = {\n        userKey: params.userKey,\n        backupKey: params.backupKey,\n        bitgoKey: params.bitgoKey,\n        walletPassphrase: params.walletPassphrase,\n        seed: params.seed,\n        tokenContractAddress: params.tokenContractAddress,\n        recoveryDestination: baseAddress,\n        startingScanIndex: idx,\n        scan: 1,\n      };\n\n      let recoveryTransaction: MPCTxs | MPCSweepTxs;\n      try {\n        recoveryTransaction = await this.recover(recoverParams);\n      } catch (e) {\n        if (e.message.startsWith('Did not find an address with sufficient funds to recover.')) {\n          lastScanIndex = idx;\n          continue;\n        }\n        throw e;\n      }\n\n      if (isUnsignedSweep) {\n        consolidationTransactions.push((recoveryTransaction as MPCSweepTxs).txRequests[0]);\n      } else {\n        consolidationTransactions.push((recoveryTransaction as MPCTxs).transactions[0]);\n      }\n      lastScanIndex = idx;\n    }\n\n    if (consolidationTransactions.length === 0) {\n      throw new Error(\n        `Did not find an address with sufficient funds to recover. Please start the next scan at address index ${\n          lastScanIndex + 1\n        }.`\n      );\n    }\n\n    if (isUnsignedSweep) {\n      // lastScanIndex will be used to inform user the last address index scanned for available funds (so they can\n      // appropriately adjust the scan range on the next iteration of consolidation recoveries). In the case of unsigned\n      // sweep consolidations, this lastScanIndex will be provided in the coinSpecific of the last txn made.\n      consolidationTransactions[\n        consolidationTransactions.length - 1\n      ].transactions[0].unsignedTx.coinSpecific.lastScanIndex = lastScanIndex;\n      return { txRequests: consolidationTransactions };\n    }\n\n    return { transactions: consolidationTransactions, lastScanIndex };\n  }\n\n  /** inherited doc */\n  setCoinSpecificFieldsInIntent(intent: PopulatedIntent, params: PrebuildTransactionWithIntentOptions): void {\n    intent.unspents = params.unspents;\n  }\n}\n"]}
|