Send Transactions
Sending transactions requires that you use either the Golang or Typescript SDKs we have made available. Confidential Compute Requests (CCRs) are not EIP-1193 compatible so you won't be able to sign CCRs directly from Metamask or providers like it.
The previous tutorial illustrated how to deploy a series of example contracts using the Golang SDK and, in the first tutorial, you setup SUAVE and then ran a script to "test the devnet".
This tutorial will reiterate how contract deployment works, and show you how to craft the transactions that occur in this script.
MEV-Share Exampleβ
The suave-geth
repo has an example version of MEV-Share. The script we'll be walking through does four things:
- deploy the MEV-Share contract
- sign and submit a transaction to be back run
- grab the extracted hint from onchain
- sign a back run transaction and submit it
Step-by-Step Breakdownβ
Let's break down what this script is doing.
Step 1: Create and Fund Test Accountsβ
testAddr1 = generatePrivKey()
testAddr2 = generatePrivKey()
if err := fundAccount(mevmClt, testAddr1.Address(), fundBalance); err != nil {
return err
}
fmt.Printf("- Funded test account: %s (%s)\n", testAddr1.Address().Hex(), fundBalance.String())
- Generate two new private keys representing test accounts.
- Fund these test accounts with a specified amount of Ether to simulate transactions.
Step 2: Deploy MEV-Share Contractβ
// Code snippet related to the step
txnResult, err := sdk.DeployContract(mevShareArtifact.Code, mevmClt)
- Deploy the MEV-Share smart contract to the SUAVE chain using the bytecode and a client instance.
- Confirm the deployment by checking the transaction receipt status.
Step 3: Send Confidential Compute Requestβ
The word "bid" is an artefact of early development. It refers to a generic "Data Identifier" used when operating on confidential data. In this context, a "bid" refers to the Ethereum L1 transaction data the is stored by the MEVM when we send it a CCR.
In practice, this transaction would be sent by a user of another blockchain who wants to share any MEV the transaction might create in that domain with searchers listening for hints on SUAVE.
refundPercent := 10
bundle := &types.SBundle{
Txs: types.Transactions{ethTxn1},
RevertingHashes: []common.Hash{},
RefundPercent: &refundPercent,
}
bundleBytes, _ := json.Marshal(bundle)
targetBlock := uint64(1)
allowedPeekers := []common.Address{mevShareContract.Address()}
confidentialDataBytes, _ := bundleBidContract.Abi.Methods["fetchBidConfidentialBundleData"].Outputs.Pack(bundleBytes)
txnResult, err := mevShareContract.SendTransaction("newBid", []interface{}{targetBlock + 1, allowedPeekers, []common.Address{}}, confidentialDataBytes)
if err != nil {
return err
}
- Craft a Confidential Compute Request using the MEV-Share contract function
newBid
. - Serialize the bundle of transactions to include in the CCR.
- Send the CCR to the specified Kettle and validate its inclusion.
Step 4: Send Backrunβ
In practice, this transaction would be sent by a searcher listening for hints on SUAVE, after they have received the hint emitted by the newBid
function that the user called above, crafted a backrun based on that hint, which they want a builder to include on the original domain where the user wants to transact.
backRunBundle := &types.SBundle{
Txs: types.Transactions{ethTxnBackrun},
RevertingHashes: []common.Hash{},
}
backRunBundleBytes, _ := json.Marshal(backRunBundle)
confidentialDataMatchBytes, _ := bundleBidContract.Abi.Methods["fetchBidConfidentialBundleData"].Outputs.Pack(backRunBundleBytes)
// backrun inputs
targetBlock := uint64(1)
allowedPeekers := []common.Address{mevShareContract.Address()}
txnResult, err := mevShareContract.SendTransaction("newMatch", []interface{}{targetBlock + 1, allowedPeekers, []common.Address{}, bidId}, confidentialDataMatchBytes)
if err != nil {
return err
}
- Create a backrun transaction bundle that will be sent after the initial bid. This is also a Confidential Compute Request.
- Serialize the backrun bundle and use the MEV-Share contract's
newMatch
function to submit it. - Verify the transaction receipt to ensure the backrun was successful.
What is not included in this script is the final step in the e2e example. We need to send this matched bundle - which includes both the original transaction from the user and the backrun from the searcher - to a builder so that it can be included in a block on the target domain. Rigil includes the ability to send such bundles to centralized relays for inclusion. There is builder solidity code which demonstrates how this might occur here.
Resultβ
If all has been successfully run you should see the following in your terminal:
suave-geth$ go run suave/devenv/cmd/main.go
Step 0: Create and fund test accounts
- Funded test account: 0x66d5a8D6B34329c0639071275b3d78D29e11EbC6 (100000000)
Step 1: Deploy mev-share contract
- Mev share contract deployed: 0x8f21Fdd6B4f4CacD33151777A46c122797c8BF17
Step 2: Send bid
- Bid sent at txn: 0xb49debcdead2b306d6ab6282b88fdad7c8d6a33d87df34b79f56d141eae7c08a
- Bid id: 30bbc65298f24e67aaf5c95bf5f0686c
Step 3: Send backrun
- Backrun sent at txn: 0xcf7880e61e94aaab48c60655c321716ecab6edab752586448b0412e93a969889
- Backrun bid id: db98b83d02694fc2b13c042ad22c233