@bitgo-beta/sdk-coin-iota 1.0.1-beta.324 → 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.
Files changed (42) hide show
  1. package/dist/src/iota.d.ts +123 -27
  2. package/dist/src/iota.d.ts.map +1 -1
  3. package/dist/src/iota.js +217 -103
  4. package/dist/src/lib/constants.d.ts +50 -1
  5. package/dist/src/lib/constants.d.ts.map +1 -1
  6. package/dist/src/lib/constants.js +68 -4
  7. package/dist/src/lib/iface.d.ts +141 -1
  8. package/dist/src/lib/iface.d.ts.map +1 -1
  9. package/dist/src/lib/iface.js +1 -1
  10. package/dist/src/lib/keyPair.d.ts +100 -6
  11. package/dist/src/lib/keyPair.d.ts.map +1 -1
  12. package/dist/src/lib/keyPair.js +103 -10
  13. package/dist/src/lib/transaction.d.ts +117 -14
  14. package/dist/src/lib/transaction.d.ts.map +1 -1
  15. package/dist/src/lib/transaction.js +190 -67
  16. package/dist/src/lib/transactionBuilder.d.ts +73 -34
  17. package/dist/src/lib/transactionBuilder.d.ts.map +1 -1
  18. package/dist/src/lib/transactionBuilder.js +90 -45
  19. package/dist/src/lib/transactionBuilderFactory.d.ts +89 -6
  20. package/dist/src/lib/transactionBuilderFactory.d.ts.map +1 -1
  21. package/dist/src/lib/transactionBuilderFactory.js +103 -16
  22. package/dist/src/lib/transferBuilder.d.ts +43 -0
  23. package/dist/src/lib/transferBuilder.d.ts.map +1 -1
  24. package/dist/src/lib/transferBuilder.js +50 -5
  25. package/dist/src/lib/transferTransaction.d.ts +93 -2
  26. package/dist/src/lib/transferTransaction.d.ts.map +1 -1
  27. package/dist/src/lib/transferTransaction.js +180 -51
  28. package/dist/src/lib/utils.d.ts +107 -8
  29. package/dist/src/lib/utils.d.ts.map +1 -1
  30. package/dist/src/lib/utils.js +134 -23
  31. package/dist/test/unit/helpers/testHelpers.d.ts +57 -0
  32. package/dist/test/unit/helpers/testHelpers.d.ts.map +1 -0
  33. package/dist/test/unit/helpers/testHelpers.js +176 -0
  34. package/dist/test/unit/iota.js +47 -152
  35. package/dist/test/unit/keyPair.js +34 -61
  36. package/dist/test/unit/transactionBuilder/transactionBuilder.js +137 -255
  37. package/dist/test/unit/transactionBuilder/transactionBuilderFactory.js +43 -108
  38. package/dist/test/unit/transactionBuilder/transferBuilder.js +296 -762
  39. package/dist/test/unit/transferTransaction.js +106 -353
  40. package/dist/test/unit/utils.js +171 -197
  41. package/dist/tsconfig.tsbuildinfo +1 -1
  42. 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
- if (txData.commands.filter((command) => !constants_1.TRANSFER_TRANSACTION_COMMANDS.includes(command.$kind)).length > 0) {
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
- super.parseFromBroadcastTx(tx);
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('count of amounts does not match count of receivers');
123
+ throw new sdk_core_1.InvalidTransactionError('Count of amounts does not match count of receivers');
97
124
  }
