Skip to content

elmol/idle-tranches

 
 

Repository files navigation

Idle Perpetual Yield Tranches

The aim of Idle Perpetual Yield Tranches is to pool capital of users (eg DAI), deposit it into a lending provider (eg Idle Finance) and split the interest received between 2 classes of users with different risk profiles.

One will gain more interest and will be more risky (BB or junior tranche) and the other will have a lower APR but more safety (AA or senior tranche). In the case of an hack or a loss of funds of the lending provider integrated (or any other protocol integrated by this provider), all funds still available will be used to refund senior tranche holders first with the aim of making them whole, and with remaining funds, if any, junior holders after.

There are no locking period or epochs and users are free to enter and exit at any time, the interest earned (and governance tokens, after being partially sold in the market) will be split between the two classes according to a predefined ratio called trancheAPRSplitRatio (eg 20% interest to AA holders and 80% to BB) so the rate is variable for both classes of tranches.

To determine the actual APR for each tranche class, we also need to know the ratio between the current underlying value of AA and BB tranches (ie AAratio = AA Value / (AA Value + BB Value)), below are some examples

Tranche apr / value split ratio

We can see that when the AAratio == trancheAPRSplitRatio we have the same APR of the lending provider (8%).

Given that AA have intrinsically less risk (due the liquidation priority over BB holders) they should also have a lower APR than BB, so ideally AAratio > trancheAPRSplitRatio (eg if the interest for AA is 20% of the total interest earned then more than 20% of the total tranches value should be of AA tranches). This behaviour should be market driven given the different rights the users have in case of hack.

The ratio between tranches heavily influences the APR of both classes of users and the aim should be to have different results with regard to the basic lending provider but not too much skewed otherwise one of the two classes won't have enough holders due to the low APR. For this reason a trancheIdealWeightRatio is set during initialization (and can be updated) which represent the ideal ratio, in value, that AA and BB should have to have an APR suitable for both classes of users.

For example in the image above, where we are splitting the interest 20% to AA and 80% to BB, an ideal AA ratio could be 50% because it would mean that holders of 50% of the pool (which are generating 50% of the interest) are retaining only the 20% of the total interest, so the APR of BB holders will be higher than the normal lending provider (12.8% vs the 8% in the example) and the one of AA is not too low considering it's liquidation priority in case of hack or loss.

To incentivize the reach of trancheIdealWeightRatio part of farmed governance tokens (eg IDLE) are redistributed to users who stakes their tranche tokens in specific tranche rewards contracts.

In case of hack, an emergency shutdown can be triggered (by both the guardian, which would be a multi-sig wallet, and the owner which will be the Idle governance) in order to pause both deposits and redeems, the redistribution of remaining funds can happens selectively, by allowing only AA holders to withdraw first directly in the main contract, or through a separate contract for more complex cases and resolutions (managed by the Idle governance).

A Fee is collected on harvests in the form AA or BB supply diluition (based on the current AA ratio in value) and it's basically a performance fee, currently set at 10% of the interest generated and it will be redirected to the Idle fee collector address.

Architecture

The main contract which will be used by users is IdleCDO which allow to deposits underlying and mint tranche tokens (ERC20), either AA or BB, and redeem principal+interest from it.

The IdleCDO references an IIdleCDOStrategy and the first strategy proposed is IdleStrategy which uses Idle finance as a lending provider. IdleCDO should be able to change its strategy without impact, so it's possibile to change the lending provider (assuming the position have been previously withdrawn). Governance tokens collected as rewards, are not redistributed to users directly in the IdleCDO contract but rather sold to the market (harvest method) and the underlyings reinvested in the downstream lending provider where possibile. Other tokens (eg IDLE that won't be sold or tokens that have no liquid markets) will get redistributed to people who staked their tranches in a separate IdleCDOTrancheRewards contract (one for AA and one for BB).

