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

feat(rpc): Implement state_getReadProof rpc call #1768

Merged
merged 48 commits into from
Sep 30, 2021
Merged
Show file tree
Hide file tree
Changes from 8 commits
Commits
Show all changes
48 commits
Select commit Hold shift + click to select a range
0e5afc8
feat: implement chain subscribe all heads
EclesioMeloJunior Aug 19, 2021
4586709
chore: add unit tests to subscribe all heads
EclesioMeloJunior Aug 20, 2021
a47cc08
Merge branch 'development' into eclesio/rpc-subscribe-all-heads
EclesioMeloJunior Aug 20, 2021
91c0756
chore: fix imports
EclesioMeloJunior Aug 23, 2021
659c503
feat: implement state_getReadProof rpc call
EclesioMeloJunior Aug 25, 2021
8160592
chore: fix unused import
EclesioMeloJunior Aug 25, 2021
3f016f9
Merge branch 'development' into eclesio/get-read-proof-rpc
EclesioMeloJunior Aug 25, 2021
8d1b73b
chore: fix lint warnings
EclesioMeloJunior Aug 25, 2021
30c4581
wip: proof generation algotithm
EclesioMeloJunior Sep 7, 2021
e0f4411
Merge branch 'development' into eclesio/get-read-proof-rpc
EclesioMeloJunior Sep 8, 2021
58c73bb
wip: append not working
EclesioMeloJunior Sep 14, 2021
4c1e63e
wip: fixing functions
EclesioMeloJunior Sep 15, 2021
26e1ad4
wip: remove unused code
EclesioMeloJunior Sep 15, 2021
1758518
feat: proof generation working
EclesioMeloJunior Sep 17, 2021
b52a767
chore: add multiple keys proof generation
EclesioMeloJunior Sep 17, 2021
b723923
chore: add unit tests
EclesioMeloJunior Sep 17, 2021
ae9e8ec
chore: add unit test
EclesioMeloJunior Sep 20, 2021
725c949
chore: remove database exposure
EclesioMeloJunior Sep 20, 2021
3079a97
chore: fix lint warnings
EclesioMeloJunior Sep 20, 2021
298eeaf
Merge branch 'development' into eclesio/get-read-proof-rpc
EclesioMeloJunior Sep 21, 2021
56a7263
chore: update GenerateTrieProof from block to storage state
EclesioMeloJunior Sep 21, 2021
449acb4
chore: remove repeated code
EclesioMeloJunior Sep 22, 2021
6ce6884
chore: resolve fix lint
EclesioMeloJunior Sep 22, 2021
f6aafb0
chore: update tests
EclesioMeloJunior Sep 22, 2021
bdfbe80
chore: simplify to sliced value itself
EclesioMeloJunior Sep 22, 2021
7f130a8
chore: deepsource ignore auto generated files
EclesioMeloJunior Sep 22, 2021
e800b0b
chore: remove unused refs
EclesioMeloJunior Sep 22, 2021
76ff9c9
chore: addressing deepsource warngins
EclesioMeloJunior Sep 22, 2021
1758e91
chore: addressing deepsource wars
EclesioMeloJunior Sep 22, 2021
3abc4b0
Merge branch 'development' into eclesio/get-read-proof-rpc
EclesioMeloJunior Sep 23, 2021
6fd30fc
chore: remove DS_Store
EclesioMeloJunior Sep 24, 2021
c68fc54
chore: fix typo
EclesioMeloJunior Sep 24, 2021
ba421be
chore: use IsEmpty as record method
EclesioMeloJunior Sep 24, 2021
d652444
chore: update comment export
EclesioMeloJunior Sep 24, 2021
9a06653
chore: resolve errors
EclesioMeloJunior Sep 27, 2021
b48fbce
chore: resolve exported
EclesioMeloJunior Sep 27, 2021
9fc6711
chore: resolve test to dont exported
EclesioMeloJunior Sep 27, 2021
87d9909
Merge branch 'development' into eclesio/get-read-proof-rpc
EclesioMeloJunior Sep 27, 2021
3406405
chore: resolve interface mocks
EclesioMeloJunior Sep 27, 2021
17c4302
chore: remove unused code
EclesioMeloJunior Sep 27, 2021
391401e
chore: resolve unues funcs
EclesioMeloJunior Sep 27, 2021
00d3f7c
chore: resolve failing unit tests
EclesioMeloJunior Sep 27, 2021
9861aa2
Merge branch 'development' into eclesio/get-read-proof-rpc
EclesioMeloJunior Sep 27, 2021
f5b151b
chore: update dot/core/services.go unit test coverage
EclesioMeloJunior Sep 28, 2021
73d6955
chore: remove skip from test
EclesioMeloJunior Sep 28, 2021
d840770
chore: remove comment
EclesioMeloJunior Sep 28, 2021
70adfe1
Merge branch 'development' into eclesio/get-read-proof-rpc
EclesioMeloJunior Sep 29, 2021
b403553
Merge branch 'development' into eclesio/get-read-proof-rpc
EclesioMeloJunior Sep 30, 2021
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
1 change: 1 addition & 0 deletions dot/core/interface.go
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,7 @@ type BlockState interface {
AddBlock(*types.Block) error
GetAllBlocksAtDepth(hash common.Hash) []common.Hash
GetBlockByHash(common.Hash) (*types.Block, error)
GetBlockStateRoot(bhash common.Hash) (common.Hash, error)
GenesisHash() common.Hash
GetSlotForBlock(common.Hash) (uint64, error)
GetFinalisedHeader(uint64, uint64) (*types.Header, error)
Expand Down
37 changes: 23 additions & 14 deletions dot/core/mocks/block_state.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

147 changes: 147 additions & 0 deletions dot/core/mocks/storage_state.go

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

47 changes: 47 additions & 0 deletions dot/core/service.go
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ import (
rtstorage "github.com/ChainSafe/gossamer/lib/runtime/storage"
"github.com/ChainSafe/gossamer/lib/services"
"github.com/ChainSafe/gossamer/lib/transaction"
"github.com/ChainSafe/gossamer/lib/trie"
"github.com/ChainSafe/gossamer/pkg/scale"
log "github.com/ChainSafe/log15"
)
Expand Down Expand Up @@ -618,3 +619,49 @@ func (s *Service) tryQueryStorage(block common.Hash, keys ...string) (QueryKeyVa

return changes, nil
}

// GetReadProofAt will return an array with the proofs for the keys passed as params
// based on the block hash passed as param as well, if block hash is nil then the current state will take place
func (s *Service) GetReadProofAt(block common.Hash, keys []common.Hash) (common.Hash, []string, error) {
if block == common.EmptyHash {
block = s.blockState.BestBlockHash()
}

stateRoot, err := s.blockState.GetBlockStateRoot(block)
if err != nil {
return common.EmptyHash, nil, err
}

ts, err := s.storageState.TrieState(&stateRoot)
if err != nil {
return common.EmptyHash, nil, err
}

proofForKeys, err := readProofForKeys(ts.Trie(), keys)
if err != nil {
return common.EmptyHash, nil, err
}

return block, proofForKeys, nil
noot marked this conversation as resolved.
Show resolved Hide resolved
}

// readProofForKeys will go through the keys and generate the proof for each of them
// and merge the result into a string array containing the hashes in the hexadecimal format
func readProofForKeys(t *trie.Trie, keys []common.Hash) ([]string, error) {
noot marked this conversation as resolved.
Show resolved Hide resolved
storageKeys := make([][]byte, len(keys))
for i, k := range keys {
storageKeys[i] = k.ToBytes()
}

proof, err := t.GenerateProof(storageKeys)
if err != nil {
return nil, err
}

proofSlice := make([]string, 0, len(proof))
for k := range proof {
proofSlice = append(proofSlice, k)
}

return proofSlice, nil
}
117 changes: 115 additions & 2 deletions dot/core/service_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,6 @@ import (
"time"

"github.com/ChainSafe/gossamer/dot/core/mocks"
coremocks "github.com/ChainSafe/gossamer/dot/core/mocks"
"github.com/ChainSafe/gossamer/dot/network"
"github.com/ChainSafe/gossamer/dot/state"
"github.com/ChainSafe/gossamer/dot/sync"
Expand Down Expand Up @@ -108,7 +107,7 @@ func TestStartService(t *testing.T) {
}

func TestAnnounceBlock(t *testing.T) {
net := new(coremocks.MockNetwork)
net := new(mocks.MockNetwork)
cfg := &Config{
Network: net,
}
Expand Down Expand Up @@ -819,3 +818,117 @@ func TestDecodeSessionKeys_WhenGetRuntimeReturnError(t *testing.T) {
require.Error(t, err, "problems")
require.Nil(t, b)
}

func TestGetReadProofAt_WhenReceivedBlockIsEmpty(t *testing.T) {
tr, entries := trie.RandomTrieTest(t, 50)
require.Len(t, entries, 50)

i := 0
var keysToProof []common.Hash
var keys [][]byte

// get the last 2 entries
for _, entry := range entries {
if i < len(entries)-2 {
i++
continue
}

keysToProof = append(keysToProof, common.BytesToHash(entry.K))
keys = append(keys, entry.K)
}
require.Len(t, keysToProof, 2)

expectedProof, err := tr.GenerateProof(keys)
require.NoError(t, err)

ts, err := storage.NewTrieState(tr)
require.NoError(t, err)
root := ts.MustRoot()

header, err := types.NewHeader(common.EmptyHash, root, common.EmptyHash, big.NewInt(1), nil)
require.NoError(t, err)
bestBlock := types.NewBlock(header, nil)

blockStateMock := new(mocks.MockBlockState)
blockStateMock.On("BestBlockHash").Return(bestBlock.Header.Hash())
blockStateMock.On("GetBlockStateRoot", bestBlock.Header.Hash()).Return(root, nil)

storageMock := new(mocks.MockStorageState)
storageMock.On("TrieState", &root).Return(ts, nil)

svc := new(Service)
svc.storageState = storageMock
svc.blockState = blockStateMock

block, proof, err := svc.GetReadProofAt(common.EmptyHash, keysToProof)
require.NoError(t, err)

blockStateMock.AssertCalled(t, "BestBlockHash")
blockStateMock.AssertCalled(t, "GetBlockStateRoot", bestBlock.Header.Hash())
storageMock.AssertCalled(t, "TrieState", &root)
require.Equal(t, bestBlock.Header.Hash(), block)

// all the proof must exists in the expected
require.Equal(t, len(expectedProof), len(proof))
for _, p := range proof {
_, has := expectedProof[p]
require.True(t, has)
}
}

func TestGetReadProofAt_OnSpecificBlock(t *testing.T) {
tr, entries := trie.RandomTrieTest(t, 50)
require.Len(t, entries, 50)

i := 0
var keysToProof []common.Hash
var keys [][]byte

// get the last 2 entries
for _, entry := range entries {
if i < len(entries)-2 {
i++
continue
}

keysToProof = append(keysToProof, common.BytesToHash(entry.K))
keys = append(keys, entry.K)
}
require.Len(t, keysToProof, 2)

expectedProof, err := tr.GenerateProof(keys)
require.NoError(t, err)

ts, err := storage.NewTrieState(tr)
require.NoError(t, err)
root := ts.MustRoot()

header, err := types.NewHeader(common.EmptyHash, root, common.EmptyHash, big.NewInt(1), nil)
require.NoError(t, err)
specifcBlock := types.NewBlock(header, nil)

blockStateMock := new(mocks.MockBlockState)
blockStateMock.On("GetBlockStateRoot", specifcBlock.Header.Hash()).Return(root, nil)

storageMock := new(mocks.MockStorageState)
storageMock.On("TrieState", &root).Return(ts, nil)

svc := new(Service)
svc.storageState = storageMock
svc.blockState = blockStateMock

block, proof, err := svc.GetReadProofAt(specifcBlock.Header.Hash(), keysToProof)
require.NoError(t, err)

blockStateMock.AssertCalled(t, "GetBlockStateRoot", specifcBlock.Header.Hash())
storageMock.AssertCalled(t, "TrieState", &root)
require.Equal(t, specifcBlock.Header.Hash(), block)

// all the proof must exists in the expected
require.Equal(t, len(expectedProof), len(proof))
for _, p := range proof {
_, has := expectedProof[p]
require.True(t, has)
}
}
Loading