2021-04-29 15:14:10 +00:00
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"
"github.com/holiman/uint256"
2022-05-07 08:02:34 +00:00
common2 "github.com/ledgerwatch/erigon-lib/common"
2021-07-01 21:31:14 +00:00
"github.com/ledgerwatch/erigon-lib/gointerfaces"
proto_cons "github.com/ledgerwatch/erigon-lib/gointerfaces/consensus"
2021-07-29 11:53:13 +00:00
"github.com/ledgerwatch/erigon-lib/kv"
2021-05-20 18:25:53 +00:00
"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"
2021-07-29 10:23:23 +00:00
"github.com/ledgerwatch/log/v3"
2021-06-05 15:17:04 +00:00
"github.com/pelletier/go-toml"
2021-04-29 15:14:10 +00:00
"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 )
2022-02-22 17:39:48 +00:00
withDataDir ( cliqueCmd )
2021-04-29 15:14:10 +00:00
withConfig ( cliqueCmd )
rootCmd . AddCommand ( cliqueCmd )
}
var cliqueCmd = & cobra . Command {
Use : "clique" ,
Short : "Run clique consensus engine" ,
RunE : func ( cmd * cobra . Command , args [ ] string ) error {
2022-01-19 03:49:07 +00:00
ctx , _ := common2 . RootContext ( )
2021-07-28 02:47:38 +00:00
logger := log . New ( )
return cliqueEngine ( ctx , logger )
2021-04-29 15:14:10 +00:00
} ,
}
2021-07-28 02:47:38 +00:00
func cliqueEngine ( ctx context . Context , logger log . Logger ) error {
2021-04-29 15:14:10 +00:00
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:<path>` to specify config file in file system, `embed:<path>` 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
}
}
2021-07-28 02:47:38 +00:00
server . db = openDB ( filepath . Join ( datadir , "clique" , "db" ) , logger )
2021-11-29 03:43:19 +00:00
server . c = clique . New ( server . chainConfig , params . CliqueSnapshot , server . db )
2021-04-29 15:14:10 +00:00
<- 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
)
2021-08-08 02:17:08 +00:00
//if metrics.Enabled {
//streamInterceptors = append(streamInterceptors, grpc_prometheus.StreamServerInterceptor)
//unaryInterceptors = append(unaryInterceptors, grpc_prometheus.UnaryServerInterceptor)
//}
2021-04-29 15:14:10 +00:00
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 ,
} ) ,
2021-09-22 04:57:09 +00:00
// 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 ,
} ) ,
2021-04-29 15:14:10 +00:00
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 )
}
2021-08-08 02:17:08 +00:00
//if metrics.Enabled {
// grpc_prometheus.Register(grpcServer)
//}
2021-04-29 15:14:10 +00:00
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
2021-07-28 02:47:38 +00:00
db kv . RwDB
2021-04-29 15:14:10 +00:00
}
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
2021-05-11 09:54:20 +00:00
case "london" :
chainConfig . LondonBlock = bigNumber
2021-04-29 15:14:10 +00:00
default :
return fmt . Errorf ( "unknown fork name [%s]" , forkName )
}
}
chainConfig . Clique = & params . 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
2021-05-07 21:07:49 +00:00
if genesisBlock , _ , err = genesis . ToBlock ( ) ; err != nil {
2021-04-29 15:14:10 +00:00
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
}