Just-In-Time Gas
How to use just-in-time gas to execute cross-chain transactions
There are some actions and contracts where cross-chain execution with a relayer doesn’t work. For example, most ‘sell’ actions depend on msg.sender
- Selling a nft
- Swapping an erc20
- Bidding in an auction
To get around this, you can simply bridge a small amount of ETH for gas and have the user execute the transaction themselves. This is only now possible because of how fast and cheap Relay makes bridging.
We will soon add functionality to our SDK to handle this flow completely, but until then, this guide will demonstrate how you could implement this ux yourself.
Overview of steps:
-
Step 1: Estimate gas needed on the Destination Chain
-
Step 2: Bridge Just-In-Time gas money from Origin to Destination chain
-
Step 3: Execute transaction/s on the Destination chain
-
Step 4 (optional): Bridge back proceeds to Origin chain
The following example assumes you are set up with the Relay SDK as well as viem. You can skip ahead to the full code and live example here.
Step 1: Estimate gas needed on the Destination Chain
const estimatedGas = await publicClient.estimateContractGas({
...wethContract,
functionName: 'withdraw',
args: [wethBalance],
account: address,
})
const gasPrice = await publicClient.getGasPrice()
const totalGasEstimation = estimatedGas * gasPrice
const totalGasEstimationWithBuffer =
totalGasEstimation + (totalGasEstimation * BigInt(5)) / BigInt(100) // add 5% buffer to handle gas fluctuation
We start by using viem’s estimateContractGas to calculate the gas required to successfully execute the transaction we want to make on the Destination Chain. We also use getGasPrice to get the current price of gas on the Destination chain. We multiply these two values together to get the total gas estimate and add a 5% buffer to handle any fluctuation.
Step 2: Bridge Just-In-Time gas money from Origin to Destination chain
const quote = await relayClient.actions.getQuote({
chainId: baseSepolia.id,
toChainId: sepolia.id,
wallet,
value: totalGasEstimationWithBuffer.toString(),
})
await relayClient.actions.execute({
quote,
wallet,
onProgress(steps) {
setStep(getCurrentStepDescription(steps))
},
})
We use the Relay SDK’s bridge action to bridge the gas money from the Origin to the Destination chain.
Step 3: Execute transaction/s on the Destination chain
const { request } = await publicClient.simulateContract({
...wethContract,
account: address,
functionName: 'withdraw',
args: [wethBalance],
chain: sepolia,
gas: estimatedGas,
})
const hash = await wallet.writeContract(request)
await publicClient.waitForTransactionReceipt({
hash,
})
In this example we withdraw
some WETH with viem’s writeContract and wait for the transaction to complete with waitForTransactionReceipt.
Step 4 (optional): Bridge back proceeds to Origin chain
const destinationEthBalance = await getBalance(wagmiConfig, {
address: address,
chainId: sepolia.id,
})
const quote = await relayClient.actions.getQuote({
chainId: sepolia.id,
toChainId: baseSepolia.id,
wallet,
value: destinationEthBalance.value.toString()
})
const bufferedGasFee = BigInt(quote?.fees?.gas ?? 0) * (BigInt(100) + BigInt(5)) / BigInt(100) // add 5% buffer to handle flucation
const amountToBridgeBack =
destinationEthBalance.value -
bufferedGasFee -
BigInt(quote?.fees?.relayer ?? 0)
await relayClient.actions.execute({
quote,
wallet,
onProgress(steps) {
setStep(getCurrentStepDescription(steps))
},
})
This step is optional and the amount you bridge back would depend on your use case. In this example, we attempt to bridge back all of the ETH that the user has on the Destination chain. We first fetch the user’s ETH balance on Destination. We then calculate the required fees in order to bridge that ETH back (gas fee + relayer fee). Once we subtract those fees from the user’s Destination ETH balance, we get the total amount that we can afford to bridge and we can execute the bridge action.
Example
Check out this codesandbox to see a live example of just-in-time bridging.