utils.go 8.97 KB
Newer Older
1 2 3
package withdrawals

import (
4
	"bytes"
5 6 7 8 9 10 11 12 13 14 15
	"context"
	"errors"
	"fmt"
	"math/big"

	"github.com/ethereum/go-ethereum/accounts/abi"
	"github.com/ethereum/go-ethereum/accounts/abi/bind"
	"github.com/ethereum/go-ethereum/common"
	"github.com/ethereum/go-ethereum/core/types"
	"github.com/ethereum/go-ethereum/crypto"
	"github.com/ethereum/go-ethereum/ethclient/gethclient"
16 17

	"github.com/ethereum-optimism/optimism/op-bindings/bindings"
18
	"github.com/ethereum-optimism/optimism/op-bindings/bindingspreview"
19
	"github.com/ethereum-optimism/optimism/op-bindings/predeploys"
20 21
)

22
var MessagePassedTopic = crypto.Keccak256Hash([]byte("MessagePassed(uint256,address,address,uint256,uint256,bytes,bytes32)"))
23

24 25 26 27
type ProofClient interface {
	GetProof(context.Context, common.Address, []string, *big.Int) (*gethclient.AccountResult, error)
}

28 29
type ReceiptClient interface {
	TransactionReceipt(context.Context, common.Hash) (*types.Receipt, error)
30 31
}

32 33 34 35
type BlockClient interface {
	BlockByNumber(context.Context, *big.Int) (*types.Block, error)
}

36 37 38
// ProvenWithdrawalParameters is the set of parameters to pass to the ProveWithdrawalTransaction
// and FinalizeWithdrawalTransaction functions
type ProvenWithdrawalParameters struct {
39 40 41 42 43
	Nonce           *big.Int
	Sender          common.Address
	Target          common.Address
	Value           *big.Int
	GasLimit        *big.Int
44
	L2OutputIndex   *big.Int
45
	Data            []byte
46
	OutputRootProof bindings.TypesOutputRootProof
47
	WithdrawalProof [][]byte // List of trie nodes to prove L2 storage
48 49
}

50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72
// ProveWithdrawalParameters calls ProveWithdrawalParametersForBlock with the most recent L2 output after the given header.
func ProveWithdrawalParameters(ctx context.Context, proofCl ProofClient, l2ReceiptCl ReceiptClient, l2BlockCl BlockClient, txHash common.Hash, header *types.Header, l2OutputOracleContract *bindings.L2OutputOracleCaller) (ProvenWithdrawalParameters, error) {
	l2OutputIndex, err := l2OutputOracleContract.GetL2OutputIndexAfter(&bind.CallOpts{}, header.Number)
	if err != nil {
		return ProvenWithdrawalParameters{}, fmt.Errorf("failed to get l2OutputIndex: %w", err)
	}
	l2BlockNumber := header.Number
	return ProveWithdrawalParametersForBlock(ctx, proofCl, l2ReceiptCl, l2BlockCl, txHash, l2BlockNumber, l2OutputIndex)
}

// ProveWithdrawalParametersFPAC calls ProveWithdrawalParametersForBlock with the most recent L2 output after the latest game.
func ProveWithdrawalParametersFPAC(ctx context.Context, proofCl ProofClient, l2ReceiptCl ReceiptClient, l2BlockCl BlockClient, txHash common.Hash, disputeGameFactoryContract *bindings.DisputeGameFactoryCaller, optimismPortal2Contract *bindingspreview.OptimismPortal2Caller) (ProvenWithdrawalParameters, error) {
	latestGame, err := FindLatestGame(ctx, disputeGameFactoryContract, optimismPortal2Contract)
	if err != nil {
		return ProvenWithdrawalParameters{}, fmt.Errorf("failed to find latest game: %w", err)
	}

	l2BlockNumber := new(big.Int).SetBytes(latestGame.ExtraData[0:32])
	l2OutputIndex := latestGame.Index
	return ProveWithdrawalParametersForBlock(ctx, proofCl, l2ReceiptCl, l2BlockCl, txHash, l2BlockNumber, l2OutputIndex)
}

