Ditto money is an elastic supply token. The total supply of Ditto is not fixed, but instead adjusts on a regular basis. These supply adjustments, called "rebases", are done in such a way that users’ proportional holdings don’t change and thus aren’t diluted. Rebases are performed to achieve a specific target price, with the idea being that a token’s nominal price will steadily be moved over time toward its target. In the case of Ditto money the price target is simply $1.
The Ditto project is a fork of Ampleforth. It includes code from other cutting-edge elastic supply projects and several novel ideas. The original architecture has been simplified into two main contracts:
- Ditto Token - ERC20 token that implements the rebase function.
- Ditto Master - Contains the rebase configuration and admin functionality.
The third necessary component is the price Oracle. The MarketOracle smart contract provides the functionality for calculating an average price using the Uniswap TWAP Oracle interface.
Important rebase parameters are found in the master contract. Most of these values are set during construction or directly in the code.
deviationThreshold
: If the current exchange rate is within this fractional distance from the target price, no supply update is performed. By default this is set to 0.05.rebaseCooldown:
Minimum time that must pass between rebases, in seconds. The default value is 4 hours.
Supply adjustments in Ditto are calculated as:
(_totalSupply * DeviationFromTargetRate) / rebaseLag
The rebaseLag
is meant to "soften" the effect of the rebase. In Ditto money, rebaseLag
is set to different values depending if the rebase is positive or negative. The default values are:
- Negative rebase lag: 2
- Positive rebase lag: 5
Most of the time, liquidity and lending pools need to be notified of rebases so that the internal accounting is updated accordingly. This needs to happen atomically in the same transaction as the rebase. If this is not done correctly, attackers can take advantage of the internal accounting mismatch to steal value from liquidity providers.
The Master contract maintains a list of external function calls that are executed at the end of every rebase. The appropriate calls must be added by the contract owner whenever a new Ditto liquidity pool is created (e.g., call sync()
in an Uniswap liquidity pool). Use addTransaction(address destination, bytes calldata data)
to add a call to the list.
Transferring DittoTokens is initially restricted to the owner. This is to allow an orderly distribution via airdrops, token sale, etc., while preventing users from listing Ditto on AMMs ahead of the Ditto team. Once the distribution lock has been released by the contract creator it cannot be re-engaged.
The market oracle must be deployed last after the token has been listed on an AMM and has sufficient liquidity. To get going, a simple getter/setter Oracle can be used where the owner sets prices manually. The Oracle is switched for the market oracle once it has been tested and confirmed to be reliable.
The oracle calculates a rate for Ditto/USD based on a DITTO/BNB and BNB/USD pool, where "USD" is some kind of USD stable coin that must have high liquidity. O)n Binance Smart Chain, BUSD pools are most suited for this purpose.