2021-03-18 21:00:00 +00:00
|
|
|
package kv
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/core/blocks"
|
|
|
|
dbIface "github.com/prysmaticlabs/prysm/beacon-chain/db/iface"
|
|
|
|
iface "github.com/prysmaticlabs/prysm/beacon-chain/state/interface"
|
2021-06-30 15:06:19 +00:00
|
|
|
state "github.com/prysmaticlabs/prysm/beacon-chain/state/v1"
|
2021-03-18 21:00:00 +00:00
|
|
|
pbp2p "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
2021-07-06 15:34:05 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/proto/eth/v1alpha1/wrapper"
|
2021-03-18 21:00:00 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
|
|
|
)
|
|
|
|
|
|
|
|
// SaveGenesisData bootstraps the beaconDB with a given genesis state.
|
|
|
|
func (s *Store) SaveGenesisData(ctx context.Context, genesisState iface.BeaconState) error {
|
|
|
|
stateRoot, err := genesisState.HashTreeRoot(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
genesisBlk := blocks.NewGenesisBlock(stateRoot[:])
|
|
|
|
genesisBlkRoot, err := genesisBlk.Block.HashTreeRoot()
|
|
|
|
if err != nil {
|
|
|
|
return errors.Wrap(err, "could not get genesis block root")
|
|
|
|
}
|
2021-07-06 15:34:05 +00:00
|
|
|
if err := s.SaveBlock(ctx, wrapper.WrappedPhase0SignedBeaconBlock(genesisBlk)); err != nil {
|
2021-03-18 21:00:00 +00:00
|
|
|
return errors.Wrap(err, "could not save genesis block")
|
|
|
|
}
|
|
|
|
if err := s.SaveState(ctx, genesisState, genesisBlkRoot); err != nil {
|
|
|
|
return errors.Wrap(err, "could not save genesis state")
|
|
|
|
}
|
|
|
|
if err := s.SaveStateSummary(ctx, &pbp2p.StateSummary{
|
|
|
|
Slot: 0,
|
|
|
|
Root: genesisBlkRoot[:],
|
|
|
|
}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := s.SaveHeadBlockRoot(ctx, genesisBlkRoot); err != nil {
|
|
|
|
return errors.Wrap(err, "could not save head block root")
|
|
|
|
}
|
|
|
|
if err := s.SaveGenesisBlockRoot(ctx, genesisBlkRoot); err != nil {
|
|
|
|
return errors.Wrap(err, "could not save genesis block root")
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-03-30 00:45:14 +00:00
|
|
|
// LoadGenesis loads a genesis state from a given file path, if no genesis exists already.
|
2021-03-18 21:00:00 +00:00
|
|
|
func (s *Store) LoadGenesis(ctx context.Context, r io.Reader) error {
|
|
|
|
b, err := ioutil.ReadAll(r)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
st := &pbp2p.BeaconState{}
|
|
|
|
if err := st.UnmarshalSSZ(b); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
gs, err := state.InitializeFromProtoUnsafe(st)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
existing, err := s.GenesisState(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// If some different genesis state existed already, return an error. The same genesis state is
|
|
|
|
// considered a no-op.
|
2021-05-24 04:55:42 +00:00
|
|
|
if existing != nil && !existing.IsNil() {
|
2021-03-18 21:00:00 +00:00
|
|
|
a, err := existing.HashTreeRoot(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
b, err := gs.HashTreeRoot(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if a == b {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return dbIface.ErrExistingGenesisState
|
|
|
|
}
|
|
|
|
|
|
|
|
if !bytes.Equal(gs.Fork().CurrentVersion, params.BeaconConfig().GenesisForkVersion) {
|
|
|
|
return fmt.Errorf("loaded genesis fork version (%#x) does not match config genesis "+
|
|
|
|
"fork version (%#x)", gs.Fork().CurrentVersion, params.BeaconConfig().GenesisForkVersion)
|
|
|
|
}
|
|
|
|
|
|
|
|
return s.SaveGenesisData(ctx, gs)
|
|
|
|
}
|
|
|
|
|
|
|
|
// EnsureEmbeddedGenesis checks that a genesis block has been generated when an embedded genesis
|
|
|
|
// state is used. If a genesis block does not exist, but a genesis state does, then we should call
|
|
|
|
// SaveGenesisData on the existing genesis state.
|
|
|
|
func (s *Store) EnsureEmbeddedGenesis(ctx context.Context) error {
|
|
|
|
gb, err := s.GenesisBlock(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-05-26 16:19:54 +00:00
|
|
|
if gb != nil && !gb.IsNil() {
|
2021-03-18 21:00:00 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
gs, err := s.GenesisState(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2021-05-24 04:55:42 +00:00
|
|
|
if gs != nil && !gs.IsNil() {
|
2021-03-18 21:00:00 +00:00
|
|
|
return s.SaveGenesisData(ctx, gs)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|