Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: adding ibc quota v2 #2296

Merged
merged 31 commits into from
Nov 15, 2023
Merged
Show file tree
Hide file tree
Changes from 26 commits
Commits
Show all changes
31 commits
Select commit Hold shift + click to select a range
5578d8c
WIP: adding quota v2
gsk967 Oct 26, 2023
aeaf433
Merge branch 'main' into sai/quota_v2
gsk967 Oct 27, 2023
8663c38
adding inflow quota check
gsk967 Oct 27, 2023
fa816a8
trying to fix ibc tests
gsk967 Oct 27, 2023
39d21a7
Merge branch 'main' into sai/quota_v2
gsk967 Oct 31, 2023
f2f4660
update the protos
gsk967 Oct 31, 2023
49112a7
enable goconst in golang-lint
gsk967 Oct 31, 2023
b1382bf
address the review comments
gsk967 Oct 31, 2023
5ad6cb2
add new params values to upgrade handler
gsk967 Oct 31, 2023
0a7ad7e
Merge branch 'main' into sai/quota_v2
gsk967 Oct 31, 2023
53513de
Merge branch 'main' into sai/quota_v2
gsk967 Nov 3, 2023
0676c7f
address the second review comments
gsk967 Nov 3, 2023
aa7e8fb
removed denom metadata tracker from docs
gsk967 Nov 3, 2023
bad905b
Merge branch 'main' into sai/quota_v2
gsk967 Nov 4, 2023
6868b11
Merge branch 'main' into sai/quota_v2
gsk967 Nov 6, 2023
3f28e31
add check for token outflows with InflowOutflowQuotaTokenBase
gsk967 Nov 7, 2023
3b73075
fix the ibc e2e tests
gsk967 Nov 8, 2023
f40788c
Merge branch 'main' into sai/quota_v2
gsk967 Nov 13, 2023
c4f2a2b
Merge branch 'main' into sai/quota_v2
gsk967 Nov 14, 2023
1842dc0
Merge branch 'main' into sai/quota_v2
gsk967 Nov 14, 2023
c6dd61b
move the new migration of uibc params to keeper
gsk967 Nov 14, 2023
dd7e859
Merge branch 'main' into sai/quota_v2
gsk967 Nov 14, 2023
4d4ca41
make keys functions to private
gsk967 Nov 14, 2023
7478cac
Merge remote-tracking branch 'origin/sai/quota_v2' into sai/quota_v2
gsk967 Nov 14, 2023
e3359d6
add tests for uibc quota inflows
gsk967 Nov 14, 2023
66bc6a6
Merge branch 'main' into sai/quota_v2
gsk967 Nov 14, 2023
d90b400
refactored GetAllInflows and GetAllOutflows funs
gsk967 Nov 14, 2023
4d3c278
Update util/store/iter.go
robert-zaremba Nov 14, 2023
b2acff9
Merge branch 'main' into sai/quota_v2
gsk967 Nov 15, 2023
1edf594
fix the build issue
gsk967 Nov 15, 2023
19aea15
refactor LoadAllDecCoins func
gsk967 Nov 15, 2023
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
8 changes: 8 additions & 0 deletions app/upgrades.go
Original file line number Diff line number Diff line change
Expand Up @@ -125,6 +125,14 @@ func (app *UmeeApp) registerUpgrade6_2(upgradeInfo upgradetypes.Plan) {
govParams := app.GovKeeper.GetParams(ctx)
govParams.MinInitialDepositRatio = sdk.NewDecWithPrec(1, 1).String()
err = app.GovKeeper.SetParams(ctx, govParams)
if err != nil {
return fromVM, err
}

// uibc migrations
uIBCKeeper := app.UIbcQuotaKeeperB.Keeper(&ctx)
uIBCKeeper.MigrateTotalOutflowSum()
err = uIBCKeeper.SetParams(uibc.DefaultParams())
return fromVM, err
},
)
Expand Down
12 changes: 11 additions & 1 deletion proto/umee/uibc/v1/genesis.proto
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,6 @@ message GenesisState {
(gogoproto.castrepeated) = "github.com/cosmos/cosmos-sdk/types.DecCoins",
(gogoproto.nullable) = false
];

