Alchemy's Account Abstraction Kit is an SDK that enables easy interactions with ERC-4337 compliant smart accounts. It supports bringing your own Account Contracts or using any of the currently available accounts. The SDK is built on top of viem
to enable a lighter-weight bundle and is published with ESM default exports though also supports CJS.
via yarn
:
yarn add @alchemy/aa-core viem
via npm
:
npm i -s @alchemy/aa-core viem
If you are using Alchemy APIs for Account Abstraction, then you can also add the @alchemy/aa-alchemy
package:
via yarn
:
yarn add @alchemy/aa-alchemy
via npm
:
npm i -s @alchemy/aa-alchemy
If you are using ethers
and want to use an ethers
compatible Provider
and Signer
you can also add the the @alchemy/aa-ethers
library (the above packages are required still).
via yarn
:
yarn add @alchemy/aa-ethers
via npm
:
npm i -s @alchemy/aa-ethers
Example Usage to Interact with Simple Accounts
import {
SimpleSmartContractAccount,
SmartAccountProvider,
type SimpleSmartAccountOwner,
} from "@alchemy/aa-core";
import { mnemonicToAccount } from "viem/accounts";
import { polygonMumbai } from "viem/chains";
import { toHex } from "viem";
const SIMPLE_ACCOUNT_FACTORY_ADDRESS =
"0x9406Cc6185a346906296840746125a0E44976454";
// 1. define the EOA owner of the Smart Account
// This is just one exapmle of how to interact with EOAs, feel free to use any other interface
const ownerAccount = mnemonicToAccount(MNEMONIC);
// All that is important for defining an owner is that it provides a `signMessage` and `getAddress` function
const owner: SimpleSmartAccountOwner = {
// this should sign a message according to ERC-191
signMessage: async (msg) =>
ownerAccount.signMessage({
message: toHex(msg),
}),
getAddress: async () => ownerAccount.address,
};
// 2. initialize the provider and connect it to the account
const provider = new SmartAccountProvider(
// the demo key below is public and rate-limited, it's better to create a new one
// you can get started with a free account @ https://www.alchemy.com/
"https://polygon-mumbai.g.alchemy.com/v2/demo", // rpcUrl
"0x5FF137D4b0FDCD49DcA30c7CF57E578a026d2789", // entryPointAddress
polygonMumbai // chain
).connect(
(rpcClient) =>
new SimpleSmartContractAccount({
entryPointAddress: "0x5FF137D4b0FDCD49DcA30c7CF57E578a026d2789",
chain: polygonMumbai,
factoryAddress: SIMPLE_ACCOUNT_FACTORY_ADDRESS,
rpcClient,
// optionally if you already know the account's address
accountAddress: "0x000...000",
})
);
// 3. send a UserOperation
const { hash } = provider.sendUserOperation({
target: "0xTargetAddress",
data: "0xcallData",
value: 0n, // value: bigint or undefined
});
import {
SimpleSmartContractAccount,
type SimpleSmartAccountOwner,
} from "@alchemy/aa-core";
import { toHex } from "viem";
import { mnemonicToAccount } from "viem/accounts";
import { polygonMumbai } from "viem/chains";
import { AlchemyProvider } from "@alchemy/aa-alchemy";
const SIMPLE_ACCOUNT_FACTORY_ADDRESS =
"0x9406Cc6185a346906296840746125a0E44976454";
// 1. define the EOA owner of the Smart Account
// This is just one exapmle of how to interact with EOAs, feel free to use any other interface
const ownerAccount = mnemonicToAccount(MNEMONIC);
// All that is important for defining an owner is that it provides a `signMessage` and `getAddress` function
const owner: SimpleSmartAccountOwner = {
// this should sign a message according to ERC-191
signMessage: async (msg) =>
ownerAccount.signMessage({
message: toHex(msg),
}),
getAddress: async () => ownerAccount.address,
};
// 2. initialize the provider and connect it to the account
let provider = new AlchemyProvider({
apiKey: API_KEY,
chain,
entryPointAddress: ENTRYPOINT_ADDRESS,
}).connect(
(rpcClient) =>
new SimpleSmartContractAccount({
entryPointAddress: ENTRYPOINT_ADDRESS,
chain: polygonMumbai, // ether a viem Chain or chainId that supports account abstraction at Alchemy
owner,
factoryAddress: SIMPLE_ACCOUNT_FACTORY_ADDRESS,
rpcClient,
})
);
// [OPTIONAL] Use Alchemy Gas Manager
prpvider = provider.withAlchemyGasManager({
provider: provider.rpcClient,
policyId: PAYMASTER_POLICY_ID,
entryPoint: ENTRYPOINT_ADDRESS,
});
// 3. send a UserOperation
const { hash } = provider.sendUserOperation({
target: "0xTargetAddress",
data: "0xcallData",
value: 0n, // value: bigint or undefined
});
import {
alchemyPaymasterAndDataMiddleware,
getChain,
SimpleSmartContractAccount,
} from "@alchemy/aa-core";
import { Alchemy, Network } from "alchemy-sdk";
import { Wallet } from "@ethersproject/wallet";
import {
EthersProviderAdapter,
convertWalletToAccountSigner,
} from "@alchemy/aa-ethers";
const SIMPLE_ACCOUNT_FACTORY_ADDRESS =
"0x9406Cc6185a346906296840746125a0E44976454";
// 1. connect to an RPC Provider and a Wallet
const alchemy = new Alchemy({
apiKey: API_KEY,
network: Network.MATIC_MUMBAI,
});
const alchemyProvider = await alchemy.config.getProvider();
const owner = Wallet.fromMnemonic(MNEMONIC);
// 2. Create the SimpleAccount signer
// signer is an ethers.js Signer
const signer = EthersProviderAdapter.fromEthersProvider(
alchemyProvider,
ENTRYPOINT_ADDRESS
).connectToAccount(
(rpcClient) =>
new SimpleSmartContractAccount({
entryPointAddress: ENTRYPOINT_ADDRESS,
chain: getChain(alchemyProvider.network.chainId),
owner: convertWalletToAccountSigner(owner),
factoryAddress: SIMPLE_ACCOUNT_FACTORY_ADDRESS,
rpcClient,
})
);
// 3. send a user op
const { hash } = signer.sendUserOperation({
target: "0xTargetAddress",
data: "0xcallData",
value: 0n, // value: bigint or undefined
});
The primary interfaces are the SmartAccountProvider
and BaseSmartContractAccount
.
The SmartAccountProvider
is an ERC-1193 compliant Provider that wraps JSON RPC methods and some Wallet Methods (signing, sendTransaction, etc). It also provides two utility methods for sending UserOperations:
sendUserOperation
-- this takes intarget
,callData
, and an optionalvalue
which then constructs a UserOperation (UO), sends it, and returns thehash
of the UO. It handles estimating gas, fetching fee data, (optionally) requesting paymasterAndData, and lastly signing. This is done via a middleware stack that runs in a specific order. The middleware order isgetDummyPaymasterData
=>estimateGas
=>getFeeData
=>getPaymasterAndData
. The paymaster fields are set to0x
by default. They can be changed usingprovider.withPaymasterMiddleware
.sendTransaction
-- this takes in a traditional Transaction Request object which then gets converted into a UO. Currently, the only data being used from the Transaction Request object isfrom
,to
,data
andvalue
. Support for other fields is coming soon.
If you want to add support for your own SmartAccounts
then you will need to provide an implementation of BaseSmartContractAccount
. You can see an example of this in SimpleSmartContractAccount. You will need to implement 4 methods:
getDummySignature
-- this method should return a signature that will notrevert
during validation. It does not have to pass validation, just not cause the contract to revert. This is required for gas estimation so that the gas estimate are accurate.encodeExecute
-- this method should return the abi encoded function data for a call to your contract'sexecute
methodsignMessage
-- this should return an ERC-191 compliant message and is used to sign UO HashesgetAccountInitCode
-- this should return the init code that will be used to create an account if one does not exist. Usually this is the concatenation of the account's factory address and the abi encoded function data of the account factory'screateAccount
method.
You can use provider.withPaymasterMiddleware
to add middleware to the stack which will set the paymasterAndData
field during sendUserOperation
calls. The withPaymasterMiddleware
method has two overrides. One of the overrides takes a dummyPaymasterData
generator function. This dummyPaymasterData
is needed to estimate gas correctly when using a paymaster and is specific to the paymaster you're using. The second override is the actually paymasterAndData
generator function. This function is called after gas estimation and fee estimation and is used to set the paymasterAndData
field. The default dummyPaymasterData
generator function returns 0x
for both the paymasterAndData
fields. The default paymasterAndData
generator function returns 0x
for both the paymasterAndData
fields.
Both of the override methods can return new gas estimates. This allows for paymaster RPC urls that handle gas estimation for you. It's important to note that if you're using an ERC-20 paymaster and your RPC endpoint does not return estimates, you should add an additional 75k gas to the gas estimate for verificationGasLimit
.
Alchemy has two separate RPC methods for interacting with our Gas Manager services. The first is alchemy_requestPaymasterAndData
and the second is alchemy_requestGasAndPaymasterAndData
.
The former is useful if you want to do your own gas estimation + fee estimation (or you're happy using the default middlewares for gas and fee estimation), but want to use the Alchemy Gas Manager service.
The latter is will handle gas + fee estimation and return paymasterAndData
in a single request.
We provide two utility methods in @alchemy/aa-alchemy
for interacting with these RPC methods:
alchemyPaymasterAndDataMiddleware
which is used in conjunction withwithPaymasterMiddleware
to add thealchemy_requestPaymasterAndData
RPC method to the middleware stack.withAlchemyGasManager
which wraps a connectedSmartAccountProvider
with the middleware overrides to usealchemy_requestGasAndPaymasterAndData
RPC method.
- clone the repo
- run
yarn
- Make changes to packages
To run tests:
TODO: currently tests require a specific mnemonic to pass and they run against the alchemy bundler which is in private beta.