mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-18 16:44:12 +00:00
143 lines
5.2 KiB
Go
143 lines
5.2 KiB
Go
package finality
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
|
|
"github.com/ledgerwatch/log/v3"
|
|
|
|
"github.com/ledgerwatch/erigon/polygon/heimdall"
|
|
|
|
"github.com/ledgerwatch/erigon-lib/common"
|
|
"github.com/ledgerwatch/erigon/polygon/bor/finality/whitelist"
|
|
)
|
|
|
|
var (
|
|
// errCheckpoint is returned when we are unable to fetch the
|
|
// latest checkpoint from the local heimdall.
|
|
errCheckpoint = errors.New("failed to fetch latest checkpoint")
|
|
|
|
// errMilestone is returned when we are unable to fetch the
|
|
// latest milestone from the local heimdall.
|
|
errMilestone = errors.New("failed to fetch latest milestone")
|
|
|
|
ErrNotInRejectedList = errors.New("MilestoneID not in rejected list")
|
|
)
|
|
|
|
// fetchWhitelistCheckpoint fetches the latest checkpoint from it's local heimdall
|
|
// and verifies the data against bor data.
|
|
func fetchWhitelistCheckpoint(ctx context.Context, heimdallClient heimdall.IHeimdallClient, verifier *borVerifier, config *config) (uint64, common.Hash, error) {
|
|
var (
|
|
blockNum uint64
|
|
blockHash common.Hash
|
|
)
|
|
|
|
// fetch the latest checkpoint from Heimdall
|
|
checkpoint, err := heimdallClient.FetchCheckpoint(ctx, -1)
|
|
if err != nil {
|
|
config.logger.Debug("[bor.heimdall] Failed to fetch latest checkpoint for whitelisting", "err", err)
|
|
return blockNum, blockHash, errCheckpoint
|
|
}
|
|
|
|
// Verify if the checkpoint fetched can be added to the local whitelist entry or not
|
|
// If verified, it returns the hash of the end block of the checkpoint. If not,
|
|
// it will return appropriate error.
|
|
hash, err := verifier.verify(ctx, config, checkpoint.StartBlock.Uint64(), checkpoint.EndBlock.Uint64(), checkpoint.RootHash.String()[2:], true)
|
|
|
|
if err != nil {
|
|
if errors.Is(err, errMissingBlocks) {
|
|
config.logger.Debug("[bor.heimdall] Got new checkpoint", "start", checkpoint.StartBlock.Uint64(), "end", checkpoint.EndBlock.Uint64(), "rootHash", checkpoint.RootHash.String())
|
|
config.logger.Debug("[bor.heimdall] Failed to whitelist checkpoint", "err", err)
|
|
} else {
|
|
config.logger.Info("[bor.heimdall] Got new checkpoint", "start", checkpoint.StartBlock.Uint64(), "end", checkpoint.EndBlock.Uint64(), "rootHash", checkpoint.RootHash.String())
|
|
config.logger.Warn("[bor.heimdall] Failed to whitelist checkpoint", "err", err)
|
|
}
|
|
|
|
return blockNum, blockHash, err
|
|
}
|
|
|
|
config.logger.Info("[bor.heimdall] Got new checkpoint", "start", checkpoint.StartBlock.Uint64(), "end", checkpoint.EndBlock.Uint64(), "rootHash", checkpoint.RootHash.String())
|
|
|
|
blockNum = checkpoint.EndBlock.Uint64()
|
|
blockHash = common.HexToHash(hash)
|
|
|
|
return blockNum, blockHash, nil
|
|
}
|
|
|
|
// fetchWhitelistMilestone fetches the latest milestone from it's local heimdall
|
|
// and verifies the data against bor data.
|
|
func fetchWhitelistMilestone(ctx context.Context, heimdallClient heimdall.IHeimdallClient, verifier *borVerifier, config *config) (uint64, common.Hash, error) {
|
|
var (
|
|
num uint64
|
|
hash common.Hash
|
|
)
|
|
|
|
// fetch latest milestone
|
|
milestone, err := heimdallClient.FetchMilestone(ctx, -1)
|
|
if errors.Is(err, heimdall.ErrServiceUnavailable) {
|
|
config.logger.Debug("[bor.heimdall] Failed to fetch latest milestone for whitelisting", "err", err)
|
|
return num, hash, err
|
|
}
|
|
|
|
if err != nil {
|
|
config.logger.Warn("[bor.heimdall] Failed to fetch latest milestone for whitelisting", "err", err)
|
|
return num, hash, errMilestone
|
|
}
|
|
|
|
config.logger.Debug("[bor.heimdall] Got new milestone", "start", milestone.StartBlock.Uint64(), "end", milestone.EndBlock.Uint64())
|
|
|
|
num = milestone.EndBlock.Uint64()
|
|
hash = milestone.Hash
|
|
|
|
// Verify if the milestone fetched can be added to the local whitelist entry or not
|
|
// If verified, it returns the hash of the end block of the milestone. If not,
|
|
// it will return appropriate error.
|
|
_, err = verifier.verify(ctx, config, milestone.StartBlock.Uint64(), milestone.EndBlock.Uint64(), milestone.Hash.String()[2:], false)
|
|
if err != nil {
|
|
whitelist.GetWhitelistingService().UnlockSprint(milestone.EndBlock.Uint64())
|
|
return num, hash, err
|
|
}
|
|
|
|
return num, hash, nil
|
|
}
|
|
|
|
func fetchNoAckMilestone(ctx context.Context, heimdallClient heimdall.IHeimdallClient, logger log.Logger) (string, error) {
|
|
var (
|
|
milestoneID string
|
|
)
|
|
|
|
milestoneID, err := heimdallClient.FetchLastNoAckMilestone(ctx)
|
|
if errors.Is(err, heimdall.ErrServiceUnavailable) {
|
|
logger.Debug("[bor.heimdall] Failed to fetch latest no-ack milestone", "err", err)
|
|
return milestoneID, err
|
|
}
|
|
|
|
if err != nil {
|
|
logger.Warn("[bor.heimdall] Failed to fetch latest no-ack milestone", "err", err)
|
|
return milestoneID, errMilestone
|
|
}
|
|
|
|
return milestoneID, nil
|
|
}
|
|
|
|
func fetchNoAckMilestoneByID(ctx context.Context, heimdallClient heimdall.IHeimdallClient, milestoneID string, logger log.Logger) error {
|
|
err := heimdallClient.FetchNoAckMilestone(ctx, milestoneID)
|
|
if errors.Is(err, heimdall.ErrServiceUnavailable) {
|
|
logger.Debug("[bor.heimdall] Failed to fetch no-ack milestone by ID", "milestoneID", milestoneID, "err", err)
|
|
return err
|
|
}
|
|
|
|
// fixme: handle different types of errors
|
|
if errors.Is(err, ErrNotInRejectedList) {
|
|
logger.Warn("[bor.heimdall] MilestoneID not in rejected list", "milestoneID", milestoneID, "err", err)
|
|
return err
|
|
}
|
|
|
|
if err != nil {
|
|
logger.Warn("[bor.heimdall] Failed to fetch no-ack milestone by ID ", "milestoneID", milestoneID, "err", err)
|
|
return errMilestone
|
|
}
|
|
|
|
return nil
|
|
}
|