admin:public(address)@externaldef__init__(_name:String[64],_symbol:String[32],_decimals:uint256):""" @notice Contract constructor @param _name Token full name @param _symbol Token symbol @param _decimals Number of decimals for token """...self.admin=msg.sender...
Getter for the name of the token. Name of the token can be changed by calling the set_name function.
Returns: token name (String[64]).
Source code
name:public(String[64])@externaldef__init__(_name:String[64],_symbol:String[32],_decimals:uint256):""" @notice Contract constructor @param _name Token full name @param _symbol Token symbol @param _decimals Number of decimals for token """init_supply:uint256=INITIAL_SUPPLY*10**_decimalsself.name=_name...
Getter of the token symbol. Symbol of the token can be changed by calling the set_name function.
Returns: token symbol (String[32]).
Source code
symbol:public(String[32])@externaldef__init__(_name:String[64],_symbol:String[32],_decimals:uint256):""" @notice Contract constructor @param _name Token full name @param _symbol Token symbol @param _decimals Number of decimals for token """...self.symbol=_symbol...
Getter for the current number of CRV tokens - claimed of unclaimed - in existence.
Returns: currently existing tokens (uint256).
Source code
@internal@viewdef_available_supply()->uint256:returnself.start_epoch_supply+(block.timestamp-self.start_epoch_time)*self.rate@external@viewdefavailable_supply()->uint256:""" @notice Current number of tokens in existence (claimed or unclaimed) """returnself._available_supply()
decimals:public(uint256)@externaldef__init__(_name:String[64],_symbol:String[32],_decimals:uint256):""" @notice Contract constructor @param _name Token full name @param _symbol Token symbol @param _decimals Number of decimals for token """...self.decimals=_decimals...
Function to transfer _value tokens from msg.sender to _to. Transfers to ZERO_ADDRESS are not allowed.
Returns: true (bool).
Emits: Transfer
Input
Type
Description
_to
address
receiver address
_value
uint256
amount of tokens to send
Warning
Vyper does not allow underflows, so the subtraction in this function will revert on an insufficient balance.
Source code
eventTransfer:_from:indexed(address)_to:indexed(address)_value:uint256@externaldeftransfer(_to:address,_value:uint256)->bool:""" @notice Transfer `_value` tokens from `msg.sender` to `_to` @dev Vyper does not allow underflows, so the subtraction in this function will revert on an insufficient balance @param _to The address to transfer to @param _value The amount to be transferred @return bool success """assert_to!=ZERO_ADDRESS# dev: transfers to 0x0 are not allowedself.balanceOf[msg.sender]-=_valueself.balanceOf[_to]+=_valuelogTransfer(msg.sender,_to,_value)returnTrue
Function to transfer _value tokens from _from_ to _to. Transfers to ZERO_ADDRESS are not allowed.
Returns: true (bool).
Emits: Transfer
Input
Type
Description
_from
address
address to send tokens from
_to
address
receiver address
_value
uint256
amount of tokens to send
Warning
Vyper does not allow underflows, so the subtraction in this function will revert on an insufficient balance.
Source code
eventTransfer:_from:indexed(address)_to:indexed(address)_value:uint256@externaldeftransferFrom(_from:address,_to:address,_value:uint256)->bool:""" @notice Transfer `_value` tokens from `_from` to `_to` @param _from address The address which you want to send tokens from @param _to address The address which you want to transfer to @param _value uint256 the amount of tokens to be transferred @return bool success """assert_to!=ZERO_ADDRESS# dev: transfers to 0x0 are not allowed# NOTE: vyper does not allow underflows# so the following subtraction would revert on insufficient balanceself.balanceOf[_from]-=_valueself.balanceOf[_to]+=_valueself.allowances[_from][msg.sender]-=_valuelogTransfer(_from,_to,_value)returnTrue
Getter method to check the amount of tokens that an owner allowed to a spender.
Returns: amount of tokens (uint256) _owner is allowed to _spender.
Input
Type
Description
_owner
address
owner address
_spender
address
spender address
Source code
allowances:HashMap[address,HashMap[address,uint256]]@external@viewdefallowance(_owner:address,_spender:address)->uint256:""" @notice Check the amount of tokens that an owner allowed to a spender @param _owner The address which owns the funds @param _spender The address which will spend the funds @return uint256 specifying the amount of tokens still available for the spender """returnself.allowances[_owner][_spender]
Function to approve _spender to transfer _value tokens on behalf of msg.sender.
Returns: true (bool).
Emits: Approval
Input
Type
Description
_spender
address
spender address
_value
uint256
amount to approce
Source code
eventApproval:_owner:indexed(address)_spender:indexed(address)_value:uint256allowances:HashMap[address,HashMap[address,uint256]]@externaldefapprove(_spender:address,_value:uint256)->bool:""" @notice Approve `_spender` to transfer `_value` tokens on behalf of `msg.sender` @dev Approval may only be from zero -> nonzero or from nonzero -> zero in order to mitigate the potential race condition described here: https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 @param _spender The address which will spend the funds @param _value The amount of tokens to be spent @return bool success """assert_value==0orself.allowances[msg.sender][_spender]==0self.allowances[msg.sender][_spender]=_valuelogApproval(msg.sender,_spender,_value)returnTrue
Function to mint _value (uint256) and assign them to _to (address).
Returns: True (bool)
Emits: Transfer
Input
Type
Description
_to
address
receiver of the minted tokens
_value
uint256
amount to mint
Source code
eventTransfer:_from:indexed(address)_to:indexed(address)_value:uint256@externaldefmint(_to:address,_value:uint256)->bool:""" @notice Mint `_value` tokens and assign them to `_to` @dev Emits a Transfer event originating from 0x00 @param _to The account that will receive the created tokens @param _value The amount that will be created @return bool success """assertmsg.sender==self.minter# dev: minter onlyassert_to!=ZERO_ADDRESS# dev: zero addressifblock.timestamp>=self.start_epoch_time+RATE_REDUCTION_TIME:self._update_mining_parameters()_total_supply:uint256=self.total_supply+_valueassert_total_supply<=self._available_supply()# dev: exceeds allowable mint amountself.total_supply=_total_supplyself.balanceOf[_to]+=_valuelogTransfer(ZERO_ADDRESS,_to,_value)returnTrue
Function to burn _value tokens belonging to the caller of the function.
Retruns: True (bool).
Emits: Transfer
Input
Type
Description
_value
uint256
amount to burn
Source code
eventTransfer:_from:indexed(address)_to:indexed(address)_value:uint256@externaldefburn(_value:uint256)->bool:""" @notice Burn `_value` tokens belonging to `msg.sender` @dev Emits a Transfer event with a destination of 0x00 @param _value The amount that will be burned @return bool success """self.balanceOf[msg.sender]-=_valueself.total_supply-=_valuelogTransfer(msg.sender,ZERO_ADDRESS,_value)returnTrue
Getter for mintable supply from start timestamp till end timestamp.
Returns: amount of mintable tokens (uint256).
Input
Type
Description
start
uint256
start timestamp
end
uint256
end timestamp
Source code
@external@viewdefmintable_in_timeframe(start:uint256,end:uint256)->uint256:""" @notice How much supply is mintable from start timestamp till end timestamp @param start Start of the time interval (timestamp) @param end End of the time interval (timestamp) @return Tokens mintable from `start` till `end` """assertstart<=end# dev: start > endto_mint:uint256=0current_epoch_time:uint256=self.start_epoch_timecurrent_rate:uint256=self.rate# Special case if end is in future (not yet minted) epochifend>current_epoch_time+RATE_REDUCTION_TIME:current_epoch_time+=RATE_REDUCTION_TIMEcurrent_rate=current_rate*RATE_DENOMINATOR/RATE_REDUCTION_COEFFICIENTassertend<=current_epoch_time+RATE_REDUCTION_TIME# dev: too far in futureforiinrange(999):# Curve will not work in 1000 years. Darn!ifend>=current_epoch_time:current_end:uint256=endifcurrent_end>current_epoch_time+RATE_REDUCTION_TIME:current_end=current_epoch_time+RATE_REDUCTION_TIMEcurrent_start:uint256=startifcurrent_start>=current_epoch_time+RATE_REDUCTION_TIME:break# We should never get here but what if...elifcurrent_start<current_epoch_time:current_start=current_epoch_timeto_mint+=current_rate*(current_end-current_start)ifstart>=current_epoch_time:breakcurrent_epoch_time-=RATE_REDUCTION_TIMEcurrent_rate=current_rate*RATE_REDUCTION_COEFFICIENT/RATE_DENOMINATOR# double-division with rounding made rate a bit less => goodassertcurrent_rate<=INITIAL_RATE# This should never happenreturnto_mint
If the formulas below do not render, please make sure to refresh the site. A solution is being worked on.
Mining parameters are used to determine the token emissions. The emissions are based on epochs (one year). With every passing epoch, the rate will be reduced, thereby reducing the entire CRV emissions.
The rate can be reduced by invoking the update_mining_parameters() function. While this function is accessible to anyone, an attempt to call it will be reverted if a year hasn't elapsed since the last update. When the function is successfully executed, the mining_epoch increments by 1, and the start_epoch_time is updated to the timestamp of that function call. Moreover, the update_mining_parameters() function will be automatically triggered if someone tries to mint CRV before the scheduled rate reduction.
with:
Effectively, every rate reduction decreases the CRV inflation by around 15.9%.
Getter for the current mining epoch. The mining epoch is incremented by 1 every time update_mining_parameters() is successfully called. At deployment, mining_epoch was set to -1.
Returns: mining epoch (int128).
Source code
mining_epoch:public(int128)@externaldef__init__(_name:String[64],_symbol:String[32],_decimals:uint256):""" @notice Contract constructor @param _name Token full name @param _symbol Token symbol @param _decimals Number of decimals for token """...self.mining_epoch=-1...@internaldef_update_mining_parameters():""" @dev Update mining rate and supply at the start of the epoch Any modifying mining call must also call this """_rate:uint256=self.rate_start_epoch_supply:uint256=self.start_epoch_supplyself.start_epoch_time+=RATE_REDUCTION_TIMEself.mining_epoch+=1if_rate==0:_rate=INITIAL_RATEelse:_start_epoch_supply+=_rate*RATE_REDUCTION_TIMEself.start_epoch_supply=_start_epoch_supply_rate=_rate*RATE_DENOMINATOR/RATE_REDUCTION_COEFFICIENTself.rate=_ratelogUpdateMiningParameters(block.timestamp,_rate,_start_epoch_supply)
Getter for the start times of the current mining epoch.
Returns: timestamp (uint256).
Source code
start_epoch_time:public(uint256)@externaldef__init__(_name:String[64],_symbol:String[32],_decimals:uint256):""" @notice Contract constructor @param _name Token full name @param _symbol Token symbol @param _decimals Number of decimals for token """...self.start_epoch_time=block.timestamp+INFLATION_DELAY-RATE_REDUCTION_TIME...
>>>CRV.start_epoch_time()1660429048->'Sat Aug 13 2022 22:17:28 GMT+0000'
rate is denominated in emissions per second. Emissions per day: 6.161965695807970181 * 86400 = 532393.8361178086
Source code
rate:public(uint256)@externaldef__init__(_name:String[64],_symbol:String[32],_decimals:uint256):""" @notice Contract constructor @param _name Token full name @param _symbol Token symbol @param _decimals Number of decimals for token """...self.rate=0...
Function to update the mining parameters for the Curve DAO Token ($CRV).
Emits: UpdateMiningParameters
Tip
This function can be called by anyone. However, the call will revert if block.timestamp is less than or equal to start_epoch_time + RATE_REDUCTION_TIME, indicating that one year has not yet passed.
Source code
eventUpdateMiningParameters:time:uint256rate:uint256supply:uint256# Supply parametersINITIAL_SUPPLY:constant(uint256)=1_303_030_303INITIAL_RATE:constant(uint256)=274_815_283*10**18/YEAR# leading to 43% premineRATE_REDUCTION_TIME:constant(uint256)=YEARRATE_REDUCTION_COEFFICIENT:constant(uint256)=1189207115002721024# 2 ** (1/4) * 1e18RATE_DENOMINATOR:constant(uint256)=10**18INFLATION_DELAY:constant(uint256)=86400# Supply variablesmining_epoch:public(int128)start_epoch_time:public(uint256)rate:public(uint256)start_epoch_supply:uint256@externaldefupdate_mining_parameters():""" @notice Update mining rate and supply at the start of the epoch @dev Callable by any address, but only once per epoch Total supply becomes slightly larger if this function is called late """assertblock.timestamp>=self.start_epoch_time+RATE_REDUCTION_TIME# dev: too soon!self._update_mining_parameters()@internaldef_update_mining_parameters():""" @dev Update mining rate and supply at the start of the epoch Any modifying mining call must also call this """_rate:uint256=self.rate_start_epoch_supply:uint256=self.start_epoch_supplyself.start_epoch_time+=RATE_REDUCTION_TIMEself.mining_epoch+=1if_rate==0:_rate=INITIAL_RATEelse:_start_epoch_supply+=_rate*RATE_REDUCTION_TIMEself.start_epoch_supply=_start_epoch_supply_rate=_rate*RATE_DENOMINATOR/RATE_REDUCTION_COEFFICIENTself.rate=_ratelogUpdateMiningParameters(block.timestamp,_rate,_start_epoch_supply)
Function to get the current mining epoch start while simultaneously updating mining parameters (if possible).
Returns: timestamp (uint256).
Source code
start_epoch_time:public(uint256)@externaldefstart_epoch_time_write()->uint256:""" @notice Get timestamp of the current mining epoch start while simultaneously updating mining parameters @return Timestamp of the epoch """_start_epoch_time:uint256=self.start_epoch_timeifblock.timestamp>=_start_epoch_time+RATE_REDUCTION_TIME:self._update_mining_parameters()returnself.start_epoch_timeelse:return_start_epoch_time
Function to get the next mining epoch start while simultaneously updating mining parameters (if possible).
Returns: timestamp (uint256).
Source code
@externaldeffuture_epoch_time_write()->uint256:""" @notice Get timestamp of the next mining epoch start while simultaneously updating mining parameters @return Timestamp of the next epoch """_start_epoch_time:uint256=self.start_epoch_timeifblock.timestamp>=_start_epoch_time+RATE_REDUCTION_TIME:self._update_mining_parameters()returnself.start_epoch_time+RATE_REDUCTION_TIMEelse:return_start_epoch_time+RATE_REDUCTION_TIME