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:
kasey 2023-06-10 22:00:38 -05:00 committed by GitHub
parent 0f228896b0
commit 6fa2d768b5
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 27 additions and 21 deletions

View File

@ -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,

View File

@ -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: