package keeper_test import ( "math/rand" "sort" "strings" "testing" "time" "github.com/stretchr/testify/suite" "github.com/cosmos/cosmos-sdk/codec" sdk "github.com/cosmos/cosmos-sdk/types" "github.com/cosmos/cosmos-sdk/types/simulation" authtypes "github.com/cosmos/cosmos-sdk/x/auth/types" abci "github.com/tendermint/tendermint/abci/types" tmproto "github.com/tendermint/tendermint/proto/tendermint/types" tmtime "github.com/tendermint/tendermint/types/time" "github.com/0glabs/0g-chain/app" "github.com/0glabs/0g-chain/x/cdp/keeper" "github.com/0glabs/0g-chain/x/cdp/types" pfkeeper "github.com/0glabs/0g-chain/x/pricefeed/keeper" pftypes "github.com/0glabs/0g-chain/x/pricefeed/types" ) const ( custom = "custom" ) type QuerierTestSuite struct { suite.Suite keeper keeper.Keeper pricefeedKeeper pfkeeper.Keeper addrs []sdk.AccAddress app app.TestApp cdps types.CDPs augmentedCDPs types.AugmentedCDPs ctx sdk.Context querier sdk.Querier cdc codec.Codec legacyAmino codec.LegacyAmino } func (suite *QuerierTestSuite) SetupTest() { tApp := app.NewTestApp() ctx := tApp.NewContext(true, tmproto.Header{Height: 1, Time: tmtime.Now()}) suite.cdc = tApp.AppCodec() suite.legacyAmino = *tApp.LegacyAmino() cdps := make(types.CDPs, 100) augmentedCDPs := make(types.AugmentedCDPs, 100) _, addrs := app.GeneratePrivKeyAddressPairs(100) coins := cs(c("btc", 10000000000), c("xrp", 10000000000)) authGS := app.NewFundedGenStateWithSameCoins(tApp.AppCodec(), coins, addrs) tApp.InitializeFromGenesisStates( authGS, NewPricefeedGenStateMulti(suite.cdc), NewCDPGenStateHighDebtLimit(suite.cdc), ) suite.ctx = ctx suite.app = tApp suite.keeper = tApp.GetCDPKeeper() suite.pricefeedKeeper = tApp.GetPriceFeedKeeper() // Set up markets oracle := addrs[9] marketParams := pftypes.Params{ Markets: pftypes.Markets{ pftypes.Market{MarketID: "xrp-usd", BaseAsset: "xrp", QuoteAsset: "usd", Oracles: []sdk.AccAddress{oracle}, Active: true}, pftypes.Market{MarketID: "btc-usd", BaseAsset: "btc", QuoteAsset: "usd", Oracles: []sdk.AccAddress{oracle}, Active: true}, }, } suite.pricefeedKeeper.SetParams(ctx, marketParams) // Set collateral prices for use in collateralization calculations _, err := suite.pricefeedKeeper.SetPrice( ctx, oracle, "xrp-usd", sdk.MustNewDecFromStr("0.75"), time.Now().Add(1*time.Hour)) suite.Nil(err) _, err = suite.pricefeedKeeper.SetPrice( ctx, oracle, "btc-usd", sdk.MustNewDecFromStr("5000"), time.Now().Add(1*time.Hour)) suite.Nil(err) for j := 0; j < 100; j++ { collateral := "xrp" amount := simulation.RandIntBetween(rand.New(rand.NewSource(int64(j))), 2500000000, 9000000000) debt := simulation.RandIntBetween(rand.New(rand.NewSource(int64(j))), 50000000, 250000000) if j%2 == 0 { collateral = "btc" amount = simulation.RandIntBetween(rand.New(rand.NewSource(int64(j))), 500000000, 5000000000) debt = simulation.RandIntBetween(rand.New(rand.NewSource(int64(j))), 1000000000, 25000000000) } err = suite.keeper.AddCdp(suite.ctx, addrs[j], c(collateral, int64(amount)), c("usdx", int64(debt)), collateral+"-a") suite.NoError(err) c, f := suite.keeper.GetCDP(suite.ctx, collateral+"-a", uint64(j+1)) suite.True(f) cdps[j] = c aCDP := suite.keeper.LoadAugmentedCDP(suite.ctx, c) augmentedCDPs[j] = aCDP } suite.cdps = cdps suite.augmentedCDPs = augmentedCDPs suite.querier = keeper.NewQuerier(suite.keeper, tApp.LegacyAmino()) suite.addrs = addrs } func (suite *QuerierTestSuite) TestQueryCdp() { ctx := suite.ctx.WithIsCheckTx(false) query := abci.RequestQuery{ Path: strings.Join([]string{custom, types.QuerierRoute, types.QueryGetCdp}, "/"), Data: suite.legacyAmino.MustMarshalJSON(types.NewQueryCdpParams(suite.cdps[0].Owner, suite.cdps[0].Type)), } bz, err := suite.querier(ctx, []string{types.QueryGetCdp}, query) suite.Nil(err) suite.NotNil(bz) var c types.AugmentedCDP suite.Nil(suite.legacyAmino.UnmarshalJSON(bz, &c)) suite.Equal(suite.augmentedCDPs[0], c) query = abci.RequestQuery{ Path: strings.Join([]string{custom, types.QuerierRoute, types.QueryGetCdp}, "/"), Data: suite.legacyAmino.MustMarshalJSON(types.NewQueryCdpParams(suite.cdps[0].Owner, "lol-a")), } _, err = suite.querier(ctx, []string{types.QueryGetCdp}, query) suite.Error(err) query = abci.RequestQuery{ Path: strings.Join([]string{custom, "nonsense"}, "/"), Data: []byte("nonsense"), } _, err = suite.querier(ctx, []string{query.Path}, query) suite.Error(err) _, err = suite.querier(ctx, []string{types.QueryGetCdp}, query) suite.Error(err) query = abci.RequestQuery{ Path: strings.Join([]string{custom, types.QuerierRoute, types.QueryGetCdp}, "/"), Data: suite.legacyAmino.MustMarshalJSON(types.NewQueryCdpParams(suite.cdps[0].Owner, "xrp-a")), } _, err = suite.querier(ctx, []string{types.QueryGetCdp}, query) suite.Error(err) } func (suite *QuerierTestSuite) TestQueryCdpsByCollateralType() { ctx := suite.ctx.WithIsCheckTx(false) query := abci.RequestQuery{ Path: strings.Join([]string{custom, types.QuerierRoute, types.QueryGetCdpsByCollateralType}, "/"), Data: suite.legacyAmino.MustMarshalJSON(types.NewQueryCdpsByCollateralTypeParams(suite.cdps[0].Type)), } bz, err := suite.querier(ctx, []string{types.QueryGetCdpsByCollateralType}, query) suite.Nil(err) suite.NotNil(bz) var c types.AugmentedCDPs suite.Nil(suite.legacyAmino.UnmarshalJSON(bz, &c)) suite.Equal(50, len(c)) query = abci.RequestQuery{ Path: strings.Join([]string{custom, types.QuerierRoute, types.QueryGetCdpsByCollateralType}, "/"), Data: suite.legacyAmino.MustMarshalJSON(types.NewQueryCdpsByCollateralTypeParams("lol-a")), } _, err = suite.querier(ctx, []string{types.QueryGetCdpsByCollateralType}, query) suite.Error(err) } func (suite *QuerierTestSuite) TestQueryCdpsByRatio() { ratioCountBtc := 0 ratioCountXrp := 0 xrpRatio := d("2.0") btcRatio := d("2500") expectedXrpIds := []int{} expectedBtcIds := []int{} for _, cdp := range suite.cdps { absoluteRatio := suite.keeper.CalculateCollateralToDebtRatio(suite.ctx, cdp.Collateral, cdp.Type, cdp.Principal) collateralizationRatio, err := suite.keeper.CalculateCollateralizationRatioFromAbsoluteRatio(suite.ctx, cdp.Type, absoluteRatio, "liquidation") suite.Nil(err) if cdp.Collateral.Denom == "xrp" { if collateralizationRatio.LT(xrpRatio) { ratioCountXrp += 1 expectedXrpIds = append(expectedXrpIds, int(cdp.ID)) } } else { if collateralizationRatio.LT(btcRatio) { ratioCountBtc += 1 expectedBtcIds = append(expectedBtcIds, int(cdp.ID)) } } } ctx := suite.ctx.WithIsCheckTx(false) query := abci.RequestQuery{ Path: strings.Join([]string{custom, types.QuerierRoute, types.QueryGetCdpsByCollateralization}, "/"), Data: suite.legacyAmino.MustMarshalJSON(types.NewQueryCdpsByRatioParams("xrp-a", xrpRatio)), } bz, err := suite.querier(ctx, []string{types.QueryGetCdpsByCollateralization}, query) suite.Nil(err) suite.NotNil(bz) var c types.AugmentedCDPs actualXrpIds := []int{} suite.Nil(suite.legacyAmino.UnmarshalJSON(bz, &c)) for _, k := range c { actualXrpIds = append(actualXrpIds, int(k.ID)) } sort.Ints(actualXrpIds) suite.Equal(expectedXrpIds, actualXrpIds) query = abci.RequestQuery{ Path: strings.Join([]string{custom, types.QuerierRoute, types.QueryGetCdpsByCollateralization}, "/"), Data: suite.legacyAmino.MustMarshalJSON(types.NewQueryCdpsByRatioParams("btc-a", btcRatio)), } bz, err = suite.querier(ctx, []string{types.QueryGetCdpsByCollateralization}, query) suite.Nil(err) suite.NotNil(bz) c = types.AugmentedCDPs{} actualBtcIds := []int{} suite.Nil(suite.legacyAmino.UnmarshalJSON(bz, &c)) for _, k := range c { actualBtcIds = append(actualBtcIds, int(k.ID)) } sort.Ints(actualBtcIds) suite.Equal(expectedBtcIds, actualBtcIds) query = abci.RequestQuery{ Path: strings.Join([]string{custom, types.QuerierRoute, types.QueryGetCdpsByCollateralization}, "/"), Data: suite.legacyAmino.MustMarshalJSON(types.NewQueryCdpsByRatioParams("xrp-a", d("0.003"))), } bz, err = suite.querier(ctx, []string{types.QueryGetCdpsByCollateralization}, query) suite.Nil(err) suite.NotNil(bz) c = types.AugmentedCDPs{} suite.Nil(suite.legacyAmino.UnmarshalJSON(bz, &c)) suite.Equal(0, len(c)) } func (suite *QuerierTestSuite) TestQueryParams() { ctx := suite.ctx.WithIsCheckTx(false) bz, err := suite.querier(ctx, []string{types.QueryGetParams}, abci.RequestQuery{}) suite.NoError(err) suite.NotNil(bz) var p types.Params err = suite.legacyAmino.UnmarshalJSON(bz, &p) suite.NoError(err) cdpGS := NewCDPGenStateHighDebtLimit(suite.app.AppCodec()) gs := types.GenesisState{} err = suite.legacyAmino.UnmarshalJSON(cdpGS["cdp"], &gs) suite.NoError(err) suite.Equal(gs.Params, p) } func (suite *QuerierTestSuite) TestQueryDeposits() { ctx := suite.ctx.WithIsCheckTx(false) query := abci.RequestQuery{ Path: strings.Join([]string{custom, types.QuerierRoute, types.QueryGetCdpDeposits}, "/"), Data: suite.legacyAmino.MustMarshalJSON(types.NewQueryCdpDeposits(suite.cdps[0].Owner, suite.cdps[0].Type)), } bz, err := suite.querier(ctx, []string{types.QueryGetCdpDeposits}, query) suite.Nil(err) suite.NotNil(bz) deposits := suite.keeper.GetDeposits(ctx, suite.cdps[0].ID) var d types.Deposits suite.Nil(suite.legacyAmino.UnmarshalJSON(bz, &d)) suite.Equal(deposits, d) } func (suite *QuerierTestSuite) TestQueryAccounts() { bz, err := suite.querier(suite.ctx, []string{types.QueryGetAccounts}, abci.RequestQuery{}) suite.Require().NoError(err) suite.Require().NotNil(bz) var accounts []authtypes.ModuleAccount suite.Require().Nil(suite.legacyAmino.UnmarshalJSON(bz, &accounts)) suite.Require().Equal(2, len(accounts)) findByName := func(name string) bool { for _, account := range accounts { if account.GetName() == name { return true } } return false } suite.Require().True(findByName("cdp")) suite.Require().True(findByName("liquidator")) } func (suite *QuerierTestSuite) TestFindIntersection() { a := types.CDPs{suite.cdps[0], suite.cdps[1], suite.cdps[2], suite.cdps[3], suite.cdps[4]} b := types.CDPs{suite.cdps[3], suite.cdps[4], suite.cdps[5], suite.cdps[6], suite.cdps[7]} expectedIntersection1 := types.CDPs{suite.cdps[3], suite.cdps[4]} intersection1 := keeper.FindIntersection(a, b) suite.Require().Equal(expectedIntersection1, intersection1) c := types.CDPs{suite.cdps[0], suite.cdps[1], suite.cdps[2], suite.cdps[3], suite.cdps[4]} d := types.CDPs{suite.cdps[5], suite.cdps[6], suite.cdps[7], suite.cdps[8], suite.cdps[9]} expectedIntersection2 := types.CDPs{} intersection2 := keeper.FindIntersection(c, d) suite.Require().Equal(expectedIntersection2, intersection2) e := types.CDPs{suite.cdps[0]} f := types.CDPs{} expectedIntersection3 := types.CDPs{} intersection3 := keeper.FindIntersection(e, f) suite.Require().Equal(expectedIntersection3, intersection3) } func (suite *QuerierTestSuite) TestFilterCDPs() { paramsType := types.NewQueryCdpsParams(1, 100, "btc-a", sdk.AccAddress{}, 0, sdk.ZeroDec()) filteredCDPs1, err := keeper.FilterCDPs(suite.ctx, suite.keeper, paramsType) suite.Require().NoError(err) suite.Require().Equal(50, len(filteredCDPs1)) paramsOwner := types.NewQueryCdpsParams(1, 100, "", suite.cdps[10].Owner, 0, sdk.ZeroDec()) filteredCDPs2, err := keeper.FilterCDPs(suite.ctx, suite.keeper, paramsOwner) suite.Require().NoError(err) suite.Require().Equal(1, len(filteredCDPs2)) suite.Require().Equal(suite.cdps[10].Owner, filteredCDPs2[0].Owner) paramsID := types.NewQueryCdpsParams(1, 100, "", sdk.AccAddress{}, 68, sdk.ZeroDec()) filteredCDPs3, err := keeper.FilterCDPs(suite.ctx, suite.keeper, paramsID) suite.Require().NoError(err) suite.Require().Equal(1, len(filteredCDPs3)) suite.Require().Equal(suite.cdps[68-1].ID, filteredCDPs3[0].ID) ratioCountBtc := 0 btcRatio := d("2500") for _, cdp := range suite.cdps { if cdp.Type == "btc-a" { absoluteRatio := suite.keeper.CalculateCollateralToDebtRatio(suite.ctx, cdp.Collateral, cdp.Type, cdp.Principal) collateralizationRatio, _ := suite.keeper.CalculateCollateralizationRatioFromAbsoluteRatio(suite.ctx, cdp.Type, absoluteRatio, "liquidation") if collateralizationRatio.LT(btcRatio) { ratioCountBtc += 1 } } } paramsTypeAndRatio := types.NewQueryCdpsParams(1, 100, "btc-a", sdk.AccAddress{}, 0, sdk.NewDec(2500)) filteredCDPs4, err := keeper.FilterCDPs(suite.ctx, suite.keeper, paramsTypeAndRatio) suite.Require().NoError(err) suite.Require().Equal(ratioCountBtc, len(filteredCDPs4)) } func (suite *QuerierTestSuite) TestQueryCdps() { ctx := suite.ctx.WithIsCheckTx(false) params := types.NewQueryCdpsParams(1, 100, "btc-a", sdk.AccAddress{}, 0, sdk.ZeroDec()) query := abci.RequestQuery{ Path: strings.Join([]string{custom, types.QuerierRoute, types.QueryGetCdps}, "/"), Data: suite.legacyAmino.MustMarshalJSON(params), } bz, err := suite.querier(ctx, []string{types.QueryGetCdps}, query) suite.Nil(err) suite.NotNil(bz) output := types.AugmentedCDPs{} suite.Nil(suite.legacyAmino.UnmarshalJSON(bz, &output)) suite.Equal(50, len(output)) } func (suite *QuerierTestSuite) TestQueryTotalPrincipal() { ctx := suite.ctx.WithIsCheckTx(false) params := types.NewQueryGetTotalPrincipalParams("btc-a") query := abci.RequestQuery{ Path: strings.Join([]string{custom, types.QuerierRoute, types.QueryGetTotalPrincipal}, "/"), Data: suite.legacyAmino.MustMarshalJSON(params), } bz, err := suite.querier(ctx, []string{types.QueryGetTotalPrincipal}, query) suite.Nil(err) suite.NotNil(bz) var output types.TotalPrincipals suite.Nil(suite.legacyAmino.UnmarshalJSON(bz, &output)) suite.Equal(1, len(output)) suite.Equal("btc-a", output[0].CollateralType) } func (suite *QuerierTestSuite) TestQueryTotalPrincipalAll() { ctx := suite.ctx.WithIsCheckTx(false) params := types.NewQueryGetTotalPrincipalParams("") query := abci.RequestQuery{ Path: strings.Join([]string{custom, types.QuerierRoute, types.QueryGetTotalPrincipal}, "/"), Data: suite.legacyAmino.MustMarshalJSON(params), } bz, err := suite.querier(ctx, []string{types.QueryGetTotalPrincipal}, query) suite.Nil(err) suite.NotNil(bz) var output types.TotalPrincipals suite.Nil(suite.legacyAmino.UnmarshalJSON(bz, &output)) var outputTypes []string for _, c := range output { outputTypes = append(outputTypes, c.CollateralType) } suite.Greater(len(output), 0) suite.Subset(outputTypes, []string{"btc-a", "xrp-a"}) } func (suite *QuerierTestSuite) TestQueryTotalCollateral() { ctx := suite.ctx.WithIsCheckTx(false) params := types.NewQueryGetTotalCollateralParams("btc-a") query := abci.RequestQuery{ Path: strings.Join([]string{custom, types.QuerierRoute, types.QueryGetTotalCollateral}, "/"), Data: suite.legacyAmino.MustMarshalJSON(params), } bz, err := suite.querier(ctx, []string{types.QueryGetTotalCollateral}, query) suite.Nil(err) suite.NotNil(bz) var output types.TotalCollaterals suite.Nil(suite.legacyAmino.UnmarshalJSON(bz, &output)) suite.Equal(1, len(output)) suite.Equal("btc-a", output[0].CollateralType) } func (suite *QuerierTestSuite) TestQueryTotalCollateralAll() { ctx := suite.ctx.WithIsCheckTx(false) params := types.NewQueryGetTotalCollateralParams("") query := abci.RequestQuery{ Path: strings.Join([]string{custom, types.QuerierRoute, types.QueryGetTotalCollateral}, "/"), Data: suite.legacyAmino.MustMarshalJSON(params), } bz, err := suite.querier(ctx, []string{types.QueryGetTotalCollateral}, query) suite.Nil(err) suite.NotNil(bz) var output types.TotalCollaterals suite.Nil(suite.legacyAmino.UnmarshalJSON(bz, &output)) var outputTypes []string for _, c := range output { outputTypes = append(outputTypes, c.CollateralType) } suite.Greater(len(output), 0) suite.Subset(outputTypes, []string{"btc-a", "xrp-a"}) } func TestQuerierTestSuite(t *testing.T) { suite.Run(t, new(QuerierTestSuite)) }