-
Notifications
You must be signed in to change notification settings - Fork 3.8k
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
feat(store/v2): add WorkingHash #20706
Changes from 2 commits
0cc9a87
bd65ac6
fbbe274
e38630e
ff4aa41
46767e2
faf3033
ee8a324
a23ee4a
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
|
@@ -35,7 +35,8 @@ type Store struct { | |||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// stateCommitment reflects the state commitment (SC) backend | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
stateCommitment store.Committer | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// commitHeader reflects the header used when committing state (note, this isn't required and only used for query purposes) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// commitHeader reflects the header used when committing state | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// note, this isn't required and only used for query purposes) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
commitHeader *coreheader.Info | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// lastCommitInfo reflects the last version/hash that has been committed | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
@@ -257,6 +258,40 @@ func (s *Store) SetCommitHeader(h *coreheader.Info) { | |||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
s.commitHeader = h | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// WorkingHash writes the changeset to SC and SS and returns the workingHash | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// of the CommitInfo. | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
func (s *Store) WorkingHash(cs *corestore.Changeset) ([]byte, error) { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if s.telemetry != nil { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
now := time.Now() | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
defer s.telemetry.MeasureSince(now, "root_store", "working_hash") | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// write the changeset to the SC and SS backends | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
eg := new(errgroup.Group) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
eg.Go(func() error { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if err := s.writeSC(cs); err != nil { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return fmt.Errorf("failed to write SC: %w", err) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return nil | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
}) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
eg.Go(func() error { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if err := s.stateStorage.ApplyChangeset(s.initialVersion, cs); err != nil { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return fmt.Errorf("failed to commit SS: %w", err) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return nil | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
}) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if err := eg.Wait(); err != nil { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return nil, err | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
workingHash := s.lastCommitInfo.Hash() | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
s.lastCommitInfo.Version -= 1 // reset lastCommitInfo to allow Commit() to work correctly | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return workingHash, nil | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
Comment on lines
+265
to
+297
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Refactor the The - if err := s.writeSC(cs); err != nil {
- return fmt.Errorf("failed to write SC: %w", err)
- }
+ scErr := s.writeSC(cs)
+ ssErr := s.stateStorage.ApplyChangeset(s.initialVersion, cs)
+ if scErr != nil || ssErr != nil {
+ return nil, fmt.Errorf("SC error: %w, SS error: %w", scErr, ssErr)
+ } Committable suggestion
Suggested change
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// Commit commits all state changes to the underlying SS and SC backends. It | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// writes a batch of the changeset to the SC tree, and retrieves the CommitInfo | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// from the SC tree. Finally, it commits the SC tree and returns the hash of the | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
@@ -388,13 +423,8 @@ func (s *Store) writeSC(cs *corestore.Changeset) error { | |||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return fmt.Errorf("failed to write batch to SC store: %w", err) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
isEmpty, err := s.stateCommitment.IsEmpty() | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if err != nil { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
return fmt.Errorf("failed to check if SC store is empty: %w", err) | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
} | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
var previousHeight, version uint64 | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if isEmpty { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
if s.lastCommitInfo.GetVersion() == 0 && s.initialVersion > 1 { | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// This case means that no commit has been made in the store, we | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
// start from initialVersion. | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
version = s.initialVersion | ||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -74,7 +74,38 @@ func (s *RootStoreTestSuite) TestGetStateStorage() { | |
} | ||
|
||
func (s *RootStoreTestSuite) TestSetInitialVersion() { | ||
s.Require().NoError(s.rootStore.SetInitialVersion(100)) | ||
initialVersion := uint64(5) | ||
s.Require().NoError(s.rootStore.SetInitialVersion(initialVersion)) | ||
|
||
// perform the initial commit | ||
cs := corestore.NewChangeset() | ||
cs.Add(testStoreKeyBytes, []byte("foo"), []byte("bar"), false) | ||
|
||
wHash, err := s.rootStore.WorkingHash(cs) | ||
s.Require().NoError(err) | ||
cHash, err := s.rootStore.Commit(corestore.NewChangeset()) | ||
s.Require().NoError(err) | ||
s.Require().Equal(wHash, cHash) | ||
|
||
// check the latest version | ||
lVersion, err := s.rootStore.GetLatestVersion() | ||
s.Require().NoError(err) | ||
s.Require().Equal(initialVersion, lVersion) | ||
|
||
// set the initial version again | ||
rInitialVersion := uint64(100) | ||
s.Require().NoError(s.rootStore.SetInitialVersion(rInitialVersion)) | ||
|
||
// perform the commit | ||
cs = corestore.NewChangeset() | ||
cs.Add(testStoreKey2Bytes, []byte("foo"), []byte("bar"), false) | ||
_, err = s.rootStore.Commit(cs) | ||
s.Require().NoError(err) | ||
lVersion, err = s.rootStore.GetLatestVersion() | ||
s.Require().NoError(err) | ||
// SetInitialVersion only works once | ||
s.Require().NotEqual(rInitialVersion, lVersion) | ||
s.Require().Equal(initialVersion+1, lVersion) | ||
} | ||
|
||
func (s *RootStoreTestSuite) TestSetCommitHeader() { | ||
|
@@ -309,3 +340,26 @@ func (s *RootStoreTestSuite) TestStateAt() { | |
} | ||
} | ||
} | ||
|
||
func (s *RootStoreTestSuite) TestWorkingHash() { | ||
// write keys over multiple versions | ||
for v := uint64(1); v <= 5; v++ { | ||
// perform changes | ||
cs := corestore.NewChangeset() | ||
for _, storeKeyBytes := range [][]byte{testStoreKeyBytes, testStoreKey2Bytes, testStoreKey3Bytes} { | ||
for i := 0; i < 100; i++ { | ||
key := fmt.Sprintf("key_%x_%03d", i, storeKeyBytes) // key000, key001, ..., key099 | ||
val := fmt.Sprintf("val%03d_%03d", i, v) // val000_1, val001_1, ..., val099_1 | ||
|
||
cs.Add(storeKeyBytes, []byte(key), []byte(val), false) | ||
} | ||
} | ||
|
||
wHash, err := s.rootStore.WorkingHash(cs) | ||
s.Require().NoError(err) | ||
// execute Commit with empty changeset | ||
cHash, err := s.rootStore.Commit(corestore.NewChangeset()) | ||
s.Require().NoError(err) | ||
s.Require().Equal(wHash, cHash) | ||
} | ||
} | ||
Comment on lines
+382
to
+403
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Improved TestWorkingHash, but consider additional scenarios. The updated Consider adding tests that modify the state between Would you like me to help draft these additional test scenarios? |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -49,11 +49,16 @@ type RootStore interface { | |
// queries based on block time need to be supported. | ||
SetCommitHeader(h *coreheader.Info) | ||
|
||
// WorkingHash returns the current WIP commitment hash by applying the Changeset | ||
// to the SC backend. It is only used to get the hash of the intermediate state | ||
// before committing, the typical use case is for the genesis block. | ||
// NOTE: It also writes the changeset to the SS backend. | ||
WorkingHash(cs *corestore.Changeset) ([]byte, error) | ||
Comment on lines
+52
to
+56
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Add a detailed method comment for The |
||
|
||
// Commit should be responsible for taking the provided changeset and flushing | ||
// it to disk. Note, depending on the implementation, the changeset, at this | ||
// point, may already be written to the SC backends. Commit() should ensure | ||
// the changeset is committed to all SC and SC backends and flushed to disk. | ||
// It must return a hash of the merkle-ized committed state. | ||
// it to disk. Note, it will overwrite the changeset if WorkingHash() was called. | ||
// Commit() should ensure the changeset is committed to all SC and SS backends | ||
// and flushed to disk. It must return a hash of the merkle-ized committed state. | ||
Comment on lines
+59
to
+61
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Clarify the The comment on the |
||
Commit(cs *corestore.Changeset) ([]byte, error) | ||
|
||
// LastCommitID returns a CommitID pertaining to the last commitment. | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Clarify the use of
commitHeader
in theStore
struct.The comment for
commitHeader
should clarify its exact role, especially how it interacts with other components and any implications it has on the state or behavior of the store.