Send ERC-4337 UserOperation
A quick overview how to use ERC-4337 easily with Blocto SDK
Introduction to ERC-4337
ERC-4337 is an Ethereum standard that achieves account abstraction on the protocol without any consensus-layer changes. ERC-4337 makes it possible to transact and create contracts in a single contract account. It has four main parts: UserOperation, Bundler, EntryPoint, and Contract Account. There are also optional components called Paymasters and Aggregators that can be added. Before we start, let's take a look with these concepts.
If you already familiar with ERC-4337, you can jump to next section How to use ERC-4337 with Blocto
UserOperations
UserOperations are like pretend transactions that help you perform actions with contract accounts. They are made by your app.
Bundlers
Bundlers are like messengers who gather UserOperations from a mempool and send them to the EntryPoint contract on the blockchain.
EntryPoint
EntryPoint is a special smart contract that takes care of checking and carrying out the instructions in the transactions.
Contract Accounts
Contract Accounts are special accounts owned by users that use smart contracts.
Paymasters and Aggregators
Paymasters are optional accounts that can help sponsor transactions for Contract Accounts.
Aggregators are also optional accounts that can check the signatures for multiple Contract Accounts at once.
How to use ERC-4337 with Blocto
Although we mention many complicated details with ERC-4337 above, sending ERC-4337 UserOperation is extremely easy with Blocto. We have already done most of works so you can start sending your first UserOperation
within 5 mins.
Only support the accounts which create EVM wallets after July 10, 2023.
EIP-4337 Supported Chain List
Ethereum
1
Yes
Ethereum Sepolia Testnet
11155111
Yes
Arbitrum Mainnet
42161
Yes
Optimism Mainnet
10
Yes
Polygon Mainnet
137
Yes
Sending like Regular Transaction
Here comes the magic: since Blocto is already a contract wallet, the Blocto SDK will automatically transform your transaction request to an ERC-4337 UserOperation ready transaction if supported. Users can choose from ERC-4337 way or traditional transaction way. That means after integrate with Blocto SDK, you are already support the "4337 mode" 🎉
To know more about how the magic works, let's start from how regular transactions work in an EVM network. A transaction typically consists of three main parts: to
, value
, and an optional data
field.
To understand this, let's look at a simple example of sending 1 ETH (the native token) from account A to account B. In this case, the transaction will contain the following information:
To: Address of account B
This is the address of account B.Value: 1 ETH
It will be 1 ETH, indicating the amount being sent.Data: null
Since it's a straightforward ETH transfer, the data field will be empty or null.
Now, let's consider a different scenario where account A wants to send 100 USDC (an ERC-20 token) to account B. ERC-20 tokens are essentially smart contracts that keep track of balances for different addresses. In this case, the transaction would look like this:
To: Address of USDC contract
Since you want to interact with USDC contract, this will be the address of the USDC smart contract.Value: 0 ETH
Since we're dealing with a contract interaction, the value will be 0 ETH. The transfer of value is represented through theData
param send to token contract.Data: Instructions to transfer 100 USDC from account A to account B
The data field will contain instructions specifying the transfer of 100 USDC from account A to account B.
The data
field is how we send instructions to a smart contract. The callData
in an ERC-4337 UserOperation
is no different: It's also the instructions send to the sender
smart contract address. That's why we can transform regular transaction to UserOperation.
Sending UserOperation
Only support with Blocto JavaScript SDK version^0.5.0
.
All components of ERC-4337 revolve around a pseudo-transaction object called a UserOperation which is used to execute actions through a smart contract account. It captures the intent of what the user wants to do. This isn't to be mistaken for a regular transaction type.
callData
bytes
Data that's passed to the sender
for execution.
true
sender
address
The address of the smart contract account to send UserOperation. If provided, should be same as login address.
false
nonce
uint256
Anti-replay protection. No need to provide since Blocto will handle it for you.
false (Will be Ignored)
initCode
bytes
Code used to deploy the account if not yet on-chain. No need to provide since Blocto will handle it for you.
false (Will be Ignored)
callGasLimit
uint256
Gas limit for the execution phase.
false
verificationGasLimit
uint256
Gas limit for the verification phase.
false
preVerificationGas
uint256
Gas to compensate the bundler for the overhead to submit a UserOperation.
false
maxFeePerGas
uint256
Similar to EIP-1559 max fee.
false
maxPriorityFeePerGas
uint256
Similar to EIP-1559 priority fee.
false
paymasterAndData
bytes
Paymaster contract address and any extra data the paymaster contract needs for verification and execution. When set to 0x
or the zero address, no paymaster is used.
false
signature
bytes
Used to validate a UserOperation during verification. No need to provide since Blocto will handle it for you.
false (Will be Ignored)
How to Encode callData
callData
You might find that callData
is the only required field of UserOperation Object to send with Blocto. The callData
is the instructions for smart contract and should be encoded as bytesLike
string.
Luckily, we have some helpful tools available to make the encoding process easier. One such tool is ethers.js
. All we need is the contract's Application Binary Interface (ABI) to work with it. The ABI provides ethers.js with the necessary information for encoding and decoding.
When using ethers.js, you have two options for the ABI: a human-readable ABI or a solidity JSON ABI. If you have the Solidity compiler, you can export the JSON ABI from there. Both types of ABI can be used with ethers.js to simplify the encoding process.
The human-readable ABI of Blocto Account Contract provided here:
And we assume an ERC-20 token contract has human-readable ABI look like this:
An ABI can be fragments and does not have to include the entire interface. That means you can includes only the parts you want to use.
With these two ABIs we can encode a callData
for our UserOperation Object that sends an amount
of an ERC-20 token to another account's address:
Why are there two ABIs?
The Blocto account is a Smart Contract Wallet. In example above, we use our Smart Contract Wallet to interact with the ERC-20 smart contract. Which is why we need encoding the data twice within the callData
.
If we only wanted to send an amount
of ETH (the native token) to another account's address, the code would look like this:
Just like the simplest regular transaction, there is no data field. Which means we don't need to encode any data within the userOp's callData
.
Submit UserOperation
After you get encoded callData, you can submit your UserOperation using JSON-RPC API:
Or using type-safe sendUserOperation
function provided:
Don't forget to add error handling. Sending user operation may fail due to user using old version of account contract, target EVM chain unsupport or some other reason.
Other RPC Methods of Blocto ERC-4337 bundler
After submit UserOperation succesfully, the next step is deal with the userOpHash
you get to see if success or not. Here's some related method provided by Blocto can help you.
Get UserOperation receipt
Fetches the UserOperation receipt based on a given userOpHash
returned from eth_sendUserOperation
.
Get UserOperation by hash
Fetches the UserOperation and transaction context based on a given userOpHash
returned from eth_sendUserOperation
.
Sample Code
Last updated