ethers-provider-flashbots-bundle
This repository contains the FlashbotsBundleProvider
ethers.js provider, an additional Provider
to ethers.js
to enable high-level access to eth_sendBundle
and eth_callBundle
rpc endpoint on mev-relay. mev-relay
is a hosted service; it is not necessary to run mev-relay
or mev-geth
to proceed with this example.
Flashbots-enabled relays and miners expose two new jsonrpc endpoints: eth_sendBundle
and eth_callBundle
. Since these are non-standard endpoints, ethers.js and other libraries do not natively support these requests (like getTransactionCount
). In order to interact with these endpoints, you will need access to another full-featured (non-Flashbots) endpoint for nonce-calculation, gas estimation, and transaction status.
One key feature this library provides is payload signing, a requirement to submit Flashbot bundles to the mev-relay
service. This library takes care of the signing process via the authSigner
passed into the constructor. Read more about relay signatures here
This library is not a fully functional ethers.js implementation, just a simple provider class, designed to interact with an existing ethers.js v5 installation.
Example
Install ethers.js and the Flashbots ethers bundle provider
npm install --save ethers
npm install --save @flashbots/ethers-provider-bundle
Open up a new TypeScript file (this also works with JavaScript if you prefer)
import { providers, Wallet } from "ethers";
import { FlashbotsBundleProvider } from "@flashbots/ethers-provider-bundle";
const provider = new providers.JsonRpcProvider({ url: ETHEREUM_RPC_URL }, 1)
const authSigner = Wallet.createRandom();
const flashbotsProvider = await FlashbotsBundleProvider.create(
provider,
authSigner
)
From here, you have a flashbotsProvider
object setup which can now perform either an eth_callBundle
(via simulate()
) or eth_sendBundle
(via sendBundle
). Each of these functions act on an array of Bundle Transactions
Bundle Transactions
Both simulate
and sendBundle
operate on a bundle of strictly-ordered transactions. While the miner requires signed transactions, the provider library will accept a mix of pre-signed transaction and TransactionRequest + Signer
transactions (which it will estimate, nonce-calculate, and sign before sending to the mev-relay
)
const wallet = new Wallet(PRIVATE_KEY)
const transaction = {
to: CONTRACT_ADDRESS,
data: CALL_DATA
}
const transactionBundle = [
{
signedTransaction: SIGNED_ORACLE_UPDATE_FROM_PENDING_POOL
},
{
signer: wallet,
transaction: transaction
}
]
Block Targeting
The last thing required for sendBundle()
is block targeting. Every bundle specifically references a single block. If your bundle is valid for multiple blocks (including all blocks until it is mined), sendBundle()
must be called for every block, ideally on one of the blocks immediately prior. This gives you a chance to re-evaluate the opportunity you are capturing and re-sign your transactions with a higher nonce, if necessary.
The block should always be a future block, never the current one.
const targetBlockNumber = (await provider.getBlockNumber()) + 1
Simulate and Send
Now that we have:
- Flashbots Provider
flashbotsProvider
- Bundle of transactions
transactionBundle
- Block Number
targetBlockNumber
We can run simulations and submit directly to miners, via the mev-relay
.
Simulate:
const signedTransactionBundle = await flashbotsProvider.signBundle(transactionBundle)
const simulation = await flashbotsProvider.simulate(signedTransactions, targetBlockNumber)
console.log(JSON.stringify(simulation, null, 2))
Send:
const flashbotsTransactionResponse = await flashbotsProvider.sendBundle(
transactionBundle,
targetBlockNumber,
)
FlashbotsTransactionResponse
After calling sendBundle
, this provider will return a Promise of an object with helper functions related to the bundle you submitted.
These functions return metadata available at transaction submission time, as well as the following functions which can wait, track, and simulate the bundle's behavior.
bundleTransactions()
- An array of transaction descriptions sent to the relay, including hash, nonce, and the raw transaction.receipts()
- Returns promise of an array of transaction receipts corresponding to the transaction hashes that were relayed as part of the bundle. Will not wait for block to be mined; could return incomplete informationwait()
- Returns a promise which will wait for target block number to be reached OR one of the transactions to become invalid due to nonce-issues (including, but not limited to, one of the transactions from your bundle being included too early). Returns the wait resolution as a status enumsimulate()
- Returns a promise of the transaction simulation, once the proper block height has been reached. Use this function to troubleshoot failing bundles and verify miner profitability
Optional eth_sendBundle arguments
Beyond target block number, an object can be passed in with optional attributes:
{
minTimestamp,
maxTimestamp,
revertingTxHashes: [tx1, tx2]
}
minTimestamp / maxTimestamp
While each bundle targets only a single block, you can add a filter for validity based on the block's timestamp. This does not allow for targeting any block number based on a timestamp or instruct miners on what timestamp to use, it merely serves as a secondary filter.
If your bundle is not valid before a certain time or includes an expiring opportunity, setting these values allows the miner to skip bundle processing earlier in the phase.
Additionally, you could target several blocks in the future, but with a strict maxTimestamp, to ensure your bundle is considered for inclusion up to a specific time, regardless of how quickly blocks are mined in that timeframe.
Reverting Transaction Hashes
Transaction bundles will not be considered for inclusion if they include any transactions that revert or fail. While this is normally desirable, there are some advanced use-cases where a searcher might WANT to bring a failing transaction to the chain. This is normally desirable for nonce management. Consider:
Transaction Nonce #1 = Failed (unrelated) token transfer
Transaction Nonce #2 = DEX trade
If a searcher wants to bring #2 to the chain, #1 must be included first, and its failure is not related to the desired transaction #2. This is especially common during high gas times.
Optional parameter revertingTxHashes
allows a searcher to specify an array of transactions that can (but are not required to) revert.
Paying for your bundle
In addition to paying for a bundle with gas price, bundles can also conditionally pay a miner via:
block.coinbase.transfer(_minerReward)
or
block.coinbase.call{value: _minerReward}("");
(assuming _minerReward is a solidity uint256
with the wei-value to be transferred directly to the miner)
The entire value of the bundle is added up at the end, so not every transaction needs to have a gas price or block.coinbase
payment, so long as at least one does, and pays enough to support the gas used in non-paying transactions.
Note: Gas-fees will ONLY benefit your bundle if the transaction is not already present in the mempool. When including a pending transaction in your bundle, it is similar to that transaction having a gas price of 0
; other transactions in your bundle will need to pay more for the gas it uses.
How to run demo.ts
Included is a simple demo of how to construct the FlashbotsProvider with auth signer authentication and submit a [non-functional] bundle. This will not yield any mev, but serves as a sample initialization to help integrate into your own functional searcher.
Flashbots on Goerli
To test Flashbots before going to mainnet, you can use the Goerli Flashbots relay, which works in conjunction with a Flashbots-enabled Goerli validator. Flashbots on Goerli requires two simple changes:
- Ensure your genericProvider passed in to the FlashbotsBundleProvider constructor is connected to Goerli (gas estimates and nonce requests need to correspond to the correct chain):
import { providers } from 'ethers'
const provider = providers.getDefaultProvider('goerli')
- Set the relay endpoint to
https://relay-goerli.flashbots.net/
const flashbotsProvider = await FlashbotsBundleProvider.create(
provider,
authSigner,
'https://relay-goerli.flashbots.net/',
'goerli')