From 1e2e4027dce04b83340b4b2f586e62a10ac7e6c7 Mon Sep 17 00:00:00 2001 From: Krish Date: Fri, 5 Jul 2024 10:11:40 +0800 Subject: [PATCH 1/4] feat: auto recover from pbss geth unclean shutdown --- op-node/rollup/derive/engine_controller.go | 80 ++++++++++++++++++++++ op-node/rollup/derive/engine_queue.go | 5 ++ op-service/eth/types.go | 2 + 3 files changed, 87 insertions(+) diff --git a/op-node/rollup/derive/engine_controller.go b/op-node/rollup/derive/engine_controller.go index f73ce21891..ee437f5289 100644 --- a/op-node/rollup/derive/engine_controller.go +++ b/op-node/rollup/derive/engine_controller.go @@ -271,6 +271,9 @@ func (e *EngineController) resetBuildingState() { // It returns true if the status is acceptable. func (e *EngineController) checkNewPayloadStatus(status eth.ExecutePayloadStatus) bool { if e.syncMode == sync.ELSync { + if status == eth.ExecutionInconsistent { + return true + } if status == eth.ExecutionValid && e.syncStatus == syncStatusStartedEL { e.syncStatus = syncStatusFinishedELButNotFinalized } @@ -348,6 +351,41 @@ func (e *EngineController) InsertUnsafePayload(ctx context.Context, envelope *et if err != nil { return NewTemporaryError(fmt.Errorf("failed to update insert payload: %w", err)) } + + //process inconsistent state + if status.Status == eth.ExecutionInconsistent { + currentL2Info, err := e.getCurrentL2Info(ctx) + if err != nil { + return NewTemporaryError(fmt.Errorf("failed to process unconsistent state: %w", err)) + } else { + log.Info("engine has inconsistent state", "unsafe", currentL2Info.Unsafe.Number, "safe", currentL2Info.Safe.Number, "final", currentL2Info.Finalized.Number) + e.SetUnsafeHead(currentL2Info.Unsafe) + if currentL2Info.Safe.Number > currentL2Info.Unsafe.Number { + log.Info("current safe is higher than unsafe block, reset it", "set safe after", currentL2Info.Unsafe.Number, "set safe before", e.safeHead.Number) + e.SetSafeHead(currentL2Info.Unsafe) + } + if currentL2Info.Finalized.Number > currentL2Info.Unsafe.Number { + log.Info("current finalized is higher than unsafe block, reset it", "set Finalized after", currentL2Info.Unsafe.Number, "set Finalized before", e.safeHead.Number) + e.SetFinalizedHead(currentL2Info.Unsafe) + } + } + + fcuReq := eth.ForkchoiceState{ + HeadBlockHash: e.unsafeHead.Hash, + SafeBlockHash: e.safeHead.Hash, + FinalizedBlockHash: e.finalizedHead.Hash, + } + + fcuRes, err := e.engine.ForkchoiceUpdate(ctx, &fcuReq, nil) + if fcuRes.PayloadStatus.Status == eth.ExecutionValid { + log.Info("engine processed data successfully") + e.needFCUCall = false + return nil + } else { + return NewTemporaryError(fmt.Errorf("engine failed to process unconsistent data: %w", err)) + } + } + if !e.checkNewPayloadStatus(status.Status) { payload := envelope.ExecutionPayload return NewTemporaryError(fmt.Errorf("cannot process unsafe payload: new - %v; parent: %v; err: %w", @@ -360,6 +398,22 @@ func (e *EngineController) InsertUnsafePayload(ctx context.Context, envelope *et SafeBlockHash: e.safeHead.Hash, FinalizedBlockHash: e.finalizedHead.Hash, } + + //update unsafe,safe,finalize and send fcu for sync + if status.Status == eth.ExecutionInconsistent { + log.Info("engine meet unconsistent here") + currentUnsafe, _ := e.engine.L2BlockRefByLabel(ctx, eth.Unsafe) + //reset unsafe + e.SetUnsafeHead(currentUnsafe) + //force reset safe,finalize + e.SetSafeHead(currentUnsafe) + e.SetFinalizedHead(currentUnsafe) + + fc.HeadBlockHash = currentUnsafe.Hash + fc.SafeBlockHash = currentUnsafe.Hash + fc.FinalizedBlockHash = currentUnsafe.Hash + } + if e.syncStatus == syncStatusFinishedELButNotFinalized { fc.SafeBlockHash = envelope.ExecutionPayload.BlockHash fc.FinalizedBlockHash = envelope.ExecutionPayload.BlockHash @@ -468,3 +522,29 @@ func (e *EngineController) TryBackupUnsafeReorg(ctx context.Context) (bool, erro func (e *EngineController) ResetBuildingState() { e.resetBuildingState() } + +// getCurrentL2Info returns the current finalized, safe and unsafe heads of the execution engine. +func (e *EngineController) getCurrentL2Info(ctx context.Context) (*sync.FindHeadsResult, error) { + finalized, err := e.engine.L2BlockRefByLabel(ctx, eth.Finalized) + if err != nil { + log.Error("err get finalized", "err", err) + return nil, fmt.Errorf("failed to find the finalized L2 block: %w", err) + } + + safe, err := e.engine.L2BlockRefByLabel(ctx, eth.Safe) + if errors.Is(err, ethereum.NotFound) { + safe = finalized + } else if err != nil { + return nil, fmt.Errorf("failed to find the safe L2 block: %w", err) + } + + unsafe, err := e.engine.L2BlockRefByLabel(ctx, eth.Unsafe) + if err != nil { + return nil, fmt.Errorf("failed to find the L2 head block: %w", err) + } + return &sync.FindHeadsResult{ + Unsafe: unsafe, + Safe: safe, + Finalized: finalized, + }, nil +} diff --git a/op-node/rollup/derive/engine_queue.go b/op-node/rollup/derive/engine_queue.go index dee9f46d0b..a394ab148a 100644 --- a/op-node/rollup/derive/engine_queue.go +++ b/op-node/rollup/derive/engine_queue.go @@ -695,6 +695,11 @@ func (eq *EngineQueue) Reset(ctx context.Context, _ eth.L1BlockRef, _ eth.System return NewTemporaryError(fmt.Errorf("failed to find the L2 Heads to start from: %w", err)) } finalized, safe, unsafe := result.Finalized, result.Safe, result.Unsafe + + if finalized.Number > safe.Number { + finalized = safe + } + l1Origin, err := eq.l1Fetcher.L1BlockRefByHash(ctx, safe.L1Origin.Hash) if err != nil { return NewTemporaryError(fmt.Errorf("failed to fetch the new L1 progress: origin: %v; err: %w", safe.L1Origin, err)) diff --git a/op-service/eth/types.go b/op-service/eth/types.go index 58d16c3cdd..18ae2cf3af 100644 --- a/op-service/eth/types.go +++ b/op-service/eth/types.go @@ -338,6 +338,8 @@ const ( ExecutionInvalidBlockHash ExecutePayloadStatus = "INVALID_BLOCK_HASH" // proof-of-stake transition only, not used in rollup ExecutionInvalidTerminalBlock ExecutePayloadStatus = "INVALID_TERMINAL_BLOCK" + // given payload is invalid + ExecutionInconsistent ExecutePayloadStatus = "INCONSISTENT" ) type PayloadStatusV1 struct { From 3df876fdef483b01c2bbffc308ba54b069d6a56a Mon Sep 17 00:00:00 2001 From: Owen <103096885+owen-reorg@users.noreply.github.com> Date: Mon, 8 Jul 2024 15:48:42 +0800 Subject: [PATCH 2/4] Update op-node/rollup/derive/engine_controller.go --- op-node/rollup/derive/engine_controller.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/op-node/rollup/derive/engine_controller.go b/op-node/rollup/derive/engine_controller.go index ee437f5289..50dbed25e4 100644 --- a/op-node/rollup/derive/engine_controller.go +++ b/op-node/rollup/derive/engine_controller.go @@ -356,7 +356,7 @@ func (e *EngineController) InsertUnsafePayload(ctx context.Context, envelope *et if status.Status == eth.ExecutionInconsistent { currentL2Info, err := e.getCurrentL2Info(ctx) if err != nil { - return NewTemporaryError(fmt.Errorf("failed to process unconsistent state: %w", err)) + return NewTemporaryError(fmt.Errorf("failed to process inconsistent state: %w", err)) } else { log.Info("engine has inconsistent state", "unsafe", currentL2Info.Unsafe.Number, "safe", currentL2Info.Safe.Number, "final", currentL2Info.Finalized.Number) e.SetUnsafeHead(currentL2Info.Unsafe) From d2b79b6df19b9d4325374e69a2356e8971724459 Mon Sep 17 00:00:00 2001 From: Owen <103096885+owen-reorg@users.noreply.github.com> Date: Mon, 8 Jul 2024 15:48:59 +0800 Subject: [PATCH 3/4] Update op-node/rollup/derive/engine_controller.go --- op-node/rollup/derive/engine_controller.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/op-node/rollup/derive/engine_controller.go b/op-node/rollup/derive/engine_controller.go index 50dbed25e4..d00a5beb0a 100644 --- a/op-node/rollup/derive/engine_controller.go +++ b/op-node/rollup/derive/engine_controller.go @@ -382,7 +382,7 @@ func (e *EngineController) InsertUnsafePayload(ctx context.Context, envelope *et e.needFCUCall = false return nil } else { - return NewTemporaryError(fmt.Errorf("engine failed to process unconsistent data: %w", err)) + return NewTemporaryError(fmt.Errorf("engine failed to process inconsistent data: %w", err)) } } From bfcf7b4cf6c9dc0e4988c5b6f14d09e5b44b42cc Mon Sep 17 00:00:00 2001 From: Owen <103096885+owen-reorg@users.noreply.github.com> Date: Mon, 8 Jul 2024 15:49:17 +0800 Subject: [PATCH 4/4] Update op-node/rollup/derive/engine_controller.go --- op-node/rollup/derive/engine_controller.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/op-node/rollup/derive/engine_controller.go b/op-node/rollup/derive/engine_controller.go index d00a5beb0a..1a7c4601b2 100644 --- a/op-node/rollup/derive/engine_controller.go +++ b/op-node/rollup/derive/engine_controller.go @@ -401,7 +401,7 @@ func (e *EngineController) InsertUnsafePayload(ctx context.Context, envelope *et //update unsafe,safe,finalize and send fcu for sync if status.Status == eth.ExecutionInconsistent { - log.Info("engine meet unconsistent here") + log.Info("engine meet inconsistent here") currentUnsafe, _ := e.engine.L2BlockRefByLabel(ctx, eth.Unsafe) //reset unsafe e.SetUnsafeHead(currentUnsafe)