// Copyright 2024 The go-ethereum Authors // This file is part of the go-ethereum library. // // The go-ethereum library is free software: you can redistribute it and/or modify // it under the terms of the GNU Lesser General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // The go-ethereum library is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public License // along with the go-ethereum library. If not, see . package blsync import ( "strings" "github.com/ethereum/go-ethereum/beacon/light" "github.com/ethereum/go-ethereum/beacon/light/api" "github.com/ethereum/go-ethereum/beacon/light/request" "github.com/ethereum/go-ethereum/beacon/light/sync" "github.com/ethereum/go-ethereum/beacon/types" "github.com/ethereum/go-ethereum/cmd/utils" "github.com/ethereum/go-ethereum/common/mclock" "github.com/ethereum/go-ethereum/ethdb/memorydb" "github.com/ethereum/go-ethereum/event" "github.com/urfave/cli/v2" ) type Client struct { scheduler *request.Scheduler chainHeadFeed *event.Feed urls []string customHeader map[string]string } func NewClient(ctx *cli.Context) *Client { if !ctx.IsSet(utils.BeaconApiFlag.Name) { utils.Fatalf("Beacon node light client API URL not specified") } var ( chainConfig = makeChainConfig(ctx) customHeader = make(map[string]string) ) for _, s := range ctx.StringSlice(utils.BeaconApiHeaderFlag.Name) { kv := strings.Split(s, ":") if len(kv) != 2 { utils.Fatalf("Invalid custom API header entry: %s", s) } customHeader[strings.TrimSpace(kv[0])] = strings.TrimSpace(kv[1]) } // create data structures var ( db = memorydb.New() threshold = ctx.Int(utils.BeaconThresholdFlag.Name) committeeChain = light.NewCommitteeChain(db, chainConfig.ChainConfig, threshold, !ctx.Bool(utils.BeaconNoFilterFlag.Name)) headTracker = light.NewHeadTracker(committeeChain, threshold) ) headSync := sync.NewHeadSync(headTracker, committeeChain) // set up scheduler and sync modules chainHeadFeed := new(event.Feed) scheduler := request.NewScheduler() checkpointInit := sync.NewCheckpointInit(committeeChain, chainConfig.Checkpoint) forwardSync := sync.NewForwardUpdateSync(committeeChain) beaconBlockSync := newBeaconBlockSync(headTracker, chainHeadFeed) scheduler.RegisterTarget(headTracker) scheduler.RegisterTarget(committeeChain) scheduler.RegisterModule(checkpointInit, "checkpointInit") scheduler.RegisterModule(forwardSync, "forwardSync") scheduler.RegisterModule(headSync, "headSync") scheduler.RegisterModule(beaconBlockSync, "beaconBlockSync") return &Client{ scheduler: scheduler, urls: ctx.StringSlice(utils.BeaconApiFlag.Name), customHeader: customHeader, chainHeadFeed: chainHeadFeed, } } // SubscribeChainHeadEvent allows callers to subscribe a provided channel to new // head updates. func (c *Client) SubscribeChainHeadEvent(ch chan<- types.ChainHeadEvent) event.Subscription { return c.chainHeadFeed.Subscribe(ch) } func (c *Client) Start() { c.scheduler.Start() // register server(s) for _, url := range c.urls { beaconApi := api.NewBeaconLightApi(url, c.customHeader) c.scheduler.RegisterServer(request.NewServer(api.NewApiServer(beaconApi), &mclock.System{})) } } func (c *Client) Stop() { c.scheduler.Stop() }