package commands import ( "context" "embed" "fmt" "math/big" "net" "os" "path/filepath" "runtime" "strings" "syscall" "time" grpc_middleware "github.com/grpc-ecosystem/go-grpc-middleware" grpc_recovery "github.com/grpc-ecosystem/go-grpc-middleware/recovery" //grpc_prometheus "github.com/grpc-ecosystem/go-grpc-prometheus" "github.com/holiman/uint256" "github.com/ledgerwatch/erigon-lib/gointerfaces" proto_cons "github.com/ledgerwatch/erigon-lib/gointerfaces/consensus" "github.com/ledgerwatch/erigon-lib/kv" "github.com/ledgerwatch/erigon/cmd/utils" "github.com/ledgerwatch/erigon/common" "github.com/ledgerwatch/erigon/consensus/clique" "github.com/ledgerwatch/erigon/core" "github.com/ledgerwatch/erigon/core/types" "github.com/ledgerwatch/erigon/params" "github.com/ledgerwatch/log/v3" "github.com/pelletier/go-toml" "github.com/spf13/cobra" "google.golang.org/grpc" "google.golang.org/grpc/keepalive" "google.golang.org/protobuf/types/known/emptypb" ) //go:embed configs var configs embed.FS func init() { withApiAddr(cliqueCmd) withDatadir(cliqueCmd) withConfig(cliqueCmd) rootCmd.AddCommand(cliqueCmd) } var cliqueCmd = &cobra.Command{ Use: "clique", Short: "Run clique consensus engine", RunE: func(cmd *cobra.Command, args []string) error { ctx, _ := utils.RootContext() logger := log.New() return cliqueEngine(ctx, logger) }, } func cliqueEngine(ctx context.Context, logger log.Logger) error { var server *CliqueServerImpl var err error if config == "test" { // Configuration will be received from the test driver server, err = grpcCliqueServer(ctx, true /* testServer */) if err != nil { return err } } else { var configuration []byte if strings.HasPrefix(config, "embed:") { filename := config[len("embed:"):] if configuration, err = configs.ReadFile(filepath.Join("configs", filename)); err != nil { return fmt.Errorf("reading embedded configuration for %s: %w", filename, err) } } else if strings.HasPrefix(config, "file:") { filename := config[len("file:"):] if configuration, err = os.ReadFile(filename); err != nil { return fmt.Errorf("reading configuration from file %s: %w", filename, err) } } else { return fmt.Errorf("unrecognized config option: [%s], `file:` to specify config file in file system, `embed:` to use embedded file, `test` to register test interface and receive config from test driver", config) } if server, err = grpcCliqueServer(ctx, false /* testServer */); err != nil { return err } if err = server.initAndConfig(configuration); err != nil { return err } } server.db = openDB(filepath.Join(datadir, "clique", "db"), logger) server.c = clique.New(server.chainConfig, ¶ms.SnapshotConfig{}, server.db) <-ctx.Done() return nil } func grpcCliqueServer(ctx context.Context, testServer bool) (*CliqueServerImpl, error) { // STARTING GRPC SERVER log.Info("Starting Clique server", "on", consensusAddr) listenConfig := net.ListenConfig{ Control: func(network, address string, _ syscall.RawConn) error { log.Info("Clique received connection", "via", network, "from", address) return nil }, } lis, err := listenConfig.Listen(ctx, "tcp", consensusAddr) if err != nil { return nil, fmt.Errorf("could not create Clique listener: %w, addr=%s", err, consensusAddr) } var ( streamInterceptors []grpc.StreamServerInterceptor unaryInterceptors []grpc.UnaryServerInterceptor ) //if metrics.Enabled { //streamInterceptors = append(streamInterceptors, grpc_prometheus.StreamServerInterceptor) //unaryInterceptors = append(unaryInterceptors, grpc_prometheus.UnaryServerInterceptor) //} streamInterceptors = append(streamInterceptors, grpc_recovery.StreamServerInterceptor()) unaryInterceptors = append(unaryInterceptors, grpc_recovery.UnaryServerInterceptor()) var grpcServer *grpc.Server cpus := uint32(runtime.GOMAXPROCS(-1)) opts := []grpc.ServerOption{ grpc.NumStreamWorkers(cpus), // reduce amount of goroutines grpc.WriteBufferSize(1024), // reduce buffers to save mem grpc.ReadBufferSize(1024), grpc.MaxConcurrentStreams(100), // to force clients reduce concurrency level grpc.KeepaliveParams(keepalive.ServerParameters{ Time: 10 * time.Minute, }), // Don't drop the connection, settings accordign to this comment on GitHub // https://github.com/grpc/grpc-go/issues/3171#issuecomment-552796779 grpc.KeepaliveEnforcementPolicy(keepalive.EnforcementPolicy{ MinTime: 10 * time.Second, PermitWithoutStream: true, }), grpc.StreamInterceptor(grpc_middleware.ChainStreamServer(streamInterceptors...)), grpc.UnaryInterceptor(grpc_middleware.ChainUnaryServer(unaryInterceptors...)), } grpcServer = grpc.NewServer(opts...) cliqueServer := NewCliqueServer(ctx) proto_cons.RegisterConsensusEngineServer(grpcServer, cliqueServer) if testServer { proto_cons.RegisterTestServer(grpcServer, cliqueServer) } //if metrics.Enabled { // grpc_prometheus.Register(grpcServer) //} go func() { if err1 := grpcServer.Serve(lis); err1 != nil { log.Error("Clique server fail", "err", err1) } }() return cliqueServer, nil } type CliqueServerImpl struct { proto_cons.UnimplementedConsensusEngineServer proto_cons.UnimplementedTestServer genesis *core.Genesis chainConfig *params.ChainConfig c *clique.Clique db kv.RwDB } func NewCliqueServer(_ context.Context) *CliqueServerImpl { return &CliqueServerImpl{} } // initAndConfig resets the Clique Engine and configures it according to given configuration func (cs *CliqueServerImpl) initAndConfig(configuration []byte) error { tree, err := toml.LoadBytes(configuration) if err != nil { return err } var ( epoch int64 period int64 vanityStr string signersStr []string gaslimit int64 timestamp int64 balances *toml.Tree chainIdStr string forksTree *toml.Tree difficultyStr string ok bool ) if epoch, ok = tree.Get("engine.params.epoch").(int64); !ok { return fmt.Errorf("engine.params.epoch absent or of wrong type") } if period, ok = tree.Get("engine.params.period").(int64); !ok { return fmt.Errorf("engine.params.period absent or of wrong type") } if vanityStr, ok = tree.Get("genesis.vanity").(string); !ok { return fmt.Errorf("genesis.vanity absent or of wrong type") } if signersStr, ok = tree.GetArray("genesis.signers").([]string); !ok { return fmt.Errorf("signers absent or of wrong type") } if gaslimit, ok = tree.Get("genesis.gas_limit").(int64); !ok { return fmt.Errorf("genesis.gaslimit absent or of wrong type") } if timestamp, ok = tree.Get("genesis.timestamp").(int64); !ok { return fmt.Errorf("genesis.timestamp absent or of wrong type") } if difficultyStr, ok = tree.Get("genesis.difficulty").(string); !ok { return fmt.Errorf("genesis.difficulty absent or of wrong type") } if balances, ok = tree.Get("genesis.balances").(*toml.Tree); !ok { return fmt.Errorf("genesis.balances absent or of wrong type") } // construct chain config var chainConfig params.ChainConfig if chainIdStr, ok = tree.Get("genesis.chain_id").(string); !ok { return fmt.Errorf("genesis.chain_id absent or of wrong type") } var chainId big.Int chainId.SetBytes(common.Hex2Bytes(chainIdStr)) chainConfig.ChainID = &chainId if forksTree, ok = tree.Get("forks").(*toml.Tree); !ok { return fmt.Errorf("forks absent or of wrong type") } for forkName, forkNumber := range forksTree.ToMap() { var number int64 if number, ok = forkNumber.(int64); !ok { return fmt.Errorf("forks.%s is of a wrong type: %T", forkName, forkNumber) } bigNumber := big.NewInt(number) switch forkName { case "homestead": chainConfig.HomesteadBlock = bigNumber case "tangerine": chainConfig.EIP150Block = bigNumber case "spurious": chainConfig.EIP155Block = bigNumber chainConfig.EIP158Block = bigNumber case "byzantium": chainConfig.ByzantiumBlock = bigNumber case "constantinople": chainConfig.ConstantinopleBlock = bigNumber case "petersburg": chainConfig.PetersburgBlock = bigNumber case "istanbul": chainConfig.IstanbulBlock = bigNumber case "berlin": chainConfig.BerlinBlock = bigNumber case "london": chainConfig.LondonBlock = bigNumber default: return fmt.Errorf("unknown fork name [%s]", forkName) } } chainConfig.Clique = ¶ms.CliqueConfig{ Epoch: uint64(epoch), Period: uint64(period), } // construct genesis var genesis core.Genesis genesis.Config = &chainConfig genesis.Timestamp = uint64(timestamp) genesis.ExtraData = common.FromHex(vanityStr) for _, signer := range signersStr { genesis.ExtraData = append(genesis.ExtraData, common.HexToAddress(signer).Bytes()...) } genesis.ExtraData = append(genesis.ExtraData, make([]byte, clique.ExtraSeal)...) genesis.GasLimit = uint64(gaslimit) genesis.Difficulty = new(big.Int).SetBytes(common.FromHex(difficultyStr)) genesis.Alloc = make(core.GenesisAlloc) for account, balance := range balances.ToMap() { genesis.Alloc[common.HexToAddress(account)] = core.GenesisAccount{ Balance: new(big.Int).SetBytes(common.FromHex(balance.(string))), } } var genesisBlock *types.Block if genesisBlock, _, err = genesis.ToBlock(); err != nil { return fmt.Errorf("creating genesis block: %w", err) } log.Info("Created genesis block", "hash", genesisBlock.Hash()) return nil } // StartTestCase implements Test interface from consensus.proto // When called, it signals to the consensus engine to reset its state and re-initialise using configuration // received from the test driver func (cs *CliqueServerImpl) StartTestCase(_ context.Context, testCase *proto_cons.StartTestCaseMessage) (*emptypb.Empty, error) { if testCase.Mechanism != "clique" { return &emptypb.Empty{}, fmt.Errorf("expected mechanism [clique], got [%s]", testCase.Mechanism) } if err := cs.initAndConfig(testCase.Config); err != nil { return &emptypb.Empty{}, err } return &emptypb.Empty{}, nil } func (cs *CliqueServerImpl) ChainSpec(context.Context, *emptypb.Empty) (*proto_cons.ChainSpecMessage, error) { var cliqueConfig []byte var forks []*proto_cons.Fork var chainId uint256.Int if cs.chainConfig.ChainID != nil { chainId.SetFromBig(cs.chainConfig.ChainID) } var genesisDiff uint256.Int if cs.genesis.Difficulty != nil { genesisDiff.SetFromBig(cs.genesis.Difficulty) } var extraWithoutSigners []byte extraWithoutSigners = append(extraWithoutSigners, cs.genesis.ExtraData[:clique.ExtraVanity]...) extraWithoutSigners = append(extraWithoutSigners, cs.genesis.ExtraData[len(cs.genesis.ExtraData)-clique.ExtraSeal:]...) tomlTree, err := toml.TreeFromMap(make(map[string]interface{})) if err != nil { return nil, err } genesisSigners := make([]string, (len(cs.genesis.ExtraData)-clique.ExtraVanity-clique.ExtraSeal)/common.AddressLength) for i := 0; i < len(genesisSigners); i++ { genesisSigners[i] = fmt.Sprintf("%x", cs.genesis.ExtraData[clique.ExtraVanity+i*common.AddressLength:]) } tomlTree.Set("genesis.signers", genesisSigners) cliqueConfig, err = tomlTree.Marshal() if err != nil { return nil, err } return &proto_cons.ChainSpecMessage{ Mechanism: "clique", MechanismConfig: cliqueConfig, Genesis: &proto_cons.Genesis{ ChainId: gointerfaces.ConvertUint256IntToH256(&chainId), Template: &proto_cons.Template{ ParentHash: gointerfaces.ConvertHashToH256(cs.genesis.ParentHash), Coinbase: gointerfaces.ConvertAddressToH160(cs.genesis.Coinbase), Difficulty: gointerfaces.ConvertUint256IntToH256(&genesisDiff), Number: cs.genesis.Number, GasLimit: cs.genesis.GasLimit, Time: cs.genesis.Timestamp, Extra: extraWithoutSigners, // Initial signers are passed in the clique-specific configuration Nonce: cs.genesis.Nonce, }, }, Forks: forks, }, nil }