// total_outflow_sum defines the total outflow sum of ibc-transfer in USD.
string total_outflow_sum = 3 [
(cosmos_proto.scalar) = "cosmos.Dec",
Expand All @@ -31,4 +30,15 @@ message GenesisState {
(gogoproto.jsontag) = "quota_duration,omitempty",
(gogoproto.moretags) = "yaml:\"quota_expires\""
];
// inflows tracks IBC inflow transfers (in USD) for each denom during quota period.
repeated cosmos.base.v1beta1.DecCoin inflows = 5 [
(gogoproto.castrepeated) = "github.com/cosmos/cosmos-sdk/types.DecCoins",
(gogoproto.nullable) = false
];
// total_inflow_sum defines tracks total sum of IBC inflow transfers (in USD) during quota period.
string total_inflow_sum = 6 [
(cosmos_proto.scalar) = "cosmos.Dec",
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
];
}
18 changes: 18 additions & 0 deletions proto/umee/uibc/v1/quota.proto
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,24 @@ message Params {
(gogoproto.jsontag) = "quota_duration,omitempty",
(gogoproto.moretags) = "yaml:\"quota_duration\""
];
// inflow_outflow_quota_base defines the inflow outflow quota base of ibc-transfer in USD
robert-zaremba marked this conversation as resolved.
Show resolved Hide resolved
string inflow_outflow_quota_base = 5 [
(cosmos_proto.scalar) = "cosmos.Dec",
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
];
// inflow_outflow_quota_rate defines the rate of total inflows
string inflow_outflow_quota_rate = 6 [
(cosmos_proto.scalar) = "cosmos.Dec",
gsk967 marked this conversation as resolved.
Show resolved Hide resolved
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
];
// inflow_outflow_quota_token_base defines the inflow outflow quota base for token
string inflow_outflow_quota_token_base = 7 [
(cosmos_proto.scalar) = "cosmos.Dec",
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
];
gsk967 marked this conversation as resolved.
Show resolved Hide resolved
}

