SDK for XGA bidders and users. This guide provides information on how to interact with the auction contracts and the L2 RPC endpoints.
Bidders
- Auction bidders can provide custom strategies either from an offchain bot or a contract.
- OpenBidder contract is a full working example
- New bidders must be registered with the protocol. Please contact [email protected] for registration.
Users
- Private transactions and bundles can be passed to beta blocks through the deployed OpenBidder.
- Beta transactions can be constructed and submitted with provided
cast
orpython
libs.
- Environment vars
- Beta Transactions
- Bridge
- Using OpenBidder
- RPC Endpoints
- Bundle JSON Requests and Responses
- Contracts Addresses
- Bidding
- Send Tx and Bundle
- Bidder Contracts
- Bundler Examples
- Contributing
- License
Copy .env.example
to .env
and change as necessary. Testnet and mainnet details have been provided.
MaxFeePerGas
= 0
An important difference with beta block transactions is that the MaxFeePerGas
is set to zero. Instead this fee is set for the L2 bid.
Make sure to setup env vars before signing txs.
Using cast (see script):
./open-bidder-contracts/script/sign-tx.sh
Using python:
poetry run python sign-eth-tx-py/sign.py <contract> <value> <function_sig> <args>
Before bidding, you will need to bridge ETH to L2 for bids.
# Fund L2 address by sending ETH to the bridge address.
cast send -i $L1_BRIDGE --value 0.01ether --rpc-url $L1_RPC_URL --private-key $PRIVATE_KEY
# Confirming the bridged Holesky ETH on L2
cast balance $WALLET_ADDRESS --rpc-url $L2_RPC_URL
Make sure to setup env vars before using open bidder, including WEI_PER_GAS
. Scripts are provided for submitting a private transaction to OpenBidder.
Using cast (see script):
./open-bidder-contracts/script/sign-submit-tx.sh
Using python:
poetry run beta_bundles_py/main.py
-
L2 RPC:
- Description: L2 Node RPC
- URL: https://xga-api.securerpc.com/v1
- Methods: eth_*
- ChainId: 7890785
-
Beta bundle RPC:
- Description: Beta bundle submission RPC
- URL: https://mainnet-auction.securerpc.com/
- Method: mev_sendBetaBundle
- Parameters:
txs
: List of txs as bundle e.g. [0x2323...,]slot
: slot number e.g. "11282389"
- ChainId: 1
-
L2 RPC (TESTNET):
- Description: L2 Node RPC (Testnet)
- URL: https://holesky-op.xga.com/v1
- Methods: eth_*
- ChainId: 42169
-
Beta bundle RPC (TESTNET):
- Description: Beta bundle submission RPC
- URL: https://holesky-rpc.xga.com/v1
- Method: mev_sendBetaBundle
- Parameters:
txs
: List of txs as bundle e.g. [0x2323...,]slot
: slot number e.g. "11282389"
- ChainId: 17000
{
"jsonrpc": "2.0",
"method": "mev_sendBetaBundle",
"params": [
{
"txs": [0x... ],
"slot": "1001"
}
],
"id": 8
}
{
'jsonrpc': '2.0',
'id': 1,
'method': 'mev_sendBetaBundle',
'result': '0x79e5cba7876f532218ac35a357209800be2362dd2e3f1e6dc5974698f0d7cee4'
}
Contract | Address |
---|---|
L1StandardBridge | 0x490B959870889D5FA0B329431683B8B3e850DD95 |
L1StandardBridge (testnet) | 0x3Ae5Ca0B05bE12d4FF9983Ed70D86de9C34e820C |
Contract | Address |
---|---|
WETH | 0x4200000000000000000000000000000000000006 |
Auctioneer | 0x86Bc75A43704E38f0FD94BdA423C50071fE17c99 |
SettlementHouse | 0x80C5FfF824d14c87C799D6F90b7D8e0a715bd33C |
Accountant | 0xE9B82BD9e839f1087d4017b4F7E928E0CADB2AF9 |
Auctioneer (testnet) | 0x82052435119AedBEC6e237B5414880bF388F1B63 |
SettlementHouse (testnet) | 0x513844F4425d54beC19B0594AE072e49bEbBF388 |
Accountant (testnet) | 0x4641748a26D6fca7C11B0366C443dc83d6eb6Be1 |
Only registered bidders can participate in the auction. Operators can onboard new bidders through the contract. Please contact Mev Protocol to proceed.
The following code snippet uses Foundry as example.
cast send $WETH "deposit()" --value 1000gwei --private-key $PRIVATE_KEY --rpc-url $L2_RPC_URL
# allow Auctioneer to spend WETH
cast send $WETH "approve(address,uint256)" $AUCTIONEER $(cast max-uint) --private-key $PRIVATE_KEY --rpc-url $L2_RPC_URL
cast call $AUCTIONEER "getOpenAuctions()" --rpc-url $L2_RPC_URL
To check for bidderId when registered, call IdMap
on the contract:
function IdMap(address bidder) external view returns (uint8 id);
Bids are packed by price, amount, bidderId
/**
* @dev Packed Bid details into a uint256 for submission.
*
* @param bidPrice Price per item.
* @param itemsToBuy Items to buy in the auction.
* @param bidderId Id for bidder
* @return packedBid for auction submission
*/
function packBid(uint256 bidPrice, uint256 itemsToBuy, uint256 bidderId)
external
pure
returns (uint256 packedBid);
For example:
# Packing a bid for 21,000 gas and 0.01 gwei per gas
cast call $AUCTIONEER \
"packBid(uint128,uint120,uint8)" 0.01gwei 21000 $(cast call $AUCTIONEER "IdMap(address)" $WALLET_ADDRESS --rpc-url $L2_RPC_URL) \
--rpc-url $L2_RPC_URL
# Placing a bid
cast send $AUCTIONEER \
"bid(uint256,uint256[] memory)" SLOT_NUMBER ["0x...(Packed Bid)"] \
--private-key $PRIVATE_KEY \
--rpc-url $L2_RPC_URL
After an auction is closed, bidders can query their bid results:
/**
* @dev Retrieve information about a bidder after auction settlement.
*
* @param slot The slot identifier of the auction.
* @param bidder The address of the bidder for whom information is requested.
* @return itemsBought The number of items bought by the bidder in the specified auction.
* @return amountOwed The amount owed by the bidder for the items bought in the specified auction.
*
* Requirements:
* - The auction must have been settled.
* - The provided `bidder` address must be valid and have participated in the auction.
*
*/
function getBidderInfo(uint256 slot, address bidder)
external
view
returns (uint120 itemsBought, uint128 amountOwed);
cast call $AUCTIONEER \
"getBidderInfo(uint256,address)" SLOT_NUMBER $WALLET_ADDRESS \
--rpc-url $L2_RPC_URL
Bidder contract requirements:
- Sufficient WETH balance and approval
- Registered by operator
- Implement
Bidder
interface
A minimal viable bidder is provided below, this contract will bid for every opened auctions:
/// SPDX-License-Identifier: UPL-1.0
pragma solidity ^0.8.25;
import {WETH} from "solmate/tokens/WETH.sol";
import {ERC6909} from "solmate/tokens/ERC6909.sol";
interface Bidder {
/**
* @dev Get the bid from a bidder for a specific slot and round.
* @param slot The auction slot.
* @return packedBids Array of bids (in a packed format). uint256(uint128(bidPrice),uint120(itemsToBuy),uint8(bidderId))
*/
function getBid(uint256 slot) external view returns (uint256[] memory packedBids);
}
interface SettlementHouse {
function submitBundle(uint256 slot, uint256 amount, bytes32[] calldata hashes) external;
}
/// @title MockBidder
contract MockBidder is Bidder {
uint256[] public bids;
ERC6909 auctioneer;
SettlementHouse house;
WETH weth;
constructor(WETH _weth, address _auctioneer, address settlement) {
weth = _weth;
auctioneer = ERC6909(_auctioneer);
house = SettlementHouse(settlement);
weth.approve(_auctioneer, type(uint256).max);
}
function setBids(uint256[] memory newBids) public {
bids = newBids;
}
function getBid(uint256) external view returns (uint256[] memory packedBids) {
return bids;
}
function submit(uint256 slot, uint256 amount, bytes32[] calldata hashes) external {
auctioneer.approve(address(house), slot, amount);
house.submitBundle(slot, amount, hashes);
}
}
cast call $AUCTIONEER \
"balanceOf(address,uint256)" $WALLET_ADDRESS SLOT_NUMBER \
--rpc-url $L2_RPC_URL
- Prioriy fee MUST be ZERO
- RPC URL should be L1
- Return signed transaction
# trasnfer 0.01 ether to self
cast mktx $WALLET_ADDRESS \
--value 0.01ether \
--priority-gas-price 0 \
--private-key $PRIVATE_KEY \
--rpc-url $L1_RPC_URL
Use the mev_sendBetaBundle
method to send the bundle to L1 RPC. The bundle will be stored in L1 RPC, and L1 RPC will return the bundle hash
.
curl -X POST \
-H 'Content-Type: application/json' \
-d '{"jsonrpc":"2.0","id":1,"method":"mev_sendBetaBundle","params":[{"slot": "SLOT_NUMBER","txs":["0x..SIGNED TX"]}]}' \
$BETA_BUNDLE_RPC
/*
* @dev Function to submit a bundle of transactions.
* @param slot The slot of the futures token being deposited
* @param amountOfGas Amount of futures tokens to deposit
* @param bundleHashes List of bundle hashes
*/
function submitBundle(uint256 slot, uint256 amountOfGas, bytes32[] calldata bundleHashes) external
Submit the bundle hash
to SettlementHouse
. During this process, the submitBundle
in SettlementHouse
will check and burn future Gas tokens.
cast send $SETTLEMENT_HOUSE \
"submitBundle(uint256,uint256,bytes32[])" SLOT_NUMBER GAS_TOKEN_AMOUNT ["0x..BUNDLE_HASH"] \
--private-key $PRIVATE_KEY \
--rpc-url $L2_RPC_URL
The block builder will query the bundle hash in the SettlementHouse
contract. If a bundle in the bundle pool is in the SettlementHouse
contract, it will be included in the beta block by the block builder.
Source code for a fully functional open bidder is provided
Source code for a fully functional bundler is provided
Contributions are welcome! If you find any issues or have suggestions for improvements, feel free to open an issue or submit a pull request.
This project is licensed under the MIT License.