diff --git a/go.mod b/go.mod index 0c601d437643..f895922b126b 100644 --- a/go.mod +++ b/go.mod @@ -26,7 +26,7 @@ require ( github.com/cosmos/go-bip39 v1.0.0 github.com/cosmos/gogogateway v1.2.0 github.com/cosmos/gogoproto v1.4.10 - github.com/cosmos/iavl v1.1.2-0.20240322194608-8a38162c01cd + github.com/cosmos/iavl v1.1.2-0.20240404133024-ff3cbb4c8c1c github.com/cosmos/ics23/go v0.10.0 github.com/cosmos/ledger-cosmos-go v0.12.1 github.com/golang/mock v1.6.0 diff --git a/go.sum b/go.sum index cc637438b37b..6f626804e261 100644 --- a/go.sum +++ b/go.sum @@ -363,8 +363,8 @@ github.com/cosmos/gogogateway v1.2.0/go.mod h1:iQpLkGWxYcnCdz5iAdLcRBSw3h7NXeOkZ github.com/cosmos/gogoproto v1.4.2/go.mod h1:cLxOsn1ljAHSV527CHOtaIP91kK6cCrZETRBrkzItWU= github.com/cosmos/gogoproto v1.4.10 h1:QH/yT8X+c0F4ZDacDv3z+xE3WU1P1Z3wQoLMBRJoKuI= github.com/cosmos/gogoproto v1.4.10/go.mod h1:3aAZzeRWpAwr+SS/LLkICX2/kDFyaYVzckBDzygIxek= -github.com/cosmos/iavl v1.1.2-0.20240322194608-8a38162c01cd h1:tta7R0AxvI6QgSB4Rkbpvn+hkwG9PrOEfcXMPWoWx4s= -github.com/cosmos/iavl v1.1.2-0.20240322194608-8a38162c01cd/go.mod h1:jLeUvm6bGT1YutCaL2fIar/8vGUE8cPZvh/gXEWDaDM= +github.com/cosmos/iavl v1.1.2-0.20240404133024-ff3cbb4c8c1c h1:mJSbQubMa4urYbWpbgVf+H+UQDJXAe3DpTn2izmN3a0= +github.com/cosmos/iavl v1.1.2-0.20240404133024-ff3cbb4c8c1c/go.mod h1:jLeUvm6bGT1YutCaL2fIar/8vGUE8cPZvh/gXEWDaDM= github.com/cosmos/ics23/go v0.10.0 h1:iXqLLgp2Lp+EdpIuwXTYIQU+AiHj9mOC2X9ab++bZDM= github.com/cosmos/ics23/go v0.10.0/go.mod h1:ZfJSmng/TBNTBkFemHHHj5YY7VAU/MBU980F4VU1NG0= github.com/cosmos/keyring v1.2.0 h1:8C1lBP9xhImmIabyXW4c3vFjjLiBdGCmfLUfeZlV1Yo= diff --git a/server/mock/store.go b/server/mock/store.go index bece8fb6d0e4..b891d6ad216c 100644 --- a/server/mock/store.go +++ b/server/mock/store.go @@ -62,6 +62,14 @@ func (ms multiStore) Commit() storetypes.CommitID { panic("not implemented") } +func (ms multiStore) SetCommitting() { + panic("not implemented") +} + +func (ms multiStore) UnsetCommitting() { + panic("not implemented") +} + func (ms multiStore) LastCommitID() storetypes.CommitID { panic("not implemented") } diff --git a/store/iavl/store.go b/store/iavl/store.go index 9d7c7315e75f..80cec4bf4b86 100644 --- a/store/iavl/store.go +++ b/store/iavl/store.go @@ -54,7 +54,8 @@ func LoadStore(db dbm.DB, logger log.Logger, key types.StoreKey, id types.Commit // provided DB. An error is returned if the version fails to load, or if called with a positive // version on an empty tree. func LoadStoreWithInitialVersion(db dbm.DB, logger log.Logger, key types.StoreKey, id types.CommitID, initialVersion uint64, cacheSize int, disableFastNode bool) (types.CommitKVStore, error) { - tree := iavl.NewMutableTree(wrapper.NewIAVLDB(db), cacheSize, disableFastNode, clog.NewNopLogger(), iavl.InitialVersionOption(initialVersion)) + // Create a new IAVL tree with the async pruning enabled + tree := iavl.NewMutableTree(wrapper.NewIAVLDB(db), cacheSize, disableFastNode, clog.NewNopLogger(), iavl.InitialVersionOption(initialVersion), iavl.AsyncPruningOption(true)) isUpgradeable, err := tree.IsUpgradeable() if err != nil { @@ -117,6 +118,17 @@ func (st *Store) GetImmutable(version int64) (*Store, error) { }, nil } +// SetCommitting marks the store as committing, which will prevent any +// parallel writes to the store. It is referenced in the async pruning. +func (st *Store) SetCommitting() { + st.tree.SetCommitting() +} + +// UnsetCommitting marks the store as not committing. +func (st *Store) UnsetCommitting() { + st.tree.UnsetCommitting() +} + // Commit commits the current store state and returns a CommitID with the new // version and hash. func (st *Store) Commit() types.CommitID { diff --git a/store/iavl/tree.go b/store/iavl/tree.go index 300a6c311236..bff4ca869fbf 100644 --- a/store/iavl/tree.go +++ b/store/iavl/tree.go @@ -23,6 +23,8 @@ type ( Get(key []byte) ([]byte, error) Set(key, value []byte) (bool, error) Remove(key []byte) ([]byte, bool, error) + SetCommitting() + UnsetCommitting() SaveVersion() ([]byte, int64, error) DeleteVersionsTo(version int64) error Version() int64 @@ -55,6 +57,14 @@ func (it *immutableTree) Remove(_ []byte) ([]byte, bool, error) { panic("cannot call 'Remove' on an immutable IAVL tree") } +func (it *immutableTree) SetCommitting() { + panic("cannot call 'SetCommitting' on an immutable IAVL tree") +} + +func (it *immutableTree) UnsetCommitting() { + panic("cannot call 'UnsetCommitting' on an immutable IAVL tree") +} + func (it *immutableTree) SaveVersion() ([]byte, int64, error) { panic("cannot call 'SaveVersion' on an immutable IAVL tree") } diff --git a/store/mem/store.go b/store/mem/store.go index bba5fde8d0d9..fb540b935d02 100644 --- a/store/mem/store.go +++ b/store/mem/store.go @@ -49,6 +49,12 @@ func (s Store) CacheWrapWithTrace(w io.Writer, tc types.TraceContext) types.Cach // Commit performs a no-op as entries are persistent between commitments. func (s *Store) Commit() (id types.CommitID) { return } +// Implements CommitStore. +func (s *Store) SetCommitting() {} + +// Implements CommitStore. +func (s *Store) UnsetCommitting() {} + func (s *Store) SetPruning(pruning pruningtypes.PruningOptions) {} // GetPruning is a no-op as pruning options cannot be directly set on this store. diff --git a/store/rootmulti/dbadapter.go b/store/rootmulti/dbadapter.go index 02b11ebd3f2a..5682ccf44820 100644 --- a/store/rootmulti/dbadapter.go +++ b/store/rootmulti/dbadapter.go @@ -32,6 +32,10 @@ func (cdsa commitDBStoreAdapter) LastCommitID() types.CommitID { } } +func (cdsa commitDBStoreAdapter) SetCommitting() {} + +func (cdsa commitDBStoreAdapter) UnsetCommitting() {} + func (cdsa commitDBStoreAdapter) SetPruning(_ pruningtypes.PruningOptions) {} // GetPruning is a no-op as pruning options cannot be directly set on this store. diff --git a/store/rootmulti/store.go b/store/rootmulti/store.go index e38a048436ee..9ce3ea60027c 100644 --- a/store/rootmulti/store.go +++ b/store/rootmulti/store.go @@ -451,7 +451,9 @@ func (rs *Store) Commit() types.CommitID { rs.logger.Debug("commit header and version mismatch", "header_height", rs.commitHeader.Height, "version", version) } + rs.SetCommitting() rs.lastCommitInfo = commitStores(version, rs.stores, rs.removalMap) + rs.UnsetCommitting() rs.lastCommitInfo.Timestamp = rs.commitHeader.Time defer rs.flushMetadata(rs.db, version, rs.lastCommitInfo) @@ -477,6 +479,20 @@ func (rs *Store) Commit() types.CommitID { } } +// SetCommitting implements Committer/CommitStore. +func (rs *Store) SetCommitting() { + for _, store := range rs.stores { + store.SetCommitting() + } +} + +// UnsetCommitting implements Committer/CommitStore. +func (rs *Store) UnsetCommitting() { + for _, store := range rs.stores { + store.UnsetCommitting() + } +} + // CacheWrap implements CacheWrapper/Store/CommitStore. func (rs *Store) CacheWrap() types.CacheWrap { return rs.CacheMultiStore().(types.CacheWrap) diff --git a/store/rootmulti/store_test.go b/store/rootmulti/store_test.go index 1d283a04c0aa..f46d02deb734 100644 --- a/store/rootmulti/store_test.go +++ b/store/rootmulti/store_test.go @@ -511,6 +511,10 @@ func TestMultiStore_Pruning(t *testing.T) { ms.Commit() } + // asyn pruning, simulate the consensus process + time.Sleep(150 * time.Millisecond) + ms.Commit() + for _, v := range tc.saved { _, err := ms.CacheMultiStoreWithVersion(v) require.NoError(t, err, "expected no error when loading height: %d", v) @@ -548,16 +552,6 @@ func TestMultiStore_Pruning_SameHeightsTwice(t *testing.T) { require.Equal(t, numVersions, lastCommitInfo.Version) - for v := int64(1); v < numVersions-int64(keepRecent); v++ { - err := ms.LoadVersion(v) - require.Error(t, err, "expected error when loading pruned height: %d", v) - } - - for v := int64(numVersions - int64(keepRecent)); v < numVersions; v++ { - err := ms.LoadVersion(v) - require.NoError(t, err, "expected no error when loading height: %d", v) - } - // Get latest err := ms.LoadVersion(numVersions - 1) require.NoError(t, err) @@ -574,8 +568,19 @@ func TestMultiStore_Pruning_SameHeightsTwice(t *testing.T) { require.Equal(t, numVersions, lastCommitInfo.Version) // Ensure that can commit one more height with no panic + time.Sleep(150 * time.Millisecond) lastCommitInfo = ms.Commit() require.Equal(t, numVersions+1, lastCommitInfo.Version) + + for v := int64(1); v < numVersions-int64(keepRecent); v++ { + err := ms.LoadVersion(v) + require.Error(t, err, "expected error when loading pruned height: %d", v) + } + + for v := int64(numVersions - int64(keepRecent)); v < numVersions; v++ { + err := ms.LoadVersion(v) + require.NoError(t, err, "expected no error when loading height: %d", v) + } } func TestMultiStore_PruningRestart(t *testing.T) { @@ -613,6 +618,8 @@ func TestMultiStore_PruningRestart(t *testing.T) { // commit one more block and ensure the heights have been pruned ms.Commit() + time.Sleep(150 * time.Millisecond) + ms.Commit() actualHeightsToPrune, err = ms.pruningManager.GetFlushAndResetPruningHeights() require.NoError(t, err) diff --git a/store/transient/store.go b/store/transient/store.go index 531cd8e25ca7..8be7dd12e39c 100644 --- a/store/transient/store.go +++ b/store/transient/store.go @@ -43,6 +43,12 @@ func (ts *Store) LastCommitID() (id types.CommitID) { return } +// Implements CommitStore +func (ts *Store) SetCommitting() {} + +// Implements CommitStore +func (ts *Store) UnsetCommitting() {} + // Implements Store. func (ts *Store) GetStoreType() types.StoreType { return types.StoreTypeTransient diff --git a/store/types/store.go b/store/types/store.go index d14c5aebccb1..d26c82bc3726 100644 --- a/store/types/store.go +++ b/store/types/store.go @@ -21,6 +21,8 @@ type Store interface { type Committer interface { Commit() CommitID LastCommitID() CommitID + SetCommitting() + UnsetCommitting() SetPruning(pruningtypes.PruningOptions) GetPruning() pruningtypes.PruningOptions