program.go 3.04 KB
Newer Older
1 2 3 4 5 6
package client

import (
	"errors"
	"fmt"
	"io"
7
	"os"
8 9

	"github.com/ethereum-optimism/optimism/op-node/rollup"
10
	preimage "github.com/ethereum-optimism/optimism/op-preimage"
11
	"github.com/ethereum-optimism/optimism/op-program/client/claim"
12 13 14
	cldr "github.com/ethereum-optimism/optimism/op-program/client/driver"
	"github.com/ethereum-optimism/optimism/op-program/client/l1"
	"github.com/ethereum-optimism/optimism/op-program/client/l2"
15
	"github.com/ethereum-optimism/optimism/op-service/eth"
16 17 18
	"github.com/ethereum/go-ethereum/common"
	"github.com/ethereum/go-ethereum/log"
	"github.com/ethereum/go-ethereum/params"
19 20
)

21 22 23 24
// Main executes the client program in a detached context and exits the current process.
// The client runtime environment must be preset before calling this function.
func Main(logger log.Logger) {
	log.Info("Starting fault proof program client")
25 26
	preimageOracle := preimage.ClientPreimageChannel()
	preimageHinter := preimage.ClientHinterChannel()
27
	if err := RunProgram(logger, preimageOracle, preimageHinter); errors.Is(err, claim.ErrClaimNotValid) {
28
		log.Error("Claim is invalid", "err", err)
29
		os.Exit(1)
30 31 32
	} else if err != nil {
		log.Error("Program failed", "err", err)
		os.Exit(2)
33
	} else {
34
		log.Info("Claim successfully verified")
35 36 37 38 39
		os.Exit(0)
	}
}

// RunProgram executes the Program, while attached to an IO based pre-image oracle, to be served by a host.
40
func RunProgram(logger log.Logger, preimageOracle io.ReadWriter, preimageHinter io.ReadWriter) error {
41 42
	pClient := preimage.NewOracleClient(preimageOracle)
	hClient := preimage.NewHintWriter(preimageHinter)
43 44
	l1PreimageOracle := l1.NewCachingOracle(l1.NewPreimageOracle(pClient, hClient))
	l2PreimageOracle := l2.NewCachingOracle(l2.NewPreimageOracle(pClient, hClient))
45 46 47

	bootInfo := NewBootstrapClient(pClient).BootInfo()
	logger.Info("Program Bootstrapped", "bootInfo", bootInfo)
48 49
	return runDerivation(
		logger,
50
		bootInfo.RollupConfig,
51 52
		bootInfo.L2ChainConfig,
		bootInfo.L1Head,
53
		bootInfo.L2OutputRoot,
54 55 56 57 58
		bootInfo.L2Claim,
		bootInfo.L2ClaimBlockNumber,
		l1PreimageOracle,
		l2PreimageOracle,
	)
59 60
}

61
// runDerivation executes the L2 state transition, given a minimal interface to retrieve data.
62
func runDerivation(logger log.Logger, cfg *rollup.Config, l2Cfg *params.ChainConfig, l1Head common.Hash, l2OutputRoot common.Hash, l2Claim common.Hash, l2ClaimBlockNum uint64, l1Oracle l1.Oracle, l2Oracle l2.Oracle) error {
63
	l1Source := l1.NewOracleL1Client(logger, l1Oracle, l1Head)
64
	l1BlobsSource := l1.NewBlobFetcher(logger, l1Oracle)
65
	engineBackend, err := l2.NewOracleBackedL2Chain(logger, l2Oracle, l1Oracle /* kzg oracle */, l2Cfg, l2OutputRoot)
66 67 68 69 70 71
	if err != nil {
		return fmt.Errorf("failed to create oracle-backed L2 chain: %w", err)
	}
	l2Source := l2.NewOracleEngine(cfg, logger, engineBackend)

	logger.Info("Starting derivation")
72
	d := cldr.NewDriver(logger, cfg, l1Source, l1BlobsSource, l2Source, l2ClaimBlockNum)
73 74
	if err := d.RunComplete(); err != nil {
		return fmt.Errorf("failed to run program to completion: %w", err)
75
	}
76
	return claim.ValidateClaim(logger, l2ClaimBlockNum, eth.Bytes32(l2Claim), l2Source)
77
}