Registry flows
In each chain we have a singleton registry contract that is responsible for holding and serving the information of the vaults.
Vaults information:
Address of accountingManager
Address of base token of the vault (token that we use for book keeping)
List of connectors
List of trusted tokens of the vault
List of trusted position blueprint
List of positions that we hold at the moment
Governance addresses
So in each vault that is defined in the registry, there is one accounting manager contract and a set of active connectors.
Flows
Creating a new vault
The maintainer (the timelock contract of the governance) can use this function to add a new vault to the registry. It needs to choose a vaultId and base token of the vault, deploy the accounting manager and set the governance addresses. Then call the function below
Adding a connector to a vault
This set of smart contracts are responsible for connecting to a protocol. Once a connector is deployed, it needs to be added to the registry to function properly. This can be done by using the function below.
Updating connector trusted tokens
In each vault there is a list of trusted tokens for each connector. By adding trusted tokens we can add positions blue prints that use those tokens. For example to be able to add a LP position on uniswap, we have to add the blue print of that position in uniswap and for doing so we have to add the tokens of that pool to the trusted tokens of the uniswap connector.
Only the vault maintainer can call this which means it's protected by timelock contract.
Adding trusted position blue prints
TrustedPostions work as the blueprint for the holding positions. For example we store the pool information of uniswapV3 in TrustedPostions and the actual information of LP tokens in the holding positions.
For each position, we might have different logic and data to calculate the value of that position based on the base token. The CalculatorConnector is the address of the connector, which will be called when the accountingManager wants to calculate the value of the position based on the base token of the vault.
Holding positions and their role in registry
Holding positions are for storing information of the actual positions. The connectors will use this function to add their positions in TVLHelper.
It’s being created when the vault is added to the registry. At that time, one empty position is being added to the vault so we make sure that there is no active position in index 0 of this array.
Later we can add positions to this array using the updateHoldingPosition function.
For each holding position of the vault, we calculate the holdingPositionId and use that as the key in the isPositionUsed mapping and use the index of that position in the array as the value. This is the reason that we added a dummy position in the array at the beginning (so we can check this mapping to see if a position exists with this Id (if the value is 0 means that it’s not been added to this mapping or it’s dummy position)).
For example, in an aave position, in “supply” function, we call calculatePositionId first and then call updateHoldingPosition to add the position to the vault.
Later, when the accountingManager is trying to get the TVL of the vault, it loops through this array and calls the getPositionTVL of the BaseConnector which will call the _getPositionTVL of aave connector. This function will return the value of our deposits into aave and convert it to the base token.
Holding positions with timestamp
For some positions, we can’t calculate the TVL of the moment (for example the positions that are on other chains). So we update them regularly and save the TVL amount in the registry. For these positions we use this function to set the time.
Last updated