Note
This repo is a fork of metamorpho, with 4 changes:
- this MetaMorpho vault does not realize the bad debt (more on this here);
- the timelock can be set to zero at deployment;
- the name and symbol are mutable;
reallocate
always reverts if the market is not enabled in the vault.
MetaMorpho is a protocol for noncustodial risk management on top of Morpho Blue. It enables anyone to create a vault depositing liquidity into multiple Morpho Blue markets. It offers a seamless experience similar to Aave and Compound.
Users of MetaMorpho are liquidity providers who want to earn from borrowing interest without having to actively manage the risk of their position. The active management of the deposited assets is the responsibility of a set of different roles (owner, curator and allocators). These roles are primarily responsible for enabling and disabling markets on Morpho Blue and managing the allocation of users’ funds.
MetaMorphoV1_1
vaults are ERC-4626 vaults, with (ERC-2612) permit.
One MetaMorpho vault is related to one loan asset on Morpho Blue.
The MetaMorphoV1_1Factory
is deploying immutable onchain instances of MetaMorpho vaults.
Users can supply or withdraw assets at any time, depending on the available liquidity on Morpho Blue. A maximum of 30 markets can be enabled on a given MetaMorpho vault. Each market has a supply cap that guarantees lenders a maximum absolute exposure to the specific market. By default, the supply cap of a market is set to 0.
There are 4 different roles for a MetaMorpho vault: owner, curator, guardian & allocator.
The vault owner can set a performance fee, cutting up to 50% of the generated interest.
The feeRecipient
can then withdraw the accumulated fee at any time.
The vault may be entitled to some rewards emitted on Morpho Blue markets the vault has supplied to.
Those rewards can be transferred to the skimRecipient
.
The vault's owner has the choice to distribute back these rewards to vault depositors however they want.
For more information about this use case, see the Rewards section.
All actions that may be against users' interests (e.g. enabling a market with a high exposure) is subject to a timelock.
To make vault setup easier, the initial timelock can be either 0 or anywhere between 24 hours and 2 weeks.
Any further timelock change must set the value between 24 hours and 2 weeks.
The owner
, or the guardian
if set, can revoke the action during the timelock.
After the timelock, the action can be executed by anyone.
Only one address can have this role.
It can:
- Do what the curator can do.
- Do what the guardian can do.
- Transfer or renounce the ownership.
- Set the curator.
- Set allocators.
- Set the rewards recipient.
- Increase the timelock.
- [Timelocked] Decrease the timelock.
- [Timelocked if already set] Set the guardian.
- Set the performance fee (capped at 50%).
- Set the fee recipient.
- Set the name and symbol of the vault.
Only one address can have this role.
It can:
- Do what allocators can do.
- Decrease the supply cap of any market.
- To softly remove a market after the curator has set the supply cap to 0, it is expected from the allocator role to reallocate the supplied liquidity to another enabled market and then to update the withdraw queue.
- [Timelocked] Increase the supply cap of any market.
- [Timelocked] Submit the forced removal of a market.
- This action is typically designed to force the removal of a market that keeps reverting thus locking the vault.
- After the timelock has elapsed, the allocator role is free to remove the market from the withdraw queue. The funds supplied to this market will be lost.
- If the market ever functions again, the allocator role can withdraw the funds that were previously lost.
- Revoke the pending cap of any market.
- Revoke the pending removal of any market.
Multiple addresses can have this role.
It can:
- Set the
supplyQueue
andwithdrawQueue
, i.e. decide on the order of the markets to supply/withdraw from.- Upon a deposit, the vault will supply up to the cap of each Morpho Blue market in the
supplyQueue
in the order set. - Upon a withdrawal, the vault will withdraw up to the liquidity of each Morpho Blue market in the
withdrawQueue
in the order set. - The
supplyQueue
only contains markets which cap has previously been non-zero. - The
withdrawQueue
contains all markets that have a non-zero cap or a non-zero vault allocation.
- Upon a deposit, the vault will supply up to the cap of each Morpho Blue market in the
- Instantaneously reallocate funds by supplying on markets of the
withdrawQueue
and withdrawing from markets that have the same loan asset as the vault's asset.
Warning If
supplyQueue
is empty, depositing to the vault is disabled.
Only one address can have this role.
It can:
- Revoke the pending timelock.
- Revoke the pending guardian (which means it can revoke any attempt to change the guardian).
- Revoke the pending cap of any market.
- Revoke the pending removal of any market.
In some cases, the vault's curator or allocators may want to keep some funds "idle", to guarantee lenders that some liquidity can be withdrawn from the vault (beyond the liquidity of each of the vault's markets).
To achieve this, they can deposit in markets with address(0)
as the oracle or the collateral, ensuring that these funds can't be borrowed.
They are thus guaranteed to be liquid; though they won't generate interest.
It is advised to use these canonical configurations for "idle" markets:
loanToken
: The vault's asset to be able to supply/withdraw funds.collateralToken
:address(0)
(not necessary since no funds will be borrowed on this market)irm
:address(0)
(Morpho Blue will skip the call to the IRM in this case, thus reducing the gas cost)oracle
:address(0)
(not necessary since no funds will be borrowed on this market)lltv
:0
(not necessary since no funds will be borrowed on this market)
Note that to allocate funds to this idle market, it is first required to enable its cap on MetaMorpho.
Enabling an infinite cap (type(uint184).max
) will always allow users to deposit on the vault.
To redistribute rewards to vault depositors, it is advised to use the Universal Rewards Distributor (URD).
Below is a typical example of how this use case would take place:
-
If not already done:
- Create a rewards distributor using the UrdFactory (can be done by anyone).
- Set the vault’s rewards recipient address to the created URD using
setSkimRecipient
.
-
Claim tokens from the Morpho Blue distribution to the vault.
NB: Anyone can claim tokens on behalf of the vault and automatically transfer them to the vault. Thus, this step might be already performed by some third-party.
-
Transfer rewards from the vault to the rewards distributor using the
skim
function.NB: Anyone can transfer rewards from the vault to the rewards distributor unless it is unset. Thus, this step might be already performed by some third-party. Note: the amount of rewards transferred corresponds to the vault's balance of reward asset.
-
Compute the new root for the vault’s rewards distributor, submit it, wait for the timelock (if any), accept the root, and let vault depositors claim their rewards according to the vault manager’s rewards re-distribution strategy.
If an enabled market is considered unsafe (e.g., risk too high), the curator/owner may want to disable this market in the following way:
-
- Revoke the pending cap of the market with the
revokePendingCap
function (this can also be done by the guardian).
- Revoke the pending cap of the market with the
-
- Set the cap of the market to 0 with the
submitCap
function. To ensure that submit cap does not revert because of a pending cap, it is recommended to batch the two previous transactions, for example using the multicall function of MetaMorpho.
- Set the cap of the market to 0 with the
-
- Withdraw all the supply of this market with the
reallocate
function. If there is not enough liquidity on the market, remove the maximum available liquidity with thereallocate
function, then put the market at the beginning of the withdraw queue (with theupdateWithdrawQueue
function).
- Withdraw all the supply of this market with the
-
- Once all the supply has been removed from the market, the market can be removed from the withdraw queue with the
updateWithdrawQueue
function.
- Once all the supply has been removed from the market, the market can be removed from the withdraw queue with the
If an enabled market starts reverting, many of the vault functions would revert as well (because of the call to totalAssets
). To turn the vault back to an operating state, the market must be forced removed by the owner/curator, who should follow these steps:
-
- Revoke the pending cap of the market with the
revokePendingCap
function (this can also be done by the guardian).
- Revoke the pending cap of the market with the
-
- Set the cap of the market to 0 with the
submitCap
function. To ensure that submit cap does not revert because of a pending cap, it is recommended to batch the two previous transactions, for example using the multicall function of MetaMorpho.
- Set the cap of the market to 0 with the
-
- Submit a removal of the market with the
submitMarketRemoval
function.
- Submit a removal of the market with the
-
- Wait for the timelock to elapse
-
- Once the timelock has elapsed, the market can be removed from the withdraw queue with the
updateWithdrawQueue
function.
- Once the timelock has elapsed, the market can be removed from the withdraw queue with the
Warning : Funds supplied in forced removed markets will be lost, this is why only markets expected to always revert should be disabled this way (because funds supplied in such markets can be considered lost anyway).
If the curator starts to submit positive caps for unsafe markets that are not in line with the vault risk strategy, the owner of the vault can:
-
- Set a new curator with the
setCurator
function.
- Set a new curator with the
-
- Revoke the pending caps submitted by the curator (this can also be done by the guardian or the new curator).
-
- If the curator had the time to accept a cap (because
timelock
has elapsed before the guardian or the owner had time to act), the owner (or the new curator) must disable the unsafe market (see above).
- If the curator had the time to accept a cap (because
If one of the allocators starts setting the withdraw queue and/or supply queue that are not in line with the vault risk strategy, or incoherently reallocating the funds, the owner of the vault should:
-
- Deprive the faulty allocator from his privileges with the
setIsAllocator
function.
- Deprive the faulty allocator from his privileges with the
-
- Reallocate the funds in a way consistent with the vault risk strategy with the
reallocate
function (this can also be done by the curator or the other allocators).
- Reallocate the funds in a way consistent with the vault risk strategy with the
-
- Set a new withdraw queue that is in line with the vault risk strategy with the
updateWithdrawQueue
function (this can also be done by the curator or the other allocators).
- Set a new withdraw queue that is in line with the vault risk strategy with the
-
- Set a new supply queue that is in line with the vault risk strategy with the
setSupplyQueue
function (this can also be done by the curator or the other allocators).
- Set a new supply queue that is in line with the vault risk strategy with the
npm install @morpho-org/metamorpho
yarn add @morpho-org/metamorpho
Bundle a supply cap raise and a reallocation to the market:
import { MetaMorphoAction } from "@morpho-org/metamorpho";
const marketParams1 = {
collateralToken: "0x...",
loanToken: "0x...",
irm: "0x...",
oracle: "0x...",
lltv: 86_0000000000000000n,
};
const marketParams2 = {
collateralToken: "0x...",
loanToken: marketParams1.loanToken,
irm: "0x...",
oracle: "0x...",
lltv: 96_5000000000000000n,
};
await metamorpho.connect(curator).multicall([
MetaMorphoAction.acceptCap(marketParams2),
MetaMorphoAction.reallocate([
{ marketParams: marketParams1, assets: 600_000000000000000000n },
{ marketParams: marketParams2, assets: 100_000000000000000000n },
]),
]);
Note
MetaMorphoV1_1Factory
has been deployed on Ethereum and Base with the metadata hash included, which appear at two places in the bytecode as it is a factory.
Install dependencies: yarn
Run forge tests: yarn test:forge
Run hardhat tests: yarn test:hardhat
You will find other useful commands in the package.json
file.
All audits are stored in the audits folder.
MetaMorpho is licensed under GPL-2.0-or-later
, see LICENSE
.