While Curve lending pools support swaps in both the wrapped and underlying coins, not all lending pools allow liquidity providers to deposit or withdraw the underlying coin.
For example, the Compound Pool allows swaps between cDai and cUSDC (wrapped coins), as well as swaps between DAI and USDC (underlying coins). However, liquidity providers are not able to deposit DAI or USDC to the pool directly. The main reason for why this is not supported by all Curve lending pools lies in the size limit of contracts. Lending pools may differ in complexity and can end up being very close to the contract byte code size limit. In order to overcome this restriction, liquidity can be added and removed to and from a lending pool in the underlying coins via a different contract, called a deposit zap, tailored to lending pools.
For an overview of the Curve lending pool implementation, please refer to the Lending Pool section.
The template source code for a lending pool deposit zap may be viewed on GitHub.
Note
Lending pool deposit zaps may differ in their API. Older pools do not implement the newer API template.
Getters generated for public arrays changed between Vyper 0.1.x and 0.2.x to accept uint256 instead of int128 in order to handle the lookups. Older deposit zap contracts (v1) use vyper 0.1.x..., while newer zaps (v2) use vyper 0.2.x....
Wrap underlying coins and deposit them in the pool.
Input
Type
Description
uamounts
uint256[N_COINS]
List of amounts of underlying coins to deposit
min_mint_amount
uint256
Minimum amount of LP token to mint from the deposit
Emits: AddLiquidity Transfer
Source code
USE_LENDING:constant(bool[N_COINS])=[True,True]...@public@nonreentrant('lock')defadd_liquidity(uamounts:uint256[N_COINS],min_mint_amount:uint256):use_lending:bool[N_COINS]=USE_LENDINGtethered:bool[N_COINS]=TETHEREDamounts:uint256[N_COINS]=ZEROSforiinrange(N_COINS):uamount:uint256=uamounts[i]ifuamount>0:# Transfer the underlying coin from owneriftethered[i]:USDT(self.underlying_coins[i]).transferFrom(msg.sender,self,uamount)else:assert_modifiable(ERC20(self.underlying_coins[i])\
.transferFrom(msg.sender,self,uamount))# Mint if neededifuse_lending[i]:ERC20(self.underlying_coins[i]).approve(self.coins[i],uamount)ok:uint256=cERC20(self.coins[i]).mint(uamount)ifok>0:raise"Could not mint coin"amounts[i]=cERC20(self.coins[i]).balanceOf(self)ERC20(self.coins[i]).approve(self.curve,amounts[i])else:amounts[i]=uamountERC20(self.underlying_coins[i]).approve(self.curve,uamount)Curve(self.curve).add_liquidity(amounts,min_mint_amount)tokens:uint256=ERC20(self.token).balanceOf(self)assert_modifiable(ERC20(self.token).transfer(msg.sender,tokens))
@public@nonreentrant('lock')defadd_liquidity(amounts:uint256[N_COINS],min_mint_amount:uint256):# Amounts is amounts of c-tokensassertnotself.is_killedtethered:bool[N_COINS]=TETHEREDuse_lending:bool[N_COINS]=USE_LENDINGfees:uint256[N_COINS]=ZEROS_fee:uint256=self.fee*N_COINS/(4*(N_COINS-1))_admin_fee:uint256=self.admin_feetoken_supply:uint256=self.token.totalSupply()rates:uint256[N_COINS]=self._current_rates()# Initial invariantD0:uint256=0old_balances:uint256[N_COINS]=self.balancesiftoken_supply>0:D0=self.get_D_mem(rates,old_balances)new_balances:uint256[N_COINS]=old_balancesforiinrange(N_COINS):iftoken_supply==0:assertamounts[i]>0# balances store amounts of c-tokensnew_balances[i]=old_balances[i]+amounts[i]# Invariant after changeD1:uint256=self.get_D_mem(rates,new_balances)assertD1>D0# We need to recalculate the invariant accounting for fees# to calculate fair user's shareD2:uint256=D1iftoken_supply>0:# Only account for fees if we are not the first to depositforiinrange(N_COINS):ideal_balance:uint256=D1*old_balances[i]/D0difference:uint256=0ifideal_balance>new_balances[i]:difference=ideal_balance-new_balances[i]else:difference=new_balances[i]-ideal_balancefees[i]=_fee*difference/FEE_DENOMINATORself.balances[i]=new_balances[i]-(fees[i]*_admin_fee/FEE_DENOMINATOR)new_balances[i]-=fees[i]D2=self.get_D_mem(rates,new_balances)else:self.balances=new_balances# Calculate, how much pool tokens to mintmint_amount:uint256=0iftoken_supply==0:mint_amount=D1# Take the dust if there was anyelse:mint_amount=token_supply*(D2-D0)/D0assertmint_amount>=min_mint_amount,"Slippage screwed you"# Take coins from the senderforiinrange(N_COINS):iftethered[i]andnotuse_lending[i]:USDT(self.coins[i]).transferFrom(msg.sender,self,amounts[i])else:assert_modifiable(cERC20(self.coins[i]).transferFrom(msg.sender,self,amounts[i]))# Mint pool tokensself.token.mint(msg.sender,mint_amount)log.AddLiquidity(msg.sender,amounts,fees,D1,token_supply+mint_amount)
@publicdefmint(_to:address,_value:uint256):""" @dev Mint an amount of the token and assigns it to an account. This encapsulates the modification of balances such that the proper events are emitted. @param _to The account that will receive the created tokens. @param _value The amount that will be created. """assertmsg.sender==self.minterassert_to!=ZERO_ADDRESSself.total_supply+=_valueself.balanceOf[_to]+=_valuelog.Transfer(ZERO_ADDRESS,_to,_value)
@privatedef_send_all(_addr:address,min_uamounts:uint256[N_COINS],one:int128):use_lending:bool[N_COINS]=USE_LENDINGtethered:bool[N_COINS]=TETHEREDforiinrange(N_COINS):if(one<0)or(i==one):ifuse_lending[i]:_coin:address=self.coins[i]_balance:uint256=cERC20(_coin).balanceOf(self)if_balance==0:# Do nothing if there are 0 coinscontinueok:uint256=cERC20(_coin).redeem(_balance)ifok>0:raise"Could not redeem coin"_ucoin:address=self.underlying_coins[i]_uamount:uint256=ERC20(_ucoin).balanceOf(self)assert_uamount>=min_uamounts[i],"Not enough coins withdrawn"# Send only if we have something to sendif_uamount>=0:iftethered[i]:USDT(_ucoin).transfer(_addr,_uamount)else:assert_modifiable(ERC20(_ucoin).transfer(_addr,_uamount))@public@nonreentrant('lock')defremove_liquidity(_amount:uint256,min_uamounts:uint256[N_COINS]):zeros:uint256[N_COINS]=ZEROSassert_modifiable(ERC20(self.token).transferFrom(msg.sender,self,_amount))Curve(self.curve).remove_liquidity(_amount,zeros)self._send_all(msg.sender,min_uamounts,-1)
@public@nonreentrant('lock')defremove_liquidity(_amount:uint256,min_amounts:uint256[N_COINS]):total_supply:uint256=self.token.totalSupply()amounts:uint256[N_COINS]=ZEROSfees:uint256[N_COINS]=ZEROStethered:bool[N_COINS]=TETHEREDuse_lending:bool[N_COINS]=USE_LENDINGforiinrange(N_COINS):value:uint256=self.balances[i]*_amount/total_supplyassertvalue>=min_amounts[i],"Withdrawal resulted in fewer coins than expected"self.balances[i]-=valueamounts[i]=valueiftethered[i]andnotuse_lending[i]:USDT(self.coins[i]).transfer(msg.sender,value)else:assert_modifiable(cERC20(self.coins[i]).transfer(msg.sender,value))self.token.burnFrom(msg.sender,_amount)# Will raise if not enoughlog.RemoveLiquidity(msg.sender,amounts,fees,total_supply-_amount)
@privatedef_burn(_to:address,_value:uint256):""" @dev Internal function that burns an amount of the token of a given account. @param _to The account whose tokens will be burned. @param _value The amount that will be burned. """assert_to!=ZERO_ADDRESSself.total_supply-=_valueself.balanceOf[_to]-=_valuelog.Transfer(_to,ZERO_ADDRESS,_value)...@publicdefburnFrom(_to:address,_value:uint256):""" @dev Burn an amount of the token from a given account. @param _to The account whose tokens will be burned. @param _value The amount that will be burned. """assertmsg.sender==self.minter,"Only minter is allowed to burn"self._burn(_to,_value)
Withdraw and unwrap coins from the pool in an imbalanced amount.
Input
Type
Description
uamounts
uint256[N_COINS]
List of amounts of underlying coins to withdraw
max_burn_amount
uint256
Maximum amount of LP token to burn in the withdrawal
Emits: Transfer RemoveLiquidityImbalance
Source code
@privatedef_send_all(_addr:address,min_uamounts:uint256[N_COINS],one:int128):use_lending:bool[N_COINS]=USE_LENDINGtethered:bool[N_COINS]=TETHEREDforiinrange(N_COINS):if(one<0)or(i==one):ifuse_lending[i]:_coin:address=self.coins[i]_balance:uint256=cERC20(_coin).balanceOf(self)if_balance==0:# Do nothing if there are 0 coinscontinueok:uint256=cERC20(_coin).redeem(_balance)ifok>0:raise"Could not redeem coin"_ucoin:address=self.underlying_coins[i]_uamount:uint256=ERC20(_ucoin).balanceOf(self)assert_uamount>=min_uamounts[i],"Not enough coins withdrawn"# Send only if we have something to sendif_uamount>=0:iftethered[i]:USDT(_ucoin).transfer(_addr,_uamount)else:assert_modifiable(ERC20(_ucoin).transfer(_addr,_uamount))@public@nonreentrant('lock')defremove_liquidity_imbalance(uamounts:uint256[N_COINS],max_burn_amount:uint256):""" Get max_burn_amount in, remove requested liquidity and transfer back what is left """use_lending:bool[N_COINS]=USE_LENDINGtethered:bool[N_COINS]=TETHERED_token:address=self.tokenamounts:uint256[N_COINS]=uamountsforiinrange(N_COINS):ifuse_lending[i]andamounts[i]>0:rate:uint256=cERC20(self.coins[i]).exchangeRateCurrent()amounts[i]=amounts[i]*LENDING_PRECISION/rate# if not use_lending - all good already# Transfrer max tokens in_tokens:uint256=ERC20(_token).balanceOf(msg.sender)if_tokens>max_burn_amount:_tokens=max_burn_amountassert_modifiable(ERC20(_token).transferFrom(msg.sender,self,_tokens))Curve(self.curve).remove_liquidity_imbalance(amounts,max_burn_amount)# Transfer unused tokens back_tokens=ERC20(_token).balanceOf(self)assert_modifiable(ERC20(_token).transfer(msg.sender,_tokens))# Unwrap and transfer all the coins we've gotself._send_all(msg.sender,ZEROS,-1)
@public@nonreentrant('lock')defremove_liquidity_imbalance(amounts:uint256[N_COINS],max_burn_amount:uint256):assertnotself.is_killedtethered:bool[N_COINS]=TETHEREDuse_lending:bool[N_COINS]=USE_LENDINGtoken_supply:uint256=self.token.totalSupply()asserttoken_supply>0_fee:uint256=self.fee*N_COINS/(4*(N_COINS-1))_admin_fee:uint256=self.admin_feerates:uint256[N_COINS]=self._current_rates()old_balances:uint256[N_COINS]=self.balancesnew_balances:uint256[N_COINS]=old_balancesD0:uint256=self.get_D_mem(rates,old_balances)foriinrange(N_COINS):new_balances[i]-=amounts[i]D1:uint256=self.get_D_mem(rates,new_balances)fees:uint256[N_COINS]=ZEROSforiinrange(N_COINS):ideal_balance:uint256=D1*old_balances[i]/D0difference:uint256=0ifideal_balance>new_balances[i]:difference=ideal_balance-new_balances[i]else:difference=new_balances[i]-ideal_balancefees[i]=_fee*difference/FEE_DENOMINATORself.balances[i]=new_balances[i]-(fees[i]*_admin_fee/FEE_DENOMINATOR)new_balances[i]-=fees[i]D2:uint256=self.get_D_mem(rates,new_balances)token_amount:uint256=(D0-D2)*token_supply/D0asserttoken_amount>0asserttoken_amount<=max_burn_amount,"Slippage screwed you"foriinrange(N_COINS):iftethered[i]andnotuse_lending[i]:USDT(self.coins[i]).transfer(msg.sender,amounts[i])else:assert_modifiable(cERC20(self.coins[i]).transfer(msg.sender,amounts[i]))self.token.burnFrom(msg.sender,token_amount)# Will raise if not enoughlog.RemoveLiquidityImbalance(msg.sender,amounts,fees,D1,token_supply-token_amount)
@privatedef_burn(_to:address,_value:uint256):""" @dev Internal function that burns an amount of the token of a given account. @param _to The account whose tokens will be burned. @param _value The amount that will be burned. """assert_to!=ZERO_ADDRESSself.total_supply-=_valueself.balanceOf[_to]-=_valuelog.Transfer(_to,ZERO_ADDRESS,_value)...@publicdefburnFrom(_to:address,_value:uint256):""" @dev Burn an amount of the token from a given account. @param _to The account whose tokens will be burned. @param _value The amount that will be burned. """assertmsg.sender==self.minter,"Only minter is allowed to burn"self._burn(_to,_value)
@public@nonreentrant('lock')defremove_liquidity_one_coin(_token_amount:uint256,i:int128,min_uamount:uint256,donate_dust:bool=False):""" Remove _amount of liquidity all in a form of coin i """use_lending:bool[N_COINS]=USE_LENDINGrates:uint256[N_COINS]=ZEROS_token:address=self.tokenforjinrange(N_COINS):ifuse_lending[j]:rates[j]=cERC20(self.coins[j]).exchangeRateCurrent()else:rates[j]=LENDING_PRECISIONdy:uint256=self._calc_withdraw_one_coin(_token_amount,i,rates)assertdy>=min_uamount,"Not enough coins removed"assert_modifiable(ERC20(self.token).transferFrom(msg.sender,self,_token_amount))amounts:uint256[N_COINS]=ZEROSamounts[i]=dy*LENDING_PRECISION/rates[i]token_amount_before:uint256=ERC20(_token).balanceOf(self)Curve(self.curve).remove_liquidity_imbalance(amounts,_token_amount)# Unwrap and transfer all the coins we've gotself._send_all(msg.sender,ZEROS,i)ifnotdonate_dust:# Transfer unused tokens backtoken_amount_after:uint256=ERC20(_token).balanceOf(self)iftoken_amount_after>token_amount_before:assert_modifiable(ERC20(_token).transfer(msg.sender,token_amount_after-token_amount_before))
@public@nonreentrant('lock')defremove_liquidity_imbalance(amounts:uint256[N_COINS],max_burn_amount:uint256):assertnotself.is_killedtethered:bool[N_COINS]=TETHEREDuse_lending:bool[N_COINS]=USE_LENDINGtoken_supply:uint256=self.token.totalSupply()asserttoken_supply>0_fee:uint256=self.fee*N_COINS/(4*(N_COINS-1))_admin_fee:uint256=self.admin_feerates:uint256[N_COINS]=self._current_rates()old_balances:uint256[N_COINS]=self.balancesnew_balances:uint256[N_COINS]=old_balancesD0:uint256=self.get_D_mem(rates,old_balances)foriinrange(N_COINS):new_balances[i]-=amounts[i]D1:uint256=self.get_D_mem(rates,new_balances)fees:uint256[N_COINS]=ZEROSforiinrange(N_COINS):ideal_balance:uint256=D1*old_balances[i]/D0difference:uint256=0ifideal_balance>new_balances[i]:difference=ideal_balance-new_balances[i]else:difference=new_balances[i]-ideal_balancefees[i]=_fee*difference/FEE_DENOMINATORself.balances[i]=new_balances[i]-(fees[i]*_admin_fee/FEE_DENOMINATOR)new_balances[i]-=fees[i]D2:uint256=self.get_D_mem(rates,new_balances)token_amount:uint256=(D0-D2)*token_supply/D0asserttoken_amount>0asserttoken_amount<=max_burn_amount,"Slippage screwed you"foriinrange(N_COINS):iftethered[i]andnotuse_lending[i]:USDT(self.coins[i]).transfer(msg.sender,amounts[i])else:assert_modifiable(cERC20(self.coins[i]).transfer(msg.sender,amounts[i]))self.token.burnFrom(msg.sender,token_amount)# Will raise if not enoughlog.RemoveLiquidityImbalance(msg.sender,amounts,fees,D1,token_supply-token_amount)
@privatedef_burn(_to:address,_value:uint256):""" @dev Internal function that burns an amount of the token of a given account. @param _to The account whose tokens will be burned. @param _value The amount that will be burned. """assert_to!=ZERO_ADDRESSself.total_supply-=_valueself.balanceOf[_to]-=_valuelog.Transfer(_to,ZERO_ADDRESS,_value)...@publicdefburnFrom(_to:address,_value:uint256):""" @dev Burn an amount of the token from a given account. @param _to The account whose tokens will be burned. @param _value The amount that will be burned. """assertmsg.sender==self.minter,"Only minter is allowed to burn"self._burn(_to,_value)
Note
The underlying pool method called when the older DepositZap contract's remove_liquidity_one_coin is called emits RemoveLiquidityImbalance whereas the newer contract emits RemoveLiquidityOne. This is because the older contracts do not have the remove_liquidity_one_coin, and instead use remove_liquidity_imbalance.
Calculate the amount received when withdrawing a single underlying coin.
Input
Type
Description
_token_amount
uint256
Amount of LP tokens to burn in the withdrawal
i
int128
Index value of the coin to withdraw
Source code
@private@constantdef_calc_withdraw_one_coin(_token_amount:uint256,i:int128,rates:uint256[N_COINS])->uint256:# First, need to calculate# * Get current D# * Solve Eqn against y_i for D - _token_amountuse_lending:bool[N_COINS]=USE_LENDING# tethered: bool[N_COINS] = TETHEREDcrv:address=self.curveA:uint256=Curve(crv).A()fee:uint256=Curve(crv).fee()*N_COINS/(4*(N_COINS-1))fee+=fee*FEE_IMPRECISION/FEE_DENOMINATOR# Overcharge to account for imprecisionprecisions:uint256[N_COINS]=PRECISION_MULtotal_supply:uint256=ERC20(self.token).totalSupply()xp:uint256[N_COINS]=PRECISION_MULS:uint256=0forjinrange(N_COINS):xp[j]*=Curve(crv).balances(j)ifuse_lending[j]:# Use stored rate b/c we have imprecision anywayxp[j]=xp[j]*rates[j]/LENDING_PRECISIONS+=xp[j]# if not use_lending - all good alreadyD0:uint256=self.get_D(A,xp)D1:uint256=D0-_token_amount*D0/total_supplyxp_reduced:uint256[N_COINS]=xp# xp = xp - fee * | xp * D1 / D0 - (xp - S * dD / D0 * (0, ... 1, ..0))|forjinrange(N_COINS):dx_expected:uint256=0b_ideal:uint256=xp[j]*D1/D0b_expected:uint256=xp[j]ifj==i:b_expected-=S*(D0-D1)/D0ifb_ideal>=b_expected:dx_expected=(b_ideal-b_expected)else:dx_expected=(b_expected-b_ideal)xp_reduced[j]-=fee*dx_expected/FEE_DENOMINATORdy:uint256=xp_reduced[i]-self.get_y(A,i,xp_reduced,D1)dy=dy/precisions[i]returndy@public@constantdefcalc_withdraw_one_coin(_token_amount:uint256,i:int128)->uint256:rates:uint256[N_COINS]=ZEROSuse_lending:bool[N_COINS]=USE_LENDINGforjinrange(N_COINS):ifuse_lending[j]:rates[j]=cERC20(self.coins[j]).exchangeRateStored()else:rates[j]=10**18returnself._calc_withdraw_one_coin(_token_amount,i,rates)
Compared to the older deposit zaps, the newer zaps mainly optimize for gas efficiency. The API is only modified in part, specifically with regards to return values and variable naming.
Getter for the pool associated with this deposit contract.
Source code
@externaldef__init__(_coins:address[N_COINS],_underlying_coins:address[N_COINS],_curve:address,_token:address):""" @notice Contract constructor @dev Where a token does not use wrapping, use the same address for `_coins` and `_underlying_coins` @param _coins List of wrapped coin addresses @param _underlying_coins List of underlying coin addresses @param _curve Pool address @param _token Pool LP token address """foriinrange(N_COINS):assert_coins[i]!=ZERO_ADDRESSassert_underlying_coins[i]!=ZERO_ADDRESS# approve underlying and wrapped coins for infinite transfers_response:Bytes[32]=raw_call(_underlying_coins[i],concat(method_id("approve(address,uint256)"),convert(_coins[i],bytes32),convert(MAX_UINT256,bytes32),),max_outsize=32,)iflen(_response)>0:assertconvert(_response,bool)_response=raw_call(_coins[i],concat(method_id("approve(address,uint256)"),convert(_curve,bytes32),convert(MAX_UINT256,bytes32),),max_outsize=32,)iflen(_response)>0:assertconvert(_response,bool)self.coins=_coinsself.underlying_coins=_underlying_coinsself.curve=_curveself.lp_token=_token
Getter for the array of underlying coins within the associated pool.
Input
Type
Description
i
int128
Index of the underlying coin for which to get the address
Source code
underlying_coins:public(address[N_COINS])...@externaldef__init__(_coins:address[N_COINS],_underlying_coins:address[N_COINS],_curve:address,_token:address):""" @notice Contract constructor @dev Where a token does not use wrapping, use the same address for `_coins` and `_underlying_coins` @param _coins List of wrapped coin addresses @param _underlying_coins List of underlying coin addresses @param _curve Pool address @param _token Pool LP token address """foriinrange(N_COINS):assert_coins[i]!=ZERO_ADDRESSassert_underlying_coins[i]!=ZERO_ADDRESS# approve underlying and wrapped coins for infinite transfers_response:Bytes[32]=raw_call(_underlying_coins[i],concat(method_id("approve(address,uint256)"),convert(_coins[i],bytes32),convert(MAX_UINT256,bytes32),),max_outsize=32,)iflen(_response)>0:assertconvert(_response,bool)_response=raw_call(_coins[i],concat(method_id("approve(address,uint256)"),convert(_curve,bytes32),convert(MAX_UINT256,bytes32),),max_outsize=32,)iflen(_response)>0:assertconvert(_response,bool)self.coins=_coinsself.underlying_coins=_underlying_coinsself.curve=_curveself.lp_token=_token
Getter for the array of wrapped coins within the associated pool.
Input
Type
Description
i
int128
Index of the coin for which to get the address
Source code
coins:public(address[N_COINS])...@externaldef__init__(_coins:address[N_COINS],_underlying_coins:address[N_COINS],_curve:address,_token:address):""" @notice Contract constructor @dev Where a token does not use wrapping, use the same address for `_coins` and `_underlying_coins` @param _coins List of wrapped coin addresses @param _underlying_coins List of underlying coin addresses @param _curve Pool address @param _token Pool LP token address """foriinrange(N_COINS):assert_coins[i]!=ZERO_ADDRESSassert_underlying_coins[i]!=ZERO_ADDRESS# approve underlying and wrapped coins for infinite transfers_response:Bytes[32]=raw_call(_underlying_coins[i],concat(method_id("approve(address,uint256)"),convert(_coins[i],bytes32),convert(MAX_UINT256,bytes32),),max_outsize=32,)iflen(_response)>0:assertconvert(_response,bool)_response=raw_call(_coins[i],concat(method_id("approve(address,uint256)"),convert(_curve,bytes32),convert(MAX_UINT256,bytes32),),max_outsize=32,)iflen(_response)>0:assertconvert(_response,bool)self.coins=_coinsself.underlying_coins=_underlying_coinsself.curve=_curveself.lp_token=_token
lp_token:public(address)...@externaldef__init__(_coins:address[N_COINS],_underlying_coins:address[N_COINS],_curve:address,_token:address):""" @notice Contract constructor @dev Where a token does not use wrapping, use the same address for `_coins` and `_underlying_coins` @param _coins List of wrapped coin addresses @param _underlying_coins List of underlying coin addresses @param _curve Pool address @param _token Pool LP token address """foriinrange(N_COINS):assert_coins[i]!=ZERO_ADDRESSassert_underlying_coins[i]!=ZERO_ADDRESS# approve underlying and wrapped coins for infinite transfers_response:Bytes[32]=raw_call(_underlying_coins[i],concat(method_id("approve(address,uint256)"),convert(_coins[i],bytes32),convert(MAX_UINT256,bytes32),),max_outsize=32,)iflen(_response)>0:assertconvert(_response,bool)_response=raw_call(_coins[i],concat(method_id("approve(address,uint256)"),convert(_curve,bytes32),convert(MAX_UINT256,bytes32),),max_outsize=32,)iflen(_response)>0:assertconvert(_response,bool)self.coins=_coinsself.underlying_coins=_underlying_coinsself.curve=_curveself.lp_token=_token
Wrap underlying coins and deposit them in the pool. Returns the amount of LP token received in exchange for the deposited amounts.
Input
Type
Description
_underlying_amounts
uint256[N_COINS]
List of amounts of underlying coins to deposit
_min_mint_amount
uint256
Minimum amount of LP token to mint from the deposit
Emits: Transfer AddLiquidity
Source code
@public@nonreentrant('lock')defadd_liquidity(uamounts:uint256[N_COINS],min_mint_amount:uint256):tethered:bool[N_COINS]=TETHEREDamounts:uint256[N_COINS]=ZEROSforiinrange(N_COINS):uamount:uint256=uamounts[i]ifuamount>0:# Transfer the underlying coin from owneriftethered[i]:USDT(self.underlying_coins[i]).transferFrom(msg.sender,self,uamount)else:assert_modifiable(ERC20(self.underlying_coins[i])\
.transferFrom(msg.sender,self,uamount))# Mint if neededERC20(self.underlying_coins[i]).approve(self.coins[i],uamount)yERC20(self.coins[i]).deposit(uamount)amounts[i]=yERC20(self.coins[i]).balanceOf(self)ERC20(self.coins[i]).approve(self.curve,amounts[i])Curve(self.curve).add_liquidity(amounts,min_mint_amount)tokens:uint256=ERC20(self.token).balanceOf(self)assert_modifiable(ERC20(self.token).transfer(msg.sender,tokens))
@external@nonreentrant('lock')defadd_liquidity(amounts:uint256[N_COINS],min_mint_amount:uint256)->uint256:""" @notice Deposit coins into the pool @param amounts List of amounts of coins to deposit @param min_mint_amount Minimum amount of LP tokens to mint from the deposit @return Amount of LP tokens received by depositing """assertnotself.is_killed# dev: is killedamp:uint256=self._A()_lp_token:address=self.lp_tokentoken_supply:uint256=ERC20(_lp_token).totalSupply()# Initial invariantD0:uint256=0old_balances:uint256[N_COINS]=self.balancesiftoken_supply>0:D0=self.get_D_mem(old_balances,amp)new_balances:uint256[N_COINS]=old_balancesforiinrange(N_COINS):iftoken_supply==0:assertamounts[i]>0# dev: initial deposit requires all coins# balances store amounts of c-tokensnew_balances[i]=old_balances[i]+amounts[i]# Invariant after changeD1:uint256=self.get_D_mem(new_balances,amp)assertD1>D0# We need to recalculate the invariant accounting for fees# to calculate fair user's shareD2:uint256=D1fees:uint256[N_COINS]=empty(uint256[N_COINS])iftoken_supply>0:# Only account for fees if we are not the first to deposit_fee:uint256=self.fee*N_COINS/(4*(N_COINS-1))_admin_fee:uint256=self.admin_feeforiinrange(N_COINS):ideal_balance:uint256=D1*old_balances[i]/D0difference:uint256=0ifideal_balance>new_balances[i]:difference=ideal_balance-new_balances[i]else:difference=new_balances[i]-ideal_balancefees[i]=_fee*difference/FEE_DENOMINATORself.balances[i]=new_balances[i]-(fees[i]*_admin_fee/FEE_DENOMINATOR)new_balances[i]-=fees[i]D2=self.get_D_mem(new_balances,amp)else:self.balances=new_balances# Calculate, how much pool tokens to mintmint_amount:uint256=0iftoken_supply==0:mint_amount=D1# Take the dust if there was anyelse:mint_amount=token_supply*(D2-D0)/D0assertmint_amount>=min_mint_amount,"Slippage screwed you"# Take coins from the senderforiinrange(N_COINS):ifamounts[i]>0:# "safeTransferFrom" which works for ERC20s which return bool or not_response:Bytes[32]=raw_call(self.coins[i],concat(method_id("transferFrom(address,address,uint256)"),convert(msg.sender,bytes32),convert(self,bytes32),convert(amounts[i],bytes32),),max_outsize=32,)# dev: failed transferiflen(_response)>0:assertconvert(_response,bool)# Mint pool tokensCurveToken(_lp_token).mint(msg.sender,mint_amount)logAddLiquidity(msg.sender,amounts,fees,D1,token_supply+mint_amount)returnmint_amount
Withdraw and unwrap coins from the pool. Returns list of amounts of underlying coins that were withdrawn.
Input
Type
Description
_amount
uint256
Quantity of LP tokens to burn in the withdrawal
_min_underlying_amounts
uint256[N_COINS]
Minimum amounts of underlying coins to receive
Emits: Transfer RemoveLiquidity
Source code
@internaldef_unwrap_and_transfer(_addr:address,_min_amounts:uint256[N_COINS])->uint256[N_COINS]:# unwrap coins and transfer them to the senderuse_lending:bool[N_COINS]=USE_LENDING_amounts:uint256[N_COINS]=empty(uint256[N_COINS])foriinrange(N_COINS):ifuse_lending[i]:_coin:address=self.coins[i]_balance:uint256=ERC20(_coin).balanceOf(self)if_balance==0:# Do nothing if there are 0 coinscontinueyERC20(_coin).withdraw(_balance)_ucoin:address=self.underlying_coins[i]_uamount:uint256=ERC20(_ucoin).balanceOf(self)assert_uamount>=_min_amounts[i],"Not enough coins withdrawn"# Send only if we have something to sendif_uamount!=0:_response:Bytes[32]=raw_call(_ucoin,concat(method_id("transfer(address,uint256)"),convert(_addr,bytes32),convert(_uamount,bytes32)),max_outsize=32)iflen(_response)>0:assertconvert(_response,bool)_amounts[i]=_uamountreturn_amounts@external@nonreentrant('lock')defremove_liquidity(_amount:uint256,_min_underlying_amounts:uint256[N_COINS])->uint256[N_COINS]:""" @notice Withdraw and unwrap coins from the pool @dev Withdrawal amounts are based on current deposit ratios @param _amount Quantity of LP tokens to burn in the withdrawal @param _min_underlying_amounts Minimum amounts of underlying coins to receive @return List of amounts of underlying coins that were withdrawn """assertERC20(self.lp_token).transferFrom(msg.sender,self,_amount)Curve(self.curve).remove_liquidity(_amount,empty(uint256[N_COINS]))returnself._unwrap_and_transfer(msg.sender,_min_underlying_amounts)
@external@nonreentrant('lock')defremove_liquidity(_amount:uint256,min_amounts:uint256[N_COINS])->uint256[N_COINS]:""" @notice Withdraw coins from the pool @dev Withdrawal amounts are based on current deposit ratios @param _amount Quantity of LP tokens to burn in the withdrawal @param min_amounts Minimum amounts of underlying coins to receive @return List of amounts of coins that were withdrawn """_lp_token:address=self.lp_tokentotal_supply:uint256=ERC20(_lp_token).totalSupply()amounts:uint256[N_COINS]=empty(uint256[N_COINS])fees:uint256[N_COINS]=empty(uint256[N_COINS])# Fees are unused but we've got them historically in eventforiinrange(N_COINS):value:uint256=self.balances[i]*_amount/total_supplyassertvalue>=min_amounts[i],"Withdrawal resulted in fewer coins than expected"self.balances[i]-=valueamounts[i]=value_response:Bytes[32]=raw_call(self.coins[i],concat(method_id("transfer(address,uint256)"),convert(msg.sender,bytes32),convert(value,bytes32),),max_outsize=32,)# dev: failed transferiflen(_response)>0:assertconvert(_response,bool)CurveToken(_lp_token).burnFrom(msg.sender,_amount)# dev: insufficient fundslogRemoveLiquidity(msg.sender,amounts,fees,total_supply-_amount)returnamounts
Withdraw and unwrap coins from the pool in an imbalanced amount. Amounts in _underlying_amounts correspond to withdrawn amounts before any fees charge for unwrapping Returns list of amounts of underlying coins that were withdrawn.
Input
Type
Description
_underlying_amounts
uint256[N_COINS]
List of amounts of underlying coins to withdraw
_max_burn_amount
uint256
Maximum amount of LP token to burn in the withdrawal
Emits: Transfer RemoveLiquidityImbalance
Source code
@internaldef_unwrap_and_transfer(_addr:address,_min_amounts:uint256[N_COINS])->uint256[N_COINS]:# unwrap coins and transfer them to the senderuse_lending:bool[N_COINS]=USE_LENDING_amounts:uint256[N_COINS]=empty(uint256[N_COINS])foriinrange(N_COINS):ifuse_lending[i]:_coin:address=self.coins[i]_balance:uint256=ERC20(_coin).balanceOf(self)if_balance==0:# Do nothing if there are 0 coinscontinueyERC20(_coin).withdraw(_balance)_ucoin:address=self.underlying_coins[i]_uamount:uint256=ERC20(_ucoin).balanceOf(self)assert_uamount>=_min_amounts[i],"Not enough coins withdrawn"# Send only if we have something to sendif_uamount!=0:_response:Bytes[32]=raw_call(_ucoin,concat(method_id("transfer(address,uint256)"),convert(_addr,bytes32),convert(_uamount,bytes32)),max_outsize=32)iflen(_response)>0:assertconvert(_response,bool)_amounts[i]=_uamountreturn_amounts@external@nonreentrant('lock')defremove_liquidity_imbalance(_underlying_amounts:uint256[N_COINS],_max_burn_amount:uint256)->uint256[N_COINS]:""" @notice Withdraw and unwrap coins from the pool in an imbalanced amount @dev Amounts in `_underlying_amounts` correspond to withdrawn amounts before any fees charge for unwrapping. @param _underlying_amounts List of amounts of underlying coins to withdraw @param _max_burn_amount Maximum amount of LP token to burn in the withdrawal @return List of amounts of underlying coins that were withdrawn """use_lending:bool[N_COINS]=USE_LENDINGlp_token:address=self.lp_tokenamounts:uint256[N_COINS]=_underlying_amountsforiinrange(N_COINS):_amount:uint256=amounts[i]ifuse_lending[i]and_amount>0:rate:uint256=yERC20(self.coins[i]).getPricePerFullShare()amounts[i]=_amount*LENDING_PRECISION/rate# if not use_lending - all good already# Transfer max tokens in_lp_amount:uint256=ERC20(lp_token).balanceOf(msg.sender)if_lp_amount>_max_burn_amount:_lp_amount=_max_burn_amountassertERC20(lp_token).transferFrom(msg.sender,self,_lp_amount)Curve(self.curve).remove_liquidity_imbalance(amounts,_max_burn_amount)# Transfer unused LP tokens back_lp_amount=ERC20(lp_token).balanceOf(self)if_lp_amount!=0:assertERC20(lp_token).transfer(msg.sender,_lp_amount)# Unwrap and transfer all the coins we've gotreturnself._unwrap_and_transfer(msg.sender,empty(uint256[N_COINS]))
@external@nonreentrant('lock')defremove_liquidity_imbalance(amounts:uint256[N_COINS],max_burn_amount:uint256)->uint256:""" @notice Withdraw coins from the pool in an imbalanced amount @param amounts List of amounts of underlying coins to withdraw @param max_burn_amount Maximum amount of LP token to burn in the withdrawal @return Actual amount of the LP token burned in the withdrawal """assertnotself.is_killed# dev: is killedamp:uint256=self._A()old_balances:uint256[N_COINS]=self.balancesnew_balances:uint256[N_COINS]=old_balancesD0:uint256=self.get_D_mem(old_balances,amp)foriinrange(N_COINS):new_balances[i]-=amounts[i]D1:uint256=self.get_D_mem(new_balances,amp)_lp_token:address=self.lp_tokentoken_supply:uint256=ERC20(_lp_token).totalSupply()asserttoken_supply!=0# dev: zero total supply_fee:uint256=self.fee*N_COINS/(4*(N_COINS-1))_admin_fee:uint256=self.admin_feefees:uint256[N_COINS]=empty(uint256[N_COINS])foriinrange(N_COINS):ideal_balance:uint256=D1*old_balances[i]/D0difference:uint256=0ifideal_balance>new_balances[i]:difference=ideal_balance-new_balances[i]else:difference=new_balances[i]-ideal_balancefees[i]=_fee*difference/FEE_DENOMINATORself.balances[i]=new_balances[i]-(fees[i]*_admin_fee/FEE_DENOMINATOR)new_balances[i]-=fees[i]D2:uint256=self.get_D_mem(new_balances,amp)token_amount:uint256=(D0-D2)*token_supply/D0asserttoken_amount!=0# dev: zero tokens burnedtoken_amount+=1# In case of rounding errors - make it unfavorable for the "attacker"asserttoken_amount<=max_burn_amount,"Slippage screwed you"CurveToken(_lp_token).burnFrom(msg.sender,token_amount)# dev: insufficient fundsforiinrange(N_COINS):ifamounts[i]!=0:_response:Bytes[32]=raw_call(self.coins[i],concat(method_id("transfer(address,uint256)"),convert(msg.sender,bytes32),convert(amounts[i],bytes32),),max_outsize=32,)# dev: failed transferiflen(_response)>0:assertconvert(_response,bool)logRemoveLiquidityImbalance(msg.sender,amounts,fees,D1,token_supply-token_amount)returntoken_amount
Withdraw and unwrap a single coin from the pool. Returns amount of underlying coin received.
Input
Type
Description
_amount
uint256
Amount of LP tokens to burn in the withdrawal
i
int128
Index value of the coin to withdraw
_min_underlying_amount
uint256
Minimum amount of underlying coin to receive
Emits: Transfer RemoveLiquidityOne
Source code
@external@nonreentrant('lock')defremove_liquidity_one_coin(_amount:uint256,i:int128,_min_underlying_amount:uint256)->uint256:""" @notice Withdraw and unwrap a single coin from the pool @param _amount Amount of LP tokens to burn in the withdrawal @param i Index value of the coin to withdraw @param _min_underlying_amount Minimum amount of underlying coin to receive @return Amount of underlying coin received """assertERC20(self.lp_token).transferFrom(msg.sender,self,_amount)Curve(self.curve).remove_liquidity_one_coin(_amount,i,0)use_lending:bool[N_COINS]=USE_LENDINGifuse_lending[i]:coin:address=self.coins[i]_balance:uint256=ERC20(coin).balanceOf(self)yERC20(coin).withdraw(_balance)coin:address=self.underlying_coins[i]_balance:uint256=ERC20(coin).balanceOf(self)assert_balance>=_min_underlying_amount,"Not enough coins removed"_response:Bytes[32]=raw_call(coin,concat(method_id("transfer(address,uint256)"),convert(msg.sender,bytes32),convert(_balance,bytes32),),max_outsize=32,)iflen(_response)>0:assertconvert(_response,bool)return_balance
@external@nonreentrant('lock')defremove_liquidity_one_coin(_token_amount:uint256,i:int128,_min_amount:uint256)->uint256:""" @notice Withdraw a single coin from the pool @param _token_amount Amount of LP tokens to burn in the withdrawal @param i Index value of the coin to withdraw @param _min_amount Minimum amount of coin to receive @return Amount of coin received """assertnotself.is_killed# dev: is killeddy:uint256=0dy_fee:uint256=0total_supply:uint256=0dy,dy_fee,total_supply=self._calc_withdraw_one_coin(_token_amount,i)assertdy>=_min_amount,"Not enough coins removed"self.balances[i]-=(dy+dy_fee*self.admin_fee/FEE_DENOMINATOR)CurveToken(self.lp_token).burnFrom(msg.sender,_token_amount)# dev: insufficient funds_response:Bytes[32]=raw_call(self.coins[i],concat(method_id("transfer(address,uint256)"),convert(msg.sender,bytes32),convert(dy,bytes32),),max_outsize=32,)# dev: failed transferiflen(_response)>0:assertconvert(_response,bool)logRemoveLiquidityOne(msg.sender,_token_amount,dy,total_supply-_token_amount)returndy