/** * 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" ptypes "github.com/gogo/protobuf/types" pb "github.com/prysmaticlabs/ethereumapis/eth/v1alpha1" "github.com/prysmaticlabs/prysm/shared/params" "github.com/sirupsen/logrus" "google.golang.org/grpc" ) var log = logrus.WithField("prefix", "forkchoice_checker") type endpoint []string func (e *endpoint) String() string { return "gRPC endpoints" } func (e *endpoint) Set(value string) error { *e = append(*e, value) return nil } func main() { params.UseDemoBeaconConfig() 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.Fatalf("fail to dial: %v", err) } clients[endpt] = pb.NewBeaconChainClient(conn) } ticker := time.NewTicker(time.Duration(params.BeaconConfig().SecondsPerSlot) * time.Second) go func() { for { select { case <-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(), &ptypes.Empty{}) if err != nil { log.Fatal(err) } logHead(endpt, head) } } // compare heads between all RPC end points, log the missmatch if there's one. func compareHeads(clients map[string]pb.BeaconChainClient) { endpt1 := randomEndpt(clients) head1, err := clients[endpt1].GetChainHead(context.Background(), &ptypes.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(), &ptypes.Empty{}) if err != nil { log.Fatal(err) } if !reflect.DeepEqual(head1, head2) { log.Error("Uh oh! Heads missmatched!") 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: 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 "" }