-
Notifications
You must be signed in to change notification settings - Fork 0
Description
Summary
When closing a YieldVault backed by an FCM (FlowCreditMarket) strategy, residual yield tokens (e.g., tauUSDFv) remain in the AutoBalancer and are burned during cleanup. This occurs because FCM's withdrawal logic only pulls enough from the AutoBalancer to repay the MOET debt — any yield profit beyond that stays in the AutoBalancer and is destroyed when the YieldVault is burned.
Impact
- Value loss: User loses accrued yield profit (the residual tokens burned)
- Affected strategies: Primarily FCM-backed strategies (
mUSDFStrategy) — non-FCM strategies have minimal risk (see below)
Mainnet Evidence (WETH → tauUSDFv)
From close tx events:
AutoBalancer.Withdrawn amount=0.01911391, balanceAfter=0.00000759
FungibleToken.Burned amount=0.00000759 ← residual tauUSDFv burned
AutoBalancer.ResourceDestroyed
Root Cause
closeYieldVaultcallsyieldVault.withdraw(amount: getYieldVaultBalance())(FlowYieldVaults.cdc:415)- For FCM strategies, the Strategy's source is
position.createSourceWithOptions(type: collateralType, pullFromTopUpSource: true) - FCM's Position Source determines withdrawal based on debt repayment requirements, not total AutoBalancer balance
- Since yield accrued, AutoBalancer holds more value than needed to repay — FCM only pulls what's necessary
Burner.burn(<-yieldVault)triggersStrategy.burnCallback→_cleanupAutoBalancer→ residual tokens destroyed
Additional Issue: Inconsistent Quote Logic in SwapSource
There's also an inconsistency in SwapConnectors.SwapSource:
minimumAvailable()usesquoteOutto estimate collateral from all yield tokenswithdrawAvailable(maxAmount)usesquoteInwhenmaxAmount == minimumAvaildue to<condition
// SwapConnectors.cdc:600-609
var quote = minimumAvail < maxAmount // ← uses < not <=
? self.swapper.quoteOut(...) // withdraws ALL (when minimumAvail < maxAmount)
: self.swapper.quoteIn(...) // calculates needed input (when minimumAvail >= maxAmount)When closing PM strategies where maxAmount == minimumAvail:
- Condition
minimumAvail < maxAmountis FALSE - Uses
quoteInwhich may return slightly less input due to swap math rounding - Tiny dust left in AutoBalancer → burned
Why Non-FCM Strategies Are Less Affected
For PM strategies (syWFLOWvStrategy, tauUSDFvStrategy, FUSDEVStrategy):
- No FCM debt constraint — on close,
maxAmount == minimumAvail - SwapSource uses
quoteIn(forDesired: maxAmount)which should withdraw nearly all input - Residual risk: Tiny dust from swap rounding (not significant yield loss)
- FCM strategies: Leave entire yield profit because
maxAmount < minimumAvail(FCM limits withdrawal to debt repayment)
| Strategy Type | Residual Risk | Amount |
|---|---|---|
FCM strategies (mUSDFStrategy) |
HIGH | Entire yield profit (significant) |
| PM strategies | LOW | Swap rounding dust (tiny) |
Proposed Fix
1. Fix SwapSource quote consistency (SwapConnectors.cdc)
Change < to <= so that when minimumAvail == maxAmount, it uses quoteOut (consistent with how minimumAvailable() calculated the value):
var quote = minimumAvail <= maxAmount // ← change < to <=
? self.swapper.quoteOut(forProvided: self.source.minimumAvailable(), reverse: false)
: self.swapper.quoteIn(forDesired: maxAmount, reverse: false)This ensures PM strategies withdraw ALL yield tokens on close, eliminating dust.
2. Sweep AutoBalancer before burn (FlowYieldVaults.cdc)
Add closeYieldVaultAndSweepAutoBalancer method that:
- Withdraws collateral via normal flow
- Sweeps any remaining AutoBalancer balance to a caller-provided receiver
- Optionally swaps residual yield tokens back to collateral
- Then burns the YieldVault
This is a safety net for FCM strategies where minimumAvail > maxAmount due to debt constraints — the SwapSource fix alone doesn't help there.
3. Update close transaction (close_yield_vault.cdc)
Use the new closeYieldVaultAndSweepAutoBalancer method and handle the swept yield tokens (swap to collateral if route exists).