Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

chore(store/v2): Increase storev2 tests coverage #21094

Merged
merged 39 commits into from
Aug 9, 2024
Merged
Show file tree
Hide file tree
Changes from 10 commits
Commits
Show all changes
39 commits
Select commit Hold shift + click to select a range
5565d5b
add signal commit test
sontrinh16 Jul 24, 2024
2de8c05
minor
sontrinh16 Jul 24, 2024
631fe55
add start migration test
sontrinh16 Jul 26, 2024
d7b6628
add test for commitment
sontrinh16 Jul 28, 2024
d5dbeed
minor
sontrinh16 Jul 28, 2024
f5af8b4
add commitment get test
sontrinh16 Jul 28, 2024
fd2d1da
undo
sontrinh16 Jul 28, 2024
4b7c856
fix test
sontrinh16 Jul 28, 2024
67afb09
add restore test for storage
sontrinh16 Jul 29, 2024
343d39d
lint
sontrinh16 Jul 29, 2024
002d0ee
add more snapshot test
sontrinh16 Jul 29, 2024
f2a50a7
add more test
sontrinh16 Jul 29, 2024
3da8451
cleanup
sontrinh16 Jul 29, 2024
1b8f120
Merge branch 'main' into son/increase_storev2_tests
sontrinh16 Jul 29, 2024
1ef2eb4
fix
sontrinh16 Jul 29, 2024
c500af3
Merge branch 'son/increase_storev2_tests' of https://github.com/cosmo…
sontrinh16 Jul 29, 2024
e4734f2
fix test
sontrinh16 Jul 29, 2024
6cba8f4
make test deterministic
sontrinh16 Jul 29, 2024
f30bff4
Merge branch 'main' into son/increase_storev2_tests
sontrinh16 Jul 29, 2024
760e53a
fix
sontrinh16 Jul 30, 2024
985c6c9
Merge branch 'son/increase_storev2_tests' of https://github.com/cosmo…
sontrinh16 Jul 30, 2024
25f1365
Merge branch 'main' into son/increase_storev2_tests
sontrinh16 Jul 30, 2024
827b78c
address comments
sontrinh16 Jul 30, 2024
36b6d5a
Merge branch 'son/increase_storev2_tests' of https://github.com/cosmo…
sontrinh16 Jul 30, 2024
fc2e743
Merge branch 'main' into son/increase_storev2_tests
sontrinh16 Aug 1, 2024
b1f61d6
update
sontrinh16 Aug 2, 2024
16e59c3
Merge branch 'main' into son/increase_storev2_tests
sontrinh16 Aug 2, 2024
da6a3cf
add mock consume chStorage
sontrinh16 Aug 5, 2024
59bb870
Merge branch 'son/increase_storev2_tests' of https://github.com/cosmo…
sontrinh16 Aug 5, 2024
86e1f95
resolve conflict
sontrinh16 Aug 6, 2024
0bf8e10
Merge branch 'main' into son/increase_storev2_tests
sontrinh16 Aug 7, 2024
4e868ef
remove API
sontrinh16 Aug 7, 2024
496c501
Merge branch 'son/increase_storev2_tests' of https://github.com/cosmo…
sontrinh16 Aug 7, 2024
660d861
Merge branch 'main' into son/increase_storev2_tests
sontrinh16 Aug 7, 2024
25624c6
Merge branch 'main' into son/increase_storev2_tests
sontrinh16 Aug 8, 2024
989940e
remove API
sontrinh16 Aug 8, 2024
67032e4
Merge branch 'son/increase_storev2_tests' of https://github.com/cosmo…
sontrinh16 Aug 8, 2024
74ded9f
Merge branch 'main' into son/increase_storev2_tests
sontrinh16 Aug 9, 2024
90f53e6
Merge branch 'main' into son/increase_storev2_tests
sontrinh16 Aug 9, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
77 changes: 77 additions & 0 deletions store/v2/commitment/store_test_suite.go
Original file line number Diff line number Diff line change
Expand Up @@ -164,3 +164,80 @@ func (s *CommitStoreTestSuite) TestStore_Pruning() {
}
}
}

