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

Network
Chain ID
EIP-4337 Support

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 the Data 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.

Field
Type
Description
Required

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

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:

// ABI of Blocto Account Contract
const bloctoAccountABI = [
  'function execute(address dest, uint256 value, bytes func)',
  'function executeBatch(address[] dest, uint256[] value, bytes[] func)',
];

And we assume an ERC-20 token contract has human-readable ABI look like this:

// Sample ABI of ERC20 contract you want to interact
const partialERC20TokenABI = [
  "function transfer(address to, uint amount) returns (bool)",
];

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:

const accountContract = new ethers.utils.Interface(bloctoAccountABI);
const erc20Token = new ethers.utils.Interface(partialERC20TokenABI);

const callData = accountContract.encodeFunctionData("execute", [
  erc20TokenContractAddress,
  ethers.constants.Zero,
  erc20Token.encodeFunctionData("transfer", [accountAddress, amount]),
]);

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:

const accountContract = new ethers.utils.Interface(bloctoAccountABI);

const callData = accountContract.encodeFunctionData("execute", [accountAddress, amount]);

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:

const userOpHash = await bloctoSDK.ethereum
  .request({
    method: 'eth_sendUserOperation',
    params: [{ callData }],
  })

Or using type-safe sendUserOperation function provided:

const userOpHash = await bloctoSDK.ethereum.sendUserOperation({ callData })

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.

Request
const result = await bloctoSDK.ethereum
  .request({
    method: 'eth_getUserOperationReceipt',
    params: [userOpHash],
  })
Response
{
  "jsonrpc": "2.0",
  "id": 1,
  "method": "eth_getUserOperationReceipt",
  "params": [
    // The hash of the UserOperation
    userOpHash,
    // The EntryPoint address
    entryPoint,
    // The contract account address
    sender,
    // nonce of the UserOperation
    nonce,
    // The paymaster address
    paymaster,
    // The actual amount paid for this UserOperation
    actualGasCost,
    // The total gas used by this UserOperation
    actualGasUsed,
    // Boolean value indicating if the execution completed without revert
    success,
    // If revert occurred, this is the reason
    reason,
    // logs generated by this UserOperation only
    logs,
    // The TransactionReceipt object for the entire bundle.
    receipt
  ]
}

Get UserOperation by hash

Fetches the UserOperation and transaction context based on a given userOpHash returned from eth_sendUserOperation.

Request
const result = await bloctoSDK.ethereum
  .request({
    method: 'eth_getUserOperationByHash',
    params: [userOpHash],
  })
Response
{
  "jsonrpc": "2.0",
  "id": 1,
  "method": "eth_getUserOperationByHash",
  "params": [
    // UserOperation object
    {
      sender,
      nonce,
      initCode,
      callData,
      callGasLimit,
      verificationGasLimit,
      preVerificationGas,
      maxFeePerGas,
      maxPriorityFeePerGas,
      paymasterAndData,
      signature
    },
    // The EntryPoint address
    entryPoint,
    // The block number this UserOperation was included in
    blockNumber,
    // The block hash this UserOperation was included in
    blockHash,
    // The transaction this UserOperation was included in
    transactionHash,
  ]
}

Sample Code

Last updated