Clearpool
Cantina Security Report
- Organization 
- @clearpool
- Engagement Type 
- Cantina Reviews 
- Period 
- - 
- Repositories 
Findings
High Risk
4 findings
4 fixed
0 acknowledged
Medium Risk
2 findings
2 fixed
0 acknowledged
Low Risk
3 findings
2 fixed
1 acknowledged
Informational
4 findings
2 fixed
2 acknowledged
High Risk4 findings
- Decimal truncation on asset vaults different to 18 decimals loss precission- Severity 
- Severity: High
- Submitted by 
- slowfi 
 - Description- Found by Clearpool team. - When using asset of less than 18 decimals, the protocol truncates the exchange rate reducing the shares calculation. This happens on - TellerWithMultiAssetSupport.sol,- AccountantWithRateProviders.soland- AtomicQueue.sol.- The decimal truncation worsen the accountant of profits with time due to interest compounding. - Recommendation- Align decimal calculation and exchange rates across all the shares and assets calculations to minimize precision loss. - Clearpool team: Fixed on commit ID 3f87019c2e5f853e99b9f1dc6ba30fe7e8b16834. This fix includes new functions to allow vaults to operate with base asset less than 18 decimals and/or exchange asset with different decimals. Rate providers are fixed to return value on 18 decimals. The protocol now operates by escalating all amounts to 18 decimal asset, doing any required calculation and then returned to the original decimal base asset amount. This reduces the decimal loss precision calculation. - Cantina Managed: Fix verified. 
- Rate provider return amount inconsistency- Severity 
- Severity: High
- Submitted by 
- slowfi 
 - Description- On commit ID 3ce9f7a82de3cb3ba72fc7945d01d4bb8fd8b82f, the new introduced functions - calculateSharesForAmountand- calculateAmountForSharesexpected the- rateProviderto return the value of- getRatefunction on 18 decimals. However the- claimFeesfunctions expected the value to be returned on quote decimals.- This inconsistency lead to improper amount calculations on with non pegged assets. - Recommendation- Consider using a standard way across the code on how - rateProviderreturns the quoted scaled amounts.- Clearpool: Fixed on commit ID 3f87019c2e5f853e99b9f1dc6ba30fe7e8b16834. - Cantina Managed: Fix verified. The fix expected all - rateProviderto return its value scaled to 18 decimals.
- Incorrect scaling in _calculateWantAmount- Severity 
- Severity: High
- Submitted by 
- slowfi 
 - (deposit path) when vault decimals ≠ 18- Description- The - _calculateWantAmountfrom- AtomicQueue.solhas three different paths to calculate the amount wanted for the exchange.- The first path may fail to obtain the right amount as the provided amount may be different than 18 decimals. This when multiplying by the rate, scaled to 18 and dividing by - 1e18the resulting amount is scaled to the original asset decimal and not to 18 as expected.- The second path may fail if the vault asset uses 6 decimal asset as it presumes the shares are in 18 decimals and vault asset may use a different scale. - Recommendation- For the first path consider using, - ONE_SHAREfor division to ensure returning in 18 decimals.- offerAmount.mulDivDown(rate, ONE_SHARE);- For the second path consider multiplying by - ONE_SHAREinstead of- 1e18to obtain the shares escaled to vault decimals.
- Inconsistent scaling in claimFees may zero out transfers- Severity 
- Severity: High
- Submitted by 
- slowfi 
 - Description- In the non-pegged branch of - claimFees, the conversion mixes units: fees are scaled to the fee asset’s decimals, but division uses a- rateexpected at 18 decimals. With e.g. base = 18 decimals, fee asset = 6 decimals, small accrued fees can round down to zero while- _feesOwedInBaseis still cleared.- Recommendation- Consider to align units by replacing with: - .mulDivDown(1e18, rate)
Medium Risk2 findings
- Public checkpoint() mutates state while paused- State 
- Severity 
- Severity: Medium
- Submitted by 
- slowfi 
 - Description- When the contract is paused ( - state._isPaused = true), the expected behavior is that sensitive state changes are halted until governance intervention. However,- checkpoint()remains callable by any address without pause restrictions. Calling- checkpoint()executes- _checkpointInterestAndFees(), which updates- _exchangeRate,- _feesOwedInBase, and- _lastAccrualTimebased on the current exchange rate.- If the pause was triggered by an out-of-bounds exchange rate in - updateExchangeRate, calling- checkpoint()during the paused state can still store that invalid rate into contract storage along with associated fee accruals. This undermines the purpose of pausing by committing potentially unsafe values that should remain frozen.- For example: - Trigger updateExchangeRatewith an out-of-bounds value so the contract pauses.
