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

fix!: allow safe leverage operations during partial oracle outages #1821

Merged
merged 21 commits into from
Feb 15, 2023
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
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
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,7 @@ Ref: https://keepachangelog.com/en/1.0.0/
- [1812](https://github.com/umee-network/umee/pull/1812) MaxCollateralShare now works during partial oracle outages when certain conditions are safe.
- [1736](https://github.com/umee-network/umee/pull/1736) Blacklisted tokens no longer add themselves back to the oracle accept list.
- [1807](https://github.com/umee-network/umee/pull/1807) Fixes BNB ibc denom in 4.1 migration
- [1821](https://github.com/umee-network/umee/pull/1821) Allow safe leverage operations during partial oracle outages.

## [v4.0.1](https://github.com/umee-network/umee/releases/tag/v4.0.1) - 2023-02-10

Expand Down
11 changes: 10 additions & 1 deletion proto/umee/leverage/v1/query.proto
Original file line number Diff line number Diff line change
Expand Up @@ -222,31 +222,40 @@ message QueryAccountSummary {
// QueryAccountSummaryResponse defines the response structure for the AccountSummary gRPC service handler.
message QueryAccountSummaryResponse {
// Supplied Value is the sum of the USD value of all tokens the account has supplied, including interest earned.
// Computation skips assets which are missing oracle prices, potentially resulting in a lower supplied
// value than if prices were all available.
string supplied_value = 1 [
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
];
// Collateral Value is the sum of the USD value of all uTokens the account has collateralized.
// Computation skips collateral which is missing an oracle price, potentially resulting in a lower collateral
// value than if prices were all available.
string collateral_value = 2 [
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
];
// Borrowed Value is the sum of the USD value of all tokens the account has borrowed, including interest owed.
// It always uses spot prices.
// Computation skips borrows which are missing oracle prices, potentially resulting in a lower borrowed
// value than if prices were all available.
string borrowed_value = 3 [
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
];
// Borrow Limit is the maximum Borrowed Value the account is allowed to reach through direct borrowing.
// The lower of spot or historic price for each collateral token is used when calculating borrow limits.
// Computation skips collateral which is missing an oracle price, potentially resulting in a lower borrow
// limit than if prices were all available.
string borrow_limit = 4 [
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
];
// Liquidation Threshold is the Borrowed Value at which the account becomes eligible for liquidation.
// Will be null if an oracle price required for computation is missing.
string liquidation_threshold = 5 [
(gogoproto.customtype) = "github.com/cosmos/cosmos-sdk/types.Dec",
(gogoproto.nullable) = false
(gogoproto.nullable) = true
];
}

Expand Down
45 changes: 45 additions & 0 deletions swagger/swagger.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -132,18 +132,33 @@ paths:
description: >-
Supplied Value is the sum of the USD value of all tokens the
account has supplied, including interest earned.

Computation skips assets which are missing oracle prices,
potentially resulting in a lower supplied

value than if prices were all available.
collateral_value:
type: string
description: >-
Collateral Value is the sum of the USD value of all uTokens
the account has collateralized.

Computation skips collateral which is missing an oracle price,
potentially resulting in a lower collateral

value than if prices were all available.
borrowed_value:
type: string
description: >-
Borrowed Value is the sum of the USD value of all tokens the
account has borrowed, including interest owed.

It always uses spot prices.

Computation skips borrows which are missing oracle prices,
potentially resulting in a lower borrowed

value than if prices were all available.
borrow_limit:
type: string
description: >-
Expand All @@ -152,11 +167,19 @@ paths:

The lower of spot or historic price for each collateral token
is used when calculating borrow limits.

Computation skips collateral which is missing an oracle price,
potentially resulting in a lower borrow

limit than if prices were all available.
liquidation_threshold:
type: string
description: >-
Liquidation Threshold is the Borrowed Value at which the
account becomes eligible for liquidation.

Will be null if an oracle price required for computation is
missing.
description: >-
QueryAccountSummaryResponse defines the response structure for the
AccountSummary gRPC service handler.
Expand Down Expand Up @@ -2023,18 +2046,33 @@ definitions:
description: >-
Supplied Value is the sum of the USD value of all tokens the account
has supplied, including interest earned.

Computation skips assets which are missing oracle prices, potentially
resulting in a lower supplied

value than if prices were all available.
collateral_value:
type: string
description: >-
Collateral Value is the sum of the USD value of all uTokens the
account has collateralized.

Computation skips collateral which is missing an oracle price,
potentially resulting in a lower collateral

value than if prices were all available.
borrowed_value:
type: string
description: >-
Borrowed Value is the sum of the USD value of all tokens the account
has borrowed, including interest owed.

It always uses spot prices.

Computation skips borrows which are missing oracle prices, potentially
resulting in a lower borrowed

value than if prices were all available.
borrow_limit:
type: string
description: >-
Expand All @@ -2043,11 +2081,18 @@ definitions:

The lower of spot or historic price for each collateral token is used
when calculating borrow limits.

Computation skips collateral which is missing an oracle price,
potentially resulting in a lower borrow

limit than if prices were all available.
liquidation_threshold:
type: string
description: >-
Liquidation Threshold is the Borrowed Value at which the account
becomes eligible for liquidation.

Will be null if an oracle price required for computation is missing.
description: >-
QueryAccountSummaryResponse defines the response structure for the
AccountSummary gRPC service handler.
Expand Down
4 changes: 3 additions & 1 deletion x/leverage/client/tests/tests.go
Original file line number Diff line number Diff line change
Expand Up @@ -252,6 +252,8 @@ func (s *IntegrationTestSuite) TestLeverageScenario() {
nil,
}

lt1 := sdk.MustNewDecFromStr("0.0085610525")

nonzeroQueries := []testQuery{
{
"query account balances",
Expand Down Expand Up @@ -294,7 +296,7 @@ func (s *IntegrationTestSuite) TestLeverageScenario() {
// (1001 / 1000000) * 34.21 * 0.25 = 0.0085610525
BorrowLimit: sdk.MustNewDecFromStr("0.0085610525"),
// (1001 / 1000000) * 0.25 * 34.21 = 0.0085610525
LiquidationThreshold: sdk.MustNewDecFromStr("0.0085610525"),
LiquidationThreshold: &lt1,
},
},
{
Expand Down
45 changes: 43 additions & 2 deletions x/leverage/keeper/borrows.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,9 @@ import (

// assertBorrowerHealth returns an error if a borrower is currently above their borrow limit,
// under either recent (historic median) or current prices. It returns an error if
// prices cannot be calculated.
// borrowed asset prices cannot be calculated, but will try to treat collateral whose prices are
// unavailable as having zero value. This can still result in a borrow limit being too low,
// unless the remaining collateral is enough to cover all borrows.
// This should be checked in msg_server.go at the end of any transaction which is restricted
// by borrow limits, i.e. Borrow, Decollateralize, Withdraw, MaxWithdraw.
func (k Keeper) assertBorrowerHealth(ctx sdk.Context, borrowerAddr sdk.AccAddress) error {
Expand All @@ -20,7 +22,7 @@ func (k Keeper) assertBorrowerHealth(ctx sdk.Context, borrowerAddr sdk.AccAddres
if err != nil {
return err
}
limit, err := k.CalculateBorrowLimit(ctx, collateral)
limit, err := k.VisibleBorrowLimit(ctx, collateral)
if err != nil {
return err
}
Expand Down Expand Up @@ -127,6 +129,45 @@ func (k Keeper) CalculateBorrowLimit(ctx sdk.Context, collateral sdk.Coins) (sdk
return limit, nil
}

// VisibleBorrowLimit uses the price oracle to determine the borrow limit (in USD) provided by
// collateral sdk.Coins, using each token's uToken exchange rate and collateral weight.
// The lower of spot price or historic price is used for each collateral token.
// An error is returned if any input coins are not uTokens.
// This function skips assets that are missing prices, which will lead to a lower borrow
// limit when prices are down instead of a complete loss of borrowing ability.
func (k Keeper) VisibleBorrowLimit(ctx sdk.Context, collateral sdk.Coins) (sdk.Dec, error) {
limit := sdk.ZeroDec()

for _, coin := range collateral {
// convert uToken collateral to base assets
baseAsset, err := k.ExchangeUToken(ctx, coin)
if err != nil {
return sdk.ZeroDec(), err
}

ts, err := k.GetTokenSettings(ctx, baseAsset.Denom)
if err != nil {
return sdk.ZeroDec(), err
}

// ignore blacklisted tokens
if !ts.Blacklist {
// get USD value of base assets using the chosen price mode
v, err := k.TokenValue(ctx, baseAsset, types.PriceModeLow)
if err == nil {
// if both spot and historic (if required) prices exist,
// add collateral coin's weighted value to borrow limit
limit = limit.Add(v.Mul(ts.CollateralWeight))
}
if nonOracleError(err) {
return sdk.ZeroDec(), err
}
}
}

return limit, nil
}

// CalculateLiquidationThreshold determines the maximum borrowed value (in USD) that a
// borrower with given collateral could reach before being eligible for liquidation, using
// each token's oracle price, uToken exchange rate, and liquidation threshold.
Expand Down
16 changes: 6 additions & 10 deletions x/leverage/keeper/collateral.go
Original file line number Diff line number Diff line change
Expand Up @@ -94,17 +94,13 @@ func (k Keeper) VisibleCollateralValue(ctx sdk.Context, collateral sdk.Coins) (s

// get USD value of base assets
v, err := k.TokenValue(ctx, baseAsset, types.PriceModeSpot)
if err != nil {
k.Logger(ctx).Info(
"collateral value skipped",
"uToken", coin.String(),
"error", err.Error(),
)
continue
if err == nil {
// for coins that did not error, add their value to the total
total = total.Add(v)
}
if nonOracleError(err) {
return sdk.ZeroDec(), err
}

// for coins that did not error, add their value to the total
total = total.Add(v)
}

return total, nil
Expand Down
33 changes: 33 additions & 0 deletions x/leverage/keeper/errors.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,33 @@
package keeper

import (
"strings"

"cosmossdk.io/errors"

"github.com/umee-network/umee/v4/util/decmath"
leveragetypes "github.com/umee-network/umee/v4/x/leverage/types"
oracletypes "github.com/umee-network/umee/v4/x/oracle/types"
)

// nonOracleError returns true if an error is non-nil
// and also not one of ErrEmptyList, ErrUnknownDenom, or ErrNoHistoricMedians
// which are errors which can result from missing prices
func nonOracleError(err error) bool {
if err == nil {
return false
}
// check typed errors
if errors.IsOf(err,
leveragetypes.ErrInvalidOraclePrice,
leveragetypes.ErrNoHistoricMedians,
oracletypes.ErrUnknownDenom,
) {
return false
}
// this error needs to be checked by string comparison
if strings.Contains(err.Error(), decmath.ErrEmptyList.Error()) {
return false
}
return true
}
38 changes: 38 additions & 0 deletions x/leverage/keeper/errors_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
package keeper

import (
"testing"

"github.com/stretchr/testify/require"

"cosmossdk.io/errors"

"github.com/umee-network/umee/v4/util/decmath"
leveragetypes "github.com/umee-network/umee/v4/x/leverage/types"
oracletypes "github.com/umee-network/umee/v4/x/oracle/types"
)

func TestErrorMatching(t *testing.T) {
// oracle errors
err1 := errors.Wrap(decmath.ErrEmptyList, "denom: UMEE")
err2 := oracletypes.ErrUnknownDenom.Wrap("UMEE")
err3 := leveragetypes.ErrNoHistoricMedians.Wrapf(
"requested %d, got %d",
16,
12,
)
// not oracle errors
err4 := leveragetypes.ErrBlacklisted
err5 := leveragetypes.ErrUToken
err6 := leveragetypes.ErrNotRegisteredToken
err7 := errors.New("foo", 1, "bar")

require.Equal(t, false, nonOracleError(nil))
require.Equal(t, false, nonOracleError(err1))
require.Equal(t, false, nonOracleError(err2))
require.Equal(t, false, nonOracleError(err3))
require.Equal(t, true, nonOracleError(err4))
require.Equal(t, true, nonOracleError(err5))
require.Equal(t, true, nonOracleError(err6))
require.Equal(t, true, nonOracleError(err7))
}
Loading