Overview
This tutorial demonstrates how to use a multi session with viem and the Biconomy Smart Account with the @biconomy/account
SDK. The provided code assumes you have a Biconomy Paymaster API key, and a valid session setup during the previous step. The following is appropriately viewed from the perspective of a dapp, looking to make txs on a users behalf.
You can get your Biconomy Paymaster API key from the dashboard here.
Prerequisites
- Biconomy Paymaster API key
- A Bundler url if you don't want to use the testnet on, for Amoy you can use
https://bundler.biconomy.io/api/v2/80002/nJPK7B3ru.dd7f7861-190d-41bd-af80-6877f74b8f44
- A session granted by a user see the previous step
Step 1: Setup
import { polygonAmoy as chain } from "viem/chains";
import {
PaymasterMode,
createSessionSmartAccountClient,
createSession,
Rule,
Policy,
DEFAULT_ERC20_MODULE,
Session,
DEFAULT_ABI_SVM_MODULE,
getBatchSessionTxParams,
} from "@biconomy/account";
const nftAddress = "0x1758f42Af7026fBbB559Dc60EcE0De3ef81f665e";
const token = "0x747A4168DB14F57871fa8cda8B5455D8C2a8e90a";
const amount = parseUnits(".0001", 6);
const withSponsorship = {
paymasterServiceData: {
mode: PaymasterMode.SPONSORED,
skipPatchCallData: true, // Required when mode is set to ERC20
},
};
Step 2: Create the sessionSmartAccountClient
The createSessionSmartAccountClient helper here is very similar to the createSmartAccountClient, in that it returns a BiconomySmartAccountV2, but there are subtle differences in the configuration. createSessionSmartAccountClient expects no signer and a mandatory smartAccountAddress in the config (so that a dapp can emulate the users smartAccount without the original signer).
const emulatedUsersSmartAccount = await createSessionSmartAccountClient(
{
accountAddress: sessionStorageClient.smartAccountAddress, // Dapp can set the account address on behalf of the user
bundlerUrl,
paymasterUrl,
chainId,
},
session, // a) Full Session, b) storage client or c) the smartAccount address (if using default storage for your environment)
true // if in batch session mode
);
Step 3: Prepare the txs
Next we can use the users smartAccount in execute txs. Txs generated for the emulatedSmartAccount must be relevant to the policies over which the user has previously signed.
const transferTx: Transaction = {
to: token,
data: encodeFunctionData({
abi: parseAbi(["function transfer(address _to, uint256 _value)"]),
functionName: "transfer",
args: [recipient, amount],
}),
};
const nftMintTx: Transaction = {
to: nftAddress,
data: encodeFunctionData({
abi: parseAbi(["function safeMint(address _to)"]),
functionName: "safeMint",
args: [sessionStorageClient.smartAccountAddress],
}),
};
Step 4: Execute txs on the users behalf
When using multiple session and sending a tx the batchSessionParams must be sent with each tx. BatchSessionParams are described here. The order of the sessionValidationTypes
array must correspond with the modules for the txs being sent. A utility function getBatchSessionTxParams is provided
const txs = [transferTx, nftMintTx];
const batchSessionParams = await getBatchSessionTxParams(
[transferTx, nftMintTx],
[0, 1],
// Order must match the order in which corresponding policies were set
sessionStorageClient.smartAccountAddress, // Storage client, full Session or simply the smartAccount address if using default storage for your environment
chain
);
const { wait } = await emulatedUsersSmartAccount.sendTransaction(
[transferTx, nftMintTx],
{
...batchSessionParams,
...withSponsorship,
}
);
const { success } = await wait();
Send the transaction using the users smartAccount and get the transaction hash. The transaction will be built into a User Operation and then send to the Bundler.
That's it! You've successfully emulated a users smartAccount to execute multiple transactions using multiple sessions on their behalf.