diff --git a/README.md b/README.md
index d9cddba8..68dc9845 100644
--- a/README.md
+++ b/README.md
@@ -131,7 +131,7 @@ This component is designed to work exclusively with chains that are already util
This is a new version of the deprecated [chain-mon faultproof-wd-mon](https://github.com/ethereum-optimism/optimism/tree/chain-mon/v1.2.1/packages/chain-mon/src/faultproof-wd-mon).
For detailed information on how the component works and the algorithms used, please refer to the component README.
-| `op-monitorism/faultproof-withdrawals` | [README](https://github.com/ethereum-optimism/monitorism/blob/main/op-monitorism/faultproof-withdrawals/README.md) |
+| `op-monitorism/faultproof-withdrawals` | [README](https://github.com/ethereum-optimism/monitorism/blob/main/op-monitorism/faultproof_withdrawals/README.md) |
| ----------------------- | --------------------------------------------------------------------------------------------------- |
diff --git a/op-monitorism/README.md b/op-monitorism/README.md
deleted file mode 100644
index a3b04fb0..00000000
--- a/op-monitorism/README.md
+++ /dev/null
@@ -1,146 +0,0 @@
-# Monitors
-
-`op-monitorism` is a collection of monitoring tools for the OP stack. Each monitor is designed to track a specific aspect of the Optimism stack and emit metrics that can be used to set up alerts.
-
-The following the commands are currently available:
-
-```bash
-NAME:
- Monitorism - OP Stack Monitoring
-
-USAGE:
- Monitorism [global options] command [command options]
-
-VERSION:
- 0.1.0-unstable
-
-DESCRIPTION:
- OP Stack Monitoring
-
-COMMANDS:
- multisig Monitors OptimismPortal pause status, Safe nonce, and Pre-Signed nonce stored in 1Password
- fault Monitors output roots posted on L1 against L2
- withdrawals Monitors proven withdrawals on L1 against L2
- balances Monitors account balances
- drippie Monitors Drippie contract
- secrets Monitors secrets revealed in the CheckSecrets dripcheck
- global_events Monitors global events with YAML configuration
- liveness_expiration Monitor the liveness expiration on Gnosis Safe.
- version Show version
- help, h Shows a list of commands or help for one command
-
-GLOBAL OPTIONS:
- --help, -h show help
- --version, -v print the version
-```
-
-Each _monitor_ has some common configuration, configurable both via cli or env with defaults.
-
-```bash
-OPTIONS:
- --log.level value [$MONITORISM_LOG_LEVEL] The lowest log level that will be output (default: INFO)
- --log.format value [$MONITORISM_LOG_FORMAT] Format the log output. Supported formats: 'text', 'terminal', 'logfmt', 'json', 'json-pretty', (default: text)
- --log.color [$MONITORISM_LOG_COLOR] Color the log output if in terminal mode (default: false)
- --metrics.enabled [$MONITORISM_METRICS_ENABLED] Enable the metrics server (default: false)
- --metrics.addr value [$MONITORISM_METRICS_ADDR] Metrics listening address (default: "0.0.0.0")
- --metrics.port value [$MONITORISM_METRICS_PORT] Metrics listening port (default: 7300)
- --loop.interval.msec value [$MONITORISM_LOOP_INTERVAL_MSEC] Loop interval of the monitor in milliseconds (default: 60000)
-```
-
-### Liveness Expiration Monitor
-
-![ab27497cea05fbd51b7b1c2ecde5bc69307ac0f27349f6bba4f3f21423116071](https://github.com/ethereum-optimism/monitorism/assets/23560242/af7a7e29-fff5-4df3-82f0-94c2f28fde84)
-
-The Liveness Expiration Monitor is made for monitoring the liveness expiration on Safes.
-
-| `op-monitorism/liveness_expiration` | [README](https://github.com/ethereum-optimism/monitorism/blob/main/op-monitorism/liveness_expiration/README.md) |
-| ----------------------------------- | --------------------------------------------------------------------------------------------------------------- |
-
-### Withdrawals Monitor
-
-![6d5477f5585cb49ff2f8bd147c2e7037772de6a1dd128ce4331596b011ce6ea9](https://github.com/user-attachments/assets/ac5e0a61-b495-4254-b32a-86abf61f0dc1)
-
-The withdrawals monitor checks for new withdrawals that have been proven to the `OptimismPortal` contract.
-Each withdrawal is checked against the `L2ToL1MessagePasser` contract.
-
-| `op-monitorism/withdrawals` | [README](https://github.com/ethereum-optimism/monitorism/blob/main/op-monitorism/withdrawals/README.md) |
-| --------------------------- | ------------------------------------------------------------------------------------------------------- |
-
-### Balances Monitor
-
-![5cd47a6e0f2fb7d921001db9eea24bb62bb892615011d03f275e02a147823827](https://github.com/user-attachments/assets/44884a76-e06d-4f58-a21f-94c2275e9d8b)
-
-The balances monitor simply emits a metric reporting the balances for the configured accounts.
-
-| `op-monitorism/balances` | [README](https://github.com/ethereum-optimism/monitorism/blob/main/op-monitorism/balances/README.md) |
-| ------------------------ | ---------------------------------------------------------------------------------------------------- |
-
-### Fault Monitor
-
-
-
-
-
-The fault monitor checks for changes in output roots posted to the `L2OutputOracle` contract.
-On change, reconstructing the output root from a trusted L2 source and looking for a match.
-
-| `op-monitorism/fault` | [README](https://github.com/ethereum-optimism/monitorism/blob/main/op-monitorism/fault/README.md) |
-| --------------------- | ------------------------------------------------------------------------------------------------- |
-
-### Multisig Monitor
-
-![7dab260ee38122980274fee27b114c590405cff2e5a68e6090290ecb786b68f2](https://github.com/user-attachments/assets/0eeb161b-923a-40fd-b561-468df3d5091d)
-
-The multisig monitor reports the paused status of the `OptimismPortal` contract.
-If set, the latest nonce of the configued `Safe` address. And also if set, the latest presigned nonce stored in One Password.
-The latest presigned nonce is identifyed by looking for items in the configued vault that follow a `ready-.json` name.
-The highest nonce of this item name format is reported.
-
-| `op-monitorism/multisig` | [README](https://github.com/ethereum-optimism/monitorism/blob/main/op-monitorism/multisig/README.md) |
-| ------------------------ | ---------------------------------------------------------------------------------------------------- |
-
-### Drippie Monitor
-
-The drippie monitor tracks the execution and executability of drips within a Drippie contract.
-
-| `op-monitorism/drippie` | [README](https://github.com/ethereum-optimism/monitorism/blob/main/op-monitorism/drippie/README.md) |
-| ----------------------- | ---------------------------------------------------------------------------------------------------- |
-
-### Secrets Monitor
-
-The secrets monitor takes a Drippie contract as a parameter and monitors for any drips within that contract that use the CheckSecrets dripcheck contract. CheckSecrets is a dripcheck that allows a drip to begin once a specific secret has been revealed (after a delay period) and cancels the drip if a second secret is revealed. It's important to monitor for these secrets being revealed as this could be a sign that the secret storage platform has been compromised and someone is attempting to exflitrate the ETH controlled by that drip.
-
-| `op-monitorism/secrets` | [README](https://github.com/ethereum-optimism/monitorism/blob/main/op-monitorism/secrets/README.md) |
-| ----------------------- | ---------------------------------------------------------------------------------------------------- |
-
-## CLI and Docs
-
-## Development
-
-After cloning, please run `./bootstrap.sh` to set up the development environment correctly.
-
-## Intro
-
-The cli has the ability to spin up a monitor for varying activities, each emmitting metrics used to setup alerts.
-
-```
-COMMANDS:
- multisig Monitors OptimismPortal pause status, Safe nonce, and Pre-Signed nonce stored in 1Password
- fault Monitors output roots posted on L1 against L2
- withdrawals Monitors proven withdrawals on L1 against L2
- balances Monitors account balances
- secrets Monitors secrets revealed in the CheckSecrets dripcheck
-```
-
-Each monitor has some common configuration, configurable both via cli or env with defaults.
-
-```
-OPTIONS:
- --log.level value [$MONITORISM_LOG_LEVEL] The lowest log level that will be output (default: INFO)
- --log.format value [$MONITORISM_LOG_FORMAT] Format the log output. Supported formats: 'text', 'terminal', 'logfmt', 'json', 'json-pretty', (default: text)
- --log.color [$MONITORISM_LOG_COLOR] Color the log output if in terminal mode (default: false)
- --metrics.enabled [$MONITORISM_METRICS_ENABLED] Enable the metrics server (default: false)
- --metrics.addr value [$MONITORISM_METRICS_ADDR] Metrics listening address (default: "0.0.0.0")
- --metrics.port value [$MONITORISM_METRICS_PORT] Metrics listening port (default: 7300)
- --loop.interval.msec value [$MONITORISM_LOOP_INTERVAL_MSEC] Loop interval of the monitor in milliseconds (default: 60000)
-```
diff --git a/op-monitorism/faultproof_withdrawals/.env.op.mainnet.example b/op-monitorism/faultproof_withdrawals/.env.op.mainnet.example
new file mode 100644
index 00000000..4c2e5da6
--- /dev/null
+++ b/op-monitorism/faultproof_withdrawals/.env.op.mainnet.example
@@ -0,0 +1,9 @@
+FAULTPROOF_WITHDRAWAL_MON_L1_GETH_URL=""
+FAULTPROOF_WITHDRAWAL_MON_L2_OP_NODE_URL=""
+FAULTPROOF_WITHDRAWAL_MON_L2_OP_GETH_URL=""
+FAULTPROOF_WITHDRAWAL_MON_OPTIMISM_PORTAL="0xbEb5Fc579115071764c7423A4f12eDde41f106Ed" # This is the address of the Optimism portal contract, this should be for the chain you are monitoring
+FAULTPROOF_WITHDRAWAL_MON_START_BLOCK_HEIGHT=20872390 # This is the block height from which the monitoring will start, decide at which block height you want to start monitoring
+FAULTPROOF_WITHDRAWAL_MON_EVENT_BLOCK_RANGE=1000 # This is the range of blocks to be monitored
+MONITORISM_LOOP_INTERVAL_MSEC=100 # This is the interval in milliseconds for the monitoring loop
+MONITORISM_METRICS_PORT=7300 # This is the port on which the metrics server will run
+MONITORISM_METRICS_ENABLED=true # This is the flag to enable/disable the metrics server
diff --git a/op-monitorism/faultproof_withdrawals/README.md b/op-monitorism/faultproof_withdrawals/README.md
index a29966de..3bcfb3ac 100644
--- a/op-monitorism/faultproof_withdrawals/README.md
+++ b/op-monitorism/faultproof_withdrawals/README.md
@@ -22,8 +22,6 @@ or
cd ../
go run ./cmd/monitorism faultproof_withdrawals --metrics.enabled --metrics.port 7300
```
-## Available Metrics and Meaning
-
# Cli options
diff --git a/op-monitorism/faultproof_withdrawals/monitor.go b/op-monitorism/faultproof_withdrawals/monitor.go
index c4fc5ba6..adfbd9c9 100644
--- a/op-monitorism/faultproof_withdrawals/monitor.go
+++ b/op-monitorism/faultproof_withdrawals/monitor.go
@@ -9,6 +9,7 @@ import (
"github.com/ethereum-optimism/monitorism/op-monitorism/faultproof_withdrawals/validator"
"github.com/ethereum-optimism/optimism/op-service/metrics"
+ "github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/ethclient"
"github.com/ethereum/go-ethereum/log"
)
@@ -123,14 +124,14 @@ func NewMonitor(ctx context.Context, log log.Logger, m metrics.Factory, cfg CLIC
startingL1BlockHeight = uint64(cfg.StartingL1BlockHeight)
}
- state, err := NewState(log, startingL1BlockHeight, latestL1Height)
+ state, err := NewState(log, startingL1BlockHeight, latestL1Height, ret.withdrawalValidator.GetLatestL2Height())
if err != nil {
return nil, fmt.Errorf("failed to create state: %w", err)
}
ret.state = *state
// log state and metrics
- ret.state.LogState(ret.log)
+ ret.state.LogState()
ret.metrics.UpdateMetricsFromState(&ret.state)
return ret, nil
@@ -228,6 +229,10 @@ func (m *Monitor) GetMaxBlock() (uint64, error) {
// Run executes the main monitoring loop.
// It retrieves new events, processes them, and updates the state accordingly.
func (m *Monitor) Run(ctx context.Context) {
+ // Defer the update function
+ defer m.metrics.UpdateMetricsFromState(&m.state)
+ defer m.state.LogState()
+
start := m.state.nextL1Height
stop, err := m.GetMaxBlock()
@@ -238,18 +243,16 @@ func (m *Monitor) Run(ctx context.Context) {
}
// review previous invalidProposalWithdrawalsEvents
- invalidProposalWithdrawalsEvents, err := m.ConsumeEvents(m.state.invalidProposalWithdrawalsEvents)
+ err = m.ConsumeEvents(m.state.potentialAttackOnInProgressGames)
if err != nil {
+ m.state.nodeConnectionFailures++
m.log.Error("failed to consume events", "error", err)
return
}
- // update state
- m.state.invalidProposalWithdrawalsEvents = *invalidProposalWithdrawalsEvents
-
// get new events
-
- newEvents, err := m.withdrawalValidator.GetEnrichedWithdrawalsEvents(start, &stop)
+ m.log.Info("getting enriched withdrawal events", "start", fmt.Sprintf("%d", start), "stop", fmt.Sprintf("%d", stop))
+ newEvents, err := m.withdrawalValidator.GetEnrichedWithdrawalsEventsMap(start, &stop)
if err != nil {
if start >= stop {
m.log.Info("no new events to process", "start", start, "stop", stop)
@@ -262,99 +265,76 @@ func (m *Monitor) Run(ctx context.Context) {
}
return
}
- newInvalidProposalWithdrawalsEvents, err := m.ConsumeEvents(newEvents)
+
+ err = m.ConsumeEvents(newEvents)
if err != nil {
+ m.state.nodeConnectionFailures++
m.log.Error("failed to consume events", "error", err)
return
}
- // update state
- if len(*newInvalidProposalWithdrawalsEvents) > 0 && newInvalidProposalWithdrawalsEvents != nil {
- m.state.invalidProposalWithdrawalsEvents = append(m.state.invalidProposalWithdrawalsEvents, *newInvalidProposalWithdrawalsEvents...)
- }
-
// update state
m.state.nextL1Height = stop
- // log state and metrics
- m.state.LogState(m.log)
- m.metrics.UpdateMetricsFromState(&m.state)
}
// ConsumeEvents processes a slice of enriched withdrawal events and updates their states.
// It returns any events detected during the consumption that requires to be re-analysed again at a later stage (when the event referenced DisputeGame completes).
-func (m *Monitor) ConsumeEvents(enrichedWithdrawalEvent []validator.EnrichedProvenWithdrawalEvent) (*[]validator.EnrichedProvenWithdrawalEvent, error) {
- var newForgeriesGameInProgressEvent []validator.EnrichedProvenWithdrawalEvent = make([]validator.EnrichedProvenWithdrawalEvent, 0)
- for _, enrichedWithdrawalEvent := range enrichedWithdrawalEvent {
+func (m *Monitor) ConsumeEvents(enrichedWithdrawalEvents map[common.Hash]validator.EnrichedProvenWithdrawalEvent) error {
+ for _, enrichedWithdrawalEvent := range enrichedWithdrawalEvents {
+ m.log.Info("processing withdrawal event", "event", &enrichedWithdrawalEvent)
err := m.withdrawalValidator.UpdateEnrichedWithdrawalEvent(&enrichedWithdrawalEvent)
+ //upgrade state to the latest L2 height after the event is processed
+ m.state.latestL2Height = m.withdrawalValidator.GetLatestL2Height()
if err != nil {
- m.state.nodeConnectionFailures++
m.log.Error("failed to update enriched withdrawal event", "error", err)
- return nil, err
+ return err
}
- consumedEvent, err := m.ConsumeEvent(enrichedWithdrawalEvent)
+ err = m.ConsumeEvent(enrichedWithdrawalEvent)
if err != nil {
m.log.Error("failed to consume event", "error", err)
- return nil, err
- } else if !consumedEvent {
- newForgeriesGameInProgressEvent = append(newForgeriesGameInProgressEvent, enrichedWithdrawalEvent)
+ return err
}
}
- return &newForgeriesGameInProgressEvent, nil
+ return nil
}
// ConsumeEvent processes a single enriched withdrawal event.
// It logs the event details and checks for any forgery detection.
-func (m *Monitor) ConsumeEvent(enrichedWithdrawalEvent validator.EnrichedProvenWithdrawalEvent) (bool, error) {
- m.log.Info("processing withdrawal event", "event", enrichedWithdrawalEvent.Event)
+func (m *Monitor) ConsumeEvent(enrichedWithdrawalEvent validator.EnrichedProvenWithdrawalEvent) error {
if enrichedWithdrawalEvent.DisputeGame.DisputeGameData.L2ChainID.Cmp(m.l2ChainID) != 0 {
m.log.Error("l2ChainID mismatch", "expected", fmt.Sprintf("%d", m.l2ChainID), "got", fmt.Sprintf("%d", enrichedWithdrawalEvent.DisputeGame.DisputeGameData.L2ChainID))
}
valid, err := m.withdrawalValidator.IsWithdrawalEventValid(&enrichedWithdrawalEvent)
if err != nil {
- m.state.nodeConnectionFailures++
m.log.Error("failed to check if forgery detected", "error", err)
- return false, err
+ return err
}
- eventConsumed := false
if !valid {
- m.state.numberOfInvalidWithdrawals++
if !enrichedWithdrawalEvent.Blacklisted {
if enrichedWithdrawalEvent.DisputeGame.DisputeGameData.Status == validator.CHALLENGER_WINS {
- m.log.Warn("WITHDRAWAL: is NOT valid, but the game is correctly resolved", "enrichedWithdrawalEvent", enrichedWithdrawalEvent)
- m.state.withdrawalsValidated++
- eventConsumed = true
+ m.state.IncrementSuspiciousEventsOnChallengerWinsGames(enrichedWithdrawalEvent)
} else if enrichedWithdrawalEvent.DisputeGame.DisputeGameData.Status == validator.DEFENDER_WINS {
- m.log.Error("WITHDRAWAL: is NOT valid, forgery detected", "enrichedWithdrawalEvent", enrichedWithdrawalEvent)
- m.state.numberOfDetectedForgery++
- // add to forgeries
- m.state.forgeriesWithdrawalsEvents = append(m.state.forgeriesWithdrawalsEvents, enrichedWithdrawalEvent)
- eventConsumed = true
+ m.state.IncrementPotentialAttackOnDefenderWinsGames(enrichedWithdrawalEvent)
} else if enrichedWithdrawalEvent.DisputeGame.DisputeGameData.Status == validator.IN_PROGRESS {
- m.log.Warn("WITHDRAWAL: is NOT valid, game is still in progress.", "enrichedWithdrawalEvent", enrichedWithdrawalEvent)
+ m.state.IncrementPotentialAttackOnInProgressGames(enrichedWithdrawalEvent)
// add to events to be re-processed
- eventConsumed = false
} else {
m.log.Error("WITHDRAWAL: is NOT valid, game status is unknown. UNKNOWN STATE SHOULD NEVER HAPPEN", "enrichedWithdrawalEvent", enrichedWithdrawalEvent)
- eventConsumed = false
}
} else {
- m.log.Warn("WITHDRAWAL: is NOT valid, but game is blacklisted", "enrichedWithdrawalEvent", enrichedWithdrawalEvent)
- m.state.withdrawalsValidated++
- eventConsumed = true
+ m.state.IncrementSuspiciousEventsOnChallengerWinsGames(enrichedWithdrawalEvent)
}
} else {
- m.log.Info("WITHDRAWAL: is valid", "enrichedWithdrawalEvent", enrichedWithdrawalEvent)
- m.state.withdrawalsValidated++
- eventConsumed = true
+ m.state.IncrementWithdrawalsValidated(enrichedWithdrawalEvent)
}
- m.state.processedProvenWithdrawalsExtension1Events++
+ m.state.eventsProcessed++
m.metrics.UpdateMetricsFromState(&m.state)
- return eventConsumed, nil
+ return nil
}
// Close gracefully shuts down the Monitor by closing the Geth clients.
diff --git a/op-monitorism/faultproof_withdrawals/monitor_live_mainnet_test.go b/op-monitorism/faultproof_withdrawals/monitor_live_mainnet_test.go
new file mode 100644
index 00000000..ad0dbf36
--- /dev/null
+++ b/op-monitorism/faultproof_withdrawals/monitor_live_mainnet_test.go
@@ -0,0 +1,173 @@
+//go:build live
+// +build live
+
+package faultproof_withdrawals
+
+import (
+ "context"
+ "io"
+ "math/big"
+ "testing"
+
+ oplog "github.com/ethereum-optimism/optimism/op-service/log"
+ opmetrics "github.com/ethereum-optimism/optimism/op-service/metrics"
+ "github.com/ethereum/go-ethereum/common"
+ "github.com/joho/godotenv"
+ "github.com/stretchr/testify/require"
+)
+
+// NewTestMonitorMainnet initializes and returns a new Monitor instance for testing.
+// It sets up the necessary environment variables and configurations required for the monitor.
+func NewTestMonitorMainnet() *Monitor {
+ envmap, err := godotenv.Read(".env.op.mainnet")
+ if err != nil {
+ panic("error")
+ }
+
+ ctx := context.Background()
+ L1GethURL := envmap["FAULTPROOF_WITHDRAWAL_MON_L1_GETH_URL"]
+ L2OpNodeURL := envmap["FAULTPROOF_WITHDRAWAL_MON_L2_OP_NODE_URL"]
+ L2OpGethURL := envmap["FAULTPROOF_WITHDRAWAL_MON_L2_OP_GETH_URL"]
+
+ FAULTPROOF_WITHDRAWAL_MON_OPTIMISM_PORTAL := "0xbEb5Fc579115071764c7423A4f12eDde41f106Ed"
+ FAULTPROOF_WITHDRAWAL_MON_EVENT_BLOCK_RANGE := uint64(1000)
+ FAULTPROOF_WITHDRAWAL_MON_START_BLOCK_HEIGHT := int64(6789100)
+
+ cfg := CLIConfig{
+ L1GethURL: L1GethURL,
+ L2OpGethURL: L2OpGethURL,
+ L2OpNodeURL: L2OpNodeURL,
+ EventBlockRange: FAULTPROOF_WITHDRAWAL_MON_EVENT_BLOCK_RANGE,
+ StartingL1BlockHeight: FAULTPROOF_WITHDRAWAL_MON_START_BLOCK_HEIGHT,
+ OptimismPortalAddress: common.HexToAddress(FAULTPROOF_WITHDRAWAL_MON_OPTIMISM_PORTAL),
+ }
+
+ clicfg := oplog.DefaultCLIConfig()
+ output_writer := io.Discard // discard log output during tests to avoid pollution of the standard output
+ log := oplog.NewLogger(output_writer, clicfg)
+
+ metricsRegistry := opmetrics.NewRegistry()
+ monitor, err := NewMonitor(ctx, log, opmetrics.With(metricsRegistry), cfg)
+ if err != nil {
+ panic(err)
+ }
+ return monitor
+}
+
+// TestSingleRunMainnet tests a single execution of the monitor's Run method.
+// It verifies that the state updates correctly after running.
+func TestSingleRunMainnet(t *testing.T) {
+ test_monitor := NewTestMonitorMainnet()
+
+ initialBlock := test_monitor.state.nextL1Height
+ blockIncrement := test_monitor.maxBlockRange
+ finalBlock := initialBlock + blockIncrement
+
+ test_monitor.Run(test_monitor.ctx)
+
+ require.Equal(t, finalBlock, test_monitor.state.nextL1Height)
+ require.Equal(t, uint64(0), test_monitor.state.withdrawalsProcessed)
+ require.Equal(t, uint64(0), test_monitor.state.eventsProcessed)
+ require.Equal(t, uint64(0), test_monitor.state.numberOfPotentialAttackOnInProgressGames)
+ require.Equal(t, uint64(0), test_monitor.state.numberOfPotentialAttacksOnDefenderWinsGames)
+ require.Equal(t, uint64(0), test_monitor.state.numberOfSuspiciousEventsOnChallengerWinsGames)
+
+ require.Equal(t, test_monitor.state.numberOfPotentialAttackOnInProgressGames, uint64(len(test_monitor.state.potentialAttackOnInProgressGames)))
+ require.Equal(t, test_monitor.state.numberOfPotentialAttacksOnDefenderWinsGames, uint64(len(test_monitor.state.potentialAttackOnDefenderWinsGames)))
+ require.Equal(t, test_monitor.state.numberOfSuspiciousEventsOnChallengerWinsGames, uint64(test_monitor.state.suspiciousEventsOnChallengerWinsGames.Len()))
+
+}
+
+// TestRun5Cycle1000BlocksMainnet tests multiple executions of the monitor's Run method over several cycles.
+// It verifies that the state updates correctly after each cycle.
+func TestRun5Cycle1000BlocksMainnet(t *testing.T) {
+ test_monitor := NewTestMonitorMainnet()
+
+ maxCycle := uint64(5)
+ initialBlock := test_monitor.state.nextL1Height
+ blockIncrement := test_monitor.maxBlockRange
+
+ for cycle := uint64(1); cycle <= maxCycle; cycle++ {
+ test_monitor.Run(test_monitor.ctx)
+ }
+
+ initialL1HeightGaugeValue, _ := GetGaugeValue(test_monitor.metrics.InitialL1HeightGauge)
+ nextL1HeightGaugeValue, _ := GetGaugeValue(test_monitor.metrics.NextL1HeightGauge)
+
+ withdrawalsProcessedCounterValue, _ := GetCounterValue(test_monitor.metrics.WithdrawalsProcessedCounter)
+ eventsProcessedCounterValue, _ := GetCounterValue(test_monitor.metrics.EventsProcessedCounter)
+
+ nodeConnectionFailuresCounterValue, _ := GetCounterValue(test_monitor.metrics.NodeConnectionFailuresCounter)
+
+ expected_end_block := blockIncrement*maxCycle + initialBlock
+ require.Equal(t, uint64(initialBlock), uint64(initialL1HeightGaugeValue))
+ require.Equal(t, uint64(expected_end_block), uint64(nextL1HeightGaugeValue))
+
+ require.Equal(t, uint64(0), uint64(eventsProcessedCounterValue))
+ require.Equal(t, uint64(0), uint64(withdrawalsProcessedCounterValue))
+ require.Equal(t, uint64(0), uint64(nodeConnectionFailuresCounterValue))
+
+ require.Equal(t, uint64(0), test_monitor.metrics.previousEventsProcessed)
+ require.Equal(t, uint64(0), test_monitor.metrics.previousWithdrawalsProcessed)
+
+}
+
+func TestRunSingleBlocksMainnet(t *testing.T) {
+ test_monitor := NewTestMonitorMainnet()
+
+ maxCycle := 1
+ initialBlock := test_monitor.state.nextL1Height
+ blockIncrement := test_monitor.maxBlockRange
+ finalBlock := initialBlock + blockIncrement
+
+ for cycle := 1; cycle <= maxCycle; cycle++ {
+ test_monitor.Run(test_monitor.ctx)
+ }
+
+ require.Equal(t, test_monitor.state.nextL1Height, finalBlock)
+ require.Equal(t, uint64(0), test_monitor.state.withdrawalsProcessed)
+ require.Equal(t, uint64(0), test_monitor.state.eventsProcessed)
+ require.Equal(t, 0, len(test_monitor.state.potentialAttackOnDefenderWinsGames))
+ require.Equal(t, 0, len(test_monitor.state.potentialAttackOnInProgressGames))
+ require.Equal(t, 0, test_monitor.state.suspiciousEventsOnChallengerWinsGames.Len())
+}
+
+func TestInvalidWithdrawalsOnMainnet(t *testing.T) {
+ test_monitor := NewTestMonitorMainnet()
+
+ // On mainnet for OP OptimismPortal, the block number 20873192 is known to have only 1 event
+ start := uint64(20873192)
+ stop := uint64(20873193)
+ newEvents, err := test_monitor.withdrawalValidator.GetEnrichedWithdrawalsEvents(start, &stop)
+ require.NoError(t, err)
+ require.Equal(t, len(newEvents), 1)
+
+ event := newEvents[0]
+ require.NotNil(t, event)
+
+ // Expected event:
+ //{WithdrawalHash: 0x45fd4bbcf3386b1fdf75929345b9243c05cd7431a707e84c293b710d40220ebd, ProofSubmitter: 0x394400571C825Da37ca4D6780417DFB514141b1f}
+ require.Equal(t, event.Event.WithdrawalHash, [32]byte(common.HexToHash("0x45fd4bbcf3386b1fdf75929345b9243c05cd7431a707e84c293b710d40220ebd")))
+ require.Equal(t, event.Event.ProofSubmitter, common.HexToAddress("0x394400571C825Da37ca4D6780417DFB514141b1f"))
+
+ //Expected DisputeGameData:
+ // Game address: 0x52cE243d552369b11D6445Cd187F6393d3B42D4a
+ require.Equal(t, event.DisputeGame.DisputeGameData.ProxyAddress, common.HexToAddress("0x52cE243d552369b11D6445Cd187F6393d3B42D4a"))
+
+ // Expected Game root claim
+ // 0xbc1c5ba13b936c6c23b7c51d425f25a8c9444771e851b6790f817a6002a14a33
+ require.Equal(t, event.DisputeGame.DisputeGameData.RootClaim, [32]byte(common.HexToHash("0xbc1c5ba13b936c6c23b7c51d425f25a8c9444771e851b6790f817a6002a14a33")))
+
+ // Expected L2 block number 1276288764
+ require.Equal(t, event.DisputeGame.DisputeGameData.L2blockNumber, big.NewInt(1276288764))
+
+ isValid, err := test_monitor.withdrawalValidator.IsWithdrawalEventValid(&event)
+ require.EqualError(t, err, "game not enriched")
+ require.False(t, isValid)
+ err = test_monitor.withdrawalValidator.UpdateEnrichedWithdrawalEvent(&event)
+ require.NoError(t, err)
+ isValid, err = test_monitor.withdrawalValidator.IsWithdrawalEventValid(&event)
+ require.NoError(t, err)
+ require.False(t, isValid)
+
+}
diff --git a/op-monitorism/faultproof_withdrawals/monitor_test.go b/op-monitorism/faultproof_withdrawals/monitor_live_sepolia_test.go
similarity index 60%
rename from op-monitorism/faultproof_withdrawals/monitor_test.go
rename to op-monitorism/faultproof_withdrawals/monitor_live_sepolia_test.go
index 9c5f04ad..d9d712f7 100644
--- a/op-monitorism/faultproof_withdrawals/monitor_test.go
+++ b/op-monitorism/faultproof_withdrawals/monitor_live_sepolia_test.go
@@ -7,8 +7,6 @@ import (
"context"
"io"
"math/big"
- "os"
- "strconv"
"testing"
"github.com/ethereum-optimism/monitorism/op-monitorism/faultproof_withdrawals/validator"
@@ -19,43 +17,30 @@ import (
"github.com/stretchr/testify/require"
)
-// TestMain runs the tests in the package and exits with the appropriate exit code.
-func TestMain(m *testing.M) {
- exitVal := m.Run()
- os.Exit(exitVal)
-}
-
-// loadEnv loads environment variables from the specified .env file.
-func loadEnv(env string) error {
- return godotenv.Load(env)
-}
-
-// NewTestMonitor initializes and returns a new Monitor instance for testing.
+// NewTestMonitorSepolia initializes and returns a new Monitor instance for testing.
// It sets up the necessary environment variables and configurations required for the monitor.
-func NewTestMonitor() *Monitor {
- loadEnv(".env.op.sepolia")
- ctx := context.Background()
- L1GethURL := os.Getenv("FAULTPROOF_WITHDRAWAL_MON_L1_GETH_URL")
- L2OpNodeURL := os.Getenv("FAULTPROOF_WITHDRAWAL_MON_L2_OP_NODE_URL")
- L2OpGethURL := os.Getenv("FAULTPROOF_WITHDRAWAL_MON_L2_OP_GETH_URL")
- EventBlockRangeStr := os.Getenv("FAULTPROOF_WITHDRAWAL_MON_EVENT_BLOCK_RANGE")
- EventBlockRange, err := strconv.ParseUint(EventBlockRangeStr, 10, 64)
+func NewTestMonitorSepolia() *Monitor {
+ envmap, err := godotenv.Read(".env.op.sepolia")
if err != nil {
- panic(err)
- }
- StartingL1BlockHeightStr := os.Getenv("FAULTPROOF_WITHDRAWAL_MON_START_BLOCK_HEIGHT")
- StartingL1BlockHeight, err := strconv.ParseInt(StartingL1BlockHeightStr, 10, 64)
- if err != nil {
- panic(err)
+ panic("error")
}
+ ctx := context.Background()
+ L1GethURL := envmap["FAULTPROOF_WITHDRAWAL_MON_L1_GETH_URL"]
+ L2OpNodeURL := envmap["FAULTPROOF_WITHDRAWAL_MON_L2_OP_NODE_URL"]
+ L2OpGethURL := envmap["FAULTPROOF_WITHDRAWAL_MON_L2_OP_GETH_URL"]
+
+ FAULTPROOF_WITHDRAWAL_MON_OPTIMISM_PORTAL := "0x16Fc5058F25648194471939df75CF27A2fdC48BC"
+ FAULTPROOF_WITHDRAWAL_MON_EVENT_BLOCK_RANGE := uint64(1000)
+ FAULTPROOF_WITHDRAWAL_MON_START_BLOCK_HEIGHT := int64(6789100)
+
cfg := CLIConfig{
L1GethURL: L1GethURL,
L2OpGethURL: L2OpGethURL,
L2OpNodeURL: L2OpNodeURL,
- EventBlockRange: EventBlockRange,
- StartingL1BlockHeight: StartingL1BlockHeight,
- OptimismPortalAddress: common.HexToAddress(os.Getenv("FAULTPROOF_WITHDRAWAL_MON_OPTIMISM_PORTAL")),
+ EventBlockRange: FAULTPROOF_WITHDRAWAL_MON_EVENT_BLOCK_RANGE,
+ StartingL1BlockHeight: FAULTPROOF_WITHDRAWAL_MON_START_BLOCK_HEIGHT,
+ OptimismPortalAddress: common.HexToAddress(FAULTPROOF_WITHDRAWAL_MON_OPTIMISM_PORTAL),
}
clicfg := oplog.DefaultCLIConfig()
@@ -70,49 +55,46 @@ func NewTestMonitor() *Monitor {
return monitor
}
-// TestSingleRun tests a single execution of the monitor's Run method.
+// TestSingleRunSepolia tests a single execution of the monitor's Run method.
// It verifies that the state updates correctly after running.
-func TestSingleRun(t *testing.T) {
- test_monitor := NewTestMonitor()
+func TestSingleRunSepolia(t *testing.T) {
+ test_monitor := NewTestMonitorSepolia()
- initialBlock := uint64(5914813)
- blockIncrement := uint64(1000)
+ initialBlock := test_monitor.state.nextL1Height
+ blockIncrement := test_monitor.maxBlockRange
finalBlock := initialBlock + blockIncrement
- test_monitor.state.nextL1Height = initialBlock
- test_monitor.maxBlockRange = blockIncrement
- test_monitor.Run(test_monitor.ctx)
+ test_monitor.Run(context.Background())
require.Equal(t, test_monitor.state.nextL1Height, finalBlock)
- require.Equal(t, test_monitor.state.withdrawalsValidated, uint64(1))
- require.Equal(t, test_monitor.state.processedProvenWithdrawalsExtension1Events, uint64(1))
- require.Equal(t, test_monitor.state.numberOfDetectedForgery, uint64(0))
- require.Equal(t, len(test_monitor.state.forgeriesWithdrawalsEvents), 0)
- require.Equal(t, len(test_monitor.state.invalidProposalWithdrawalsEvents), 0)
+ require.Equal(t, uint64(1), test_monitor.state.withdrawalsProcessed)
+ require.Equal(t, uint64(1), test_monitor.state.eventsProcessed)
+ require.Equal(t, 0, len(test_monitor.state.potentialAttackOnDefenderWinsGames))
+ require.Equal(t, 0, len(test_monitor.state.potentialAttackOnInProgressGames))
+ require.Equal(t, 0, test_monitor.state.suspiciousEventsOnChallengerWinsGames.Len())
}
-// TestConsumeEvents tests the consumption of enriched withdrawal events.
+// TestConsumeEventsSepolia tests the consumption of enriched withdrawal events.
// It verifies that new events can be processed correctly.
-func TestConsumeEvents(t *testing.T) {
- test_monitor := NewTestMonitor()
+func TestConsumeEventsSepolia(t *testing.T) {
+ test_monitor := NewTestMonitorSepolia()
- initialBlock := uint64(5914813)
- blockIncrement := uint64(1000)
+ initialBlock := test_monitor.state.nextL1Height
+ blockIncrement := test_monitor.maxBlockRange
finalBlock := initialBlock + blockIncrement
- newEvents, err := test_monitor.withdrawalValidator.GetEnrichedWithdrawalsEvents(initialBlock, &finalBlock)
+ newEvents, err := test_monitor.withdrawalValidator.GetEnrichedWithdrawalsEventsMap(initialBlock, &finalBlock)
require.NoError(t, err)
- require.NotEqual(t, len(newEvents), 0)
+ require.NotEqual(t, 0, len(newEvents))
- newInvalidProposalWithdrawalsEvents, err := test_monitor.ConsumeEvents(newEvents)
+ err = test_monitor.ConsumeEvents(newEvents)
require.NoError(t, err)
- require.Equal(t, len(*newInvalidProposalWithdrawalsEvents), 0)
}
-// TestConsumeEventValid_DEFENDER_WINS tests the consumption of a valid event where the defender wins.
+// TestConsumeEventValid_DEFENDER_WINS_Sepolia tests the consumption of a valid event where the defender wins.
// It checks that the state updates correctly after processing the event.
-func TestConsumeEventValid_DEFENDER_WINS(t *testing.T) {
- test_monitor := NewTestMonitor()
+func TestConsumeEventValid_DEFENDER_WINS_Sepolia(t *testing.T) {
+ test_monitor := NewTestMonitorSepolia()
expectedRootClaim := common.HexToHash("0x763d50048ccdb85fded935ff88c9e6b2284fd981da8ed7ae892f36b8761f7597")
@@ -146,20 +128,23 @@ func TestConsumeEventValid_DEFENDER_WINS(t *testing.T) {
},
}
- consumedEvent, err := test_monitor.ConsumeEvent(validEvent)
+ eventsMap := map[common.Hash]validator.EnrichedProvenWithdrawalEvent{
+ validEvent.Event.WithdrawalHash: validEvent,
+ }
+ err := test_monitor.ConsumeEvents(eventsMap)
require.NoError(t, err)
- require.True(t, consumedEvent)
- require.Equal(t, test_monitor.state.withdrawalsValidated, uint64(1))
- require.Equal(t, test_monitor.state.processedProvenWithdrawalsExtension1Events, uint64(1))
- require.Equal(t, test_monitor.state.numberOfDetectedForgery, uint64(0))
- require.Equal(t, len(test_monitor.state.forgeriesWithdrawalsEvents), 0)
- require.Equal(t, len(test_monitor.state.invalidProposalWithdrawalsEvents), 0)
+ require.Equal(t, uint64(1), test_monitor.state.withdrawalsProcessed)
+ require.Equal(t, uint64(1), test_monitor.state.eventsProcessed)
+ require.Equal(t, 0, len(test_monitor.state.potentialAttackOnDefenderWinsGames))
+ require.Equal(t, 0, len(test_monitor.state.potentialAttackOnInProgressGames))
+ require.Equal(t, 0, test_monitor.state.suspiciousEventsOnChallengerWinsGames.Len())
+
}
-// TestConsumeEventValid_CHALLENGER_WINS tests the consumption of a valid event where the challenger wins.
+// TestConsumeEventValid_CHALLENGER_WINS_Sepolia tests the consumption of a valid event where the challenger wins.
// It checks that the state updates correctly after processing the event.
-func TestConsumeEventValid_CHALLENGER_WINS(t *testing.T) {
- test_monitor := NewTestMonitor()
+func TestConsumeEventValid_CHALLENGER_WINS_Sepolia(t *testing.T) {
+ test_monitor := NewTestMonitorSepolia()
expectedRootClaim := common.HexToHash("0x763d50048ccdb85fded935ff88c9e6b2284fd981da8ed7ae892f36b8761f7597")
rootClaim := common.HexToHash("0x763d50048ccdb85fded935ff88c9e6b2284fd981da8ed7ae892f36b8761f7596") // different root claim, last number is 6 instead of 7
@@ -194,20 +179,23 @@ func TestConsumeEventValid_CHALLENGER_WINS(t *testing.T) {
},
}
- consumedEvent, err := test_monitor.ConsumeEvent(event)
+ eventsMap := map[common.Hash]validator.EnrichedProvenWithdrawalEvent{
+ event.Event.WithdrawalHash: event,
+ }
+ err := test_monitor.ConsumeEvents(eventsMap)
require.NoError(t, err)
- require.True(t, consumedEvent)
- require.Equal(t, test_monitor.state.withdrawalsValidated, uint64(1))
- require.Equal(t, test_monitor.state.processedProvenWithdrawalsExtension1Events, uint64(1))
- require.Equal(t, test_monitor.state.numberOfDetectedForgery, uint64(0))
- require.Equal(t, len(test_monitor.state.forgeriesWithdrawalsEvents), 0)
- require.Equal(t, len(test_monitor.state.invalidProposalWithdrawalsEvents), 0)
+ require.Equal(t, uint64(1), test_monitor.state.withdrawalsProcessed)
+ require.Equal(t, uint64(1), test_monitor.state.eventsProcessed)
+ require.Equal(t, 0, len(test_monitor.state.potentialAttackOnDefenderWinsGames))
+ require.Equal(t, 0, len(test_monitor.state.potentialAttackOnInProgressGames))
+ require.Equal(t, 1, test_monitor.state.suspiciousEventsOnChallengerWinsGames.Len())
+
}
-// TestConsumeEventValid_Blacklisted tests the consumption of a valid event that is blacklisted.
+// TestConsumeEventValid_BlacklistedSepolia tests the consumption of a valid event that is blacklisted.
// It checks that the state updates correctly after processing the event.
-func TestConsumeEventValid_Blacklisted(t *testing.T) {
- test_monitor := NewTestMonitor()
+func TestConsumeEventValid_BlacklistedSepolia(t *testing.T) {
+ test_monitor := NewTestMonitorSepolia()
expectedRootClaim := common.HexToHash("0x763d50048ccdb85fded935ff88c9e6b2284fd981da8ed7ae892f36b8761f7597")
rootClaim := common.HexToHash("0x763d50048ccdb85fded935ff88c9e6b2284fd981da8ed7ae892f36b8761f7596") // different root claim, last number is 6 instead of 7
@@ -242,20 +230,23 @@ func TestConsumeEventValid_Blacklisted(t *testing.T) {
},
}
- consumedEvent, err := test_monitor.ConsumeEvent(event)
+ eventsMap := map[common.Hash]validator.EnrichedProvenWithdrawalEvent{
+ event.Event.WithdrawalHash: event,
+ }
+ err := test_monitor.ConsumeEvents(eventsMap)
require.NoError(t, err)
- require.True(t, consumedEvent)
- require.Equal(t, test_monitor.state.withdrawalsValidated, uint64(1))
- require.Equal(t, test_monitor.state.processedProvenWithdrawalsExtension1Events, uint64(1))
- require.Equal(t, test_monitor.state.numberOfDetectedForgery, uint64(0))
- require.Equal(t, len(test_monitor.state.forgeriesWithdrawalsEvents), 0)
- require.Equal(t, len(test_monitor.state.invalidProposalWithdrawalsEvents), 0)
+ require.Equal(t, uint64(1), test_monitor.state.withdrawalsProcessed)
+ require.Equal(t, uint64(1), test_monitor.state.eventsProcessed)
+ require.Equal(t, 1, len(test_monitor.state.potentialAttackOnDefenderWinsGames))
+ require.Equal(t, 0, len(test_monitor.state.potentialAttackOnInProgressGames))
+ require.Equal(t, 0, test_monitor.state.suspiciousEventsOnChallengerWinsGames.Len())
+
}
-// TestConsumeEventForgery1 tests the consumption of an event that indicates a forgery.
+// TestConsumeEventForgery1Sepolia tests the consumption of an event that indicates a forgery.
// It checks that the state updates correctly after processing the event.
-func TestConsumeEventForgery1(t *testing.T) {
- test_monitor := NewTestMonitor()
+func TestConsumeEventForgery1Sepolia(t *testing.T) {
+ test_monitor := NewTestMonitorSepolia()
expectedRootClaim := common.HexToHash("0x763d50048ccdb85fded935ff88c9e6b2284fd981da8ed7ae892f36b8761f7597")
@@ -289,20 +280,22 @@ func TestConsumeEventForgery1(t *testing.T) {
},
}
- consumedEvent, err := test_monitor.ConsumeEvent(validEvent)
+ eventsMap := map[common.Hash]validator.EnrichedProvenWithdrawalEvent{
+ validEvent.Event.WithdrawalHash: validEvent,
+ }
+ err := test_monitor.ConsumeEvents(eventsMap)
require.NoError(t, err)
- require.True(t, consumedEvent)
- require.Equal(t, test_monitor.state.withdrawalsValidated, uint64(0))
- require.Equal(t, test_monitor.state.processedProvenWithdrawalsExtension1Events, uint64(1))
- require.Equal(t, test_monitor.state.numberOfDetectedForgery, uint64(1))
- require.Equal(t, len(test_monitor.state.forgeriesWithdrawalsEvents), 1)
- require.Equal(t, len(test_monitor.state.invalidProposalWithdrawalsEvents), 0)
+ require.Equal(t, uint64(1), test_monitor.state.withdrawalsProcessed)
+ require.Equal(t, uint64(1), test_monitor.state.eventsProcessed)
+ require.Equal(t, 0, len(test_monitor.state.potentialAttackOnDefenderWinsGames))
+ require.Equal(t, 0, len(test_monitor.state.potentialAttackOnInProgressGames))
+ require.Equal(t, 0, test_monitor.state.suspiciousEventsOnChallengerWinsGames.Len())
}
-// TestConsumeEventForgery2 tests the consumption of another event that indicates a forgery.
+// TestConsumeEventForgery2Sepolia tests the consumption of another event that indicates a forgery.
// It checks that the state updates correctly after processing the event.
-func TestConsumeEventForgery2(t *testing.T) {
- test_monitor := NewTestMonitor()
+func TestConsumeEventForgery2Sepolia(t *testing.T) {
+ test_monitor := NewTestMonitorSepolia()
expectedRootClaim := common.HexToHash("0x763d50048ccdb85fded935ff88c9e6b2284fd981da8ed7ae892f36b8761f7597")
rootClaim := common.HexToHash("0x763d50048ccdb85fded935ff88c9e6b2284fd981da8ed7ae892f36b8761f7596") // different root claim, last number is 6 instead of 7
@@ -337,12 +330,15 @@ func TestConsumeEventForgery2(t *testing.T) {
},
}
- consumedEvent, err := test_monitor.ConsumeEvent(event)
+ eventsMap := map[common.Hash]validator.EnrichedProvenWithdrawalEvent{
+ event.Event.WithdrawalHash: event,
+ }
+ err := test_monitor.ConsumeEvents(eventsMap)
require.NoError(t, err)
- require.True(t, consumedEvent)
- require.Equal(t, test_monitor.state.withdrawalsValidated, uint64(0))
- require.Equal(t, test_monitor.state.processedProvenWithdrawalsExtension1Events, uint64(1))
- require.Equal(t, test_monitor.state.numberOfDetectedForgery, uint64(1))
- require.Equal(t, len(test_monitor.state.forgeriesWithdrawalsEvents), 1)
- require.Equal(t, len(test_monitor.state.invalidProposalWithdrawalsEvents), 0)
+ require.Equal(t, uint64(1), test_monitor.state.withdrawalsProcessed)
+ require.Equal(t, uint64(1), test_monitor.state.eventsProcessed)
+ require.Equal(t, 1, len(test_monitor.state.potentialAttackOnDefenderWinsGames))
+ require.Equal(t, 0, len(test_monitor.state.potentialAttackOnInProgressGames))
+ require.Equal(t, 0, test_monitor.state.suspiciousEventsOnChallengerWinsGames.Len())
+
}
diff --git a/op-monitorism/faultproof_withdrawals/state.go b/op-monitorism/faultproof_withdrawals/state.go
index bd13d90f..2b4bd268 100644
--- a/op-monitorism/faultproof_withdrawals/state.go
+++ b/op-monitorism/faultproof_withdrawals/state.go
@@ -8,66 +8,156 @@ import (
"github.com/ethereum-optimism/optimism/op-service/metrics"
"github.com/ethereum/go-ethereum/common"
"github.com/ethereum/go-ethereum/log"
+ lru "github.com/hashicorp/golang-lru"
"github.com/prometheus/client_golang/prometheus"
+ dto "github.com/prometheus/client_model/go"
+)
+
+const (
+ suspiciousEventsOnChallengerWinsGamesCacheSize = 1000
)
type State struct {
+ logger log.Logger
nextL1Height uint64
latestL1Height uint64
initialL1Height uint64
+ latestL2Height uint64
- processedProvenWithdrawalsExtension1Events uint64
-
- numberOfDetectedForgery uint64
- numberOfInvalidWithdrawals uint64
- withdrawalsValidated uint64
+ eventsProcessed uint64 // This counts the events that we have taken care of, and we are aware of.
+ withdrawalsProcessed uint64 // This counts the withdrawals that have being completed and processed and we are not tracking anymore. eventProcessed >= withdrawalsProcessed. withdrawalsProcessed does not includes potential attacks with games in progress.
nodeConnectionFailures uint64
- forgeriesWithdrawalsEvents []validator.EnrichedProvenWithdrawalEvent
- invalidProposalWithdrawalsEvents []validator.EnrichedProvenWithdrawalEvent
+ // possible attacks detected
+
+ // Forgeries detected on games that are already resolved
+ potentialAttackOnDefenderWinsGames map[common.Hash]validator.EnrichedProvenWithdrawalEvent
+ numberOfPotentialAttacksOnDefenderWinsGames uint64
+
+ // Forgeries detected on games that are still in progress
+ // Faultproof system should make them invalid
+ potentialAttackOnInProgressGames map[common.Hash]validator.EnrichedProvenWithdrawalEvent
+ numberOfPotentialAttackOnInProgressGames uint64
+
+ // Suspicious events
+ // It is unlikely that someone is going to use a withdrawal hash on a games that resolved with ChallengerWins. If this happens, maybe there is a bug somewhere in the UI used by the users or it is a malicious attack that failed
+ suspiciousEventsOnChallengerWinsGames *lru.Cache
+ numberOfSuspiciousEventsOnChallengerWinsGames uint64
}
-func NewState(log log.Logger, nextL1Height uint64, latestL1Height uint64) (*State, error) {
+func NewState(logger log.Logger, nextL1Height uint64, latestL1Height uint64, latestL2Height uint64) (*State, error) {
if nextL1Height > latestL1Height {
- log.Info("nextL1Height is greater than latestL1Height, starting from latest", "nextL1Height", nextL1Height, "latestL1Height", latestL1Height)
+ logger.Info("nextL1Height is greater than latestL1Height, starting from latest", "nextL1Height", nextL1Height, "latestL1Height", latestL1Height)
nextL1Height = latestL1Height
}
ret := State{
- processedProvenWithdrawalsExtension1Events: 0,
- nextL1Height: nextL1Height,
- latestL1Height: latestL1Height,
- numberOfDetectedForgery: 0,
- withdrawalsValidated: 0,
- nodeConnectionFailures: 0,
- numberOfInvalidWithdrawals: 0,
- initialL1Height: nextL1Height,
+ potentialAttackOnDefenderWinsGames: make(map[common.Hash]validator.EnrichedProvenWithdrawalEvent),
+ numberOfPotentialAttacksOnDefenderWinsGames: 0,
+ suspiciousEventsOnChallengerWinsGames: func() *lru.Cache {
+ cache, err := lru.New(suspiciousEventsOnChallengerWinsGamesCacheSize)
+ if err != nil {
+ logger.Error("Failed to create LRU cache", "error", err)
+ return nil
+ }
+ return cache
+ }(),
+ numberOfSuspiciousEventsOnChallengerWinsGames: 0,
+
+ potentialAttackOnInProgressGames: make(map[common.Hash]validator.EnrichedProvenWithdrawalEvent),
+ numberOfPotentialAttackOnInProgressGames: 0,
+
+ eventsProcessed: 0,
+
+ withdrawalsProcessed: 0,
+ nodeConnectionFailures: 0,
+
+ nextL1Height: nextL1Height,
+ latestL1Height: latestL1Height,
+ initialL1Height: nextL1Height,
+ latestL2Height: latestL2Height,
+ logger: logger,
}
return &ret, nil
}
-func (s *State) LogState(log log.Logger) {
+func (s *State) LogState() {
blockToProcess, syncPercentage := s.GetPercentages()
- log.Info("STATE:",
- "withdrawalsValidated", fmt.Sprintf("%d", s.withdrawalsValidated),
+ s.logger.Info("STATE:",
+ "withdrawalsProcessed", fmt.Sprintf("%d", s.withdrawalsProcessed),
+
"initialL1Height", fmt.Sprintf("%d", s.initialL1Height),
"nextL1Height", fmt.Sprintf("%d", s.nextL1Height),
"latestL1Height", fmt.Sprintf("%d", s.latestL1Height),
+ "latestL2Height", fmt.Sprintf("%d", s.latestL2Height),
"blockToProcess", fmt.Sprintf("%d", blockToProcess),
"syncPercentage", fmt.Sprintf("%d%%", syncPercentage),
- "processedProvenWithdrawalsExtension1Events", fmt.Sprintf("%d", s.processedProvenWithdrawalsExtension1Events),
- "numberOfDetectedForgery", fmt.Sprintf("%d", s.numberOfDetectedForgery),
- "numberOfInvalidWithdrawals", fmt.Sprintf("%d", s.numberOfInvalidWithdrawals),
+
+ "eventsProcessed", fmt.Sprintf("%d", s.eventsProcessed),
"nodeConnectionFailures", fmt.Sprintf("%d", s.nodeConnectionFailures),
- "forgeriesWithdrawalsEvents", fmt.Sprintf("%d", len(s.forgeriesWithdrawalsEvents)),
- "invalidProposalWithdrawalsEvents", fmt.Sprintf("%d", len(s.invalidProposalWithdrawalsEvents)),
+
+ "potentialAttackOnDefenderWinsGames", fmt.Sprintf("%d", s.numberOfPotentialAttacksOnDefenderWinsGames),
+ "potentialAttackOnInProgressGames", fmt.Sprintf("%d", s.numberOfPotentialAttackOnInProgressGames),
+ "suspiciousEventsOnChallengerWinsGames", fmt.Sprintf("%d", s.numberOfSuspiciousEventsOnChallengerWinsGames),
)
}
+func (s *State) IncrementWithdrawalsValidated(enrichedWithdrawalEvent validator.EnrichedProvenWithdrawalEvent) {
+ s.logger.Info("STATE WITHDRAWAL: valid", "TxHash", fmt.Sprintf("%v", enrichedWithdrawalEvent.Event.Raw.TxHash), "enrichedWithdrawalEvent", &enrichedWithdrawalEvent)
+ s.withdrawalsProcessed++
+}
+
+func (s *State) IncrementPotentialAttackOnDefenderWinsGames(enrichedWithdrawalEvent validator.EnrichedProvenWithdrawalEvent) {
+ key := enrichedWithdrawalEvent.Event.Raw.TxHash
+
+ s.logger.Error("STATE WITHDRAWAL: is NOT valid, forgery detected", "TxHash", fmt.Sprintf("%v", enrichedWithdrawalEvent.Event.Raw.TxHash), "enrichedWithdrawalEvent", &enrichedWithdrawalEvent)
+ s.potentialAttackOnDefenderWinsGames[key] = enrichedWithdrawalEvent
+ s.numberOfPotentialAttacksOnDefenderWinsGames++
+
+ if _, ok := s.potentialAttackOnInProgressGames[key]; ok {
+ s.logger.Error("STATE WITHDRAWAL: added to potential attacks. Removing from inProgress", "TxHash", fmt.Sprintf("%v", enrichedWithdrawalEvent.Event.Raw.TxHash), "enrichedWithdrawalEvent", &enrichedWithdrawalEvent)
+ delete(s.potentialAttackOnInProgressGames, key)
+ s.numberOfPotentialAttackOnInProgressGames--
+ }
+
+ s.withdrawalsProcessed++
+}
+
+func (s *State) IncrementPotentialAttackOnInProgressGames(enrichedWithdrawalEvent validator.EnrichedProvenWithdrawalEvent) {
+ key := enrichedWithdrawalEvent.Event.Raw.TxHash
+ // check if key already exists
+ if _, ok := s.potentialAttackOnInProgressGames[key]; ok {
+ s.logger.Error("STATE WITHDRAWAL:is NOT valid, game is still in progress", "TxHash", fmt.Sprintf("%v", enrichedWithdrawalEvent.Event.Raw.TxHash), "enrichedWithdrawalEvent", &enrichedWithdrawalEvent)
+ } else {
+ s.logger.Error("STATE WITHDRAWAL:is NOT valid, game is still in progress. New game found In Progress", "TxHash", fmt.Sprintf("%v", enrichedWithdrawalEvent.Event.Raw.TxHash), "enrichedWithdrawalEvent", &enrichedWithdrawalEvent)
+ s.numberOfPotentialAttackOnInProgressGames++
+ }
+
+ // eventually update the map with the new enrichedWithdrawalEvent
+ s.potentialAttackOnInProgressGames[key] = enrichedWithdrawalEvent
+}
+
+func (s *State) IncrementSuspiciousEventsOnChallengerWinsGames(enrichedWithdrawalEvent validator.EnrichedProvenWithdrawalEvent) {
+ key := enrichedWithdrawalEvent.Event.Raw.TxHash
+
+ s.logger.Error("STATE WITHDRAWAL:is NOT valid, but the game is correctly resolved", "TxHash", fmt.Sprintf("%v", enrichedWithdrawalEvent.Event.Raw.TxHash), "enrichedWithdrawalEvent", &enrichedWithdrawalEvent)
+ s.suspiciousEventsOnChallengerWinsGames.Add(key, enrichedWithdrawalEvent)
+ s.numberOfSuspiciousEventsOnChallengerWinsGames++
+
+ if _, ok := s.potentialAttackOnInProgressGames[key]; ok {
+ s.logger.Error("STATE WITHDRAWAL: added to suspicious attacks. Removing from inProgress", "TxHash", fmt.Sprintf("%v", enrichedWithdrawalEvent.Event.Raw.TxHash), "enrichedWithdrawalEvent", &enrichedWithdrawalEvent)
+ delete(s.potentialAttackOnInProgressGames, key)
+ s.numberOfPotentialAttackOnInProgressGames--
+ }
+
+ s.withdrawalsProcessed++
+
+}
+
func (s *State) GetPercentages() (uint64, uint64) {
blockToProcess := s.latestL1Height - s.nextL1Height
divisor := float64(s.latestL1Height) * 100
@@ -80,23 +170,99 @@ func (s *State) GetPercentages() (uint64, uint64) {
}
type Metrics struct {
- InitialL1HeightGauge prometheus.Gauge
- NextL1HeightGauge prometheus.Gauge
- LatestL1HeightGauge prometheus.Gauge
- ProcessedProvenWithdrawalsEventsExtensions1Counter prometheus.Counter
- NumberOfDetectedForgeryGauge prometheus.Gauge
- NumberOfInvalidWithdrawalsGauge prometheus.Gauge
- WithdrawalsValidatedCounter prometheus.Counter
- NodeConnectionFailuresCounter prometheus.Counter
- ForgeriesWithdrawalsEventsGauge prometheus.Gauge
- InvalidProposalWithdrawalsEventsGauge prometheus.Gauge
- ForgeriesWithdrawalsEventsGaugeVec *prometheus.GaugeVec
- InvalidProposalWithdrawalsEventsGaugeVec *prometheus.GaugeVec
+ InitialL1HeightGauge prometheus.Gauge
+ NextL1HeightGauge prometheus.Gauge
+ LatestL1HeightGauge prometheus.Gauge
+ LatestL2HeightGauge prometheus.Gauge
+
+ EventsProcessedCounter prometheus.Counter
+ WithdrawalsProcessedCounter prometheus.Counter
+
+ NodeConnectionFailuresCounter prometheus.Counter
+
+ PotentialAttackOnDefenderWinsGamesGauge prometheus.Gauge
+ PotentialAttackOnInProgressGamesGauge prometheus.Gauge
+ SuspiciousEventsOnChallengerWinsGamesGauge prometheus.Gauge
+
+ PotentialAttackOnDefenderWinsGamesGaugeVec *prometheus.GaugeVec
+ PotentialAttackOnInProgressGamesGaugeVec *prometheus.GaugeVec
+ SuspiciousEventsOnChallengerWinsGamesGaugeVec *prometheus.GaugeVec
// Previous values for counters
- previousProcessedProvenWithdrawalsExtension1Events uint64
- previousWithdrawalsValidated uint64
- previousNodeConnectionFailures uint64
+ previousEventsProcessed uint64
+ previousWithdrawalsProcessed uint64
+ previousNodeConnectionFailures uint64
+}
+
+func (m *Metrics) String() string {
+ initialL1HeightGaugeValue, _ := GetGaugeValue(m.InitialL1HeightGauge)
+ nextL1HeightGaugeValue, _ := GetGaugeValue(m.NextL1HeightGauge)
+ latestL1HeightGaugeValue, _ := GetGaugeValue(m.LatestL1HeightGauge)
+ latestL2HeightGaugeValue, _ := GetGaugeValue(m.LatestL2HeightGauge)
+
+ withdrawalsProcessedCounterValue, _ := GetCounterValue(m.WithdrawalsProcessedCounter)
+ eventsProcessedCounterValue, _ := GetCounterValue(m.EventsProcessedCounter)
+
+ nodeConnectionFailuresCounterValue, _ := GetCounterValue(m.NodeConnectionFailuresCounter)
+
+ potentialAttackOnDefenderWinsGamesGaugeValue, _ := GetGaugeValue(m.PotentialAttackOnDefenderWinsGamesGauge)
+ potentialAttackOnInProgressGamesGaugeValue, _ := GetGaugeValue(m.PotentialAttackOnInProgressGamesGauge)
+
+ forgeriesWithdrawalsEventsGaugeVecValue, _ := GetGaugeVecValue(m.PotentialAttackOnDefenderWinsGamesGaugeVec, prometheus.Labels{})
+ invalidProposalWithdrawalsEventsGaugeVecValue, _ := GetGaugeVecValue(m.PotentialAttackOnInProgressGamesGaugeVec, prometheus.Labels{})
+
+ return fmt.Sprintf(
+ "InitialL1HeightGauge: %d\nNextL1HeightGauge: %d\nLatestL1HeightGauge: %d\n latestL2HeightGaugeValue: %d\n eventsProcessedCounterValue: %d\nwithdrawalsProcessedCounterValue: %d\nnodeConnectionFailuresCounterValue: %d\n potentialAttackOnDefenderWinsGamesGaugeValue: %d\n potentialAttackOnInProgressGamesGaugeValue: %d\n forgeriesWithdrawalsEventsGaugeVecValue: %d\n invalidProposalWithdrawalsEventsGaugeVecValue: %d\n previousEventsProcessed: %d\n previousWithdrawalsProcessed: %d\n previousNodeConnectionFailures: %d\n",
+ uint64(initialL1HeightGaugeValue),
+ uint64(nextL1HeightGaugeValue),
+ uint64(latestL1HeightGaugeValue),
+ uint64(latestL2HeightGaugeValue),
+ uint64(eventsProcessedCounterValue),
+ uint64(withdrawalsProcessedCounterValue),
+ uint64(nodeConnectionFailuresCounterValue),
+ uint64(potentialAttackOnDefenderWinsGamesGaugeValue),
+ uint64(potentialAttackOnInProgressGamesGaugeValue),
+ uint64(forgeriesWithdrawalsEventsGaugeVecValue),
+ uint64(invalidProposalWithdrawalsEventsGaugeVecValue),
+ m.previousEventsProcessed,
+ m.previousWithdrawalsProcessed,
+ m.previousNodeConnectionFailures,
+ )
+}
+
+// Generic function to get the value of any prometheus.Counter
+func GetCounterValue(counter prometheus.Counter) (float64, error) {
+ metric := &dto.Metric{}
+ err := counter.Write(metric)
+ if err != nil {
+ return 0, err
+ }
+ return metric.GetCounter().GetValue(), nil
+}
+
+// Generic function to get the value of any prometheus.Gauge
+func GetGaugeValue(gauge prometheus.Gauge) (float64, error) {
+ metric := &dto.Metric{}
+ err := gauge.Write(metric)
+ if err != nil {
+ return 0, err
+ }
+ return metric.GetGauge().GetValue(), nil
+}
+
+// Function to get the value of a specific Gauge within a GaugeVec
+func GetGaugeVecValue(gaugeVec *prometheus.GaugeVec, labels prometheus.Labels) (float64, error) {
+ gauge, err := gaugeVec.GetMetricWith(labels)
+ if err != nil {
+ return 0, err
+ }
+
+ metric := &dto.Metric{}
+ err = gauge.Write(metric)
+ if err != nil {
+ return 0, err
+ }
+ return metric.GetGauge().GetValue(), nil
}
func NewMetrics(m metrics.Factory) *Metrics {
@@ -116,56 +282,64 @@ func NewMetrics(m metrics.Factory) *Metrics {
Name: "latest_l1_height",
Help: "Latest L1 Height",
}),
- ProcessedProvenWithdrawalsEventsExtensions1Counter: m.NewCounter(prometheus.CounterOpts{
+ LatestL2HeightGauge: m.NewGauge(prometheus.GaugeOpts{
Namespace: MetricsNamespace,
- Name: "processed_provenwithdrawalsextension1_events_total",
- Help: "Total number of processed provenwithdrawalsextension1 events",
+ Name: "latest_l2_height",
+ Help: "Latest L2 Height",
}),
- NumberOfDetectedForgeryGauge: m.NewGauge(prometheus.GaugeOpts{
+ EventsProcessedCounter: m.NewCounter(prometheus.CounterOpts{
Namespace: MetricsNamespace,
- Name: "number_of_detected_forgeries",
- Help: "Number of detected forgeries",
+ Name: "events_processed_total",
+ Help: "Total number of events processed",
}),
- NumberOfInvalidWithdrawalsGauge: m.NewGauge(prometheus.GaugeOpts{
+ WithdrawalsProcessedCounter: m.NewCounter(prometheus.CounterOpts{
Namespace: MetricsNamespace,
- Name: "number_of_invalid_withdrawals",
- Help: "Number of invalid withdrawals",
- }),
- WithdrawalsValidatedCounter: m.NewCounter(prometheus.CounterOpts{
- Namespace: MetricsNamespace,
- Name: "withdrawals_validated_total",
- Help: "Total number of withdrawals validated",
+ Name: "withdrawals_processed_total",
+ Help: "Total number of withdrawals processed",
}),
NodeConnectionFailuresCounter: m.NewCounter(prometheus.CounterOpts{
Namespace: MetricsNamespace,
Name: "node_connection_failures_total",
Help: "Total number of node connection failures",
}),
- ForgeriesWithdrawalsEventsGauge: m.NewGauge(prometheus.GaugeOpts{
+ PotentialAttackOnDefenderWinsGamesGauge: m.NewGauge(prometheus.GaugeOpts{
Namespace: MetricsNamespace,
- Name: "forgeries_withdrawals_events_count",
- Help: "Number of forgeries withdrawals events",
+ Name: "potential_attack_on_defender_wins_games_count",
+ Help: "Number of potential attacks on defender wins games",
}),
- InvalidProposalWithdrawalsEventsGauge: m.NewGauge(prometheus.GaugeOpts{
+ PotentialAttackOnInProgressGamesGauge: m.NewGauge(prometheus.GaugeOpts{
Namespace: MetricsNamespace,
- Name: "invalid_proposal_withdrawals_events_count",
- Help: "Number of invalid proposal withdrawals events",
+ Name: "potential_attack_on_in_progress_games_count",
+ Help: "Number of potential attacks on in progress games",
}),
- ForgeriesWithdrawalsEventsGaugeVec: m.NewGaugeVec(
+ SuspiciousEventsOnChallengerWinsGamesGauge: m.NewGauge(prometheus.GaugeOpts{
+ Namespace: MetricsNamespace,
+ Name: "suspicious_events_on_challenger_wins_games_count",
+ Help: "Number of suspicious events on challenger wins games",
+ }),
+ PotentialAttackOnDefenderWinsGamesGaugeVec: m.NewGaugeVec(
prometheus.GaugeOpts{
Namespace: MetricsNamespace,
- Name: "forgeries_withdrawals_events_info",
- Help: "Information about forgeries withdrawals events.",
+ Name: "potential_attack_on_defender_wins_games_gauge_vec",
+ Help: "Information about potential attacks on defender wins games.",
},
- []string{"withdrawal_hash", "proof_submitter", "status", "blacklisted", "withdrawal_hash_present", "enriched", "event_block_number", "event_tx_hash", "event_index"},
+ []string{"withdrawal_hash", "proof_submitter", "status", "blacklisted", "withdrawal_hash_present", "enriched", "event_block_number", "event_tx_hash"},
),
- InvalidProposalWithdrawalsEventsGaugeVec: m.NewGaugeVec(
+ PotentialAttackOnInProgressGamesGaugeVec: m.NewGaugeVec(
prometheus.GaugeOpts{
Namespace: MetricsNamespace,
- Name: "invalid_proposal_withdrawals_events_info",
- Help: "Information about invalid proposal withdrawals events.",
+ Name: "potential_attack_on_in_progress_games_gauge_vec",
+ Help: "Information about potential attacks on in progress games.",
},
- []string{"withdrawal_hash", "proof_submitter", "status", "blacklisted", "withdrawal_hash_present", "enriched", "event_block_number", "event_tx_hash", "event_index"},
+ []string{"withdrawal_hash", "proof_submitter", "status", "blacklisted", "withdrawal_hash_present", "enriched", "event_block_number", "event_tx_hash"},
+ ),
+ SuspiciousEventsOnChallengerWinsGamesGaugeVec: m.NewGaugeVec(
+ prometheus.GaugeOpts{
+ Namespace: MetricsNamespace,
+ Name: "suspicious_events_on_challenger_wins_games_info",
+ Help: "Information about suspicious events on challenger wins games.",
+ },
+ []string{"withdrawal_hash", "proof_submitter", "status", "blacklisted", "withdrawal_hash_present", "enriched", "event_block_number", "event_tx_hash"},
),
}
@@ -178,26 +352,26 @@ func (m *Metrics) UpdateMetricsFromState(state *State) {
m.InitialL1HeightGauge.Set(float64(state.initialL1Height))
m.NextL1HeightGauge.Set(float64(state.nextL1Height))
m.LatestL1HeightGauge.Set(float64(state.latestL1Height))
+ m.LatestL2HeightGauge.Set(float64(state.latestL2Height))
- m.NumberOfDetectedForgeryGauge.Set(float64(state.numberOfDetectedForgery))
- m.NumberOfInvalidWithdrawalsGauge.Set(float64(state.numberOfInvalidWithdrawals))
- m.ForgeriesWithdrawalsEventsGauge.Set(float64(len(state.forgeriesWithdrawalsEvents)))
- m.InvalidProposalWithdrawalsEventsGauge.Set(float64(len(state.invalidProposalWithdrawalsEvents)))
+ m.PotentialAttackOnDefenderWinsGamesGauge.Set(float64(state.numberOfPotentialAttacksOnDefenderWinsGames))
+ m.PotentialAttackOnInProgressGamesGauge.Set(float64(state.numberOfPotentialAttackOnInProgressGames))
+ m.SuspiciousEventsOnChallengerWinsGamesGauge.Set(float64(state.numberOfSuspiciousEventsOnChallengerWinsGames))
// Update Counters by calculating deltas
// Processed Withdrawals
- processedWithdrawalsDelta := state.processedProvenWithdrawalsExtension1Events - m.previousProcessedProvenWithdrawalsExtension1Events
- if processedWithdrawalsDelta > 0 {
- m.ProcessedProvenWithdrawalsEventsExtensions1Counter.Add(float64(processedWithdrawalsDelta))
+ eventsProcessedDelta := state.eventsProcessed - m.previousEventsProcessed
+ if eventsProcessedDelta > 0 {
+ m.EventsProcessedCounter.Add(float64(eventsProcessedDelta))
}
- m.previousProcessedProvenWithdrawalsExtension1Events = state.processedProvenWithdrawalsExtension1Events
+ m.previousEventsProcessed = state.eventsProcessed
// Withdrawals Validated
- withdrawalsValidatedDelta := state.withdrawalsValidated - m.previousWithdrawalsValidated
- if withdrawalsValidatedDelta > 0 {
- m.WithdrawalsValidatedCounter.Add(float64(withdrawalsValidatedDelta))
+ withdrawalsProcessedDelta := state.withdrawalsProcessed - m.previousWithdrawalsProcessed
+ if withdrawalsProcessedDelta > 0 {
+ m.WithdrawalsProcessedCounter.Add(float64(withdrawalsProcessedDelta))
}
- m.previousWithdrawalsValidated = state.withdrawalsValidated
+ m.previousWithdrawalsProcessed = state.withdrawalsProcessed
// Node Connection Failures
nodeConnectionFailuresDelta := state.nodeConnectionFailures - m.previousNodeConnectionFailures
@@ -207,12 +381,12 @@ func (m *Metrics) UpdateMetricsFromState(state *State) {
m.previousNodeConnectionFailures = state.nodeConnectionFailures
// Update metrics for forgeries withdrawals events
- for index, event := range state.forgeriesWithdrawalsEvents {
+ for _, event := range state.potentialAttackOnDefenderWinsGames {
withdrawalHash := common.BytesToHash(event.Event.WithdrawalHash[:]).Hex()
proofSubmitter := event.Event.ProofSubmitter.String()
status := event.DisputeGame.DisputeGameData.Status.String()
- m.ForgeriesWithdrawalsEventsGaugeVec.WithLabelValues(
+ m.PotentialAttackOnDefenderWinsGamesGaugeVec.WithLabelValues(
withdrawalHash,
proofSubmitter,
status,
@@ -221,20 +395,19 @@ func (m *Metrics) UpdateMetricsFromState(state *State) {
fmt.Sprintf("%v", event.Enriched),
fmt.Sprintf("%v", event.Event.Raw.BlockNumber),
event.Event.Raw.TxHash.String(),
- fmt.Sprintf("%v", index),
).Set(1) // Set a value for existence
}
// Clear the previous values
- m.InvalidProposalWithdrawalsEventsGaugeVec.Reset()
+ m.PotentialAttackOnInProgressGamesGaugeVec.Reset()
// Update metrics for invalid proposal withdrawals events
- for index, event := range state.invalidProposalWithdrawalsEvents {
+ for _, event := range state.potentialAttackOnInProgressGames {
withdrawalHash := common.BytesToHash(event.Event.WithdrawalHash[:]).Hex()
proofSubmitter := event.Event.ProofSubmitter.String()
status := event.DisputeGame.DisputeGameData.Status.String()
- m.InvalidProposalWithdrawalsEventsGaugeVec.WithLabelValues(
+ m.PotentialAttackOnInProgressGamesGaugeVec.WithLabelValues(
withdrawalHash,
proofSubmitter,
status,
@@ -243,7 +416,32 @@ func (m *Metrics) UpdateMetricsFromState(state *State) {
fmt.Sprintf("%v", event.Enriched),
fmt.Sprintf("%v", event.Event.Raw.BlockNumber),
event.Event.Raw.TxHash.String(),
- fmt.Sprintf("%v", index),
).Set(1) // Set a value for existence
}
+
+ // Clear the previous values
+ m.SuspiciousEventsOnChallengerWinsGamesGaugeVec.Reset()
+ // Update metrics for invalid proposal withdrawals events
+ for _, key := range state.suspiciousEventsOnChallengerWinsGames.Keys() {
+ enrichedEvent, ok := state.suspiciousEventsOnChallengerWinsGames.Get(key)
+ if ok {
+ event := enrichedEvent.(validator.EnrichedProvenWithdrawalEvent)
+
+ withdrawalHash := common.BytesToHash(event.Event.WithdrawalHash[:]).Hex()
+ proofSubmitter := event.Event.ProofSubmitter.String()
+ status := event.DisputeGame.DisputeGameData.Status.String()
+
+ m.SuspiciousEventsOnChallengerWinsGamesGaugeVec.WithLabelValues(
+ withdrawalHash,
+ proofSubmitter,
+ status,
+ fmt.Sprintf("%v", event.Blacklisted),
+ fmt.Sprintf("%v", event.WithdrawalHashPresentOnL2),
+ fmt.Sprintf("%v", event.Enriched),
+ fmt.Sprintf("%v", event.Event.Raw.BlockNumber),
+ event.Event.Raw.TxHash.String(),
+ ).Set(1) // Set a value for existence
+ }
+ }
+
}
diff --git a/op-monitorism/faultproof_withdrawals/validator/fault_dispute_game_helper.go b/op-monitorism/faultproof_withdrawals/validator/fault_dispute_game_helper.go
index 92d8b867..1b559e1c 100644
--- a/op-monitorism/faultproof_withdrawals/validator/fault_dispute_game_helper.go
+++ b/op-monitorism/faultproof_withdrawals/validator/fault_dispute_game_helper.go
@@ -69,7 +69,7 @@ func (gs GameStatus) String() string {
// String provides a string representation of DisputeGameData.
func (d DisputeGameData) String() string {
- return fmt.Sprintf("DisputeGame[ disputeGameProxyAddress=%v rootClaim=%s l2blockNumber=%s l2ChainID=%s status=%v createdAt=%v resolvedAt=%v ]",
+ return fmt.Sprintf("DisputeGame[ disputeGameProxyAddress: %v rootClaim: %s l2blockNumber: %s l2ChainID: %s status: %v createdAt: %v resolvedAt: %v ]",
d.ProxyAddress,
common.BytesToHash(d.RootClaim[:]),
d.L2blockNumber.String(),
diff --git a/op-monitorism/faultproof_withdrawals/validator/op_node_helper.go b/op-monitorism/faultproof_withdrawals/validator/op_node_helper.go
index 67d63cb4..7fa0543c 100644
--- a/op-monitorism/faultproof_withdrawals/validator/op_node_helper.go
+++ b/op-monitorism/faultproof_withdrawals/validator/op_node_helper.go
@@ -17,55 +17,82 @@ import (
// OpNodeHelper assists in interacting with the op-node
type OpNodeHelper struct {
// objects
- l2OpNodeClient *ethclient.Client // The op-node (consensus) client.
- rpc_l2Client *rpc.Client // The RPC client for the L2 node.
- ctx context.Context // Context for managing cancellation and timeouts.
- l2OutputRootCache *lru.Cache // Cache for storing L2 output roots.
+ l2OpNodeClient *ethclient.Client // The op-node (consensus) client.
+ l2OpGethClient *ethclient.Client // The op-geth client.
+ rpc_l2Client *rpc.Client // The RPC client for the L2 node.
+ ctx context.Context // Context for managing cancellation and timeouts.
+ l2OutputRootCache *lru.Cache // Cache for storing L2 output roots.
+ LatestKnownL2BlockNumber uint64 // The latest known L2 block number.
}
const outputRootCacheSize = 1000 // Size of the output root cache.
// NewOpNodeHelper initializes a new OpNodeHelper.
// It creates a cache for storing output roots and binds to the L2 node client.
-func NewOpNodeHelper(ctx context.Context, l2OpNodeClient *ethclient.Client) (*OpNodeHelper, error) {
+func NewOpNodeHelper(ctx context.Context, l2OpNodeClient *ethclient.Client, l2OpGethClient *ethclient.Client) (*OpNodeHelper, error) {
l2OutputRootCache, err := lru.New(outputRootCacheSize)
if err != nil {
return nil, fmt.Errorf("failed to create cache: %w", err)
}
rpc_l2Client := l2OpNodeClient.Client()
- return &OpNodeHelper{
- l2OpNodeClient: l2OpNodeClient,
- rpc_l2Client: rpc_l2Client,
- ctx: ctx,
- l2OutputRootCache: l2OutputRootCache,
- }, nil
+ ret := OpNodeHelper{
+ l2OpNodeClient: l2OpNodeClient,
+ l2OpGethClient: l2OpGethClient,
+ rpc_l2Client: rpc_l2Client,
+ ctx: ctx,
+ l2OutputRootCache: l2OutputRootCache,
+ LatestKnownL2BlockNumber: 0,
+ }
+
+ //ignoring the return value as it is already stored in the struct by the method
+ latestBlockNumber, err := ret.GetLatestKnownL2BlockNumber()
+ if err != nil {
+ return nil, fmt.Errorf("failed to get latest known L2 block number: %w", err)
+ }
+
+ ret.LatestKnownL2BlockNumber = latestBlockNumber
+ return &ret, nil
+
+}
+
+// get latest known L2 block number
+func (on *OpNodeHelper) GetLatestKnownL2BlockNumber() (uint64, error) {
+ LatestKnownL2BlockNumber, err := on.l2OpGethClient.BlockNumber(on.ctx)
+ if err != nil {
+ return 0, fmt.Errorf("failed to get latest known L2 block number: %w", err)
+ }
+ on.LatestKnownL2BlockNumber = LatestKnownL2BlockNumber
+ return LatestKnownL2BlockNumber, nil
}
// GetOutputRootFromTrustedL2Node retrieves the output root for a given L2 block number from a trusted L2 node.
-// It returns the output root as a bytes32 array.
+// It returns the output root as a Bytes32 array.
func (on *OpNodeHelper) GetOutputRootFromTrustedL2Node(l2blockNumber *big.Int) ([32]byte, error) {
ret, found := on.l2OutputRootCache.Get(l2blockNumber)
+
if !found {
var result OutputResponse
l2blockNumberHex := hexutil.EncodeBig(l2blockNumber)
err := on.rpc_l2Client.CallContext(on.ctx, &result, "optimism_outputAtBlock", l2blockNumberHex)
+ //check if error contains "failed to determine L2BlockRef of height"
if err != nil {
return [32]byte{}, fmt.Errorf("failed to get output at block for game block:%v : %w", l2blockNumberHex, err)
}
trustedRootProof, err := StringToBytes32(result.OutputRoot)
if err != nil {
- return [32]byte{}, fmt.Errorf("failed to convert output root to bytes32: %w", err)
+ return [32]byte{}, fmt.Errorf("failed to convert output root to Bytes32: %w", err)
}
- ret = trustedRootProof
+ ret = [32]byte(trustedRootProof)
+ on.l2OutputRootCache.Add(l2blockNumber, ret)
}
return ret.([32]byte), nil
}
// GetOutputRootFromCalculation retrieves the output root by calculating it from the given block number.
-// It returns the calculated output root as a bytes32 array.
+// It returns the calculated output root as a Bytes32 array.
func (on *OpNodeHelper) GetOutputRootFromCalculation(blockNumber *big.Int) ([32]byte, error) {
block, err := on.l2OpNodeClient.BlockByNumber(on.ctx, blockNumber)
if err != nil {
@@ -78,6 +105,6 @@ func (on *OpNodeHelper) GetOutputRootFromCalculation(blockNumber *big.Int) ([32]
return [32]byte{}, fmt.Errorf("failed to get proof: %w", err)
}
- outputRoot := eth.OutputRoot(ð.OutputV0{StateRoot: eth.Bytes32(block.Root()), MessagePasserStorageRoot: eth.Bytes32(proof.StorageHash), BlockHash: block.Hash()})
+ outputRoot := eth.OutputRoot(ð.OutputV0{StateRoot: [32]byte(block.Root()), MessagePasserStorageRoot: [32]byte(proof.StorageHash), BlockHash: block.Hash()})
return outputRoot, nil
}
diff --git a/op-monitorism/faultproof_withdrawals/validator/optimism_portal2_helper.go b/op-monitorism/faultproof_withdrawals/validator/optimism_portal2_helper.go
index ef784449..32f5484f 100644
--- a/op-monitorism/faultproof_withdrawals/validator/optimism_portal2_helper.go
+++ b/op-monitorism/faultproof_withdrawals/validator/optimism_portal2_helper.go
@@ -41,13 +41,13 @@ type OptimismPortal2Helper struct {
}
// String provides a string representation of WithdrawalProvenExtension1Event.
-func (e *WithdrawalProvenExtension1Event) String() string {
- return fmt.Sprintf("WithdrawalHash: %x, ProofSubmitter: %v, Raw: %v", e.WithdrawalHash, e.ProofSubmitter, e.Raw)
+func (e WithdrawalProvenExtension1Event) String() string {
+ return fmt.Sprintf("WithdrawalHash: %s, ProofSubmitter: %v, Raw: %v", common.BytesToHash(e.WithdrawalHash[:]), e.ProofSubmitter, e.Raw)
}
// String provides a string representation of WithdrawalProvenEvent.
-func (e *WithdrawalProvenEvent) String() string {
- return fmt.Sprintf("WithdrawalHash: %x, Raw: %v", e.WithdrawalHash, e.Raw)
+func (e WithdrawalProvenEvent) String() string {
+ return fmt.Sprintf("WithdrawalHash: %s, Raw: %v", common.BytesToHash(e.WithdrawalHash[:]), e.Raw)
}
// String provides a string representation of SubmittedProofData.
@@ -126,9 +126,9 @@ func (op *OptimismPortal2Helper) GetProvenWithdrawalsEvents(start uint64, end *u
return events, nil
}
-// GetProvenWithdrawalsExtension1EventsIterartor creates an iterator for proven withdrawal extension 1 events within the specified block range.
+// GetProvenWithdrawalsExtension1EventsIterator creates an iterator for proven withdrawal extension 1 events within the specified block range.
// It returns the iterator along with any error encountered.
-func (op *OptimismPortal2Helper) GetProvenWithdrawalsExtension1EventsIterartor(start uint64, end *uint64) (*l1.OptimismPortal2WithdrawalProvenExtension1Iterator, error) {
+func (op *OptimismPortal2Helper) GetProvenWithdrawalsExtension1EventsIterator(start uint64, end *uint64) (*l1.OptimismPortal2WithdrawalProvenExtension1Iterator, error) {
filterOpts := &bind.FilterOpts{Context: op.ctx, Start: start, End: end}
iterator, err := op.optimismPortal2.FilterWithdrawalProvenExtension1(filterOpts, nil, nil)
if err != nil {
@@ -141,7 +141,7 @@ func (op *OptimismPortal2Helper) GetProvenWithdrawalsExtension1EventsIterartor(s
// GetProvenWithdrawalsExtension1Events retrieves proven withdrawal extension 1 events within the specified block range.
// It returns a slice of WithdrawalProvenExtension1Event along with any error encountered.
func (op *OptimismPortal2Helper) GetProvenWithdrawalsExtension1Events(start uint64, end *uint64) ([]WithdrawalProvenExtension1Event, error) {
- iterator, err := op.GetProvenWithdrawalsExtension1EventsIterartor(start, end)
+ iterator, err := op.GetProvenWithdrawalsExtension1EventsIterator(start, end)
if err != nil {
return nil, fmt.Errorf("failed to get proven withdrawals extension1 iterator error:%w", err)
}
diff --git a/op-monitorism/faultproof_withdrawals/validator/proven_withdrawal_validator.go b/op-monitorism/faultproof_withdrawals/validator/proven_withdrawal_validator.go
index e7c21c1f..f20e235c 100644
--- a/op-monitorism/faultproof_withdrawals/validator/proven_withdrawal_validator.go
+++ b/op-monitorism/faultproof_withdrawals/validator/proven_withdrawal_validator.go
@@ -45,7 +45,7 @@ func (e *EnrichedProvenWithdrawalEvent) String() string {
return fmt.Sprintf("Event: %v, DisputeGame: %v, ExpectedRootClaim: %s, Blacklisted: %v, withdrawalHashPresentOnL2: %v, Enriched: %v",
e.Event,
e.DisputeGame,
- common.BytesToHash(e.ExpectedRootClaim[:]),
+ common.Bytes2Hex(e.ExpectedRootClaim[:]),
e.Blacklisted,
e.WithdrawalHashPresentOnL2,
e.Enriched)
@@ -74,7 +74,7 @@ func NewWithdrawalValidator(ctx context.Context, l1GethClient *ethclient.Client,
return nil, fmt.Errorf("failed to create l2 to l1 message passer helper: %w", err)
}
- l2NodeHelper, err := NewOpNodeHelper(ctx, l2OpNodeClient)
+ l2NodeHelper, err := NewOpNodeHelper(ctx, l2OpNodeClient, l2OpGethClient)
if err != nil {
return nil, fmt.Errorf("failed to create l2 node helper: %w", err)
}
@@ -112,11 +112,20 @@ func (wv *ProvenWithdrawalValidator) UpdateEnrichedWithdrawalEvent(event *Enrich
// Check if the game root claim is valid on L2 only if not confirmed already that it is on L2
if !event.Enriched {
- trustedRootClaim, err := wv.l2NodeHelper.GetOutputRootFromTrustedL2Node(event.DisputeGame.DisputeGameData.L2blockNumber)
+ latest_known_l2_block, err := wv.l2NodeHelper.GetLatestKnownL2BlockNumber()
if err != nil {
- return fmt.Errorf("failed to get trustedRootClaim from Op-node: %w", err)
+ return fmt.Errorf("failed to get latest known L2 block number: %w", err)
}
- event.ExpectedRootClaim = trustedRootClaim
+ if latest_known_l2_block >= event.DisputeGame.DisputeGameData.L2blockNumber.Uint64() {
+ trustedRootClaim, err := wv.l2NodeHelper.GetOutputRootFromTrustedL2Node(event.DisputeGame.DisputeGameData.L2blockNumber)
+ if err != nil {
+ return fmt.Errorf("failed to get trustedRootClaim from Op-node: %w", err)
+ }
+ event.ExpectedRootClaim = trustedRootClaim
+ } else {
+ event.ExpectedRootClaim = [32]byte{}
+ }
+
}
// Check if the withdrawal exists on L2 only if not confirmed already that it is on L2
@@ -194,11 +203,43 @@ func (wv *ProvenWithdrawalValidator) GetEnrichedWithdrawalsEvents(start uint64,
return enrichedProvenWithdrawalEvents, nil
}
+// GetEnrichedWithdrawalsEvents retrieves enriched withdrawal events within the specified block range.
+// It returns a slice of EnrichedProvenWithdrawalEvent along with any error encountered.
+func (wv *ProvenWithdrawalValidator) GetEnrichedWithdrawalsEventsMap(start uint64, end *uint64) (map[common.Hash]EnrichedProvenWithdrawalEvent, error) {
+ iterator, err := wv.optimismPortal2Helper.GetProvenWithdrawalsExtension1EventsIterator(start, end)
+ if err != nil {
+ return nil, fmt.Errorf("failed to get proven withdrawals extension1 iterator error:%w", err)
+ }
+
+ enrichedProvenWithdrawalEvents := make(map[common.Hash]EnrichedProvenWithdrawalEvent)
+
+ for iterator.Next() {
+ event := iterator.Event
+
+ enrichedWithdrawalEvent, err := wv.GetEnrichedWithdrawalEvent(&WithdrawalProvenExtension1Event{
+ WithdrawalHash: event.WithdrawalHash,
+ ProofSubmitter: event.ProofSubmitter,
+ Raw: Raw{
+ BlockNumber: event.Raw.BlockNumber,
+ TxHash: event.Raw.TxHash,
+ },
+ })
+ if err != nil {
+ return nil, fmt.Errorf("failed to get enriched withdrawal event: %w", err)
+ }
+
+ key := enrichedWithdrawalEvent.Event.Raw.TxHash
+ enrichedProvenWithdrawalEvents[key] = *enrichedWithdrawalEvent
+ }
+
+ return enrichedProvenWithdrawalEvents, nil
+}
+
// IsWithdrawalEventValid checks if the enriched withdrawal event is valid.
// It returns true if the event is valid, otherwise returns false.
func (wv *ProvenWithdrawalValidator) IsWithdrawalEventValid(enrichedWithdrawalEvent *EnrichedProvenWithdrawalEvent) (bool, error) {
- if enrichedWithdrawalEvent.ExpectedRootClaim == [32]byte{} {
- return false, fmt.Errorf("trustedRootClaim is nil, game not enriched")
+ if !enrichedWithdrawalEvent.Enriched {
+ return false, fmt.Errorf("game not enriched")
}
validGameRootClaim := enrichedWithdrawalEvent.DisputeGame.DisputeGameData.RootClaim == enrichedWithdrawalEvent.ExpectedRootClaim
@@ -208,3 +249,7 @@ func (wv *ProvenWithdrawalValidator) IsWithdrawalEventValid(enrichedWithdrawalEv
return false, nil
}
}
+
+func (wv *ProvenWithdrawalValidator) GetLatestL2Height() uint64 {
+ return wv.l2NodeHelper.LatestKnownL2BlockNumber
+}
diff --git a/op-monitorism/faultproof_withdrawals/validator/utils.go b/op-monitorism/faultproof_withdrawals/validator/utils.go
index 7d0c1443..40ffe2d7 100644
--- a/op-monitorism/faultproof_withdrawals/validator/utils.go
+++ b/op-monitorism/faultproof_withdrawals/validator/utils.go
@@ -2,6 +2,7 @@ package validator
import (
"encoding/hex"
+ "fmt"
"strings"
"time"
@@ -14,6 +15,11 @@ type Raw struct {
TxHash common.Hash // The hash of the transaction.
}
+// String provides a string representation of Raw.
+func (r Raw) String() string {
+ return fmt.Sprintf("{BlockNumber: %d, TxHash: %s}", r.BlockNumber, r.TxHash.String())
+}
+
// Timestamp represents a Unix timestamp.
type Timestamp uint64
diff --git a/op-monitorism/go.mod b/op-monitorism/go.mod
index 080c5ee6..4ce4ec20 100644
--- a/op-monitorism/go.mod
+++ b/op-monitorism/go.mod
@@ -11,6 +11,7 @@ require (
github.com/hashicorp/golang-lru v0.5.0
github.com/joho/godotenv v1.5.1
github.com/prometheus/client_golang v1.20.2
+ github.com/prometheus/client_model v0.6.1
github.com/stretchr/testify v1.9.0
github.com/urfave/cli/v2 v2.27.4
gopkg.in/yaml.v3 v3.0.1
@@ -50,8 +51,6 @@ require (
github.com/google/uuid v1.6.0 // indirect
github.com/gorilla/websocket v1.5.3 // indirect
github.com/holiman/uint256 v1.3.1 // indirect
- github.com/huin/goupnp v1.3.0 // indirect
- github.com/jackpal/go-nat-pmp v1.0.2 // indirect
github.com/klauspost/compress v1.17.9 // indirect
github.com/kr/pretty v0.3.1 // indirect
github.com/kr/text v0.2.0 // indirect
@@ -62,7 +61,6 @@ require (
github.com/olekukonko/tablewriter v0.0.5 // indirect
github.com/pkg/errors v0.9.1 // indirect
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 // indirect
- github.com/prometheus/client_model v0.6.1 // indirect
github.com/prometheus/common v0.55.0 // indirect
github.com/prometheus/procfs v0.15.1 // indirect
github.com/rivo/uniseg v0.4.4 // indirect
diff --git a/op-monitorism/go.sum b/op-monitorism/go.sum
index 03a2f695..49f5cb7a 100644
--- a/op-monitorism/go.sum
+++ b/op-monitorism/go.sum
@@ -262,7 +262,6 @@ golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJ
golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
-golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM=
golang.org/x/sync v0.8.0 h1:3NFvSEYkUoMifnESzZl15y791HH1qU2xm6eCJU5ZPXQ=
golang.org/x/sync v0.8.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=