These are the main contracts used:

  • IdleCDO.sol: contract which holds all the users pooled assets (both underlyings, eg DAI, and interest bearing tokens, eg idleDAI) and entry point for the user to mint tranche tokens and burn them to redeem principal + interest. When users deposit into the CDO they will: update the global accounting of the system (ie split accrued rewards) and mint their choosen tranche tokens. Funds won't get put in lending right away. The harvest method will be called periodically to put new deposits in lending, get fees and update the accounting. During the harvest call some predefined rewards will be sold into the market (via uniswap) and released linearly over x (currently set a 1500) blocks, to increase the value of all tranche holders, and part of the gov tokens will be sent to IdleCDOTrancheRewards contracts, if those are set, to incentivize the ideal ratio trancheIdealWeightRatio. On redeem users will burn their tranche tokens and get underlyings using a checkpointed price (set at last harvest to avoid potential theft of interest, updated when dumping gov tokens to increase the tranche price)

  • IdleCDOTrancheRewards.sol: contract for staking tranche tokens and getting rewards (for incentivizing the trancheIdealWeightRatio). During harvest depositReward can be called to transfer rewards that will be released linearly over x (currently set a 1500) blocks

  • IdleCDOTranche.sol: ERC20 representing a specific (either AA or BB) tranche token. Only IdleCDO contract can mint and burn tranche tokens.

  • IdleStrategy.sol: IdleCDO strategy for lending assets in Idle Finance. This contract it's just a proxy for interacting with Idle Finance and should have no funds at end of each transaction, but it can have stkAAVE if those are redistributed by the lending provider (this is made to let the IdleCDO properly convert stkAAVE to AAVE) and can be claimed only by the IdleCDO contract or the strategy owner. More info on how idleTokens works can be found here. The tldr of the Idle Finance protocol is this (using DAI as an example): In Idle Finance you deposit DAI to earn interest, on deposits you get back idleDAI, an interest bearing token that always increase in price and represent your position in Idle + the interest earned by that position (similar to Compound's cTokens). Idle lend those funds to other protocols such as Compound and Aave and continously rebalance the pooled user funds in order to achieve the highest avg yield. On redeems you burn your idleDAI and get back DAI principal + DAI earned as interest + a set of governance tokens rewards (currently IDLE, COMP, stkAAVE)

Notes:

  • IdleCDO, IdleStrategy and IdleCDOTrancheRewards are upgradable contracts.
  • There are no 'loose' scripts but only hardhat tasks which are used both for interacting with contracts and tests in fork (integration task)
  • The integration task should be useful also to understand the complete workflow

Setup

yarn install

Testing

For unit tests run:

yarn test

For integration tests:

Copy the .env.public in a new .env file and fill out the keys OR in terminal:

export ALCHEMY_API_KEY=XXXX

then uncomment the following in hardhat.config.js (be sure to have a pinned block):

forking: {
  url: `https://eth-mainnet.alchemyapi.io/v2/${process.env.ALCHEMY_API_KEY}`,
  blockNumber: XXXXX,
}

then run

yarn test-integration

For running any other task:

npx hardhat TASK_NAME

Deploy

npx hardhat deploy-with-factory-params --network YOUR_CONFIGURED_NETWORK --cdoname CDO_NAME 

CDO_NAME should be the name of the key of the deployTokens object in lib/addresses.js with all params for deployment. This is an example of config for deploying an IdleCDO with Lido strategy:

  lido: {
    underlying: mainnetContracts.stETH,    // underlying token for the IdleCDO
    decimals: 18,                          // underlying token decimals
    proxyCdoAddress: CDOs.idleDAI.cdoAddr, // address of another IdleCDO where we will get the implementation to use
    strategyName: 'IdleLidoStrategy',      // name of the strategy contract
    strategyParams: [                      // strategy params for `initialize` call
      mainnetContracts.wstETH,
      mainnetContracts.stETH,
      'owner'                              // The string 'owner' can be used to replace a specific address with the deployer of the strategy
    ],
    AAStaking: false,                      // if rewards are distributed in IdleCDO to AA holders
    BBStaking: false,                      // if rewards are distributed in IdleCDO to BB holders
    stkAAVEActive: false,                  // if IdleCDO needs to manage stkAAVE
    limit: '1000000',                      // deposit limit for this IdleCDO
    AARatio: '10000'                       // Interest rate split for AA holders. `100000` means 100% to AA
  },

Coverage

npx hardhat coverage

Code Contributions

We welcome new contributors and code contributions with open arms! Please be sure to follow our contribution guidelines when proposing any new code. Idle Finance is a decentralized protocol managed by a decentralized governance, any new code contributions are more likely to be accepted into future deployments and proposals if they have been openly discussed within the community first in our forum https://gov.idle.finance/

About

No description, website, or topics provided.

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Languages

  • JavaScript 60.9%
  • Solidity 39.1%