func (s *CommitStoreTestSuite) TestStore_GetProof() {
storeKeys := []string{storeKey1, storeKey2}
commitStore, err := s.NewStore(dbm.NewMemDB(), storeKeys, coretesting.NewNopLogger())
s.Require().NoError(err)

toVersion := uint64(10)
keyCount := 5

// commit some changes
for version := uint64(1); version <= toVersion; version++ {
cs := corestore.NewChangeset()
for _, storeKey := range storeKeys {
for i := 0; i < keyCount; i++ {
cs.Add([]byte(storeKey), []byte(fmt.Sprintf("key-%d-%d", version, i)), []byte(fmt.Sprintf("value-%d-%d", version, i)), false)
}
}
err := commitStore.WriteChangeset(cs)
s.Require().NoError(err)
_, err = commitStore.Commit(version)
s.Require().NoError(err)
}

// get proof
for version := uint64(1); version <= toVersion; version++ {
for _, storeKey := range storeKeys {
for i := 0; i < keyCount; i++ {
_, err := commitStore.GetProof([]byte(storeKey), version, []byte(fmt.Sprintf("key-%d-%d", version, i)))
s.Require().NoError(err)
}
}
}

// prune version 1
s.Require().NoError(commitStore.Prune(1))

// check if proof for version 1 is pruned
_, err = commitStore.GetProof([]byte(storeKeys[0]), 1, []byte(fmt.Sprintf("key-%d-%d", 1, 0)))
s.Require().Error(err)
// check the commit info
commit, _ := commitStore.GetCommitInfo(1)
s.Require().Nil(commit)
}

func (s *CommitStoreTestSuite) TestStore_Get() {
storeKeys := []string{storeKey1, storeKey2}
commitStore, err := s.NewStore(dbm.NewMemDB(), storeKeys, coretesting.NewNopLogger())
s.Require().NoError(err)

toVersion := uint64(10)
keyCount := 5

// commit some changes
for version := uint64(1); version <= toVersion; version++ {
cs := corestore.NewChangeset()
for _, storeKey := range storeKeys {
for i := 0; i < keyCount; i++ {
cs.Add([]byte(storeKey), []byte(fmt.Sprintf("key-%d-%d", version, i)), []byte(fmt.Sprintf("value-%d-%d", version, i)), false)
}
}
err := commitStore.WriteChangeset(cs)
s.Require().NoError(err)
_, err = commitStore.Commit(version)
s.Require().NoError(err)
}

// get proof
for version := uint64(1); version <= toVersion; version++ {
for _, storeKey := range storeKeys {
for i := 0; i < keyCount; i++ {
val, err := commitStore.Get([]byte(storeKey), version, []byte(fmt.Sprintf("key-%d-%d", version, i)))
s.Require().NoError(err)
s.Require().Equal([]byte(fmt.Sprintf("value-%d-%d", version, i)), val)
}
}
}
}
101 changes: 101 additions & 0 deletions store/v2/migration/manager_test.go
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
package migration

import (
"encoding/binary"
"fmt"
"testing"

Expand Down Expand Up @@ -117,3 +118,103 @@ func TestMigrateState(t *testing.T) {
})
}
}

