The PriceAggregator contract serves as a sophisticated tool designed to compute the aggregated weighted average price of the crvUSD currency across a variety of liquidity pools, employing an advanced methodology that incorporates the exponential moving-average (EMA) of Total Value Locked (TVL) to ensure accurate and reliable pricing data.
The calculated price is primarily used as an oracle for calculating the interest rate, but also for PegKeepers to determine whether to mint and deposit or withdraw and burn.
Contract Source & Deployment
Source code is available on Github. Relevant contract deployments can be found here.
The PriceAggregator contract calculates the weighted average price of crvUSD across multiple liquidity pools, considering only those pools with sufficient liquidity (MIN_LIQUIDITY = 100,000 * 10**18). This calculation is based on the exponential moving-average (EMA) of the Total Value Locked (TVL) for each pool, determining the liquidity considered in the price aggregation.
The price calculation begins with determining the EMA of the TVL from different Curve Stableswap liquidity pools using the _ema_tvl function. This internal function computes the EMA TVLs based on the following formula, which adjusts for the time since the last update to smooth out short-term volatility in the TVL data, providing a more stable and representative average value over the specified time window (TVL_MA_TIME set to 50,000 seconds):
The code snippet provided illustrates the implementation of the above formula in the contract.
_ema_tvl
TVL_MA_TIME:public(constant(uint256))=50000# s@internal@viewdef_ema_tvl()->DynArray[uint256,MAX_PAIRS]:tvls:DynArray[uint256,MAX_PAIRS]=[]last_timestamp:uint256=self.last_timestampalpha:uint256=10**18iflast_timestamp<block.timestamp:alpha=self.exp(-convert((block.timestamp-last_timestamp)*10**18/TVL_MA_TIME,int256))n_price_pairs:uint256=self.n_price_pairsforiinrange(MAX_PAIRS):ifi==n_price_pairs:breaktvl:uint256=self.last_tvl[i]ifalpha!=10**18:# alpha = 1.0 when dt = 0# alpha = 0.0 when dt = infnew_tvl:uint256=self.price_pairs[i].pool.totalSupply()# We don't do virtual price here to save on gastvl=(new_tvl*(10**18-alpha)+tvl*alpha)/10**18tvls.append(tvl)returntvls
The _price function then uses these EMA TVLs to calculate the aggregated prices by considering the liquidity of each pool. A pool's liquidity must meet or exceed 100,000 * 10**18 to be included in the calculation. The function adjusts the price from the pool's price_oracle based on the position of crvUSD in the liquidity pair, ensuring consistent price representation across pools.
_price
@internal@viewdef_price(tvls:DynArray[uint256,MAX_PAIRS])->uint256:n:uint256=self.n_price_pairsprices:uint256[MAX_PAIRS]=empty(uint256[MAX_PAIRS])D:uint256[MAX_PAIRS]=empty(uint256[MAX_PAIRS])Dsum:uint256=0DPsum:uint256=0foriinrange(MAX_PAIRS):ifi==n:breakprice_pair:PricePair=self.price_pairs[i]pool_supply:uint256=tvls[i]ifpool_supply>=MIN_LIQUIDITY:p:uint256=price_pair.pool.price_oracle()ifprice_pair.is_inverse:p=10**36/pprices[i]=pD[i]=pool_supplyDsum+=pool_supplyDPsum+=pool_supply*pifDsum==0:return10**18# Placeholder for no active poolsp_avg:uint256=DPsum/Dsume:uint256[MAX_PAIRS]=empty(uint256[MAX_PAIRS])e_min:uint256=max_value(uint256)foriinrange(MAX_PAIRS):ifi==n:breakp:uint256=prices[i]e[i]=(max(p,p_avg)-min(p,p_avg))**2/(SIGMA**2/10**18)e_min=min(e[i],e_min)wp_sum:uint256=0w_sum:uint256=0foriinrange(MAX_PAIRS):ifi==n:breakw:uint256=D[i]*self.exp(-convert(e[i]-e_min,int256))/10**18w_sum+=wwp_sum+=w*prices[i]returnwp_sum/w_sum
The process involves:
Storing the price of crvUSD in a prices[i] array for each qualifying pool.
Recording each qualifying pool's supply (TVL) in D[i], adding this supply to Dsum, and accumulating the product of the crvUSD price and pool supply in DPsum.
Iterating over all price pairs to perform the above steps.
Finally, the contract:
Calculates an average price:
Computes a variance measure e for each pool's price relative to the average, adjusting by SIGMA to normalize:
Applies an exponential decay based on these variance measures to weigh each pool's contribution to the final average price, reducing the influence of prices far from the minimum variance.
Sums up all w to store it in w_sum and calculates the product of w * prices[i], which is stored in wp_sum.
Finally calculates the weighted average price as wp_sum / w_sum, with weights adjusted for both liquidity and price variance.
Function to calculate the weighted price of crvUSD.
Returns: price (uint256).
Source code
MAX_PAIRS:constant(uint256)=20MIN_LIQUIDITY:constant(uint256)=100_000*10**18# Only take into account pools with enough liquiditySTABLECOIN:immutable(address)SIGMA:immutable(uint256)price_pairs:public(PricePair[MAX_PAIRS])n_price_pairs:uint256@external@viewdefprice()->uint256:returnself._price(self._ema_tvl())@internal@viewdef_price(tvls:DynArray[uint256,MAX_PAIRS])->uint256:n:uint256=self.n_price_pairsprices:uint256[MAX_PAIRS]=empty(uint256[MAX_PAIRS])D:uint256[MAX_PAIRS]=empty(uint256[MAX_PAIRS])Dsum:uint256=0DPsum:uint256=0foriinrange(MAX_PAIRS):ifi==n:breakprice_pair:PricePair=self.price_pairs[i]pool_supply:uint256=tvls[i]ifpool_supply>=MIN_LIQUIDITY:p:uint256=price_pair.pool.price_oracle()ifprice_pair.is_inverse:p=10**36/pprices[i]=pD[i]=pool_supplyDsum+=pool_supplyDPsum+=pool_supply*pifDsum==0:return10**18# Placeholder for no active poolsp_avg:uint256=DPsum/Dsume:uint256[MAX_PAIRS]=empty(uint256[MAX_PAIRS])e_min:uint256=max_value(uint256)foriinrange(MAX_PAIRS):ifi==n:breakp:uint256=prices[i]e[i]=(max(p,p_avg)-min(p,p_avg))**2/(SIGMA**2/10**18)e_min=min(e[i],e_min)wp_sum:uint256=0w_sum:uint256=0foriinrange(MAX_PAIRS):ifi==n:breakw:uint256=D[i]*self.exp(-convert(e[i]-e_min,int256))/10**18w_sum+=wwp_sum+=w*prices[i]returnwp_sum/w_sum
Getter for the last price. This variable was set to (1.00) when initializing the contract and is now updated every time price_w is called.
Returns: last price of crvUSD (uint256).
Source code
last_price:public(uint256)@externaldef__init__(stablecoin:address,sigma:uint256,admin:address):STABLECOIN=stablecoinSIGMA=sigma# The change is so rare that we can change the whole thing altogetherself.admin=adminself.last_price=10**18self.last_timestamp=block.timestamp@externaldefprice_w()->uint256:ifself.last_timestamp==block.timestamp:returnself.last_priceelse:ema_tvl:DynArray[uint256,MAX_PAIRS]=self._ema_tvl()self.last_timestamp=block.timestampforiinrange(MAX_PAIRS):ifi==len(ema_tvl):breakself.last_tvl[i]=ema_tvl[i]p:uint256=self._price(ema_tvl)self.last_price=preturnp
Getter for the exponential moving-average of the TVL in price_pairs.
Returns: array of ema tvls (DynArray[uint256, MAX_PAIRS]).
Source code
TVL_MA_TIME:public(constant(uint256))=50000# slast_tvl:public(uint256[MAX_PAIRS])@external@viewdefema_tvl()->DynArray[uint256,MAX_PAIRS]:returnself._ema_tvl()@internal@viewdef_ema_tvl()->DynArray[uint256,MAX_PAIRS]:tvls:DynArray[uint256,MAX_PAIRS]=[]last_timestamp:uint256=self.last_timestampalpha:uint256=10**18iflast_timestamp<block.timestamp:alpha=self.exp(-convert((block.timestamp-last_timestamp)*10**18/TVL_MA_TIME,int256))n_price_pairs:uint256=self.n_price_pairsforiinrange(MAX_PAIRS):ifi==n_price_pairs:breaktvl:uint256=self.last_tvl[i]ifalpha!=10**18:# alpha = 1.0 when dt = 0# alpha = 0.0 when dt = infnew_tvl:uint256=self.price_pairs[i].pool.totalSupply()# We don't do virtual price here to save on gastvl=(new_tvl*(10**18-alpha)+tvl*alpha)/10**18tvls.append(tvl)returntvls
Function to calculate and write the price. If called successfully, updates last_tvl, last_price and last_timestamp.
Returns: price (uint256).
Source code
@externaldefprice_w()->uint256:ifself.last_timestamp==block.timestamp:returnself.last_priceelse:ema_tvl:DynArray[uint256,MAX_PAIRS]=self._ema_tvl()self.last_timestamp=block.timestampforiinrange(MAX_PAIRS):ifi==len(ema_tvl):breakself.last_tvl[i]=ema_tvl[i]p:uint256=self._price(ema_tvl)self.last_price=preturnp@internal@viewdef_price(tvls:DynArray[uint256,MAX_PAIRS])->uint256:n:uint256=self.n_price_pairsprices:uint256[MAX_PAIRS]=empty(uint256[MAX_PAIRS])D:uint256[MAX_PAIRS]=empty(uint256[MAX_PAIRS])Dsum:uint256=0DPsum:uint256=0foriinrange(MAX_PAIRS):ifi==n:breakprice_pair:PricePair=self.price_pairs[i]pool_supply:uint256=tvls[i]ifpool_supply>=MIN_LIQUIDITY:p:uint256=price_pair.pool.price_oracle()ifprice_pair.is_inverse:p=10**36/pprices[i]=pD[i]=pool_supplyDsum+=pool_supplyDPsum+=pool_supply*pifDsum==0:return10**18# Placeholder for no active poolsp_avg:uint256=DPsum/Dsume:uint256[MAX_PAIRS]=empty(uint256[MAX_PAIRS])e_min:uint256=max_value(uint256)foriinrange(MAX_PAIRS):ifi==n:breakp:uint256=prices[i]e[i]=(max(p,p_avg)-min(p,p_avg))**2/(SIGMA**2/10**18)e_min=min(e[i],e_min)wp_sum:uint256=0w_sum:uint256=0foriinrange(MAX_PAIRS):ifi==n:breakw:uint256=D[i]*self.exp(-convert(e[i]-e_min,int256))/10**18w_sum+=wwp_sum+=w*prices[i]returnwp_sum/w_sum
All price pairs added to the contract are considered when calculating the price of crvUSD. Adding or removing price pairs can only be done by the admin of the contract, which is the Curve DAO.
This function is only callable by the admin of the contract.
Function to add a price pair to the PriceAggregator.
Emits: AddPricePair
Input
Type
Description
_pool
address
Price pair to add
Source code
eventAddPricePair:n:uint256pool:Stableswapis_inverse:bool@externaldefadd_price_pair(_pool:Stableswap):assertmsg.sender==self.adminprice_pair:PricePair=empty(PricePair)price_pair.pool=_poolcoins:address[2]=[_pool.coins(0),_pool.coins(1)]ifcoins[0]==STABLECOIN:price_pair.is_inverse=Trueelse:assertcoins[1]==STABLECOINn:uint256=self.n_price_pairsself.price_pairs[n]=price_pair# Should revert if too many pairsself.last_tvl[n]=_pool.totalSupply()self.n_price_pairs=n+1logAddPricePair(n,_pool,price_pair.is_inverse)
This function is only callable by the admin of the contract.
Function to remove a price pair from the contract. If a prior pool than the latest added one gets removed, the function will move the latest added price pair to the removed pair pairs index to not mess up price_pairs.
Getter for the admin of the contract, which is the Curve DAO OwnershipAgent.
Returns: admin (address).
Source code
admin:public(address)@externaldef__init__(stablecoin:address,sigma:uint256,admin:address):STABLECOIN=stablecoinSIGMA=sigma# The change is so rare that we can change the whole thing altogetherself.admin=admin
This function is only callable by the admin of the contract.
Function to set a new admin.
Emits: SetAdmin
Input
Type
Description
_admin
address
new admin address
Source code
eventSetAdmin:admin:addressadmin:public(address)@externaldefset_admin(_admin:address):# We are not doing commit / apply because the owner will be a voting DAO anyway# which has vote delaysassertmsg.sender==self.adminself.admin=_adminlogSetAdmin(_admin)
Getter for the sigma value. SIGMA is a predefined constant that influences the adjustment of price deviations, affecting how variations in individual stablecoin prices contribute to the overall average stablecoin price.