@bitgo-beta/sdk-coin-flrp 1.0.1-beta.21 → 1.0.1-beta.211
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/flrp.d.ts +82 -61
- package/dist/src/flrp.d.ts.map +1 -1
- package/dist/src/flrp.js +293 -134
- package/dist/src/lib/atomicTransactionBuilder.d.ts +42 -6
- package/dist/src/lib/atomicTransactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/atomicTransactionBuilder.js +233 -29
- package/dist/src/lib/constants.d.ts +160 -1
- package/dist/src/lib/constants.d.ts.map +1 -1
- package/dist/src/lib/constants.js +213 -3
- package/dist/src/lib/delegatorTxBuilder.d.ts +58 -0
- package/dist/src/lib/delegatorTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/delegatorTxBuilder.js +224 -0
- package/dist/src/lib/exportInCTxBuilder.d.ts +1 -1
- package/dist/src/lib/exportInCTxBuilder.d.ts.map +1 -1
- package/dist/src/lib/exportInCTxBuilder.js +46 -17
- package/dist/src/lib/exportInPTxBuilder.d.ts +1 -1
- package/dist/src/lib/exportInPTxBuilder.d.ts.map +1 -1
- package/dist/src/lib/exportInPTxBuilder.js +70 -6
- package/dist/src/lib/iface.d.ts +52 -1
- package/dist/src/lib/iface.d.ts.map +1 -1
- package/dist/src/lib/iface.js +1 -1
- package/dist/src/lib/importInCTxBuilder.d.ts +67 -0
- package/dist/src/lib/importInCTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/importInCTxBuilder.js +403 -0
- package/dist/src/lib/importInPTxBuilder.d.ts +73 -0
- package/dist/src/lib/importInPTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/importInPTxBuilder.js +464 -0
- package/dist/src/lib/index.d.ts +7 -0
- package/dist/src/lib/index.d.ts.map +1 -1
- package/dist/src/lib/index.js +15 -2
- package/dist/src/lib/keyPair.d.ts +4 -4
- package/dist/src/lib/keyPair.d.ts.map +1 -1
- package/dist/src/lib/keyPair.js +12 -10
- package/dist/src/lib/permissionlessValidatorTxBuilder.d.ts +81 -0
- package/dist/src/lib/permissionlessValidatorTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/permissionlessValidatorTxBuilder.js +248 -0
- package/dist/src/lib/transaction.d.ts +111 -0
- package/dist/src/lib/transaction.d.ts.map +1 -0
- package/dist/src/lib/transaction.js +322 -0
- package/dist/src/lib/transactionBuilder.d.ts +85 -0
- package/dist/src/lib/transactionBuilder.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilder.js +167 -0
- package/dist/src/lib/transactionBuilderFactory.d.ts +37 -0
- package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -0
- package/dist/src/lib/transactionBuilderFactory.js +91 -0
- package/dist/src/lib/types.d.ts +78 -0
- package/dist/src/lib/types.d.ts.map +1 -0
- package/dist/src/lib/types.js +5 -0
- package/dist/src/lib/utils.d.ts +81 -19
- package/dist/src/lib/utils.d.ts.map +1 -1
- package/dist/src/lib/utils.js +286 -93
- package/dist/src/lib/validatorTxBuilder.d.ts +40 -0
- package/dist/src/lib/validatorTxBuilder.d.ts.map +1 -0
- package/dist/src/lib/validatorTxBuilder.js +180 -0
- package/dist/test/resources/account.d.ts +49 -0
- package/dist/test/resources/account.d.ts.map +1 -0
- package/dist/test/resources/account.js +52 -0
- package/dist/test/unit/delegatorTxBuilder.test.d.ts +2 -0
- package/dist/test/unit/delegatorTxBuilder.test.d.ts.map +1 -0
- package/dist/test/unit/delegatorTxBuilder.test.js +233 -0
- package/dist/test/unit/lib/atomicTransactionBuilder.js +37 -11
- package/dist/test/unit/lib/exportInCTxBuilder.d.ts +2 -0
- package/dist/test/unit/lib/exportInCTxBuilder.d.ts.map +1 -0
- package/dist/test/unit/lib/exportInCTxBuilder.js +584 -0
- package/dist/test/unit/lib/exportInPTxBuilder.d.ts +2 -0
- package/dist/test/unit/lib/exportInPTxBuilder.d.ts.map +1 -0
- package/dist/test/unit/lib/exportInPTxBuilder.js +377 -0
- package/dist/test/unit/lib/importInCTxBuilder.d.ts +2 -0
- package/dist/test/unit/lib/importInCTxBuilder.d.ts.map +1 -0
- package/dist/test/unit/lib/importInCTxBuilder.js +258 -0
- package/dist/test/unit/lib/importInPTxBuilder.d.ts +2 -0
- package/dist/test/unit/lib/importInPTxBuilder.d.ts.map +1 -0
- package/dist/test/unit/lib/importInPTxBuilder.js +501 -0
- package/dist/test/unit/lib/keyPair.d.ts +2 -0
- package/dist/test/unit/lib/keyPair.d.ts.map +1 -0
- package/dist/test/unit/lib/keyPair.js +158 -0
- package/dist/test/unit/lib/transaction.d.ts +2 -0
- package/dist/test/unit/lib/transaction.d.ts.map +1 -0
- package/dist/test/unit/lib/transaction.js +460 -0
- package/dist/test/unit/lib/utils.js +176 -43
- package/dist/test/unit/permissionlessValidatorTxBuilder.test.d.ts +2 -0
- package/dist/test/unit/permissionlessValidatorTxBuilder.test.d.ts.map +1 -0
- package/dist/test/unit/permissionlessValidatorTxBuilder.test.js +271 -0
- package/dist/test/unit/transactionBuilder.test.d.ts +2 -0
- package/dist/test/unit/transactionBuilder.test.d.ts.map +1 -0
- package/dist/test/unit/transactionBuilder.test.js +114 -0
- package/dist/test/unit/validatorTxBuilder.test.d.ts +2 -0
- package/dist/test/unit/validatorTxBuilder.test.d.ts.map +1 -0
- package/dist/test/unit/validatorTxBuilder.test.js +293 -0
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +15 -13
- package/.eslintignore +0 -5
- package/.eslintrc.json +0 -7
- package/.mocharc.yml +0 -8
- package/CHANGELOG.md +0 -0
- package/dist/test/unit/lib/exportTxBuilder.d.ts +0 -2
- package/dist/test/unit/lib/exportTxBuilder.d.ts.map +0 -1
- package/dist/test/unit/lib/exportTxBuilder.js +0 -45
|
@@ -0,0 +1,322 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
3
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
4
|
+
};
|
|
5
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
6
|
+
exports.Transaction = void 0;
|
|
7
|
+
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
8
|
+
const buffer_1 = require("buffer");
|
|
9
|
+
const iface_1 = require("./iface");
|
|
10
|
+
const utils_1 = __importDefault(require("./utils"));
|
|
11
|
+
const constants_1 = require("./constants");
|
|
12
|
+
/**
|
|
13
|
+
* Flare P-chain transaction implementation using FlareJS
|
|
14
|
+
* Based on AVAX transaction patterns adapted for Flare network
|
|
15
|
+
*/
|
|
16
|
+
class Transaction extends sdk_core_1.BaseTransaction {
|
|
17
|
+
constructor(coinConfig) {
|
|
18
|
+
super(coinConfig);
|
|
19
|
+
this._threshold = 2;
|
|
20
|
+
this._locktime = BigInt(0);
|
|
21
|
+
this._fromAddresses = [];
|
|
22
|
+
this._rewardAddresses = [];
|
|
23
|
+
this._utxos = [];
|
|
24
|
+
this._fee = {};
|
|
25
|
+
this._memo = new Uint8Array(); // FlareJS memo field
|
|
26
|
+
this._network = coinConfig.network;
|
|
27
|
+
this._assetId = constants_1.FLR_ASSET_ID; // Default FLR asset
|
|
28
|
+
this._blockchainID = this._network.blockchainID || '';
|
|
29
|
+
this._networkID = this._network.networkID || 0;
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* Get the base transaction from FlareJS UnsignedTx
|
|
33
|
+
* TODO: Implement proper FlareJS transaction extraction
|
|
34
|
+
*/
|
|
35
|
+
get flareTransaction() {
|
|
36
|
+
return this._flareTransaction;
|
|
37
|
+
}
|
|
38
|
+
get signature() {
|
|
39
|
+
if (this.credentials.length === 0) {
|
|
40
|
+
return [];
|
|
41
|
+
}
|
|
42
|
+
// TODO: Extract signatures from FlareJS credentials
|
|
43
|
+
// For now, return placeholder
|
|
44
|
+
return [];
|
|
45
|
+
}
|
|
46
|
+
get credentials() {
|
|
47
|
+
// TODO: Extract credentials from FlareJS transaction
|
|
48
|
+
// For now, return empty array
|
|
49
|
+
return [];
|
|
50
|
+
}
|
|
51
|
+
get hasCredentials() {
|
|
52
|
+
return this.credentials !== undefined && this.credentials.length > 0;
|
|
53
|
+
}
|
|
54
|
+
/** @inheritdoc */
|
|
55
|
+
canSign({ key }) {
|
|
56
|
+
// TODO: Implement proper signing validation for FlareJS
|
|
57
|
+
return true;
|
|
58
|
+
}
|
|
59
|
+
/**
|
|
60
|
+
* Sign a Flare transaction using FlareJS
|
|
61
|
+
* @param {KeyPair} keyPair
|
|
62
|
+
*/
|
|
63
|
+
async sign(keyPair) {
|
|
64
|
+
const prv = keyPair.getPrivateKey();
|
|
65
|
+
if (!prv) {
|
|
66
|
+
throw new sdk_core_1.SigningError('Missing private key');
|
|
67
|
+
}
|
|
68
|
+
if (!this.flareTransaction) {
|
|
69
|
+
throw new sdk_core_1.InvalidTransactionError('empty transaction to sign');
|
|
70
|
+
}
|
|
71
|
+
if (!this.hasCredentials) {
|
|
72
|
+
throw new sdk_core_1.InvalidTransactionError('empty credentials to sign');
|
|
73
|
+
}
|
|
74
|
+
// TODO: Implement FlareJS signing process
|
|
75
|
+
// This will involve:
|
|
76
|
+
// 1. Creating FlareJS signature using private key
|
|
77
|
+
// 2. Attaching signature to appropriate credential
|
|
78
|
+
// 3. Updating transaction with signed credentials
|
|
79
|
+
throw new Error('FlareJS signing not yet implemented - placeholder');
|
|
80
|
+
}
|
|
81
|
+
/**
|
|
82
|
+
* Set memo from string
|
|
83
|
+
* @param {string} memo - Memo text
|
|
84
|
+
*/
|
|
85
|
+
setMemo(memo) {
|
|
86
|
+
this._memo = utils_1.default.stringToBytes(memo);
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Set memo from various formats
|
|
90
|
+
* @param {string | Record<string, unknown> | Uint8Array} memo - Memo data
|
|
91
|
+
*/
|
|
92
|
+
setMemoData(memo) {
|
|
93
|
+
this._memo = utils_1.default.createMemoBytes(memo);
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Get memo as bytes (FlareJS format)
|
|
97
|
+
* @returns {Uint8Array} Memo bytes
|
|
98
|
+
*/
|
|
99
|
+
getMemoBytes() {
|
|
100
|
+
return this._memo;
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* Get memo as string
|
|
104
|
+
* @returns {string} Memo string
|
|
105
|
+
*/
|
|
106
|
+
getMemoString() {
|
|
107
|
+
return utils_1.default.parseMemoBytes(this._memo);
|
|
108
|
+
}
|
|
109
|
+
/**
|
|
110
|
+
* Check if transaction has memo
|
|
111
|
+
* @returns {boolean} Whether memo exists and is not empty
|
|
112
|
+
*/
|
|
113
|
+
hasMemo() {
|
|
114
|
+
return this._memo.length > 0;
|
|
115
|
+
}
|
|
116
|
+
toHexString(byteArray) {
|
|
117
|
+
return buffer_1.Buffer.from(byteArray).toString(constants_1.HEX_ENCODING);
|
|
118
|
+
}
|
|
119
|
+
/** @inheritdoc */
|
|
120
|
+
toBroadcastFormat() {
|
|
121
|
+
if (!this.flareTransaction) {
|
|
122
|
+
throw new sdk_core_1.InvalidTransactionError('Empty transaction data');
|
|
123
|
+
}
|
|
124
|
+
// TODO: Implement FlareJS transaction serialization
|
|
125
|
+
// For now, return placeholder
|
|
126
|
+
return constants_1.FLARE_TX_HEX_PLACEHOLDER;
|
|
127
|
+
}
|
|
128
|
+
toJson() {
|
|
129
|
+
if (!this.flareTransaction) {
|
|
130
|
+
throw new sdk_core_1.InvalidTransactionError('Empty transaction data');
|
|
131
|
+
}
|
|
132
|
+
return {
|
|
133
|
+
id: this.id,
|
|
134
|
+
inputs: this.inputs,
|
|
135
|
+
fromAddresses: this.fromAddresses,
|
|
136
|
+
threshold: this._threshold,
|
|
137
|
+
locktime: this._locktime.toString(),
|
|
138
|
+
type: this.type,
|
|
139
|
+
signatures: this.signature,
|
|
140
|
+
outputs: this.outputs,
|
|
141
|
+
changeOutputs: this.changeOutputs,
|
|
142
|
+
sourceChain: this._network.blockchainID,
|
|
143
|
+
destinationChain: this._network.cChainBlockchainID,
|
|
144
|
+
memo: this.getMemoString(), // Include memo in JSON representation
|
|
145
|
+
};
|
|
146
|
+
}
|
|
147
|
+
setTransaction(tx) {
|
|
148
|
+
this._flareTransaction = tx;
|
|
149
|
+
}
|
|
150
|
+
/**
|
|
151
|
+
* Set the transaction type
|
|
152
|
+
* @param {TransactionType} transactionType The transaction type to be set
|
|
153
|
+
*/
|
|
154
|
+
setTransactionType(transactionType) {
|
|
155
|
+
const supportedTypes = [
|
|
156
|
+
sdk_core_1.TransactionType.Export,
|
|
157
|
+
sdk_core_1.TransactionType.Import,
|
|
158
|
+
sdk_core_1.TransactionType.AddValidator,
|
|
159
|
+
sdk_core_1.TransactionType.AddDelegator,
|
|
160
|
+
sdk_core_1.TransactionType.AddPermissionlessValidator,
|
|
161
|
+
sdk_core_1.TransactionType.AddPermissionlessDelegator,
|
|
162
|
+
];
|
|
163
|
+
if (!supportedTypes.includes(transactionType)) {
|
|
164
|
+
throw new Error(`Transaction type ${transactionType} is not supported`);
|
|
165
|
+
}
|
|
166
|
+
this._type = transactionType;
|
|
167
|
+
}
|
|
168
|
+
/**
|
|
169
|
+
* Returns the portion of the transaction that needs to be signed in Buffer format.
|
|
170
|
+
* Only needed for coins that support adding signatures directly (e.g. TSS).
|
|
171
|
+
*/
|
|
172
|
+
get signablePayload() {
|
|
173
|
+
if (!this.flareTransaction) {
|
|
174
|
+
throw new sdk_core_1.InvalidTransactionError('Empty transaction for signing');
|
|
175
|
+
}
|
|
176
|
+
// TODO: Implement FlareJS signable payload extraction
|
|
177
|
+
// For now, return placeholder
|
|
178
|
+
return buffer_1.Buffer.from(constants_1.FLARE_SIGNABLE_PAYLOAD);
|
|
179
|
+
}
|
|
180
|
+
get id() {
|
|
181
|
+
if (!this.flareTransaction) {
|
|
182
|
+
throw new sdk_core_1.InvalidTransactionError('Empty transaction for ID generation');
|
|
183
|
+
}
|
|
184
|
+
// TODO: Implement FlareJS transaction ID generation
|
|
185
|
+
// For now, return placeholder
|
|
186
|
+
return constants_1.FLARE_TRANSACTION_ID_PLACEHOLDER;
|
|
187
|
+
}
|
|
188
|
+
get fromAddresses() {
|
|
189
|
+
return this._fromAddresses.map((address) => {
|
|
190
|
+
// TODO: Format addresses using FlareJS utilities
|
|
191
|
+
return address;
|
|
192
|
+
});
|
|
193
|
+
}
|
|
194
|
+
get rewardAddresses() {
|
|
195
|
+
return this._rewardAddresses.map((address) => {
|
|
196
|
+
// TODO: Format addresses using FlareJS utilities
|
|
197
|
+
return address;
|
|
198
|
+
});
|
|
199
|
+
}
|
|
200
|
+
/**
|
|
201
|
+
* Get the list of outputs. Amounts are expressed in absolute value.
|
|
202
|
+
*/
|
|
203
|
+
get outputs() {
|
|
204
|
+
switch (this.type) {
|
|
205
|
+
case sdk_core_1.TransactionType.Export:
|
|
206
|
+
// TODO: Extract export outputs from FlareJS transaction
|
|
207
|
+
return [];
|
|
208
|
+
case sdk_core_1.TransactionType.Import:
|
|
209
|
+
// TODO: Extract import outputs from FlareJS transaction
|
|
210
|
+
return [];
|
|
211
|
+
case sdk_core_1.TransactionType.AddValidator:
|
|
212
|
+
case sdk_core_1.TransactionType.AddPermissionlessValidator:
|
|
213
|
+
// TODO: Extract validator outputs from FlareJS transaction
|
|
214
|
+
return [
|
|
215
|
+
{
|
|
216
|
+
address: this._nodeID || constants_1.PLACEHOLDER_NODE_ID,
|
|
217
|
+
value: this._stakeAmount?.toString() || '0',
|
|
218
|
+
},
|
|
219
|
+
];
|
|
220
|
+
case sdk_core_1.TransactionType.AddDelegator:
|
|
221
|
+
case sdk_core_1.TransactionType.AddPermissionlessDelegator:
|
|
222
|
+
// TODO: Extract delegator outputs from FlareJS transaction
|
|
223
|
+
return [
|
|
224
|
+
{
|
|
225
|
+
address: this._nodeID || constants_1.PLACEHOLDER_NODE_ID,
|
|
226
|
+
value: this._stakeAmount?.toString() || '0',
|
|
227
|
+
},
|
|
228
|
+
];
|
|
229
|
+
default:
|
|
230
|
+
return [];
|
|
231
|
+
}
|
|
232
|
+
}
|
|
233
|
+
get fee() {
|
|
234
|
+
return { fee: '0', ...this._fee };
|
|
235
|
+
}
|
|
236
|
+
get changeOutputs() {
|
|
237
|
+
// TODO: Extract change outputs from FlareJS transaction
|
|
238
|
+
// For now, return empty array
|
|
239
|
+
return [];
|
|
240
|
+
}
|
|
241
|
+
get inputs() {
|
|
242
|
+
// TODO: Extract inputs from FlareJS transaction
|
|
243
|
+
// For now, return placeholder based on UTXOs
|
|
244
|
+
return this._utxos.map((utxo) => ({
|
|
245
|
+
id: utxo.txid + iface_1.INPUT_SEPARATOR + utxo.outputidx,
|
|
246
|
+
address: this.fromAddresses.sort().join(iface_1.ADDRESS_SEPARATOR),
|
|
247
|
+
value: utxo.amount,
|
|
248
|
+
}));
|
|
249
|
+
}
|
|
250
|
+
/**
|
|
251
|
+
* Flare wrapper to create signature and return it for credentials
|
|
252
|
+
* @param prv
|
|
253
|
+
* @return hexstring
|
|
254
|
+
*/
|
|
255
|
+
createSignature(prv) {
|
|
256
|
+
// TODO: Implement FlareJS signature creation
|
|
257
|
+
// This should use FlareJS signing utilities
|
|
258
|
+
const signval = utils_1.default.createSignature(this._network, this.signablePayload, prv);
|
|
259
|
+
return signval.toString(constants_1.HEX_ENCODING);
|
|
260
|
+
}
|
|
261
|
+
/**
|
|
262
|
+
* Check if transaction is for C-chain (cross-chain)
|
|
263
|
+
*/
|
|
264
|
+
get isTransactionForCChain() {
|
|
265
|
+
return this.type === sdk_core_1.TransactionType.Export || this.type === sdk_core_1.TransactionType.Import;
|
|
266
|
+
}
|
|
267
|
+
/** @inheritdoc */
|
|
268
|
+
explainTransaction() {
|
|
269
|
+
const txJson = this.toJson();
|
|
270
|
+
const displayOrder = [...constants_1.DISPLAY_ORDER_BASE];
|
|
271
|
+
// Add memo to display order if present
|
|
272
|
+
if (this.hasMemo()) {
|
|
273
|
+
displayOrder.push(constants_1.MEMO_FIELD);
|
|
274
|
+
}
|
|
275
|
+
// Calculate total output amount
|
|
276
|
+
const outputAmount = txJson.outputs
|
|
277
|
+
.reduce((sum, output) => {
|
|
278
|
+
return sum + BigInt(output.value || '0');
|
|
279
|
+
}, BigInt(0))
|
|
280
|
+
.toString();
|
|
281
|
+
// Calculate total change amount
|
|
282
|
+
const changeAmount = txJson.changeOutputs
|
|
283
|
+
.reduce((sum, output) => {
|
|
284
|
+
return sum + BigInt(output.value || '0');
|
|
285
|
+
}, BigInt(0))
|
|
286
|
+
.toString();
|
|
287
|
+
let rewardAddresses;
|
|
288
|
+
const stakingTypes = [
|
|
289
|
+
sdk_core_1.TransactionType.AddValidator,
|
|
290
|
+
sdk_core_1.TransactionType.AddDelegator,
|
|
291
|
+
sdk_core_1.TransactionType.AddPermissionlessValidator,
|
|
292
|
+
sdk_core_1.TransactionType.AddPermissionlessDelegator,
|
|
293
|
+
];
|
|
294
|
+
if (stakingTypes.includes(txJson.type)) {
|
|
295
|
+
rewardAddresses = this.rewardAddresses;
|
|
296
|
+
displayOrder.splice(6, 0, constants_1.REWARD_ADDRESSES_FIELD);
|
|
297
|
+
}
|
|
298
|
+
// Add cross-chain information for export/import
|
|
299
|
+
if (this.isTransactionForCChain) {
|
|
300
|
+
displayOrder.push(constants_1.SOURCE_CHAIN_FIELD, constants_1.DESTINATION_CHAIN_FIELD);
|
|
301
|
+
}
|
|
302
|
+
const explanation = {
|
|
303
|
+
displayOrder,
|
|
304
|
+
id: txJson.id,
|
|
305
|
+
inputs: txJson.inputs,
|
|
306
|
+
outputs: txJson.outputs.map((o) => ({ address: o.address, amount: o.value })),
|
|
307
|
+
outputAmount,
|
|
308
|
+
changeOutputs: txJson.changeOutputs.map((o) => ({ address: o.address, amount: o.value })),
|
|
309
|
+
changeAmount,
|
|
310
|
+
rewardAddresses,
|
|
311
|
+
fee: this.fee,
|
|
312
|
+
type: txJson.type,
|
|
313
|
+
};
|
|
314
|
+
// Add memo to explanation if present
|
|
315
|
+
if (this.hasMemo()) {
|
|
316
|
+
explanation.memo = this.getMemoString();
|
|
317
|
+
}
|
|
318
|
+
return explanation;
|
|
319
|
+
}
|
|
320
|
+
}
|
|
321
|
+
exports.Transaction = Transaction;
|
|
322
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transaction.js","sourceRoot":"","sources":["../../../src/lib/transaction.ts"],"names":[],"mappings":"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","sourcesContent":["import { UnsignedTx, Credential } from '@flarenetwork/flarejs';\nimport {\n  BaseKey,\n  BaseTransaction,\n  Entry,\n  InvalidTransactionError,\n  SigningError,\n  TransactionFee,\n  TransactionType,\n} from '@bitgo-beta/sdk-core';\nimport { FlareNetwork, BaseCoin as CoinConfig } from '@bitgo-beta/statics';\nimport { Buffer } from 'buffer';\nimport {\n  ADDRESS_SEPARATOR,\n  DecodedUtxoObj,\n  INPUT_SEPARATOR,\n  TransactionExplanation,\n  Tx,\n  TxData,\n  FlrpEntry,\n} from './iface';\nimport { KeyPair } from './keyPair';\nimport utils from './utils';\nimport {\n  FLR_ASSET_ID,\n  FLARE_TX_HEX_PLACEHOLDER,\n  FLARE_SIGNABLE_PAYLOAD,\n  FLARE_TRANSACTION_ID_PLACEHOLDER,\n  PLACEHOLDER_NODE_ID,\n  HEX_ENCODING,\n  MEMO_FIELD,\n  DISPLAY_ORDER_BASE,\n  REWARD_ADDRESSES_FIELD,\n  SOURCE_CHAIN_FIELD,\n  DESTINATION_CHAIN_FIELD,\n} from './constants';\n\n/**\n * Flare P-chain transaction implementation using FlareJS\n * Based on AVAX transaction patterns adapted for Flare network\n */\nexport class Transaction extends BaseTransaction {\n  protected _flareTransaction: Tx;\n  public _type: TransactionType;\n  public _network: FlareNetwork;\n  public _networkID: number;\n  public _assetId: string;\n  public _blockchainID: string;\n  public _nodeID: string;\n  public _startTime: bigint;\n  public _endTime: bigint;\n  public _stakeAmount: bigint;\n  public _threshold = 2;\n  public _locktime = BigInt(0);\n  public _fromAddresses: string[] = [];\n  public _rewardAddresses: string[] = [];\n  public _utxos: DecodedUtxoObj[] = [];\n  public _to: string[];\n  public _fee: Partial<TransactionFee> = {};\n  public _blsPublicKey: string;\n  public _blsSignature: string;\n  public _memo: Uint8Array = new Uint8Array(); // FlareJS memo field\n\n  constructor(coinConfig: Readonly<CoinConfig>) {\n    super(coinConfig);\n    this._network = coinConfig.network as FlareNetwork;\n    this._assetId = FLR_ASSET_ID; // Default FLR asset\n    this._blockchainID = this._network.blockchainID || '';\n    this._networkID = this._network.networkID || 0;\n  }\n\n  /**\n   * Get the base transaction from FlareJS UnsignedTx\n   * TODO: Implement proper FlareJS transaction extraction\n   */\n  get flareTransaction(): UnsignedTx {\n    return this._flareTransaction as UnsignedTx;\n  }\n\n  get signature(): string[] {\n    if (this.credentials.length === 0) {\n      return [];\n    }\n    // TODO: Extract signatures from FlareJS credentials\n    // For now, return placeholder\n    return [];\n  }\n\n  get credentials(): Credential[] {\n    // TODO: Extract credentials from FlareJS transaction\n    // For now, return empty array\n    return [];\n  }\n\n  get hasCredentials(): boolean {\n    return this.credentials !== undefined && this.credentials.length > 0;\n  }\n\n  /** @inheritdoc */\n  canSign({ key }: BaseKey): boolean {\n    // TODO: Implement proper signing validation for FlareJS\n    return true;\n  }\n\n  /**\n   * Sign a Flare transaction using FlareJS\n   * @param {KeyPair} keyPair\n   */\n  async sign(keyPair: KeyPair): Promise<void> {\n    const prv = keyPair.getPrivateKey() as Buffer;\n\n    if (!prv) {\n      throw new SigningError('Missing private key');\n    }\n\n    if (!this.flareTransaction) {\n      throw new InvalidTransactionError('empty transaction to sign');\n    }\n\n    if (!this.hasCredentials) {\n      throw new InvalidTransactionError('empty credentials to sign');\n    }\n\n    // TODO: Implement FlareJS signing process\n    // This will involve:\n    // 1. Creating FlareJS signature using private key\n    // 2. Attaching signature to appropriate credential\n    // 3. Updating transaction with signed credentials\n\n    throw new Error('FlareJS signing not yet implemented - placeholder');\n  }\n\n  /**\n   * Set memo from string\n   * @param {string} memo - Memo text\n   */\n  setMemo(memo: string): void {\n    this._memo = utils.stringToBytes(memo);\n  }\n\n  /**\n   * Set memo from various formats\n   * @param {string | Record<string, unknown> | Uint8Array} memo - Memo data\n   */\n  setMemoData(memo: string | Record<string, unknown> | Uint8Array): void {\n    this._memo = utils.createMemoBytes(memo);\n  }\n\n  /**\n   * Get memo as bytes (FlareJS format)\n   * @returns {Uint8Array} Memo bytes\n   */\n  getMemoBytes(): Uint8Array {\n    return this._memo;\n  }\n\n  /**\n   * Get memo as string\n   * @returns {string} Memo string\n   */\n  getMemoString(): string {\n    return utils.parseMemoBytes(this._memo);\n  }\n\n  /**\n   * Check if transaction has memo\n   * @returns {boolean} Whether memo exists and is not empty\n   */\n  hasMemo(): boolean {\n    return this._memo.length > 0;\n  }\n\n  toHexString(byteArray: Uint8Array): string {\n    return Buffer.from(byteArray).toString(HEX_ENCODING);\n  }\n\n  /** @inheritdoc */\n  toBroadcastFormat(): string {\n    if (!this.flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction data');\n    }\n\n    // TODO: Implement FlareJS transaction serialization\n    // For now, return placeholder\n    return FLARE_TX_HEX_PLACEHOLDER;\n  }\n\n  toJson(): TxData {\n    if (!this.flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction data');\n    }\n\n    return {\n      id: this.id,\n      inputs: this.inputs,\n      fromAddresses: this.fromAddresses,\n      threshold: this._threshold,\n      locktime: this._locktime.toString(),\n      type: this.type,\n      signatures: this.signature,\n      outputs: this.outputs,\n      changeOutputs: this.changeOutputs,\n      sourceChain: this._network.blockchainID,\n      destinationChain: this._network.cChainBlockchainID,\n      memo: this.getMemoString(), // Include memo in JSON representation\n    };\n  }\n\n  setTransaction(tx: Tx): void {\n    this._flareTransaction = tx;\n  }\n\n  /**\n   * Set the transaction type\n   * @param {TransactionType} transactionType The transaction type to be set\n   */\n  setTransactionType(transactionType: TransactionType): void {\n    const supportedTypes = [\n      TransactionType.Export,\n      TransactionType.Import,\n      TransactionType.AddValidator,\n      TransactionType.AddDelegator,\n      TransactionType.AddPermissionlessValidator,\n      TransactionType.AddPermissionlessDelegator,\n    ];\n\n    if (!supportedTypes.includes(transactionType)) {\n      throw new Error(`Transaction type ${transactionType} is not supported`);\n    }\n    this._type = transactionType;\n  }\n\n  /**\n   * Returns the portion of the transaction that needs to be signed in Buffer format.\n   * Only needed for coins that support adding signatures directly (e.g. TSS).\n   */\n  get signablePayload(): Buffer {\n    if (!this.flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction for signing');\n    }\n\n    // TODO: Implement FlareJS signable payload extraction\n    // For now, return placeholder\n    return Buffer.from(FLARE_SIGNABLE_PAYLOAD);\n  }\n\n  get id(): string {\n    if (!this.flareTransaction) {\n      throw new InvalidTransactionError('Empty transaction for ID generation');\n    }\n\n    // TODO: Implement FlareJS transaction ID generation\n    // For now, return placeholder\n    return FLARE_TRANSACTION_ID_PLACEHOLDER;\n  }\n\n  get fromAddresses(): string[] {\n    return this._fromAddresses.map((address) => {\n      // TODO: Format addresses using FlareJS utilities\n      return address;\n    });\n  }\n\n  get rewardAddresses(): string[] {\n    return this._rewardAddresses.map((address) => {\n      // TODO: Format addresses using FlareJS utilities\n      return address;\n    });\n  }\n\n  /**\n   * Get the list of outputs. Amounts are expressed in absolute value.\n   */\n  get outputs(): Entry[] {\n    switch (this.type) {\n      case TransactionType.Export:\n        // TODO: Extract export outputs from FlareJS transaction\n        return [];\n      case TransactionType.Import:\n        // TODO: Extract import outputs from FlareJS transaction\n        return [];\n      case TransactionType.AddValidator:\n      case TransactionType.AddPermissionlessValidator:\n        // TODO: Extract validator outputs from FlareJS transaction\n        return [\n          {\n            address: this._nodeID || PLACEHOLDER_NODE_ID,\n            value: this._stakeAmount?.toString() || '0',\n          },\n        ];\n      case TransactionType.AddDelegator:\n      case TransactionType.AddPermissionlessDelegator:\n        // TODO: Extract delegator outputs from FlareJS transaction\n        return [\n          {\n            address: this._nodeID || PLACEHOLDER_NODE_ID,\n            value: this._stakeAmount?.toString() || '0',\n          },\n        ];\n      default:\n        return [];\n    }\n  }\n\n  get fee(): TransactionFee {\n    return { fee: '0', ...this._fee };\n  }\n\n  get changeOutputs(): Entry[] {\n    // TODO: Extract change outputs from FlareJS transaction\n    // For now, return empty array\n    return [];\n  }\n\n  get inputs(): FlrpEntry[] {\n    // TODO: Extract inputs from FlareJS transaction\n    // For now, return placeholder based on UTXOs\n    return this._utxos.map((utxo) => ({\n      id: utxo.txid + INPUT_SEPARATOR + utxo.outputidx,\n      address: this.fromAddresses.sort().join(ADDRESS_SEPARATOR),\n      value: utxo.amount,\n    }));\n  }\n\n  /**\n   * Flare wrapper to create signature and return it for credentials\n   * @param prv\n   * @return hexstring\n   */\n  createSignature(prv: Buffer): string {\n    // TODO: Implement FlareJS signature creation\n    // This should use FlareJS signing utilities\n    const signval = utils.createSignature(this._network, this.signablePayload, prv);\n    return signval.toString(HEX_ENCODING);\n  }\n\n  /**\n   * Check if transaction is for C-chain (cross-chain)\n   */\n  get isTransactionForCChain(): boolean {\n    return this.type === TransactionType.Export || this.type === TransactionType.Import;\n  }\n\n  /** @inheritdoc */\n  explainTransaction(): TransactionExplanation {\n    const txJson = this.toJson();\n    const displayOrder = [...DISPLAY_ORDER_BASE];\n\n    // Add memo to display order if present\n    if (this.hasMemo()) {\n      displayOrder.push(MEMO_FIELD);\n    }\n\n    // Calculate total output amount\n    const outputAmount = txJson.outputs\n      .reduce((sum, output) => {\n        return sum + BigInt(output.value || '0');\n      }, BigInt(0))\n      .toString();\n\n    // Calculate total change amount\n    const changeAmount = txJson.changeOutputs\n      .reduce((sum, output) => {\n        return sum + BigInt(output.value || '0');\n      }, BigInt(0))\n      .toString();\n\n    let rewardAddresses;\n    const stakingTypes = [\n      TransactionType.AddValidator,\n      TransactionType.AddDelegator,\n      TransactionType.AddPermissionlessValidator,\n      TransactionType.AddPermissionlessDelegator,\n    ];\n\n    if (stakingTypes.includes(txJson.type)) {\n      rewardAddresses = this.rewardAddresses;\n      displayOrder.splice(6, 0, REWARD_ADDRESSES_FIELD);\n    }\n\n    // Add cross-chain information for export/import\n    if (this.isTransactionForCChain) {\n      displayOrder.push(SOURCE_CHAIN_FIELD, DESTINATION_CHAIN_FIELD);\n    }\n\n    const explanation: TransactionExplanation & { memo?: string } = {\n      displayOrder,\n      id: txJson.id,\n      inputs: txJson.inputs,\n      outputs: txJson.outputs.map((o) => ({ address: o.address, amount: o.value })),\n      outputAmount,\n      changeOutputs: txJson.changeOutputs.map((o) => ({ address: o.address, amount: o.value })),\n      changeAmount,\n      rewardAddresses,\n      fee: this.fee,\n      type: txJson.type,\n    };\n\n    // Add memo to explanation if present\n    if (this.hasMemo()) {\n      explanation.memo = this.getMemoString();\n    }\n\n    return explanation;\n  }\n}\n"]}
|
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
import { BaseTransactionBuilder } from '@bitgo-beta/sdk-core';
|
|
2
|
+
import { BaseCoin as CoinConfig } from '@bitgo-beta/statics';
|
|
3
|
+
import { DecodedUtxoObj, Tx } from './iface';
|
|
4
|
+
import { KeyPair } from './keyPair';
|
|
5
|
+
import { Transaction } from './transaction';
|
|
6
|
+
export declare abstract class TransactionBuilder extends BaseTransactionBuilder {
|
|
7
|
+
protected _transaction: Transaction;
|
|
8
|
+
protected recoverSigner: boolean;
|
|
9
|
+
_signer: KeyPair[];
|
|
10
|
+
protected _recoveryMode: boolean;
|
|
11
|
+
constructor(_coinConfig: Readonly<CoinConfig>);
|
|
12
|
+
/**
|
|
13
|
+
* Initialize the transaction builder fields using the decoded transaction data
|
|
14
|
+
*
|
|
15
|
+
* @param {Tx} tx the transaction data
|
|
16
|
+
* @returns itself
|
|
17
|
+
*/
|
|
18
|
+
initBuilder(tx: Tx): this;
|
|
19
|
+
/**
|
|
20
|
+
* Validates the threshold
|
|
21
|
+
* @param threshold
|
|
22
|
+
*/
|
|
23
|
+
validateThreshold(threshold: number): void;
|
|
24
|
+
/**
|
|
25
|
+
* Check the UTXO has expected fields.
|
|
26
|
+
* @param UTXO
|
|
27
|
+
*/
|
|
28
|
+
validateUtxo(value: DecodedUtxoObj): void;
|
|
29
|
+
/**
|
|
30
|
+
* Check the list of UTXOS is empty and check each UTXO.
|
|
31
|
+
* @param values
|
|
32
|
+
*/
|
|
33
|
+
validateUtxos(values: DecodedUtxoObj[]): void;
|
|
34
|
+
/**
|
|
35
|
+
* Validates locktime
|
|
36
|
+
* @param locktime
|
|
37
|
+
*/
|
|
38
|
+
validateLocktime(locktime: bigint): void;
|
|
39
|
+
/**
|
|
40
|
+
* Threshold is an int that names the number of unique signatures required to spend the output.
|
|
41
|
+
* Must be less than or equal to the length of Addresses.
|
|
42
|
+
* @param {number} value
|
|
43
|
+
*/
|
|
44
|
+
threshold(value: number): this;
|
|
45
|
+
/**
|
|
46
|
+
* Locktime is a long that contains the unix timestamp that this output can be spent after.
|
|
47
|
+
* The unix timestamp is specific to the second.
|
|
48
|
+
* @param value
|
|
49
|
+
*/
|
|
50
|
+
locktime(value: string | number): this;
|
|
51
|
+
/**
|
|
52
|
+
* When using recovery key must be set here
|
|
53
|
+
* @param {boolean}[recoverSigner=true] whether it's recovery signer
|
|
54
|
+
*/
|
|
55
|
+
recoverMode(recoverSigner?: boolean): this;
|
|
56
|
+
/**
|
|
57
|
+
* fromPubKey is a list of unique addresses that correspond to the private keys that can be used to spend this output
|
|
58
|
+
* @param {string | string[]} senderPubKey
|
|
59
|
+
*/
|
|
60
|
+
fromPubKey(senderPubKey: string | string[]): this;
|
|
61
|
+
/**
|
|
62
|
+
* List of UTXO required as inputs.
|
|
63
|
+
* A UTXO is a standalone representation of a transaction output.
|
|
64
|
+
*
|
|
65
|
+
* @param {DecodedUtxoObj[]} list of UTXOS
|
|
66
|
+
*/
|
|
67
|
+
utxos(value: DecodedUtxoObj[]): this;
|
|
68
|
+
/**
|
|
69
|
+
* Build the Flare transaction using FlareJS API
|
|
70
|
+
* @protected
|
|
71
|
+
*/
|
|
72
|
+
protected abstract buildFlareTransaction(): Promise<void> | void;
|
|
73
|
+
/** @inheritdoc */
|
|
74
|
+
protected fromImplementation(rawTransaction: string): Transaction;
|
|
75
|
+
/**
|
|
76
|
+
* Get the transaction instance
|
|
77
|
+
*/
|
|
78
|
+
get transaction(): Transaction;
|
|
79
|
+
/**
|
|
80
|
+
* Validate required fields before building transaction
|
|
81
|
+
* @protected
|
|
82
|
+
*/
|
|
83
|
+
protected validateRequiredFields(): void;
|
|
84
|
+
}
|
|
85
|
+
//# sourceMappingURL=transactionBuilder.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"transactionBuilder.d.ts","sourceRoot":"","sources":["../../../src/lib/transactionBuilder.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,sBAAsB,EAAyB,MAAM,sBAAsB,CAAC;AACrF,OAAO,EAAE,QAAQ,IAAI,UAAU,EAAE,MAAM,qBAAqB,CAAC;AAC7D,OAAO,EAAE,cAAc,EAAE,EAAE,EAAE,MAAM,SAAS,CAAC;AAC7C,OAAO,EAAE,OAAO,EAAE,MAAM,WAAW,CAAC;AACpC,OAAO,EAAE,WAAW,EAAE,MAAM,eAAe,CAAC;AAiB5C,8BAAsB,kBAAmB,SAAQ,sBAAsB;IACrE,SAAS,CAAC,YAAY,EAAE,WAAW,CAAC;IACpC,SAAS,CAAC,aAAa,UAAS;IACzB,OAAO,EAAE,OAAO,EAAE,CAAM;IAG/B,SAAS,CAAC,aAAa,UAAS;gBAEpB,WAAW,EAAE,QAAQ,CAAC,UAAU,CAAC;IAK7C;;;;;OAKG;IACH,WAAW,CAAC,EAAE,EAAE,EAAE,GAAG,IAAI;IAyBzB;;;OAGG;IACH,iBAAiB,CAAC,SAAS,EAAE,MAAM,GAAG,IAAI;IAM1C;;;OAGG;IACH,YAAY,CAAC,KAAK,EAAE,cAAc,GAAG,IAAI;IAMzC;;;OAGG;IACH,aAAa,CAAC,MAAM,EAAE,cAAc,EAAE,GAAG,IAAI;IAO7C;;;OAGG;IACH,gBAAgB,CAAC,QAAQ,EAAE,MAAM,GAAG,IAAI;IAOxC;;;;OAIG;IACH,SAAS,CAAC,KAAK,EAAE,MAAM,GAAG,IAAI;IAM9B;;;;OAIG;IACH,QAAQ,CAAC,KAAK,EAAE,MAAM,GAAG,MAAM,GAAG,IAAI;IAMtC;;;OAGG;IACH,WAAW,CAAC,aAAa,UAAO,GAAG,IAAI;IAYvC;;;OAGG;IACH,UAAU,CAAC,YAAY,EAAE,MAAM,GAAG,MAAM,EAAE,GAAG,IAAI;IAMjD;;;;;OAKG;IACH,KAAK,CAAC,KAAK,EAAE,cAAc,EAAE,GAAG,IAAI;IAMpC;;;OAGG;IACH,SAAS,CAAC,QAAQ,CAAC,qBAAqB,IAAI,OAAO,CAAC,IAAI,CAAC,GAAG,IAAI;IAEhE,kBAAkB;IAClB,SAAS,CAAC,kBAAkB,CAAC,cAAc,EAAE,MAAM,GAAG,WAAW;IAajE;;OAEG;IACH,IAAI,WAAW,IAAI,WAAW,CAE7B;IAED;;;OAGG;IACH,SAAS,CAAC,sBAAsB,IAAI,IAAI;CAQzC"}
|
|
@@ -0,0 +1,167 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.TransactionBuilder = void 0;
|
|
4
|
+
const sdk_core_1 = require("@bitgo-beta/sdk-core");
|
|
5
|
+
const transaction_1 = require("./transaction");
|
|
6
|
+
const constants_1 = require("./constants");
|
|
7
|
+
class TransactionBuilder extends sdk_core_1.BaseTransactionBuilder {
|
|
8
|
+
constructor(_coinConfig) {
|
|
9
|
+
super(_coinConfig);
|
|
10
|
+
this.recoverSigner = false;
|
|
11
|
+
this._signer = [];
|
|
12
|
+
// Recovery mode flag for transaction building
|
|
13
|
+
this._recoveryMode = false;
|
|
14
|
+
this._transaction = new transaction_1.Transaction(_coinConfig);
|
|
15
|
+
}
|
|
16
|
+
/**
|
|
17
|
+
* Initialize the transaction builder fields using the decoded transaction data
|
|
18
|
+
*
|
|
19
|
+
* @param {Tx} tx the transaction data
|
|
20
|
+
* @returns itself
|
|
21
|
+
*/
|
|
22
|
+
initBuilder(tx) {
|
|
23
|
+
// Validate network and blockchain IDs if available
|
|
24
|
+
const txData = tx;
|
|
25
|
+
if (txData.networkID !== undefined && txData.networkID !== this._transaction._networkID) {
|
|
26
|
+
throw new Error(constants_1.ERROR_NETWORK_ID_MISMATCH);
|
|
27
|
+
}
|
|
28
|
+
if (txData.blockchainID) {
|
|
29
|
+
const blockchainID = Buffer.isBuffer(txData.blockchainID)
|
|
30
|
+
? txData.blockchainID
|
|
31
|
+
: Buffer.from(txData.blockchainID, constants_1.HEX_ENCODING);
|
|
32
|
+
const transactionBlockchainID = Buffer.isBuffer(this._transaction._blockchainID)
|
|
33
|
+
? this._transaction._blockchainID
|
|
34
|
+
: Buffer.from(this._transaction._blockchainID, constants_1.HEX_ENCODING);
|
|
35
|
+
if (!blockchainID.equals(transactionBlockchainID)) {
|
|
36
|
+
throw new Error(constants_1.ERROR_BLOCKCHAIN_ID_MISMATCH_BUILDER);
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
this._transaction.setTransaction(tx);
|
|
40
|
+
return this;
|
|
41
|
+
}
|
|
42
|
+
// region Validators
|
|
43
|
+
/**
|
|
44
|
+
* Validates the threshold
|
|
45
|
+
* @param threshold
|
|
46
|
+
*/
|
|
47
|
+
validateThreshold(threshold) {
|
|
48
|
+
if (!threshold || threshold !== 2) {
|
|
49
|
+
throw new sdk_core_1.BuildTransactionError(constants_1.ERROR_INVALID_THRESHOLD);
|
|
50
|
+
}
|
|
51
|
+
}
|
|
52
|
+
/**
|
|
53
|
+
* Check the UTXO has expected fields.
|
|
54
|
+
* @param UTXO
|
|
55
|
+
*/
|
|
56
|
+
validateUtxo(value) {
|
|
57
|
+
constants_1.UTXO_REQUIRED_FIELDS.forEach((field) => {
|
|
58
|
+
if (!value.hasOwnProperty(field))
|
|
59
|
+
throw new sdk_core_1.BuildTransactionError(`${constants_1.ERROR_UTXOS_MISSING_FIELD} ${field}`);
|
|
60
|
+
});
|
|
61
|
+
}
|
|
62
|
+
/**
|
|
63
|
+
* Check the list of UTXOS is empty and check each UTXO.
|
|
64
|
+
* @param values
|
|
65
|
+
*/
|
|
66
|
+
validateUtxos(values) {
|
|
67
|
+
if (values.length === 0) {
|
|
68
|
+
throw new sdk_core_1.BuildTransactionError(constants_1.ERROR_UTXOS_EMPTY_ARRAY);
|
|
69
|
+
}
|
|
70
|
+
values.forEach(this.validateUtxo);
|
|
71
|
+
}
|
|
72
|
+
/**
|
|
73
|
+
* Validates locktime
|
|
74
|
+
* @param locktime
|
|
75
|
+
*/
|
|
76
|
+
validateLocktime(locktime) {
|
|
77
|
+
if (!locktime || locktime < BigInt(0)) {
|
|
78
|
+
throw new sdk_core_1.BuildTransactionError(constants_1.ERROR_INVALID_LOCKTIME);
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
// endregion
|
|
82
|
+
/**
|
|
83
|
+
* Threshold is an int that names the number of unique signatures required to spend the output.
|
|
84
|
+
* Must be less than or equal to the length of Addresses.
|
|
85
|
+
* @param {number} value
|
|
86
|
+
*/
|
|
87
|
+
threshold(value) {
|
|
88
|
+
this.validateThreshold(value);
|
|
89
|
+
this._transaction._threshold = value;
|
|
90
|
+
return this;
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* Locktime is a long that contains the unix timestamp that this output can be spent after.
|
|
94
|
+
* The unix timestamp is specific to the second.
|
|
95
|
+
* @param value
|
|
96
|
+
*/
|
|
97
|
+
locktime(value) {
|
|
98
|
+
this.validateLocktime(BigInt(value));
|
|
99
|
+
this._transaction._locktime = BigInt(value);
|
|
100
|
+
return this;
|
|
101
|
+
}
|
|
102
|
+
/**
|
|
103
|
+
* When using recovery key must be set here
|
|
104
|
+
* @param {boolean}[recoverSigner=true] whether it's recovery signer
|
|
105
|
+
*/
|
|
106
|
+
recoverMode(recoverSigner = true) {
|
|
107
|
+
this.recoverSigner = recoverSigner;
|
|
108
|
+
this._recoveryMode = recoverSigner;
|
|
109
|
+
// Recovery operations typically need single signature
|
|
110
|
+
if (recoverSigner && !this._transaction._threshold) {
|
|
111
|
+
this._transaction._threshold = 1;
|
|
112
|
+
}
|
|
113
|
+
return this;
|
|
114
|
+
}
|
|
115
|
+
/**
|
|
116
|
+
* fromPubKey is a list of unique addresses that correspond to the private keys that can be used to spend this output
|
|
117
|
+
* @param {string | string[]} senderPubKey
|
|
118
|
+
*/
|
|
119
|
+
fromPubKey(senderPubKey) {
|
|
120
|
+
const pubKeys = senderPubKey instanceof Array ? senderPubKey : [senderPubKey];
|
|
121
|
+
this._transaction._fromAddresses = pubKeys; // Store as strings directly
|
|
122
|
+
return this;
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* List of UTXO required as inputs.
|
|
126
|
+
* A UTXO is a standalone representation of a transaction output.
|
|
127
|
+
*
|
|
128
|
+
* @param {DecodedUtxoObj[]} list of UTXOS
|
|
129
|
+
*/
|
|
130
|
+
utxos(value) {
|
|
131
|
+
this.validateUtxos(value);
|
|
132
|
+
this._transaction._utxos = value;
|
|
133
|
+
return this;
|
|
134
|
+
}
|
|
135
|
+
/** @inheritdoc */
|
|
136
|
+
fromImplementation(rawTransaction) {
|
|
137
|
+
// Parse the raw transaction and initialize the builder
|
|
138
|
+
try {
|
|
139
|
+
const parsedTx = JSON.parse(rawTransaction);
|
|
140
|
+
this.initBuilder(parsedTx);
|
|
141
|
+
return this._transaction;
|
|
142
|
+
}
|
|
143
|
+
catch (error) {
|
|
144
|
+
throw new Error(`${constants_1.ERROR_PARSE_RAW_TRANSACTION}: ${error instanceof Error ? error.message : constants_1.ERROR_UNKNOWN_PARSING}`);
|
|
145
|
+
}
|
|
146
|
+
}
|
|
147
|
+
/**
|
|
148
|
+
* Get the transaction instance
|
|
149
|
+
*/
|
|
150
|
+
get transaction() {
|
|
151
|
+
return this._transaction;
|
|
152
|
+
}
|
|
153
|
+
/**
|
|
154
|
+
* Validate required fields before building transaction
|
|
155
|
+
* @protected
|
|
156
|
+
*/
|
|
157
|
+
validateRequiredFields() {
|
|
158
|
+
if (this._transaction._fromAddresses.length === 0) {
|
|
159
|
+
throw new Error(constants_1.ERROR_FROM_ADDRESSES_REQUIRED);
|
|
160
|
+
}
|
|
161
|
+
if (this._transaction._utxos.length === 0) {
|
|
162
|
+
throw new Error(constants_1.ERROR_UTXOS_REQUIRED_BUILDER);
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
exports.TransactionBuilder = TransactionBuilder;
|
|
167
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transactionBuilder.js","sourceRoot":"","sources":["../../../src/lib/transactionBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import { BaseTransactionBuilder, BuildTransactionError } from '@bitgo-beta/sdk-core';\nimport { BaseCoin as CoinConfig } from '@bitgo-beta/statics';\nimport { DecodedUtxoObj, Tx } from './iface';\nimport { KeyPair } from './keyPair';\nimport { Transaction } from './transaction';\nimport { RawTransactionData } from './types';\nimport {\n  ERROR_NETWORK_ID_MISMATCH,\n  ERROR_BLOCKCHAIN_ID_MISMATCH_BUILDER,\n  ERROR_INVALID_THRESHOLD,\n  ERROR_INVALID_LOCKTIME,\n  ERROR_UTXOS_EMPTY_ARRAY,\n  ERROR_UTXOS_MISSING_FIELD,\n  ERROR_FROM_ADDRESSES_REQUIRED,\n  ERROR_UTXOS_REQUIRED_BUILDER,\n  ERROR_PARSE_RAW_TRANSACTION,\n  ERROR_UNKNOWN_PARSING,\n  UTXO_REQUIRED_FIELDS,\n  HEX_ENCODING,\n} from './constants';\n\nexport abstract class TransactionBuilder extends BaseTransactionBuilder {\n  protected _transaction: Transaction;\n  protected recoverSigner = false;\n  public _signer: KeyPair[] = [];\n\n  // Recovery mode flag for transaction building\n  protected _recoveryMode = false;\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    this._transaction = new Transaction(_coinConfig);\n  }\n\n  /**\n   * Initialize the transaction builder fields using the decoded transaction data\n   *\n   * @param {Tx} tx the transaction data\n   * @returns itself\n   */\n  initBuilder(tx: Tx): this {\n    // Validate network and blockchain IDs if available\n    const txData = tx as unknown as RawTransactionData;\n\n    if (txData.networkID !== undefined && txData.networkID !== this._transaction._networkID) {\n      throw new Error(ERROR_NETWORK_ID_MISMATCH);\n    }\n\n    if (txData.blockchainID) {\n      const blockchainID = Buffer.isBuffer(txData.blockchainID)\n        ? txData.blockchainID\n        : Buffer.from(txData.blockchainID, HEX_ENCODING);\n      const transactionBlockchainID = Buffer.isBuffer(this._transaction._blockchainID)\n        ? this._transaction._blockchainID\n        : Buffer.from(this._transaction._blockchainID, HEX_ENCODING);\n      if (!blockchainID.equals(transactionBlockchainID)) {\n        throw new Error(ERROR_BLOCKCHAIN_ID_MISMATCH_BUILDER);\n      }\n    }\n\n    this._transaction.setTransaction(tx);\n    return this;\n  }\n\n  // region Validators\n  /**\n   * Validates the threshold\n   * @param threshold\n   */\n  validateThreshold(threshold: number): void {\n    if (!threshold || threshold !== 2) {\n      throw new BuildTransactionError(ERROR_INVALID_THRESHOLD);\n    }\n  }\n\n  /**\n   * Check the UTXO has expected fields.\n   * @param UTXO\n   */\n  validateUtxo(value: DecodedUtxoObj): void {\n    UTXO_REQUIRED_FIELDS.forEach((field) => {\n      if (!value.hasOwnProperty(field)) throw new BuildTransactionError(`${ERROR_UTXOS_MISSING_FIELD} ${field}`);\n    });\n  }\n\n  /**\n   * Check the list of UTXOS is empty and check each UTXO.\n   * @param values\n   */\n  validateUtxos(values: DecodedUtxoObj[]): void {\n    if (values.length === 0) {\n      throw new BuildTransactionError(ERROR_UTXOS_EMPTY_ARRAY);\n    }\n    values.forEach(this.validateUtxo);\n  }\n\n  /**\n   * Validates locktime\n   * @param locktime\n   */\n  validateLocktime(locktime: bigint): void {\n    if (!locktime || locktime < BigInt(0)) {\n      throw new BuildTransactionError(ERROR_INVALID_LOCKTIME);\n    }\n  }\n  // endregion\n\n  /**\n   * Threshold is an int that names the number of unique signatures required to spend the output.\n   * Must be less than or equal to the length of Addresses.\n   * @param {number} value\n   */\n  threshold(value: number): this {\n    this.validateThreshold(value);\n    this._transaction._threshold = value;\n    return this;\n  }\n\n  /**\n   * Locktime is a long that contains the unix timestamp that this output can be spent after.\n   * The unix timestamp is specific to the second.\n   * @param value\n   */\n  locktime(value: string | number): this {\n    this.validateLocktime(BigInt(value));\n    this._transaction._locktime = BigInt(value);\n    return this;\n  }\n\n  /**\n   * When using recovery key must be set here\n   * @param {boolean}[recoverSigner=true] whether it's recovery signer\n   */\n  recoverMode(recoverSigner = true): this {\n    this.recoverSigner = recoverSigner;\n    this._recoveryMode = recoverSigner;\n\n    // Recovery operations typically need single signature\n    if (recoverSigner && !this._transaction._threshold) {\n      this._transaction._threshold = 1;\n    }\n\n    return this;\n  }\n\n  /**\n   * fromPubKey is a list of unique addresses that correspond to the private keys that can be used to spend this output\n   * @param {string | string[]} senderPubKey\n   */\n  fromPubKey(senderPubKey: string | string[]): this {\n    const pubKeys = senderPubKey instanceof Array ? senderPubKey : [senderPubKey];\n    this._transaction._fromAddresses = pubKeys; // Store as strings directly\n    return this;\n  }\n\n  /**\n   * List of UTXO required as inputs.\n   * A UTXO is a standalone representation of a transaction output.\n   *\n   * @param {DecodedUtxoObj[]} list of UTXOS\n   */\n  utxos(value: DecodedUtxoObj[]): this {\n    this.validateUtxos(value);\n    this._transaction._utxos = value;\n    return this;\n  }\n\n  /**\n   * Build the Flare transaction using FlareJS API\n   * @protected\n   */\n  protected abstract buildFlareTransaction(): Promise<void> | void;\n\n  /** @inheritdoc */\n  protected fromImplementation(rawTransaction: string): Transaction {\n    // Parse the raw transaction and initialize the builder\n    try {\n      const parsedTx = JSON.parse(rawTransaction);\n      this.initBuilder(parsedTx);\n      return this._transaction;\n    } catch (error) {\n      throw new Error(\n        `${ERROR_PARSE_RAW_TRANSACTION}: ${error instanceof Error ? error.message : ERROR_UNKNOWN_PARSING}`\n      );\n    }\n  }\n\n  /**\n   * Get the transaction instance\n   */\n  get transaction(): Transaction {\n    return this._transaction;\n  }\n\n  /**\n   * Validate required fields before building transaction\n   * @protected\n   */\n  protected validateRequiredFields(): void {\n    if (this._transaction._fromAddresses.length === 0) {\n      throw new Error(ERROR_FROM_ADDRESSES_REQUIRED);\n    }\n    if (this._transaction._utxos.length === 0) {\n      throw new Error(ERROR_UTXOS_REQUIRED_BUILDER);\n    }\n  }\n}\n"]}
|