- While paused, call checkpoint()from any address.
- _exchangeRateand accrual data are still updated in storage despite the paused state.
 - This behavior allows invalid or manipulated data to be persisted, potentially affecting redemptions, pricing, or downstream accounting. - Recommendation- Consider to prevent - checkpoint()from executing while the contract is paused, unless called by an explicitly authorized recovery role. This ensures that no invalid exchange rate or accrual data is committed during the paused period, preserving the integrity of the pause mechanism.- Clearpool: Added requiresAuth to restrict the call of this function to controlled addresses. Fixed on commit ID cb325156c14bc83dff16909cf59471e4de31d2d5. - Cantina Managed: Fix verified. 
- Invalid Exchange Rate Causes Permanent State Corruption in Fee Calculations- State 
- Severity 
- Severity: Medium
- Submitted by 
- slowfi 
 - Description- The - updateExchangeRate()function stores invalid exchange rates that are subsequently used as the base for all future interest calculations. The function executes- _checkpointInterestAndFees()before validation, but always stores the new exchange rate.- When an invalid rate is provided: - _checkpointInterestAndFees()executes with the valid old rate
- Validation fails and the contract pauses
- The invalid rate is stored
- Future calls to _checkpointInterestAndFeesuse this invalid rate as the base
 - This means any subsequent checkpoint will calculate interest using potentially invalid data, and the error compounds over time. - Recommendation- Consider to enforce pause semantics consistently across all writers: - Consider to make _checkpointInterestAndFees()a no-op when paused and optionally revertcheckpoint()while paused. This requires to remove the the if control pause flow statement ofupdateExchangeRate()to work.
- Consider to pause-gate any privileged function that indirectly checkpoints or ensure they early-exit without accrual when paused.
- Consider to add tests covering: after a pause, calls to checkpoint()or admin setters do not mutate_exchangeRate,_feesOwedInBase, or_lastAccrualTimeuntil unpaused.
 - Clearpool: Fixed on commit ID cb325156c14bc83dff16909cf59471e4de31d2d5. The fix erase the pause control flow on - updateExchangeRateto let authorized roles to change the exchange rate in case is improperly set on previous interactions. Having the contract paused prevents further operations from happening, but we still allow to store the checkpoint on subsequent calls as it is consider a potential negligible amount that may not disrupt normal operations due to the off-chain monitoring systems in place.- Cantina Managed: Fix verified according to Clearpool team requirements. 
Low Risk3 findings
- Insufficient event coverage for state changes- State 
- Severity 
- Severity: Low
- Submitted by 
- slowfi 
 - Description- Several state-changing paths do not emit dedicated events, reducing observability and making off-chain monitoring and accounting harder: - 
Accrual checkpoints — checkpoint()calls_checkpointInterestAndFees()which mutates_exchangeRate,_feesOwedInBase, and_lastAccrualTimewithout emitting any event. Accruals triggered indirectly from other functions also leave no checkpoint trace.
- 
Share-lock configuration — setShareLockPeriod(uint64)updatesshareLockPeriodwithout emitting an event, despite affecting transfer restrictions (beforeTransfer) and the refundable window used byrefundDeposit.
- 
Pause via updateExchangeRate— When bounds/time checks fail,updateExchangeRatesetsstate._isPaused = truebut does not emitPaused()(and duplicates pause logic). This creates inconsistent signaling compared to the dedicatedpause()function that does emit.
 - Recommendation- Consider to emit explicit events for these transitions and to reuse existing pause logic to avoid divergence. This would align all pause triggers with consistent on-chain signaling, and improve monitoring of accrual and configuration changes. 
- Inconsistent enforcement when lowering maxLendingRate- Severity 
- Severity: Low
- Submitted by 
- slowfi 
 - Description- setMaxLendingRateonly updates the cap and emits- MaxLendingRateUpdated. If governance lowers- maxLendingRatebelow the current- lendingInfo._lendingRate, the contract continues accruing at the now out-of-policy lending rate until a separate- setLendingRatetransaction is sent. This creates a window where accruals don’t reflect the newly enforced maximum. Raising the cap is benign.- Recommendation- Consider to align the active lending rate with the new cap in the same transaction when the cap is reduced: - Finalize accruals up to the change (i.e., checkpoint first).
- Clamp lendingInfo._lendingRatetomin(currentRate, newMax).
- Emit a rate-update event to make the adjustment observable.
 - Alternatively, revert if the new cap is below the current rate unless the call explicitly opts into clamping (e.g., via a parameter), ensuring policy and accrual remain consistent immediately after the cap change. 
