@bitgo-beta/sdk-coin-iota 1.0.1-beta.324 → 1.0.1-beta.326
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/iota.d.ts +123 -27
- package/dist/src/iota.d.ts.map +1 -1
- package/dist/src/iota.js +217 -103
- package/dist/src/lib/constants.d.ts +50 -1
- package/dist/src/lib/constants.d.ts.map +1 -1
- package/dist/src/lib/constants.js +68 -4
- package/dist/src/lib/iface.d.ts +141 -1
- package/dist/src/lib/iface.d.ts.map +1 -1
- package/dist/src/lib/iface.js +1 -1
- package/dist/src/lib/keyPair.d.ts +100 -6
- package/dist/src/lib/keyPair.d.ts.map +1 -1
- package/dist/src/lib/keyPair.js +103 -10
- package/dist/src/lib/transaction.d.ts +117 -14
- package/dist/src/lib/transaction.d.ts.map +1 -1
- package/dist/src/lib/transaction.js +190 -67
- package/dist/src/lib/transactionBuilder.d.ts +73 -34
- package/dist/src/lib/transactionBuilder.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilder.js +90 -45
- package/dist/src/lib/transactionBuilderFactory.d.ts +89 -6
- package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -1
- package/dist/src/lib/transactionBuilderFactory.js +103 -16
- package/dist/src/lib/transferBuilder.d.ts +43 -0
- package/dist/src/lib/transferBuilder.d.ts.map +1 -1
- package/dist/src/lib/transferBuilder.js +50 -5
- package/dist/src/lib/transferTransaction.d.ts +93 -2
- package/dist/src/lib/transferTransaction.d.ts.map +1 -1
- package/dist/src/lib/transferTransaction.js +180 -51
- package/dist/src/lib/utils.d.ts +107 -8
- package/dist/src/lib/utils.d.ts.map +1 -1
- package/dist/src/lib/utils.js +134 -23
- package/dist/test/unit/helpers/testHelpers.d.ts +57 -0
- package/dist/test/unit/helpers/testHelpers.d.ts.map +1 -0
- package/dist/test/unit/helpers/testHelpers.js +176 -0
- package/dist/test/unit/iota.js +47 -152
- package/dist/test/unit/keyPair.js +34 -61
- package/dist/test/unit/transactionBuilder/transactionBuilder.js +137 -255
- package/dist/test/unit/transactionBuilder/transactionBuilderFactory.js +43 -108
- package/dist/test/unit/transactionBuilder/transferBuilder.js +296 -762
- package/dist/test/unit/transferTransaction.js +106 -353
- package/dist/test/unit/utils.js +171 -197
- package/dist/tsconfig.tsbuildinfo +1 -1
- package/package.json +7 -7
|
@@ -68,12 +68,33 @@ class TransferTransaction extends transaction_1.Transaction {
|
|
|
68
68
|
this.paymentObjects = txData.paymentObjects;
|
|
69
69
|
this.updateIsSimulateTx();
|
|
70
70
|
}
|
|
71
|
+
/**
|
|
72
|
+
* Parses a transfer transaction from its broadcast format (base64 or raw bytes).
|
|
73
|
+
* Extracts recipients, amounts, and payment objects from the transaction data.
|
|
74
|
+
*/
|
|
71
75
|
parseFromBroadcastTx(tx) {
|
|
72
76
|
const txData = transactions_1.Transaction.from(tx).getData();
|
|
73
|
-
|
|
77
|
+
this.validateTransferCommands(txData);
|
|
78
|
+
super.parseFromBroadcastTx(tx);
|
|
79
|
+
const { inputObjects, amounts, receivers } = this.parseTransactionInputs(txData);
|
|
80
|
+
this.validateAmountsMatchReceivers(amounts, receivers);
|
|
81
|
+
this.assignParsedObjects(inputObjects);
|
|
82
|
+
this.assignRecipients(receivers, amounts);
|
|
83
|
+
this.updateIsSimulateTx();
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Validates that the transaction only contains supported transfer commands.
|
|
87
|
+
*/
|
|
88
|
+
validateTransferCommands(txData) {
|
|
89
|
+
const hasUnsupportedCommands = txData.commands.some((command) => !constants_1.TRANSFER_TRANSACTION_COMMANDS.includes(command.$kind));
|
|
90
|
+
if (hasUnsupportedCommands) {
|
|
74
91
|
throw new sdk_core_1.InvalidTransactionError('Unsupported commands in the transaction');
|
|
75
92
|
}
|
|
76
|
-
|
|
93
|
+
}
|
|
94
|
+
/**
|
|
95
|
+
* Parses transaction inputs to extract objects, amounts, and receiver addresses.
|
|
96
|
+
*/
|
|
97
|
+
parseTransactionInputs(txData) {
|
|
77
98
|
const inputObjects = [];
|
|
78
99
|
const amounts = [];
|
|
79
100
|
const receivers = [];
|
|
@@ -92,59 +113,120 @@ class TransferTransaction extends transaction_1.Transaction {
|
|
|
92
113
|
}
|
|
93
114
|
}
|
|
94
115
|
});
|
|
116
|
+
return { inputObjects, amounts, receivers };
|
|
117
|
+
}
|
|
118
|
+
/**
|
|
119
|
+
* Validates that the number of amounts matches the number of receivers.
|
|
120
|
+
*/
|
|
121
|
+
validateAmountsMatchReceivers(amounts, receivers) {
|
|
95
122
|
if (amounts.length !== receivers.length) {
|
|
96
|
-
throw new sdk_core_1.InvalidTransactionError('
|
|
123
|
+
throw new sdk_core_1.InvalidTransactionError('Count of amounts does not match count of receivers');
|
|
97
124
|
}
|
|
98
|
-
|
|
99
|
-
|
|
125
|
+
}
|
|
126
|
+
/**
|
|
127
|
+
* Assigns parsed input objects to either gas payment objects or payment objects.
|
|
128
|
+
* If no gas objects exist and sender pays own gas, objects become gas objects.
|
|
129
|
+
* Otherwise, they become payment objects.
|
|
130
|
+
*/
|
|
131
|
+
assignParsedObjects(inputObjects) {
|
|
132
|
+
if (inputObjects.length === 0) {
|
|
133
|
+
return;
|
|
134
|
+
}
|
|
135
|
+
const noGasObjectsExist = !this.gasPaymentObjects || this.gasPaymentObjects.length === 0;
|
|
136
|
+
const senderPaysOwnGas = !this.gasSponsor || this.sender === this.gasSponsor;
|
|
137
|
+
if (noGasObjectsExist && senderPaysOwnGas) {
|
|
100
138
|
this.gasPaymentObjects = inputObjects;
|
|
101
139
|
}
|
|
102
|
-
else
|
|
140
|
+
else {
|
|
103
141
|
this._paymentObjects = inputObjects;
|
|
104
142
|
}
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
143
|
+
}
|
|
144
|
+
/**
|
|
145
|
+
* Creates and assigns recipients from parsed addresses and amounts.
|
|
146
|
+
*/
|
|
147
|
+
assignRecipients(receivers, amounts) {
|
|
148
|
+
this._recipients = receivers.map((address, index) => ({
|
|
149
|
+
address,
|
|
150
|
+
amount: amounts[index],
|
|
151
|
+
}));
|
|
113
152
|
}
|
|
114
153
|
messageWithIntent(message) {
|
|
115
154
|
return (0, cryptography_1.messageWithIntent)('TransactionData', message);
|
|
116
155
|
}
|
|
156
|
+
/**
|
|
157
|
+
* Populates the IOTA transaction with inputs and commands for the transfer.
|
|
158
|
+
* This determines which objects to use for payment (either payment objects or gas objects),
|
|
159
|
+
* consolidates them into a single coin, and then splits/transfers to recipients.
|
|
160
|
+
*/
|
|
117
161
|
populateTxInputsAndCommands() {
|
|
118
|
-
const
|
|
119
|
-
|
|
120
|
-
: this.consolidatePaymentObjects();
|
|
121
|
-
this.splitAndTransferToRecipients(consolidatedCoin);
|
|
162
|
+
const sourceCoin = this.getConsolidatedSourceCoin();
|
|
163
|
+
this.splitAndTransferToRecipients(sourceCoin);
|
|
122
164
|
}
|
|
123
|
-
|
|
124
|
-
|
|
165
|
+
/**
|
|
166
|
+
* Determines which objects to use as the payment source and consolidates them.
|
|
167
|
+
* If payment objects are provided, use those. Otherwise, if the sender is paying
|
|
168
|
+
* their own gas, use the gas objects for payment.
|
|
169
|
+
*/
|
|
170
|
+
getConsolidatedSourceCoin() {
|
|
171
|
+
if (this.hasPaymentObjects()) {
|
|
172
|
+
return this.consolidatePaymentObjects();
|
|
173
|
+
}
|
|
174
|
+
return this.consolidateGasObjects();
|
|
175
|
+
}
|
|
176
|
+
/**
|
|
177
|
+
* Checks if payment objects exist and if gas objects should be used instead.
|
|
178
|
+
* Gas objects are used when: no payment objects exist AND sender pays their own gas.
|
|
179
|
+
*/
|
|
180
|
+
hasPaymentObjects() {
|
|
181
|
+
const hasPaymentObjects = this.paymentObjects && this.paymentObjects.length > 0;
|
|
182
|
+
if (hasPaymentObjects) {
|
|
183
|
+
return true;
|
|
184
|
+
}
|
|
185
|
+
// If no payment objects, only use gas objects if sender pays own gas
|
|
125
186
|
const senderPaysOwnGas = !this.gasSponsor || this.gasSponsor === this.sender;
|
|
126
|
-
return !
|
|
187
|
+
return !senderPaysOwnGas;
|
|
127
188
|
}
|
|
189
|
+
/**
|
|
190
|
+
* Consolidates payment objects into a single coin object.
|
|
191
|
+
* If multiple payment objects exist, they are merged in batches.
|
|
192
|
+
*/
|
|
128
193
|
consolidatePaymentObjects() {
|
|
129
194
|
if (!this.paymentObjects || this.paymentObjects.length === 0) {
|
|
130
195
|
throw new sdk_core_1.InvalidTransactionError('Payment objects are required');
|
|
131
196
|
}
|
|
132
197
|
const firstObject = this._iotaTransaction.object(transactions_1.Inputs.ObjectRef(this.paymentObjects[0]));
|
|
198
|
+
// Single object doesn't need consolidation
|
|
133
199
|
if (this.paymentObjects.length === 1) {
|
|
134
200
|
return firstObject;
|
|
135
201
|
}
|
|
202
|
+
// Merge remaining objects into the first one
|
|
136
203
|
return this.mergeObjectsInBatches(firstObject, this.paymentObjects.slice(1), constants_1.MAX_INPUT_OBJECTS);
|
|
137
204
|
}
|
|
205
|
+
/**
|
|
206
|
+
* Consolidates gas payment objects into a single coin object.
|
|
207
|
+
* If the number of gas objects exceeds the maximum, they are merged in batches.
|
|
208
|
+
*/
|
|
138
209
|
consolidateGasObjects() {
|
|
139
210
|
if (!this.gasPaymentObjects || this.gasPaymentObjects.length === 0) {
|
|
140
211
|
throw new sdk_core_1.InvalidTransactionError('Gas payment objects are required');
|
|
141
212
|
}
|
|
142
213
|
const gasObject = this._iotaTransaction.gas;
|
|
214
|
+
// If within the limit, no consolidation needed
|
|
143
215
|
if (this.gasPaymentObjects.length <= constants_1.MAX_GAS_PAYMENT_OBJECTS) {
|
|
144
216
|
return gasObject;
|
|
145
217
|
}
|
|
218
|
+
// Merge excess gas objects to stay within limits
|
|
146
219
|
return this.mergeObjectsInBatches(gasObject, this.gasPaymentObjects, constants_1.MAX_INPUT_OBJECTS);
|
|
147
220
|
}
|
|
221
|
+
/**
|
|
222
|
+
* Merges multiple coin objects into a target coin in batches.
|
|
223
|
+
* This is necessary because IOTA has limits on the number of objects per merge command.
|
|
224
|
+
*
|
|
225
|
+
* @param targetCoin - The coin to merge into
|
|
226
|
+
* @param objectsToMerge - Array of objects to merge
|
|
227
|
+
* @param batchSize - Maximum number of objects to merge per batch
|
|
228
|
+
* @returns The consolidated coin object
|
|
229
|
+
*/
|
|
148
230
|
mergeObjectsInBatches(targetCoin, objectsToMerge, batchSize) {
|
|
149
231
|
let consolidatedCoin = targetCoin;
|
|
150
232
|
for (let startIndex = 0; startIndex < objectsToMerge.length; startIndex += batchSize) {
|
|
@@ -154,6 +236,11 @@ class TransferTransaction extends transaction_1.Transaction {
|
|
|
154
236
|
}
|
|
155
237
|
return consolidatedCoin;
|
|
156
238
|
}
|
|
239
|
+
/**
|
|
240
|
+
* Splits the source coin into the amounts needed for each recipient and transfers them.
|
|
241
|
+
* This creates split coin commands for all recipient amounts, then transfer commands
|
|
242
|
+
* to send each split coin to the corresponding recipient.
|
|
243
|
+
*/
|
|
157
244
|
splitAndTransferToRecipients(sourceCoin) {
|
|
158
245
|
const recipientAmounts = this._recipients.map((recipient) => recipient.amount);
|
|
159
246
|
const splitCoins = this._iotaTransaction.splitCoins(sourceCoin, recipientAmounts);
|
|
@@ -161,55 +248,97 @@ class TransferTransaction extends transaction_1.Transaction {
|
|
|
161
248
|
this._iotaTransaction.transferObjects([splitCoins[index]], recipient.address);
|
|
162
249
|
});
|
|
163
250
|
}
|
|
251
|
+
/**
|
|
252
|
+
* Validates all transfer transaction data before building.
|
|
253
|
+
* Checks recipients, payment objects, and ensures no duplicate object IDs.
|
|
254
|
+
*/
|
|
164
255
|
validateTxDataImplementation() {
|
|
165
|
-
|
|
256
|
+
this.validateRecipientsList();
|
|
257
|
+
this.validatePaymentObjectsExist();
|
|
258
|
+
this.validateNoDuplicateObjects();
|
|
259
|
+
}
|
|
260
|
+
/**
|
|
261
|
+
* Validates that recipients exist and don't exceed the maximum allowed.
|
|
262
|
+
*/
|
|
263
|
+
validateRecipientsList() {
|
|
264
|
+
if (!this.recipients || this.recipients.length === 0) {
|
|
166
265
|
throw new sdk_core_1.InvalidTransactionError('Transaction recipients are required');
|
|
167
266
|
}
|
|
168
267
|
if (this.recipients.length > constants_1.MAX_RECIPIENTS) {
|
|
169
268
|
throw new sdk_core_1.InvalidTransactionError(`Recipients count (${this.recipients.length}) exceeds maximum allowed (${constants_1.MAX_RECIPIENTS})`);
|
|
170
269
|
}
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
179
|
-
|
|
270
|
+
}
|
|
271
|
+
/**
|
|
272
|
+
* Validates that either payment objects or gas objects exist for funding the transfer.
|
|
273
|
+
* When a gas sponsor is used, payment objects are required.
|
|
274
|
+
* Otherwise, either payment objects or gas objects can be used.
|
|
275
|
+
*/
|
|
276
|
+
validatePaymentObjectsExist() {
|
|
277
|
+
const hasPaymentObjects = this.paymentObjects && this.paymentObjects.length > 0;
|
|
278
|
+
if (hasPaymentObjects) {
|
|
279
|
+
return; // Payment objects exist, validation passes
|
|
180
280
|
}
|
|
181
|
-
//
|
|
281
|
+
// No payment objects - check if gas objects can be used instead
|
|
282
|
+
const hasGasSponsor = this.gasSponsor && this.gasSponsor !== this.sender;
|
|
283
|
+
if (hasGasSponsor) {
|
|
284
|
+
throw new sdk_core_1.InvalidTransactionError('Payment objects are required when using a gas sponsor');
|
|
285
|
+
}
|
|
286
|
+
// No gas sponsor - gas objects must exist
|
|
287
|
+
const hasGasObjects = this.gasPaymentObjects && this.gasPaymentObjects.length > 0;
|
|
288
|
+
if (!hasGasObjects) {
|
|
289
|
+
throw new sdk_core_1.InvalidTransactionError('Payment or Gas objects are required');
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
/**
|
|
293
|
+
* Validates that there are no duplicate object IDs within payment objects,
|
|
294
|
+
* gas payment objects, or between the two groups.
|
|
295
|
+
*/
|
|
296
|
+
validateNoDuplicateObjects() {
|
|
182
297
|
const paymentObjectIds = this.paymentObjects?.map((obj) => obj.objectId) ?? [];
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
298
|
+
// Check for duplicates within payment objects
|
|
299
|
+
this.checkForDuplicateIds(paymentObjectIds, 'payment objects');
|
|
300
|
+
if (!this.gasPaymentObjects || this.gasPaymentObjects.length === 0) {
|
|
301
|
+
return;
|
|
302
|
+
}
|
|
303
|
+
const gasObjectIds = this.gasPaymentObjects.map((gas) => gas.objectId);
|
|
304
|
+
// Check for duplicates within gas payment objects
|
|
305
|
+
this.checkForDuplicateIds(gasObjectIds, 'gas payment objects');
|
|
306
|
+
// Check for overlaps between payment and gas objects
|
|
307
|
+
const overlappingIds = paymentObjectIds.filter((id) => gasObjectIds.includes(id));
|
|
308
|
+
if (overlappingIds.length > 0) {
|
|
309
|
+
throw new sdk_core_1.InvalidTransactionError('Payment objects cannot be the same as gas payment objects: ' + overlappingIds.join(', '));
|
|
310
|
+
}
|
|
311
|
+
}
|
|
312
|
+
/**
|
|
313
|
+
* Helper to check for duplicate IDs in an array.
|
|
314
|
+
*/
|
|
315
|
+
checkForDuplicateIds(ids, objectType) {
|
|
316
|
+
const uniqueIds = new Set(ids);
|
|
317
|
+
if (uniqueIds.size !== ids.length) {
|
|
318
|
+
throw new sdk_core_1.InvalidTransactionError(`Duplicate object IDs found in ${objectType}`);
|
|
198
319
|
}
|
|
199
320
|
}
|
|
200
321
|
/**
|
|
201
322
|
* @inheritDoc
|
|
323
|
+
* Provides a human-readable explanation of the transfer transaction.
|
|
202
324
|
*/
|
|
203
|
-
explainTransactionImplementation(
|
|
325
|
+
explainTransactionImplementation(_json, explanationResult) {
|
|
204
326
|
const outputs = this.recipients.map((recipient) => recipient);
|
|
205
|
-
const
|
|
206
|
-
const outputAmount = outputAmountBN.toString();
|
|
327
|
+
const outputAmount = this.calculateTotalOutputAmount();
|
|
207
328
|
return {
|
|
208
329
|
...explanationResult,
|
|
209
330
|
outputAmount,
|
|
210
331
|
outputs,
|
|
211
332
|
};
|
|
212
333
|
}
|
|
334
|
+
/**
|
|
335
|
+
* Calculates the total amount being transferred to all recipients.
|
|
336
|
+
*/
|
|
337
|
+
calculateTotalOutputAmount() {
|
|
338
|
+
return this.recipients
|
|
339
|
+
.reduce((accumulator, current) => accumulator.plus(current.amount), new bignumber_js_1.default(0))
|
|
340
|
+
.toString();
|
|
341
|
+
}
|
|
213
342
|
}
|
|
214
343
|
exports.TransferTransaction = TransferTransaction;
|
|
215
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transferTransaction.js","sourceRoot":"","sources":["../../../src/lib/transferTransaction.ts"],"names":[],"mappings":"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","sourcesContent":["import { InvalidTransactionError, toHex, TransactionRecipient, TransactionType } from '@bitgo-beta/sdk-core';\nimport { BaseCoin as CoinConfig } from '@bitgo-beta/statics';\nimport { Transaction } from './transaction';\nimport { TransactionExplanation, TransactionObjectInput, TransferTxData, TxData } from './iface';\nimport {\n  Inputs as IotaInputs,\n  TransactionObjectArgument,\n  Transaction as IotaTransaction,\n} from '@iota/iota-sdk/transactions';\nimport { fromBase64 } from '@iota/iota-sdk/utils';\nimport { messageWithIntent as iotaMessageWithIntent } from '@iota/iota-sdk/cryptography';\nimport { BcsReader } from '@iota/bcs';\nimport { MAX_GAS_PAYMENT_OBJECTS, MAX_INPUT_OBJECTS, MAX_RECIPIENTS, TRANSFER_TRANSACTION_COMMANDS } from './constants';\nimport utils from './utils';\nimport BigNumber from 'bignumber.js';\n\nexport class TransferTransaction extends Transaction {\n  private _recipients: TransactionRecipient[];\n  private _paymentObjects?: TransactionObjectInput[];\n\n  constructor(coinConfig: Readonly<CoinConfig>) {\n    super(coinConfig);\n    this._type = TransactionType.Send;\n    this._inputs = [];\n    this._outputs = [];\n  }\n\n  get recipients(): TransactionRecipient[] {\n    return this._recipients;\n  }\n\n  set recipients(value: TransactionRecipient[]) {\n    this._recipients = value;\n    this._rebuildRequired = true;\n  }\n\n  get paymentObjects(): TransactionObjectInput[] | undefined {\n    return this._paymentObjects;\n  }\n\n  set paymentObjects(value: TransactionObjectInput[] | undefined) {\n    this._paymentObjects = value;\n    this._rebuildRequired = true;\n  }\n\n  /**\n   * @inheritDoc\n   */\n  addInputsAndOutputs(): void {\n    if (!this._iotaTransaction) {\n      return;\n    }\n    const totalAmount = this.recipients.reduce((accumulator, current) => accumulator + Number(current.amount), 0);\n    this._outputs = this.recipients.map((recipient, index) => ({\n      address: recipient.address,\n      value: recipient.amount.toString(),\n      coin: this._coinConfig.name,\n    }));\n    this._inputs = [\n      {\n        address: this.sender,\n        value: totalAmount.toString(),\n        coin: this._coinConfig.name,\n      },\n    ];\n  }\n\n  toJson(): TransferTxData {\n    return {\n      ...super.toJson(),\n      recipients: this.recipients,\n      paymentObjects: this.paymentObjects,\n    };\n  }\n\n  parseFromJSON(txData: TransferTxData): void {\n    super.parseFromJSON(txData);\n    this.recipients = txData.recipients;\n    this.paymentObjects = txData.paymentObjects;\n    this.updateIsSimulateTx();\n  }\n\n  parseFromBroadcastTx(tx: string | Uint8Array): void {\n    const txData = IotaTransaction.from(tx).getData();\n    if (txData.commands.filter((command) => !TRANSFER_TRANSACTION_COMMANDS.includes(command.$kind)).length > 0) {\n      throw new InvalidTransactionError('Unsupported commands in the transaction');\n    }\n    super.parseFromBroadcastTx(tx);\n    const inputObjects: TransactionObjectInput[] = [];\n    const amounts: string[] = [];\n    const receivers: string[] = [];\n    txData.inputs.forEach((input) => {\n      if (input.$kind === 'Object' && 'ImmOrOwnedObject' in input.Object) {\n        inputObjects.push(input.Object.ImmOrOwnedObject as TransactionObjectInput);\n      }\n      if (input.$kind === 'Pure' && 'bytes' in input.Pure) {\n        const value = fromBase64(input.Pure.bytes);\n        const hexValue = '0x' + toHex(value);\n        if (utils.isValidAddress(hexValue)) {\n          receivers.push(hexValue);\n        } else {\n          amounts.push(new BcsReader(value).read64().toString());\n        }\n      }\n    });\n    if (amounts.length !== receivers.length) {\n      throw new InvalidTransactionError('count of amounts does not match count of receivers');\n    }\n    if (\n      (!this.gasPaymentObjects || this.gasPaymentObjects.length === 0) &&\n      (!this.gasSponsor || this.sender === this.gasSponsor)\n    ) {\n      this.gasPaymentObjects = inputObjects;\n    } else if (inputObjects.length > 0) {\n      this._paymentObjects = inputObjects;\n    }\n    this.recipients = [];\n    receivers.forEach((recipient, index) => {\n      this._recipients.push({\n        address: recipient,\n        amount: amounts[index],\n      });\n    });\n    this.updateIsSimulateTx();\n  }\n\n  protected messageWithIntent(message: Uint8Array): Uint8Array {\n    return iotaMessageWithIntent('TransactionData', message);\n  }\n\n  protected populateTxInputsAndCommands(): void {\n    const consolidatedCoin = this.shouldUseGasObjectsForPayment()\n      ? this.consolidateGasObjects()\n      : this.consolidatePaymentObjects();\n    this.splitAndTransferToRecipients(consolidatedCoin);\n  }\n\n  private shouldUseGasObjectsForPayment(): boolean {\n    const paymentObjectExists = this.paymentObjects && this.paymentObjects.length > 0;\n    const senderPaysOwnGas = !this.gasSponsor || this.gasSponsor === this.sender;\n    return !paymentObjectExists && senderPaysOwnGas && Boolean(this.gasPaymentObjects);\n  }\n\n  private consolidatePaymentObjects(): TransactionObjectArgument {\n    if (!this.paymentObjects || this.paymentObjects.length === 0) {\n      throw new InvalidTransactionError('Payment objects are required');\n    }\n\n    const firstObject = this._iotaTransaction.object(IotaInputs.ObjectRef(this.paymentObjects[0]));\n\n    if (this.paymentObjects.length === 1) {\n      return firstObject;\n    }\n\n    return this.mergeObjectsInBatches(firstObject, this.paymentObjects.slice(1), MAX_INPUT_OBJECTS);\n  }\n\n  private consolidateGasObjects(): TransactionObjectArgument {\n    if (!this.gasPaymentObjects || this.gasPaymentObjects.length === 0) {\n      throw new InvalidTransactionError('Gas payment objects are required');\n    }\n\n    const gasObject = this._iotaTransaction.gas;\n\n    if (this.gasPaymentObjects.length <= MAX_GAS_PAYMENT_OBJECTS) {\n      return gasObject;\n    }\n\n    return this.mergeObjectsInBatches(gasObject, this.gasPaymentObjects, MAX_INPUT_OBJECTS);\n  }\n\n  private mergeObjectsInBatches(\n    targetCoin: TransactionObjectArgument,\n    objectsToMerge: TransactionObjectInput[],\n    batchSize: number\n  ): TransactionObjectArgument {\n    let consolidatedCoin = targetCoin;\n\n    for (let startIndex = 0; startIndex < objectsToMerge.length; startIndex += batchSize) {\n      const batch = objectsToMerge.slice(startIndex, startIndex + batchSize);\n      const batchAsTransactionObjects = batch.map((obj) => this._iotaTransaction.object(IotaInputs.ObjectRef(obj)));\n\n      [consolidatedCoin] = this._iotaTransaction.mergeCoins(consolidatedCoin, batchAsTransactionObjects);\n    }\n\n    return consolidatedCoin;\n  }\n\n  private splitAndTransferToRecipients(sourceCoin: TransactionObjectArgument): void {\n    const recipientAmounts = this._recipients.map((recipient) => recipient.amount);\n    const splitCoins = this._iotaTransaction.splitCoins(sourceCoin, recipientAmounts);\n\n    this._recipients.forEach((recipient, index) => {\n      this._iotaTransaction.transferObjects([splitCoins[index]], recipient.address);\n    });\n  }\n\n  protected validateTxDataImplementation(): void {\n    if (!this.recipients || this.recipients?.length === 0) {\n      throw new InvalidTransactionError('Transaction recipients are required');\n    }\n\n    if (this.recipients.length > MAX_RECIPIENTS) {\n      throw new InvalidTransactionError(\n        `Recipients count (${this.recipients.length}) exceeds maximum allowed (${MAX_RECIPIENTS})`\n      );\n    }\n\n    if (!this.paymentObjects || this.paymentObjects?.length === 0) {\n      if (!this.gasSponsor || this.gasSponsor === this.sender) {\n        if (!this.gasPaymentObjects || this.gasPaymentObjects?.length === 0) {\n          throw new InvalidTransactionError('Payment or Gas objects are required');\n        }\n      } else {\n        throw new InvalidTransactionError('Payment objects are required');\n      }\n    }\n\n    // Check for duplicate object IDs in payment objects\n    const paymentObjectIds = this.paymentObjects?.map((obj) => obj.objectId) ?? [];\n    const uniquePaymentIds = new Set(paymentObjectIds);\n    if (uniquePaymentIds.size !== paymentObjectIds.length) {\n      throw new InvalidTransactionError('Duplicate object IDs found in payment objects');\n    }\n\n    if (this.gasPaymentObjects && this.gasPaymentObjects.length > 0) {\n      const gasObjectIds = this.gasPaymentObjects.map((gas) => gas.objectId);\n\n      // Check for duplicate object IDs in gas payment objects\n      const uniqueGasIds = new Set(gasObjectIds);\n      if (uniqueGasIds.size !== gasObjectIds.length) {\n        throw new InvalidTransactionError('Duplicate object IDs found in gas payment objects');\n      }\n\n      const duplicates = paymentObjectIds.filter((payment) => gasObjectIds.includes(payment));\n      if (duplicates.length > 0) {\n        throw new InvalidTransactionError(\n          'Payment objects cannot be the same as gas payment objects: ' + duplicates.join(', ')\n        );\n      }\n    }\n  }\n\n  /**\n   * @inheritDoc\n   */\n  protected explainTransactionImplementation(\n    json: TxData,\n    explanationResult: TransactionExplanation\n  ): TransactionExplanation {\n    const outputs: TransactionRecipient[] = this.recipients.map((recipient) => recipient);\n    const outputAmountBN = this.recipients.reduce(\n      (accumulator, current) => accumulator.plus(current.amount),\n      new BigNumber(0)\n    );\n    const outputAmount = outputAmountBN.toString();\n\n    return {\n      ...explanationResult,\n      outputAmount,\n      outputs,\n    };\n  }\n}\n"]}
|
|
344
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transferTransaction.js","sourceRoot":"","sources":["../../../src/lib/transferTransaction.ts"],"names":[],"mappings":"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","sourcesContent":["import { InvalidTransactionError, toHex, TransactionRecipient, TransactionType } from '@bitgo-beta/sdk-core';\nimport { BaseCoin as CoinConfig } from '@bitgo-beta/statics';\nimport { Transaction } from './transaction';\nimport { TransactionExplanation, TransactionObjectInput, TransferTxData, TxData } from './iface';\nimport {\n  Inputs as IotaInputs,\n  TransactionObjectArgument,\n  Transaction as IotaTransaction,\n} from '@iota/iota-sdk/transactions';\nimport { fromBase64 } from '@iota/iota-sdk/utils';\nimport { messageWithIntent as iotaMessageWithIntent } from '@iota/iota-sdk/cryptography';\nimport { BcsReader } from '@iota/bcs';\nimport { MAX_GAS_PAYMENT_OBJECTS, MAX_INPUT_OBJECTS, MAX_RECIPIENTS, TRANSFER_TRANSACTION_COMMANDS } from './constants';\nimport utils from './utils';\nimport BigNumber from 'bignumber.js';\n\nexport class TransferTransaction extends Transaction {\n  private _recipients: TransactionRecipient[];\n  private _paymentObjects?: TransactionObjectInput[];\n\n  constructor(coinConfig: Readonly<CoinConfig>) {\n    super(coinConfig);\n    this._type = TransactionType.Send;\n    this._inputs = [];\n    this._outputs = [];\n  }\n\n  get recipients(): TransactionRecipient[] {\n    return this._recipients;\n  }\n\n  set recipients(value: TransactionRecipient[]) {\n    this._recipients = value;\n    this._rebuildRequired = true;\n  }\n\n  get paymentObjects(): TransactionObjectInput[] | undefined {\n    return this._paymentObjects;\n  }\n\n  set paymentObjects(value: TransactionObjectInput[] | undefined) {\n    this._paymentObjects = value;\n    this._rebuildRequired = true;\n  }\n\n  /**\n   * @inheritDoc\n   */\n  addInputsAndOutputs(): void {\n    if (!this._iotaTransaction) {\n      return;\n    }\n    const totalAmount = this.recipients.reduce((accumulator, current) => accumulator + Number(current.amount), 0);\n    this._outputs = this.recipients.map((recipient, index) => ({\n      address: recipient.address,\n      value: recipient.amount.toString(),\n      coin: this._coinConfig.name,\n    }));\n    this._inputs = [\n      {\n        address: this.sender,\n        value: totalAmount.toString(),\n        coin: this._coinConfig.name,\n      },\n    ];\n  }\n\n  toJson(): TransferTxData {\n    return {\n      ...super.toJson(),\n      recipients: this.recipients,\n      paymentObjects: this.paymentObjects,\n    };\n  }\n\n  parseFromJSON(txData: TransferTxData): void {\n    super.parseFromJSON(txData);\n    this.recipients = txData.recipients;\n    this.paymentObjects = txData.paymentObjects;\n    this.updateIsSimulateTx();\n  }\n\n  /**\n   * Parses a transfer transaction from its broadcast format (base64 or raw bytes).\n   * Extracts recipients, amounts, and payment objects from the transaction data.\n   */\n  parseFromBroadcastTx(tx: string | Uint8Array): void {\n    const txData = IotaTransaction.from(tx).getData();\n    this.validateTransferCommands(txData);\n\n    super.parseFromBroadcastTx(tx);\n\n    const { inputObjects, amounts, receivers } = this.parseTransactionInputs(txData);\n    this.validateAmountsMatchReceivers(amounts, receivers);\n    this.assignParsedObjects(inputObjects);\n    this.assignRecipients(receivers, amounts);\n    this.updateIsSimulateTx();\n  }\n\n  /**\n   * Validates that the transaction only contains supported transfer commands.\n   */\n  private validateTransferCommands(txData: ReturnType<IotaTransaction['getData']>): void {\n    const hasUnsupportedCommands = txData.commands.some(\n      (command) => !TRANSFER_TRANSACTION_COMMANDS.includes(command.$kind)\n    );\n\n    if (hasUnsupportedCommands) {\n      throw new InvalidTransactionError('Unsupported commands in the transaction');\n    }\n  }\n\n  /**\n   * Parses transaction inputs to extract objects, amounts, and receiver addresses.\n   */\n  private parseTransactionInputs(txData: ReturnType<IotaTransaction['getData']>): {\n    inputObjects: TransactionObjectInput[];\n    amounts: string[];\n    receivers: string[];\n  } {\n    const inputObjects: TransactionObjectInput[] = [];\n    const amounts: string[] = [];\n    const receivers: string[] = [];\n\n    txData.inputs.forEach((input) => {\n      if (input.$kind === 'Object' && 'ImmOrOwnedObject' in input.Object) {\n        inputObjects.push(input.Object.ImmOrOwnedObject as TransactionObjectInput);\n      }\n\n      if (input.$kind === 'Pure' && 'bytes' in input.Pure) {\n        const value = fromBase64(input.Pure.bytes);\n        const hexValue = '0x' + toHex(value);\n\n        if (utils.isValidAddress(hexValue)) {\n          receivers.push(hexValue);\n        } else {\n          amounts.push(new BcsReader(value).read64().toString());\n        }\n      }\n    });\n\n    return { inputObjects, amounts, receivers };\n  }\n\n  /**\n   * Validates that the number of amounts matches the number of receivers.\n   */\n  private validateAmountsMatchReceivers(amounts: string[], receivers: string[]): void {\n    if (amounts.length !== receivers.length) {\n      throw new InvalidTransactionError('Count of amounts does not match count of receivers');\n    }\n  }\n\n  /**\n   * Assigns parsed input objects to either gas payment objects or payment objects.\n   * If no gas objects exist and sender pays own gas, objects become gas objects.\n   * Otherwise, they become payment objects.\n   */\n  private assignParsedObjects(inputObjects: TransactionObjectInput[]): void {\n    if (inputObjects.length === 0) {\n      return;\n    }\n\n    const noGasObjectsExist = !this.gasPaymentObjects || this.gasPaymentObjects.length === 0;\n    const senderPaysOwnGas = !this.gasSponsor || this.sender === this.gasSponsor;\n\n    if (noGasObjectsExist && senderPaysOwnGas) {\n      this.gasPaymentObjects = inputObjects;\n    } else {\n      this._paymentObjects = inputObjects;\n    }\n  }\n\n  /**\n   * Creates and assigns recipients from parsed addresses and amounts.\n   */\n  private assignRecipients(receivers: string[], amounts: string[]): void {\n    this._recipients = receivers.map((address, index) => ({\n      address,\n      amount: amounts[index],\n    }));\n  }\n\n  protected messageWithIntent(message: Uint8Array): Uint8Array {\n    return iotaMessageWithIntent('TransactionData', message);\n  }\n\n  /**\n   * Populates the IOTA transaction with inputs and commands for the transfer.\n   * This determines which objects to use for payment (either payment objects or gas objects),\n   * consolidates them into a single coin, and then splits/transfers to recipients.\n   */\n  protected populateTxInputsAndCommands(): void {\n    const sourceCoin = this.getConsolidatedSourceCoin();\n    this.splitAndTransferToRecipients(sourceCoin);\n  }\n\n  /**\n   * Determines which objects to use as the payment source and consolidates them.\n   * If payment objects are provided, use those. Otherwise, if the sender is paying\n   * their own gas, use the gas objects for payment.\n   */\n  private getConsolidatedSourceCoin(): TransactionObjectArgument {\n    if (this.hasPaymentObjects()) {\n      return this.consolidatePaymentObjects();\n    }\n    return this.consolidateGasObjects();\n  }\n\n  /**\n   * Checks if payment objects exist and if gas objects should be used instead.\n   * Gas objects are used when: no payment objects exist AND sender pays their own gas.\n   */\n  private hasPaymentObjects(): boolean {\n    const hasPaymentObjects = this.paymentObjects && this.paymentObjects.length > 0;\n    if (hasPaymentObjects) {\n      return true;\n    }\n\n    // If no payment objects, only use gas objects if sender pays own gas\n    const senderPaysOwnGas = !this.gasSponsor || this.gasSponsor === this.sender;\n    return !senderPaysOwnGas;\n  }\n\n  /**\n   * Consolidates payment objects into a single coin object.\n   * If multiple payment objects exist, they are merged in batches.\n   */\n  private consolidatePaymentObjects(): TransactionObjectArgument {\n    if (!this.paymentObjects || this.paymentObjects.length === 0) {\n      throw new InvalidTransactionError('Payment objects are required');\n    }\n\n    const firstObject = this._iotaTransaction.object(IotaInputs.ObjectRef(this.paymentObjects[0]));\n\n    // Single object doesn't need consolidation\n    if (this.paymentObjects.length === 1) {\n      return firstObject;\n    }\n\n    // Merge remaining objects into the first one\n    return this.mergeObjectsInBatches(firstObject, this.paymentObjects.slice(1), MAX_INPUT_OBJECTS);\n  }\n\n  /**\n   * Consolidates gas payment objects into a single coin object.\n   * If the number of gas objects exceeds the maximum, they are merged in batches.\n   */\n  private consolidateGasObjects(): TransactionObjectArgument {\n    if (!this.gasPaymentObjects || this.gasPaymentObjects.length === 0) {\n      throw new InvalidTransactionError('Gas payment objects are required');\n    }\n\n    const gasObject = this._iotaTransaction.gas;\n\n    // If within the limit, no consolidation needed\n    if (this.gasPaymentObjects.length <= MAX_GAS_PAYMENT_OBJECTS) {\n      return gasObject;\n    }\n\n    // Merge excess gas objects to stay within limits\n    return this.mergeObjectsInBatches(gasObject, this.gasPaymentObjects, MAX_INPUT_OBJECTS);\n  }\n\n  /**\n   * Merges multiple coin objects into a target coin in batches.\n   * This is necessary because IOTA has limits on the number of objects per merge command.\n   *\n   * @param targetCoin - The coin to merge into\n   * @param objectsToMerge - Array of objects to merge\n   * @param batchSize - Maximum number of objects to merge per batch\n   * @returns The consolidated coin object\n   */\n  private mergeObjectsInBatches(\n    targetCoin: TransactionObjectArgument,\n    objectsToMerge: TransactionObjectInput[],\n    batchSize: number\n  ): TransactionObjectArgument {\n    let consolidatedCoin = targetCoin;\n\n    for (let startIndex = 0; startIndex < objectsToMerge.length; startIndex += batchSize) {\n      const batch = objectsToMerge.slice(startIndex, startIndex + batchSize);\n      const batchAsTransactionObjects = batch.map((obj) => this._iotaTransaction.object(IotaInputs.ObjectRef(obj)));\n\n      [consolidatedCoin] = this._iotaTransaction.mergeCoins(consolidatedCoin, batchAsTransactionObjects);\n    }\n\n    return consolidatedCoin;\n  }\n\n  /**\n   * Splits the source coin into the amounts needed for each recipient and transfers them.\n   * This creates split coin commands for all recipient amounts, then transfer commands\n   * to send each split coin to the corresponding recipient.\n   */\n  private splitAndTransferToRecipients(sourceCoin: TransactionObjectArgument): void {\n    const recipientAmounts = this._recipients.map((recipient) => recipient.amount);\n    const splitCoins = this._iotaTransaction.splitCoins(sourceCoin, recipientAmounts);\n\n    this._recipients.forEach((recipient, index) => {\n      this._iotaTransaction.transferObjects([splitCoins[index]], recipient.address);\n    });\n  }\n\n  /**\n   * Validates all transfer transaction data before building.\n   * Checks recipients, payment objects, and ensures no duplicate object IDs.\n   */\n  protected validateTxDataImplementation(): void {\n    this.validateRecipientsList();\n    this.validatePaymentObjectsExist();\n    this.validateNoDuplicateObjects();\n  }\n\n  /**\n   * Validates that recipients exist and don't exceed the maximum allowed.\n   */\n  private validateRecipientsList(): void {\n    if (!this.recipients || this.recipients.length === 0) {\n      throw new InvalidTransactionError('Transaction recipients are required');\n    }\n\n    if (this.recipients.length > MAX_RECIPIENTS) {\n      throw new InvalidTransactionError(\n        `Recipients count (${this.recipients.length}) exceeds maximum allowed (${MAX_RECIPIENTS})`\n      );\n    }\n  }\n\n  /**\n   * Validates that either payment objects or gas objects exist for funding the transfer.\n   * When a gas sponsor is used, payment objects are required.\n   * Otherwise, either payment objects or gas objects can be used.\n   */\n  private validatePaymentObjectsExist(): void {\n    const hasPaymentObjects = this.paymentObjects && this.paymentObjects.length > 0;\n    if (hasPaymentObjects) {\n      return; // Payment objects exist, validation passes\n    }\n\n    // No payment objects - check if gas objects can be used instead\n    const hasGasSponsor = this.gasSponsor && this.gasSponsor !== this.sender;\n    if (hasGasSponsor) {\n      throw new InvalidTransactionError('Payment objects are required when using a gas sponsor');\n    }\n\n    // No gas sponsor - gas objects must exist\n    const hasGasObjects = this.gasPaymentObjects && this.gasPaymentObjects.length > 0;\n    if (!hasGasObjects) {\n      throw new InvalidTransactionError('Payment or Gas objects are required');\n    }\n  }\n\n  /**\n   * Validates that there are no duplicate object IDs within payment objects,\n   * gas payment objects, or between the two groups.\n   */\n  private validateNoDuplicateObjects(): void {\n    const paymentObjectIds = this.paymentObjects?.map((obj) => obj.objectId) ?? [];\n\n    // Check for duplicates within payment objects\n    this.checkForDuplicateIds(paymentObjectIds, 'payment objects');\n\n    if (!this.gasPaymentObjects || this.gasPaymentObjects.length === 0) {\n      return;\n    }\n\n    const gasObjectIds = this.gasPaymentObjects.map((gas) => gas.objectId);\n\n    // Check for duplicates within gas payment objects\n    this.checkForDuplicateIds(gasObjectIds, 'gas payment objects');\n\n    // Check for overlaps between payment and gas objects\n    const overlappingIds = paymentObjectIds.filter((id) => gasObjectIds.includes(id));\n    if (overlappingIds.length > 0) {\n      throw new InvalidTransactionError(\n        'Payment objects cannot be the same as gas payment objects: ' + overlappingIds.join(', ')\n      );\n    }\n  }\n\n  /**\n   * Helper to check for duplicate IDs in an array.\n   */\n  private checkForDuplicateIds(ids: string[], objectType: string): void {\n    const uniqueIds = new Set(ids);\n    if (uniqueIds.size !== ids.length) {\n      throw new InvalidTransactionError(`Duplicate object IDs found in ${objectType}`);\n    }\n  }\n\n  /**\n   * @inheritDoc\n   * Provides a human-readable explanation of the transfer transaction.\n   */\n  protected explainTransactionImplementation(\n    _json: TxData,\n    explanationResult: TransactionExplanation\n  ): TransactionExplanation {\n    const outputs = this.recipients.map((recipient) => recipient);\n    const outputAmount = this.calculateTotalOutputAmount();\n\n    return {\n      ...explanationResult,\n      outputAmount,\n      outputs,\n    };\n  }\n\n  /**\n   * Calculates the total amount being transferred to all recipients.\n   */\n  private calculateTotalOutputAmount(): string {\n    return this.recipients\n      .reduce((accumulator, current) => accumulator.plus(current.amount), new BigNumber(0))\n      .toString();\n  }\n}\n"]}
|
package/dist/src/lib/utils.d.ts
CHANGED
|
@@ -1,22 +1,121 @@
|
|
|
1
1
|
import { BaseUtils } from '@bitgo-beta/sdk-core';
|
|
2
|
+
/**
|
|
3
|
+
* Utility class for IOTA-specific validation and conversion operations.
|
|
4
|
+
* Implements the BaseUtils interface and provides methods for validating
|
|
5
|
+
* addresses, keys, signatures, and transaction data.
|
|
6
|
+
*/
|
|
2
7
|
export declare class Utils implements BaseUtils {
|
|
3
|
-
/**
|
|
8
|
+
/**
|
|
9
|
+
* Validates an IOTA address format.
|
|
10
|
+
* IOTA addresses are 64-character hex strings prefixed with '0x'.
|
|
11
|
+
*
|
|
12
|
+
* @param address - The address to validate
|
|
13
|
+
* @returns true if the address is valid
|
|
14
|
+
*
|
|
15
|
+
* @example
|
|
16
|
+
* ```typescript
|
|
17
|
+
* utils.isValidAddress('0x1234...') // true
|
|
18
|
+
* utils.isValidAddress('invalid') // false
|
|
19
|
+
* ```
|
|
20
|
+
*/
|
|
4
21
|
isValidAddress(address: string): boolean;
|
|
5
|
-
/**
|
|
22
|
+
/**
|
|
23
|
+
* Validates an IOTA block ID (digest).
|
|
24
|
+
* Block IDs are 32-byte base58-encoded strings.
|
|
25
|
+
*
|
|
26
|
+
* @param hash - The block ID to validate
|
|
27
|
+
* @returns true if the block ID is valid
|
|
28
|
+
*/
|
|
6
29
|
isValidBlockId(hash: string): boolean;
|
|
7
|
-
/**
|
|
30
|
+
/**
|
|
31
|
+
* Validates an IOTA transaction ID (digest).
|
|
32
|
+
* Transaction IDs are 32-byte base58-encoded strings.
|
|
33
|
+
*
|
|
34
|
+
* @param txId - The transaction ID to validate
|
|
35
|
+
* @returns true if the transaction ID is valid
|
|
36
|
+
*/
|
|
37
|
+
isValidTransactionId(txId: string): boolean;
|
|
38
|
+
/**
|
|
39
|
+
* Validates an Ed25519 private key format.
|
|
40
|
+
*
|
|
41
|
+
* @param key - The private key to validate (hex string)
|
|
42
|
+
* @returns true if the private key is valid
|
|
43
|
+
*/
|
|
8
44
|
isValidPrivateKey(key: string): boolean;
|
|
9
|
-
/**
|
|
45
|
+
/**
|
|
46
|
+
* Validates an Ed25519 public key format.
|
|
47
|
+
*
|
|
48
|
+
* @param key - The public key to validate (hex string)
|
|
49
|
+
* @returns true if the public key is valid
|
|
50
|
+
*/
|
|
10
51
|
isValidPublicKey(key: string): boolean;
|
|
11
|
-
/**
|
|
52
|
+
/**
|
|
53
|
+
* Validates an IOTA signature format.
|
|
54
|
+
* Signatures must be base64-encoded and exactly 64 bytes when decoded.
|
|
55
|
+
*
|
|
56
|
+
* @param signature - The base64-encoded signature to validate
|
|
57
|
+
* @returns true if the signature is valid
|
|
58
|
+
*/
|
|
12
59
|
isValidSignature(signature: string): boolean;
|
|
13
|
-
/**
|
|
14
|
-
|
|
60
|
+
/**
|
|
61
|
+
* Validates a raw IOTA transaction format.
|
|
62
|
+
* Attempts to parse the transaction using the IOTA SDK.
|
|
63
|
+
*
|
|
64
|
+
* @param rawTransaction - The raw transaction (base64 string or Uint8Array)
|
|
65
|
+
* @returns true if the transaction can be parsed
|
|
66
|
+
*/
|
|
67
|
+
isValidRawTransaction(rawTransaction: string | Uint8Array): boolean;
|
|
68
|
+
/**
|
|
69
|
+
* Validates a hex string with a specific length requirement.
|
|
70
|
+
* Checks for '0x' or '0X' prefix followed by the specified number of hex characters.
|
|
71
|
+
*
|
|
72
|
+
* @param value - The hex string to validate
|
|
73
|
+
* @param length - The required length (number of hex characters, excluding prefix)
|
|
74
|
+
* @returns true if the hex string matches the format and length
|
|
75
|
+
*/
|
|
15
76
|
isValidHex(value: string, length: number): boolean;
|
|
77
|
+
/**
|
|
78
|
+
* Converts a value to a base64-encoded string.
|
|
79
|
+
* Handles both Uint8Array and hex string inputs.
|
|
80
|
+
*
|
|
81
|
+
* @param value - The value to encode (Uint8Array or hex string)
|
|
82
|
+
* @returns Base64-encoded string
|
|
83
|
+
*
|
|
84
|
+
* @example
|
|
85
|
+
* ```typescript
|
|
86
|
+
* utils.getBase64String(new Uint8Array([1, 2, 3]))
|
|
87
|
+
* utils.getBase64String('0x010203')
|
|
88
|
+
* ```
|
|
89
|
+
*/
|
|
16
90
|
getBase64String(value: string | Uint8Array): string;
|
|
91
|
+
/**
|
|
92
|
+
* Derives an IOTA address from an Ed25519 public key.
|
|
93
|
+
* Uses the IOTA SDK's address derivation algorithm.
|
|
94
|
+
*
|
|
95
|
+
* @param publicKey - The Ed25519 public key (hex string)
|
|
96
|
+
* @returns The derived IOTA address
|
|
97
|
+
*
|
|
98
|
+
* @example
|
|
99
|
+
* ```typescript
|
|
100
|
+
* const address = utils.getAddressFromPublicKey('8c26e54e36c902c5...')
|
|
101
|
+
* // Returns: '0x9882188ba3e8070a...'
|
|
102
|
+
* ```
|
|
103
|
+
*/
|
|
17
104
|
getAddressFromPublicKey(publicKey: string): string;
|
|
18
|
-
isValidRawTransaction(rawTransaction: string | Uint8Array): boolean;
|
|
19
105
|
}
|
|
106
|
+
/**
|
|
107
|
+
* Singleton instance of the Utils class.
|
|
108
|
+
* Use this for all IOTA utility operations throughout the SDK.
|
|
109
|
+
*
|
|
110
|
+
* @example
|
|
111
|
+
* ```typescript
|
|
112
|
+
* import utils from './utils';
|
|
113
|
+
*
|
|
114
|
+
* if (utils.isValidAddress(address)) {
|
|
115
|
+
* // Process valid address
|
|
116
|
+
* }
|
|
117
|
+
* ```
|
|
118
|
+
*/
|
|
20
119
|
declare const utils: Utils;
|
|
21
120
|
export default utils;
|
|
22
121
|
//# sourceMappingURL=utils.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAA8D,MAAM,sBAAsB,CAAC;AAW7G,qBAAa,KAAM,YAAW,SAAS;
|
|
1
|
+
{"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAA8D,MAAM,sBAAsB,CAAC;AAW7G;;;;GAIG;AACH,qBAAa,KAAM,YAAW,SAAS;IAKrC;;;;;;;;;;;;OAYG;IACH,cAAc,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO;IAIxC;;;;;;OAMG;IACH,cAAc,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO;IAIrC;;;;;;OAMG;IACH,oBAAoB,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO;IAQ3C;;;;;OAKG;IACH,iBAAiB,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO;IAIvC;;;;;OAKG;IACH,gBAAgB,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO;IAQtC;;;;;;OAMG;IACH,gBAAgB,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO;IAU5C;;;;;;OAMG;IACH,qBAAqB,CAAC,cAAc,EAAE,MAAM,GAAG,UAAU,GAAG,OAAO;IAanE;;;;;;;OAOG;IACH,UAAU,CAAC,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,OAAO;IAKlD;;;;;;;;;;;;OAYG;IACH,eAAe,CAAC,KAAK,EAAE,MAAM,GAAG,UAAU,GAAG,MAAM;IAOnD;;;;;;;;;;;;OAYG;IACH,uBAAuB,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM;CAInD;AAED;;;;;;;;;;;;GAYG;AACH,QAAA,MAAM,KAAK,OAAc,CAAC;AAE1B,eAAe,KAAK,CAAC"}
|