mirror of
https://github.com/scroll-tech/scroll.git
synced 2026-01-15 00:48:01 -05:00
137 lines
4.2 KiB
Go
137 lines
4.2 KiB
Go
package app
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
"os/signal"
|
|
"time"
|
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
"github.com/scroll-tech/go-ethereum/ethclient"
|
|
"github.com/scroll-tech/go-ethereum/log"
|
|
"github.com/scroll-tech/go-ethereum/params"
|
|
"github.com/scroll-tech/go-ethereum/rpc"
|
|
"github.com/urfave/cli/v2"
|
|
|
|
"scroll-tech/common/database"
|
|
"scroll-tech/common/observability"
|
|
"scroll-tech/common/utils"
|
|
"scroll-tech/common/version"
|
|
|
|
"scroll-tech/rollup/internal/config"
|
|
"scroll-tech/rollup/internal/controller/relayer"
|
|
"scroll-tech/rollup/internal/controller/watcher"
|
|
butils "scroll-tech/rollup/internal/utils"
|
|
)
|
|
|
|
var app *cli.App
|
|
|
|
func init() {
|
|
// Set up gas-oracle app info.
|
|
app = cli.NewApp()
|
|
app.Action = action
|
|
app.Name = "gas-oracle"
|
|
app.Usage = "The Scroll Gas Oracle"
|
|
app.Description = "Scroll Gas Oracle."
|
|
app.Version = version.Version
|
|
app.Flags = append(app.Flags, utils.CommonFlags...)
|
|
app.Commands = []*cli.Command{}
|
|
app.Before = func(ctx *cli.Context) error {
|
|
return utils.LogSetup(ctx)
|
|
}
|
|
// Register `gas-oracle-test` app for integration-test.
|
|
utils.RegisterSimulation(app, utils.GasOracleApp)
|
|
}
|
|
|
|
func action(ctx *cli.Context) error {
|
|
// Load config file.
|
|
cfgFile := ctx.String(utils.ConfigFileFlag.Name)
|
|
cfg, err := config.NewConfig(cfgFile)
|
|
if err != nil {
|
|
log.Crit("failed to load config file", "config file", cfgFile, "error", err)
|
|
}
|
|
subCtx, cancel := context.WithCancel(ctx.Context)
|
|
// Init db connection
|
|
db, err := database.InitDB(cfg.DBConfig)
|
|
if err != nil {
|
|
log.Crit("failed to init db connection", "err", err)
|
|
}
|
|
defer func() {
|
|
cancel()
|
|
if err = database.CloseDB(db); err != nil {
|
|
log.Crit("failed to close db connection", "error", err)
|
|
}
|
|
}()
|
|
|
|
registry := prometheus.DefaultRegisterer
|
|
observability.Server(ctx, db)
|
|
|
|
l1client, err := ethclient.Dial(cfg.L1Config.Endpoint)
|
|
if err != nil {
|
|
log.Crit("failed to connect l1 geth", "config file", cfgFile, "error", err)
|
|
}
|
|
|
|
// Init l2geth connection
|
|
l2client, err := ethclient.Dial(cfg.L2Config.Endpoint)
|
|
if err != nil {
|
|
log.Crit("failed to connect l2 geth", "config file", cfgFile, "error", err)
|
|
}
|
|
|
|
genesisPath := ctx.String(utils.Genesis.Name)
|
|
genesis, err := utils.ReadGenesis(genesisPath)
|
|
if err != nil {
|
|
log.Crit("failed to read genesis", "genesis file", genesisPath, "error", err)
|
|
}
|
|
|
|
l1watcher := watcher.NewL1WatcherClient(ctx.Context, l1client, cfg.L1Config.StartHeight, db, registry)
|
|
|
|
l1relayer, err := relayer.NewLayer1Relayer(ctx.Context, db, cfg.L1Config.RelayerConfig, genesis.Config, relayer.ServiceTypeL1GasOracle, registry)
|
|
if err != nil {
|
|
log.Crit("failed to create new l1 relayer", "config file", cfgFile, "error", err)
|
|
}
|
|
l2relayer, err := relayer.NewLayer2Relayer(ctx.Context, l2client, db, cfg.L2Config.RelayerConfig, ¶ms.ChainConfig{}, false /* initGenesis */, relayer.ServiceTypeL2GasOracle, registry)
|
|
if err != nil {
|
|
log.Crit("failed to create new l2 relayer", "config file", cfgFile, "error", err)
|
|
}
|
|
// Start l1 watcher process
|
|
go utils.LoopWithContext(subCtx, 10*time.Second, func(ctx context.Context) {
|
|
// Fetch the latest block number to decrease the delay when fetching gas prices
|
|
// Use latest block number - 1 to prevent frequent reorg
|
|
number, loopErr := butils.GetLatestConfirmedBlockNumber(ctx, l1client, rpc.LatestBlockNumber)
|
|
if loopErr != nil {
|
|
log.Error("failed to get block number", "err", loopErr)
|
|
return
|
|
}
|
|
|
|
if loopErr = l1watcher.FetchBlockHeader(number - 1); loopErr != nil {
|
|
log.Error("Failed to fetch L1 block header", "lastest", number-1, "err", loopErr)
|
|
return
|
|
}
|
|
})
|
|
|
|
// Start l1relayer process
|
|
go utils.Loop(subCtx, 10*time.Second, l1relayer.ProcessGasPriceOracle)
|
|
go utils.Loop(subCtx, 2*time.Second, l2relayer.ProcessGasPriceOracle)
|
|
|
|
// Finish start all message relayer functions
|
|
log.Info("Start gas-oracle successfully", "version", version.Version)
|
|
|
|
// Catch CTRL-C to ensure a graceful shutdown.
|
|
interrupt := make(chan os.Signal, 1)
|
|
signal.Notify(interrupt, os.Interrupt)
|
|
|
|
// Wait until the interrupt signal is received from an OS signal.
|
|
<-interrupt
|
|
|
|
return nil
|
|
}
|
|
|
|
// Run message_relayer cmd instance.
|
|
func Run() {
|
|
if err := app.Run(os.Args); err != nil {
|
|
_, _ = fmt.Fprintln(os.Stderr, err)
|
|
os.Exit(1)
|
|
}
|
|
}
|