package cachekv_test import ( "fmt" "testing" "github.com/stretchr/testify/require" tmrand "github.com/tendermint/tendermint/libs/rand" dbm "github.com/tendermint/tm-db" "github.com/cosmos/cosmos-sdk/store/cachekv" "github.com/cosmos/cosmos-sdk/store/dbadapter" "github.com/cosmos/cosmos-sdk/store/types" ) func newCacheKVStore() types.CacheKVStore { mem := dbadapter.Store{DB: dbm.NewMemDB()} return cachekv.NewStore(mem) } func keyFmt(i int) []byte { return bz(fmt.Sprintf("key%0.8d", i)) } func valFmt(i int) []byte { return bz(fmt.Sprintf("value%0.8d", i)) } func TestCacheKVStore(t *testing.T) { mem := dbadapter.Store{DB: dbm.NewMemDB()} st := cachekv.NewStore(mem) require.Empty(t, st.Get(keyFmt(1)), "Expected `key1` to be empty") // put something in mem and in cache mem.Set(keyFmt(1), valFmt(1)) st.Set(keyFmt(1), valFmt(1)) require.Equal(t, valFmt(1), st.Get(keyFmt(1))) // update it in cache, shoudn't change mem st.Set(keyFmt(1), valFmt(2)) require.Equal(t, valFmt(2), st.Get(keyFmt(1))) require.Equal(t, valFmt(1), mem.Get(keyFmt(1))) // write it. should change mem st.Write() require.Equal(t, valFmt(2), mem.Get(keyFmt(1))) require.Equal(t, valFmt(2), st.Get(keyFmt(1))) // more writes and checks st.Write() st.Write() require.Equal(t, valFmt(2), mem.Get(keyFmt(1))) require.Equal(t, valFmt(2), st.Get(keyFmt(1))) // make a new one, check it st = cachekv.NewStore(mem) require.Equal(t, valFmt(2), st.Get(keyFmt(1))) // make a new one and delete - should not be removed from mem st = cachekv.NewStore(mem) st.Delete(keyFmt(1)) require.Empty(t, st.Get(keyFmt(1))) require.Equal(t, mem.Get(keyFmt(1)), valFmt(2)) // Write. should now be removed from both st.Write() require.Empty(t, st.Get(keyFmt(1)), "Expected `key1` to be empty") require.Empty(t, mem.Get(keyFmt(1)), "Expected `key1` to be empty") } func TestCacheKVStoreNoNilSet(t *testing.T) { mem := dbadapter.Store{DB: dbm.NewMemDB()} st := cachekv.NewStore(mem) require.Panics(t, func() { st.Set([]byte("key"), nil) }, "setting a nil value should panic") require.Panics(t, func() { st.Set(nil, []byte("value")) }, "setting a nil key should panic") require.Panics(t, func() { st.Set([]byte(""), []byte("value")) }, "setting an empty key should panic") } func TestCacheKVStoreNested(t *testing.T) { mem := dbadapter.Store{DB: dbm.NewMemDB()} st := cachekv.NewStore(mem) // set. check its there on st and not on mem. st.Set(keyFmt(1), valFmt(1)) require.Empty(t, mem.Get(keyFmt(1))) require.Equal(t, valFmt(1), st.Get(keyFmt(1))) // make a new from st and check st2 := cachekv.NewStore(st) require.Equal(t, valFmt(1), st2.Get(keyFmt(1))) // update the value on st2, check it only effects st2 st2.Set(keyFmt(1), valFmt(3)) require.Equal(t, []byte(nil), mem.Get(keyFmt(1))) require.Equal(t, valFmt(1), st.Get(keyFmt(1))) require.Equal(t, valFmt(3), st2.Get(keyFmt(1))) // st2 writes to its parent, st. doesnt effect mem st2.Write() require.Equal(t, []byte(nil), mem.Get(keyFmt(1))) require.Equal(t, valFmt(3), st.Get(keyFmt(1))) // updates mem st.Write() require.Equal(t, valFmt(3), mem.Get(keyFmt(1))) } func TestCacheKVIteratorBounds(t *testing.T) { st := newCacheKVStore() // set some items nItems := 5 for i := 0; i < nItems; i++ { st.Set(keyFmt(i), valFmt(i)) } // iterate over all of them itr := st.Iterator(nil, nil) var i = 0 for ; itr.Valid(); itr.Next() { k, v := itr.Key(), itr.Value() require.Equal(t, keyFmt(i), k) require.Equal(t, valFmt(i), v) i++ } require.Equal(t, nItems, i) // iterate over none itr = st.Iterator(bz("money"), nil) i = 0 for ; itr.Valid(); itr.Next() { i++ } require.Equal(t, 0, i) // iterate over lower itr = st.Iterator(keyFmt(0), keyFmt(3)) i = 0 for ; itr.Valid(); itr.Next() { k, v := itr.Key(), itr.Value() require.Equal(t, keyFmt(i), k) require.Equal(t, valFmt(i), v) i++ } require.Equal(t, 3, i) // iterate over upper itr = st.Iterator(keyFmt(2), keyFmt(4)) i = 2 for ; itr.Valid(); itr.Next() { k, v := itr.Key(), itr.Value() require.Equal(t, keyFmt(i), k) require.Equal(t, valFmt(i), v) i++ } require.Equal(t, 4, i) } func TestCacheKVMergeIteratorBasics(t *testing.T) { st := newCacheKVStore() // set and delete an item in the cache, iterator should be empty k, v := keyFmt(0), valFmt(0) st.Set(k, v) st.Delete(k) assertIterateDomain(t, st, 0) // now set it and assert its there st.Set(k, v) assertIterateDomain(t, st, 1) // write it and assert its there st.Write() assertIterateDomain(t, st, 1) // remove it in cache and assert its not st.Delete(k) assertIterateDomain(t, st, 0) // write the delete and assert its not there st.Write() assertIterateDomain(t, st, 0) // add two keys and assert theyre there k1, v1 := keyFmt(1), valFmt(1) st.Set(k, v) st.Set(k1, v1) assertIterateDomain(t, st, 2) // write it and assert theyre there st.Write() assertIterateDomain(t, st, 2) // remove one in cache and assert its not st.Delete(k1) assertIterateDomain(t, st, 1) // write the delete and assert its not there st.Write() assertIterateDomain(t, st, 1) // delete the other key in cache and asserts its empty st.Delete(k) assertIterateDomain(t, st, 0) } func TestCacheKVMergeIteratorDeleteLast(t *testing.T) { st := newCacheKVStore() // set some items and write them nItems := 5 for i := 0; i < nItems; i++ { st.Set(keyFmt(i), valFmt(i)) } st.Write() // set some more items and leave dirty for i := nItems; i < nItems*2; i++ { st.Set(keyFmt(i), valFmt(i)) } // iterate over all of them assertIterateDomain(t, st, nItems*2) // delete them all for i := 0; i < nItems*2; i++ { last := nItems*2 - 1 - i st.Delete(keyFmt(last)) assertIterateDomain(t, st, last) } } func TestCacheKVMergeIteratorDeletes(t *testing.T) { st := newCacheKVStore() truth := dbm.NewMemDB() // set some items and write them nItems := 10 for i := 0; i < nItems; i++ { doOp(t, st, truth, opSet, i) } st.Write() // delete every other item, starting from 0 for i := 0; i < nItems; i += 2 { doOp(t, st, truth, opDel, i) assertIterateDomainCompare(t, st, truth) } // reset st = newCacheKVStore() truth = dbm.NewMemDB() // set some items and write them for i := 0; i < nItems; i++ { doOp(t, st, truth, opSet, i) } st.Write() // delete every other item, starting from 1 for i := 1; i < nItems; i += 2 { doOp(t, st, truth, opDel, i) assertIterateDomainCompare(t, st, truth) } } func TestCacheKVMergeIteratorChunks(t *testing.T) { st := newCacheKVStore() // Use the truth to check values on the merge iterator truth := dbm.NewMemDB() // sets to the parent setRange(t, st, truth, 0, 20) setRange(t, st, truth, 40, 60) st.Write() // sets to the cache setRange(t, st, truth, 20, 40) setRange(t, st, truth, 60, 80) assertIterateDomainCheck(t, st, truth, []keyRange{{0, 80}}) // remove some parents and some cache deleteRange(t, st, truth, 15, 25) assertIterateDomainCheck(t, st, truth, []keyRange{{0, 15}, {25, 80}}) // remove some parents and some cache deleteRange(t, st, truth, 35, 45) assertIterateDomainCheck(t, st, truth, []keyRange{{0, 15}, {25, 35}, {45, 80}}) // write, add more to the cache, and delete some cache st.Write() setRange(t, st, truth, 38, 42) deleteRange(t, st, truth, 40, 43) assertIterateDomainCheck(t, st, truth, []keyRange{{0, 15}, {25, 35}, {38, 40}, {45, 80}}) } func TestCacheKVMergeIteratorRandom(t *testing.T) { st := newCacheKVStore() truth := dbm.NewMemDB() start, end := 25, 975 max := 1000 setRange(t, st, truth, start, end) // do an op, test the iterator for i := 0; i < 2000; i++ { doRandomOp(t, st, truth, max) assertIterateDomainCompare(t, st, truth) } } //------------------------------------------------------------------------------------------- // do some random ops const ( opSet = 0 opSetRange = 1 opDel = 2 opDelRange = 3 opWrite = 4 totalOps = 5 // number of possible operations ) func randInt(n int) int { return tmrand.NewRand().Int() % n } // useful for replaying a error case if we find one func doOp(t *testing.T, st types.CacheKVStore, truth dbm.DB, op int, args ...int) { switch op { case opSet: k := args[0] st.Set(keyFmt(k), valFmt(k)) err := truth.Set(keyFmt(k), valFmt(k)) require.NoError(t, err) case opSetRange: start := args[0] end := args[1] setRange(t, st, truth, start, end) case opDel: k := args[0] st.Delete(keyFmt(k)) err := truth.Delete(keyFmt(k)) require.NoError(t, err) case opDelRange: start := args[0] end := args[1] deleteRange(t, st, truth, start, end) case opWrite: st.Write() } } func doRandomOp(t *testing.T, st types.CacheKVStore, truth dbm.DB, maxKey int) { r := randInt(totalOps) switch r { case opSet: k := randInt(maxKey) st.Set(keyFmt(k), valFmt(k)) err := truth.Set(keyFmt(k), valFmt(k)) require.NoError(t, err) case opSetRange: start := randInt(maxKey - 2) end := randInt(maxKey-start) + start setRange(t, st, truth, start, end) case opDel: k := randInt(maxKey) st.Delete(keyFmt(k)) err := truth.Delete(keyFmt(k)) require.NoError(t, err) case opDelRange: start := randInt(maxKey - 2) end := randInt(maxKey-start) + start deleteRange(t, st, truth, start, end) case opWrite: st.Write() } } //------------------------------------------------------------------------------------------- // iterate over whole domain func assertIterateDomain(t *testing.T, st types.KVStore, expectedN int) { itr := st.Iterator(nil, nil) var i = 0 for ; itr.Valid(); itr.Next() { k, v := itr.Key(), itr.Value() require.Equal(t, keyFmt(i), k) require.Equal(t, valFmt(i), v) i++ } require.Equal(t, expectedN, i) } func assertIterateDomainCheck(t *testing.T, st types.KVStore, mem dbm.DB, r []keyRange) { // iterate over each and check they match the other itr := st.Iterator(nil, nil) itr2, err := mem.Iterator(nil, nil) // ground truth require.NoError(t, err) krc := newKeyRangeCounter(r) i := 0 for ; krc.valid(); krc.next() { require.True(t, itr.Valid()) require.True(t, itr2.Valid()) // check the key/val matches the ground truth k, v := itr.Key(), itr.Value() k2, v2 := itr2.Key(), itr2.Value() require.Equal(t, k, k2) require.Equal(t, v, v2) // check they match the counter require.Equal(t, k, keyFmt(krc.key())) itr.Next() itr2.Next() i++ } require.False(t, itr.Valid()) require.False(t, itr2.Valid()) } func assertIterateDomainCompare(t *testing.T, st types.KVStore, mem dbm.DB) { // iterate over each and check they match the other itr := st.Iterator(nil, nil) itr2, err := mem.Iterator(nil, nil) // ground truth require.NoError(t, err) checkIterators(t, itr, itr2) checkIterators(t, itr2, itr) } func checkIterators(t *testing.T, itr, itr2 types.Iterator) { for ; itr.Valid(); itr.Next() { require.True(t, itr2.Valid()) k, v := itr.Key(), itr.Value() k2, v2 := itr2.Key(), itr2.Value() require.Equal(t, k, k2) require.Equal(t, v, v2) itr2.Next() } require.False(t, itr.Valid()) require.False(t, itr2.Valid()) } //-------------------------------------------------------- func setRange(t *testing.T, st types.KVStore, mem dbm.DB, start, end int) { for i := start; i < end; i++ { st.Set(keyFmt(i), valFmt(i)) err := mem.Set(keyFmt(i), valFmt(i)) require.NoError(t, err) } } func deleteRange(t *testing.T, st types.KVStore, mem dbm.DB, start, end int) { for i := start; i < end; i++ { st.Delete(keyFmt(i)) err := mem.Delete(keyFmt(i)) require.NoError(t, err) } } //-------------------------------------------------------- type keyRange struct { start int end int } func (kr keyRange) len() int { return kr.end - kr.start } func newKeyRangeCounter(kr []keyRange) *keyRangeCounter { return &keyRangeCounter{keyRanges: kr} } // we can iterate over this and make sure our real iterators have all the right keys type keyRangeCounter struct { rangeIdx int idx int keyRanges []keyRange } func (krc *keyRangeCounter) valid() bool { maxRangeIdx := len(krc.keyRanges) - 1 maxRange := krc.keyRanges[maxRangeIdx] // if we're not in the max range, we're valid if krc.rangeIdx <= maxRangeIdx && krc.idx < maxRange.len() { return true } return false } func (krc *keyRangeCounter) next() { thisKeyRange := krc.keyRanges[krc.rangeIdx] if krc.idx == thisKeyRange.len()-1 { krc.rangeIdx++ krc.idx = 0 } else { krc.idx++ } } func (krc *keyRangeCounter) key() int { thisKeyRange := krc.keyRanges[krc.rangeIdx] return thisKeyRange.start + krc.idx } //-------------------------------------------------------- func bz(s string) []byte { return []byte(s) } func BenchmarkCacheKVStoreGetNoKeyFound(b *testing.B) { b.ReportAllocs() st := newCacheKVStore() b.ResetTimer() // assumes b.N < 2**24 for i := 0; i < b.N; i++ { st.Get([]byte{byte((i & 0xFF0000) >> 16), byte((i & 0xFF00) >> 8), byte(i & 0xFF)}) } } func BenchmarkCacheKVStoreGetKeyFound(b *testing.B) { b.ReportAllocs() st := newCacheKVStore() for i := 0; i < b.N; i++ { arr := []byte{byte((i & 0xFF0000) >> 16), byte((i & 0xFF00) >> 8), byte(i & 0xFF)} st.Set(arr, arr) } b.ResetTimer() // assumes b.N < 2**24 for i := 0; i < b.N; i++ { st.Get([]byte{byte((i & 0xFF0000) >> 16), byte((i & 0xFF00) >> 8), byte(i & 0xFF)}) } }