mirror of
https://github.com/0glabs/0g-chain.git
synced 2025-01-15 01:35:21 +00:00
911f9f59d6
* add regression test for hooks; refactor deposit to commit pool and shares in one place with commit after validation checks; panic on pool and share set methods if the record is invalid * use correct hook ordering; add regression test for order - fails if hooks are not called before or after share record set
75 lines
2.8 KiB
Go
75 lines
2.8 KiB
Go
package keeper
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
|
sdkerrors "github.com/cosmos/cosmos-sdk/types/errors"
|
|
|
|
"github.com/kava-labs/kava/x/swap/types"
|
|
)
|
|
|
|
// Withdraw removes liquidity from an existing pool from an owners deposit, converting the provided shares for
|
|
// the returned pool liquidity.
|
|
//
|
|
// If 100% of the owners shares are removed, then the deposit is deleted. In addition, if all the pool shares
|
|
// are removed then the pool is deleted.
|
|
//
|
|
// The number of shares must be large enough to result in at least 1 unit of the smallest reserve in the pool.
|
|
// If the share input is below the minimum required for positive liquidity to be remove from both reserves, a
|
|
// insufficient error is returned.
|
|
//
|
|
// In addition, if the withdrawn liquidity for each reserve is below the provided minimum, a slippage exceeded
|
|
// error is returned.
|
|
func (k Keeper) Withdraw(ctx sdk.Context, owner sdk.AccAddress, shares sdk.Int, minCoinA, minCoinB sdk.Coin) error {
|
|
poolID := types.PoolID(minCoinA.Denom, minCoinB.Denom)
|
|
|
|
shareRecord, found := k.GetDepositorShares(ctx, owner, poolID)
|
|
if !found {
|
|
return sdkerrors.Wrapf(types.ErrDepositNotFound, "no deposit for account %s and pool %s", owner, poolID)
|
|
}
|
|
|
|
if shares.GT(shareRecord.SharesOwned) {
|
|
return sdkerrors.Wrapf(types.ErrInvalidShares, "withdraw of %s shares greater than %s shares owned", shares, shareRecord.SharesOwned)
|
|
}
|
|
|
|
poolRecord, found := k.GetPool(ctx, poolID)
|
|
if !found {
|
|
panic(fmt.Sprintf("pool %s not found", poolID))
|
|
}
|
|
|
|
pool, err := types.NewDenominatedPoolWithExistingShares(poolRecord.Reserves(), poolRecord.TotalShares)
|
|
if err != nil {
|
|
panic(fmt.Sprintf("invalid pool %s: %s", poolID, err))
|
|
}
|
|
|
|
withdrawnAmount := pool.RemoveLiquidity(shares)
|
|
if withdrawnAmount.AmountOf(minCoinA.Denom).IsZero() || withdrawnAmount.AmountOf(minCoinB.Denom).IsZero() {
|
|
return sdkerrors.Wrap(types.ErrInsufficientLiquidity, "shares must be increased")
|
|
}
|
|
if withdrawnAmount.AmountOf(minCoinA.Denom).LT(minCoinA.Amount) || withdrawnAmount.AmountOf(minCoinB.Denom).LT(minCoinB.Amount) {
|
|
return sdkerrors.Wrap(types.ErrSlippageExceeded, "minimum withdraw not met")
|
|
}
|
|
|
|
k.updatePool(ctx, poolID, pool)
|
|
k.BeforePoolDepositModified(ctx, poolID, owner, shareRecord.SharesOwned)
|
|
k.updateShares(ctx, owner, poolID, shareRecord.SharesOwned.Sub(shares))
|
|
|
|
err = k.supplyKeeper.SendCoinsFromModuleToAccount(ctx, types.ModuleAccountName, owner, withdrawnAmount)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
|
|
ctx.EventManager().EmitEvent(
|
|
sdk.NewEvent(
|
|
types.EventTypeSwapWithdraw,
|
|
sdk.NewAttribute(types.AttributeKeyPoolID, poolID),
|
|
sdk.NewAttribute(types.AttributeKeyOwner, owner.String()),
|
|
sdk.NewAttribute(sdk.AttributeKeyAmount, withdrawnAmount.String()),
|
|
sdk.NewAttribute(types.AttributeKeyShares, shares.String()),
|
|
),
|
|
)
|
|
|
|
return nil
|
|
}
|