erigon-pulse/polygon/bor/finality/whitelist.go
2024-01-09 19:20:42 +01:00

240 lines
6.6 KiB
Go

package finality
import (
"context"
"errors"
"fmt"
"time"
"github.com/ledgerwatch/log/v3"
"github.com/ledgerwatch/erigon/polygon/heimdall"
"github.com/ledgerwatch/erigon-lib/kv"
"github.com/ledgerwatch/erigon/polygon/bor/finality/flags"
"github.com/ledgerwatch/erigon/polygon/bor/finality/whitelist"
"github.com/ledgerwatch/erigon/turbo/services"
)
type config struct {
heimdall heimdall.IHeimdallClient
borDB kv.RwDB
chainDB kv.RwDB
blockReader services.BlockReader
logger log.Logger
borAPI BorAPI
closeCh chan struct{}
}
type BorAPI interface {
GetRootHash(start uint64, end uint64) (string, error)
}
func Whitelist(heimdall heimdall.IHeimdallClient, borDB kv.RwDB, chainDB kv.RwDB, blockReader services.BlockReader, logger log.Logger, borAPI BorAPI, closeCh chan struct{}) {
if !flags.Milestone {
return
}
config := &config{
heimdall: heimdall,
borDB: borDB,
chainDB: chainDB,
blockReader: blockReader,
logger: logger,
borAPI: borAPI,
closeCh: closeCh,
}
go startCheckpointWhitelistService(config)
go startMilestoneWhitelistService(config)
go startNoAckMilestoneService(config)
go startNoAckMilestoneByIDService(config)
}
const (
whitelistTimeout = 30 * time.Second
noAckMilestoneTimeout = 4 * time.Second
)
// StartCheckpointWhitelistService starts the goroutine to fetch checkpoints and update the
// checkpoint whitelist map.
func startCheckpointWhitelistService(config *config) {
const (
tickerDuration = 100 * time.Second
fnName = "whitelist checkpoint"
)
RetryHeimdallHandler(handleWhitelistCheckpoint, config, tickerDuration, whitelistTimeout, fnName)
}
// startMilestoneWhitelistService starts the goroutine to fetch milestiones and update the
// milestone whitelist map.
func startMilestoneWhitelistService(config *config) {
const (
tickerDuration = 12 * time.Second
fnName = "whitelist milestone"
)
RetryHeimdallHandler(handleMilestone, config, tickerDuration, whitelistTimeout, fnName)
}
func startNoAckMilestoneService(config *config) {
const (
tickerDuration = 6 * time.Second
fnName = "no-ack-milestone service"
)
RetryHeimdallHandler(handleNoAckMilestone, config, tickerDuration, noAckMilestoneTimeout, fnName)
}
func startNoAckMilestoneByIDService(config *config) {
const (
tickerDuration = 1 * time.Minute
fnName = "no-ack-milestone-by-id service"
)
RetryHeimdallHandler(handleNoAckMilestoneByID, config, tickerDuration, noAckMilestoneTimeout, fnName)
}
type heimdallHandler func(ctx context.Context, heimdallClient heimdall.IHeimdallClient, config *config) error
func RetryHeimdallHandler(fn heimdallHandler, config *config, tickerDuration time.Duration, timeout time.Duration, fnName string) {
retryHeimdallHandler(fn, config, tickerDuration, timeout, fnName)
}
func retryHeimdallHandler(fn heimdallHandler, config *config, tickerDuration time.Duration, timeout time.Duration, fnName string) {
// a shortcut helps with tests and early exit
select {
case <-config.closeCh:
return
default:
}
if config.heimdall == nil {
config.logger.Error("[bor] engine not available")
return
}
// first run for fetching milestones
firstCtx, cancel := context.WithTimeout(context.Background(), timeout)
err := fn(firstCtx, config.heimdall, config)
cancel()
if err != nil {
if !errors.Is(err, errMissingBlocks) {
config.logger.Warn(fmt.Sprintf("[bor] unable to start the %s service - first run", fnName), "err", err)
}
}
ticker := time.NewTicker(tickerDuration)
defer ticker.Stop()
for {
defer func() {
r := recover()
if r != nil {
log.Warn(fmt.Sprintf("[bor] service %s- run failed with panic", fnName), "err", r)
}
}()
select {
case <-ticker.C:
ctx, cancel := context.WithTimeout(context.Background(), timeout)
err := fn(ctx, config.heimdall, config)
cancel()
if err != nil {
if errors.Is(err, errMissingBlocks) {
config.logger.Debug(fmt.Sprintf("[bor] unable to handle %s", fnName), "err", err)
} else {
config.logger.Warn(fmt.Sprintf("[bor] unable to handle %s", fnName), "err", err)
}
}
case <-config.closeCh:
return
}
}
}
// handleWhitelistCheckpoint handles the checkpoint whitelist mechanism.
func handleWhitelistCheckpoint(ctx context.Context, heimdallClient heimdall.IHeimdallClient, config *config) error {
service := whitelist.GetWhitelistingService()
// Create a new bor verifier, which will be used to verify checkpoints and milestones
verifier := newBorVerifier()
blockNum, blockHash, err := fetchWhitelistCheckpoint(ctx, heimdallClient, verifier, config)
// If the array is empty, we're bound to receive an error. Non-nill error and non-empty array
// means that array has partial elements and it failed for some block. We'll add those partial
// elements anyway.
if err != nil {
return err
}
service.ProcessCheckpoint(blockNum, blockHash)
return nil
}
// handleMilestone handles the milestone mechanism.
func handleMilestone(ctx context.Context, heimdallClient heimdall.IHeimdallClient, config *config) error {
service := whitelist.GetWhitelistingService()
// Create a new bor verifier, which will be used to verify checkpoints and milestones
verifier := newBorVerifier()
num, hash, err := fetchWhitelistMilestone(ctx, heimdallClient, verifier, config)
// If the current chain head is behind the received milestone, add it to the future milestone
// list. Also, the hash mismatch (end block hash) error will lead to rewind so also
// add that milestone to the future milestone list.
if errors.Is(err, errMissingBlocks) || errors.Is(err, errHashMismatch) {
service.ProcessFutureMilestone(num, hash)
return nil
}
if errors.Is(err, heimdall.ErrServiceUnavailable) {
return nil
}
if err != nil {
return err
}
service.ProcessMilestone(num, hash)
return nil
}
func handleNoAckMilestone(ctx context.Context, heimdallClient heimdall.IHeimdallClient, config *config) error {
service := whitelist.GetWhitelistingService()
milestoneID, err := fetchNoAckMilestone(ctx, heimdallClient, config.logger)
if errors.Is(err, heimdall.ErrServiceUnavailable) {
return nil
}
if err != nil {
return err
}
service.RemoveMilestoneID(milestoneID)
return nil
}
func handleNoAckMilestoneByID(ctx context.Context, heimdallClient heimdall.IHeimdallClient, config *config) error {
service := whitelist.GetWhitelistingService()
milestoneIDs := service.GetMilestoneIDsList()
for _, milestoneID := range milestoneIDs {
err := fetchNoAckMilestoneByID(ctx, heimdallClient, milestoneID, config.logger)
if err == nil {
service.RemoveMilestoneID(milestoneID)
}
}
return nil
}