diff --git a/.github/workflows/checks.yml b/.github/workflows/checks.yml index 538a74fcfb..51eaa337a9 100644 --- a/.github/workflows/checks.yml +++ b/.github/workflows/checks.yml @@ -17,7 +17,7 @@ jobs: - name: Set up Go uses: actions/setup-go@v3 with: - go-version: ^1.20 + go-version: 1.21.1 id: go - name: Check out code into the Go module directory @@ -32,6 +32,9 @@ jobs: - name: Lint run: make lint + - name: Lint contracts + run: cd suave && forge fmt --check + - name: Build contracts run: cd suave && forge build diff --git a/Makefile b/Makefile index 470296bb4d..3d9a8f2307 100644 --- a/Makefile +++ b/Makefile @@ -54,3 +54,6 @@ devnet-up: devnet-down: docker-compose -f ./suave/devenv/docker-compose.yml down + +fmt-contracts: + cd suave && forge fmt diff --git a/README.md b/README.md index 9720810c33..fcc3ebd77e 100644 --- a/README.md +++ b/README.md @@ -7,50 +7,12 @@ `suave-geth` is a work-in-progress Golang SUAVE client consisting of two separable components: chain nodes and execution nodes. SUAVE clients offer confidential execution for smart contracts, allowing confidential processing with extended precompiles for enhanced MEV functionalities, including transaction simulation via geth RPC, block building, and relay boosting, all handled by dedicated execution nodes. -For a deeper dive, check out the [technical details section](#suave-geth-technical-details), [simple MEV-share walk through](suave/cmd/suavecli/README.md), and the [demo video from EthCC](https://drive.google.com/file/d/1IHuLtxwjRvRpYjMG3oRuAgS5MUZtmAXq/view?usp=sharing). +For a deeper dive, check out the following links: ---- - -**Table of Contents** - -- [SUAVE](#suave) - - [Getting Started](#getting-started) - - [Starting a local devnet](#starting-a-local-devnet) - - [Building from source](#building-from-source) - - [Using Docker](#using-docker) - - [Testing the devnet](#testing-the-devnet) - - [SUAPP examples](#suapp-examples) - - [What do I use SUAVE for?](#what-do-i-use-suave-for) - - [How do I execute a contract confidentially?](#how-do-i-execute-a-contract-confidentially) - - [How do I run a SUAVE chain node?](#how-do-i-run-a-suave-chain-node) - - [How do I run a SUAVE execution node?](#how-do-i-run-a-suave-execution-node) - - [suave-geth technical details](#suave-geth-technical-details) - - [SUAVE Runtime (MEVM)](#suave-runtime-mevm) - - [Confidential execution of smart contracts](#confidential-execution-of-smart-contracts) - - [Confidential compute requests](#confidential-compute-requests) - - [SUAVE Bids](#suave-bids) - - [SUAVE library](#suave-library) - - [Confidential APIs](#confidential-apis) - - [Confidential Store](#confidential-store) - - [SUAVE Mempool](#suave-mempool) - - [Notable differences from standard issue go-ethereum](#notable-differences-from-standard-issue-go-ethereum) - - [Changes to RPC methods](#changes-to-rpc-methods) - - [SuavePrecompiledContract](#suaveprecompiledcontract) - - [SUAVE precompile wrapper](#suave-precompile-wrapper) - - [SuaveExecutionBackend](#suaveexecutionbackend) - - [EVM Interpreter](#evm-interpreter) - - [Basic Eth block building RPC](#basic-eth-block-building-rpc) - - [SUAVE precompiles](#suave-precompiles) - - [IsConfidential](#isconfidential) - - [ConfidentialInputs](#confidentialinputs) - - [ConfidentialStore](#confidentialstore) - - [ConfidentialRetrieve](#confidentialretrieve) - - [NewBid](#newbid) - - [FetchBids](#fetchbids) - - [SimulateBundle](#simulatebundle) - - [ExtractHint](#extracthint) - - [BuildEthBlock](#buildethblock) - - [SubmitEthBlockBidToRelay](#submitethblockbidtorelay) +- [Suave Specs](https://github.com/flashbots/suave-specs) +- [Simple MEV-share walk through](suave/cmd/suavecli/README.md) +- [Demo video from EthCC](https://drive.google.com/file/d/1IHuLtxwjRvRpYjMG3oRuAgS5MUZtmAXq/view?usp=sharing). +- [Suapp Examples](https://github.com/flashbots/suapp-examples) --- @@ -108,546 +70,7 @@ Execute a RPC request with curl like this: $ curl 'http://localhost:8545' --header 'Content-Type: application/json' --data '{ "jsonrpc":"2.0", "method":"eth_blockNumber", "params":[], "id":83 }' ``` -#### SUAPP examples - -See https://github.com/flashbots/suapp-examples 🚀 - ---- - -### What do I use SUAVE for? - -1. **Deploy confidential smart contracts.** - Smart contracts on SUAVE follow the same rules as on Ethereum with the added advantage of being able to access additional precompiles during confidential execution. Precompiles are available through the [SUAVE library](#suave-library). - -2. **NEW! Request confidential execution using the new confidential computation request.** - Contracts called using confidential compute requests have access to confidential data and APIs through SUAVE precompiles. Confidential computation is *not* reproducible on-chain, thus, users are required to whitelist a specific execution node trusted to provide the result. Eventually proofs and trusted enclaves will help to verify the results of execution. - After the initial confidential computation, its result replaces the calldata for on-chain execution. This grants different behaviors to confidential computation and regular on-chain transactions since confidential APIs are inaccessible during regular chain state transition. - - - See [confidential compute requests](#confidential-compute-requests) for more details. - -### How do I execute a contract confidentially? - -Let’s take a look at how you can request confidential computation through an execution node. - -1. Pick your favorite execution node. You’ll need its URL and wallet address. Note that the execution node is fully trusted to provide the result of your confidential computation. - -2. Craft your confidential computation record. This is a regular Ethereum transaction (fields are similar to `LegacyTx`), where you specify the desired contract address and its (public) calldata. I’m assuming you have found or deployed a smart contract which you intend to call. Don’t sign the transaction quite yet! - - ```go - allowedPeekers := []common.Address{newBlockBidPeeker, newBundleBidPeeker, buildEthBlockPeeker} // express which contracts should have access to your data (by their addresses) - confidentialComputeRecord := &types.ConfidentialComputeRecord{ - KettleAddress: "0x4E2B0c0e428AE1CDE26d5BcF17Ba83f447068E5B", - Nonce: suaveAccNonce, - To: &newBundleBidAddress, - Value: nil, - Gas: 1000000, - GasPrice: 50, - Data: bundleBidAbi.Pack("newBid", targetBlock, allowedPeekers) - } - ``` - -3. Wrap your compute record into a `ConfidentialComputeRequest` transaction type, and specify the confidential data. - - ```go - confidentialDataBytes := hexutil.Encode(ethBundle) - confidentialComputeRequest := types.SignTx(types.NewTx(&types.ConfidentialComputeRequest{ - ConfidentialComputeRecord: confidentialComputeRecord, - ConfidentialInputs: confidentialDataBytes, - }), suaveSigner, privKey) - ``` - -4. Request confidential computation by submitting your transaction to the execution node you chose via `eth_sendRawTransaction`. - - ```go - suaveClient.Call("eth_sendRawTransaction", confidentialComputeRequest) - ``` - -5. All done! Once the execution node processes your computation request, the execution node will submit it as `SuaveTransaction` to the mempool. - -For more on confidential compute requests see [confidential compute requests](#confidential-compute-requests). - -### How do I run a SUAVE chain node? - -1. Build the client with `make geth`. -2. Run the node. Pass in `--dev` to enable local devnet. Example: - - ```go - ./build/bin/geth --dev --dev.gaslimit 30000000 --datadir suave_dev --http --ws --allow-insecure-unlock --unlock "0xd52d1935D1239ADf94C59fA0F586fE00250694d5" - ``` - -3. Do your thing! - -### How do I run a SUAVE execution node? - -Not all nodes serve confidential compute requests. You’ll need: -- A SUAVE node (see above). -- An account. If you are doing this for testing, simply run `geth --suave account new`. Take note of the address. -- Access to Ethereum’s RPC. When starting your node, pass in `--suave.eth.remote_endpoint` to point to your Ethereum RPC. - ```go - ./build/bin/geth --dev --dev.gaslimit 30000000 --datadir suave_dev --http --allow-insecure-unlock --unlock "0x" --ws --suave.eth.remote_endpoint "http://" - ``` -Note that simply enabling http jsonrpc and allowing direct access might not be the wisest. Look into proxyd and other restricted access solutions. - -## suave-geth technical details - -### SUAVE Runtime (MEVM) - -[`SuaveExecutionBackend`](#suaveexecutionbackend) 🤝 EVM = MEVM - -More specifically, `SuaveExecutionBackend` and `Runtime` add functionality to the stock EVM which allows it both confidential computation and interaction with APIs. - -```mermaid -graph TB - A[EVM]-->|1|B((StateDB)) - A-->|2|C((Context)) - A-->|3|D((chainConfig)) - A-->|4|E((Config)) - A-->|5|F((interpreter)) - D-->|6|R[ChainRules] - E-->|7|S[Tracer] - A-->|8|T[NewRuntime] - T-->|9|Z((Runtime)) - Z-->|10|F - A-->|11|U[NewRuntimeSuaveExecutionBackend] - U-->|12|V((SuaveExecutionBackend)) - V-->|13|F - class A,B,C,D,E,F yellow - class G,H,I,J,K,L,M,N,O red - class P,Q green - class R blue - class S orange - class T,U purple - class Z,V lightgreen - classDef yellow fill:#f5cf58,stroke:#444,stroke-width:2px, color:#333; - classDef red fill:#d98686,stroke:#444,stroke-width:2px, color:#333; - classDef green fill:#82a682,stroke:#444,stroke-width:2px, color:#333; - classDef blue fill:#9abedc,stroke:#444,stroke-width:2px, color:#333; - classDef orange fill:#f3b983,stroke:#444,stroke-width:2px, color:#333; - classDef purple fill:#ab92b5,stroke:#444,stroke-width:2px, color:#333; - classDef lightgreen fill:#b3c69f,stroke:#444,stroke-width:2px, color:#333; -``` - -The capabilities enabled by this modified runtime are exposed to the virtual machine via `SuaveContext` and its components. - -```go -type SuaveContext struct { - Backend *SuaveExecutionBackend - ConfidentialComputeRequestTx *types.Transaction - ConfidentialInputs []byte - CallerStack []*common.Address -} - -type SuaveExecutionBackend struct { - ConfidentialStoreEngine *suave.ConfidentialStoreEngine - MempoolBackend suave.MempoolBackend - ConfidentialEthBackend suave.ConfidentialEthBackend -} -``` - -All of these newly offered APIs are available to your solidity smart contract through the use of precompiles! See below for how confidential computation and smart contracts interact. - -### Confidential execution of smart contracts - -The virtual machine (MEVM) inside SUAVE nodes have two modes of operation: regular and confidential. Regular on-chain environment is your usual Ethereum virtual machine environment. - -Confidential environment is available to users through a new type of transaction - `ConfidentialComputeRequest` via the usual jsonrpc methods `eth_sendRawTransaction`, `eth_sendTransaction` and `eth_call`. Simulations (`eth_call`) requested with a new optional argument `IsConfidential` are also executed in the confidential mode. For more on confidential requests see [confidential compute requests](#confidential-compute-requests). - -The confidential execution environment provides additional precompiles, both directly and through a convenient [library](#suave-library). Confidential execution is *not* verifiable during on-chain state transition, instead the result of the confidential execution is cached in the transaction (`SuaveTransaction`). Users requesting confidential compute requests specify which execution nodes they trust with execution, and the execution nodes' signature is used for verifying the transaction on-chain. - -The cached result of confidential execution is used as calldata in the transaction that inevitably makes its way onto the SUAVE chain. - -Other than ability to access new precompiles, the contracts aiming to be executed confidentially are written as usual in Solidity (or any other language) and compiled to EVM bytecode. - -### Confidential compute requests - -We introduce a few new transaction types. - -* `ConfidentialComputeRecord` - - This type serves as an onchain record of computation. It's a part of both the [Confidential Compute Request](#confidential-compute-request) and [Suave Transaction](#suave-transaction). - - ```go - type ConfidentialComputeRecord struct { - KettleAddress common.Address - ConfidentialInputsHash common.Hash - - // LegacyTx fields - Nonce uint64 - GasPrice *big.Int - Gas uint64 - To *common.Address `rlp:"nil"` - Value *big.Int - Data []byte - - // Signature fields - } - ``` - -* `ConfidentialComputeRequest` - - This type facilitates users in interacting with the MEVM through the `eth_sendRawTransaction` method. After processing, the request's `ConfidentialComputeRecord` is embedded into `SuaveTransaction.ConfidentialComputeRequest` and serves as an onchain record of computation. - - ```go - type ConfidentialComputeRequest struct { - ConfidentialComputeRecord - ConfidentialInputs []byte - } - ``` - -* `SuaveTransaction` - - A specialized transaction type that encapsulates the result of a confidential computation request. It includes the `ConfidentialComputeRequest`, signed by the user, which ensures that the result comes from the expected computor, as the `SuaveTransaction`'s signer must match the `KettleAddress`. - - ```go - type SuaveTransaction struct { - KettleAddress common.Address - ConfidentialComputeRequest ConfidentialComputeRecord - ConfidentialComputeResult []byte - /* Execution node's signature fields */ - } - ``` - -![image](suave/docs/conf_comp_request_flow.png) - - -The basic flow is as follows: - -1. User crafts a confidential computation request (`ConfidentialComputeRequest`): - 1. Sets their GasPrice, GasLimit, To address and calldata as they would for a `LegacyTx` - 2. Choses an execution node of their liking, that is an address whose signature over the confidential computation result will be trusted - 3. The above becomes the `ConfidentialComputeRecord` that will eventually make its way onto the chain - 4. Sets the `ConfidentialInputs` of the request (if any) - 5. Signs and sends the confidential computation request (consisting of (3 and 4) to an execution node via `eth_sendRawTransaction` -2. The execution node executes the transaction in the confidential mode, providing access to the usual confidential APIs -3. The execution node creates a `SuaveTransaction` using the confidential computation request and the result of its execution, the node then signs and submits the transaction into the mempool -4. The transaction makes its way into a block, by executing the `ConfidentialComputeResult` as calldata, as long as the execution node's signature matches the requested executor node in (1.2.) - -The initial confidential computation has access to both the public and confidential data, but only the public data becomes part of the transaction propagated through the mempool. Any confidential data passed in by the user is discarded after the execution. - -Architecture reference -![image](suave/docs/execution_node_architecture.png) - -Mind, that the results are not reproducible as they are based on confidential data that is dropped after execution. On-chain state transition only depends on the result of the confidential computation, so it is fully reproducible. - -### SUAVE Bids - -On the SUAVE chain, bids serve as the primary transaction unit, and are used for interactions between smart contracts and the Confidential Store. - -A `Bid` is a data structure encapsulating key information about a transaction on the SUAVE chain. - -```go -type Bid struct { - Id BidId `json:"id"` - Salt BidId `json:"salt"` - DecryptionCondition uint64 `json:"decryptionCondition"` - AllowedPeekers []common.Address `json:"allowedPeekers"` - AllowedStores []common.Address `json:"allowedStores"` - Version string `json:"version"` -} -``` - -Each `Bid` has a `Id` (uuid v5), a `Salt` (random uuid v4), a `DecryptionCondition`, an array of `AllowedPeekers` and `AllowedStores`, and a `Version`. The `DecryptionCondition` signifies the block number at which the bid can be decrypted and is typically derived from the source contract or may even be a contract itself. The `AllowedPeekers` are the addresses that are permitted to access the data associated with the bid, providing an added layer of access control. The `AllowedStores` are the confidential stores which should be granted access to the bid's data (currently not enforced). The `Version` indicates the version of the protocol used for the bid. - -### SUAVE library - -Along with the SUAVE precompiles, we provide a convenient wrapper for calling them from Solidity. The [library](suave/sol/libraries/Suave.sol) makes the precompiles easier to call by providing the signatures, and the library functions themselves simply perform a `staticcall` of the requested precompile. - -```solidity -library Suave { - error PeekerReverted(address, bytes); - - type BidId is bytes16; - - struct Bid { - BidId id; - BidId salt; - uint64 decryptionCondition; - address[] allowedPeekers; - address[] allowedStores; - string version; - } - - function isConfidential() internal view returns (bool b) - function confidentialInputs() internal view returns (bytes memory) - function newBid(uint64 decryptionCondition, address[] memory allowedPeekers, string memory BidType) internal view returns (Bid memory) - function fetchBids(uint64 cond, string memory namespace) internal view returns (Bid[] memory) - function confidentialStoreStore(BidId bidId, string memory key, bytes memory data) internal view - function confidentialStoreRetrieve(BidId bidId, string memory key) internal view returns (bytes memory) - function simulateBundle(bytes memory bundleData) internal view returns (bool, uint64) - function extractHint(bytes memory bundleData) internal view returns (bytes memory) - function buildEthBlock(BuildBlockArgs memory blockArgs, BidId bid, string memory namespace) internal view returns (bytes memory, bytes memory) - function submitEthBlockBidToRelay(string memory relayUrl, bytes memory builderBid) internal view returns (bool, bytes memory) -} -``` - -### Confidential APIs - -Confidential precompiles have access to the following [Confidential APIs](suave/core/types.go) during execution. -This is subject to change! - -```go -type ConfidentialStoreEngine interface { - Initialize(bid Bid, creationTx *types.Transaction, key string, value []byte) (Bid, error) - Store(bidId BidId, sourceTx *types.Transaction, caller common.Address, key string, value []byte) (Bid, error) - Retrieve(bid BidId, caller common.Address, key string) ([]byte, error) -} - -type MempoolBackend interface { - SubmitBid(Bid) error - FetchBidById(BidId) (Bid, error) - FetchBidsByProtocolAndBlock(blockNumber uint64, namespace string) []Bid -} - -type ConfidentialEthBackend interface { - BuildEthBlock(ctx context.Context, args *BuildBlockArgs, txs types.Transactions) (*engine.ExecutionPayloadEnvelope, error) - BuildEthBlockFromBundles(ctx context.Context, args *BuildBlockArgs, bundles []types.SBundle) (*engine.ExecutionPayloadEnvelope, error) -} -``` - -### Confidential Store - -The Confidential Store is an integral part of the SUAVE chain, designed to facilitate secure and privacy-preserving transactions and smart contract interactions. It functions as a key-value store where users can safely store and retrieve confidential data related to their bids. The Confidential Store restricts access (both reading and writing) only to the allowed peekers of each bid, allowing developers to define the entire data model of their application! - -The current, and certainly not final, implementation of the Confidential Store is managed by the `ConfidentialStoreEngine`. The engine consists of a storage backend, which holds the raw data, and a transport topic, which relays synchronization messages between nodes. -We provide two storage backends to the confidential store engine: the `LocalConfidentialStore`, storing data in memory in a simple dictionary, and `RedisStoreBackend`, storing data in redis. To enable redis as the storage backed, pass redis endpoint via `--suave.confidential.redis-store-endpoint`. -For synchronization of confidential stores via transport we provide an implementation using a shared Redis PubSub in `RedisPubSubTransport`, as well as a *crude* synchronization protocol. To enable redis transport, pass redis endpoint via `--suave.confidential.redis-transport-endpoint`. Note that Redis transport only synchronizes *current* state, there is no initial synchronization - a newly connected node will not have access to old data. -Redis as either storage backend or transport is *temporary* and will be removed once we have a well-tested p2p solution. - -![image](suave/docs/confidential_store_engine.png) - -The `ConfidentialStoreEngine` provides the following key methods: - -- **Initialize**: This method is used to initialize a bid. The method is trusted, meaning it is not directly accessible through precompiles. The method returns the initialized bid, importantly with the `Id` field set. -- **Store**: This method stores a given value under a specified key in a bid's `dataMap`. Access is restricted only to addresses listed in the bid's `AllowedPeekers`. -- **Retrieve**: This method retrieves data associated with a given key from a bid's `dataMap`. Similar to the `Store` method, access is restricted only to addresses listed in the bid's `AllowedPeekers`. - -It is important to note that the actual implementation of the Confidential Store will vary depending on future requirements and the privacy mechanisms used. - -### SUAVE Mempool - -The SUAVE mempool is a temporary storage pool for transactions waiting to be added to the blockchain. This mempool, `MempoolOnConfidentialStore`, operates on the Confidential Store, hence facilitating the privacy-preserving handling of bid transactions. The `MempoolOnConfidentialStore` is designed to handle SUAVE bids, namely the submission, retrieval, and grouping of bids by decryption condition such as block number and protocol. It provides a secure and efficient mechanism for managing these transactions while preserving their confidentiality. - -The `MempoolOnConfidentialStore` interacts directly with the `ConfidentialStoreBackend` interface. - -```go -type MempoolOnConfidentialStore struct { - cs suave.ConfidentialStoreBackend -} -``` -It is initialized with a predefined `mempoolConfidentialStoreBid` that's only accessible by a particular address `mempoolConfStoreAddr`. - -```go -mempoolConfidentialStoreBid = suave.Bid{Id: mempoolConfStoreId, AllowedPeekers: []common.Address{mempoolConfStoreAddr}} -``` -The `MempoolOnConfidentialStore` includes three primary methods: - -- **SubmitBid**: This method submits a bid to the mempool. The bid is stored in the Confidential Store with its ID as the key. Additionally, the bid is grouped by block number and protocol, which are also stored in the Confidential Store. - -- **FetchBidById**: This method retrieves a bid from the mempool using its ID. - -- **FetchBidsByProtocolAndBlock**: This method fetches all bids from a particular block that match a specified protocol. - -The mempool operates on the underlying Confidential Store, thereby maintaining the confidentiality of the bids throughout the transaction process. As such, all data access is subject to the Confidential Store's security controls, ensuring privacy and integrity. Please note that while this initial implementation provides an idea of the ideal functionality, the final version will most likely incorporate additional features or modifications. - -## Notable differences from standard issue go-ethereum - -### Changes to RPC methods - -1. New `IsConfidential` and `KettleAddress` fields are added to TransactionArgs, used in `eth_sendTransaction` and `eth_call` methods. -If `IsConfidential` is set to true, the call will be performed as a confidential call, using the `KettleAddress` passed in for constructing `ConfidentialComputeRequest`. -`SuaveTransaction` is the result of `eth_sendTransaction`! - -2. New optional argument - `confidential_data` is added to `eth_sendRawTransaction`, `eth_sendTransaction` and `eth_call` methods. -The confidential data is made available to the EVM in the confidential mode via a precompile, but does not become a part of the transaction that makes it to chain. This allows performing computation based on confidential data (like simulating a bundle, putting the data into confidential store). - - -### SuavePrecompiledContract - -We introduce a new interface [SuavePrecompiledContract](core/vm/contracts.go) for SUAVE precompiles. - -``` -type SuavePrecompiledContract interface { - PrecompiledContract - RunConfidential(backend *SuaveExecutionBackend, input []byte) ([]byte, error) -} -``` - -The method `RunConfidential` is invoked during confidential execution, and the suave execution backend which provides access to confidential APIs is passed in as input. - -### SUAVE precompile wrapper - -We introduce [SuavePrecompiledContractWrapper](core/vm/suave.go) implementing the `PrecompiledContract` interface. The new structure captures the confidential APIs in its constructor, and passes the confidential APIs during the usual contract's `Run` method to a separate method - `RunConfidential` - - -### SuaveExecutionBackend - -We introduce [SuaveExecutionBackend](core/vm/suave.go), which allows access to confidential capabilities during execution: -* Access to confidential APIs -* Access to confidential input -* Caller stack tracing - -The backend is only available to confidential execution! - -### EVM Interpreter - -The [EVM interpreter](core/vm/interpreter.go) is modified to allow for confidential computation's needs: -* We introduce `IsConfidential` to the interpreter's config -* We modify the `Run` function to accept confidential APIs `func (in *EVMInterpreter) Run(*SuaveExecutionBackend, *Contract, []byte, bool) ([]byte, err)` -* We modify the `Run` function to trace the caller stack - - -Like `eth_sendTransaction`, this method accepts an additional, optional confidential inputs argument. - - -### Basic Eth block building RPC - -We implement two rpc methods that allow building Ethereum blocks from a list of either transactions or bundles: `BuildEth2Block` and `BuildEth2BlockFromBundles`. - -These methods are defined in [BlockChainAPI](internal/ethapi/api.go) - -```go -func (s *BlockChainAPI) BuildEth2Block(ctx context.Context, buildArgs *types.BuildBlockArgs, txs types.Transactions) (*engine.ExecutionPayloadEnvelope, error) -func (s *BlockChainAPI) BuildEth2BlockFromBundles(ctx context.Context, buildArgs *types.BuildBlockArgs, bundles []types.SBundle) (*engine.ExecutionPayloadEnvelope, error) - -``` - -The methods are implemented in [worker](miner/worker.go), by `buildBlockFromTxs` and `buildBlockFromBundles` respectively. - -`buildBlockFromTxs` will simply build a block out of the transactions provided, while `buildBlockFromBundles` will in addition forward the block profit to the requested fee recipient, as needed for boost relay payments. - - -## SUAVE precompiles - -Additional precompiles available via the EVM. -Only `IsConfidential` is available during on-chain execution, and simply returns false. - -For details and implementation see [contracts_suave.go](core/vm/contracts_suave.go) - -### IsConfidential - -| | | -| ------- | ------------ | -| Address | `0x42010000` | -| Inputs | None | -| Outputs | boolean | - -Outputs whether execution mode is regular (on-chain) or confidential. - - -### ConfidentialInputs - -| | | -| ------- | ------------ | -| Address | `0x42010001` | -| Inputs | None | -| Outputs | bytes | - -Outputs the confidential inputs passed in with the confidential computation request. - - -NOTE: currently all precompiles have access to the data passed in. This might change in the future. - -### ConfidentialStore - -| | | -| ------- | ------------------------------------------- | -| Address | `0x42020000` | -| Inputs | (Suave.BidId bidId, string key, bytes data) | -| Outputs | None | - - -Stores the value in underlying confidential store. -Requires that the caller is present in the `AllowedPeekers` of the bid passed in! - -### ConfidentialRetrieve - -| | | -| ------- | ------------------------------- | -| Address | `0x42020001` | -| Inputs | (Suave.BidId bidId, string key) | -| Outputs | bytes | - - -Retrieves the value from underlying confidential store. -Requires that the caller is present in the `AllowedPeekers` of the bid passed in! - - -### NewBid - - -| | | -| ------- | ----------------------------------------------------- | -| Address | `0x42030000` | -| Inputs | (uint64 decryptionCondition, string[] allowedPeekers) | -| Outputs | Suave.Bid | - -Initializes the bid in ConfidentialStore. All bids must be initialized before attempting to store data on them. -Initialization of bids can *only* be done through this precompile! - -### FetchBids - -| | | -| ------- | -------------------------- | -| Address | `0x42030001` | -| Inputs | uint64 DecryptionCondition | -| Outputs | Suave.Bid[] | - -Returns all bids matching the decryption condition. -This method is subject to change! In the near future bids will be stored in a different way, possibly changing how they are accessed. - -### SimulateBundle - -| | | -| ------- | -------------------------- | -| Address | `0x42100000` | -| Inputs | bytes bundleArgs (json) | -| Outputs | (bool success, uint64 egp) | - -Simulates the bundle by building a block containing it, returns whether the apply was successful and the EGP of the resulting block. - -### ExtractHint - -| | | -| ------- | ----------------------- | -| Address | `0x42100037` | -| Inputs | bytes bundleData (json) | -| Outputs | bytes hintData (json) | - -Parses the bundle data and extracts the hint - "To" address and the calldata. - -The return structure is encoded as follows: -``` -struct { - To common.Address - Data []byte -} -``` - - -### BuildEthBlock - -| | | -| ------- | --------------------------------------------------- | -| Address | `0x42100001` | -| Inputs | (Suave.BuildBlockArgs blockArgs, Suave.BidId bidId) | -| Outputs | (bytes builderBid, bytes blockPayload) | - - -Builds an Ethereum block based on the bid passed in. -The bid can either hold `ethBundle` in its confidential store, or be a "merged bid", ie contain a list of bids in `mergedBids` in its confidential store. The merged bids should themselves hold `ethBundle`. -The block is built *in order*, without any attepmts at re-ordering. The block will contain the transactions unless they failed to apply. The caller should check whether the bids applied successfully, ie whether they revert only if are allowed to. - -### SubmitEthBlockBidToRelay - -| | | -| ------- | ----------------------------------------- | -| Address | `0x42100002` | -| Inputs | (string relayUrl, bytes builderBid (json) | -| Outputs | (bytes error) | - -Submits provided builderBid to a boost relay. If the submission is successful, returns nothing, otherwise returns an error string. - ---- +## What next? -Made with ☀️ by the ⚡🤖 collective. +- [suapp-examples](https://github.com/flashbots/suapp-examples) is a collection of example SUAVE apps and boilerplate to get started quickly and right-footed. +- [suave-specs](https://github.com/flashbots/suave-specs) is the spec repository for SUAVE which contains all the technical documentation. \ No newline at end of file diff --git a/accounts/abi/type2.go b/accounts/abi/type2.go new file mode 100644 index 0000000000..4c3b1d10d1 --- /dev/null +++ b/accounts/abi/type2.go @@ -0,0 +1,123 @@ +package abi + +import ( + "fmt" + "reflect" + "strings" +) + +func NewTypeFromString(s string) (Type, error) { + if strings.HasPrefix(s, "tuple") { + sig, args, err := newTypeForTuple(s) + if err != nil { + return Type{}, err + } + return NewType(sig, "", args) + } + return NewType(s, "", nil) +} + +func (t Type) Pack(v interface{}) ([]byte, error) { + return t.pack(reflect.ValueOf(v)) +} + +func (t Type) Unpack(data []byte, obj interface{}) error { + fmt.Println(toGoType(0, t, data)) + + return nil +} + +// newTypeForTuple implements the format described in https://blog.ricmoo.com/human-readable-contract-abis-in-ethers-js-141902f4d917 +func newTypeForTuple(s string) (string, []ArgumentMarshaling, error) { + if !strings.HasPrefix(s, "tuple") { + return "", nil, fmt.Errorf("'tuple' prefix not found") + } + + pos := strings.Index(s, "(") + if pos == -1 { + return "", nil, fmt.Errorf("not a tuple, '(' not found") + } + + // this is the type of the tuple. It can either be + // tuple, tuple[] or tuple[x] + sig := s[:pos] + s = s[pos:] + + // Now, decode the arguments of the tuple + // tuple(arg1, arg2, tuple(arg3, arg4)). + // We need to find the commas that are not inside a nested tuple. + // We do this by keeping a counter of the number of open parens. + + var ( + parenthesisCount int + fields []string + ) + + lastComma := 1 + for indx, c := range s { + switch c { + case '(': + parenthesisCount++ + case ')': + parenthesisCount-- + if parenthesisCount == 0 { + fields = append(fields, s[lastComma:indx]) + + // this should be the end of the tuple + if indx != len(s)-1 { + return "", nil, fmt.Errorf("invalid tuple, it does not end with ')'") + } + } + case ',': + if parenthesisCount == 1 { + fields = append(fields, s[lastComma:indx]) + lastComma = indx + 1 + } + } + } + + // trim the args of spaces + for i := range fields { + fields[i] = strings.TrimSpace(fields[i]) + } + + // decode the type of each field + var args []ArgumentMarshaling + for _, field := range fields { + // anonymous fields are not supported so the first + // string should be the identifier of the field. + + spacePos := strings.Index(field, " ") + if spacePos == -1 { + return "", nil, fmt.Errorf("invalid tuple field name not found '%s'", field) + } + + name := field[:spacePos] + field = field[spacePos+1:] + + if strings.HasPrefix(field, "tuple") { + // decode a recursive tuple + sig, elems, err := newTypeForTuple(field) + if err != nil { + return "", nil, err + } + args = append(args, ArgumentMarshaling{ + Name: name, + Type: sig, + Components: elems, + }) + } else { + // basic type. Try to decode it to see + // if it is a correct abi type. + if _, err := NewType(field, "", nil); err != nil { + return "", nil, fmt.Errorf("invalid tuple basic field '%s': %v", field, err) + } + args = append(args, ArgumentMarshaling{ + Name: name, + Type: field, + }) + } + } + + return sig, args, nil +} diff --git a/accounts/abi/type2_test.go b/accounts/abi/type2_test.go new file mode 100644 index 0000000000..2dd7e1cef4 --- /dev/null +++ b/accounts/abi/type2_test.go @@ -0,0 +1,94 @@ +package abi + +import ( + "testing" + + "github.com/stretchr/testify/require" +) + +func TestNewType2FromString(t *testing.T) { + type struct1 struct { + A uint64 + B []struct { + X struct { + C uint64 + } + D uint64 + } + E struct { + F uint64 + G uint64 + } + } + + cases := []struct { + input string + want string + obj interface{} + }{ + { + "uint64[]", + "uint64[]", + []uint64{1, 2, 3}, + }, + { + "tuple(a uint64, b uint64)", + "(uint64,uint64)", + &struct { + A uint64 + B uint64 + }{A: 1, B: 2}, + }, + { + "tuple(a uint64, b tuple(c uint64), d uint64)", + "(uint64,(uint64),uint64)", + &struct { + A uint64 + B struct { + C uint64 + } + D uint64 + }{A: 1, B: struct{ C uint64 }{C: 2}, D: 3}, + }, + { + "tuple(a uint64, b tuple[](x tuple(c uint64), d uint64), e tuple(f uint64, g uint64))", + "(uint64,((uint64),uint64)[],(uint64,uint64))", + &struct1{ + A: 1, + B: []struct { + X struct { + C uint64 + } + D uint64 + }{ + { + X: struct { + C uint64 + }{C: 2}, + D: 3, + }, + { + X: struct { + C uint64 + }{C: 4}, + D: 5, + }, + }, + E: struct { + F uint64 + G uint64 + }{F: 6, G: 7}, + }, + }, + } + + for _, c := range cases { + typ, err := NewTypeFromString(c.input) + + require.NoError(t, err) + require.Equal(t, c.want, typ.String()) + + _, err = typ.Pack(c.obj) + require.NoError(t, err) + } +} diff --git a/cmd/geth/main.go b/cmd/geth/main.go index 42266d5bf4..e71a162df4 100644 --- a/cmd/geth/main.go +++ b/cmd/geth/main.go @@ -272,6 +272,16 @@ func main() { // prepare manipulates memory cache allowance and setups metric system. // This function should be called before launching devp2p stack. func prepare(ctx *cli.Context) { + if err := prepareSuaveDev(ctx); err != nil { + log.Error("failed to setup suave dev mode", "err", err) + os.Exit(1) + } + + if err := prepareSuaveNetworksRemapping(ctx); err != nil { + log.Error("failed to setup suave networks remapping", "err", err) + os.Exit(1) + } + // If we're running a known preset, log it for convenience. switch { case ctx.IsSet(utils.RinkebyFlag.Name): @@ -341,14 +351,6 @@ func geth(ctx *cli.Context) error { return fmt.Errorf("invalid command: %q", args[0]) } - if err := prepareSuaveDev(ctx); err != nil { - return fmt.Errorf("failed to setup suave development mode: %v", err) - } - - if err := prepareSuaveNetworksRemapping(ctx); err != nil { - return err - } - prepare(ctx) stack, backend := makeFullNode(ctx) defer stack.Close() @@ -532,7 +534,6 @@ func prepareSuaveDev(ctx *cli.Context) error { utils.DeveloperFlag.Name: "true", utils.DeveloperGasLimitFlag.Name: "30000000", utils.HTTPEnabledFlag.Name: "true", - utils.HTTPPortFlag.Name: "8545", utils.HTTPVirtualHostsFlag.Name: "*", utils.HTTPCORSDomainFlag.Name: "*", utils.WSEnabledFlag.Name: "true", diff --git a/core/types/sbundle.go b/core/types/sbundle.go index cbe2dbc2a5..c88dd4f3bc 100644 --- a/core/types/sbundle.go +++ b/core/types/sbundle.go @@ -12,6 +12,7 @@ import ( type SBundle struct { BlockNumber *big.Int `json:"blockNumber,omitempty"` // if BlockNumber is set it must match DecryptionCondition! + MaxBlock *big.Int `json:"maxBlock,omitempty"` Txs Transactions `json:"txs"` RevertingHashes []common.Hash `json:"revertingHashes,omitempty"` RefundPercent *int `json:"percent,omitempty"` @@ -19,6 +20,7 @@ type SBundle struct { type RpcSBundle struct { BlockNumber *hexutil.Big `json:"blockNumber,omitempty"` + MaxBlock *hexutil.Big `json:"maxBlock,omitempty"` Txs []hexutil.Bytes `json:"txs"` RevertingHashes []common.Hash `json:"revertingHashes,omitempty"` RefundPercent *int `json:"percent,omitempty"` @@ -66,6 +68,7 @@ func (s *SBundle) UnmarshalJSON(data []byte) error { } s.BlockNumber = (*big.Int)(rpcSBundle.BlockNumber) + s.MaxBlock = (*big.Int)(rpcSBundle.MaxBlock) s.Txs = txs s.RevertingHashes = rpcSBundle.RevertingHashes s.RefundPercent = rpcSBundle.RefundPercent @@ -76,7 +79,8 @@ func (s *SBundle) UnmarshalJSON(data []byte) error { type RPCMevShareBundle struct { Version string `json:"version"` Inclusion struct { - Block string `json:"block"` + Block string `json:"block"` + MaxBlock string `json:"maxBlock"` } `json:"inclusion"` Body []struct { Tx string `json:"tx"` diff --git a/core/vm/contracts_suave_eth.go b/core/vm/contracts_suave_eth.go index 02c826f170..38d62583f2 100644 --- a/core/vm/contracts_suave_eth.go +++ b/core/vm/contracts_suave_eth.go @@ -422,6 +422,7 @@ func (c *suaveRuntime) fillMevShareBundle(bidId types.BidId) ([]byte, error) { } shareBundle.Inclusion.Block = hexutil.EncodeUint64(bid.DecryptionCondition) + shareBundle.Inclusion.MaxBlock = hexutil.EncodeUint64(bid.DecryptionCondition + 25) // Assumes 25 block inclusion range for _, tx := range append(userBundle.Txs, matchBundle.Txs...) { txBytes, err := tx.MarshalBinary() diff --git a/core/vm/contracts_suave_runtime_adapter_test.go b/core/vm/contracts_suave_runtime_adapter_test.go new file mode 100644 index 0000000000..dde519b234 --- /dev/null +++ b/core/vm/contracts_suave_runtime_adapter_test.go @@ -0,0 +1,93 @@ +package vm + +import ( + "testing" + + "github.com/ethereum/go-ethereum/accounts/abi" + "github.com/ethereum/go-ethereum/common" + "github.com/ethereum/go-ethereum/core/types" + "github.com/ethereum/go-ethereum/suave/artifacts" + "github.com/stretchr/testify/require" +) + +var _ SuaveRuntime = &mockRuntime{} + +type mockRuntime struct { +} + +func (m *mockRuntime) buildEthBlock(blockArgs types.BuildBlockArgs, bidId types.BidId, namespace string) ([]byte, []byte, error) { + return []byte{0x1}, []byte{0x1}, nil +} + +func (m *mockRuntime) confidentialInputs() ([]byte, error) { + return []byte{0x1}, nil +} + +func (m *mockRuntime) confidentialRetrieve(bidId types.BidId, key string) ([]byte, error) { + return []byte{0x1}, nil +} + +func (m *mockRuntime) confidentialStore(bidId types.BidId, key string, data1 []byte) error { + return nil +} + +func (m *mockRuntime) ethcall(contractAddr common.Address, input1 []byte) ([]byte, error) { + return []byte{0x1}, nil +} + +func (m *mockRuntime) extractHint(bundleData []byte) ([]byte, error) { + return []byte{0x1}, nil +} + +func (m *mockRuntime) fetchBids(cond uint64, namespace string) ([]types.Bid, error) { + return []types.Bid{{}}, nil +} + +func (m *mockRuntime) fillMevShareBundle(bidId types.BidId) ([]byte, error) { + return []byte{0x1}, nil +} + +func (m *mockRuntime) newBid(decryptionCondition uint64, allowedPeekers []common.Address, allowedStores []common.Address, bidType string) (types.Bid, error) { + return types.Bid{}, nil +} + +func (m *mockRuntime) signEthTransaction(txn []byte, chainId string, signingKey string) ([]byte, error) { + return []byte{0x1}, nil +} + +func (m *mockRuntime) simulateBundle(bundleData []byte) (uint64, error) { + return 1, nil +} + +func (m *mockRuntime) submitBundleJsonRPC(url string, method string, params []byte) ([]byte, error) { + return []byte{0x1}, nil +} + +func (m *mockRuntime) submitEthBlockBidToRelay(relayUrl string, builderBid []byte) ([]byte, error) { + return []byte{0x1}, nil +} + +func (m *mockRuntime) doHTTPRequest(request types.HttpRequest) ([]byte, error) { + return []byte{0x1}, nil +} + +func TestRuntimeAdapter(t *testing.T) { + adapter := &SuaveRuntimeAdapter{ + impl: &mockRuntime{}, + } + + for name, addr := range artifacts.SuaveMethods { + abiMethod, ok := artifacts.SuaveAbi.Methods[name] + if !ok { + t.Fatalf("abi method '%s' not found", name) + } + + inputVals := abi.GenerateRandomTypeForMethod(abiMethod) + + packedInput, err := abiMethod.Inputs.Pack(inputVals...) + require.NoError(t, err) + + _, err = adapter.run(addr, packedInput) + require.NoError(t, err) + } +} diff --git a/core/vm/contracts_suave_test.go b/core/vm/contracts_suave_test.go index a37cdddd2d..995c088c86 100644 --- a/core/vm/contracts_suave_test.go +++ b/core/vm/contracts_suave_test.go @@ -2,21 +2,13 @@ package vm import ( "context" - "math/big" "net/http" "net/http/httptest" - "regexp" - "strings" "testing" - "github.com/ethereum/go-ethereum/accounts/abi" "github.com/ethereum/go-ethereum/beacon/engine" "github.com/ethereum/go-ethereum/common" - "github.com/ethereum/go-ethereum/core/rawdb" - "github.com/ethereum/go-ethereum/core/state" "github.com/ethereum/go-ethereum/core/types" - "github.com/ethereum/go-ethereum/params" - "github.com/ethereum/go-ethereum/suave/artifacts" suave "github.com/ethereum/go-ethereum/suave/core" "github.com/ethereum/go-ethereum/suave/cstore" "github.com/stretchr/testify/require" @@ -74,102 +66,6 @@ func (m *mockSuaveBackend) Subscribe() (<-chan cstore.DAMessage, context.CancelF func (m *mockSuaveBackend) Publish(cstore.DAMessage) {} -var dummyBlockContext = BlockContext{ - CanTransfer: func(StateDB, common.Address, *big.Int) bool { return true }, - Transfer: func(StateDB, common.Address, common.Address, *big.Int) {}, - BlockNumber: big.NewInt(0), -} - -func TestSuavePrecompileStub(t *testing.T) { - // This test ensures that the Suave precompile stubs work as expected - // for encoding/decoding. - mockSuaveBackend := &mockSuaveBackend{} - stubEngine := cstore.NewConfidentialStoreEngine(mockSuaveBackend, mockSuaveBackend, cstore.MockSigner{}, cstore.MockChainSigner{}) - - reqTx := types.NewTx(&types.ConfidentialComputeRequest{ - ConfidentialComputeRecord: types.ConfidentialComputeRecord{ - KettleAddress: common.Address{}, - }, - }) - - suaveContext := SuaveContext{ - Backend: &SuaveExecutionBackend{ - ConfidentialStore: stubEngine.NewTransactionalStore(reqTx), - ConfidentialEthBackend: mockSuaveBackend, - }, - ConfidentialComputeRequestTx: reqTx, - } - - statedb, _ := state.New(types.EmptyRootHash, state.NewDatabase(rawdb.NewMemoryDatabase()), nil) - vmenv := NewConfidentialEVM(suaveContext, dummyBlockContext, TxContext{}, statedb, params.AllEthashProtocolChanges, Config{IsConfidential: true}) - - // The objective of the unit test is to make sure that the encoding of the precompile - // inputs works as expected from the ABI specification. Thus, we will skip any errors - // that are generated by the logic of the precompile. - // Note: Once code generated is in place, we can remove this and only test the - // encodings in isolation outside the logic. - expectedErrors := []string{ - // json error when the precompile expects to decode a json object encoded as []byte - // in the precompile input. - "invalid character", - "not allowed to store", - "not allowed to retrieve", - "unknown bid version", - // error from a precompile that expects to make an http request from an input value. - "could not send request to relay", - // error in 'buildEthBlock' when it expects to retrieve bids in abi format from the - // confidential store. - "could not unpack merged bid ids", - "no caller of confidentialRetrieve (0000000000000000000000000000000042020001) is allowed on 00000000000000000000000000000000", - "precompile fillMevShareBundle (0000000000000000000000000000000043200001) not allowed on 00000000000000000000000000000000", - "no caller of confidentialStore (0000000000000000000000000000000042020000) is allowed on 00000000000000000000000000000000", - "precompile buildEthBlock (0000000000000000000000000000000042100001) not allowed on 00000000000000000000000000000000", - "only GET and POST methods are supported", - "domain is not allowed", - } - - expectedVariableErrors := []*regexp.Regexp{ - regexp.MustCompile("key not formatted properly: invalid hex character.*in private key"), - } - - for name, addr := range artifacts.SuaveMethods { - abiMethod, ok := artifacts.SuaveAbi.Methods[name] - if !ok { - t.Fatalf("abi method '%s' not found", name) - } - - inputVals := abi.GenerateRandomTypeForMethod(abiMethod) - - packedInput, err := abiMethod.Inputs.Pack(inputVals...) - require.NoError(t, err) - - _, _, err = vmenv.Call(AccountRef(common.Address{}), addr, packedInput, 100000000, big.NewInt(0)) - if err != nil { - found := false - for _, expectedError := range expectedErrors { - if strings.Contains(err.Error(), expectedError) { - found = true - break - } - } - - if found { - continue - } - - for _, expectedErrRe := range expectedVariableErrors { - if expectedErrRe.Match([]byte(err.Error())) { - found = true - break - } - } - if !found { - t.Fatal(err) - } - } - } -} - func newTestBackend(t *testing.T) *suaveRuntime { confStore := cstore.NewLocalConfidentialStore() confEngine := cstore.NewConfidentialStoreEngine(confStore, &cstore.MockTransport{}, cstore.MockSigner{}, cstore.MockChainSigner{}) diff --git a/core/vm/suave.go b/core/vm/suave.go index 95dd473b60..62af185b78 100644 --- a/core/vm/suave.go +++ b/core/vm/suave.go @@ -93,6 +93,7 @@ func (p *SuavePrecompiledContractWrapper) Run(input []byte) ([]byte, error) { ret, err := stub.run(p.addr, input) if err != nil && ret == nil { ret = []byte(err.Error()) + err = ErrExecutionReverted } return ret, err diff --git a/suave/artifacts/SuaveLib.json b/suave/artifacts/SuaveLib.json index de85d4fd4d..425fea3bec 100644 --- a/suave/artifacts/SuaveLib.json +++ b/suave/artifacts/SuaveLib.json @@ -1 +1 @@ -[{"type":"function","name":"buildEthBlock","inputs":[{"name":"blockArgs","type":"tuple","internalType":"struct Suave.BuildBlockArgs","components":[{"name":"slot","type":"uint64","internalType":"uint64"},{"name":"proposerPubkey","type":"bytes","internalType":"bytes"},{"name":"parent","type":"bytes32","internalType":"bytes32"},{"name":"timestamp","type":"uint64","internalType":"uint64"},{"name":"feeRecipient","type":"address","internalType":"address"},{"name":"gasLimit","type":"uint64","internalType":"uint64"},{"name":"random","type":"bytes32","internalType":"bytes32"},{"name":"withdrawals","type":"tuple[]","internalType":"struct Suave.Withdrawal[]","components":[{"name":"index","type":"uint64","internalType":"uint64"},{"name":"validator","type":"uint64","internalType":"uint64"},{"name":"Address","type":"address","internalType":"address"},{"name":"amount","type":"uint64","internalType":"uint64"}]},{"name":"extra","type":"bytes","internalType":"bytes"}]},{"name":"bidId","type":"bytes16","internalType":"struct Suave.BidId"},{"name":"namespace","type":"string","internalType":"string"}],"outputs":[{"name":"output1","type":"bytes","internalType":"bytes"},{"name":"output2","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"confidentialInputs","outputs":[{"name":"output1","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"confidentialRetrieve","inputs":[{"name":"bidId","type":"bytes16","internalType":"struct Suave.BidId"},{"name":"key","type":"string","internalType":"string"}],"outputs":[{"name":"output1","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"confidentialStore","inputs":[{"name":"bidId","type":"bytes16","internalType":"struct Suave.BidId"},{"name":"key","type":"string","internalType":"string"},{"name":"data1","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"doHTTPRequest","inputs":[{"name":"request","type":"tuple","internalType":"struct Suave.HttpRequest","components":[{"name":"url","type":"string","internalType":"string"},{"name":"method","type":"string","internalType":"string"},{"name":"headers","type":"string[]","internalType":"string[]"},{"name":"body","type":"bytes","internalType":"bytes"}]}],"outputs":[{"name":"response","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"ethcall","inputs":[{"name":"contractAddr","type":"address","internalType":"address"},{"name":"input1","type":"bytes","internalType":"bytes"}],"outputs":[{"name":"output1","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"extractHint","inputs":[{"name":"bundleData","type":"bytes","internalType":"bytes"}],"outputs":[{"name":"output1","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"fetchBids","inputs":[{"name":"cond","type":"uint64","internalType":"uint64"},{"name":"namespace","type":"string","internalType":"string"}],"outputs":[{"name":"bid","type":"tuple[]","internalType":"struct Suave.Bid[]","components":[{"name":"id","type":"bytes16","internalType":"struct Suave.BidId"},{"name":"salt","type":"bytes16","internalType":"struct Suave.BidId"},{"name":"decryptionCondition","type":"uint64","internalType":"uint64"},{"name":"allowedPeekers","type":"address[]","internalType":"address[]"},{"name":"allowedStores","type":"address[]","internalType":"address[]"},{"name":"version","type":"string","internalType":"string"}]}]},{"type":"function","name":"fillMevShareBundle","inputs":[{"name":"bidId","type":"bytes16","internalType":"struct Suave.BidId"}],"outputs":[{"name":"encodedBundle","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"newBid","inputs":[{"name":"decryptionCondition","type":"uint64","internalType":"uint64"},{"name":"allowedPeekers","type":"address[]","internalType":"address[]"},{"name":"allowedStores","type":"address[]","internalType":"address[]"},{"name":"bidType","type":"string","internalType":"string"}],"outputs":[{"name":"bid","type":"tuple","internalType":"struct Suave.Bid","components":[{"name":"id","type":"bytes16","internalType":"struct Suave.BidId"},{"name":"salt","type":"bytes16","internalType":"struct Suave.BidId"},{"name":"decryptionCondition","type":"uint64","internalType":"uint64"},{"name":"allowedPeekers","type":"address[]","internalType":"address[]"},{"name":"allowedStores","type":"address[]","internalType":"address[]"},{"name":"version","type":"string","internalType":"string"}]}]},{"type":"function","name":"signEthTransaction","inputs":[{"name":"txn","type":"bytes","internalType":"bytes"},{"name":"chainId","type":"string","internalType":"string"},{"name":"signingKey","type":"string","internalType":"string"}],"outputs":[{"name":"output1","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"simulateBundle","inputs":[{"name":"bundleData","type":"bytes","internalType":"bytes"}],"outputs":[{"name":"output1","type":"uint64","internalType":"uint64"}]},{"type":"function","name":"submitBundleJsonRPC","inputs":[{"name":"url","type":"string","internalType":"string"},{"name":"method","type":"string","internalType":"string"},{"name":"params","type":"bytes","internalType":"bytes"}],"outputs":[{"name":"output1","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"submitEthBlockBidToRelay","inputs":[{"name":"relayUrl","type":"string","internalType":"string"},{"name":"builderBid","type":"bytes","internalType":"bytes"}],"outputs":[{"name":"output1","type":"bytes","internalType":"bytes"}]}] \ No newline at end of file +[{"type":"error","name":"PeekerReverted","inputs":[{"name":"addr","type":"address"},{"name":"err","type":"bytes"}]},{"type":"function","name":"buildEthBlock","inputs":[{"name":"blockArgs","type":"tuple","internalType":"struct Suave.BuildBlockArgs","components":[{"name":"slot","type":"uint64","internalType":"uint64"},{"name":"proposerPubkey","type":"bytes","internalType":"bytes"},{"name":"parent","type":"bytes32","internalType":"bytes32"},{"name":"timestamp","type":"uint64","internalType":"uint64"},{"name":"feeRecipient","type":"address","internalType":"address"},{"name":"gasLimit","type":"uint64","internalType":"uint64"},{"name":"random","type":"bytes32","internalType":"bytes32"},{"name":"withdrawals","type":"tuple[]","internalType":"struct Suave.Withdrawal[]","components":[{"name":"index","type":"uint64","internalType":"uint64"},{"name":"validator","type":"uint64","internalType":"uint64"},{"name":"Address","type":"address","internalType":"address"},{"name":"amount","type":"uint64","internalType":"uint64"}]},{"name":"extra","type":"bytes","internalType":"bytes"}]},{"name":"bidId","type":"bytes16","internalType":"struct Suave.BidId"},{"name":"namespace","type":"string","internalType":"string"}],"outputs":[{"name":"output1","type":"bytes","internalType":"bytes"},{"name":"output2","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"confidentialInputs","outputs":[{"name":"output1","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"confidentialRetrieve","inputs":[{"name":"bidId","type":"bytes16","internalType":"struct Suave.BidId"},{"name":"key","type":"string","internalType":"string"}],"outputs":[{"name":"output1","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"confidentialStore","inputs":[{"name":"bidId","type":"bytes16","internalType":"struct Suave.BidId"},{"name":"key","type":"string","internalType":"string"},{"name":"data1","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"doHTTPRequest","inputs":[{"name":"request","type":"tuple","internalType":"struct Suave.HttpRequest","components":[{"name":"url","type":"string","internalType":"string"},{"name":"method","type":"string","internalType":"string"},{"name":"headers","type":"string[]","internalType":"string[]"},{"name":"body","type":"bytes","internalType":"bytes"}]}],"outputs":[{"name":"response","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"ethcall","inputs":[{"name":"contractAddr","type":"address","internalType":"address"},{"name":"input1","type":"bytes","internalType":"bytes"}],"outputs":[{"name":"output1","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"extractHint","inputs":[{"name":"bundleData","type":"bytes","internalType":"bytes"}],"outputs":[{"name":"output1","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"fetchBids","inputs":[{"name":"cond","type":"uint64","internalType":"uint64"},{"name":"namespace","type":"string","internalType":"string"}],"outputs":[{"name":"bid","type":"tuple[]","internalType":"struct Suave.Bid[]","components":[{"name":"id","type":"bytes16","internalType":"struct Suave.BidId"},{"name":"salt","type":"bytes16","internalType":"struct Suave.BidId"},{"name":"decryptionCondition","type":"uint64","internalType":"uint64"},{"name":"allowedPeekers","type":"address[]","internalType":"address[]"},{"name":"allowedStores","type":"address[]","internalType":"address[]"},{"name":"version","type":"string","internalType":"string"}]}]},{"type":"function","name":"fillMevShareBundle","inputs":[{"name":"bidId","type":"bytes16","internalType":"struct Suave.BidId"}],"outputs":[{"name":"encodedBundle","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"newBid","inputs":[{"name":"decryptionCondition","type":"uint64","internalType":"uint64"},{"name":"allowedPeekers","type":"address[]","internalType":"address[]"},{"name":"allowedStores","type":"address[]","internalType":"address[]"},{"name":"bidType","type":"string","internalType":"string"}],"outputs":[{"name":"bid","type":"tuple","internalType":"struct Suave.Bid","components":[{"name":"id","type":"bytes16","internalType":"struct Suave.BidId"},{"name":"salt","type":"bytes16","internalType":"struct Suave.BidId"},{"name":"decryptionCondition","type":"uint64","internalType":"uint64"},{"name":"allowedPeekers","type":"address[]","internalType":"address[]"},{"name":"allowedStores","type":"address[]","internalType":"address[]"},{"name":"version","type":"string","internalType":"string"}]}]},{"type":"function","name":"signEthTransaction","inputs":[{"name":"txn","type":"bytes","internalType":"bytes"},{"name":"chainId","type":"string","internalType":"string"},{"name":"signingKey","type":"string","internalType":"string"}],"outputs":[{"name":"output1","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"simulateBundle","inputs":[{"name":"bundleData","type":"bytes","internalType":"bytes"}],"outputs":[{"name":"output1","type":"uint64","internalType":"uint64"}]},{"type":"function","name":"submitBundleJsonRPC","inputs":[{"name":"url","type":"string","internalType":"string"},{"name":"method","type":"string","internalType":"string"},{"name":"params","type":"bytes","internalType":"bytes"}],"outputs":[{"name":"output1","type":"bytes","internalType":"bytes"}]},{"type":"function","name":"submitEthBlockBidToRelay","inputs":[{"name":"relayUrl","type":"string","internalType":"string"},{"name":"builderBid","type":"bytes","internalType":"bytes"}],"outputs":[{"name":"output1","type":"bytes","internalType":"bytes"}]}] \ No newline at end of file diff --git a/suave/e2e/contracts.go b/suave/e2e/contracts.go index acafc71d57..ac03d0904e 100644 --- a/suave/e2e/contracts.go +++ b/suave/e2e/contracts.go @@ -18,7 +18,6 @@ var ( MevShareBundleSenderContract = newArtifact("bids.sol/MevShareBundleSenderContract.json") buildEthBlockContract = newArtifact("bids.sol/EthBlockBidContract.json") ethBlockBidSenderContract = newArtifact("bids.sol/EthBlockBidSenderContract.json") - suaveLibContract = newArtifact("SuaveAbi.sol/SuaveAbi.json") exampleCallSourceContract = newArtifact("example.sol/ExampleEthCallSource.json") exampleCallTargetContract = newArtifact("example.sol/ExampleEthCallTarget.json") ) diff --git a/suave/e2e/workflow_test.go b/suave/e2e/workflow_test.go index 6ad6218f86..f78a800fb5 100644 --- a/suave/e2e/workflow_test.go +++ b/suave/e2e/workflow_test.go @@ -735,7 +735,7 @@ func TestBlockBuildingPrecompiles(t *testing.T) { require.NoError(t, err) { // Test the bundle simulation precompile through eth_call - calldata, err := suaveLibContract.Abi.Methods["simulateBundle"].Inputs.Pack(bundleBytes) + calldata, err := artifacts.SuaveAbi.Methods["simulateBundle"].Inputs.Pack(bundleBytes) require.NoError(t, err) var simResult hexutil.Bytes @@ -785,7 +785,7 @@ func TestBlockBuildingPrecompiles(t *testing.T) { FeeRecipient: common.Address{0x42}, } - packed, err := suaveLibContract.Abi.Methods["buildEthBlock"].Inputs.Pack(payloadArgsTuple, bid.Id, "") + packed, err := artifacts.SuaveAbi.Methods["buildEthBlock"].Inputs.Pack(payloadArgsTuple, bid.Id, "") require.NoError(t, err) var simResult hexutil.Bytes @@ -799,7 +799,7 @@ func TestBlockBuildingPrecompiles(t *testing.T) { require.NotNil(t, simResult) - unpacked, err := suaveLibContract.Abi.Methods["buildEthBlock"].Outputs.Unpack(simResult) + unpacked, err := artifacts.SuaveAbi.Methods["buildEthBlock"].Outputs.Unpack(simResult) require.NoError(t, err) // TODO: test builder bid @@ -1528,7 +1528,7 @@ func requireNoRpcError(t *testing.T, rpcErr error) { require.NoError(t, rpcErr, decodedError) } - unpacked, err := suaveLibContract.Abi.Errors["PeekerReverted"].Inputs.Unpack(decodedError[4:]) + unpacked, err := artifacts.SuaveAbi.Errors["PeekerReverted"].Inputs.Unpack(decodedError[4:]) if err != nil { require.NoError(t, err, rpcErr.Error()) } else { diff --git a/suave/gen/main.go b/suave/gen/main.go index 468bea8904..3a1d9f3fd9 100644 --- a/suave/gen/main.go +++ b/suave/gen/main.go @@ -608,6 +608,17 @@ func generateABI(out string, dd desc) error { return arg } + // encode PeekerReverted(address, bytes) + peekerReverted := &abiField{ + Type: "error", + Name: "PeekerReverted", + Inputs: []arguments{ + {Name: "addr", Type: "address"}, + {Name: "err", Type: "bytes"}, + }, + } + abiEncode = append(abiEncode, peekerReverted) + for _, f := range dd.Functions { field := &abiField{ Name: f.Name, diff --git a/suave/gen/main_test.go b/suave/gen/main_test.go index fd8056dcc0..486c977180 100644 --- a/suave/gen/main_test.go +++ b/suave/gen/main_test.go @@ -1,8 +1,11 @@ package main import ( + "encoding/hex" "testing" + "github.com/ethereum/go-ethereum/common" + "github.com/ethereum/go-ethereum/suave/artifacts" "github.com/stretchr/testify/require" ) @@ -43,3 +46,18 @@ func TestEncodeTypeToGolang(t *testing.T) { require.Equal(t, c.expected, actual) } } + +func TestDecodeABI_PeekerReverted(t *testing.T) { + errMsg, err := hex.DecodeString("75fff4670000000000000000000000000000000000000000000000000000000042100000000000000000000000000000000000000000000000000000000000000000004000000000000000000000000000000000000000000000000000000000000000036261640000000000000000000000000000000000000000000000000000000000") + require.NoError(t, err) + + errorEvnt := artifacts.SuaveAbi.Errors["PeekerReverted"] + vals, err := errorEvnt.Inputs.Unpack(errMsg[4:]) + require.NoError(t, err) + + addr := vals[0].(common.Address) + reason := vals[1].([]byte) + + require.Equal(t, addr.String(), "0x0000000000000000000000000000000042100000") + require.Equal(t, string(reason), "bad") +} diff --git a/suave/sol/libraries/SuaveAbi.sol b/suave/sol/libraries/SuaveAbi.sol deleted file mode 100644 index c2c552cc9c..0000000000 --- a/suave/sol/libraries/SuaveAbi.sol +++ /dev/null @@ -1,19 +0,0 @@ -pragma solidity ^0.8.8; - -import {Suave} from "./Suave.sol"; - -contract SuaveAbi { - error PeekerReverted(address, bytes); - - function newBid(uint64 decryptionCondition, address[] memory allowedPeekers, address[] memory allowedStores, string memory BidType) external view returns (Suave.Bid memory) {} - function fetchBids(uint64 cond, string memory namespace) external view returns (Suave.Bid[] memory) {} - function confidentialStore(Suave.BidId bidId, string memory key, bytes memory data) external view {} - function confidentialRetrieve(Suave.BidId bidId, string memory key) external view returns (bytes memory) {} - function signEthTransaction(bytes memory txn, string memory chainId, string memory signingKey) external view returns (bytes memory) {} - function simulateBundle(bytes memory bundleData) external view returns (uint64) {} - function extractHint(bytes memory bundleData) external view returns (bytes memory) {} - function buildEthBlock(Suave.BuildBlockArgs memory blockArgs, Suave.BidId bid, string memory namespace) external view returns (bytes memory, bytes memory) {} - function submitEthBlockBidToRelay(string memory relayUrl, bytes memory builderBid) external view returns (bytes memory) {} - function fillMevShareBundle(Suave.BidId bidId) external view returns (bytes memory) {} - function submitBundleJsonRPC(string memory url, string memory method, bytes memory params) external view returns (bytes memory) {} -} diff --git a/suave/sol/standard_peekers/bids.sol b/suave/sol/standard_peekers/bids.sol index b57bd1755e..4c866a5a73 100644 --- a/suave/sol/standard_peekers/bids.sol +++ b/suave/sol/standard_peekers/bids.sol @@ -3,324 +3,357 @@ pragma solidity ^0.8.8; import "../libraries/Suave.sol"; contract AnyBidContract { + event BidEvent(Suave.BidId bidId, uint64 decryptionCondition, address[] allowedPeekers); - event BidEvent( - Suave.BidId bidId, - uint64 decryptionCondition, - address[] allowedPeekers - ); - - function fetchBidConfidentialBundleData() public returns (bytes memory) { - require(Suave.isConfidential()); - - bytes memory confidentialInputs = Suave.confidentialInputs(); - return abi.decode(confidentialInputs, (bytes)); - } - - // Bids to this contract should not be trusted! - function emitBid(Suave.Bid calldata bid) public { - emit BidEvent(bid.id, bid.decryptionCondition, bid.allowedPeekers); - } + function fetchBidConfidentialBundleData() public returns (bytes memory) { + require(Suave.isConfidential()); + + bytes memory confidentialInputs = Suave.confidentialInputs(); + return abi.decode(confidentialInputs, (bytes)); + } + + // Bids to this contract should not be trusted! + function emitBid(Suave.Bid calldata bid) public { + emit BidEvent(bid.id, bid.decryptionCondition, bid.allowedPeekers); + } } contract BundleBidContract is AnyBidContract { + function newBid(uint64 decryptionCondition, address[] memory bidAllowedPeekers, address[] memory bidAllowedStores) + external + payable + returns (bytes memory) + { + require(Suave.isConfidential()); - function newBid(uint64 decryptionCondition, address[] memory bidAllowedPeekers, address[] memory bidAllowedStores) external payable returns (bytes memory) { - require(Suave.isConfidential()); + bytes memory bundleData = this.fetchBidConfidentialBundleData(); - bytes memory bundleData = this.fetchBidConfidentialBundleData(); + uint64 egp = Suave.simulateBundle(bundleData); - uint64 egp = Suave.simulateBundle(bundleData); + Suave.Bid memory bid = + Suave.newBid(decryptionCondition, bidAllowedPeekers, bidAllowedStores, "default:v0:ethBundles"); - Suave.Bid memory bid = Suave.newBid(decryptionCondition, bidAllowedPeekers, bidAllowedStores, "default:v0:ethBundles"); + Suave.confidentialStore(bid.id, "default:v0:ethBundles", bundleData); + Suave.confidentialStore(bid.id, "default:v0:ethBundleSimResults", abi.encode(egp)); - Suave.confidentialStore(bid.id, "default:v0:ethBundles", bundleData); - Suave.confidentialStore(bid.id, "default:v0:ethBundleSimResults", abi.encode(egp)); + return emitAndReturn(bid, bundleData); + } - return emitAndReturn(bid, bundleData); - } - - function emitAndReturn(Suave.Bid memory bid, bytes memory) internal virtual returns (bytes memory) { - emit BidEvent(bid.id, bid.decryptionCondition, bid.allowedPeekers); - return bytes.concat(this.emitBid.selector, abi.encode(bid)); - } + function emitAndReturn(Suave.Bid memory bid, bytes memory) internal virtual returns (bytes memory) { + emit BidEvent(bid.id, bid.decryptionCondition, bid.allowedPeekers); + return bytes.concat(this.emitBid.selector, abi.encode(bid)); + } } contract EthBundleSenderContract is BundleBidContract { - string[] public builderUrls; - - constructor(string[] memory builderUrls_) { - builderUrls = builderUrls_; - } - - function emitAndReturn(Suave.Bid memory bid, bytes memory bundleData) internal virtual override returns (bytes memory) { - for (uint i = 0; i < builderUrls.length; i++) { - Suave.submitBundleJsonRPC(builderUrls[i], "eth_sendBundle", bundleData); - } - - return BundleBidContract.emitAndReturn(bid, bundleData); - } + string[] public builderUrls; + + constructor(string[] memory builderUrls_) { + builderUrls = builderUrls_; + } + + function emitAndReturn(Suave.Bid memory bid, bytes memory bundleData) + internal + virtual + override + returns (bytes memory) + { + for (uint256 i = 0; i < builderUrls.length; i++) { + Suave.submitBundleJsonRPC(builderUrls[i], "eth_sendBundle", bundleData); + } + + return BundleBidContract.emitAndReturn(bid, bundleData); + } } contract MevShareBidContract is AnyBidContract { - - event HintEvent( - Suave.BidId bidId, - bytes hint - ); - - event MatchEvent( - Suave.BidId matchBidId, - bytes matchHint - ); - - function newBid(uint64 decryptionCondition, address[] memory bidAllowedPeekers, address[] memory bidAllowedStores) external payable returns (bytes memory) { - // 0. check confidential execution - require(Suave.isConfidential()); - - // 1. fetch bundle data - bytes memory bundleData = this.fetchBidConfidentialBundleData(); - - // 2. sim bundle - uint64 egp = Suave.simulateBundle(bundleData); - - // 3. extract hint - bytes memory hint = Suave.extractHint(bundleData); - - // // 4. store bundle and sim results - Suave.Bid memory bid = Suave.newBid(decryptionCondition, bidAllowedPeekers, bidAllowedStores, "mevshare:v0:unmatchedBundles"); - Suave.confidentialStore(bid.id, "mevshare:v0:ethBundles", bundleData); - Suave.confidentialStore(bid.id, "mevshare:v0:ethBundleSimResults", abi.encode(egp)); - emit BidEvent(bid.id, bid.decryptionCondition, bid.allowedPeekers); - emit HintEvent(bid.id, hint); - - // // 5. return "callback" to emit hint onchain - return bytes.concat(this.emitBidAndHint.selector, abi.encode(bid, hint)); - } - - function emitBidAndHint(Suave.Bid calldata bid, bytes memory hint) public { - emit BidEvent(bid.id, bid.decryptionCondition, bid.allowedPeekers); - emit HintEvent(bid.id, hint); - } - - function newMatch(uint64 decryptionCondition, address[] memory bidAllowedPeekers, address[] memory bidAllowedStores, Suave.BidId shareBidId) external payable returns (bytes memory) { - // WARNING : this function will copy the original mev share bid - // into a new key with potentially different permsissions - - require(Suave.isConfidential()); - // 1. fetch confidential data - bytes memory matchBundleData = this.fetchBidConfidentialBundleData(); - - // 2. sim match alone for validity - uint64 egp = Suave.simulateBundle(matchBundleData); - - // 3. extract hint - bytes memory matchHint = Suave.extractHint(matchBundleData); - - Suave.Bid memory bid = Suave.newBid(decryptionCondition, bidAllowedPeekers, bidAllowedStores, "mevshare:v0:matchBids"); - Suave.confidentialStore(bid.id, "mevshare:v0:ethBundles", matchBundleData); - Suave.confidentialStore(bid.id, "mevshare:v0:ethBundleSimResults", abi.encode(0)); - - //4. merge bids - Suave.BidId[] memory bids = new Suave.BidId[](2); - bids[0] = shareBidId; - bids[1] = bid.id; - Suave.confidentialStore(bid.id, "mevshare:v0:mergedBids", abi.encode(bids)); - - return emitMatchBidAndHint(bid, matchHint); - } - - function emitMatchBidAndHint(Suave.Bid memory bid, bytes memory matchHint) internal virtual returns (bytes memory) { - emit BidEvent(bid.id, bid.decryptionCondition, bid.allowedPeekers); - emit MatchEvent(bid.id, matchHint); - - return bytes.concat(this.emitBid.selector, abi.encode(bid)); - } + event HintEvent(Suave.BidId bidId, bytes hint); + + event MatchEvent(Suave.BidId matchBidId, bytes matchHint); + + function newBid(uint64 decryptionCondition, address[] memory bidAllowedPeekers, address[] memory bidAllowedStores) + external + payable + returns (bytes memory) + { + // 0. check confidential execution + require(Suave.isConfidential()); + + // 1. fetch bundle data + bytes memory bundleData = this.fetchBidConfidentialBundleData(); + + // 2. sim bundle + uint64 egp = Suave.simulateBundle(bundleData); + + // 3. extract hint + bytes memory hint = Suave.extractHint(bundleData); + + // // 4. store bundle and sim results + Suave.Bid memory bid = + Suave.newBid(decryptionCondition, bidAllowedPeekers, bidAllowedStores, "mevshare:v0:unmatchedBundles"); + Suave.confidentialStore(bid.id, "mevshare:v0:ethBundles", bundleData); + Suave.confidentialStore(bid.id, "mevshare:v0:ethBundleSimResults", abi.encode(egp)); + emit BidEvent(bid.id, bid.decryptionCondition, bid.allowedPeekers); + emit HintEvent(bid.id, hint); + + // // 5. return "callback" to emit hint onchain + return bytes.concat(this.emitBidAndHint.selector, abi.encode(bid, hint)); + } + + function emitBidAndHint(Suave.Bid calldata bid, bytes memory hint) public { + emit BidEvent(bid.id, bid.decryptionCondition, bid.allowedPeekers); + emit HintEvent(bid.id, hint); + } + + function newMatch( + uint64 decryptionCondition, + address[] memory bidAllowedPeekers, + address[] memory bidAllowedStores, + Suave.BidId shareBidId + ) external payable returns (bytes memory) { + // WARNING : this function will copy the original mev share bid + // into a new key with potentially different permsissions + + require(Suave.isConfidential()); + // 1. fetch confidential data + bytes memory matchBundleData = this.fetchBidConfidentialBundleData(); + + // 2. sim match alone for validity + uint64 egp = Suave.simulateBundle(matchBundleData); + + // 3. extract hint + bytes memory matchHint = Suave.extractHint(matchBundleData); + + Suave.Bid memory bid = + Suave.newBid(decryptionCondition, bidAllowedPeekers, bidAllowedStores, "mevshare:v0:matchBids"); + Suave.confidentialStore(bid.id, "mevshare:v0:ethBundles", matchBundleData); + Suave.confidentialStore(bid.id, "mevshare:v0:ethBundleSimResults", abi.encode(0)); + + //4. merge bids + Suave.BidId[] memory bids = new Suave.BidId[](2); + bids[0] = shareBidId; + bids[1] = bid.id; + Suave.confidentialStore(bid.id, "mevshare:v0:mergedBids", abi.encode(bids)); + + return emitMatchBidAndHint(bid, matchHint); + } + + function emitMatchBidAndHint(Suave.Bid memory bid, bytes memory matchHint) + internal + virtual + returns (bytes memory) + { + emit BidEvent(bid.id, bid.decryptionCondition, bid.allowedPeekers); + emit MatchEvent(bid.id, matchHint); + + return bytes.concat(this.emitBid.selector, abi.encode(bid)); + } } contract MevShareBundleSenderContract is MevShareBidContract { - string[] public builderUrls; - - constructor(string[] memory builderUrls_) { - builderUrls = builderUrls_; - } - - function emitMatchBidAndHint(Suave.Bid memory bid, bytes memory matchHint) internal virtual override returns (bytes memory) { - bytes memory bundleData = Suave.fillMevShareBundle(bid.id); - for (uint i = 0; i < builderUrls.length; i++) { - Suave.submitBundleJsonRPC(builderUrls[i], "mev_sendBundle", bundleData); - } - - return MevShareBidContract.emitMatchBidAndHint(bid, matchHint); - } + string[] public builderUrls; + + constructor(string[] memory builderUrls_) { + builderUrls = builderUrls_; + } + + function emitMatchBidAndHint(Suave.Bid memory bid, bytes memory matchHint) + internal + virtual + override + returns (bytes memory) + { + bytes memory bundleData = Suave.fillMevShareBundle(bid.id); + for (uint256 i = 0; i < builderUrls.length; i++) { + Suave.submitBundleJsonRPC(builderUrls[i], "mev_sendBundle", bundleData); + } + + return MevShareBidContract.emitMatchBidAndHint(bid, matchHint); + } } /* Not tested or implemented on the precompile side */ struct EgpBidPair { - uint64 egp; // in wei, beware overflow - Suave.BidId bidId; + uint64 egp; // in wei, beware overflow + Suave.BidId bidId; } contract EthBlockBidContract is AnyBidContract { - - event BuilderBoostBidEvent( - Suave.BidId bidId, - bytes builderBid - ); - - function idsEqual(Suave.BidId _l, Suave.BidId _r) public pure returns (bool) { - bytes memory l = abi.encodePacked(_l); - bytes memory r = abi.encodePacked(_r); - for (uint i = 0; i < l.length; i++) { - if (bytes(l)[i] != r[i]) { - return false; - } - } - - return true; - } - - function buildMevShare(Suave.BuildBlockArgs memory blockArgs, uint64 blockHeight) public returns (bytes memory) { - require(Suave.isConfidential()); - - Suave.Bid[] memory allShareMatchBids = Suave.fetchBids(blockHeight, "mevshare:v0:matchBids"); - Suave.Bid[] memory allShareUserBids = Suave.fetchBids(blockHeight, "mevshare:v0:unmatchedBundles"); - - if (allShareUserBids.length == 0) { - revert Suave.PeekerReverted(address(this), "no bids"); - } - - Suave.Bid[] memory allBids = new Suave.Bid[](allShareUserBids.length); - for (uint i = 0; i < allShareUserBids.length; i++) { - // TODO: sort matches by egp first! - Suave.Bid memory bidToInsert = allShareUserBids[i]; // will be updated with the best match if any - for (uint j = 0; j < allShareMatchBids.length; j++) { - // TODO: should be done once at the start and sorted - Suave.BidId[] memory mergedBidIds = abi.decode(Suave.confidentialRetrieve(allShareMatchBids[j].id, "mevshare:v0:mergedBids"), (Suave.BidId[])); - if (idsEqual(mergedBidIds[0], allShareUserBids[i].id)) { - bidToInsert = allShareMatchBids[j]; - break; - } - } - allBids[i] = bidToInsert; - } - - EgpBidPair[] memory bidsByEGP = new EgpBidPair[](allBids.length); - for (uint i = 0; i < allBids.length; i++) { - bytes memory simResults = Suave.confidentialRetrieve(allBids[i].id, "mevshare:v0:ethBundleSimResults"); - uint64 egp = abi.decode(simResults, (uint64)); - bidsByEGP[i] = EgpBidPair(egp, allBids[i].id); - } - - // Bubble sort, cause why not - uint n = bidsByEGP.length; - for (uint i = 0; i < n - 1; i++) { - for (uint j = i + 1; j < n; j++) { - if (bidsByEGP[i].egp < bidsByEGP[j].egp) { - EgpBidPair memory temp = bidsByEGP[i]; - bidsByEGP[i] = bidsByEGP[j]; - bidsByEGP[j] = temp; - } - } - } - - Suave.BidId[] memory allBidIds = new Suave.BidId[](allBids.length); - for (uint i = 0; i < bidsByEGP.length; i++) { - allBidIds[i] = bidsByEGP[i].bidId; - } - - return buildAndEmit(blockArgs, blockHeight, allBidIds, "mevshare:v0"); - } - - function buildFromPool(Suave.BuildBlockArgs memory blockArgs, uint64 blockHeight) public returns (bytes memory) { - require(Suave.isConfidential()); - - Suave.Bid[] memory allBids = Suave.fetchBids(blockHeight, "default:v0:ethBundles"); - if (allBids.length == 0) { - revert Suave.PeekerReverted(address(this), "no bids"); - } - - EgpBidPair[] memory bidsByEGP = new EgpBidPair[](allBids.length); - for (uint i = 0; i < allBids.length; i++) { - bytes memory simResults = Suave.confidentialRetrieve(allBids[i].id, "default:v0:ethBundleSimResults"); - uint64 egp = abi.decode(simResults, (uint64)); - bidsByEGP[i] = EgpBidPair(egp, allBids[i].id); - } - - // Bubble sort, cause why not - uint n = bidsByEGP.length; - for (uint i = 0; i < n - 1; i++) { - for (uint j = i + 1; j < n; j++) { - if (bidsByEGP[i].egp < bidsByEGP[j].egp) { - EgpBidPair memory temp = bidsByEGP[i]; - bidsByEGP[i] = bidsByEGP[j]; - bidsByEGP[j] = temp; - } - } - } - - Suave.BidId[] memory allBidIds = new Suave.BidId[](allBids.length); - for (uint i = 0; i < bidsByEGP.length; i++) { - allBidIds[i] = bidsByEGP[i].bidId; - } - - return buildAndEmit(blockArgs, blockHeight, allBidIds, ""); - } - - function buildAndEmit(Suave.BuildBlockArgs memory blockArgs, uint64 blockHeight, Suave.BidId[] memory bids, string memory namespace) public virtual returns (bytes memory) { - require(Suave.isConfidential()); - - (Suave.Bid memory blockBid, bytes memory builderBid) = this.doBuild(blockArgs, blockHeight, bids, namespace); - - emit BuilderBoostBidEvent(blockBid.id, builderBid); - emit BidEvent(blockBid.id, blockBid.decryptionCondition, blockBid.allowedPeekers); - return bytes.concat(this.emitBuilderBidAndBid.selector, abi.encode(blockBid, builderBid)); - } - - function doBuild(Suave.BuildBlockArgs memory blockArgs, uint64 blockHeight, Suave.BidId[] memory bids, string memory namespace) public view returns (Suave.Bid memory, bytes memory) { - address[] memory allowedPeekers = new address[](2); - allowedPeekers[0] = address(this); - allowedPeekers[1] = Suave.BUILD_ETH_BLOCK; - - Suave.Bid memory blockBid = Suave.newBid(blockHeight, allowedPeekers, allowedPeekers, "default:v0:mergedBids"); - Suave.confidentialStore(blockBid.id, "default:v0:mergedBids", abi.encode(bids)); - - (bytes memory builderBid, bytes memory payload) = Suave.buildEthBlock(blockArgs, blockBid.id, namespace); - Suave.confidentialStore(blockBid.id, "default:v0:builderPayload", payload); // only through this.unlock - - return (blockBid, builderBid); - } - - function emitBuilderBidAndBid(Suave.Bid memory bid, bytes memory builderBid) public returns (Suave.Bid memory, bytes memory) { - emit BuilderBoostBidEvent(bid.id, builderBid); - emit BidEvent(bid.id, bid.decryptionCondition, bid.allowedPeekers); - return (bid, builderBid); - } - - function unlock(Suave.BidId bidId, bytes memory signedBlindedHeader) public view returns (bytes memory) { - require(Suave.isConfidential()); - - // TODO: verify the header is correct - // TODO: incorporate protocol name - bytes memory payload = Suave.confidentialRetrieve(bidId, "default:v0:builderPayload"); - return payload; - } + event BuilderBoostBidEvent(Suave.BidId bidId, bytes builderBid); + + function idsEqual(Suave.BidId _l, Suave.BidId _r) public pure returns (bool) { + bytes memory l = abi.encodePacked(_l); + bytes memory r = abi.encodePacked(_r); + for (uint256 i = 0; i < l.length; i++) { + if (bytes(l)[i] != r[i]) { + return false; + } + } + + return true; + } + + function buildMevShare(Suave.BuildBlockArgs memory blockArgs, uint64 blockHeight) public returns (bytes memory) { + require(Suave.isConfidential()); + + Suave.Bid[] memory allShareMatchBids = Suave.fetchBids(blockHeight, "mevshare:v0:matchBids"); + Suave.Bid[] memory allShareUserBids = Suave.fetchBids(blockHeight, "mevshare:v0:unmatchedBundles"); + + if (allShareUserBids.length == 0) { + revert Suave.PeekerReverted(address(this), "no bids"); + } + + Suave.Bid[] memory allBids = new Suave.Bid[](allShareUserBids.length); + for (uint256 i = 0; i < allShareUserBids.length; i++) { + // TODO: sort matches by egp first! + Suave.Bid memory bidToInsert = allShareUserBids[i]; // will be updated with the best match if any + for (uint256 j = 0; j < allShareMatchBids.length; j++) { + // TODO: should be done once at the start and sorted + Suave.BidId[] memory mergedBidIds = abi.decode( + Suave.confidentialRetrieve(allShareMatchBids[j].id, "mevshare:v0:mergedBids"), (Suave.BidId[]) + ); + if (idsEqual(mergedBidIds[0], allShareUserBids[i].id)) { + bidToInsert = allShareMatchBids[j]; + break; + } + } + allBids[i] = bidToInsert; + } + + EgpBidPair[] memory bidsByEGP = new EgpBidPair[](allBids.length); + for (uint256 i = 0; i < allBids.length; i++) { + bytes memory simResults = Suave.confidentialRetrieve(allBids[i].id, "mevshare:v0:ethBundleSimResults"); + uint64 egp = abi.decode(simResults, (uint64)); + bidsByEGP[i] = EgpBidPair(egp, allBids[i].id); + } + + // Bubble sort, cause why not + uint256 n = bidsByEGP.length; + for (uint256 i = 0; i < n - 1; i++) { + for (uint256 j = i + 1; j < n; j++) { + if (bidsByEGP[i].egp < bidsByEGP[j].egp) { + EgpBidPair memory temp = bidsByEGP[i]; + bidsByEGP[i] = bidsByEGP[j]; + bidsByEGP[j] = temp; + } + } + } + + Suave.BidId[] memory allBidIds = new Suave.BidId[](allBids.length); + for (uint256 i = 0; i < bidsByEGP.length; i++) { + allBidIds[i] = bidsByEGP[i].bidId; + } + + return buildAndEmit(blockArgs, blockHeight, allBidIds, "mevshare:v0"); + } + + function buildFromPool(Suave.BuildBlockArgs memory blockArgs, uint64 blockHeight) public returns (bytes memory) { + require(Suave.isConfidential()); + + Suave.Bid[] memory allBids = Suave.fetchBids(blockHeight, "default:v0:ethBundles"); + if (allBids.length == 0) { + revert Suave.PeekerReverted(address(this), "no bids"); + } + + EgpBidPair[] memory bidsByEGP = new EgpBidPair[](allBids.length); + for (uint256 i = 0; i < allBids.length; i++) { + bytes memory simResults = Suave.confidentialRetrieve(allBids[i].id, "default:v0:ethBundleSimResults"); + uint64 egp = abi.decode(simResults, (uint64)); + bidsByEGP[i] = EgpBidPair(egp, allBids[i].id); + } + + // Bubble sort, cause why not + uint256 n = bidsByEGP.length; + for (uint256 i = 0; i < n - 1; i++) { + for (uint256 j = i + 1; j < n; j++) { + if (bidsByEGP[i].egp < bidsByEGP[j].egp) { + EgpBidPair memory temp = bidsByEGP[i]; + bidsByEGP[i] = bidsByEGP[j]; + bidsByEGP[j] = temp; + } + } + } + + Suave.BidId[] memory allBidIds = new Suave.BidId[](allBids.length); + for (uint256 i = 0; i < bidsByEGP.length; i++) { + allBidIds[i] = bidsByEGP[i].bidId; + } + + return buildAndEmit(blockArgs, blockHeight, allBidIds, ""); + } + + function buildAndEmit( + Suave.BuildBlockArgs memory blockArgs, + uint64 blockHeight, + Suave.BidId[] memory bids, + string memory namespace + ) public virtual returns (bytes memory) { + require(Suave.isConfidential()); + + (Suave.Bid memory blockBid, bytes memory builderBid) = this.doBuild(blockArgs, blockHeight, bids, namespace); + + emit BuilderBoostBidEvent(blockBid.id, builderBid); + emit BidEvent(blockBid.id, blockBid.decryptionCondition, blockBid.allowedPeekers); + return bytes.concat(this.emitBuilderBidAndBid.selector, abi.encode(blockBid, builderBid)); + } + + function doBuild( + Suave.BuildBlockArgs memory blockArgs, + uint64 blockHeight, + Suave.BidId[] memory bids, + string memory namespace + ) public view returns (Suave.Bid memory, bytes memory) { + address[] memory allowedPeekers = new address[](2); + allowedPeekers[0] = address(this); + allowedPeekers[1] = Suave.BUILD_ETH_BLOCK; + + Suave.Bid memory blockBid = Suave.newBid(blockHeight, allowedPeekers, allowedPeekers, "default:v0:mergedBids"); + Suave.confidentialStore(blockBid.id, "default:v0:mergedBids", abi.encode(bids)); + + (bytes memory builderBid, bytes memory payload) = Suave.buildEthBlock(blockArgs, blockBid.id, namespace); + Suave.confidentialStore(blockBid.id, "default:v0:builderPayload", payload); // only through this.unlock + + return (blockBid, builderBid); + } + + function emitBuilderBidAndBid(Suave.Bid memory bid, bytes memory builderBid) + public + returns (Suave.Bid memory, bytes memory) + { + emit BuilderBoostBidEvent(bid.id, builderBid); + emit BidEvent(bid.id, bid.decryptionCondition, bid.allowedPeekers); + return (bid, builderBid); + } + + function unlock(Suave.BidId bidId, bytes memory signedBlindedHeader) public view returns (bytes memory) { + require(Suave.isConfidential()); + + // TODO: verify the header is correct + // TODO: incorporate protocol name + bytes memory payload = Suave.confidentialRetrieve(bidId, "default:v0:builderPayload"); + return payload; + } } contract EthBlockBidSenderContract is EthBlockBidContract { - string boostRelayUrl; - - constructor(string memory boostRelayUrl_) { - boostRelayUrl = boostRelayUrl_; - } - - function buildAndEmit(Suave.BuildBlockArgs memory blockArgs, uint64 blockHeight, Suave.BidId[] memory bids, string memory namespace) public virtual override returns (bytes memory) { - require(Suave.isConfidential()); - - (Suave.Bid memory blockBid, bytes memory builderBid) = this.doBuild(blockArgs, blockHeight, bids, namespace); - Suave.submitEthBlockBidToRelay(boostRelayUrl, builderBid); - - emit BidEvent(blockBid.id, blockBid.decryptionCondition, blockBid.allowedPeekers); - return bytes.concat(this.emitBid.selector, abi.encode(blockBid)); - } + string boostRelayUrl; + + constructor(string memory boostRelayUrl_) { + boostRelayUrl = boostRelayUrl_; + } + + function buildAndEmit( + Suave.BuildBlockArgs memory blockArgs, + uint64 blockHeight, + Suave.BidId[] memory bids, + string memory namespace + ) public virtual override returns (bytes memory) { + require(Suave.isConfidential()); + + (Suave.Bid memory blockBid, bytes memory builderBid) = this.doBuild(blockArgs, blockHeight, bids, namespace); + Suave.submitEthBlockBidToRelay(boostRelayUrl, builderBid); + + emit BidEvent(blockBid.id, blockBid.decryptionCondition, blockBid.allowedPeekers); + return bytes.concat(this.emitBid.selector, abi.encode(blockBid)); + } }