2022-11-02 12:12:44 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2022-11-20 17:44:42 +00:00
|
|
|
"fmt"
|
|
|
|
"os"
|
2022-11-02 12:12:44 +00:00
|
|
|
|
2022-11-28 23:00:40 +00:00
|
|
|
sentinelrpc "github.com/ledgerwatch/erigon-lib/gointerfaces/sentinel"
|
2022-11-26 23:03:58 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
2022-11-02 12:12:44 +00:00
|
|
|
"github.com/ledgerwatch/erigon-lib/kv/mdbx"
|
|
|
|
"github.com/ledgerwatch/erigon/cl/clparams"
|
2022-11-20 17:44:42 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/cltypes"
|
2022-12-05 00:25:12 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/fork"
|
2022-12-23 21:31:08 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/rpc"
|
2022-11-25 15:38:22 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cmd/erigon-cl/core"
|
|
|
|
"github.com/ledgerwatch/erigon/cmd/erigon-cl/core/rawdb"
|
2022-12-11 14:12:38 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cmd/erigon-cl/core/state"
|
2022-11-25 15:38:22 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cmd/erigon-cl/network"
|
|
|
|
"github.com/ledgerwatch/erigon/cmd/erigon-cl/stages"
|
2022-11-20 17:44:42 +00:00
|
|
|
lcCli "github.com/ledgerwatch/erigon/cmd/sentinel/cli"
|
|
|
|
"github.com/ledgerwatch/erigon/cmd/sentinel/cli/flags"
|
|
|
|
"github.com/ledgerwatch/erigon/cmd/sentinel/sentinel"
|
2022-12-05 00:25:12 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cmd/sentinel/sentinel/handshake"
|
2022-11-20 17:44:42 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cmd/sentinel/sentinel/service"
|
|
|
|
sentinelapp "github.com/ledgerwatch/erigon/turbo/app"
|
2022-11-02 12:12:44 +00:00
|
|
|
"github.com/ledgerwatch/log/v3"
|
2022-11-20 17:44:42 +00:00
|
|
|
"github.com/urfave/cli/v2"
|
2022-11-02 12:12:44 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func main() {
|
2022-11-28 22:29:48 +00:00
|
|
|
app := sentinelapp.MakeApp(runConsensusLayerNode, flags.CLDefaultFlags)
|
2022-11-20 17:44:42 +00:00
|
|
|
if err := app.Run(os.Args); err != nil {
|
|
|
|
_, printErr := fmt.Fprintln(os.Stderr, err)
|
|
|
|
if printErr != nil {
|
|
|
|
log.Warn("Fprintln error", "err", printErr)
|
|
|
|
}
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
}
|
2022-11-02 12:12:44 +00:00
|
|
|
|
2022-11-20 17:44:42 +00:00
|
|
|
func runConsensusLayerNode(cliCtx *cli.Context) error {
|
2022-11-02 12:12:44 +00:00
|
|
|
ctx := context.Background()
|
2022-11-28 22:29:48 +00:00
|
|
|
cfg, _ := lcCli.SetupConsensusClientCfg(cliCtx)
|
2022-12-22 18:59:24 +00:00
|
|
|
var db kv.RwDB
|
|
|
|
var err error
|
|
|
|
if cfg.Chaindata == "" {
|
|
|
|
db, err = mdbx.NewTemporaryMdbx()
|
|
|
|
} else {
|
|
|
|
db, err = mdbx.Open(cfg.Chaindata, log.Root(), false)
|
|
|
|
}
|
2022-11-26 23:03:58 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Error("Error opening database", "err", err)
|
|
|
|
}
|
|
|
|
defer db.Close()
|
2022-11-20 17:44:42 +00:00
|
|
|
// Fetch the checkpoint state.
|
2022-11-26 23:03:58 +00:00
|
|
|
cpState, err := getCheckpointState(ctx, db)
|
2022-11-20 17:44:42 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Error("Could not get checkpoint", "err", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Info("Starting sync from checkpoint.")
|
|
|
|
|
|
|
|
// Start the sentinel service
|
2022-12-17 15:05:56 +00:00
|
|
|
log.Root().SetHandler(log.LvlFilterHandler(log.Lvl(cfg.LogLvl), log.StderrHandler))
|
2022-11-28 22:29:48 +00:00
|
|
|
log.Info("[Sentinel] running sentinel with configuration", "cfg", cfg)
|
2022-12-05 00:25:12 +00:00
|
|
|
s, err := startSentinel(cliCtx, *cfg, cpState)
|
2022-11-20 17:44:42 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Error("Could not start sentinel service", "err", err)
|
|
|
|
}
|
|
|
|
|
2022-11-25 15:38:22 +00:00
|
|
|
genesisCfg, _, beaconConfig := clparams.GetConfigsByNetwork(clparams.MainnetNetwork)
|
2022-12-23 21:31:08 +00:00
|
|
|
beaconRpc := rpc.NewBeaconRpcP2P(ctx, s, beaconConfig, genesisCfg)
|
|
|
|
downloader := network.NewForwardBeaconDownloader(ctx, beaconRpc)
|
|
|
|
bdownloader := network.NewBackwardBeaconDownloader(ctx, beaconRpc)
|
2022-12-17 15:05:56 +00:00
|
|
|
|
2022-12-03 02:16:26 +00:00
|
|
|
gossipManager := network.NewGossipReceiver(ctx, s)
|
|
|
|
gossipManager.AddReceiver(sentinelrpc.GossipType_BeaconBlockGossipType, downloader)
|
|
|
|
go gossipManager.Loop()
|
2022-12-17 15:05:56 +00:00
|
|
|
stageloop, err := stages.NewConsensusStagedSync(ctx, db, downloader, bdownloader, genesisCfg, beaconConfig, cpState, nil, false)
|
2022-12-03 02:16:26 +00:00
|
|
|
if err != nil {
|
2022-11-26 23:03:58 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-12-03 02:16:26 +00:00
|
|
|
Loop:
|
|
|
|
for {
|
|
|
|
if err := stageloop.Run(db, nil, false, true); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
break Loop
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
2022-11-20 17:44:42 +00:00
|
|
|
}
|
|
|
|
|
2022-12-11 14:12:38 +00:00
|
|
|
func startSentinel(cliCtx *cli.Context, cfg lcCli.ConsensusClientCliCfg, beaconState *state.BeaconState) (sentinelrpc.SentinelClient, error) {
|
2022-12-05 00:25:12 +00:00
|
|
|
forkDigest, err := fork.ComputeForkDigest(cfg.BeaconCfg, cfg.GenesisCfg)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-11-20 17:44:42 +00:00
|
|
|
s, err := service.StartSentinelService(&sentinel.SentinelConfig{
|
2022-11-28 22:29:48 +00:00
|
|
|
IpAddr: cfg.Addr,
|
|
|
|
Port: int(cfg.Port),
|
|
|
|
TCPPort: cfg.ServerTcpPort,
|
|
|
|
GenesisConfig: cfg.GenesisCfg,
|
|
|
|
NetworkConfig: cfg.NetworkCfg,
|
|
|
|
BeaconConfig: cfg.BeaconCfg,
|
|
|
|
NoDiscovery: cfg.NoDiscovery,
|
2022-12-05 00:25:12 +00:00
|
|
|
}, nil, &service.ServerConfig{Network: cfg.ServerProtocol, Addr: cfg.ServerAddr}, nil, &cltypes.Status{
|
|
|
|
ForkDigest: forkDigest,
|
2022-12-11 14:12:38 +00:00
|
|
|
FinalizedRoot: beaconState.FinalizedCheckpoint().Root,
|
|
|
|
FinalizedEpoch: beaconState.FinalizedCheckpoint().Epoch,
|
|
|
|
HeadSlot: beaconState.FinalizedCheckpoint().Epoch * 32,
|
|
|
|
HeadRoot: beaconState.FinalizedCheckpoint().Root,
|
2022-12-05 00:25:12 +00:00
|
|
|
}, handshake.FullClientRule)
|
2022-11-20 17:44:42 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Error("Could not start sentinel", "err", err)
|
|
|
|
return nil, err
|
|
|
|
}
|
2022-11-28 22:29:48 +00:00
|
|
|
log.Info("Sentinel started", "addr", cfg.ServerAddr)
|
2022-11-20 17:44:42 +00:00
|
|
|
return s, nil
|
|
|
|
}
|
|
|
|
|
2022-12-11 14:12:38 +00:00
|
|
|
func getCheckpointState(ctx context.Context, db kv.RwDB) (*state.BeaconState, error) {
|
2022-11-20 17:44:42 +00:00
|
|
|
|
2022-11-02 12:12:44 +00:00
|
|
|
uri := clparams.GetCheckpointSyncEndpoint(clparams.MainnetNetwork)
|
|
|
|
|
2022-11-25 15:38:22 +00:00
|
|
|
state, err := core.RetrieveBeaconState(ctx, uri)
|
2022-11-02 12:12:44 +00:00
|
|
|
if err != nil {
|
|
|
|
log.Error("[Checkpoint Sync] Failed", "reason", err)
|
2022-11-20 17:44:42 +00:00
|
|
|
return nil, err
|
2022-11-02 12:12:44 +00:00
|
|
|
}
|
|
|
|
tx, err := db.BeginRw(ctx)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("[DB] Failed", "reason", err)
|
2022-11-20 17:44:42 +00:00
|
|
|
return nil, err
|
2022-11-02 12:12:44 +00:00
|
|
|
}
|
|
|
|
defer tx.Rollback()
|
|
|
|
|
2022-11-25 15:38:22 +00:00
|
|
|
if err := rawdb.WriteBeaconState(tx, state); err != nil {
|
2022-11-02 12:12:44 +00:00
|
|
|
log.Error("[DB] Failed", "reason", err)
|
2022-11-20 17:44:42 +00:00
|
|
|
return nil, err
|
2022-11-02 12:12:44 +00:00
|
|
|
}
|
2022-11-20 17:44:42 +00:00
|
|
|
log.Info("Checkpoint sync successful: hurray!")
|
2022-11-26 23:03:58 +00:00
|
|
|
return state, tx.Commit()
|
2022-11-02 12:12:44 +00:00
|
|
|
}
|