// IBCTransferStatus status of ibc-transfer quota check for inflow and outflow
Expand Down
4 changes: 2 additions & 2 deletions tests/e2e/e2e_ibc_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -169,8 +169,8 @@ func (s *E2ETest) TestIBCTokenTransfer() {
// supply will be not be decreased because sending more than total quota from umee to gaia
s.checkSupply(umeeAPIEndpoint, uatomIBCHash, atomFromGaia.Amount)

// ✅ << BELOW TOKEN QUTOA 5$ but ATOM_QUOTA (5$)+ UMEE_QUOTA(90$) <= TOTAL QUOTA (120$) >>
// send $15 ATOM from umee to gaia
// ✅ << BELOW TOKEN QUTOA 5$ but ATOM_QUOTA (5$)+ UMEE_QUOTA(90$) <= TOTAL QUOTA (120$)
// send $5 ATOM from umee to gaia
sendAtom := mulCoin(atomQuota, "0.05")
s.SendIBC(s.Chain.ID, setup.GaiaChainID, "", sendAtom, false, "below both quotas")
// remaing supply decreased uatom on umee
Expand Down
9 changes: 9 additions & 0 deletions util/store/store.go
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@
if bz := store.Get(key); len(bz) > 0 {
var c TPtr = new(T)
if err := c.Unmarshal(bz); err != nil {
panic(fmt.Sprintf("error unmarshaling %s into %T: %s", errField, c, err))

Check warning on line 29 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L29

Added line #L29 was not covered by tests
}
return c
}
Expand All @@ -39,7 +39,7 @@
func SetValue[T Marshalable](store sdk.KVStore, key []byte, value T, errField string) error {
bz, err := value.Marshal()
if err != nil {
return fmt.Errorf("can't marshal %s: %s", errField, err)

Check warning on line 42 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L42

Added line #L42 was not covered by tests
}
store.Set(key, bz)
return nil
Expand All @@ -47,26 +47,26 @@

// GetBinValue is similar to GetValue (loads value in the store),
// but uses UnmarshalBinary interface instead of protobuf
func GetBinValue[TPtr PtrBinMarshalable[T], T any](store sdk.KVStore, key []byte, errField string) (TPtr, error) {

Check warning on line 50 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L50

Added line #L50 was not covered by tests
if bz := store.Get(key); len(bz) > 0 {
var c TPtr = new(T)
if err := c.UnmarshalBinary(bz); err != nil {
return nil, fmt.Errorf("error unmarshaling %s into %T: %s", errField, c, err)

Check warning on line 54 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L52-L54

Added lines #L52 - L54 were not covered by tests
}
return c, nil

Check warning on line 56 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L56

Added line #L56 was not covered by tests
}
return nil, nil

Check warning on line 58 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L58

Added line #L58 was not covered by tests
}

// SetBinValue is similar to SetValue (stores value in the store),
// but uses UnmarshalBinary interface instead of protobuf
func SetBinValue[T BinMarshalable](store sdk.KVStore, key []byte, value T, errField string) error {
bz, err := value.MarshalBinary()
if err != nil {
return fmt.Errorf("can't marshal %s: %s", errField, err)

Check warning on line 66 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L63-L66

Added lines #L63 - L66 were not covered by tests
}
store.Set(key, bz)
return nil

Check warning on line 69 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L68-L69

Added lines #L68 - L69 were not covered by tests
}

// GetValueCdc is similar to GetValue, but uses codec for marshaling. For Protobuf objects the
Expand All @@ -74,24 +74,24 @@
// instead of GetValue.
// Returns a boolean indicating whether any data was found. If the return is false, the object
// is not changed by this function.
func GetValueCdc(store sdk.KVStore, cdc codec.Codec, key []byte, object codec.ProtoMarshaler, errField string) bool {
if bz := store.Get(key); len(bz) > 0 {
err := cdc.Unmarshal(bz, object)
if err != nil {
panic(errField + " could not be unmarshaled: " + err.Error())

Check warning on line 81 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L77-L81

Added lines #L77 - L81 were not covered by tests
}
return true

Check warning on line 83 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L83

Added line #L83 was not covered by tests
}
return false

Check warning on line 85 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L85

Added line #L85 was not covered by tests
}

// SetValueCdc is similar to the SetValue, but uses codec for marshaling. For Protobuf objects the
// result is the same, unless codec.Any is used. In the latter case this function MUST be used,
// instead of SetValue.
func SetValueCdc(store sdk.KVStore, cdc codec.Codec, key []byte, object codec.ProtoMarshaler, errField string) error {
bz, err := cdc.Marshal(object)

Check warning on line 92 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L91-L92

Added lines #L91 - L92 were not covered by tests
if err != nil {
return fmt.Errorf("failed to encode %s, %s", errField, err.Error())

Check warning on line 94 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L94

Added line #L94 was not covered by tests
}
store.Set(key, bz)
return nil
Expand All @@ -114,7 +114,7 @@
func SetInt(store sdk.KVStore, key []byte, val sdkmath.Int, errField string) error {
if val.IsNil() || val.IsZero() {
store.Delete(key)
return nil

Check warning on line 117 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L117

Added line #L117 was not covered by tests
}
return SetValue(store, key, &val, errField)
}
Expand All @@ -135,7 +135,7 @@
func SetDec(store sdk.KVStore, key []byte, val sdk.Dec, errField string) error {
if val.IsNil() || val.IsZero() {
store.Delete(key)
return nil

Check warning on line 138 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L138

Added line #L138 was not covered by tests
}
return SetValue(store, key, &val, errField)
}
Expand All @@ -155,7 +155,7 @@
func SetAddress(store sdk.KVStore, key []byte, val sdk.AccAddress) {
if val == nil || val.Empty() {
store.Delete(key)
return

Check warning on line 158 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L158

Added line #L158 was not covered by tests
}
store.Set(key, val)
}
Expand Down Expand Up @@ -187,8 +187,8 @@
case uint32:
bz = make([]byte, 4)
binary.LittleEndian.PutUint32(bz, v)
case byte:
bz = []byte{v}

Check warning on line 191 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L190-L191

Added lines #L190 - L191 were not covered by tests
}
store.Set(key, bz)
}
Expand All @@ -204,8 +204,17 @@
return T(binary.LittleEndian.Uint64(bz)), true
case int32, uint32:
return T(binary.LittleEndian.Uint32(bz)), true
case byte:
return T(bz[0]), true

Check warning on line 208 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L207-L208

Added lines #L207 - L208 were not covered by tests
}
panic("not possible: all types must be covered above")

Check warning on line 210 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L210

Added line #L210 was not covered by tests
}

// DeleteByPrefixStoreIterator will delete all keys stored in prefix store
func DeleteByPrefixStoreIterator(store sdk.KVStore) {
gsk967 marked this conversation as resolved.
Show resolved Hide resolved
iter := sdk.KVStorePrefixIterator(store, nil)
defer iter.Close()
gsk967 marked this conversation as resolved.
Show resolved Hide resolved
for ; iter.Valid(); iter.Next() {
store.Delete(iter.Key())
}

Check warning on line 219 in util/store/store.go

View check run for this annotation

Codecov / codecov/patch

util/store/store.go#L214-L219

Added lines #L214 - L219 were not covered by tests
}
26 changes: 9 additions & 17 deletions x/uibc/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,26 +4,12 @@

The `x/uibc` is a Cosmos Module providing:

