Fees are distributed on a weekly basis. The porportional amount of fees that each user is to receive is calculated based on their veCRV balance relative to the total veCRV supply.This amount is calculated at the start of the week. The actual distribution occurs at the end of the week based on the fees that were collected. As such, a user that creates a new vote-lock should expect to receive their first fee payout at the end of the following epoch week.
Changing the Reward Token
Changing the reward token from 3CRV to, for example, crvUSD, would require the creation of a new FeeDistributor, as the reward token cannot be configured within the existing contract.
The available 3CRV balance to distribute is tracked via the “token checkpoint”. This is updated at minimum every 24 hours. Fees that are received between the last checkpoint of the previous week and first checkpoint of the new week will be split evenly between the weeks.
The token checkpoint tracks the balance of 3CRV within the distributor to determine the amount of fees to distribute in the given week. The checkpoint can be updated at most once every 24 hours. Fees that are received between the last checkpoint of the previous week and first checkpoint of the new week will be split evenly between the weeks.
To ensure full distribution of fees in the current week, the burn process must be completed prior to the last checkpoint within the week.
A token checkpoint is automatically taken during any claim action, if the last checkpoint is more than 24 hours old.
Source code
eventCheckpointToken:time:uint256tokens:uint256can_checkpoint_token:public(bool)@internaldef_checkpoint_token():token_balance:uint256=ERC20(self.token).balanceOf(self)to_distribute:uint256=token_balance-self.token_last_balanceself.token_last_balance=token_balancet:uint256=self.last_token_timesince_last:uint256=block.timestamp-tself.last_token_time=block.timestampthis_week:uint256=t/WEEK*WEEKnext_week:uint256=0foriinrange(20):next_week=this_week+WEEKifblock.timestamp<next_week:ifsince_last==0andblock.timestamp==t:self.tokens_per_week[this_week]+=to_distributeelse:self.tokens_per_week[this_week]+=to_distribute*(block.timestamp-t)/since_lastbreakelse:ifsince_last==0andnext_week==t:self.tokens_per_week[this_week]+=to_distributeelse:self.tokens_per_week[this_week]+=to_distribute*(next_week-t)/since_lastt=next_weekthis_week=next_weeklogCheckpointToken(block.timestamp,to_distribute)@externaldefcheckpoint_token():""" @notice Update the token checkpoint @dev Calculates the total number of tokens to be distributed in a given week. During setup for the initial distribution this function is only callable by the contract owner. Beyond initial distro, it can be enabled for anyone to call. """assert(msg.sender==self.admin)or\
(self.can_checkpoint_tokenand(block.timestamp>self.last_token_time+TOKEN_CHECKPOINT_DEADLINE))self._checkpoint_token()
Function to perform multiple claims in a single call.
This is useful to claim for multiple accounts at once, or for making many claims against the same account if that account has performed more than 50 veCRV related actions.
Returns: true (boolean).
Input
Type
Description
_recievers
address
receiver address
Source code
@internaldef_checkpoint_total_supply():ve:address=self.voting_escrowt:uint256=self.time_cursorrounded_timestamp:uint256=block.timestamp/WEEK*WEEKVotingEscrow(ve).checkpoint()foriinrange(20):ift>rounded_timestamp:breakelse:epoch:uint256=self._find_timestamp_epoch(ve,t)pt:Point=VotingEscrow(ve).point_history(epoch)dt:int128=0ift>pt.ts:# If the point is at 0 epoch, it can actually be earlier than the first deposit# Then make dt 0dt=convert(t-pt.ts,int128)self.ve_supply[t]=convert(max(pt.bias-pt.slope*dt,0),uint256)t+=WEEKself.time_cursor=t@externaldefcheckpoint_total_supply():""" @notice Update the veCRV total supply checkpoint @dev The checkpoint is also updated by the first claimant each new epoch week. This function may be called independently of a claim, to reduce claiming gas costs. """self._checkpoint_total_supply()
Returns: amount of 3CRV (uint256) received in the claim.
Note
For off-chain integrators, this function can be called as though it were a view method in order to check the claimable amount.
Every veCRV related action (locking, extending a lock, increasing the locktime) increments a user’s veCRV epoch. A call to claim will consider at most 50 user epochs. For accounts that performed many veCRV actions, it may be required to call claim more than once to receive the fees. In such cases it can be more efficient to use claim_many.
Input
Type
Description
_addr
address
receiver address
Source code
eventClaimed:recipient:indexed(address)amount:uint256claim_epoch:uint256max_epoch:uint256@internaldef_claim(addr:address,ve:address,_last_token_time:uint256)->uint256:# Minimal user_epoch is 0 (if user had no point)user_epoch:uint256=0to_distribute:uint256=0max_user_epoch:uint256=VotingEscrow(ve).user_point_epoch(addr)_start_time:uint256=self.start_timeifmax_user_epoch==0:# No lock = no feesreturn0week_cursor:uint256=self.time_cursor_of[addr]ifweek_cursor==0:# Need to do the initial binary searchuser_epoch=self._find_timestamp_user_epoch(ve,addr,_start_time,max_user_epoch)else:user_epoch=self.user_epoch_of[addr]ifuser_epoch==0:user_epoch=1user_point:Point=VotingEscrow(ve).user_point_history(addr,user_epoch)ifweek_cursor==0:week_cursor=(user_point.ts+WEEK-1)/WEEK*WEEKifweek_cursor>=_last_token_time:return0ifweek_cursor<_start_time:week_cursor=_start_timeold_user_point:Point=empty(Point)# Iterate over weeksforiinrange(50):ifweek_cursor>=_last_token_time:breakifweek_cursor>=user_point.tsanduser_epoch<=max_user_epoch:user_epoch+=1old_user_point=user_pointifuser_epoch>max_user_epoch:user_point=empty(Point)else:user_point=VotingEscrow(ve).user_point_history(addr,user_epoch)else:# Calc# + i * 2 is for rounding errorsdt:int128=convert(week_cursor-old_user_point.ts,int128)balance_of:uint256=convert(max(old_user_point.bias-dt*old_user_point.slope,0),uint256)ifbalance_of==0anduser_epoch>max_user_epoch:breakifbalance_of>0:to_distribute+=balance_of*self.tokens_per_week[week_cursor]/self.ve_supply[week_cursor]week_cursor+=WEEKuser_epoch=min(max_user_epoch,user_epoch-1)self.user_epoch_of[addr]=user_epochself.time_cursor_of[addr]=week_cursorlogClaimed(addr,to_distribute,user_epoch,max_user_epoch)returnto_distribute@external@nonreentrant('lock')defclaim(_addr:address=msg.sender)->uint256:""" @notice Claim fees for `_addr` @dev Each call to claim look at a maximum of 50 user veCRV points. For accounts with many veCRV related actions, this function may need to be called more than once to claim all available fees. In the `Claimed` event that fires, if `claim_epoch` is less than `max_epoch`, the account may claim again. @param _addr Address to claim fees for @return uint256 Amount of fees claimed in the call """assertnotself.is_killedifblock.timestamp>=self.time_cursor:self._checkpoint_total_supply()last_token_time:uint256=self.last_token_timeifself.can_checkpoint_tokenand(block.timestamp>last_token_time+TOKEN_CHECKPOINT_DEADLINE):self._checkpoint_token()last_token_time=block.timestamplast_token_time=last_token_time/WEEK*WEEKamount:uint256=self._claim(_addr,self.voting_escrow,last_token_time)ifamount!=0:token:address=self.tokenassertERC20(token).transfer(_addr,amount)self.token_last_balance-=amountreturnamount
Function to perform multiple claims in a single call.
This is useful to claim for multiple accounts at once, or for making many claims against the same account if that account has performed more than 50 veCRV related actions.
Returns: true (boolean).
Input
Type
Description
_recievers
address
receiver address
Source code
eventClaimed:recipient:indexed(address)amount:uint256claim_epoch:uint256max_epoch:uint256@internaldef_claim(addr:address,ve:address,_last_token_time:uint256)->uint256:# Minimal user_epoch is 0 (if user had no point)user_epoch:uint256=0to_distribute:uint256=0max_user_epoch:uint256=VotingEscrow(ve).user_point_epoch(addr)_start_time:uint256=self.start_timeifmax_user_epoch==0:# No lock = no feesreturn0week_cursor:uint256=self.time_cursor_of[addr]ifweek_cursor==0:# Need to do the initial binary searchuser_epoch=self._find_timestamp_user_epoch(ve,addr,_start_time,max_user_epoch)else:user_epoch=self.user_epoch_of[addr]ifuser_epoch==0:user_epoch=1user_point:Point=VotingEscrow(ve).user_point_history(addr,user_epoch)ifweek_cursor==0:week_cursor=(user_point.ts+WEEK-1)/WEEK*WEEKifweek_cursor>=_last_token_time:return0ifweek_cursor<_start_time:week_cursor=_start_timeold_user_point:Point=empty(Point)# Iterate over weeksforiinrange(50):ifweek_cursor>=_last_token_time:breakifweek_cursor>=user_point.tsanduser_epoch<=max_user_epoch:user_epoch+=1old_user_point=user_pointifuser_epoch>max_user_epoch:user_point=empty(Point)else:user_point=VotingEscrow(ve).user_point_history(addr,user_epoch)else:# Calc# + i * 2 is for rounding errorsdt:int128=convert(week_cursor-old_user_point.ts,int128)balance_of:uint256=convert(max(old_user_point.bias-dt*old_user_point.slope,0),uint256)ifbalance_of==0anduser_epoch>max_user_epoch:breakifbalance_of>0:to_distribute+=balance_of*self.tokens_per_week[week_cursor]/self.ve_supply[week_cursor]week_cursor+=WEEKuser_epoch=min(max_user_epoch,user_epoch-1)self.user_epoch_of[addr]=user_epochself.time_cursor_of[addr]=week_cursorlogClaimed(addr,to_distribute,user_epoch,max_user_epoch)returnto_distribute@external@nonreentrant('lock')defclaim_many(_receivers:address[20])->bool:""" @notice Make multiple fee claims in a single call @dev Used to claim for many accounts at once, or to make multiple claims for the same address when that address has significant veCRV history @param _receivers List of addresses to claim for. Claiming terminates at the first `ZERO_ADDRESS`. @return bool success """assertnotself.is_killedifblock.timestamp>=self.time_cursor:self._checkpoint_total_supply()last_token_time:uint256=self.last_token_timeifself.can_checkpoint_tokenand(block.timestamp>last_token_time+TOKEN_CHECKPOINT_DEADLINE):self._checkpoint_token()last_token_time=block.timestamplast_token_time=last_token_time/WEEK*WEEKvoting_escrow:address=self.voting_escrowtoken:address=self.tokentotal:uint256=0foraddrin_receivers:ifaddr==ZERO_ADDRESS:breakamount:uint256=self._claim(addr,voting_escrow,last_token_time)ifamount!=0:assertERC20(token).transfer(addr,amount)total+=amountiftotal!=0:self.token_last_balance-=totalreturnTrue
Function to receive 3CRV into the contract and trigger a token checkpoint.
Input
Type
Description
_recievers
address
receiver address
Source code
@externaldefburn(_coin:address)->bool:""" @notice Receive 3CRV into the contract and trigger a token checkpoint @param _coin Address of the coin being received (must be 3CRV) @return bool success """assert_coin==self.tokenassertnotself.is_killedamount:uint256=ERC20(_coin).balanceOf(msg.sender)ifamount!=0:ERC20(_coin).transferFrom(msg.sender,self,amount)ifself.can_checkpoint_tokenand(block.timestamp>self.last_token_time+TOKEN_CHECKPOINT_DEADLINE):self._checkpoint_token()returnTrue
The FeeDistributor contract can be killed. Doing so transfers the entire 3CRV balance to the emergency_return address and blocks the ability to claim or burn. The contract cannot be unkilled.
Killing transfers the entire 3CRV balance to the emergency_return address and blocks the ability to claim or burn. The contract cannot be unkilled.
Guarded Method
This function is only callable by the admin of the contract.
Function to kill the fee distributor contract.
Source code
is_killed:public(bool)@externaldefkill_me():""" @notice Kill the contract @dev Killing transfers the entire 3CRV balance to the emergency return address and blocks the ability to claim or burn. The contract cannot be unkilled. """assertmsg.sender==self.adminself.is_killed=Truetoken:address=self.tokenassertERC20(token).transfer(self.emergency_return,ERC20(token).balanceOf(self))
Function to recover ERC20 tokens from the contract. Tokens are sent to the emergency return address.
Returns: true (bool).
Input
Type
Description
_coin
address
Coin Address to recover
Source code
@externaldefrecover_balance(_coin:address)->bool:""" @notice Recover ERC20 tokens from this contract @dev Tokens are sent to the emergency return address. @param _coin Token address @return bool success """assertmsg.sender==self.adminassert_coin!=self.tokenamount:uint256=ERC20(_coin).balanceOf(self)response:Bytes[32]=raw_call(_coin,concat(method_id("transfer(address,uint256)"),convert(self.emergency_return,bytes32),convert(amount,bytes32),),max_outsize=32,)iflen(response)!=0:assertconvert(response,bool)returnTrue
This function is only callable by the admin of the contract.
Function to commit transfer of the ownership.
Input
Type
Description
_addr
address
Address to transfer ownership to
Source code
eventCommitAdmin:admin:addressadmin:public(address)future_admin:public(address)@externaldefcommit_admin(_addr:address):""" @notice Commit transfer of ownership @param _addr New admin address """assertmsg.sender==self.admin# dev: access deniedself.future_admin=_addrlogCommitAdmin(_addr)
This function is only callable by the admin of the contract.
Function to apply transfer of the ownership.
Input
Type
Description
_addr
address
Address to transfer ownership to
Source code
eventApplyAdmin:admin:addressadmin:public(address)future_admin:public(address)@externaldefapply_admin():""" @notice Apply transfer of ownership """assertmsg.sender==self.adminassertself.future_admin!=ZERO_ADDRESSfuture_admin:address=self.future_adminself.admin=future_adminlogApplyAdmin(future_admin)
Getter for the veCRV balance for _user at _timestamp.
Returns: veCRV balance (uint256).
Input
Type
Description
_user
address
address to query balance for
_timestamp
uint256
epoch time
Source code
@view@externaldefve_for_at(_user:address,_timestamp:uint256)->uint256:""" @notice Get the veCRV balance for `_user` at `_timestamp` @param _user Address to query balance for @param _timestamp Epoch time @return uint256 veCRV balance """ve:address=self.voting_escrowmax_user_epoch:uint256=VotingEscrow(ve).user_point_epoch(_user)epoch:uint256=self._find_timestamp_user_epoch(ve,_user,_timestamp,max_user_epoch)pt:Point=VotingEscrow(ve).user_point_history(_user,epoch)returnconvert(max(pt.bias-pt.slope*convert(_timestamp-pt.ts,int128),0),uint256)
Getter for the epoch time for fee distribution to start.
Returns: epoch time (uint256).
Input
Type
Description
_user
address
address to query balance for
_timestamp
uint256
epoch time
Source code
start_time:public(uint256)@externaldef__init__(_voting_escrow:address,_start_time:uint256,_token:address,_admin:address,_emergency_return:address):""" @notice Contract constructor @param _voting_escrow VotingEscrow contract address @param _start_time Epoch time for fee distribution to start @param _token Fee token address (3CRV) @param _admin Admin address @param _emergency_return Address to transfer `_token` balance to if this contract is killed """t:uint256=_start_time/WEEK*WEEKself.start_time=tself.last_token_time=tself.time_cursor=tself.token=_tokenself.voting_escrow=_voting_escrowself.admin=_adminself.emergency_return=_emergency_return
voting_escrow:public(address)@externaldef__init__(_voting_escrow:address,_start_time:uint256,_token:address,_admin:address,_emergency_return:address):""" @notice Contract constructor @param _voting_escrow VotingEscrow contract address @param _start_time Epoch time for fee distribution to start @param _token Fee token address (3CRV) @param _admin Admin address @param _emergency_return Address to transfer `_token` balance to if this contract is killed """t:uint256=_start_time/WEEK*WEEKself.start_time=tself.last_token_time=tself.time_cursor=tself.token=_tokenself.voting_escrow=_voting_escrowself.admin=_adminself.emergency_return=_emergency_return
token:public(address)@externaldef__init__(_voting_escrow:address,_start_time:uint256,_token:address,_admin:address,_emergency_return:address):""" @notice Contract constructor @param _voting_escrow VotingEscrow contract address @param _start_time Epoch time for fee distribution to start @param _token Fee token address (3CRV) @param _admin Admin address @param _emergency_return Address to transfer `_token` balance to if this contract is killed """t:uint256=_start_time/WEEK*WEEKself.start_time=tself.last_token_time=tself.time_cursor=tself.token=_tokenself.voting_escrow=_voting_escrowself.admin=_adminself.emergency_return=_emergency_return
This function is only callable by the admin of the contract.
Funtion to toggle permission for checkpointing by an account.
Source code
eventToggleAllowCheckpointToken:toggle_flag:bool@externaldeftoggle_allow_checkpoint_token():""" @notice Toggle permission for checkpointing by any account """assertmsg.sender==self.adminflag:bool=notself.can_checkpoint_tokenself.can_checkpoint_token=flaglogToggleAllowCheckpointToken(flag)