• Adrian Sutton's avatar
    op-challenger: Hook up the super game type (#13794) · 5f4757cb
    Adrian Sutton authored
    * op-challenger: Implement split adapter and trace accessor creation for super cannon
    
    * op-challenger: Implement register task for super cannon
    
    * op-challenger: Actually register super cannon game handler
    
    * op-challenger: Lazy create clients
    
    Make sync validator configurable per game type.
    
    * op-challenger: Test split adapter
    
    Remove ClaimTimestamp from ClaimInfo as we always use the poststate timestamp
    5f4757cb
split_adapter_test.go 4.66 KB
package super

import (
	"context"
	"errors"
	"math/big"
	"testing"

	"github.com/ethereum-optimism/optimism/op-challenger/game/fault/trace/split"
	"github.com/ethereum-optimism/optimism/op-challenger/game/fault/types"
	interopTypes "github.com/ethereum-optimism/optimism/op-program/client/interop/types"
	"github.com/ethereum-optimism/optimism/op-service/eth"
	"github.com/ethereum-optimism/optimism/op-service/testlog"
	"github.com/ethereum/go-ethereum/common"
	"github.com/ethereum/go-ethereum/log"
	"github.com/stretchr/testify/require"
)

var creatorError = errors.New("captured args")

func TestSplitAdapter(t *testing.T) {
	depth := types.Depth(30)
	prestateTimestamp := uint64(150)
	poststateTimestamp := uint64(200)

	t.Run("FromAbsolutePrestate", func(t *testing.T) {
		creator, rootProvider, adapter := setupSplitAdapterTest(t, depth, prestateTimestamp, poststateTimestamp)
		prestateResponse := eth.SuperRootResponse{
			Timestamp: prestateTimestamp,
		}
		rootProvider.Add(prestateResponse)
		postClaim := types.Claim{
			ClaimData: types.ClaimData{
				Value:    common.Hash{0xaa, 0xbb, 0xcc, 0xdd, 0xee, 0xff},
				Position: types.NewPosition(depth, big.NewInt(0)),
			},
		}
		expectedClaimInfo := ClaimInfo{
			AgreedPrestate: responseToSuper(prestateResponse).Marshal(),
			Claim:          postClaim.Value,
		}
		_, err := adapter(context.Background(), depth, types.Claim{}, postClaim)
		require.ErrorIs(t, err, creatorError)
		require.Equal(t, split.CreateLocalContext(types.Claim{}, postClaim), creator.localContext)
		require.Equal(t, expectedClaimInfo, creator.claimInfo)
	})

	t.Run("AfterClaimedBlock", func(t *testing.T) {
		creator, rootProvider, adapter := setupSplitAdapterTest(t, depth, prestateTimestamp, poststateTimestamp)
		prestateResponse := eth.SuperRootResponse{
			Timestamp: poststateTimestamp,
		}
		rootProvider.Add(prestateResponse)
		preClaim := types.Claim{
			ClaimData: types.ClaimData{
				Value:    common.Hash{0x11},
				Position: types.NewPosition(depth, big.NewInt(999_999)),
			},
		}
		postClaim := types.Claim{
			ClaimData: types.ClaimData{
				Value:    common.Hash{0xaa, 0xbb, 0xcc, 0xdd, 0xee, 0xff},
				Position: types.NewPosition(depth, big.NewInt(1_000_000)),
			},
		}
		expectedClaimInfo := ClaimInfo{
			AgreedPrestate: responseToSuper(prestateResponse).Marshal(),
			Claim:          postClaim.Value,
		}
		_, err := adapter(context.Background(), depth, preClaim, postClaim)
		require.ErrorIs(t, err, creatorError)
		require.Equal(t, split.CreateLocalContext(preClaim, postClaim), creator.localContext)
		require.Equal(t, expectedClaimInfo, creator.claimInfo)
	})

	t.Run("MiddleOfTimestampTransition", func(t *testing.T) {
		creator, rootProvider, adapter := setupSplitAdapterTest(t, depth, prestateTimestamp, poststateTimestamp)
		prestateResponse := eth.SuperRootResponse{
			Timestamp: prestateTimestamp,
		}
		rootProvider.Add(prestateResponse)
		rootProvider.Add(eth.SuperRootResponse{
			Timestamp: prestateTimestamp + 1,
		})
		preClaim := types.Claim{
			ClaimData: types.ClaimData{
				Value:    common.Hash{0x11},
				Position: types.NewPosition(depth, big.NewInt(2)),
			},
		}
		postClaim := types.Claim{
			ClaimData: types.ClaimData{
				Value:    common.Hash{0xaa, 0xbb, 0xcc, 0xdd, 0xee, 0xff},
				Position: types.NewPosition(depth, big.NewInt(3)),
			},
		}
		expectedPrestate := interopTypes.TransitionState{
			SuperRoot: responseToSuper(prestateResponse).Marshal(),
			Step:      3,
		}
		expectedClaimInfo := ClaimInfo{
			AgreedPrestate: expectedPrestate.Marshal(),
			Claim:          postClaim.Value,
		}
		_, err := adapter(context.Background(), depth, preClaim, postClaim)
		require.ErrorIs(t, err, creatorError)
		require.Equal(t, split.CreateLocalContext(preClaim, postClaim), creator.localContext)
		require.Equal(t, expectedClaimInfo, creator.claimInfo)
	})
}

func setupSplitAdapterTest(t *testing.T, depth types.Depth, prestateTimestamp uint64, poststateTimestamp uint64) (*capturingCreator, *stubRootProvider, split.ProviderCreator) {
	creator := &capturingCreator{}
	rootProvider := &stubRootProvider{}
	prestateProvider := NewSuperRootPrestateProvider(rootProvider, prestateTimestamp)
	traceProvider := NewSuperTraceProvider(testlog.Logger(t, log.LvlInfo), prestateProvider, rootProvider, eth.BlockID{}, depth, prestateTimestamp, poststateTimestamp)
	adapter := SuperRootSplitAdapter(traceProvider, creator.Create)
	return creator, rootProvider, adapter
}

type capturingCreator struct {
	localContext common.Hash
	claimInfo    ClaimInfo
}

func (c *capturingCreator) Create(_ context.Context, localContext common.Hash, _ types.Depth, claimInfo ClaimInfo) (types.TraceProvider, error) {
	c.localContext = localContext
	c.claimInfo = claimInfo
	return nil, creatorError
}