2021-06-07 16:08:03 +00:00
|
|
|
package v0_15
|
|
|
|
|
|
|
|
import (
|
2021-08-17 17:59:07 +00:00
|
|
|
"fmt"
|
2021-06-07 16:08:03 +00:00
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
|
|
|
"testing"
|
2021-08-12 15:31:58 +00:00
|
|
|
"time"
|
2021-06-07 16:08:03 +00:00
|
|
|
|
|
|
|
"github.com/cosmos/cosmos-sdk/codec"
|
|
|
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
2021-08-12 15:31:58 +00:00
|
|
|
"github.com/cosmos/cosmos-sdk/x/auth"
|
|
|
|
"github.com/cosmos/cosmos-sdk/x/auth/vesting"
|
2021-07-15 14:41:55 +00:00
|
|
|
"github.com/cosmos/cosmos-sdk/x/genutil"
|
2021-08-12 15:31:58 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
2021-07-07 13:23:06 +00:00
|
|
|
"github.com/stretchr/testify/require"
|
2021-08-19 15:20:26 +00:00
|
|
|
tmtypes "github.com/tendermint/tendermint/types"
|
2021-07-07 13:23:06 +00:00
|
|
|
|
2021-06-07 16:08:03 +00:00
|
|
|
"github.com/kava-labs/kava/app"
|
2021-08-19 15:20:26 +00:00
|
|
|
v0_15cdp "github.com/kava-labs/kava/x/cdp/types"
|
2021-06-07 16:08:03 +00:00
|
|
|
v0_14committee "github.com/kava-labs/kava/x/committee/legacy/v0_14"
|
|
|
|
v0_15committee "github.com/kava-labs/kava/x/committee/types"
|
2021-08-17 17:59:07 +00:00
|
|
|
"github.com/kava-labs/kava/x/hard"
|
2021-07-05 22:01:25 +00:00
|
|
|
v0_14incentive "github.com/kava-labs/kava/x/incentive/legacy/v0_14"
|
|
|
|
v0_15incentive "github.com/kava-labs/kava/x/incentive/types"
|
2021-08-19 18:33:40 +00:00
|
|
|
"github.com/kava-labs/kava/x/swap"
|
2021-06-07 16:08:03 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestMain(m *testing.M) {
|
|
|
|
config := sdk.GetConfig()
|
|
|
|
app.SetBech32AddressPrefixes(config)
|
|
|
|
app.SetBip44CoinType(config)
|
|
|
|
|
|
|
|
os.Exit(m.Run())
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCommittee(t *testing.T) {
|
|
|
|
bz, err := ioutil.ReadFile(filepath.Join("testdata", "kava-7-committee-state.json"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
var oldGenState v0_14committee.GenesisState
|
|
|
|
cdc := codec.New()
|
|
|
|
sdk.RegisterCodec(cdc)
|
|
|
|
v0_14committee.RegisterCodec(cdc)
|
|
|
|
|
|
|
|
require.NotPanics(t, func() {
|
|
|
|
cdc.MustUnmarshalJSON(bz, &oldGenState)
|
|
|
|
})
|
|
|
|
|
|
|
|
newGenState := Committee(oldGenState)
|
|
|
|
err = newGenState.Validate()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
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>
2021-07-13 11:38:15 +00:00
|
|
|
require.Equal(t, len(oldGenState.Committees)+2, len(newGenState.Committees)) // New gen state has 2 additional committees
|
2021-06-07 16:08:03 +00:00
|
|
|
for i := 0; i < len(oldGenState.Committees); i++ {
|
|
|
|
require.Equal(t, len(oldGenState.Committees[i].Permissions), len(newGenState.Committees[i].GetPermissions()))
|
|
|
|
}
|
|
|
|
|
|
|
|
oldSPCP := oldGenState.Committees[0].Permissions[0].(v0_14committee.SubParamChangePermission)
|
|
|
|
newSPCP := newGenState.Committees[0].GetPermissions()[0].(v0_15committee.SubParamChangePermission)
|
|
|
|
require.Equal(t, len(oldSPCP.AllowedParams), len(newSPCP.AllowedParams))
|
|
|
|
require.Equal(t, len(oldSPCP.AllowedAssetParams), len(newSPCP.AllowedAssetParams))
|
|
|
|
require.Equal(t, len(oldSPCP.AllowedCollateralParams), len(newSPCP.AllowedCollateralParams))
|
|
|
|
require.Equal(t, len(oldSPCP.AllowedMarkets), len(newSPCP.AllowedMarkets))
|
|
|
|
require.Equal(t, len(oldSPCP.AllowedMoneyMarkets), len(newSPCP.AllowedMoneyMarkets))
|
|
|
|
}
|
2021-08-19 15:20:26 +00:00
|
|
|
|
|
|
|
func TestIncentive_Full(t *testing.T) {
|
|
|
|
t.Skip() // skip to avoid having to commit a large genesis file to the repo
|
|
|
|
|
|
|
|
genDoc, err := tmtypes.GenesisDocFromFile(filepath.Join("testdata", "genesis.json"))
|
2021-07-05 22:01:25 +00:00
|
|
|
require.NoError(t, err)
|
2021-08-19 15:20:26 +00:00
|
|
|
|
|
|
|
cdc := makeV014Codec()
|
|
|
|
|
|
|
|
var oldState genutil.AppMap
|
|
|
|
cdc.MustUnmarshalJSON(genDoc.AppState, &oldState)
|
|
|
|
|
2021-07-05 22:01:25 +00:00
|
|
|
var oldIncentiveGenState v0_14incentive.GenesisState
|
2021-08-19 15:20:26 +00:00
|
|
|
cdc.MustUnmarshalJSON(oldState[v0_14incentive.ModuleName], &oldIncentiveGenState)
|
2021-07-05 22:01:25 +00:00
|
|
|
|
2021-08-19 15:20:26 +00:00
|
|
|
var oldCDPGenState v0_15cdp.GenesisState
|
|
|
|
cdc.MustUnmarshalJSON(oldState[v0_15cdp.ModuleName], &oldCDPGenState)
|
|
|
|
|
|
|
|
newGenState := Incentive(app.MakeCodec(), oldIncentiveGenState, oldCDPGenState.CDPs)
|
|
|
|
require.NoError(t, newGenState.Validate())
|
|
|
|
|
|
|
|
// TODO check params, indexes, and accumulation times
|
|
|
|
|
|
|
|
// Ensure the usdx claim indexes match global
|
|
|
|
globalIndexes := newGenState.USDXRewardState.MultiRewardIndexes
|
|
|
|
for _, claim := range newGenState.USDXMintingClaims {
|
|
|
|
|
|
|
|
for _, globalIndex := range globalIndexes {
|
|
|
|
expectedFactor, found := globalIndex.RewardIndexes.Get(v0_15incentive.USDXMintingRewardDenom)
|
|
|
|
require.True(t, found)
|
|
|
|
|
|
|
|
factor, found := claim.RewardIndexes.Get(globalIndex.CollateralType)
|
|
|
|
require.True(t, found)
|
|
|
|
|
|
|
|
require.Equal(t, expectedFactor, factor)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure there is a usdx claim for every cdp
|
|
|
|
for _, cdp := range oldCDPGenState.CDPs {
|
|
|
|
numClaims := 0
|
|
|
|
for _, claim := range newGenState.USDXMintingClaims {
|
|
|
|
if cdp.Owner.Equals(claim.Owner) {
|
|
|
|
numClaims++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
require.Equal(t, 1, numClaims, "cdp '%s' has invalid number of claims '%d'", cdp.Owner, numClaims)
|
|
|
|
|
|
|
|
// also check cdp indexes are valid
|
|
|
|
require.True(t, cdp.InterestFactor.GTE(sdk.OneDec()), "found cdp with interest factor < 1")
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check reward amounts
|
|
|
|
for _, claim := range newGenState.USDXMintingClaims {
|
|
|
|
|
|
|
|
// check a few high value accounts
|
|
|
|
switch claim.Owner.String() {
|
|
|
|
// check reward is: additional reward + existing unclaimed reward
|
|
|
|
// note, non zero unclaimed rewards could change if the user submits a claim tx before launch
|
|
|
|
case "kava1k8lymw58tduy9gm6jkt04ddkjd83nf7sm8xthl":
|
|
|
|
require.Equal(t, sdk.NewInt(370982556999+0), claim.Reward.Amount)
|
|
|
|
case "kava1p3ucd3ptpw902fluyjzhq3ffgq4ntddaysyq8h":
|
|
|
|
require.Equal(t, sdk.NewInt(77550672285+16960713469), claim.Reward.Amount)
|
|
|
|
case "kava1qe6ahdnhnfugle29054d8uqg7fa44ryx934yc6":
|
|
|
|
require.Equal(t, sdk.NewInt(40874651319+0), claim.Reward.Amount)
|
|
|
|
case "kava12h6pq2xqzgtxttrzg7q2rplsyxtv2dc5gwh8rl":
|
|
|
|
require.Equal(t, sdk.NewInt(30867752254+0), claim.Reward.Amount)
|
|
|
|
case "kava10hczxv0p3eadcwgt5u79yhahsyuw98u26qan50":
|
|
|
|
require.Equal(t, sdk.NewInt(22429344254+0), claim.Reward.Amount)
|
|
|
|
case "kava15wyjwhj6zh79m7adm69pwl3nsq9z8gs9ezs4k7":
|
|
|
|
require.Equal(t, sdk.NewInt(10252596901+0), claim.Reward.Amount)
|
|
|
|
case "kava1yg4840l77dfs5zqflldhut27en2mhvvc8vj93x":
|
|
|
|
require.Equal(t, sdk.NewInt(9898765520+0), claim.Reward.Amount)
|
|
|
|
case "kava1x242qk6jf2rv23ruvk6fmxp97gg2y75a9r2caq":
|
|
|
|
require.Equal(t, sdk.NewInt(7761701231+0), claim.Reward.Amount)
|
|
|
|
case "kava1tstf3u4cw7u4xyu7wxdrnmrpvvmfamq3twcj7f":
|
|
|
|
require.Equal(t, sdk.NewInt(2466900572+0), claim.Reward.Amount)
|
|
|
|
}
|
|
|
|
|
|
|
|
// check no rewards have been reduced
|
|
|
|
for _, oldClaim := range oldIncentiveGenState.USDXMintingClaims {
|
|
|
|
if oldClaim.Owner.Equals(claim.Owner) {
|
|
|
|
require.Truef(t, claim.Reward.IsGTE(oldClaim.Reward), "found claim with reduced rewards, old %s, new %s", oldClaim, claim)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-07-05 22:01:25 +00:00
|
|
|
|
|
|
|
require.Equal(t, len(oldIncentiveGenState.HardLiquidityProviderClaims), len(newGenState.HardLiquidityProviderClaims))
|
2021-07-07 16:50:14 +00:00
|
|
|
// 1 new DelegatorClaim should have been created for each existing HardLiquidityProviderClaim
|
|
|
|
require.Equal(t, len(oldIncentiveGenState.HardLiquidityProviderClaims), len(newGenState.DelegatorClaims))
|
2021-07-05 22:01:25 +00:00
|
|
|
}
|
2021-07-15 14:41:55 +00:00
|
|
|
|
2021-08-19 15:20:26 +00:00
|
|
|
func TestIncentive_Full_TotalRewards(t *testing.T) {
|
|
|
|
t.Skip() // skip to avoid having to commit a large genesis file to the repo
|
|
|
|
|
|
|
|
genDoc, err := tmtypes.GenesisDocFromFile(filepath.Join("testdata", "genesis.json"))
|
2021-07-15 14:41:55 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
2021-08-19 15:20:26 +00:00
|
|
|
cdc := makeV014Codec()
|
2021-07-15 14:41:55 +00:00
|
|
|
|
2021-08-19 15:20:26 +00:00
|
|
|
var oldState genutil.AppMap
|
|
|
|
cdc.MustUnmarshalJSON(genDoc.AppState, &oldState)
|
|
|
|
|
|
|
|
var oldIncentiveGenState v0_14incentive.GenesisState
|
|
|
|
cdc.MustUnmarshalJSON(oldState[v0_14incentive.ModuleName], &oldIncentiveGenState)
|
|
|
|
|
|
|
|
var oldCDPGenState v0_15cdp.GenesisState
|
|
|
|
cdc.MustUnmarshalJSON(oldState[v0_15cdp.ModuleName], &oldCDPGenState)
|
|
|
|
|
|
|
|
newGenState := Incentive(app.MakeCodec(), oldIncentiveGenState, oldCDPGenState.CDPs)
|
|
|
|
|
|
|
|
// total previous rewards
|
|
|
|
oldTotalRewards := sdk.NewCoins() // total synced unclaimed rewards
|
|
|
|
for _, claim := range oldIncentiveGenState.HardLiquidityProviderClaims {
|
|
|
|
oldTotalRewards = oldTotalRewards.Add(claim.Reward...)
|
|
|
|
}
|
|
|
|
for _, claim := range oldIncentiveGenState.USDXMintingClaims {
|
|
|
|
oldTotalRewards = oldTotalRewards.Add(claim.Reward)
|
|
|
|
}
|
|
|
|
|
|
|
|
// total new rewards
|
|
|
|
newTotalRewards := sdk.NewCoins() // total synced unclaimed rewards
|
|
|
|
for _, claim := range newGenState.USDXMintingClaims {
|
|
|
|
newTotalRewards = newTotalRewards.Add(claim.Reward)
|
|
|
|
}
|
|
|
|
for _, claim := range newGenState.HardLiquidityProviderClaims {
|
|
|
|
newTotalRewards = newTotalRewards.Add(claim.Reward...)
|
|
|
|
}
|
|
|
|
|
|
|
|
// rewards added in migration
|
|
|
|
additionalRewards := sdk.NewCoins()
|
|
|
|
var missedRewards map[string]sdk.Coin
|
|
|
|
cdc.MustUnmarshalJSON([]byte(missedUSDXMintingRewards), &missedRewards)
|
|
|
|
for _, c := range missedRewards {
|
|
|
|
additionalRewards = additionalRewards.Add(c)
|
|
|
|
}
|
2021-07-15 14:41:55 +00:00
|
|
|
|
2021-08-19 15:20:26 +00:00
|
|
|
require.Equal(t, oldTotalRewards.Add(additionalRewards...), newTotalRewards)
|
2021-07-15 14:41:55 +00:00
|
|
|
}
|
2021-08-12 15:31:58 +00:00
|
|
|
|
2021-08-19 18:33:40 +00:00
|
|
|
func TestIncentive_SwpLPRewards(t *testing.T) {
|
|
|
|
expectedAnnualRewards := sdk.NewCoin("swp", sdk.NewInt(24000000e6))
|
|
|
|
expectedAnnualRewardsMap := map[string]sdk.Coin{
|
|
|
|
"ukava:usdx": sdk.NewCoin("swp", sdk.NewInt(7000000e6)),
|
|
|
|
"swp:usdx": sdk.NewCoin("swp", sdk.NewInt(6000000e6)),
|
|
|
|
"busd:usdx": sdk.NewCoin("swp", sdk.NewInt(5000000e6)),
|
|
|
|
"bnb:usdx": sdk.NewCoin("swp", sdk.NewInt(1500000e6)),
|
|
|
|
"btcb:usdx": sdk.NewCoin("swp", sdk.NewInt(1500000e6)),
|
|
|
|
"hard:usdx": sdk.NewCoin("swp", sdk.NewInt(1500000e6)),
|
|
|
|
"usdx:xrpb": sdk.NewCoin("swp", sdk.NewInt(1500000e6)),
|
|
|
|
}
|
|
|
|
|
|
|
|
bz, err := ioutil.ReadFile(filepath.Join("testdata", "kava-7-test-incentive-state.json"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
var oldIncentiveGenState v0_14incentive.GenesisState
|
|
|
|
cdc := app.MakeCodec()
|
|
|
|
require.NotPanics(t, func() {
|
|
|
|
cdc.MustUnmarshalJSON(bz, &oldIncentiveGenState)
|
|
|
|
})
|
|
|
|
secondsPerYear := sdk.NewInt(31536000)
|
|
|
|
|
|
|
|
newGenState := v0_15incentive.GenesisState{}
|
|
|
|
require.NotPanics(t, func() {
|
|
|
|
newGenState = Incentive(app.MakeCodec(), oldIncentiveGenState, v0_15cdp.CDPs{})
|
|
|
|
})
|
|
|
|
err = newGenState.Validate()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
actualAnnualRewards := sdk.NewCoin("swp", sdk.ZeroInt())
|
|
|
|
|
|
|
|
for _, rp := range newGenState.Params.SwapRewardPeriods {
|
|
|
|
expected, found := expectedAnnualRewardsMap[rp.CollateralType]
|
|
|
|
require.True(t, found)
|
|
|
|
require.True(t, len(rp.RewardsPerSecond) == 1)
|
|
|
|
require.True(t, rp.RewardsPerSecond[0].Denom == "swp")
|
|
|
|
annualRewardsAmount := rp.RewardsPerSecond[0].Amount.ToDec().Mul(secondsPerYear.ToDec()).RoundInt()
|
|
|
|
annualRewardsCoin := sdk.NewCoin("swp", annualRewardsAmount)
|
|
|
|
actualAnnualRewards = actualAnnualRewards.Add(annualRewardsCoin)
|
|
|
|
// allow for +- 0.1% deviation for targeted annual rewards
|
|
|
|
inRange := assertRewardsWithinRange(expected, annualRewardsCoin, sdk.MustNewDecFromStr("0.001"))
|
|
|
|
require.True(t, inRange, fmt.Sprintf("expected annual rewards: %s, actual %s", expected, annualRewardsCoin))
|
|
|
|
}
|
|
|
|
inRange := assertRewardsWithinRange(expectedAnnualRewards, actualAnnualRewards, sdk.MustNewDecFromStr("0.001"))
|
|
|
|
require.True(t, inRange, fmt.Sprintf("expected annual rewards: %s, actual %s", expectedAnnualRewards, actualAnnualRewards))
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestIncentive_SwpDelegatorRewards(t *testing.T) {
|
|
|
|
expectedAnnualRewards := sdk.NewCoin("swp", sdk.NewInt(6250000e6))
|
|
|
|
bz, err := ioutil.ReadFile(filepath.Join("testdata", "kava-7-test-incentive-state.json"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
var oldIncentiveGenState v0_14incentive.GenesisState
|
|
|
|
cdc := app.MakeCodec()
|
|
|
|
require.NotPanics(t, func() {
|
|
|
|
cdc.MustUnmarshalJSON(bz, &oldIncentiveGenState)
|
|
|
|
})
|
|
|
|
secondsPerYear := sdk.NewInt(31536000)
|
|
|
|
|
|
|
|
newGenState := v0_15incentive.GenesisState{}
|
|
|
|
require.NotPanics(t, func() {
|
|
|
|
newGenState = Incentive(app.MakeCodec(), oldIncentiveGenState, v0_15cdp.CDPs{})
|
|
|
|
})
|
|
|
|
|
|
|
|
for _, rp := range newGenState.Params.DelegatorRewardPeriods {
|
|
|
|
swpRewardsPerSecondDelegators := sdk.NewCoin("swp", rp.RewardsPerSecond.AmountOf("swp"))
|
|
|
|
annualRewardsAmount := swpRewardsPerSecondDelegators.Amount.ToDec().Mul(secondsPerYear.ToDec()).RoundInt()
|
|
|
|
annualRewardsCoin := sdk.NewCoin("swp", annualRewardsAmount)
|
|
|
|
inRange := assertRewardsWithinRange(expectedAnnualRewards, annualRewardsCoin, sdk.MustNewDecFromStr("0.001"))
|
|
|
|
require.True(t, inRange, fmt.Sprintf("expected annual rewards: %s, actual %s", expectedAnnualRewards, annualRewardsCoin))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestIncentive_SwpPoolsValid(t *testing.T) {
|
|
|
|
bz, err := ioutil.ReadFile(filepath.Join("testdata", "kava-7-test-incentive-state.json"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
appState := genutil.AppMap{v0_14incentive.ModuleName: bz, v0_15cdp.ModuleName: app.MakeCodec().MustMarshalJSON(v0_15cdp.DefaultGenesisState())}
|
|
|
|
|
|
|
|
MigrateAppState(appState)
|
|
|
|
cdc := app.MakeCodec()
|
|
|
|
var swapGS swap.GenesisState
|
|
|
|
cdc.MustUnmarshalJSON(appState[swap.ModuleName], &swapGS)
|
|
|
|
var incentiveGS v0_15incentive.GenesisState
|
|
|
|
cdc.MustUnmarshalJSON(appState[v0_15incentive.ModuleName], &incentiveGS)
|
|
|
|
|
|
|
|
for _, rp := range incentiveGS.Params.SwapRewardPeriods {
|
|
|
|
found := false
|
|
|
|
for _, pool := range swapGS.Params.AllowedPools {
|
|
|
|
if pool.Name() == rp.CollateralType {
|
|
|
|
found = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
require.True(t, found, fmt.Sprintf("incentivized pool %s not found in swap genesis state", rp.CollateralType))
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
func assertRewardsWithinRange(expected, actual sdk.Coin, tolerance sdk.Dec) bool {
|
|
|
|
upper := expected.Amount.ToDec().Mul(sdk.OneDec().Add(tolerance))
|
|
|
|
lower := expected.Amount.ToDec().Mul(sdk.OneDec().Sub(tolerance))
|
|
|
|
if actual.Amount.ToDec().GTE(lower) && actual.Amount.ToDec().LTE(upper) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2021-08-16 16:14:48 +00:00
|
|
|
func TestSwap(t *testing.T) {
|
|
|
|
swapGS := Swap()
|
|
|
|
err := swapGS.Validate()
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, 7, len(swapGS.Params.AllowedPools))
|
|
|
|
require.Equal(t, 0, len(swapGS.PoolRecords))
|
|
|
|
require.Equal(t, 0, len(swapGS.ShareRecords))
|
|
|
|
}
|
|
|
|
|
2021-08-12 15:31:58 +00:00
|
|
|
func TestAuth_ParametersEqual(t *testing.T) {
|
|
|
|
bz, err := ioutil.ReadFile(filepath.Join("testdata", "kava-7-test-auth-state.json"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
var genesisState auth.GenesisState
|
|
|
|
cdc := app.MakeCodec()
|
|
|
|
cdc.MustUnmarshalJSON(bz, &genesisState)
|
|
|
|
|
2021-08-17 17:59:07 +00:00
|
|
|
migratedGenesisState := Auth(cdc, genesisState, GenesisTime)
|
2021-08-12 15:31:58 +00:00
|
|
|
|
|
|
|
assert.Equal(t, genesisState.Params, migratedGenesisState.Params, "expected auth parameters to not change")
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAuth_AccountConversion(t *testing.T) {
|
|
|
|
bz, err := ioutil.ReadFile(filepath.Join("testdata", "kava-7-test-auth-state.json"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
cdc := app.MakeCodec()
|
2021-08-16 18:25:15 +00:00
|
|
|
|
|
|
|
var genesisState auth.GenesisState
|
2021-08-12 15:31:58 +00:00
|
|
|
cdc.MustUnmarshalJSON(bz, &genesisState)
|
|
|
|
|
2021-08-17 17:59:07 +00:00
|
|
|
migratedGenesisState := MigrateAccounts(genesisState, GenesisTime)
|
2021-08-16 18:25:15 +00:00
|
|
|
var originalGenesisState auth.GenesisState
|
|
|
|
cdc.MustUnmarshalJSON(bz, &originalGenesisState)
|
2021-08-12 15:31:58 +00:00
|
|
|
require.Equal(t, len(genesisState.Accounts), len(migratedGenesisState.Accounts), "expected the number of accounts after migration to be equal")
|
|
|
|
err = auth.ValidateGenesis(migratedGenesisState)
|
|
|
|
require.NoError(t, err, "expected migrated genesis to be valid")
|
|
|
|
|
|
|
|
for i, acc := range migratedGenesisState.Accounts {
|
2021-08-16 18:25:15 +00:00
|
|
|
oldAcc := originalGenesisState.Accounts[i]
|
2021-08-12 15:31:58 +00:00
|
|
|
|
|
|
|
// total owned coins does not change
|
|
|
|
require.Equal(t, oldAcc.GetCoins(), acc.GetCoins(), "expected base coins to not change")
|
|
|
|
|
|
|
|
// ensure spenable coins at genesis time is equal
|
|
|
|
require.Equal(t, oldAcc.SpendableCoins(GenesisTime), acc.SpendableCoins(GenesisTime), "expected spendable coins to not change")
|
|
|
|
// check 30 days
|
|
|
|
futureDate := GenesisTime.Add(30 * 24 * time.Hour)
|
|
|
|
require.Equal(t, oldAcc.SpendableCoins(futureDate), acc.SpendableCoins(futureDate), "expected spendable coins to not change")
|
|
|
|
// check 90 days
|
|
|
|
futureDate = GenesisTime.Add(90 * 24 * time.Hour)
|
|
|
|
require.Equal(t, oldAcc.SpendableCoins(futureDate), acc.SpendableCoins(futureDate), "expected spendable coins to not change")
|
|
|
|
// check 180 days
|
|
|
|
futureDate = GenesisTime.Add(180 * 24 * time.Hour)
|
|
|
|
require.Equal(t, oldAcc.SpendableCoins(futureDate), acc.SpendableCoins(futureDate), "expected spendable coins to not change")
|
|
|
|
// check 365 days
|
|
|
|
futureDate = GenesisTime.Add(365 * 24 * time.Hour)
|
|
|
|
require.Equal(t, oldAcc.SpendableCoins(futureDate), acc.SpendableCoins(futureDate), "expected spendable coins to not change")
|
2021-08-16 18:25:15 +00:00
|
|
|
// check 2 years
|
|
|
|
futureDate = GenesisTime.Add(2 * 365 * 24 * time.Hour)
|
|
|
|
require.Equal(t, oldAcc.SpendableCoins(futureDate), acc.SpendableCoins(futureDate), "expected spendable coins to not change")
|
2021-08-12 15:31:58 +00:00
|
|
|
|
|
|
|
if vacc, ok := acc.(*vesting.PeriodicVestingAccount); ok {
|
|
|
|
// old account must be a periodic vesting account
|
|
|
|
oldVacc, ok := oldAcc.(*vesting.PeriodicVestingAccount)
|
|
|
|
require.True(t, ok)
|
|
|
|
|
|
|
|
// total delegated coins must match
|
|
|
|
oldTotalDelegated := oldVacc.DelegatedFree.Add(oldVacc.DelegatedVesting...)
|
|
|
|
newTotalDelegated := vacc.DelegatedFree.Add(vacc.DelegatedVesting...)
|
|
|
|
require.Equal(t, oldTotalDelegated, newTotalDelegated, "expected total amount of tracked delegations to not change")
|
|
|
|
|
|
|
|
// delegated vesting must be less or equal to original vesting
|
|
|
|
require.True(t, vacc.DelegatedVesting.IsAllLTE(vacc.OriginalVesting), "expected delegated vesting to be less or equal to original vesting")
|
|
|
|
|
|
|
|
// vested coins must be nil for the new account
|
|
|
|
require.Equal(t, sdk.Coins(nil), vacc.GetVestedCoins(GenesisTime), "expected no vested coins at genesis time")
|
|
|
|
|
|
|
|
// vesting coins must not be nil
|
|
|
|
require.NotEqual(t, sdk.Coins(nil), vacc.GetVestingCoins(GenesisTime), "expected vesting coins to be greater than 0")
|
|
|
|
|
|
|
|
// new account as less than or equal
|
|
|
|
require.LessOrEqual(t, len(vacc.VestingPeriods), len(oldVacc.VestingPeriods), "expected vesting periods of new account to be less than or equal to old")
|
2021-08-16 18:25:15 +00:00
|
|
|
|
|
|
|
// end time should not change
|
|
|
|
require.Equal(t, oldVacc.EndTime, vacc.EndTime, "expected end time to not change")
|
2021-08-12 15:31:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-08-17 17:59:07 +00:00
|
|
|
|
2021-08-19 05:10:58 +00:00
|
|
|
func TestAuth_Regression_BalancesEqual(t *testing.T) {
|
|
|
|
bz, err := ioutil.ReadFile(filepath.Join("testdata", "block-1543671-auth-state.json"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
cdc := app.MakeCodec()
|
|
|
|
|
|
|
|
var originalGenesisState auth.GenesisState
|
|
|
|
cdc.MustUnmarshalJSON(bz, &originalGenesisState)
|
|
|
|
|
|
|
|
var genesisState auth.GenesisState
|
|
|
|
cdc.MustUnmarshalJSON(bz, &genesisState)
|
|
|
|
|
|
|
|
genesisState = Auth(cdc, genesisState, GenesisTime)
|
|
|
|
|
|
|
|
for i, oldAcc := range originalGenesisState.Accounts {
|
|
|
|
acc := genesisState.Accounts[i]
|
|
|
|
|
|
|
|
// total owned coins does not change, despite additional swp tokens
|
|
|
|
swpBalance := sdk.NewCoins(sdk.NewCoin("swp", acc.GetCoins().AmountOf("swp")))
|
|
|
|
require.Equal(t, oldAcc.GetCoins(), acc.GetCoins().Sub(swpBalance), "expected base coins to not change")
|
|
|
|
|
|
|
|
// ensure spendable coins at genesis time is equal, despite additional swp tokens
|
|
|
|
swpBalance = sdk.NewCoins(sdk.NewCoin("swp", acc.SpendableCoins(GenesisTime).AmountOf("swp")))
|
|
|
|
require.Equal(t, oldAcc.SpendableCoins(GenesisTime), acc.SpendableCoins(GenesisTime).Sub(swpBalance), "expected spendable coins to not change")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-08-17 17:59:07 +00:00
|
|
|
func TestAuth_MakeAirdropMap(t *testing.T) {
|
|
|
|
cdc := app.MakeCodec()
|
|
|
|
aidropTokenAmount := sdk.NewInt(1000000000000)
|
|
|
|
totalSwpTokens := sdk.ZeroInt()
|
|
|
|
var loadedAirdropMap map[string]sdk.Coin
|
|
|
|
cdc.MustUnmarshalJSON([]byte(swpAirdropMap), &loadedAirdropMap)
|
|
|
|
for _, coin := range loadedAirdropMap {
|
|
|
|
totalSwpTokens = totalSwpTokens.Add(coin.Amount)
|
|
|
|
}
|
|
|
|
require.Equal(t, aidropTokenAmount, totalSwpTokens)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAuth_TestAllDepositorsIncluded(t *testing.T) {
|
|
|
|
var deposits hard.Deposits
|
|
|
|
cdc := app.MakeCodec()
|
|
|
|
bz, err := ioutil.ReadFile("./data/hard-deposits-block-1543671.json")
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Sprintf("Couldn't open hard deposit snapshot file: %v", err))
|
|
|
|
}
|
|
|
|
cdc.MustUnmarshalJSON(bz, &deposits)
|
|
|
|
|
|
|
|
depositorsInSnapShot := 0
|
|
|
|
for _, dep := range deposits {
|
|
|
|
if dep.Amount.AmountOf("usdx").IsPositive() {
|
|
|
|
depositorsInSnapShot++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
var loadedAirdropMap map[string]sdk.Coin
|
|
|
|
cdc.MustUnmarshalJSON([]byte(swpAirdropMap), &loadedAirdropMap)
|
|
|
|
keys := make([]string, 0, len(loadedAirdropMap))
|
|
|
|
for k := range loadedAirdropMap {
|
|
|
|
keys = append(keys, k)
|
|
|
|
}
|
|
|
|
require.Equal(t, depositorsInSnapShot, len(keys))
|
|
|
|
}
|
|
|
|
|
2021-08-19 05:10:58 +00:00
|
|
|
func TestAuth_SwpSupply_AfterAirdrop(t *testing.T) {
|
2021-08-17 17:59:07 +00:00
|
|
|
swpSupply := sdk.NewCoin("swp", sdk.ZeroInt())
|
|
|
|
expectedSwpSupply := sdk.NewCoin("swp", sdk.NewInt(1000000000000))
|
|
|
|
bz, err := ioutil.ReadFile(filepath.Join("testdata", "block-1543671-auth-state.json"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
var genesisState auth.GenesisState
|
|
|
|
cdc := app.MakeCodec()
|
|
|
|
cdc.MustUnmarshalJSON(bz, &genesisState)
|
|
|
|
|
2021-08-19 05:10:58 +00:00
|
|
|
migratedGenesisState := ApplySwpAirdrop(cdc, genesisState)
|
|
|
|
|
|
|
|
for _, acc := range migratedGenesisState.Accounts {
|
|
|
|
swpAmount := acc.GetCoins().AmountOf("swp")
|
|
|
|
if swpAmount.IsPositive() {
|
|
|
|
swpCoin := sdk.NewCoin("swp", swpAmount)
|
|
|
|
swpSupply = swpSupply.Add(swpCoin)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
require.Equal(t, expectedSwpSupply, swpSupply)
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestAuth_SwpSupply_TotalSupply(t *testing.T) {
|
|
|
|
swpSupply := sdk.NewCoin("swp", sdk.ZeroInt())
|
|
|
|
expectedSwpSupply := sdk.NewCoin("swp", sdk.NewInt(250000000e6))
|
|
|
|
bz, err := ioutil.ReadFile(filepath.Join("testdata", "block-1543671-auth-state.json"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
var genesisState auth.GenesisState
|
|
|
|
cdc := app.MakeCodec()
|
|
|
|
cdc.MustUnmarshalJSON(bz, &genesisState)
|
|
|
|
|
2021-08-17 17:59:07 +00:00
|
|
|
migratedGenesisState := Auth(cdc, genesisState, GenesisTime)
|
|
|
|
|
|
|
|
for _, acc := range migratedGenesisState.Accounts {
|
|
|
|
swpAmount := acc.GetCoins().AmountOf("swp")
|
|
|
|
if swpAmount.IsPositive() {
|
|
|
|
swpCoin := sdk.NewCoin("swp", swpAmount)
|
|
|
|
swpSupply = swpSupply.Add(swpCoin)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
require.Equal(t, expectedSwpSupply, swpSupply)
|
|
|
|
}
|
2021-08-19 05:10:58 +00:00
|
|
|
|
|
|
|
func TestAuth_SwpSupply_SpendableCoins(t *testing.T) {
|
|
|
|
bz, err := ioutil.ReadFile(filepath.Join("testdata", "block-1543671-auth-state.json"))
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
var genesisState auth.GenesisState
|
|
|
|
cdc := app.MakeCodec()
|
|
|
|
cdc.MustUnmarshalJSON(bz, &genesisState)
|
|
|
|
|
|
|
|
swpTreasuryExpectedSpendableCoins := sdk.NewCoins(SwpTreasuryCoins.Sub(SwpTreasuryOriginalVesting))
|
|
|
|
foundTreasury := false
|
|
|
|
|
|
|
|
swpTeamExpectedSpendableCoins := sdk.Coins(nil)
|
|
|
|
foundTeam := false
|
|
|
|
|
|
|
|
kavaDistExpectedSpendableCoins := KavaDistCoins
|
|
|
|
foundKavadist := false
|
|
|
|
|
|
|
|
swpEcosystemExpectedSpendableCoins := sdk.NewCoins(EcoSystemCoins)
|
|
|
|
foundEcosystem := false
|
|
|
|
|
|
|
|
migratedGenesisState := Auth(cdc, genesisState, GenesisTime)
|
|
|
|
|
|
|
|
for _, acc := range migratedGenesisState.Accounts {
|
|
|
|
if acc.GetAddress().Equals(SwpTreasuryAddr) {
|
|
|
|
foundTreasury = true
|
|
|
|
pva, ok := acc.(*vesting.PeriodicVestingAccount)
|
|
|
|
require.True(t, ok)
|
|
|
|
spendableCoins := pva.SpendableCoins(GenesisTime)
|
|
|
|
require.Equal(t, swpTreasuryExpectedSpendableCoins, spendableCoins)
|
|
|
|
}
|
|
|
|
if acc.GetAddress().Equals(SwpTeamAddr) {
|
|
|
|
foundTeam = true
|
|
|
|
pva, ok := acc.(*vesting.PeriodicVestingAccount)
|
|
|
|
require.True(t, ok)
|
|
|
|
spendableCoins := pva.SpendableCoins(GenesisTime)
|
|
|
|
require.Equal(t, swpTeamExpectedSpendableCoins, spendableCoins)
|
|
|
|
}
|
|
|
|
if acc.GetAddress().Equals(KavaDistAddr) {
|
|
|
|
foundKavadist = true
|
|
|
|
spendableCoins := acc.SpendableCoins(GenesisTime)
|
|
|
|
spendableSwp := sdk.NewCoin("swp", spendableCoins.AmountOf("swp"))
|
|
|
|
require.Equal(t, kavaDistExpectedSpendableCoins, spendableSwp)
|
|
|
|
}
|
|
|
|
if acc.GetAddress().Equals(SwpEcoSystemAddr) {
|
|
|
|
foundEcosystem = true
|
|
|
|
spendableCoins := acc.SpendableCoins(GenesisTime)
|
|
|
|
require.Equal(t, swpEcosystemExpectedSpendableCoins, spendableCoins)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
require.True(t, foundTreasury)
|
|
|
|
require.True(t, foundTeam)
|
|
|
|
require.True(t, foundKavadist)
|
|
|
|
require.True(t, foundEcosystem)
|
|
|
|
|
|
|
|
}
|