github.com/Finschia/finschia-sdk@v0.49.1/x/params/keeper/keeper_test.go (about) 1 package keeper_test 2 3 import ( 4 "reflect" 5 "testing" 6 7 "github.com/stretchr/testify/require" 8 "github.com/stretchr/testify/suite" 9 tmproto "github.com/tendermint/tendermint/proto/tendermint/types" 10 11 "github.com/Finschia/finschia-sdk/baseapp" 12 "github.com/Finschia/finschia-sdk/simapp" 13 "github.com/Finschia/finschia-sdk/store/prefix" 14 sdk "github.com/Finschia/finschia-sdk/types" 15 "github.com/Finschia/finschia-sdk/x/params/types" 16 "github.com/Finschia/finschia-sdk/x/params/types/proposal" 17 ) 18 19 type KeeperTestSuite struct { 20 suite.Suite 21 22 app *simapp.SimApp 23 ctx sdk.Context 24 25 queryClient proposal.QueryClient 26 } 27 28 func (suite *KeeperTestSuite) SetupTest() { 29 suite.app, suite.ctx = createTestApp(true) 30 31 queryHelper := baseapp.NewQueryServerTestHelper(suite.ctx, suite.app.InterfaceRegistry()) 32 proposal.RegisterQueryServer(queryHelper, suite.app.ParamsKeeper) 33 suite.queryClient = proposal.NewQueryClient(queryHelper) 34 } 35 36 func TestKeeperTestSuite(t *testing.T) { 37 suite.Run(t, new(KeeperTestSuite)) 38 } 39 40 // returns context and app 41 func createTestApp(isCheckTx bool) (*simapp.SimApp, sdk.Context) { 42 app := simapp.Setup(isCheckTx) 43 ctx := app.BaseApp.NewContext(isCheckTx, tmproto.Header{}) 44 45 return app, ctx 46 } 47 48 func validateNoOp(_ interface{}) error { return nil } 49 50 func TestKeeper(t *testing.T) { 51 kvs := []struct { 52 key string 53 param int64 54 }{ 55 {"key1", 10}, 56 {"key2", 55}, 57 {"key3", 182}, 58 {"key4", 17582}, 59 {"key5", 2768554}, 60 {"key6", 1157279}, 61 {"key7", 9058701}, 62 } 63 64 table := types.NewKeyTable( 65 types.NewParamSetPair([]byte("key1"), int64(0), validateNoOp), 66 types.NewParamSetPair([]byte("key2"), int64(0), validateNoOp), 67 types.NewParamSetPair([]byte("key3"), int64(0), validateNoOp), 68 types.NewParamSetPair([]byte("key4"), int64(0), validateNoOp), 69 types.NewParamSetPair([]byte("key5"), int64(0), validateNoOp), 70 types.NewParamSetPair([]byte("key6"), int64(0), validateNoOp), 71 types.NewParamSetPair([]byte("key7"), int64(0), validateNoOp), 72 types.NewParamSetPair([]byte("extra1"), bool(false), validateNoOp), 73 types.NewParamSetPair([]byte("extra2"), string(""), validateNoOp), 74 ) 75 76 cdc, ctx, skey, keeper := testComponents() 77 78 store := prefix.NewStore(ctx.KVStore(skey), []byte("test/")) 79 space := keeper.Subspace("test") 80 require.False(t, space.HasKeyTable()) 81 space = space.WithKeyTable(table) 82 require.True(t, space.HasKeyTable()) 83 84 // Set params 85 for i, kv := range kvs { 86 require.NotPanics(t, func() { space.Set(ctx, []byte(kv.key), kv.param) }, "space.Set panics, tc #%d", i) 87 } 88 89 // Test space.Get 90 for i, kv := range kvs { 91 var param int64 92 require.NotPanics(t, func() { space.Get(ctx, []byte(kv.key), ¶m) }, "space.Get panics, tc #%d", i) 93 require.Equal(t, kv.param, param, "stored param not equal, tc #%d", i) 94 } 95 96 // Test space.GetRaw 97 for i, kv := range kvs { 98 var param int64 99 bz := space.GetRaw(ctx, []byte(kv.key)) 100 err := cdc.UnmarshalJSON(bz, ¶m) 101 require.Nil(t, err, "err is not nil, tc #%d", i) 102 require.Equal(t, kv.param, param, "stored param not equal, tc #%d", i) 103 } 104 105 // Test store.Get equals space.Get 106 for i, kv := range kvs { 107 var param int64 108 bz := store.Get([]byte(kv.key)) 109 require.NotNil(t, bz, "KVStore.Get returns nil, tc #%d", i) 110 err := cdc.UnmarshalJSON(bz, ¶m) 111 require.NoError(t, err, "UnmarshalJSON returns error, tc #%d", i) 112 require.Equal(t, kv.param, param, "stored param not equal, tc #%d", i) 113 } 114 115 // Test invalid space.Get 116 for i, kv := range kvs { 117 var param bool 118 require.Panics(t, func() { space.Get(ctx, []byte(kv.key), ¶m) }, "invalid space.Get not panics, tc #%d", i) 119 } 120 121 // Test invalid space.Set 122 for i, kv := range kvs { 123 require.Panics(t, func() { space.Set(ctx, []byte(kv.key), true) }, "invalid space.Set not panics, tc #%d", i) 124 } 125 126 // Test GetSubspace 127 for i, kv := range kvs { 128 var gparam, param int64 129 gspace, ok := keeper.GetSubspace("test") 130 require.True(t, ok, "cannot retrieve subspace, tc #%d", i) 131 132 require.NotPanics(t, func() { gspace.Get(ctx, []byte(kv.key), &gparam) }) 133 require.NotPanics(t, func() { space.Get(ctx, []byte(kv.key), ¶m) }) 134 require.Equal(t, gparam, param, "GetSubspace().Get not equal with space.Get, tc #%d", i) 135 136 require.NotPanics(t, func() { gspace.Set(ctx, []byte(kv.key), int64(i)) }) 137 require.NotPanics(t, func() { space.Get(ctx, []byte(kv.key), ¶m) }) 138 require.Equal(t, int64(i), param, "GetSubspace().Set not equal with space.Get, tc #%d", i) 139 } 140 } 141 142 func indirect(ptr interface{}) interface{} { 143 return reflect.ValueOf(ptr).Elem().Interface() 144 } 145 146 func TestSubspace(t *testing.T) { 147 cdc, ctx, key, keeper := testComponents() 148 149 kvs := []struct { 150 key string 151 param interface{} 152 zero interface{} 153 ptr interface{} 154 }{ 155 {"string", "test", "", new(string)}, 156 {"bool", true, false, new(bool)}, 157 {"int16", int16(1), int16(0), new(int16)}, 158 {"int32", int32(1), int32(0), new(int32)}, 159 {"int64", int64(1), int64(0), new(int64)}, 160 {"uint16", uint16(1), uint16(0), new(uint16)}, 161 {"uint32", uint32(1), uint32(0), new(uint32)}, 162 {"uint64", uint64(1), uint64(0), new(uint64)}, 163 {"int", sdk.NewInt(1), sdk.Int{}, new(sdk.Int)}, 164 {"uint", sdk.NewUint(1), sdk.Uint{}, new(sdk.Uint)}, 165 {"dec", sdk.NewDec(1), sdk.Dec{}, new(sdk.Dec)}, 166 {"struct", s{1}, s{0}, new(s)}, 167 } 168 169 table := types.NewKeyTable( 170 types.NewParamSetPair([]byte("string"), "", validateNoOp), 171 types.NewParamSetPair([]byte("bool"), false, validateNoOp), 172 types.NewParamSetPair([]byte("int16"), int16(0), validateNoOp), 173 types.NewParamSetPair([]byte("int32"), int32(0), validateNoOp), 174 types.NewParamSetPair([]byte("int64"), int64(0), validateNoOp), 175 types.NewParamSetPair([]byte("uint16"), uint16(0), validateNoOp), 176 types.NewParamSetPair([]byte("uint32"), uint32(0), validateNoOp), 177 types.NewParamSetPair([]byte("uint64"), uint64(0), validateNoOp), 178 types.NewParamSetPair([]byte("int"), sdk.Int{}, validateNoOp), 179 types.NewParamSetPair([]byte("uint"), sdk.Uint{}, validateNoOp), 180 types.NewParamSetPair([]byte("dec"), sdk.Dec{}, validateNoOp), 181 types.NewParamSetPair([]byte("struct"), s{}, validateNoOp), 182 ) 183 184 store := prefix.NewStore(ctx.KVStore(key), []byte("test/")) 185 space := keeper.Subspace("test").WithKeyTable(table) 186 187 // Test space.Set 188 for i, kv := range kvs { 189 require.NotPanics(t, func() { space.Set(ctx, []byte(kv.key), kv.param) }, "space.Set panics, tc #%d", i) 190 } 191 192 // Test space.Get, space.GetIfExists 193 for i, kv := range kvs { 194 require.NotPanics(t, func() { space.GetIfExists(ctx, []byte("invalid"), kv.ptr) }, "space.GetIfExists panics when no value exists, tc #%d", i) 195 require.Equal(t, kv.zero, indirect(kv.ptr), "space.GetIfExists unmarshalls when no value exists, tc #%d", i) 196 require.Panics(t, func() { space.Get(ctx, []byte("invalid"), kv.ptr) }, "invalid space.Get not panics when no value exists, tc #%d", i) 197 require.Equal(t, kv.zero, indirect(kv.ptr), "invalid space.Get unmarshalls when no value exists, tc #%d", i) 198 199 require.NotPanics(t, func() { space.GetIfExists(ctx, []byte(kv.key), kv.ptr) }, "space.GetIfExists panics, tc #%d", i) 200 require.Equal(t, kv.param, indirect(kv.ptr), "stored param not equal, tc #%d", i) 201 require.NotPanics(t, func() { space.Get(ctx, []byte(kv.key), kv.ptr) }, "space.Get panics, tc #%d", i) 202 require.Equal(t, kv.param, indirect(kv.ptr), "stored param not equal, tc #%d", i) 203 204 require.Panics(t, func() { space.Get(ctx, []byte("invalid"), kv.ptr) }, "invalid space.Get not panics when no value exists, tc #%d", i) 205 require.Equal(t, kv.param, indirect(kv.ptr), "invalid space.Get unmarshalls when no value existt, tc #%d", i) 206 207 require.Panics(t, func() { space.Get(ctx, []byte(kv.key), nil) }, "invalid space.Get not panics when the pointer is nil, tc #%d", i) 208 require.Panics(t, func() { space.Get(ctx, []byte(kv.key), new(invalid)) }, "invalid space.Get not panics when the pointer is different type, tc #%d", i) 209 } 210 211 // Test store.Get equals space.Get 212 for i, kv := range kvs { 213 bz := store.Get([]byte(kv.key)) 214 require.NotNil(t, bz, "store.Get() returns nil, tc #%d", i) 215 err := cdc.UnmarshalJSON(bz, kv.ptr) 216 require.NoError(t, err, "cdc.UnmarshalJSON() returns error, tc #%d", i) 217 require.Equal(t, kv.param, indirect(kv.ptr), "stored param not equal, tc #%d", i) 218 } 219 } 220 221 type paramJSON struct { 222 Param1 int64 `json:"param1,omitempty" yaml:"param1,omitempty"` 223 Param2 string `json:"param2,omitempty" yaml:"param2,omitempty"` 224 } 225 226 func TestJSONUpdate(t *testing.T) { 227 _, ctx, _, keeper := testComponents() 228 229 key := []byte("key") 230 231 space := keeper.Subspace("test").WithKeyTable(types.NewKeyTable(types.NewParamSetPair(key, paramJSON{}, validateNoOp))) 232 233 var param paramJSON 234 235 err := space.Update(ctx, key, []byte(`{"param1": "10241024"}`)) 236 require.NoError(t, err) 237 space.Get(ctx, key, ¶m) 238 require.Equal(t, paramJSON{10241024, ""}, param) 239 240 err = space.Update(ctx, key, []byte(`{"param2": "helloworld"}`)) 241 require.NoError(t, err) 242 space.Get(ctx, key, ¶m) 243 require.Equal(t, paramJSON{10241024, "helloworld"}, param) 244 245 err = space.Update(ctx, key, []byte(`{"param1": "20482048"}`)) 246 require.NoError(t, err) 247 space.Get(ctx, key, ¶m) 248 require.Equal(t, paramJSON{20482048, "helloworld"}, param) 249 250 err = space.Update(ctx, key, []byte(`{"param1": "40964096", "param2": "goodbyeworld"}`)) 251 require.NoError(t, err) 252 space.Get(ctx, key, ¶m) 253 require.Equal(t, paramJSON{40964096, "goodbyeworld"}, param) 254 }