-
Notifications
You must be signed in to change notification settings - Fork 2
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #77 from onomyprotocol/dong/psm-spec
PSM: Add spec for psm module
- Loading branch information
Showing
1 changed file
with
195 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,195 @@ | ||
# PSM Module | ||
|
||
The PSM module is a stabilization mechanism for the nomUSD stablecoin, allowing users to swap approved stablecoins for nomUSD at a 1-1 ratio. | ||
|
||
## Contents | ||
- [Concept](#concept) | ||
- [State](#state) | ||
- [Messages](#messages) | ||
- [Events](#events) | ||
- [Params](#params) | ||
- [Keepers](#keepers) | ||
- [Hooks](#hooks) | ||
- [Future Improvements](#future-improvements) | ||
|
||
## Concept | ||
|
||
PSM issues nomUSD in exchange for approved stablecoins, with a maximum issuance limit set by governance. It also allows users to swap nomUSD for stablecoins at a 1-1 ratio.nomUSD is issued from PSM which is backed by stablecoins within it. Parameters such as limit per stablecoin, swap-in fees and swap-out fees are governed and can change based on economic conditions. PSM is an effective support module for Vaults module in case users need nomUSD to repay off debts and close vault (redeem collateral). | ||
|
||
### Key Features | ||
|
||
Key Features | ||
|
||
- **Swap to nomUSD**: Users can convert from stablecoin to nomUSD provided that the stablecoin has been added to the list of accepted stablecoins. | ||
- **Swap to Stablecoin**: Users can convert from nomUSD to stablecoin provided that the stablecoin has been added to the list of accepted stablecoins. | ||
|
||
## State | ||
|
||
### Params | ||
|
||
Params contains: | ||
|
||
- `LimitTotal`: Maximum amount of nomUSD that can be provided. | ||
- `AcceptablePriceRatio`: The spread price between nomUSD and the stablecoins is at an acceptable level where the fees will remain the same. | ||
- `AdjustmentFee`: "AdjustmentFee" is a parameter used to adjust the amount of change in swap fees based on the deviation of the stablecoin from the target price. AdjustmentFee determines the number of iterations that one of the fees (entry fee or exit fee) will be adjusted according to the current ratio between the target price and the actual price. The larger the AdjustmentFee, the more severe the fee adjustment will be. Specifically, it affects how the feeOut or feeIn is calculated by multiplying (or dividing) it by the price ratio. | ||
|
||
```protobuf | ||
message Params { | ||
// total $nomUSD can mint | ||
bytes limit_total = 1 [ | ||
(cosmos_proto.scalar) = "cosmos.Int", | ||
(gogoproto.customtype) = "cosmossdk.io/math.Int", | ||
(gogoproto.nullable) = false, | ||
(amino.dont_omitempty) = true | ||
]; | ||
// The price cannot be exactly 1, an acceptable such as 0.9999 (AcceptablePriceRatio = 0.0001) | ||
bytes acceptable_price_ratio = 2 [ | ||
(cosmos_proto.scalar) = "cosmos.Dec", | ||
(gogoproto.customtype) = "cosmossdk.io/math.LegacyDec", | ||
(gogoproto.nullable) = false | ||
]; | ||
// feeIn adjustment factor | ||
int64 adjustment_fee = 3; | ||
} | ||
``` | ||
|
||
### Stablecoin Manager | ||
|
||
Each stablecoin will contains: | ||
|
||
#### Stablecoin Manager Params | ||
|
||
Params config for each stablecoin type: | ||
|
||
- `Denom`: stablecoin name | ||
- `LimitTotal`: limit total stablecoin | ||
- `FeeIn`: stablecoin to nomUSD exchange fee, fee_in when 1 stablecoin = 1nomUSD | ||
- `FeeOut`: nomUSD to stablecoin exchange fee, fee_out when 1 stablecoin = 1nomUSD | ||
- `TotalStablecoinLock`: amount of stablecoins locked in exchange for nomUSD, default start at 0 | ||
- `FeeMaxStablecoin`: maximum fee for when either fee = 0, default at fee_in+fee_out | ||
|
||
```protobuf | ||
message Stablecoin { | ||
// stablecoin name | ||
string denom = 1; | ||
// limit total stablecoin module support | ||
bytes limit_total = 2 [ | ||
(cosmos_proto.scalar) = "cosmos.Int", | ||
(gogoproto.customtype) = "cosmossdk.io/math.Int", | ||
(gogoproto.nullable) = false, | ||
(amino.dont_omitempty) = true | ||
]; | ||
// stablecoin to nomUSD exchange fee, fee_in when 1 stablecoin = 1nomUSD | ||
bytes fee_in = 3 [ | ||
(cosmos_proto.scalar) = "cosmos.Dec", | ||
(gogoproto.customtype) = "cosmossdk.io/math.LegacyDec", | ||
(gogoproto.nullable) = false | ||
]; | ||
// nomUSD to stablecoin exchange fee, fee_out when 1 stablecoin = 1nomUSD | ||
bytes fee_out = 4 [ | ||
(cosmos_proto.scalar) = "cosmos.Dec", | ||
(gogoproto.customtype) = "cosmossdk.io/math.LegacyDec", | ||
(gogoproto.nullable) = false | ||
]; | ||
// amount of stablecoins locked in exchange for nomUSD | ||
bytes total_stablecoin_lock = 5 [ | ||
(cosmos_proto.scalar) = "cosmos.Int", | ||
(gogoproto.customtype) = "cosmossdk.io/math.Int", | ||
(gogoproto.nullable) = false, | ||
(amino.dont_omitempty) = true | ||
]; | ||
// maximum fee for when either fee = 0 | ||
bytes fee_max_stablecoin = 6 [ | ||
(cosmos_proto.scalar) = "cosmos.Dec", | ||
(gogoproto.customtype) = "cosmossdk.io/math.LegacyDec", | ||
(gogoproto.nullable) = false | ||
]; | ||
} | ||
``` | ||
|
||
|
||
## Messages | ||
|
||
### MsgSwapTonomUSD | ||
Allows users to swap accepted stablecoins for nomUSD. `Address` is the sender's address, `Coin` is the amount of stablecoin the user sent | ||
|
||
```{.go} | ||
type MsgSwapTonomUSD struct { | ||
Address string | ||
Coin *types.Coin | ||
} | ||
``` | ||
|
||
**State Modifications:** | ||
|
||
- Safety Checks that are being done before running swap logic: | ||
- Check stablecoin is suport | ||
- Check limit swap | ||
- Check balance user and calculate amount of coins received and fee in | ||
|
||
- Transfer stablecoin from user to psm module. | ||
- Mint nomUSD and send for user | ||
|
||
### MsgSwapToStablecoin | ||
Allows users to swap accepted nomUSD for stablecoins. `Address` is the sender's address, `ToDenom` is the stablecoin name to receive ,`Amount` is the amount of nomUSD the user sent | ||
|
||
```{.go} | ||
type MsgSwapToStablecoin struct { | ||
Address string | ||
ToDenom string | ||
Amount math.Int | ||
} | ||
``` | ||
|
||
**State Modifications:** | ||
|
||
- Safety Checks that are being done before running swap logic: | ||
- Check stablecoin is suport | ||
- Check total stablecoin lock enough to swap | ||
- Check balance user and calculate amount of coins received and fee out | ||
|
||
- Transfer nomUSD from user to psm module and burn. | ||
- Transfer stablecoin to user | ||
|
||
## Events | ||
|
||
The PSM module emits events for various operations: | ||
- **AddStablecoin**: Emitted when a new stablecoin is added. | ||
- **UpdateStablecoin**: Emitted when a stablecoin is updates | ||
- **SwapToStablecoin**: Emitted when exchanging nomUSD for stablecoin | ||
- **SwapTonomUSD**: Emitted when exchanging stablecoin for nomUSD | ||
|
||
|
||
## ABCI | ||
|
||
### Fee recalculate | ||
To maintain the peg of 1 nomUSD = 1 USD, the swap fees need to be adjusted whenever the price of stablecoins fluctuates. This adjustment ensures that deviations from the target price of 1 USD per nomUSD are counterbalanced by the fees. | ||
|
||
Adjustment Logic: | ||
- If the stablecoin price is above 1, fee_out (the fee for converting from nomUSD to the stablecoin) will be higher, and fee_in (the fee for converting from the stablecoin to nomUSD) will be lower. This setup discourages swaps that would increase the stablecoin holdings when its value is above 1, helping to bring the price back down. | ||
- If the stablecoin price is below 1, fee_out will be lower, and fee_in will be higher. This makes it cheaper to convert nomUSD to the stablecoin and more costly to convert the stablecoin to nomUSD, which encourages activity that pushes the price back up toward the target. | ||
|
||
|
||
#### How to calculate fee: | ||
The fee adjustments are scaled using the `AdjustmentFee` parameter (k), which controls the responsiveness of the fee to price deviations. | ||
|
||
Suppose: | ||
- `newPrice`: new market price of stablecoin relative to nomUSD. | ||
- `feeIn`: inbound fee (to exchange stablecoin to nomUSD). | ||
- `feeOut`: outbound fee (to exchange nomUSD to stablecoin). | ||
- `maxFee`: maximum total fee for both directions (usually feeIn + feeOut). | ||
- `k`: adjustment factor (`AdjustmentFee`) that controls the sensitivity of the fee to price changes. | ||
|
||
Calculate new price ratio:`rate`= 1/`newPrice` | ||
| ||
Fee adjustment: | ||
- `rate` < 1: | ||
`newFeeOut` = `feeOut`/ `rate`^`k` | ||
`newFeeIn`=`maxFee`−`newFeeOut` | ||
`newFeeOut` will not exceed `maxFee` | ||
- `rate` > 1: | ||
`newFeeIn` = `feeIn`*`rate`^`k` | ||
`newFeeOut`=`maxFee`−`newFeeIn` | ||
`newFeeIn` will not exceed `maxFee` |