// ProveWithdrawalParametersForBlock queries L1 & L2 to generate all withdrawal parameters and proof necessary to prove a withdrawal on L1.
73
// The header provided is very important. It should be a block (timestamp) for which there is a submitted output in the L2 Output Oracle
74
// contract. If not, the withdrawal will fail as it the storage proof cannot be verified if there is no submitted state root.
75
func ProveWithdrawalParametersForBlock(ctx context.Context, proofCl ProofClient, l2ReceiptCl ReceiptClient, l2BlockCl BlockClient, txHash common.Hash, l2BlockNumber, l2OutputIndex *big.Int) (ProvenWithdrawalParameters, error) {
76
	// Transaction receipt
77
	receipt, err := l2ReceiptCl.TransactionReceipt(ctx, txHash)
78
	if err != nil {
79
		return ProvenWithdrawalParameters{}, err
80 81
	}
	// Parse the receipt
82
	ev, err := ParseMessagePassed(receipt)
83
	if err != nil {
84
		return ProvenWithdrawalParameters{}, err
85 86 87
	}
	// Generate then verify the withdrawal proof
	withdrawalHash, err := WithdrawalHash(ev)
88
	if !bytes.Equal(withdrawalHash[:], ev.WithdrawalHash[:]) {
89
		return ProvenWithdrawalParameters{}, errors.New("Computed withdrawal hash incorrectly")
90
	}
91
	if err != nil {
92
		return ProvenWithdrawalParameters{}, err
93 94
	}
	slot := StorageSlotOfWithdrawalHash(withdrawalHash)
95 96 97

	// Fetch the block from the L2 node
	l2Block, err := l2BlockCl.BlockByNumber(ctx, l2BlockNumber)
98
	if err != nil {
99
		return ProvenWithdrawalParameters{}, fmt.Errorf("failed to get l2Block: %w", err)
100
	}
101 102

	p, err := proofCl.GetProof(ctx, predeploys.L2ToL1MessagePasserAddr, []string{slot.String()}, l2Block.Number())
103
	if err != nil {
104
		return ProvenWithdrawalParameters{}, err
105 106
	}
	if len(p.StorageProof) != 1 {
107
		return ProvenWithdrawalParameters{}, errors.New("invalid amount of storage proofs")
108 109
	}

110 111 112 113 114
	err = VerifyProof(l2Block.Root(), p)
	if err != nil {
		return ProvenWithdrawalParameters{}, err
	}

115 116 117 118 119 120
	// Encode it as expected by the contract
	trieNodes := make([][]byte, len(p.StorageProof[0].Proof))
	for i, s := range p.StorageProof[0].Proof {
		trieNodes[i] = common.FromHex(s)
	}

121
	return ProvenWithdrawalParameters{
122 123 124 125 126 127 128
		Nonce:         ev.Nonce,
		Sender:        ev.Sender,
		Target:        ev.Target,
		Value:         ev.Value,
		GasLimit:      ev.GasLimit,
		L2OutputIndex: l2OutputIndex,
		Data:          ev.Data,
129
		OutputRootProof: bindings.TypesOutputRootProof{
130
			Version:                  [32]byte{}, // Empty for version 1
131
			StateRoot:                l2Block.Root(),
132
			MessagePasserStorageRoot: p.StorageHash,
133
			LatestBlockhash:          l2Block.Hash(),
134
		},
135
		WithdrawalProof: trieNodes,
136 137 138
	}, nil
}

