2021-01-29 14:33:24 +00:00
|
|
|
package stateroot
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/binary"
|
|
|
|
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/core/state"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/core/storage"
|
|
|
|
"github.com/nspcc-dev/neo-go/pkg/io"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2021-02-01 16:00:07 +00:00
|
|
|
prefixGC = 0x01
|
|
|
|
prefixLocal = 0x02
|
|
|
|
prefixValidated = 0x03
|
2021-01-29 14:33:24 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func (s *Module) addLocalStateRoot(sr *state.MPTRoot) error {
|
|
|
|
key := makeStateRootKey(sr.Index)
|
|
|
|
if err := s.putStateRoot(key, sr); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
data := make([]byte, 4)
|
|
|
|
binary.LittleEndian.PutUint32(data, sr.Index)
|
|
|
|
if err := s.Store.Put([]byte{byte(storage.DataMPT), prefixLocal}, data); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
s.currentLocal.Store(sr.Root)
|
|
|
|
s.localHeight.Store(sr.Index)
|
2021-02-01 16:00:07 +00:00
|
|
|
if s.bc.GetConfig().StateRootInHeader {
|
|
|
|
s.validatedHeight.Store(sr.Index)
|
|
|
|
updateStateHeightMetric(sr.Index)
|
|
|
|
}
|
2021-01-29 14:33:24 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Module) putStateRoot(key []byte, sr *state.MPTRoot) error {
|
|
|
|
w := io.NewBufBinWriter()
|
|
|
|
sr.EncodeBinary(w.BinWriter)
|
|
|
|
return s.Store.Put(key, w.Bytes())
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Module) getStateRoot(key []byte) (*state.MPTRoot, error) {
|
|
|
|
data, err := s.Store.Get(key)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2021-03-25 19:11:55 +00:00
|
|
|
sr := &state.MPTRoot{}
|
2021-01-29 14:33:24 +00:00
|
|
|
r := io.NewBinReaderFromBuf(data)
|
|
|
|
sr.DecodeBinary(r)
|
|
|
|
return sr, r.Err
|
|
|
|
}
|
|
|
|
|
|
|
|
func makeStateRootKey(index uint32) []byte {
|
|
|
|
key := make([]byte, 5)
|
|
|
|
key[0] = byte(storage.DataMPT)
|
|
|
|
binary.BigEndian.PutUint32(key, index)
|
|
|
|
return key
|
|
|
|
}
|
2021-02-01 16:00:07 +00:00
|
|
|
|
|
|
|
// AddStateRoot adds validated state root provided by network.
|
|
|
|
func (s *Module) AddStateRoot(sr *state.MPTRoot) error {
|
|
|
|
if err := s.VerifyStateRoot(sr); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
key := makeStateRootKey(sr.Index)
|
|
|
|
local, err := s.getStateRoot(key)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if local.Witness != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if err := s.putStateRoot(key, sr); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
data := make([]byte, 4)
|
|
|
|
binary.LittleEndian.PutUint32(data, sr.Index)
|
|
|
|
if err := s.Store.Put([]byte{byte(storage.DataMPT), prefixValidated}, data); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
s.validatedHeight.Store(sr.Index)
|
|
|
|
if !s.bc.GetConfig().StateRootInHeader {
|
|
|
|
updateStateHeightMetric(sr.Index)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|