func TestStartMigrateState(t *testing.T) {
for _, noCommitStore := range []bool{false, true} {
t.Run(fmt.Sprintf("Migrate noCommitStore=%v", noCommitStore), func(t *testing.T) {
m, orgCommitStore := setupMigrationManager(t, noCommitStore)

chDone := make(chan struct{})
chChangeset := make(chan *VersionedChangeset, 1)

// apply changeset
toVersion := uint64(10)
keyCount := 5
changesets := []corestore.Changeset{}

for version := uint64(1); version <= toVersion; version++ {
cs := corestore.NewChangeset()
for _, storeKey := range storeKeys {
for i := 0; i < keyCount; i++ {
cs.Add([]byte(storeKey), []byte(fmt.Sprintf("key-%d-%d", version, i)), []byte(fmt.Sprintf("value-%d-%d", version, i)), false)
}
}
changesets = append(changesets, *cs)
require.NoError(t, orgCommitStore.WriteChangeset(cs))
_, err := orgCommitStore.Commit(version)
require.NoError(t, err)
}

// feed changesets to channel
go func() {
for version := uint64(1); version <= toVersion; version++ {
chChangeset <- &VersionedChangeset{
Version: version,
Changeset: &changesets[version-1],
}
}
}()

// check if migrate process complete
go func() {
for {
migrateVersion := m.GetMigratedVersion()
if migrateVersion == toVersion-1 {
break
}
}

chDone <- struct{}{}
}()

err := m.Start(toVersion-1, chChangeset, chDone)
require.NoError(t, err)

// expecting error for conflicting process, since Migrate trigger snapshotter create migration,
// which start a snapshot process already.
_, err = m.snapshotsManager.Create(toVersion - 1)
require.Error(t, err)

if m.stateCommitment != nil {
// check the migrated state
for version := uint64(1); version < toVersion; version++ {
for _, storeKey := range storeKeys {
for i := 0; i < keyCount; i++ {
val, err := m.stateCommitment.Get([]byte(storeKey), toVersion-1, []byte(fmt.Sprintf("key-%d-%d", version, i)))
require.NoError(t, err)
require.Equal(t, []byte(fmt.Sprintf("value-%d-%d", version, i)), val)
}
}
}
// check the latest state
val, err := m.stateCommitment.Get([]byte("store1"), toVersion-1, []byte("key-100-1"))
require.NoError(t, err)
require.Nil(t, val)
val, err = m.stateCommitment.Get([]byte("store2"), toVersion-1, []byte("key-100-0"))
require.NoError(t, err)
require.Nil(t, val)
}

// check the storage
for version := uint64(1); version < toVersion; version++ {
for _, storeKey := range storeKeys {
for i := 0; i < keyCount; i++ {
val, err := m.stateStorage.Get([]byte(storeKey), toVersion-1, []byte(fmt.Sprintf("key-%d-%d", version, i)))
require.NoError(t, err)
require.Equal(t, []byte(fmt.Sprintf("value-%d-%d", version, i)), val)
}
}
}

// check if migration db write change set to storage
for version := uint64(1); version < toVersion; version++ {
buf := make([]byte, 8)
binary.BigEndian.PutUint64(buf, version)
csKey := []byte(fmt.Sprintf(migrateChangesetKeyFmt, buf))
csVal, err := m.db.Get(csKey)
require.NoError(t, err)
require.NotEmpty(t, csVal)
}
})
}
}
104 changes: 104 additions & 0 deletions store/v2/pruning/manager_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -152,3 +152,107 @@ func TestPruningOption(t *testing.T) {
})
}
}

func (s *PruningManagerTestSuite) TestSignalCommit() {
// commit version 1
cs := corestore.NewChangeset()
for _, storeKey := range storeKeys {
cs.Add([]byte(storeKey), []byte(fmt.Sprintf("key-%d-%d", 1, 0)), []byte(fmt.Sprintf("value-%d-%d", 1, 0)), false)
}

s.Require().NoError(s.sc.WriteChangeset(cs))
_, err := s.sc.Commit(1)
s.Require().NoError(err)

s.Require().NoError(s.ss.ApplyChangeset(1, cs))

// commit version 2
for _, storeKey := range storeKeys {
cs.Add([]byte(storeKey), []byte(fmt.Sprintf("key-%d-%d", 2, 0)), []byte(fmt.Sprintf("value-%d-%d", 2, 0)), false)
}

// signaling commit has started
s.Require().NoError(s.manager.SignalCommit(true, 2))

s.Require().NoError(s.sc.WriteChangeset(cs))
_, err = s.sc.Commit(2)
s.Require().NoError(err)

s.Require().NoError(s.ss.ApplyChangeset(2, cs))

// try prune before signaling commit has finished
s.Require().NoError(s.manager.Prune(2))

// proof is removed no matter SignalCommit has not yet inform that commit process has finish
// since commitInfo is remove async with tree data
checkSCPrune := func() bool {
count := 0
for _, storeKey := range storeKeys {
_, err := s.sc.GetProof([]byte(storeKey), 1, []byte(fmt.Sprintf("key-%d-%d", 1, 0)))
if err != nil {
count++
}
}

return count == len(storeKeys)
}
s.Require().Eventually(checkSCPrune, 10*time.Second, 1*time.Second)

// data from state commitment should not be pruned since we haven't signal the commit process has finished
val, err := s.sc.Get([]byte(storeKeys[0]), 1, []byte(fmt.Sprintf("key-%d-%d", 1, 0)))
s.Require().NoError(err)
s.Require().Equal(val, []byte(fmt.Sprintf("value-%d-%d", 1, 0)))

// signaling commit has finished, version 1 should be pruned
s.Require().NoError(s.manager.SignalCommit(false, 2))

checkSCPrune = func() bool {
count := 0
for _, storeKey := range storeKeys {
_, err := s.sc.GetProof([]byte(storeKey), 1, []byte(fmt.Sprintf("key-%d-%d", 1, 0)))
if err != nil {
count++
}
}

return count == len(storeKeys)
}
s.Require().Eventually(checkSCPrune, 10*time.Second, 1*time.Second)

// try with signal commit start and finish accordingly
// commit changesets with pruning
toVersion := uint64(100)
keyCount := 10
for version := uint64(3); version <= toVersion; version++ {
cs := corestore.NewChangeset()
for _, storeKey := range storeKeys {
for i := 0; i < keyCount; i++ {
cs.Add([]byte(storeKey), []byte(fmt.Sprintf("key-%d-%d", version, i)), []byte(fmt.Sprintf("value-%d-%d", version, i)), false)
}
}
s.Require().NoError(s.manager.SignalCommit(true, version))

s.Require().NoError(s.sc.WriteChangeset(cs))
_, err := s.sc.Commit(version)
s.Require().NoError(err)

s.Require().NoError(s.ss.ApplyChangeset(version, cs))

s.Require().NoError(s.manager.SignalCommit(false, version))

}

// wait for the pruning to finish in the commitment store
checkSCPrune = func() bool {
count := 0
for _, storeKey := range storeKeys {
_, err := s.sc.GetProof([]byte(storeKey), toVersion-1, []byte(fmt.Sprintf("key-%d-%d", toVersion-1, 0)))
if err != nil {
count++
}
}

return count == len(storeKeys)
}
s.Require().Eventually(checkSCPrune, 10*time.Second, 1*time.Second)
}
2 changes: 1 addition & 1 deletion store/v2/root/store_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -630,7 +630,7 @@ func (s *RootStoreTestSuite) TestMultiStore_PruningRestart() {
return false
}
// wait for async pruning process to finish
s.Require().Eventually(checkErr, 5*time.Second, 100*time.Millisecond, "expected error when loading height: %d", v)
s.Require().Eventually(checkErr, 10*time.Second, 1*time.Second, "expected error when loading height: %d", v)
}
}

