PegKeepers are contracts that help stabilize the peg of crvUSD. Each Keeper is allocated a specific amount of crvUSD to secure the peg. The DAO decides this balance and can be raised or lowered by calling set_debt_ceiling() in the Factory.
The underlying actions of the PegKeepers can be divided into two actions, which get executed when calling update():
crvUSD price > 1: The PegKeeper mints and deposits crvUSD single-sidedly into the pool to which it is "linked", and receives LP tokens in exchange. This increases the balance of crvUSD in the pool and therefore decreases the price. It is important to note that the LP tokens are not staked in the gauge (if there is one). Thus, the PegKeeper does not receive CRV emissions.
crvUSD price < 1: If PegKeepers hold a balance of the corresponding LP token, they can single-sidedly withdraw crvUSD from the liquidity pool and burn it. This action reduces the supply of crvUSD in the pool and should subsequently increase its price.
Note
PegKeepers do not actually mint or burn crvUSD tokens. They have a defined allocated balance of crvUSD tokens that they can use for deposits. It is important to note that PegKeepers cannot do anything else apart from depositing and withdrawing. Therefore, crvUSD token balances of the PegKeepers that are not deposited into a pool may not be counted as circulating supply, although technically they are.
Contract Source & Deployment
Source code for this contract is available on Github.
The most important function in the PegKeeper contract is the update() function. When invoked, the PegKeeper either mints and single-sidedly deposits crvUSD into the StableSwap pool, or it withdraws crvUSD from the pool by redeeming the LP tokens received from previous deposits.
Deposit and Mint: This mechanism is triggered when the price of crvUSD > 1. Minting and depositing into the pool will increase the crvUSD supply and decrease its price. The LP tokens that the PegKeeper receives when depositing crvUSD into the pool are not staked in the gauge (if the pool has one), which means the PegKeeper does not receive CRV inflation rewards.
Withdraw and Burn: This mechanism is triggered when the price of crvUSD < 1. By withdrawing crvUSD from the pool, the supply of crvUSD decreases, which increases its price.
PegKeepers have unlimited approval for the liquidity pool, allowing them to deposit and withdraw crvUSD.
Function to either mint and deposit or withdraw and burn based on the balances within the pools. A share (caller_share) of the generated profit will be awarded to the function's caller. By default, this is set to msg.sender, but there is also the possibility to input a _beneficiary address to which the rewards will be sent.
Returns: caller profit (uint256).
Emits: Provide or Withdraw
Note
There is an ACTION_DELAY of 15 minutes before calling the function again.
Source code: Mint and Deposit
eventProvide:amount:uint256@external@nonpayabledefupdate(_beneficiary:address=msg.sender)->uint256:""" @notice Provide or withdraw coins from the pool to stabilize it @param _beneficiary Beneficiary address @return Amount of profit received by beneficiary """ifself.last_change+ACTION_DELAY>block.timestamp:return0balance_pegged:uint256=POOL.balances(I)balance_peg:uint256=POOL.balances(1-I)*PEG_MULinitial_profit:uint256=self._calc_profit()p_agg:uint256=AGGREGATOR.price()# Current USD per stablecoin# Checking the balance will ensure no-loss of the stabilizer, but to ensure stabilization# we need to exclude "bad" p_agg, so we add an extra check for itifbalance_peg>balance_pegged:assertp_agg>=10**18self._provide((balance_peg-balance_pegged)/5)# this dumps stablecoinelse:assertp_agg<=10**18self._withdraw((balance_pegged-balance_peg)/5)# this pumps stablecoin# Send generated profitnew_profit:uint256=self._calc_profit()assertnew_profit>=initial_profit,"peg unprofitable"lp_amount:uint256=new_profit-initial_profitcaller_profit:uint256=lp_amount*self.caller_share/SHARE_PRECISIONifcaller_profit>0:POOL.transfer(_beneficiary,caller_profit)returncaller_profit@internaldef_provide(_amount:uint256):# We already have all reserves here# ERC20(PEGGED).mint(self, _amount)if_amount==0:returnamounts:uint256[2]=empty(uint256[2])amounts[I]=_amountPOOL.add_liquidity(amounts,0)self.last_change=block.timestampself.debt+=_amountlogProvide(_amount)
Source code: Withdraw and Burn
eventWithdraw:amount:uint256@external@nonpayabledefupdate(_beneficiary:address=msg.sender)->uint256:""" @notice Provide or withdraw coins from the pool to stabilize it @param _beneficiary Beneficiary address @return Amount of profit received by beneficiary """ifself.last_change+ACTION_DELAY>block.timestamp:return0balance_pegged:uint256=POOL.balances(I)balance_peg:uint256=POOL.balances(1-I)*PEG_MULinitial_profit:uint256=self._calc_profit()p_agg:uint256=AGGREGATOR.price()# Current USD per stablecoin# Checking the balance will ensure no-loss of the stabilizer, but to ensure stabilization# we need to exclude "bad" p_agg, so we add an extra check for itifbalance_peg>balance_pegged:assertp_agg>=10**18self._provide((balance_peg-balance_pegged)/5)# this dumps stablecoinelse:assertp_agg<=10**18self._withdraw((balance_pegged-balance_peg)/5)# this pumps stablecoin# Send generated profitnew_profit:uint256=self._calc_profit()assertnew_profit>=initial_profit,"peg unprofitable"lp_amount:uint256=new_profit-initial_profitcaller_profit:uint256=lp_amount*self.caller_share/SHARE_PRECISIONifcaller_profit>0:POOL.transfer(_beneficiary,caller_profit)returncaller_profit@internaldef_withdraw(_amount:uint256):if_amount==0:returndebt:uint256=self.debtamount:uint256=min(_amount,debt)amounts:uint256[2]=empty(uint256[2])amounts[I]=amountPOOL.remove_liquidity_imbalance(amounts,max_value(uint256))self.last_change=block.timestampself.debt-=amountlogWithdraw(amount)
Function which retrieves the timestamp of when the balances of the PegKeeper were last altered. This variable is updated each time update() (_provide or _withdraw) is called. This variable is of importance for update(), as there is a mandatory delay of 15 * 60 seconds before the function can be called again.
Function to estimate the profit from calling update(). The caller of the function will receive 20% of the total profits.
Returns: expected amount of profit going to the caller (uint256).
Warning
Please note that this method provides an estimate and may not reflect the precise profit. The actual profit tends to be higher due to the increasing virtual price of the LP token.
Source code
ACTION_DELAY:constant(uint256)=15*60@external@viewdefestimate_caller_profit()->uint256:""" @notice Estimate profit from calling update() @dev This method is not precise, real profit is always more because of increasing virtual price @return Expected amount of profit going to beneficiary """ifself.last_change+ACTION_DELAY>block.timestamp:return0balance_pegged:uint256=POOL.balances(I)balance_peg:uint256=POOL.balances(1-I)*PEG_MULinitial_profit:uint256=self._calc_profit()p_agg:uint256=AGGREGATOR.price()# Current USD per stablecoin# Checking the balance will ensure no-loss of the stabilizer, but to ensure stabilization# we need to exclude "bad" p_agg, so we add an extra check for itnew_profit:uint256=0ifbalance_peg>balance_pegged:ifp_agg<10**18:return0new_profit=self._calc_future_profit((balance_peg-balance_pegged)/5,True)# this dumps stablecoinelse:ifp_agg>10**18:return0new_profit=self._calc_future_profit((balance_pegged-balance_peg)/5,False)# this pumps stablecoinifnew_profit<initial_profit:return0lp_amount:uint256=new_profit-initial_profitreturnlp_amount*self.caller_share/SHARE_PRECISION
Getter for the caller share which is the share of the profit generated when calling the update() function. The share is intended to incentivize the call of the function. The precision of the variable is set to .
Returns: caller share (uint256).
Source code
SHARE_PRECISION:constant(uint256)=10**5caller_share:public(uint256)@externaldef__init__(_pool:CurvePool,_index:uint256,_receiver:address,_caller_share:uint256,_factory:address,_aggregator:StableAggregator,_admin:address):""" @notice Contract constructor @param _pool Contract pool address @param _index Index of the pegged @param _receiver Receiver of the profit @param _caller_share Caller's share of profit @param _factory Factory which should be able to take coins away @param _aggregator Price aggregator which shows the price of pegged in real "dollars" @param _admin Admin account """assert_index<2POOL=_poolI=_indexpegged:address=_pool.coins(_index)PEGGED=peggedERC20(pegged).approve(_pool.address,max_value(uint256))ERC20(pegged).approve(_factory,max_value(uint256))PEG_MUL=10**(18-ERC20(_pool.coins(1-_index)).decimals())self.admin=_adminassert_receiver!=empty(address)self.receiver=_receiverlogApplyNewAdmin(msg.sender)logApplyNewReceiver(_receiver)assert_caller_share<=SHARE_PRECISION# dev: bad part valueself.caller_share=_caller_sharelogSetNewCallerShare(_caller_share)FACTORY=_factoryAGGREGATOR=_aggregatorIS_INVERSE=(_index==0)
This function is only callable by the admin of the contract.
Function to set the caller share to _new_caller_share.
Emits: SetNewCallerShare
Input
Type
Description
_new_caller_share
uint256
New caller share
Source code
eventSetNewCallerShare:caller_share:uint256SHARE_PRECISION:constant(uint256)=10**5caller_share:public(uint256)@external@nonpayabledefset_new_caller_share(_new_caller_share:uint256):""" @notice Set new update caller's part @param _new_caller_share Part with SHARE_PRECISION """assertmsg.sender==self.admin# dev: only adminassert_new_caller_share<=SHARE_PRECISION# dev: bad part valueself.caller_share=_new_caller_sharelogSetNewCallerShare(_new_caller_share)
Function to withdraw the profit generated by the PegKeeper.
Returns: amount of LP tokens (uint256).
Emits: Profit
Source code
eventProfit:lp_amount:uint256@external@nonpayabledefwithdraw_profit()->uint256:""" @notice Withdraw profit generated by Peg Keeper @return Amount of LP Token received """lp_amount:uint256=self._calc_profit()POOL.transfer(self.receiver,lp_amount)logProfit(lp_amount)returnlp_amount
PegKeepers have an admin and a receiver. Both of these variables can be changed by calling the respective admin-guarded functions, but such changes must first be approved by a DAO vote. After approval, the newly designated admin or receiver is required to apply these changes within a timeframe of 3 * 86400 seconds, which equates to a timespan of three days. Should there be an attempt to implement these changes after this period, the function will revert.
admin:public(address)@externaldef__init__(_pool:CurvePool,_index:uint256,_receiver:address,_caller_share:uint256,_factory:address,_aggregator:StableAggregator,_admin:address):""" @notice Contract constructor @param _pool Contract pool address @param _index Index of the pegged @param _receiver Receiver of the profit @param _caller_share Caller's share of profit @param _factory Factory which should be able to take coins away @param _aggregator Price aggregator which shows the price of pegged in real "dollars" @param _admin Admin account """...self.admin=_admin...
This function is only callable by the admin of the contract.
Function to commit a new admin.
Emits: CommitNewAdmin
Input
Type
Description
_new_admin
address
new admin address
Source code
eventCommitNewAdmin:admin:address@external@nonpayabledefcommit_new_admin(_new_admin:address):""" @notice Commit new admin of the Peg Keeper @param _new_admin Address of the new admin """assertmsg.sender==self.admin# dev: only adminassertself.new_admin_deadline==0# dev: active actiondeadline:uint256=block.timestamp+ADMIN_ACTIONS_DELAYself.new_admin_deadline=deadlineself.future_admin=_new_adminlogCommitNewAdmin(_new_admin)
This function is only callable by the future_admin of the contract.
Function to apply the new admin of the PegKeeper.
Emits: ApplyNewAdmin
Source code
eventApplyNewAdmin:admin:address@external@nonpayabledefapply_new_admin():""" @notice Apply new admin of the Peg Keeper @dev Should be executed from new admin """new_admin:address=self.future_adminassertmsg.sender==new_admin# dev: only new adminassertblock.timestamp>=self.new_admin_deadline# dev: insufficient timeassertself.new_admin_deadline!=0# dev: no active actionself.admin=new_adminself.new_admin_deadline=0logApplyNewAdmin(new_admin)
Getter for the timestamp indicating the deadline by which the future_admin can apply the admin change. Once the deadline is over, the address will no longer be able to apply the changes. The deadline is set for a timeperiod of three days.
Getter for the receiver of the PegKeeper's profits.
Returns: receiver (address).
Source code
receiver:public(address)@externaldef__init__(_pool:CurvePool,_index:uint256,_receiver:address,_caller_share:uint256,_factory:address,_aggregator:StableAggregator,_admin:address):""" @notice Contract constructor @param _pool Contract pool address @param _index Index of the pegged @param _receiver Receiver of the profit @param _caller_share Caller's share of profit @param _factory Factory which should be able to take coins away @param _aggregator Price aggregator which shows the price pegged in real "dollars" @param _admin Admin account """...assert_receiver!=empty(address)self.receiver=_receiver...
This function is only callable by the admin of the contract.
Function to commit a new receiver address.
Emits: CommitNewReceiver
Input
Type
Description
_new_receiver
address
new receiver address
Source code
eventCommitNewReceiver:receiver:address@external@nonpayabledefcommit_new_receiver(_new_receiver:address):""" @notice Commit new receiver of profit @param _new_receiver Address of the new receiver """assertmsg.sender==self.admin# dev: only adminassertself.new_receiver_deadline==0# dev: active actiondeadline:uint256=block.timestamp+ADMIN_ACTIONS_DELAYself.new_receiver_deadline=deadlineself.future_receiver=_new_receiverlogCommitNewReceiver(_new_receiver)
Function to apply the new receiver address of the PegKeeper's profit.
Emits: ApplyNewReceiver
Source code
eventApplyNewReceiver:receiver:address@external@nonpayabledefapply_new_receiver():""" @notice Apply new receiver of profit """assertblock.timestamp>=self.new_receiver_deadline# dev: insufficient timeassertself.new_receiver_deadline!=0# dev: no active actionnew_receiver:address=self.future_receiverself.receiver=new_receiverself.new_receiver_deadline=0logApplyNewReceiver(new_receiver)
Getter for the timestamp indicating the deadline by which the future_receiver can apply the receiver change. Once the deadline is over, the address will no longer be able to apply the changes. The deadline is set for a timeperiod of three days.
This function is only callable by the admin of the contract.
Function to revert admin or receiver changes. Calling this function sets the admin and receiver deadline back to 0 and emits ApplyNewAdmin and ApplyNewReceiver events to revert the changes.
Emits: ApplyNewAdmin and ApplyNewReceiver
Source code
eventApplyNewReceiver:receiver:addresseventApplyNewAdmin:admin:address@external@nonpayabledefrevert_new_options():""" @notice Revert new admin of the Peg Keeper or new receiver @dev Should be executed from admin """assertmsg.sender==self.admin# dev: only adminself.new_admin_deadline=0self.new_receiver_deadline=0logApplyNewAdmin(self.admin)logApplyNewReceiver(self.receiver)
Getter for the crvUSD debt of the PegKeeper. When the PegKeeper deposits crvUSD into the pool, the debt is incremented by the deposited amount. Conversely, if the PegKeeper withdraws, the debt is reduced by the withdrawn amount. debt is used to calculate the DebtFraction of the PegKeepers.
FACTORY:immutable(address)@externaldef__init__(_pool:CurvePool,_index:uint256,_receiver:address,_caller_share:uint256,_factory:address,_aggregator:StableAggregator,_admin:address):""" @notice Contract constructor @param _pool Contract pool address @param _index Index of the pegged @param _receiver Receiver of the profit @param _caller_share Caller's share of profit @param _factory Factory which should be able to take coins away @param _aggregator Price aggregator which shows the price of pegged in real "dollars" @param _admin Admin account """...FACTORY=_factory...
Getter for the address of the pegged token (crvUSD). Pegged asset is determined by the index of the token in the corresponding pool. Index value is stored in I.
Returns: pegged token contract (address).
Source code
PEGGED:immutable(address)@externaldef__init__(_pool:CurvePool,_index:uint256,_receiver:address,_caller_share:uint256,_factory:address,_aggregator:StableAggregator,_admin:address):""" @notice Contract constructor @param _pool Contract pool address @param _index Index of the pegged @param _receiver Receiver of the profit @param _caller_share Caller's share of profit @param _factory Factory which should be able to take coins away @param _aggregator Price aggregator which shows the price of pegged in real "dollars" @param _admin Admin account """...PEGGED=pegged...
Getter for the pool contract address in which the PegKeeper deposits and withdraws.
Returns: pool contract (address).
Source code
POOL:immutable(CurvePool)@externaldef__init__(_pool:CurvePool,_index:uint256,_receiver:address,_caller_share:uint256,_factory:address,_aggregator:StableAggregator,_admin:address):""" @notice Contract constructor @param _pool Contract pool address @param _index Index of the pegged @param _receiver Receiver of the profit @param _caller_share Caller's share of profit @param _factory Factory which should be able to take coins away @param _aggregator Price aggregator which shows the price of pegged in real "dollars" @param _admin Admin account """...POOL=_pool...
Getter for the price aggregator contract for crvUSD. This contract is used to determine the value of crvUSD.
Returns: price aggregator contract (address).
Source code
AGGREGATOR:immutable(StableAggregator)@externaldef__init__(_pool:CurvePool,_index:uint256,_receiver:address,_caller_share:uint256,_factory:address,_aggregator:StableAggregator,_admin:address):""" @notice Contract constructor @param _pool Contract pool address @param _index Index of the pegged @param _receiver Receiver of the profit @param _caller_share Caller's share of profit @param _factory Factory which should be able to take coins away @param _aggregator Price aggregator which shows the price of pegged in real "dollars" @param _admin Admin account """...AGGREGATOR=_aggregator...