mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-03 08:37:37 +00:00
Checkpoint sync: get block using state.latest_block_header.slot (#12447)
Co-authored-by: Kasey Kirkham <kasey@users.noreply.github.com> Co-authored-by: Nishant Das <nishdas93@gmail.com> Co-authored-by: prylabs-bulldozer[bot] <58059840+prylabs-bulldozer[bot]@users.noreply.github.com>
This commit is contained in:
parent
0f228896b0
commit
6fa2d768b5
@ -11,6 +11,7 @@ import (
|
|||||||
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state"
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state"
|
||||||
"github.com/prysmaticlabs/prysm/v4/consensus-types/interfaces"
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/interfaces"
|
||||||
"github.com/prysmaticlabs/prysm/v4/consensus-types/primitives"
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/primitives"
|
||||||
|
"github.com/prysmaticlabs/prysm/v4/encoding/bytesutil"
|
||||||
"github.com/prysmaticlabs/prysm/v4/encoding/ssz/detect"
|
"github.com/prysmaticlabs/prysm/v4/encoding/ssz/detect"
|
||||||
"github.com/prysmaticlabs/prysm/v4/io/file"
|
"github.com/prysmaticlabs/prysm/v4/io/file"
|
||||||
"github.com/prysmaticlabs/prysm/v4/runtime/version"
|
"github.com/prysmaticlabs/prysm/v4/runtime/version"
|
||||||
@ -19,6 +20,8 @@ import (
|
|||||||
"golang.org/x/mod/semver"
|
"golang.org/x/mod/semver"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var errCheckpointBlockMismatch = errors.New("mismatch between checkpoint sync state and block")
|
||||||
|
|
||||||
// OriginData represents the BeaconState and ReadOnlySignedBeaconBlock necessary to start an empty Beacon Node
|
// OriginData represents the BeaconState and ReadOnlySignedBeaconBlock necessary to start an empty Beacon Node
|
||||||
// using Checkpoint Sync.
|
// using Checkpoint Sync.
|
||||||
type OriginData struct {
|
type OriginData struct {
|
||||||
@ -75,37 +78,40 @@ func DownloadFinalizedData(ctx context.Context, client *Client) (*OriginData, er
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.Wrap(err, "error unmarshaling finalized state to correct version")
|
return nil, errors.Wrap(err, "error unmarshaling finalized state to correct version")
|
||||||
}
|
}
|
||||||
if s.Slot() != s.LatestBlockHeader().Slot {
|
|
||||||
return nil, fmt.Errorf("finalized state slot does not match latest block header slot %d != %d", s.Slot(), s.LatestBlockHeader().Slot)
|
|
||||||
}
|
|
||||||
|
|
||||||
sr, err := s.HashTreeRoot(ctx)
|
slot := s.LatestBlockHeader().Slot
|
||||||
|
bb, err := client.GetBlock(ctx, IdFromSlot(slot))
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.Wrapf(err, "failed to compute htr for finalized state at slot=%d", s.Slot())
|
return nil, errors.Wrapf(err, "error requesting block by slot = %d", slot)
|
||||||
}
|
|
||||||
header := s.LatestBlockHeader()
|
|
||||||
header.StateRoot = sr[:]
|
|
||||||
br, err := header.HashTreeRoot()
|
|
||||||
if err != nil {
|
|
||||||
return nil, errors.Wrap(err, "error while computing block root using state data")
|
|
||||||
}
|
|
||||||
|
|
||||||
bb, err := client.GetBlock(ctx, IdFromRoot(br))
|
|
||||||
if err != nil {
|
|
||||||
return nil, errors.Wrapf(err, "error requesting block by root = %#x", br)
|
|
||||||
}
|
}
|
||||||
b, err := vu.UnmarshalBeaconBlock(bb)
|
b, err := vu.UnmarshalBeaconBlock(bb)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.Wrap(err, "unable to unmarshal block to a supported type using the detected fork schedule")
|
return nil, errors.Wrap(err, "unable to unmarshal block to a supported type using the detected fork schedule")
|
||||||
}
|
}
|
||||||
realBlockRoot, err := b.Block().HashTreeRoot()
|
br, err := b.Block().HashTreeRoot()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, errors.Wrap(err, "error computing hash_tree_root of retrieved block")
|
return nil, errors.Wrap(err, "error computing hash_tree_root of retrieved block")
|
||||||
}
|
}
|
||||||
|
bodyRoot, err := b.Block().Body().HashTreeRoot()
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrap(err, "error computing hash_tree_root of retrieved block body")
|
||||||
|
}
|
||||||
|
|
||||||
log.Printf("BeaconState slot=%d, Block slot=%d", s.Slot(), b.Block().Slot())
|
sbr := bytesutil.ToBytes32(s.LatestBlockHeader().BodyRoot)
|
||||||
log.Printf("BeaconState htr=%#x, Block state_root=%#x", sr, b.Block().StateRoot())
|
if sbr != bodyRoot {
|
||||||
log.Printf("BeaconState latest_block_header htr=%#x, block htr=%#x", br, realBlockRoot)
|
return nil, errors.Wrapf(errCheckpointBlockMismatch, "state body root = %#x, block body root = %#x", sbr, bodyRoot)
|
||||||
|
}
|
||||||
|
sr, err := s.HashTreeRoot(ctx)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.Wrapf(err, "failed to compute htr for finalized state at slot=%d", s.Slot())
|
||||||
|
}
|
||||||
|
|
||||||
|
log.
|
||||||
|
WithField("block_slot", b.Block().Slot()).
|
||||||
|
WithField("state_slot", s.Slot()).
|
||||||
|
WithField("state_root", sr).
|
||||||
|
WithField("block_root", br).
|
||||||
|
Info("Downloaded checkpoint sync state and block.")
|
||||||
return &OriginData{
|
return &OriginData{
|
||||||
st: s,
|
st: s,
|
||||||
b: b,
|
b: b,
|
||||||
|
@ -440,7 +440,7 @@ func TestDownloadFinalizedData(t *testing.T) {
|
|||||||
case renderGetStatePath(IdFinalized):
|
case renderGetStatePath(IdFinalized):
|
||||||
res.StatusCode = http.StatusOK
|
res.StatusCode = http.StatusOK
|
||||||
res.Body = io.NopCloser(bytes.NewBuffer(ms))
|
res.Body = io.NopCloser(bytes.NewBuffer(ms))
|
||||||
case renderGetBlockPath(IdFromRoot(br)):
|
case renderGetBlockPath(IdFromSlot(b.Block().Slot())):
|
||||||
res.StatusCode = http.StatusOK
|
res.StatusCode = http.StatusOK
|
||||||
res.Body = io.NopCloser(bytes.NewBuffer(mb))
|
res.Body = io.NopCloser(bytes.NewBuffer(mb))
|
||||||
default:
|
default:
|
||||||
|
Loading…
Reference in New Issue
Block a user