- IBC Denom Metadata Tracker for [ICS-20](https://github.com/cosmos/ibc/tree/main/spec/app/ics-020-fungible-token-transfer) transferred tokens to backfill denom metadata into the x/bank standard Cosmos SDK module.
- IBC Quota is an ICS-4 middleware for the ICS-20 token transfer app to apply quota mechanism.

## Content

- [IBC Denom Metadata Tracker](#ibc-denom-metadata-tracker)
- [IBC Quota](#ibc-quota)

## IBC Denom Metadata Tracker

`x/bank.types.Metadata` is a structure which provides essential information about denom, such as display denom name, description, symbol, list of units (unit name and decimal exponent), and the default unit (`Base`).

ICS-20 is a x/bank token transfer protocol over IBC.
The core implementation doesn't create bank `Metadata` when a new token is transferred for the very first time. It's worth to note that token received through IBC is identified by the port ID, channel ID and the source denom ID.
The purpose of the `x/uibc/ics20` module is to wrap the core IBC module and create a denom `Metadata` whenever it is missing. Look at the [`TrackDenomMetadata`](ics20/keeper/keeper.go) function for more details.

### Considerations

The IBC ICS-20 doesn't carry any metadata information, so we only fill up the base denom. Importantly, we don't know about the `Exponent`, and we set `Exponent := 0`. In many cases this is wrong, and should be overwritten by chain governance.

## IBC Quota

Hack or lending abuse is impossible to stop once the funds leave the chain. One mitigation is to limit the IBC inflows and outflows and be able to stop a chain and recover the funds with a migration.
Expand All @@ -43,9 +29,9 @@ All outflows are measured in token average USD value using our x/oracle `AvgKeep

We define 2 Quotas for ICS-20 transfers. Each quota only tracks tokens x/leverage Token Registry.

- `Params.TokenQuota`: upper limit of a sum of all outflows per token. Initially it's set to 0.6m USD per token. It limits the outflows value for each token.
- `Params.TokenQuota`: upper limit of a sum of all outflows per token. It's set to 1.2M USD per token. It limits the outflows value for each token.
NOTE: we measure per token as defined in the x/leverage, not the IBC Denom Path (there can be multiple paths). Since creating a channel is permission less, we want to use same quota token.
- `Params.TotalQuota`: upper limit of a sum of all token outflows combined. Initially it's set to 1m USD. Example of IBC outflows reaching the total quota: 300k USD worth of ATOM, 200k USD worth of STATOM, 250k USD worth of UMEE and 250k USD worth JUNO.
- `Params.TotalQuota`: upper limit of a sum of all token outflows combined. For example if it's set to 1.6M USD then IBC outflows reaching the total quota will be 600k USD worth of ATOM, 500k USD worth of STATOM, 250k USD worth of UMEE and 250k USD worth JUNO.

If a quota parameter is set to zero then we consider it as unlimited.

Expand All @@ -57,7 +43,11 @@ Transfer of tokens, which are not registered in the x/leverage Token Registry ar

#### Inflows

We only allow inflows of tokens registered in x/leverage Token Registry. Other inflow transfers will be rejected.
All inflows are measured in token average USD value using our x/oracle `AvgKeeper`. The `AvgKeeper` aggregates TVWAP prices over 16h window.
We are only tracking inflows for tokens which are registered in x/leverage Token Registry.

- `Genesis.TotalInflowSum` : Sum of all IBC Tokens Inflows which are registered in x/leverage Token Registry.
- `Genesis.Inflows`: IBC Inflow of each registered token.

#### ICS-20 Quota control

Expand All @@ -75,6 +65,8 @@ In the state we store:
- Running sum of total outflow values, serialized as `sdk.Dec`.
- Running sum of per token outflow values, serialized as `sdk.Dec`.
- Next quota expire time (after which the quota reset happens).
- Running sum of total inflow values, serialized as `sdk.Dec`.
- Running sum of per token inflow values, serialized as `sdk.Dec`.

### Messages

Expand Down
15 changes: 15 additions & 0 deletions x/uibc/genesis.go
Original file line number Diff line number Diff line change
Expand Up @@ -18,8 +18,10 @@ func NewGenesisState(params Params, outflows sdk.DecCoins, outflowSum sdk.Dec) *
func DefaultGenesisState() *GenesisState {
return &GenesisState{
Params: DefaultParams(),
Inflows: nil,
Outflows: nil,
TotalOutflowSum: sdk.NewDec(0),
TotalInflowSum: sdk.NewDec(0),
}
}

Expand All @@ -38,9 +40,22 @@ func (gs GenesisState) Validate() error {
}
}

for _, o := range gs.Inflows {
if o.Amount.IsNil() {
return sdkerrors.ErrInvalidRequest.Wrap("ibc denom inflow must be defined")
}
if err := o.Validate(); err != nil {
return err
}
}

if gs.TotalOutflowSum.IsNegative() {
return fmt.Errorf("total outflow sum cannot be negative : %s ", gs.TotalOutflowSum.String())
}

if gs.TotalInflowSum.IsNegative() {
return fmt.Errorf("total inflow sum cannot be negative : %s ", gs.TotalInflowSum.String())
}

return nil
}
165 changes: 136 additions & 29 deletions x/uibc/genesis.pb.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

Loading
Loading