Skip to content

Commit

Permalink
Merge pull request #271 from cosmos/bugfix/stable-app-hash
Browse files Browse the repository at this point in the history
Bugfix: stable app hash
  • Loading branch information
ebuchman authored Oct 27, 2017
2 parents fc82979 + aa87858 commit f9dde9f
Show file tree
Hide file tree
Showing 2 changed files with 83 additions and 27 deletions.
65 changes: 38 additions & 27 deletions state/kvcache.go
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,9 @@ package state

// MemKVCache is designed to wrap MemKVStore as a cache
type MemKVCache struct {
store SimpleDB
cache *MemKVStore
store SimpleDB
readCache *MemKVStore
writeCache *MemKVStore
}

var _ SimpleDB = (*MemKVCache)(nil)
Expand All @@ -18,23 +19,34 @@ func NewMemKVCache(store SimpleDB) *MemKVCache {
}

return &MemKVCache{
store: store,
cache: NewMemKVStore(),
store: store,
readCache: NewMemKVStore(),
writeCache: NewMemKVStore(),
}
}

// Set sets a key, fulfills KVStore interface
func (c *MemKVCache) Set(key []byte, value []byte) {
c.cache.Set(key, value)
// if we write it, remove from read cache
c.readCache.Remove(key)
c.writeCache.Set(key, value)
}

// Get gets a key, fulfills KVStore interface
func (c *MemKVCache) Get(key []byte) (value []byte) {
value, ok := c.cache.m[string(key)]
if !ok {
value = c.store.Get(key)
c.cache.Set(key, value)
// see if we have a cached write
value, ok := c.writeCache.m[string(key)]
if ok {
return value
}
// then a cached read
value, ok = c.readCache.m[string(key)]
if ok {
return value
}
// then read directly and cache it
value = c.store.Get(key)
c.readCache.Set(key, value)
return value
}

Expand All @@ -44,19 +56,19 @@ func (c *MemKVCache) Has(key []byte) bool {
return value != nil
}

// Remove uses nil value as a flag to delete... not ideal but good enough
// for testing
// Remove uses nil value as a flag to delete...
// not ideal but good enough for testing
func (c *MemKVCache) Remove(key []byte) (value []byte) {
value = c.Get(key)
c.cache.Set(key, nil)
c.Set(key, nil)
return value
}

// List is also inefficiently implemented...
func (c *MemKVCache) List(start, end []byte, limit int) []Model {
orig := c.store.List(start, end, 0)
cached := c.cache.List(start, end, 0)
keys := c.combineLists(orig, cached)
writen := c.writeCache.List(start, end, 0)
keys := c.combineLists(orig, writen)

// apply limit (too late)
if limit > 0 && len(keys) > 0 {
Expand All @@ -69,19 +81,17 @@ func (c *MemKVCache) List(start, end []byte, limit int) []Model {
return keys
}

func (c *MemKVCache) combineLists(orig, cache []Model) []Model {
func (c *MemKVCache) combineLists(lists ...[]Model) []Model {
store := NewMemKVStore()
for _, m := range orig {
store.Set(m.Key, m.Value)
}
for _, m := range cache {
if m.Value == nil {
store.Remove([]byte(m.Key))
} else {
store.Set([]byte(m.Key), m.Value)
for _, cache := range lists {
for _, m := range cache {
if m.Value == nil {
store.Remove([]byte(m.Key))
} else {
store.Set([]byte(m.Key), m.Value)
}
}
}

return store.List(nil, nil, 0)
}

Expand Down Expand Up @@ -131,8 +141,8 @@ func (c *MemKVCache) Commit(sub SimpleDB) error {

// applyCache will apply all the cache methods to the underlying store
func (c *MemKVCache) applyCache() {
for _, k := range c.cache.keysInRange(nil, nil) {
v := c.cache.m[k]
for _, k := range c.writeCache.keysInRange(nil, nil) {
v := c.writeCache.m[k]
if v == nil {
c.store.Remove([]byte(k))
} else {
Expand All @@ -143,5 +153,6 @@ func (c *MemKVCache) applyCache() {

// Discard will remove reference to this
func (c *MemKVCache) Discard() {
c.cache = NewMemKVStore()
c.readCache = NewMemKVStore()
c.writeCache = NewMemKVStore()
}
45 changes: 45 additions & 0 deletions state/merkle_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -99,3 +99,48 @@ func TestStateCommitHash(t *testing.T) {
}

}

// Ensure getting data doesn't cause it to write
func TestGetDoesntWrite(t *testing.T) {
assert, require := assert.New(t), require.New(t)

// make the store...
tree := iavl.NewVersionedTree(0, db.NewMemDB())
store := NewState(tree, 5)

k, v := []byte("foo"), []byte("bar")
k2, v2 := []byte("abc"), []byte("def")
nok := []byte("baz")

// one set should change something
store.Append().Set(k, v)
hash, err := store.Commit(1)
require.NoError(err)
require.NotNil(hash)
// calling hash returns last committed state
hash1 := store.LatestHash()
require.Equal(hash, hash1)

// a second set will update the state
store.Append().Set(k2, v2)
hash, err = store.Commit(2)
require.NoError(err)
assert.NotEqual(hash1, hash)
// calling hash returns last committed state
hash2 := store.LatestHash()
require.Equal(hash, hash2)

// a missed get will not do anything....
val := store.Append().Get(nok)
assert.Nil(val)
hash, err = store.Commit(3)
require.NoError(err)
assert.Equal(hash2, hash)

// a proper get will not do anything....
val = store.Append().Get(k2)
assert.Equal(v2, val)
hash, err = store.Commit(4)
require.NoError(err)
assert.Equal(hash2, hash)
}

0 comments on commit f9dde9f

Please sign in to comment.