forked from TrueCloudLab/neoneo-go
Merge pull request #1083 from nspcc-dev/fix/stateroot
Request verified state roots
This commit is contained in:
commit
3a6186af54
8 changed files with 81 additions and 12 deletions
|
@ -1776,12 +1776,18 @@ func (bc *Blockchain) isTxStillRelevant(t *transaction.Transaction) bool {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// StateHeight returns height of the verified state root.
|
||||||
|
func (bc *Blockchain) StateHeight() uint32 {
|
||||||
|
h, _ := bc.dao.GetCurrentStateRootHeight()
|
||||||
|
return h
|
||||||
|
}
|
||||||
|
|
||||||
// AddStateRoot add new (possibly unverified) state root to the blockchain.
|
// AddStateRoot add new (possibly unverified) state root to the blockchain.
|
||||||
func (bc *Blockchain) AddStateRoot(r *state.MPTRoot) error {
|
func (bc *Blockchain) AddStateRoot(r *state.MPTRoot) error {
|
||||||
our, err := bc.GetStateRoot(r.Index)
|
our, err := bc.GetStateRoot(r.Index)
|
||||||
if err == nil {
|
if err == nil {
|
||||||
if our.Flag == state.Verified {
|
if our.Flag == state.Verified {
|
||||||
return nil
|
return bc.updateStateHeight(r.Index)
|
||||||
} else if r.Witness == nil && our.Witness != nil {
|
} else if r.Witness == nil && our.Witness != nil {
|
||||||
r.Witness = our.Witness
|
r.Witness = our.Witness
|
||||||
}
|
}
|
||||||
|
@ -1803,10 +1809,24 @@ func (bc *Blockchain) AddStateRoot(r *state.MPTRoot) error {
|
||||||
}
|
}
|
||||||
flag = state.Verified
|
flag = state.Verified
|
||||||
}
|
}
|
||||||
return bc.dao.PutStateRoot(&state.MPTRootState{
|
err = bc.dao.PutStateRoot(&state.MPTRootState{
|
||||||
MPTRoot: *r,
|
MPTRoot: *r,
|
||||||
Flag: flag,
|
Flag: flag,
|
||||||
})
|
})
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return bc.updateStateHeight(r.Index)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (bc *Blockchain) updateStateHeight(newHeight uint32) error {
|
||||||
|
h, err := bc.dao.GetCurrentStateRootHeight()
|
||||||
|
if err != nil {
|
||||||
|
return errors.WithMessage(err, "can't get current state root height")
|
||||||
|
} else if newHeight == h+1 {
|
||||||
|
return bc.dao.PutCurrentStateRootHeight(h + 1)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// verifyStateRoot checks if state root is valid.
|
// verifyStateRoot checks if state root is valid.
|
||||||
|
|
|
@ -49,6 +49,7 @@ type Blockchainer interface {
|
||||||
References(t *transaction.Transaction) ([]transaction.InOut, error)
|
References(t *transaction.Transaction) ([]transaction.InOut, error)
|
||||||
mempool.Feer // fee interface
|
mempool.Feer // fee interface
|
||||||
PoolTx(*transaction.Transaction) error
|
PoolTx(*transaction.Transaction) error
|
||||||
|
StateHeight() uint32
|
||||||
SubscribeForBlocks(ch chan<- *block.Block)
|
SubscribeForBlocks(ch chan<- *block.Block)
|
||||||
SubscribeForExecutions(ch chan<- *state.AppExecResult)
|
SubscribeForExecutions(ch chan<- *state.AppExecResult)
|
||||||
SubscribeForNotifications(ch chan<- *state.NotificationEvent)
|
SubscribeForNotifications(ch chan<- *state.NotificationEvent)
|
||||||
|
|
|
@ -32,6 +32,7 @@ type DAO interface {
|
||||||
GetContractState(hash util.Uint160) (*state.Contract, error)
|
GetContractState(hash util.Uint160) (*state.Contract, error)
|
||||||
GetCurrentBlockHeight() (uint32, error)
|
GetCurrentBlockHeight() (uint32, error)
|
||||||
GetCurrentHeaderHeight() (i uint32, h util.Uint256, err error)
|
GetCurrentHeaderHeight() (i uint32, h util.Uint256, err error)
|
||||||
|
GetCurrentStateRootHeight() (uint32, error)
|
||||||
GetHeaderHashes() ([]util.Uint256, error)
|
GetHeaderHashes() ([]util.Uint256, error)
|
||||||
GetNEP5Balances(acc util.Uint160) (*state.NEP5Balances, error)
|
GetNEP5Balances(acc util.Uint160) (*state.NEP5Balances, error)
|
||||||
GetNEP5TransferLog(acc util.Uint160, index uint32) (*state.NEP5TransferLog, error)
|
GetNEP5TransferLog(acc util.Uint160, index uint32) (*state.NEP5TransferLog, error)
|
||||||
|
@ -434,6 +435,27 @@ func (dao *Simple) InitMPT(height uint32) error {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// GetCurrentStateRootHeight returns current state root height.
|
||||||
|
func (dao *Simple) GetCurrentStateRootHeight() (uint32, error) {
|
||||||
|
key := []byte{byte(storage.DataMPT)}
|
||||||
|
val, err := dao.Store.Get(key)
|
||||||
|
if err != nil {
|
||||||
|
if err == storage.ErrKeyNotFound {
|
||||||
|
err = nil
|
||||||
|
}
|
||||||
|
return 0, err
|
||||||
|
}
|
||||||
|
return binary.LittleEndian.Uint32(val), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// PutCurrentStateRootHeight updates current state root height.
|
||||||
|
func (dao *Simple) PutCurrentStateRootHeight(height uint32) error {
|
||||||
|
key := []byte{byte(storage.DataMPT)}
|
||||||
|
val := make([]byte, 4)
|
||||||
|
binary.LittleEndian.PutUint32(val, height)
|
||||||
|
return dao.Store.Put(key, val)
|
||||||
|
}
|
||||||
|
|
||||||
// GetStateRoot returns state root of a given height.
|
// GetStateRoot returns state root of a given height.
|
||||||
func (dao *Simple) GetStateRoot(height uint32) (*state.MPTRootState, error) {
|
func (dao *Simple) GetStateRoot(height uint32) (*state.MPTRootState, error) {
|
||||||
r := new(state.MPTRootState)
|
r := new(state.MPTRootState)
|
||||||
|
|
|
@ -154,7 +154,9 @@ func (chain testChain) IsLowPriority(util.Fixed8) bool {
|
||||||
func (chain testChain) PoolTx(*transaction.Transaction) error {
|
func (chain testChain) PoolTx(*transaction.Transaction) error {
|
||||||
panic("TODO")
|
panic("TODO")
|
||||||
}
|
}
|
||||||
|
func (chain testChain) StateHeight() uint32 {
|
||||||
|
panic("TODO")
|
||||||
|
}
|
||||||
func (chain testChain) SubscribeForBlocks(ch chan<- *block.Block) {
|
func (chain testChain) SubscribeForBlocks(ch chan<- *block.Block) {
|
||||||
panic("TODO")
|
panic("TODO")
|
||||||
}
|
}
|
||||||
|
|
|
@ -620,12 +620,35 @@ func (s *Server) handleGetRootsCmd(p Peer, gr *payload.GetStateRoots) error {
|
||||||
}
|
}
|
||||||
|
|
||||||
// handleStateRootsCmd processees `roots` request.
|
// handleStateRootsCmd processees `roots` request.
|
||||||
func (s *Server) handleRootsCmd(rs *payload.StateRoots) error {
|
func (s *Server) handleRootsCmd(p Peer, rs *payload.StateRoots) error {
|
||||||
|
h := s.chain.StateHeight()
|
||||||
for i := range rs.Roots {
|
for i := range rs.Roots {
|
||||||
|
if rs.Roots[i].Index <= h {
|
||||||
|
continue
|
||||||
|
}
|
||||||
_ = s.chain.AddStateRoot(&rs.Roots[i])
|
_ = s.chain.AddStateRoot(&rs.Roots[i])
|
||||||
}
|
}
|
||||||
|
// request more state roots from peer if needed
|
||||||
|
return s.requestStateRoot(p)
|
||||||
|
}
|
||||||
|
|
||||||
|
// requestStateRoot sends `getroots` message to get verified state roots.
|
||||||
|
func (s *Server) requestStateRoot(p Peer) error {
|
||||||
|
stateHeight := s.chain.StateHeight()
|
||||||
|
hdrHeight := s.chain.BlockHeight()
|
||||||
|
count := uint32(payload.MaxStateRootsAllowed)
|
||||||
|
if diff := hdrHeight - stateHeight; diff < count {
|
||||||
|
count = diff
|
||||||
|
}
|
||||||
|
if count == 0 {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
gr := &payload.GetStateRoots{
|
||||||
|
Start: stateHeight + 1,
|
||||||
|
Count: count,
|
||||||
|
}
|
||||||
|
return p.EnqueueP2PMessage(s.MkMsg(CMDGetRoots, gr))
|
||||||
|
}
|
||||||
|
|
||||||
// handleStateRootCmd processees `stateroot` request.
|
// handleStateRootCmd processees `stateroot` request.
|
||||||
func (s *Server) handleStateRootCmd(r *state.MPTRoot) error {
|
func (s *Server) handleStateRootCmd(r *state.MPTRoot) error {
|
||||||
|
@ -772,7 +795,7 @@ func (s *Server) handleMessage(peer Peer, msg *Message) error {
|
||||||
return s.handlePong(peer, pong)
|
return s.handlePong(peer, pong)
|
||||||
case CMDRoots:
|
case CMDRoots:
|
||||||
rs := msg.Payload.(*payload.StateRoots)
|
rs := msg.Payload.(*payload.StateRoots)
|
||||||
return s.handleRootsCmd(rs)
|
return s.handleRootsCmd(peer, rs)
|
||||||
case CMDStateRoot:
|
case CMDStateRoot:
|
||||||
r := msg.Payload.(*state.MPTRoot)
|
r := msg.Payload.(*state.MPTRoot)
|
||||||
return s.handleStateRootCmd(r)
|
return s.handleStateRootCmd(r)
|
||||||
|
|
|
@ -251,6 +251,9 @@ func (p *TCPPeer) StartProtocol() {
|
||||||
if p.LastBlockIndex() > p.server.chain.BlockHeight() {
|
if p.LastBlockIndex() > p.server.chain.BlockHeight() {
|
||||||
err = p.server.requestBlocks(p)
|
err = p.server.requestBlocks(p)
|
||||||
}
|
}
|
||||||
|
if err == nil {
|
||||||
|
err = p.server.requestStateRoot(p)
|
||||||
|
}
|
||||||
if err == nil {
|
if err == nil {
|
||||||
timer.Reset(p.server.ProtoTickInterval)
|
timer.Reset(p.server.ProtoTickInterval)
|
||||||
}
|
}
|
||||||
|
|
|
@ -745,10 +745,9 @@ func (s *Server) verifyProof(ps request.Params) (interface{}, *response.Error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (s *Server) getStateHeight(_ request.Params) (interface{}, *response.Error) {
|
func (s *Server) getStateHeight(_ request.Params) (interface{}, *response.Error) {
|
||||||
height := s.chain.BlockHeight()
|
|
||||||
return &result.StateHeight{
|
return &result.StateHeight{
|
||||||
BlockHeight: height,
|
BlockHeight: s.chain.BlockHeight(),
|
||||||
StateHeight: height,
|
StateHeight: s.chain.StateHeight(),
|
||||||
}, nil
|
}, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -246,9 +246,8 @@ var rpcTestCases = map[string][]rpcTestCase{
|
||||||
sh, ok := res.(*result.StateHeight)
|
sh, ok := res.(*result.StateHeight)
|
||||||
require.True(t, ok)
|
require.True(t, ok)
|
||||||
|
|
||||||
h := e.chain.BlockHeight()
|
require.Equal(t, e.chain.BlockHeight(), sh.BlockHeight)
|
||||||
require.Equal(t, h, sh.BlockHeight)
|
require.Equal(t, e.chain.StateHeight(), sh.StateHeight)
|
||||||
require.Equal(t, h, sh.StateHeight)
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
Loading…
Reference in a new issue