mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-12 20:50:05 +00:00
70380660b3
* generate SyncCommittee SSZ * refactor error handling * rewards: use http2.HandleError * add light client protobuf files * add light client helpers
247 lines
8.8 KiB
Go
247 lines
8.8 KiB
Go
package blockchain
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state"
|
|
"github.com/prysmaticlabs/prysm/v4/config/params"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/interfaces"
|
|
types "github.com/prysmaticlabs/prysm/v4/consensus-types/primitives"
|
|
"github.com/prysmaticlabs/prysm/v4/encoding/bytesutil"
|
|
ethpbv1 "github.com/prysmaticlabs/prysm/v4/proto/eth/v1"
|
|
ethpbv2 "github.com/prysmaticlabs/prysm/v4/proto/eth/v2"
|
|
"github.com/prysmaticlabs/prysm/v4/proto/migration"
|
|
"github.com/prysmaticlabs/prysm/v4/time/slots"
|
|
)
|
|
|
|
const (
|
|
finalityBranchNumOfLeaves = 6
|
|
)
|
|
|
|
// CreateLightClientFinalityUpdate - implements https://github.com/ethereum/consensus-specs/blob/3d235740e5f1e641d3b160c8688f26e7dc5a1894/specs/altair/light-client/full-node.md#create_light_client_finality_update
|
|
// def create_light_client_finality_update(update: LightClientUpdate) -> LightClientFinalityUpdate:
|
|
//
|
|
// return LightClientFinalityUpdate(
|
|
// attested_header=update.attested_header,
|
|
// finalized_header=update.finalized_header,
|
|
// finality_branch=update.finality_branch,
|
|
// sync_aggregate=update.sync_aggregate,
|
|
// signature_slot=update.signature_slot,
|
|
// )
|
|
func CreateLightClientFinalityUpdate(update *ethpbv2.LightClientUpdate) *ethpbv2.LightClientFinalityUpdate {
|
|
return ðpbv2.LightClientFinalityUpdate{
|
|
AttestedHeader: update.AttestedHeader,
|
|
FinalizedHeader: update.FinalizedHeader,
|
|
FinalityBranch: update.FinalityBranch,
|
|
SyncAggregate: update.SyncAggregate,
|
|
SignatureSlot: update.SignatureSlot,
|
|
}
|
|
}
|
|
|
|
// CreateLightClientOptimisticUpdate - implements https://github.com/ethereum/consensus-specs/blob/3d235740e5f1e641d3b160c8688f26e7dc5a1894/specs/altair/light-client/full-node.md#create_light_client_optimistic_update
|
|
// def create_light_client_optimistic_update(update: LightClientUpdate) -> LightClientOptimisticUpdate:
|
|
//
|
|
// return LightClientOptimisticUpdate(
|
|
// attested_header=update.attested_header,
|
|
// sync_aggregate=update.sync_aggregate,
|
|
// signature_slot=update.signature_slot,
|
|
// )
|
|
func CreateLightClientOptimisticUpdate(update *ethpbv2.LightClientUpdate) *ethpbv2.LightClientOptimisticUpdate {
|
|
return ðpbv2.LightClientOptimisticUpdate{
|
|
AttestedHeader: update.AttestedHeader,
|
|
SyncAggregate: update.SyncAggregate,
|
|
SignatureSlot: update.SignatureSlot,
|
|
}
|
|
}
|
|
|
|
func NewLightClientOptimisticUpdateFromBeaconState(
|
|
ctx context.Context,
|
|
state state.BeaconState,
|
|
block interfaces.ReadOnlySignedBeaconBlock,
|
|
attestedState state.BeaconState) (*ethpbv2.LightClientUpdate, error) {
|
|
// assert compute_epoch_at_slot(attested_state.slot) >= ALTAIR_FORK_EPOCH
|
|
attestedEpoch := slots.ToEpoch(attestedState.Slot())
|
|
if attestedEpoch < types.Epoch(params.BeaconConfig().AltairForkEpoch) {
|
|
return nil, fmt.Errorf("invalid attested epoch %d", attestedEpoch)
|
|
}
|
|
|
|
// assert sum(block.message.body.sync_aggregate.sync_committee_bits) >= MIN_SYNC_COMMITTEE_PARTICIPANTS
|
|
syncAggregate, err := block.Block().Body().SyncAggregate()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not get sync aggregate %v", err)
|
|
}
|
|
|
|
if syncAggregate.SyncCommitteeBits.Count() < params.BeaconConfig().MinSyncCommitteeParticipants {
|
|
return nil, fmt.Errorf("invalid sync committee bits count %d", syncAggregate.SyncCommitteeBits.Count())
|
|
}
|
|
|
|
// assert state.slot == state.latest_block_header.slot
|
|
if state.Slot() != state.LatestBlockHeader().Slot {
|
|
return nil, fmt.Errorf("state slot %d not equal to latest block header slot %d", state.Slot(), state.LatestBlockHeader().Slot)
|
|
}
|
|
|
|
// assert hash_tree_root(header) == hash_tree_root(block.message)
|
|
header := state.LatestBlockHeader()
|
|
stateRoot, err := state.HashTreeRoot(ctx)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not get state root %v", err)
|
|
}
|
|
header.StateRoot = stateRoot[:]
|
|
|
|
headerRoot, err := header.HashTreeRoot()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not get header root %v", err)
|
|
}
|
|
|
|
blockRoot, err := block.Block().HashTreeRoot()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not get block root %v", err)
|
|
}
|
|
|
|
if headerRoot != blockRoot {
|
|
return nil, fmt.Errorf("header root %#x not equal to block root %#x", headerRoot, blockRoot)
|
|
}
|
|
|
|
// assert attested_state.slot == attested_state.latest_block_header.slot
|
|
if attestedState.Slot() != attestedState.LatestBlockHeader().Slot {
|
|
return nil, fmt.Errorf("attested state slot %d not equal to attested latest block header slot %d", attestedState.Slot(), attestedState.LatestBlockHeader().Slot)
|
|
}
|
|
|
|
// attested_header = attested_state.latest_block_header.copy()
|
|
attestedHeader := attestedState.LatestBlockHeader()
|
|
|
|
// attested_header.state_root = hash_tree_root(attested_state)
|
|
attestedStateRoot, err := attestedState.HashTreeRoot(ctx)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not get attested state root %v", err)
|
|
}
|
|
attestedHeader.StateRoot = attestedStateRoot[:]
|
|
|
|
// assert hash_tree_root(attested_header) == block.message.parent_root
|
|
attestedHeaderRoot, err := attestedHeader.HashTreeRoot()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not get attested header root %v", err)
|
|
}
|
|
|
|
if attestedHeaderRoot != block.Block().ParentRoot() {
|
|
return nil, fmt.Errorf("attested header root %#x not equal to block parent root %#x", attestedHeaderRoot, block.Block().ParentRoot())
|
|
}
|
|
|
|
// Return result
|
|
attestedHeaderResult := ðpbv1.BeaconBlockHeader{
|
|
Slot: attestedHeader.Slot,
|
|
ProposerIndex: attestedHeader.ProposerIndex,
|
|
ParentRoot: attestedHeader.ParentRoot,
|
|
StateRoot: attestedHeader.StateRoot,
|
|
BodyRoot: attestedHeader.BodyRoot,
|
|
}
|
|
|
|
syncAggregateResult := ðpbv1.SyncAggregate{
|
|
SyncCommitteeBits: syncAggregate.SyncCommitteeBits,
|
|
SyncCommitteeSignature: syncAggregate.SyncCommitteeSignature,
|
|
}
|
|
|
|
result := ðpbv2.LightClientUpdate{
|
|
AttestedHeader: attestedHeaderResult,
|
|
SyncAggregate: syncAggregateResult,
|
|
SignatureSlot: block.Block().Slot(),
|
|
}
|
|
|
|
return result, nil
|
|
}
|
|
|
|
func NewLightClientFinalityUpdateFromBeaconState(
|
|
ctx context.Context,
|
|
state state.BeaconState,
|
|
block interfaces.ReadOnlySignedBeaconBlock,
|
|
attestedState state.BeaconState,
|
|
finalizedBlock interfaces.ReadOnlySignedBeaconBlock) (*ethpbv2.LightClientUpdate, error) {
|
|
result, err := NewLightClientOptimisticUpdateFromBeaconState(
|
|
ctx,
|
|
state,
|
|
block,
|
|
attestedState,
|
|
)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Indicate finality whenever possible
|
|
var finalizedHeader *ethpbv1.BeaconBlockHeader
|
|
var finalityBranch [][]byte
|
|
|
|
if finalizedBlock != nil && !finalizedBlock.IsNil() {
|
|
if finalizedBlock.Block().Slot() != 0 {
|
|
tempFinalizedHeader, err := finalizedBlock.Header()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not get finalized header %v", err)
|
|
}
|
|
finalizedHeader = migration.V1Alpha1SignedHeaderToV1(tempFinalizedHeader).GetMessage()
|
|
|
|
finalizedHeaderRoot, err := finalizedHeader.HashTreeRoot()
|
|
if err != nil {
|
|
return nil, fmt.Errorf("could not get finalized header root %v", err)
|
|
}
|
|
|
|
if finalizedHeaderRoot != bytesutil.ToBytes32(attestedState.FinalizedCheckpoint().Root) {
|
|
return nil, fmt.Errorf("finalized header root %#x not equal to attested finalized checkpoint root %#x", finalizedHeaderRoot, bytesutil.ToBytes32(attestedState.FinalizedCheckpoint().Root))
|
|
}
|
|
} else {
|
|
if !bytes.Equal(attestedState.FinalizedCheckpoint().Root, make([]byte, 32)) {
|
|
return nil, fmt.Errorf("invalid finalized header root %v", attestedState.FinalizedCheckpoint().Root)
|
|
}
|
|
|
|
finalizedHeader = ðpbv1.BeaconBlockHeader{
|
|
Slot: 0,
|
|
ProposerIndex: 0,
|
|
ParentRoot: make([]byte, 32),
|
|
StateRoot: make([]byte, 32),
|
|
BodyRoot: make([]byte, 32),
|
|
}
|
|
}
|
|
|
|
var bErr error
|
|
finalityBranch, bErr = attestedState.FinalizedRootProof(ctx)
|
|
if bErr != nil {
|
|
return nil, fmt.Errorf("could not get finalized root proof %v", bErr)
|
|
}
|
|
} else {
|
|
finalizedHeader = ðpbv1.BeaconBlockHeader{
|
|
Slot: 0,
|
|
ProposerIndex: 0,
|
|
ParentRoot: make([]byte, 32),
|
|
StateRoot: make([]byte, 32),
|
|
BodyRoot: make([]byte, 32),
|
|
}
|
|
|
|
finalityBranch = make([][]byte, finalityBranchNumOfLeaves)
|
|
for i := 0; i < finalityBranchNumOfLeaves; i++ {
|
|
finalityBranch[i] = make([]byte, 32)
|
|
}
|
|
}
|
|
|
|
result.FinalizedHeader = finalizedHeader
|
|
result.FinalityBranch = finalityBranch
|
|
return result, nil
|
|
}
|
|
|
|
func NewLightClientUpdateFromFinalityUpdate(update *ethpbv2.LightClientFinalityUpdate) *ethpbv2.LightClientUpdate {
|
|
return ðpbv2.LightClientUpdate{
|
|
AttestedHeader: update.AttestedHeader,
|
|
FinalizedHeader: update.FinalizedHeader,
|
|
FinalityBranch: update.FinalityBranch,
|
|
SyncAggregate: update.SyncAggregate,
|
|
SignatureSlot: update.SignatureSlot,
|
|
}
|
|
}
|
|
|
|
func NewLightClientUpdateFromOptimisticUpdate(update *ethpbv2.LightClientOptimisticUpdate) *ethpbv2.LightClientUpdate {
|
|
return ðpbv2.LightClientUpdate{
|
|
AttestedHeader: update.AttestedHeader,
|
|
SyncAggregate: update.SyncAggregate,
|
|
SignatureSlot: update.SignatureSlot,
|
|
}
|
|
}
|