Expand Down
66 changes: 66 additions & 0 deletions store/v2/storage/storage_test_suite.go
Original file line number Diff line number Diff line change
Expand Up @@ -640,6 +640,72 @@
s.Require().Equal([]byte("val200"), bz)
}

func (s *StorageTestSuite) TestDatabase_Restore() {
db, err := s.NewDB(s.T().TempDir())
s.Require().NoError(err)
defer db.Close()

toVersion := uint64(10)
keyCount := 10

// for versions 1-10, set 10 keys
for v := uint64(1); v <= toVersion; v++ {
cs := corestore.NewChangesetWithPairs(map[string]corestore.KVPairs{storeKey1: {}})
for i := 0; i < keyCount; i++ {
key := fmt.Sprintf("key%03d", i)
val := fmt.Sprintf("val%03d-%03d", i, v)

cs.AddKVPair(storeKey1Bytes, corestore.KVPair{Key: []byte(key), Value: []byte(val)})
}

s.Require().NoError(db.ApplyChangeset(v, cs))
}

latestVersion, err := db.GetLatestVersion()
s.Require().NoError(err)
s.Require().Equal(uint64(10), latestVersion)

chStorage := make(chan *corestore.StateChanges, 5)

go func() {
for i := uint64(11); i <= 15; i++ {
kvPairs := []corestore.KVPair{}
for j := 0; j < keyCount; j++ {
key := fmt.Sprintf("key%03d-%03d", j, i)
val := fmt.Sprintf("val%03d-%03d", j, i)

kvPairs = append(kvPairs, corestore.KVPair{Key: []byte(key), Value: []byte(val)})
}
chStorage <- &corestore.StateChanges{
Actor: storeKey1Bytes,
StateChanges: kvPairs,
}
}
close(chStorage)
}()
Dismissed Show dismissed Hide dismissed

// restore
err = db.Restore(9, chStorage)
s.Require().Error(err)

// restore
err = db.Restore(11, chStorage)
s.Require().NoError(err)

// check the storage
for i := uint64(11); i <= 15; i++ {
for j := 0; j < keyCount; j++ {
key := fmt.Sprintf("key%03d-%03d", j, i)
val := fmt.Sprintf("val%03d-%03d", j, i)

v, err := db.Get(storeKey1Bytes, 11, []byte(key))
s.Require().NoError(err)
fmt.Println(string(v), val, i)
s.Require().Equal([]byte(val), v)
}
}
}

func DBApplyChangeset(
t *testing.T,
db store.VersionedDatabase,
Expand Down
Loading