diff --git a/CHANGELOG.md b/CHANGELOG.md index e3f2ad584f0f..6ec1f2e5132e 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -36,7 +36,9 @@ Ref: https://keepachangelog.com/en/1.0.0/ # Changelog -## [Unreleased] +## Unreleased + +* (cache-store) [#52](https://github.com/evmos/cosmos-sdk/pull/52) Add a deep copy method for the store. ## [v0.50.9](https://github.com/cosmos/cosmos-sdk/releases/tag/v0.50.9) - 2024-08-07 diff --git a/store/cachekv/store.go b/store/cachekv/store.go index 08cfc2b325f6..aafc669285f4 100644 --- a/store/cachekv/store.go +++ b/store/cachekv/store.go @@ -171,6 +171,38 @@ func (store *Store) CacheWrapWithTrace(w io.Writer, tc types.TraceContext) types return NewStore(tracekv.NewStore(store, w, tc)) } +// Copy creates a deep copy of the Store object +func (store *Store) Copy() *Store { + store.mtx.Lock() + defer store.mtx.Unlock() + + // Copy cache + cacheCopy := make(map[string]*cValue, len(store.cache)) + for key, val := range store.cache { + newVal := *val // Create a copy of the cValue + cacheCopy[key] = &newVal + } + + // Copy unsortedCache + unsortedCacheCopy := make(map[string]struct{}, len(store.unsortedCache)) + for key := range store.unsortedCache { + unsortedCacheCopy[key] = struct{}{} + } + + // Copy sortedCache + sortedCacheCopy := store.sortedCache.Copy() + + // Create new Store with copied values + newStore := &Store{ + cache: cacheCopy, + unsortedCache: unsortedCacheCopy, + sortedCache: sortedCacheCopy, + parent: store.parent, + } + + return newStore +} + //---------------------------------------- // Iteration diff --git a/store/cachemulti/store.go b/store/cachemulti/store.go index 696911370c5d..5e5359a46ea2 100644 --- a/store/cachemulti/store.go +++ b/store/cachemulti/store.go @@ -168,3 +168,35 @@ func (cms Store) GetKVStore(key types.StoreKey) types.KVStore { } return store.(types.KVStore) } + +// Copy creates a deep copy of the Store object +func (cms Store) Copy() types.CacheMultiStore { + // Deep copy the db field + newDB := cms.db.(*cachekv.Store).Copy() + + // Deep copy the cachekv stores map + newStores := make(map[types.StoreKey]types.CacheWrap, len(cms.stores)) + for key, store := range cms.stores { + store, ok := store.(*cachekv.Store) + if ok { + newStores[key] = store.Copy() + } + } + + // Deep copy the keys map + newKeys := make(map[string]types.StoreKey, len(cms.keys)) + for key, value := range cms.keys { + newKeys[key] = value + } + + // Create new Store with copied values + newStore := Store{ + db: newDB, + stores: newStores, + keys: newKeys, + traceWriter: cms.traceWriter, + traceContext: cms.traceContext, + } + + return newStore +} diff --git a/store/types/store.go b/store/types/store.go index 8980179950e2..64e6d855143b 100644 --- a/store/types/store.go +++ b/store/types/store.go @@ -157,7 +157,8 @@ type MultiStore interface { // From MultiStore.CacheMultiStore().... type CacheMultiStore interface { MultiStore - Write() // Writes operations to underlying KVStore + Write() // Writes operations to underlying KVStore + Copy() CacheMultiStore // Returns a deep copy of the CacheMultiStore } // CommitMultiStore is an interface for a MultiStore without cache capabilities.