prysm-pulse/tools/forkchecker/forkchecker.go
Radosław Kapka 0b261cba5e
Unify log fields (#13654)
* unify fields

* fix tests
2024-02-22 22:40:36 +00:00

150 lines
4.2 KiB
Go

/**
* Fork choice checker
*
* A gRPC client that polls beacon node at every slot to log or compare nodes current head.
*
* Example: 2 beacon nodes with 2 gRPC end points, 127.0.0.1:4000 and 127.0.0.1:4001
* For logging heads: forkchecker --endpoint 127.0.0.1:4000 --endpoint 127.0.0.1:4001
* For comparing heads: forkchecker --endpoint 127.0.0.1:4000 --endpoint 127.0.0.1:4001 --compare
*/
package main
import (
"context"
"encoding/hex"
"flag"
"reflect"
"time"
"github.com/prysmaticlabs/prysm/v5/config/params"
"github.com/prysmaticlabs/prysm/v5/consensus-types/primitives"
pb "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1"
"github.com/sirupsen/logrus"
"google.golang.org/grpc"
"google.golang.org/protobuf/types/known/emptypb"
)
var log = logrus.WithField("prefix", "forkchoice_checker")
type endpoint []string
func (_ *endpoint) String() string {
return "gRPC endpoints"
}
// Set adds endpoint value to list.
func (e *endpoint) Set(value string) error {
*e = append(*e, value)
return nil
}
func main() {
var endpts endpoint
clients := make(map[string]pb.BeaconChainClient)
flag.Var(&endpts, "endpoint", "Specify gRPC end points for beacon node")
compare := flag.Bool("compare", false, "Enable head comparisons between all end points")
flag.Parse()
for _, endpt := range endpts {
conn, err := grpc.Dial(endpt, grpc.WithInsecure())
if err != nil {
log.WithError(err).Fatal("fail to dial")
}
clients[endpt] = pb.NewBeaconChainClient(conn)
}
ticker := time.NewTicker(time.Duration(params.BeaconConfig().SecondsPerSlot) * time.Second)
go func() {
for range ticker.C {
if *compare {
compareHeads(clients)
} else {
displayHeads(clients)
}
}
}()
select {}
}
// log heads for all RPC end points
func displayHeads(clients map[string]pb.BeaconChainClient) {
for endpt, client := range clients {
head, err := client.GetChainHead(context.Background(), &emptypb.Empty{})
if err != nil {
log.Fatal(err)
}
logHead(endpt, head)
}
}
// compare heads between all RPC end points, log the mismatch if there's one.
func compareHeads(clients map[string]pb.BeaconChainClient) {
endpt1 := randomEndpt(clients)
head1, err := clients[endpt1].GetChainHead(context.Background(), &emptypb.Empty{})
if err != nil {
log.Fatal(err)
}
log.Infof("Comparing all heads for head slot :%d", head1.HeadSlot)
if (head1.HeadSlot+1)%params.BeaconConfig().SlotsPerEpoch == 0 {
p, err := clients[endpt1].GetValidatorParticipation(context.Background(), &pb.GetValidatorParticipationRequest{})
if err != nil {
log.Fatal(err)
}
logParticipation(endpt1, p.Participation)
}
for endpt2, client := range clients {
head2, err := client.GetChainHead(context.Background(), &emptypb.Empty{})
if err != nil {
log.Fatal(err)
}
if !reflect.DeepEqual(head1, head2) {
log.Error("Uh oh! Heads mismatched!")
logHead(endpt1, head1)
logHead(endpt2, head2)
if (head1.HeadSlot+1)%params.BeaconConfig().SlotsPerEpoch == 0 {
p, err := clients[endpt2].GetValidatorParticipation(context.Background(), &pb.GetValidatorParticipationRequest{
QueryFilter: &pb.GetValidatorParticipationRequest_Epoch{
Epoch: primitives.Epoch(head2.HeadSlot / params.BeaconConfig().SlotsPerEpoch),
},
})
if err != nil {
log.Fatal(err)
}
logParticipation(endpt2, p.Participation)
}
}
}
}
func logHead(endpt string, head *pb.ChainHead) {
log.WithFields(
logrus.Fields{
"headSlot": head.HeadSlot,
"headRoot": hex.EncodeToString(head.HeadBlockRoot),
"justifiedEpoch": head.JustifiedEpoch,
"justifiedRoot": hex.EncodeToString(head.JustifiedBlockRoot),
"finalizedEpoch": head.FinalizedEpoch,
"finalizedRoot": hex.EncodeToString(head.FinalizedBlockRoot),
}).Info("Head from beacon node ", endpt)
}
func logParticipation(endpt string, p *pb.ValidatorParticipation) {
log.WithFields(
logrus.Fields{
"votedEther": p.VotedEther,
"totalEther": p.EligibleEther,
"participationRate": p.GlobalParticipationRate,
}).Info("Participation rate from beacon node ", endpt)
}
func randomEndpt(clients map[string]pb.BeaconChainClient) string {
for endpt := range clients {
return endpt
}
return ""
}