@bitgo/sdk-coin-sui 5.23.2 → 5.23.4

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.
@@ -90,7 +90,7 @@ class TokenTransferBuilder extends transactionBuilder_1.TransactionBuilder {
90
90
  (input?.value !== null && typeof input?.value === 'object' && 'BalanceWithdrawal' in (input.value ?? {})));
91
91
  if (withdrawalInput) {
92
92
  const bw = withdrawalInput.BalanceWithdrawal ?? withdrawalInput.value?.BalanceWithdrawal;
93
- this._fundsInAddressBalance = new bignumber_js_1.default(String(bw.amount));
93
+ this._fundsInAddressBalance = new bignumber_js_1.default(String(bw.reservation?.MaxAmountU64 ?? bw.amount));
94
94
  }
95
95
  if (txData.inputObjects && txData.inputObjects.length > 0) {
96
96
  this.inputObjects(txData.inputObjects);
@@ -199,4 +199,4 @@ class TokenTransferBuilder extends transactionBuilder_1.TransactionBuilder {
199
199
  }
200
200
  }
201
201
  exports.TokenTransferBuilder = TokenTransferBuilder;
202
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tokenTransferBuilder.js","sourceRoot":"","sources":["../../../src/lib/tokenTransferBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\nimport { TransactionType, Recipient, BuildTransactionError, BaseKey } from '@bitgo/sdk-core';\nimport { BaseCoin as CoinConfig, SuiCoin } from '@bitgo/statics';\nimport { SuiTransaction, SuiTransactionType, TokenTransferProgrammableTransaction } from './iface';\nimport { Transaction } from './transaction';\nimport { TransactionBuilder } from './transactionBuilder';\nimport { TokenTransferTransaction } from './tokenTransferTransaction';\nimport { SuiObjectRef } from './mystenlab/types';\nimport utils from './utils';\nimport {\n  Inputs,\n  TransactionBlock as ProgrammingTransactionBlockBuilder,\n  TransactionArgument,\n} from './mystenlab/builder';\nimport BigNumber from 'bignumber.js';\n\nexport class TokenTransferBuilder extends TransactionBuilder<TokenTransferProgrammableTransaction> {\n  protected _recipients: Recipient[];\n  protected _inputObjects: SuiObjectRef[];\n  /**\n   * Balance held in the address balance system for the token being transferred.\n   * When set, this amount is included in the total available balance.\n   * At execution time, tx.withdrawal() + 0x2::coin::redeem_funds converts it\n   * to a Coin<T> that is merged with any coin objects before splitting.\n   */\n  protected _fundsInAddressBalance: BigNumber = new BigNumber(0);\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    this._transaction = new TokenTransferTransaction(_coinConfig);\n  }\n\n  protected get transactionType(): TransactionType {\n    return TransactionType.Send;\n  }\n\n  /**\n   * The full coin type string derived from the coin config (e.g. `0xabc::my_token::MY_TOKEN`).\n   */\n  private get tokenCoinType(): string {\n    const config = this._coinConfig as SuiCoin;\n    return `${config.packageId}::${config.module}::${config.symbol}`;\n  }\n\n  /** @inheritdoc */\n  validateTransaction(transaction: TokenTransferTransaction): void {\n    if (!transaction.suiTransaction) {\n      return;\n    }\n    this.validateTransactionFields();\n  }\n\n  /** @inheritdoc */\n  sign(key: BaseKey): void {\n    this.transaction.setSuiTransaction(this.buildSuiTransaction());\n    super.sign(key);\n  }\n\n  /** @inheritdoc */\n  protected fromImplementation(rawTransaction: string): Transaction<TokenTransferProgrammableTransaction> {\n    const tx = new TokenTransferTransaction(this._coinConfig);\n    this.validateRawTransaction(rawTransaction);\n    tx.fromRawTransaction(rawTransaction);\n    this.initBuilder(tx);\n    return this.transaction;\n  }\n\n  /** @inheritdoc */\n  protected async buildImplementation(): Promise<Transaction<TokenTransferProgrammableTransaction>> {\n    this.transaction.setSuiTransaction(this.buildSuiTransaction());\n    this.transaction.transactionType(this.transactionType);\n\n    if (this._signer) {\n      this.transaction.sign(this._signer);\n    }\n\n    this._signatures.forEach((signature) => {\n      this.transaction.addSignature(signature.publicKey, signature.signature);\n    });\n\n    this.transaction.loadInputsAndOutputs();\n    return this.transaction;\n  }\n\n  /** @inheritdoc */\n  initBuilder(tx: TokenTransferTransaction): void {\n    this._transaction = tx;\n\n    if (tx.signature && tx.signature.length > 0) {\n      this._signatures = [tx.suiSignature];\n    }\n    const txData = tx.toJson();\n    this.type(SuiTransactionType.TokenTransfer);\n    this.sender(txData.sender);\n    this.gasData(txData.gasData);\n\n    // Only restore meaningful expirations (Epoch or ValidDuring) — ignore None.\n    if (txData.expiration && !('None' in txData.expiration)) {\n      this._expiration = txData.expiration;\n    }\n\n    const recipients = utils.getRecipients(tx.suiTransaction);\n    this.send(recipients);\n\n    // Reconstruct fundsInAddressBalance from BalanceWithdrawal input if present.\n    // After BCS deserialization inputs are CallArg format: { BalanceWithdrawal: {...} }\n    // During building they are TransactionBlockInput format: { kind:'Input', value: { BalanceWithdrawal: {...} } }\n    const withdrawalInput = (tx.suiTransaction?.tx?.inputs as any[])?.find(\n      (input: any) =>\n        (input !== null && typeof input === 'object' && 'BalanceWithdrawal' in input) ||\n        (input?.value !== null && typeof input?.value === 'object' && 'BalanceWithdrawal' in (input.value ?? {}))\n    );\n    if (withdrawalInput) {\n      const bw = withdrawalInput.BalanceWithdrawal ?? withdrawalInput.value?.BalanceWithdrawal;\n      this._fundsInAddressBalance = new BigNumber(String(bw.amount));\n    }\n\n    if (txData.inputObjects && txData.inputObjects.length > 0) {\n      this.inputObjects(txData.inputObjects);\n    }\n  }\n\n  send(recipients: Recipient[]): this {\n    this.validateRecipients(recipients);\n    this._recipients = recipients;\n    return this;\n  }\n\n  inputObjects(inputObject: SuiObjectRef[]): this {\n    this.validateInputObjectRefs(inputObject);\n    this._inputObjects = inputObject;\n    return this;\n  }\n\n  /**\n   * Set the amount of token funds held in the Sui address balance system for this sender.\n   *\n   * @param {string} amount - amount in base units held in address balance\n   */\n  fundsInAddressBalance(amount: string): this {\n    this._fundsInAddressBalance = new BigNumber(amount);\n    return this;\n  }\n\n  /**\n   * Validates all fields are defined correctly\n   */\n  private validateTransactionFields(): void {\n    assert(this._type, new BuildTransactionError('type is required before building'));\n    assert(this._sender, new BuildTransactionError('sender is required before building'));\n    assert(\n      this._recipients && this._recipients.length > 0,\n      new BuildTransactionError('at least one recipient is required before building')\n    );\n    assert(this._gasData, new BuildTransactionError('gasData is required before building'));\n    this.validateGasData(this._gasData);\n\n    // Must have at least coin objects OR address balance\n    assert(\n      (this._inputObjects && this._inputObjects.length > 0) || this._fundsInAddressBalance.gt(0),\n      new BuildTransactionError('input objects or fundsInAddressBalance required before building')\n    );\n    if (this._inputObjects && this._inputObjects.length > 0) {\n      this.validateInputObjectRefs(this._inputObjects);\n    }\n  }\n\n  /** Validates the individual object refs (does not require non-empty array). */\n  private validateInputObjectRefs(inputObjects: SuiObjectRef[]): void {\n    if (inputObjects) {\n      inputObjects.forEach((inputObject) => {\n        this.validateSuiObjectRef(inputObject, 'input object');\n      });\n    }\n  }\n\n  /**\n   * Build SuiTransaction.\n   *\n   * Two build paths:\n   *\n   * Path A — coin objects only (fundsInAddressBalance = 0):\n   *   MergeCoins(inputObject[0], [inputObject[1..]]) → SplitCoins → TransferObjects\n   *\n   * Path B — coin objects + address balance (or address balance only):\n   *   MoveCall(0x2::coin::redeem_funds, [withdrawal(amount, coinType)]) → Coin<T>\n   *   MergeCoins(inputObject[0] | addrCoin, [rest...]) → SplitCoins → TransferObjects\n   *\n   * @return {SuiTransaction<TokenTransferProgrammableTransaction>}\n   * @protected\n   */\n  protected buildSuiTransaction(): SuiTransaction<TokenTransferProgrammableTransaction> {\n    this.validateTransactionFields();\n\n    const programmableTxBuilder = new ProgrammingTransactionBlockBuilder();\n\n    const inputObjects: TransactionArgument[] = (this._inputObjects ?? []).map((object) =>\n      programmableTxBuilder.object(Inputs.ObjectRef(object))\n    );\n\n    // If address balance is available, withdraw it as Coin<T> and add to the pool\n    if (this._fundsInAddressBalance.gt(0)) {\n      const coinType = this.tokenCoinType;\n      const [addrCoin] = programmableTxBuilder.moveCall({\n        target: '0x2::coin::redeem_funds',\n        typeArguments: [coinType],\n        arguments: [\n          programmableTxBuilder.withdrawal({\n            amount: BigInt(this._fundsInAddressBalance.toFixed()),\n            type: coinType,\n          }),\n        ],\n      });\n      inputObjects.push(addrCoin);\n    }\n\n    const mergedObject = inputObjects.shift() as TransactionArgument;\n    if (inputObjects.length > 0) {\n      programmableTxBuilder.mergeCoins(mergedObject, inputObjects);\n    }\n\n    this._recipients.forEach((recipient) => {\n      const splitObject = programmableTxBuilder.splitCoins(mergedObject, [\n        programmableTxBuilder.pure(Number(recipient.amount)),\n      ]);\n      programmableTxBuilder.transferObjects([splitObject], programmableTxBuilder.object(recipient.address));\n    });\n\n    const txData = programmableTxBuilder.blockData;\n    return {\n      type: this._type,\n      sender: this._sender,\n      tx: {\n        inputs: [...txData.inputs],\n        transactions: [...txData.transactions],\n      },\n      gasData: {\n        ...this._gasData,\n      },\n      expiration: this._expiration,\n      fundsInAddressBalance: this._fundsInAddressBalance.gt(0) ? this._fundsInAddressBalance.toFixed() : undefined,\n    };\n  }\n}\n"]}
202
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tokenTransferBuilder.js","sourceRoot":"","sources":["../../../src/lib/tokenTransferBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import assert from 'assert';\nimport { TransactionType, Recipient, BuildTransactionError, BaseKey } from '@bitgo/sdk-core';\nimport { BaseCoin as CoinConfig, SuiCoin } from '@bitgo/statics';\nimport { SuiTransaction, SuiTransactionType, TokenTransferProgrammableTransaction } from './iface';\nimport { Transaction } from './transaction';\nimport { TransactionBuilder } from './transactionBuilder';\nimport { TokenTransferTransaction } from './tokenTransferTransaction';\nimport { SuiObjectRef } from './mystenlab/types';\nimport utils from './utils';\nimport {\n  Inputs,\n  TransactionBlock as ProgrammingTransactionBlockBuilder,\n  TransactionArgument,\n} from './mystenlab/builder';\nimport BigNumber from 'bignumber.js';\n\nexport class TokenTransferBuilder extends TransactionBuilder<TokenTransferProgrammableTransaction> {\n  protected _recipients: Recipient[];\n  protected _inputObjects: SuiObjectRef[];\n  /**\n   * Balance held in the address balance system for the token being transferred.\n   * When set, this amount is included in the total available balance.\n   * At execution time, tx.withdrawal() + 0x2::coin::redeem_funds converts it\n   * to a Coin<T> that is merged with any coin objects before splitting.\n   */\n  protected _fundsInAddressBalance: BigNumber = new BigNumber(0);\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    this._transaction = new TokenTransferTransaction(_coinConfig);\n  }\n\n  protected get transactionType(): TransactionType {\n    return TransactionType.Send;\n  }\n\n  /**\n   * The full coin type string derived from the coin config (e.g. `0xabc::my_token::MY_TOKEN`).\n   */\n  private get tokenCoinType(): string {\n    const config = this._coinConfig as SuiCoin;\n    return `${config.packageId}::${config.module}::${config.symbol}`;\n  }\n\n  /** @inheritdoc */\n  validateTransaction(transaction: TokenTransferTransaction): void {\n    if (!transaction.suiTransaction) {\n      return;\n    }\n    this.validateTransactionFields();\n  }\n\n  /** @inheritdoc */\n  sign(key: BaseKey): void {\n    this.transaction.setSuiTransaction(this.buildSuiTransaction());\n    super.sign(key);\n  }\n\n  /** @inheritdoc */\n  protected fromImplementation(rawTransaction: string): Transaction<TokenTransferProgrammableTransaction> {\n    const tx = new TokenTransferTransaction(this._coinConfig);\n    this.validateRawTransaction(rawTransaction);\n    tx.fromRawTransaction(rawTransaction);\n    this.initBuilder(tx);\n    return this.transaction;\n  }\n\n  /** @inheritdoc */\n  protected async buildImplementation(): Promise<Transaction<TokenTransferProgrammableTransaction>> {\n    this.transaction.setSuiTransaction(this.buildSuiTransaction());\n    this.transaction.transactionType(this.transactionType);\n\n    if (this._signer) {\n      this.transaction.sign(this._signer);\n    }\n\n    this._signatures.forEach((signature) => {\n      this.transaction.addSignature(signature.publicKey, signature.signature);\n    });\n\n    this.transaction.loadInputsAndOutputs();\n    return this.transaction;\n  }\n\n  /** @inheritdoc */\n  initBuilder(tx: TokenTransferTransaction): void {\n    this._transaction = tx;\n\n    if (tx.signature && tx.signature.length > 0) {\n      this._signatures = [tx.suiSignature];\n    }\n    const txData = tx.toJson();\n    this.type(SuiTransactionType.TokenTransfer);\n    this.sender(txData.sender);\n    this.gasData(txData.gasData);\n\n    // Only restore meaningful expirations (Epoch or ValidDuring) — ignore None.\n    if (txData.expiration && !('None' in txData.expiration)) {\n      this._expiration = txData.expiration;\n    }\n\n    const recipients = utils.getRecipients(tx.suiTransaction);\n    this.send(recipients);\n\n    // Reconstruct fundsInAddressBalance from BalanceWithdrawal input if present.\n    // After BCS deserialization inputs are CallArg format: { BalanceWithdrawal: {...} }\n    // During building they are TransactionBlockInput format: { kind:'Input', value: { BalanceWithdrawal: {...} } }\n    const withdrawalInput = (tx.suiTransaction?.tx?.inputs as any[])?.find(\n      (input: any) =>\n        (input !== null && typeof input === 'object' && 'BalanceWithdrawal' in input) ||\n        (input?.value !== null && typeof input?.value === 'object' && 'BalanceWithdrawal' in (input.value ?? {}))\n    );\n    if (withdrawalInput) {\n      const bw = withdrawalInput.BalanceWithdrawal ?? withdrawalInput.value?.BalanceWithdrawal;\n      this._fundsInAddressBalance = new BigNumber(String(bw.reservation?.MaxAmountU64 ?? bw.amount));\n    }\n\n    if (txData.inputObjects && txData.inputObjects.length > 0) {\n      this.inputObjects(txData.inputObjects);\n    }\n  }\n\n  send(recipients: Recipient[]): this {\n    this.validateRecipients(recipients);\n    this._recipients = recipients;\n    return this;\n  }\n\n  inputObjects(inputObject: SuiObjectRef[]): this {\n    this.validateInputObjectRefs(inputObject);\n    this._inputObjects = inputObject;\n    return this;\n  }\n\n  /**\n   * Set the amount of token funds held in the Sui address balance system for this sender.\n   *\n   * @param {string} amount - amount in base units held in address balance\n   */\n  fundsInAddressBalance(amount: string): this {\n    this._fundsInAddressBalance = new BigNumber(amount);\n    return this;\n  }\n\n  /**\n   * Validates all fields are defined correctly\n   */\n  private validateTransactionFields(): void {\n    assert(this._type, new BuildTransactionError('type is required before building'));\n    assert(this._sender, new BuildTransactionError('sender is required before building'));\n    assert(\n      this._recipients && this._recipients.length > 0,\n      new BuildTransactionError('at least one recipient is required before building')\n    );\n    assert(this._gasData, new BuildTransactionError('gasData is required before building'));\n    this.validateGasData(this._gasData);\n\n    // Must have at least coin objects OR address balance\n    assert(\n      (this._inputObjects && this._inputObjects.length > 0) || this._fundsInAddressBalance.gt(0),\n      new BuildTransactionError('input objects or fundsInAddressBalance required before building')\n    );\n    if (this._inputObjects && this._inputObjects.length > 0) {\n      this.validateInputObjectRefs(this._inputObjects);\n    }\n  }\n\n  /** Validates the individual object refs (does not require non-empty array). */\n  private validateInputObjectRefs(inputObjects: SuiObjectRef[]): void {\n    if (inputObjects) {\n      inputObjects.forEach((inputObject) => {\n        this.validateSuiObjectRef(inputObject, 'input object');\n      });\n    }\n  }\n\n  /**\n   * Build SuiTransaction.\n   *\n   * Two build paths:\n   *\n   * Path A — coin objects only (fundsInAddressBalance = 0):\n   *   MergeCoins(inputObject[0], [inputObject[1..]]) → SplitCoins → TransferObjects\n   *\n   * Path B — coin objects + address balance (or address balance only):\n   *   MoveCall(0x2::coin::redeem_funds, [withdrawal(amount, coinType)]) → Coin<T>\n   *   MergeCoins(inputObject[0] | addrCoin, [rest...]) → SplitCoins → TransferObjects\n   *\n   * @return {SuiTransaction<TokenTransferProgrammableTransaction>}\n   * @protected\n   */\n  protected buildSuiTransaction(): SuiTransaction<TokenTransferProgrammableTransaction> {\n    this.validateTransactionFields();\n\n    const programmableTxBuilder = new ProgrammingTransactionBlockBuilder();\n\n    const inputObjects: TransactionArgument[] = (this._inputObjects ?? []).map((object) =>\n      programmableTxBuilder.object(Inputs.ObjectRef(object))\n    );\n\n    // If address balance is available, withdraw it as Coin<T> and add to the pool\n    if (this._fundsInAddressBalance.gt(0)) {\n      const coinType = this.tokenCoinType;\n      const [addrCoin] = programmableTxBuilder.moveCall({\n        target: '0x2::coin::redeem_funds',\n        typeArguments: [coinType],\n        arguments: [\n          programmableTxBuilder.withdrawal({\n            amount: BigInt(this._fundsInAddressBalance.toFixed()),\n            type: coinType,\n          }),\n        ],\n      });\n      inputObjects.push(addrCoin);\n    }\n\n    const mergedObject = inputObjects.shift() as TransactionArgument;\n    if (inputObjects.length > 0) {\n      programmableTxBuilder.mergeCoins(mergedObject, inputObjects);\n    }\n\n    this._recipients.forEach((recipient) => {\n      const splitObject = programmableTxBuilder.splitCoins(mergedObject, [\n        programmableTxBuilder.pure(Number(recipient.amount)),\n      ]);\n      programmableTxBuilder.transferObjects([splitObject], programmableTxBuilder.object(recipient.address));\n    });\n\n    const txData = programmableTxBuilder.blockData;\n    return {\n      type: this._type,\n      sender: this._sender,\n      tx: {\n        inputs: [...txData.inputs],\n        transactions: [...txData.transactions],\n      },\n      gasData: {\n        ...this._gasData,\n      },\n      expiration: this._expiration,\n      fundsInAddressBalance: this._fundsInAddressBalance.gt(0) ? this._fundsInAddressBalance.toFixed() : undefined,\n    };\n  }\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"transferBuilder.d.ts","sourceRoot":"","sources":["../../../src/lib/transferBuilder.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAC1D,OAAO,EAAE,QAAQ,IAAI,UAAU,EAAE,MAAM,gBAAgB,CAAC;AACxD,OAAO,EAAE,OAAO,EAAyB,SAAS,EAAE,eAAe,EAAE,MAAM,iBAAiB,CAAC;AAC7F,OAAO,EAAE,cAAc,EAAsB,+BAA+B,EAAE,MAAM,SAAS,CAAC;AAC9F,OAAO,EAAE,WAAW,EAAE,MAAM,eAAe,CAAC;AAC5C,OAAO,EAAE,mBAAmB,EAAE,MAAM,uBAAuB,CAAC;AAU5D,OAAO,SAAS,MAAM,cAAc,CAAC;AAErC,qBAAa,eAAgB,SAAQ,kBAAkB,CAAC,+BAA+B,CAAC;IACtF,SAAS,CAAC,WAAW,EAAE,SAAS,EAAE,CAAC;IACnC;;;;;;OAMG;IACH,SAAS,CAAC,sBAAsB,EAAE,SAAS,CAAoB;gBAEnD,WAAW,EAAE,QAAQ,CAAC,UAAU,CAAC;IAK7C,SAAS,KAAK,eAAe,IAAI,eAAe,CAE/C;IAED,IAAI,CAAC,UAAU,EAAE,SAAS,EAAE,GAAG,IAAI;IAMnC;;;;;;OAMG;IACH,qBAAqB,CAAC,MAAM,EAAE,MAAM,GAAG,IAAI;IAK3C;;;;;OAKG;IACH,qBAAqB,CAAC,kBAAkB,EAAE,SAAS,GAAG,SAAS;IAI/D,kBAAkB;IAClB,mBAAmB,CAAC,WAAW,EAAE,mBAAmB,GAAG,IAAI;IAO3D,kBAAkB;IAClB,IAAI,CAAC,GAAG,EAAE,OAAO,GAAG,IAAI;IAKxB,kBAAkB;IAClB,SAAS,CAAC,kBAAkB,CAAC,cAAc,EAAE,MAAM,GAAG,WAAW,CAAC,+BAA+B,CAAC;IAQlG,kBAAkB;cACF,mBAAmB,IAAI,OAAO,CAAC,WAAW,CAAC,+BAA+B,CAAC,CAAC;IAgB5F;;;;OAIG;IACH,WAAW,CAAC,EAAE,EAAE,mBAAmB,GAAG,IAAI;IAqC1C;;OAEG;IACH,OAAO,CAAC,yBAAyB;IA4BjC;;;;;;;;;;;;;;;;;;;;;;;OAuBG;IACH,SAAS,CAAC,mBAAmB,IAAI,cAAc,CAAC,+BAA+B,CAAC;CA4HjF"}
1
+ {"version":3,"file":"transferBuilder.d.ts","sourceRoot":"","sources":["../../../src/lib/transferBuilder.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,kBAAkB,EAAE,MAAM,sBAAsB,CAAC;AAC1D,OAAO,EAAE,QAAQ,IAAI,UAAU,EAAE,MAAM,gBAAgB,CAAC;AACxD,OAAO,EAAE,OAAO,EAAyB,SAAS,EAAE,eAAe,EAAE,MAAM,iBAAiB,CAAC;AAC7F,OAAO,EAAE,cAAc,EAAsB,+BAA+B,EAAE,MAAM,SAAS,CAAC;AAC9F,OAAO,EAAE,WAAW,EAAE,MAAM,eAAe,CAAC;AAC5C,OAAO,EAAE,mBAAmB,EAAE,MAAM,uBAAuB,CAAC;AAU5D,OAAO,SAAS,MAAM,cAAc,CAAC;AAErC,qBAAa,eAAgB,SAAQ,kBAAkB,CAAC,+BAA+B,CAAC;IACtF,SAAS,CAAC,WAAW,EAAE,SAAS,EAAE,CAAC;IACnC;;;;;;OAMG;IACH,SAAS,CAAC,sBAAsB,EAAE,SAAS,CAAoB;gBAEnD,WAAW,EAAE,QAAQ,CAAC,UAAU,CAAC;IAK7C,SAAS,KAAK,eAAe,IAAI,eAAe,CAE/C;IAED,IAAI,CAAC,UAAU,EAAE,SAAS,EAAE,GAAG,IAAI;IAMnC;;;;;;OAMG;IACH,qBAAqB,CAAC,MAAM,EAAE,MAAM,GAAG,IAAI;IAK3C;;;;;OAKG;IACH,qBAAqB,CAAC,kBAAkB,EAAE,SAAS,GAAG,SAAS;IAI/D,kBAAkB;IAClB,mBAAmB,CAAC,WAAW,EAAE,mBAAmB,GAAG,IAAI;IAO3D,kBAAkB;IAClB,IAAI,CAAC,GAAG,EAAE,OAAO,GAAG,IAAI;IAKxB,kBAAkB;IAClB,SAAS,CAAC,kBAAkB,CAAC,cAAc,EAAE,MAAM,GAAG,WAAW,CAAC,+BAA+B,CAAC;IAQlG,kBAAkB;cACF,mBAAmB,IAAI,OAAO,CAAC,WAAW,CAAC,+BAA+B,CAAC,CAAC;IAgB5F;;;;OAIG;IACH,WAAW,CAAC,EAAE,EAAE,mBAAmB,GAAG,IAAI;IAqC1C;;OAEG;IACH,OAAO,CAAC,yBAAyB;IA4BjC;;;;;;;;;;;;;;;;;;;;;;;OAuBG;IACH,SAAS,CAAC,mBAAmB,IAAI,cAAc,CAAC,+BAA+B,CAAC;CA+IjF"}
@@ -114,7 +114,7 @@ class TransferBuilder extends transactionBuilder_1.TransactionBuilder {
114
114
  (input?.value !== null && typeof input?.value === 'object' && 'BalanceWithdrawal' in (input.value ?? {})));
115
115
  if (withdrawalInput) {
116
116
  const bw = withdrawalInput.BalanceWithdrawal ?? withdrawalInput.value?.BalanceWithdrawal;
117
- this._fundsInAddressBalance = new bignumber_js_1.default(String(bw.amount));
117
+ this._fundsInAddressBalance = new bignumber_js_1.default(String(bw.reservation?.MaxAmountU64 ?? bw.amount));
118
118
  }
119
119
  const recipients = utils_1.default.getRecipients(tx.suiTransaction);
120
120
  this.send(recipients);
@@ -218,12 +218,30 @@ class TransferBuilder extends transactionBuilder_1.TransactionBuilder {
218
218
  typeArguments: ['0x2::sui::SUI'],
219
219
  arguments: [programmableTxBuilder.withdrawal({ amount: BigInt(this._fundsInAddressBalance.toFixed()) })],
220
220
  });
221
+ // addrCoin is a command result (from redeem_funds) and must be explicitly consumed.
222
+ // Split for every recipient — keeps a 1:1 SplitCoins↔TransferObjects structure for the
223
+ // transaction parser. After all splits, consume the source coin explicitly:
224
+ // - If there is change (fundsInAddressBalance > total recipient amount): return addrCoin
225
+ // to sender via TransferObjects.
226
+ // - If send-all (no change): addrCoin has 0 balance; destroy it by merging into the
227
+ // sponsor's gas coin (a 0-value merge is valid in Sui and deletes the source object).
228
+ const totalRecipientAmount = this._recipients.reduce((sum, r) => sum.plus(r.amount), new bignumber_js_1.default(0));
229
+ const hasChange = this._fundsInAddressBalance.gt(totalRecipientAmount);
221
230
  this._recipients.forEach((recipient) => {
222
231
  const splitObject = programmableTxBuilder.splitCoins(addrCoin, [
223
232
  programmableTxBuilder.pure(Number(recipient.amount)),
224
233
  ]);
225
234
  programmableTxBuilder.transferObjects([splitObject], programmableTxBuilder.object(recipient.address));
226
235
  });
236
+ if (hasChange) {
237
+ // Return the remaining balance (change) to the sender.
238
+ programmableTxBuilder.transferObjects([addrCoin], programmableTxBuilder.object(this._sender));
239
+ }
240
+ else {
241
+ // Send-all: addrCoin has 0 balance after all splits. Merge it into the sponsor's gas
242
+ // coin to destroy the zero-balance object (coin::join accepts a 0-value source).
243
+ programmableTxBuilder.mergeCoins(programmableTxBuilder.gas, [addrCoin]);
244
+ }
227
245
  const txData1b = programmableTxBuilder.blockData;
228
246
  return {
229
247
  type: this._type,
@@ -279,4 +297,4 @@ class TransferBuilder extends transactionBuilder_1.TransactionBuilder {
279
297
  }
280
298
  }
281
299
  exports.TransferBuilder = TransferBuilder;
282
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transferBuilder.js","sourceRoot":"","sources":["../../../src/lib/transferBuilder.ts"],"names":[],"mappings":";;;;;;AAAA,6DAA0D;AAE1D,8CAA6F;AAC7F,mCAA8F;AAE9F,+DAA4D;AAC5D,oDAA4B;AAC5B,iDAK6B;AAC7B,oDAA4B;AAC5B,2CAAgE;AAChE,gEAAqC;AAErC,MAAa,eAAgB,SAAQ,uCAAmD;IAWtF,YAAY,WAAiC;QAC3C,KAAK,CAAC,WAAW,CAAC,CAAC;QAVrB;;;;;;WAMG;QACO,2BAAsB,GAAc,IAAI,sBAAS,CAAC,CAAC,CAAC,CAAC;QAI7D,IAAI,CAAC,YAAY,GAAG,IAAI,yCAAmB,CAAC,WAAW,CAAC,CAAC;IAC3D,CAAC;IAED,IAAc,eAAe;QAC3B,OAAO,0BAAe,CAAC,IAAI,CAAC;IAC9B,CAAC;IAED,IAAI,CAAC,UAAuB;QAC1B,IAAI,CAAC,kBAAkB,CAAC,UAAU,CAAC,CAAC;QACpC,IAAI,CAAC,WAAW,GAAG,UAAU,CAAC;QAC9B,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;;;OAMG;IACH,qBAAqB,CAAC,MAAc;QAClC,IAAI,CAAC,sBAAsB,GAAG,IAAI,sBAAS,CAAC,MAAM,CAAC,CAAC;QACpD,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;;OAKG;IACH,qBAAqB,CAAC,kBAA6B;QACjD,OAAO,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,sBAAsB,CAAC,CAAC;IAC9D,CAAC;IAED,kBAAkB;IAClB,mBAAmB,CAAC,WAAgC;QAClD,IAAI,CAAC,WAAW,CAAC,cAAc,EAAE,CAAC;YAChC,OAAO;QACT,CAAC;QACD,IAAI,CAAC,yBAAyB,EAAE,CAAC;IACnC,CAAC;IAED,kBAAkB;IAClB,IAAI,CAAC,GAAY;QACf,IAAI,CAAC,WAAW,CAAC,iBAAiB,CAAC,IAAI,CAAC,mBAAmB,EAAE,CAAC,CAAC;QAC/D,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IAClB,CAAC;IAED,kBAAkB;IACR,kBAAkB,CAAC,cAAsB;QACjD,MAAM,EAAE,GAAG,IAAI,yCAAmB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACrD,IAAI,CAAC,sBAAsB,CAAC,cAAc,CAAC,CAAC;QAC5C,EAAE,CAAC,kBAAkB,CAAC,cAAc,CAAC,CAAC;QACtC,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;QACrB,OAAO,IAAI,CAAC,WAAW,CAAC;IAC1B,CAAC;IAED,kBAAkB;IACR,KAAK,CAAC,mBAAmB;QACjC,IAAI,CAAC,WAAW,CAAC,iBAAiB,CAAC,IAAI,CAAC,mBAAmB,EAAE,CAAC,CAAC;QAC/D,IAAI,CAAC,WAAW,CAAC,eAAe,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;QAEvD,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;YACjB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACtC,CAAC;QAED,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,SAAS,EAAE,EAAE;YACrC,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,SAAS,CAAC,SAAS,EAAE,SAAS,CAAC,SAAS,CAAC,CAAC;QAC1E,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,WAAW,CAAC,oBAAoB,EAAE,CAAC;QACxC,OAAO,IAAI,CAAC,WAAW,CAAC;IAC1B,CAAC;IAED;;;;OAIG;IACH,WAAW,CAAC,EAAuB;QACjC,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;QAEvB,IAAI,EAAE,CAAC,SAAS,IAAI,EAAE,CAAC,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC5C,IAAI,CAAC,WAAW,GAAG,CAAC,EAAE,CAAC,YAAY,CAAC,CAAC;QACvC,CAAC;QAED,MAAM,MAAM,GAAG,EAAE,CAAC,MAAM,EAAE,CAAC;QAC3B,IAAI,CAAC,IAAI,CAAC,0BAAkB,CAAC,QAAQ,CAAC,CAAC;QACvC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QAC3B,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;QAE7B,IAAI,MAAM,CAAC,UAAU,IAAI,CAAC,CAAC,MAAM,IAAI,MAAM,CAAC,UAAU,CAAC,EAAE,CAAC;YACxD,IAAI,CAAC,WAAW,GAAG,MAAM,CAAC,UAAU,CAAC;QACvC,CAAC;QAED,IAAI,MAAM,CAAC,YAAY,EAAE,CAAC;YACxB,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC;QACzC,CAAC;QAED,6EAA6E;QAC7E,oFAAoF;QACpF,+GAA+G;QAC/G,MAAM,eAAe,GAAI,EAAE,CAAC,cAAc,EAAE,EAAE,EAAE,MAAgB,EAAE,IAAI,CACpE,CAAC,KAAU,EAAE,EAAE,CACb,CAAC,KAAK,KAAK,IAAI,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,mBAAmB,IAAI,KAAK,CAAC;YAC7E,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,IAAI,OAAO,KAAK,EAAE,KAAK,KAAK,QAAQ,IAAI,mBAAmB,IAAI,CAAC,KAAK,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC,CAC5G,CAAC;QACF,IAAI,eAAe,EAAE,CAAC;YACpB,MAAM,EAAE,GAAG,eAAe,CAAC,iBAAiB,IAAI,eAAe,CAAC,KAAK,EAAE,iBAAiB,CAAC;YACzF,IAAI,CAAC,sBAAsB,GAAG,IAAI,sBAAS,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC;QACjE,CAAC;QAED,MAAM,UAAU,GAAG,eAAK,CAAC,aAAa,CAAC,EAAE,CAAC,cAAc,CAAC,CAAC;QAC1D,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;IACxB,CAAC;IAED;;OAEG;IACK,yBAAyB;QAC/B,IAAA,gBAAM,EAAC,IAAI,CAAC,KAAK,EAAE,IAAI,gCAAqB,CAAC,kCAAkC,CAAC,CAAC,CAAC;QAClF,IAAA,gBAAM,EAAC,IAAI,CAAC,OAAO,EAAE,IAAI,gCAAqB,CAAC,oCAAoC,CAAC,CAAC,CAAC;QACtF,IAAA,gBAAM,EACJ,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,EAC/C,IAAI,gCAAqB,CAAC,oDAAoD,CAAC,CAChF,CAAC;QACF,IAAA,gBAAM,EAAC,IAAI,CAAC,QAAQ,EAAE,IAAI,gCAAqB,CAAC,qCAAqC,CAAC,CAAC,CAAC;QACxF,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAEpC,8CAA8C;QAC9C,IAAI,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACxD,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;QACpD,CAAC;QAED,0EAA0E;QAC1E,0EAA0E;QAC1E,yEAAyE;QACzE,yDAAyD;QACzD,IAAI,IAAI,CAAC,sBAAsB,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;YACtC,MAAM,oBAAoB,GAAG,IAAI,CAAC,WAAW,CAAC,MAAM,CAClD,CAAC,GAAG,EAAE,CAAC,EAAE,EAAE,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,sBAAS,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,EAC7C,IAAI,sBAAS,CAAC,CAAC,CAAC,CACjB,CAAC;YACF,IAAA,gBAAM,EAAC,oBAAoB,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,IAAI,gCAAqB,CAAC,+CAA+C,CAAC,CAAC,CAAC;QACjH,CAAC;IACH,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;OAuBG;IACO,mBAAmB;QAC3B,IAAI,CAAC,yBAAyB,EAAE,CAAC;QACjC,MAAM,qBAAqB,GAAG,IAAI,0BAAkC,EAAE,CAAC;QAEvE,IAAI,IAAI,CAAC,OAAO,KAAK,IAAI,CAAC,QAAQ,CAAC,KAAK,IAAI,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAChG,iCAAiC;YACjC,wEAAwE;YACxE,kCAAkC;YAClC,oEAAoE;YACpE,MAAM,YAAY,GAA0B,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,MAAM,EAAE,EAAE,CAC5E,qBAAqB,CAAC,MAAM,CAAC,gBAAM,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CACvD,CAAC;YAEF,yEAAyE;YACzE,sEAAsE;YACtE,IAAI,IAAI,CAAC,sBAAsB,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;gBACtC,MAAM,CAAC,QAAQ,CAAC,GAAG,qBAAqB,CAAC,QAAQ,CAAC;oBAChD,MAAM,EAAE,yBAAyB;oBACjC,aAAa,EAAE,CAAC,eAAe,CAAC;oBAChC,SAAS,EAAE,CAAC,qBAAqB,CAAC,UAAU,CAAC,EAAE,MAAM,EAAE,MAAM,CAAC,IAAI,CAAC,sBAAsB,CAAC,OAAO,EAAE,CAAC,EAAE,CAAC,CAAC;iBACzG,CAAC,CAAC;gBACH,YAAY,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YAC9B,CAAC;YAED,MAAM,YAAY,GAAG,YAAY,CAAC,KAAK,EAAyB,CAAC;YACjE,IAAI,YAAY,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAC5B,qBAAqB,CAAC,UAAU,CAAC,YAAY,EAAE,YAAY,CAAC,CAAC;YAC/D,CAAC;YACD,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,SAAS,EAAE,EAAE;gBACrC,MAAM,WAAW,GAAG,qBAAqB,CAAC,UAAU,CAAC,YAAY,EAAE;oBACjE,qBAAqB,CAAC,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;iBACrD,CAAC,CAAC;gBACH,qBAAqB,CAAC,eAAe,CAAC,CAAC,WAAW,CAAC,EAAE,qBAAqB,CAAC,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC;YACxG,CAAC,CAAC,CAAC;YACH,MAAM,MAAM,GAAG,qBAAqB,CAAC,SAAS,CAAC;YAC/C,OAAO;gBACL,IAAI,EAAE,IAAI,CAAC,KAAK;gBAChB,MAAM,EAAE,IAAI,CAAC,OAAO;gBACpB,EAAE,EAAE;oBACF,MAAM,EAAE,CAAC,GAAG,MAAM,CAAC,MAAM,CAAC;oBAC1B,YAAY,EAAE,CAAC,GAAG,MAAM,CAAC,YAAY,CAAC;iBACvC;gBACD,OAAO,EAAE;oBACP,GAAG,IAAI,CAAC,QAAQ;iBACjB;gBACD,UAAU,EAAE,IAAI,CAAC,WAAW;gBAC5B,qBAAqB,EAAE,IAAI,CAAC,sBAAsB,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,sBAAsB,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,SAAS;aAC7G,CAAC;QACJ,CAAC;aAAM,IAAI,IAAI,CAAC,OAAO,KAAK,IAAI,CAAC,QAAQ,CAAC,KAAK,IAAI,IAAI,CAAC,sBAAsB,CAAC,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;YACrF,mEAAmE;YACnE,iFAAiF;YACjF,8FAA8F;YAC9F,MAAM,CAAC,QAAQ,CAAC,GAAG,qBAAqB,CAAC,QAAQ,CAAC;gBAChD,MAAM,EAAE,yBAAyB;gBACjC,aAAa,EAAE,CAAC,eAAe,CAAC;gBAChC,SAAS,EAAE,CAAC,qBAAqB,CAAC,UAAU,CAAC,EAAE,MAAM,EAAE,MAAM,CAAC,IAAI,CAAC,sBAAsB,CAAC,OAAO,EAAE,CAAC,EAAE,CAAC,CAAC;aACzG,CAAC,CAAC;YACH,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,SAAS,EAAE,EAAE;gBACrC,MAAM,WAAW,GAAG,qBAAqB,CAAC,UAAU,CAAC,QAAQ,EAAE;oBAC7D,qBAAqB,CAAC,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;iBACrD,CAAC,CAAC;gBACH,qBAAqB,CAAC,eAAe,CAAC,CAAC,WAAW,CAAC,EAAE,qBAAqB,CAAC,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CAAC;YACxG,CAAC,CAAC,CAAC;YACH,MAAM,QAAQ,GAAG,qBAAqB,CAAC,SAAS,CAAC;YACjD,OAAO;gBACL,IAAI,EAAE,IAAI,CAAC,KAAK;gBAChB,MAAM,EAAE,IAAI,CAAC,OAAO;gBACpB,EAAE,EAAE;oBACF,MAAM,EAAE,CAAC,GAAG,QAAQ,CAAC,MAAM,CAAC;oBAC5B,YAAY,EAAE,CAAC,GAAG,QAAQ,CAAC,YAAY,CAAC;iBACzC;gBACD,OAAO,EAAE;oBACP,GAAG,IAAI,CAAC,QAAQ;iBACjB;gBACD,UAAU,EAAE,IAAI,CAAC,WAAW;gBAC5B,qBAAqB,EAAE,IAAI,CAAC,sBAAsB,CAAC,OAAO,EAAE;aAC7D,CAAC;QACJ,CAAC;aAAM,CAAC;YACN,+GAA+G;YAC/G,gHAAgH;YAChH,IAAI,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,MAAM,IAAI,2BAAe,EAAE,CAAC;gBACpD,MAAM,iBAAiB,GAAG,IAAI,CAAC,QAAQ,CAAC,OAAO;qBAC5C,KAAK,CAAC,2BAAe,GAAG,CAAC,CAAC;qBAC1B,GAAG,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,gBAAM,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC,CAAC;gBAE7C,4GAA4G;gBAC5G,wGAAwG;gBACxG,4GAA4G;gBAC5G,wBAAwB;gBACxB,OAAO,iBAAiB,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBACpC,qBAAqB,CAAC,UAAU,CAC9B,qBAAqB,CAAC,GAAG,EACzB,iBAAiB,CAAC,MAAM,CAAC,CAAC,EAAE,4BAAgB,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,qBAAqB,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CACxG,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,CAAC,SAAS,EAAE,EAAE;gBACrC,MAAM,IAAI,GAAG,qBAAqB,CAAC,GAAG,CACpC,sBAAuB,CAAC,UAAU,CAAC,qBAAqB,CAAC,GAAG,EAAE;oBAC5D,qBAAqB,CAAC,IAAI,CAAC,MAAM,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;iBACrD,CAAC,CACH,CAAC;gBACF,qBAAqB,CAAC,GAAG,CACvB,sBAAuB,CAAC,eAAe,CAAC,CAAC,IAAI,CAAC,EAAE,qBAAqB,CAAC,MAAM,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC,CACjG,CAAC;YACJ,CAAC,CAAC,CAAC;YACH,MAAM,MAAM,GAAG,qBAAqB,CAAC,SAAS,CAAC;YAC/C,OAAO;gBACL,IAAI,EAAE,IAAI,CAAC,KAAK;gBAChB,MAAM,EAAE,IAAI,CAAC,OAAO;gBACpB,EAAE,EAAE;oBACF,MAAM,EAAE,CAAC,GAAG,MAAM,CAAC,MAAM,CAAC;oBAC1B,YAAY,EAAE,CAAC,GAAG,MAAM,CAAC,YAAY,CAAC;iBACvC;gBACD,OAAO,EAAE;oBACP,GAAG,IAAI,CAAC,QAAQ;oBAChB,OAAO,EAAE,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,2BAAe,GAAG,CAAC,CAAC;iBAC7D;gBACD,UAAU,EAAE,IAAI,CAAC,WAAW;gBAC5B,qBAAqB,EAAE,IAAI,CAAC,sBAAsB,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,sBAAsB,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,SAAS;aAC7G,CAAC;QACJ,CAAC;IACH,CAAC;CACF;AArTD,0CAqTC","sourcesContent":["import { TransactionBuilder } from './transactionBuilder';\nimport { BaseCoin as CoinConfig } from '@bitgo/statics';\nimport { BaseKey, BuildTransactionError, Recipient, TransactionType } from '@bitgo/sdk-core';\nimport { SuiTransaction, SuiTransactionType, TransferProgrammableTransaction } from './iface';\nimport { Transaction } from './transaction';\nimport { TransferTransaction } from './transferTransaction';\nimport assert from 'assert';\nimport {\n  Inputs,\n  Transactions as TransactionsConstructor,\n  TransactionBlock as ProgrammingTransactionBlockBuilder,\n  TransactionArgument,\n} from './mystenlab/builder';\nimport utils from './utils';\nimport { MAX_COMMAND_ARGS, MAX_GAS_OBJECTS } from './constants';\nimport BigNumber from 'bignumber.js';\n\nexport class TransferBuilder extends TransactionBuilder<TransferProgrammableTransaction> {\n  protected _recipients: Recipient[];\n  /**\n   * Balance held in the Sui address balance system (not in coin objects).\n   * When set, this amount is included in the total available balance for transfer.\n   * At execution time, Sui's GasCoin automatically draws from both coin objects\n   * (gasData.payment) and address balance, so SplitCoins(GasCoin, [amount])\n   * can spend funds from either source.\n   */\n  protected _fundsInAddressBalance: BigNumber = new BigNumber(0);\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    this._transaction = new TransferTransaction(_coinConfig);\n  }\n\n  protected get transactionType(): TransactionType {\n    return TransactionType.Send;\n  }\n\n  send(recipients: Recipient[]): this {\n    this.validateRecipients(recipients);\n    this._recipients = recipients;\n    return this;\n  }\n\n  /**\n   * Set the amount of funds held in the Sui address balance system for this sender.\n   * This is the `fundsInAddressBalance` value from the `suix_getBalance` RPC response.\n   * It is added to the coin object balances when computing total available funds.\n   *\n   * @param {string} amount - amount in MIST held in address balance\n   */\n  fundsInAddressBalance(amount: string): this {\n    this._fundsInAddressBalance = new BigNumber(amount);\n    return this;\n  }\n\n  /**\n   * Returns the total available balance: sum of all gas payment coin object\n   * balances plus any funds held in the address balance system.\n   *\n   * @param {BigNumber} coinObjectsBalance - sum of balances from gasData.payment coin objects\n   */\n  totalAvailableBalance(coinObjectsBalance: BigNumber): BigNumber {\n    return coinObjectsBalance.plus(this._fundsInAddressBalance);\n  }\n\n  /** @inheritdoc */\n  validateTransaction(transaction: TransferTransaction): void {\n    if (!transaction.suiTransaction) {\n      return;\n    }\n    this.validateTransactionFields();\n  }\n\n  /** @inheritdoc */\n  sign(key: BaseKey): void {\n    this.transaction.setSuiTransaction(this.buildSuiTransaction());\n    super.sign(key);\n  }\n\n  /** @inheritdoc */\n  protected fromImplementation(rawTransaction: string): Transaction<TransferProgrammableTransaction> {\n    const tx = new TransferTransaction(this._coinConfig);\n    this.validateRawTransaction(rawTransaction);\n    tx.fromRawTransaction(rawTransaction);\n    this.initBuilder(tx);\n    return this.transaction;\n  }\n\n  /** @inheritdoc */\n  protected async buildImplementation(): Promise<Transaction<TransferProgrammableTransaction>> {\n    this.transaction.setSuiTransaction(this.buildSuiTransaction());\n    this.transaction.transactionType(this.transactionType);\n\n    if (this._signer) {\n      this.transaction.sign(this._signer);\n    }\n\n    this._signatures.forEach((signature) => {\n      this.transaction.addSignature(signature.publicKey, signature.signature);\n    });\n\n    this.transaction.loadInputsAndOutputs();\n    return this.transaction;\n  }\n\n  /**\n   * Initialize the transaction builder fields using the decoded transaction data\n   *\n   * @param {Transaction} tx the transaction data\n   */\n  initBuilder(tx: TransferTransaction): void {\n    this._transaction = tx;\n\n    if (tx.signature && tx.signature.length > 0) {\n      this._signatures = [tx.suiSignature];\n    }\n\n    const txData = tx.toJson();\n    this.type(SuiTransactionType.Transfer);\n    this.sender(txData.sender);\n    this.gasData(txData.gasData);\n\n    if (txData.expiration && !('None' in txData.expiration)) {\n      this._expiration = txData.expiration;\n    }\n\n    if (txData.inputObjects) {\n      this.inputObjects(txData.inputObjects);\n    }\n\n    // Reconstruct fundsInAddressBalance from BalanceWithdrawal input if present.\n    // After BCS deserialization inputs are CallArg format: { BalanceWithdrawal: {...} }\n    // During building they are TransactionBlockInput format: { kind:'Input', value: { BalanceWithdrawal: {...} } }\n    const withdrawalInput = (tx.suiTransaction?.tx?.inputs as any[])?.find(\n      (input: any) =>\n        (input !== null && typeof input === 'object' && 'BalanceWithdrawal' in input) ||\n        (input?.value !== null && typeof input?.value === 'object' && 'BalanceWithdrawal' in (input.value ?? {}))\n    );\n    if (withdrawalInput) {\n      const bw = withdrawalInput.BalanceWithdrawal ?? withdrawalInput.value?.BalanceWithdrawal;\n      this._fundsInAddressBalance = new BigNumber(String(bw.amount));\n    }\n\n    const recipients = utils.getRecipients(tx.suiTransaction);\n    this.send(recipients);\n  }\n\n  /**\n   * Validates all fields are defined\n   */\n  private validateTransactionFields(): void {\n    assert(this._type, new BuildTransactionError('type is required before building'));\n    assert(this._sender, new BuildTransactionError('sender is required before building'));\n    assert(\n      this._recipients && this._recipients.length > 0,\n      new BuildTransactionError('at least one recipient is required before building')\n    );\n    assert(this._gasData, new BuildTransactionError('gasData is required before building'));\n    this.validateGasData(this._gasData);\n\n    // If inputObjects are provided, validate them\n    if (this._inputObjects && this._inputObjects.length > 0) {\n      this.validateInputObjectsBase(this._inputObjects);\n    }\n\n    // When fundsInAddressBalance is set, validate that total recipient amount\n    // does not exceed available address balance. Coin object balances are not\n    // stored in the builder (gasData.payment holds only ObjectRefs), so only\n    // the address balance portion can be cross-checked here.\n    if (this._fundsInAddressBalance.gt(0)) {\n      const totalRecipientAmount = this._recipients.reduce(\n        (acc, r) => acc.plus(new BigNumber(r.amount)),\n        new BigNumber(0)\n      );\n      assert(totalRecipientAmount.gt(0), new BuildTransactionError('total recipient amount must be greater than 0'));\n    }\n  }\n\n  /**\n   * Build transfer programmable transaction.\n   *\n   * Three build paths:\n   *\n   * Path 1a — Sponsored with coin objects (sender ≠ gasData.owner, inputObjects provided):\n   *   [optional withdrawal(fundsInAddressBalance) → redeem_funds → Coin<SUI>]\n   *   MergeCoins(inputObject[0], [inputObject[1..], addrCoin?])\n   *   SplitCoins(mergedObject, [amount]) → TransferObjects\n   *   Handles Cases 4 (coins only), 6 (coins, sponsor addr-bal gas), 8 (coins+addr-bal).\n   *\n   * Path 1b — Sponsored, address balance only (sender ≠ gasData.owner, no inputObjects):\n   *   withdrawal(fundsInAddressBalance) → redeem_funds → Coin<SUI>\n   *   SplitCoins(addrCoin, [amount]) → TransferObjects\n   *   Handles Case 5 (sponsor coin-object gas) and Case 7/Phase-4b (sponsor addr-bal gas).\n   *   Caller must set ValidDuring expiration when gasData.payment = [] (Cases 7, 9).\n   *\n   * Path 2 — Self-pay (sender === gasData.owner):\n   *   SplitCoins(GasCoin, [amount]) → TransferObjects\n   *   GasCoin at Sui execution time = gasData.payment objects merged + fundsInAddressBalance.\n   *   Handles Case 1 (coins), Case 2 (addr-bal only, caller sets ValidDuring), Case 3 (mixed).\n   *\n   * @protected\n   */\n  protected buildSuiTransaction(): SuiTransaction<TransferProgrammableTransaction> {\n    this.validateTransactionFields();\n    const programmableTxBuilder = new ProgrammingTransactionBlockBuilder();\n\n    if (this._sender !== this._gasData.owner && this._inputObjects && this._inputObjects.length > 0) {\n      // Path 1: sponsored transaction.\n      // The fee payer (gasData.owner) pays gas. The sender's funds come from:\n      //   - coin objects (inputObjects)\n      //   - address balance via tx.withdrawal() + 0x2::coin::redeem_funds\n      const inputObjects: TransactionArgument[] = this._inputObjects.map((object) =>\n        programmableTxBuilder.object(Inputs.ObjectRef(object))\n      );\n\n      // If the sender also has address balance, withdraw it as a Coin<SUI> and\n      // merge it into the coin-object pool before splitting for recipients.\n      if (this._fundsInAddressBalance.gt(0)) {\n        const [addrCoin] = programmableTxBuilder.moveCall({\n          target: '0x2::coin::redeem_funds',\n          typeArguments: ['0x2::sui::SUI'],\n          arguments: [programmableTxBuilder.withdrawal({ amount: BigInt(this._fundsInAddressBalance.toFixed()) })],\n        });\n        inputObjects.push(addrCoin);\n      }\n\n      const mergedObject = inputObjects.shift() as TransactionArgument;\n      if (inputObjects.length > 0) {\n        programmableTxBuilder.mergeCoins(mergedObject, inputObjects);\n      }\n      this._recipients.forEach((recipient) => {\n        const splitObject = programmableTxBuilder.splitCoins(mergedObject, [\n          programmableTxBuilder.pure(Number(recipient.amount)),\n        ]);\n        programmableTxBuilder.transferObjects([splitObject], programmableTxBuilder.object(recipient.address));\n      });\n      const txData = programmableTxBuilder.blockData;\n      return {\n        type: this._type,\n        sender: this._sender,\n        tx: {\n          inputs: [...txData.inputs],\n          transactions: [...txData.transactions],\n        },\n        gasData: {\n          ...this._gasData,\n        },\n        expiration: this._expiration,\n        fundsInAddressBalance: this._fundsInAddressBalance.gt(0) ? this._fundsInAddressBalance.toFixed() : undefined,\n      };\n    } else if (this._sender !== this._gasData.owner && this._fundsInAddressBalance.gt(0)) {\n      // Path 1b: sponsored, address balance only — no coin inputObjects.\n      // Sender's funds come entirely from address balance; gas is paid by the sponsor.\n      // withdrawal(fundsInAddressBalance) → redeem_funds → Coin<SUI> → SplitCoins → TransferObjects\n      const [addrCoin] = programmableTxBuilder.moveCall({\n        target: '0x2::coin::redeem_funds',\n        typeArguments: ['0x2::sui::SUI'],\n        arguments: [programmableTxBuilder.withdrawal({ amount: BigInt(this._fundsInAddressBalance.toFixed()) })],\n      });\n      this._recipients.forEach((recipient) => {\n        const splitObject = programmableTxBuilder.splitCoins(addrCoin, [\n          programmableTxBuilder.pure(Number(recipient.amount)),\n        ]);\n        programmableTxBuilder.transferObjects([splitObject], programmableTxBuilder.object(recipient.address));\n      });\n      const txData1b = programmableTxBuilder.blockData;\n      return {\n        type: this._type,\n        sender: this._sender,\n        tx: {\n          inputs: [...txData1b.inputs],\n          transactions: [...txData1b.transactions],\n        },\n        gasData: {\n          ...this._gasData,\n        },\n        expiration: this._expiration,\n        fundsInAddressBalance: this._fundsInAddressBalance.toFixed(),\n      };\n    } else {\n      // number of objects passed as gas payment should be strictly less than `MAX_GAS_OBJECTS`. When the transaction\n      // requires a larger number of inputs we use the merge command to merge the rest of the objects into the gasCoin\n      if (this._gasData.payment.length >= MAX_GAS_OBJECTS) {\n        const gasPaymentObjects = this._gasData.payment\n          .slice(MAX_GAS_OBJECTS - 1)\n          .map((object) => Inputs.ObjectRef(object));\n\n        // limit for total number of `args: CallArg[]` for a single command is MAX_COMMAND_ARGS so the max length of\n        // `sources[]` for a `mergeCoins(destination, sources[])` command is MAX_COMMAND_ARGS - 1 (1 used up for\n        // `destination`). We need to create a total of `gasPaymentObjects/(MAX_COMMAND_ARGS - 1)` merge commands to\n        // merge all the objects\n        while (gasPaymentObjects.length > 0) {\n          programmableTxBuilder.mergeCoins(\n            programmableTxBuilder.gas,\n            gasPaymentObjects.splice(0, MAX_COMMAND_ARGS - 1).map((object) => programmableTxBuilder.object(object))\n          );\n        }\n      }\n\n      this._recipients.forEach((recipient) => {\n        const coin = programmableTxBuilder.add(\n          TransactionsConstructor.SplitCoins(programmableTxBuilder.gas, [\n            programmableTxBuilder.pure(Number(recipient.amount)),\n          ])\n        );\n        programmableTxBuilder.add(\n          TransactionsConstructor.TransferObjects([coin], programmableTxBuilder.object(recipient.address))\n        );\n      });\n      const txData = programmableTxBuilder.blockData;\n      return {\n        type: this._type,\n        sender: this._sender,\n        tx: {\n          inputs: [...txData.inputs],\n          transactions: [...txData.transactions],\n        },\n        gasData: {\n          ...this._gasData,\n          payment: this._gasData.payment.slice(0, MAX_GAS_OBJECTS - 1),\n        },\n        expiration: this._expiration,\n        fundsInAddressBalance: this._fundsInAddressBalance.gt(0) ? this._fundsInAddressBalance.toFixed() : undefined,\n      };\n    }\n  }\n}\n"]}
300
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transferBuilder.js","sourceRoot":"","sources":["../../../src/lib/transferBuilder.ts"],"names":[],"mappings":"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","sourcesContent":["import { TransactionBuilder } from './transactionBuilder';\nimport { BaseCoin as CoinConfig } from '@bitgo/statics';\nimport { BaseKey, BuildTransactionError, Recipient, TransactionType } from '@bitgo/sdk-core';\nimport { SuiTransaction, SuiTransactionType, TransferProgrammableTransaction } from './iface';\nimport { Transaction } from './transaction';\nimport { TransferTransaction } from './transferTransaction';\nimport assert from 'assert';\nimport {\n  Inputs,\n  Transactions as TransactionsConstructor,\n  TransactionBlock as ProgrammingTransactionBlockBuilder,\n  TransactionArgument,\n} from './mystenlab/builder';\nimport utils from './utils';\nimport { MAX_COMMAND_ARGS, MAX_GAS_OBJECTS } from './constants';\nimport BigNumber from 'bignumber.js';\n\nexport class TransferBuilder extends TransactionBuilder<TransferProgrammableTransaction> {\n  protected _recipients: Recipient[];\n  /**\n   * Balance held in the Sui address balance system (not in coin objects).\n   * When set, this amount is included in the total available balance for transfer.\n   * At execution time, Sui's GasCoin automatically draws from both coin objects\n   * (gasData.payment) and address balance, so SplitCoins(GasCoin, [amount])\n   * can spend funds from either source.\n   */\n  protected _fundsInAddressBalance: BigNumber = new BigNumber(0);\n\n  constructor(_coinConfig: Readonly<CoinConfig>) {\n    super(_coinConfig);\n    this._transaction = new TransferTransaction(_coinConfig);\n  }\n\n  protected get transactionType(): TransactionType {\n    return TransactionType.Send;\n  }\n\n  send(recipients: Recipient[]): this {\n    this.validateRecipients(recipients);\n    this._recipients = recipients;\n    return this;\n  }\n\n  /**\n   * Set the amount of funds held in the Sui address balance system for this sender.\n   * This is the `fundsInAddressBalance` value from the `suix_getBalance` RPC response.\n   * It is added to the coin object balances when computing total available funds.\n   *\n   * @param {string} amount - amount in MIST held in address balance\n   */\n  fundsInAddressBalance(amount: string): this {\n    this._fundsInAddressBalance = new BigNumber(amount);\n    return this;\n  }\n\n  /**\n   * Returns the total available balance: sum of all gas payment coin object\n   * balances plus any funds held in the address balance system.\n   *\n   * @param {BigNumber} coinObjectsBalance - sum of balances from gasData.payment coin objects\n   */\n  totalAvailableBalance(coinObjectsBalance: BigNumber): BigNumber {\n    return coinObjectsBalance.plus(this._fundsInAddressBalance);\n  }\n\n  /** @inheritdoc */\n  validateTransaction(transaction: TransferTransaction): void {\n    if (!transaction.suiTransaction) {\n      return;\n    }\n    this.validateTransactionFields();\n  }\n\n  /** @inheritdoc */\n  sign(key: BaseKey): void {\n    this.transaction.setSuiTransaction(this.buildSuiTransaction());\n    super.sign(key);\n  }\n\n  /** @inheritdoc */\n  protected fromImplementation(rawTransaction: string): Transaction<TransferProgrammableTransaction> {\n    const tx = new TransferTransaction(this._coinConfig);\n    this.validateRawTransaction(rawTransaction);\n    tx.fromRawTransaction(rawTransaction);\n    this.initBuilder(tx);\n    return this.transaction;\n  }\n\n  /** @inheritdoc */\n  protected async buildImplementation(): Promise<Transaction<TransferProgrammableTransaction>> {\n    this.transaction.setSuiTransaction(this.buildSuiTransaction());\n    this.transaction.transactionType(this.transactionType);\n\n    if (this._signer) {\n      this.transaction.sign(this._signer);\n    }\n\n    this._signatures.forEach((signature) => {\n      this.transaction.addSignature(signature.publicKey, signature.signature);\n    });\n\n    this.transaction.loadInputsAndOutputs();\n    return this.transaction;\n  }\n\n  /**\n   * Initialize the transaction builder fields using the decoded transaction data\n   *\n   * @param {Transaction} tx the transaction data\n   */\n  initBuilder(tx: TransferTransaction): void {\n    this._transaction = tx;\n\n    if (tx.signature && tx.signature.length > 0) {\n      this._signatures = [tx.suiSignature];\n    }\n\n    const txData = tx.toJson();\n    this.type(SuiTransactionType.Transfer);\n    this.sender(txData.sender);\n    this.gasData(txData.gasData);\n\n    if (txData.expiration && !('None' in txData.expiration)) {\n      this._expiration = txData.expiration;\n    }\n\n    if (txData.inputObjects) {\n      this.inputObjects(txData.inputObjects);\n    }\n\n    // Reconstruct fundsInAddressBalance from BalanceWithdrawal input if present.\n    // After BCS deserialization inputs are CallArg format: { BalanceWithdrawal: {...} }\n    // During building they are TransactionBlockInput format: { kind:'Input', value: { BalanceWithdrawal: {...} } }\n    const withdrawalInput = (tx.suiTransaction?.tx?.inputs as any[])?.find(\n      (input: any) =>\n        (input !== null && typeof input === 'object' && 'BalanceWithdrawal' in input) ||\n        (input?.value !== null && typeof input?.value === 'object' && 'BalanceWithdrawal' in (input.value ?? {}))\n    );\n    if (withdrawalInput) {\n      const bw = withdrawalInput.BalanceWithdrawal ?? withdrawalInput.value?.BalanceWithdrawal;\n      this._fundsInAddressBalance = new BigNumber(String(bw.reservation?.MaxAmountU64 ?? bw.amount));\n    }\n\n    const recipients = utils.getRecipients(tx.suiTransaction);\n    this.send(recipients);\n  }\n\n  /**\n   * Validates all fields are defined\n   */\n  private validateTransactionFields(): void {\n    assert(this._type, new BuildTransactionError('type is required before building'));\n    assert(this._sender, new BuildTransactionError('sender is required before building'));\n    assert(\n      this._recipients && this._recipients.length > 0,\n      new BuildTransactionError('at least one recipient is required before building')\n    );\n    assert(this._gasData, new BuildTransactionError('gasData is required before building'));\n    this.validateGasData(this._gasData);\n\n    // If inputObjects are provided, validate them\n    if (this._inputObjects && this._inputObjects.length > 0) {\n      this.validateInputObjectsBase(this._inputObjects);\n    }\n\n    // When fundsInAddressBalance is set, validate that total recipient amount\n    // does not exceed available address balance. Coin object balances are not\n    // stored in the builder (gasData.payment holds only ObjectRefs), so only\n    // the address balance portion can be cross-checked here.\n    if (this._fundsInAddressBalance.gt(0)) {\n      const totalRecipientAmount = this._recipients.reduce(\n        (acc, r) => acc.plus(new BigNumber(r.amount)),\n        new BigNumber(0)\n      );\n      assert(totalRecipientAmount.gt(0), new BuildTransactionError('total recipient amount must be greater than 0'));\n    }\n  }\n\n  /**\n   * Build transfer programmable transaction.\n   *\n   * Three build paths:\n   *\n   * Path 1a — Sponsored with coin objects (sender ≠ gasData.owner, inputObjects provided):\n   *   [optional withdrawal(fundsInAddressBalance) → redeem_funds → Coin<SUI>]\n   *   MergeCoins(inputObject[0], [inputObject[1..], addrCoin?])\n   *   SplitCoins(mergedObject, [amount]) → TransferObjects\n   *   Handles Cases 4 (coins only), 6 (coins, sponsor addr-bal gas), 8 (coins+addr-bal).\n   *\n   * Path 1b — Sponsored, address balance only (sender ≠ gasData.owner, no inputObjects):\n   *   withdrawal(fundsInAddressBalance) → redeem_funds → Coin<SUI>\n   *   SplitCoins(addrCoin, [amount]) → TransferObjects\n   *   Handles Case 5 (sponsor coin-object gas) and Case 7/Phase-4b (sponsor addr-bal gas).\n   *   Caller must set ValidDuring expiration when gasData.payment = [] (Cases 7, 9).\n   *\n   * Path 2 — Self-pay (sender === gasData.owner):\n   *   SplitCoins(GasCoin, [amount]) → TransferObjects\n   *   GasCoin at Sui execution time = gasData.payment objects merged + fundsInAddressBalance.\n   *   Handles Case 1 (coins), Case 2 (addr-bal only, caller sets ValidDuring), Case 3 (mixed).\n   *\n   * @protected\n   */\n  protected buildSuiTransaction(): SuiTransaction<TransferProgrammableTransaction> {\n    this.validateTransactionFields();\n    const programmableTxBuilder = new ProgrammingTransactionBlockBuilder();\n\n    if (this._sender !== this._gasData.owner && this._inputObjects && this._inputObjects.length > 0) {\n      // Path 1: sponsored transaction.\n      // The fee payer (gasData.owner) pays gas. The sender's funds come from:\n      //   - coin objects (inputObjects)\n      //   - address balance via tx.withdrawal() + 0x2::coin::redeem_funds\n      const inputObjects: TransactionArgument[] = this._inputObjects.map((object) =>\n        programmableTxBuilder.object(Inputs.ObjectRef(object))\n      );\n\n      // If the sender also has address balance, withdraw it as a Coin<SUI> and\n      // merge it into the coin-object pool before splitting for recipients.\n      if (this._fundsInAddressBalance.gt(0)) {\n        const [addrCoin] = programmableTxBuilder.moveCall({\n          target: '0x2::coin::redeem_funds',\n          typeArguments: ['0x2::sui::SUI'],\n          arguments: [programmableTxBuilder.withdrawal({ amount: BigInt(this._fundsInAddressBalance.toFixed()) })],\n        });\n        inputObjects.push(addrCoin);\n      }\n\n      const mergedObject = inputObjects.shift() as TransactionArgument;\n      if (inputObjects.length > 0) {\n        programmableTxBuilder.mergeCoins(mergedObject, inputObjects);\n      }\n      this._recipients.forEach((recipient) => {\n        const splitObject = programmableTxBuilder.splitCoins(mergedObject, [\n          programmableTxBuilder.pure(Number(recipient.amount)),\n        ]);\n        programmableTxBuilder.transferObjects([splitObject], programmableTxBuilder.object(recipient.address));\n      });\n      const txData = programmableTxBuilder.blockData;\n      return {\n        type: this._type,\n        sender: this._sender,\n        tx: {\n          inputs: [...txData.inputs],\n          transactions: [...txData.transactions],\n        },\n        gasData: {\n          ...this._gasData,\n        },\n        expiration: this._expiration,\n        fundsInAddressBalance: this._fundsInAddressBalance.gt(0) ? this._fundsInAddressBalance.toFixed() : undefined,\n      };\n    } else if (this._sender !== this._gasData.owner && this._fundsInAddressBalance.gt(0)) {\n      // Path 1b: sponsored, address balance only — no coin inputObjects.\n      // Sender's funds come entirely from address balance; gas is paid by the sponsor.\n      // withdrawal(fundsInAddressBalance) → redeem_funds → Coin<SUI> → SplitCoins → TransferObjects\n      const [addrCoin] = programmableTxBuilder.moveCall({\n        target: '0x2::coin::redeem_funds',\n        typeArguments: ['0x2::sui::SUI'],\n        arguments: [programmableTxBuilder.withdrawal({ amount: BigInt(this._fundsInAddressBalance.toFixed()) })],\n      });\n      // addrCoin is a command result (from redeem_funds) and must be explicitly consumed.\n      // Split for every recipient — keeps a 1:1 SplitCoins↔TransferObjects structure for the\n      // transaction parser.  After all splits, consume the source coin explicitly:\n      //   - If there is change (fundsInAddressBalance > total recipient amount): return addrCoin\n      //     to sender via TransferObjects.\n      //   - If send-all (no change): addrCoin has 0 balance; destroy it by merging into the\n      //     sponsor's gas coin (a 0-value merge is valid in Sui and deletes the source object).\n      const totalRecipientAmount = this._recipients.reduce((sum, r) => sum.plus(r.amount), new BigNumber(0));\n      const hasChange = this._fundsInAddressBalance.gt(totalRecipientAmount);\n\n      this._recipients.forEach((recipient) => {\n        const splitObject = programmableTxBuilder.splitCoins(addrCoin, [\n          programmableTxBuilder.pure(Number(recipient.amount)),\n        ]);\n        programmableTxBuilder.transferObjects([splitObject], programmableTxBuilder.object(recipient.address));\n      });\n\n      if (hasChange) {\n        // Return the remaining balance (change) to the sender.\n        programmableTxBuilder.transferObjects([addrCoin], programmableTxBuilder.object(this._sender));\n      } else {\n        // Send-all: addrCoin has 0 balance after all splits.  Merge it into the sponsor's gas\n        // coin to destroy the zero-balance object (coin::join accepts a 0-value source).\n        programmableTxBuilder.mergeCoins(programmableTxBuilder.gas, [addrCoin]);\n      }\n      const txData1b = programmableTxBuilder.blockData;\n      return {\n        type: this._type,\n        sender: this._sender,\n        tx: {\n          inputs: [...txData1b.inputs],\n          transactions: [...txData1b.transactions],\n        },\n        gasData: {\n          ...this._gasData,\n        },\n        expiration: this._expiration,\n        fundsInAddressBalance: this._fundsInAddressBalance.toFixed(),\n      };\n    } else {\n      // number of objects passed as gas payment should be strictly less than `MAX_GAS_OBJECTS`. When the transaction\n      // requires a larger number of inputs we use the merge command to merge the rest of the objects into the gasCoin\n      if (this._gasData.payment.length >= MAX_GAS_OBJECTS) {\n        const gasPaymentObjects = this._gasData.payment\n          .slice(MAX_GAS_OBJECTS - 1)\n          .map((object) => Inputs.ObjectRef(object));\n\n        // limit for total number of `args: CallArg[]` for a single command is MAX_COMMAND_ARGS so the max length of\n        // `sources[]` for a `mergeCoins(destination, sources[])` command is MAX_COMMAND_ARGS - 1 (1 used up for\n        // `destination`). We need to create a total of `gasPaymentObjects/(MAX_COMMAND_ARGS - 1)` merge commands to\n        // merge all the objects\n        while (gasPaymentObjects.length > 0) {\n          programmableTxBuilder.mergeCoins(\n            programmableTxBuilder.gas,\n            gasPaymentObjects.splice(0, MAX_COMMAND_ARGS - 1).map((object) => programmableTxBuilder.object(object))\n          );\n        }\n      }\n\n      this._recipients.forEach((recipient) => {\n        const coin = programmableTxBuilder.add(\n          TransactionsConstructor.SplitCoins(programmableTxBuilder.gas, [\n            programmableTxBuilder.pure(Number(recipient.amount)),\n          ])\n        );\n        programmableTxBuilder.add(\n          TransactionsConstructor.TransferObjects([coin], programmableTxBuilder.object(recipient.address))\n        );\n      });\n      const txData = programmableTxBuilder.blockData;\n      return {\n        type: this._type,\n        sender: this._sender,\n        tx: {\n          inputs: [...txData.inputs],\n          transactions: [...txData.transactions],\n        },\n        gasData: {\n          ...this._gasData,\n          payment: this._gasData.payment.slice(0, MAX_GAS_OBJECTS - 1),\n        },\n        expiration: this._expiration,\n        fundsInAddressBalance: this._fundsInAddressBalance.gt(0) ? this._fundsInAddressBalance.toFixed() : undefined,\n      };\n    }\n  }\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"utils.d.ts","sourceRoot":"","sources":["../../../src/lib/utils.ts"],"names":[],"mappings":"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"}