@bitgo-beta/sdk-coin-iota 1.0.1-beta.323 → 1.0.1-beta.325
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 -70
- 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 +97 -1
- package/dist/src/lib/transferTransaction.d.ts.map +1 -1
- package/dist/src/lib/transferTransaction.js +223 -52
- 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 +334 -319
- 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,82 +113,232 @@ 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('
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
123
|
+
throw new sdk_core_1.InvalidTransactionError('Count of amounts does not match count of receivers');
|
|
124
|
+
}
|
|
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) {
|
|
138
|
+
this.gasPaymentObjects = inputObjects;
|
|
139
|
+
}
|
|
140
|
+
else {
|
|
141
|
+
this._paymentObjects = inputObjects;
|
|
142
|
+
}
|
|
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
|
+
}));
|
|
107
152
|
}
|
|
108
153
|
messageWithIntent(message) {
|
|
109
154
|
return (0, cryptography_1.messageWithIntent)('TransactionData', message);
|
|
110
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
|
+
*/
|
|
111
161
|
populateTxInputsAndCommands() {
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
162
|
+
const sourceCoin = this.getConsolidatedSourceCoin();
|
|
163
|
+
this.splitAndTransferToRecipients(sourceCoin);
|
|
164
|
+
}
|
|
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;
|
|
127
184
|
}
|
|
185
|
+
// If no payment objects, only use gas objects if sender pays own gas
|
|
186
|
+
const senderPaysOwnGas = !this.gasSponsor || this.gasSponsor === this.sender;
|
|
187
|
+
return !senderPaysOwnGas;
|
|
128
188
|
}
|
|
189
|
+
/**
|
|
190
|
+
* Consolidates payment objects into a single coin object.
|
|
191
|
+
* If multiple payment objects exist, they are merged in batches.
|
|
192
|
+
*/
|
|
193
|
+
consolidatePaymentObjects() {
|
|
194
|
+
if (!this.paymentObjects || this.paymentObjects.length === 0) {
|
|
195
|
+
throw new sdk_core_1.InvalidTransactionError('Payment objects are required');
|
|
196
|
+
}
|
|
197
|
+
const firstObject = this._iotaTransaction.object(transactions_1.Inputs.ObjectRef(this.paymentObjects[0]));
|
|
198
|
+
// Single object doesn't need consolidation
|
|
199
|
+
if (this.paymentObjects.length === 1) {
|
|
200
|
+
return firstObject;
|
|
201
|
+
}
|
|
202
|
+
// Merge remaining objects into the first one
|
|
203
|
+
return this.mergeObjectsInBatches(firstObject, this.paymentObjects.slice(1), constants_1.MAX_INPUT_OBJECTS);
|
|
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
|
+
*/
|
|
209
|
+
consolidateGasObjects() {
|
|
210
|
+
if (!this.gasPaymentObjects || this.gasPaymentObjects.length === 0) {
|
|
211
|
+
throw new sdk_core_1.InvalidTransactionError('Gas payment objects are required');
|
|
212
|
+
}
|
|
213
|
+
const gasObject = this._iotaTransaction.gas;
|
|
214
|
+
// If within the limit, no consolidation needed
|
|
215
|
+
if (this.gasPaymentObjects.length <= constants_1.MAX_GAS_PAYMENT_OBJECTS) {
|
|
216
|
+
return gasObject;
|
|
217
|
+
}
|
|
218
|
+
// Merge excess gas objects to stay within limits
|
|
219
|
+
return this.mergeObjectsInBatches(gasObject, this.gasPaymentObjects, constants_1.MAX_INPUT_OBJECTS);
|
|
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
|
+
*/
|
|
230
|
+
mergeObjectsInBatches(targetCoin, objectsToMerge, batchSize) {
|
|
231
|
+
let consolidatedCoin = targetCoin;
|
|
232
|
+
for (let startIndex = 0; startIndex < objectsToMerge.length; startIndex += batchSize) {
|
|
233
|
+
const batch = objectsToMerge.slice(startIndex, startIndex + batchSize);
|
|
234
|
+
const batchAsTransactionObjects = batch.map((obj) => this._iotaTransaction.object(transactions_1.Inputs.ObjectRef(obj)));
|
|
235
|
+
[consolidatedCoin] = this._iotaTransaction.mergeCoins(consolidatedCoin, batchAsTransactionObjects);
|
|
236
|
+
}
|
|
237
|
+
return consolidatedCoin;
|
|
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
|
+
*/
|
|
244
|
+
splitAndTransferToRecipients(sourceCoin) {
|
|
245
|
+
const recipientAmounts = this._recipients.map((recipient) => recipient.amount);
|
|
246
|
+
const splitCoins = this._iotaTransaction.splitCoins(sourceCoin, recipientAmounts);
|
|
247
|
+
this._recipients.forEach((recipient, index) => {
|
|
248
|
+
this._iotaTransaction.transferObjects([splitCoins[index]], recipient.address);
|
|
249
|
+
});
|
|
250
|
+
}
|
|
251
|
+
/**
|
|
252
|
+
* Validates all transfer transaction data before building.
|
|
253
|
+
* Checks recipients, payment objects, and ensures no duplicate object IDs.
|
|
254
|
+
*/
|
|
129
255
|
validateTxDataImplementation() {
|
|
130
|
-
|
|
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) {
|
|
131
265
|
throw new sdk_core_1.InvalidTransactionError('Transaction recipients are required');
|
|
132
266
|
}
|
|
133
267
|
if (this.recipients.length > constants_1.MAX_RECIPIENTS) {
|
|
134
268
|
throw new sdk_core_1.InvalidTransactionError(`Recipients count (${this.recipients.length}) exceeds maximum allowed (${constants_1.MAX_RECIPIENTS})`);
|
|
135
269
|
}
|
|
136
|
-
|
|
137
|
-
|
|
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
|
|
138
280
|
}
|
|
139
|
-
//
|
|
140
|
-
const
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
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() {
|
|
297
|
+
const paymentObjectIds = this.paymentObjects?.map((obj) => obj.objectId) ?? [];
|
|
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}`);
|
|
156
319
|
}
|
|
157
320
|
}
|
|
158
321
|
/**
|
|
159
322
|
* @inheritDoc
|
|
323
|
+
* Provides a human-readable explanation of the transfer transaction.
|
|
160
324
|
*/
|
|
161
|
-
explainTransactionImplementation(
|
|
325
|
+
explainTransactionImplementation(_json, explanationResult) {
|
|
162
326
|
const outputs = this.recipients.map((recipient) => recipient);
|
|
163
|
-
const
|
|
164
|
-
const outputAmount = outputAmountBN.toString();
|
|
327
|
+
const outputAmount = this.calculateTotalOutputAmount();
|
|
165
328
|
return {
|
|
166
329
|
...explanationResult,
|
|
167
330
|
outputAmount,
|
|
168
331
|
outputs,
|
|
169
332
|
};
|
|
170
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
|
+
}
|
|
171
342
|
}
|
|
172
343
|
exports.TransferTransaction = TransferTransaction;
|
|
173
|
-
//# 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_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    this._paymentObjects = inputObjects;\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    if (this.paymentObjects) {\n      let firstInputObj: TransactionObjectArgument = this._iotaTransaction.object(\n        IotaInputs.ObjectRef(this.paymentObjects[0])\n      );\n      if (this.paymentObjects.length > 1) {\n        let idx = 1;\n        while (idx < this.paymentObjects.length) {\n          [firstInputObj] = this._iotaTransaction.mergeCoins(\n            firstInputObj,\n            this.paymentObjects\n              .slice(idx, idx + MAX_INPUT_OBJECTS)\n              .map((input) => this._iotaTransaction.object(IotaInputs.ObjectRef(input)))\n          );\n          idx += MAX_INPUT_OBJECTS;\n        }\n      }\n\n      const coins = this._iotaTransaction.splitCoins(\n        firstInputObj,\n        this._recipients.map((recipient) => recipient.amount)\n      );\n      this._recipients.forEach((recipient, idx) => {\n        this._iotaTransaction.transferObjects([coins[idx]], recipient.address);\n      });\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      throw new InvalidTransactionError('Payment objects are required');\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 = this.paymentObjects.filter((payment) => gasObjectIds.includes(payment.objectId));\n      if (duplicates.length > 0) {\n        throw new InvalidTransactionError(\n          'Payment objects cannot be the same as gas payment objects: ' + duplicates.map((d) => d.objectId).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"}
|