98
- if ((!this.gasPaymentObjects || this.gasPaymentObjects.length === 0) &&
99
- (!this.gasSponsor || this.sender === this.gasSponsor)) {
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 if (inputObjects.length > 0) {
140
+ else {
103
141
  this._paymentObjects = inputObjects;
104
142
  }
105
- this.recipients = [];
106
- receivers.forEach((recipient, index) => {
107
- this._recipients.push({
108
- address: recipient,
109
- amount: amounts[index],
110
- });
111
- });
112
- this.updateIsSimulateTx();
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 consolidatedCoin = this.shouldUseGasObjectsForPayment()
119
- ? this.consolidateGasObjects()
120
- : this.consolidatePaymentObjects();
121
- this.splitAndTransferToRecipients(consolidatedCoin);
162
+ const sourceCoin = this.getConsolidatedSourceCoin();
163
+ this.splitAndTransferToRecipients(sourceCoin);
122
164
  }
123
- shouldUseGasObjectsForPayment() {
124
- const paymentObjectExists = this.paymentObjects && this.paymentObjects.length > 0;
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 !paymentObjectExists && senderPaysOwnGas && Boolean(this.gasPaymentObjects);
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
- if (!this.recipients || this.recipients?.length === 0) {
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
- if (!this.paymentObjects || this.paymentObjects?.length === 0) {
172
- if (!this.gasSponsor || this.gasSponsor === this.sender) {
173
- if (!this.gasPaymentObjects || this.gasPaymentObjects?.length === 0) {
174
- throw new sdk_core_1.InvalidTransactionError('Payment or Gas objects are required');
175
- }
176
- }
177
- else {
178
- throw new sdk_core_1.InvalidTransactionError('Payment objects are required');
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
- // Check for duplicate object IDs in payment objects
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
- const uniquePaymentIds = new Set(paymentObjectIds);
184
- if (uniquePaymentIds.size !== paymentObjectIds.length) {
185
- throw new sdk_core_1.InvalidTransactionError('Duplicate object IDs found in payment objects');
186
- }
187
- if (this.gasPaymentObjects && this.gasPaymentObjects.length > 0) {
188
- const gasObjectIds = this.gasPaymentObjects.map((gas) => gas.objectId);
189
- // Check for duplicate object IDs in gas payment objects
190
- const uniqueGasIds = new Set(gasObjectIds);
191
- if (uniqueGasIds.size !== gasObjectIds.length) {
192
- throw new sdk_core_1.InvalidTransactionError('Duplicate object IDs found in gas payment objects');
193
- }
194
- const duplicates = paymentObjectIds.filter((payment) => gasObjectIds.includes(payment));
195
- if (duplicates.length > 0) {
196
- throw new sdk_core_1.InvalidTransactionError('Payment objects cannot be the same as gas payment objects: ' + duplicates.join(', '));
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(json, explanationResult) {
325
+ explainTransactionImplementation(_json, explanationResult) {
204
326
  const outputs = this.recipients.map((recipient) => recipient);
205
- const outputAmountBN = this.recipients.reduce((accumulator, current) => accumulator.plus(current.amount), new bignumber_js_1.default(0));
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":";;;;;;AAAA,mDAA6G;AAE7G,+CAA4C;AAE5C,8DAIqC;AACrC,gDAAkD;AAClD,8DAAyF;AACzF,mCAAsC;AACtC,2CAAwH;AACxH,oDAA4B;AAC5B,gEAAqC;AAErC,MAAa,mBAAoB,SAAQ,yBAAW;IAIlD,YAAY,UAAgC;QAC1C,KAAK,CAAC,UAAU,CAAC,CAAC;QAClB,IAAI,CAAC,KAAK,GAAG,0BAAe,CAAC,IAAI,CAAC;QAClC,IAAI,CAAC,OAAO,GAAG,EAAE,CAAC;QAClB,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;IACrB,CAAC;IAED,IAAI,UAAU;QACZ,OAAO,IAAI,CAAC,WAAW,CAAC;IAC1B,CAAC;IAED,IAAI,UAAU,CAAC,KAA6B;QAC1C,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;QACzB,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;IAC/B,CAAC;IAED,IAAI,cAAc;QAChB,OAAO,IAAI,CAAC,eAAe,CAAC;IAC9B,CAAC;IAED,IAAI,cAAc,CAAC,KAA2C;QAC5D,IAAI,CAAC,eAAe,GAAG,KAAK,CAAC;QAC7B,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;IAC/B,CAAC;IAED;;OAEG;IACH,mBAAmB;QACjB,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC3B,OAAO;QACT,CAAC;QACD,MAAM,WAAW,GAAG,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,CAAC,WAAW,EAAE,OAAO,EAAE,EAAE,CAAC,WAAW,GAAG,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC,CAAC;QAC9G,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,SAAS,EAAE,KAAK,EAAE,EAAE,CAAC,CAAC;YACzD,OAAO,EAAE,SAAS,CAAC,OAAO;YAC1B,KAAK,EAAE,SAAS,CAAC,MAAM,CAAC,QAAQ,EAAE;YAClC,IAAI,EAAE,IAAI,CAAC,WAAW,CAAC,IAAI;SAC5B,CAAC,CAAC,CAAC;QACJ,IAAI,CAAC,OAAO,GAAG;YACb;gBACE,OAAO,EAAE,IAAI,CAAC,MAAM;gBACpB,KAAK,EAAE,WAAW,CAAC,QAAQ,EAAE;gBAC7B,IAAI,EAAE,IAAI,CAAC,WAAW,CAAC,IAAI;aAC5B;SACF,CAAC;IACJ,CAAC;IAED,MAAM;QACJ,OAAO;YACL,GAAG,KAAK,CAAC,MAAM,EAAE;YACjB,UAAU,EAAE,IAAI,CAAC,UAAU;YAC3B,cAAc,EAAE,IAAI,CAAC,cAAc;SACpC,CAAC;IACJ,CAAC;IAED,aAAa,CAAC,MAAsB;QAClC,KAAK,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC;QAC5B,IAAI,CAAC,UAAU,GAAG,MAAM,CAAC,UAAU,CAAC;QACpC,IAAI,CAAC,cAAc,GAAG,MAAM,CAAC,cAAc,CAAC;QAC5C,IAAI,CAAC,kBAAkB,EAAE,CAAC;IAC5B,CAAC;IAED;;;OAGG;IACH,oBAAoB,CAAC,EAAuB;QAC1C,MAAM,MAAM,GAAG,0BAAe,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,OAAO,EAAE,CAAC;QAClD,IAAI,CAAC,wBAAwB,CAAC,MAAM,CAAC,CAAC;QAEtC,KAAK,CAAC,oBAAoB,CAAC,EAAE,CAAC,CAAC;QAE/B,MAAM,EAAE,YAAY,EAAE,OAAO,EAAE,SAAS,EAAE,GAAG,IAAI,CAAC,sBAAsB,CAAC,MAAM,CAAC,CAAC;QACjF,IAAI,CAAC,6BAA6B,CAAC,OAAO,EAAE,SAAS,CAAC,CAAC;QACvD,IAAI,CAAC,mBAAmB,CAAC,YAAY,CAAC,CAAC;QACvC,IAAI,CAAC,gBAAgB,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;QAC1C,IAAI,CAAC,kBAAkB,EAAE,CAAC;IAC5B,CAAC;IAED;;OAEG;IACK,wBAAwB,CAAC,MAA8C;QAC7E,MAAM,sBAAsB,GAAG,MAAM,CAAC,QAAQ,CAAC,IAAI,CACjD,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,yCAA6B,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,CACpE,CAAC;QAEF,IAAI,sBAAsB,EAAE,CAAC;YAC3B,MAAM,IAAI,kCAAuB,CAAC,yCAAyC,CAAC,CAAC;QAC/E,CAAC;IACH,CAAC;IAED;;OAEG;IACK,sBAAsB,CAAC,MAA8C;QAK3E,MAAM,YAAY,GAA6B,EAAE,CAAC;QAClD,MAAM,OAAO,GAAa,EAAE,CAAC;QAC7B,MAAM,SAAS,GAAa,EAAE,CAAC;QAE/B,MAAM,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,EAAE;YAC9B,IAAI,KAAK,CAAC,KAAK,KAAK,QAAQ,IAAI,kBAAkB,IAAI,KAAK,CAAC,MAAM,EAAE,CAAC;gBACnE,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,gBAA0C,CAAC,CAAC;YAC7E,CAAC;YAED,IAAI,KAAK,CAAC,KAAK,KAAK,MAAM,IAAI,OAAO,IAAI,KAAK,CAAC,IAAI,EAAE,CAAC;gBACpD,MAAM,KAAK,GAAG,IAAA,kBAAU,EAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;gBAC3C,MAAM,QAAQ,GAAG,IAAI,GAAG,IAAA,gBAAK,EAAC,KAAK,CAAC,CAAC;gBAErC,IAAI,eAAK,CAAC,cAAc,CAAC,QAAQ,CAAC,EAAE,CAAC;oBACnC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;gBAC3B,CAAC;qBAAM,CAAC;oBACN,OAAO,CAAC,IAAI,CAAC,IAAI,eAAS,CAAC,KAAK,CAAC,CAAC,MAAM,EAAE,CAAC,QAAQ,EAAE,CAAC,CAAC;gBACzD,CAAC;YACH,CAAC;QACH,CAAC,CAAC,CAAC;QAEH,OAAO,EAAE,YAAY,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC;IAC9C,CAAC;IAED;;OAEG;IACK,6BAA6B,CAAC,OAAiB,EAAE,SAAmB;QAC1E,IAAI,OAAO,CAAC,MAAM,KAAK,SAAS,CAAC,MAAM,EAAE,CAAC;YACxC,MAAM,IAAI,kCAAuB,CAAC,oDAAoD,CAAC,CAAC;QAC1F,CAAC;IACH,CAAC;IAED;;;;OAIG;IACK,mBAAmB,CAAC,YAAsC;QAChE,IAAI,YAAY,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC9B,OAAO;QACT,CAAC;QAED,MAAM,iBAAiB,GAAG,CAAC,IAAI,CAAC,iBAAiB,IAAI,IAAI,CAAC,iBAAiB,CAAC,MAAM,KAAK,CAAC,CAAC;QACzF,MAAM,gBAAgB,GAAG,CAAC,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,MAAM,KAAK,IAAI,CAAC,UAAU,CAAC;QAE7E,IAAI,iBAAiB,IAAI,gBAAgB,EAAE,CAAC;YAC1C,IAAI,CAAC,iBAAiB,GAAG,YAAY,CAAC;QACxC,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,eAAe,GAAG,YAAY,CAAC;QACtC,CAAC;IACH,CAAC;IAED;;OAEG;IACK,gBAAgB,CAAC,SAAmB,EAAE,OAAiB;QAC7D,IAAI,CAAC,WAAW,GAAG,SAAS,CAAC,GAAG,CAAC,CAAC,OAAO,EAAE,KAAK,EAAE,EAAE,CAAC,CAAC;YACpD,OAAO;YACP,MAAM,EAAE,OAAO,CAAC,KAAK,CAAC;SACvB,CAAC,CAAC,CAAC;IACN,CAAC;IAES,iBAAiB,CAAC,OAAmB;QAC7C,OAAO,IAAA,gCAAqB,EAAC,iBAAiB,EAAE,OAAO,CAAC,CAAC;IAC3D,CAAC;IAED;;;;OAIG;IACO,2BAA2B;QACnC,MAAM,UAAU,GAAG,IAAI,CAAC,yBAAyB,EAAE,CAAC;QACpD,IAAI,CAAC,4BAA4B,CAAC,UAAU,CAAC,CAAC;IAChD,CAAC;IAED;;;;OAIG;IACK,yBAAyB;QAC/B,IAAI,IAAI,CAAC,iBAAiB,EAAE,EAAE,CAAC;YAC7B,OAAO,IAAI,CAAC,yBAAyB,EAAE,CAAC;QAC1C,CAAC;QACD,OAAO,IAAI,CAAC,qBAAqB,EAAE,CAAC;IACtC,CAAC;IAED;;;OAGG;IACK,iBAAiB;QACvB,MAAM,iBAAiB,GAAG,IAAI,CAAC,cAAc,IAAI,IAAI,CAAC,cAAc,CAAC,MAAM,GAAG,CAAC,CAAC;QAChF,IAAI,iBAAiB,EAAE,CAAC;YACtB,OAAO,IAAI,CAAC;QACd,CAAC;QAED,qEAAqE;QACrE,MAAM,gBAAgB,GAAG,CAAC,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,UAAU,KAAK,IAAI,CAAC,MAAM,CAAC;QAC7E,OAAO,CAAC,gBAAgB,CAAC;IAC3B,CAAC;IAED;;;OAGG;IACK,yBAAyB;QAC/B,IAAI,CAAC,IAAI,CAAC,cAAc,IAAI,IAAI,CAAC,cAAc,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC7D,MAAM,IAAI,kCAAuB,CAAC,8BAA8B,CAAC,CAAC;QACpE,CAAC;QAED,MAAM,WAAW,GAAG,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,qBAAU,CAAC,SAAS,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QAE/F,2CAA2C;QAC3C,IAAI,IAAI,CAAC,cAAc,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACrC,OAAO,WAAW,CAAC;QACrB,CAAC;QAED,6CAA6C;QAC7C,OAAO,IAAI,CAAC,qBAAqB,CAAC,WAAW,EAAE,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,6BAAiB,CAAC,CAAC;IAClG,CAAC;IAED;;;OAGG;IACK,qBAAqB;QAC3B,IAAI,CAAC,IAAI,CAAC,iBAAiB,IAAI,IAAI,CAAC,iBAAiB,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACnE,MAAM,IAAI,kCAAuB,CAAC,kCAAkC,CAAC,CAAC;QACxE,CAAC;QAED,MAAM,SAAS,GAAG,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC;QAE5C,+CAA+C;QAC/C,IAAI,IAAI,CAAC,iBAAiB,CAAC,MAAM,IAAI,mCAAuB,EAAE,CAAC;YAC7D,OAAO,SAAS,CAAC;QACnB,CAAC;QAED,iDAAiD;QACjD,OAAO,IAAI,CAAC,qBAAqB,CAAC,SAAS,EAAE,IAAI,CAAC,iBAAiB,EAAE,6BAAiB,CAAC,CAAC;IAC1F,CAAC;IAED;;;;;;;;OAQG;IACK,qBAAqB,CAC3B,UAAqC,EACrC,cAAwC,EACxC,SAAiB;QAEjB,IAAI,gBAAgB,GAAG,UAAU,CAAC;QAElC,KAAK,IAAI,UAAU,GAAG,CAAC,EAAE,UAAU,GAAG,cAAc,CAAC,MAAM,EAAE,UAAU,IAAI,SAAS,EAAE,CAAC;YACrF,MAAM,KAAK,GAAG,cAAc,CAAC,KAAK,CAAC,UAAU,EAAE,UAAU,GAAG,SAAS,CAAC,CAAC;YACvE,MAAM,yBAAyB,GAAG,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,qBAAU,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;YAE9G,CAAC,gBAAgB,CAAC,GAAG,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,gBAAgB,EAAE,yBAAyB,CAAC,CAAC;QACrG,CAAC;QAED,OAAO,gBAAgB,CAAC;IAC1B,CAAC;IAED;;;;OAIG;IACK,4BAA4B,CAAC,UAAqC;QACxE,MAAM,gBAAgB,GAAG,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,SAAS,EAAE,EAAE,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;QAC/E,MAAM,UAAU,GAAG,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,UAAU,EAAE,gBAAgB,CAAC,CAAC;QAElF,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,SAAS,EAAE,KAAK,EAAE,EAAE;YAC5C,IAAI,CAAC,gBAAgB,CAAC,eAAe,CAAC,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,EAAE,SAAS,CAAC,OAAO,CAAC,CAAC;QAChF,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACO,4BAA4B;QACpC,IAAI,CAAC,sBAAsB,EAAE,CAAC;QAC9B,IAAI,CAAC,2BAA2B,EAAE,CAAC;QACnC,IAAI,CAAC,0BAA0B,EAAE,CAAC;IACpC,CAAC;IAED;;OAEG;IACK,sBAAsB;QAC5B,IAAI,CAAC,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACrD,MAAM,IAAI,kCAAuB,CAAC,qCAAqC,CAAC,CAAC;QAC3E,CAAC;QAED,IAAI,IAAI,CAAC,UAAU,CAAC,MAAM,GAAG,0BAAc,EAAE,CAAC;YAC5C,MAAM,IAAI,kCAAuB,CAC/B,qBAAqB,IAAI,CAAC,UAAU,CAAC,MAAM,8BAA8B,0BAAc,GAAG,CAC3F,CAAC;QACJ,CAAC;IACH,CAAC;IAED;;;;OAIG;IACK,2BAA2B;QACjC,MAAM,iBAAiB,GAAG,IAAI,CAAC,cAAc,IAAI,IAAI,CAAC,cAAc,CAAC,MAAM,GAAG,CAAC,CAAC;QAChF,IAAI,iBAAiB,EAAE,CAAC;YACtB,OAAO,CAAC,2CAA2C;QACrD,CAAC;QAED,gEAAgE;QAChE,MAAM,aAAa,GAAG,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,UAAU,KAAK,IAAI,CAAC,MAAM,CAAC;QACzE,IAAI,aAAa,EAAE,CAAC;YAClB,MAAM,IAAI,kCAAuB,CAAC,uDAAuD,CAAC,CAAC;QAC7F,CAAC;QAED,0CAA0C;QAC1C,MAAM,aAAa,GAAG,IAAI,CAAC,iBAAiB,IAAI,IAAI,CAAC,iBAAiB,CAAC,MAAM,GAAG,CAAC,CAAC;QAClF,IAAI,CAAC,aAAa,EAAE,CAAC;YACnB,MAAM,IAAI,kCAAuB,CAAC,qCAAqC,CAAC,CAAC;QAC3E,CAAC;IACH,CAAC;IAED;;;OAGG;IACK,0BAA0B;QAChC,MAAM,gBAAgB,GAAG,IAAI,CAAC,cAAc,EAAE,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QAE/E,8CAA8C;QAC9C,IAAI,CAAC,oBAAoB,CAAC,gBAAgB,EAAE,iBAAiB,CAAC,CAAC;QAE/D,IAAI,CAAC,IAAI,CAAC,iBAAiB,IAAI,IAAI,CAAC,iBAAiB,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACnE,OAAO;QACT,CAAC;QAED,MAAM,YAAY,GAAG,IAAI,CAAC,iBAAiB,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAEvE,kDAAkD;QAClD,IAAI,CAAC,oBAAoB,CAAC,YAAY,EAAE,qBAAqB,CAAC,CAAC;QAE/D,qDAAqD;QACrD,MAAM,cAAc,GAAG,gBAAgB,CAAC,MAAM,CAAC,CAAC,EAAE,EAAE,EAAE,CAAC,YAAY,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,CAAC;QAClF,IAAI,cAAc,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC9B,MAAM,IAAI,kCAAuB,CAC/B,6DAA6D,GAAG,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAC1F,CAAC;QACJ,CAAC;IACH,CAAC;IAED;;OAEG;IACK,oBAAoB,CAAC,GAAa,EAAE,UAAkB;QAC5D,MAAM,SAAS,GAAG,IAAI,GAAG,CAAC,GAAG,CAAC,CAAC;QAC/B,IAAI,SAAS,CAAC,IAAI,KAAK,GAAG,CAAC,MAAM,EAAE,CAAC;YAClC,MAAM,IAAI,kCAAuB,CAAC,iCAAiC,UAAU,EAAE,CAAC,CAAC;QACnF,CAAC;IACH,CAAC;IAED;;;OAGG;IACO,gCAAgC,CACxC,KAAa,EACb,iBAAyC;QAEzC,MAAM,OAAO,GAAG,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,SAAS,EAAE,EAAE,CAAC,SAAS,CAAC,CAAC;QAC9D,MAAM,YAAY,GAAG,IAAI,CAAC,0BAA0B,EAAE,CAAC;QAEvD,OAAO;YACL,GAAG,iBAAiB;YACpB,YAAY;YACZ,OAAO;SACR,CAAC;IACJ,CAAC;IAED;;OAEG;IACK,0BAA0B;QAChC,OAAO,IAAI,CAAC,UAAU;aACnB,MAAM,CAAC,CAAC,WAAW,EAAE,OAAO,EAAE,EAAE,CAAC,WAAW,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,IAAI,sBAAS,CAAC,CAAC,CAAC,CAAC;aACpF,QAAQ,EAAE,CAAC;IAChB,CAAC;CACF;AAjZD,kDAiZC","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"]}
@@ -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
- /** @inheritdoc */
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
- /** @inheritdoc */
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
- /** @inheritdoc */
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
- /** @inheritdoc */
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
- /** @inheritdoc */
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
- /** @inheritdoc */
14
- isValidTransactionId(txId: string): boolean;
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;IACrC,kBAAkB;IAClB,cAAc,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO;IAIxC,kBAAkB;IAClB,cAAc,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO;IAIrC,kBAAkB;IAClB,iBAAiB,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO;IAIvC,kBAAkB;IAClB,gBAAgB,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO;IAItC,kBAAkB;IAClB,gBAAgB,CAAC,SAAS,EAAE,MAAM,GAAG,OAAO;IAQ5C,kBAAkB;IAClB,oBAAoB,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO;IAI3C,UAAU,CAAC,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,MAAM,GAAG,OAAO;IAKlD,eAAe,CAAC,KAAK,EAAE,MAAM,GAAG,UAAU,GAAG,MAAM;IAQnD,uBAAuB,CAAC,SAAS,EAAE,MAAM,GAAG,MAAM;IAKlD,qBAAqB,CAAC,cAAc,EAAE,MAAM,GAAG,UAAU,GAAG,OAAO;CAQpE;AAED,QAAA,MAAM,KAAK,OAAc,CAAC;AAE1B,eAAe,KAAK,CAAC"}
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"}