2023-09-13 10:49:49 +00:00
|
|
|
package whitelist
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/ledgerwatch/erigon-lib/common"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
|
|
"github.com/ledgerwatch/erigon/core/types"
|
2024-01-09 18:20:42 +00:00
|
|
|
"github.com/ledgerwatch/erigon/polygon/bor/finality/rawdb"
|
2023-09-13 10:49:49 +00:00
|
|
|
"github.com/ledgerwatch/log/v3"
|
|
|
|
)
|
|
|
|
|
|
|
|
type finality[T rawdb.BlockFinality[T]] struct {
|
|
|
|
sync.RWMutex
|
|
|
|
db kv.RwDB
|
|
|
|
Hash common.Hash // Whitelisted Hash, populated by reaching out to heimdall
|
|
|
|
Number uint64 // Number , populated by reaching out to heimdall
|
|
|
|
interval uint64 // Interval, until which we can allow importing
|
|
|
|
doExist bool
|
|
|
|
}
|
|
|
|
|
|
|
|
type finalityService interface {
|
|
|
|
IsValidChain(currentHeader uint64, chain []*types.Header) bool
|
|
|
|
Get() (bool, uint64, common.Hash)
|
|
|
|
Process(block uint64, hash common.Hash)
|
|
|
|
Purge()
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsValidChain checks the validity of chain by comparing it
|
|
|
|
// against the local checkpoint entry
|
|
|
|
func (f *finality[T]) IsValidChain(currentHeader uint64, chain []*types.Header) bool {
|
|
|
|
// Return if we've received empty chain
|
|
|
|
if len(chain) == 0 {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
res := isValidChain(currentHeader, chain, f.doExist, f.Number, f.Hash, f.interval)
|
|
|
|
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *finality[T]) Process(block uint64, hash common.Hash) {
|
|
|
|
f.doExist = true
|
|
|
|
f.Hash = hash
|
|
|
|
f.Number = block
|
|
|
|
|
|
|
|
err := rawdb.WriteLastFinality[T](f.db, block, hash)
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
log.Error("Error in writing whitelist state to db", "err", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get returns the existing whitelisted
|
|
|
|
// entries of checkpoint of the form (doExist,block number,block hash.)
|
|
|
|
func (f *finality[T]) Get() (bool, uint64, common.Hash) {
|
|
|
|
f.RLock()
|
|
|
|
defer f.RUnlock()
|
|
|
|
|
|
|
|
if f.doExist {
|
|
|
|
return f.doExist, f.Number, f.Hash
|
|
|
|
}
|
|
|
|
|
|
|
|
block, hash, err := rawdb.ReadFinality[T](f.db)
|
|
|
|
if err != nil {
|
|
|
|
return false, f.Number, f.Hash
|
|
|
|
}
|
|
|
|
|
|
|
|
return true, block, hash
|
|
|
|
}
|
|
|
|
|
|
|
|
// Purge purges the whitlisted checkpoint
|
|
|
|
func (f *finality[T]) Purge() {
|
|
|
|
f.Lock()
|
|
|
|
defer f.Unlock()
|
|
|
|
|
|
|
|
f.doExist = false
|
|
|
|
}
|