2021-06-21 21:05:17 +00:00
|
|
|
package keeper_test
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2022-09-23 16:38:22 +00:00
|
|
|
"strings"
|
2021-07-22 12:53:18 +00:00
|
|
|
"time"
|
2021-06-21 21:05:17 +00:00
|
|
|
|
2023-04-05 23:21:59 +00:00
|
|
|
sdkmath "cosmossdk.io/math"
|
2024-05-01 03:17:24 +00:00
|
|
|
pricefeedtypes "github.com/0glabs/0g-chain/x/pricefeed/types"
|
2024-02-06 22:54:10 +00:00
|
|
|
db "github.com/cometbft/cometbft-db"
|
|
|
|
"github.com/cometbft/cometbft/libs/log"
|
2021-06-21 21:05:17 +00:00
|
|
|
"github.com/cosmos/cosmos-sdk/codec"
|
|
|
|
"github.com/cosmos/cosmos-sdk/store"
|
2023-04-04 00:08:45 +00:00
|
|
|
storetypes "github.com/cosmos/cosmos-sdk/store/types"
|
2021-06-21 21:05:17 +00:00
|
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
2023-01-26 16:50:27 +00:00
|
|
|
minttypes "github.com/cosmos/cosmos-sdk/x/mint/types"
|
2022-01-08 00:39:27 +00:00
|
|
|
paramtypes "github.com/cosmos/cosmos-sdk/x/params/types"
|
2021-07-22 12:53:18 +00:00
|
|
|
stakingtypes "github.com/cosmos/cosmos-sdk/x/staking/types"
|
2021-06-21 21:05:17 +00:00
|
|
|
"github.com/stretchr/testify/suite"
|
|
|
|
|
2024-05-01 03:17:24 +00:00
|
|
|
"github.com/0glabs/0g-chain/app"
|
|
|
|
cdptypes "github.com/0glabs/0g-chain/x/cdp/types"
|
|
|
|
earntypes "github.com/0glabs/0g-chain/x/earn/types"
|
|
|
|
tmprototypes "github.com/tendermint/tendermint/proto/tendermint/types"
|
2022-01-08 00:39:27 +00:00
|
|
|
|
2024-05-01 03:17:24 +00:00
|
|
|
hardtypes "github.com/0glabs/0g-chain/x/hard/types"
|
|
|
|
"github.com/0glabs/0g-chain/x/incentive/keeper"
|
|
|
|
"github.com/0glabs/0g-chain/x/incentive/types"
|
2021-06-21 21:05:17 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// NewTestContext sets up a basic context with an in-memory db
|
2023-04-04 00:08:45 +00:00
|
|
|
func NewTestContext(requiredStoreKeys ...storetypes.StoreKey) sdk.Context {
|
2021-06-21 21:05:17 +00:00
|
|
|
memDB := db.NewMemDB()
|
|
|
|
cms := store.NewCommitMultiStore(memDB)
|
|
|
|
|
|
|
|
for _, key := range requiredStoreKeys {
|
2023-04-04 00:08:45 +00:00
|
|
|
cms.MountStoreWithDB(key, storetypes.StoreTypeIAVL, nil)
|
2021-06-21 21:05:17 +00:00
|
|
|
}
|
|
|
|
|
2022-01-08 00:39:27 +00:00
|
|
|
if err := cms.LoadLatestVersion(); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return sdk.NewContext(cms, tmprototypes.Header{}, false, log.NewNopLogger())
|
2021-06-21 21:05:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// unitTester is a wrapper around suite.Suite, with common functionality for keeper unit tests.
|
|
|
|
// It can be embedded in structs the same way as suite.Suite.
|
|
|
|
type unitTester struct {
|
|
|
|
suite.Suite
|
|
|
|
keeper keeper.Keeper
|
|
|
|
ctx sdk.Context
|
|
|
|
|
2022-01-08 00:39:27 +00:00
|
|
|
cdc codec.Codec
|
2023-04-04 00:08:45 +00:00
|
|
|
incentiveStoreKey storetypes.StoreKey
|
2021-06-21 21:05:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (suite *unitTester) SetupSuite() {
|
2022-09-28 02:46:14 +00:00
|
|
|
tApp := app.NewTestApp()
|
|
|
|
suite.cdc = tApp.AppCodec()
|
2021-06-21 21:05:17 +00:00
|
|
|
|
|
|
|
suite.incentiveStoreKey = sdk.NewKVStoreKey(types.StoreKey)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (suite *unitTester) SetupTest() {
|
|
|
|
suite.ctx = NewTestContext(suite.incentiveStoreKey)
|
2022-09-23 16:38:22 +00:00
|
|
|
suite.keeper = suite.NewKeeper(&fakeParamSubspace{}, nil, nil, nil, nil, nil, nil, nil, nil, nil)
|
2021-06-21 21:05:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (suite *unitTester) TearDownTest() {
|
|
|
|
suite.keeper = keeper.Keeper{}
|
|
|
|
suite.ctx = sdk.Context{}
|
|
|
|
}
|
|
|
|
|
2022-09-23 16:38:22 +00:00
|
|
|
func (suite *unitTester) NewKeeper(
|
|
|
|
paramSubspace types.ParamSubspace,
|
|
|
|
bk types.BankKeeper, cdpk types.CdpKeeper, hk types.HardKeeper,
|
|
|
|
ak types.AccountKeeper, stk types.StakingKeeper, swk types.SwapKeeper,
|
|
|
|
svk types.SavingsKeeper, lqk types.LiquidKeeper, ek types.EarnKeeper,
|
|
|
|
) keeper.Keeper {
|
2022-10-05 18:39:50 +00:00
|
|
|
return keeper.NewKeeper(
|
|
|
|
suite.cdc, suite.incentiveStoreKey, paramSubspace,
|
|
|
|
bk, cdpk, hk, ak, stk, swk, svk, lqk, ek,
|
|
|
|
nil, nil, nil,
|
|
|
|
)
|
2021-06-21 21:05:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (suite *unitTester) storeGlobalBorrowIndexes(indexes types.MultiRewardIndexes) {
|
|
|
|
for _, i := range indexes {
|
|
|
|
suite.keeper.SetHardBorrowRewardIndexes(suite.ctx, i.CollateralType, i.RewardIndexes)
|
|
|
|
}
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-06-21 21:05:17 +00:00
|
|
|
func (suite *unitTester) storeGlobalSupplyIndexes(indexes types.MultiRewardIndexes) {
|
|
|
|
for _, i := range indexes {
|
|
|
|
suite.keeper.SetHardSupplyRewardIndexes(suite.ctx, i.CollateralType, i.RewardIndexes)
|
|
|
|
}
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-07-07 13:23:06 +00:00
|
|
|
func (suite *unitTester) storeGlobalDelegatorIndexes(multiRewardIndexes types.MultiRewardIndexes) {
|
|
|
|
// Hardcoded to use bond denom
|
|
|
|
multiRewardIndex, _ := multiRewardIndexes.GetRewardIndex(types.BondDenom)
|
2021-07-07 16:50:14 +00:00
|
|
|
suite.keeper.SetDelegatorRewardIndexes(suite.ctx, types.BondDenom, multiRewardIndex.RewardIndexes)
|
2021-07-07 13:23:06 +00:00
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-07-07 13:23:06 +00:00
|
|
|
func (suite *unitTester) storeGlobalSwapIndexes(indexes types.MultiRewardIndexes) {
|
|
|
|
for _, i := range indexes {
|
|
|
|
suite.keeper.SetSwapRewardIndexes(suite.ctx, i.CollateralType, i.RewardIndexes)
|
|
|
|
}
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2022-04-21 14:19:03 +00:00
|
|
|
func (suite *unitTester) storeGlobalSavingsIndexes(indexes types.MultiRewardIndexes) {
|
|
|
|
for _, i := range indexes {
|
|
|
|
suite.keeper.SetSavingsRewardIndexes(suite.ctx, i.CollateralType, i.RewardIndexes)
|
|
|
|
}
|
|
|
|
}
|
2021-06-21 21:05:17 +00:00
|
|
|
|
2022-09-22 18:26:08 +00:00
|
|
|
func (suite *unitTester) storeGlobalEarnIndexes(indexes types.MultiRewardIndexes) {
|
|
|
|
for _, i := range indexes {
|
|
|
|
suite.keeper.SetEarnRewardIndexes(suite.ctx, i.CollateralType, i.RewardIndexes)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-07-07 16:50:14 +00:00
|
|
|
func (suite *unitTester) storeHardClaim(claim types.HardLiquidityProviderClaim) {
|
2021-06-21 21:05:17 +00:00
|
|
|
suite.keeper.SetHardLiquidityProviderClaim(suite.ctx, claim)
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-07-07 16:50:14 +00:00
|
|
|
func (suite *unitTester) storeDelegatorClaim(claim types.DelegatorClaim) {
|
|
|
|
suite.keeper.SetDelegatorClaim(suite.ctx, claim)
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
Swap users accumulate rewards (#950)
* add swap claim type
* add store methods for swap claims
* add swap claims to genesis state
* tidy up cdp and hard hook doc comments
* add uncalled hooks to the swap keeper
* add swap rewards sync method
* add swap rewards init method
* connect swap rewards via swap hooks
* Update querier and client for swap claims (#951)
* refactor querier to dedupe code
* add swap claims querier endpoint
* add swap claim querying to rest
* add swap claim querying to cli
* add keeper method to iterate swap reward indexes
* simplify reward-factors query endpoint, add swap
* update swap hook to match latest swap branch
* rename func to not collide with latest swap branch
* Squash merge swap-acceptance branch (#956)
* add failing acceptance test for a user depositing into a pool
* implement GetAccount test helper
* implement swap.MsgDeposit for creating and adding liquidity to a pool
* update aliases, add event types, and fix typo/compiler errors in handler
test
* use only aliases names in handler test (don't use swap types -- ensures
we have run aliasgen), add assertion for even type message
* implement account and module account balance checks in handler test
* fill out handler assertions for testing keeper state and events
* update signed json representation and register swap/MsgDeposit for
proper encoding
* fill out boilerplate to get handler test to compile
* alias gen for pool
* add handling of message type; fill in deposit keeper method for
succesful compile; noop but test assertions now run up to module acc not
nil check
* add module account permissions for swap module -- fixes module account
creation; pass account keeper and supply keeper into swap keeper to
allow the ability to work with user and module accounts
* implement create pool logic for msg deposit; allows creation of a of new
pool, checking params to see if it is allowed. Initi shares are set,
and senders number of shares are stored
* Swap migrations scaffolding (#925)
* swap module scaffolding
* global swap fee
* can't think of a reason for begin blocker; removing for abci.go for now;
* test pair types; refactor pair name logic; simplify pairs validation and
fix stack overflow error
* check comparison
* use test package
* init swap module genesis
* add basic marshall tests
* remove reward apy from pairs
* fix integration helpers
* use max swap fee constant; fix validation of swap fee; add tests to
cover param validation and param set setup
* use noerror over nil
* start genesis tests
* test param set validation mirrors param validation
* add genesis tests
* remove print statement
* add subtests for genesis test cases; add extra querier test for unknown
route; add keeper params testing
* add spec
* update swagger
* find replace hard -> swap in comments
* remove unused method
* rename pairs to allowed pools; pool is more commonly used, and
allowedPool makes it more clear what swap parameter is for. In
addition, we won't conflict with Pool data structure for storing a
created pool in the store.
* remove generated link
* missed spec rename
* validate token order for allowed pools
* fix swagger
* json should be snakecase; change allowedPools to allowed_pools
* add legacy types
* add swap genesis to v0_15 migration
* add legacy types
* add swap genesis to v0_15 migration
* migration revisions
Co-authored-by: Nick DeLuca <nickdeluca08@gmail.com>
* keeper todos
* update keeper tests
* type todos
* update types tests
* tx deposit cli cmd
* tx deposit rest
* Swap module simulation scaffolding (#924)
* sims scaffolding
* add noop operation
* genesis revisions
* add param changes
* mvoe persistance methods to main keeper file, consolidate tests
* make helper methods private. they are tested via deposit method, and
unit testing them would make test suite brittle and refactoring
difficult
* use more clear coin variables
* code 1 is reserved, use code 2 and sequence all errors
* remove todo
* Implement deadline for swap module module message. This is implemented in
handler with a interface to easily apply to it to all messages, and
separate msg validation concerns from the keeper
* move allowed pools to params -- let pool and pool_test focus on pool domain logic, not
parameter & governance concerns
* update alias
* add unitless implementatin of constant product liquidity pool to
isolate and enapsulate liquidity logic. Swap methods interfaces are
added, but implementation not yet added
* nits and todos
* add ErrInvalidPool
* add tests for edge cases around pool depletion; add explicit panic for
edge case that results in a pool reserve being zero; handle pool
reinitialization if it is empty
* touch up comments and flush out the rest of assertions
* add data structures for keeper state storage separate from pool domain
objects, and improve structure for easier querying
* rename pool name to pool key for events
* add support for a denominated pool that uses sdk.Coins and sdk.Coin
arguments, keeping tracking of the units in the base pool. This gives
nice separation between pool logic, and coin/denom logic
* refactor keeper to use new records for storage, and implement pool
deposit using the denominated pool
* address previous PR comment - reminder for migration if changing
account permissions
* msg deposit should validate that denoms are not equal
* add godoc comments
* golint and some poolName -> poolID cleanup
* implement adding liquidity to an existing pool
* hardcode pools in sims
* touch up comment
* withdraw keeper logic
* withdraw type updates
* add withdraw msg tx handler
* initial withdraw test
* fix panic
* use new denominated pool with existing shares
* fix: check args on deposit cmd
* add slippage limit check for depositing to an existing pool
* send coins just before event emission
* check liquidity returned is greater than zero for both coins; ensure
returned number of shares are greater than zero
* add deadline to msgwithdraw
* register msgwithdraw
* scaffold msgwithdraw types test
* register the correct msg
* modify swap functions to also return the amount paid for the pool swap
fee. This will be used to calculate slippage and for event tracking
* add slippage types
* add expected withdrawal coins
* calculate slippage against expected coins
* update withdraw keeper tests
* spelling, improve comments on add liquidity math
* typo
* typo
* grammer
* typo / grammer
* remove pool_id from withdraw msg
* add slippage to tx cmd
* TestWithdraw_Partial
* nit
* add withdraw no pool, no deposit record tests
* drop event check on partial withdraw test
* fix broken link
* fix broken link
* resolve merge conflicts
* ensure swap fee can not be equal to 1; add full implementation of swap
pool methods; these implementation ensure that the pool invariant is
always greater or equal to the previous invariant
* refactor duplicated code into private swap methods
* add runtime assertion to always ensure invariant is greater or equal
to the previous invariant sub fee on swaps
* improve comments for base pool swap functions
* add swap exact input and output methods to denominated pool that wrap
the base pool interface for swapping
* comment touch ups
* more comment touchups
* fix msg deposit struct tag (#943)
* use better name for swap calculation private methods
* nits: golint
* fix misspelling in method name
* Add HARD token governance committee for Hard module (#941)
* add hard gov token committee
* revisions: update migration
* revisions: update test/data file
* initial revisions
* add TokenCommittee JSONMarshal test
* fix SetPermissions method
* remove BaseCommittee Type field
* add incentive params to allowed params
* Add SWP token governance committee for Swap module (#946)
* add swp token commitee to migration
* update test, add gen export utility method
* final revisions: add TODO
* remove slippage from withdraw to use min values for coins; add
additional validation test cases
* update alias for swap module
* add withdraw tests to handler for increased coverage; note: first pass,
improvements still yet to be made here
* refact withdraw keeper to use min amounts; panic for cases that do not
happen in normal situations
* lint fixes
* use total shares to track if pool should be deleted; add more in depth
withdraw comment
* add exact args for withdraw cmd
* extract record update methods
* update depositor share record if it exists -- do not overwrite an
existing record; ensures no loss of shares if the same address deposits
more than once
* Swap queries: deposit, pool, pools (#949)
* query deposits types
* implement deposit querier keeper methods
* query deposits CLI
* query deposits REST
* query types for pool/pools
* pool/pools querier keeper methods
* pool/pools CLI
* pool/pools REST
* basic pool/pools query tests
* basic deposit querier test
* iterate share records via owner bytes
* nit: add example for querying deposits by owner only
Co-authored-by: karzak <kjydavis3@gmail.com>
* feat: add REST tx handler for swap LP withdrawals
Co-authored-by: Nick DeLuca <nickdeluca08@gmail.com>
Co-authored-by: Denali Marsh <denali@kava.io>
Co-authored-by: denalimarsh <denalimarsh@gmail.com>
Co-authored-by: karzak <kjydavis3@gmail.com>
* expand incentive cli query docs
Co-authored-by: Nick DeLuca <nickdeluca08@gmail.com>
Co-authored-by: Denali Marsh <denali@kava.io>
Co-authored-by: denalimarsh <denalimarsh@gmail.com>
Co-authored-by: karzak <kjydavis3@gmail.com>
* minor update to godoc comment
Co-authored-by: Nick DeLuca <nickdeluca08@gmail.com>
Co-authored-by: Denali Marsh <denali@kava.io>
Co-authored-by: denalimarsh <denalimarsh@gmail.com>
Co-authored-by: karzak <kjydavis3@gmail.com>
2021-07-13 12:35:02 +00:00
|
|
|
func (suite *unitTester) storeSwapClaim(claim types.SwapClaim) {
|
|
|
|
suite.keeper.SetSwapClaim(suite.ctx, claim)
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2022-04-21 14:19:03 +00:00
|
|
|
func (suite *unitTester) storeSavingsClaim(claim types.SavingsClaim) {
|
|
|
|
suite.keeper.SetSavingsClaim(suite.ctx, claim)
|
|
|
|
}
|
2021-07-07 16:50:14 +00:00
|
|
|
|
2022-09-22 18:26:08 +00:00
|
|
|
func (suite *unitTester) storeEarnClaim(claim types.EarnClaim) {
|
|
|
|
suite.keeper.SetEarnClaim(suite.ctx, claim)
|
|
|
|
}
|
|
|
|
|
2022-10-05 18:39:50 +00:00
|
|
|
type TestKeeperBuilder struct {
|
|
|
|
cdc codec.Codec
|
2023-04-04 00:08:45 +00:00
|
|
|
key storetypes.StoreKey
|
2022-10-05 18:39:50 +00:00
|
|
|
paramSubspace types.ParamSubspace
|
|
|
|
accountKeeper types.AccountKeeper
|
|
|
|
bankKeeper types.BankKeeper
|
|
|
|
cdpKeeper types.CdpKeeper
|
|
|
|
hardKeeper types.HardKeeper
|
|
|
|
stakingKeeper types.StakingKeeper
|
|
|
|
swapKeeper types.SwapKeeper
|
|
|
|
savingsKeeper types.SavingsKeeper
|
|
|
|
liquidKeeper types.LiquidKeeper
|
|
|
|
earnKeeper types.EarnKeeper
|
|
|
|
|
|
|
|
// Keepers used for APY queries
|
2023-01-26 16:50:27 +00:00
|
|
|
mintKeeper types.MintKeeper
|
2022-10-05 18:39:50 +00:00
|
|
|
distrKeeper types.DistrKeeper
|
|
|
|
pricefeedKeeper types.PricefeedKeeper
|
|
|
|
}
|
|
|
|
|
|
|
|
func (suite *unitTester) NewTestKeeper(
|
|
|
|
paramSubspace types.ParamSubspace,
|
|
|
|
) *TestKeeperBuilder {
|
|
|
|
if !paramSubspace.HasKeyTable() {
|
|
|
|
paramSubspace = paramSubspace.WithKeyTable(types.ParamKeyTable())
|
|
|
|
}
|
|
|
|
|
|
|
|
return &TestKeeperBuilder{
|
|
|
|
cdc: suite.cdc,
|
|
|
|
key: suite.incentiveStoreKey,
|
|
|
|
paramSubspace: paramSubspace,
|
|
|
|
accountKeeper: nil,
|
|
|
|
bankKeeper: nil,
|
|
|
|
cdpKeeper: nil,
|
|
|
|
hardKeeper: nil,
|
|
|
|
stakingKeeper: nil,
|
|
|
|
swapKeeper: nil,
|
|
|
|
savingsKeeper: nil,
|
|
|
|
liquidKeeper: nil,
|
|
|
|
earnKeeper: nil,
|
2023-01-26 16:50:27 +00:00
|
|
|
mintKeeper: nil,
|
2022-10-05 18:39:50 +00:00
|
|
|
distrKeeper: nil,
|
|
|
|
pricefeedKeeper: nil,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tk *TestKeeperBuilder) WithPricefeedKeeper(k types.PricefeedKeeper) *TestKeeperBuilder {
|
|
|
|
tk.pricefeedKeeper = k
|
|
|
|
return tk
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tk *TestKeeperBuilder) WithDistrKeeper(k types.DistrKeeper) *TestKeeperBuilder {
|
|
|
|
tk.distrKeeper = k
|
|
|
|
return tk
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tk *TestKeeperBuilder) WithBankKeeper(k types.BankKeeper) *TestKeeperBuilder {
|
|
|
|
tk.bankKeeper = k
|
|
|
|
return tk
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tk *TestKeeperBuilder) WithStakingKeeper(k types.StakingKeeper) *TestKeeperBuilder {
|
|
|
|
tk.stakingKeeper = k
|
|
|
|
return tk
|
|
|
|
}
|
|
|
|
|
2023-01-26 16:50:27 +00:00
|
|
|
func (tk *TestKeeperBuilder) WithMintKeeper(k types.MintKeeper) *TestKeeperBuilder {
|
|
|
|
tk.mintKeeper = k
|
2022-10-05 18:39:50 +00:00
|
|
|
return tk
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tk *TestKeeperBuilder) WithEarnKeeper(k types.EarnKeeper) *TestKeeperBuilder {
|
|
|
|
tk.earnKeeper = k
|
|
|
|
return tk
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tk *TestKeeperBuilder) WithLiquidKeeper(k types.LiquidKeeper) *TestKeeperBuilder {
|
|
|
|
tk.liquidKeeper = k
|
|
|
|
return tk
|
|
|
|
}
|
|
|
|
|
|
|
|
func (tk *TestKeeperBuilder) Build() keeper.Keeper {
|
|
|
|
return keeper.NewKeeper(
|
|
|
|
tk.cdc, tk.key, tk.paramSubspace,
|
|
|
|
tk.bankKeeper, tk.cdpKeeper, tk.hardKeeper, tk.accountKeeper,
|
|
|
|
tk.stakingKeeper, tk.swapKeeper, tk.savingsKeeper, tk.liquidKeeper,
|
2023-01-26 16:50:27 +00:00
|
|
|
tk.earnKeeper, tk.mintKeeper, tk.distrKeeper, tk.pricefeedKeeper,
|
2022-10-05 18:39:50 +00:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2021-07-22 12:53:18 +00:00
|
|
|
// fakeParamSubspace is a stub paramSpace to simplify keeper unit test setup.
|
2021-06-21 21:05:17 +00:00
|
|
|
type fakeParamSubspace struct {
|
|
|
|
params types.Params
|
|
|
|
}
|
|
|
|
|
2022-01-08 00:39:27 +00:00
|
|
|
func (subspace *fakeParamSubspace) GetParamSet(_ sdk.Context, ps paramtypes.ParamSet) {
|
2021-06-21 21:05:17 +00:00
|
|
|
*(ps.(*types.Params)) = subspace.params
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2022-01-08 00:39:27 +00:00
|
|
|
func (subspace *fakeParamSubspace) SetParamSet(_ sdk.Context, ps paramtypes.ParamSet) {
|
2021-06-21 21:05:17 +00:00
|
|
|
subspace.params = *(ps.(*types.Params))
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-06-21 21:05:17 +00:00
|
|
|
func (subspace *fakeParamSubspace) HasKeyTable() bool {
|
2021-07-07 13:23:06 +00:00
|
|
|
// return true so the keeper does not try to call WithKeyTable, which does nothing
|
2021-06-21 21:05:17 +00:00
|
|
|
return true
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2022-01-08 00:39:27 +00:00
|
|
|
func (subspace *fakeParamSubspace) WithKeyTable(paramtypes.KeyTable) paramtypes.Subspace {
|
2021-06-21 21:05:17 +00:00
|
|
|
// return an non-functional subspace to satisfy the interface
|
2022-01-08 00:39:27 +00:00
|
|
|
return paramtypes.Subspace{}
|
2021-06-21 21:05:17 +00:00
|
|
|
}
|
|
|
|
|
2021-07-22 12:53:18 +00:00
|
|
|
// fakeSwapKeeper is a stub swap keeper.
|
|
|
|
// It can be used to return values to the incentive keeper without having to initialize a full swap keeper.
|
|
|
|
type fakeSwapKeeper struct {
|
2023-04-05 23:21:59 +00:00
|
|
|
poolShares map[string]sdkmath.Int
|
|
|
|
depositShares map[string](map[string]sdkmath.Int)
|
2021-07-22 12:53:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var _ types.SwapKeeper = newFakeSwapKeeper()
|
|
|
|
|
|
|
|
func newFakeSwapKeeper() *fakeSwapKeeper {
|
|
|
|
return &fakeSwapKeeper{
|
2023-04-05 23:21:59 +00:00
|
|
|
poolShares: map[string]sdkmath.Int{},
|
|
|
|
depositShares: map[string](map[string]sdkmath.Int){},
|
2021-07-22 12:53:18 +00:00
|
|
|
}
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2023-04-05 23:21:59 +00:00
|
|
|
func (k *fakeSwapKeeper) addPool(id string, shares sdkmath.Int) *fakeSwapKeeper {
|
2021-07-22 12:53:18 +00:00
|
|
|
k.poolShares[id] = shares
|
|
|
|
return k
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2023-04-05 23:21:59 +00:00
|
|
|
func (k *fakeSwapKeeper) addDeposit(poolID string, depositor sdk.AccAddress, shares sdkmath.Int) *fakeSwapKeeper {
|
2021-07-26 20:07:18 +00:00
|
|
|
if k.depositShares[poolID] == nil {
|
2023-04-05 23:21:59 +00:00
|
|
|
k.depositShares[poolID] = map[string]sdkmath.Int{}
|
2021-07-26 20:07:18 +00:00
|
|
|
}
|
|
|
|
k.depositShares[poolID][depositor.String()] = shares
|
|
|
|
return k
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2023-04-05 23:21:59 +00:00
|
|
|
func (k *fakeSwapKeeper) GetPoolShares(_ sdk.Context, poolID string) (sdkmath.Int, bool) {
|
2021-07-22 12:53:18 +00:00
|
|
|
shares, ok := k.poolShares[poolID]
|
|
|
|
return shares, ok
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2023-04-05 23:21:59 +00:00
|
|
|
func (k *fakeSwapKeeper) GetDepositorSharesAmount(_ sdk.Context, depositor sdk.AccAddress, poolID string) (sdkmath.Int, bool) {
|
2021-07-26 20:07:18 +00:00
|
|
|
shares, found := k.depositShares[poolID][depositor.String()]
|
|
|
|
return shares, found
|
2021-07-22 12:53:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// fakeHardKeeper is a stub hard keeper.
|
|
|
|
// It can be used to return values to the incentive keeper without having to initialize a full hard keeper.
|
|
|
|
type fakeHardKeeper struct {
|
|
|
|
borrows fakeHardState
|
|
|
|
deposits fakeHardState
|
|
|
|
}
|
|
|
|
|
|
|
|
type fakeHardState struct {
|
|
|
|
total sdk.Coins
|
|
|
|
interestFactors map[string]sdk.Dec
|
|
|
|
}
|
|
|
|
|
|
|
|
func newFakeHardState() fakeHardState {
|
|
|
|
return fakeHardState{
|
|
|
|
total: nil,
|
|
|
|
interestFactors: map[string]sdk.Dec{}, // initialize map to avoid panics on read
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ types.HardKeeper = newFakeHardKeeper()
|
|
|
|
|
|
|
|
func newFakeHardKeeper() *fakeHardKeeper {
|
|
|
|
return &fakeHardKeeper{
|
|
|
|
borrows: newFakeHardState(),
|
|
|
|
deposits: newFakeHardState(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeHardKeeper) addTotalBorrow(coin sdk.Coin, factor sdk.Dec) *fakeHardKeeper {
|
|
|
|
k.borrows.total = k.borrows.total.Add(coin)
|
|
|
|
k.borrows.interestFactors[coin.Denom] = factor
|
|
|
|
return k
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-07-22 12:53:18 +00:00
|
|
|
func (k *fakeHardKeeper) addTotalSupply(coin sdk.Coin, factor sdk.Dec) *fakeHardKeeper {
|
|
|
|
k.deposits.total = k.deposits.total.Add(coin)
|
|
|
|
k.deposits.interestFactors[coin.Denom] = factor
|
|
|
|
return k
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeHardKeeper) GetBorrowedCoins(_ sdk.Context) (sdk.Coins, bool) {
|
|
|
|
if k.borrows.total == nil {
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
return k.borrows.total, true
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-07-22 12:53:18 +00:00
|
|
|
func (k *fakeHardKeeper) GetSuppliedCoins(_ sdk.Context) (sdk.Coins, bool) {
|
|
|
|
if k.deposits.total == nil {
|
|
|
|
return nil, false
|
|
|
|
}
|
|
|
|
return k.deposits.total, true
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-07-22 12:53:18 +00:00
|
|
|
func (k *fakeHardKeeper) GetBorrowInterestFactor(_ sdk.Context, denom string) (sdk.Dec, bool) {
|
|
|
|
f, ok := k.borrows.interestFactors[denom]
|
|
|
|
return f, ok
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-07-22 12:53:18 +00:00
|
|
|
func (k *fakeHardKeeper) GetSupplyInterestFactor(_ sdk.Context, denom string) (sdk.Dec, bool) {
|
|
|
|
f, ok := k.deposits.interestFactors[denom]
|
|
|
|
return f, ok
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-07-22 12:53:18 +00:00
|
|
|
func (k *fakeHardKeeper) GetBorrow(_ sdk.Context, _ sdk.AccAddress) (hardtypes.Borrow, bool) {
|
|
|
|
panic("unimplemented")
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-07-22 12:53:18 +00:00
|
|
|
func (k *fakeHardKeeper) GetDeposit(_ sdk.Context, _ sdk.AccAddress) (hardtypes.Deposit, bool) {
|
|
|
|
panic("unimplemented")
|
|
|
|
}
|
|
|
|
|
|
|
|
// fakeStakingKeeper is a stub staking keeper.
|
|
|
|
// It can be used to return values to the incentive keeper without having to initialize a full staking keeper.
|
|
|
|
type fakeStakingKeeper struct {
|
|
|
|
delegations stakingtypes.Delegations
|
|
|
|
validators stakingtypes.Validators
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ types.StakingKeeper = newFakeStakingKeeper()
|
|
|
|
|
|
|
|
func newFakeStakingKeeper() *fakeStakingKeeper { return &fakeStakingKeeper{} }
|
|
|
|
|
|
|
|
func (k *fakeStakingKeeper) addBondedTokens(amount int64) *fakeStakingKeeper {
|
|
|
|
if len(k.validators) != 0 {
|
|
|
|
panic("cannot set total bonded if keeper already has validators set")
|
|
|
|
}
|
|
|
|
// add a validator with all the tokens
|
|
|
|
k.validators = append(k.validators, stakingtypes.Validator{
|
2022-01-08 00:39:27 +00:00
|
|
|
Status: stakingtypes.Bonded,
|
2023-04-05 23:21:59 +00:00
|
|
|
Tokens: sdkmath.NewInt(amount),
|
2021-07-22 12:53:18 +00:00
|
|
|
})
|
|
|
|
return k
|
|
|
|
}
|
|
|
|
|
2023-04-05 23:21:59 +00:00
|
|
|
func (k *fakeStakingKeeper) TotalBondedTokens(_ sdk.Context) sdkmath.Int {
|
2021-07-22 12:53:18 +00:00
|
|
|
total := sdk.ZeroInt()
|
|
|
|
for _, val := range k.validators {
|
2022-01-08 00:39:27 +00:00
|
|
|
if val.GetStatus() == stakingtypes.Bonded {
|
2021-07-22 12:53:18 +00:00
|
|
|
total = total.Add(val.GetBondedTokens())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return total
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-07-22 12:53:18 +00:00
|
|
|
func (k *fakeStakingKeeper) GetDelegatorDelegations(_ sdk.Context, delegator sdk.AccAddress, maxRetrieve uint16) []stakingtypes.Delegation {
|
|
|
|
return k.delegations
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-07-22 12:53:18 +00:00
|
|
|
func (k *fakeStakingKeeper) GetValidator(_ sdk.Context, addr sdk.ValAddress) (stakingtypes.Validator, bool) {
|
|
|
|
for _, val := range k.validators {
|
|
|
|
if val.GetOperator().Equals(addr) {
|
|
|
|
return val, true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return stakingtypes.Validator{}, false
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-07-22 12:53:18 +00:00
|
|
|
func (k *fakeStakingKeeper) GetValidatorDelegations(_ sdk.Context, valAddr sdk.ValAddress) []stakingtypes.Delegation {
|
|
|
|
var delegations stakingtypes.Delegations
|
|
|
|
for _, d := range k.delegations {
|
2022-01-08 00:39:27 +00:00
|
|
|
if d.GetValidatorAddr().Equals(valAddr) {
|
2021-07-22 12:53:18 +00:00
|
|
|
delegations = append(delegations, d)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return delegations
|
|
|
|
}
|
|
|
|
|
|
|
|
// fakeCDPKeeper is a stub cdp keeper.
|
|
|
|
// It can be used to return values to the incentive keeper without having to initialize a full cdp keeper.
|
|
|
|
type fakeCDPKeeper struct {
|
|
|
|
interestFactor *sdk.Dec
|
2023-04-05 23:21:59 +00:00
|
|
|
totalPrincipal sdkmath.Int
|
2021-07-22 12:53:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var _ types.CdpKeeper = newFakeCDPKeeper()
|
|
|
|
|
|
|
|
func newFakeCDPKeeper() *fakeCDPKeeper {
|
|
|
|
return &fakeCDPKeeper{
|
|
|
|
interestFactor: nil,
|
|
|
|
totalPrincipal: sdk.ZeroInt(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeCDPKeeper) addInterestFactor(f sdk.Dec) *fakeCDPKeeper {
|
|
|
|
k.interestFactor = &f
|
|
|
|
return k
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2023-04-05 23:21:59 +00:00
|
|
|
func (k *fakeCDPKeeper) addTotalPrincipal(p sdkmath.Int) *fakeCDPKeeper {
|
2021-07-22 12:53:18 +00:00
|
|
|
k.totalPrincipal = p
|
|
|
|
return k
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeCDPKeeper) GetInterestFactor(_ sdk.Context, collateralType string) (sdk.Dec, bool) {
|
|
|
|
if k.interestFactor != nil {
|
|
|
|
return *k.interestFactor, true
|
|
|
|
}
|
|
|
|
return sdk.Dec{}, false
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2023-04-05 23:21:59 +00:00
|
|
|
func (k *fakeCDPKeeper) GetTotalPrincipal(_ sdk.Context, collateralType string, principalDenom string) sdkmath.Int {
|
2021-07-22 12:53:18 +00:00
|
|
|
return k.totalPrincipal
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-07-22 12:53:18 +00:00
|
|
|
func (k *fakeCDPKeeper) GetCdpByOwnerAndCollateralType(_ sdk.Context, owner sdk.AccAddress, collateralType string) (cdptypes.CDP, bool) {
|
|
|
|
return cdptypes.CDP{}, false
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-07-22 12:53:18 +00:00
|
|
|
func (k *fakeCDPKeeper) GetCollateral(_ sdk.Context, collateralType string) (cdptypes.CollateralParam, bool) {
|
|
|
|
return cdptypes.CollateralParam{}, false
|
|
|
|
}
|
|
|
|
|
2022-09-22 18:26:08 +00:00
|
|
|
// fakeEarnKeeper is a stub earn keeper.
|
|
|
|
// It can be used to return values to the incentive keeper without having to initialize a full earn keeper.
|
|
|
|
type fakeEarnKeeper struct {
|
|
|
|
vaultShares map[string]earntypes.VaultShare
|
|
|
|
depositShares map[string]earntypes.VaultShares
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ types.EarnKeeper = newFakeEarnKeeper()
|
|
|
|
|
|
|
|
func newFakeEarnKeeper() *fakeEarnKeeper {
|
|
|
|
return &fakeEarnKeeper{
|
|
|
|
vaultShares: map[string]earntypes.VaultShare{},
|
|
|
|
depositShares: map[string]earntypes.VaultShares{},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeEarnKeeper) addVault(vaultDenom string, shares earntypes.VaultShare) *fakeEarnKeeper {
|
|
|
|
k.vaultShares[vaultDenom] = shares
|
|
|
|
return k
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeEarnKeeper) addDeposit(
|
|
|
|
depositor sdk.AccAddress,
|
|
|
|
shares earntypes.VaultShare,
|
|
|
|
) *fakeEarnKeeper {
|
|
|
|
if k.depositShares[depositor.String()] == nil {
|
|
|
|
k.depositShares[depositor.String()] = earntypes.NewVaultShares()
|
|
|
|
}
|
|
|
|
|
|
|
|
k.depositShares[depositor.String()] = k.depositShares[depositor.String()].Add(shares)
|
|
|
|
|
|
|
|
return k
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeEarnKeeper) GetVaultTotalShares(
|
|
|
|
ctx sdk.Context,
|
|
|
|
denom string,
|
|
|
|
) (shares earntypes.VaultShare, found bool) {
|
|
|
|
vaultShares, found := k.vaultShares[denom]
|
|
|
|
return vaultShares, found
|
|
|
|
}
|
|
|
|
|
2022-10-05 18:39:50 +00:00
|
|
|
func (k *fakeEarnKeeper) GetVaultTotalValue(ctx sdk.Context, denom string) (sdk.Coin, error) {
|
|
|
|
vaultShares, found := k.vaultShares[denom]
|
|
|
|
if !found {
|
|
|
|
return sdk.NewCoin(denom, sdk.ZeroInt()), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return sdk.NewCoin(denom, vaultShares.Amount.RoundInt()), nil
|
|
|
|
}
|
|
|
|
|
2022-09-22 18:26:08 +00:00
|
|
|
func (k *fakeEarnKeeper) GetVaultAccountShares(
|
|
|
|
ctx sdk.Context,
|
|
|
|
acc sdk.AccAddress,
|
|
|
|
) (shares earntypes.VaultShares, found bool) {
|
|
|
|
accShares, found := k.depositShares[acc.String()]
|
|
|
|
return accShares, found
|
|
|
|
}
|
|
|
|
|
2022-09-23 16:38:22 +00:00
|
|
|
func (k *fakeEarnKeeper) IterateVaultRecords(
|
|
|
|
ctx sdk.Context,
|
|
|
|
cb func(record earntypes.VaultRecord) (stop bool),
|
|
|
|
) {
|
|
|
|
for _, vaultShares := range k.vaultShares {
|
|
|
|
cb(earntypes.VaultRecord{
|
|
|
|
TotalShares: vaultShares,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// fakeLiquidKeeper is a stub liquid keeper.
|
|
|
|
// It can be used to return values to the incentive keeper without having to initialize a full liquid keeper.
|
|
|
|
type fakeLiquidKeeper struct {
|
2023-04-05 23:21:59 +00:00
|
|
|
derivatives map[string]sdkmath.Int
|
2022-09-28 20:20:01 +00:00
|
|
|
lastRewardClaim map[string]time.Time
|
2022-09-23 16:38:22 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var _ types.LiquidKeeper = newFakeLiquidKeeper()
|
|
|
|
|
|
|
|
func newFakeLiquidKeeper() *fakeLiquidKeeper {
|
|
|
|
return &fakeLiquidKeeper{
|
2023-04-05 23:21:59 +00:00
|
|
|
derivatives: map[string]sdkmath.Int{},
|
2022-09-28 20:20:01 +00:00
|
|
|
lastRewardClaim: map[string]time.Time{},
|
2022-09-23 16:38:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-28 20:20:01 +00:00
|
|
|
func (k *fakeLiquidKeeper) addDerivative(
|
|
|
|
ctx sdk.Context,
|
|
|
|
denom string,
|
2023-04-05 23:21:59 +00:00
|
|
|
supply sdkmath.Int,
|
2022-09-28 20:20:01 +00:00
|
|
|
) *fakeLiquidKeeper {
|
2022-09-23 16:38:22 +00:00
|
|
|
k.derivatives[denom] = supply
|
2022-09-28 20:20:01 +00:00
|
|
|
k.lastRewardClaim[denom] = ctx.BlockTime()
|
2022-09-23 16:38:22 +00:00
|
|
|
return k
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeLiquidKeeper) IsDerivativeDenom(ctx sdk.Context, denom string) bool {
|
|
|
|
return strings.HasPrefix(denom, "bkava-")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeLiquidKeeper) GetAllDerivativeDenoms(ctx sdk.Context) (denoms []string) {
|
|
|
|
for denom := range k.derivatives {
|
|
|
|
denoms = append(denoms, denom)
|
|
|
|
}
|
|
|
|
|
|
|
|
return denoms
|
|
|
|
}
|
|
|
|
|
2022-09-28 20:20:01 +00:00
|
|
|
func (k *fakeLiquidKeeper) GetTotalDerivativeValue(ctx sdk.Context) (sdk.Coin, error) {
|
2022-09-23 16:38:22 +00:00
|
|
|
totalSupply := sdk.ZeroInt()
|
|
|
|
for _, supply := range k.derivatives {
|
|
|
|
totalSupply = totalSupply.Add(supply)
|
|
|
|
}
|
|
|
|
|
2022-09-28 20:20:01 +00:00
|
|
|
return sdk.NewCoin("ukava", totalSupply), nil
|
2022-09-23 16:38:22 +00:00
|
|
|
}
|
|
|
|
|
2022-09-28 20:20:01 +00:00
|
|
|
func (k *fakeLiquidKeeper) GetDerivativeValue(ctx sdk.Context, denom string) (sdk.Coin, error) {
|
2022-09-23 16:38:22 +00:00
|
|
|
supply, found := k.derivatives[denom]
|
|
|
|
if !found {
|
2022-09-28 20:20:01 +00:00
|
|
|
return sdk.NewCoin("ukava", sdk.ZeroInt()), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return sdk.NewCoin("ukava", supply), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeLiquidKeeper) CollectStakingRewardsByDenom(
|
|
|
|
ctx sdk.Context,
|
|
|
|
derivativeDenom string,
|
|
|
|
destinationModAccount string,
|
|
|
|
) (sdk.Coins, error) {
|
|
|
|
amt := k.getRewardAmount(ctx, derivativeDenom)
|
|
|
|
|
|
|
|
return sdk.NewCoins(sdk.NewCoin("ukava", amt)), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeLiquidKeeper) getRewardAmount(
|
|
|
|
ctx sdk.Context,
|
|
|
|
derivativeDenom string,
|
2023-04-05 23:21:59 +00:00
|
|
|
) sdkmath.Int {
|
2022-09-28 20:20:01 +00:00
|
|
|
amt, found := k.derivatives[derivativeDenom]
|
|
|
|
if !found {
|
|
|
|
// No error
|
|
|
|
return sdk.ZeroInt()
|
|
|
|
}
|
|
|
|
|
|
|
|
lastRewardClaim, found := k.lastRewardClaim[derivativeDenom]
|
|
|
|
if !found {
|
|
|
|
panic("last reward claim not found")
|
|
|
|
}
|
|
|
|
|
|
|
|
duration := int64(ctx.BlockTime().Sub(lastRewardClaim).Seconds())
|
|
|
|
if duration <= 0 {
|
2022-09-23 16:38:22 +00:00
|
|
|
return sdk.ZeroInt()
|
|
|
|
}
|
|
|
|
|
2022-09-28 20:20:01 +00:00
|
|
|
// Reward amount just set to 10% of the derivative supply per second
|
|
|
|
return amt.QuoRaw(10).MulRaw(duration)
|
2022-09-23 16:38:22 +00:00
|
|
|
}
|
|
|
|
|
2022-10-05 18:39:50 +00:00
|
|
|
type fakeDistrKeeper struct {
|
|
|
|
communityTax sdk.Dec
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ types.DistrKeeper = newFakeDistrKeeper()
|
|
|
|
|
|
|
|
func newFakeDistrKeeper() *fakeDistrKeeper {
|
|
|
|
return &fakeDistrKeeper{}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeDistrKeeper) setCommunityTax(percent sdk.Dec) *fakeDistrKeeper {
|
|
|
|
k.communityTax = percent
|
|
|
|
return k
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeDistrKeeper) GetCommunityTax(ctx sdk.Context) (percent sdk.Dec) {
|
|
|
|
return k.communityTax
|
|
|
|
}
|
|
|
|
|
2023-01-26 16:50:27 +00:00
|
|
|
type fakeMintKeeper struct {
|
|
|
|
minter minttypes.Minter
|
2022-10-05 18:39:50 +00:00
|
|
|
}
|
|
|
|
|
2023-01-26 16:50:27 +00:00
|
|
|
var _ types.MintKeeper = newFakeMintKeeper()
|
2022-10-05 18:39:50 +00:00
|
|
|
|
2023-01-26 16:50:27 +00:00
|
|
|
func newFakeMintKeeper() *fakeMintKeeper {
|
|
|
|
return &fakeMintKeeper{}
|
2022-10-05 18:39:50 +00:00
|
|
|
}
|
|
|
|
|
2023-01-26 16:50:27 +00:00
|
|
|
func (k *fakeMintKeeper) setMinter(minter minttypes.Minter) *fakeMintKeeper {
|
|
|
|
k.minter = minter
|
2022-10-05 18:39:50 +00:00
|
|
|
return k
|
|
|
|
}
|
|
|
|
|
2023-01-26 16:50:27 +00:00
|
|
|
func (k *fakeMintKeeper) GetMinter(ctx sdk.Context) (minter minttypes.Minter) {
|
|
|
|
return k.minter
|
2022-10-05 18:39:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type fakePricefeedKeeper struct {
|
|
|
|
prices map[string]pricefeedtypes.CurrentPrice
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ types.PricefeedKeeper = newFakePricefeedKeeper()
|
|
|
|
|
|
|
|
func newFakePricefeedKeeper() *fakePricefeedKeeper {
|
|
|
|
return &fakePricefeedKeeper{
|
|
|
|
prices: map[string]pricefeedtypes.CurrentPrice{},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakePricefeedKeeper) setPrice(price pricefeedtypes.CurrentPrice) *fakePricefeedKeeper {
|
|
|
|
k.prices[price.MarketID] = price
|
|
|
|
return k
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakePricefeedKeeper) GetCurrentPrice(ctx sdk.Context, marketID string) (pricefeedtypes.CurrentPrice, error) {
|
|
|
|
price, found := k.prices[marketID]
|
|
|
|
if !found {
|
|
|
|
return pricefeedtypes.CurrentPrice{}, fmt.Errorf("price not found for market %s", marketID)
|
|
|
|
}
|
|
|
|
|
|
|
|
return price, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type fakeBankKeeper struct {
|
2023-04-05 23:21:59 +00:00
|
|
|
supply map[string]sdkmath.Int
|
2022-10-05 18:39:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var _ types.BankKeeper = newFakeBankKeeper()
|
|
|
|
|
|
|
|
func newFakeBankKeeper() *fakeBankKeeper {
|
|
|
|
return &fakeBankKeeper{
|
2023-04-05 23:21:59 +00:00
|
|
|
supply: map[string]sdkmath.Int{},
|
2022-10-05 18:39:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeBankKeeper) setSupply(coins ...sdk.Coin) *fakeBankKeeper {
|
|
|
|
for _, coin := range coins {
|
|
|
|
k.supply[coin.Denom] = coin.Amount
|
|
|
|
}
|
|
|
|
|
|
|
|
return k
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeBankKeeper) SendCoinsFromModuleToAccount(
|
|
|
|
ctx sdk.Context,
|
|
|
|
senderModule string,
|
|
|
|
recipientAddr sdk.AccAddress,
|
|
|
|
amt sdk.Coins,
|
|
|
|
) error {
|
|
|
|
panic("not implemented")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeBankKeeper) GetAllBalances(ctx sdk.Context, addr sdk.AccAddress) sdk.Coins {
|
|
|
|
panic("not implemented")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *fakeBankKeeper) GetSupply(ctx sdk.Context, denom string) sdk.Coin {
|
|
|
|
supply, found := k.supply[denom]
|
|
|
|
if !found {
|
|
|
|
return sdk.NewCoin(denom, sdk.ZeroInt())
|
|
|
|
}
|
|
|
|
|
|
|
|
return sdk.NewCoin(denom, supply)
|
|
|
|
}
|
|
|
|
|
2021-07-22 12:53:18 +00:00
|
|
|
// Assorted Testing Data
|
|
|
|
|
|
|
|
// note: amino panics when encoding times ≥ the start of year 10000.
|
|
|
|
var distantFuture = time.Date(9000, 1, 1, 0, 0, 0, 0, time.UTC)
|
|
|
|
|
2021-06-21 21:05:17 +00:00
|
|
|
func arbitraryCoins() sdk.Coins {
|
|
|
|
return cs(c("btcb", 1))
|
|
|
|
}
|
|
|
|
|
|
|
|
func arbitraryAddress() sdk.AccAddress {
|
|
|
|
_, addresses := app.GeneratePrivKeyAddressPairs(1)
|
|
|
|
return addresses[0]
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-06-21 21:05:17 +00:00
|
|
|
func arbitraryValidatorAddress() sdk.ValAddress {
|
|
|
|
return generateValidatorAddresses(1)[0]
|
|
|
|
}
|
2022-05-09 18:37:36 +00:00
|
|
|
|
2021-06-21 21:05:17 +00:00
|
|
|
func generateValidatorAddresses(n int) []sdk.ValAddress {
|
|
|
|
_, addresses := app.GeneratePrivKeyAddressPairs(n)
|
|
|
|
var valAddresses []sdk.ValAddress
|
|
|
|
for _, a := range addresses {
|
|
|
|
valAddresses = append(valAddresses, sdk.ValAddress(a))
|
|
|
|
}
|
|
|
|
return valAddresses
|
|
|
|
}
|
|
|
|
|
|
|
|
var nonEmptyMultiRewardIndexes = types.MultiRewardIndexes{
|
|
|
|
{
|
|
|
|
CollateralType: "bnb",
|
|
|
|
RewardIndexes: types.RewardIndexes{
|
|
|
|
{
|
|
|
|
CollateralType: "hard",
|
|
|
|
RewardFactor: d("0.02"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
CollateralType: "ukava",
|
|
|
|
RewardFactor: d("0.04"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
CollateralType: "btcb",
|
|
|
|
RewardIndexes: types.RewardIndexes{
|
|
|
|
{
|
|
|
|
CollateralType: "hard",
|
|
|
|
RewardFactor: d("0.2"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
CollateralType: "ukava",
|
|
|
|
RewardFactor: d("0.4"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
func extractCollateralTypes(indexes types.MultiRewardIndexes) []string {
|
|
|
|
var denoms []string
|
|
|
|
for _, ri := range indexes {
|
|
|
|
denoms = append(denoms, ri.CollateralType)
|
|
|
|
}
|
|
|
|
return denoms
|
|
|
|
}
|
|
|
|
|
|
|
|
func increaseAllRewardFactors(indexes types.MultiRewardIndexes) types.MultiRewardIndexes {
|
|
|
|
increasedIndexes := make(types.MultiRewardIndexes, len(indexes))
|
|
|
|
copy(increasedIndexes, indexes)
|
|
|
|
|
|
|
|
for i := range increasedIndexes {
|
|
|
|
increasedIndexes[i].RewardIndexes = increaseRewardFactors(increasedIndexes[i].RewardIndexes)
|
|
|
|
}
|
|
|
|
return increasedIndexes
|
|
|
|
}
|
|
|
|
|
|
|
|
func increaseRewardFactors(indexes types.RewardIndexes) types.RewardIndexes {
|
|
|
|
increasedIndexes := make(types.RewardIndexes, len(indexes))
|
|
|
|
copy(increasedIndexes, indexes)
|
|
|
|
|
|
|
|
for i := range increasedIndexes {
|
|
|
|
increasedIndexes[i].RewardFactor = increasedIndexes[i].RewardFactor.MulInt64(2)
|
|
|
|
}
|
|
|
|
return increasedIndexes
|
|
|
|
}
|
|
|
|
|
|
|
|
func appendUniqueMultiRewardIndex(indexes types.MultiRewardIndexes) types.MultiRewardIndexes {
|
|
|
|
const uniqueDenom = "uniquedenom"
|
|
|
|
|
|
|
|
for _, mri := range indexes {
|
|
|
|
if mri.CollateralType == uniqueDenom {
|
|
|
|
panic(fmt.Sprintf("tried to add unique multi reward index with denom '%s', but denom already existed", uniqueDenom))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return append(indexes, types.NewMultiRewardIndex(
|
|
|
|
uniqueDenom,
|
|
|
|
types.RewardIndexes{
|
|
|
|
{
|
|
|
|
CollateralType: "hard",
|
|
|
|
RewardFactor: d("0.02"),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
CollateralType: "ukava",
|
|
|
|
RewardFactor: d("0.04"),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
),
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
func appendUniqueEmptyMultiRewardIndex(indexes types.MultiRewardIndexes) types.MultiRewardIndexes {
|
|
|
|
const uniqueDenom = "uniquedenom"
|
|
|
|
|
|
|
|
for _, mri := range indexes {
|
|
|
|
if mri.CollateralType == uniqueDenom {
|
|
|
|
panic(fmt.Sprintf("tried to add unique multi reward index with denom '%s', but denom already existed", uniqueDenom))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return append(indexes, types.NewMultiRewardIndex(uniqueDenom, nil))
|
|
|
|
}
|
|
|
|
|
|
|
|
func appendUniqueRewardIndexToFirstItem(indexes types.MultiRewardIndexes) types.MultiRewardIndexes {
|
|
|
|
newIndexes := make(types.MultiRewardIndexes, len(indexes))
|
|
|
|
copy(newIndexes, indexes)
|
|
|
|
|
|
|
|
newIndexes[0].RewardIndexes = appendUniqueRewardIndex(newIndexes[0].RewardIndexes)
|
|
|
|
return newIndexes
|
|
|
|
}
|
|
|
|
|
|
|
|
func appendUniqueRewardIndex(indexes types.RewardIndexes) types.RewardIndexes {
|
|
|
|
const uniqueDenom = "uniquereward"
|
|
|
|
|
|
|
|
for _, mri := range indexes {
|
|
|
|
if mri.CollateralType == uniqueDenom {
|
|
|
|
panic(fmt.Sprintf("tried to add unique reward index with denom '%s', but denom already existed", uniqueDenom))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return append(
|
|
|
|
indexes,
|
|
|
|
types.NewRewardIndex(uniqueDenom, d("0.02")),
|
|
|
|
)
|
|
|
|
}
|