Methods
The SDK provides the following API methods for a smart account.
createSmartAccountClient
This method is used to create an instance of the Biconomy Smart Account. This method requires a smart account configuration object to be passed and returns the smart account API instance.
This method creates the smart account based on the config object. To configure a smart account on another chain, you need to instantiate another smart account API instance with configuration of that chain.
Usage
let provider = new ethers.JsonRpcProvider("rpcUrl");
let signer = new ethers.Wallet("private key", provider);
const smartAccount = await createSmartAccountClient({
signer,
bundlerUrl: "", // <-- Read about this at https://docs.biconomy.io/dashboard#bundler-url
biconomyPaymasterApiKey: "", // <-- Read about at https://docs.biconomy.io/dashboard/paymaster
rpcUrl: "" // Recommended for signers that are not a viem wallet or an ethers signer. It's advised to pass RPC url in case of custom signers such as privy, dynamic etc. If rpcUrl is not provided then a default public rpc will be used - which will likely be heavily throttled and can often silently fail
});
Parameters
required params are explicitly mentioned
- config (
object
, required): A BiconomySmartAccountConfig object containing configuration options for creating the smart account.- signer(
Signer
, required) OR defaultValidationModule (BaseValidationModule
, required): One either needs to pass the signer instance or the default validation module which gets used to detect address of the smart account. If not passed explictly, ECDSA module gets used as default. - bundlerUrl (
string
, required) OR bundler (IBundler
, required) : bundler url which will be internally used to create bundler instance or the bundler instance. Bundler instance can also be used if one wants to customise the bundler. Refer to bundler integration for more details on bundler. - chainId (
ChainId
enum): The identifier for the blockchain network. (e.g., ChainId.POLYGON_MUMBAI). - biconomyPaymasterApiKey(
string
) OR paymaster (IPaymaster
): one can either pass paymaster API key or custom paymaster instance to use the paymaster. - entryPointAddress (
string
): DEFAULT_ENTRY_POINT_ADDRESS will be used if not passed, otherwise the passed address will be used. On specific chains like Chiliz Mainnet it is a different address, so will need to be passed explicitly. Refer to below notes on this. - activeValidationModule (
BaseValidationModule
): The run-time validation module (must be one of enabled validation modules) to sign and validate next userOp. - rpcUrl (
string
): Recommended for signers that are not a viem wallet or an ethers signer. It's advised to pass RPC url in case of custom signers such as privy, dynamic etc. If rpcUrl is not provided then a default public rpc will be used - which will likely be heavily throttled and can often silently fail. - index (
number
): index to create multiple smart accounts for an EOA. First account gets created with 0 index.
- signer(
Returns
smartAccount
(BiconomySmartAccount
): An instance of the Biconomy Smart Account.
Building on Chiliz Mainnet or the Spicy Testnet? Note that the entry point address on this is different as it was deployed by us on the Biconomy team. The address of the entry point is : 0x00000061FEfce24A79343c27127435286BB7A4E1
Smart Account Get Methods
getAccountAddress( )
This method retrieves the counterfactual address of the smartAccount
instance.
Usage
const address = await smartAccount.getAccountAddress();
Returns
- address (
Promise<string>
): A Promise resolving to the account address associated with the smartAccount instance.
getNonce( )
This method is used to retrieve the nonce associated with the smartAccount
instance.
Usage
const nonce = await smartAccount.getNonce();
console.log(nonce.toNumber());
const nonceKey = 10;
const nonce = await biconomySmartAccount.getNonce(nonceKey);
Parameters
- nonceKey(
number
): one can also pass the optional parameter nonceKey in the case of two-dimensional nonces.
Returns
- address (
Promise<BigNumber>
): A Promise resolving to current nonce of the smart account.
index
This is used to retrieve the index of the current active smart account.
Usage
const index = smartAccount.index;
Returns
- index (
number
): A number indicating the index of current active smart account.
deploy()
getBalances()
getSupportedTokens()
Transaction Methods
sendTransaction( )
This method is used to Send a transaction to a bundler for execution. It internally executes a build and send UserOp.
Usage
import { createClient } from "viem";
import { createSmartAccountClient, PaymasterMode } from "@biconomy/account";
import { createWalletClient, http } from "viem";
import { polygonMumbai } from "viem/chains";
const signer = createWalletClient({
account,
chain: polygonMumbai,
transport: http(),
});
const smartWallet = await createSmartAccountClient({ signer, bundlerUrl }); // Retrieve bundler url from dasboard
const encodedCall = encodeFunctionData({
abi: parseAbi(["function safeMint(address to) public"]),
functionName: "safeMint",
args: ["0x..."],
});
const transaction = {
to: nftAddress,
data: encodedCall,
};
const { waitForTxHash } = await smartWallet.sendTransaction(transaction);
const { transactionHash, userOperationReceipt } = await wait();
Parameters
-
manyOrOneTransactions (
Transaction | Transaction[]
, required): An array of transactions to be batched which will be executed in the provided order. You can also pass a single transaction.Transaction: {
to: string;
} & ValueOrData -
buildUseropDto (
BuildUserOpOptions
, optional): These options can be passed to customize how a userOp is built.type BuildUserOpOptions = {
forceEncodeForBatch?: boolean;
nonceOptions?: NonceOptions;
params?: ModuleInfo;
paymasterServiceData?: PaymasterUserOperationDto;
simulationType?: SimulationType;
stateOverrideSet?: StateOverrideSet;
useEmptyDeployCallData?: boolean;
};Let's look at each of these params:
-
forceEncodeForBatch (
boolean
): When a transactions array is passed, by default the Biconomy SDK encodes it for executeBatch() executor function and execute() function for single transaction. However, in some cases, there may be a preference to encode a single transaction for a batch, especially if the custom module only decodes for executeBatch. In such cases, set this flag to true; otherwise, it remains false by default. -
nonceOptions(
NonceOptions
) : This can be used to execute multiple user operations in parallel for the same smart account.type NonceOptions = {
nonceKey?: number;
nonceOverride?: number;
};
// nonceOptions usage
let i = 0;
const userOp = await smartAccount.buildUserOp([tx1], {
nonceOptions: { nonceKey: i++ },
});nonceKey can be initialised at any arbitrary number and incremented as one builds user operations to be sent in parallel. The nonceKey will create a batch or space in which the nonce can safely increment without colliding with other transactions. The nonceOverride will directly override the nonce and should only be used if you know the order in which you are sending the userOps.
-
params (
ModuleInfo
): This param can be used to pass session validation module parameters. Refer to the tutorial to learn more about the session keys.type ModuleInfo = {
sessionID?: string;
sessionSigner?: Signer;
sessionValidationModule?: string;
additionalSessionData?: string;
batchSessionParams?: SessionParams[];
}; -
paymasterServiceData (
PaymasterUserOperationDto
): ThepaymasterServiceData
includes details about the kind of sponsorship and payment token in case mode is ERC20. It contains information about the paymaster service, which is used to calculate thepaymasterAndData
field in the user operation. Note that this is only applicable if you're using Biconomy paymaster.type PaymasterUserOperationDto = {
mode: PaymasterMode;
calculateGasLimits?: boolean; //this flag defaults to true, signifying the paymaster will undertake gas limit calculations to ensure efficient operation execution
expiryDuration?: number;
webhookData?: Record<string, any>;
smartAccountInfo?: SmartAccountData;
feeTokenAddress?: string;
}; -
simulationType(
SimulationType
, enum): The simulationType flag can be of 2 ways:validation
which will only simulate the validation phase, checks if user op is valid but does not check if execution will succeed. By default this flag is set to validation.validation_and_execution
checks if user op is valid and if user op execution will succeed.
-
stateOverrideSet(
StateOverrideSet
): for overriding the blockchain state during simulations or gas estimation.
-
Returns
-
userOpResponse (
Promise<UserOpResponse>
): The method returns an object of typeUserOpResponse
which has auserOpHash
and two methods:wait()
andwaitForTxHash()
.type UserOpResponse = {
userOpHash: string;
wait(_confirmations?: number): Promise<UserOpReceipt>;
waitForTxHash(): Promise<UserOpStatus>;
};
type UserOpReceipt = {
/* The request hash of the UserOperation. */
userOpHash: string;
/* The entry point address used for the UserOperation. */
entryPoint: string;
/* The paymaster used for this UserOperation (or empty). */
paymaster: string;
/* The actual amount paid (by account or paymaster) for this UserOperation. */
actualGasCost: Hex;
/* The total gas used by this UserOperation (including preVerification, creation, validation, and execution). */
actualGasUsed: Hex;
/* Indicates whether the execution completed without reverting. */
success: "true" | "false";
/* In case of revert, this is the revert reason. */
reason: string;
/* The logs generated by this UserOperation (not including logs of other UserOperations in the same bundle). */
logs: Array<any>; // The logs generated by this UserOperation (not including logs of other UserOperations in the same bundle)
/* The TransactionReceipt object for the entire bundle, not only for this UserOperation. */
receipt: any;
};The
wait()
method resolves when the user operation is dispatched by the bundler on-chain and gets mined. ThewaitForTxHash()
method returns aUserOpStatus
object which includes the transaction hash and the receipt once added on-chain.const { transactionHash } = await userOpResponse.waitForTxHash();
console.log("transaction Hash", transactionHash);
const userOpReceipt = await userOpResponse.wait();
if(userOpReceipt.success == 'true') { // indicates that the user operation was successful without any revert
console.log("UserOp receipt", userOpReceipt)
console.log("actual transaction receipt", userOpReceipt.receipt)
}
buildUserOp( )
This method is used for configuring and setting up properties of the partial userOp
object. It converts an individual transaction or batch of transactions into a partial user operation populating fields such as initCode, sender, nonce, maxFeePerGas, maxPriorityFeePerGas, callGasLimit, verificationGasLimit and preVerificationGas (as this step also involves estimating gas for the userOp internally)
Usage
For example, in the context of creating a userOp for an addComment
transaction, an instance of the contract is created. A basic transaction object is then created that holds the necessary address and data from the transaction. Finally, a partial userOp is created using the Smart Account's buildUserOp
method. Now this can be signed and sent to the bundler.
const contractAddress = "contract address";
const provider = new ethers.JsonRpcProvider("rpc url");
const blogContract = new ethers.Contract(
contractAddress,
abi, // contract abi
provider
);
const createComment = await blogContract.populateTransaction.addComment(
"comment"
);
const tx1 = {
to: contractAddress,
data: createComment.data,
};
const userOp = await smartAccount.buildUserOp([tx1]);
Parameters
- transactions (
Transaction[]
, required): The required argument is an array of transactions which will be executed in provided order. You can pass multiple transactions into a userOp if you would like to batch them together into one transaction. - buildUseropDto (
BuildUserOpOptions
): One can also pass these options to customize how a userOp is built.
Returns
- partialUserOp (
Promise<Partial<UserOperationStruct>>
): A Promise resolving toUserOperationStruct
which can be further signed and sent to the bundler.
senduserOp( )
This method is used to submit a User Operation object to the User Operation pool of the client. It signs the UserOperation using activeValidationModule instance and submits it to the bundler for on-chain processing.
Usage
const userOp = await smartAccount.buildUserOp([transaction]);
const userOpResponse = await smartAccount.sendUserOp(userOp);
const { receipt } = await userOpResponse.wait(1);
Parameters
-
userOp (
Partial<UserOperationStruct>
, required): TheuserOp
object includes essential fields likesender
,nonce
,callData
,callGasLimit
andgas
related properties. -
params (
SendUserOpParams
, optional): This gets used when the active validation module is complex and requires additional information for signature generation. TheSendUserOpParams
object can contain fields such assessionID
,sessionSigner
,sessionValidationModule
,additionalSessionData
,batchSessionParams
, andsimulationType
. These parameters are used to customize the behavior of thesendUserOp
method and are optional.const userOpResponse = await moduleSmartAccount?.sendUserOp(userOp, {
sessionSigner: sessionSigner,
sessionValidationModule: moduleAddr,
});Similar to building a
userOp
we need to ensure that any modules used for additional validation or execution logic are specified in thesendUserOp
method. Currently, this only applies for session key module requirements.
Please note that simulationType
allows for more debugging insights about callData
on why an internal transaction fails. It is set to "validation" by default, but can be changed to "validation_and_execution" for more detailed tracing.
Returns
-
userOpsResponse (
UserOpResponse
): The method returns an object of typeUserOpResponse
which has auserOpHash
and two methods:wait()
andwaitForTxHash()
.type UserOpResponse = {
userOpHash: string;
wait(_confirmations?: number): Promise<UserOpReceipt>;
waitForTxHash(): Promise<UserOpStatus>;
};
type UserOpReceipt = {
userOpHash: string;
entryPoint: string;
sender: string;
nonce: number;
paymaster: string;
actualGasCost: BigNumber;
actualGasUsed: BigNumber;
success: boolean;
reason: string;
logs: Array<ethers.providers.Log>;
receipt: ethers.providers.TransactionReceipt;
};The
wait()
method resolves when the user operation is dispatched by the bundler on-chain and gets mined. ThewaitForTxHash()
method returns aUserOpStatus
object which includes the transaction hash and the receipt once added on-chain.
sendSignedUserOp( )
This method is designed to dispatch signed user operations to the bundler. This method is particularly useful when handling operations that have been grouped together with a multi-chain module, as it allows for the submission of these combined operations in a single request. It can also be useful in the case of two different instances of smart account, for example one backend instance to build userOp, while another instance to obtain the signed userOp on the frontend and subsequently dispatch the signed userOp using the backend instance.
Usage
const userOpResponse = await smartAccount.sendSignedUserOp(userOp);
Please ensure that the user operations have been correctly signed before using this method
Parameters
-
userOp (
UserOperation
, required): TheuserOp
object includes essential fields likesender
,nonce
,callData
,gas
related properties, andsignature
. -
params (
SendUserOpParams
): TheSendUserOpParams
object can contain fields such assessionID
,sessionSigner
,sessionValidationModule
,additionalSessionData
,batchSessionParams
, andsimulationType
. These parameters are used to customize the behavior of thesendUserOp
method and are optional.
Returns
- userOpsResponse (
UserOpResponse
): The method returns an object of typeUserOpResponse
which has auserOpHash
and two methods:wait()
andwaitForTxHash()
. Thewait()
method resolves when the user operation is dispatched by the bundler on-chain and gets mined. ThewaitForTxHash()
method returns aUserOpStatus
object which includes the transaction hash and the receipt once added on-chain.