- solve can be DoSed when a single request is invalid- State 
- Severity 
- Severity: Low
- Submitted by 
- high byte 
 - Summary- when a single request is invalid, instead of skipping it the entire solve will revert. - it is recommended to - continueinstead.- also noteworthy that - transferFromcan also fail so it is best to try-catch around that too.- Clearpool: Acknowledged as the fix for this issue introduced a new issue that does not have a trivial fix. - Cantina Managed: Acknowledged by Clearpool team. 
Informational4 findings
- Use of raw 1e4 instead of a named constant- State 
- Severity 
- Severity: Informational
- Submitted by 
- slowfi 
 - Description- In - AccountantWithRateProviders, the checks for- _allowedExchangeRateChangeUpperand- _allowedExchangeRateChangeLoweruse the literal- 1e4as the denominator for basis points calculations. While functionally correct, using a named constant such as- BASIS_POINTSimproves clarity and maintainability, making it explicit that these values are expressed in basis points.- Recommendation- Consider to replace the raw - 1e4literal with a descriptive constant (e.g.,- uint256 internal constant BASIS_POINTS = 1e4;) and use it consistently across the codebase for all basis point math. This reduces the risk of misinterpretation and improves code readability.
- Unused return value and unnecessary computation in calculateExchangeRateWithInterest- State 
- Acknowledged
- Severity 
- Severity: Informational
- Submitted by 
- slowfi 
 - Description- calculateExchangeRateWithInterest()returns- (newRate, interestAccrued), but across its call sites (- getRate,- getRateSafe,- getRateInQuote,- previewFeesOwed, and- _checkpointInterestAndFees) only- newRateis consumed. This implies the function performs extra work (computing- interestAccrued) that is not needed for current usage, increasing gas and cognitive load.- Recommendation- Consider to decouple responsibilities so callers compute only what they need. For example, expose a lightweight function that returns just the updated rate, and keep a separate path for callers that require - interestAccrued. Align call sites to the minimal interface, reducing redundant calculations and improving readability.- Clearpool: Acknowledged. Decided to keep this value for off-chain and external integration. - Cantina Managed: Acknowledged by Clearpool team. 
- Increasing Interest Rate creates price drift risk in AtomicQueue (mitigated by current withdraw-only usage)- State 
- Acknowledged
- Severity 
- Severity: Informational
- Submitted by 
- slowfi 
 - Description- AtomicQueueprices requests at execution time using the current NAV from the accountant. When there is time between request creation and fulfillment, the exchange rate can drift (e.g., due to continuous interest accrual).- Example scenario: - User creates request: “Buy vault shares with 1,000 USDC.”
- At request time: 1 share = 10 USDC → expects ~100 shares.
- Later, interest accrual raises NAV: 1 share = 10.50 USDC.
- On execution, user only receives ~95.2 shares instead of 100.
 - This creates an expectation gap for flows like “buy shares with USDC,” where appreciating share price means the user ultimately receives fewer shares than anticipated. - In the current Clearpool protocol, the queue is used only for withdrawals (users offer vault shares → receive assets), and the solver role is operated by Clearpool. In this constrained mode, rate drift tends to work in favor of withdrawing users (more assets per share as time passes) and operational risk is reduced by having the protocol as solver. Nevertheless, the mechanism remains sensitive to time-based price movement and would resurface as a user-facing slippage problem if the queue later supports deposits/buy-share intents. - Recommendation- 
Consider to encode user price expectations in requests if bi-directional usage is introduced later: - Add a per-request minAmountOut(for withdrawals) ormaxSharesPerAsset/ max acceptable rate (for deposits), enforced at fulfillment.
- Keep short deadlines to limit drift exposure.
 
- Add a per-request 
- 
If the intent is withdraw-only, consider to enforce that at the contract level (restrict allowed offer/wantpairs) and document that execution uses real-time NAV so users understand outcomes may improve/change with time.
- 
For future deposit support, consider UI and solver policies that quote and respect slippage bounds, or snapshot a reference rate at intent time and enforce a tolerance at solve. 
 - Clearpool: Acknowledged. This will be a to-do item if Clearpool supports deposit via AtomicQueue in the future. For now it is not an essential fix. - Cantina Managed: Acknowledged by Clearpool team. 
- bound check incomplete- Severity 
- Severity: Informational
- Submitted by 
- high byte 
 - Summary- this (together with the complementary function) only performs bound check, but does not check if - _allowedExchangeRateChangeUpper > _allowedExchangeRateChangeLower, basically allowing within-bounds values but in reverse order.