139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166
// FindLatestGame finds the latest game in the DisputeGameFactory contract.
func FindLatestGame(ctx context.Context, disputeGameFactoryContract *bindings.DisputeGameFactoryCaller, optimismPortal2Contract *bindingspreview.OptimismPortal2Caller) (*bindings.IDisputeGameFactoryGameSearchResult, error) {
	respectedGameType, err := optimismPortal2Contract.RespectedGameType(&bind.CallOpts{})
	if err != nil {
		return nil, fmt.Errorf("failed to get respected game type: %w", err)
	}

	gameCount, err := disputeGameFactoryContract.GameCount(&bind.CallOpts{})
	if err != nil {
		return nil, fmt.Errorf("failed to get game count: %w", err)
	}
	if gameCount.Cmp(common.Big0) == 0 {
		return nil, errors.New("no games")
	}

	searchStart := new(big.Int).Sub(gameCount, common.Big1)
	latestGames, err := disputeGameFactoryContract.FindLatestGames(&bind.CallOpts{}, respectedGameType, searchStart, common.Big1)
	if err != nil {
		return nil, fmt.Errorf("failed to get latest games: %w", err)
	}
	if len(latestGames) == 0 {
		return nil, errors.New("no latest games")
	}

	latestGame := latestGames[0]
	return &latestGame, nil
}

167 168 169 170 171 172 173
// Standard ABI types copied from golang ABI tests
var (
	Uint256Type, _ = abi.NewType("uint256", "", nil)
	BytesType, _   = abi.NewType("bytes", "", nil)
	AddressType, _ = abi.NewType("address", "", nil)
)

174 175
// WithdrawalHash computes the hash of the withdrawal that was stored in the L2toL1MessagePasser
// contract state.
176
// TODO:
177 178 179
//   - I don't like having to use the ABI Generated struct
//   - There should be a better way to run the ABI encoding
//   - These needs to be fuzzed against the solidity
180
func WithdrawalHash(ev *bindings.L2ToL1MessagePasserMessagePassed) (common.Hash, error) {
181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196
	//  abi.encode(nonce, msg.sender, _target, msg.value, _gasLimit, _data)
	args := abi.Arguments{
		{Name: "nonce", Type: Uint256Type},
		{Name: "sender", Type: AddressType},
		{Name: "target", Type: AddressType},
		{Name: "value", Type: Uint256Type},
		{Name: "gasLimit", Type: Uint256Type},
		{Name: "data", Type: BytesType},
	}
	enc, err := args.Pack(ev.Nonce, ev.Sender, ev.Target, ev.Value, ev.GasLimit, ev.Data)
	if err != nil {
		return common.Hash{}, fmt.Errorf("failed to pack for withdrawal hash: %w", err)
	}
	return crypto.Keccak256Hash(enc), nil
}

197
// ParseMessagePassed parses MessagePassed events from
198 199
// a transaction receipt. It does not support multiple withdrawals
// per receipt.
200
func ParseMessagePassed(receipt *types.Receipt) (*bindings.L2ToL1MessagePasserMessagePassed, error) {
201
	contract, err := bindings.NewL2ToL1MessagePasser(common.Address{}, nil)
202 203 204
	if err != nil {
		return nil, err
	}
205 206

	for _, log := range receipt.Logs {
207
		if len(log.Topics) == 0 || log.Topics[0] != MessagePassedTopic {
208
			continue
209
		}
210

211
		ev, err := contract.ParseMessagePassed(*log)
212 213
		if err != nil {
			return nil, fmt.Errorf("failed to parse log: %w", err)
214
		}
215
		return ev, nil
216
	}
217
	return nil, errors.New("Unable to find MessagePassed event")
218 219
}

220
// StorageSlotOfWithdrawalHash determines the storage slot of the L2ToL1MessagePasser contract to look at
221 222
// given a WithdrawalHash
func StorageSlotOfWithdrawalHash(hash common.Hash) common.Hash {
223
	// The withdrawals mapping is the 0th storage slot in the L2ToL1MessagePasser contract.
224 225 226 227 228 229
	// To determine the storage slot, use keccak256(withdrawalHash ++ p)
	// Where p is the 32 byte value of the storage slot and ++ is concatenation
	buf := make([]byte, 64)
	copy(buf, hash[:])
	return crypto.Keccak256Hash(buf)
}