From 74e4fe2d746557198275f8297136b07f55bf802a Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Mon, 12 Aug 2024 12:54:45 +0100 Subject: [PATCH 01/51] simplify account requirements --- .../privateKeyToBiconomySmartAccount.ts | 21 +--- .../biconomy/signerToBiconomySmartAccount.ts | 13 +-- .../kernel/signerToEcdsaKernelSmartAccount.ts | 18 ++-- .../light/privateKeyToLightSmartAccount.ts | 20 +--- .../light/signerToLightSmartAccount.ts | 17 ++- .../safe/privateKeyToSafeSmartAccount.ts | 20 +--- .../accounts/safe/signerToSafeSmartAccount.ts | 17 ++- .../simple/privateKeyToSimpleSmartAccount.ts | 22 +--- .../simple/signerToSimpleSmartAccount.ts | 17 ++- .../permissionless/accounts/toSmartAccount.ts | 23 ++-- .../trust/privateKeyToTrustSmartAccount.ts | 20 +--- .../trust/signerToTrustSmartAccount.ts | 12 +-- .../accounts/trust/utils/getAccountAddress.ts | 6 +- .../accounts/trust/utils/signMessage.ts | 7 +- .../accounts/trust/utils/signUserOperation.ts | 5 +- packages/permissionless/accounts/types.ts | 18 ++-- packages/permissionless/actions/erc7579.ts | 72 +++---------- .../actions/erc7579/accountId.ts | 27 +++-- .../actions/erc7579/installModule.ts | 32 ++---- .../actions/erc7579/installModules.ts | 32 ++---- .../actions/erc7579/isModuleInstalled.ts | 44 ++++---- .../actions/erc7579/supportsExecutionMode.ts | 47 ++++---- .../actions/erc7579/supportsModule.ts | 41 ++++--- .../actions/erc7579/uninstallModule.ts | 39 ++----- .../actions/erc7579/uninstallModules.ts | 32 ++---- .../actions/smartAccount/deployContract.ts | 36 +++---- .../prepareUserOperationRequest.ts | 102 +++++++----------- .../actions/smartAccount/sendTransaction.ts | 21 +--- .../actions/smartAccount/sendTransactions.ts | 30 ++---- .../actions/smartAccount/sendUserOperation.ts | 29 ++--- .../actions/smartAccount/signMessage.ts | 8 +- .../actions/smartAccount/signTypedData.ts | 8 +- .../actions/smartAccount/writeContract.ts | 13 +-- .../clients/createBundlerClient.ts | 5 +- .../clients/decorators/smartAccount.ts | 87 +++++++-------- .../eip7677/actions/getPaymasterData.ts | 4 +- .../eip7677/actions/getPaymasterStubData.ts | 4 +- .../decorators/paymasterActionsEip7677.ts | 9 +- packages/permissionless/types/index.ts | 14 +-- 39 files changed, 344 insertions(+), 648 deletions(-) diff --git a/packages/permissionless/accounts/biconomy/privateKeyToBiconomySmartAccount.ts b/packages/permissionless/accounts/biconomy/privateKeyToBiconomySmartAccount.ts index 39f36efb..a7b4665c 100644 --- a/packages/permissionless/accounts/biconomy/privateKeyToBiconomySmartAccount.ts +++ b/packages/permissionless/accounts/biconomy/privateKeyToBiconomySmartAccount.ts @@ -1,11 +1,4 @@ -import type { - Chain, - Client, - Hex, - PublicActions, - PublicRpcSchema, - Transport -} from "viem" +import type { Account, Chain, Client, Hex, Transport } from "viem" import { privateKeyToAccount } from "viem/accounts" import type { ENTRYPOINT_ADDRESS_V06_TYPE, Prettify } from "../../types" import { @@ -30,15 +23,10 @@ export type PrivateKeyToBiconomySmartAccountParameters< export async function privateKeyToBiconomySmartAccount< entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE, TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined + TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined >( - client: Client< - TTransport, - TChain, - undefined, - PublicRpcSchema, - PublicActions - >, + client: Client, { privateKey, ...rest @@ -49,6 +37,7 @@ export async function privateKeyToBiconomySmartAccount< entryPoint, TTransport, TChain, + TClientAccount, "privateKey" >(client, { signer: privateKeyAccount, diff --git a/packages/permissionless/accounts/biconomy/signerToBiconomySmartAccount.ts b/packages/permissionless/accounts/biconomy/signerToBiconomySmartAccount.ts index edc52b4e..fe53e130 100644 --- a/packages/permissionless/accounts/biconomy/signerToBiconomySmartAccount.ts +++ b/packages/permissionless/accounts/biconomy/signerToBiconomySmartAccount.ts @@ -1,4 +1,4 @@ -import type { PublicActions, PublicRpcSchema, TypedData } from "viem" +import type { Account, TypedData } from "viem" import { type Address, type Chain, @@ -212,16 +212,11 @@ export async function signerToBiconomySmartAccount< entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined, TSource extends string = string, TAddress extends Address = Address >( - client: Client< - TTransport, - TChain, - undefined, - PublicRpcSchema, - PublicActions - >, + client: Client, { signer, address, @@ -311,7 +306,7 @@ export async function signerToBiconomySmartAccount< TTypedData, TPrimaryType, TChain, - undefined + TClientAccount >(client, { account: viemSigner, ...typedData diff --git a/packages/permissionless/accounts/kernel/signerToEcdsaKernelSmartAccount.ts b/packages/permissionless/accounts/kernel/signerToEcdsaKernelSmartAccount.ts index 3eb63024..ac0e5d1b 100644 --- a/packages/permissionless/accounts/kernel/signerToEcdsaKernelSmartAccount.ts +++ b/packages/permissionless/accounts/kernel/signerToEcdsaKernelSmartAccount.ts @@ -1,4 +1,4 @@ -import type { PublicActions, PublicRpcSchema, TypedData } from "viem" +import type { Account, TypedData } from "viem" import { type Address, type Chain, @@ -316,14 +316,15 @@ const getAccountInitCode = async ({ const getAccountAddress = async < entryPoint extends EntryPoint, TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined + TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined >({ client, entryPoint: entryPointAddress, factory, factoryData }: { - client: Client + client: Client entryPoint: entryPoint factory: Address factoryData: Hex @@ -373,16 +374,11 @@ export async function signerToEcdsaKernelSmartAccount< entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined, TSource extends string = string, TAddress extends Address = Address >( - client: Client< - TTransport, - TChain, - undefined, - PublicRpcSchema, - PublicActions - >, + client: Client, { signer, address, @@ -487,7 +483,7 @@ export async function signerToEcdsaKernelSmartAccount< TTypedData, TPrimaryType, TChain, - undefined + TClientAccount >(client, { account: viemSigner, ...typedData, diff --git a/packages/permissionless/accounts/light/privateKeyToLightSmartAccount.ts b/packages/permissionless/accounts/light/privateKeyToLightSmartAccount.ts index 7d7d7651..b5a5556d 100644 --- a/packages/permissionless/accounts/light/privateKeyToLightSmartAccount.ts +++ b/packages/permissionless/accounts/light/privateKeyToLightSmartAccount.ts @@ -1,11 +1,4 @@ -import type { - Chain, - Client, - Hex, - PublicActions, - PublicRpcSchema, - Transport -} from "viem" +import type { Account, Chain, Client, Hex, Transport } from "viem" import { privateKeyToAccount } from "viem/accounts" import type { Prettify } from "../../types" import type { EntryPoint } from "../../types" @@ -31,15 +24,10 @@ export type PrivateKeyToLightSmartAccountParameters< export async function privateKeyToLightSmartAccount< entryPoint extends EntryPoint, TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined + TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined >( - client: Client< - TTransport, - TChain, - undefined, - PublicRpcSchema, - PublicActions - >, + client: Client, { privateKey, ...rest }: PrivateKeyToLightSmartAccountParameters ): Promise> { const privateKeyAccount = privateKeyToAccount(privateKey) diff --git a/packages/permissionless/accounts/light/signerToLightSmartAccount.ts b/packages/permissionless/accounts/light/signerToLightSmartAccount.ts index 8f043ad5..7f369f77 100644 --- a/packages/permissionless/accounts/light/signerToLightSmartAccount.ts +++ b/packages/permissionless/accounts/light/signerToLightSmartAccount.ts @@ -1,11 +1,10 @@ import { + type Account, type Address, type Chain, type Client, type Hex, type LocalAccount, - type PublicActions, - type PublicRpcSchema, type Transport, type TypedData, type TypedDataDefinition, @@ -80,7 +79,8 @@ const getAccountInitCode = async ( const getAccountAddress = async < entryPoint extends EntryPoint, TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined + TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined >({ client, factoryAddress, @@ -88,7 +88,7 @@ const getAccountAddress = async < owner, index = BigInt(0) }: { - client: Client + client: Client factoryAddress: Address owner: Address entryPoint: entryPoint @@ -191,16 +191,11 @@ export async function signerToLightSmartAccount< entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined, TSource extends string = string, TAddress extends Address = Address >( - client: Client< - TTransport, - TChain, - undefined, - PublicRpcSchema, - PublicActions - >, + client: Client, { signer, address, diff --git a/packages/permissionless/accounts/safe/privateKeyToSafeSmartAccount.ts b/packages/permissionless/accounts/safe/privateKeyToSafeSmartAccount.ts index 8404f27e..e96946d6 100644 --- a/packages/permissionless/accounts/safe/privateKeyToSafeSmartAccount.ts +++ b/packages/permissionless/accounts/safe/privateKeyToSafeSmartAccount.ts @@ -1,11 +1,4 @@ -import type { - Chain, - Client, - Hex, - PublicActions, - PublicRpcSchema, - Transport -} from "viem" +import type { Account, Chain, Client, Hex, Transport } from "viem" import { privateKeyToAccount } from "viem/accounts" import type { EntryPoint, Prettify } from "../../types" import { @@ -30,15 +23,10 @@ export type PrivateKeyToSafeSmartAccountParameters< export async function privateKeyToSafeSmartAccount< entryPoint extends EntryPoint, TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined + TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined >( - client: Client< - TTransport, - TChain, - undefined, - PublicRpcSchema, - PublicActions - >, + client: Client, { privateKey, ...rest }: PrivateKeyToSafeSmartAccountParameters ): Promise> { const privateKeyAccount = privateKeyToAccount(privateKey) diff --git a/packages/permissionless/accounts/safe/signerToSafeSmartAccount.ts b/packages/permissionless/accounts/safe/signerToSafeSmartAccount.ts index 00311de5..a404e350 100644 --- a/packages/permissionless/accounts/safe/signerToSafeSmartAccount.ts +++ b/packages/permissionless/accounts/safe/signerToSafeSmartAccount.ts @@ -1,11 +1,10 @@ import { + type Account, type Address, type Chain, type Client, type Hex, type LocalAccount, - type PublicActions, - type PublicRpcSchema, type SignableMessage, type Transport, type TypedData, @@ -864,7 +863,8 @@ const getAccountInitCode = async ({ const getAccountAddress = async < TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined + TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined >({ client, owner, @@ -884,7 +884,7 @@ const getAccountAddress = async < attesters = [], attestersThreshold = 0 }: { - client: Client + client: Client owner: Address safeModuleSetupAddress: Address safe4337ModuleAddress: Address @@ -1087,17 +1087,12 @@ export async function signerToSafeSmartAccount< entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined, TSource extends string = string, TAddress extends Address = Address, TErc7579 extends Address | undefined = undefined >( - client: Client< - TTransport, - TChain, - undefined, - PublicRpcSchema, - PublicActions - >, + client: Client, args: SignerToSafeSmartAccountParameters< entryPoint, TSource, diff --git a/packages/permissionless/accounts/simple/privateKeyToSimpleSmartAccount.ts b/packages/permissionless/accounts/simple/privateKeyToSimpleSmartAccount.ts index 927d47bc..135f827d 100644 --- a/packages/permissionless/accounts/simple/privateKeyToSimpleSmartAccount.ts +++ b/packages/permissionless/accounts/simple/privateKeyToSimpleSmartAccount.ts @@ -1,12 +1,4 @@ -import type { - Address, - Chain, - Client, - Hex, - PublicActions, - PublicRpcSchema, - Transport -} from "viem" +import type { Account, Address, Chain, Client, Hex, Transport } from "viem" import { privateKeyToAccount } from "viem/accounts" import type { EntryPoint, Prettify } from "../../types" import { @@ -31,15 +23,10 @@ export type PrivateKeyToSimpleSmartAccountParameters< export async function privateKeyToSimpleSmartAccount< entryPoint extends EntryPoint, TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined + TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined >( - client: Client< - TTransport, - TChain, - undefined, - PublicRpcSchema, - PublicActions - >, + client: Client, { privateKey, ...rest @@ -51,6 +38,7 @@ export async function privateKeyToSimpleSmartAccount< entryPoint, TTransport, TChain, + TClientAccount, "privateKey", Address >(client, { diff --git a/packages/permissionless/accounts/simple/signerToSimpleSmartAccount.ts b/packages/permissionless/accounts/simple/signerToSimpleSmartAccount.ts index 7c987bf9..16dd0ea2 100644 --- a/packages/permissionless/accounts/simple/signerToSimpleSmartAccount.ts +++ b/packages/permissionless/accounts/simple/signerToSimpleSmartAccount.ts @@ -1,11 +1,10 @@ import { + type Account, type Address, type Chain, type Client, type Hex, type LocalAccount, - type PublicActions, - type PublicRpcSchema, type Transport, concatHex, encodeFunctionData @@ -76,7 +75,8 @@ const getAccountInitCode = async ( const getAccountAddress = async < entryPoint extends EntryPoint, TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined + TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined >({ client, factoryAddress, @@ -84,7 +84,7 @@ const getAccountAddress = async < owner, index = BigInt(0) }: { - client: Client + client: Client factoryAddress: Address owner: Address entryPoint: entryPoint @@ -142,16 +142,11 @@ export async function signerToSimpleSmartAccount< entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined, TSource extends string = string, TAddress extends Address = Address >( - client: Client< - TTransport, - TChain, - undefined, - PublicRpcSchema, - PublicActions - >, + client: Client, { signer, factoryAddress: _factoryAddress, diff --git a/packages/permissionless/accounts/toSmartAccount.ts b/packages/permissionless/accounts/toSmartAccount.ts index c4637dd5..9d199174 100644 --- a/packages/permissionless/accounts/toSmartAccount.ts +++ b/packages/permissionless/accounts/toSmartAccount.ts @@ -1,13 +1,12 @@ import { type Abi, + type Account, type Address, type Chain, type Client, type CustomSource, type EncodeDeployDataParameters, type Hex, - type PublicActions, - type PublicRpcSchema, type SignableMessage, type Transport, type TypedDataDefinition, @@ -32,6 +31,7 @@ export function toSmartAccount< TSource extends string = string, transport extends Transport = Transport, chain extends Chain | undefined = Chain | undefined, + clientAccount extends Account | undefined = Account | undefined, TAbi extends Abi | readonly unknown[] = Abi >({ address, @@ -50,13 +50,7 @@ export function toSmartAccount< signTypedData }: TAccountSource & { source: TSource - client: Client< - transport, - chain, - undefined, - PublicRpcSchema, - PublicActions - > + client: Client entryPoint: TEntryPoint getNonce: (key?: bigint) => Promise getInitCode: () => Promise @@ -86,7 +80,7 @@ export function toSmartAccount< signUserOperation: ( userOperation: UserOperation> ) => Promise -}): SmartAccount { +}): SmartAccount { const account = toAccount({ address: address, signMessage: async ({ message }: { message: SignableMessage }) => { @@ -171,5 +165,12 @@ export function toSmartAccount< getDummySignature, encodeDeployCallData, signUserOperation - } + } as SmartAccount< + TEntryPoint, + TSource, + transport, + chain, + clientAccount, + TAbi + > } diff --git a/packages/permissionless/accounts/trust/privateKeyToTrustSmartAccount.ts b/packages/permissionless/accounts/trust/privateKeyToTrustSmartAccount.ts index 1613b028..45f33611 100644 --- a/packages/permissionless/accounts/trust/privateKeyToTrustSmartAccount.ts +++ b/packages/permissionless/accounts/trust/privateKeyToTrustSmartAccount.ts @@ -1,11 +1,4 @@ -import type { - Chain, - Client, - Hex, - PublicActions, - PublicRpcSchema, - Transport -} from "viem" +import type { Account, Chain, Client, Hex, Transport } from "viem" import { privateKeyToAccount } from "viem/accounts" import type { ENTRYPOINT_ADDRESS_V06_TYPE, Prettify } from "../../types" import { @@ -30,15 +23,10 @@ export type PrivateKeyToTrustSmartAccountParameters< export async function privateKeyToTrustSmartAccount< entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE, TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined + TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined >( - client: Client< - TTransport, - TChain, - undefined, - PublicRpcSchema, - PublicActions - >, + client: Client, { privateKey, ...rest }: PrivateKeyToTrustSmartAccountParameters ): Promise> { const privateKeyAccount = privateKeyToAccount(privateKey) diff --git a/packages/permissionless/accounts/trust/signerToTrustSmartAccount.ts b/packages/permissionless/accounts/trust/signerToTrustSmartAccount.ts index 6cc97f84..5384ac25 100644 --- a/packages/permissionless/accounts/trust/signerToTrustSmartAccount.ts +++ b/packages/permissionless/accounts/trust/signerToTrustSmartAccount.ts @@ -1,11 +1,10 @@ import { + type Account, type Address, type Chain, type Client, type Hex, type LocalAccount, - type PublicActions, - type PublicRpcSchema, type Transport, type TypedData, type TypedDataDefinition, @@ -100,16 +99,11 @@ export async function signerToTrustSmartAccount< entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined, TSource extends string = string, TAddress extends Address = Address >( - client: Client< - TTransport, - TChain, - undefined, - PublicRpcSchema, - PublicActions - >, + client: Client, { signer, factoryAddress = TRUST_ADDRESSES.factoryAddress, diff --git a/packages/permissionless/accounts/trust/utils/getAccountAddress.ts b/packages/permissionless/accounts/trust/utils/getAccountAddress.ts index f52d45cf..ffb65dff 100644 --- a/packages/permissionless/accounts/trust/utils/getAccountAddress.ts +++ b/packages/permissionless/accounts/trust/utils/getAccountAddress.ts @@ -1,4 +1,5 @@ import { + type Account, type Address, type Chain, type Client, @@ -12,9 +13,10 @@ import { getFactoryData } from "./getFactoryData" export const getAccountAddress = async < entryPoint extends EntryPoint, TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined + TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined >( - client: Client, + client: Client, { factoryAddress, entryPoint: entryPointAddress, diff --git a/packages/permissionless/accounts/trust/utils/signMessage.ts b/packages/permissionless/accounts/trust/utils/signMessage.ts index 76558165..75099d24 100644 --- a/packages/permissionless/accounts/trust/utils/signMessage.ts +++ b/packages/permissionless/accounts/trust/utils/signMessage.ts @@ -1,12 +1,13 @@ -import type { Chain, Client, Transport } from "viem" +import type { Account, Chain, Client, Transport } from "viem" import type { SignMessageParameters } from "viem" import { signMessage as viem_signMessage } from "viem/actions" export const signMessage = < TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined + TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined >( - client: Client, + client: Client, { account, message }: SignMessageParameters ) => { return viem_signMessage(client, { account, message }) diff --git a/packages/permissionless/accounts/trust/utils/signUserOperation.ts b/packages/permissionless/accounts/trust/utils/signUserOperation.ts index 0ff0393a..b5e01d89 100644 --- a/packages/permissionless/accounts/trust/utils/signUserOperation.ts +++ b/packages/permissionless/accounts/trust/utils/signUserOperation.ts @@ -10,9 +10,10 @@ import { signMessage } from "./signMessage" export const signUserOperation = async < entryPoint extends EntryPoint, TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined + TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined >( - client: Client, + client: Client, { account, userOperation, diff --git a/packages/permissionless/accounts/types.ts b/packages/permissionless/accounts/types.ts index a11d787f..8f1541e9 100644 --- a/packages/permissionless/accounts/types.ts +++ b/packages/permissionless/accounts/types.ts @@ -7,10 +7,9 @@ import { type LocalAccount } from "viem" import type { + Account, Chain, EncodeDeployDataParameters, - PublicActions, - PublicRpcSchema, Transport } from "viem" import type { UserOperation } from "../types" @@ -34,18 +33,13 @@ export class SignTransactionNotSupportedBySmartAccount extends BaseError { export type SmartAccount< entryPoint extends EntryPoint, - TSource extends string, - transport extends Transport, - chain extends Chain | undefined, + TSource extends string = string, + transport extends Transport = Transport, + chain extends Chain | undefined = Chain | undefined, + clientAccount extends Account | undefined = Account | undefined, TAbi extends Abi | readonly unknown[] = Abi > = LocalAccount & { - client: Client< - transport, - chain, - undefined, - PublicRpcSchema, - PublicActions - > + client: Client entryPoint: entryPoint getNonce: (key?: bigint) => Promise getInitCode: () => Promise diff --git a/packages/permissionless/actions/erc7579.ts b/packages/permissionless/actions/erc7579.ts index f131ede2..d2f62948 100644 --- a/packages/permissionless/actions/erc7579.ts +++ b/packages/permissionless/actions/erc7579.ts @@ -36,77 +36,33 @@ import { export type Erc7579Actions< TEntryPoint extends EntryPoint, - TTransport extends Transport, - TChain extends Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined + TSmartAccount extends SmartAccount | undefined > = { accountId: ( args?: TSmartAccount extends undefined - ? GetAccountParameter< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - > + ? GetAccountParameter : undefined ) => Promise installModule: ( - args: InstallModuleParameters< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - > + args: InstallModuleParameters ) => Promise installModules: ( - args: InstallModulesParameters< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - > + args: InstallModulesParameters ) => Promise isModuleInstalled: ( - args: IsModuleInstalledParameters< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - > + args: IsModuleInstalledParameters ) => Promise supportsExecutionMode: ( - args: SupportsExecutionModeParameters< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - > + args: SupportsExecutionModeParameters ) => Promise supportsModule: ( - args: SupportsModuleParameters< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - > + args: SupportsModuleParameters ) => Promise uninstallModule: ( - args: UninstallModuleParameters< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - > + args: UninstallModuleParameters ) => Promise uninstallModules: ( - args: UninstallModulesParameters< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - > + args: UninstallModulesParameters ) => Promise } @@ -136,14 +92,14 @@ export function erc7579Actions(_args: { entryPoint: TEntryPoint }) { return < - TTransport extends Transport, - TChain extends Chain | undefined, - TSmartAccount extends - | SmartAccount + TTransport extends Transport = Transport, + TChain extends Chain | undefined = Chain | undefined, + TSmartAccount extends SmartAccount | undefined = + | SmartAccount | undefined >( client: Client - ): Erc7579Actions => ({ + ): Erc7579Actions => ({ accountId: (args) => accountId(client, args), installModule: (args) => installModule( diff --git a/packages/permissionless/actions/erc7579/accountId.ts b/packages/permissionless/actions/erc7579/accountId.ts index d7e370bd..ea681270 100644 --- a/packages/permissionless/actions/erc7579/accountId.ts +++ b/packages/permissionless/actions/erc7579/accountId.ts @@ -7,6 +7,8 @@ import { decodeFunctionResult, encodeFunctionData } from "viem" +import { call, readContract } from "viem/actions" +import { getAction } from "viem/utils" import type { SmartAccount } from "../../accounts/types" import type { GetAccountParameter } from "../../types" import type { EntryPoint } from "../../types/entrypoint" @@ -17,12 +19,12 @@ export async function accountId< TEntryPoint extends EntryPoint, TTransport extends Transport, TChain extends Chain | undefined, - TSmartAccount extends - | SmartAccount + TSmartAccount extends SmartAccount | undefined = + | SmartAccount | undefined >( client: Client, - args?: GetAccountParameter + args?: GetAccountParameter ): Promise { let account_ = client.account @@ -36,12 +38,7 @@ export async function accountId< }) } - const account = parseAccount(account_) as SmartAccount< - TEntryPoint, - string, - TTransport, - TChain - > + const account = parseAccount(account_) as SmartAccount const publicClient = account.client @@ -61,7 +58,11 @@ export async function accountId< ] as const try { - return await publicClient.readContract({ + return await getAction( + publicClient, + readContract, + "readContract" + )({ abi, functionName: "accountId", address: account.address @@ -71,7 +72,11 @@ export async function accountId< const factory = await account.getFactory() const factoryData = await account.getFactoryData() - const result = await publicClient.call({ + const result = await getAction( + publicClient, + call, + "call" + )({ factory: factory, factoryData: factoryData, to: account.address, diff --git a/packages/permissionless/actions/erc7579/installModule.ts b/packages/permissionless/actions/erc7579/installModule.ts index 2e5a1341..f61f47de 100644 --- a/packages/permissionless/actions/erc7579/installModule.ts +++ b/packages/permissionless/actions/erc7579/installModule.ts @@ -22,12 +22,8 @@ import { type ModuleType, parseModuleTypeId } from "./supportsModule" export type InstallModuleParameters< TEntryPoint extends EntryPoint, - TTransport extends Transport, - TChain extends Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined -> = GetAccountParameter & { + TSmartAccount extends SmartAccount | undefined +> = GetAccountParameter & { type: ModuleType address: Address context: Hex @@ -40,16 +36,12 @@ export async function installModule< TEntryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined = - | SmartAccount + TSmartAccount extends SmartAccount | undefined = + | SmartAccount | undefined >( client: Client, - parameters: Prettify< - InstallModuleParameters - > + parameters: Prettify> ): Promise { const { account: account_ = client.account, @@ -67,12 +59,7 @@ export async function installModule< }) } - const account = parseAccount(account_) as SmartAccount< - TEntryPoint, - string, - TTransport, - TChain - > + const account = parseAccount(account_) as SmartAccount const installModuleCallData = await account.encodeCallData({ to: account.address, @@ -123,10 +110,5 @@ export async function installModule< }, account: account, middleware - } as SendUserOperationParameters< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - >) + } as SendUserOperationParameters) } diff --git a/packages/permissionless/actions/erc7579/installModules.ts b/packages/permissionless/actions/erc7579/installModules.ts index 0f943c4c..c9a81468 100644 --- a/packages/permissionless/actions/erc7579/installModules.ts +++ b/packages/permissionless/actions/erc7579/installModules.ts @@ -22,12 +22,8 @@ import { type ModuleType, parseModuleTypeId } from "./supportsModule" export type InstallModulesParameters< TEntryPoint extends EntryPoint, - TTransport extends Transport, - TChain extends Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined -> = GetAccountParameter & + TSmartAccount extends SmartAccount | undefined +> = GetAccountParameter & Middleware & { modules: { type: ModuleType @@ -43,17 +39,13 @@ export async function installModules< TEntryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined = - | SmartAccount + TSmartAccount extends SmartAccount | undefined = + | SmartAccount | undefined | undefined >( client: Client, - parameters: Prettify< - InstallModulesParameters - > + parameters: Prettify> ): Promise { const { account: account_ = client.account, @@ -70,12 +62,7 @@ export async function installModules< }) } - const account = parseAccount(account_) as SmartAccount< - TEntryPoint, - string, - TTransport, - TChain - > + const account = parseAccount(account_) as SmartAccount const installModulesCallData = await account.encodeCallData( await Promise.all( @@ -130,10 +117,5 @@ export async function installModules< }, account: account, middleware - } as SendUserOperationParameters< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - >) + } as SendUserOperationParameters) } diff --git a/packages/permissionless/actions/erc7579/isModuleInstalled.ts b/packages/permissionless/actions/erc7579/isModuleInstalled.ts index 6ab614b1..851ae77a 100644 --- a/packages/permissionless/actions/erc7579/isModuleInstalled.ts +++ b/packages/permissionless/actions/erc7579/isModuleInstalled.ts @@ -10,6 +10,8 @@ import { encodeFunctionData, getAddress } from "viem" +import { call, readContract } from "viem/actions" +import { getAction } from "viem/utils" import type { SmartAccount } from "../../accounts/types" import type { GetAccountParameter } from "../../types/" import type { EntryPoint } from "../../types/entrypoint" @@ -19,14 +21,8 @@ import { type ModuleType, parseModuleTypeId } from "./supportsModule" export type IsModuleInstalledParameters< TEntryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined = - | SmartAccount - | undefined -> = GetAccountParameter & { + TSmartAccount extends SmartAccount | undefined +> = GetAccountParameter & { type: ModuleType address: Address context: Hex @@ -36,19 +32,12 @@ export async function isModuleInstalled< TEntryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined = - | SmartAccount + TSmartAccount extends SmartAccount | undefined = + | SmartAccount | undefined >( client: Client, - parameters: IsModuleInstalledParameters< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - > + parameters: IsModuleInstalledParameters ): Promise { const { account: account_ = client.account, address, context } = parameters @@ -58,12 +47,7 @@ export async function isModuleInstalled< }) } - const account = parseAccount(account_) as SmartAccount< - TEntryPoint, - string, - TTransport, - TChain - > + const account = parseAccount(account_) as SmartAccount const publicClient = account.client @@ -95,7 +79,11 @@ export async function isModuleInstalled< ] as const try { - return await publicClient.readContract({ + return await getAction( + publicClient, + readContract, + "readContract" + )({ abi, functionName: "isModuleInstalled", args: [ @@ -110,7 +98,11 @@ export async function isModuleInstalled< const factory = await account.getFactory() const factoryData = await account.getFactoryData() - const result = await publicClient.call({ + const result = await getAction( + publicClient, + call, + "call" + )({ factory: factory, factoryData: factoryData, to: account.address, diff --git a/packages/permissionless/actions/erc7579/supportsExecutionMode.ts b/packages/permissionless/actions/erc7579/supportsExecutionMode.ts index da0d27a3..39820b41 100644 --- a/packages/permissionless/actions/erc7579/supportsExecutionMode.ts +++ b/packages/permissionless/actions/erc7579/supportsExecutionMode.ts @@ -11,6 +11,8 @@ import { toBytes, toHex } from "viem" +import { call, readContract } from "viem/actions" +import { getAction } from "viem/utils" import type { SmartAccount } from "../../accounts/types" import type { GetAccountParameter, Prettify } from "../../types/" import type { EntryPoint } from "../../types/entrypoint" @@ -28,16 +30,11 @@ export type ExecutionMode = { export type SupportsExecutionModeParameters< TEntryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined = - | SmartAccount + TSmartAccount extends SmartAccount | undefined = + | SmartAccount | undefined, callType extends CallType = CallType -> = GetAccountParameter & - ExecutionMode +> = GetAccountParameter & ExecutionMode function parseCallType(callType: CallType) { switch (callType) { @@ -72,21 +69,12 @@ export async function supportsExecutionMode< TEntryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined = - | SmartAccount + TSmartAccount extends SmartAccount | undefined = + | SmartAccount | undefined >( client: Client, - args: Prettify< - SupportsExecutionModeParameters< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - > - > + args: Prettify> ): Promise { const { account: account_ = client.account, @@ -102,12 +90,7 @@ export async function supportsExecutionMode< }) } - const account = parseAccount(account_) as SmartAccount< - TEntryPoint, - string, - TTransport, - TChain - > + const account = parseAccount(account_) as SmartAccount const publicClient = account.client @@ -138,7 +121,11 @@ export async function supportsExecutionMode< ] as const try { - return await publicClient.readContract({ + return await getAction( + publicClient, + readContract, + "readContract" + )({ abi, functionName: "supportsExecutionMode", args: [encodedMode], @@ -149,7 +136,11 @@ export async function supportsExecutionMode< const factory = await account.getFactory() const factoryData = await account.getFactoryData() - const result = await publicClient.call({ + const result = await getAction( + publicClient, + call, + "call" + )({ factory: factory, factoryData: factoryData, to: account.address, diff --git a/packages/permissionless/actions/erc7579/supportsModule.ts b/packages/permissionless/actions/erc7579/supportsModule.ts index 3396d5f5..17f36799 100644 --- a/packages/permissionless/actions/erc7579/supportsModule.ts +++ b/packages/permissionless/actions/erc7579/supportsModule.ts @@ -7,6 +7,8 @@ import { decodeFunctionResult, encodeFunctionData } from "viem" +import { call, readContract } from "viem/actions" +import { getAction } from "viem/utils" import type { SmartAccount } from "../../accounts/types" import type { GetAccountParameter, Prettify } from "../../types/" import type { EntryPoint } from "../../types/entrypoint" @@ -17,14 +19,10 @@ export type ModuleType = "validator" | "executor" | "fallback" | "hook" export type SupportsModuleParameters< TEntryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined = - | SmartAccount + TSmartAccount extends SmartAccount | undefined = + | SmartAccount | undefined -> = GetAccountParameter & { +> = GetAccountParameter & { type: ModuleType } @@ -47,16 +45,12 @@ export async function supportsModule< TEntryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined = - | SmartAccount + TSmartAccount extends SmartAccount | undefined = + | SmartAccount | undefined >( client: Client, - args: Prettify< - SupportsModuleParameters - > + args: Prettify> ): Promise { const { account: account_ = client.account } = args @@ -66,12 +60,7 @@ export async function supportsModule< }) } - const account = parseAccount(account_) as SmartAccount< - TEntryPoint, - string, - TTransport, - TChain - > + const account = parseAccount(account_) as SmartAccount const publicClient = account.client @@ -95,7 +84,11 @@ export async function supportsModule< ] as const try { - return await publicClient.readContract({ + return await getAction( + publicClient, + readContract, + "readContract" + )({ abi, functionName: "supportsModule", args: [parseModuleTypeId(args.type)], @@ -106,7 +99,11 @@ export async function supportsModule< const factory = await account.getFactory() const factoryData = await account.getFactoryData() - const result = await publicClient.call({ + const result = await getAction( + publicClient, + call, + "call" + )({ factory: factory, factoryData: factoryData, to: account.address, diff --git a/packages/permissionless/actions/erc7579/uninstallModule.ts b/packages/permissionless/actions/erc7579/uninstallModule.ts index 617eb7dd..495f044f 100644 --- a/packages/permissionless/actions/erc7579/uninstallModule.ts +++ b/packages/permissionless/actions/erc7579/uninstallModule.ts @@ -22,14 +22,10 @@ import { type ModuleType, parseModuleTypeId } from "./supportsModule" export type UninstallModuleParameters< TEntryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined = - | SmartAccount + TSmartAccount extends SmartAccount | undefined = + | SmartAccount | undefined -> = GetAccountParameter & { +> = GetAccountParameter & { type: ModuleType address: Address context: Hex @@ -42,21 +38,12 @@ export async function uninstallModule< TEntryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined = - | SmartAccount + TSmartAccount extends SmartAccount | undefined = + | SmartAccount | undefined >( client: Client, - parameters: Prettify< - UninstallModuleParameters< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - > - > + parameters: Prettify> ): Promise { const { account: account_ = client.account, @@ -74,12 +61,7 @@ export async function uninstallModule< }) } - const account = parseAccount(account_) as SmartAccount< - TEntryPoint, - string, - TTransport, - TChain - > + const account = parseAccount(account_) as SmartAccount const uninstallModuleCallData = await account.encodeCallData({ to: account.address, @@ -130,10 +112,5 @@ export async function uninstallModule< }, account: account, middleware - } as SendUserOperationParameters< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - >) + } as SendUserOperationParameters) } diff --git a/packages/permissionless/actions/erc7579/uninstallModules.ts b/packages/permissionless/actions/erc7579/uninstallModules.ts index e9a12853..43b47dc5 100644 --- a/packages/permissionless/actions/erc7579/uninstallModules.ts +++ b/packages/permissionless/actions/erc7579/uninstallModules.ts @@ -22,14 +22,10 @@ import { type ModuleType, parseModuleTypeId } from "./supportsModule" export type UninstallModulesParameters< TEntryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined = - | SmartAccount + TSmartAccount extends SmartAccount | undefined = + | SmartAccount | undefined -> = GetAccountParameter & { +> = GetAccountParameter & { modules: [ { type: ModuleType @@ -46,21 +42,12 @@ export async function uninstallModules< TEntryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined = - | SmartAccount + TSmartAccount extends SmartAccount | undefined = + | SmartAccount | undefined >( client: Client, - parameters: Prettify< - UninstallModulesParameters< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - > - > + parameters: Prettify> ): Promise { const { account: account_ = client.account, @@ -137,10 +124,5 @@ export async function uninstallModules< }, account: account, middleware - } as SendUserOperationParameters< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - >) + } as SendUserOperationParameters) } diff --git a/packages/permissionless/actions/smartAccount/deployContract.ts b/packages/permissionless/actions/smartAccount/deployContract.ts index bbb1ee2a..50bfd3b8 100644 --- a/packages/permissionless/actions/smartAccount/deployContract.ts +++ b/packages/permissionless/actions/smartAccount/deployContract.ts @@ -1,5 +1,6 @@ import type { Abi, + Account, Chain, Client, DeployContractParameters, @@ -9,7 +10,6 @@ import type { } from "viem" import { getAction } from "viem/utils" import type { SmartAccount } from "../../accounts/types" -import type { Prettify } from "../../types/" import type { EntryPoint } from "../../types/entrypoint" import { parseAccount } from "../../utils/" import { AccountOrClientNotFoundError } from "../../utils/signUserOperationHashWithECDSA" @@ -23,13 +23,8 @@ import { export type DeployContractParametersWithPaymaster< entryPoint extends EntryPoint, TAbi extends Abi | readonly unknown[] = Abi | readonly unknown[], - TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TAccount extends - | SmartAccount - | undefined = - | SmartAccount - | undefined, + TAccount extends Account | undefined = Account | undefined, TChainOverride extends Chain | undefined = Chain | undefined > = DeployContractParameters & Middleware @@ -66,14 +61,18 @@ export async function deployContract< entryPoint extends EntryPoint, TTransport extends Transport, TChain extends Chain | undefined, - TAccount extends - | SmartAccount - | undefined = - | SmartAccount - | undefined + TAbi extends Abi | readonly unknown[], + TAccount extends SmartAccount | undefined, + TChainOverride extends Chain | undefined >( - client: Client, - args: Prettify> + client: Client, + args: DeployContractParametersWithPaymaster< + entryPoint, + TAbi, + TChain, + TAccount, + TChainOverride + > ): Promise { const { abi, @@ -91,12 +90,7 @@ export async function deployContract< }) } - const account = parseAccount(account_) as SmartAccount< - entryPoint, - string, - TTransport, - TChain - > + const account = parseAccount(account_) as SmartAccount const userOpHash = await getAction( client, @@ -115,7 +109,7 @@ export async function deployContract< }, account: account, middleware - } as SendUserOperationParameters) + } as SendUserOperationParameters) const userOperationReceipt = await getAction( client, diff --git a/packages/permissionless/actions/smartAccount/prepareUserOperationRequest.ts b/packages/permissionless/actions/smartAccount/prepareUserOperationRequest.ts index 444a4150..f719bf8a 100644 --- a/packages/permissionless/actions/smartAccount/prepareUserOperationRequest.ts +++ b/packages/permissionless/actions/smartAccount/prepareUserOperationRequest.ts @@ -72,12 +72,8 @@ export type Middleware = { export type PrepareUserOperationRequestParameters< entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends - | SmartAccount - | undefined = - | SmartAccount + TAccount extends SmartAccount | undefined = + | SmartAccount | undefined > = { userOperation: entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE @@ -111,7 +107,7 @@ export type PrepareUserOperationRequestParameters< | "paymasterData" | "signature" > -} & GetAccountParameter & +} & GetAccountParameter & Middleware export type PrepareUserOperationRequestReturnType< @@ -122,21 +118,12 @@ async function prepareUserOperationRequestForEntryPointV06< entryPoint extends EntryPoint = ENTRYPOINT_ADDRESS_V06_TYPE, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TAccount extends - | SmartAccount - | undefined = - | SmartAccount + TAccount extends SmartAccount | undefined = + | SmartAccount | undefined >( client: Client, - args: Prettify< - PrepareUserOperationRequestParameters< - entryPoint, - TTransport, - TChain, - TAccount - > - >, + args: PrepareUserOperationRequestParameters, stateOverrides?: StateOverrides ): Promise>> { const { @@ -146,12 +133,9 @@ async function prepareUserOperationRequestForEntryPointV06< } = args if (!account_) throw new AccountOrClientNotFoundError() - const account = parseAccount(account_) as SmartAccount< - ENTRYPOINT_ADDRESS_V06_TYPE, - string, - Transport, - TChain - > + const account = parseAccount( + account_ + ) as SmartAccount const [sender, nonce, initCode, callData] = await Promise.all([ partialUserOperation.sender || account.address, @@ -271,21 +255,12 @@ async function prepareUserOperationRequestEntryPointV07< entryPoint extends EntryPoint = ENTRYPOINT_ADDRESS_V07_TYPE, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TAccount extends - | SmartAccount - | undefined = - | SmartAccount + TAccount extends SmartAccount | undefined = + | SmartAccount | undefined >( client: Client, - args: Prettify< - PrepareUserOperationRequestParameters< - entryPoint, - TTransport, - TChain, - TAccount - > - >, + args: Prettify>, stateOverrides?: StateOverrides ): Promise>> { const { @@ -295,12 +270,9 @@ async function prepareUserOperationRequestEntryPointV07< } = args if (!account_) throw new AccountOrClientNotFoundError() - const account = parseAccount(account_) as SmartAccount< - ENTRYPOINT_ADDRESS_V07_TYPE, - string, - Transport, - TChain - > + const account = parseAccount( + account_ + ) as SmartAccount const [sender, nonce, factory, factoryData, callData] = await Promise.all([ partialUserOperation.sender || account.address, @@ -427,46 +399,46 @@ export async function prepareUserOperationRequest< entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TAccount extends - | SmartAccount - | undefined = - | SmartAccount + TAccount extends SmartAccount | undefined = + | SmartAccount | undefined >( client: Client, - args: Prettify< - PrepareUserOperationRequestParameters< - entryPoint, - TTransport, - TChain, - TAccount - > - >, + args: PrepareUserOperationRequestParameters, stateOverrides?: StateOverrides ): Promise>> { const { account: account_ = client.account } = args if (!account_) throw new AccountOrClientNotFoundError() - const account = parseAccount(account_) as SmartAccount< - entryPoint, - string, - TTransport, - TChain - > + const account = parseAccount(account_) as SmartAccount const entryPointVersion = getEntryPointVersion(account.entryPoint) if (entryPointVersion === "v0.6") { return prepareUserOperationRequestForEntryPointV06( - client, - args, + client as unknown as Client< + Transport, + Chain | undefined, + SmartAccount + >, + args as unknown as PrepareUserOperationRequestParameters< + ENTRYPOINT_ADDRESS_V06_TYPE, + SmartAccount + >, stateOverrides ) as Promise> } return prepareUserOperationRequestEntryPointV07( - client, - args, + client as unknown as Client< + Transport, + Chain | undefined, + SmartAccount + >, + args as unknown as PrepareUserOperationRequestParameters< + ENTRYPOINT_ADDRESS_V07_TYPE, + SmartAccount + >, stateOverrides ) as Promise> } diff --git a/packages/permissionless/actions/smartAccount/sendTransaction.ts b/packages/permissionless/actions/smartAccount/sendTransaction.ts index 3f64cb4c..72d4208f 100644 --- a/packages/permissionless/actions/smartAccount/sendTransaction.ts +++ b/packages/permissionless/actions/smartAccount/sendTransaction.ts @@ -16,12 +16,9 @@ import { sendUserOperation } from "./sendUserOperation" export type SendTransactionWithPaymasterParameters< entryPoint extends EntryPoint, - TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TAccount extends - | SmartAccount - | undefined = - | SmartAccount + TAccount extends SmartAccount | undefined = + | SmartAccount | undefined, TChainOverride extends Chain | undefined = Chain | undefined > = SendTransactionParameters & @@ -76,9 +73,7 @@ export type SendTransactionWithPaymasterParameters< export async function sendTransaction< TTransport extends Transport, TChain extends Chain | undefined, - TAccount extends - | SmartAccount - | undefined, + TAccount extends SmartAccount | undefined, entryPoint extends EntryPoint, TChainOverride extends Chain | undefined = Chain | undefined >( @@ -86,7 +81,6 @@ export async function sendTransaction< args: Prettify< SendTransactionWithPaymasterParameters< entryPoint, - TTransport, TChain, TAccount, TChainOverride @@ -110,12 +104,7 @@ export async function sendTransaction< }) } - const account = parseAccount(account_) as SmartAccount< - entryPoint, - string, - TTransport, - TChain - > + const account = parseAccount(account_) as SmartAccount if (!to) throw new Error("Missing to address") @@ -135,7 +124,7 @@ export async function sendTransaction< entryPoint, TTransport, TChain, - SmartAccount + SmartAccount >, "sendUserOperation" )({ diff --git a/packages/permissionless/actions/smartAccount/sendTransactions.ts b/packages/permissionless/actions/smartAccount/sendTransactions.ts index ef03a44d..863fb24b 100644 --- a/packages/permissionless/actions/smartAccount/sendTransactions.ts +++ b/packages/permissionless/actions/smartAccount/sendTransactions.ts @@ -18,17 +18,13 @@ import { sendUserOperation } from "./sendUserOperation" export type SendTransactionsWithPaymasterParameters< entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends - | SmartAccount - | undefined = - | SmartAccount + TAccount extends SmartAccount | undefined = + | SmartAccount | undefined | undefined > = { transactions: { to: Address; value: bigint; data: Hex }[] -} & GetAccountParameter & +} & GetAccountParameter & Middleware & { maxFeePerGas?: bigint maxPriorityFeePerGas?: bigint @@ -84,19 +80,12 @@ export type SendTransactionsWithPaymasterParameters< export async function sendTransactions< TTransport extends Transport, TChain extends Chain | undefined, - TAccount extends - | SmartAccount - | undefined, + TAccount extends SmartAccount | undefined, entryPoint extends EntryPoint >( client: Client, args: Prettify< - SendTransactionsWithPaymasterParameters< - entryPoint, - TTransport, - TChain, - TAccount - > + SendTransactionsWithPaymasterParameters > ): Promise { const { @@ -114,12 +103,7 @@ export async function sendTransactions< }) } - const account = parseAccount(account_) as SmartAccount< - entryPoint, - string, - TTransport, - TChain - > + const account = parseAccount(account_) as SmartAccount if (account.type !== "local") { throw new Error("RPC account type not supported") @@ -142,7 +126,7 @@ export async function sendTransactions< entryPoint, TTransport, TChain, - SmartAccount + SmartAccount >, "sendUserOperation" )({ diff --git a/packages/permissionless/actions/smartAccount/sendUserOperation.ts b/packages/permissionless/actions/smartAccount/sendUserOperation.ts index b78d19ac..b5a2dbeb 100644 --- a/packages/permissionless/actions/smartAccount/sendUserOperation.ts +++ b/packages/permissionless/actions/smartAccount/sendUserOperation.ts @@ -22,12 +22,8 @@ import { export type SendUserOperationParameters< entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends - | SmartAccount - | undefined = - | SmartAccount + TAccount extends SmartAccount | undefined = + | SmartAccount | undefined > = { userOperation: entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE @@ -61,33 +57,24 @@ export type SendUserOperationParameters< | "paymasterData" | "signature" > -} & GetAccountParameter & +} & GetAccountParameter & Middleware export async function sendUserOperation< entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TAccount extends - | SmartAccount - | undefined = - | SmartAccount + TAccount extends SmartAccount | undefined = + | SmartAccount | undefined >( client: Client, - args: Prettify< - SendUserOperationParameters - > + args: Prettify> ): Promise { const { account: account_ = client.account } = args if (!account_) throw new AccountOrClientNotFoundError() - const account = parseAccount(account_) as SmartAccount< - entryPoint, - string, - TTransport, - TChain - > + const account = parseAccount(account_) as SmartAccount const userOperation = await getAction( client, @@ -95,8 +82,6 @@ export async function sendUserOperation< "prepareUserOperationRequest" )({ ...args, account } as PrepareUserOperationRequestParameters< entryPoint, - TTransport, - TChain, TAccount >) diff --git a/packages/permissionless/actions/smartAccount/signMessage.ts b/packages/permissionless/actions/smartAccount/signMessage.ts index 8c0deebc..4a08426a 100644 --- a/packages/permissionless/actions/smartAccount/signMessage.ts +++ b/packages/permissionless/actions/smartAccount/signMessage.ts @@ -59,13 +59,11 @@ export async function signMessage< entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TAccount extends - | SmartAccount - | undefined = - | SmartAccount + TAccount extends SmartAccount | undefined = + | SmartAccount | undefined >( - client: Client, + client: Client, { account: account_ = client.account, message diff --git a/packages/permissionless/actions/smartAccount/signTypedData.ts b/packages/permissionless/actions/smartAccount/signTypedData.ts index 438f3b39..bae146d1 100644 --- a/packages/permissionless/actions/smartAccount/signTypedData.ts +++ b/packages/permissionless/actions/smartAccount/signTypedData.ts @@ -118,13 +118,11 @@ export async function signTypedData< TPrimaryType extends string, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TAccount extends - | SmartAccount - | undefined = - | SmartAccount + TAccount extends SmartAccount | undefined = + | SmartAccount | undefined >( - client: Client, + client: Client, { account: account_ = client.account, domain, diff --git a/packages/permissionless/actions/smartAccount/writeContract.ts b/packages/permissionless/actions/smartAccount/writeContract.ts index 67a231aa..0ac942b0 100644 --- a/packages/permissionless/actions/smartAccount/writeContract.ts +++ b/packages/permissionless/actions/smartAccount/writeContract.ts @@ -72,12 +72,9 @@ import { */ export type WriteContractWithPaymasterParameters< entryPoint extends EntryPoint, - TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TAccount extends - | SmartAccount - | undefined = - | SmartAccount + TAccount extends SmartAccount | undefined = + | SmartAccount | undefined, TAbi extends Abi | readonly unknown[] = Abi | readonly unknown[], TFunctionName extends ContractFunctionName< @@ -104,9 +101,7 @@ export async function writeContract< entryPoint extends EntryPoint, TTransport extends Transport, TChain extends Chain | undefined, - TAccount extends - | SmartAccount - | undefined, + TAccount extends SmartAccount | undefined, const TAbi extends Abi | readonly unknown[], TFunctionName extends ContractFunctionName< TAbi, @@ -129,7 +124,6 @@ export async function writeContract< ...request }: WriteContractWithPaymasterParameters< entryPoint, - TTransport, TChain, TAccount, TAbi, @@ -159,7 +153,6 @@ export async function writeContract< ...request } as unknown as SendTransactionWithPaymasterParameters< entryPoint, - TTransport, TChain, TAccount, TChainOverride diff --git a/packages/permissionless/clients/createBundlerClient.ts b/packages/permissionless/clients/createBundlerClient.ts index 4ea177dd..7e295133 100644 --- a/packages/permissionless/clients/createBundlerClient.ts +++ b/packages/permissionless/clients/createBundlerClient.ts @@ -12,11 +12,12 @@ import { type BundlerActions, bundlerActions } from "./decorators/bundler" export type BundlerClient< entryPoint extends EntryPoint, - TChain extends Chain | undefined = Chain | undefined + TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined > = Client< Transport, TChain, - Account | undefined, + TClientAccount, BundlerRpcSchema, BundlerActions > diff --git a/packages/permissionless/clients/decorators/smartAccount.ts b/packages/permissionless/clients/decorators/smartAccount.ts index 39c472c2..0a20aac7 100644 --- a/packages/permissionless/clients/decorators/smartAccount.ts +++ b/packages/permissionless/clients/decorators/smartAccount.ts @@ -47,10 +47,8 @@ export type SmartAccountActions< entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined = - | SmartAccount + TSmartAccount extends SmartAccount | undefined = + | SmartAccount | undefined > = { /** @@ -299,13 +297,11 @@ export type SmartAccountActions< const TAbi extends Abi | readonly unknown[], TChainOverride extends Chain | undefined = undefined >( - args: Prettify< - DeployContractParameters< - TAbi, - TChain, - TSmartAccount, - TChainOverride - > + args: DeployContractParameters< + TAbi, + TChain, + TSmartAccount, + TChainOverride > ) => Promise /** @@ -390,16 +386,14 @@ export type SmartAccountActions< > > prepareUserOperationRequest: ( - args: Prettify< - Parameters< - typeof prepareUserOperationRequest< - entryPoint, - TTransport, - TChain, - TSmartAccount - > - >[1] - >, + args: Parameters< + typeof prepareUserOperationRequest< + entryPoint, + TTransport, + TChain, + TSmartAccount + > + >[1], stateOverrides?: StateOverrides ) => Promise>> sendUserOperation: ( @@ -465,12 +459,7 @@ export type SmartAccountActions< */ sendTransactions: ( args: Prettify< - SendTransactionsWithPaymasterParameters< - entryPoint, - TTransport, - TChain, - TSmartAccount - > + SendTransactionsWithPaymasterParameters > ) => ReturnType< typeof sendTransactions @@ -483,10 +472,8 @@ export function smartAccountActions({ return < TTransport extends Transport, TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends - | SmartAccount - | undefined = - | SmartAccount + TSmartAccount extends SmartAccount | undefined = + | SmartAccount | undefined >( client: Client @@ -500,11 +487,34 @@ export function smartAccountActions({ }, stateOverrides ), - deployContract: (args) => - deployContract(client, { + deployContract: < + const TAbi extends Abi | readonly unknown[], + TChainOverride extends Chain | undefined = undefined + >( + args: DeployContractParameters< + TAbi, + TChain, + TSmartAccount, + TChainOverride + > + ) => + deployContract< + entryPoint, + TTransport, + TChain, + TAbi, + TSmartAccount, + TChainOverride + >(client, { ...args, middleware - } as DeployContractParametersWithPaymaster), + } as unknown as DeployContractParametersWithPaymaster< + entryPoint, + TAbi, + TChain, + TSmartAccount, + TChainOverride + >), sendTransaction: (args) => sendTransaction( client, @@ -513,7 +523,6 @@ export function smartAccountActions({ middleware } as SendTransactionWithPaymasterParameters< entryPoint, - TTransport, TChain, TSmartAccount > @@ -532,12 +541,7 @@ export function smartAccountActions({ { ...args, middleware - } as SendUserOperationParameters< - entryPoint, - TTransport, - TChain, - TSmartAccount - > + } as SendUserOperationParameters ), signMessage: (args) => signMessage( @@ -598,7 +602,6 @@ export function smartAccountActions({ middleware } as WriteContractWithPaymasterParameters< entryPoint, - TTransport, TChain, TSmartAccount, TAbi diff --git a/packages/permissionless/experimental/eip7677/actions/getPaymasterData.ts b/packages/permissionless/experimental/eip7677/actions/getPaymasterData.ts index c656a105..9482f8e2 100644 --- a/packages/permissionless/experimental/eip7677/actions/getPaymasterData.ts +++ b/packages/permissionless/experimental/eip7677/actions/getPaymasterData.ts @@ -1,4 +1,5 @@ import { + type Account, type Address, type Chain, ChainNotFoundError, @@ -81,13 +82,14 @@ export type GetPaymasterDataReturnType = export async function getPaymasterData< TEntryPoint extends EntryPoint, TChain extends Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined, TTransport extends Transport = Transport, TChainOverride extends Chain | undefined = Chain | undefined >( client: Client< TTransport, TChain, - undefined, + TClientAccount, Eip7677RpcSchema >, { diff --git a/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.ts b/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.ts index 0d0ba502..da062abc 100644 --- a/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.ts +++ b/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.ts @@ -1,4 +1,5 @@ import { + type Account, type Address, type Chain, ChainNotFoundError, @@ -88,12 +89,13 @@ export async function getPaymasterStubData< TEntryPoint extends EntryPoint, TChain extends Chain | undefined, TTransport extends Transport = Transport, + TClientAccount extends Account | undefined = Account | undefined, TChainOverride extends Chain | undefined = Chain | undefined >( client: Client< TTransport, TChain, - undefined, + TClientAccount, Eip7677RpcSchema >, { diff --git a/packages/permissionless/experimental/eip7677/clients/decorators/paymasterActionsEip7677.ts b/packages/permissionless/experimental/eip7677/clients/decorators/paymasterActionsEip7677.ts index 452d0658..9ee6b955 100644 --- a/packages/permissionless/experimental/eip7677/clients/decorators/paymasterActionsEip7677.ts +++ b/packages/permissionless/experimental/eip7677/clients/decorators/paymasterActionsEip7677.ts @@ -1,4 +1,4 @@ -import type { Chain, Client, Transport } from "viem" +import type { Account, Chain, Client, Transport } from "viem" import type { EntryPoint } from "../../../../types/entrypoint" import { type GetPaymasterDataParameters, @@ -38,7 +38,8 @@ const paymasterActionsEip7677 = (entryPoint: TEntryPoint) => < TTransport extends Transport, - TChain extends Chain | undefined = Chain | undefined + TChain extends Chain | undefined = Chain | undefined, + TClientAccount extends Account | undefined = Account | undefined >( client: Client ): PaymasterActionsEip7677 => ({ @@ -47,7 +48,7 @@ const paymasterActionsEip7677 = client as Client< TTransport, TChain, - undefined, + TClientAccount, Eip7677RpcSchema >, { @@ -62,7 +63,7 @@ const paymasterActionsEip7677 = client as Client< TTransport, TChain, - undefined, + TClientAccount, Eip7677RpcSchema >, { diff --git a/packages/permissionless/types/index.ts b/packages/permissionless/types/index.ts index 05462f29..6ce78630 100644 --- a/packages/permissionless/types/index.ts +++ b/packages/permissionless/types/index.ts @@ -25,14 +25,14 @@ export type GetAccountParameterWithClient< export type GetAccountParameter< entryPoint extends EntryPoint, - TTransport extends Transport, - TChain extends Chain | undefined, - TAccount extends - | SmartAccount - | undefined + TAccount extends SmartAccount | undefined > = IsUndefined extends true - ? { account: SmartAccount } - : { account?: SmartAccount } + ? { + account: SmartAccount + } + : { + account?: SmartAccount + } export type Prettify = { [K in keyof T]: T[K] From e137fbee1d16c07a17eb8907b011f4bc904e310e Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Mon, 12 Aug 2024 13:12:38 +0100 Subject: [PATCH 02/51] Default public account to undefined --- .../accounts/biconomy/privateKeyToBiconomySmartAccount.ts | 2 +- .../accounts/biconomy/signerToBiconomySmartAccount.ts | 2 +- .../accounts/kernel/signerToEcdsaKernelSmartAccount.ts | 4 ++-- .../accounts/light/privateKeyToLightSmartAccount.ts | 2 +- .../accounts/light/signerToLightSmartAccount.ts | 4 ++-- .../accounts/safe/privateKeyToSafeSmartAccount.ts | 2 +- .../permissionless/accounts/safe/signerToSafeSmartAccount.ts | 4 ++-- .../accounts/simple/privateKeyToSimpleSmartAccount.ts | 2 +- .../accounts/simple/signerToSimpleSmartAccount.ts | 4 ++-- .../accounts/trust/privateKeyToTrustSmartAccount.ts | 2 +- .../accounts/trust/signerToTrustSmartAccount.ts | 2 +- .../permissionless/accounts/trust/utils/getAccountAddress.ts | 2 +- packages/permissionless/accounts/trust/utils/signMessage.ts | 2 +- .../permissionless/accounts/trust/utils/signUserOperation.ts | 2 +- packages/permissionless/clients/createBundlerClient.ts | 2 +- .../experimental/eip7677/actions/getPaymasterData.ts | 2 +- .../experimental/eip7677/actions/getPaymasterStubData.ts | 2 +- .../eip7677/clients/decorators/paymasterActionsEip7677.ts | 2 +- 18 files changed, 22 insertions(+), 22 deletions(-) diff --git a/packages/permissionless/accounts/biconomy/privateKeyToBiconomySmartAccount.ts b/packages/permissionless/accounts/biconomy/privateKeyToBiconomySmartAccount.ts index a7b4665c..3aec0f37 100644 --- a/packages/permissionless/accounts/biconomy/privateKeyToBiconomySmartAccount.ts +++ b/packages/permissionless/accounts/biconomy/privateKeyToBiconomySmartAccount.ts @@ -24,7 +24,7 @@ export async function privateKeyToBiconomySmartAccount< entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined + TClientAccount extends Account | undefined = undefined >( client: Client, { diff --git a/packages/permissionless/accounts/biconomy/signerToBiconomySmartAccount.ts b/packages/permissionless/accounts/biconomy/signerToBiconomySmartAccount.ts index fe53e130..329d5e3b 100644 --- a/packages/permissionless/accounts/biconomy/signerToBiconomySmartAccount.ts +++ b/packages/permissionless/accounts/biconomy/signerToBiconomySmartAccount.ts @@ -212,7 +212,7 @@ export async function signerToBiconomySmartAccount< entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined, + TClientAccount extends Account | undefined = undefined, TSource extends string = string, TAddress extends Address = Address >( diff --git a/packages/permissionless/accounts/kernel/signerToEcdsaKernelSmartAccount.ts b/packages/permissionless/accounts/kernel/signerToEcdsaKernelSmartAccount.ts index ac0e5d1b..a16c1203 100644 --- a/packages/permissionless/accounts/kernel/signerToEcdsaKernelSmartAccount.ts +++ b/packages/permissionless/accounts/kernel/signerToEcdsaKernelSmartAccount.ts @@ -317,7 +317,7 @@ const getAccountAddress = async < entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined + TClientAccount extends Account | undefined = undefined >({ client, entryPoint: entryPointAddress, @@ -374,7 +374,7 @@ export async function signerToEcdsaKernelSmartAccount< entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined, + TClientAccount extends Account | undefined = undefined, TSource extends string = string, TAddress extends Address = Address >( diff --git a/packages/permissionless/accounts/light/privateKeyToLightSmartAccount.ts b/packages/permissionless/accounts/light/privateKeyToLightSmartAccount.ts index b5a5556d..c243054e 100644 --- a/packages/permissionless/accounts/light/privateKeyToLightSmartAccount.ts +++ b/packages/permissionless/accounts/light/privateKeyToLightSmartAccount.ts @@ -25,7 +25,7 @@ export async function privateKeyToLightSmartAccount< entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined + TClientAccount extends Account | undefined = undefined >( client: Client, { privateKey, ...rest }: PrivateKeyToLightSmartAccountParameters diff --git a/packages/permissionless/accounts/light/signerToLightSmartAccount.ts b/packages/permissionless/accounts/light/signerToLightSmartAccount.ts index 7f369f77..d3a43aea 100644 --- a/packages/permissionless/accounts/light/signerToLightSmartAccount.ts +++ b/packages/permissionless/accounts/light/signerToLightSmartAccount.ts @@ -80,7 +80,7 @@ const getAccountAddress = async < entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined + TClientAccount extends Account | undefined = undefined >({ client, factoryAddress, @@ -191,7 +191,7 @@ export async function signerToLightSmartAccount< entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined, + TClientAccount extends Account | undefined = undefined, TSource extends string = string, TAddress extends Address = Address >( diff --git a/packages/permissionless/accounts/safe/privateKeyToSafeSmartAccount.ts b/packages/permissionless/accounts/safe/privateKeyToSafeSmartAccount.ts index e96946d6..a16bdfa6 100644 --- a/packages/permissionless/accounts/safe/privateKeyToSafeSmartAccount.ts +++ b/packages/permissionless/accounts/safe/privateKeyToSafeSmartAccount.ts @@ -24,7 +24,7 @@ export async function privateKeyToSafeSmartAccount< entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined + TClientAccount extends Account | undefined = undefined >( client: Client, { privateKey, ...rest }: PrivateKeyToSafeSmartAccountParameters diff --git a/packages/permissionless/accounts/safe/signerToSafeSmartAccount.ts b/packages/permissionless/accounts/safe/signerToSafeSmartAccount.ts index a404e350..f21062ad 100644 --- a/packages/permissionless/accounts/safe/signerToSafeSmartAccount.ts +++ b/packages/permissionless/accounts/safe/signerToSafeSmartAccount.ts @@ -864,7 +864,7 @@ const getAccountInitCode = async ({ const getAccountAddress = async < TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined + TClientAccount extends Account | undefined = undefined >({ client, owner, @@ -1087,7 +1087,7 @@ export async function signerToSafeSmartAccount< entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined, + TClientAccount extends Account | undefined = undefined, TSource extends string = string, TAddress extends Address = Address, TErc7579 extends Address | undefined = undefined diff --git a/packages/permissionless/accounts/simple/privateKeyToSimpleSmartAccount.ts b/packages/permissionless/accounts/simple/privateKeyToSimpleSmartAccount.ts index 135f827d..43e9ec1f 100644 --- a/packages/permissionless/accounts/simple/privateKeyToSimpleSmartAccount.ts +++ b/packages/permissionless/accounts/simple/privateKeyToSimpleSmartAccount.ts @@ -24,7 +24,7 @@ export async function privateKeyToSimpleSmartAccount< entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined + TClientAccount extends Account | undefined = undefined >( client: Client, { diff --git a/packages/permissionless/accounts/simple/signerToSimpleSmartAccount.ts b/packages/permissionless/accounts/simple/signerToSimpleSmartAccount.ts index 16dd0ea2..8e448b2c 100644 --- a/packages/permissionless/accounts/simple/signerToSimpleSmartAccount.ts +++ b/packages/permissionless/accounts/simple/signerToSimpleSmartAccount.ts @@ -76,7 +76,7 @@ const getAccountAddress = async < entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined + TClientAccount extends Account | undefined = undefined >({ client, factoryAddress, @@ -142,7 +142,7 @@ export async function signerToSimpleSmartAccount< entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined, + TClientAccount extends Account | undefined = undefined, TSource extends string = string, TAddress extends Address = Address >( diff --git a/packages/permissionless/accounts/trust/privateKeyToTrustSmartAccount.ts b/packages/permissionless/accounts/trust/privateKeyToTrustSmartAccount.ts index 45f33611..13ddc59c 100644 --- a/packages/permissionless/accounts/trust/privateKeyToTrustSmartAccount.ts +++ b/packages/permissionless/accounts/trust/privateKeyToTrustSmartAccount.ts @@ -24,7 +24,7 @@ export async function privateKeyToTrustSmartAccount< entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined + TClientAccount extends Account | undefined = undefined >( client: Client, { privateKey, ...rest }: PrivateKeyToTrustSmartAccountParameters diff --git a/packages/permissionless/accounts/trust/signerToTrustSmartAccount.ts b/packages/permissionless/accounts/trust/signerToTrustSmartAccount.ts index 5384ac25..4869b6a3 100644 --- a/packages/permissionless/accounts/trust/signerToTrustSmartAccount.ts +++ b/packages/permissionless/accounts/trust/signerToTrustSmartAccount.ts @@ -99,7 +99,7 @@ export async function signerToTrustSmartAccount< entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined, + TClientAccount extends Account | undefined = undefined, TSource extends string = string, TAddress extends Address = Address >( diff --git a/packages/permissionless/accounts/trust/utils/getAccountAddress.ts b/packages/permissionless/accounts/trust/utils/getAccountAddress.ts index ffb65dff..dcf054ae 100644 --- a/packages/permissionless/accounts/trust/utils/getAccountAddress.ts +++ b/packages/permissionless/accounts/trust/utils/getAccountAddress.ts @@ -14,7 +14,7 @@ export const getAccountAddress = async < entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined + TClientAccount extends Account | undefined = undefined >( client: Client, { diff --git a/packages/permissionless/accounts/trust/utils/signMessage.ts b/packages/permissionless/accounts/trust/utils/signMessage.ts index 75099d24..57596437 100644 --- a/packages/permissionless/accounts/trust/utils/signMessage.ts +++ b/packages/permissionless/accounts/trust/utils/signMessage.ts @@ -5,7 +5,7 @@ import { signMessage as viem_signMessage } from "viem/actions" export const signMessage = < TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined + TClientAccount extends Account | undefined = undefined >( client: Client, { account, message }: SignMessageParameters diff --git a/packages/permissionless/accounts/trust/utils/signUserOperation.ts b/packages/permissionless/accounts/trust/utils/signUserOperation.ts index b5e01d89..22b5f32b 100644 --- a/packages/permissionless/accounts/trust/utils/signUserOperation.ts +++ b/packages/permissionless/accounts/trust/utils/signUserOperation.ts @@ -11,7 +11,7 @@ export const signUserOperation = async < entryPoint extends EntryPoint, TTransport extends Transport = Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined + TClientAccount extends Account | undefined = undefined >( client: Client, { diff --git a/packages/permissionless/clients/createBundlerClient.ts b/packages/permissionless/clients/createBundlerClient.ts index 7e295133..28fe4d48 100644 --- a/packages/permissionless/clients/createBundlerClient.ts +++ b/packages/permissionless/clients/createBundlerClient.ts @@ -13,7 +13,7 @@ import { type BundlerActions, bundlerActions } from "./decorators/bundler" export type BundlerClient< entryPoint extends EntryPoint, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined + TClientAccount extends Account | undefined = undefined > = Client< Transport, TChain, diff --git a/packages/permissionless/experimental/eip7677/actions/getPaymasterData.ts b/packages/permissionless/experimental/eip7677/actions/getPaymasterData.ts index 9482f8e2..a1fadf4e 100644 --- a/packages/permissionless/experimental/eip7677/actions/getPaymasterData.ts +++ b/packages/permissionless/experimental/eip7677/actions/getPaymasterData.ts @@ -82,7 +82,7 @@ export type GetPaymasterDataReturnType = export async function getPaymasterData< TEntryPoint extends EntryPoint, TChain extends Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined, + TClientAccount extends Account | undefined = undefined, TTransport extends Transport = Transport, TChainOverride extends Chain | undefined = Chain | undefined >( diff --git a/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.ts b/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.ts index da062abc..19108e2b 100644 --- a/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.ts +++ b/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.ts @@ -89,7 +89,7 @@ export async function getPaymasterStubData< TEntryPoint extends EntryPoint, TChain extends Chain | undefined, TTransport extends Transport = Transport, - TClientAccount extends Account | undefined = Account | undefined, + TClientAccount extends Account | undefined = undefined, TChainOverride extends Chain | undefined = Chain | undefined >( client: Client< diff --git a/packages/permissionless/experimental/eip7677/clients/decorators/paymasterActionsEip7677.ts b/packages/permissionless/experimental/eip7677/clients/decorators/paymasterActionsEip7677.ts index 9ee6b955..bffdb614 100644 --- a/packages/permissionless/experimental/eip7677/clients/decorators/paymasterActionsEip7677.ts +++ b/packages/permissionless/experimental/eip7677/clients/decorators/paymasterActionsEip7677.ts @@ -39,7 +39,7 @@ const paymasterActionsEip7677 = < TTransport extends Transport, TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = Account | undefined + TClientAccount extends Account | undefined = undefined >( client: Client ): PaymasterActionsEip7677 => ({ From b5f998abd9ab2c0e9bf80876252a4cc5eaad7e3e Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Mon, 12 Aug 2024 13:15:52 +0100 Subject: [PATCH 03/51] Fix build --- .../accounts/kernel/signerToEcdsaKernelSmartAccount.ts | 2 +- .../permissionless/accounts/light/signerToLightSmartAccount.ts | 2 +- .../permissionless/accounts/safe/signerToSafeSmartAccount.ts | 2 +- .../accounts/simple/signerToSimpleSmartAccount.ts | 2 +- 4 files changed, 4 insertions(+), 4 deletions(-) diff --git a/packages/permissionless/accounts/kernel/signerToEcdsaKernelSmartAccount.ts b/packages/permissionless/accounts/kernel/signerToEcdsaKernelSmartAccount.ts index a16c1203..d1714f6c 100644 --- a/packages/permissionless/accounts/kernel/signerToEcdsaKernelSmartAccount.ts +++ b/packages/permissionless/accounts/kernel/signerToEcdsaKernelSmartAccount.ts @@ -431,7 +431,7 @@ export async function signerToEcdsaKernelSmartAccount< // Fetch account address and chain id const [accountAddress, chainId] = await Promise.all([ address ?? - getAccountAddress({ + getAccountAddress({ client, factory: entryPointVersion === "v0.6" diff --git a/packages/permissionless/accounts/light/signerToLightSmartAccount.ts b/packages/permissionless/accounts/light/signerToLightSmartAccount.ts index d3a43aea..65a65f08 100644 --- a/packages/permissionless/accounts/light/signerToLightSmartAccount.ts +++ b/packages/permissionless/accounts/light/signerToLightSmartAccount.ts @@ -225,7 +225,7 @@ export async function signerToLightSmartAccount< const [accountAddress, chainId] = await Promise.all([ address ?? - getAccountAddress({ + getAccountAddress({ client, factoryAddress, entryPoint: entryPointAddress, diff --git a/packages/permissionless/accounts/safe/signerToSafeSmartAccount.ts b/packages/permissionless/accounts/safe/signerToSafeSmartAccount.ts index f21062ad..419b022e 100644 --- a/packages/permissionless/accounts/safe/signerToSafeSmartAccount.ts +++ b/packages/permissionless/accounts/safe/signerToSafeSmartAccount.ts @@ -1175,7 +1175,7 @@ export async function signerToSafeSmartAccount< const accountAddress = address ?? - (await getAccountAddress({ + (await getAccountAddress({ client, owner: viemSigner.address, safeModuleSetupAddress, diff --git a/packages/permissionless/accounts/simple/signerToSimpleSmartAccount.ts b/packages/permissionless/accounts/simple/signerToSimpleSmartAccount.ts index 8e448b2c..0a9a8333 100644 --- a/packages/permissionless/accounts/simple/signerToSimpleSmartAccount.ts +++ b/packages/permissionless/accounts/simple/signerToSimpleSmartAccount.ts @@ -167,7 +167,7 @@ export async function signerToSimpleSmartAccount< const [accountAddress, chainId] = await Promise.all([ address ?? - getAccountAddress({ + getAccountAddress({ client, factoryAddress, entryPoint: entryPointAddress, From 18e4f0ac9d94d66869b113e85499d7a5d9c65c82 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Thu, 15 Aug 2024 17:40:03 +0100 Subject: [PATCH 04/51] Make smart accounts compatible with viem --- bun.lockb | Bin 579160 -> 579552 bytes package.json | 2 +- .../biconomy/abi/BiconomySmartAccountAbi.ts | 99 ++- .../privateKeyToBiconomySmartAccount.ts | 46 -- .../biconomy/signerToBiconomySmartAccount.ts | 449 ---------- .../biconomy/toBiconomySmartAccount.ts | 363 ++++++++ packages/permissionless/accounts/index.ts | 148 ++-- .../accounts/kernel/abi/KernelV3AccountAbi.ts | 705 ++++++++++++++++ ...ccount.ts => toEcdsaKernelSmartAccount.ts} | 468 ++++++----- .../accounts/kernel/utils/encodeCallData.ts | 122 +-- .../accounts/kernel/utils/getNonceKey.ts | 11 +- .../accounts/kernel/utils/isKernelV2.ts | 5 +- .../accounts/kernel/utils/signMessage.ts | 49 +- .../accounts/kernel/utils/signTypedData.ts | 50 +- .../accounts/kernel/utils/wrapMessageHash.ts | 11 +- .../light/privateKeyToLightSmartAccount.ts | 39 - ...SmartAccount.ts => toLightSmartAccount.ts} | 388 ++++----- .../safe/privateKeyToSafeSmartAccount.ts | 38 - ...eSmartAccount.ts => toSafeSmartAccount.ts} | 774 +++++++++--------- .../simple/privateKeyToSimpleSmartAccount.ts | 48 -- ...martAccount.ts => toSimpleSmartAccount.ts} | 354 ++++---- .../permissionless/accounts/toSmartAccount.ts | 176 ---- .../trust/privateKeyToTrustSmartAccount.ts | 38 - .../trust/signerToTrustSmartAccount.ts | 236 ------ .../accounts/trust/toTrustSmartAccount.ts | 238 ++++++ .../accounts/trust/utils/encodeCallData.ts | 46 +- .../accounts/trust/utils/getAccountAddress.ts | 23 +- .../accounts/trust/utils/getDummySignature.ts | 12 - .../accounts/trust/utils/signMessage.ts | 14 - .../accounts/trust/utils/signTransaction.ts | 15 - .../accounts/trust/utils/signUserOperation.ts | 39 - packages/permissionless/accounts/types.ts | 62 +- .../actions/bundler/chainId.test.ts | 25 - .../permissionless/actions/bundler/chainId.ts | 42 - .../bundler/estimateUserOperationGas.test.ts | 205 ----- .../bundler/estimateUserOperationGas.ts | 146 ---- .../bundler/getUserOperationByHash.test.ts | 139 ---- .../actions/bundler/getUserOperationByHash.ts | 110 --- .../bundler/getUserOperationReceipt.test.ts | 127 --- .../bundler/getUserOperationReceipt.ts | 127 --- .../actions/bundler/sendUserOperation.test.ts | 126 --- .../actions/bundler/sendUserOperation.ts | 72 -- .../bundler/supportedEntryPoints.test.ts | 27 - .../actions/bundler/supportedEntryPoints.ts | 40 - .../waitForUserOperationReceipt.test.ts | 130 --- .../bundler/waitForUserOperationReceipt.ts | 130 --- .../actions/erc7579/accountId.ts | 31 +- .../actions/erc7579/installModule.ts | 25 +- .../actions/public/getAccountNonce.ts | 27 +- .../actions/public/getSenderAddress.ts | 61 +- .../clients/createBundlerClient.test.ts | 26 - .../clients/createBundlerClient.ts | 60 -- packages/permissionless/package.json | 2 +- packages/permissionless/types/entrypoint.ts | 13 - packages/permissionless/types/index.ts | 38 +- .../permissionless/types/userOperation.ts | 41 - ...llData.test.ts => encode7579Calls.test.ts} | 23 +- ...code7579CallData.ts => encode7579Calls.ts} | 36 +- .../utils/getUserOperationHash.ts | 158 ---- .../utils/isSmartAccountDeployed.ts | 9 +- 60 files changed, 2634 insertions(+), 4430 deletions(-) delete mode 100644 packages/permissionless/accounts/biconomy/privateKeyToBiconomySmartAccount.ts delete mode 100644 packages/permissionless/accounts/biconomy/signerToBiconomySmartAccount.ts create mode 100644 packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts rename packages/permissionless/accounts/kernel/{signerToEcdsaKernelSmartAccount.ts => toEcdsaKernelSmartAccount.ts} (56%) delete mode 100644 packages/permissionless/accounts/light/privateKeyToLightSmartAccount.ts rename packages/permissionless/accounts/light/{signerToLightSmartAccount.ts => toLightSmartAccount.ts} (51%) delete mode 100644 packages/permissionless/accounts/safe/privateKeyToSafeSmartAccount.ts rename packages/permissionless/accounts/safe/{signerToSafeSmartAccount.ts => toSafeSmartAccount.ts} (69%) delete mode 100644 packages/permissionless/accounts/simple/privateKeyToSimpleSmartAccount.ts rename packages/permissionless/accounts/simple/{signerToSimpleSmartAccount.ts => toSimpleSmartAccount.ts} (54%) delete mode 100644 packages/permissionless/accounts/toSmartAccount.ts delete mode 100644 packages/permissionless/accounts/trust/privateKeyToTrustSmartAccount.ts delete mode 100644 packages/permissionless/accounts/trust/signerToTrustSmartAccount.ts create mode 100644 packages/permissionless/accounts/trust/toTrustSmartAccount.ts delete mode 100644 packages/permissionless/accounts/trust/utils/getDummySignature.ts delete mode 100644 packages/permissionless/accounts/trust/utils/signMessage.ts delete mode 100644 packages/permissionless/accounts/trust/utils/signTransaction.ts delete mode 100644 packages/permissionless/accounts/trust/utils/signUserOperation.ts delete mode 100644 packages/permissionless/actions/bundler/chainId.test.ts delete mode 100644 packages/permissionless/actions/bundler/chainId.ts delete mode 100644 packages/permissionless/actions/bundler/estimateUserOperationGas.test.ts delete mode 100644 packages/permissionless/actions/bundler/estimateUserOperationGas.ts delete mode 100644 packages/permissionless/actions/bundler/getUserOperationByHash.test.ts delete mode 100644 packages/permissionless/actions/bundler/getUserOperationByHash.ts delete mode 100644 packages/permissionless/actions/bundler/getUserOperationReceipt.test.ts delete mode 100644 packages/permissionless/actions/bundler/getUserOperationReceipt.ts delete mode 100644 packages/permissionless/actions/bundler/sendUserOperation.test.ts delete mode 100644 packages/permissionless/actions/bundler/sendUserOperation.ts delete mode 100644 packages/permissionless/actions/bundler/supportedEntryPoints.test.ts delete mode 100644 packages/permissionless/actions/bundler/supportedEntryPoints.ts delete mode 100644 packages/permissionless/actions/bundler/waitForUserOperationReceipt.test.ts delete mode 100644 packages/permissionless/actions/bundler/waitForUserOperationReceipt.ts delete mode 100644 packages/permissionless/clients/createBundlerClient.test.ts delete mode 100644 packages/permissionless/clients/createBundlerClient.ts delete mode 100644 packages/permissionless/types/entrypoint.ts rename packages/permissionless/utils/{encode7579CallData.test.ts => encode7579Calls.test.ts} (87%) rename packages/permissionless/utils/{encode7579CallData.ts => encode7579Calls.ts} (76%) delete mode 100644 packages/permissionless/utils/getUserOperationHash.ts diff --git a/bun.lockb b/bun.lockb index bf430555cb36d7033be23ad6d1bd7f6e254b22c1..0ad1a05128932a977f673b53bd95eda7f1d56e98 100755 GIT binary patch delta 119148 zcmeEvcYIXU+VzkaQlx|?BB2CA2rz;4P(n)(1rb5<2wOy| z1%n2SNK+{yQbdAcMFks{1cSzo3Vv(tXHUq?$9v!Re&72i^Ydms>skBRy`3|o_oL^o z{rvfBABu{Yp51T3jJki;AM^B}Up{IdzV(?|HCk2geC|-YLG{!3zqw@Dzz@SMex@zG ztGT=JgrezxRLP5ou&nBql^Nrhem={{&Bm^jv-lbexjHHH5Q z_)pzvS(gKA8~z3IsR#bB%d%{sC(^Pm1NH@y|9+Fs$N4raBCki3_~LGsG!Bqy#%3iZ zC8PYyT8n=mkabuOq}~nbSf)Qaa>C^Fu~vK=Np}nUwcsD0n4Ob}x@RR%$@Y)WwN666 z68<%}NS#juYg$(0{C}B%asDiw)3&x!K*uO&qo>(r8{0{_?QV6BdD_?Zc%|6C+e@mS zorYDL=dF(xZ>m4dpOKiI{XIA1L#1OK~>0vUK?r9vj7{Cvk%t$to=H_O5h=6}%D zvaSVIzs<6m0j~wG1I(W2PfbnEva*v?rvj`h-7Tvw{8`D_>651>TMzZHtmfeJfvnAR zAjfWEA|_UvWxd%`YTFF{2JruY{I3GO(@QrmFaI?d?3XVfFtCTIP&O*i6g*I|M|w*J zx5CfCodV8+=k&F#M!;Wy?6(O(_S-;%{gLhl@Sbr}!5jNaKJ|cgtvoBg3JhlWBN7l# z0@?B-z}mo94L%2CMISf#fWgV>X%o0|B2mya(3_(HEr8ZwY0xW!F!jI#JsaqOvuH?* zJP3gnZH9wG@Q}ek4+L615`MPqvZ2y5JAhY%uL3p!&IJAo`2C$Sgl_=Z^3?|G#Y@F( zV0HLE1Lm=%xrtd5veAN*!=(#z+eD|aMMuEd1G|7&Kl#boNr{sa$0a9C_D}Icp9w#E z%;!9K)>I;LO)QN_t=n&<3xaAo;t2%&#Z`Q>_M!C*UCV8ZC3(pEfZ$%b%0| zt5amx&kI9#tnY>~lFe!0<=}sGk?b}nivG00hk>ll-D4%)WPdU$Wm$iMvl{cEbEc#L zIVH=G5A(|z=bw1-(K13P}Gkg9m|JDjm>=tj8|Hw*Xnc=bYo0dGn5=bh6?}(pfJ9 z*{TeGTDpbLHf=Y-ymt;V#Lj z7mz)1!(0ih17y|DxEpeDe&3nf2&h}6Zn2M*2L#L1KWSaR-#&Mh@g<~_YsI$}AHmDo}=k>w#}yFNfj;Q*hoB(ym%hNAm`#)GpW1A*-7?m$*FX=38!Nzj*?IXoZ8A((WX;F#+y zr*EC69YdGSyDv#TGmxJ7q#2wx(VsLiG9$I=R%ytLtauw-$XiQ8;HwFP8=7B zyaT+?PD%dAoq*mE9HSrqDewdXKDt!0k5l zmi!Vb#&PboSGJ_1Kn`P%*Cqe&XcuO{)S_tIWU1Kl!Z@3mTyZr3*Scnvt>ypVa<0W z!(%338L$cTK}Mefyc)bU@ERb7RNpk}fHwo*_pZcm1F{E}0sjU36zQ1Hxb&RlG_2Y{ zL+ZSb{%;K7ulJ>kyM7?J0LYr>WVLILSgYLVfdaR{$2qX*BbnV9K;|hekS zvQo=x1^niujLwMD^4Jo1Lowug8NDVzW>*H}G(81mKKgn#C2MkaavJJwS=l*>Ia9Ke zvwtX)^rwJKpPY<&nU(Dr%1o;@qaoX~eoOZ?F ztitG@Wyb0eiu7lvr6=L^PPS^HO4JjRl9DrWvaNz&q~&W2rutKHaT{wTrKhDOV{D)L zRmxidL|k6}l(Y;#UcOYcAy9V@GG-gD{#^ zzdi7JVB53O0QHY$wF7Sq&ip#NTzchv4s6EVuNVer`d%Pg(zJ?8&&*mtE}_5OE`1yS z5lB7sbPDXA%@E%l`XP@?KO8>;C@rA5&p)Pvt?F(O&}}W z;R=`TnFc^sXi{~TehBXZ%ws@x2<)0b!RHak3Vi|O$@L+SU4ADDXapRGE^P~Zp_a>P z1H1+N7GT{vF6(CC>Dn$my`Q<#rTw#jH^Dy?*aFxQ$R4r&LH{%G?emt^8o1Wrp!zO7 zhBpEkFrb0-KsNM9a9=}dX)kbQ+yuym{B%y%)j@-+4Zi=kth*`T5lA=hPg!@3n&wH@ z#6jS~yx!nB6Yv~{rXl==hVT1J7V5o**S%$!GorvYk}

4JIrgG@jxgC`=SOSc(ZYH*Ul9tN)lvP=JJDfl6fJ-FH6>KkOX zKWO+QgSP|O!&Pr{;ckHSf4yhFWSeZV3W`AXCW%v+Kd7lAl*}D zGQU4acF4Pc^snpV%F|nKPt0;=comQ}Po0En;T|;>f!wjW8gv8M)%PPEd#urEIkXDz zl#ByC6Nrx(Dm|E*m^CREJ7n}8xj`>6>F3;u_0KIg9S(L)GLRK{W0>Gez}n#KdOka( zXC$W~AFBom=GHpfROD=_G+-m%b8xIz0-0YeAcr6|IVUR}ae)Iq2}^_JD zDXH%Q@=nwR$TL_Wch+v|ggj(x*(Ac*}I$>BhnDPLP8n@UF4W z7&*BDPn=c3S?eE0x%5kxI#Mp4xkpHWp)>IsBw%x<;kk`(#ycg+1i<}#GAr`)SeeSF z5YJ`?o>;n$lXQX0Mc@+g9nv%1*lgy8moUlfW~{*Vf(rUJ496lilW*nfKR3(-XuzWDt0?Y?~qh zuL3!MwI@k|RWRVJIUn{W`_qySoRpdPjKTYX&7h|On*s*|8w0%tGqTcCc|F>lCJhNZ zLg$0$u|=a{aH^-I``I&A%M2Og20+e%aD%_3O9tl+c19d`s~wqgHh+!oU=KDn{@lb= zJygY6W`?It)sIMfz&W&MQ?Uk_!AGf5fu^W&Bk*cKPR5o~q-%e|WaPHhe5zDzKJY5= zFM+J!6yskoO@?e1kUbW?N7kGNIvZNUqzgQ1y?FxX)s+yoqX3Te1|Vm5;CVU|oO5G5 zkQt6J*!hI>*tN}CrG6rNMrXL0|8<{AdGEl_DWOplc7IXdG(##C~L~L1mkPq{{VxDY83xEy5e*`lB zb4ZVTQ}U-w@pG+OTkes8EC8~3+4oAukASnE=L0!FiSvaI0c;eJ{9nOR9VJuE9DZGui$ zeUXe_6(CDXP0#VCr&(672c$VIfUM0=K8| z1<0Ew#hj&S}}$ahh!q9(dnf z1L@gH*E5pOKkFo)!v)gHO+dE#K_I$6FFzdy2ev3VE1N8pKB)40dQVeHvw7kM_-bKd>UAr6X$UVT-j67 zWPp}{vw#|y@@)CQt#XzA0b3bo?3X|uhJhUh*GmipPU-D3Q@;hW!c&1I(*kbpo=s6IfptSylDlomi1TouS6^W)1^yx_CWZWDiUl0P{c z+sfiylF!{h_DC*}b1E=I$@r!uBgda~X1A+`?vhV-%OpDHWVULa7g*Y3@pBH|A5J2B zj{I&Q+ka@U6xZ-|sosM?R_kT>Ie-!Sr1;fH&lxoim(S5uo_D;mKSbJuKWpzWwmQy~`YwrY$0dpah*Z!k2bGL2f zq)hXj$S~ zdaT{zrhpHbyGA=Dxv{QSoroE+cFjoK2X*T5v0m2@rvSbs@Ueg@PSGf@>zETUGuHM* zS=O!aUGB6QW*BUxWFyRy4wEU)u)F2Mbf> z*{A{-8|7v$t~E}{oLIXYzP9k8f2Vmp*P-_9oz!VDuHjBWUaaR?@GdH{p{tV#)z8V#XlJLMT(I}1$W+6?pB+bc8Ak6_(Z zSq)dGaCB=Qodd_~L8svESo;8cj6^rgj1B|wIb|6!c4st^-HyVBdtEb~f_bs_Hu%_9 zo7M5u@-l7Ym>9c1ELKG-cAsHYRC#?@v{Q0Vtlby|pc3fZqFG*h02q2G|1zgM!D|;ZH#)*oQXw}_d(qc?T`!HBfXc$=xSaS#*K6H7!*OlZ%EQocz z?D!YN+J8e~R5%rT53(bZf&CMV4XUd1=mL|>ql)TcU}1Iv3v=4!d+n28L&5YswcB-; zkqFo0l4V$!gI=|Ff^n(2RhtKp3kHmNx|||RvL0ZjE~s;!(W>e3e8sS;PSKY-jD@4a z&<#D%KTtW;=!SV^qS=Cpc0hcwLV=C5vP2x8NHLA7)9m*WI;; zld{BTKi>o2nYpjCil-L$Gd6;fNT!VK2)XCXpk( zu1qK5p;*s%@C|iR7suE>G=Z(alo;;y%!k>T+l}jWCt_)=>p922G}coC^LUJtx-`a< z2CI*@w!`Y8tv_H5b0VQW-Pf}E(`Q@#0s~sU*z4&6cDqioyuvyOYa)I2z&L4(ZmxY> ze~ZDa&*|teke+OW=Y3e+^a7|qz>GP2*b@)dMK^N=EDoHWm7WjDnBG%&AhOripbBfT zw%9p`V6{e+hcnM}7HlLd<+{^}P_gbBgPj!RbH@#KHY%TI6$WLfZpP2B20E#kG49*$ za8e%ic?#}eHOd}{asAtR)dDc}}o#HVube-2b5s$~Z+YNV89_K78cs$lF z9d21&(Y(5P-@S1}MPso*Mu1`HFkiSQKLv&>V4@`TMHVTYo zVp%R^UTmtR-47&Febo_}9hSmyMA&%6Z!njB7$q z3OfyqixX=b^?JeauaC8VgO547ouXsWu0*GFz0b2JkyEB@eT?Vt3M+c79zFOL!-{pv za%1ebVBs7zJ3y5rC*?_>yK|DxV;a6A=*>LrJ+QEGLhTWzO9{9Ko>dnJm0LGS}|De4B#vTKk;dR#@@0336vwM%1B-oJ9RkIDlW`!<( z8LVQTv3~^PV7Q#jHD0^b1Q}4-Z|8w=TFBvi1Pn`0nrJtkC`_h|A511tId=Qa6^so# z2gVJmiqmE~HdTMcaAWhj7mONqdo<-WFxCYrahm+)6l{$3v`;~|IAwEU!qZ@}-pW;0 zr8O^E_LK0iBlNawe+0($3${Bte3FbMZe9sqch5;qtLJ=nKAg-SlXtAwJ^|KKr{`tk z+Q~9`aV8D(y0V>!!dUxf_^@qrA)r;arb^BA0pObAlpw_c_^_%`f9~;()1*-JUU{@r zuqoEH*eTf*YfniJ^Z@fI0%JY(Hfw)jG)%f#UiZxzPU-VL+n*s7m9ypvFb3HAzTrb`p#0^TP0X5hkr)Gk*;)d|m5PDDAl-a-U?1X~Uj)`28mfy9ISO{6nbqc1X0c#Qi8-H->l_$o zGD^tw+8=_k3uH>wzegDMOAM&rFf2wKR|Q}lbsY9K_gD8gtzPrl?e3LcvQ(Q5=p`_& zS6nc0k$45{c0Kc}%$Mq5@jU6ZW576edW7A%^PP=*eV#Yr?C2E!*onVh9PgCvjq&V& zg_W~6CcO52av#z?tUkZWWrx9o(`j#vy$KdKA{p)L7n)IH1Kf!VosIi^_A)qGb7|F4 zFz%)@IG+1uaCEr4*Zt1MH+-HvI0ra|Z^U?tVZ~{y@uGk=vBKI?VYwf$tiC#39IW2j zT3%s&R$;YV98fbWtd}dSDoX-tzY1#^tX?|5Pb;iz9}M_XDy$+{J#@O?Dy*1?0>1nT z>y3bAU$#^>M(MfnU>rL@&))rr^tgVO@wk_v$Mr;rhQ(RxQswEVHc+GN4Kg-tC4{M9S zx)8EvHloV9Q8B7WLjgXguh}~MXkgCf``kj!ehrh8kiQ` z{8xZ+j_OGg{-I%V0Du0i9lj#4zL+=;j57)i*cMIc7EQP;w-_c)WM6%aCZ>>}MzR!NGA_*U&Ype^@ zgBw!@*f6Oh_TPhGoCA7A+gGg*)Sr)_ z`4+>zgH}Zji@~^k>HIwJgZ0r5N6nuKq>qOs`wPzo1txXl-t!e$4=H1&*KYZA;C5Ml zB07vj&(il`NeDBzdPAVfZP1)qU_GUH?2voGc(*`h61<)(ogX7|4=mF?cr0H3 zjFa-I&;A8Y6N&-oSP<9~(Mz5zFx?LORai2a$}x&RfempAXT;cdZIq()D)DRq8|Xyl z$Jm}{Wdp-Z&BF@}Fw?#Vz4ilOvR>GbgJ5PPP=1r=q#v=uQO-CpJiFnR@+2(Q5|>F_ zo=U;GNdqx@(S@?A^gQ)U2kWFq=p|T^Gs^LVZ_*t(J;qLeWm=5xc@eC;evS7VtbWK< zmu+`@J}~Y)@gE^$O+4SglJg_#1v~~yEl}47z~n}C3KrLzw7p)DT-b2!!@)KcjPnF# z;PBc7))ov$*(R^MyvRv8<#V@v(b;&)XQ#a=t-^tbi{2(M$+J8$It)ZMzDAp6QeD_E zh8Yj8qv)FDhT$0+JHlsR%tUUwwO^{}25bp;ftfDF3|axk$t_Lz6if!6&kuFB1SUSZ zHM}1fx3r3<0M7&T@a7r*H7u^*3$?p(Yv9c1HZ%{6rO4CGPO#RxOl*$szqUFlB|dxf zHnab!BD_X&ZF5TD{0L5&KzM)Y+U}&3`rLPHcQ%&#>`mKEdY%yBrwps8oZV=LbeP`x z!iRzB3RXPAJOmG$Dwpm3Cbgx?r@ErUUWTUM3U-9Ws^Wbx7T;+w-YKz7VRpB8MLH7W zhAy87)=uZgS+oVLiz>SsyWcBL=@_3K^J>L1L9fpS;}%g(hrR;F-a)o_PWu&%RmDiY z5FNg=LK^1vECK7}6b_H^*t@u1%ihE81Zymk<5tc69*n_q>-_`Fr!(gE(0;d`|G3%Q z538HLy}SjBMeB3MuDU01cJK`D2i6@L&Ntl29|B{}9Kz@@kb!zBHF!;mfE$l{Ltk?? z{^YYCg|idVVq3!8e-CU37;1=olJ?@QjLwJa`xMw<%{UeAeBCMi*=MhQUFNF%Hbw9#_Kr`b{qZfu5URT|Mq!izJ;7+^1vFP%g-#qL)(!+ ziR|`x!%#Pj+S5mz(sMrhq;bk^q)D+n#bVkcU@92r3>eP$*TGl^oHo$@0^@cjEb^!n zgp>=S!$7*I!bfn4fyJpTd(0Wb&^RoMn70F$2R>IgU@|>1KG6<`gYy{k3Sqy3$+Rp-4R1XrhqSJtXDL{MN_`fUg4I8ex!w6)IhOIr zgcI;RFrE;|56|OG-jfZYs%~)t*ccrL&AsbARa(V`Iq|;CdW;PEC=rZ9EbrmgfDM+C zX5hW}2hv1L25jAZ4ZE;Au7HOppzK{=gK?Kb7oc8sKdk5;ls)J})v7AWE`T=%X)qtL zNj>us{vDPI<~lBK97G!mHo$1n;UIl068;D_sDgF<7>_g+YzbH(%Dd6*UcVtZ z@tJCMxy$GN*XJq)XutWnOf$SH$@IEweW6O}?f!+Vd2|Q1&4SjEcnJOp-! zO1<0_W4phR>4e8I>>phWlOdf07KIEwD)UifXf)Yi8-6R-ZA-PO1r4k_QexfV0l4s6 z)v6A9v^~e z#&ocbPTBi0o&s2jx<#ILrIwYfP4|jYReF`n=b3xjvPMaJTYfL)>Yd$_2!0eV{Lkp1MDvX?!;m-UAlXn>p~a9R`QolTW}h9gX|XH9tykT$n6_ z;Ne`tB*uO)&*(Coo59*c)0?Baz!0^`B%1 zz+}X8&3Lc@(Dbd{a}?}Or*LG9`?{Z1t0w5ote<81*43T)G8l8Jt#{I&!B`u)@kjg; z=rFt%cl&=)DNPZ-8D35xbTN*TAAV7#@VamMRkdmcHR;zt4%qEHFM#2I!pVLHmV7Z4 z^;=--a5atvli}ly)k5lS~R?E6}6GLw2CSvzE?%Hx(?VPOr-$r)G(Lc0I-U1 zZQTH7IvDH(7*`nPJO;M%xj6BV&Ve;jB47W~reGG11Uxx!^l)n#&1{s1f`P=Z<;bb1DRpCw8x2M*3S(8v5y!6cSdcFXYQA}xI zdfKxW)+|*Ph1uuFzZNo0McxeabC@%jxo3DI^sBax!I~;XH)aK??y9ibHE~&4+V?c9 z99_JpYg3nbU%CNSy5wE2Ss>+o71o~>*5u|cejSUKr+Z*!N!<&*o?EVVS*eo28kkv9 zW$!iYjxsdW^AyYsU4gp)3Rw5jl9a!~l(r{b>#`>6ghwl^nCk+*4Hedv*JFn0bhBYi z&}COSb-hcSzRK0v)4c`Ooh;R-VdC8Zx}(9(F6&ioy$5Txwz68IJGAu;tjD!=cN>@W zuu5%@QM|=vmicN}_i437+rY3t2kRdCTJ~$_vevOPEd4WL{FP5@_?y7rY~}H(pV#dE zv{P$X15X*8`8unoZbd=b^0vo(*47qUs;o5%_Xd1Lux6>q{_sV+tSQpZ>tV_;H|W5j zPmO_0OJ@_-k$syQiLTFqDNX#Y!on{UjPGq&GBdh$3Ro|}lEJ#JbHxxn0#k;_?F(2L z71qfLt6ywDeHoS>N_UGcss{x zNlPZ_yD-N{-}ega;7bFR{R1p+9(d7klgsNKe7h>`fpx8JSFL)wd@ZZ@c3Gn-Rs*gh(y7aH22uZyn= z3S;b4SUkkAjp3EwvtYPoWX0H@z~YI3s}v5`n*G$q+YuMn&t=7^!fvh@_kI0TtKQg- zUxQ2EE7>dVzmO1KcTAirCFaMeR(+7_qd2S;$qpBw>-z@|AD+A;g4j|pR!AP%js~&n z1LWVf60L_}@_aQL47WWzt8RnElUQ!SH3#DG(fRS}H3+Pyt_R;VC@`M+^2Bo#j9rIE z_b0s`&mb4y`C)aZ!y2z5TVPhel7n%`;DGuxEPW7rzK4Z(k9gJE@{Wo=dC%*fe1}RI zfX-VBZ;BHcALF@fNZ>F{gQatH7Y|V>1Cgrso!AKV24lYka}+A1@6C3{p)w-4A>%@m zM#dMNp7pT$=)>lFShqN3zsK0k?sDnZ_4-|cCk<@4WO)!~7iOu-72atLb6LIgaN5&h zveWb})IMsMOSPGaSro4}-hs>DfOzSyiqqL%08eW~;hcHM>)sZxQih-jjfcCe+f?ch zY!L}CJBzs;W)E%J=U{Ra!c?h9a|PM(VS4bWHduSe%j- zPps|+qf{x<*h5AIMiocA=P|H5B|mq9%eqU2~XmnM}E?YhI#OQ!0X8e zYpov^Uw}0j1>@R`)$AUJj&;i7@$MfMmtVLlN^r%vV(9c7g0q7YIXuSgPFAfF(2t#y zW$j@_p`A0qIJj<=SpyGoV4NU)!-8+O!PqA3h3|Pi-NpxI{v25BDLkAZhZn$jox^nx zm&~dYBnR}$Ua#E^OrC!6Xc;~Wj6GH{wW_>0!R4He@9c3;#AISE?H;fgf%mTXdMFQ! zCCLM05g2z3dGArpFD1#Vv=Ly#pvldAGnlD4o{@ewtfE}omm<2}W5VZvG2aS3{4hKm z{R^H(lcZi19(xEpjKZdh2fT;DIG6BqO(>SY6d=ORRZuizO6hULbp zy$f?BQsLDJp8synl2O31)*S0(5~xXc0;Zl>{N0A9dN$iuRu>tJhUrBHco~M@5H1J9 z%K9|M^D!)m_1v5jXpiS^nDW?UAA>m*fwfe5T^F9kr^u#{9>6+y5$q1#&#k9oE$J2U z7_8e7aGBF)Npu)UtP@$*DQp^c0Ijcp)dRX)6?riE!8oA$!8W|fblF{X }s;;Vmn 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 delta 117681 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

uf{b^>zEw-CIbBpL~s@#;eNifa$G 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 X!a#wJT`_kpyhfBAQ^AW|HNW_OR@9uN diff --git a/package.json b/package.json index ffc72a6f..dcaeb6cd 100644 --- a/package.json +++ b/package.json @@ -61,6 +61,6 @@ "get-port": "^7.0.0", "tsc-alias": "^1.8.8", "vitest": "^1.2.0", - "viem": "^2.16.2" + "viem": "^2.18.0" } } diff --git a/packages/permissionless/accounts/biconomy/abi/BiconomySmartAccountAbi.ts b/packages/permissionless/accounts/biconomy/abi/BiconomySmartAccountAbi.ts index db6ee1ac..ee697284 100644 --- a/packages/permissionless/accounts/biconomy/abi/BiconomySmartAccountAbi.ts +++ b/packages/permissionless/accounts/biconomy/abi/BiconomySmartAccountAbi.ts @@ -1,59 +1,36 @@ -/** - * The exeuctor abi, used to execute transactions on the Biconomy Modular Smart Account - */ -export const BiconomyExecuteAbi = [ +export const FactoryAbi = [ { inputs: [ { internalType: "address", - name: "dest", + name: "moduleSetupContract", type: "address" }, - { - internalType: "uint256", - name: "value", - type: "uint256" - }, { internalType: "bytes", - name: "func", + name: "moduleSetupData", type: "bytes" - } - ], - name: "execute_ncC", - outputs: [], - stateMutability: "nonpayable", - type: "function" - }, - { - inputs: [ - { - internalType: "address[]", - name: "dest", - type: "address[]" }, { - internalType: "uint256[]", - name: "value", - type: "uint256[]" - }, + internalType: "uint256", + name: "index", + type: "uint256" + } + ], + name: "deployCounterFactualAccount", + outputs: [ { - internalType: "bytes[]", - name: "func", - type: "bytes[]" + internalType: "address", + name: "proxy", + type: "address" } ], - name: "executeBatch_y6U", - outputs: [], stateMutability: "nonpayable", type: "function" } ] as const -/** - * The init abi, used to initialise Biconomy Modular Smart Account / setup default ECDSA module - */ -export const BiconomyInitAbi = [ +export const BiconomyAbi = [ { inputs: [ { @@ -101,5 +78,51 @@ export const BiconomyInitAbi = [ ], stateMutability: "nonpayable", type: "function" + }, + { + inputs: [ + { + internalType: "address", + name: "dest", + type: "address" + }, + { + internalType: "uint256", + name: "value", + type: "uint256" + }, + { + internalType: "bytes", + name: "func", + type: "bytes" + } + ], + name: "execute_ncC", + outputs: [], + stateMutability: "nonpayable", + type: "function" + }, + { + inputs: [ + { + internalType: "address[]", + name: "dest", + type: "address[]" + }, + { + internalType: "uint256[]", + name: "value", + type: "uint256[]" + }, + { + internalType: "bytes[]", + name: "func", + type: "bytes[]" + } + ], + name: "executeBatch_y6U", + outputs: [], + stateMutability: "nonpayable", + type: "function" } -] +] as const diff --git a/packages/permissionless/accounts/biconomy/privateKeyToBiconomySmartAccount.ts b/packages/permissionless/accounts/biconomy/privateKeyToBiconomySmartAccount.ts deleted file mode 100644 index 3aec0f37..00000000 --- a/packages/permissionless/accounts/biconomy/privateKeyToBiconomySmartAccount.ts +++ /dev/null @@ -1,46 +0,0 @@ -import type { Account, Chain, Client, Hex, Transport } from "viem" -import { privateKeyToAccount } from "viem/accounts" -import type { ENTRYPOINT_ADDRESS_V06_TYPE, Prettify } from "../../types" -import { - type BiconomySmartAccount, - type SignerToBiconomySmartAccountParameters, - signerToBiconomySmartAccount -} from "./signerToBiconomySmartAccount" - -export type PrivateKeyToBiconomySmartAccountParameters< - entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE -> = Prettify< - { - privateKey: Hex - } & Omit, "signer"> -> - -/** - * @description Creates a Biconomy Smart Account from a private key. - * - * @returns A Private Key Biconomy Smart Account using ECDSA as default validation module. - */ -export async function privateKeyToBiconomySmartAccount< - entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined ->( - client: Client, - { - privateKey, - ...rest - }: PrivateKeyToBiconomySmartAccountParameters -): Promise> { - const privateKeyAccount = privateKeyToAccount(privateKey) - return signerToBiconomySmartAccount< - entryPoint, - TTransport, - TChain, - TClientAccount, - "privateKey" - >(client, { - signer: privateKeyAccount, - ...rest - }) -} diff --git a/packages/permissionless/accounts/biconomy/signerToBiconomySmartAccount.ts b/packages/permissionless/accounts/biconomy/signerToBiconomySmartAccount.ts deleted file mode 100644 index 329d5e3b..00000000 --- a/packages/permissionless/accounts/biconomy/signerToBiconomySmartAccount.ts +++ /dev/null @@ -1,449 +0,0 @@ -import type { Account, TypedData } from "viem" -import { - type Address, - type Chain, - type Client, - type Hex, - type LocalAccount, - type Transport, - type TypedDataDefinition, - concatHex, - encodeAbiParameters, - encodeFunctionData, - encodePacked, - getContractAddress, - hexToBigInt, - keccak256, - parseAbiParameters -} from "viem" -import { getChainId, signMessage, signTypedData } from "viem/actions" -import { getAccountNonce } from "../../actions/public/getAccountNonce" -import type { Prettify } from "../../types" -import type { ENTRYPOINT_ADDRESS_V06_TYPE } from "../../types/entrypoint" -import { getEntryPointVersion } from "../../utils" -import { getUserOperationHash } from "../../utils/getUserOperationHash" -import { isSmartAccountDeployed } from "../../utils/isSmartAccountDeployed" -import { toSmartAccount } from "../toSmartAccount" -import { - SignTransactionNotSupportedBySmartAccount, - type SmartAccount, - type SmartAccountSigner -} from "../types" -import { - BiconomyExecuteAbi, - BiconomyInitAbi -} from "./abi/BiconomySmartAccountAbi" - -export type BiconomySmartAccount< - entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE, - transport extends Transport = Transport, - chain extends Chain | undefined = Chain | undefined -> = SmartAccount - -/** - * The account creation ABI for Biconomy Smart Account (from the biconomy SmartAccountFactory) - */ - -const createAccountAbi = [ - { - inputs: [ - { - internalType: "address", - name: "moduleSetupContract", - type: "address" - }, - { - internalType: "bytes", - name: "moduleSetupData", - type: "bytes" - }, - { - internalType: "uint256", - name: "index", - type: "uint256" - } - ], - name: "deployCounterFactualAccount", - outputs: [ - { - internalType: "address", - name: "proxy", - type: "address" - } - ], - stateMutability: "nonpayable", - type: "function" - } -] as const - -/** - * Default addresses for Biconomy Smart Account - */ -const BICONOMY_ADDRESSES: { - ECDSA_OWNERSHIP_REGISTRY_MODULE: Address - ACCOUNT_V2_0_LOGIC: Address - FACTORY_ADDRESS: Address - DEFAULT_FALLBACK_HANDLER_ADDRESS: Address -} = { - ECDSA_OWNERSHIP_REGISTRY_MODULE: - "0x0000001c5b32F37F5beA87BDD5374eB2aC54eA8e", - ACCOUNT_V2_0_LOGIC: "0x0000002512019Dafb59528B82CB92D3c5D2423aC", - FACTORY_ADDRESS: "0x000000a56Aaca3e9a4C479ea6b6CD0DbcB6634F5", - DEFAULT_FALLBACK_HANDLER_ADDRESS: - "0x0bBa6d96BD616BedC6BFaa341742FD43c60b83C1" -} - -const BICONOMY_PROXY_CREATION_CODE = - "0x6080346100aa57601f61012038819003918201601f19168301916001600160401b038311848410176100af578084926020946040528339810103126100aa57516001600160a01b0381168082036100aa5715610065573055604051605a90816100c68239f35b60405162461bcd60e51b815260206004820152601e60248201527f496e76616c696420696d706c656d656e746174696f6e206164647265737300006044820152606490fd5b600080fd5b634e487b7160e01b600052604160045260246000fdfe608060405230546000808092368280378136915af43d82803e156020573d90f35b3d90fdfea2646970667358221220a03b18dce0be0b4c9afe58a9eb85c35205e2cf087da098bbf1d23945bf89496064736f6c63430008110033" - -/** - * Get the account initialization code for Biconomy smart account with ECDSA as default authorization module - * @param owner - * @param index - * @param factoryAddress - * @param ecdsaValidatorAddress - */ -const getAccountInitCode = async ({ - owner, - index, - ecdsaModuleAddress -}: { - owner: Address - index: bigint - ecdsaModuleAddress: Address -}): Promise => { - if (!owner) throw new Error("Owner account not found") - - // Build the module setup data - const ecdsaOwnershipInitData = encodeFunctionData({ - abi: BiconomyInitAbi, - functionName: "initForSmartAccount", - args: [owner] - }) - - // Build the account init code - return encodeFunctionData({ - abi: createAccountAbi, - functionName: "deployCounterFactualAccount", - args: [ecdsaModuleAddress, ecdsaOwnershipInitData, index] - }) -} - -const getAccountAddress = async ({ - factoryAddress, - accountLogicAddress, - fallbackHandlerAddress, - ecdsaModuleAddress, - owner, - index = BigInt(0) -}: { - factoryAddress: Address - accountLogicAddress: Address - fallbackHandlerAddress: Address - ecdsaModuleAddress: Address - owner: Address - index?: bigint -}): Promise

=> { - // Build the module setup data - const ecdsaOwnershipInitData = encodeFunctionData({ - abi: BiconomyInitAbi, - functionName: "initForSmartAccount", - args: [owner] - }) - - // Build account init code - const initialisationData = encodeFunctionData({ - abi: BiconomyInitAbi, - functionName: "init", - args: [ - fallbackHandlerAddress, - ecdsaModuleAddress, - ecdsaOwnershipInitData - ] - }) - - const deploymentCode = encodePacked( - ["bytes", "uint256"], - [BICONOMY_PROXY_CREATION_CODE, hexToBigInt(accountLogicAddress)] - ) - - const salt = keccak256( - encodePacked( - ["bytes32", "uint256"], - [keccak256(encodePacked(["bytes"], [initialisationData])), index] - ) - ) - - return getContractAddress({ - from: factoryAddress, - salt, - bytecode: deploymentCode, - opcode: "CREATE2" - }) -} - -export type SignerToBiconomySmartAccountParameters< - entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE, - TSource extends string = string, - TAddress extends Address = Address -> = Prettify<{ - nonceKey?: bigint - signer: SmartAccountSigner - entryPoint: entryPoint - address?: Address - index?: bigint - factoryAddress?: Address - accountLogicAddress?: Address - fallbackHandlerAddress?: Address - ecdsaModuleAddress?: Address -}> - -/** - * Build a Biconomy modular smart account from a private key, that use the ECDSA signer behind the scene - * @param client - * @param privateKey - * @param entryPoint - * @param index - * @param factoryAddress - * @param accountLogicAddress - * @param ecdsaModuleAddress - */ -export async function signerToBiconomySmartAccount< - entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined, - TSource extends string = string, - TAddress extends Address = Address ->( - client: Client, - { - signer, - address, - entryPoint: entryPointAddress, - index = BigInt(0), - factoryAddress = BICONOMY_ADDRESSES.FACTORY_ADDRESS, - accountLogicAddress = BICONOMY_ADDRESSES.ACCOUNT_V2_0_LOGIC, - fallbackHandlerAddress = BICONOMY_ADDRESSES.DEFAULT_FALLBACK_HANDLER_ADDRESS, - ecdsaModuleAddress = BICONOMY_ADDRESSES.ECDSA_OWNERSHIP_REGISTRY_MODULE, - nonceKey - }: SignerToBiconomySmartAccountParameters -): Promise> { - const entryPointVersion = getEntryPointVersion(entryPointAddress) - - if (entryPointVersion !== "v0.6") { - throw new Error("Only EntryPoint 0.6 is supported") - } - - // Get the private key related account - const viemSigner: LocalAccount = { - ...signer, - signTransaction: (_, __) => { - throw new SignTransactionNotSupportedBySmartAccount() - } - } as LocalAccount - - // Helper to generate the init code for the smart account - const generateInitCode = () => - getAccountInitCode({ - owner: viemSigner.address, - index, - ecdsaModuleAddress - }) - - // Fetch account address and chain id - const [accountAddress, chainId] = await Promise.all([ - address ?? - getAccountAddress({ - owner: viemSigner.address, - ecdsaModuleAddress, - factoryAddress, - accountLogicAddress, - fallbackHandlerAddress, - index - }), - client.chain?.id ?? getChainId(client) - ]) - - if (!accountAddress) throw new Error("Account address not found") - - let smartAccountDeployed = await isSmartAccountDeployed( - client, - accountAddress - ) - - return toSmartAccount({ - address: accountAddress, - async signMessage({ message }) { - let signature: Hex = await signMessage(client, { - account: viemSigner, - message - }) - const potentiallyIncorrectV = Number.parseInt( - signature.slice(-2), - 16 - ) - if (![27, 28].includes(potentiallyIncorrectV)) { - const correctV = potentiallyIncorrectV + 27 - signature = (signature.slice(0, -2) + - correctV.toString(16)) as Hex - } - return encodeAbiParameters( - [{ type: "bytes" }, { type: "address" }], - [signature, ecdsaModuleAddress] - ) - }, - async signTransaction(_, __) { - throw new SignTransactionNotSupportedBySmartAccount() - }, - async signTypedData< - const TTypedData extends TypedData | Record, - TPrimaryType extends - | keyof TTypedData - | "EIP712Domain" = keyof TTypedData - >(typedData: TypedDataDefinition) { - let signature: Hex = await signTypedData< - TTypedData, - TPrimaryType, - TChain, - TClientAccount - >(client, { - account: viemSigner, - ...typedData - }) - const potentiallyIncorrectV = Number.parseInt( - signature.slice(-2), - 16 - ) - if (![27, 28].includes(potentiallyIncorrectV)) { - const correctV = potentiallyIncorrectV + 27 - signature = (signature.slice(0, -2) + - correctV.toString(16)) as Hex - } - return encodeAbiParameters( - [{ type: "bytes" }, { type: "address" }], - [signature, ecdsaModuleAddress] - ) - }, - client: client, - publicKey: accountAddress, - entryPoint: entryPointAddress, - source: "biconomySmartAccount", - - // Get the nonce of the smart account - async getNonce(key?: bigint) { - return getAccountNonce(client, { - sender: accountAddress, - entryPoint: entryPointAddress, - key: key ?? nonceKey - }) - }, - - // Sign a user operation - async signUserOperation(userOperation) { - const hash = getUserOperationHash({ - userOperation: { - ...userOperation, - signature: "0x" - }, - entryPoint: entryPointAddress, - chainId: chainId - }) - const signature = await signMessage(client, { - account: viemSigner, - message: { raw: hash } - }) - // userOp signature is encoded module signature + module address - const signatureWithModuleAddress = encodeAbiParameters( - parseAbiParameters("bytes, address"), - [signature, ecdsaModuleAddress] - ) - return signatureWithModuleAddress - }, - - async getFactory() { - if (smartAccountDeployed) return undefined - - smartAccountDeployed = await isSmartAccountDeployed( - client, - accountAddress - ) - - if (smartAccountDeployed) return undefined - - return factoryAddress - }, - - async getFactoryData() { - if (smartAccountDeployed) return undefined - - smartAccountDeployed = await isSmartAccountDeployed( - client, - accountAddress - ) - - if (smartAccountDeployed) return undefined - return generateInitCode() - }, - - // Encode the init code - async getInitCode() { - if (smartAccountDeployed) return "0x" - - smartAccountDeployed = await isSmartAccountDeployed( - client, - accountAddress - ) - - if (smartAccountDeployed) return "0x" - - return concatHex([factoryAddress, await generateInitCode()]) - }, - - // Encode the deploy call data - async encodeDeployCallData(_) { - throw new Error("Doesn't support account deployment") - }, - - // Encode a call - async encodeCallData(args) { - if (Array.isArray(args)) { - // Encode a batched call - const argsArray = args as { - to: Address - value: bigint - data: Hex - }[] - - return encodeFunctionData({ - abi: BiconomyExecuteAbi, - functionName: "executeBatch_y6U", - args: [ - argsArray.map((a) => a.to), - argsArray.map((a) => a.value), - argsArray.map((a) => a.data) - ] - }) - } - const { to, value, data } = args as { - to: Address - value: bigint - data: Hex - } - // Encode a simple call - return encodeFunctionData({ - abi: BiconomyExecuteAbi, - functionName: "execute_ncC", - args: [to, value, data] - }) - }, - - // Get simple dummy signature for ECDSA module authorization - async getDummySignature(_userOperation) { - const moduleAddress = - BICONOMY_ADDRESSES.ECDSA_OWNERSHIP_REGISTRY_MODULE - const dynamicPart = moduleAddress.substring(2).padEnd(40, "0") - return `0x0000000000000000000000000000000000000000000000000000000000000040000000000000000000000000${dynamicPart}000000000000000000000000000000000000000000000000000000000000004181d4b4981670cb18f99f0b4a66446df1bf5b204d24cfcb659bf38ba27a4359b5711649ec2423c5e1247245eba2964679b6a1dbb85c992ae40b9b00c6935b02ff1b00000000000000000000000000000000000000000000000000000000000000` - } - }) -} diff --git a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts new file mode 100644 index 00000000..5d64af92 --- /dev/null +++ b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts @@ -0,0 +1,363 @@ +import type { Assign, Prettify } from "viem" +import { + type Address, + type Client, + type Hex, + type LocalAccount, + encodeAbiParameters, + encodeFunctionData, + encodePacked, + getContractAddress, + hexToBigInt, + keccak256, + parseAbiParameters +} from "viem" +import { + type SmartAccount, + type SmartAccountImplementation, + entryPoint06Abi, + entryPoint06Address, + getUserOperationHash, + toSmartAccount +} from "viem/account-abstraction" +import { signMessage } from "viem/actions" +import { getAccountNonce } from "../../actions/public/getAccountNonce" +import { isSmartAccountDeployed } from "../../utils/isSmartAccountDeployed" +import { BiconomyAbi, FactoryAbi } from "./abi/BiconomySmartAccountAbi" + +/** + * The account creation ABI for Biconomy Smart Account (from the biconomy SmartAccountFactory) + */ + +/** + * Default addresses for Biconomy Smart Account + */ +const BICONOMY_ADDRESSES: { + ECDSA_OWNERSHIP_REGISTRY_MODULE: Address + ACCOUNT_V2_0_LOGIC: Address + FACTORY_ADDRESS: Address + DEFAULT_FALLBACK_HANDLER_ADDRESS: Address +} = { + ECDSA_OWNERSHIP_REGISTRY_MODULE: + "0x0000001c5b32F37F5beA87BDD5374eB2aC54eA8e", + ACCOUNT_V2_0_LOGIC: "0x0000002512019Dafb59528B82CB92D3c5D2423aC", + FACTORY_ADDRESS: "0x000000a56Aaca3e9a4C479ea6b6CD0DbcB6634F5", + DEFAULT_FALLBACK_HANDLER_ADDRESS: + "0x0bBa6d96BD616BedC6BFaa341742FD43c60b83C1" +} + +const BICONOMY_PROXY_CREATION_CODE = + "0x6080346100aa57601f61012038819003918201601f19168301916001600160401b038311848410176100af578084926020946040528339810103126100aa57516001600160a01b0381168082036100aa5715610065573055604051605a90816100c68239f35b60405162461bcd60e51b815260206004820152601e60248201527f496e76616c696420696d706c656d656e746174696f6e206164647265737300006044820152606490fd5b600080fd5b634e487b7160e01b600052604160045260246000fdfe608060405230546000808092368280378136915af43d82803e156020573d90f35b3d90fdfea2646970667358221220a03b18dce0be0b4c9afe58a9eb85c35205e2cf087da098bbf1d23945bf89496064736f6c63430008110033" + +/** + * Get the account initialization code for Biconomy smart account with ECDSA as default authorization module + * @param owner + * @param index + * @param factoryAddress + * @param ecdsaValidatorAddress + */ +const getAccountInitCode = async ({ + owner, + index, + ecdsaModuleAddress +}: { + owner: Address + index: bigint + ecdsaModuleAddress: Address +}): Promise => { + if (!owner) throw new Error("Owner account not found") + + // Build the module setup data + const ecdsaOwnershipInitData = encodeFunctionData({ + abi: BiconomyAbi, + functionName: "initForSmartAccount", + args: [owner] + }) + + // Build the account init code + return encodeFunctionData({ + abi: FactoryAbi, + functionName: "deployCounterFactualAccount", + args: [ecdsaModuleAddress, ecdsaOwnershipInitData, index] + }) +} + +const getAccountAddress = async ({ + factoryAddress, + accountLogicAddress, + fallbackHandlerAddress, + ecdsaModuleAddress, + owner, + index = BigInt(0) +}: { + factoryAddress: Address + accountLogicAddress: Address + fallbackHandlerAddress: Address + ecdsaModuleAddress: Address + owner: Address + index?: bigint +}): Promise
=> { + // Build the module setup data + const ecdsaOwnershipInitData = encodeFunctionData({ + abi: BiconomyAbi, + functionName: "initForSmartAccount", + args: [owner] + }) + + // Build account init code + const initialisationData = encodeFunctionData({ + abi: BiconomyAbi, + functionName: "init", + args: [ + fallbackHandlerAddress, + ecdsaModuleAddress, + ecdsaOwnershipInitData + ] + }) + + const deploymentCode = encodePacked( + ["bytes", "uint256"], + [BICONOMY_PROXY_CREATION_CODE, hexToBigInt(accountLogicAddress)] + ) + + const salt = keccak256( + encodePacked( + ["bytes32", "uint256"], + [keccak256(encodePacked(["bytes"], [initialisationData])), index] + ) + ) + + return getContractAddress({ + from: factoryAddress, + salt, + bytecode: deploymentCode, + opcode: "CREATE2" + }) +} + +export type ToBiconomySmartAccountParameters< + entryPointAddress extends + typeof entryPoint06Address = typeof entryPoint06Address, + entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi +> = Prettify<{ + client: Client + owner: LocalAccount + address?: Address | undefined + entryPoint?: { + address: entryPointAddress + version: "0.6" + abi?: entryPointAbi + } + nonceKey?: bigint + index?: bigint + factoryAddress?: Address + accountLogicAddress?: Address + fallbackHandlerAddress?: Address + ecdsaModuleAddress?: Address +}> + +export type BiconomySmartAccountImplementation< + entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi +> = Assign< + SmartAccountImplementation< + entryPointAbi, + "0.6", + { + abi: typeof BiconomyAbi + factory: { abi: typeof FactoryAbi; address: Address } + } + >, + { sign: NonNullable } +> + +export type ToBiconomySmartAccountReturnType< + entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi +> = Prettify>> + +/** + * Build a Biconomy modular smart account from a private key, that use the ECDSA signer behind the scene + * @param client + * @param privateKey + * @param entryPoint + * @param index + * @param factoryAddress + * @param accountLogicAddress + * @param ecdsaModuleAddress + */ + +export async function toBiconomySmartAccount< + entryPointAddress extends + typeof entryPoint06Address = typeof entryPoint06Address, + entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi +>( + parameters: ToBiconomySmartAccountParameters< + entryPointAddress, + entryPointAbi + > +): Promise> { + // ): Promise> { + + const { owner, client, index = 0n, address } = parameters + + const entryPoint = { + address: parameters.entryPoint?.address ?? entryPoint06Address, + abi: parameters.entryPoint?.abi ?? entryPoint06Abi, + version: "0.6" + } as const + + const factoryAddress = + parameters.factoryAddress ?? BICONOMY_ADDRESSES.FACTORY_ADDRESS + + let accountAddress: Address + + const ecdsaModuleAddress = + parameters.ecdsaModuleAddress ?? + BICONOMY_ADDRESSES.ECDSA_OWNERSHIP_REGISTRY_MODULE + + const accountLogicAddress = + parameters.accountLogicAddress ?? BICONOMY_ADDRESSES.ACCOUNT_V2_0_LOGIC + + const fallbackHandlerAddress = + parameters.fallbackHandlerAddress ?? + BICONOMY_ADDRESSES.DEFAULT_FALLBACK_HANDLER_ADDRESS + + const getAddress = async () => { + if (accountAddress) return accountAddress + + accountAddress = + address ?? + (await getAccountAddress({ + owner: owner.address, + ecdsaModuleAddress, + factoryAddress, + accountLogicAddress, + fallbackHandlerAddress, + index + })) + + return accountAddress + } + + return toSmartAccount({ + client, + entryPoint, + extend: { + abi: BiconomyAbi, + factory: { + abi: FactoryAbi, + address: factoryAddress + } + }, + getAddress, + async getNonce(args) { + const address = await getAddress() + return getAccountNonce(client, { + address, + entryPointAddress: entryPoint.address, + key: args?.key ?? parameters?.nonceKey + }) + }, + encodeCalls: async (calls) => { + if (calls.length > 1) { + // Encode a batched call + return encodeFunctionData({ + abi: BiconomyAbi, + functionName: "executeBatch_y6U", + args: [ + calls.map((a) => a.to), + calls.map((a) => a.value ?? 0n), + calls.map((a) => a.data ?? "0x") + ] + }) + } + const { to, value, data } = calls[0] + // Encode a simple call + return encodeFunctionData({ + abi: BiconomyAbi, + functionName: "execute_ncC", + args: [to, value ?? 0n, data ?? "0x"] + }) + }, + getFactoryArgs: async () => { + return { + factory: factoryAddress, + factoryData: await getAccountInitCode({ + owner: owner.address, + index, + ecdsaModuleAddress + }) + } + }, + // Get simple dummy signature for ECDSA module authorization + async getStubSignature() { + const dynamicPart = ecdsaModuleAddress.substring(2).padEnd(40, "0") + return `0x0000000000000000000000000000000000000000000000000000000000000040000000000000000000000000${dynamicPart}000000000000000000000000000000000000000000000000000000000000004181d4b4981670cb18f99f0b4a66446df1bf5b204d24cfcb659bf38ba27a4359b5711649ec2423c5e1247245eba2964679b6a1dbb85c992ae40b9b00c6935b02ff1b00000000000000000000000000000000000000000000000000000000000000` + }, + async sign({ hash }) { + return this.signMessage({ message: hash }) + }, + async signMessage({ message }) { + let signature = await owner.signMessage({ + message + }) + + const potentiallyIncorrectV = Number.parseInt( + signature.slice(-2), + 16 + ) + if (![27, 28].includes(potentiallyIncorrectV)) { + const correctV = potentiallyIncorrectV + 27 + signature = (signature.slice(0, -2) + + correctV.toString(16)) as Hex + } + return encodeAbiParameters( + [{ type: "bytes" }, { type: "address" }], + [signature, ecdsaModuleAddress] + ) + }, + async signTypedData(typedData) { + let signature = await owner.signTypedData(typedData) + + const potentiallyIncorrectV = Number.parseInt( + signature.slice(-2), + 16 + ) + if (![27, 28].includes(potentiallyIncorrectV)) { + const correctV = potentiallyIncorrectV + 27 + signature = (signature.slice(0, -2) + + correctV.toString(16)) as Hex + } + return encodeAbiParameters( + [{ type: "bytes" }, { type: "address" }], + [signature, ecdsaModuleAddress] + ) + }, + // Sign a user operation + async signUserOperation(parameters) { + const { chainId = client.chain?.id, ...userOperation } = parameters + + if (!chainId) throw new Error("Chain id not found") + + const hash = getUserOperationHash({ + userOperation: { + ...userOperation, + sender: userOperation.sender ?? (await getAddress()), + signature: "0x" + }, + entryPointAddress: entryPoint.address, + entryPointVersion: entryPoint.version, + chainId: chainId + }) + const signature = await signMessage(client, { + account: owner, + message: { raw: hash } + }) + // userOp signature is encoded module signature + module address + const signatureWithModuleAddress = encodeAbiParameters( + parseAbiParameters("bytes, address"), + [signature, ecdsaModuleAddress] + ) + return signatureWithModuleAddress + } + }) as Promise> +} diff --git a/packages/permissionless/accounts/index.ts b/packages/permissionless/accounts/index.ts index d28cb06b..982bde04 100644 --- a/packages/permissionless/accounts/index.ts +++ b/packages/permissionless/accounts/index.ts @@ -1,105 +1,77 @@ import { - type PrivateKeyToSimpleSmartAccountParameters, - privateKeyToSimpleSmartAccount -} from "./simple/privateKeyToSimpleSmartAccount" + type SimpleSmartAccountImplementation, + type ToSimpleSmartAccountParameters, + type ToSimpleSmartAccountReturnType, + toSimpleSmartAccount +} from "./simple/toSimpleSmartAccount" import { - type SignerToSimpleSmartAccountParameters, - type SimpleSmartAccount, - signerToSimpleSmartAccount -} from "./simple/signerToSimpleSmartAccount" + type LightAccountVersion, + type LightSmartAccountImplementation, + type ToLightSmartAccountParameters, + type ToLightSmartAccountReturnType, + toLightSmartAccount +} from "./light/toLightSmartAccount" import { - type PrivateKeyToLightSmartAccountParameters, - privateKeyToLightSmartAccount -} from "./light/privateKeyToLightSmartAccount" + type ToTrustSmartAccountParameters, + type ToTrustSmartAccountReturnType, + type TrustSmartAccountImplementation, + toTrustSmartAccount +} from "./trust/toTrustSmartAccount" import { - type LightSmartAccount, - type SignerToLightSmartAccountParameters, - signerToLightSmartAccount -} from "./light/signerToLightSmartAccount" - -import { - type SignerToTrustSmartAccountParameters, - type TrustSmartAccount, - signerToTrustSmartAccount -} from "./trust/signerToTrustSmartAccount" - -import { - type PrivateKeyToTrustSmartAccountParameters, - privateKeyToTrustSmartAccount -} from "./trust/privateKeyToTrustSmartAccount" - -import { - type PrivateKeyToSafeSmartAccountParameters, - privateKeyToSafeSmartAccount -} from "./safe/privateKeyToSafeSmartAccount" - -import { - type SafeSmartAccount, + type SafeSmartAccountImplementation, type SafeVersion, - type SignerToSafeSmartAccountParameters, - signerToSafeSmartAccount -} from "./safe/signerToSafeSmartAccount" - -import { - type KernelEcdsaSmartAccount, - type SignerToEcdsaKernelSmartAccountParameters, - signerToEcdsaKernelSmartAccount -} from "./kernel/signerToEcdsaKernelSmartAccount" - -import { - type BiconomySmartAccount, - type SignerToBiconomySmartAccountParameters, - signerToBiconomySmartAccount -} from "./biconomy/signerToBiconomySmartAccount" + type ToSafeSmartAccountParameters, + type ToSafeSmartAccountReturnType, + toSafeSmartAccount +} from "./safe/toSafeSmartAccount" import { - type PrivateKeyToBiconomySmartAccountParameters, - privateKeyToBiconomySmartAccount -} from "./biconomy/privateKeyToBiconomySmartAccount" + type EcdsaKernelSmartAccountImplementation, + type KernelVersion, + type ToEcdsaKernelSmartAccountParameters, + type ToEcdsaKernelSmartAccountReturnType, + toEcdsaKernelSmartAccount +} from "./kernel/toEcdsaKernelSmartAccount" import { - SignTransactionNotSupportedBySmartAccount, - type SmartAccount, - type SmartAccountSigner -} from "./types" + type BiconomySmartAccountImplementation, + type ToBiconomySmartAccountParameters, + type ToBiconomySmartAccountReturnType, + toBiconomySmartAccount +} from "./biconomy/toBiconomySmartAccount" -import { toSmartAccount } from "./toSmartAccount" +import { SignTransactionNotSupportedBySmartAccount } from "./types" export { + type ToSimpleSmartAccountParameters, + type SimpleSmartAccountImplementation, + type ToSimpleSmartAccountReturnType, + toSimpleSmartAccount, + type LightAccountVersion, + type ToLightSmartAccountParameters, + type LightSmartAccountImplementation, + type ToLightSmartAccountReturnType, + toLightSmartAccount, + type ToTrustSmartAccountParameters, + type TrustSmartAccountImplementation, + type ToTrustSmartAccountReturnType, + toTrustSmartAccount, + type ToSafeSmartAccountParameters, + type SafeSmartAccountImplementation, + type ToSafeSmartAccountReturnType, + toSafeSmartAccount, + type ToEcdsaKernelSmartAccountParameters, + type EcdsaKernelSmartAccountImplementation, + type ToEcdsaKernelSmartAccountReturnType, type SafeVersion, - type SmartAccountSigner, - type SafeSmartAccount, - signerToSafeSmartAccount, - type SimpleSmartAccount, - signerToSimpleSmartAccount, - type LightSmartAccount, - signerToLightSmartAccount, - type TrustSmartAccount, - signerToTrustSmartAccount, - privateKeyToTrustSmartAccount, - SignTransactionNotSupportedBySmartAccount, - privateKeyToBiconomySmartAccount, - privateKeyToSimpleSmartAccount, - privateKeyToLightSmartAccount, - type SmartAccount, - privateKeyToSafeSmartAccount, - type KernelEcdsaSmartAccount, - signerToEcdsaKernelSmartAccount, - type BiconomySmartAccount, - signerToBiconomySmartAccount, - toSmartAccount, - type SignerToSimpleSmartAccountParameters, - type SignerToLightSmartAccountParameters, - type SignerToSafeSmartAccountParameters, - type PrivateKeyToSimpleSmartAccountParameters, - type PrivateKeyToLightSmartAccountParameters, - type PrivateKeyToSafeSmartAccountParameters, - type SignerToEcdsaKernelSmartAccountParameters, - type SignerToBiconomySmartAccountParameters, - type PrivateKeyToBiconomySmartAccountParameters, - type SignerToTrustSmartAccountParameters, - type PrivateKeyToTrustSmartAccountParameters + type KernelVersion, + toEcdsaKernelSmartAccount, + type ToBiconomySmartAccountReturnType, + type ToBiconomySmartAccountParameters, + type BiconomySmartAccountImplementation, + toBiconomySmartAccount, + SignTransactionNotSupportedBySmartAccount } diff --git a/packages/permissionless/accounts/kernel/abi/KernelV3AccountAbi.ts b/packages/permissionless/accounts/kernel/abi/KernelV3AccountAbi.ts index 31820fbb..5571635c 100644 --- a/packages/permissionless/accounts/kernel/abi/KernelV3AccountAbi.ts +++ b/packages/permissionless/accounts/kernel/abi/KernelV3AccountAbi.ts @@ -106,3 +106,708 @@ export const KernelV3ExecuteAbi = [ stateMutability: "payable" } ] as const + +export const KernelV3AccountAbi = [ + { + type: "constructor", + inputs: [ + { + name: "_entrypoint", + type: "address", + internalType: "contract IEntryPoint" + } + ], + stateMutability: "nonpayable" + }, + { type: "fallback", stateMutability: "payable" }, + { type: "receive", stateMutability: "payable" }, + { + type: "function", + name: "accountId", + inputs: [], + outputs: [ + { + name: "accountImplementationId", + type: "string", + internalType: "string" + } + ], + stateMutability: "pure" + }, + { + type: "function", + name: "currentNonce", + inputs: [], + outputs: [{ name: "", type: "uint32", internalType: "uint32" }], + stateMutability: "view" + }, + { + type: "function", + name: "eip712Domain", + inputs: [], + outputs: [ + { name: "fields", type: "bytes1", internalType: "bytes1" }, + { name: "name", type: "string", internalType: "string" }, + { name: "version", type: "string", internalType: "string" }, + { name: "chainId", type: "uint256", internalType: "uint256" }, + { + name: "verifyingContract", + type: "address", + internalType: "address" + }, + { name: "salt", type: "bytes32", internalType: "bytes32" }, + { name: "extensions", type: "uint256[]", internalType: "uint256[]" } + ], + stateMutability: "view" + }, + { + type: "function", + name: "entrypoint", + inputs: [], + outputs: [ + { name: "", type: "address", internalType: "contract IEntryPoint" } + ], + stateMutability: "view" + }, + { + type: "function", + name: "execute", + inputs: [ + { name: "execMode", type: "bytes32", internalType: "ExecMode" }, + { name: "executionCalldata", type: "bytes", internalType: "bytes" } + ], + outputs: [], + stateMutability: "payable" + }, + { + type: "function", + name: "executeFromExecutor", + inputs: [ + { name: "execMode", type: "bytes32", internalType: "ExecMode" }, + { name: "executionCalldata", type: "bytes", internalType: "bytes" } + ], + outputs: [ + { name: "returnData", type: "bytes[]", internalType: "bytes[]" } + ], + stateMutability: "payable" + }, + { + type: "function", + name: "executeUserOp", + inputs: [ + { + name: "userOp", + type: "tuple", + internalType: "struct PackedUserOperation", + components: [ + { + name: "sender", + type: "address", + internalType: "address" + }, + { name: "nonce", type: "uint256", internalType: "uint256" }, + { name: "initCode", type: "bytes", internalType: "bytes" }, + { name: "callData", type: "bytes", internalType: "bytes" }, + { + name: "accountGasLimits", + type: "bytes32", + internalType: "bytes32" + }, + { + name: "preVerificationGas", + type: "uint256", + internalType: "uint256" + }, + { + name: "gasFees", + type: "bytes32", + internalType: "bytes32" + }, + { + name: "paymasterAndData", + type: "bytes", + internalType: "bytes" + }, + { name: "signature", type: "bytes", internalType: "bytes" } + ] + }, + { name: "userOpHash", type: "bytes32", internalType: "bytes32" } + ], + outputs: [], + stateMutability: "payable" + }, + { + type: "function", + name: "executorConfig", + inputs: [ + { + name: "executor", + type: "address", + internalType: "contract IExecutor" + } + ], + outputs: [ + { + name: "", + type: "tuple", + internalType: "struct ExecutorManager.ExecutorConfig", + components: [ + { + name: "hook", + type: "address", + internalType: "contract IHook" + } + ] + } + ], + stateMutability: "view" + }, + { + type: "function", + name: "initialize", + inputs: [ + { + name: "_rootValidator", + type: "bytes21", + internalType: "ValidationId" + }, + { name: "hook", type: "address", internalType: "contract IHook" }, + { name: "validatorData", type: "bytes", internalType: "bytes" }, + { name: "hookData", type: "bytes", internalType: "bytes" } + ], + outputs: [], + stateMutability: "nonpayable" + }, + { + type: "function", + name: "installModule", + inputs: [ + { name: "moduleType", type: "uint256", internalType: "uint256" }, + { name: "module", type: "address", internalType: "address" }, + { name: "initData", type: "bytes", internalType: "bytes" } + ], + outputs: [], + stateMutability: "payable" + }, + { + type: "function", + name: "installValidations", + inputs: [ + { name: "vIds", type: "bytes21[]", internalType: "ValidationId[]" }, + { + name: "configs", + type: "tuple[]", + internalType: "struct ValidationManager.ValidationConfig[]", + components: [ + { name: "nonce", type: "uint32", internalType: "uint32" }, + { + name: "hook", + type: "address", + internalType: "contract IHook" + } + ] + }, + { + name: "validationData", + type: "bytes[]", + internalType: "bytes[]" + }, + { name: "hookData", type: "bytes[]", internalType: "bytes[]" } + ], + outputs: [], + stateMutability: "payable" + }, + { + type: "function", + name: "invalidateNonce", + inputs: [{ name: "nonce", type: "uint32", internalType: "uint32" }], + outputs: [], + stateMutability: "payable" + }, + { + type: "function", + name: "isAllowedSelector", + inputs: [ + { name: "vId", type: "bytes21", internalType: "ValidationId" }, + { name: "selector", type: "bytes4", internalType: "bytes4" } + ], + outputs: [{ name: "", type: "bool", internalType: "bool" }], + stateMutability: "view" + }, + { + type: "function", + name: "isModuleInstalled", + inputs: [ + { name: "moduleType", type: "uint256", internalType: "uint256" }, + { name: "module", type: "address", internalType: "address" }, + { name: "additionalContext", type: "bytes", internalType: "bytes" } + ], + outputs: [{ name: "", type: "bool", internalType: "bool" }], + stateMutability: "view" + }, + { + type: "function", + name: "isValidSignature", + inputs: [ + { name: "hash", type: "bytes32", internalType: "bytes32" }, + { name: "signature", type: "bytes", internalType: "bytes" } + ], + outputs: [{ name: "", type: "bytes4", internalType: "bytes4" }], + stateMutability: "view" + }, + { + type: "function", + name: "permissionConfig", + inputs: [{ name: "pId", type: "bytes4", internalType: "PermissionId" }], + outputs: [ + { + name: "", + type: "tuple", + internalType: "struct ValidationManager.PermissionConfig", + components: [ + { + name: "permissionFlag", + type: "bytes2", + internalType: "PassFlag" + }, + { + name: "signer", + type: "address", + internalType: "contract ISigner" + }, + { + name: "policyData", + type: "bytes22[]", + internalType: "PolicyData[]" + } + ] + } + ], + stateMutability: "view" + }, + { + type: "function", + name: "rootValidator", + inputs: [], + outputs: [{ name: "", type: "bytes21", internalType: "ValidationId" }], + stateMutability: "view" + }, + { + type: "function", + name: "selectorConfig", + inputs: [{ name: "selector", type: "bytes4", internalType: "bytes4" }], + outputs: [ + { + name: "", + type: "tuple", + internalType: "struct SelectorManager.SelectorConfig", + components: [ + { + name: "hook", + type: "address", + internalType: "contract IHook" + }, + { + name: "target", + type: "address", + internalType: "address" + }, + { + name: "callType", + type: "bytes1", + internalType: "CallType" + } + ] + } + ], + stateMutability: "view" + }, + { + type: "function", + name: "supportsExecutionMode", + inputs: [{ name: "mode", type: "bytes32", internalType: "ExecMode" }], + outputs: [{ name: "", type: "bool", internalType: "bool" }], + stateMutability: "pure" + }, + { + type: "function", + name: "supportsModule", + inputs: [ + { name: "moduleTypeId", type: "uint256", internalType: "uint256" } + ], + outputs: [{ name: "", type: "bool", internalType: "bool" }], + stateMutability: "pure" + }, + { + type: "function", + name: "uninstallModule", + inputs: [ + { name: "moduleType", type: "uint256", internalType: "uint256" }, + { name: "module", type: "address", internalType: "address" }, + { name: "deInitData", type: "bytes", internalType: "bytes" } + ], + outputs: [], + stateMutability: "payable" + }, + { + type: "function", + name: "uninstallValidation", + inputs: [ + { name: "vId", type: "bytes21", internalType: "ValidationId" }, + { name: "deinitData", type: "bytes", internalType: "bytes" }, + { name: "hookDeinitData", type: "bytes", internalType: "bytes" } + ], + outputs: [], + stateMutability: "payable" + }, + { + type: "function", + name: "upgradeTo", + inputs: [ + { + name: "_newImplementation", + type: "address", + internalType: "address" + } + ], + outputs: [], + stateMutability: "payable" + }, + { + type: "function", + name: "validNonceFrom", + inputs: [], + outputs: [{ name: "", type: "uint32", internalType: "uint32" }], + stateMutability: "view" + }, + { + type: "function", + name: "validateUserOp", + inputs: [ + { + name: "userOp", + type: "tuple", + internalType: "struct PackedUserOperation", + components: [ + { + name: "sender", + type: "address", + internalType: "address" + }, + { name: "nonce", type: "uint256", internalType: "uint256" }, + { name: "initCode", type: "bytes", internalType: "bytes" }, + { name: "callData", type: "bytes", internalType: "bytes" }, + { + name: "accountGasLimits", + type: "bytes32", + internalType: "bytes32" + }, + { + name: "preVerificationGas", + type: "uint256", + internalType: "uint256" + }, + { + name: "gasFees", + type: "bytes32", + internalType: "bytes32" + }, + { + name: "paymasterAndData", + type: "bytes", + internalType: "bytes" + }, + { name: "signature", type: "bytes", internalType: "bytes" } + ] + }, + { name: "userOpHash", type: "bytes32", internalType: "bytes32" }, + { + name: "missingAccountFunds", + type: "uint256", + internalType: "uint256" + } + ], + outputs: [ + { + name: "validationData", + type: "uint256", + internalType: "ValidationData" + } + ], + stateMutability: "payable" + }, + { + type: "function", + name: "validationConfig", + inputs: [ + { name: "vId", type: "bytes21", internalType: "ValidationId" } + ], + outputs: [ + { + name: "", + type: "tuple", + internalType: "struct ValidationManager.ValidationConfig", + components: [ + { name: "nonce", type: "uint32", internalType: "uint32" }, + { + name: "hook", + type: "address", + internalType: "contract IHook" + } + ] + } + ], + stateMutability: "view" + }, + { + type: "event", + name: "ModuleInstalled", + inputs: [ + { + name: "moduleTypeId", + type: "uint256", + indexed: false, + internalType: "uint256" + }, + { + name: "module", + type: "address", + indexed: false, + internalType: "address" + } + ], + anonymous: false + }, + { + type: "event", + name: "ModuleUninstallResult", + inputs: [ + { + name: "module", + type: "address", + indexed: false, + internalType: "address" + }, + { + name: "result", + type: "bool", + indexed: false, + internalType: "bool" + } + ], + anonymous: false + }, + { + type: "event", + name: "ModuleUninstalled", + inputs: [ + { + name: "moduleTypeId", + type: "uint256", + indexed: false, + internalType: "uint256" + }, + { + name: "module", + type: "address", + indexed: false, + internalType: "address" + } + ], + anonymous: false + }, + { + type: "event", + name: "NonceInvalidated", + inputs: [ + { + name: "nonce", + type: "uint32", + indexed: false, + internalType: "uint32" + } + ], + anonymous: false + }, + { + type: "event", + name: "PermissionInstalled", + inputs: [ + { + name: "permission", + type: "bytes4", + indexed: false, + internalType: "PermissionId" + }, + { + name: "nonce", + type: "uint32", + indexed: false, + internalType: "uint32" + } + ], + anonymous: false + }, + { + type: "event", + name: "PermissionUninstalled", + inputs: [ + { + name: "permission", + type: "bytes4", + indexed: false, + internalType: "PermissionId" + } + ], + anonymous: false + }, + { + type: "event", + name: "Received", + inputs: [ + { + name: "sender", + type: "address", + indexed: false, + internalType: "address" + }, + { + name: "amount", + type: "uint256", + indexed: false, + internalType: "uint256" + } + ], + anonymous: false + }, + { + type: "event", + name: "RootValidatorUpdated", + inputs: [ + { + name: "rootValidator", + type: "bytes21", + indexed: false, + internalType: "ValidationId" + } + ], + anonymous: false + }, + { + type: "event", + name: "SelectorSet", + inputs: [ + { + name: "selector", + type: "bytes4", + indexed: false, + internalType: "bytes4" + }, + { + name: "vId", + type: "bytes21", + indexed: false, + internalType: "ValidationId" + }, + { + name: "allowed", + type: "bool", + indexed: false, + internalType: "bool" + } + ], + anonymous: false + }, + { + type: "event", + name: "TryExecuteUnsuccessful", + inputs: [ + { + name: "batchExecutionindex", + type: "uint256", + indexed: false, + internalType: "uint256" + }, + { + name: "result", + type: "bytes", + indexed: false, + internalType: "bytes" + } + ], + anonymous: false + }, + { + type: "event", + name: "Upgraded", + inputs: [ + { + name: "implementation", + type: "address", + indexed: true, + internalType: "address" + } + ], + anonymous: false + }, + { + type: "event", + name: "ValidatorInstalled", + inputs: [ + { + name: "validator", + type: "address", + indexed: false, + internalType: "contract IValidator" + }, + { + name: "nonce", + type: "uint32", + indexed: false, + internalType: "uint32" + } + ], + anonymous: false + }, + { + type: "event", + name: "ValidatorUninstalled", + inputs: [ + { + name: "validator", + type: "address", + indexed: false, + internalType: "contract IValidator" + } + ], + anonymous: false + }, + { type: "error", name: "EnableNotApproved", inputs: [] }, + { type: "error", name: "ExecutionReverted", inputs: [] }, + { type: "error", name: "InvalidCallType", inputs: [] }, + { type: "error", name: "InvalidCaller", inputs: [] }, + { type: "error", name: "InvalidExecutor", inputs: [] }, + { type: "error", name: "InvalidFallback", inputs: [] }, + { type: "error", name: "InvalidMode", inputs: [] }, + { type: "error", name: "InvalidModuleType", inputs: [] }, + { type: "error", name: "InvalidNonce", inputs: [] }, + { type: "error", name: "InvalidSelector", inputs: [] }, + { type: "error", name: "InvalidSignature", inputs: [] }, + { type: "error", name: "InvalidValidationType", inputs: [] }, + { type: "error", name: "InvalidValidator", inputs: [] }, + { type: "error", name: "NonceInvalidationError", inputs: [] }, + { type: "error", name: "NotSupportedCallType", inputs: [] }, + { type: "error", name: "OnlyExecuteUserOp", inputs: [] }, + { type: "error", name: "PermissionDataLengthMismatch", inputs: [] }, + { type: "error", name: "PermissionNotAlllowedForSignature", inputs: [] }, + { type: "error", name: "PermissionNotAlllowedForUserOp", inputs: [] }, + { type: "error", name: "PolicyDataTooLarge", inputs: [] }, + { + type: "error", + name: "PolicyFailed", + inputs: [{ name: "i", type: "uint256", internalType: "uint256" }] + }, + { type: "error", name: "PolicySignatureOrderError", inputs: [] }, + { type: "error", name: "RootValidatorCannotBeRemoved", inputs: [] }, + { type: "error", name: "SignerPrefixNotPresent", inputs: [] } +] as const diff --git a/packages/permissionless/accounts/kernel/signerToEcdsaKernelSmartAccount.ts b/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts similarity index 56% rename from packages/permissionless/accounts/kernel/signerToEcdsaKernelSmartAccount.ts rename to packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts index d1714f6c..23eb7edf 100644 --- a/packages/permissionless/accounts/kernel/signerToEcdsaKernelSmartAccount.ts +++ b/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts @@ -1,35 +1,30 @@ -import type { Account, TypedData } from "viem" +import type { Assign } from "viem" import { type Address, - type Chain, type Client, type Hex, type LocalAccount, - type Transport, type TypedDataDefinition, concatHex, encodeFunctionData, toHex, zeroAddress } from "viem" +import { + type SmartAccount, + type SmartAccountImplementation, + type UserOperation, + entryPoint06Abi, + type entryPoint06Address, + entryPoint07Abi, + entryPoint07Address, + getUserOperationHash, + toSmartAccount +} from "viem/account-abstraction" import { signMessage as _signMessage, getChainId } from "viem/actions" +import { getAction } from "viem/utils" import { getAccountNonce } from "../../actions/public/getAccountNonce" import { getSenderAddress } from "../../actions/public/getSenderAddress" -import type { - ENTRYPOINT_ADDRESS_V07_TYPE, - EntryPoint, - Prettify -} from "../../types" -import type { ENTRYPOINT_ADDRESS_V06_TYPE } from "../../types/entrypoint" -import { ENTRYPOINT_ADDRESS_V06, getEntryPointVersion } from "../../utils" -import { getUserOperationHash } from "../../utils/getUserOperationHash" -import { isSmartAccountDeployed } from "../../utils/isSmartAccountDeployed" -import { toSmartAccount } from "../toSmartAccount" -import type { SmartAccount } from "../types" -import { - SignTransactionNotSupportedBySmartAccount, - type SmartAccountSigner -} from "../types" import { KernelInitAbi } from "./abi/KernelAccountAbi" import { KernelV3InitAbi, KernelV3_1AccountAbi } from "./abi/KernelV3AccountAbi" import { KernelV3MetaFactoryDeployWithFactoryAbi } from "./abi/KernelV3MetaFactoryAbi" @@ -44,12 +39,6 @@ import { isKernelV2 } from "./utils/isKernelV2" import { signMessage } from "./utils/signMessage" import { signTypedData } from "./utils/signTypedData" -export type KernelEcdsaSmartAccount< - entryPoint extends EntryPoint, - transport extends Transport = Transport, - chain extends Chain | undefined = Chain | undefined -> = SmartAccount - /** * The account creation ABI for a kernel smart account (from the KernelFactory) */ @@ -85,8 +74,8 @@ const createAccountAbi = [ } ] as const -export type KernelVersion = - entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE +export type KernelVersion = + entryPointVersion extends "0.6" ? "0.2.1" | "0.2.2" | "0.2.3" | "0.2.4" : "0.3.0-beta" | "0.3.1" @@ -94,7 +83,7 @@ export type KernelVersion = * Default addresses map for different kernel smart account versions */ export const KERNEL_VERSION_TO_ADDRESSES_MAP: { - [key in KernelVersion]: { + [key in KernelVersion<"0.6" | "0.7">]: { ECDSA_VALIDATOR: Address ACCOUNT_LOGIC: Address FACTORY_ADDRESS: Address @@ -139,16 +128,16 @@ export const KERNEL_VERSION_TO_ADDRESSES_MAP: { * Get supported Kernel Smart Account version based on entryPoint * @param entryPoint */ -const getDefaultKernelVersion = ( - entryPoint: TEntryPoint, - version?: KernelVersion -): KernelVersion => { +const getDefaultKernelVersion = ( + entryPointVersion: TEntryPointVersion, + version?: KernelVersion +): KernelVersion => { if (version) { return version } return ( - entryPoint === ENTRYPOINT_ADDRESS_V06 ? "0.2.2" : "0.3.0-beta" - ) as KernelVersion + entryPointVersion === "0.6" ? "0.2.2" : "0.3.0-beta" + ) as KernelVersion } type KERNEL_ADDRESSES = { @@ -173,7 +162,7 @@ const getDefaultAddresses = ({ metaFactoryAddress: _metaFactoryAddress, kernelVersion }: Partial & { - kernelVersion: KernelVersion + kernelVersion: KernelVersion<"0.6" | "0.7"> }): KERNEL_ADDRESSES => { const addresses = KERNEL_VERSION_TO_ADDRESSES_MAP[kernelVersion] const ecdsaValidatorAddress = @@ -203,20 +192,21 @@ export const getEcdsaRootIdentifierForKernelV3 = ( * @param owner * @param ecdsaValidatorAddress */ -const getInitialisationData = ({ +const getInitializationData = ({ + entryPoint: { version: entryPointVersion, address: entryPointAddress }, kernelVersion, - entryPoint: entryPointAddress, owner, ecdsaValidatorAddress }: { - kernelVersion: KernelVersion - entryPoint: entryPoint + kernelVersion: KernelVersion + entryPoint: { + version: entryPointVersion + address: typeof entryPoint06Address | typeof entryPoint07Address + } owner: Address ecdsaValidatorAddress: Address }) => { - const entryPointVersion = getEntryPointVersion(entryPointAddress) - - if (entryPointVersion === "v0.6") { + if (entryPointVersion === "0.6") { return encodeFunctionData({ abi: KernelInitAbi, functionName: "initialize", @@ -259,8 +249,8 @@ const getInitialisationData = ({ * @param accountLogicAddress * @param ecdsaValidatorAddress */ -const getAccountInitCode = async ({ - entryPoint: entryPointAddress, +const getAccountInitCode = async ({ + entryPoint: { version: entryPointVersion, address: entryPointAddress }, kernelVersion, owner, index, @@ -268,39 +258,39 @@ const getAccountInitCode = async ({ accountLogicAddress, ecdsaValidatorAddress }: { - kernelVersion: KernelVersion - entryPoint: entryPoint + kernelVersion: KernelVersion + entryPoint: { + version: entryPointVersion + address: typeof entryPoint06Address | typeof entryPoint07Address + } owner: Address index: bigint factoryAddress: Address accountLogicAddress: Address ecdsaValidatorAddress: Address }): Promise => { - if (!owner) throw new Error("Owner account not found") - const entryPointVersion = getEntryPointVersion(entryPointAddress) - // Build the account initialization data - const initialisationData = getInitialisationData({ + const initializationData = getInitializationData({ + entryPoint: { version: entryPointVersion, address: entryPointAddress }, kernelVersion, - entryPoint: entryPointAddress, ecdsaValidatorAddress, owner }) // Build the account init code - if (entryPointVersion === "v0.6") { + if (entryPointVersion === "0.6") { return encodeFunctionData({ abi: createAccountAbi, functionName: "createAccount", - args: [accountLogicAddress, initialisationData, index] + args: [accountLogicAddress, initializationData, index] }) } return encodeFunctionData({ abi: KernelV3MetaFactoryDeployWithFactoryAbi, functionName: "deployWithFactory", - args: [factoryAddress, initialisationData, toHex(index, { size: 32 })] + args: [factoryAddress, initializationData, toHex(index, { size: 32 })] }) } @@ -313,53 +303,95 @@ const getAccountInitCode = async ({ * @param initCodeProvider * @param factoryAddress */ -const getAccountAddress = async < - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined ->({ +const getAccountAddress = async ({ client, - entryPoint: entryPointAddress, + entryPointVersion, + entryPointAddress, factory, factoryData }: { - client: Client - entryPoint: entryPoint + client: Client + entryPointVersion: "0.6" | "0.7" + entryPointAddress: typeof entryPoint06Address | typeof entryPoint07Address factory: Address factoryData: Hex }): Promise
=> { // Find the init code for this account - const entryPointVersion = getEntryPointVersion(entryPointAddress) - if (entryPointVersion === "v0.6") { - return getSenderAddress(client, { + if (entryPointVersion === "0.6") { + return getSenderAddress(client, { initCode: concatHex([factory, factoryData]), - entryPoint: entryPointAddress as ENTRYPOINT_ADDRESS_V06_TYPE + entryPointAddress: entryPointAddress as typeof entryPoint06Address }) } - return getSenderAddress(client, { + return getSenderAddress(client, { factory: factory, factoryData: factoryData, - entryPoint: entryPointAddress as ENTRYPOINT_ADDRESS_V07_TYPE + entryPointAddress: entryPointAddress as typeof entryPoint07Address }) } -export type SignerToEcdsaKernelSmartAccountParameters< - entryPoint extends EntryPoint, - TSource extends string = string, - TAddress extends Address = Address -> = Prettify<{ - signer: SmartAccountSigner - version?: KernelVersion - entryPoint: entryPoint +export type ToEcdsaKernelSmartAccountParameters< + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi, + kernelVersion extends + KernelVersion = entryPointVersion extends "0.6" + ? KernelVersion<"0.6"> + : KernelVersion<"0.7"> +> = { + client: Client + owner: LocalAccount + entryPoint?: { + address: entryPointAddress + abi: entryPointAbi + version: entryPointVersion + } address?: Address + version?: kernelVersion index?: bigint factoryAddress?: Address metaFactoryAddress?: Address accountLogicAddress?: Address ecdsaValidatorAddress?: Address nonceKey?: bigint -}> +} + +export type EcdsaKernelSmartAccountImplementation< + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi +> = Assign< + SmartAccountImplementation< + entryPointAbi, + entryPointVersion + // { + // // entryPoint === ENTRYPOINT_ADDRESS_V06 ? "0.2.2" : "0.3.0-beta" + // abi: entryPointVersion extends "0.6" ? typeof BiconomyAbi + // factory: { abi: typeof FactoryAbi; address: Address } + // } + >, + { sign: NonNullable } +> + +export type ToEcdsaKernelSmartAccountReturnType< + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi +> = SmartAccount< + EcdsaKernelSmartAccountImplementation +> /** * Build a kernel smart account from a private key, that use the ECDSA signer behind the scene * @param client @@ -370,30 +402,48 @@ export type SignerToEcdsaKernelSmartAccountParameters< * @param accountLogicAddress * @param ecdsaValidatorAddress */ -export async function signerToEcdsaKernelSmartAccount< - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined, - TSource extends string = string, - TAddress extends Address = Address +export async function toEcdsaKernelSmartAccount< + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address = typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7" = "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi >( - client: Client, - { - signer, + parameters: ToEcdsaKernelSmartAccountParameters< + entryPointAddress, + entryPointVersion, + entryPointAbi + > +): Promise< + ToEcdsaKernelSmartAccountReturnType +> { + const { + client, address, + index = 0n, + owner, version, - entryPoint: entryPointAddress, - index = BigInt(0), + ecdsaValidatorAddress: _ecdsaValidatorAddress, factoryAddress: _factoryAddress, metaFactoryAddress: _metaFactoryAddress, - accountLogicAddress: _accountLogicAddress, - ecdsaValidatorAddress: _ecdsaValidatorAddress, - nonceKey - }: SignerToEcdsaKernelSmartAccountParameters -): Promise> { - const entryPointVersion = getEntryPointVersion(entryPointAddress) - const kernelVersion = getDefaultKernelVersion(entryPointAddress, version) + accountLogicAddress: _accountLogicAddress + } = parameters + + const entryPoint = { + address: parameters.entryPoint?.address ?? entryPoint07Address, + abi: + parameters.entryPoint?.abi ?? + (parameters.entryPoint?.version ?? "0.7") === "0.6" + ? entryPoint06Abi + : entryPoint07Abi, + version: parameters.entryPoint?.version ?? "0.7" + } as const + + const kernelVersion = getDefaultKernelVersion(entryPoint.version, version) const { accountLogicAddress, @@ -408,57 +458,95 @@ export async function signerToEcdsaKernelSmartAccount< kernelVersion }) - // Get the private key related account - const viemSigner: LocalAccount = { - ...signer, - signTransaction: (_, __) => { - throw new SignTransactionNotSupportedBySmartAccount() - } - } as LocalAccount - // Helper to generate the init code for the smart account const generateInitCode = () => getAccountInitCode({ + entryPoint: { + version: entryPoint.version, + address: entryPoint.address + }, kernelVersion, - entryPoint: entryPointAddress, - owner: viemSigner.address, + owner: owner.address, index, factoryAddress, accountLogicAddress, ecdsaValidatorAddress }) - // Fetch account address and chain id - const [accountAddress, chainId] = await Promise.all([ - address ?? - getAccountAddress({ + let accountAddress: Address + + const getAddress = async () => { + if (accountAddress) return accountAddress + accountAddress = + address ?? + (await getAccountAddress({ client, factory: - entryPointVersion === "v0.6" + entryPoint.version === "0.6" ? factoryAddress : metaFactoryAddress, factoryData: await generateInitCode(), - entryPoint: entryPointAddress - }), - client.chain?.id ?? getChainId(client) - ]) + entryPointAddress: entryPoint.address, + entryPointVersion: entryPoint.version + })) + return accountAddress + } - if (!accountAddress) throw new Error("Account address not found") + let chainId: number - let smartAccountDeployed = await isSmartAccountDeployed( - client, - accountAddress - ) + const getMemoizedChainId = async () => { + if (chainId) return chainId + chainId = client.chain + ? client.chain.id + : await getAction(client, getChainId, "getChainId")({}) + return chainId + } return toSmartAccount({ - address: accountAddress, + client, + entryPoint, + getAddress, + async encodeCalls(calls) { + return encodeCallData({ calls, kernelVersion }) + }, + async getFactoryArgs() { + return { + factory: + entryPoint.version === "0.6" + ? factoryAddress + : metaFactoryAddress, + factoryData: await generateInitCode() + } + }, + async getNonce(args) { + return getAccountNonce(client, { + address: await getAddress(), + entryPointAddress: entryPoint.address, + key: getNonceKeyWithEncoding( + kernelVersion, + ecdsaValidatorAddress, + args?.key ?? parameters?.nonceKey ?? 0n + // @dev specify the custom nonceKey here when integrating the said feature + /*, nonceKey */ + ) + }) + }, + async getStubSignature() { + if (isKernelV2(kernelVersion)) { + return concatHex([ROOT_MODE_KERNEL_V2, DUMMY_ECDSA_SIGNATURE]) + } + return DUMMY_ECDSA_SIGNATURE + }, + async sign({ hash }) { + return this.signMessage({ message: hash }) + }, async signMessage({ message }) { - const signature = await signMessage(client, { - account: viemSigner, + const signature = await signMessage({ + owner: owner, message, accountAddress, - accountVersion: kernelVersion, - chainId + kernelVersion, + chainId: await getMemoizedChainId() }) if (isKernelV2(kernelVersion)) { @@ -470,26 +558,13 @@ export async function signerToEcdsaKernelSmartAccount< signature ]) }, - async signTransaction(_, __) { - throw new SignTransactionNotSupportedBySmartAccount() - }, - async signTypedData< - const TTypedData extends TypedData | Record, - TPrimaryType extends - | keyof TTypedData - | "EIP712Domain" = keyof TTypedData - >(typedData: TypedDataDefinition) { - const signature = await signTypedData< - TTypedData, - TPrimaryType, - TChain, - TClientAccount - >(client, { - account: viemSigner, - ...typedData, + async signTypedData(typedData) { + const signature = await signTypedData({ + owner, + chainId: await getMemoizedChainId(), + ...(typedData as TypedDataDefinition), accountAddress, - accountVersion: kernelVersion, - chainId + kernelVersion }) if (isKernelV2(kernelVersion)) { @@ -501,111 +576,32 @@ export async function signerToEcdsaKernelSmartAccount< signature ]) }, - client: client, - publicKey: accountAddress, - entryPoint: entryPointAddress, - source: "kernelEcdsaSmartAccount", - - // Get the nonce of the smart account - async getNonce(key?: bigint) { - return getAccountNonce(client, { - sender: accountAddress, - entryPoint: entryPointAddress, - key: - key ?? - nonceKey ?? - getNonceKeyWithEncoding( - kernelVersion, - ecdsaValidatorAddress - // @dev specify the custom nonceKey here when integrating the said feature - /*, nonceKey */ - ) - }) - }, - // Sign a user operation - async signUserOperation(userOperation) { + async signUserOperation(parameters) { + const { chainId = await getMemoizedChainId(), ...userOperation } = + parameters + const hash = getUserOperationHash({ userOperation: { ...userOperation, + sender: userOperation.sender ?? (await getAddress()), signature: "0x" - }, - entryPoint: entryPointAddress, + } as UserOperation, + entryPointAddress: entryPoint.address, + entryPointVersion: entryPoint.version, chainId: chainId }) - const signature = await _signMessage(client, { - account: viemSigner, + const signature = await owner.signMessage({ message: { raw: hash } }) + // Always use the sudo mode, since we will use external paymaster if (isKernelV2(kernelVersion)) { return concatHex(["0x00000000", signature]) } return signature - }, - - // Encode the init code - async getInitCode() { - if (smartAccountDeployed) return "0x" - - smartAccountDeployed = await isSmartAccountDeployed( - client, - accountAddress - ) - - if (smartAccountDeployed) return "0x" - - const _factoryAddress = - entryPointVersion === "v0.6" - ? factoryAddress - : metaFactoryAddress - return concatHex([_factoryAddress, await generateInitCode()]) - }, - - async getFactory() { - if (smartAccountDeployed) return undefined - - smartAccountDeployed = await isSmartAccountDeployed( - client, - accountAddress - ) - - if (smartAccountDeployed) return undefined - - return entryPointVersion === "v0.6" - ? factoryAddress - : metaFactoryAddress - }, - - async getFactoryData() { - if (smartAccountDeployed) return undefined - - smartAccountDeployed = await isSmartAccountDeployed( - client, - accountAddress - ) - - if (smartAccountDeployed) return undefined - - return generateInitCode() - }, - - // Encode the deploy call data - async encodeDeployCallData(_) { - throw new Error("Simple account doesn't support account deployment") - }, - - // Encode a call - async encodeCallData(_tx) { - return encodeCallData(_tx, kernelVersion) - }, - - // Get simple dummy signature - async getDummySignature(_userOperation) { - if (isKernelV2(kernelVersion)) { - return concatHex([ROOT_MODE_KERNEL_V2, DUMMY_ECDSA_SIGNATURE]) - } - return DUMMY_ECDSA_SIGNATURE } - }) + }) as Promise< + ToEcdsaKernelSmartAccountReturnType + > } diff --git a/packages/permissionless/accounts/kernel/utils/encodeCallData.ts b/packages/permissionless/accounts/kernel/utils/encodeCallData.ts index deac9b3b..1b58ef7c 100644 --- a/packages/permissionless/accounts/kernel/utils/encodeCallData.ts +++ b/packages/permissionless/accounts/kernel/utils/encodeCallData.ts @@ -1,44 +1,31 @@ -import { - type Address, - type Hex, - concatHex, - encodeAbiParameters, - encodeFunctionData, - toHex -} from "viem" -import type { EntryPoint } from "../../../types" +import { type Address, type Hex, encodeFunctionData } from "viem" +import { encode7579Calls } from "../../../utils/encode7579Calls" import { KernelExecuteAbi } from "../abi/KernelAccountAbi" -import { KernelV3ExecuteAbi } from "../abi/KernelV3AccountAbi" -import { CALL_TYPE, EXEC_TYPE } from "../constants" -import type { KernelVersion } from "../signerToEcdsaKernelSmartAccount" -import { getExecMode } from "./getExecMode" +import type { KernelVersion } from "../toEcdsaKernelSmartAccount" import { isKernelV2 } from "./isKernelV2" -export const encodeCallData = ( - _tx: - | { - to: Address - value: bigint - data: Hex - } - | { - to: Address - value: bigint - data: Hex - }[], - accountVersion: KernelVersion -) => { - if (isKernelV2(accountVersion)) { - if (Array.isArray(_tx)) { +export const encodeCallData = ({ + kernelVersion, + calls +}: { + calls: readonly { + to: Address + value?: bigint | undefined + data?: Hex | undefined + }[] + kernelVersion: KernelVersion<"0.6" | "0.7"> +}) => { + if (isKernelV2(kernelVersion)) { + if (calls.length > 1) { // Encode a batched call return encodeFunctionData({ abi: KernelExecuteAbi, functionName: "executeBatch", args: [ - _tx.map((tx) => ({ + calls.map((tx) => ({ to: tx.to, - value: tx.value, - data: tx.data + value: tx.value ?? 0n, + data: tx.data ?? "0x" })) ] }) @@ -47,70 +34,17 @@ export const encodeCallData = ( return encodeFunctionData({ abi: KernelExecuteAbi, functionName: "execute", - args: [_tx.to, _tx.value, _tx.data, 0] + args: [calls[0].to, calls[0].value ?? 0n, calls[0].data ?? "0x", 0] }) } - if (Array.isArray(_tx)) { - // Encode a batched call - const calldata = encodeAbiParameters( - [ - { - name: "executionBatch", - type: "tuple[]", - components: [ - { - name: "target", - type: "address" - }, - { - name: "value", - type: "uint256" - }, - { - name: "callData", - type: "bytes" - } - ] - } - ], - [ - _tx.map((arg) => { - return { - target: arg.to, - value: arg.value, - callData: arg.data - } - }) - ] - ) - return encodeFunctionData({ - abi: KernelV3ExecuteAbi, - functionName: "execute", - args: [ - getExecMode({ - callType: CALL_TYPE.BATCH, - execType: EXEC_TYPE.DEFAULT - }), - calldata - ] - }) - } - - const calldata = concatHex([ - _tx.to, - toHex(_tx.value, { size: 32 }), - _tx.data - ]) - return encodeFunctionData({ - abi: KernelV3ExecuteAbi, - functionName: "execute", - args: [ - getExecMode({ - callType: CALL_TYPE.SINGLE, - execType: EXEC_TYPE.DEFAULT - }), - calldata - ] + return encode7579Calls({ + mode: { + type: Array.isArray(calls) ? "batchcall" : "call", + revertOnError: false, + selector: "0x", + context: "0x" + }, + callData: calls }) } diff --git a/packages/permissionless/accounts/kernel/utils/getNonceKey.ts b/packages/permissionless/accounts/kernel/utils/getNonceKey.ts index 92702549..089f6fc7 100644 --- a/packages/permissionless/accounts/kernel/utils/getNonceKey.ts +++ b/packages/permissionless/accounts/kernel/utils/getNonceKey.ts @@ -1,21 +1,21 @@ import { type Address, concatHex, maxUint16, pad, toHex } from "viem" import type { EntryPoint } from "../../../types" import { VALIDATOR_MODE, VALIDATOR_TYPE } from "../constants" -import type { KernelVersion } from "../signerToEcdsaKernelSmartAccount" +import type { KernelVersion } from "../toEcdsaKernelSmartAccount" import { isKernelV2 } from "./isKernelV2" export const getNonceKeyWithEncoding = ( - accountVerion: KernelVersion, + kernelVersion: KernelVersion<"0.6" | "0.7">, validatorAddress: Address, nonceKey = 0n ) => { - if (isKernelV2(accountVerion)) { + if (isKernelV2(kernelVersion)) { return nonceKey } if (nonceKey > maxUint16) throw new Error( - `nonce key must be equal or less than 2 bytes(maxUint16) for Kernel version ${accountVerion}` + `nonce key must be equal or less than 2 bytes(maxUint16) for Kernel version ${kernelVersion}` ) const validatorMode = VALIDATOR_MODE.DEFAULT @@ -29,6 +29,5 @@ export const getNonceKeyWithEncoding = ( ]), { size: 24 } ) // 24 bytes - const encodedNonceKey = BigInt(encoding) - return encodedNonceKey + return BigInt(encoding) } diff --git a/packages/permissionless/accounts/kernel/utils/isKernelV2.ts b/packages/permissionless/accounts/kernel/utils/isKernelV2.ts index 68476b63..16fc02f9 100644 --- a/packages/permissionless/accounts/kernel/utils/isKernelV2.ts +++ b/packages/permissionless/accounts/kernel/utils/isKernelV2.ts @@ -1,7 +1,6 @@ -import type { EntryPoint } from "../../../types" -import type { KernelVersion } from "../signerToEcdsaKernelSmartAccount" +import type { KernelVersion } from "../toEcdsaKernelSmartAccount" -export const isKernelV2 = (version: KernelVersion): boolean => { +export const isKernelV2 = (version: KernelVersion<"0.6" | "0.7">): boolean => { const regex = /0\.2\.\d+/ return regex.test(version) } diff --git a/packages/permissionless/accounts/kernel/utils/signMessage.ts b/packages/permissionless/accounts/kernel/utils/signMessage.ts index 98860ff5..510a1d99 100644 --- a/packages/permissionless/accounts/kernel/utils/signMessage.ts +++ b/packages/permissionless/accounts/kernel/utils/signMessage.ts @@ -1,49 +1,40 @@ import { - type Account, - type Chain, - type Client, type LocalAccount, - type SignMessageParameters, type SignMessageReturnType, - type Transport, - hashMessage, - publicActions + type SignableMessage, + hashMessage } from "viem" import { signMessage as _signMessage } from "viem/actions" import { isKernelV2 } from "./isKernelV2" import { type WrapMessageHashParams, wrapMessageHash } from "./wrapMessageHash" -export async function signMessage< - TChain extends Chain | undefined, - TAccount extends Account | undefined ->( - client: Client, - { - account: account_ = client.account, - message, - accountAddress, - accountVersion - }: SignMessageParameters & WrapMessageHashParams -): Promise { +export async function signMessage({ + message, + owner, + accountAddress, + kernelVersion: accountVersion, + chainId +}: { + chainId: number + message: SignableMessage + owner: LocalAccount +} & WrapMessageHashParams): Promise { if (isKernelV2(accountVersion)) { - return _signMessage(client, { - account: account_ as LocalAccount, + return owner.signMessage({ message }) } const wrappedMessageHash = wrapMessageHash(hashMessage(message), { - accountVersion, + kernelVersion: accountVersion, accountAddress, - chainId: client.chain - ? client.chain.id - : await client.extend(publicActions).getChainId() + chainId + // chainId: client.chain + // ? client.chain.id + // : await client.extend(publicActions).getChainId() }) - const signature = await _signMessage(client, { - account: account_ as LocalAccount, + return owner.signMessage({ message: { raw: wrappedMessageHash } }) - - return signature } diff --git a/packages/permissionless/accounts/kernel/utils/signTypedData.ts b/packages/permissionless/accounts/kernel/utils/signTypedData.ts index 901bc204..ff7bd81f 100644 --- a/packages/permissionless/accounts/kernel/utils/signTypedData.ts +++ b/packages/permissionless/accounts/kernel/utils/signTypedData.ts @@ -1,43 +1,32 @@ import { - type Account, - type Chain, - type Client, type LocalAccount, - type SignTypedDataParameters, type SignTypedDataReturnType, - type Transport, - type TypedData, + type TypedDataDefinition, getTypesForEIP712Domain, hashTypedData, - publicActions, validateTypedData } from "viem" - -import { - signMessage as _signMessage, - signTypedData as _signTypedData -} from "viem/actions" import { isKernelV2 } from "./isKernelV2" import { type WrapMessageHashParams, wrapMessageHash } from "./wrapMessageHash" -export async function signTypedData< - const typedData extends TypedData | Record, - primaryType extends keyof typedData | "EIP712Domain", - chain extends Chain | undefined, - account extends Account | undefined ->( - client: Client, - parameters: SignTypedDataParameters & - WrapMessageHashParams +export async function signTypedData( + parameters: TypedDataDefinition & + WrapMessageHashParams & { + owner: LocalAccount + } ): Promise { const { - account: account_, + owner, accountAddress, - accountVersion, + kernelVersion: accountVersion, + chainId, ...typedData - } = parameters as unknown as SignTypedDataParameters & WrapMessageHashParams + } = parameters + if (isKernelV2(accountVersion)) { - return _signTypedData(client, { account: account_, ...typedData }) + return owner.signTypedData({ + ...typedData + }) } const { message, primaryType, types: _types, domain } = typedData const types = { @@ -59,17 +48,12 @@ export async function signTypedData< const typedHash = hashTypedData({ message, primaryType, types, domain }) const wrappedMessageHash = wrapMessageHash(typedHash, { - accountVersion, + kernelVersion: accountVersion, accountAddress, - chainId: client.chain - ? client.chain.id - : await client.extend(publicActions).getChainId() + chainId: chainId }) - const signature = await _signMessage(client, { - account: account_ as LocalAccount, + return owner.signMessage({ message: { raw: wrappedMessageHash } }) - - return signature } diff --git a/packages/permissionless/accounts/kernel/utils/wrapMessageHash.ts b/packages/permissionless/accounts/kernel/utils/wrapMessageHash.ts index 47b5ff20..a2e72a5b 100644 --- a/packages/permissionless/accounts/kernel/utils/wrapMessageHash.ts +++ b/packages/permissionless/accounts/kernel/utils/wrapMessageHash.ts @@ -6,23 +6,22 @@ import { stringToHex } from "viem" import { type Address, domainSeparator } from "viem" -import type { EntryPoint } from "../../../types" -import type { KernelVersion } from "../signerToEcdsaKernelSmartAccount" +import type { KernelVersion } from "../toEcdsaKernelSmartAccount" export type WrapMessageHashParams = { - accountVersion: KernelVersion + kernelVersion: KernelVersion<"0.6" | "0.7"> accountAddress: Address - chainId?: number + chainId: number } export const wrapMessageHash = ( messageHash: Hex, - { accountAddress, accountVersion, chainId }: WrapMessageHashParams + { accountAddress, kernelVersion, chainId }: WrapMessageHashParams ) => { const _domainSeparator = domainSeparator({ domain: { name: "Kernel", - version: accountVersion, + version: kernelVersion, chainId, verifyingContract: accountAddress } diff --git a/packages/permissionless/accounts/light/privateKeyToLightSmartAccount.ts b/packages/permissionless/accounts/light/privateKeyToLightSmartAccount.ts deleted file mode 100644 index c243054e..00000000 --- a/packages/permissionless/accounts/light/privateKeyToLightSmartAccount.ts +++ /dev/null @@ -1,39 +0,0 @@ -import type { Account, Chain, Client, Hex, Transport } from "viem" -import { privateKeyToAccount } from "viem/accounts" -import type { Prettify } from "../../types" -import type { EntryPoint } from "../../types" -import { - type LightSmartAccount, - type SignerToLightSmartAccountParameters, - signerToLightSmartAccount -} from "./signerToLightSmartAccount" - -export type PrivateKeyToLightSmartAccountParameters< - entryPoint extends EntryPoint -> = Prettify< - { - privateKey: Hex - } & Omit, "signer"> -> - -/** - * @description Creates an Light Account from a private key. - * - * @returns A Private Key Light Account. - */ -export async function privateKeyToLightSmartAccount< - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined ->( - client: Client, - { privateKey, ...rest }: PrivateKeyToLightSmartAccountParameters -): Promise> { - const privateKeyAccount = privateKeyToAccount(privateKey) - - return signerToLightSmartAccount(client, { - signer: privateKeyAccount, - ...rest - }) -} diff --git a/packages/permissionless/accounts/light/signerToLightSmartAccount.ts b/packages/permissionless/accounts/light/toLightSmartAccount.ts similarity index 51% rename from packages/permissionless/accounts/light/signerToLightSmartAccount.ts rename to packages/permissionless/accounts/light/toLightSmartAccount.ts index 65a65f08..20ae28df 100644 --- a/packages/permissionless/accounts/light/signerToLightSmartAccount.ts +++ b/packages/permissionless/accounts/light/toLightSmartAccount.ts @@ -1,42 +1,29 @@ import { - type Account, type Address, - type Chain, + type Assign, type Client, type Hex, type LocalAccount, - type Transport, - type TypedData, - type TypedDataDefinition, concatHex, encodeFunctionData, hashMessage, hashTypedData } from "viem" +import { + type SmartAccount, + type SmartAccountImplementation, + type UserOperation, + entryPoint06Abi, + type entryPoint06Address, + entryPoint07Abi, + entryPoint07Address, + getUserOperationHash, + toSmartAccount +} from "viem/account-abstraction" import { getChainId, signMessage } from "viem/actions" +import { getAction } from "viem/utils" import { getAccountNonce } from "../../actions/public/getAccountNonce" import { getSenderAddress } from "../../actions/public/getSenderAddress" -import type { - ENTRYPOINT_ADDRESS_V06_TYPE, - ENTRYPOINT_ADDRESS_V07_TYPE, - Prettify -} from "../../types" -import type { EntryPoint } from "../../types/entrypoint" -import { getEntryPointVersion } from "../../utils" -import { getUserOperationHash } from "../../utils/getUserOperationHash" -import { isSmartAccountDeployed } from "../../utils/isSmartAccountDeployed" -import { toSmartAccount } from "../toSmartAccount" -import { - SignTransactionNotSupportedBySmartAccount, - type SmartAccount, - type SmartAccountSigner -} from "../types" - -export type LightSmartAccount< - entryPoint extends EntryPoint, - transport extends Transport = Transport, - chain extends Chain | undefined = Chain | undefined -> = SmartAccount const getAccountInitCode = async ( owner: Address, @@ -77,63 +64,71 @@ const getAccountInitCode = async ( } const getAccountAddress = async < - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7" >({ client, factoryAddress, - entryPoint: entryPointAddress, + entryPointAddress, + entryPointVersion, owner, index = BigInt(0) }: { - client: Client + client: Client factoryAddress: Address owner: Address - entryPoint: entryPoint + entryPointAddress: entryPointAddress + entryPointVersion: entryPointVersion index?: bigint }): Promise
=> { - const entryPointVersion = getEntryPointVersion(entryPointAddress) - const factoryData = await getAccountInitCode(owner, index) - if (entryPointVersion === "v0.6") { - return getSenderAddress(client, { + if (entryPointVersion === "0.6") { + return getSenderAddress(client, { initCode: concatHex([factoryAddress, factoryData]), - entryPoint: entryPointAddress as ENTRYPOINT_ADDRESS_V06_TYPE + entryPointAddress: entryPointAddress as typeof entryPoint06Address }) } // Get the sender address based on the init code - return getSenderAddress(client, { + return getSenderAddress(client, { factory: factoryAddress, factoryData, - entryPoint: entryPointAddress as ENTRYPOINT_ADDRESS_V07_TYPE + entryPointAddress: entryPointAddress as typeof entryPoint07Address }) } export type LightAccountVersion = "1.1.0" -export type SignerToLightSmartAccountParameters< - entryPoint extends EntryPoint, - TSource extends string = string, - TAddress extends Address = Address -> = Prettify<{ - signer: SmartAccountSigner +export type ToLightSmartAccountParameters< + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi +> = { + client: Client + entryPoint?: { + address: entryPointAddress + abi: entryPointAbi + version: entryPointVersion + } + owner: LocalAccount lightAccountVersion: LightAccountVersion - entryPoint: entryPoint factoryAddress?: Address index?: bigint address?: Address nonceKey?: bigint -}> +} -async function signWith1271WrapperV1< - TSource extends string = string, - TAddress extends Address = Address ->( - signer: SmartAccountSigner, +async function signWith1271WrapperV1( + signer: LocalAccount, chainId: number, accountAddress: Address, hashedMessage: Hex @@ -182,36 +177,78 @@ const getDefaultAddresses = ( } } +export type LightSmartAccountImplementation< + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi +> = Assign< + SmartAccountImplementation< + entryPointAbi, + entryPointVersion + // { + // // entryPoint === ENTRYPOINT_ADDRESS_V06 ? "0.2.2" : "0.3.0-beta" + // abi: entryPointVersion extends "0.6" ? typeof BiconomyAbi + // factory: { abi: typeof FactoryAbi; address: Address } + // } + >, + { sign: NonNullable } +> + +export type ToLightSmartAccountReturnType< + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi +> = SmartAccount< + LightSmartAccountImplementation +> + /** * @description Creates an Light Account from a private key. * * @returns A Private Key Light Account. */ -export async function signerToLightSmartAccount< - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined, - TSource extends string = string, - TAddress extends Address = Address +export async function toLightSmartAccount< + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi >( - client: Client, - { - signer, - address, + parameters: ToLightSmartAccountParameters< + entryPointAddress, + entryPointVersion, + entryPointAbi + > +): Promise> { + const { lightAccountVersion, - entryPoint: entryPointAddress, - index = BigInt(0), factoryAddress: _factoryAddress, + address, + owner, + client, + index = BigInt(0), nonceKey - }: SignerToLightSmartAccountParameters -): Promise> { - const viemSigner: LocalAccount = { - ...signer, - signTransaction: (_, __) => { - throw new SignTransactionNotSupportedBySmartAccount() - } - } as LocalAccount + } = parameters + + const entryPoint = { + address: parameters.entryPoint?.address ?? entryPoint07Address, + abi: + parameters.entryPoint?.abi ?? + (parameters.entryPoint?.version ?? "0.7") === "0.6" + ? entryPoint06Abi + : entryPoint07Abi, + version: parameters.entryPoint?.version ?? "0.7" + } as const if (lightAccountVersion !== "1.1.0") { throw new Error( @@ -223,118 +260,38 @@ export async function signerToLightSmartAccount< factoryAddress: _factoryAddress }) - const [accountAddress, chainId] = await Promise.all([ - address ?? - getAccountAddress({ + let accountAddress: Address + const getAddress = async () => { + if (accountAddress) return accountAddress + accountAddress = + address ?? + (await getAccountAddress({ client, factoryAddress, - entryPoint: entryPointAddress, - owner: viemSigner.address, + entryPointAddress: entryPoint.address, + entryPointVersion: entryPoint.version, + owner: owner.address, index - }), - client.chain?.id ?? getChainId(client) - ]) + })) + return accountAddress + } - if (!accountAddress) throw new Error("Account address not found") + let chainId: number - let smartAccountDeployed = await isSmartAccountDeployed( - client, - accountAddress - ) + const getMemoizedChainId = async () => { + if (chainId) return chainId + chainId = client.chain + ? client.chain.id + : await getAction(client, getChainId, "getChainId")({}) + return chainId + } return toSmartAccount({ - address: accountAddress, - signMessage: async ({ message }) => { - return signWith1271WrapperV1( - signer, - chainId, - accountAddress, - hashMessage(message) - ) - }, - signTransaction: (_, __) => { - throw new SignTransactionNotSupportedBySmartAccount() - }, - async signTypedData< - const TTypedData extends TypedData | Record, - TPrimaryType extends - | keyof TTypedData - | "EIP712Domain" = keyof TTypedData - >(typedData: TypedDataDefinition) { - return signWith1271WrapperV1( - signer, - chainId, - accountAddress, - hashTypedData(typedData) - ) - }, - client: client, - publicKey: accountAddress, - entryPoint: entryPointAddress, - source: "LightSmartAccount", - async getNonce(key?: bigint) { - return getAccountNonce(client, { - sender: accountAddress, - entryPoint: entryPointAddress, - key: key ?? nonceKey - }) - }, - async signUserOperation(userOperation) { - return signMessage(client, { - account: viemSigner, - message: { - raw: getUserOperationHash({ - userOperation, - entryPoint: entryPointAddress, - chainId: chainId - }) - } - }) - }, - async getInitCode() { - if (smartAccountDeployed) return "0x" - - smartAccountDeployed = await isSmartAccountDeployed( - client, - accountAddress - ) - - if (smartAccountDeployed) return "0x" - - return concatHex([ - factoryAddress, - await getAccountInitCode(viemSigner.address, index) - ]) - }, - async getFactory() { - if (smartAccountDeployed) return undefined - smartAccountDeployed = await isSmartAccountDeployed( - client, - accountAddress - ) - if (smartAccountDeployed) return undefined - return factoryAddress - }, - async getFactoryData() { - if (smartAccountDeployed) return undefined - smartAccountDeployed = await isSmartAccountDeployed( - client, - accountAddress - ) - if (smartAccountDeployed) return undefined - return getAccountInitCode(viemSigner.address, index) - }, - async encodeDeployCallData(_) { - throw new Error("Light account doesn't support account deployment") - }, - async encodeCallData(args) { - if (Array.isArray(args)) { - const argsArray = args as { - to: Address - value: bigint - data: Hex - }[] - + client, + entryPoint, + getAddress, + async encodeCalls(calls) { + if (calls.length > 1) { return encodeFunctionData({ abi: [ { @@ -363,19 +320,13 @@ export async function signerToLightSmartAccount< ], functionName: "executeBatch", args: [ - argsArray.map((a) => a.to), - argsArray.map((a) => a.value), - argsArray.map((a) => a.data) + calls.map((a) => a.to), + calls.map((a) => a.value ?? 0n), + calls.map((a) => a.data ?? "0x") ] }) } - const { to, value, data } = args as { - to: Address - value: bigint - data: Hex - } - return encodeFunctionData({ abi: [ { @@ -403,11 +354,66 @@ export async function signerToLightSmartAccount< } ], functionName: "execute", - args: [to, value, data] + args: [calls[0].to, calls[0].value ?? 0n, calls[0].data ?? "0x"] + }) + }, + async getFactoryArgs() { + return { + factory: factoryAddress, + factoryData: await getAccountInitCode(owner.address, index) + } + }, + async getNonce(args) { + return getAccountNonce(client, { + address: await getAddress(), + entryPointAddress: entryPoint.address, + key: args?.key ?? nonceKey }) }, - async getDummySignature(_userOperation) { + async getStubSignature() { return "0xfffffffffffffffffffffffffffffff0000000000000000000000000000000007aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa1c" + }, + async sign({ hash }) { + return this.signMessage({ message: hash }) + }, + async signMessage({ message }) { + return signWith1271WrapperV1( + owner, + await getMemoizedChainId(), + await getAddress(), + hashMessage(message) + ) + }, + async signTypedData(typedData) { + return signWith1271WrapperV1( + owner, + await getMemoizedChainId(), + await getAddress(), + hashTypedData(typedData) + ) + }, + async signUserOperation(parameters) { + const { chainId = await getMemoizedChainId(), ...userOperation } = + parameters + + const hash = getUserOperationHash({ + userOperation: { + ...userOperation, + signature: "0x" + } as UserOperation, + entryPointAddress: entryPoint.address, + entryPointVersion: entryPoint.version, + chainId: chainId + }) + + return signMessage(client, { + account: owner, + message: { + raw: hash + } + }) } - }) + }) as Promise< + ToLightSmartAccountReturnType + > } diff --git a/packages/permissionless/accounts/safe/privateKeyToSafeSmartAccount.ts b/packages/permissionless/accounts/safe/privateKeyToSafeSmartAccount.ts deleted file mode 100644 index a16bdfa6..00000000 --- a/packages/permissionless/accounts/safe/privateKeyToSafeSmartAccount.ts +++ /dev/null @@ -1,38 +0,0 @@ -import type { Account, Chain, Client, Hex, Transport } from "viem" -import { privateKeyToAccount } from "viem/accounts" -import type { EntryPoint, Prettify } from "../../types" -import { - type SafeSmartAccount, - type SignerToSafeSmartAccountParameters, - signerToSafeSmartAccount -} from "./signerToSafeSmartAccount" - -export type PrivateKeyToSafeSmartAccountParameters< - entryPoint extends EntryPoint -> = Prettify< - { - privateKey: Hex - } & Omit, "signer"> -> - -/** - * @description Creates an Simple Account from a private key. - * - * @returns A Private Key Simple Account. - */ -export async function privateKeyToSafeSmartAccount< - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined ->( - client: Client, - { privateKey, ...rest }: PrivateKeyToSafeSmartAccountParameters -): Promise> { - const privateKeyAccount = privateKeyToAccount(privateKey) - - return signerToSafeSmartAccount(client, { - signer: privateKeyAccount, - ...rest - }) -} diff --git a/packages/permissionless/accounts/safe/signerToSafeSmartAccount.ts b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts similarity index 69% rename from packages/permissionless/accounts/safe/signerToSafeSmartAccount.ts rename to packages/permissionless/accounts/safe/toSafeSmartAccount.ts index 419b022e..ee3d8d10 100644 --- a/packages/permissionless/accounts/safe/signerToSafeSmartAccount.ts +++ b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts @@ -1,6 +1,7 @@ import { type Account, type Address, + type Assign, type Chain, type Client, type Hex, @@ -24,28 +25,25 @@ import { toHex, zeroAddress } from "viem" +import { + type SmartAccount, + type SmartAccountImplementation, + type UserOperation, + entryPoint06Abi, + type entryPoint06Address, + entryPoint07Abi, + entryPoint07Address, + toSmartAccount +} from "viem/account-abstraction" import { getChainId, readContract, signMessage, signTypedData } from "viem/actions" +import { getAction } from "viem/utils" import { getAccountNonce } from "../../actions/public/getAccountNonce" -import type { EntryPointVersion, GetEntryPointVersion } from "../../types" -import type { EntryPoint, UserOperation } from "../../types" -import { encode7579CallData } from "../../utils/encode7579CallData" -import { - getEntryPointVersion, - isUserOperationVersion06, - isUserOperationVersion07 -} from "../../utils/getEntryPointVersion" -import { isSmartAccountDeployed } from "../../utils/isSmartAccountDeployed" -import { toSmartAccount } from "../toSmartAccount" -import { - SignTransactionNotSupportedBySmartAccount, - type SmartAccount, - type SmartAccountSigner -} from "../types" +import { encode7579Calls } from "../../utils/encode7579Calls" export type SafeVersion = "1.4.1" @@ -422,7 +420,7 @@ const EIP712_SAFE_OPERATION_TYPE_V07 = { const SAFE_VERSION_TO_ADDRESSES_MAP: { [key in SafeVersion]: { - [key in EntryPointVersion]: { + [key in "0.6" | "0.7"]: { SAFE_MODULE_SETUP_ADDRESS: Address SAFE_4337_MODULE_ADDRESS: Address SAFE_PROXY_FACTORY_ADDRESS: Address @@ -433,7 +431,7 @@ const SAFE_VERSION_TO_ADDRESSES_MAP: { } } = { "1.4.1": { - "v0.6": { + "0.6": { SAFE_MODULE_SETUP_ADDRESS: "0x8EcD4ec46D4D2a6B64fE960B3D64e8B94B2234eb", SAFE_4337_MODULE_ADDRESS: @@ -446,7 +444,7 @@ const SAFE_VERSION_TO_ADDRESSES_MAP: { MULTI_SEND_CALL_ONLY_ADDRESS: "0x9641d764fc13c8B624c04430C7356C1C7C8102e2" }, - "v0.7": { + "0.7": { SAFE_MODULE_SETUP_ADDRESS: "0x2dd68b007B46fBe91B9A7c3EDa5A7a1063cB5b47", SAFE_4337_MODULE_ADDRESS: @@ -541,12 +539,6 @@ const encodeMultiSend = ( }) } -export type SafeSmartAccount< - entryPoint extends EntryPoint, - transport extends Transport = Transport, - chain extends Chain | undefined = Chain | undefined -> = SmartAccount - const get7579LaunchPadInitData = ({ safe4337ModuleAddress, safeSingletonAddress, @@ -761,7 +753,7 @@ const getInitializerCode = async ({ }) } -function getPaymasterAndData(unpackedUserOperation: UserOperation<"v0.7">) { +function getPaymasterAndData(unpackedUserOperation: UserOperation) { return unpackedUserOperation.paymaster ? concat([ unpackedUserOperation.paymaster, @@ -957,7 +949,7 @@ const getAccountAddress = async < const getDefaultAddresses = ( safeVersion: SafeVersion, - entryPointAddress: EntryPoint, + entryPointVersion: "0.6" | "0.7", { addModuleLibAddress: _addModuleLibAddress, safeModuleSetupAddress: _safeModuleSetupAddress, @@ -976,8 +968,6 @@ const getDefaultAddresses = ( multiSendCallOnlyAddress?: Address } ) => { - const entryPointVersion = getEntryPointVersion(entryPointAddress) - const safeModuleSetupAddress = _safeModuleSetupAddress ?? _addModuleLibAddress ?? @@ -1002,9 +992,8 @@ const getDefaultAddresses = ( const multiSendCallOnlyAddress = _multiSendCallOnlyAddress ?? - SAFE_VERSION_TO_ADDRESSES_MAP[safeVersion][ - getEntryPointVersion(entryPointAddress) - ].MULTI_SEND_CALL_ONLY_ADDRESS + SAFE_VERSION_TO_ADDRESSES_MAP[safeVersion][entryPointVersion] + .MULTI_SEND_CALL_ONLY_ADDRESS return { safeModuleSetupAddress, @@ -1041,15 +1030,26 @@ type GetErc7579Params = attestersThreshold?: number } -export type SignerToSafeSmartAccountParameters< - entryPoint extends EntryPoint, - TSource extends string = string, - TAddress extends Address = Address, +export type ToSafeSmartAccountParameters< + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address = typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7" = "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi, TErc7579 extends Address | undefined = Address | undefined > = { - signer: SmartAccountSigner + client: Client + owner: LocalAccount safeVersion: SafeVersion - entryPoint: entryPoint + entryPoint?: { + address: entryPointAddress + abi: entryPointAbi + version: entryPointVersion + } safe4337ModuleAddress?: Address erc7569LaunchpadAddress?: Address erc7579LaunchpadAddress?: TErc7579 @@ -1062,60 +1062,111 @@ export type SignerToSafeSmartAccountParameters< nonceKey?: bigint } & GetErc7579Params -function isErc7579Args( - args: SignerToSafeSmartAccountParameters< - EntryPoint, - string, - Address, +function isErc7579Args< + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address = typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7" = "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi +>( + args: ToSafeSmartAccountParameters< + entryPointAddress, + entryPointVersion, + entryPointAbi, Address | undefined > -): args is SignerToSafeSmartAccountParameters< - EntryPoint, - string, - Address, +): args is ToSafeSmartAccountParameters< + entryPointAddress, + entryPointVersion, + entryPointAbi, Address > { return args.erc7579LaunchpadAddress !== undefined } +export type SafeSmartAccountImplementation< + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi +> = Assign< + SmartAccountImplementation< + entryPointAbi, + entryPointVersion + // { + // // entryPoint === ENTRYPOINT_ADDRESS_V06 ? "0.2.2" : "0.3.0-beta" + // abi: entryPointVersion extends "0.6" ? typeof BiconomyAbi + // factory: { abi: typeof FactoryAbi; address: Address } + // } + >, + { sign: NonNullable } +> + +export type ToSafeSmartAccountReturnType< + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi +> = SmartAccount< + SafeSmartAccountImplementation +> + /** * @description Creates an Simple Account from a private key. * * @returns A Private Key Simple Account. */ -export async function signerToSafeSmartAccount< - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined, - TSource extends string = string, - TAddress extends Address = Address, +export async function toSafeSmartAccount< + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address = typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7" = "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi, TErc7579 extends Address | undefined = undefined >( - client: Client, - args: SignerToSafeSmartAccountParameters< - entryPoint, - TSource, - TAddress, + parameters: ToSafeSmartAccountParameters< + entryPointAddress, + entryPointVersion, + entryPointAbi, TErc7579 > -): Promise> { - const chainId = client.chain?.id ?? (await getChainId(client)) - +): Promise> { const { - signer, + client, + owner, address, safeVersion, - entryPoint: entryPointAddress, safe4337ModuleAddress: _safe4337ModuleAddress, safeProxyFactoryAddress: _safeProxyFactoryAddress, safeSingletonAddress: _safeSingletonAddress, - erc7579LaunchpadAddress = args.erc7579LaunchpadAddress, + erc7579LaunchpadAddress, saltNonce = BigInt(0), validUntil = 0, validAfter = 0, nonceKey - } = args + } = parameters + + const entryPoint = { + address: parameters.entryPoint?.address ?? entryPoint07Address, + abi: + parameters.entryPoint?.abi ?? + (parameters.entryPoint?.version ?? "0.7") === "0.6" + ? entryPoint06Abi + : entryPoint07Abi, + version: parameters.entryPoint?.version ?? "0.7" + } as const let _safeModuleSetupAddress: Address | undefined = undefined let _multiSendAddress: Address | undefined = undefined @@ -1133,30 +1184,23 @@ export async function signerToSafeSmartAccount< let attesters: Address[] = [] let attestersThreshold = 0 - if (!isErc7579Args(args)) { - _safeModuleSetupAddress = args.safeModuleSetupAddress - _multiSendAddress = args.multiSendAddress - _multiSendCallOnlyAddress = args.multiSendCallOnlyAddress - safeModules = args.safeModules - setupTransactions = args.setupTransactions ?? [] + if (!isErc7579Args(parameters)) { + _safeModuleSetupAddress = parameters.safeModuleSetupAddress + _multiSendAddress = parameters.multiSendAddress + _multiSendCallOnlyAddress = parameters.multiSendCallOnlyAddress + safeModules = parameters.safeModules + setupTransactions = parameters.setupTransactions ?? [] } - if (isErc7579Args(args)) { - validators = args.validators ?? [] - executors = args.executors ?? [] - fallbacks = args.fallbacks ?? [] - hooks = args.hooks ?? [] - attesters = args.attesters ?? [] - attestersThreshold = args.attestersThreshold ?? 0 + if (isErc7579Args(parameters)) { + validators = parameters.validators ?? [] + executors = parameters.executors ?? [] + fallbacks = parameters.fallbacks ?? [] + hooks = parameters.hooks ?? [] + attesters = parameters.attesters ?? [] + attestersThreshold = parameters.attestersThreshold ?? 0 } - const viemSigner: LocalAccount = { - ...signer, - signTransaction: (_, __) => { - throw new SignTransactionNotSupportedBySmartAccount() - } - } as LocalAccount - const { safeModuleSetupAddress, safe4337ModuleAddress, @@ -1164,7 +1208,7 @@ export async function signerToSafeSmartAccount< safeSingletonAddress, multiSendAddress, multiSendCallOnlyAddress - } = getDefaultAddresses(safeVersion, entryPointAddress, { + } = getDefaultAddresses(safeVersion, entryPoint.version, { safeModuleSetupAddress: _safeModuleSetupAddress, safe4337ModuleAddress: _safe4337ModuleAddress, safeProxyFactoryAddress: _safeProxyFactoryAddress, @@ -1173,212 +1217,142 @@ export async function signerToSafeSmartAccount< multiSendCallOnlyAddress: _multiSendCallOnlyAddress }) - const accountAddress = - address ?? - (await getAccountAddress({ - client, - owner: viemSigner.address, - safeModuleSetupAddress, - safe4337ModuleAddress, - safeProxyFactoryAddress, - safeSingletonAddress, - multiSendAddress, - erc7579LaunchpadAddress, - saltNonce, - setupTransactions, - safeModules, - validators, - executors, - fallbacks, - hooks, - attesters, - attestersThreshold - })) - - if (!accountAddress) throw new Error("Account address not found") + let accountAddress: Address + + const getAddress = async () => { + if (accountAddress) return accountAddress + accountAddress = + address ?? + (await getAccountAddress({ + client, + owner: owner.address, + safeModuleSetupAddress, + safe4337ModuleAddress, + safeProxyFactoryAddress, + safeSingletonAddress, + multiSendAddress, + erc7579LaunchpadAddress, + saltNonce, + setupTransactions, + safeModules, + validators, + executors, + fallbacks, + hooks, + attesters, + attestersThreshold + })) + return accountAddress + } - let safeDeployed = await isSmartAccountDeployed(client, accountAddress) + let chainId: number - const safeSmartAccount: SafeSmartAccount = - toSmartAccount({ - address: accountAddress, - client: client, - publicKey: accountAddress, - entryPoint: entryPointAddress, - source: "SafeSmartAccount", - async signMessage({ message }) { - const messageHash = hashTypedData({ - domain: { - chainId: chainId, - verifyingContract: accountAddress - }, - types: { - SafeMessage: [{ name: "message", type: "bytes" }] - }, - primaryType: "SafeMessage", - message: { - message: generateSafeMessageMessage(message) - } - }) + const getMemoizedChainId = async () => { + if (chainId) return chainId + chainId = client.chain + ? client.chain.id + : await getAction(client, getChainId, "getChainId")({}) + return chainId + } - return adjustVInSignature( - "eth_sign", - await signMessage(client, { - account: viemSigner, - message: { - raw: toBytes(messageHash) - } - }) - ) - }, - async signTransaction(_, __) { - throw new SignTransactionNotSupportedBySmartAccount() - }, - async signTypedData< - const TTypedData extends TypedData | Record, - TPrimaryType extends - | keyof TTypedData - | "EIP712Domain" = keyof TTypedData - >(typedData: TypedDataDefinition) { - return adjustVInSignature( - "eth_signTypedData", - await signTypedData(client, { - account: viemSigner, - domain: { - chainId: chainId, - verifyingContract: accountAddress - }, - types: { - SafeMessage: [{ name: "message", type: "bytes" }] - }, - primaryType: "SafeMessage", - message: { - message: generateSafeMessageMessage< - TTypedData, - TPrimaryType - >(typedData) - } + return toSmartAccount({ + client, + entryPoint, + getAddress, + async encodeCalls(calls) { + const hasMultipleCalls = calls.length > 1 + + if (erc7579LaunchpadAddress) { + const safeDeployed = await this.isDeployed() + + if (!safeDeployed) { + const initData = get7579LaunchPadInitData({ + safe4337ModuleAddress, + safeSingletonAddress, + erc7579LaunchpadAddress, + owner: owner.address, + validators, + executors, + fallbacks, + hooks, + attesters, + attestersThreshold }) - ) - }, - async getNonce(key?: bigint) { - return getAccountNonce(client, { - sender: accountAddress, - entryPoint: entryPointAddress, - key: key ?? nonceKey - }) - }, - async signUserOperation( - userOperation: UserOperation> - ) { - const message = { - safe: accountAddress, - callData: userOperation.callData, - nonce: userOperation.nonce, - initCode: userOperation.initCode ?? "0x", - maxFeePerGas: userOperation.maxFeePerGas, - maxPriorityFeePerGas: userOperation.maxPriorityFeePerGas, - preVerificationGas: userOperation.preVerificationGas, - verificationGasLimit: userOperation.verificationGasLimit, - callGasLimit: userOperation.callGasLimit, - paymasterAndData: userOperation.paymasterAndData ?? "0x", - validAfter: validAfter, - validUntil: validUntil, - entryPoint: entryPointAddress - } - - let isDeployed = false - if ( - isUserOperationVersion06(entryPointAddress, userOperation) - ) { - message.paymasterAndData = userOperation.paymasterAndData - isDeployed = userOperation.initCode === "0x" - } - - if ( - isUserOperationVersion07(entryPointAddress, userOperation) - ) { - if (userOperation.factory && userOperation.factoryData) { - message.initCode = concatHex([ - userOperation.factory, - userOperation.factoryData - ]) - } - message.paymasterAndData = - getPaymasterAndData(userOperation) - isDeployed = !userOperation.factory - } - - let verifyingContract = safe4337ModuleAddress - - if (erc7579LaunchpadAddress && !isDeployed) { - verifyingContract = userOperation.sender + return encodeFunctionData({ + abi: setupSafeAbi, + functionName: "setupSafe", + args: [ + { + ...initData, + validators: initData.validators.map( + (validator) => ({ + module: validator.address, + initData: validator.context + }) + ), + callData: encode7579Calls({ + mode: { + type: hasMultipleCalls + ? "batchcall" + : "call", + revertOnError: false, + selector: "0x", + context: "0x" + }, + callData: calls + }) + } + ] + }) } - const signatures = [ - { - signer: viemSigner.address, - data: await signTypedData(client, { - account: viemSigner, - domain: { - chainId, - verifyingContract - }, - types: - getEntryPointVersion(entryPointAddress) === - "v0.6" - ? EIP712_SAFE_OPERATION_TYPE_V06 - : EIP712_SAFE_OPERATION_TYPE_V07, - primaryType: "SafeOp", - message: message - }) - } - ] - - signatures.sort((left, right) => - left.signer - .toLowerCase() - .localeCompare(right.signer.toLowerCase()) - ) - - const signatureBytes = concat(signatures.map((sig) => sig.data)) + return encode7579Calls({ + mode: { + type: hasMultipleCalls ? "batchcall" : "call", + revertOnError: false, + selector: "0x", + context: "0x" + }, + callData: calls + }) + } - return encodePacked( - ["uint48", "uint48", "bytes"], - [validAfter, validUntil, signatureBytes] + let to: Address + let value: bigint + let data: Hex + let operationType = 0 + + if (hasMultipleCalls) { + to = multiSendCallOnlyAddress + value = BigInt(0) + + data = encodeMultiSend( + calls.map((tx) => ({ + to: tx.to, + value: tx.value ?? 0n, + data: tx.data ?? "0x", + operation: 0 + })) ) - }, - async getInitCode() { - safeDeployed = - safeDeployed || - (await isSmartAccountDeployed(client, accountAddress)) - - if (safeDeployed) return "0x" - - return concatHex([ - (await this.getFactory()) ?? "0x", - (await this.getFactoryData()) ?? "0x" - ]) - }, - async getFactory() { - safeDeployed = - safeDeployed || - (await isSmartAccountDeployed(client, accountAddress)) - - if (safeDeployed) return undefined - - return safeProxyFactoryAddress - }, - async getFactoryData() { - safeDeployed = - safeDeployed || - (await isSmartAccountDeployed(client, accountAddress)) - - if (safeDeployed) return undefined + operationType = 1 + } else { + to = calls[0].to + data = calls[0].data ?? "0x" + value = calls[0].value ?? 0n + } - return await getAccountInitCode({ - owner: viemSigner.address, + return encodeFunctionData({ + abi: executeUserOpWithErrorStringAbi, + functionName: "executeUserOpWithErrorString", + args: [to, value, data, operationType] + }) + }, + async getFactoryArgs() { + return { + factory: safeProxyFactoryAddress, + factoryData: await getAccountInitCode({ + owner: owner.address, safeModuleSetupAddress, safe4337ModuleAddress, safeSingletonAddress, @@ -1394,114 +1368,146 @@ export async function signerToSafeSmartAccount< attesters, attestersThreshold }) - }, - async encodeDeployCallData(_) { - throw new Error( - "Safe account doesn't support account deployment" - ) - }, - async encodeCallData(args) { - const isArray = Array.isArray(args) - - if (erc7579LaunchpadAddress) { - // First transaction will be slower because we need to enable 7579 modules - safeDeployed = - safeDeployed || - (await isSmartAccountDeployed(client, accountAddress)) - - if (!safeDeployed) { - const initData = get7579LaunchPadInitData({ - safe4337ModuleAddress, - safeSingletonAddress, - erc7579LaunchpadAddress, - owner: viemSigner.address, - validators, - executors, - fallbacks, - hooks, - attesters, - attestersThreshold - }) - - return encodeFunctionData({ - abi: setupSafeAbi, - functionName: "setupSafe", - args: [ - { - ...initData, - validators: initData.validators.map( - (validator) => ({ - module: validator.address, - initData: validator.context - }) - ), - callData: encode7579CallData({ - mode: { - type: isArray - ? "batchcall" - : "call", - revertOnError: false, - selector: "0x", - context: "0x" - }, - callData: args - }) - } - ] - }) + } + }, + async getNonce(args) { + return getAccountNonce(client, { + address: await getAddress(), + entryPointAddress: entryPoint.address, + key: args?.key ?? nonceKey + }) + }, + async getStubSignature() { + return "0x000000000000000000000000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff" + }, + async sign({ hash }) { + return this.signMessage({ message: hash }) + }, + async signMessage({ message }) { + const messageHash = hashTypedData({ + domain: { + chainId: chainId, + verifyingContract: accountAddress + }, + types: { + SafeMessage: [{ name: "message", type: "bytes" }] + }, + primaryType: "SafeMessage", + message: { + message: generateSafeMessageMessage(message) + } + }) + + return adjustVInSignature( + "eth_sign", + await signMessage(client, { + account: owner, + message: { + raw: toBytes(messageHash) } + }) + ) + }, + async signTypedData(typedData) { + return adjustVInSignature( + "eth_signTypedData", + await signTypedData(client, { + account: owner, + domain: { + chainId: chainId, + verifyingContract: accountAddress + }, + types: { + SafeMessage: [{ name: "message", type: "bytes" }] + }, + primaryType: "SafeMessage", + message: { + message: generateSafeMessageMessage(typedData) + } + }) + ) + }, + async signUserOperation(parameters) { + const { chainId = await getMemoizedChainId(), ...userOperation } = + parameters + + const message = { + safe: accountAddress, + callData: userOperation.callData, + nonce: userOperation.nonce, + initCode: userOperation.initCode ?? "0x", + maxFeePerGas: userOperation.maxFeePerGas, + maxPriorityFeePerGas: userOperation.maxPriorityFeePerGas, + preVerificationGas: userOperation.preVerificationGas, + verificationGasLimit: userOperation.verificationGasLimit, + callGasLimit: userOperation.callGasLimit, + paymasterAndData: userOperation.paymasterAndData ?? "0x", + validAfter: validAfter, + validUntil: validUntil, + entryPoint: entryPoint.address + } + + let isDeployed = false - return encode7579CallData({ - mode: { - type: isArray ? "batchcall" : "call", - revertOnError: false, - selector: "0x", - context: "0x" + if ("initCode" in userOperation) { + message.paymasterAndData = + userOperation.paymasterAndData ?? "0x" + isDeployed = userOperation.initCode === "0x" + } + + if ("factory" in userOperation) { + if (userOperation.factory && userOperation.factoryData) { + message.initCode = concatHex([ + userOperation.factory, + userOperation.factoryData + ]) + } + message.paymasterAndData = getPaymasterAndData({ + ...userOperation, + sender: userOperation.sender ?? (await getAddress()) + }) + isDeployed = !userOperation.factory + } + + let verifyingContract = safe4337ModuleAddress + + if (erc7579LaunchpadAddress && !isDeployed) { + verifyingContract = userOperation.sender ?? (await getAddress()) + } + + const signatures = [ + { + signer: owner.address, + data: await signTypedData(client, { + account: owner, + domain: { + chainId, + verifyingContract }, - callData: args + types: + entryPoint.version === "0.6" + ? EIP712_SAFE_OPERATION_TYPE_V06 + : EIP712_SAFE_OPERATION_TYPE_V07, + primaryType: "SafeOp", + message: message }) } + ] - let to: Address - let value: bigint - let data: Hex - let operationType = 0 - - if (isArray) { - const argsArray = args as { - to: Address - value: bigint - data: Hex - }[] - - to = multiSendCallOnlyAddress - value = BigInt(0) - - data = encodeMultiSend( - argsArray.map((tx) => ({ ...tx, operation: 0 })) - ) - operationType = 1 - } else { - const singleTransaction = args as { - to: Address - value: bigint - data: Hex - } - to = singleTransaction.to - data = singleTransaction.data - value = singleTransaction.value - } + signatures.sort((left, right) => + left.signer + .toLowerCase() + .localeCompare(right.signer.toLowerCase()) + ) - return encodeFunctionData({ - abi: executeUserOpWithErrorStringAbi, - functionName: "executeUserOpWithErrorString", - args: [to, value, data, operationType] - }) - }, - async getDummySignature(_userOperation) { - return "0x000000000000000000000000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff" - } - }) + const signatureBytes = concat(signatures.map((sig) => sig.data)) - return safeSmartAccount + return encodePacked( + ["uint48", "uint48", "bytes"], + [validAfter, validUntil, signatureBytes] + ) + } + }) as Promise< + ToSafeSmartAccountReturnType + > } diff --git a/packages/permissionless/accounts/simple/privateKeyToSimpleSmartAccount.ts b/packages/permissionless/accounts/simple/privateKeyToSimpleSmartAccount.ts deleted file mode 100644 index 43e9ec1f..00000000 --- a/packages/permissionless/accounts/simple/privateKeyToSimpleSmartAccount.ts +++ /dev/null @@ -1,48 +0,0 @@ -import type { Account, Address, Chain, Client, Hex, Transport } from "viem" -import { privateKeyToAccount } from "viem/accounts" -import type { EntryPoint, Prettify } from "../../types" -import { - type SignerToSimpleSmartAccountParameters, - type SimpleSmartAccount, - signerToSimpleSmartAccount -} from "./signerToSimpleSmartAccount" - -export type PrivateKeyToSimpleSmartAccountParameters< - entryPoint extends EntryPoint -> = Prettify< - { - privateKey: Hex - } & Omit, "signer"> -> - -/** - * @description Creates an Simple Account from a private key. - * - * @returns A Private Key Simple Account. - */ -export async function privateKeyToSimpleSmartAccount< - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined ->( - client: Client, - { - privateKey, - ...rest - }: PrivateKeyToSimpleSmartAccountParameters -): Promise> { - const privateKeyAccount = privateKeyToAccount(privateKey) - - return signerToSimpleSmartAccount< - entryPoint, - TTransport, - TChain, - TClientAccount, - "privateKey", - Address - >(client, { - signer: privateKeyAccount, - ...rest - }) -} diff --git a/packages/permissionless/accounts/simple/signerToSimpleSmartAccount.ts b/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts similarity index 54% rename from packages/permissionless/accounts/simple/signerToSimpleSmartAccount.ts rename to packages/permissionless/accounts/simple/toSimpleSmartAccount.ts index 0a9a8333..bc1fcb37 100644 --- a/packages/permissionless/accounts/simple/signerToSimpleSmartAccount.ts +++ b/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts @@ -1,38 +1,28 @@ import { - type Account, type Address, - type Chain, + type Assign, type Client, type Hex, type LocalAccount, - type Transport, concatHex, encodeFunctionData } from "viem" +import { + type SmartAccount, + type SmartAccountImplementation, + type UserOperation, + entryPoint06Abi, + type entryPoint06Address, + entryPoint07Abi, + entryPoint07Address, + getUserOperationHash, + toSmartAccount +} from "viem/account-abstraction" import { getChainId, signMessage } from "viem/actions" +import { getAction } from "viem/utils" import { getAccountNonce } from "../../actions/public/getAccountNonce" import { getSenderAddress } from "../../actions/public/getSenderAddress" -import type { - ENTRYPOINT_ADDRESS_V06_TYPE, - ENTRYPOINT_ADDRESS_V07_TYPE, - Prettify -} from "../../types" -import type { EntryPoint } from "../../types/entrypoint" import { getEntryPointVersion } from "../../utils" -import { getUserOperationHash } from "../../utils/getUserOperationHash" -import { isSmartAccountDeployed } from "../../utils/isSmartAccountDeployed" -import { toSmartAccount } from "../toSmartAccount" -import { - SignTransactionNotSupportedBySmartAccount, - type SmartAccount, - type SmartAccountSigner -} from "../types" - -export type SimpleSmartAccount< - entryPoint extends EntryPoint, - transport extends Transport = Transport, - chain extends Chain | undefined = Chain | undefined -> = SmartAccount const getAccountInitCode = async ( owner: Address, @@ -73,21 +63,20 @@ const getAccountInitCode = async ( } const getAccountAddress = async < - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address >({ client, factoryAddress, - entryPoint: entryPointAddress, + entryPointAddress, owner, index = BigInt(0) }: { - client: Client + client: Client factoryAddress: Address owner: Address - entryPoint: entryPoint + entryPointAddress: entryPointAddress index?: bigint }): Promise
=> { const entryPointVersion = getEntryPointVersion(entryPointAddress) @@ -95,173 +84,166 @@ const getAccountAddress = async < const factoryData = await getAccountInitCode(owner, index) if (entryPointVersion === "v0.6") { - return getSenderAddress(client, { + return getSenderAddress(client, { initCode: concatHex([factoryAddress, factoryData]), - entryPoint: entryPointAddress as ENTRYPOINT_ADDRESS_V06_TYPE + entryPointAddress: entryPointAddress as typeof entryPoint06Address }) } // Get the sender address based on the init code - return getSenderAddress(client, { + return getSenderAddress(client, { factory: factoryAddress, factoryData, - entryPoint: entryPointAddress as ENTRYPOINT_ADDRESS_V07_TYPE + entryPointAddress: entryPointAddress as typeof entryPoint07Address }) } -export type SignerToSimpleSmartAccountParameters< - entryPoint extends EntryPoint, - TSource extends string = string, - TAddress extends Address = Address -> = Prettify<{ - signer: SmartAccountSigner +export type ToSimpleSmartAccountParameters< + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address = typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7" = "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi +> = { + client: Client + owner: LocalAccount factoryAddress?: Address - entryPoint: entryPoint + entryPoint?: { + address: entryPointAddress + abi: entryPointAbi + version: entryPointVersion + } index?: bigint address?: Address nonceKey?: bigint -}> +} const getFactoryAddress = ( - entryPoint: EntryPoint, + entryPointVersion: "0.6" | "0.7", factoryAddress?: Address ): Address => { if (factoryAddress) return factoryAddress - if (getEntryPointVersion(entryPoint) === "v0.6") { + + if (entryPointVersion === "0.6") { return "0x9406Cc6185a346906296840746125a0E44976454" } return "0x91E60e0613810449d098b0b5Ec8b51A0FE8c8985" } +export type SimpleSmartAccountImplementation< + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi +> = Assign< + SmartAccountImplementation< + entryPointAbi, + entryPointVersion + // { + // // entryPoint === ENTRYPOINT_ADDRESS_V06 ? "0.2.2" : "0.3.0-beta" + // abi: entryPointVersion extends "0.6" ? typeof BiconomyAbi + // factory: { abi: typeof FactoryAbi; address: Address } + // } + >, + { sign: NonNullable } +> + +export type ToSimpleSmartAccountReturnType< + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi +> = SmartAccount< + SimpleSmartAccountImplementation +> + /** * @description Creates an Simple Account from a private key. * * @returns A Private Key Simple Account. */ -export async function signerToSimpleSmartAccount< - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined, - TSource extends string = string, - TAddress extends Address = Address +export async function toSimpleSmartAccount< + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address = typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7" = "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi >( - client: Client, - { - signer, + parameters: ToSimpleSmartAccountParameters< + entryPointAddress, + entryPointVersion, + entryPointAbi + > +): Promise> { + const { + client, + owner, factoryAddress: _factoryAddress, entryPoint: entryPointAddress, index = BigInt(0), address, nonceKey - }: SignerToSimpleSmartAccountParameters -): Promise> { - const viemSigner: LocalAccount = { - ...signer, - signTransaction: (_, __) => { - throw new SignTransactionNotSupportedBySmartAccount() - } - } as LocalAccount + } = parameters - const factoryAddress = getFactoryAddress(entryPointAddress, _factoryAddress) + const entryPoint = { + address: parameters.entryPoint?.address ?? entryPoint07Address, + abi: + parameters.entryPoint?.abi ?? + (parameters.entryPoint?.version ?? "0.7") === "0.6" + ? entryPoint06Abi + : entryPoint07Abi, + version: parameters.entryPoint?.version ?? "0.7" + } as const - const [accountAddress, chainId] = await Promise.all([ - address ?? - getAccountAddress({ - client, - factoryAddress, - entryPoint: entryPointAddress, - owner: viemSigner.address, - index - }), - client.chain?.id ?? getChainId(client) - ]) - - if (!accountAddress) throw new Error("Account address not found") - - let smartAccountDeployed = await isSmartAccountDeployed( - client, - accountAddress + const factoryAddress = getFactoryAddress( + entryPoint.version, + _factoryAddress ) - return toSmartAccount({ - address: accountAddress, - signMessage: async (_) => { - throw new Error("Simple account isn't 1271 compliant") - }, - signTransaction: (_, __) => { - throw new SignTransactionNotSupportedBySmartAccount() - }, - signTypedData: async (_) => { - throw new Error("Simple account isn't 1271 compliant") - }, - client: client, - publicKey: accountAddress, - entryPoint: entryPointAddress, - source: "SimpleSmartAccount", - async getNonce(key?: bigint) { - return getAccountNonce(client, { - sender: accountAddress, - entryPoint: entryPointAddress, - key: key ?? nonceKey - }) - }, - async signUserOperation(userOperation) { - return signMessage(client, { - account: viemSigner, - message: { - raw: getUserOperationHash({ - userOperation, - entryPoint: entryPointAddress, - chainId: chainId - }) - } - }) - }, - async getInitCode() { - if (smartAccountDeployed) return "0x" + let accountAddress: Address - smartAccountDeployed = await isSmartAccountDeployed( + const getAddress = async () => { + if (accountAddress) return accountAddress + accountAddress = + address ?? + (await getAccountAddress({ client, - accountAddress - ) + factoryAddress, + entryPointAddress: entryPoint.address, + owner: owner.address, + index + })) + return accountAddress + } - if (smartAccountDeployed) return "0x" + let chainId: number - return concatHex([ - factoryAddress, - await getAccountInitCode(viemSigner.address, index) - ]) - }, - async getFactory() { - if (smartAccountDeployed) return undefined - smartAccountDeployed = await isSmartAccountDeployed( - client, - accountAddress - ) - if (smartAccountDeployed) return undefined - return factoryAddress - }, - async getFactoryData() { - if (smartAccountDeployed) return undefined - smartAccountDeployed = await isSmartAccountDeployed( - client, - accountAddress - ) - if (smartAccountDeployed) return undefined - return getAccountInitCode(viemSigner.address, index) - }, - async encodeDeployCallData(_) { - throw new Error("Simple account doesn't support account deployment") - }, - async encodeCallData(args) { - if (Array.isArray(args)) { - const argsArray = args as { - to: Address - value: bigint - data: Hex - }[] + const getMemoizedChainId = async () => { + if (chainId) return chainId + chainId = client.chain + ? client.chain.id + : await getAction(client, getChainId, "getChainId")({}) + return chainId + } + return toSmartAccount({ + client, + entryPoint, + getAddress, + async encodeCalls(calls) { + if (calls.length > 1) { if (getEntryPointVersion(entryPointAddress) === "v0.6") { return encodeFunctionData({ abi: [ @@ -286,8 +268,8 @@ export async function signerToSimpleSmartAccount< ], functionName: "executeBatch", args: [ - argsArray.map((a) => a.to), - argsArray.map((a) => a.data) + calls.map((a) => a.to), + calls.map((a) => a.data ?? "0x") ] }) } @@ -319,19 +301,13 @@ export async function signerToSimpleSmartAccount< ], functionName: "executeBatch", args: [ - argsArray.map((a) => a.to), - argsArray.map((a) => a.value), - argsArray.map((a) => a.data) + calls.map((a) => a.to), + calls.map((a) => a.value ?? 0n), + calls.map((a) => a.data ?? "0x") ] }) } - const { to, value, data } = args as { - to: Address - value: bigint - data: Hex - } - return encodeFunctionData({ abi: [ { @@ -359,11 +335,55 @@ export async function signerToSimpleSmartAccount< } ], functionName: "execute", - args: [to, value, data] + args: [calls[0].to, calls[0].value ?? 0n, calls[0].data ?? "0x"] }) }, - async getDummySignature(_userOperation) { + async getFactoryArgs() { + return { + factory: factoryAddress, + factoryData: await getAccountInitCode(owner.address, index) + } + }, + async getNonce(args) { + return getAccountNonce(client, { + address: accountAddress, + entryPointAddress: entryPoint.address, + key: args?.key ?? nonceKey + }) + }, + async getStubSignature() { return "0xfffffffffffffffffffffffffffffff0000000000000000000000000000000007aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa1c" + }, + async sign({ hash }) { + return this.signMessage({ message: hash }) + }, + signMessage: async (_) => { + throw new Error("Simple account isn't 1271 compliant") + }, + signTypedData: async (_) => { + throw new Error("Simple account isn't 1271 compliant") + }, + async signUserOperation(parameters) { + const { chainId = await getMemoizedChainId(), ...userOperation } = + parameters + return signMessage(client, { + account: owner, + message: { + raw: getUserOperationHash({ + userOperation: { + ...userOperation, + sender: + userOperation.sender ?? (await getAddress()), + signature: "0x" + } as UserOperation, + entryPointAddress: entryPoint.address, + entryPointVersion: entryPoint.version, + chainId: chainId + }) + } + }) } - }) + }) as Promise< + ToSimpleSmartAccountReturnType + > } diff --git a/packages/permissionless/accounts/toSmartAccount.ts b/packages/permissionless/accounts/toSmartAccount.ts deleted file mode 100644 index 9d199174..00000000 --- a/packages/permissionless/accounts/toSmartAccount.ts +++ /dev/null @@ -1,176 +0,0 @@ -import { - type Abi, - type Account, - type Address, - type Chain, - type Client, - type CustomSource, - type EncodeDeployDataParameters, - type Hex, - type SignableMessage, - type Transport, - type TypedDataDefinition, - concat, - encodeAbiParameters -} from "viem" -import { toAccount } from "viem/accounts" -import type { UserOperation } from "../types" -import type { EntryPoint, GetEntryPointVersion } from "../types/entrypoint" -import { isSmartAccountDeployed } from "../utils" -import { - SignTransactionNotSupportedBySmartAccount, - type SmartAccount -} from "./types" - -const MAGIC_BYTES = - "0x6492649264926492649264926492649264926492649264926492649264926492" - -export function toSmartAccount< - TAccountSource extends CustomSource, - TEntryPoint extends EntryPoint, - TSource extends string = string, - transport extends Transport = Transport, - chain extends Chain | undefined = Chain | undefined, - clientAccount extends Account | undefined = Account | undefined, - TAbi extends Abi | readonly unknown[] = Abi ->({ - address, - client, - source, - entryPoint, - getNonce, - getInitCode, - getFactory, - getFactoryData, - encodeCallData, - getDummySignature, - encodeDeployCallData, - signUserOperation, - signMessage, - signTypedData -}: TAccountSource & { - source: TSource - client: Client - entryPoint: TEntryPoint - getNonce: (key?: bigint) => Promise - getInitCode: () => Promise - getFactory: () => Promise
- getFactoryData: () => Promise - encodeCallData: ( - args: - | { - to: Address - value: bigint - data: Hex - } - | { - to: Address - value: bigint - data: Hex - }[] - ) => Promise - getDummySignature( - userOperation: UserOperation> - ): Promise - encodeDeployCallData: ({ - abi, - args, - bytecode - }: EncodeDeployDataParameters) => Promise - signUserOperation: ( - userOperation: UserOperation> - ) => Promise -}): SmartAccount { - const account = toAccount({ - address: address, - signMessage: async ({ message }: { message: SignableMessage }) => { - const isDeployed = await isSmartAccountDeployed(client, address) - const signature = await signMessage({ message }) - - if (isDeployed) return signature - - const abiEncodedMessage = encodeAbiParameters( - [ - { - type: "address", - name: "create2Factory" - }, - { - type: "bytes", - name: "factoryCalldata" - }, - { - type: "bytes", - name: "originalERC1271Signature" - } - ], - [ - (await getFactory()) ?? "0x", // "0x should never happen if it's deployed" - (await getFactoryData()) ?? "0x", // "0x should never happen if it's deployed" - signature - ] - ) - - return concat([abiEncodedMessage, MAGIC_BYTES]) - }, - signTypedData: async (typedData) => { - const isDeployed = await isSmartAccountDeployed(client, address) - const signature = await signTypedData( - typedData as TypedDataDefinition - ) - - if (isDeployed) return signature - - const abiEncodedMessage = encodeAbiParameters( - [ - { - type: "address", - name: "create2Factory" - }, - { - type: "bytes", - name: "factoryCalldata" - }, - { - type: "bytes", - name: "originalERC1271Signature" - } - ], - [ - (await getFactory()) ?? "0x", // "0x should never happen if it's deployed" - (await getFactoryData()) ?? "0x", // "0x should never happen if it's deployed" - signature - ] - ) - - return concat([abiEncodedMessage, MAGIC_BYTES]) - }, - async signTransaction(_, __) { - throw new SignTransactionNotSupportedBySmartAccount() - } - }) - - return { - ...account, - source, - client, - type: "local", - entryPoint, - publicKey: address, - getNonce, - getInitCode, - getFactory, - getFactoryData, - encodeCallData, - getDummySignature, - encodeDeployCallData, - signUserOperation - } as SmartAccount< - TEntryPoint, - TSource, - transport, - chain, - clientAccount, - TAbi - > -} diff --git a/packages/permissionless/accounts/trust/privateKeyToTrustSmartAccount.ts b/packages/permissionless/accounts/trust/privateKeyToTrustSmartAccount.ts deleted file mode 100644 index 13ddc59c..00000000 --- a/packages/permissionless/accounts/trust/privateKeyToTrustSmartAccount.ts +++ /dev/null @@ -1,38 +0,0 @@ -import type { Account, Chain, Client, Hex, Transport } from "viem" -import { privateKeyToAccount } from "viem/accounts" -import type { ENTRYPOINT_ADDRESS_V06_TYPE, Prettify } from "../../types" -import { - type SignerToTrustSmartAccountParameters, - type TrustSmartAccount, - signerToTrustSmartAccount -} from "./signerToTrustSmartAccount" - -export type PrivateKeyToTrustSmartAccountParameters< - entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE -> = Prettify< - { - privateKey: Hex - } & Omit, "signer"> -> - -/** - * @description Creates an Trust Account from a private key. - * - * @returns A Private Key Trust Account. - */ -export async function privateKeyToTrustSmartAccount< - entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined ->( - client: Client, - { privateKey, ...rest }: PrivateKeyToTrustSmartAccountParameters -): Promise> { - const privateKeyAccount = privateKeyToAccount(privateKey) - - return signerToTrustSmartAccount(client, { - signer: privateKeyAccount, - ...rest - }) -} diff --git a/packages/permissionless/accounts/trust/signerToTrustSmartAccount.ts b/packages/permissionless/accounts/trust/signerToTrustSmartAccount.ts deleted file mode 100644 index 4869b6a3..00000000 --- a/packages/permissionless/accounts/trust/signerToTrustSmartAccount.ts +++ /dev/null @@ -1,236 +0,0 @@ -import { - type Account, - type Address, - type Chain, - type Client, - type Hex, - type LocalAccount, - type Transport, - type TypedData, - type TypedDataDefinition, - concatHex, - hashMessage, - hashTypedData -} from "viem" -import { getChainId } from "viem/actions" -import { getAccountNonce } from "../../actions/public/getAccountNonce" - -import { toSmartAccount } from "../toSmartAccount" -import { - SignTransactionNotSupportedBySmartAccount, - type SmartAccount, - type SmartAccountSigner -} from "../types" - -import type { ENTRYPOINT_ADDRESS_V06_TYPE } from "../../types" -import { isSmartAccountDeployed } from "../../utils/isSmartAccountDeployed" - -import { encodeCallData } from "./utils/encodeCallData" -import { getAccountAddress } from "./utils/getAccountAddress" -import { getDummySignature } from "./utils/getDummySignature" -import { getFactoryData } from "./utils/getFactoryData" -import { signTransaction } from "./utils/signTransaction" -import { signUserOperation } from "./utils/signUserOperation" - -async function _signTypedData< - TSource extends string = string, - TAddress extends Address = Address ->( - signer: SmartAccountSigner, - chainId: number, - accountAddress: Address, - hashedMessage: Hex -): Promise { - return signer.signTypedData({ - domain: { - chainId: Number(chainId), - name: "Barz", - verifyingContract: accountAddress, - version: "v0.2.0" - }, - types: { - BarzMessage: [{ name: "message", type: "bytes" }] - }, - message: { - message: hashedMessage - }, - primaryType: "BarzMessage" - }) -} - -/** - * Default addresses for Trust Smart Account - */ -export const TRUST_ADDRESSES: { - secp256k1VerificationFacetAddress: Address - factoryAddress: Address -} = { - secp256k1VerificationFacetAddress: - "0x81b9E3689390C7e74cF526594A105Dea21a8cdD5", - factoryAddress: "0x729c310186a57833f622630a16d13f710b83272a" -} - -export type TrustSmartAccount< - entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE, - transport extends Transport = Transport, - chain extends Chain | undefined = Chain | undefined -> = SmartAccount - -export type SignerToTrustSmartAccountParameters< - entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE, - TSource extends string = string, - TAddress extends Address = Address -> = { - signer: SmartAccountSigner - factoryAddress?: Address - entryPoint: entryPoint - index?: bigint - address?: Address - secp256k1VerificationFacetAddress?: Address - nonceKey?: bigint -} - -/** - * @description Creates an Trust Smart Account from a private key. - * - * @returns A Private Key Trust Smart Account. - */ -export async function signerToTrustSmartAccount< - entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined, - TSource extends string = string, - TAddress extends Address = Address ->( - client: Client, - { - signer, - factoryAddress = TRUST_ADDRESSES.factoryAddress, - entryPoint: entryPointAddress, - index = 0n, - secp256k1VerificationFacetAddress = TRUST_ADDRESSES.secp256k1VerificationFacetAddress, - address, - nonceKey - }: SignerToTrustSmartAccountParameters -): Promise> { - const viemSigner: LocalAccount = { - ...signer, - signTransaction: (_, __) => { - throw new SignTransactionNotSupportedBySmartAccount() - } - } - - const [accountAddress, chainId] = await Promise.all([ - address ?? - getAccountAddress(client, { - factoryAddress, - secp256k1VerificationFacetAddress, - entryPoint: entryPointAddress, - bytes: viemSigner.publicKey, - index - }), - client.chain?.id ?? getChainId(client) - ]) - - if (!accountAddress) throw new Error("Account address not found") - - let smartAccountDeployed = await isSmartAccountDeployed( - client, - accountAddress - ) - - return toSmartAccount({ - address: accountAddress, - client: client, - publicKey: accountAddress, - entryPoint: entryPointAddress, - source: "TrustSmartAccount", - signMessage: ({ message }) => { - return _signTypedData( - signer, - chainId, - accountAddress, - hashMessage(message) - ) - }, - signTransaction: signTransaction, - signTypedData< - const TTypedData extends TypedData | Record, - TPrimaryType extends - | keyof TTypedData - | "EIP712Domain" = keyof TTypedData - >(typedData: TypedDataDefinition) { - return _signTypedData( - signer, - chainId, - accountAddress, - hashTypedData(typedData) - ) - }, - getNonce: async (key?: bigint) => { - return getAccountNonce(client, { - sender: accountAddress, - entryPoint: entryPointAddress, - key: key ?? nonceKey - }) - }, - signUserOperation: async (userOperation) => { - return signUserOperation(client, { - account: viemSigner, - userOperation, - entryPoint: entryPointAddress, - chainId: chainId - }) - }, - getInitCode: async () => { - smartAccountDeployed = - smartAccountDeployed || - (await isSmartAccountDeployed(client, accountAddress)) - - if (smartAccountDeployed) { - return "0x" - } - - return concatHex([ - factoryAddress, - await getFactoryData({ - bytes: viemSigner.publicKey, - secp256k1VerificationFacetAddress, - index - }) - ]) - }, - async getFactory() { - smartAccountDeployed = - smartAccountDeployed || - (await isSmartAccountDeployed(client, accountAddress)) - - if (smartAccountDeployed) return undefined - - return factoryAddress - }, - async getFactoryData() { - smartAccountDeployed = - smartAccountDeployed || - (await isSmartAccountDeployed(client, accountAddress)) - - if (smartAccountDeployed) return undefined - - return getFactoryData({ - bytes: viemSigner.publicKey, - secp256k1VerificationFacetAddress, - index - }) - }, - async encodeDeployCallData(_) { - throw new Error("Trust account doesn't support account deployment") - }, - async encodeCallData(args) { - return encodeCallData({ args }) - }, - async getDummySignature(userOperation) { - return getDummySignature(userOperation) - } - }) -} diff --git a/packages/permissionless/accounts/trust/toTrustSmartAccount.ts b/packages/permissionless/accounts/trust/toTrustSmartAccount.ts new file mode 100644 index 00000000..6423fb97 --- /dev/null +++ b/packages/permissionless/accounts/trust/toTrustSmartAccount.ts @@ -0,0 +1,238 @@ +import { + type Address, + type Assign, + type Client, + type Hex, + type LocalAccount, + hashMessage, + hashTypedData +} from "viem" +import { getChainId, signMessage } from "viem/actions" +import { getAccountNonce } from "../../actions/public/getAccountNonce" + +import { + type SmartAccount, + type SmartAccountImplementation, + type UserOperation, + entryPoint06Abi, + entryPoint06Address, + type entryPoint07Abi, + getUserOperationHash, + toSmartAccount +} from "viem/account-abstraction" +import { getAction } from "viem/utils" +import { encodeCallData } from "./utils/encodeCallData" +import { getAccountAddress } from "./utils/getAccountAddress" +import { getFactoryData } from "./utils/getFactoryData" + +async function _signTypedData( + signer: LocalAccount, + chainId: number, + accountAddress: Address, + hashedMessage: Hex +): Promise { + return signer.signTypedData({ + domain: { + chainId: Number(chainId), + name: "Barz", + verifyingContract: accountAddress, + version: "v0.2.0" + }, + types: { + BarzMessage: [{ name: "message", type: "bytes" }] + }, + message: { + message: hashedMessage + }, + primaryType: "BarzMessage" + }) +} + +/** + * Default addresses for Trust Smart Account + */ +export const TRUST_ADDRESSES: { + secp256k1VerificationFacetAddress: Address + factoryAddress: Address +} = { + secp256k1VerificationFacetAddress: + "0x81b9E3689390C7e74cF526594A105Dea21a8cdD5", + factoryAddress: "0x729c310186a57833f622630a16d13f710b83272a" +} + +export type ToTrustSmartAccountParameters< + entryPointAddress extends + typeof entryPoint06Address = typeof entryPoint06Address, + entryPointVersion extends "0.6" = "0.6", + entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi +> = { + client: Client + owner: LocalAccount + factoryAddress?: Address + entryPoint?: { + address: entryPointAddress + abi: entryPointAbi + version: entryPointVersion + } + index?: bigint + address?: Address + secp256k1VerificationFacetAddress?: Address + nonceKey?: bigint +} + +export type TrustSmartAccountImplementation< + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi +> = Assign< + SmartAccountImplementation< + entryPointAbi, + entryPointVersion + // { + // // entryPoint === ENTRYPOINT_ADDRESS_V06 ? "0.2.2" : "0.3.0-beta" + // abi: entryPointVersion extends "0.6" ? typeof BiconomyAbi + // factory: { abi: typeof FactoryAbi; address: Address } + // } + >, + { sign: NonNullable } +> + +export type ToTrustSmartAccountReturnType< + entryPointVersion extends "0.6" = "0.6", + entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi +> = SmartAccount< + TrustSmartAccountImplementation +> + +/** + * @description Creates an Trust Smart Account from a private key. + * + * @returns A Private Key Trust Smart Account. + */ +export async function toTrustSmartAccount< + entryPointAddress extends + typeof entryPoint06Address = typeof entryPoint06Address, + entryPointVersion extends "0.6" = "0.6", + entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi +>( + parameters: ToTrustSmartAccountParameters< + entryPointAddress, + entryPointVersion, + entryPointAbi + > +): Promise> { + const { + owner, + client, + index = 0n, + address, + factoryAddress = TRUST_ADDRESSES.factoryAddress, + secp256k1VerificationFacetAddress = TRUST_ADDRESSES.secp256k1VerificationFacetAddress + } = parameters + + let accountAddress: Address + + const entryPoint = { + address: parameters.entryPoint?.address ?? entryPoint06Address, + abi: parameters.entryPoint?.abi ?? entryPoint06Abi, + version: parameters.entryPoint?.version ?? "0.6" + } as const + + const getAddress = async () => { + if (accountAddress) return accountAddress + accountAddress = + address ?? + (await getAccountAddress(client, { + factoryAddress, + secp256k1VerificationFacetAddress, + entryPoint: entryPoint.address, + bytes: owner.address, + index + })) + return accountAddress + } + + let chainId: number + + const getMemoizedChainId = async () => { + if (chainId) return chainId + chainId = client.chain + ? client.chain.id + : await getAction(client, getChainId, "getChainId")({}) + return chainId + } + + return toSmartAccount({ + client, + entryPoint, + getAddress, + async encodeCalls(calls) { + return encodeCallData(calls) + }, + async getFactoryArgs() { + return { + factory: factoryAddress, + factoryData: await getFactoryData({ + bytes: owner.address, + secp256k1VerificationFacetAddress, + index + }) + } + }, + async getNonce(args) { + return getAccountNonce(client, { + address: await getAddress(), + entryPointAddress: entryPoint.address, + key: args?.key ?? parameters?.nonceKey + }) + }, + async getStubSignature() { + return "0xfffffffffffffffffffffffffffffff0000000000000000000000000000000007aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa1c" + }, + async sign({ hash }) { + return this.signMessage({ message: hash }) + }, + async signMessage({ message }) { + return _signTypedData( + owner, + await getMemoizedChainId(), + await getAddress(), + hashMessage(message) + ) + }, + async signTypedData(typedData) { + return _signTypedData( + owner, + await getMemoizedChainId(), + await getAddress(), + hashTypedData(typedData) + ) + }, + async signUserOperation(parameters) { + const { chainId = await getMemoizedChainId(), ...userOperation } = + parameters + + return signMessage(client, { + account: owner, + message: { + raw: getUserOperationHash({ + userOperation: { + ...userOperation, + sender: + userOperation.sender ?? (await getAddress()), + signature: "0x" + } as UserOperation, + entryPointAddress: entryPoint.address, + entryPointVersion: entryPoint.version, + chainId: chainId + }) + } + }) + } + }) as Promise< + ToTrustSmartAccountReturnType + > +} diff --git a/packages/permissionless/accounts/trust/utils/encodeCallData.ts b/packages/permissionless/accounts/trust/utils/encodeCallData.ts index 54fa4690..c5ee95c7 100644 --- a/packages/permissionless/accounts/trust/utils/encodeCallData.ts +++ b/packages/permissionless/accounts/trust/utils/encodeCallData.ts @@ -1,27 +1,13 @@ -import { type Address, type Hex, encodeFunctionData } from "viem" - -export const encodeCallData = async ({ - args -}: { - args: - | { - to: `0x${string}` - value: bigint - data: `0x${string}` - } - | { - to: `0x${string}` - value: bigint - data: `0x${string}` - }[] -}) => { - if (Array.isArray(args)) { - const argsArray = args as { - to: Address - value: bigint - data: Hex - }[] +import { encodeFunctionData } from "viem" +export const encodeCallData = async ( + calls: readonly { + to: `0x${string}` + value?: bigint | undefined + data?: `0x${string}` | undefined + }[] +) => { + if (calls.length > 1) { return encodeFunctionData({ abi: [ { @@ -50,19 +36,13 @@ export const encodeCallData = async ({ ], functionName: "executeBatch", args: [ - argsArray.map((a) => a.to), - argsArray.map((a) => a.value), - argsArray.map((a) => a.data) + calls.map((a) => a.to), + calls.map((a) => a.value ?? 0n), + calls.map((a) => a.data ?? "0x") ] }) } - const { to, value, data } = args as { - to: Address - value: bigint - data: Hex - } - return encodeFunctionData({ abi: [ { @@ -90,6 +70,6 @@ export const encodeCallData = async ({ } ], functionName: "execute", - args: [to, value, data] + args: [calls[0].to, calls[0].value ?? 0n, calls[0].data ?? "0x"] }) } diff --git a/packages/permissionless/accounts/trust/utils/getAccountAddress.ts b/packages/permissionless/accounts/trust/utils/getAccountAddress.ts index dcf054ae..96d3f17f 100644 --- a/packages/permissionless/accounts/trust/utils/getAccountAddress.ts +++ b/packages/permissionless/accounts/trust/utils/getAccountAddress.ts @@ -1,22 +1,13 @@ -import { - type Account, - type Address, - type Chain, - type Client, - type Transport, - concatHex -} from "viem" +import { type Address, type Client, concatHex } from "viem" +import type { entryPoint06Address } from "viem/account-abstraction" import { getSenderAddress } from "../../../actions/public/getSenderAddress" -import type { ENTRYPOINT_ADDRESS_V06_TYPE, EntryPoint } from "../../../types" import { getFactoryData } from "./getFactoryData" export const getAccountAddress = async < - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined + entryPointAddress extends + typeof entryPoint06Address = typeof entryPoint06Address >( - client: Client, + client: Client, { factoryAddress, entryPoint: entryPointAddress, @@ -26,7 +17,7 @@ export const getAccountAddress = async < }: { factoryAddress: Address bytes: `0x${string}` - entryPoint: entryPoint + entryPoint: entryPointAddress secp256k1VerificationFacetAddress: Address index?: bigint } @@ -39,6 +30,6 @@ export const getAccountAddress = async < return getSenderAddress(client, { initCode: concatHex([factoryAddress, factoryData]), - entryPoint: entryPointAddress as ENTRYPOINT_ADDRESS_V06_TYPE + entryPointAddress: entryPointAddress }) } diff --git a/packages/permissionless/accounts/trust/utils/getDummySignature.ts b/packages/permissionless/accounts/trust/utils/getDummySignature.ts deleted file mode 100644 index 5ddfd3f2..00000000 --- a/packages/permissionless/accounts/trust/utils/getDummySignature.ts +++ /dev/null @@ -1,12 +0,0 @@ -import type { Hex } from "viem" -import type { - EntryPoint, - GetEntryPointVersion, - UserOperation -} from "../../../types" - -export const getDummySignature = async ( - _userOperation: UserOperation> -) => { - return "0xfffffffffffffffffffffffffffffff0000000000000000000000000000000007aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa1c" as Hex -} diff --git a/packages/permissionless/accounts/trust/utils/signMessage.ts b/packages/permissionless/accounts/trust/utils/signMessage.ts deleted file mode 100644 index 57596437..00000000 --- a/packages/permissionless/accounts/trust/utils/signMessage.ts +++ /dev/null @@ -1,14 +0,0 @@ -import type { Account, Chain, Client, Transport } from "viem" -import type { SignMessageParameters } from "viem" -import { signMessage as viem_signMessage } from "viem/actions" - -export const signMessage = < - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined ->( - client: Client, - { account, message }: SignMessageParameters -) => { - return viem_signMessage(client, { account, message }) -} diff --git a/packages/permissionless/accounts/trust/utils/signTransaction.ts b/packages/permissionless/accounts/trust/utils/signTransaction.ts deleted file mode 100644 index 208d557c..00000000 --- a/packages/permissionless/accounts/trust/utils/signTransaction.ts +++ /dev/null @@ -1,15 +0,0 @@ -import type { SerializeTransactionFn, TransactionSerializable } from "viem" -import { SignTransactionNotSupportedBySmartAccount } from "../../types" - -export const signTransaction = < - serializer extends - SerializeTransactionFn = SerializeTransactionFn, - transaction extends Parameters[0] = Parameters[0] ->( - _transaction: transaction, - _args?: { - serializer?: serializer - } -) => { - throw new SignTransactionNotSupportedBySmartAccount() -} diff --git a/packages/permissionless/accounts/trust/utils/signUserOperation.ts b/packages/permissionless/accounts/trust/utils/signUserOperation.ts deleted file mode 100644 index 22b5f32b..00000000 --- a/packages/permissionless/accounts/trust/utils/signUserOperation.ts +++ /dev/null @@ -1,39 +0,0 @@ -import type { Account, Address, Chain, Client, Transport } from "viem" -import type { - EntryPoint, - GetEntryPointVersion, - UserOperation -} from "../../../types" -import { getUserOperationHash } from "../../../utils" -import { signMessage } from "./signMessage" - -export const signUserOperation = async < - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined ->( - client: Client, - { - account, - userOperation, - entryPoint: entryPointAddress, - chainId - }: { - account: Account | Address - userOperation: UserOperation> - entryPoint: entryPoint - chainId: number - } -) => { - return signMessage(client, { - account: account, - message: { - raw: getUserOperationHash({ - userOperation, - entryPoint: entryPointAddress, - chainId: chainId - }) - } - }) -} diff --git a/packages/permissionless/accounts/types.ts b/packages/permissionless/accounts/types.ts index 8f1541e9..d722253e 100644 --- a/packages/permissionless/accounts/types.ts +++ b/packages/permissionless/accounts/types.ts @@ -1,19 +1,4 @@ -import { - type Abi, - type Address, - BaseError, - type Client, - type Hex, - type LocalAccount -} from "viem" -import type { - Account, - Chain, - EncodeDeployDataParameters, - Transport -} from "viem" -import type { UserOperation } from "../types" -import type { EntryPoint, GetEntryPointVersion } from "../types/entrypoint" +import { BaseError } from "viem" export class SignTransactionNotSupportedBySmartAccount extends BaseError { override name = "SignTransactionNotSupportedBySmartAccount" @@ -30,48 +15,3 @@ export class SignTransactionNotSupportedBySmartAccount extends BaseError { ) } } - -export type SmartAccount< - entryPoint extends EntryPoint, - TSource extends string = string, - transport extends Transport = Transport, - chain extends Chain | undefined = Chain | undefined, - clientAccount extends Account | undefined = Account | undefined, - TAbi extends Abi | readonly unknown[] = Abi -> = LocalAccount & { - client: Client - entryPoint: entryPoint - getNonce: (key?: bigint) => Promise - getInitCode: () => Promise - getFactory: () => Promise
- getFactoryData: () => Promise - encodeCallData: ( - args: - | { - to: Address - value: bigint - data: Hex - } - | { - to: Address - value: bigint - data: Hex - }[] - ) => Promise - getDummySignature( - userOperation: UserOperation> - ): Promise - encodeDeployCallData: ({ - abi, - args, - bytecode - }: EncodeDeployDataParameters) => Promise - signUserOperation: ( - userOperation: UserOperation> - ) => Promise -} - -export type SmartAccountSigner< - TSource extends string = string, - TAddress extends Address = Address -> = Omit, "signTransaction"> diff --git a/packages/permissionless/actions/bundler/chainId.test.ts b/packages/permissionless/actions/bundler/chainId.test.ts deleted file mode 100644 index 8d5ecf5e..00000000 --- a/packages/permissionless/actions/bundler/chainId.test.ts +++ /dev/null @@ -1,25 +0,0 @@ -import { http } from "viem" -import { foundry } from "viem/chains" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - type BundlerClient, - createBundlerClient -} from "../../clients/createBundlerClient" -import type { ENTRYPOINT_ADDRESS_V06_TYPE } from "../../types/entrypoint" -import { ENTRYPOINT_ADDRESS_V06 } from "../../utils" -import { chainId } from "./chainId" - -describe("chainId", () => { - testWithRpc("chainId", async ({ rpc }) => { - const { altoRpc } = rpc - const entryPoint = ENTRYPOINT_ADDRESS_V06 - - const bundlerClient = createBundlerClient({ - transport: http(altoRpc), - entryPoint - }) - const id = await chainId(bundlerClient) - expect(id).toBe(foundry.id) - }) -}) diff --git a/packages/permissionless/actions/bundler/chainId.ts b/packages/permissionless/actions/bundler/chainId.ts deleted file mode 100644 index c17c0b28..00000000 --- a/packages/permissionless/actions/bundler/chainId.ts +++ /dev/null @@ -1,42 +0,0 @@ -import type { Account, Chain, Client, Transport } from "viem" -import type { BundlerClient } from "../../clients/createBundlerClient" -import type { EntryPoint } from "../../types" -import type { BundlerRpcSchema } from "../../types/bundler" - -/** - * Returns the supported chain id by the bundler service - * - * - Docs: https://docs.pimlico.io/permissionless/reference/bundler-actions/chainId - * - * @param client {@link BundlerClient} that you created using viem's createClient and extended it with bundlerActions. - * @returns Supported chain id - * - * - * @example - * import { createClient } from "viem" - * import { chainId } from "permissionless/actions" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http(BUNDLER_URL) - * }) - * - * const bundlerChainId = chainId(bundlerClient) - * // Return 5n for Goerli - * - */ -export const chainId = async < - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends Account | undefined = Account | undefined ->( - client: Client> -) => { - return Number( - await client.request({ - method: "eth_chainId", - params: [] - }) - ) -} diff --git a/packages/permissionless/actions/bundler/estimateUserOperationGas.test.ts b/packages/permissionless/actions/bundler/estimateUserOperationGas.test.ts deleted file mode 100644 index 9329a493..00000000 --- a/packages/permissionless/actions/bundler/estimateUserOperationGas.test.ts +++ /dev/null @@ -1,205 +0,0 @@ -import { http, parseEther } from "viem" -import { generatePrivateKey } from "viem/accounts" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getPimlicoPaymasterClient, - getSimpleAccountClient -} from "../../../permissionless-test/src/utils" -import { createBundlerClient } from "../../clients/createBundlerClient" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" -import { estimateUserOperationGas } from "./estimateUserOperationGas" - -describe("eth_estimateUserOperationGas", () => { - testWithRpc("eth_estimateUserOperationGas_v06", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const bundlerClientV06 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V06 - }) - - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) - }) - - const userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: "0x5af0d9827e0c53e4799bb226655a1de152a425a5", - data: "0x", - value: 0n - }) - } - }) - - const { preVerificationGas, verificationGasLimit, callGasLimit } = - await estimateUserOperationGas( - bundlerClientV06, - { - userOperation, - entryPoint: ENTRYPOINT_ADDRESS_V06 - }, - { - "0xa5cc3c03994DB5b0d9A5eEdD10CabaB0813678AC": { - balance: parseEther("1"), - stateDiff: { - "0x3ea2f1d0abf3fc66cf29eebb70cbd4e7fe762ef8a09bcc06c8edf641230afec0": - "0x00000000000000000000000000000000000000000000000000000000000001a4" - } - } - } - ) - - expect(preVerificationGas).toBeTruthy() - expect(verificationGasLimit).toBeTruthy() - expect(callGasLimit).toBeTruthy() - }) - - testWithRpc("eth_estimateUserOperationGas_v07", async ({ rpc }) => { - const { anvilRpc, altoRpc } = rpc - const bundlerClientV07 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) - - const smartAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc - }) - - const userOperation = - await smartAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await smartAccountClient.account.encodeCallData({ - to: "0x5af0d9827e0c53e4799bb226655a1de152a425a5", - data: "0x", - value: 0n - }) - } - }) - - const { - preVerificationGas, - verificationGasLimit, - callGasLimit, - paymasterVerificationGasLimit, - paymasterPostOpGasLimit - } = await estimateUserOperationGas(bundlerClientV07, { - entryPoint: ENTRYPOINT_ADDRESS_V07, - userOperation - }) - - expect(preVerificationGas).toBeTruthy() - expect(verificationGasLimit).toBeTruthy() - expect(callGasLimit).toBeTruthy() - expect(paymasterVerificationGasLimit).toBe(0n) - expect(paymasterPostOpGasLimit).toBe(0n) - }) - - testWithRpc( - "eth_estimateUserOperationGas_V07_with_error", - async ({ rpc }) => { - const { anvilRpc, altoRpc } = rpc - const bundlerClientV07 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) - - const smartAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc - }) - - const userOperation = - await smartAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: - await smartAccountClient.account.encodeCallData({ - to: "0x5af0d9827e0c53e4799bb226655a1de152a425a5", - data: "0x", - value: 1000n - }) - } - }) - - await expect(() => - estimateUserOperationGas( - bundlerClientV07, - { - userOperation, - entryPoint: ENTRYPOINT_ADDRESS_V07 - }, - { - [smartAccountClient.account.address]: { - balance: 0n - } - } - ) - ).rejects.toThrowError(/AA21/) - } - ) - - testWithRpc( - "eth_estimateUserOperationGas_V07_withPaymaster", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - const bundlerClientV07 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) - - const smartAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) - }) - - const userOperation = - await smartAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: - await smartAccountClient.account.encodeCallData({ - to: "0x5af0d9827e0c53e4799bb226655a1de152a425a5", - data: "0x", - value: 0n - }) - } - }) - - const { - preVerificationGas, - verificationGasLimit, - callGasLimit, - paymasterVerificationGasLimit, - paymasterPostOpGasLimit - } = await estimateUserOperationGas(bundlerClientV07, { - userOperation, - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) - - expect(preVerificationGas).toBeTruthy() - expect(verificationGasLimit).toBeTruthy() - expect(callGasLimit).toBeTruthy() - expect(paymasterVerificationGasLimit).toBeTruthy() - expect(paymasterPostOpGasLimit).toBeTruthy() - } - ) -}) diff --git a/packages/permissionless/actions/bundler/estimateUserOperationGas.ts b/packages/permissionless/actions/bundler/estimateUserOperationGas.ts deleted file mode 100644 index 00f518be..00000000 --- a/packages/permissionless/actions/bundler/estimateUserOperationGas.ts +++ /dev/null @@ -1,146 +0,0 @@ -import type { Account, BaseError, Chain, Client, Hex, Transport } from "viem" -import type { PartialBy } from "viem/chains" -import type { BundlerClient } from "../../clients/createBundlerClient" -import type { Prettify } from "../../types/" -import type { BundlerRpcSchema, StateOverrides } from "../../types/bundler" -import type { EntryPoint, GetEntryPointVersion } from "../../types/entrypoint" -import type { UserOperation } from "../../types/userOperation" -import { getEntryPointVersion } from "../../utils" -import { deepHexlify } from "../../utils/deepHexlify" -import { - type GetEstimateUserOperationGasErrorReturnType, - getEstimateUserOperationGasError -} from "../../utils/errors/getEstimateUserOperationGasError" - -export type EstimateUserOperationGasParameters = - { - userOperation: GetEntryPointVersion extends "v0.6" - ? PartialBy< - UserOperation<"v0.6">, - "callGasLimit" | "preVerificationGas" | "verificationGasLimit" - > - : PartialBy< - UserOperation<"v0.7">, - | "callGasLimit" - | "preVerificationGas" - | "verificationGasLimit" - | "paymasterVerificationGasLimit" - | "paymasterPostOpGasLimit" - > - entryPoint: entryPoint - } - -export type EstimateUserOperationGasReturnType = - GetEntryPointVersion extends "v0.6" - ? { - preVerificationGas: bigint - verificationGasLimit: bigint - callGasLimit: bigint - } - : { - preVerificationGas: bigint - verificationGasLimit: bigint - callGasLimit: bigint - paymasterVerificationGasLimit?: bigint - paymasterPostOpGasLimit?: bigint - } - -export type EstimateUserOperationErrorType = - GetEstimateUserOperationGasErrorReturnType - -/** - * Estimates preVerificationGas, verificationGasLimit and callGasLimit for user operation - * - * - Docs: https://docs.pimlico.io/permissionless/reference/bundler-actions/estimateUserOperationGas - * - * @param client {@link BundlerClient} that you created using viem's createClient and extended it with bundlerActions. - * @param args {@link EstimateUserOperationGasParameters} - * @returns preVerificationGas, verificationGasLimit and callGasLimit as {@link EstimateUserOperationGasReturnType} - * - * - * @example - * import { createClient } from "viem" - * import { estimateUserOperationGas } from "permissionless/actions" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http(BUNDLER_URL) - * }) - * - * const gasParameters = estimateUserOperationGas(bundlerClient, { - * serOperation: signedUserOperation, - * entryPoint: entryPoint - * }) - * - * // Return {preVerificationGas: 43492n, verificationGasLimit: 59436n, callGasLimit: 9000n} - * - */ -export const estimateUserOperationGas = async < - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends Account | undefined = Account | undefined ->( - client: Client>, - args: Prettify>, - stateOverrides?: StateOverrides -): Promise> => { - const { userOperation, entryPoint } = args - - const userOperationWithBigIntAsHex = deepHexlify(userOperation) - const stateOverridesWithBigIntAsHex = deepHexlify(stateOverrides) - - try { - const response = await client.request({ - method: "eth_estimateUserOperationGas", - params: stateOverrides - ? [ - userOperationWithBigIntAsHex, - entryPoint, - stateOverridesWithBigIntAsHex - ] - : [userOperationWithBigIntAsHex, entryPoint] - }) - - const entryPointVersion = getEntryPointVersion(entryPoint) - - if (entryPointVersion === "v0.6") { - const responseV06 = response as { - preVerificationGas: Hex - verificationGasLimit: Hex - callGasLimit: Hex - } - - return { - preVerificationGas: BigInt(responseV06.preVerificationGas || 0), - verificationGasLimit: BigInt( - responseV06.verificationGasLimit || 0 - ), - callGasLimit: BigInt(responseV06.callGasLimit || 0) - } as EstimateUserOperationGasReturnType - } - - const responseV07 = response as { - preVerificationGas: Hex - verificationGasLimit: Hex - callGasLimit: Hex - paymasterVerificationGasLimit?: Hex - paymasterPostOpGasLimit?: Hex - } - - return { - preVerificationGas: BigInt(responseV07.preVerificationGas || 0), - verificationGasLimit: BigInt(responseV07.verificationGasLimit || 0), - callGasLimit: BigInt(responseV07.callGasLimit || 0), - paymasterVerificationGasLimit: - responseV07.paymasterVerificationGasLimit - ? BigInt(responseV07.paymasterVerificationGasLimit) - : undefined, - paymasterPostOpGasLimit: responseV07.paymasterPostOpGasLimit - ? BigInt(responseV07.paymasterPostOpGasLimit) - : undefined - } as EstimateUserOperationGasReturnType - } catch (err) { - throw getEstimateUserOperationGasError(err as BaseError, args) - } -} diff --git a/packages/permissionless/actions/bundler/getUserOperationByHash.test.ts b/packages/permissionless/actions/bundler/getUserOperationByHash.test.ts deleted file mode 100644 index f9693cd5..00000000 --- a/packages/permissionless/actions/bundler/getUserOperationByHash.test.ts +++ /dev/null @@ -1,139 +0,0 @@ -import { http, isHash, zeroAddress } from "viem" -import { generatePrivateKey } from "viem/accounts" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getPimlicoPaymasterClient, - getSimpleAccountClient -} from "../../../permissionless-test/src/utils" -import { createBundlerClient } from "../../clients/createBundlerClient" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" -import { getUserOperationByHash } from "./getUserOperationByHash" - -describe("getUserOperationByHash", () => { - testWithRpc("getUserOperationByHash_V06", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - const bundlerClientV06 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V06 - }) - - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) - }) - - const userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - } - }) - - userOperation.signature = - await simpleAccountClient.account.signUserOperation(userOperation) - - const opHash = await bundlerClientV06.sendUserOperation({ - userOperation - }) - - expect(isHash(opHash)).toBe(true) - - await bundlerClientV06.waitForUserOperationReceipt({ - hash: opHash, - timeout: 10000 - }) - - const userOperationFromUserOpHash = await getUserOperationByHash( - bundlerClientV06, - { hash: opHash } - ) - - expect(userOperationFromUserOpHash).not.toBeNull() - expect(userOperationFromUserOpHash?.entryPoint).toBe( - ENTRYPOINT_ADDRESS_V06 - ) - - for (const key in userOperationFromUserOpHash?.userOperation) { - const expected = userOperationFromUserOpHash?.userOperation[key] - const actual = userOperation[key] - - if (typeof expected === "string" && typeof actual === "string") { - expect(expected.toLowerCase()).toBe(actual.toLowerCase()) - } else { - expect(expected).toBe(actual) - } - } - }) - - testWithRpc("getUserOperationByHash_V07", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const bundlerClientV07 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) - - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) - }) - - const userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - } - }) - - userOperation.signature = - await simpleAccountClient.account.signUserOperation(userOperation) - - const opHash = await bundlerClientV07.sendUserOperation({ - userOperation - }) - - expect(isHash(opHash)).toBe(true) - - await bundlerClientV07.waitForUserOperationReceipt({ - hash: opHash, - timeout: 10000 - }) - - const userOperationFromUserOpHash = await getUserOperationByHash( - bundlerClientV07, - { hash: opHash } - ) - - for (const key in userOperationFromUserOpHash?.userOperation) { - const expected = userOperationFromUserOpHash?.userOperation[key] - const actual = userOperation[key] - - if (typeof expected === "string" && typeof actual === "string") { - expect(expected.toLowerCase()).toBe(actual.toLowerCase()) - } else { - expect(expected).toBe(actual) - } - } - }) -}) diff --git a/packages/permissionless/actions/bundler/getUserOperationByHash.ts b/packages/permissionless/actions/bundler/getUserOperationByHash.ts deleted file mode 100644 index 97c0ee4c..00000000 --- a/packages/permissionless/actions/bundler/getUserOperationByHash.ts +++ /dev/null @@ -1,110 +0,0 @@ -import type { Account, Address, Chain, Client, Hash, Transport } from "viem" -import type { BundlerClient } from "../../clients/createBundlerClient" -import type { Prettify } from "../../types/" -import type { BundlerRpcSchema } from "../../types/bundler" -import type { EntryPoint, GetEntryPointVersion } from "../../types/entrypoint" -import type { UserOperation } from "../../types/userOperation" -import { ENTRYPOINT_ADDRESS_V06 } from "../../utils/getEntryPointVersion" - -export type GetUserOperationByHashParameters = { - hash: Hash -} - -export type GetUserOperationByHashReturnType = { - userOperation: UserOperation> - entryPoint: Address - transactionHash: Hash - blockHash: Hash - blockNumber: bigint -} - -/** - * Returns the user operation from userOpHash - * - * - Docs: https://docs.pimlico.io/permissionless/reference/bundler-actions/getUserOperationByHash - * - * @param client {@link BundlerClient} that you created using viem's createClient and extended it with bundlerActions. - * @param args {@link GetUserOperationByHashParameters} UserOpHash that was returned by {@link sendUserOperation} - * @returns userOperation along with entryPoint, transactionHash, blockHash, blockNumber if found or null - * - * - * @example - * import { createClient } from "viem" - * import { getUserOperationByHash } from "permissionless/actions" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http(BUNDLER_URL) - * }) - * - * getUserOperationByHash(bundlerClient, {hash: userOpHash}) - * - */ -export const getUserOperationByHash = async < - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends Account | undefined = Account | undefined ->( - client: Client>, - { hash }: Prettify -): Promise> | null> => { - const params: [Hash] = [hash] - - const response = await client.request({ - method: "eth_getUserOperationByHash", - params - }) - - if (!response) return null - - const { - userOperation, - entryPoint: entryPointAddress, - transactionHash, - blockHash, - blockNumber - } = response - - return { - userOperation: (entryPointAddress === ENTRYPOINT_ADDRESS_V06 - ? { - ...userOperation, - nonce: BigInt(userOperation.nonce), - callGasLimit: BigInt(userOperation.callGasLimit), - verificationGasLimit: BigInt( - userOperation.verificationGasLimit - ), - preVerificationGas: BigInt(userOperation.preVerificationGas), - maxFeePerGas: BigInt(userOperation.maxFeePerGas), - maxPriorityFeePerGas: BigInt( - userOperation.maxPriorityFeePerGas - ) - } - : { - ...userOperation, - nonce: BigInt(userOperation.nonce), - callGasLimit: BigInt(userOperation.callGasLimit), - verificationGasLimit: BigInt( - userOperation.verificationGasLimit - ), - preVerificationGas: BigInt(userOperation.preVerificationGas), - maxFeePerGas: BigInt(userOperation.maxFeePerGas), - maxPriorityFeePerGas: BigInt( - userOperation.maxPriorityFeePerGas - ), - paymasterVerificationGasLimit: - userOperation.paymasterVerificationGasLimit - ? BigInt(userOperation.paymasterVerificationGasLimit) - : undefined, - paymasterPostOpGasLimit: - userOperation.paymasterVerificationGasLimit - ? BigInt(userOperation.paymasterPostOpGasLimit) - : undefined - }) as UserOperation>, - entryPoint: entryPointAddress, - transactionHash: transactionHash, - blockHash: blockHash, - blockNumber: BigInt(blockNumber) - } -} diff --git a/packages/permissionless/actions/bundler/getUserOperationReceipt.test.ts b/packages/permissionless/actions/bundler/getUserOperationReceipt.test.ts deleted file mode 100644 index e7693263..00000000 --- a/packages/permissionless/actions/bundler/getUserOperationReceipt.test.ts +++ /dev/null @@ -1,127 +0,0 @@ -import { http, isHash, zeroAddress } from "viem" -import { generatePrivateKey } from "viem/accounts" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getPimlicoPaymasterClient, - getSimpleAccountClient -} from "../../../permissionless-test/src/utils" -import { createBundlerClient } from "../../clients/createBundlerClient" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" -import { getUserOperationReceipt } from "./getUserOperationReceipt" - -describe("getUserOperationReceipt", () => { - testWithRpc("getUserOperationReceipt_V06", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const bundlerClientV06 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V06 - }) - - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) - }) - - const userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - } - }) - - userOperation.signature = - await simpleAccountClient.account.signUserOperation(userOperation) - - const opHash = await bundlerClientV06.sendUserOperation({ - userOperation - }) - - expect(isHash(opHash)).toBe(true) - - const userOperationReceipt = - await bundlerClientV06.waitForUserOperationReceipt({ - hash: opHash, - timeout: 100000 - }) - expect(userOperationReceipt).not.toBeNull() - expect(userOperationReceipt?.userOpHash).toBe(opHash) - expect(userOperationReceipt?.receipt.transactionHash).toBeTruthy() - - const receipt = await getUserOperationReceipt(bundlerClientV06, { - hash: opHash - }) - - expect(receipt?.receipt.transactionHash).toBe( - userOperationReceipt?.receipt.transactionHash - ) - }) - - testWithRpc("getUserOperationReceipt_V07", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const bundlerClientV07 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) - - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) - }) - - const userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - } - }) - - userOperation.signature = - await simpleAccountClient.account.signUserOperation(userOperation) - - const opHash = await bundlerClientV07.sendUserOperation({ - userOperation - }) - - expect(isHash(opHash)).toBe(true) - - const userOperationReceipt = - await bundlerClientV07.waitForUserOperationReceipt({ - hash: opHash, - timeout: 100000 - }) - expect(userOperationReceipt).not.toBeNull() - expect(userOperationReceipt?.userOpHash).toBe(opHash) - expect(userOperationReceipt?.receipt.transactionHash).toBeTruthy() - - const receipt = await getUserOperationReceipt(bundlerClientV07, { - hash: opHash - }) - - expect(receipt?.receipt.transactionHash).toBe( - userOperationReceipt?.receipt.transactionHash - ) - }) -}) diff --git a/packages/permissionless/actions/bundler/getUserOperationReceipt.ts b/packages/permissionless/actions/bundler/getUserOperationReceipt.ts deleted file mode 100644 index 476d32f9..00000000 --- a/packages/permissionless/actions/bundler/getUserOperationReceipt.ts +++ /dev/null @@ -1,127 +0,0 @@ -import type { - Account, - Address, - Chain, - Client, - Hash, - Hex, - Log, - Transport -} from "viem" -import type { BundlerClient } from "../../clients/createBundlerClient" -import type { Prettify } from "../../types" -import type { BundlerRpcSchema } from "../../types/bundler" -import type { EntryPoint } from "../../types/entrypoint" -import type { TStatus } from "../../types/userOperation" -import { transactionReceiptStatus } from "../../utils/deepHexlify" - -export type GetUserOperationReceiptParameters = { - hash: Hash -} - -export type GetUserOperationReceiptReturnType = { - userOpHash: Hash - entryPoint: Address - sender: Address - nonce: bigint - paymaster?: Address - actualGasUsed: bigint - actualGasCost: bigint - success: boolean - reason?: string - receipt: { - transactionHash: Hex - transactionIndex: bigint - blockHash: Hash - blockNumber: bigint - from: Address - to: Address | null - cumulativeGasUsed: bigint - status: TStatus - gasUsed: bigint - contractAddress: Address | null - logsBloom: Hex - effectiveGasPrice: bigint - } - logs: Log[] -} - -/** - * Returns the user operation receipt from userOpHash - * - * - Docs: https://docs.pimlico.io/permissionless/reference/bundler-actions/getUserOperationReceipt - * - * @param client {@link BundlerClient} that you created using viem's createClient and extended it with bundlerActions. - * @param args {@link GetUserOperationReceiptParameters} UserOpHash that was returned by {@link sendUserOperation} - * @returns user operation receipt {@link GetUserOperationReceiptReturnType} if found or null - * - * - * @example - * import { createClient } from "viem" - * import { getUserOperationReceipt } from "permissionless/actions" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http(BUNDLER_URL) - * }) - * - * getUserOperationReceipt(bundlerClient, {hash: userOpHash}) - * - */ -export const getUserOperationReceipt = async < - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends Account | undefined = Account | undefined ->( - client: Client>, - { hash }: Prettify -): Promise | null> => { - const params: [Hash] = [hash] - - const response = await client.request({ - method: "eth_getUserOperationReceipt", - params - }) - - if (!response) return null - - const userOperationReceipt: GetUserOperationReceiptReturnType = { - userOpHash: response.userOpHash, - entryPoint: response.entryPoint, - sender: response.sender, - nonce: BigInt(response.nonce), - paymaster: response.paymaster, - actualGasUsed: BigInt(response.actualGasUsed), - actualGasCost: BigInt(response.actualGasCost), - success: response.success, - reason: response.reason, - receipt: { - transactionHash: response.receipt.transactionHash, - transactionIndex: BigInt(response.receipt.transactionIndex), - blockHash: response.receipt.blockHash, - blockNumber: BigInt(response.receipt.blockNumber), - from: response.receipt.from, - to: response.receipt.to, - cumulativeGasUsed: BigInt(response.receipt.cumulativeGasUsed), - status: transactionReceiptStatus[response.receipt.status], - gasUsed: BigInt(response.receipt.gasUsed), - contractAddress: response.receipt.contractAddress, - logsBloom: response.receipt.logsBloom, - effectiveGasPrice: BigInt(response.receipt.effectiveGasPrice) - }, - logs: response.logs.map((log) => ({ - data: log.data, - blockNumber: BigInt(log.blockNumber), - blockHash: log.blockHash, - transactionHash: log.transactionHash, - logIndex: Number(log.logIndex), - transactionIndex: Number(log.transactionIndex), - address: log.address, - topics: log.topics, - removed: log.removed - })) - } - - return userOperationReceipt -} diff --git a/packages/permissionless/actions/bundler/sendUserOperation.test.ts b/packages/permissionless/actions/bundler/sendUserOperation.test.ts deleted file mode 100644 index e1085f95..00000000 --- a/packages/permissionless/actions/bundler/sendUserOperation.test.ts +++ /dev/null @@ -1,126 +0,0 @@ -import { http, isHash, zeroAddress } from "viem" -import { generatePrivateKey } from "viem/accounts" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getPimlicoPaymasterClient, - getSimpleAccountClient -} from "../../../permissionless-test/src/utils" -import { createBundlerClient } from "../../clients/createBundlerClient" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" -import { getUserOperationReceipt } from "./getUserOperationReceipt" - -describe("sendUserOperation", () => { - testWithRpc("sendUserOperation_V06", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const bundlerClientV06 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V06 - }) - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) - }) - - const userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - } - }) - - userOperation.signature = - await simpleAccountClient.account.signUserOperation(userOperation) - - const opHash = await bundlerClientV06.sendUserOperation({ - userOperation - }) - - expect(isHash(opHash)).toBe(true) - - const userOperationReceipt = - await bundlerClientV06.waitForUserOperationReceipt({ - hash: opHash, - timeout: 100000 - }) - expect(userOperationReceipt).not.toBeNull() - expect(userOperationReceipt?.userOpHash).toBe(opHash) - expect(userOperationReceipt?.receipt.transactionHash).toBeTruthy() - - const receipt = await getUserOperationReceipt(bundlerClientV06, { - hash: opHash - }) - - expect(receipt?.receipt.transactionHash).toBe( - userOperationReceipt?.receipt.transactionHash - ) - }) - - testWithRpc("sendUserOperation_V07", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const bundlerClientV07 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) - - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) - }) - - const userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - } - }) - - userOperation.signature = - await simpleAccountClient.account.signUserOperation(userOperation) - - const opHash = await bundlerClientV07.sendUserOperation({ - userOperation - }) - - expect(isHash(opHash)).toBe(true) - - const userOperationReceipt = - await bundlerClientV07.waitForUserOperationReceipt({ - hash: opHash, - timeout: 100000 - }) - expect(userOperationReceipt).not.toBeNull() - expect(userOperationReceipt?.userOpHash).toBe(opHash) - expect(userOperationReceipt?.receipt.transactionHash).toBeTruthy() - - const receipt = await getUserOperationReceipt(bundlerClientV07, { - hash: opHash - }) - - expect(receipt?.receipt.transactionHash).toBe( - userOperationReceipt?.receipt.transactionHash - ) - }) -}) diff --git a/packages/permissionless/actions/bundler/sendUserOperation.ts b/packages/permissionless/actions/bundler/sendUserOperation.ts deleted file mode 100644 index 04b60ffe..00000000 --- a/packages/permissionless/actions/bundler/sendUserOperation.ts +++ /dev/null @@ -1,72 +0,0 @@ -import type { Account, BaseError, Chain, Client, Hash, Transport } from "viem" -import type { BundlerClient } from "../../clients/createBundlerClient" -import type { Prettify } from "../../types/" -import type { BundlerRpcSchema } from "../../types/bundler" -import type { EntryPoint, GetEntryPointVersion } from "../../types/entrypoint" -import type { - UserOperation, - UserOperationWithBigIntAsHex -} from "../../types/userOperation" -import { deepHexlify } from "../../utils/deepHexlify" -import { getSendUserOperationError } from "../../utils/errors/getSendUserOperationError" - -export type SendUserOperationParameters = { - userOperation: UserOperation> - entryPoint: entryPoint -} - -/** - * Sends user operation to the bundler - * - * - Docs: https://docs.pimlico.io/permissionless/reference/bundler-actions/sendUserOperation - * - * @param client {@link BundlerClient} that you created using viem's createClient and extended it with bundlerActions. - * @param args {@link SendUserOperationParameters}. - * @returns UserOpHash that you can use to track user operation as {@link Hash}. - * - * @example - * import { createClient } from "viem" - * import { sendUserOperation } from "permissionless/actions" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http(BUNDLER_URL) - * }) - * - * const userOpHash = sendUserOperation(bundlerClient, { - * userOperation: signedUserOperation, - * entryPoint: entryPoint - * }) - * - * // Return '0xe9fad2cd67f9ca1d0b7a6513b2a42066784c8df938518da2b51bb8cc9a89ea34' - */ -export const sendUserOperation = async < - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends Account | undefined = Account | undefined ->( - client: Client>, - args: Prettify> -): Promise => { - const { userOperation, entryPoint } = args - - try { - const userOperationHash = await client.request({ - method: "eth_sendUserOperation", - params: [ - deepHexlify(userOperation) as UserOperationWithBigIntAsHex< - GetEntryPointVersion - >, - entryPoint - ] - }) - - return userOperationHash - } catch (err) { - throw getSendUserOperationError( - err as BaseError, - args as SendUserOperationParameters - ) - } -} diff --git a/packages/permissionless/actions/bundler/supportedEntryPoints.test.ts b/packages/permissionless/actions/bundler/supportedEntryPoints.test.ts deleted file mode 100644 index dca58f4d..00000000 --- a/packages/permissionless/actions/bundler/supportedEntryPoints.test.ts +++ /dev/null @@ -1,27 +0,0 @@ -import { http } from "viem" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { createBundlerClient } from "../../clients/createBundlerClient" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" -import { supportedEntryPoints } from "./supportedEntryPoints" - -describe("supportedEntryPoints", () => { - testWithRpc("supportedEntryPoints_V06", async ({ rpc }) => { - const bundlerClientV06 = createBundlerClient({ - transport: http(rpc.altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V06 - }) - - const entryPoints = await supportedEntryPoints(bundlerClientV06) - expect(entryPoints).contain(ENTRYPOINT_ADDRESS_V06) - }) - - testWithRpc("supportedEntryPoints_V07", async ({ rpc }) => { - const bundlerClientV07 = createBundlerClient({ - transport: http(rpc.altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) - const entryPoints = await supportedEntryPoints(bundlerClientV07) - expect(entryPoints).contain(ENTRYPOINT_ADDRESS_V07) - }) -}) diff --git a/packages/permissionless/actions/bundler/supportedEntryPoints.ts b/packages/permissionless/actions/bundler/supportedEntryPoints.ts deleted file mode 100644 index 208ade1f..00000000 --- a/packages/permissionless/actions/bundler/supportedEntryPoints.ts +++ /dev/null @@ -1,40 +0,0 @@ -import type { Account, Chain, Client, Transport } from "viem" -import type { BundlerClient } from "../../clients/createBundlerClient" -import type { BundlerRpcSchema } from "../../types/bundler" -import type { EntryPoint } from "../../types/entrypoint" - -/** - * Returns the supported entrypoints by the bundler service - * - * - Docs: https://docs.pimlico.io/permissionless/reference/bundler-actions/supportedEntryPoints - * - * @param client {@link BundlerClient} that you created using viem's createClient and extended it with bundlerActions. - * @returns Supported entryPoints - * - * - * @example - * import { createClient } from "viem" - * import { supportedEntryPoints } from "permissionless/actions" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http(BUNDLER_URL) - * }) - * - * const entryPointsSupported = supportedEntryPoints(bundlerClient) - * // Return ['0x5FF137D4b0FDCD49DcA30c7CF57E578a026d2789'] - * - */ -export const supportedEntryPoints = async < - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends Account | undefined = Account | undefined ->( - client: Client> -): Promise => { - return client.request({ - method: "eth_supportedEntryPoints", - params: [] - }) -} diff --git a/packages/permissionless/actions/bundler/waitForUserOperationReceipt.test.ts b/packages/permissionless/actions/bundler/waitForUserOperationReceipt.test.ts deleted file mode 100644 index df30228b..00000000 --- a/packages/permissionless/actions/bundler/waitForUserOperationReceipt.test.ts +++ /dev/null @@ -1,130 +0,0 @@ -import { http, isHash, zeroAddress } from "viem" -import { generatePrivateKey } from "viem/accounts" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getPimlicoPaymasterClient, - getSimpleAccountClient -} from "../../../permissionless-test/src/utils" -import { createBundlerClient } from "../../clients/createBundlerClient" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" -import { waitForUserOperationReceipt } from "./waitForUserOperationReceipt" - -describe("waitForUserOperationReceipt", () => { - testWithRpc("waitForUserOperationReceipt_V06", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const bundlerClientV06 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V06 - }) - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) - }) - - const userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - } - }) - - userOperation.signature = - await simpleAccountClient.account.signUserOperation(userOperation) - - const opHash = await bundlerClientV06.sendUserOperation({ - userOperation - }) - - expect(isHash(opHash)).toBe(true) - - const userOperationReceipt = await waitForUserOperationReceipt( - bundlerClientV06, - { - hash: opHash, - timeout: 100000 - } - ) - expect(userOperationReceipt).not.toBeNull() - expect(userOperationReceipt?.userOpHash).toBe(opHash) - expect(userOperationReceipt?.receipt.transactionHash).toBeTruthy() - - const receipt = await bundlerClientV06.getUserOperationReceipt({ - hash: opHash - }) - - expect(receipt?.receipt.transactionHash).toBe( - userOperationReceipt?.receipt.transactionHash - ) - }) - - testWithRpc("waitForUserOperationReceipt_V07", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const bundlerClientV07 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) - - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) - }) - - const userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - } - }) - - userOperation.signature = - await simpleAccountClient.account.signUserOperation(userOperation) - - const opHash = await bundlerClientV07.sendUserOperation({ - userOperation - }) - - expect(isHash(opHash)).toBe(true) - - const userOperationReceipt = await waitForUserOperationReceipt( - bundlerClientV07, - { - hash: opHash, - timeout: 100000 - } - ) - expect(userOperationReceipt).not.toBeNull() - expect(userOperationReceipt?.userOpHash).toBe(opHash) - expect(userOperationReceipt?.receipt.transactionHash).toBeTruthy() - - const receipt = await bundlerClientV07.getUserOperationReceipt({ - hash: opHash - }) - - expect(receipt?.receipt.transactionHash).toBe( - userOperationReceipt?.receipt.transactionHash - ) - }) -}) diff --git a/packages/permissionless/actions/bundler/waitForUserOperationReceipt.ts b/packages/permissionless/actions/bundler/waitForUserOperationReceipt.ts deleted file mode 100644 index 8ed2a875..00000000 --- a/packages/permissionless/actions/bundler/waitForUserOperationReceipt.ts +++ /dev/null @@ -1,130 +0,0 @@ -import { - type Account, - BaseError, - type Chain, - type Client, - type Hash, - type Transport, - stringify -} from "viem" -import { getAction } from "viem/utils" -import type { Prettify } from "../../types/" -import { observe } from "../../utils/observe" -import { - type GetUserOperationReceiptReturnType, - getUserOperationReceipt -} from "./getUserOperationReceipt" - -export class WaitForUserOperationReceiptTimeoutError extends BaseError { - override name = "WaitForUserOperationReceiptTimeoutError" - constructor({ hash }: { hash: Hash }) { - super( - `Timed out while waiting for user operation with hash "${hash}" to be confirmed.` - ) - } -} - -export type WaitForUserOperationReceiptParameters = { - /** The hash of the transaction. */ - hash: Hash - /** - * Polling frequency (in ms). Defaults to the client's pollingInterval config. - * @default client.pollingInterval - */ - pollingInterval?: number - /** Optional timeout (in milliseconds) to wait before stopping polling. */ - timeout?: number -} - -/** - * Waits for the User Operation to be included on a [Block](https://viem.sh/docs/glossary/terms.html#block) (one confirmation), and then returns the [User Operation Receipt](https://docs.pimlico.io/permissionless/reference/bundler-actions/getUserOperationReceipt). - * - * - Docs: https://docs.pimlico.io/permissionless/reference/bundler-actions/waitForUserOperationReceipt - * - * @param client - Bundler Client to use - * @param parameters - {@link WaitForUserOperationReceiptParameters} - * @returns The transaction receipt. {@link GetUserOperationReceiptReturnType} - * - * @example - * import { createBundlerClient, waitForUserOperationReceipt, http } from 'viem' - * import { mainnet } from 'viem/chains' - * - * const client = createBundlerClient({ - * chain: mainnet, - * transport: http(), - * }) - * const userOperationReceipt = await waitForUserOperationReceipt(client, { - * hash: '0x4ca7ee652d57678f26e887c149ab0735f41de37bcad58c9f6d3ed5824f15b74d', - * }) - */ -export const waitForUserOperationReceipt = < - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends Account | undefined = Account | undefined ->( - bundlerClient: Client, - { - hash, - pollingInterval = bundlerClient.pollingInterval, - timeout = bundlerClient.transport.timeout - }: Prettify -): Promise> => { - const observerId = stringify([ - "waitForUserOperationReceipt", - bundlerClient.uid, - hash - ]) - - let userOperationReceipt: GetUserOperationReceiptReturnType - - return new Promise((resolve, reject) => { - const unobserve = observe( - observerId, - { resolve, reject }, - async (emit) => { - let timeoutTimer: ReturnType - - const _removeInterval = setInterval(async () => { - const done = (fn: () => void) => { - clearInterval(_removeInterval) - fn() - unobserve() - if (timeout) clearTimeout(timeoutTimer) - } - try { - const _userOperationReceipt = await getAction( - bundlerClient, - getUserOperationReceipt, - "getUserOperationReceipt" - )({ hash }) - - if (_userOperationReceipt !== null) { - userOperationReceipt = _userOperationReceipt - } - - if (userOperationReceipt) { - done(() => emit.resolve(userOperationReceipt)) - return - } - } catch (err) { - done(() => emit.reject(err)) - return - } - }, pollingInterval) - - if (timeout) { - timeoutTimer = setTimeout(() => { - clearInterval(_removeInterval) - unobserve() - reject( - new WaitForUserOperationReceiptTimeoutError({ - hash - }) - ) - clearTimeout(timeoutTimer) - }, timeout) - } - } - ) - }) -} diff --git a/packages/permissionless/actions/erc7579/accountId.ts b/packages/permissionless/actions/erc7579/accountId.ts index ea681270..708d42ca 100644 --- a/packages/permissionless/actions/erc7579/accountId.ts +++ b/packages/permissionless/actions/erc7579/accountId.ts @@ -1,30 +1,22 @@ import { - type CallParameters, - type Chain, type Client, ContractFunctionExecutionError, - type Transport, decodeFunctionResult, encodeFunctionData } from "viem" +import type { + GetSmartAccountParameter, + SmartAccount +} from "viem/account-abstraction" import { call, readContract } from "viem/actions" import { getAction } from "viem/utils" -import type { SmartAccount } from "../../accounts/types" -import type { GetAccountParameter } from "../../types" -import type { EntryPoint } from "../../types/entrypoint" -import { parseAccount } from "../../utils/" import { AccountOrClientNotFoundError } from "../../utils/signUserOperationHashWithECDSA" export async function accountId< - TEntryPoint extends EntryPoint, - TTransport extends Transport, - TChain extends Chain | undefined, - TSmartAccount extends SmartAccount | undefined = - | SmartAccount - | undefined + TSmartAccount extends SmartAccount | undefined = SmartAccount | undefined >( - client: Client, - args?: GetAccountParameter + client: Client, + args?: GetSmartAccountParameter ): Promise { let account_ = client.account @@ -38,7 +30,7 @@ export async function accountId< }) } - const account = parseAccount(account_) as SmartAccount + const account = account_ as SmartAccount const publicClient = account.client @@ -65,12 +57,11 @@ export async function accountId< )({ abi, functionName: "accountId", - address: account.address + address: await account.getAddress() }) } catch (error) { if (error instanceof ContractFunctionExecutionError) { - const factory = await account.getFactory() - const factoryData = await account.getFactoryData() + const { factory, factoryData } = await account.getFactoryArgs() const result = await getAction( publicClient, @@ -84,7 +75,7 @@ export async function accountId< abi, functionName: "accountId" }) - } as unknown as CallParameters) + }) if (!result || !result.data) { throw new Error("accountId result is empty") diff --git a/packages/permissionless/actions/erc7579/installModule.ts b/packages/permissionless/actions/erc7579/installModule.ts index f61f47de..56302bc9 100644 --- a/packages/permissionless/actions/erc7579/installModule.ts +++ b/packages/permissionless/actions/erc7579/installModule.ts @@ -1,16 +1,15 @@ import { type Address, - type Chain, type Client, type Hex, - type Transport, encodeFunctionData, getAddress } from "viem" +import type { + GetSmartAccountParameter, + SmartAccount +} from "viem/account-abstraction" import { getAction } from "viem/utils" -import type { SmartAccount } from "../../accounts/types" -import type { GetAccountParameter, Prettify } from "../../types/" -import type { EntryPoint } from "../../types/entrypoint" import { parseAccount } from "../../utils/" import { AccountOrClientNotFoundError } from "../../utils/signUserOperationHashWithECDSA" import type { Middleware } from "../smartAccount/prepareUserOperationRequest" @@ -21,9 +20,8 @@ import { import { type ModuleType, parseModuleTypeId } from "./supportsModule" export type InstallModuleParameters< - TEntryPoint extends EntryPoint, - TSmartAccount extends SmartAccount | undefined -> = GetAccountParameter & { + TSmartAccount extends SmartAccount | undefined = SmartAccount | undefined +> = GetSmartAccountParameter & { type: ModuleType address: Address context: Hex @@ -33,15 +31,10 @@ export type InstallModuleParameters< } & Middleware export async function installModule< - TEntryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends SmartAccount | undefined = - | SmartAccount - | undefined + TSmartAccount extends SmartAccount | undefined = SmartAccount | undefined >( - client: Client, - parameters: Prettify> + client: Client, + parameters: Prettify> ): Promise { const { account: account_ = client.account, diff --git a/packages/permissionless/actions/public/getAccountNonce.ts b/packages/permissionless/actions/public/getAccountNonce.ts index 218f4299..97113c75 100644 --- a/packages/permissionless/actions/public/getAccountNonce.ts +++ b/packages/permissionless/actions/public/getAccountNonce.ts @@ -1,12 +1,14 @@ -import type { Address, Chain, Client, Transport } from "viem" +import type { Address, Client } from "viem" +import type { + entryPoint06Address, + entryPoint07Address +} from "viem/account-abstraction" import { readContract } from "viem/actions" import { getAction } from "viem/utils" -import type { Prettify } from "../../types/" -import type { EntryPoint } from "../../types/entrypoint" export type GetAccountNonceParams = { - sender: Address - entryPoint: EntryPoint + address: Address + entryPointAddress: typeof entryPoint06Address | typeof entryPoint07Address key?: bigint } @@ -36,21 +38,18 @@ export type GetAccountNonceParams = { * * // Return 0n */ -export const getAccountNonce = async < - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined ->( - client: Client, - args: Prettify +export const getAccountNonce = async ( + client: Client, + args: GetAccountNonceParams ): Promise => { - const { sender, entryPoint, key = BigInt(0) } = args + const { address, entryPointAddress, key = BigInt(0) } = args return await getAction( client, readContract, "readContract" )({ - address: entryPoint, + address: entryPointAddress, abi: [ { inputs: [ @@ -75,6 +74,6 @@ export const getAccountNonce = async < } ], functionName: "getNonce", - args: [sender, key] + args: [address, key] }) } diff --git a/packages/permissionless/actions/public/getSenderAddress.ts b/packages/permissionless/actions/public/getSenderAddress.ts index 4cca5a30..163e7d68 100644 --- a/packages/permissionless/actions/public/getSenderAddress.ts +++ b/packages/permissionless/actions/public/getSenderAddress.ts @@ -1,52 +1,51 @@ import { type Address, BaseError, - type Chain, type Client, type ContractFunctionExecutionErrorType, ContractFunctionRevertedError, type Hex, InvalidInputRpcError, + type Prettify, RpcRequestError, - type Transport, UnknownRpcError, concat, decodeErrorResult } from "viem" +import type { + entryPoint06Address, + entryPoint07Address +} from "viem/account-abstraction" import { simulateContract } from "viem/actions" import { getAction } from "viem/utils" -import type { Prettify } from "../../types/" -import type { - ENTRYPOINT_ADDRESS_V06_TYPE, - EntryPoint -} from "../../types/entrypoint" -export type GetSenderAddressParams = - entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE - ? { - initCode: Hex - entryPoint: entryPoint - factory?: never - factoryData?: never - } - : { - entryPoint: entryPoint - factory: Address - factoryData: Hex - initCode?: never - } +export type GetSenderAddressParams< + entryPoint extends typeof entryPoint06Address | typeof entryPoint07Address +> = entryPoint extends typeof entryPoint06Address + ? { + initCode: Hex + entryPointAddress: entryPoint + factory?: never + factoryData?: never + } + : { + entryPointAddress: entryPoint + factory: Address + factoryData: Hex + initCode?: never + } export class InvalidEntryPointError extends BaseError { override name = "InvalidEntryPointError" constructor({ cause, - entryPoint - }: { cause?: BaseError; entryPoint?: Address } = {}) { + entryPointAddress + }: { cause?: BaseError; entryPointAddress?: Address } = {}) { super( `The entry point address (\`entryPoint\`${ - entryPoint ? ` = ${entryPoint}` : "" + entryPointAddress ? ` = ${entryPointAddress}` : "" }) is not a valid entry point. getSenderAddress did not revert with a SenderAddressResult error.`, { cause @@ -81,14 +80,12 @@ export class InvalidEntryPointError extends BaseError { * // Return '0x7a88a206ba40b37a8c07a2b5688cf8b287318b63' */ export const getSenderAddress = async < - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined + entryPoint extends typeof entryPoint06Address | typeof entryPoint07Address >( - client: Client, + client: Client, args: Prettify> ): Promise
=> { - const { initCode, entryPoint, factory, factoryData } = args + const { initCode, entryPointAddress, factory, factoryData } = args if (!initCode && !factory && !factoryData) { throw new Error( @@ -102,7 +99,7 @@ export const getSenderAddress = async < simulateContract, "simulateContract" )({ - address: entryPoint, + address: entryPointAddress, abi: [ { inputs: [ @@ -224,12 +221,12 @@ export const getSenderAddress = async < if (revertError instanceof UnknownRpcError) { const parsedBody = JSON.parse( + // biome-ignore lint/suspicious/noExplicitAny: (revertError as unknown as any).cause.body ) const revertData = parsedBody.error.data const hexStringRegex = /0x[a-fA-F0-9]+/ - // biome-ignore lint/suspicious/noExplicitAny: const match = revertData.match(hexStringRegex) if (!match) { @@ -263,5 +260,5 @@ export const getSenderAddress = async < throw e } - throw new InvalidEntryPointError({ entryPoint }) + throw new InvalidEntryPointError({ entryPointAddress }) } diff --git a/packages/permissionless/clients/createBundlerClient.test.ts b/packages/permissionless/clients/createBundlerClient.test.ts deleted file mode 100644 index 42cf80ea..00000000 --- a/packages/permissionless/clients/createBundlerClient.test.ts +++ /dev/null @@ -1,26 +0,0 @@ -import { http } from "viem" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../permissionless-test/src/testWithRpc" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../utils" -import { createBundlerClient } from "./createBundlerClient" - -describe("createBundlerClient", () => { - testWithRpc("createBundlerClient_V06", async ({ rpc }) => { - const bundlerClientV06 = createBundlerClient({ - transport: http(rpc.altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V06 - }) - - const entryPoints = await bundlerClientV06.supportedEntryPoints() - expect(entryPoints).contain(ENTRYPOINT_ADDRESS_V06) - }) - - testWithRpc("createBundlerClient_V07", async ({ rpc }) => { - const bundlerClientV07 = createBundlerClient({ - transport: http(rpc.altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) - const entryPoints = await bundlerClientV07.supportedEntryPoints() - expect(entryPoints).contain(ENTRYPOINT_ADDRESS_V07) - }) -}) diff --git a/packages/permissionless/clients/createBundlerClient.ts b/packages/permissionless/clients/createBundlerClient.ts deleted file mode 100644 index 28fe4d48..00000000 --- a/packages/permissionless/clients/createBundlerClient.ts +++ /dev/null @@ -1,60 +0,0 @@ -import type { - Account, - Chain, - Client, - PublicClientConfig, - Transport -} from "viem" -import { createClient } from "viem" -import type { BundlerRpcSchema } from "../types/bundler" -import type { EntryPoint } from "../types/entrypoint" -import { type BundlerActions, bundlerActions } from "./decorators/bundler" - -export type BundlerClient< - entryPoint extends EntryPoint, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined -> = Client< - Transport, - TChain, - TClientAccount, - BundlerRpcSchema, - BundlerActions -> -/** - * Creates a EIP-4337 compliant Bundler Client with a given [Transport](https://viem.sh/docs/clients/intro.html) configured for a [Chain](https://viem.sh/docs/clients/chains.html). - * - * - Docs: https://docs.pimlico.io/permissionless/reference/clients/bundlerClient - * - * A Bundler Client is an interface to "erc 4337" [JSON-RPC API](https://eips.ethereum.org/EIPS/eip-4337#rpc-methods-eth-namespace) methods such as sending user operation, estimating gas for a user operation, get user operation receipt, etc through Bundler Actions. - * - * @param config - {@link PublicClientConfig} - * @returns A Bundler Client. {@link BundlerClient} - * - * @example - * import { createPublicClient, http } from 'viem' - * import { mainnet } from 'viem/chains' - * - * const bundlerClient = createBundlerClient({ - * chain: mainnet, - * transport: http(BUNDLER_URL), - * }) - */ -export const createBundlerClient = < - entryPoint extends EntryPoint, - transport extends Transport = Transport, - chain extends Chain | undefined = undefined ->( - parameters: PublicClientConfig & { - entryPoint: entryPoint - } -): BundlerClient => { - const { key = "public", name = "Bundler Client" } = parameters - const client = createClient({ - ...parameters, - key, - name, - type: "bundlerClient" - }) - return client.extend(bundlerActions(parameters.entryPoint)) -} diff --git a/packages/permissionless/package.json b/packages/permissionless/package.json index fb318ce1..91560671 100644 --- a/packages/permissionless/package.json +++ b/packages/permissionless/package.json @@ -86,6 +86,6 @@ } }, "peerDependencies": { - "viem": ">=2.14.1 <2.18.0" + "viem": "^2.18.0" } } diff --git a/packages/permissionless/types/entrypoint.ts b/packages/permissionless/types/entrypoint.ts deleted file mode 100644 index 32753e5e..00000000 --- a/packages/permissionless/types/entrypoint.ts +++ /dev/null @@ -1,13 +0,0 @@ -export type EntryPointVersion = "v0.6" | "v0.7" - -export type ENTRYPOINT_ADDRESS_V06_TYPE = - "0x5FF137D4b0FDCD49DcA30c7CF57E578a026d2789" -export type ENTRYPOINT_ADDRESS_V07_TYPE = - "0x0000000071727De22E5E9d8BAf0edAc6f37da032" - -export type GetEntryPointVersion = - entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE ? "v0.6" : "v0.7" - -export type EntryPoint = - | ENTRYPOINT_ADDRESS_V06_TYPE - | ENTRYPOINT_ADDRESS_V07_TYPE diff --git a/packages/permissionless/types/index.ts b/packages/permissionless/types/index.ts index 6ce78630..a0f68677 100644 --- a/packages/permissionless/types/index.ts +++ b/packages/permissionless/types/index.ts @@ -1,15 +1,8 @@ import type { Account, Chain, Client, Transport } from "viem" -import type { SmartAccount } from "../accounts/types" -import type { EntryPoint } from "./entrypoint" -import type { UserOperation } from "./userOperation" -export type { UserOperation } -export type { - EntryPointVersion, - ENTRYPOINT_ADDRESS_V06_TYPE, - ENTRYPOINT_ADDRESS_V07_TYPE, - GetEntryPointVersion, - EntryPoint -} from "./entrypoint" +import type { + SmartAccount, + SmartAccountImplementation +} from "viem/account-abstraction" export type { PackedUserOperation } from "./userOperation" @@ -23,25 +16,18 @@ export type GetAccountParameterWithClient< ? { account: Account; client?: Client } : { client: Client; account?: Account } -export type GetAccountParameter< - entryPoint extends EntryPoint, - TAccount extends SmartAccount | undefined -> = IsUndefined extends true - ? { - account: SmartAccount - } - : { - account?: SmartAccount - } - -export type Prettify = { - [K in keyof T]: T[K] -} & {} - export type PartialBy = Omit & Partial> export type PartialPick = Partial> +export type GetAccountParameter< + implementation extends + | SmartAccountImplementation + | undefined = SmartAccountImplementation +> = IsUndefined extends true + ? { account: SmartAccount } + : { account?: SmartAccount } + // biome-ignore lint/suspicious/noExplicitAny: generic type export type UnionOmit = T extends any ? Omit diff --git a/packages/permissionless/types/userOperation.ts b/packages/permissionless/types/userOperation.ts index abd644df..c5f3cedf 100644 --- a/packages/permissionless/types/userOperation.ts +++ b/packages/permissionless/types/userOperation.ts @@ -46,47 +46,6 @@ export type UserOperationWithBigIntAsHex< paymasterAndData?: never } -export type UserOperation = - entryPointVersion extends "v0.6" - ? { - sender: Address - nonce: bigint - initCode: Hex - callData: Hex - callGasLimit: bigint - verificationGasLimit: bigint - preVerificationGas: bigint - maxFeePerGas: bigint - maxPriorityFeePerGas: bigint - paymasterAndData: Hex - signature: Hex - factory?: never - factoryData?: never - paymaster?: never - paymasterVerificationGasLimit?: never - paymasterPostOpGasLimit?: never - paymasterData?: never - } - : { - sender: Address - nonce: bigint - factory?: Address - factoryData?: Hex - callData: Hex - callGasLimit: bigint - verificationGasLimit: bigint - preVerificationGas: bigint - maxFeePerGas: bigint - maxPriorityFeePerGas: bigint - paymaster?: Address - paymasterVerificationGasLimit?: bigint - paymasterPostOpGasLimit?: bigint - paymasterData?: Hex - signature: Hex - initCode?: never - paymasterAndData?: never - } - export type Hex32 = `0x${string & { length: 64 }}` export type PackedUserOperation = { diff --git a/packages/permissionless/utils/encode7579CallData.test.ts b/packages/permissionless/utils/encode7579Calls.test.ts similarity index 87% rename from packages/permissionless/utils/encode7579CallData.test.ts rename to packages/permissionless/utils/encode7579Calls.test.ts index 58b1fbf1..f47271ba 100644 --- a/packages/permissionless/utils/encode7579CallData.test.ts +++ b/packages/permissionless/utils/encode7579Calls.test.ts @@ -1,17 +1,19 @@ import { describe, expect, test } from "vitest" -import { encode7579CallData } from "./encode7579CallData" +import { encode7579Calls } from "./encode7579Calls" -describe("encode7579CallData", () => { +describe("encode7579Calls", () => { test("single call", async () => { - const callData = encode7579CallData({ + const callData = encode7579Calls({ mode: { type: "call" }, - callData: { - to: "0x1234567890123456789012345678901234567890", - value: BigInt(12345), - data: "0x1234567890123456789012345678901234567890" - } + callData: [ + { + to: "0x1234567890123456789012345678901234567890", + value: BigInt(12345), + data: "0x1234567890123456789012345678901234567890" + } + ] }) expect(callData).toBe( @@ -20,7 +22,7 @@ describe("encode7579CallData", () => { }) test("batch call", async () => { - const callData = encode7579CallData({ + const callData = encode7579Calls({ mode: { type: "batchcall" }, @@ -45,11 +47,10 @@ describe("encode7579CallData", () => { test("should throw error for batch call with different mode", async () => { expect(() => { - encode7579CallData({ + encode7579Calls({ mode: { type: "call" }, - // @ts-expect-error callData: [ { to: "0x1234567890123456789012345678901234567890", diff --git a/packages/permissionless/utils/encode7579CallData.ts b/packages/permissionless/utils/encode7579Calls.ts similarity index 76% rename from packages/permissionless/utils/encode7579CallData.ts rename to packages/permissionless/utils/encode7579Calls.ts index 8390bad2..bc551175 100644 --- a/packages/permissionless/utils/encode7579CallData.ts +++ b/packages/permissionless/utils/encode7579Calls.ts @@ -14,29 +14,17 @@ import { export type EncodeCallDataParams = { mode: ExecutionMode - callData: callType extends "batchcall" - ? { - to: Address - value: bigint - data: Hex - }[] - : { - to: Address - value: bigint - data: Hex - } + callData: readonly { + to: Address + value?: bigint | undefined + data?: Hex | undefined + }[] } -export function encode7579CallData({ +export function encode7579Calls({ mode, callData }: EncodeCallDataParams): Hex { - if (Array.isArray(callData) && mode?.type !== "batchcall") { - throw new Error( - `mode ${JSON.stringify(mode)} does not supported for batchcall calldata` - ) - } - const executeAbi = [ { type: "function", @@ -58,7 +46,7 @@ export function encode7579CallData({ } ] as const - if (Array.isArray(callData)) { + if (callData.length > 1) { return encodeFunctionData({ abi: executeAbi, functionName: "execute", @@ -89,8 +77,8 @@ export function encode7579CallData({ callData.map((arg) => { return { target: arg.to, - value: arg.value, - callData: arg.data + value: arg.value ?? 0n, + callData: arg.data ?? "0x" } }) ] @@ -105,9 +93,9 @@ export function encode7579CallData({ args: [ encodeExecutionMode(mode), concatHex([ - callData.to, - toHex(callData.value, { size: 32 }), - callData.data + callData[0].to, + toHex(callData[0].value ?? 0n, { size: 32 }), + callData[0].data ?? "0x" ]) ] }) diff --git a/packages/permissionless/utils/getUserOperationHash.ts b/packages/permissionless/utils/getUserOperationHash.ts deleted file mode 100644 index de3ea6e2..00000000 --- a/packages/permissionless/utils/getUserOperationHash.ts +++ /dev/null @@ -1,158 +0,0 @@ -import type { Address, Hash, Hex } from "viem" -import { concat, encodeAbiParameters, keccak256, pad, toHex } from "viem" -import type { EntryPoint, GetEntryPointVersion } from "../types" -import type { UserOperation } from "../types/userOperation" -import { isUserOperationVersion06 } from "./getEntryPointVersion" - -function packUserOp({ - userOperation, - entryPoint: entryPointAddress -}: { - userOperation: UserOperation> - entryPoint: entryPoint -}): Hex { - if (isUserOperationVersion06(entryPointAddress, userOperation)) { - const hashedInitCode = keccak256(userOperation.initCode) - const hashedCallData = keccak256(userOperation.callData) - const hashedPaymasterAndData = keccak256(userOperation.paymasterAndData) - - return encodeAbiParameters( - [ - { type: "address" }, - { type: "uint256" }, - { type: "bytes32" }, - { type: "bytes32" }, - { type: "uint256" }, - { type: "uint256" }, - { type: "uint256" }, - { type: "uint256" }, - { type: "uint256" }, - { type: "bytes32" } - ], - [ - userOperation.sender as Address, - userOperation.nonce, - hashedInitCode, - hashedCallData, - userOperation.callGasLimit, - userOperation.verificationGasLimit, - userOperation.preVerificationGas, - userOperation.maxFeePerGas, - userOperation.maxPriorityFeePerGas, - hashedPaymasterAndData - ] - ) - } - - const hashedInitCode = keccak256( - userOperation.factory && userOperation.factoryData - ? concat([userOperation.factory, userOperation.factoryData]) - : "0x" - ) - const hashedCallData = keccak256(userOperation.callData) - const hashedPaymasterAndData = keccak256( - userOperation.paymaster - ? concat([ - userOperation.paymaster, - pad( - toHex( - userOperation.paymasterVerificationGasLimit || - BigInt(0) - ), - { - size: 16 - } - ), - pad( - toHex(userOperation.paymasterPostOpGasLimit || BigInt(0)), - { - size: 16 - } - ), - userOperation.paymasterData || "0x" - ]) - : "0x" - ) - - return encodeAbiParameters( - [ - { type: "address" }, - { type: "uint256" }, - { type: "bytes32" }, - { type: "bytes32" }, - { type: "bytes32" }, - { type: "uint256" }, - { type: "bytes32" }, - { type: "bytes32" } - ], - [ - userOperation.sender as Address, - userOperation.nonce, - hashedInitCode, - hashedCallData, - concat([ - pad(toHex(userOperation.verificationGasLimit), { - size: 16 - }), - pad(toHex(userOperation.callGasLimit), { size: 16 }) - ]), - userOperation.preVerificationGas, - concat([ - pad(toHex(userOperation.maxPriorityFeePerGas), { - size: 16 - }), - pad(toHex(userOperation.maxFeePerGas), { size: 16 }) - ]), - hashedPaymasterAndData - ] - ) -} - -export type GetUserOperationHashParams = { - userOperation: UserOperation> - entryPoint: entryPoint - chainId: number -} - -/** - * - * Returns user operation hash that is a unique identifier of the user operation. - * - * - Docs: https://docs.pimlico.io/permissionless/reference/utils/getUserOperationHash - * - * @param args: userOperation, entryPoint, chainId as {@link GetUserOperationHashParams} - * @returns userOperationHash as {@link Hash} - * - * @example - * import { getUserOperationHash } from "permissionless/utils" - * - * const userOperationHash = getUserOperationHash({ - * userOperation, - * entryPoint, - * chainId - * }) - * - * // Returns "0xe9fad2cd67f9ca1d0b7a6513b2a42066784c8df938518da2b51bb8cc9a89ea34" - * - */ -export const getUserOperationHash = ({ - userOperation, - entryPoint: entryPointAddress, - chainId -}: GetUserOperationHashParams): Hash => { - const encoded = encodeAbiParameters( - [{ type: "bytes32" }, { type: "address" }, { type: "uint256" }], - [ - keccak256( - packUserOp({ - userOperation, - entryPoint: entryPointAddress - }) - ), - entryPointAddress, - BigInt(chainId) - ] - ) as `0x${string}` - - return keccak256(encoded) -} diff --git a/packages/permissionless/utils/isSmartAccountDeployed.ts b/packages/permissionless/utils/isSmartAccountDeployed.ts index b39a4c6c..f8d5ab8b 100644 --- a/packages/permissionless/utils/isSmartAccountDeployed.ts +++ b/packages/permissionless/utils/isSmartAccountDeployed.ts @@ -1,16 +1,13 @@ import type { Address, Client } from "viem" -import { getBytecode } from "viem/actions" +import { getCode } from "viem/actions" export const isSmartAccountDeployed = async ( client: Client, address: Address ): Promise => { - const contractCode = await getBytecode(client, { + const contractCode = await getCode(client, { address: address }) - if ((contractCode?.length ?? 0) > 2) { - return true - } - return false + return Boolean(contractCode) } From fdc419147ff6111c785f28f0b0fc232be78f5179 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Tue, 20 Aug 2024 13:19:08 +0100 Subject: [PATCH 05/51] Fix build --- .../biconomy/toBiconomySmartAccount.ts | 37 +- .../kernel/toEcdsaKernelSmartAccount.ts | 17 +- .../accounts/kernel/utils/getNonceKey.ts | 1 - packages/permissionless/actions/index.ts | 52 +- packages/permissionless/actions/pimlico.ts | 8 +- .../pimlico/getUserOperationGasPrice.ts | 18 +- .../actions/pimlico/getUserOperationStatus.ts | 23 +- .../pimlico/sendCompressedUserOperation.ts | 32 +- .../actions/pimlico/sponsorUserOperation.ts | 158 ++--- .../pimlico/validateSponsorshipPolicies.ts | 51 +- .../actions/smartAccount.test.ts | 198 ------ .../permissionless/actions/smartAccount.ts | 56 -- .../smartAccount/deployContract.test.ts | 118 ---- .../actions/smartAccount/deployContract.ts | 123 ---- .../prepareUserOperationRequest.test.ts | 520 --------------- .../prepareUserOperationRequest.ts | 444 ------------- .../smartAccount/sendTransaction.test.ts | 108 ---- .../actions/smartAccount/sendTransaction.ts | 151 ----- .../smartAccount/sendTransactions.test.ts | 126 ---- .../actions/smartAccount/sendTransactions.ts | 153 ----- .../smartAccount/sendUserOperation.test.ts | 133 ---- .../actions/smartAccount/sendUserOperation.ts | 98 --- .../actions/smartAccount/signMessage.test.ts | 172 ----- .../actions/smartAccount/signMessage.ts | 81 --- .../smartAccount/signTypedData.test.ts | 196 ------ .../actions/smartAccount/signTypedData.ts | 165 ----- .../smartAccount/writeContract.test.ts | 196 ------ .../actions/smartAccount/writeContract.ts | 161 ----- packages/permissionless/actions/stackup.ts | 17 - .../actions/stackup/accounts.ts | 41 -- .../actions/stackup/sponsorUserOperation.ts | 126 ---- .../clients/createSmartAccountClient.test.ts | 55 -- .../clients/createSmartAccountClient.ts | 121 ---- .../clients/decorators/bundler.ts | 253 -------- .../clients/decorators/pimlico.ts | 182 ++---- .../clients/decorators/smartAccount.test.ts | 55 -- .../clients/decorators/smartAccount.ts | 610 ------------------ .../clients/decorators/stackup.ts | 77 --- packages/permissionless/clients/pimlico.ts | 159 ++--- packages/permissionless/clients/stackup.ts | 63 -- packages/permissionless/errors/bundler.ts | 62 -- packages/permissionless/types/bundler.ts | 131 ---- packages/permissionless/types/index.ts | 10 - packages/permissionless/types/pimlico.ts | 67 +- packages/permissionless/types/stackup.ts | 63 -- .../permissionless/types/userOperation.ts | 61 -- .../utils/errors/getBundlerError.ts | 224 ------- .../getEstimateUserOperationGasError.ts | 40 -- .../utils/errors/getSendUserOperationError.ts | 26 - .../utils/getEntryPointVersion.ts | 31 - .../utils/getPackedUserOperation.ts | 20 +- .../utils/getRequiredPrefund.ts | 25 +- packages/permissionless/utils/index.ts | 24 +- packages/permissionless/utils/observe.ts | 74 --- .../utils/signUserOperationHashWithECDSA.ts | 134 ---- .../utils/walletClientToSmartAccountSigner.ts | 12 +- 56 files changed, 381 insertions(+), 5978 deletions(-) delete mode 100644 packages/permissionless/actions/smartAccount.test.ts delete mode 100644 packages/permissionless/actions/smartAccount.ts delete mode 100644 packages/permissionless/actions/smartAccount/deployContract.test.ts delete mode 100644 packages/permissionless/actions/smartAccount/deployContract.ts delete mode 100644 packages/permissionless/actions/smartAccount/prepareUserOperationRequest.test.ts delete mode 100644 packages/permissionless/actions/smartAccount/prepareUserOperationRequest.ts delete mode 100644 packages/permissionless/actions/smartAccount/sendTransaction.test.ts delete mode 100644 packages/permissionless/actions/smartAccount/sendTransaction.ts delete mode 100644 packages/permissionless/actions/smartAccount/sendTransactions.test.ts delete mode 100644 packages/permissionless/actions/smartAccount/sendTransactions.ts delete mode 100644 packages/permissionless/actions/smartAccount/sendUserOperation.test.ts delete mode 100644 packages/permissionless/actions/smartAccount/sendUserOperation.ts delete mode 100644 packages/permissionless/actions/smartAccount/signMessage.test.ts delete mode 100644 packages/permissionless/actions/smartAccount/signMessage.ts delete mode 100644 packages/permissionless/actions/smartAccount/signTypedData.test.ts delete mode 100644 packages/permissionless/actions/smartAccount/signTypedData.ts delete mode 100644 packages/permissionless/actions/smartAccount/writeContract.test.ts delete mode 100644 packages/permissionless/actions/smartAccount/writeContract.ts delete mode 100644 packages/permissionless/actions/stackup.ts delete mode 100644 packages/permissionless/actions/stackup/accounts.ts delete mode 100644 packages/permissionless/actions/stackup/sponsorUserOperation.ts delete mode 100644 packages/permissionless/clients/createSmartAccountClient.test.ts delete mode 100644 packages/permissionless/clients/createSmartAccountClient.ts delete mode 100644 packages/permissionless/clients/decorators/bundler.ts delete mode 100644 packages/permissionless/clients/decorators/smartAccount.test.ts delete mode 100644 packages/permissionless/clients/decorators/smartAccount.ts delete mode 100644 packages/permissionless/clients/decorators/stackup.ts delete mode 100644 packages/permissionless/clients/stackup.ts delete mode 100644 packages/permissionless/errors/bundler.ts delete mode 100644 packages/permissionless/types/bundler.ts delete mode 100644 packages/permissionless/types/stackup.ts delete mode 100644 packages/permissionless/types/userOperation.ts delete mode 100644 packages/permissionless/utils/errors/getBundlerError.ts delete mode 100644 packages/permissionless/utils/errors/getEstimateUserOperationGasError.ts delete mode 100644 packages/permissionless/utils/errors/getSendUserOperationError.ts delete mode 100644 packages/permissionless/utils/getEntryPointVersion.ts delete mode 100644 packages/permissionless/utils/observe.ts delete mode 100644 packages/permissionless/utils/signUserOperationHashWithECDSA.ts diff --git a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts index 5d64af92..cbaffd85 100644 --- a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts +++ b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts @@ -22,7 +22,6 @@ import { } from "viem/account-abstraction" import { signMessage } from "viem/actions" import { getAccountNonce } from "../../actions/public/getAccountNonce" -import { isSmartAccountDeployed } from "../../utils/isSmartAccountDeployed" import { BiconomyAbi, FactoryAbi } from "./abi/BiconomySmartAccountAbi" /** @@ -146,7 +145,7 @@ export type ToBiconomySmartAccountParameters< entryPoint?: { address: entryPointAddress version: "0.6" - abi?: entryPointAbi + abi: entryPointAbi } nonceKey?: bigint index?: bigint @@ -159,14 +158,7 @@ export type ToBiconomySmartAccountParameters< export type BiconomySmartAccountImplementation< entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi > = Assign< - SmartAccountImplementation< - entryPointAbi, - "0.6", - { - abi: typeof BiconomyAbi - factory: { abi: typeof FactoryAbi; address: Address } - } - >, + SmartAccountImplementation, { sign: NonNullable } > @@ -199,11 +191,17 @@ export async function toBiconomySmartAccount< const { owner, client, index = 0n, address } = parameters - const entryPoint = { - address: parameters.entryPoint?.address ?? entryPoint06Address, - abi: parameters.entryPoint?.abi ?? entryPoint06Abi, - version: "0.6" - } as const + const entryPoint = + parameters.entryPoint ?? + ({ + address: entryPoint06Address, + abi: entryPoint06Abi, + version: "0.6" + } as { + address: entryPointAddress + version: "0.6" + abi: entryPointAbi + }) const factoryAddress = parameters.factoryAddress ?? BICONOMY_ADDRESSES.FACTORY_ADDRESS @@ -241,13 +239,6 @@ export async function toBiconomySmartAccount< return toSmartAccount({ client, entryPoint, - extend: { - abi: BiconomyAbi, - factory: { - abi: FactoryAbi, - address: factoryAddress - } - }, getAddress, async getNonce(args) { const address = await getAddress() @@ -359,5 +350,5 @@ export async function toBiconomySmartAccount< ) return signatureWithModuleAddress } - }) as Promise> + }) } diff --git a/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts b/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts index 23eb7edf..662f2833 100644 --- a/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts +++ b/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts @@ -193,7 +193,7 @@ export const getEcdsaRootIdentifierForKernelV3 = ( * @param ecdsaValidatorAddress */ const getInitializationData = ({ - entryPoint: { version: entryPointVersion, address: entryPointAddress }, + entryPoint: { version: entryPointVersion }, kernelVersion, owner, ecdsaValidatorAddress @@ -201,7 +201,6 @@ const getInitializationData = ({ kernelVersion: KernelVersion entryPoint: { version: entryPointVersion - address: typeof entryPoint06Address | typeof entryPoint07Address } owner: Address ecdsaValidatorAddress: Address @@ -250,7 +249,7 @@ const getInitializationData = ({ * @param ecdsaValidatorAddress */ const getAccountInitCode = async ({ - entryPoint: { version: entryPointVersion, address: entryPointAddress }, + entryPointVersion, kernelVersion, owner, index, @@ -259,10 +258,7 @@ const getAccountInitCode = async ({ ecdsaValidatorAddress }: { kernelVersion: KernelVersion - entryPoint: { - version: entryPointVersion - address: typeof entryPoint06Address | typeof entryPoint07Address - } + entryPointVersion: entryPointVersion owner: Address index: bigint factoryAddress: Address @@ -271,7 +267,7 @@ const getAccountInitCode = async ({ }): Promise => { // Build the account initialization data const initializationData = getInitializationData({ - entryPoint: { version: entryPointVersion, address: entryPointAddress }, + entryPoint: { version: entryPointVersion }, kernelVersion, ecdsaValidatorAddress, owner @@ -461,10 +457,7 @@ export async function toEcdsaKernelSmartAccount< // Helper to generate the init code for the smart account const generateInitCode = () => getAccountInitCode({ - entryPoint: { - version: entryPoint.version, - address: entryPoint.address - }, + entryPointVersion: entryPoint.version, kernelVersion, owner: owner.address, index, diff --git a/packages/permissionless/accounts/kernel/utils/getNonceKey.ts b/packages/permissionless/accounts/kernel/utils/getNonceKey.ts index 089f6fc7..0af2b1a1 100644 --- a/packages/permissionless/accounts/kernel/utils/getNonceKey.ts +++ b/packages/permissionless/accounts/kernel/utils/getNonceKey.ts @@ -1,5 +1,4 @@ import { type Address, concatHex, maxUint16, pad, toHex } from "viem" -import type { EntryPoint } from "../../../types" import { VALIDATOR_MODE, VALIDATOR_TYPE } from "../constants" import type { KernelVersion } from "../toEcdsaKernelSmartAccount" import { isKernelV2 } from "./isKernelV2" diff --git a/packages/permissionless/actions/index.ts b/packages/permissionless/actions/index.ts index 3b20e772..0d2236be 100644 --- a/packages/permissionless/actions/index.ts +++ b/packages/permissionless/actions/index.ts @@ -1,60 +1,12 @@ -import type { - EstimateUserOperationErrorType, - EstimateUserOperationGasParameters, - EstimateUserOperationGasReturnType -} from "./bundler/estimateUserOperationGas" -import type { GetUserOperationByHashParameters } from "./bundler/getUserOperationByHash" -import type { GetUserOperationByHashReturnType } from "./bundler/getUserOperationByHash" -import type { - GetUserOperationReceiptParameters, - GetUserOperationReceiptReturnType -} from "./bundler/getUserOperationReceipt" -import type { SendUserOperationParameters } from "./bundler/sendUserOperation" - import type { GetSenderAddressParams } from "./public/getSenderAddress" import { InvalidEntryPointError, getSenderAddress } from "./public/getSenderAddress" -import { chainId } from "./bundler/chainId" -import { estimateUserOperationGas } from "./bundler/estimateUserOperationGas" -import { getUserOperationByHash } from "./bundler/getUserOperationByHash" -import { getUserOperationReceipt } from "./bundler/getUserOperationReceipt" -import { sendUserOperation } from "./bundler/sendUserOperation" -import { supportedEntryPoints } from "./bundler/supportedEntryPoints" -import { waitForUserOperationReceipt } from "./bundler/waitForUserOperationReceipt" -import { - type WaitForUserOperationReceiptParameters, - WaitForUserOperationReceiptTimeoutError -} from "./bundler/waitForUserOperationReceipt" import type { GetAccountNonceParams } from "./public/getAccountNonce" import { getAccountNonce } from "./public/getAccountNonce" -export type { - SendUserOperationParameters, - EstimateUserOperationGasParameters, - EstimateUserOperationGasReturnType, - GetUserOperationByHashParameters, - GetUserOperationByHashReturnType, - GetUserOperationReceiptParameters, - GetUserOperationReceiptReturnType, - GetSenderAddressParams, - GetAccountNonceParams, - WaitForUserOperationReceiptParameters, - EstimateUserOperationErrorType -} +export type { GetSenderAddressParams, GetAccountNonceParams } -export { - sendUserOperation, - estimateUserOperationGas, - supportedEntryPoints, - chainId, - getUserOperationByHash, - getUserOperationReceipt, - getSenderAddress, - getAccountNonce, - InvalidEntryPointError, - waitForUserOperationReceipt, - WaitForUserOperationReceiptTimeoutError -} +export { getSenderAddress, getAccountNonce, InvalidEntryPointError } diff --git a/packages/permissionless/actions/pimlico.ts b/packages/permissionless/actions/pimlico.ts index 77e500b9..2ba5d170 100644 --- a/packages/permissionless/actions/pimlico.ts +++ b/packages/permissionless/actions/pimlico.ts @@ -18,11 +18,11 @@ import { } from "./pimlico/sponsorUserOperation" import type { - PimlicoBundlerActions, + PimlicoActions, PimlicoPaymasterClientActions } from "../clients/decorators/pimlico" import { - pimlicoBundlerActions, + pimlicoActions, pimlicoPaymasterActions } from "../clients/decorators/pimlico" @@ -36,7 +36,7 @@ export type { GetUserOperationGasPriceReturnType, GetUserOperationStatusParameters, GetUserOperationStatusReturnType, - PimlicoBundlerActions, + PimlicoActions as PimlicoBundlerActions, PimlicoPaymasterClientActions, PimlicoSponsorUserOperationParameters, SendCompressedUserOperationParameters, @@ -48,7 +48,7 @@ export type { export { getUserOperationGasPrice, getUserOperationStatus, - pimlicoBundlerActions, + pimlicoActions as pimlicoBundlerActions, pimlicoPaymasterActions, sendCompressedUserOperation, sponsorUserOperation, diff --git a/packages/permissionless/actions/pimlico/getUserOperationGasPrice.ts b/packages/permissionless/actions/pimlico/getUserOperationGasPrice.ts index ca81bb95..1b2e7592 100644 --- a/packages/permissionless/actions/pimlico/getUserOperationGasPrice.ts +++ b/packages/permissionless/actions/pimlico/getUserOperationGasPrice.ts @@ -1,6 +1,5 @@ import type { Account, Chain, Client, Transport } from "viem" -import type { Prettify } from "../../types/" -import type { PimlicoBundlerRpcSchema } from "../../types/pimlico" +import type { PimlicoRpcSchema } from "../../types/pimlico" export type GetUserOperationGasPriceReturnType = { slow: { @@ -38,13 +37,14 @@ export type GetUserOperationGasPriceReturnType = { * await getUserOperationGasPrice(bundlerClient) * */ -export const getUserOperationGasPrice = async < - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends Account | undefined = Account | undefined ->( - client: Client -): Promise> => { +export const getUserOperationGasPrice = async ( + client: Client< + Transport, + Chain | undefined, + Account | undefined, + PimlicoRpcSchema + > +): Promise => { const gasPrice = await client.request({ method: "pimlico_getUserOperationGasPrice", params: [] diff --git a/packages/permissionless/actions/pimlico/getUserOperationStatus.ts b/packages/permissionless/actions/pimlico/getUserOperationStatus.ts index d5edc66c..93b1fcc2 100644 --- a/packages/permissionless/actions/pimlico/getUserOperationStatus.ts +++ b/packages/permissionless/actions/pimlico/getUserOperationStatus.ts @@ -1,8 +1,6 @@ import type { Account, Chain, Client, Hash, Transport } from "viem" -import type { PimlicoBundlerClient } from "../../clients/pimlico" -import type { Prettify } from "../../types/" import type { - PimlicoBundlerRpcSchema, + PimlicoRpcSchema, PimlicoUserOperationStatus } from "../../types/pimlico" @@ -17,7 +15,7 @@ export type GetUserOperationStatusReturnType = PimlicoUserOperationStatus * * - Docs: https://docs.pimlico.io/permissionless/reference/pimlico-bundler-actions/getUserOperationStatus * - * @param client {@link PimlicoBundlerClient} that you created using viem's createClient whose transport url is pointing to the Pimlico's bundler. + * @param client {@link PimlicoClient} that you created using viem's createClient whose transport url is pointing to the Pimlico's bundler. * @param hash {@link Hash} UserOpHash that you must have received from sendUserOperation. * @returns status & transaction hash if included {@link GetUserOperationStatusReturnType} * @@ -35,14 +33,15 @@ export type GetUserOperationStatusReturnType = PimlicoUserOperationStatus * await getUserOperationStatus(bundlerClient, { hash: userOpHash }) * */ -export const getUserOperationStatus = async < - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends Account | undefined = Account | undefined ->( - client: Client, - { hash }: Prettify -): Promise> => { +export const getUserOperationStatus = async ( + client: Client< + Transport, + Chain | undefined, + Account | undefined, + PimlicoRpcSchema + >, + { hash }: GetUserOperationStatusParameters +): Promise => { return client.request({ method: "pimlico_getUserOperationStatus", params: [hash] diff --git a/packages/permissionless/actions/pimlico/sendCompressedUserOperation.ts b/packages/permissionless/actions/pimlico/sendCompressedUserOperation.ts index ed47b16e..45835929 100644 --- a/packages/permissionless/actions/pimlico/sendCompressedUserOperation.ts +++ b/packages/permissionless/actions/pimlico/sendCompressedUserOperation.ts @@ -7,13 +7,16 @@ import type { Hex, Transport } from "viem" -import type { Prettify } from "../../types/" -import type { PimlicoBundlerRpcSchema } from "../../types/pimlico" +import type { + entryPoint06Address, + entryPoint07Address +} from "viem/account-abstraction" +import type { PimlicoRpcSchema } from "../../types/pimlico" export type SendCompressedUserOperationParameters = { compressedUserOperation: Hex inflatorAddress: Address - entryPoint: Address + entryPointAddress: typeof entryPoint06Address | typeof entryPoint07Address } /** @@ -41,22 +44,19 @@ export type SendCompressedUserOperationParameters = { * }) * // Return '0xe9fad2cd67f9ca1d0b7a6513b2a42066784c8df938518da2b51bb8cc9a89ea34' */ -export const sendCompressedUserOperation = async < - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends Account | undefined = Account | undefined ->( - client: Client, - args: Prettify +export const sendCompressedUserOperation = async ( + client: Client< + Transport, + Chain | undefined, + Account | undefined, + PimlicoRpcSchema + >, + args: SendCompressedUserOperationParameters ): Promise => { - const { compressedUserOperation, inflatorAddress, entryPoint } = args + const { compressedUserOperation, inflatorAddress, entryPointAddress } = args return client.request({ method: "pimlico_sendCompressedUserOperation", - params: [ - compressedUserOperation as Hex, - inflatorAddress as Address, - entryPoint as Address - ] + params: [compressedUserOperation, inflatorAddress, entryPointAddress] }) } diff --git a/packages/permissionless/actions/pimlico/sponsorUserOperation.ts b/packages/permissionless/actions/pimlico/sponsorUserOperation.ts index a8786d9e..1ab7bf1f 100644 --- a/packages/permissionless/actions/pimlico/sponsorUserOperation.ts +++ b/packages/permissionless/actions/pimlico/sponsorUserOperation.ts @@ -1,56 +1,76 @@ -import type { Account, Address, Chain, Client, Hex, Transport } from "viem" -import type { PartialBy } from "viem/types/utils" -import type { Prettify } from "../../types/" import type { - ENTRYPOINT_ADDRESS_V06_TYPE, - EntryPoint, - GetEntryPointVersion -} from "../../types/entrypoint" -import type { PimlicoPaymasterRpcSchema } from "../../types/pimlico" -import type { - UserOperation, - UserOperationWithBigIntAsHex -} from "../../types/userOperation" + Account, + Address, + Chain, + Client, + Hex, + OneOf, + PartialBy, + Transport +} from "viem" +import { + type UserOperation, + entryPoint06Address, + type entryPoint07Address +} from "viem/account-abstraction" +import type { PimlicoRpcSchema } from "../../types/pimlico" import { deepHexlify } from "../../utils/deepHexlify" -import { ENTRYPOINT_ADDRESS_V06 } from "../../utils/getEntryPointVersion" export type PimlicoSponsorUserOperationParameters< - entryPoint extends EntryPoint + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address = + | typeof entryPoint06Address + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7" = "0.6" | "0.7" > = { - userOperation: entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE - ? PartialBy< - UserOperation<"v0.6">, - "callGasLimit" | "preVerificationGas" | "verificationGasLimit" - > - : PartialBy< - UserOperation<"v0.7">, - | "callGasLimit" - | "preVerificationGas" - | "verificationGasLimit" - | "paymasterVerificationGasLimit" - | "paymasterPostOpGasLimit" - > - entryPoint: entryPoint + userOperation: OneOf< + | (entryPointVersion extends "0.6" + ? PartialBy< + UserOperation<"0.6">, + | "callGasLimit" + | "preVerificationGas" + | "verificationGasLimit" + > + : never) + | (entryPointVersion extends "0.7" + ? PartialBy< + UserOperation<"0.7">, + | "callGasLimit" + | "preVerificationGas" + | "verificationGasLimit" + | "paymasterVerificationGasLimit" + | "paymasterPostOpGasLimit" + > + : never) + > + entryPointAddress: entryPointAddress sponsorshipPolicyId?: string } -export type SponsorUserOperationReturnType = - entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE - ? { - callGasLimit: bigint - verificationGasLimit: bigint - preVerificationGas: bigint - paymasterAndData: Hex - } - : { - callGasLimit: bigint - verificationGasLimit: bigint - preVerificationGas: bigint - paymaster: Address - paymasterVerificationGasLimit: bigint - paymasterPostOpGasLimit: bigint - paymasterData: Hex - } +export type SponsorUserOperationReturnType< + entryPointVersion extends "0.6" | "0.7" = "0.7" +> = OneOf< + | (entryPointVersion extends "0.6" + ? { + callGasLimit: bigint + verificationGasLimit: bigint + preVerificationGas: bigint + paymasterAndData: Hex + } + : never) + | (entryPointVersion extends "0.7" + ? { + callGasLimit: bigint + verificationGasLimit: bigint + preVerificationGas: bigint + paymaster: Address + paymasterVerificationGasLimit: bigint + paymasterPostOpGasLimit: bigint + paymasterData: Hex + } + : never) +> /** * Returns paymasterAndData & updated gas parameters required to sponsor a userOperation. @@ -78,44 +98,38 @@ export type SponsorUserOperationReturnType = * */ export const sponsorUserOperation = async < - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends Account | undefined = Account | undefined + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address = + | typeof entryPoint06Address + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7" = "0.6" | "0.7" >( client: Client< - TTransport, - TChain, - TAccount, - PimlicoPaymasterRpcSchema + Transport, + Chain | undefined, + Account | undefined, + PimlicoRpcSchema >, - args: Prettify> -): Promise>> => { + args: PimlicoSponsorUserOperationParameters< + entryPointAddress, + entryPointVersion + > +): Promise> => { const response = await client.request({ method: "pm_sponsorUserOperation", params: args.sponsorshipPolicyId ? [ - deepHexlify( - args.userOperation - ) as UserOperationWithBigIntAsHex< - GetEntryPointVersion - >, - args.entryPoint, + deepHexlify(args.userOperation), + args.entryPointAddress, { sponsorshipPolicyId: args.sponsorshipPolicyId } ] - : [ - deepHexlify( - args.userOperation - ) as UserOperationWithBigIntAsHex< - GetEntryPointVersion - >, - args.entryPoint - ] + : [deepHexlify(args.userOperation), args.entryPointAddress] }) - if (args.entryPoint === ENTRYPOINT_ADDRESS_V06) { + if (args.entryPointAddress === entryPoint06Address) { const responseV06 = response as { paymasterAndData: Hex preVerificationGas: Hex @@ -131,7 +145,7 @@ export const sponsorUserOperation = async < preVerificationGas: BigInt(responseV06.preVerificationGas), verificationGasLimit: BigInt(responseV06.verificationGasLimit), callGasLimit: BigInt(responseV06.callGasLimit) - } as SponsorUserOperationReturnType + } as SponsorUserOperationReturnType } const responseV07 = response as { @@ -155,5 +169,5 @@ export const sponsorUserOperation = async < ), paymasterPostOpGasLimit: BigInt(responseV07.paymasterPostOpGasLimit), paymasterData: responseV07.paymasterData - } as SponsorUserOperationReturnType + } as SponsorUserOperationReturnType } diff --git a/packages/permissionless/actions/pimlico/validateSponsorshipPolicies.ts b/packages/permissionless/actions/pimlico/validateSponsorshipPolicies.ts index 2eb511c0..ffe303d0 100644 --- a/packages/permissionless/actions/pimlico/validateSponsorshipPolicies.ts +++ b/packages/permissionless/actions/pimlico/validateSponsorshipPolicies.ts @@ -1,18 +1,22 @@ import type { Account, Chain, Client, Transport } from "viem" -import type { Prettify } from "../../types/" -import type { EntryPoint, GetEntryPointVersion } from "../../types/entrypoint" -import type { PimlicoPaymasterRpcSchema } from "../../types/pimlico" import type { UserOperation, - UserOperationWithBigIntAsHex -} from "../../types/userOperation" + entryPoint06Address, + entryPoint07Address +} from "viem/account-abstraction" +import type { PimlicoRpcSchema } from "../../types/pimlico" import { deepHexlify } from "../../utils/deepHexlify" export type ValidateSponsorshipPoliciesParameters< - entryPoint extends EntryPoint + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address = + | typeof entryPoint06Address + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7" = "0.6" | "0.7" > = { - userOperation: UserOperation> - entryPoint: entryPoint + userOperation: UserOperation + entryPointAddress: entryPointAddress sponsorshipPolicyIds: string[] } @@ -62,26 +66,29 @@ export type ValidateSponsorshipPolicies = { * ] */ export const validateSponsorshipPolicies = async < - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends Account | undefined = Account | undefined + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address = + | typeof entryPoint06Address + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7" = "0.6" | "0.7" >( client: Client< - TTransport, - TChain, - TAccount, - PimlicoPaymasterRpcSchema + Transport, + Chain | undefined, + Account | undefined, + PimlicoRpcSchema >, - args: Prettify> -): Promise[]> => { + args: ValidateSponsorshipPoliciesParameters< + entryPointAddress, + entryPointVersion + > +): Promise => { return await client.request({ method: "pm_validateSponsorshipPolicies", params: [ - deepHexlify(args.userOperation) as UserOperationWithBigIntAsHex< - GetEntryPointVersion - >, - args.entryPoint, + deepHexlify(args.userOperation), + args.entryPointAddress, args.sponsorshipPolicyIds ] }) diff --git a/packages/permissionless/actions/smartAccount.test.ts b/packages/permissionless/actions/smartAccount.test.ts deleted file mode 100644 index c2a63c7e..00000000 --- a/packages/permissionless/actions/smartAccount.test.ts +++ /dev/null @@ -1,198 +0,0 @@ -import type { - Abi, - Account, - Address, - Chain, - Client, - Hash, - Hex, - SignTypedDataParameters, - Transport, - TypedData -} from "viem" -import type { SignMessageParameters } from "viem" -import { describe, expectTypeOf, test } from "vitest" -import type { SmartAccount } from "../accounts" -import type { BundlerRpcSchema } from "../types/bundler" -import type { - ENTRYPOINT_ADDRESS_V06_TYPE, - EntryPoint -} from "../types/entrypoint" -import { - type DeployContractParametersWithPaymaster, - type PrepareUserOperationRequestParameters, - type PrepareUserOperationRequestReturnType, - type SendTransactionWithPaymasterParameters, - type SendTransactionsWithPaymasterParameters, - type SendUserOperationParameters, - type WriteContractWithPaymasterParameters, - deployContract, - prepareUserOperationRequest, - sendTransaction, - sendTransactions, - sendUserOperation, - signMessage, - signTypedData, - writeContract -} from "./smartAccount" - -describe("index", () => { - test("sendUserOperation", () => { - expectTypeOf(deployContract).toBeFunction() - expectTypeOf(deployContract) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - expectTypeOf(deployContract) - .parameter(1) - .toMatchTypeOf>() - expectTypeOf(deployContract).returns.toMatchTypeOf>() - }) - test("prepareUserOperationRequest", () => { - expectTypeOf(prepareUserOperationRequest).toBeFunction() - expectTypeOf(prepareUserOperationRequest) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - expectTypeOf(prepareUserOperationRequest) - .parameter(1) - .toMatchTypeOf>() - expectTypeOf(prepareUserOperationRequest).returns.toMatchTypeOf< - Promise> - >() - }) - test("sendTransaction", () => { - expectTypeOf(sendTransaction).toBeFunction() - expectTypeOf(sendTransaction) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - expectTypeOf(sendTransaction) - .parameter(1) - .toMatchTypeOf>() - expectTypeOf(sendTransaction).returns.toMatchTypeOf>() - }) - test("sendTransactions", () => { - expectTypeOf(sendTransactions).toBeFunction() - expectTypeOf(sendTransactions) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - expectTypeOf(sendTransactions) - .parameter(1) - .toMatchTypeOf< - SendTransactionsWithPaymasterParameters - >() - expectTypeOf(sendTransactions).returns.toMatchTypeOf>() - }) - test("writeContract", () => { - expectTypeOf(writeContract).toBeFunction() - expectTypeOf(writeContract) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - expectTypeOf( - writeContract< - ENTRYPOINT_ADDRESS_V06_TYPE, - Chain, - SmartAccount, - Abi - > - ) - .parameter(1) - .toMatchTypeOf< - WriteContractWithPaymasterParameters< - ENTRYPOINT_ADDRESS_V06_TYPE, - Chain, - SmartAccount, - Abi - > - >() - expectTypeOf(writeContract).returns.toMatchTypeOf>() - }) - test("sendUserOperation", () => { - expectTypeOf(sendUserOperation).toBeFunction() - expectTypeOf(sendUserOperation) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - expectTypeOf(sendUserOperation) - .parameter(1) - .toMatchTypeOf>() - expectTypeOf(sendUserOperation).returns.toMatchTypeOf>() - }) - test("signMessage", () => { - expectTypeOf(signMessage).toBeFunction() - expectTypeOf(signMessage) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - expectTypeOf(signMessage) - .parameter(1) - .toMatchTypeOf>>() - expectTypeOf(signMessage).returns.toMatchTypeOf>() - }) - test("signTypedData", () => { - expectTypeOf(signTypedData).toBeFunction() - expectTypeOf(signTypedData) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - expectTypeOf( - signTypedData - ) - .parameter(1) - .toMatchTypeOf< - SignTypedDataParameters - >() - expectTypeOf(signTypedData).returns.toMatchTypeOf>() - }) -}) diff --git a/packages/permissionless/actions/smartAccount.ts b/packages/permissionless/actions/smartAccount.ts deleted file mode 100644 index 856c15d5..00000000 --- a/packages/permissionless/actions/smartAccount.ts +++ /dev/null @@ -1,56 +0,0 @@ -import { - type DeployContractParametersWithPaymaster, - deployContract -} from "./smartAccount/deployContract" - -import { - type Middleware, - type PrepareUserOperationRequestParameters, - type PrepareUserOperationRequestReturnType, - type SponsorUserOperationReturnType, - prepareUserOperationRequest -} from "./smartAccount/prepareUserOperationRequest" - -import { - type SendTransactionWithPaymasterParameters, - sendTransaction -} from "./smartAccount/sendTransaction" - -import { - type SendUserOperationParameters, - sendUserOperation -} from "./smartAccount/sendUserOperation" - -import { signMessage } from "./smartAccount/signMessage" - -import { signTypedData } from "./smartAccount/signTypedData" - -import { - type SendTransactionsWithPaymasterParameters, - sendTransactions -} from "./smartAccount/sendTransactions" - -import { - type WriteContractWithPaymasterParameters, - writeContract -} from "./smartAccount/writeContract" - -export { - deployContract, - type DeployContractParametersWithPaymaster, - prepareUserOperationRequest, - type PrepareUserOperationRequestParameters, - type PrepareUserOperationRequestReturnType, - type SponsorUserOperationReturnType, - sendTransaction, - sendUserOperation, - type SendUserOperationParameters, - signMessage, - signTypedData, - type SendTransactionWithPaymasterParameters, - type Middleware, - sendTransactions, - type SendTransactionsWithPaymasterParameters, - type WriteContractWithPaymasterParameters, - writeContract -} diff --git a/packages/permissionless/actions/smartAccount/deployContract.test.ts b/packages/permissionless/actions/smartAccount/deployContract.test.ts deleted file mode 100644 index c74b96c1..00000000 --- a/packages/permissionless/actions/smartAccount/deployContract.test.ts +++ /dev/null @@ -1,118 +0,0 @@ -import type { Chain, Client, Transport } from "viem" -import { generatePrivateKey } from "viem/accounts" -import { foundry } from "viem/chains" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getCoreSmartAccounts, - getPimlicoPaymasterClient -} from "../../../permissionless-test/src/utils" -import type { SmartAccount } from "../../accounts" -import type { SmartAccountClient } from "../../clients/createSmartAccountClient" -import type { - ENTRYPOINT_ADDRESS_V06_TYPE, - ENTRYPOINT_ADDRESS_V07_TYPE, - EntryPoint -} from "../../types/entrypoint" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" -import { deployContract } from "./deployContract" - -describe.each(getCoreSmartAccounts())( - "deployContract $name", - ({ - getSmartAccountClient, - supportsEntryPointV06, - supportsEntryPointV07 - }) => { - testWithRpc.skipIf(!supportsEntryPointV06)( - "deployContract_V06", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) - }) - - await expect(async () => - deployContract( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - account: smartClient.account, - chain: foundry, - abi: [ - { - inputs: [], - stateMutability: "payable", - type: "constructor" - } - ], - bytecode: - "0x608060405260358060116000396000f3006080604052600080fd00a165627a7a72305820f86ff341f0dff29df244305f8aa88abaf10e3a0719fa6ea1dcdd01b8b7d750970029" - } - ) - ).rejects.toThrowError( - /^.*doesn't support account deployment.*$/i - ) - } - ) - - testWithRpc.skipIf(!supportsEntryPointV07)( - "deployContract_V07", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = (await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) - })) as SmartAccountClient< - ENTRYPOINT_ADDRESS_V07_TYPE, - Transport, - Chain, - SmartAccount - > - - await expect(async () => - deployContract( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - account: smartClient.account, - chain: foundry, - abi: [ - { - inputs: [], - stateMutability: "payable", - type: "constructor" - } - ], - bytecode: - "0x608060405260358060116000396000f3006080604052600080fd00a165627a7a72305820f86ff341f0dff29df244305f8aa88abaf10e3a0719fa6ea1dcdd01b8b7d750970029" - } - ) - ).rejects.toThrowError( - /^.*doesn't support account deployment.*$/i - ) - } - ) - } -) diff --git a/packages/permissionless/actions/smartAccount/deployContract.ts b/packages/permissionless/actions/smartAccount/deployContract.ts deleted file mode 100644 index 50bfd3b8..00000000 --- a/packages/permissionless/actions/smartAccount/deployContract.ts +++ /dev/null @@ -1,123 +0,0 @@ -import type { - Abi, - Account, - Chain, - Client, - DeployContractParameters, - EncodeDeployDataParameters, - Hash, - Transport -} from "viem" -import { getAction } from "viem/utils" -import type { SmartAccount } from "../../accounts/types" -import type { EntryPoint } from "../../types/entrypoint" -import { parseAccount } from "../../utils/" -import { AccountOrClientNotFoundError } from "../../utils/signUserOperationHashWithECDSA" -import { waitForUserOperationReceipt } from "../bundler/waitForUserOperationReceipt" -import type { Middleware } from "./prepareUserOperationRequest" -import { - type SendUserOperationParameters, - sendUserOperation -} from "./sendUserOperation" - -export type DeployContractParametersWithPaymaster< - entryPoint extends EntryPoint, - TAbi extends Abi | readonly unknown[] = Abi | readonly unknown[], - TChain extends Chain | undefined = Chain | undefined, - TAccount extends Account | undefined = Account | undefined, - TChainOverride extends Chain | undefined = Chain | undefined -> = DeployContractParameters & - Middleware - -/** - * Deploys a contract to the network, given bytecode and constructor arguments. - * This function also allows you to sponsor this transaction if sender is a smartAccount - * - * - Docs: https://viem.sh/docs/contract/deployContract.html - * - Examples: https://stackblitz.com/github/wagmi-dev/viem/tree/main/examples/contracts/deploying-contracts - * - * @param client - Client to use - * @param parameters - {@link DeployContractParameters} - * @returns The [Transaction](https://viem.sh/docs/glossary/terms.html#transaction) hash. - * - * @example - * import { createWalletClient, http } from 'viem' - * import { privateKeyToAccount } from 'viem/accounts' - * import { mainnet } from 'viem/chains' - * import { deployContract } from 'viem/contract' - * - * const client = createWalletClient({ - * account: privateKeyToAccount('0x…'), - * chain: mainnet, - * transport: http(), - * }) - * const hash = await deployContract(client, { - * abi: [], - * account: '0x…, - * bytecode: '0x608060405260405161083e38038061083e833981016040819052610...', - * }) - */ -export async function deployContract< - entryPoint extends EntryPoint, - TTransport extends Transport, - TChain extends Chain | undefined, - TAbi extends Abi | readonly unknown[], - TAccount extends SmartAccount | undefined, - TChainOverride extends Chain | undefined ->( - client: Client, - args: DeployContractParametersWithPaymaster< - entryPoint, - TAbi, - TChain, - TAccount, - TChainOverride - > -): Promise { - const { - abi, - args: constructorArgs, - bytecode, - middleware, - ...request - } = args - - const { account: account_ = client.account } = request - - if (!account_) { - throw new AccountOrClientNotFoundError({ - docsPath: "/docs/actions/wallet/sendTransaction" - }) - } - - const account = parseAccount(account_) as SmartAccount - - const userOpHash = await getAction( - client, - sendUserOperation, - "sendUserOperation" - )({ - userOperation: { - sender: account.address, - maxFeePerGas: request.maxFeePerGas, - maxPriorityFeePerGas: request.maxPriorityFeePerGas, - callData: await account.encodeDeployCallData({ - abi, - bytecode, - args: constructorArgs - } as EncodeDeployDataParameters) - }, - account: account, - middleware - } as SendUserOperationParameters) - - const userOperationReceipt = await getAction( - client, - waitForUserOperationReceipt, - "waitForUserOperationReceipt" - )({ - hash: userOpHash - }) - - return userOperationReceipt?.receipt.transactionHash -} diff --git a/packages/permissionless/actions/smartAccount/prepareUserOperationRequest.test.ts b/packages/permissionless/actions/smartAccount/prepareUserOperationRequest.test.ts deleted file mode 100644 index 3bf09aec..00000000 --- a/packages/permissionless/actions/smartAccount/prepareUserOperationRequest.test.ts +++ /dev/null @@ -1,520 +0,0 @@ -import { type Chain, type Client, type Transport, zeroAddress } from "viem" -import { generatePrivateKey } from "viem/accounts" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getCoreSmartAccounts, - getPimlicoBundlerClient, - getPimlicoPaymasterClient -} from "../../../permissionless-test/src/utils" -import type { SmartAccount } from "../../accounts" -import type { EntryPoint } from "../../types/entrypoint" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" -import { pimlicoPaymasterActions } from "../pimlico" -import { prepareUserOperationRequest } from "./prepareUserOperationRequest" - -describe.each(getCoreSmartAccounts())( - "prepareUserOperationRequest $name", - ({ - getSmartAccountClient, - supportsEntryPointV06, - supportsEntryPointV07 - }) => { - testWithRpc.skipIf(!supportsEntryPointV06)( - "prepareUserOperationRequest_v06", - async ({ rpc }) => { - const { anvilRpc, altoRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc - }) - - const userOperation = await prepareUserOperationRequest( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - userOperation: { - callData: await smartClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - } - } - ) - expect(userOperation).toBeTruthy() - expect(userOperation.sender).toBe(smartClient.account.address) - expect(userOperation.nonce).toBe( - await smartClient.account.getNonce() - ) - expect(userOperation.initCode).toBe( - await smartClient.account.getInitCode() - ) - expect(userOperation.callData).toBe( - await smartClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - ) - expect(userOperation.callGasLimit).toBeTruthy() - expect(userOperation.verificationGasLimit).toBeTruthy() - expect(userOperation.maxFeePerGas).toBeTruthy() - expect(userOperation.maxPriorityFeePerGas).toBeTruthy() - expect(userOperation.paymasterAndData).toBe("0x") - expect(userOperation.signature).toBe( - // @ts-ignore: since tests return all smart account client, some of them don't support V06. - // The TS error is because in that case, getDummySignature would not accept the userOperation of type UserOperation - await smartClient.account.getDummySignature(userOperation) - ) - - expect(userOperation.factory).toBe(undefined) - expect(userOperation.factoryData).toBe(undefined) - expect(userOperation.paymaster).toBe(undefined) - expect(userOperation.paymasterVerificationGasLimit).toBe( - undefined - ) - expect(userOperation.paymasterPostOpGasLimit).toBe(undefined) - expect(userOperation.paymasterData).toBe(undefined) - } - ) - - testWithRpc.skipIf(!supportsEntryPointV06)( - "prepareUserOperationRequest_v06", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc - }) - - const pimlicoBundlerClient = getPimlicoBundlerClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - altoRpc: altoRpc - }) - - const pimlicoPaymasterActions = getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc: paymasterRpc - }) - - const userOperation = await prepareUserOperationRequest( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - userOperation: { - callData: await smartClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - }, - middleware: { - gasPrice: async () => - ( - await pimlicoBundlerClient.getUserOperationGasPrice() - ).fast, - sponsorUserOperation: - pimlicoPaymasterActions.sponsorUserOperation - } - } - ) - - expect(userOperation).toBeTruthy() - expect(userOperation.sender).toBe(smartClient.account.address) - expect(userOperation.nonce).toBe( - await smartClient.account.getNonce() - ) - expect(userOperation.initCode).toBe( - await smartClient.account.getInitCode() - ) - expect(userOperation.callData).toBe( - await smartClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - ) - expect(userOperation.callGasLimit).toBeTruthy() - expect(userOperation.verificationGasLimit).toBeTruthy() - expect(userOperation.maxFeePerGas).toBeTruthy() - expect(userOperation.maxPriorityFeePerGas).toBeTruthy() - expect(userOperation.paymasterAndData).toBeTruthy() - expect(userOperation.signature).toBe( - // @ts-ignore: since tests return all smart account client, some of them don't support V06. - // The TS error is because in that case, getDummySignature would not accept the userOperation of type UserOperation - await smartClient.account.getDummySignature(userOperation) - ) - - expect(userOperation.factory).toBe(undefined) - expect(userOperation.factoryData).toBe(undefined) - expect(userOperation.paymaster).toBe(undefined) - expect(userOperation.paymasterVerificationGasLimit).toBe( - undefined - ) - expect(userOperation.paymasterPostOpGasLimit).toBe(undefined) - expect(userOperation.paymasterData).toBe(undefined) - } - ) - - testWithRpc.skipIf(!supportsEntryPointV06)( - "prepareUserOperationRequest_v06", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc - }) - - const pimlicoBundlerClient = getPimlicoBundlerClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - altoRpc: altoRpc - }) - const pimlicoPaymasterActions = getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc: paymasterRpc - }) - - const userOperation = await prepareUserOperationRequest( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - userOperation: { - callData: await smartClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - }, - middleware: async (args: { - userOperation: T - entryPoint: EntryPoint - }) => { - const gasPrice = ( - await pimlicoBundlerClient.getUserOperationGasPrice() - ).fast - - return { - ...args.userOperation, - maxFeePerGas: gasPrice.maxFeePerGas, - maxPriorityFeePerGas: - gasPrice.maxPriorityFeePerGas, - ...(await pimlicoPaymasterActions.sponsorUserOperation( - { - userOperation: { - ...args.userOperation, - maxFeePerGas: gasPrice.maxFeePerGas, - maxPriorityFeePerGas: - gasPrice.maxPriorityFeePerGas - } - } - )) - } as T - } - } - ) - - expect(userOperation).toBeTruthy() - expect(userOperation.sender).toBe(smartClient.account.address) - expect(userOperation.nonce).toBe( - await smartClient.account.getNonce() - ) - expect(userOperation.initCode).toBe( - await smartClient.account.getInitCode() - ) - expect(userOperation.callData).toBe( - await smartClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - ) - expect(userOperation.callGasLimit).toBeTruthy() - expect(userOperation.verificationGasLimit).toBeTruthy() - expect(userOperation.maxFeePerGas).toBeTruthy() - expect(userOperation.maxPriorityFeePerGas).toBeTruthy() - expect(userOperation.paymasterAndData).toBeTruthy() - expect(userOperation.signature).toBe( - // @ts-ignore: since tests return all smart account client, some of them don't support V06. - // The TS error is because in that case, getDummySignature would not accept the userOperation of type UserOperation - await smartClient.account.getDummySignature(userOperation) - ) - - expect(userOperation.factory).toBe(undefined) - expect(userOperation.factoryData).toBe(undefined) - expect(userOperation.paymaster).toBe(undefined) - expect(userOperation.paymasterVerificationGasLimit).toBe( - undefined - ) - expect(userOperation.paymasterPostOpGasLimit).toBe(undefined) - expect(userOperation.paymasterData).toBe(undefined) - } - ) - testWithRpc.skipIf(!supportsEntryPointV07)( - "prepareUserOperationRequest_v07", - async ({ rpc }) => { - const { anvilRpc, altoRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc - }) - - const userOperation = await prepareUserOperationRequest( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - userOperation: { - callData: await smartClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - } - } - ) - - expect(userOperation).toBeTruthy() - expect(userOperation.sender).toBe(smartClient.account.address) - expect(userOperation.nonce).toBe( - await smartClient.account.getNonce() - ) - expect(userOperation.factory).toBe( - await smartClient.account.getFactory() - ) - expect(userOperation.factoryData).toBe( - await smartClient.account.getFactoryData() - ) - expect(userOperation.callData).toBe( - await smartClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - ) - expect(userOperation.callGasLimit).toBeTruthy() - expect(userOperation.verificationGasLimit).toBeTruthy() - expect(userOperation.maxFeePerGas).toBeTruthy() - expect(userOperation.maxPriorityFeePerGas).toBeTruthy() - expect(userOperation.paymaster).toBe(undefined) - expect(userOperation.paymasterVerificationGasLimit).toBe( - undefined - ) - expect(userOperation.signature).toBe( - // @ts-ignore: since tests return all smart account client, some of them don't support V07. - // The TS error is because in that case, getDummySignature would not accept the userOperation of type UserOperation - await smartClient.account.getDummySignature(userOperation) - ) - expect(userOperation.paymasterPostOpGasLimit).toBe(0n) - expect(userOperation.paymasterData).toBe(undefined) - expect(userOperation.paymasterAndData).toBe(undefined) - expect(userOperation.initCode).toBe(undefined) - } - ) - - testWithRpc.skipIf(!supportsEntryPointV07)( - "prepareUserOperationRequest_v07", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc - }) - - const pimlicoBundlerClient = getPimlicoBundlerClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - altoRpc: altoRpc - }) - - const pimlicoPaymasterActions = getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc: paymasterRpc - }) - - const userOperation = await prepareUserOperationRequest( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - userOperation: { - callData: await smartClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - }, - middleware: { - gasPrice: async () => - ( - await pimlicoBundlerClient.getUserOperationGasPrice() - ).fast, - sponsorUserOperation: - pimlicoPaymasterActions.sponsorUserOperation - } - } - ) - - expect(userOperation).toBeTruthy() - expect(userOperation.sender).toBe(smartClient.account.address) - expect(userOperation.nonce).toBe( - await smartClient.account.getNonce() - ) - expect(userOperation.factory).toBe( - await smartClient.account.getFactory() - ) - expect(userOperation.factoryData).toBe( - await smartClient.account.getFactoryData() - ) - expect(userOperation.callData).toBe( - await smartClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - ) - expect(userOperation.callGasLimit).toBeTruthy() - expect(userOperation.verificationGasLimit).toBeTruthy() - expect(userOperation.maxFeePerGas).toBeTruthy() - expect(userOperation.maxPriorityFeePerGas).toBeTruthy() - expect(userOperation.paymaster).toBeTruthy() - expect(userOperation.paymasterVerificationGasLimit).toBeTruthy() - expect(userOperation.signature).toBe( - // @ts-ignore: since tests return all smart account client, some of them don't support V07. - // The TS error is because in that case, getDummySignature would not accept the userOperation of type UserOperation - await smartClient.account.getDummySignature(userOperation) - ) - expect(userOperation.paymasterPostOpGasLimit).toBe(1n) - expect(userOperation.paymasterData).toBeTruthy() - expect(userOperation.paymasterAndData).toBe(undefined) - expect(userOperation.initCode).toBe(undefined) - } - ) - - testWithRpc.skipIf(!supportsEntryPointV07)( - "prepareUserOperationRequest_v07", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc - }) - - const pimlicoBundlerClient = getPimlicoBundlerClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - altoRpc: altoRpc - }) - const pimlicoPaymasterActions = getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc: paymasterRpc - }) - - const userOperation = await prepareUserOperationRequest( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - userOperation: { - callData: await smartClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - }, - middleware: async (args: { - userOperation: T - entryPoint: EntryPoint - }) => { - const gasPrice = ( - await pimlicoBundlerClient.getUserOperationGasPrice() - ).fast - - return { - ...args.userOperation, - maxFeePerGas: gasPrice.maxFeePerGas, - maxPriorityFeePerGas: - gasPrice.maxPriorityFeePerGas, - ...(await pimlicoPaymasterActions.sponsorUserOperation( - { - userOperation: { - ...args.userOperation, - maxFeePerGas: gasPrice.maxFeePerGas, - maxPriorityFeePerGas: - gasPrice.maxPriorityFeePerGas - } - } - )) - } as T - } - } - ) - - expect(userOperation).toBeTruthy() - expect(userOperation.sender).toBe(smartClient.account.address) - expect(userOperation.nonce).toBe( - await smartClient.account.getNonce() - ) - expect(userOperation.factory).toBe( - await smartClient.account.getFactory() - ) - expect(userOperation.factoryData).toBe( - await smartClient.account.getFactoryData() - ) - expect(userOperation.callData).toBe( - await smartClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - ) - expect(userOperation.callGasLimit).toBeTruthy() - expect(userOperation.verificationGasLimit).toBeTruthy() - expect(userOperation.maxFeePerGas).toBeTruthy() - expect(userOperation.maxPriorityFeePerGas).toBeTruthy() - expect(userOperation.paymaster).toBeTruthy() - expect(userOperation.paymasterVerificationGasLimit).toBeTruthy() - expect(userOperation.signature).toBe( - // @ts-ignore: since tests return all smart account client, some of them don't support V07. - // The TS error is because in that case, getDummySignature would not accept the userOperation of type UserOperation - await smartClient.account.getDummySignature(userOperation) - ) - expect(userOperation.paymasterPostOpGasLimit).toBe(1n) - expect(userOperation.paymasterData).toBeTruthy() - expect(userOperation.paymasterAndData).toBe(undefined) - expect(userOperation.initCode).toBe(undefined) - } - ) - } -) diff --git a/packages/permissionless/actions/smartAccount/prepareUserOperationRequest.ts b/packages/permissionless/actions/smartAccount/prepareUserOperationRequest.ts deleted file mode 100644 index f719bf8a..00000000 --- a/packages/permissionless/actions/smartAccount/prepareUserOperationRequest.ts +++ /dev/null @@ -1,444 +0,0 @@ -import type { Chain, Client, Transport } from "viem" -import { estimateFeesPerGas } from "viem/actions" -import { getAction } from "viem/utils" -import type { SmartAccount } from "../../accounts/types" -import type { PartialPick } from "../../types" -import type { - GetAccountParameter, - PartialBy, - Prettify, - UserOperation -} from "../../types/" -import type { StateOverrides } from "../../types/bundler" -import type { - ENTRYPOINT_ADDRESS_V06_TYPE, - ENTRYPOINT_ADDRESS_V07_TYPE, - EntryPoint, - GetEntryPointVersion -} from "../../types/entrypoint" -import { AccountOrClientNotFoundError, parseAccount } from "../../utils/" -import { getEntryPointVersion } from "../../utils/getEntryPointVersion" -import { estimateUserOperationGas } from "../bundler/estimateUserOperationGas" - -export type SponsorUserOperationReturnType = - entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE - ? Prettify< - Pick< - UserOperation<"v0.6">, - | "callGasLimit" - | "verificationGasLimit" - | "preVerificationGas" - | "paymasterAndData" - > & - PartialPick< - UserOperation<"v0.6">, - "maxFeePerGas" | "maxPriorityFeePerGas" - > - > - : Prettify< - Pick< - UserOperation<"v0.7">, - | "callGasLimit" - | "verificationGasLimit" - | "preVerificationGas" - | "paymaster" - | "paymasterVerificationGasLimit" - | "paymasterPostOpGasLimit" - | "paymasterData" - > & - PartialPick< - UserOperation<"v0.7">, - "maxFeePerGas" | "maxPriorityFeePerGas" - > - > - -export type Middleware = { - middleware?: - | ((args: { - userOperation: UserOperation> - entryPoint: entryPoint - }) => Promise>>) - | { - gasPrice?: () => Promise<{ - maxFeePerGas: bigint - maxPriorityFeePerGas: bigint - }> - sponsorUserOperation?: (args: { - userOperation: UserOperation> - entryPoint: entryPoint - }) => Promise> - } -} - -export type PrepareUserOperationRequestParameters< - entryPoint extends EntryPoint, - TAccount extends SmartAccount | undefined = - | SmartAccount - | undefined -> = { - userOperation: entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE - ? PartialBy< - UserOperation<"v0.6">, - | "sender" - | "nonce" - | "initCode" - | "callGasLimit" - | "verificationGasLimit" - | "preVerificationGas" - | "maxFeePerGas" - | "maxPriorityFeePerGas" - | "paymasterAndData" - | "signature" - > - : PartialBy< - UserOperation<"v0.7">, - | "sender" - | "nonce" - | "factory" - | "factoryData" - | "callGasLimit" - | "verificationGasLimit" - | "preVerificationGas" - | "maxFeePerGas" - | "maxPriorityFeePerGas" - | "paymaster" - | "paymasterVerificationGasLimit" - | "paymasterPostOpGasLimit" - | "paymasterData" - | "signature" - > -} & GetAccountParameter & - Middleware - -export type PrepareUserOperationRequestReturnType< - entryPoint extends EntryPoint -> = UserOperation> - -async function prepareUserOperationRequestForEntryPointV06< - entryPoint extends EntryPoint = ENTRYPOINT_ADDRESS_V06_TYPE, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends SmartAccount | undefined = - | SmartAccount - | undefined ->( - client: Client, - args: PrepareUserOperationRequestParameters, - stateOverrides?: StateOverrides -): Promise>> { - const { - account: account_ = client.account, - userOperation: partialUserOperation, - middleware - } = args - if (!account_) throw new AccountOrClientNotFoundError() - - const account = parseAccount( - account_ - ) as SmartAccount - - const [sender, nonce, initCode, callData] = await Promise.all([ - partialUserOperation.sender || account.address, - partialUserOperation.nonce || account.getNonce(), - partialUserOperation.initCode || account.getInitCode(), - partialUserOperation.callData - ]) - - const userOperation: UserOperation<"v0.6"> = { - sender, - nonce, - initCode, - callData, - paymasterAndData: "0x", - signature: partialUserOperation.signature || "0x", - maxFeePerGas: partialUserOperation.maxFeePerGas || BigInt(0), - maxPriorityFeePerGas: - partialUserOperation.maxPriorityFeePerGas || BigInt(0), - callGasLimit: partialUserOperation.callGasLimit || BigInt(0), - verificationGasLimit: - partialUserOperation.verificationGasLimit || BigInt(0), - preVerificationGas: partialUserOperation.preVerificationGas || BigInt(0) - } - - if (userOperation.signature === "0x") { - userOperation.signature = await account.getDummySignature(userOperation) - } - - if (typeof middleware === "function") { - return middleware({ - userOperation, - entryPoint: account.entryPoint - } as { - userOperation: UserOperation> - entryPoint: entryPoint - }) as Promise> - } - - if (middleware && typeof middleware !== "function" && middleware.gasPrice) { - const gasPrice = await middleware.gasPrice() - userOperation.maxFeePerGas = gasPrice.maxFeePerGas - userOperation.maxPriorityFeePerGas = gasPrice.maxPriorityFeePerGas - } - - if (!userOperation.maxFeePerGas || !userOperation.maxPriorityFeePerGas) { - const estimateGas = await estimateFeesPerGas(account.client) - userOperation.maxFeePerGas = - userOperation.maxFeePerGas || estimateGas.maxFeePerGas - userOperation.maxPriorityFeePerGas = - userOperation.maxPriorityFeePerGas || - estimateGas.maxPriorityFeePerGas - } - - if ( - middleware && - typeof middleware !== "function" && - middleware.sponsorUserOperation - ) { - const sponsorUserOperationData = (await middleware.sponsorUserOperation( - { - userOperation, - entryPoint: account.entryPoint - } as { - userOperation: UserOperation> - entryPoint: entryPoint - } - )) as SponsorUserOperationReturnType - - userOperation.callGasLimit = sponsorUserOperationData.callGasLimit - userOperation.verificationGasLimit = - sponsorUserOperationData.verificationGasLimit - userOperation.preVerificationGas = - sponsorUserOperationData.preVerificationGas - userOperation.paymasterAndData = - sponsorUserOperationData.paymasterAndData - userOperation.maxFeePerGas = - sponsorUserOperationData.maxFeePerGas || userOperation.maxFeePerGas - userOperation.maxPriorityFeePerGas = - sponsorUserOperationData.maxPriorityFeePerGas || - userOperation.maxPriorityFeePerGas - return userOperation as PrepareUserOperationRequestReturnType - } - - if ( - !userOperation.callGasLimit || - !userOperation.verificationGasLimit || - !userOperation.preVerificationGas - ) { - const gasParameters = await getAction( - client, - estimateUserOperationGas, - "estimateUserOperationGas" - )( - { - userOperation, - entryPoint: account.entryPoint - } as { - userOperation: UserOperation> - entryPoint: entryPoint - }, - // @ts-ignore getAction takes only two params but when compiled this will work - stateOverrides - ) - - userOperation.callGasLimit |= gasParameters.callGasLimit - userOperation.verificationGasLimit = - userOperation.verificationGasLimit || - gasParameters.verificationGasLimit - userOperation.preVerificationGas = - userOperation.preVerificationGas || gasParameters.preVerificationGas - } - - return userOperation as PrepareUserOperationRequestReturnType -} - -async function prepareUserOperationRequestEntryPointV07< - entryPoint extends EntryPoint = ENTRYPOINT_ADDRESS_V07_TYPE, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends SmartAccount | undefined = - | SmartAccount - | undefined ->( - client: Client, - args: Prettify>, - stateOverrides?: StateOverrides -): Promise>> { - const { - account: account_ = client.account, - userOperation: partialUserOperation, - middleware - } = args - if (!account_) throw new AccountOrClientNotFoundError() - - const account = parseAccount( - account_ - ) as SmartAccount - - const [sender, nonce, factory, factoryData, callData] = await Promise.all([ - partialUserOperation.sender || account.address, - partialUserOperation.nonce || account.getNonce(), - partialUserOperation.factory || account.getFactory(), - partialUserOperation.factoryData || account.getFactoryData(), - partialUserOperation.callData - ]) - - const userOperation: UserOperation<"v0.7"> = { - sender, - nonce, - factory: factory, - factoryData: factoryData, - callData, - callGasLimit: partialUserOperation.callGasLimit || BigInt(0), - verificationGasLimit: - partialUserOperation.verificationGasLimit || BigInt(0), - preVerificationGas: - partialUserOperation.preVerificationGas || BigInt(0), - maxFeePerGas: partialUserOperation.maxFeePerGas || BigInt(0), - maxPriorityFeePerGas: - partialUserOperation.maxPriorityFeePerGas || BigInt(0), - signature: partialUserOperation.signature || "0x" - } - - if (userOperation.signature === "0x") { - userOperation.signature = await account.getDummySignature(userOperation) - } - - if (typeof middleware === "function") { - return middleware({ - userOperation, - entryPoint: account.entryPoint - } as { - userOperation: UserOperation> - entryPoint: entryPoint - }) as Promise> - } - - if (middleware && typeof middleware !== "function" && middleware.gasPrice) { - const gasPrice = await middleware.gasPrice() - userOperation.maxFeePerGas = gasPrice.maxFeePerGas - userOperation.maxPriorityFeePerGas = gasPrice.maxPriorityFeePerGas - } - - if (!userOperation.maxFeePerGas || !userOperation.maxPriorityFeePerGas) { - const estimateGas = await estimateFeesPerGas(account.client) - userOperation.maxFeePerGas = - userOperation.maxFeePerGas || estimateGas.maxFeePerGas - userOperation.maxPriorityFeePerGas = - userOperation.maxPriorityFeePerGas || - estimateGas.maxPriorityFeePerGas - } - - if ( - middleware && - typeof middleware !== "function" && - middleware.sponsorUserOperation - ) { - const sponsorUserOperationData = (await middleware.sponsorUserOperation( - { - userOperation, - entryPoint: account.entryPoint - } as { - userOperation: UserOperation> - entryPoint: entryPoint - } - )) as SponsorUserOperationReturnType - - userOperation.callGasLimit = sponsorUserOperationData.callGasLimit - userOperation.verificationGasLimit = - sponsorUserOperationData.verificationGasLimit - userOperation.preVerificationGas = - sponsorUserOperationData.preVerificationGas - userOperation.paymaster = sponsorUserOperationData.paymaster - userOperation.paymasterVerificationGasLimit = - sponsorUserOperationData.paymasterVerificationGasLimit - userOperation.paymasterPostOpGasLimit = - sponsorUserOperationData.paymasterPostOpGasLimit - userOperation.paymasterData = sponsorUserOperationData.paymasterData - userOperation.maxFeePerGas = - sponsorUserOperationData.maxFeePerGas || userOperation.maxFeePerGas - userOperation.maxPriorityFeePerGas = - sponsorUserOperationData.maxPriorityFeePerGas || - userOperation.maxPriorityFeePerGas - - return userOperation as PrepareUserOperationRequestReturnType - } - - if ( - !userOperation.callGasLimit || - !userOperation.verificationGasLimit || - !userOperation.preVerificationGas - ) { - const gasParameters = await getAction( - client, - estimateUserOperationGas, - "estimateUserOperationGas" - )( - { - userOperation, - entryPoint: account.entryPoint - }, - // @ts-ignore getAction takes only two params but when compiled this will work - stateOverrides - ) - - userOperation.callGasLimit |= gasParameters.callGasLimit - userOperation.verificationGasLimit = - userOperation.verificationGasLimit || - gasParameters.verificationGasLimit - userOperation.preVerificationGas = - userOperation.preVerificationGas || gasParameters.preVerificationGas - userOperation.paymasterPostOpGasLimit = - userOperation.paymasterPostOpGasLimit || - gasParameters.paymasterPostOpGasLimit - } - - return userOperation as PrepareUserOperationRequestReturnType -} - -export async function prepareUserOperationRequest< - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends SmartAccount | undefined = - | SmartAccount - | undefined ->( - client: Client, - args: PrepareUserOperationRequestParameters, - stateOverrides?: StateOverrides -): Promise>> { - const { account: account_ = client.account } = args - if (!account_) throw new AccountOrClientNotFoundError() - - const account = parseAccount(account_) as SmartAccount - - const entryPointVersion = getEntryPointVersion(account.entryPoint) - - if (entryPointVersion === "v0.6") { - return prepareUserOperationRequestForEntryPointV06( - client as unknown as Client< - Transport, - Chain | undefined, - SmartAccount - >, - args as unknown as PrepareUserOperationRequestParameters< - ENTRYPOINT_ADDRESS_V06_TYPE, - SmartAccount - >, - stateOverrides - ) as Promise> - } - - return prepareUserOperationRequestEntryPointV07( - client as unknown as Client< - Transport, - Chain | undefined, - SmartAccount - >, - args as unknown as PrepareUserOperationRequestParameters< - ENTRYPOINT_ADDRESS_V07_TYPE, - SmartAccount - >, - stateOverrides - ) as Promise> -} diff --git a/packages/permissionless/actions/smartAccount/sendTransaction.test.ts b/packages/permissionless/actions/smartAccount/sendTransaction.test.ts deleted file mode 100644 index 35f5c1fa..00000000 --- a/packages/permissionless/actions/smartAccount/sendTransaction.test.ts +++ /dev/null @@ -1,108 +0,0 @@ -import { type Chain, type Client, type Transport, zeroAddress } from "viem" -import { generatePrivateKey } from "viem/accounts" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getCoreSmartAccounts, - getPimlicoPaymasterClient, - getPublicClient -} from "../../../permissionless-test/src/utils" -import type { SmartAccount } from "../../accounts" -import type { EntryPoint } from "../../types/entrypoint" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" -import { sendTransaction } from "./sendTransaction" - -describe.each(getCoreSmartAccounts())( - "sendTransaction $name", - ({ - getSmartAccountClient, - supportsEntryPointV06, - supportsEntryPointV07 - }) => { - testWithRpc.skipIf(!supportsEntryPointV06)( - "sendTransaction_v06", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) - }) - - const transactionHash = await sendTransaction( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - to: zeroAddress, - data: "0x", - value: 0n - } - ) - - expect(transactionHash).toBeTruthy() - - const publicClient = getPublicClient(anvilRpc) - - const receipt = await publicClient.getTransactionReceipt({ - hash: transactionHash - }) - - expect(receipt).toBeTruthy() - expect(receipt.transactionHash).toBe(transactionHash) - expect(receipt.status).toBe("success") - } - ) - - testWithRpc.skipIf(!supportsEntryPointV07)( - "sendTransaction_v07", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) - }) - - const transactionHash = await sendTransaction( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - to: zeroAddress, - data: "0x", - value: 0n - } - ) - - expect(transactionHash).toBeTruthy() - - const publicClient = getPublicClient(anvilRpc) - - const receipt = await publicClient.getTransactionReceipt({ - hash: transactionHash - }) - - expect(receipt).toBeTruthy() - expect(receipt.transactionHash).toBe(transactionHash) - expect(receipt.status).toBe("success") - } - ) - } -) diff --git a/packages/permissionless/actions/smartAccount/sendTransaction.ts b/packages/permissionless/actions/smartAccount/sendTransaction.ts deleted file mode 100644 index 72d4208f..00000000 --- a/packages/permissionless/actions/smartAccount/sendTransaction.ts +++ /dev/null @@ -1,151 +0,0 @@ -import type { - Chain, - Client, - Hash, - SendTransactionParameters, - Transport -} from "viem" -import { getAction } from "viem/utils" -import type { SmartAccount } from "../../accounts/types" -import type { Prettify } from "../../types/" -import type { EntryPoint } from "../../types/entrypoint" -import { AccountOrClientNotFoundError, parseAccount } from "../../utils/" -import { waitForUserOperationReceipt } from "../bundler/waitForUserOperationReceipt" -import type { Middleware } from "./prepareUserOperationRequest" -import { sendUserOperation } from "./sendUserOperation" - -export type SendTransactionWithPaymasterParameters< - entryPoint extends EntryPoint, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends SmartAccount | undefined = - | SmartAccount - | undefined, - TChainOverride extends Chain | undefined = Chain | undefined -> = SendTransactionParameters & - Middleware - -/** - * Creates, signs, and sends a new transaction to the network. - * This function also allows you to sponsor this transaction if sender is a smartAccount - * - * - Docs: https://viem.sh/docs/actions/wallet/sendTransaction.html - * - Examples: https://stackblitz.com/github/wagmi-dev/viem/tree/main/examples/transactions/sending-transactions - * - JSON-RPC Methods: - * - JSON-RPC Accounts: [`eth_sendTransaction`](https://ethereum.org/en/developers/docs/apis/json-rpc/#eth_sendtransaction) - * - Local Accounts: [`eth_sendRawTransaction`](https://ethereum.org/en/developers/docs/apis/json-rpc/#eth_sendrawtransaction) - * - * @param client - Client to use - * @param parameters - {@link SendTransactionParameters} - * @returns The [Transaction](https://viem.sh/docs/glossary/terms.html#transaction) hash. - * - * @example - * import { createWalletClient, custom } from 'viem' - * import { mainnet } from 'viem/chains' - * import { sendTransaction } from 'viem/wallet' - * - * const client = createWalletClient({ - * chain: mainnet, - * transport: custom(window.ethereum), - * }) - * const hash = await sendTransaction(client, { - * account: '0xA0Cf798816D4b9b9866b5330EEa46a18382f251e', - * to: '0x70997970c51812dc3a010c7d01b50e0d17dc79c8', - * value: 1000000000000000000n, - * }) - * - * @example - * // Account Hoisting - * import { createWalletClient, http } from 'viem' - * import { privateKeyToAccount } from 'viem/accounts' - * import { mainnet } from 'viem/chains' - * import { sendTransaction } from 'viem/wallet' - * - * const client = createWalletClient({ - * account: privateKeyToAccount('0x…'), - * chain: mainnet, - * transport: http(), - * }) - * const hash = await sendTransaction(client, { - * to: '0x70997970c51812dc3a010c7d01b50e0d17dc79c8', - * value: 1000000000000000000n, - * }) - */ -export async function sendTransaction< - TTransport extends Transport, - TChain extends Chain | undefined, - TAccount extends SmartAccount | undefined, - entryPoint extends EntryPoint, - TChainOverride extends Chain | undefined = Chain | undefined ->( - client: Client, - args: Prettify< - SendTransactionWithPaymasterParameters< - entryPoint, - TChain, - TAccount, - TChainOverride - > - > -): Promise { - const { - account: account_ = client.account, - data, - maxFeePerGas, - maxPriorityFeePerGas, - to, - value, - nonce, - middleware - } = args - - if (!account_) { - throw new AccountOrClientNotFoundError({ - docsPath: "/docs/actions/wallet/sendTransaction" - }) - } - - const account = parseAccount(account_) as SmartAccount - - if (!to) throw new Error("Missing to address") - - if (account.type !== "local") { - throw new Error("RPC account type not supported") - } - - const callData = await account.encodeCallData({ - to, - value: value || BigInt(0), - data: data || "0x" - }) - - const userOpHash = await getAction( - client, - sendUserOperation< - entryPoint, - TTransport, - TChain, - SmartAccount - >, - "sendUserOperation" - )({ - userOperation: { - sender: account.address, - maxFeePerGas: maxFeePerGas, - maxPriorityFeePerGas: maxPriorityFeePerGas, - callData: callData, - nonce: nonce ? BigInt(nonce) : undefined - }, - account, - middleware - }) - - const userOperationReceipt = await getAction( - client, - waitForUserOperationReceipt, - "waitForUserOperationReceipt" - )({ - hash: userOpHash - }) - - return userOperationReceipt?.receipt.transactionHash -} diff --git a/packages/permissionless/actions/smartAccount/sendTransactions.test.ts b/packages/permissionless/actions/smartAccount/sendTransactions.test.ts deleted file mode 100644 index ef70b4b2..00000000 --- a/packages/permissionless/actions/smartAccount/sendTransactions.test.ts +++ /dev/null @@ -1,126 +0,0 @@ -import { type Chain, type Client, type Transport, zeroAddress } from "viem" -import { generatePrivateKey } from "viem/accounts" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getCoreSmartAccounts, - getPimlicoPaymasterClient, - getPublicClient -} from "../../../permissionless-test/src/utils" -import type { SmartAccount } from "../../accounts" -import type { EntryPoint } from "../../types/entrypoint" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" -import { sendTransactions } from "./sendTransactions" - -describe.each(getCoreSmartAccounts())( - "sendTransactions $name", - ({ - getSmartAccountClient, - supportsEntryPointV06, - supportsEntryPointV07 - }) => { - testWithRpc.skipIf(!supportsEntryPointV06)( - "sendTransactions_v06", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) - }) - - const transactionHash = await sendTransactions( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - transactions: [ - { - to: zeroAddress, - data: "0x", - value: 0n - }, - { - to: zeroAddress, - data: "0x", - value: 0n - } - ] - } - ) - - expect(transactionHash).toBeTruthy() - - const publicClient = getPublicClient(anvilRpc) - - const receipt = await publicClient.getTransactionReceipt({ - hash: transactionHash - }) - - expect(receipt).toBeTruthy() - expect(receipt.transactionHash).toBe(transactionHash) - expect(receipt.status).toBe("success") - } - ) - - testWithRpc.skipIf(!supportsEntryPointV07)( - "sendTransactions_v07", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) - }) - - const transactionHash = await sendTransactions( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - transactions: [ - { - to: zeroAddress, - data: "0x", - value: 0n - }, - { - to: zeroAddress, - data: "0x", - value: 0n - } - ] - } - ) - - expect(transactionHash).toBeTruthy() - - const publicClient = getPublicClient(anvilRpc) - - const receipt = await publicClient.getTransactionReceipt({ - hash: transactionHash - }) - - expect(receipt).toBeTruthy() - expect(receipt.transactionHash).toBe(transactionHash) - expect(receipt.status).toBe("success") - } - ) - } -) diff --git a/packages/permissionless/actions/smartAccount/sendTransactions.ts b/packages/permissionless/actions/smartAccount/sendTransactions.ts deleted file mode 100644 index 863fb24b..00000000 --- a/packages/permissionless/actions/smartAccount/sendTransactions.ts +++ /dev/null @@ -1,153 +0,0 @@ -import type { - Address, - Chain, - Client, - Hash, - Hex, - SendTransactionParameters, - Transport -} from "viem" -import { getAction } from "viem/utils" -import type { SmartAccount } from "../../accounts/types" -import type { GetAccountParameter, Prettify } from "../../types/" -import type { EntryPoint } from "../../types/entrypoint" -import { AccountOrClientNotFoundError, parseAccount } from "../../utils/" -import { waitForUserOperationReceipt } from "../bundler/waitForUserOperationReceipt" -import type { Middleware } from "./prepareUserOperationRequest" -import { sendUserOperation } from "./sendUserOperation" - -export type SendTransactionsWithPaymasterParameters< - entryPoint extends EntryPoint, - TAccount extends SmartAccount | undefined = - | SmartAccount - | undefined - | undefined -> = { - transactions: { to: Address; value: bigint; data: Hex }[] -} & GetAccountParameter & - Middleware & { - maxFeePerGas?: bigint - maxPriorityFeePerGas?: bigint - nonce?: bigint - } - -/** - * Creates, signs, and sends a new transactions to the network. - * This function also allows you to sponsor this transaction if sender is a smartAccount - * - * @param client - Client to use - * @param parameters - {@link SendTransactionParameters} - * @returns The [Transaction](https://viem.sh/docs/glossary/terms.html#transaction) hash. - * - * @example - * import { createWalletClient, custom } from 'viem' - * import { mainnet } from 'viem/chains' - * import { sendTransaction } from 'viem/wallet' - * - * const client = createWalletClient({ - * chain: mainnet, - * transport: custom(window.ethereum), - * }) - * const hash = await sendTransaction(client, [{ - * account: '0xA0Cf798816D4b9b9866b5330EEa46a18382f251e', - * to: '0x70997970c51812dc3a010c7d01b50e0d17dc79c8', - * value: 1000000000000000000n, - * }, { - * to: '0x61897970c51812dc3a010c7d01b50e0d17dc1234', - * value: 10000000000000000n, - * }]) - * - * @example - * // Account Hoisting - * import { createWalletClient, http } from 'viem' - * import { privateKeyToAccount } from 'viem/accounts' - * import { mainnet } from 'viem/chains' - * import { sendTransaction } from 'viem/wallet' - * - * const client = createWalletClient({ - * account: privateKeyToAccount('0x…'), - * chain: mainnet, - * transport: http(), - * }) - * const hash = await sendTransactions(client, [{ - * to: '0x70997970c51812dc3a010c7d01b50e0d17dc79c8', - * value: 1000000000000000000n, - * }, { - * to: '0x61897970c51812dc3a010c7d01b50e0d17dc1234', - * value: 10000000000000000n, - * }]) - */ -export async function sendTransactions< - TTransport extends Transport, - TChain extends Chain | undefined, - TAccount extends SmartAccount | undefined, - entryPoint extends EntryPoint ->( - client: Client, - args: Prettify< - SendTransactionsWithPaymasterParameters - > -): Promise { - const { - account: account_ = client.account, - transactions, - middleware, - maxFeePerGas, - maxPriorityFeePerGas, - nonce - } = args - - if (!account_) { - throw new AccountOrClientNotFoundError({ - docsPath: "/docs/actions/wallet/sendTransaction" - }) - } - - const account = parseAccount(account_) as SmartAccount - - if (account.type !== "local") { - throw new Error("RPC account type not supported") - } - - const callData = await account.encodeCallData( - transactions.map(({ to, value, data }) => { - if (!to) throw new Error("Missing to address") - return { - to, - value: value || BigInt(0), - data: data || "0x" - } - }) - ) - - const userOpHash = await getAction( - client, - sendUserOperation< - entryPoint, - TTransport, - TChain, - SmartAccount - >, - "sendUserOperation" - )({ - userOperation: { - sender: account.address, - maxFeePerGas: maxFeePerGas, - maxPriorityFeePerGas: maxPriorityFeePerGas, - callData: callData, - nonce: nonce - }, - account: account, - middleware - }) - - const userOperationReceipt = await getAction( - client, - waitForUserOperationReceipt, - "waitForUserOperationReceipt" - )({ - hash: userOpHash - }) - - return userOperationReceipt?.receipt.transactionHash -} diff --git a/packages/permissionless/actions/smartAccount/sendUserOperation.test.ts b/packages/permissionless/actions/smartAccount/sendUserOperation.test.ts deleted file mode 100644 index a5b28858..00000000 --- a/packages/permissionless/actions/smartAccount/sendUserOperation.test.ts +++ /dev/null @@ -1,133 +0,0 @@ -import { type Chain, type Client, type Transport, zeroAddress } from "viem" -import { generatePrivateKey } from "viem/accounts" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getBundlerClient, - getCoreSmartAccounts, - getPimlicoPaymasterClient, - getPublicClient -} from "../../../permissionless-test/src/utils" -import type { SmartAccount } from "../../accounts" -import type { EntryPoint } from "../../types/entrypoint" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" -import { sendUserOperation } from "./sendUserOperation" - -describe.each(getCoreSmartAccounts())( - "sendUserOperation $name", - ({ - getSmartAccountClient, - supportsEntryPointV06, - supportsEntryPointV07 - }) => { - testWithRpc.skipIf(!supportsEntryPointV06)( - "sendUserOperation_v06", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) - }) - - const userOperationHash = await sendUserOperation( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - userOperation: { - callData: await smartClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - } - } - ) - - expect(userOperationHash).toBeTruthy() - - const bundlerClient = getBundlerClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - altoRpc - }) - - const receipt = await bundlerClient.waitForUserOperationReceipt( - { - hash: userOperationHash - } - ) - - expect(receipt).toBeTruthy() - expect(receipt.userOpHash).toBe(userOperationHash) - expect(receipt.entryPoint.toLowerCase()).toBe( - ENTRYPOINT_ADDRESS_V06.toLowerCase() - ) - expect(receipt.receipt.status).toBe("success") - } - ) - - testWithRpc.skipIf(!supportsEntryPointV07)( - "sendUserOperation_v07", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) - }) - - const userOperationHash = await sendUserOperation( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - userOperation: { - callData: await smartClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - } - } - ) - - expect(userOperationHash).toBeTruthy() - - const bundlerClient = getBundlerClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - altoRpc - }) - - const receipt = await bundlerClient.waitForUserOperationReceipt( - { - hash: userOperationHash - } - ) - - expect(receipt).toBeTruthy() - expect(receipt.userOpHash).toBe(userOperationHash) - expect(receipt.entryPoint.toLowerCase()).toBe( - ENTRYPOINT_ADDRESS_V07.toLowerCase() - ) - expect(receipt.receipt.status).toBe("success") - } - ) - } -) diff --git a/packages/permissionless/actions/smartAccount/sendUserOperation.ts b/packages/permissionless/actions/smartAccount/sendUserOperation.ts deleted file mode 100644 index b5a2dbeb..00000000 --- a/packages/permissionless/actions/smartAccount/sendUserOperation.ts +++ /dev/null @@ -1,98 +0,0 @@ -import type { Chain, Client, Hash, Transport } from "viem" -import { getAction } from "viem/utils" -import type { SmartAccount } from "../../accounts/types" -import type { - ENTRYPOINT_ADDRESS_V06_TYPE, - EntryPoint, - GetEntryPointVersion -} from "../../types/entrypoint" -import type { - GetAccountParameter, - PartialBy, - Prettify, - UserOperation -} from "../../types/index" -import { AccountOrClientNotFoundError, parseAccount } from "../../utils/" -import { sendUserOperation as sendUserOperationBundler } from "../bundler/sendUserOperation" -import { - type Middleware, - type PrepareUserOperationRequestParameters, - prepareUserOperationRequest -} from "./prepareUserOperationRequest" - -export type SendUserOperationParameters< - entryPoint extends EntryPoint, - TAccount extends SmartAccount | undefined = - | SmartAccount - | undefined -> = { - userOperation: entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE - ? PartialBy< - UserOperation<"v0.6">, - | "sender" - | "nonce" - | "initCode" - | "callGasLimit" - | "verificationGasLimit" - | "preVerificationGas" - | "maxFeePerGas" - | "maxPriorityFeePerGas" - | "paymasterAndData" - | "signature" - > - : PartialBy< - UserOperation<"v0.7">, - | "sender" - | "nonce" - | "factory" - | "factoryData" - | "callGasLimit" - | "verificationGasLimit" - | "preVerificationGas" - | "maxFeePerGas" - | "maxPriorityFeePerGas" - | "paymaster" - | "paymasterVerificationGasLimit" - | "paymasterPostOpGasLimit" - | "paymasterData" - | "signature" - > -} & GetAccountParameter & - Middleware - -export async function sendUserOperation< - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends SmartAccount | undefined = - | SmartAccount - | undefined ->( - client: Client, - args: Prettify> -): Promise { - const { account: account_ = client.account } = args - if (!account_) throw new AccountOrClientNotFoundError() - - const account = parseAccount(account_) as SmartAccount - - const userOperation = await getAction( - client, - prepareUserOperationRequest, - "prepareUserOperationRequest" - )({ ...args, account } as PrepareUserOperationRequestParameters< - entryPoint, - TAccount - >) - - userOperation.signature = await account.signUserOperation( - userOperation as UserOperation> - ) - - return sendUserOperationBundler(client, { - userOperation: userOperation as UserOperation< - GetEntryPointVersion - >, - entryPoint: account.entryPoint - }) -} diff --git a/packages/permissionless/actions/smartAccount/signMessage.test.ts b/packages/permissionless/actions/smartAccount/signMessage.test.ts deleted file mode 100644 index 61e9685d..00000000 --- a/packages/permissionless/actions/smartAccount/signMessage.test.ts +++ /dev/null @@ -1,172 +0,0 @@ -import type { Chain, Client, Transport } from "viem" -import { generatePrivateKey } from "viem/accounts" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getCoreSmartAccounts, - getPimlicoPaymasterClient, - getPublicClient -} from "../../../permissionless-test/src/utils" -import type { SmartAccount } from "../../accounts" -import type { EntryPoint } from "../../types/entrypoint" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" -import { signMessage } from "./signMessage" - -describe.each(getCoreSmartAccounts())( - "signMessage $name", - ({ - getSmartAccountClient, - isEip1271Compliant, - supportsEntryPointV06, - supportsEntryPointV07, - name - }) => { - testWithRpc.skipIf(isEip1271Compliant || !supportsEntryPointV06)( - "not isEip1271Compliant_v06", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) - }) - - await expect(async () => - signMessage( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - message: - "slowly and steadily burning the private keys" - } - ) - ).rejects.toThrow() - } - ) - - testWithRpc.skipIf(!isEip1271Compliant || !supportsEntryPointV06)( - "isEip1271Compliant_v06", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) - }) - - const signature = await signMessage( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - message: "slowly and steadily burning the private keys" - } - ) - - const publicClient = getPublicClient(anvilRpc) - - const isVerified = await publicClient.verifyMessage({ - address: smartClient.account.address, - message: "slowly and steadily burning the private keys", - signature - }) - - expect(isVerified).toBeTruthy() - } - ) - - testWithRpc.skipIf(isEip1271Compliant || !supportsEntryPointV07)( - "not isEip1271Compliant_v07", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) - }) - - await expect(async () => - signMessage( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - message: - "slowly and steadily burning the private keys" - } - ) - ).rejects.toThrow() - } - ) - - testWithRpc.skipIf(!isEip1271Compliant || !supportsEntryPointV07)( - "isEip1271Compliant_v07", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) - }) - - const signature = await signMessage( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - { - message: "slowly and steadily burning the private keys" - } - ) - - const publicClient = getPublicClient(anvilRpc) - - if (name === "Safe 7579") { - // Due to 7579 launchpad, we can't verify the signature as of now. - // Awaiting for the fix - return - } - - const isVerified = await publicClient.verifyMessage({ - address: smartClient.account.address, - message: "slowly and steadily burning the private keys", - signature - }) - - expect(isVerified).toBeTruthy() - } - ) - } -) diff --git a/packages/permissionless/actions/smartAccount/signMessage.ts b/packages/permissionless/actions/smartAccount/signMessage.ts deleted file mode 100644 index 4a08426a..00000000 --- a/packages/permissionless/actions/smartAccount/signMessage.ts +++ /dev/null @@ -1,81 +0,0 @@ -import type { - Chain, - Client, - SignMessageParameters, - SignMessageReturnType, - Transport -} from "viem" -import type { SmartAccount } from "../../accounts/types" -import type { EntryPoint } from "../../types/entrypoint" -import { AccountOrClientNotFoundError, parseAccount } from "../../utils/" - -/** - * Calculates an Ethereum-specific signature in [EIP-191 format](https://eips.ethereum.org/EIPS/eip-191): `keccak256("\x19Ethereum Signed Message:\n" + len(message) + message))`. - * - * - Docs: https://viem.sh/docs/actions/wallet/signMessage.html - * - JSON-RPC Methods: - * - JSON-RPC Accounts: [`personal_sign`](https://docs.metamask.io/guide/signing-data.html#personal-sign) - * - Local Accounts: Signs locally. No JSON-RPC request. - * - * With the calculated signature, you can: - * - use [`verifyMessage`](https://viem.sh/docs/utilities/verifyMessage.html) to verify the signature, - * - use [`recoverMessageAddress`](https://viem.sh/docs/utilities/recoverMessageAddress.html) to recover the signing address from a signature. - * - * @param client - Client to use - * @param parameters - {@link SignMessageParameters} - * @returns The signed message. {@link SignMessageReturnType} - * - * @example - * import { createWalletClient, custom } from 'viem' - * import { mainnet } from 'viem/chains' - * import { signMessage } from 'viem/wallet' - * - * const client = createWalletClient({ - * chain: mainnet, - * transport: custom(window.ethereum), - * }) - * const signature = await signMessage(client, { - * account: '0xA0Cf798816D4b9b9866b5330EEa46a18382f251e', - * message: 'hello world', - * }) - * - * @example - * // Account Hoisting - * import { createWalletClient, custom } from 'viem' - * import { privateKeyToAccount } from 'viem/accounts' - * import { mainnet } from 'viem/chains' - * import { signMessage } from 'viem/wallet' - * - * const client = createWalletClient({ - * account: privateKeyToAccount('0x…'), - * chain: mainnet, - * transport: custom(window.ethereum), - * }) - * const signature = await signMessage(client, { - * message: 'hello world', - * }) - */ -export async function signMessage< - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends SmartAccount | undefined = - | SmartAccount - | undefined ->( - client: Client, - { - account: account_ = client.account, - message - }: SignMessageParameters -): Promise { - if (!account_) - throw new AccountOrClientNotFoundError({ - docsPath: "/docs/actions/wallet/signMessage" - }) - - const account = parseAccount(account_) - if (account.type === "local") return account.signMessage({ message }) - - throw new Error("Sign message is not supported by this account") -} diff --git a/packages/permissionless/actions/smartAccount/signTypedData.test.ts b/packages/permissionless/actions/smartAccount/signTypedData.test.ts deleted file mode 100644 index 7c3bc1f5..00000000 --- a/packages/permissionless/actions/smartAccount/signTypedData.test.ts +++ /dev/null @@ -1,196 +0,0 @@ -import { type Chain, type Client, type Transport, getAddress } from "viem" -import { generatePrivateKey } from "viem/accounts" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getCoreSmartAccounts, - getPimlicoPaymasterClient, - getPublicClient -} from "../../../permissionless-test/src/utils" -import type { SmartAccount } from "../../accounts" -import type { EntryPoint } from "../../types/entrypoint" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" -import { signTypedData } from "./signTypedData" - -const typedData = { - domain: { - name: "Ether Mail", - version: "1", - chainId: 1, - verifyingContract: getAddress( - "0xCcCCccccCCCCcCCCCCCcCcCccCcCCCcCcccccccC" - ) - }, - types: { - Person: [ - { name: "name", type: "string" }, - { name: "wallet", type: "address" } - ], - Mail: [ - { name: "from", type: "Person" }, - { name: "to", type: "Person" }, - { name: "contents", type: "string" } - ] - }, - primaryType: "Mail" as const, - message: { - from: { - name: "Cow", - wallet: "0xCD2a3d9F938E13CD947Ec05AbC7FE734Df8DD826" - }, - to: { - name: "Bob", - wallet: "0xbBbBBBBbbBBBbbbBbbBbbbbBBbBbbbbBbBbbBBbB" - }, - contents: "Hello, Bob!" - } -} - -describe.each(getCoreSmartAccounts())( - "signTypedData $name", - ({ - getSmartAccountClient, - isEip1271Compliant, - supportsEntryPointV06, - supportsEntryPointV07, - name - }) => { - testWithRpc.skipIf(isEip1271Compliant || !supportsEntryPointV06)( - "not isEip1271Compliant_v06", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) - }) - - await expect(async () => - signTypedData( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - typedData - ) - ).rejects.toThrow() - } - ) - - testWithRpc.skipIf(!isEip1271Compliant || !supportsEntryPointV06)( - "isEip1271Compliant_v06", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) - }) - - const signature = await signTypedData( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - typedData - ) - - const publicClient = getPublicClient(anvilRpc) - - const isVerified = await publicClient.verifyTypedData({ - ...typedData, - address: smartClient.account.address, - signature - }) - - expect(isVerified).toBeTruthy() - } - ) - - testWithRpc.skipIf(isEip1271Compliant || !supportsEntryPointV07)( - "not isEip1271Compliant_v07", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) - }) - - await expect(async () => - signTypedData( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - typedData - ) - ).rejects.toThrow() - } - ) - - testWithRpc.skipIf(!isEip1271Compliant || !supportsEntryPointV07)( - "isEip1271Compliant_v07", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) - }) - - const signature = await signTypedData( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - typedData - ) - - const publicClient = getPublicClient(anvilRpc) - - if (name === "Safe 7579") { - // Due to 7579 launchpad, we can't verify the signature as of now. - // Awaiting for the fix - return - } - - const isVerified = await publicClient.verifyTypedData({ - ...typedData, - address: smartClient.account.address, - signature - }) - - expect(isVerified).toBeTruthy() - } - ) - } -) diff --git a/packages/permissionless/actions/smartAccount/signTypedData.ts b/packages/permissionless/actions/smartAccount/signTypedData.ts deleted file mode 100644 index bae146d1..00000000 --- a/packages/permissionless/actions/smartAccount/signTypedData.ts +++ /dev/null @@ -1,165 +0,0 @@ -import { - type Chain, - type Client, - type SignTypedDataParameters, - type SignTypedDataReturnType, - type Transport, - type TypedData, - type TypedDataDefinition, - type TypedDataDomain, - getTypesForEIP712Domain, - validateTypedData -} from "viem" -import type { SmartAccount } from "../../accounts/types" -import type { EntryPoint } from "../../types/entrypoint" -import { AccountOrClientNotFoundError, parseAccount } from "../../utils/" - -/** - * Signs typed data and calculates an Ethereum-specific signature in [https://eips.ethereum.org/EIPS/eip-712](https://eips.ethereum.org/EIPS/eip-712): `sign(keccak256("\x19\x01" ‖ domainSeparator ‖ hashStruct(message)))` - * - * - Docs: https://viem.sh/docs/actions/wallet/signTypedData.html - * - JSON-RPC Methods: - * - JSON-RPC Accounts: [`eth_signTypedData_v4`](https://docs.metamask.io/guide/signing-data.html#signtypeddata-v4) - * - Local Accounts: Signs locally. No JSON-RPC request. - * - * @param client - Client to use - * @param parameters - {@link SignTypedDataParameters} - * @returns The signed data. {@link SignTypedDataReturnType} - * - * @example - * import { createWalletClient, custom } from 'viem' - * import { mainnet } from 'viem/chains' - * import { signTypedData } from 'viem/wallet' - * - * const client = createWalletClient({ - * chain: mainnet, - * transport: custom(window.ethereum), - * }) - * const signature = await signTypedData(client, { - * account: '0xA0Cf798816D4b9b9866b5330EEa46a18382f251e', - * domain: { - * name: 'Ether Mail', - * version: '1', - * chainId: 1, - * verifyingContract: '0xCcCCccccCCCCcCCCCCCcCcCccCcCCCcCcccccccC', - * }, - * types: { - * Person: [ - * { name: 'name', type: 'string' }, - * { name: 'wallet', type: 'address' }, - * ], - * Mail: [ - * { name: 'from', type: 'Person' }, - * { name: 'to', type: 'Person' }, - * { name: 'contents', type: 'string' }, - * ], - * }, - * primaryType: 'Mail', - * message: { - * from: { - * name: 'Cow', - * wallet: '0xCD2a3d9F938E13CD947Ec05AbC7FE734Df8DD826', - * }, - * to: { - * name: 'Bob', - * wallet: '0xbBbBBBBbbBBBbbbBbbBbbbbBBbBbbbbBbBbbBBbB', - * }, - * contents: 'Hello, Bob!', - * }, - * }) - * - * @example - * // Account Hoisting - * import { createWalletClient, http } from 'viem' - * import { privateKeyToAccount } from 'viem/accounts' - * import { mainnet } from 'viem/chains' - * import { signTypedData } from 'viem/wallet' - * - * const client = createWalletClient({ - * account: privateKeyToAccount('0x…'), - * chain: mainnet, - * transport: http(), - * }) - * const signature = await signTypedData(client, { - * domain: { - * name: 'Ether Mail', - * version: '1', - * chainId: 1, - * verifyingContract: '0xCcCCccccCCCCcCCCCCCcCcCccCcCCCcCcccccccC', - * }, - * types: { - * Person: [ - * { name: 'name', type: 'string' }, - * { name: 'wallet', type: 'address' }, - * ], - * Mail: [ - * { name: 'from', type: 'Person' }, - * { name: 'to', type: 'Person' }, - * { name: 'contents', type: 'string' }, - * ], - * }, - * primaryType: 'Mail', - * message: { - * from: { - * name: 'Cow', - * wallet: '0xCD2a3d9F938E13CD947Ec05AbC7FE734Df8DD826', - * }, - * to: { - * name: 'Bob', - * wallet: '0xbBbBBBBbbBBBbbbBbbBbbbbBBbBbbbbBbBbbBBbB', - * }, - * contents: 'Hello, Bob!', - * }, - * }) - */ -export async function signTypedData< - entryPoint extends EntryPoint, - const TTypedData extends TypedData | { [key: string]: unknown }, - TPrimaryType extends string, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends SmartAccount | undefined = - | SmartAccount - | undefined ->( - client: Client, - { - account: account_ = client.account, - domain, - message, - primaryType, - types: types_ - }: SignTypedDataParameters -): Promise { - if (!account_) { - throw new AccountOrClientNotFoundError({ - docsPath: "/docs/actions/wallet/signMessage" - }) - } - - const account = parseAccount(account_) - - const types = { - EIP712Domain: getTypesForEIP712Domain({ domain } as { - domain: TypedDataDomain - }), - ...(types_ as TTypedData) - } - validateTypedData({ - domain, - message, - primaryType, - types - } as TypedDataDefinition) - - if (account.type === "local") { - return account.signTypedData({ - domain, - primaryType, - types, - message - } as TypedDataDefinition) - } - - throw new Error("Sign type message is not supported by this account") -} diff --git a/packages/permissionless/actions/smartAccount/writeContract.test.ts b/packages/permissionless/actions/smartAccount/writeContract.test.ts deleted file mode 100644 index 7c3bc1f5..00000000 --- a/packages/permissionless/actions/smartAccount/writeContract.test.ts +++ /dev/null @@ -1,196 +0,0 @@ -import { type Chain, type Client, type Transport, getAddress } from "viem" -import { generatePrivateKey } from "viem/accounts" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getCoreSmartAccounts, - getPimlicoPaymasterClient, - getPublicClient -} from "../../../permissionless-test/src/utils" -import type { SmartAccount } from "../../accounts" -import type { EntryPoint } from "../../types/entrypoint" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" -import { signTypedData } from "./signTypedData" - -const typedData = { - domain: { - name: "Ether Mail", - version: "1", - chainId: 1, - verifyingContract: getAddress( - "0xCcCCccccCCCCcCCCCCCcCcCccCcCCCcCcccccccC" - ) - }, - types: { - Person: [ - { name: "name", type: "string" }, - { name: "wallet", type: "address" } - ], - Mail: [ - { name: "from", type: "Person" }, - { name: "to", type: "Person" }, - { name: "contents", type: "string" } - ] - }, - primaryType: "Mail" as const, - message: { - from: { - name: "Cow", - wallet: "0xCD2a3d9F938E13CD947Ec05AbC7FE734Df8DD826" - }, - to: { - name: "Bob", - wallet: "0xbBbBBBBbbBBBbbbBbbBbbbbBBbBbbbbBbBbbBBbB" - }, - contents: "Hello, Bob!" - } -} - -describe.each(getCoreSmartAccounts())( - "signTypedData $name", - ({ - getSmartAccountClient, - isEip1271Compliant, - supportsEntryPointV06, - supportsEntryPointV07, - name - }) => { - testWithRpc.skipIf(isEip1271Compliant || !supportsEntryPointV06)( - "not isEip1271Compliant_v06", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) - }) - - await expect(async () => - signTypedData( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - typedData - ) - ).rejects.toThrow() - } - ) - - testWithRpc.skipIf(!isEip1271Compliant || !supportsEntryPointV06)( - "isEip1271Compliant_v06", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) - }) - - const signature = await signTypedData( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - typedData - ) - - const publicClient = getPublicClient(anvilRpc) - - const isVerified = await publicClient.verifyTypedData({ - ...typedData, - address: smartClient.account.address, - signature - }) - - expect(isVerified).toBeTruthy() - } - ) - - testWithRpc.skipIf(isEip1271Compliant || !supportsEntryPointV07)( - "not isEip1271Compliant_v07", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) - }) - - await expect(async () => - signTypedData( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - typedData - ) - ).rejects.toThrow() - } - ) - - testWithRpc.skipIf(!isEip1271Compliant || !supportsEntryPointV07)( - "isEip1271Compliant_v07", - async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - - const smartClient = await getSmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) - }) - - const signature = await signTypedData( - smartClient as Client< - Transport, - Chain, - SmartAccount - >, - typedData - ) - - const publicClient = getPublicClient(anvilRpc) - - if (name === "Safe 7579") { - // Due to 7579 launchpad, we can't verify the signature as of now. - // Awaiting for the fix - return - } - - const isVerified = await publicClient.verifyTypedData({ - ...typedData, - address: smartClient.account.address, - signature - }) - - expect(isVerified).toBeTruthy() - } - ) - } -) diff --git a/packages/permissionless/actions/smartAccount/writeContract.ts b/packages/permissionless/actions/smartAccount/writeContract.ts deleted file mode 100644 index 0ac942b0..00000000 --- a/packages/permissionless/actions/smartAccount/writeContract.ts +++ /dev/null @@ -1,161 +0,0 @@ -import { - type Abi, - type Chain, - type Client, - type ContractFunctionArgs, - type ContractFunctionName, - type EncodeFunctionDataParameters, - type Hash, - type Transport, - type WriteContractParameters, - encodeFunctionData -} from "viem" -import { getAction } from "viem/utils" -import type { SmartAccount } from "../../accounts/types" -import type { EntryPoint } from "../../types/entrypoint" -import type { Middleware } from "./prepareUserOperationRequest" -import { - type SendTransactionWithPaymasterParameters, - sendTransaction -} from "./sendTransaction" - -/** - * Executes a write function on a contract. - * This function also allows you to sponsor this transaction if sender is a smartAccount - * - * - Docs: https://viem.sh/docs/contract/writeContract.html - * - Examples: https://stackblitz.com/github/wagmi-dev/viem/tree/main/examples/contracts/writing-to-contracts - * - * A "write" function on a Solidity contract modifies the state of the blockchain. These types of functions require gas to be executed, and hence a [Transaction](https://viem.sh/docs/glossary/terms.html) is needed to be broadcast in order to change the state. - * - * Internally, uses a [Wallet Client](https://viem.sh/docs/clients/wallet.html) to call the [`sendTransaction` action](https://viem.sh/docs/actions/wallet/sendTransaction.html) with [ABI-encoded `data`](https://viem.sh/docs/contract/encodeFunctionData.html). - * - * __Warning: The `write` internally sends a transaction – it does not validate if the contract write will succeed (the contract may throw an error). It is highly recommended to [simulate the contract write with `contract.simulate`](https://viem.sh/docs/contract/writeContract.html#usage) before you execute it.__ - * - * @param client - Client to use - * @param parameters - {@link WriteContractParameters} - * @returns A [Transaction Hash](https://viem.sh/docs/glossary/terms.html#hash). - * - * @example - * import { createWalletClient, custom, parseAbi } from 'viem' - * import { mainnet } from 'viem/chains' - * import { writeContract } from 'viem/contract' - * - * const client = createWalletClient({ - * chain: mainnet, - * transport: custom(window.ethereum), - * }) - * const hash = await writeContract(client, { - * address: '0xFBA3912Ca04dd458c843e2EE08967fC04f3579c2', - * abi: parseAbi(['function mint(uint32 tokenId) nonpayable']), - * functionName: 'mint', - * args: [69420], - * }) - * - * @example - * // With Validation - * import { createWalletClient, http, parseAbi } from 'viem' - * import { mainnet } from 'viem/chains' - * import { simulateContract, writeContract } from 'viem/contract' - * - * const client = createWalletClient({ - * chain: mainnet, - * transport: http(), - * }) - * const { request } = await simulateContract(client, { - * address: '0xFBA3912Ca04dd458c843e2EE08967fC04f3579c2', - * abi: parseAbi(['function mint(uint32 tokenId) nonpayable']), - * functionName: 'mint', - * args: [69420], - * } - * const hash = await writeContract(client, request) - */ -export type WriteContractWithPaymasterParameters< - entryPoint extends EntryPoint, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends SmartAccount | undefined = - | SmartAccount - | undefined, - TAbi extends Abi | readonly unknown[] = Abi | readonly unknown[], - TFunctionName extends ContractFunctionName< - TAbi, - "nonpayable" | "payable" - > = ContractFunctionName, - TArgs extends ContractFunctionArgs< - TAbi, - "nonpayable" | "payable", - TFunctionName - > = ContractFunctionArgs, - TChainOverride extends Chain | undefined = undefined -> = WriteContractParameters< - TAbi, - TFunctionName, - TArgs, - TChain, - TAccount, - TChainOverride -> & - Middleware - -export async function writeContract< - entryPoint extends EntryPoint, - TTransport extends Transport, - TChain extends Chain | undefined, - TAccount extends SmartAccount | undefined, - const TAbi extends Abi | readonly unknown[], - TFunctionName extends ContractFunctionName< - TAbi, - "nonpayable" | "payable" - > = ContractFunctionName, - TArgs extends ContractFunctionArgs< - TAbi, - "nonpayable" | "payable", - TFunctionName - > = ContractFunctionArgs, - TChainOverride extends Chain | undefined = undefined ->( - client: Client, - { - abi, - address, - args, - dataSuffix, - functionName, - ...request - }: WriteContractWithPaymasterParameters< - entryPoint, - TChain, - TAccount, - TAbi, - TFunctionName, - TArgs, - TChainOverride - > -): Promise { - const data = encodeFunctionData({ - abi, - args, - functionName - } as EncodeFunctionDataParameters) - const hash = await getAction( - client, - sendTransaction< - TTransport, - TChain, - TAccount, - entryPoint, - TChainOverride - >, - "sendTransaction" - )({ - data: `${data}${dataSuffix ? dataSuffix.replace("0x", "") : ""}`, - to: address, - ...request - } as unknown as SendTransactionWithPaymasterParameters< - entryPoint, - TChain, - TAccount, - TChainOverride - >) - return hash -} diff --git a/packages/permissionless/actions/stackup.ts b/packages/permissionless/actions/stackup.ts deleted file mode 100644 index e3d866b8..00000000 --- a/packages/permissionless/actions/stackup.ts +++ /dev/null @@ -1,17 +0,0 @@ -import type { StackupPaymasterClientActions } from "../clients/decorators/stackup" -import { stackupPaymasterActions } from "../clients/decorators/stackup" -import { type AccountsParameters, accounts } from "./stackup/accounts" -import { - type SponsorUserOperationParameters, - type SponsorUserOperationReturnType, - sponsorUserOperation -} from "./stackup/sponsorUserOperation" - -export type { - SponsorUserOperationParameters, - SponsorUserOperationReturnType, - AccountsParameters, - StackupPaymasterClientActions -} - -export { sponsorUserOperation, accounts, stackupPaymasterActions } diff --git a/packages/permissionless/actions/stackup/accounts.ts b/packages/permissionless/actions/stackup/accounts.ts deleted file mode 100644 index b1d11a61..00000000 --- a/packages/permissionless/actions/stackup/accounts.ts +++ /dev/null @@ -1,41 +0,0 @@ -import type { Address } from "viem" -import type { StackupPaymasterClient } from "../../clients/stackup" -import type { EntryPoint } from "../../types" - -export type AccountsParameters = { - entryPoint: entryPoint -} - -/** - * Returns all the Paymaster addresses associated with an EntryPoint that’s owned by this service. - * - * https://docs.stackup.sh/docs/paymaster-api-rpc-methods#pm_accounts - * - * @param args {@link AccountsParameters} entryPoint for which you want to get list of supported paymasters. - * @returns paymaster addresses - * - * @example - * import { createClient } from "viem" - * import { accounts } from "permissionless/actions/stackup" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http("https://api.stackup.sh/v2/paymaster/YOUR_API_KEY_HERE") - * }) - * - * await accounts(bundlerClient, { - * entryPoint: entryPoint - * }}) - * - */ -export const accounts = async ( - client: StackupPaymasterClient, - { entryPoint: entryPointAddress }: AccountsParameters -): Promise => { - const response = await client.request({ - method: "pm_accounts", - params: [entryPointAddress] - }) - - return response -} diff --git a/packages/permissionless/actions/stackup/sponsorUserOperation.ts b/packages/permissionless/actions/stackup/sponsorUserOperation.ts deleted file mode 100644 index 3b376c91..00000000 --- a/packages/permissionless/actions/stackup/sponsorUserOperation.ts +++ /dev/null @@ -1,126 +0,0 @@ -import type { Address, Hex } from "viem" -import type { PartialBy } from "viem/types/utils" -import type { StackupPaymasterClient } from "../../clients/stackup" -import type { - ENTRYPOINT_ADDRESS_V06_TYPE, - EntryPoint -} from "../../types/entrypoint" -import type { StackupPaymasterContext } from "../../types/stackup" -import type { UserOperation } from "../../types/userOperation" -import { deepHexlify } from "../../utils/deepHexlify" -import { ENTRYPOINT_ADDRESS_V06 } from "../../utils/getEntryPointVersion" - -export type SponsorUserOperationParameters = { - userOperation: entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE - ? PartialBy< - UserOperation<"v0.6">, - "callGasLimit" | "preVerificationGas" | "verificationGasLimit" - > - : PartialBy< - UserOperation<"v0.7">, - | "callGasLimit" - | "preVerificationGas" - | "verificationGasLimit" - | "paymasterVerificationGasLimit" - | "paymasterPostOpGasLimit" - > - entryPoint: entryPoint - context: StackupPaymasterContext -} - -export type SponsorUserOperationReturnType = - entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE - ? Pick< - UserOperation<"v0.6">, - | "callGasLimit" - | "verificationGasLimit" - | "preVerificationGas" - | "paymasterAndData" - > - : Pick< - UserOperation<"v0.7">, - | "callGasLimit" - | "verificationGasLimit" - | "preVerificationGas" - | "paymaster" - | "paymasterVerificationGasLimit" - | "paymasterPostOpGasLimit" - | "paymasterData" - > - -/** - * Returns paymasterAndData & updated gas parameters required to sponsor a userOperation. - * - * - Docs: https://docs.pimlico.io/permissionless/reference/stackup-paymaster-actions/sponsorUserOperation - * - * @param client {@link PimlicoBundlerClient} that you created using viem's createClient whose transport url is pointing to the Pimlico's bundler. - * @param args {@link sponsorUserOperationParameters} UserOperation you want to sponsor & entryPoint. - * @returns paymasterAndData & updated gas parameters, see {@link SponsorUserOperationReturnType} - * - * - * @example - * import { createClient } from "viem" - * import { sponsorUserOperation } from "permissionless/actions/stackup" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http("https://api.stackup.sh/v2/paymaster/YOUR_API_KEY_HERE") - * }) - * - * await sponsorUserOperation(bundlerClient, { - * userOperation: userOperationWithDummySignature, - * entryPoint: entryPoint - * }}) - * - */ -export const sponsorUserOperation = async ( - client: StackupPaymasterClient, - args: SponsorUserOperationParameters -): Promise> => { - const response = await client.request({ - method: "pm_sponsorUserOperation", - params: [deepHexlify(args.userOperation), args.entryPoint, args.context] - }) - - if (args.entryPoint === ENTRYPOINT_ADDRESS_V06) { - const responseV06 = response as { - paymasterAndData: Hex - preVerificationGas: Hex - verificationGasLimit: Hex - callGasLimit: Hex - paymaster?: never - paymasterVerificationGasLimit?: never - paymasterPostOpGasLimit?: never - paymasterData?: never - } - return { - paymasterAndData: responseV06.paymasterAndData, - preVerificationGas: BigInt(responseV06.preVerificationGas), - verificationGasLimit: BigInt(responseV06.verificationGasLimit), - callGasLimit: BigInt(responseV06.callGasLimit) - } as SponsorUserOperationReturnType - } - - const responseV07 = response as { - preVerificationGas: Hex - verificationGasLimit: Hex - callGasLimit: Hex - paymaster: Address - paymasterVerificationGasLimit: Hex - paymasterPostOpGasLimit: Hex - paymasterData: Hex - paymasterAndData?: never - } - - return { - callGasLimit: BigInt(responseV07.callGasLimit), - verificationGasLimit: BigInt(responseV07.verificationGasLimit), - preVerificationGas: BigInt(responseV07.preVerificationGas), - paymaster: responseV07.paymaster, - paymasterVerificationGasLimit: BigInt( - responseV07.paymasterVerificationGasLimit - ), - paymasterPostOpGasLimit: BigInt(responseV07.paymasterPostOpGasLimit), - paymasterData: responseV07.paymasterData - } as SponsorUserOperationReturnType -} diff --git a/packages/permissionless/clients/createSmartAccountClient.test.ts b/packages/permissionless/clients/createSmartAccountClient.test.ts deleted file mode 100644 index 194b93f9..00000000 --- a/packages/permissionless/clients/createSmartAccountClient.test.ts +++ /dev/null @@ -1,55 +0,0 @@ -import { http } from "viem" -import { generatePrivateKey, privateKeyToAccount } from "viem/accounts" -import { foundry } from "viem/chains" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../permissionless-test/src/testWithRpc" -import { getPublicClient } from "../../permissionless-test/src/utils" -import { signerToSimpleSmartAccount } from "../accounts" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../utils" -import { createSmartAccountClient } from "./createSmartAccountClient" - -describe("createSmartAccountClient", () => { - testWithRpc("createSmartAccountClient_V06", async ({ rpc }) => { - const publicClient = getPublicClient(rpc.anvilRpc) - - const simpleSmartAccount = await signerToSimpleSmartAccount( - publicClient, - { - entryPoint: ENTRYPOINT_ADDRESS_V06, - signer: privateKeyToAccount(generatePrivateKey()) - } - ) - - const smartAccountClient = createSmartAccountClient({ - chain: foundry, - account: simpleSmartAccount, - bundlerTransport: http(rpc.altoRpc) - }) - - expect(smartAccountClient.account.address).toBe( - simpleSmartAccount.address - ) - }) - - testWithRpc("createSmartAccountClient_V07", async ({ rpc }) => { - const publicClient = getPublicClient(rpc.anvilRpc) - - const simpleSmartAccount = await signerToSimpleSmartAccount( - publicClient, - { - entryPoint: ENTRYPOINT_ADDRESS_V07, - signer: privateKeyToAccount(generatePrivateKey()) - } - ) - - const smartAccountClient = createSmartAccountClient({ - chain: foundry, - account: simpleSmartAccount, - bundlerTransport: http(rpc.altoRpc) - }) - - expect(smartAccountClient.account.address).toBe( - simpleSmartAccount.address - ) - }) -}) diff --git a/packages/permissionless/clients/createSmartAccountClient.ts b/packages/permissionless/clients/createSmartAccountClient.ts deleted file mode 100644 index 433c2135..00000000 --- a/packages/permissionless/clients/createSmartAccountClient.ts +++ /dev/null @@ -1,121 +0,0 @@ -import type { - Chain, - Client, - ClientConfig, - Transport, - WalletClientConfig -} from "viem" -import { createClient } from "viem" -import type { SmartAccount } from "../accounts/types" -import type { Middleware } from "../actions/smartAccount/prepareUserOperationRequest" -import type { Prettify } from "../types/" -import type { BundlerRpcSchema } from "../types/bundler" -import type { EntryPoint } from "../types/entrypoint" -import { - type SmartAccountActions, - smartAccountActions -} from "./decorators/smartAccount" - -/** - * TODO: - * - Add docs - * - Fix typing, 'accounts' is required to signMessage, signTypedData, signTransaction, but not needed here, since account is embedded in the client - */ -export type SmartAccountClient< - entryPoint extends EntryPoint, - transport extends Transport = Transport, - chain extends Chain | undefined = Chain | undefined, - account extends - | SmartAccount - | undefined = - | SmartAccount - | undefined -> = Prettify< - Client< - transport, - chain, - account, - BundlerRpcSchema, - SmartAccountActions - > -> - -export type SmartAccountClientConfig< - entryPoint extends EntryPoint, - transport extends Transport = Transport, - chain extends Chain | undefined = Chain | undefined, - account extends - | SmartAccount - | undefined = - | SmartAccount - | undefined - | undefined -> = Prettify< - Pick< - ClientConfig, - "cacheTime" | "chain" | "key" | "name" | "pollingInterval" - > & - Middleware & { - account: account - bundlerTransport: Transport - } & { - entryPoint?: entryPoint - } -> - -/** - * Creates a EIP-4337 compliant Bundler Client with a given [Transport](https://viem.sh/docs/clients/intro.html) configured for a [Chain](https://viem.sh/docs/clients/chains.html). - * - * - Docs: https://docs.pimlico.io/permissionless/reference/clients/smartAccountClient - * - * A Bundler Client is an interface to "erc 4337" [JSON-RPC API](https://eips.ethereum.org/EIPS/eip-4337#rpc-methods-eth-namespace) methods such as sending user operation, estimating gas for a user operation, get user operation receipt, etc through Bundler Actions. - * - * @param parameters - {@link WalletClientConfig} - * @returns A Bundler Client. {@link SmartAccountClient} - * - * @example - * import { createPublicClient, http } from 'viem' - * import { mainnet } from 'viem/chains' - * - * const smartAccountClient = createSmartAccountClient({ - * chain: mainnet, - * transport: http(BUNDLER_URL), - * }) - */ - -export function createSmartAccountClient< - TTransport extends Transport, - TChain extends Chain | undefined, - TEntryPoint extends EntryPoint, - TSmartAccount extends - | SmartAccount - | undefined = - | SmartAccount - | undefined ->( - parameters: SmartAccountClientConfig< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - > -): SmartAccountClient { - const { - key = "Account", - name = "Smart Account Client", - bundlerTransport - } = parameters - const client = createClient({ - ...parameters, - key, - name, - transport: bundlerTransport, - type: "smartAccountClient" - }) - - return client.extend( - smartAccountActions({ - middleware: parameters.middleware - }) - ) as SmartAccountClient -} diff --git a/packages/permissionless/clients/decorators/bundler.ts b/packages/permissionless/clients/decorators/bundler.ts deleted file mode 100644 index bd91cef8..00000000 --- a/packages/permissionless/clients/decorators/bundler.ts +++ /dev/null @@ -1,253 +0,0 @@ -import type { Client, Hash } from "viem" -import { chainId } from "../../actions/bundler/chainId" -import { - type EstimateUserOperationGasParameters, - type EstimateUserOperationGasReturnType, - estimateUserOperationGas -} from "../../actions/bundler/estimateUserOperationGas" -import { - type GetUserOperationByHashParameters, - type GetUserOperationByHashReturnType, - getUserOperationByHash -} from "../../actions/bundler/getUserOperationByHash" -import { - type GetUserOperationReceiptParameters, - type GetUserOperationReceiptReturnType, - getUserOperationReceipt -} from "../../actions/bundler/getUserOperationReceipt" -import { - type SendUserOperationParameters, - sendUserOperation -} from "../../actions/bundler/sendUserOperation" -import { supportedEntryPoints } from "../../actions/bundler/supportedEntryPoints" -import { - type WaitForUserOperationReceiptParameters, - waitForUserOperationReceipt -} from "../../actions/bundler/waitForUserOperationReceipt" -import type { Prettify } from "../../types/" -import type { StateOverrides } from "../../types/bundler" -import type { EntryPoint } from "../../types/entrypoint" -import type { BundlerClient } from "../createBundlerClient" - -export type BundlerActions = { - /** - * - * Sends user operation to the bundler - * - * - Docs: https://docs.pimlico.io/permissionless/reference/bundler-actions/sendUserOperation - * - * @param args {@link SendUserOperationParameters}. - * @returns UserOpHash that you can use to track user operation as {@link Hash}. - * - * @example - * import { createClient } from "viem" - * import { bundlerActions } from "permissionless" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http("https://api.pimlico.io/v2/goerli/rpc?apikey=YOUR_API_KEY_HERE") - * }).extend(bundlerActions) - * - * const userOpHash = await bundlerClient.sendUserOperation({ - * userOperation: signedUserOperation, - * entryPoint: entryPoint - * }) - * - * // Return '0xe9fad2cd67f9ca1d0b7a6513b2a42066784c8df938518da2b51bb8cc9a89ea34' - */ - sendUserOperation: ( - args: Prettify< - Omit, "entryPoint"> - > - ) => Promise - /** - * - * Estimates preVerificationGas, verificationGasLimit and callGasLimit for user operation - * - * - Docs: https://docs.pimlico.io/permissionless/reference/bundler-actions/estimateUserOperationGas - * - * @param args {@link EstimateUserOperationGasParameters} - * @returns preVerificationGas, verificationGasLimit and callGasLimit as {@link EstimateUserOperationGasReturnType} - * - * @example - * import { createClient } from "viem" - * import { bundlerActions } from "permissionless" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http(BUNDLER_URL) - * }).extend(bundlerActions) - * - * const gasParameters = await bundlerClient.estimateUserOperationGas({ - * userOperation: signedUserOperation, - * entryPoint: entryPoint - * }) - * - * // Return {preVerificationGas: 43492n, verificationGasLimit: 59436n, callGasLimit: 9000n} - */ - estimateUserOperationGas: ( - args: Prettify< - Omit, "entryPoint"> - >, - stateOverrides?: StateOverrides - ) => Promise>> - /** - * - * Returns the supported entrypoints by the bundler service - * - * - Docs: https://docs.pimlico.io/permissionless/reference/bundler-actions/supportedEntryPoints - * - * @returns Supported entryPoints - * - * @example - * import { createClient } from "viem" - * import { bundlerActions } from "permissionless" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http(BUNDLER_URL) - * }).extend(bundlerActions) - * - * const supportedEntryPoints = await bundlerClient.supportedEntryPoints() - * - * // Return ['0x5FF137D4b0FDCD49DcA30c7CF57E578a026d2789'] - */ - supportedEntryPoints: () => Promise - /** - * - * Returns the supported chain id by the bundler service - * - * - Docs: https://docs.pimlico.io/permissionless/reference/bundler-actions/chainId - * - * @returns Supported chain id - * - * @example - * import { createClient } from "viem" - * import { bundlerActions } from "permissionless" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http(BUNDLER_URL) - * }).extend(bundlerActions) - * - * const chainId = await bundlerClient.chainId() - * // Return 5n for Goerli - */ - chainId: () => Promise - /** - * - * Returns the user operation from userOpHash - * - * - Docs: https://docs.pimlico.io/permissionless/reference/bundler-actions/getUserOperationByHash - * - * @param args {@link GetUserOperationByHash} UserOpHash that was returned by {@link sendUserOperation} - * @returns userOperation along with entryPoint, transactionHash, blockHash, blockNumber if found or null - * - * @example - * import { createClient } from "viem" - * import { bundlerActions } from "permissionless" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http(BUNDLER_URL) - * }).extend(bundlerActions) - * - * await bundlerClient.getUserOperationByHash(userOpHash) - * - */ - getUserOperationByHash: ( - args: Prettify - ) => Promise> | null> - /** - * - * Returns the user operation receipt from userOpHash - * - * - Docs: https://docs.pimlico.io/permissionless/reference/bundler-actions/getUserOperationReceipt - * - * @param args {@link GetUserOperationReceiptParameters} UserOpHash that was returned by {@link sendUserOperation} - * @returns user operation receipt {@link GetUserOperationReceiptReturnType} if found or null - * - * @example - * import { createClient } from "viem" - * import { bundlerActions } from "permissionless" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http(BUNDLER_URL) - * }).extend(bundlerActions) - * - * await bundlerClient.getUserOperationReceipt({hash: userOpHash}) - * - */ - getUserOperationReceipt: ( - args: Prettify - ) => Promise | null> - - /** - * Waits for the User Operation to be included on a [Block](https://viem.sh/docs/glossary/terms.html#block) (one confirmation), and then returns the [User Operation Receipt](https://docs.pimlico.io/permissionless/reference/bundler-actions/getUserOperationReceipt). - * - * - Docs: https://docs.pimlico.io/permissionless/reference/bundler-actions/waitForUserOperationReceipt - * - * @param client - Bundler Client to use - * @param parameters - {@link WaitForUserOperationReceiptParameters} - * @returns The transaction receipt. {@link GetUserOperationReceiptReturnType} - * - * @example - * import { createBundlerClient, waitForUserOperationReceipt, http } from 'viem' - * import { mainnet } from 'viem/chains' - * - * const bundlerClient = createBundlerClient({ - * chain: mainnet, - * transport: http(), - * }) - * const userOperationReceipt = await bundlerClient.waitForUserOperationReceipt({ - * hash: '0x4ca7ee652d57678f26e887c149ab0735f41de37bcad58c9f6d3ed5824f15b74d', - * }) - */ - waitForUserOperationReceipt: ( - args: Prettify - ) => Promise> -} - -const bundlerActions = - (entryPointAddress: entryPoint) => - (client: Client): BundlerActions => ({ - sendUserOperation: async ( - args: Omit, "entryPoint"> - ): Promise => - sendUserOperation(client as BundlerClient, { - ...args, - entryPoint: entryPointAddress - }), - estimateUserOperationGas: ( - args: Omit< - EstimateUserOperationGasParameters, - "entryPoint" - >, - stateOverrides?: StateOverrides - ) => - estimateUserOperationGas( - client as BundlerClient, - { ...args, entryPoint: entryPointAddress }, - stateOverrides - ), - supportedEntryPoints: (): Promise => - supportedEntryPoints(client as BundlerClient), - chainId: () => chainId(client as BundlerClient), - getUserOperationByHash: (args: GetUserOperationByHashParameters) => - getUserOperationByHash( - client as BundlerClient, - args - ), - getUserOperationReceipt: (args: GetUserOperationReceiptParameters) => - getUserOperationReceipt(client as BundlerClient, args), - waitForUserOperationReceipt: ( - args: WaitForUserOperationReceiptParameters - ) => - waitForUserOperationReceipt( - client as BundlerClient, - args - ) - }) - -export { bundlerActions } diff --git a/packages/permissionless/clients/decorators/pimlico.ts b/packages/permissionless/clients/decorators/pimlico.ts index 69fc4904..69baa268 100644 --- a/packages/permissionless/clients/decorators/pimlico.ts +++ b/packages/permissionless/clients/decorators/pimlico.ts @@ -1,4 +1,8 @@ -import type { Client, Hash } from "viem" +import type { Client, Hash, Prettify } from "viem" +import type { + entryPoint06Address, + entryPoint07Address +} from "viem/account-abstraction" import { type SendCompressedUserOperationParameters, type ValidateSponsorshipPolicies, @@ -20,11 +24,15 @@ import { type SponsorUserOperationReturnType, sponsorUserOperation } from "../../actions/pimlico/sponsorUserOperation" -import type { Prettify } from "../../types/" -import type { EntryPoint } from "../../types/entrypoint" -import type { PimlicoBundlerClient, PimlicoPaymasterClient } from "../pimlico" -export type PimlicoBundlerActions = { +export type PimlicoActions< + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address = + | typeof entryPoint06Address + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7" = "0.6" | "0.7" +> = { /** * Returns the live gas prices that you can use to send a user operation. * @@ -98,144 +106,56 @@ export type PimlicoBundlerActions = { Omit > ) => Promise -} - -export const pimlicoBundlerActions = - (entryPointAddress: entryPoint) => - (client: Client): PimlicoBundlerActions => ({ - getUserOperationGasPrice: async () => - getUserOperationGasPrice( - client as PimlicoBundlerClient - ), - getUserOperationStatus: async ( - args: GetUserOperationStatusParameters - ) => - getUserOperationStatus( - client as PimlicoBundlerClient, - args - ), - sendCompressedUserOperation: async ( - args: Omit - ) => - sendCompressedUserOperation( - client as PimlicoBundlerClient, - { - ...args, - entryPoint: entryPointAddress - } - ) - }) - -export type PimlicoPaymasterClientActions = { - /** - * Returns paymasterAndData & updated gas parameters required to sponsor a userOperation. - * - * https://docs.pimlico.io/permissionless/reference/pimlico-paymaster-actions/sponsorUserOperation - * - * @param args {@link PimlicoSponsorUserOperationParameters} UserOperation you want to sponsor & entryPoint. - * @returns paymasterAndData & updated gas parameters, see {@link SponsorUserOperationReturnType} - * - * @example - * import { createClient } from "viem" - * import { sponsorUserOperation } from "permissionless/actions/pimlico" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http("https://api.pimlico.io/v2/goerli/rpc?apikey=YOUR_API_KEY_HERE") - * }).extend(pimlicoPaymasterActions) - * - * await bundlerClient.sponsorUserOperation(bundlerClient, { - * userOperation: userOperationWithDummySignature, - * entryPoint: entryPoint - * }}) - * - */ sponsorUserOperation: ( args: Omit< - PimlicoSponsorUserOperationParameters, + PimlicoSponsorUserOperationParameters< + entryPointAddress, + entryPointVersion + >, "entryPoint" > - ) => Promise>> - + ) => Promise>> validateSponsorshipPolicies: ( args: Prettify< Omit< - ValidateSponsorshipPoliciesParameters, + ValidateSponsorshipPoliciesParameters< + entryPointAddress, + entryPointVersion + >, "entryPoint" > > ) => Promise[]> } -/** - * Returns valid sponsorship policies for a userOperation from the list of ids passed - * - Docs: https://docs.pimlico.io/permissionless/reference/pimlico-paymaster-actions/ValidateSponsorshipPolicies - * - * @param args {@link ValidateSponsorshipPoliciesParameters} UserOperation you want to sponsor & entryPoint. - * @returns valid sponsorship policies, see {@link ValidateSponsorshipPolicies} - * - * @example - * import { createClient } from "viem" - * import { validateSponsorshipPolicies } from "permissionless/actions/pimlico" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http("https://api.pimlico.io/v2/goerli/rpc?apikey=YOUR_API_KEY_HERE") - * }).extend(pimlicoPaymasterActions) - - * - * await bundlerClient.validateSponsorshipPolicies({ - * userOperation: userOperationWithDummySignature, - * entryPoint: entryPoint, - * sponsorshipPolicyIds: ["sp_shiny_puma"] - * }) - * Returns - * [ - * { - * sponsorshipPolicyId: "sp_shiny_puma", - * data: { - * name: "Shiny Puma", - * author: "Pimlico", - * icon: "...", - * description: "This policy is for testing purposes only" - * } - * } - * ] - */ -export const pimlicoPaymasterActions = - (entryPointAddress: entryPoint) => - (client: Client): PimlicoPaymasterClientActions => ({ - sponsorUserOperation: async ( - args: Omit< - PimlicoSponsorUserOperationParameters, - "entryPoint" - > - ) => - sponsorUserOperation( - client as PimlicoPaymasterClient, - { - ...args, - entryPoint: entryPointAddress - } - ), - validateSponsorshipPolicies: async ( - args: Omit< - ValidateSponsorshipPoliciesParameters, - "entryPoint" - > +export const pimlicoActions = + < + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address + >( + entryPointAddress: entryPointAddress + ) => + (client: Client): PimlicoActions => ({ + getUserOperationGasPrice: async () => getUserOperationGasPrice(client), + getUserOperationStatus: async ( + args: GetUserOperationStatusParameters + ) => getUserOperationStatus(client, args), + sendCompressedUserOperation: async ( + args: Omit ) => - validateSponsorshipPolicies( - client as PimlicoPaymasterClient, - { ...args, entryPoint: entryPointAddress } - ) + sendCompressedUserOperation(client, { + ...args, + entryPointAddress + }), + sponsorUserOperation: async (args) => + sponsorUserOperation(client, { + ...args, + entryPointAddress + }), + validateSponsorshipPolicies: async (args) => + validateSponsorshipPolicies(client, { + ...args, + entryPointAddress + }) }) - -/** - * TODO: Add support for pimlicoActions after we support all the actions of v2 of the Pimlico API. - */ -// export const pimlicoActions = (client: Client) => { -// return { -// ...pimlicoBundlerActions(client), -// ...pimlicoPaymasterActions(client) -// } -// } diff --git a/packages/permissionless/clients/decorators/smartAccount.test.ts b/packages/permissionless/clients/decorators/smartAccount.test.ts deleted file mode 100644 index d9f222ef..00000000 --- a/packages/permissionless/clients/decorators/smartAccount.test.ts +++ /dev/null @@ -1,55 +0,0 @@ -import { http } from "viem" -import { generatePrivateKey, privateKeyToAccount } from "viem/accounts" -import { foundry } from "viem/chains" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { getPublicClient } from "../../../permissionless-test/src/utils" -import { signerToSimpleSmartAccount } from "../../accounts" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" -import { createSmartAccountClient } from "../createSmartAccountClient" - -describe("createSmartAccountClient", () => { - testWithRpc("createSmartAccountClient_V06", async ({ rpc }) => { - const publicClient = getPublicClient(rpc.anvilRpc) - - const simpleSmartAccount = await signerToSimpleSmartAccount( - publicClient, - { - entryPoint: ENTRYPOINT_ADDRESS_V06, - signer: privateKeyToAccount(generatePrivateKey()) - } - ) - - const smartAccountClient = createSmartAccountClient({ - chain: foundry, - account: simpleSmartAccount, - bundlerTransport: http(rpc.altoRpc) - }) - - expect(smartAccountClient.account.address).toBe( - simpleSmartAccount.address - ) - }) - - testWithRpc("createSmartAccountClient_V07", async ({ rpc }) => { - const publicClient = getPublicClient(rpc.anvilRpc) - - const simpleSmartAccount = await signerToSimpleSmartAccount( - publicClient, - { - entryPoint: ENTRYPOINT_ADDRESS_V07, - signer: privateKeyToAccount(generatePrivateKey()) - } - ) - - const smartAccountClient = createSmartAccountClient({ - chain: foundry, - account: simpleSmartAccount, - bundlerTransport: http(rpc.altoRpc) - }) - - expect(smartAccountClient.account.address).toBe( - simpleSmartAccount.address - ) - }) -}) diff --git a/packages/permissionless/clients/decorators/smartAccount.ts b/packages/permissionless/clients/decorators/smartAccount.ts deleted file mode 100644 index 0a20aac7..00000000 --- a/packages/permissionless/clients/decorators/smartAccount.ts +++ /dev/null @@ -1,610 +0,0 @@ -import type { - Abi, - Chain, - Client, - ContractFunctionArgs, - ContractFunctionName, - DeployContractParameters, - Hash, - SendTransactionParameters, - Transport, - TypedData, - WriteContractParameters -} from "viem" -import type { SmartAccount } from "../../accounts/types" -import { - type SendTransactionsWithPaymasterParameters, - sendTransactions -} from "../../actions/smartAccount" -import { - type DeployContractParametersWithPaymaster, - deployContract -} from "../../actions/smartAccount/deployContract" -import { - type Middleware, - type PrepareUserOperationRequestReturnType, - prepareUserOperationRequest -} from "../../actions/smartAccount/prepareUserOperationRequest" -import { - type SendTransactionWithPaymasterParameters, - sendTransaction -} from "../../actions/smartAccount/sendTransaction" -import { - type SendUserOperationParameters, - sendUserOperation -} from "../../actions/smartAccount/sendUserOperation" -import { signMessage } from "../../actions/smartAccount/signMessage" -import { signTypedData } from "../../actions/smartAccount/signTypedData" -import { - type WriteContractWithPaymasterParameters, - writeContract -} from "../../actions/smartAccount/writeContract" -import type { Prettify } from "../../types/" -import type { StateOverrides } from "../../types/bundler" -import type { EntryPoint } from "../../types/entrypoint" - -export type SmartAccountActions< - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends SmartAccount | undefined = - | SmartAccount - | undefined -> = { - /** - * Creates, signs, and sends a new transaction to the network. - * This function also allows you to sponsor this transaction if sender is a smartAccount - * - * - Docs: https://viem.sh/docs/actions/wallet/sendTransaction.html - * - Examples: https://stackblitz.com/github/wagmi-dev/viem/tree/main/examples/transactions/sending-transactions - * - JSON-RPC Methods: - * - JSON-RPC Accounts: [`eth_sendTransaction`](https://ethereum.org/en/developers/docs/apis/json-rpc/#eth_sendtransaction) - * - Local Accounts: [`eth_sendRawTransaction`](https://ethereum.org/en/developers/docs/apis/json-rpc/#eth_sendrawtransaction) - * - * @param args - {@link SendTransactionParameters} - * @returns The [Transaction](https://viem.sh/docs/glossary/terms.html#transaction) hash. {@link SendTransactionReturnType} - * - * @example - * import { createWalletClient, custom } from 'viem' - * import { mainnet } from 'viem/chains' - * - * const client = createWalletClient({ - * chain: mainnet, - * transport: custom(window.ethereum), - * }) - * const hash = await client.sendTransaction({ - * account: '0xA0Cf798816D4b9b9866b5330EEa46a18382f251e', - * to: '0x70997970c51812dc3a010c7d01b50e0d17dc79c8', - * value: 1000000000000000000n, - * }) - * - * @example - * // Account Hoisting - * import { createWalletClient, http } from 'viem' - * import { privateKeyToAccount } from 'viem/accounts' - * import { mainnet } from 'viem/chains' - * - * const client = createWalletClient({ - * account: privateKeyToAccount('0x…'), - * chain: mainnet, - * transport: http(), - * }) - * const hash = await client.sendTransaction({ - * to: '0x70997970c51812dc3a010c7d01b50e0d17dc79c8', - * value: 1000000000000000000n, - * }) - */ - sendTransaction: ( - args: SendTransactionParameters - ) => Promise - /** - * Calculates an Ethereum-specific signature in [EIP-191 format](https://eips.ethereum.org/EIPS/eip-191): `keccak256("\x19Ethereum Signed Message:\n" + len(message) + message))`. - * - * - Docs: https://viem.sh/docs/actions/wallet/signMessage.html - * - JSON-RPC Methods: - * - JSON-RPC Accounts: [`personal_sign`](https://docs.metamask.io/guide/signing-data.html#personal-sign) - * - Local Accounts: Signs locally. No JSON-RPC request. - * - * With the calculated signature, you can: - * - use [`verifyMessage`](https://viem.sh/docs/utilities/verifyMessage.html) to verify the signature, - * - use [`recoverMessageAddress`](https://viem.sh/docs/utilities/recoverMessageAddress.html) to recover the signing address from a signature. - * - * @param args - {@link SignMessageParameters} - * @returns The signed message. {@link SignMessageReturnType} - * - * @example - * import { createWalletClient, custom } from 'viem' - * import { mainnet } from 'viem/chains' - * - * const client = createWalletClient({ - * chain: mainnet, - * transport: custom(window.ethereum), - * }) - * const signature = await client.signMessage({ - * account: '0xA0Cf798816D4b9b9866b5330EEa46a18382f251e', - * message: 'hello world', - * }) - * - * @example - * // Account Hoisting - * import { createWalletClient, http } from 'viem' - * import { privateKeyToAccount } from 'viem/accounts' - * import { mainnet } from 'viem/chains' - * - * const client = createWalletClient({ - * account: privateKeyToAccount('0x…'), - * chain: mainnet, - * transport: http(), - * }) - * const signature = await client.signMessage({ - * message: 'hello world', - * }) - */ - signMessage: ( - args: Parameters< - typeof signMessage - >[1] - ) => ReturnType< - typeof signMessage - > - /** - * Signs typed data and calculates an Ethereum-specific signature in [EIP-191 format](https://eips.ethereum.org/EIPS/eip-191): `keccak256("\x19Ethereum Signed Message:\n" + len(message) + message))`. - * - * - Docs: https://viem.sh/docs/actions/wallet/signTypedData.html - * - JSON-RPC Methods: - * - JSON-RPC Accounts: [`eth_signTypedData_v4`](https://docs.metamask.io/guide/signing-data.html#signtypeddata-v4) - * - Local Accounts: Signs locally. No JSON-RPC request. - * - * @param client - Client to use - * @param args - {@link SignTypedDataParameters} - * @returns The signed data. {@link SignTypedDataReturnType} - * - * @example - * import { createWalletClient, custom } from 'viem' - * import { mainnet } from 'viem/chains' - * - * const client = createWalletClient({ - * chain: mainnet, - * transport: custom(window.ethereum), - * }) - * const signature = await client.signTypedData({ - * account: '0xA0Cf798816D4b9b9866b5330EEa46a18382f251e', - * domain: { - * name: 'Ether Mail', - * version: '1', - * chainId: 1, - * verifyingContract: '0xCcCCccccCCCCcCCCCCCcCcCccCcCCCcCcccccccC', - * }, - * types: { - * Person: [ - * { name: 'name', type: 'string' }, - * { name: 'wallet', type: 'address' }, - * ], - * Mail: [ - * { name: 'from', type: 'Person' }, - * { name: 'to', type: 'Person' }, - * { name: 'contents', type: 'string' }, - * ], - * }, - * primaryType: 'Mail', - * message: { - * from: { - * name: 'Cow', - * wallet: '0xCD2a3d9F938E13CD947Ec05AbC7FE734Df8DD826', - * }, - * to: { - * name: 'Bob', - * wallet: '0xbBbBBBBbbBBBbbbBbbBbbbbBBbBbbbbBbBbbBBbB', - * }, - * contents: 'Hello, Bob!', - * }, - * }) - * - * @example - * // Account Hoisting - * import { createWalletClient, http } from 'viem' - * import { privateKeyToAccount } from 'viem/accounts' - * import { mainnet } from 'viem/chains' - * - * const client = createWalletClient({ - * account: privateKeyToAccount('0x…'), - * chain: mainnet, - * transport: http(), - * }) - * const signature = await client.signTypedData({ - * domain: { - * name: 'Ether Mail', - * version: '1', - * chainId: 1, - * verifyingContract: '0xCcCCccccCCCCcCCCCCCcCcCccCcCCCcCcccccccC', - * }, - * types: { - * Person: [ - * { name: 'name', type: 'string' }, - * { name: 'wallet', type: 'address' }, - * ], - * Mail: [ - * { name: 'from', type: 'Person' }, - * { name: 'to', type: 'Person' }, - * { name: 'contents', type: 'string' }, - * ], - * }, - * primaryType: 'Mail', - * message: { - * from: { - * name: 'Cow', - * wallet: '0xCD2a3d9F938E13CD947Ec05AbC7FE734Df8DD826', - * }, - * to: { - * name: 'Bob', - * wallet: '0xbBbBBBBbbBBBbbbBbbBbbbbBBbBbbbbBbBbbBBbB', - * }, - * contents: 'Hello, Bob!', - * }, - * }) - */ - signTypedData: < - const TTypedData extends TypedData | { [key: string]: unknown }, - TPrimaryType extends string - >( - args: Parameters< - typeof signTypedData< - entryPoint, - TTypedData, - TPrimaryType, - TTransport, - TChain, - TSmartAccount - > - >[1] - ) => ReturnType< - typeof signTypedData< - entryPoint, - TTypedData, - TPrimaryType, - TTransport, - TChain, - TSmartAccount - > - > - /** - * Deploys a contract to the network, given bytecode and constructor arguments. - * This function also allows you to sponsor this transaction if sender is a smartAccount - * - * - Docs: https://viem.sh/docs/contract/deployContract.html - * - Examples: https://stackblitz.com/github/wagmi-dev/viem/tree/main/examples/contracts/deploying-contracts - * - * @param args - {@link DeployContractParameters} - * @returns The [Transaction](https://viem.sh/docs/glossary/terms.html#transaction) hash. {@link DeployContractReturnType} - * - * @example - * import { createWalletClient, http } from 'viem' - * import { privateKeyToAccount } from 'viem/accounts' - * import { mainnet } from 'viem/chains' - * - * const client = createWalletClient({ - * account: privateKeyToAccount('0x…'), - * chain: mainnet, - * transport: http(), - * }) - * const hash = await client.deployContract({ - * abi: [], - * account: '0x…, - * bytecode: '0x608060405260405161083e38038061083e833981016040819052610...', - * }) - */ - deployContract: < - const TAbi extends Abi | readonly unknown[], - TChainOverride extends Chain | undefined = undefined - >( - args: DeployContractParameters< - TAbi, - TChain, - TSmartAccount, - TChainOverride - > - ) => Promise - /** - * Executes a write function on a contract. - * This function also allows you to sponsor this transaction if sender is a smartAccount - * - * - Docs: https://viem.sh/docs/contract/writeContract.html - * - Examples: https://stackblitz.com/github/wagmi-dev/viem/tree/main/examples/contracts/writing-to-contracts - * - * A "write" function on a Solidity contract modifies the state of the blockchain. These types of functions require gas to be executed, and hence a [Transaction](https://viem.sh/docs/glossary/terms.html) is needed to be broadcast in order to change the state. - * - * Internally, uses a [Wallet Client](https://viem.sh/docs/clients/wallet.html) to call the [`sendTransaction` action](https://viem.sh/docs/actions/wallet/sendTransaction.html) with [ABI-encoded `data`](https://viem.sh/docs/contract/encodeFunctionData.html). - * - * __Warning: The `write` internally sends a transaction – it does not validate if the contract write will succeed (the contract may throw an error). It is highly recommended to [simulate the contract write with `contract.simulate`](https://viem.sh/docs/contract/writeContract.html#usage) before you execute it.__ - * - * @param args - {@link WriteContractParameters} - * @returns A [Transaction Hash](https://viem.sh/docs/glossary/terms.html#hash). {@link WriteContractReturnType} - * - * @example - * import { createWalletClient, custom, parseAbi } from 'viem' - * import { mainnet } from 'viem/chains' - * - * const client = createWalletClient({ - * chain: mainnet, - * transport: custom(window.ethereum), - * }) - * const hash = await client.writeContract({ - * address: '0xFBA3912Ca04dd458c843e2EE08967fC04f3579c2', - * abi: parseAbi(['function mint(uint32 tokenId) nonpayable']), - * functionName: 'mint', - * args: [69420], - * }) - * - * @example - * // With Validation - * import { createWalletClient, custom, parseAbi } from 'viem' - * import { mainnet } from 'viem/chains' - * - * const client = createWalletClient({ - * chain: mainnet, - * transport: custom(window.ethereum), - * }) - * const { request } = await client.simulateContract({ - * address: '0xFBA3912Ca04dd458c843e2EE08967fC04f3579c2', - * abi: parseAbi(['function mint(uint32 tokenId) nonpayable']), - * functionName: 'mint', - * args: [69420], - * } - * const hash = await client.writeContract(request) - */ - writeContract: < - const TAbi extends Abi | readonly unknown[], - TFunctionName extends ContractFunctionName< - TAbi, - "nonpayable" | "payable" - > = ContractFunctionName, - TArgs extends ContractFunctionArgs< - TAbi, - "nonpayable" | "payable", - TFunctionName - > = ContractFunctionArgs, - TChainOverride extends Chain | undefined = undefined - >( - args: WriteContractParameters< - TAbi, - TFunctionName, - TArgs, - TChain, - TSmartAccount, - TChainOverride - > - ) => ReturnType< - typeof writeContract< - entryPoint, - TTransport, - TChain, - TSmartAccount, - TAbi, - TFunctionName, - TArgs, - TChainOverride - > - > - prepareUserOperationRequest: ( - args: Parameters< - typeof prepareUserOperationRequest< - entryPoint, - TTransport, - TChain, - TSmartAccount - > - >[1], - stateOverrides?: StateOverrides - ) => Promise>> - sendUserOperation: ( - args: Prettify< - Parameters< - typeof sendUserOperation< - entryPoint, - TTransport, - TChain, - TSmartAccount - > - >[1] - > - ) => Promise - /** - * Creates, signs, and sends a new transaction to the network. - * This function also allows you to sponsor this transaction if sender is a smartAccount - * - * - Docs: https://viem.sh/docs/actions/wallet/sendTransaction.html - * - Examples: https://stackblitz.com/github/wagmi-dev/viem/tree/main/examples/transactions/sending-transactions - * - JSON-RPC Methods: - * - JSON-RPC Accounts: [`eth_sendTransaction`](https://ethereum.org/en/developers/docs/apis/json-rpc/#eth_sendtransaction) - * - Local Accounts: [`eth_sendRawTransaction`](https://ethereum.org/en/developers/docs/apis/json-rpc/#eth_sendrawtransaction) - * - * @param args - {@link SendTransactionParameters} - * @returns The [Transaction](https://viem.sh/docs/glossary/terms.html#transaction) hash. {@link SendTransactionReturnType} - * - * @example - * import { createWalletClient, custom } from 'viem' - * import { mainnet } from 'viem/chains' - * - * const client = createWalletClient({ - * chain: mainnet, - * transport: custom(window.ethereum), - * }) - * const hash = await client.sendTransaction([{ - * account: '0xA0Cf798816D4b9b9866b5330EEa46a18382f251e', - * to: '0x70997970c51812dc3a010c7d01b50e0d17dc79c8', - * value: 1000000000000000000n - * }, { - * to: '0x61897970c51812dc3a010c7d01b50e0d17dc1234', - * value: 10000000000000000n - * }) - * - * @example - * // Account Hoisting - * import { createWalletClient, http } from 'viem' - * import { privateKeyToAccount } from 'viem/accounts' - * import { mainnet } from 'viem/chains' - * - * const client = createWalletClient({ - * account: privateKeyToAccount('0x…'), - * chain: mainnet, - * transport: http(), - * }) - * const hash = await client.sendTransaction([{ - * to: '0x70997970c51812dc3a010c7d01b50e0d17dc79c8', - * value: 1000000000000000000n - * }, { - * to: '0x61897970c51812dc3a010c7d01b50e0d17dc1234', - * value: 10000000000000000n - * }]) - */ - sendTransactions: ( - args: Prettify< - SendTransactionsWithPaymasterParameters - > - ) => ReturnType< - typeof sendTransactions - > -} - -export function smartAccountActions({ - middleware -}: Middleware) { - return < - TTransport extends Transport, - TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends SmartAccount | undefined = - | SmartAccount - | undefined - >( - client: Client - ): SmartAccountActions => ({ - prepareUserOperationRequest: (args, stateOverrides) => - prepareUserOperationRequest( - client, - { - ...args, - middleware - }, - stateOverrides - ), - deployContract: < - const TAbi extends Abi | readonly unknown[], - TChainOverride extends Chain | undefined = undefined - >( - args: DeployContractParameters< - TAbi, - TChain, - TSmartAccount, - TChainOverride - > - ) => - deployContract< - entryPoint, - TTransport, - TChain, - TAbi, - TSmartAccount, - TChainOverride - >(client, { - ...args, - middleware - } as unknown as DeployContractParametersWithPaymaster< - entryPoint, - TAbi, - TChain, - TSmartAccount, - TChainOverride - >), - sendTransaction: (args) => - sendTransaction( - client, - { - ...args, - middleware - } as SendTransactionWithPaymasterParameters< - entryPoint, - TChain, - TSmartAccount - > - ), - sendTransactions: (args) => - sendTransactions( - client, - { - ...args, - middleware - } - ), - sendUserOperation: (args) => - sendUserOperation( - client, - { - ...args, - middleware - } as SendUserOperationParameters - ), - signMessage: (args) => - signMessage( - client, - args - ), - signTypedData: < - const TTypedData extends TypedData | { [key: string]: unknown }, - TPrimaryType extends string - >( - args: Parameters< - typeof signTypedData< - entryPoint, - TTypedData, - TPrimaryType, - TTransport, - TChain, - TSmartAccount - > - >[1] - ) => - signTypedData< - entryPoint, - TTypedData, - TPrimaryType, - TTransport, - TChain, - TSmartAccount - >(client, args), - writeContract: < - const TAbi extends Abi | readonly unknown[], - TFunctionName extends ContractFunctionName< - TAbi, - "nonpayable" | "payable" - > = ContractFunctionName, - TArgs extends ContractFunctionArgs< - TAbi, - "nonpayable" | "payable", - TFunctionName - > = ContractFunctionArgs< - TAbi, - "nonpayable" | "payable", - TFunctionName - >, - TChainOverride extends Chain | undefined = undefined - >( - args: WriteContractParameters< - TAbi, - TFunctionName, - TArgs, - TChain, - TSmartAccount, - TChainOverride - > - ) => - writeContract(client, { - ...args, - middleware - } as WriteContractWithPaymasterParameters< - entryPoint, - TChain, - TSmartAccount, - TAbi - >) - }) -} diff --git a/packages/permissionless/clients/decorators/stackup.ts b/packages/permissionless/clients/decorators/stackup.ts deleted file mode 100644 index 00f7063d..00000000 --- a/packages/permissionless/clients/decorators/stackup.ts +++ /dev/null @@ -1,77 +0,0 @@ -import type { Address, Client } from "viem" -import { - type AccountsParameters, - accounts -} from "../../actions/stackup/accounts" -import { - type SponsorUserOperationParameters, - type SponsorUserOperationReturnType, - sponsorUserOperation -} from "../../actions/stackup/sponsorUserOperation" -import type { EntryPoint } from "../../types/entrypoint" -import type { StackupPaymasterClient } from "../stackup" - -export type StackupPaymasterClientActions = { - /** - * Returns paymasterAndData & updated gas parameters required to sponsor a userOperation. - * - * https://docs.stackup.sh/docs/paymaster-api-rpc-methods#pm_sponsoruseroperation - * - * @param args {@link SponsorUserOperationParameters} UserOperation you want to sponsor & entryPoint. - * @returns paymasterAndData & updated gas parameters, see {@link SponsorUserOperationReturnType} - * - * @example - * import { createClient } from "viem" - * import { stackupPaymasterActions } from "permissionless/actions/stackup" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http("https://api.stackup.sh/v1/paymaster/YOUR_API_KEY_HERE") - * }).extend(stackupPaymasterActions) - * - * await bundlerClient.sponsorUserOperation(bundlerClient, { - * userOperation: userOperationWithDummySignature, - * entryPoint: entryPoint - * }}) - * - */ - sponsorUserOperation: ( - args: Omit, "entrypoint"> - ) => Promise> - - /** - * Returns all the Paymaster addresses associated with an EntryPoint that’s owned by this service. - * - * https://docs.stackup.sh/docs/paymaster-api-rpc-methods#pm_accounts - * - * @param args {@link AccountsParameters} entryPoint for which you want to get list of supported paymasters. - * @returns paymaster addresses - * - * @example - * import { createClient } from "viem" - * import { stackupPaymasterActions } from "permissionless/actions/stackup" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http("https://api.stackup.sh/v1/paymaster/YOUR_API_KEY_HERE") - * }).extend(stackupPaymasterActions) - * - * await bundlerClient.accounts(bundlerClient, { - * entryPoint: entryPoint - * }}) - * - */ - accounts: (args: AccountsParameters) => Promise -} - -export const stackupPaymasterActions = - (entryPointAddress: entryPoint) => - (client: Client): StackupPaymasterClientActions => ({ - sponsorUserOperation: async (args) => - sponsorUserOperation(client as StackupPaymasterClient, { - ...args, - entryPoint: entryPointAddress - }), - accounts: async (args) => - accounts(client as StackupPaymasterClient, args) - }) diff --git a/packages/permissionless/clients/pimlico.ts b/packages/permissionless/clients/pimlico.ts index 1eda6396..2452b445 100644 --- a/packages/permissionless/clients/pimlico.ts +++ b/packages/permissionless/clients/pimlico.ts @@ -1,39 +1,71 @@ import type { - Account, + BundlerRpcSchema, Chain, Client, + ClientConfig, + Prettify, PublicClientConfig, + RpcSchema, Transport } from "viem" import { createClient } from "viem" -import type { EntryPoint } from "../types/entrypoint" -import type { - PimlicoBundlerRpcSchema, - PimlicoPaymasterRpcSchema -} from "../types/pimlico" -import { type BundlerActions, bundlerActions } from "./decorators/bundler" import { - type PimlicoBundlerActions, - type PimlicoPaymasterClientActions, - pimlicoBundlerActions, - pimlicoPaymasterActions -} from "./decorators/pimlico" + type BundlerActions, + type SmartAccount, + bundlerActions, + type entryPoint06Address, + entryPoint07Address +} from "viem/account-abstraction" +import type { PimlicoRpcSchema } from "../types/pimlico" +import { pimlicoActions } from "./decorators/pimlico" -export type PimlicoBundlerClient = Client< - Transport, - Chain | undefined, - Account | undefined, - PimlicoBundlerRpcSchema, - PimlicoBundlerActions & BundlerActions +export type PimlicoClient< + transport extends Transport = Transport, + chain extends Chain | undefined = Chain | undefined, + account extends SmartAccount | undefined = SmartAccount | undefined, + client extends Client | undefined = Client | undefined, + rpcSchema extends RpcSchema | undefined = undefined +> = Prettify< + Client< + transport, + chain extends Chain + ? chain + : // biome-ignore lint/suspicious/noExplicitAny: We need any to infer the chain type + client extends Client + ? chain + : undefined, + account, + rpcSchema extends RpcSchema + ? [...BundlerRpcSchema, ...PimlicoRpcSchema, ...rpcSchema] + : [...BundlerRpcSchema, ...PimlicoRpcSchema], + BundlerActions + > > -export type PimlicoPaymasterClient = Client< - Transport, - Chain | undefined, - Account | undefined, - PimlicoPaymasterRpcSchema, - PimlicoPaymasterClientActions -> +export type PimlicoClientConfig< + transport extends Transport = Transport, + chain extends Chain | undefined = Chain | undefined, + account extends SmartAccount | undefined = SmartAccount | undefined, + rpcSchema extends RpcSchema | undefined = undefined, + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address + | undefined = undefined +> = Prettify< + Pick< + ClientConfig, + | "account" + | "cacheTime" + | "chain" + | "key" + | "name" + | "pollingInterval" + | "rpcSchema" + | "transport" + > +> & { + entryPointAddress?: entryPointAddress +} /** * Creates a pimlico specific Bundler Client with a given [Transport](https://viem.sh/docs/clients/intro.html) configured for a [Chain](https://viem.sh/docs/clients/chains.html). @@ -54,16 +86,33 @@ export type PimlicoPaymasterClient = Client< * transport: http("https://api.pimlico.io/v2/goerli/rpc?apikey=YOUR_API_KEY_HERE"), * }) */ -export const createPimlicoBundlerClient = < - entryPoint extends EntryPoint, - transport extends Transport = Transport, - chain extends Chain | undefined = undefined +export function createPimlicoBundlerClient< + transport extends Transport, + chain extends Chain | undefined = undefined, + account extends SmartAccount | undefined = undefined, + client extends Client | undefined = undefined, + rpcSchema extends RpcSchema | undefined = undefined, + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address = typeof entryPoint07Address >( - parameters: PublicClientConfig & { - entryPoint: entryPoint - } -): PimlicoBundlerClient => { - const { key = "public", name = "Pimlico Bundler Client" } = parameters + parameters: PimlicoClientConfig< + transport, + chain, + account, + rpcSchema, + entryPointAddress + > +): PimlicoClient + +export function createPimlicoBundlerClient( + parameters: PimlicoClientConfig +): PimlicoClient { + const { + key = "public", + name = "Pimlico Bundler Client", + entryPointAddress + } = parameters const client = createClient({ ...parameters, key, @@ -71,44 +120,6 @@ export const createPimlicoBundlerClient = < type: "pimlicoBundlerClient" }) return client - .extend(bundlerActions(parameters.entryPoint)) - .extend(pimlicoBundlerActions(parameters.entryPoint)) -} - -/** - * Creates a pimlico specific Paymaster Client with a given [Transport](https://viem.sh/docs/clients/intro.html) configured for a [Chain](https://viem.sh/docs/clients/chains.html). - * - * - Docs: https://docs.pimlico.io/permissionless/reference/clients/pimlicoPaymasterClient - * - * A Pimlico Paymaster Client is an interface to "pimlico paymaster endpoints" [JSON-RPC API](https://docs.pimlico.io/reference/verifying-paymaster/endpoints) methods such as sponsoring user operation, etc through Pimlico Paymaster Actions. - * - * @param config - {@link PublicClientConfig} - * @returns A Pimlico Paymaster Client. {@link PimlicoPaymasterClient} - * - * @example - * import { createPublicClient, http } from 'viem' - * import { mainnet } from 'viem/chains' - * - * const pimlicoPaymasterClient = createPimlicoPaymasterClient({ - * chain: mainnet, - * transport: http("https://api.pimlico.io/v2/goerli/rpc?apikey=YOUR_API_KEY_HERE"), - * }) - */ -export const createPimlicoPaymasterClient = < - entryPoint extends EntryPoint, - transport extends Transport = Transport, - chain extends Chain | undefined = undefined ->( - parameters: PublicClientConfig & { - entryPoint: entryPoint - } -): PimlicoPaymasterClient => { - const { key = "public", name = "Pimlico Paymaster Client" } = parameters - const client = createClient({ - ...parameters, - key, - name, - type: "pimlicoPaymasterClient" - }) - return client.extend(pimlicoPaymasterActions(parameters.entryPoint)) + .extend(bundlerActions) + .extend(pimlicoActions(entryPointAddress ?? entryPoint07Address)) } diff --git a/packages/permissionless/clients/stackup.ts b/packages/permissionless/clients/stackup.ts deleted file mode 100644 index 1c3aacd7..00000000 --- a/packages/permissionless/clients/stackup.ts +++ /dev/null @@ -1,63 +0,0 @@ -import { - type Account, - type Chain, - type Client, - type PublicClientConfig, - type Transport, - createClient -} from "viem" -import type { EntryPoint } from "../types/entrypoint" -import type { StackupPaymasterRpcSchema } from "../types/stackup" -import { type BundlerActions, bundlerActions } from "./decorators/bundler" -import { - type StackupPaymasterClientActions, - stackupPaymasterActions -} from "./decorators/stackup" - -export type StackupPaymasterClient = Client< - Transport, - Chain | undefined, - Account | undefined, - StackupPaymasterRpcSchema, - StackupPaymasterClientActions & BundlerActions -> - -/** - * Creates a Stackup specific Paymaster Client with a given [Transport](https://viem.sh/docs/clients/intro.html) configured for a [Chain](https://viem.sh/docs/clients/chains.html). - * - * - Docs: https://docs.pimlico.io/permissionless/reference/clients/stackupPaymasterClient - * - * A Stackup Paymaster Client is an interface to "stackup paymaster endpoints" [JSON-RPC API](https://docs.stackup.sh/docs/paymaster-api-rpc-methods) methods such as sponsoring user operation, etc through Stackup Paymaster Actions. - * - * @param config - {@link PublicClientConfig} - * @returns A Stackup Paymaster Client. {@link StackupPaymasterClient} - * - * @example - * import { createPublicClient, http } from 'viem' - * import { mainnet } from 'viem/chains' - * - * const stackupPaymasterClient = createStackupPaymasterClient({ - * chain: mainnet, - * transport: http("https://api.stackup.sh/v2/paymaster/YOUR_API_KEY_HERE"), - * }) - */ -export const createStackupPaymasterClient = < - entryPoint extends EntryPoint, - transport extends Transport = Transport, - chain extends Chain | undefined = undefined ->( - parameters: PublicClientConfig & { - entryPoint: entryPoint - } -): StackupPaymasterClient => { - const { key = "public", name = "Stackup Paymaster Client" } = parameters - const client = createClient({ - ...parameters, - key, - name, - type: "stackupPaymasterClient" - }) - return client - .extend(bundlerActions(parameters.entryPoint)) - .extend(stackupPaymasterActions(parameters.entryPoint)) -} diff --git a/packages/permissionless/errors/bundler.ts b/packages/permissionless/errors/bundler.ts deleted file mode 100644 index aac2e7c7..00000000 --- a/packages/permissionless/errors/bundler.ts +++ /dev/null @@ -1,62 +0,0 @@ -import { BaseError } from "viem" - -export type InvalidBeneficiaryAddressErrorType = - InvalidBeneficiaryAddressError & { - name: "InvalidBeneficiaryAddressError" - } -export class InvalidBeneficiaryAddressError extends BaseError { - static message = /aa9[01]/ - override name = "InvalidBeneficiaryAddressError" - constructor({ - cause, - docsPath - }: { - cause?: BaseError - docsPath?: string - }) { - super( - [ - "The bundler did not set a beneficiary address when bundling the user operation.", - "", - "Possible solutions:", - "• If you encounter this error when running self-hosted bundler, make sure you have configured the bundler correctly.", - "• If you are using a bundler provider, reach out to them.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type InvalidAggregatorErrorType = InvalidAggregatorError & { - name: "InvalidAggregatorError" -} -export class InvalidAggregatorError extends BaseError { - static message = /aa96/ - override name = "InvalidAggregatorError" - constructor({ - cause, - docsPath - }: { - cause?: BaseError - docsPath?: string - }) { - super( - [ - "The bundler tried to bundle the user operation with an invalid aggregator.", - "", - "Possible solutions:", - "• If you are using your own bundler, configure it to use a valid aggregator.", - "• If you are using a bundler provider, reach out to them.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} diff --git a/packages/permissionless/types/bundler.ts b/packages/permissionless/types/bundler.ts deleted file mode 100644 index 70355664..00000000 --- a/packages/permissionless/types/bundler.ts +++ /dev/null @@ -1,131 +0,0 @@ -import type { Address, Hash, Hex } from "viem" -import type { PartialBy } from "viem/types/utils" -import type { EntryPoint, GetEntryPointVersion } from "./entrypoint" -import type { UserOperationWithBigIntAsHex } from "./userOperation" - -export type BundlerRpcSchema = [ - { - Method: "eth_sendUserOperation" - Parameters: [ - userOperation: UserOperationWithBigIntAsHex< - GetEntryPointVersion - >, - entryPoint: entryPoint - ] - ReturnType: Hash - }, - { - Method: "eth_estimateUserOperationGas" - Parameters: [ - userOperation: GetEntryPointVersion extends "v0.6" - ? PartialBy< - UserOperationWithBigIntAsHex<"v0.6">, - | "callGasLimit" - | "preVerificationGas" - | "verificationGasLimit" - > - : PartialBy< - UserOperationWithBigIntAsHex<"v0.7">, - | "callGasLimit" - | "preVerificationGas" - | "verificationGasLimit" - | "paymasterVerificationGasLimit" - | "paymasterPostOpGasLimit" - >, - entryPoint: entryPoint, - stateOverrides?: StateOverrides - ] - ReturnType: GetEntryPointVersion extends "v0.6" - ? { - preVerificationGas: Hex - verificationGasLimit: Hex - callGasLimit: Hex - } - : { - preVerificationGas: Hex - verificationGasLimit: Hex - callGasLimit?: Hex | null - paymasterVerificationGasLimit?: Hex | null - paymasterPostOpGasLimit?: Hex | null - } - }, - { - Method: "eth_supportedEntryPoints" - Parameters: [] - ReturnType: Address[] - }, - { - Method: "eth_chainId" - Parameters: [] - ReturnType: Hex - }, - { - Method: "eth_getUserOperationByHash" - Parameters: [hash: Hash] - ReturnType: { - userOperation: UserOperationWithBigIntAsHex< - GetEntryPointVersion - > - entryPoint: entryPoint - transactionHash: Hash - blockHash: Hash - blockNumber: Hex - } - }, - { - Method: "eth_getUserOperationReceipt" - Parameters: [hash: Hash] - ReturnType: UserOperationReceiptWithBigIntAsHex - } -] - -type UserOperationReceiptWithBigIntAsHex = { - userOpHash: Hash - entryPoint: Address - sender: Address - nonce: Hex - paymaster?: Address - actualGasUsed: Hex - actualGasCost: Hex - success: boolean - reason?: string - receipt: { - transactionHash: Hex - transactionIndex: Hex - blockHash: Hash - blockNumber: Hex - from: Address - to: Address | null - cumulativeGasUsed: Hex - status: "0x0" | "0x1" - gasUsed: Hex - contractAddress: Address | null - logsBloom: Hex - effectiveGasPrice: Hex - } - logs: { - data: Hex - blockNumber: Hex - blockHash: Hash - transactionHash: Hash - logIndex: Hex - transactionIndex: Hex - address: Address - topics: [Hex, ...Hex[]] | [] - removed: boolean - }[] -} - -export type StateOverrides = { - [x: string]: { - balance?: bigint | undefined - nonce?: bigint | number | undefined - code?: Hex | undefined - state?: { - [x: Hex]: Hex - } - stateDiff?: { - [x: Hex]: Hex - } - } -} diff --git a/packages/permissionless/types/index.ts b/packages/permissionless/types/index.ts index a0f68677..78b80d52 100644 --- a/packages/permissionless/types/index.ts +++ b/packages/permissionless/types/index.ts @@ -4,18 +4,8 @@ import type { SmartAccountImplementation } from "viem/account-abstraction" -export type { PackedUserOperation } from "./userOperation" - export type IsUndefined = [undefined] extends [T] ? true : false -export type GetAccountParameterWithClient< - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends Account | undefined = Account | undefined -> = IsUndefined extends true - ? { account: Account; client?: Client } - : { client: Client; account?: Account } - export type PartialBy = Omit & Partial> export type PartialPick = Partial> diff --git a/packages/permissionless/types/pimlico.ts b/packages/permissionless/types/pimlico.ts index 839044ed..08b2ccfb 100644 --- a/packages/permissionless/types/pimlico.ts +++ b/packages/permissionless/types/pimlico.ts @@ -1,7 +1,9 @@ -import type { Address, Hash, Hex } from "viem" -import type { PartialBy } from "viem/types/utils" -import type { EntryPoint, GetEntryPointVersion } from "./entrypoint" -import type { UserOperationWithBigIntAsHex } from "./userOperation" +import type { Address, Hash, Hex, OneOf, PartialBy } from "viem" +import type { + UserOperation, + entryPoint06Address, + entryPoint07Address +} from "viem/account-abstraction" type PimlicoUserOperationGasPriceWithBigIntAsHex = { slow: { @@ -30,7 +32,12 @@ export type PimlicoUserOperationStatus = { transactionHash: Hash | null } -export type PimlicoBundlerRpcSchema = [ +export type PimlicoRpcSchema< + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address = typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7" = "0.7" +> = [ { Method: "pimlico_getUserOperationGasPrice" Parameters: [] @@ -49,34 +56,36 @@ export type PimlicoBundlerRpcSchema = [ entryPoint: Address ] ReturnType: Hash - } -] - -export type PimlicoPaymasterRpcSchema = [ + }, { Method: "pm_sponsorUserOperation" Parameters: [ - userOperation: GetEntryPointVersion extends "v0.6" - ? PartialBy< - UserOperationWithBigIntAsHex<"v0.6">, - | "callGasLimit" - | "preVerificationGas" - | "verificationGasLimit" - > - : PartialBy< - UserOperationWithBigIntAsHex<"v0.7">, - | "callGasLimit" - | "preVerificationGas" - | "verificationGasLimit" - | "paymasterVerificationGasLimit" - | "paymasterPostOpGasLimit" - >, - entryPoint: entryPoint, + userOperation: OneOf< + | (entryPointVersion extends "0.6" + ? PartialBy< + UserOperation<"0.6", Hex>, + | "callGasLimit" + | "preVerificationGas" + | "verificationGasLimit" + > + : never) + | (entryPointVersion extends "0.7" + ? PartialBy< + UserOperation<"0.7", Hex>, + | "callGasLimit" + | "preVerificationGas" + | "verificationGasLimit" + | "paymasterVerificationGasLimit" + | "paymasterPostOpGasLimit" + > + : never) + >, + entryPoint: entryPointAddress, metadata?: { sponsorshipPolicyId?: string } ] - ReturnType: GetEntryPointVersion extends "v0.6" + ReturnType: entryPointAddress extends "0.6" ? { paymasterAndData: Hex preVerificationGas: Hex @@ -101,10 +110,8 @@ export type PimlicoPaymasterRpcSchema = [ { Method: "pm_validateSponsorshipPolicies" Parameters: [ - userOperation: UserOperationWithBigIntAsHex< - GetEntryPointVersion - >, - entryPoint: entryPoint, + userOperation: UserOperation, + entryPoint: entryPointAddress, sponsorshipPolicyIds: string[] ] ReturnType: { diff --git a/packages/permissionless/types/stackup.ts b/packages/permissionless/types/stackup.ts deleted file mode 100644 index 197a95f6..00000000 --- a/packages/permissionless/types/stackup.ts +++ /dev/null @@ -1,63 +0,0 @@ -import type { Address, Hex } from "viem" -import type { PartialBy } from "viem/types/utils" -import type { ENTRYPOINT_ADDRESS_V06_TYPE, EntryPoint } from "./entrypoint" -import type { UserOperationWithBigIntAsHex } from "./userOperation" - -interface StackupPaymasterContextType { - type: "erc20token" | "payg" -} - -export type StackupPaymasterContext = - | (StackupPaymasterContextType & { type: "erc20token"; token: string }) - | (StackupPaymasterContextType & { type: "payg" }) - -export type StackupPaymasterRpcSchema = [ - { - Method: "pm_sponsorUserOperation" - Parameters: [ - userOperation: entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE - ? PartialBy< - UserOperationWithBigIntAsHex<"v0.6">, - | "callGasLimit" - | "preVerificationGas" - | "verificationGasLimit" - > - : PartialBy< - UserOperationWithBigIntAsHex<"v0.7">, - | "callGasLimit" - | "preVerificationGas" - | "verificationGasLimit" - | "paymasterVerificationGasLimit" - | "paymasterPostOpGasLimit" - >, - entryPoint: entryPoint, - context: StackupPaymasterContext - ] - ReturnType: entryPoint extends ENTRYPOINT_ADDRESS_V06_TYPE - ? { - paymasterAndData: Hex - preVerificationGas: Hex - verificationGasLimit: Hex - callGasLimit: Hex - paymaster?: never - paymasterVerificationGasLimit?: never - paymasterPostOpGasLimit?: never - paymasterData?: never - } - : { - preVerificationGas: Hex - verificationGasLimit: Hex - callGasLimit: Hex - paymaster: Address - paymasterVerificationGasLimit: Hex - paymasterPostOpGasLimit: Hex - paymasterData: Hex - paymasterAndData?: never - } - }, - { - Method: "pm_accounts" - Parameters: [entryPoint: Address] - ReturnType: Address[] - } -] diff --git a/packages/permissionless/types/userOperation.ts b/packages/permissionless/types/userOperation.ts deleted file mode 100644 index c5f3cedf..00000000 --- a/packages/permissionless/types/userOperation.ts +++ /dev/null @@ -1,61 +0,0 @@ -import type { Address } from "viem" -import type { Hex } from "viem" -import type { EntryPointVersion } from "./entrypoint" - -export type TStatus = "success" | "reverted" - -export type UserOperationWithBigIntAsHex< - entryPointVersion extends EntryPointVersion -> = entryPointVersion extends "v0.6" - ? { - sender: Address - nonce: Hex - initCode: Hex - callData: Hex - callGasLimit: Hex - verificationGasLimit: Hex - preVerificationGas: Hex - maxFeePerGas: Hex - maxPriorityFeePerGas: Hex - paymasterAndData: Hex - signature: Hex - factory?: never - factoryData?: never - paymaster?: never - paymasterVerificationGasLimit?: never - paymasterPostOpGasLimit?: never - paymasterData?: never - } - : { - sender: Address - nonce: Hex - factory: Address - factoryData: Hex - callData: Hex - callGasLimit: Hex - verificationGasLimit: Hex - preVerificationGas: Hex - maxFeePerGas: Hex - maxPriorityFeePerGas: Hex - paymaster: Address - paymasterVerificationGasLimit: Hex - paymasterPostOpGasLimit: Hex - paymasterData: Hex - signature: Hex - initCode?: never - paymasterAndData?: never - } - -export type Hex32 = `0x${string & { length: 64 }}` - -export type PackedUserOperation = { - sender: Address - nonce: bigint - initCode: Hex - callData: Hex - accountGasLimits: Hex32 - preVerificationGas: bigint - gasFees: Hex32 - paymasterAndData: Hex - signature: Hex -} diff --git a/packages/permissionless/utils/errors/getBundlerError.ts b/packages/permissionless/utils/errors/getBundlerError.ts deleted file mode 100644 index eeadf0fa..00000000 --- a/packages/permissionless/utils/errors/getBundlerError.ts +++ /dev/null @@ -1,224 +0,0 @@ -import { - type Address, - BaseError, - ExecutionRevertedError, - type ExecutionRevertedErrorType, - UnknownNodeError, - type UnknownNodeErrorType -} from "viem" -import { SenderAlreadyDeployedError } from "../../errors" -import { - InitCodeDidNotDeploySenderError, - type InitCodeDidNotDeploySenderErrorType, - InitCodeRevertedError, - type InitCodeRevertedErrorType, - InvalidSmartAccountNonceError, - type InvalidSmartAccountNonceErrorType, - SenderAddressMismatchError, - type SenderAddressMismatchErrorType, - type SenderAlreadyDeployedErrorType, - SenderNotDeployedError, - type SenderNotDeployedErrorType, - SmartAccountInsufficientFundsError, - type SmartAccountInsufficientFundsErrorType, - SmartAccountSignatureValidityPeriodError, - type SmartAccountSignatureValidityPeriodErrorType, - SmartAccountValidationRevertedError, - type SmartAccountValidationRevertedErrorType -} from "../../errors/account" -import { - PaymasterDataRejectedError, - type PaymasterDataRejectedErrorType, - PaymasterDepositTooLowError, - type PaymasterDepositTooLowErrorType, - PaymasterNotDeployedError, - type PaymasterNotDeployedErrorType, - PaymasterValidationRevertedError, - type PaymasterValidationRevertedErrorType, - PaymasterValidityPeriodError, - type PaymasterValidityPeriodErrorType -} from "../../errors/paymaster" -import type { - EntryPoint, - GetEntryPointVersion, - UserOperation -} from "../../types" - -export type GetBundlerErrorParameters = { - userOperation: Partial>> - entryPoint: Address -} - -export type GetBundlerErrorReturnType = - | ExecutionRevertedErrorType - | UnknownNodeErrorType - | SenderAlreadyDeployedErrorType - | InitCodeRevertedErrorType - | SenderAddressMismatchErrorType - | InitCodeDidNotDeploySenderErrorType - | SenderNotDeployedErrorType - | SmartAccountInsufficientFundsErrorType - | SmartAccountSignatureValidityPeriodErrorType - | SmartAccountValidationRevertedErrorType - | InvalidSmartAccountNonceErrorType - | PaymasterNotDeployedErrorType - | PaymasterDepositTooLowErrorType - | PaymasterValidityPeriodErrorType - | PaymasterValidationRevertedErrorType - | PaymasterDataRejectedErrorType - -export function getBundlerError( - err: BaseError, - args: GetBundlerErrorParameters -) { - const message = (err.details || "").toLowerCase() - - const executionRevertedError = - err instanceof BaseError - ? err.walk( - (e) => - (e as { code: number }).code === - ExecutionRevertedError.code - ) - : err - - if (executionRevertedError instanceof BaseError) { - return new ExecutionRevertedError({ - cause: err, - message: executionRevertedError.details - }) as ExecutionRevertedErrorType - } - - // TODO: Add validation Errors - if (args.userOperation.sender === undefined) - return new UnknownNodeError({ cause: err }) - if (args.userOperation.nonce === undefined) - return new UnknownNodeError({ cause: err }) - - if (SenderAlreadyDeployedError.message.test(message)) { - return new SenderAlreadyDeployedError({ - cause: err, - sender: args.userOperation.sender, - docsPath: - "https://docs.pimlico.io/bundler/reference/entrypoint-errors/aa10" - }) as SenderAlreadyDeployedErrorType - } - - if (InitCodeRevertedError.message.test(message)) { - return new InitCodeRevertedError({ - cause: err, - docsPath: - "https://docs.pimlico.io/bundler/reference/entrypoint-errors/aa13" - }) as InitCodeRevertedErrorType - } - - if (SenderAddressMismatchError.message.test(message)) { - return new SenderAddressMismatchError({ - cause: err, - sender: args.userOperation.sender, - docsPath: - "https://docs.pimlico.io/bundler/reference/entrypoint-errors/aa14" - }) as SenderAddressMismatchErrorType - } - - if (InitCodeDidNotDeploySenderError.message.test(message)) { - return new InitCodeDidNotDeploySenderError({ - cause: err, - sender: args.userOperation.sender, - docsPath: - "https://docs.pimlico.io/bundler/reference/entrypoint-errors/aa15" - }) as InitCodeDidNotDeploySenderErrorType - } - - if (SenderNotDeployedError.message.test(message)) { - return new SenderNotDeployedError({ - cause: err, - sender: args.userOperation.sender, - docsPath: - "https://docs.pimlico.io/bundler/reference/entrypoint-errors/aa20" - }) as SenderNotDeployedErrorType - } - - if (SmartAccountInsufficientFundsError.message.test(message)) { - return new SmartAccountInsufficientFundsError({ - cause: err, - sender: args.userOperation.sender, - docsPath: - "https://docs.pimlico.io/bundler/reference/entrypoint-errors/aa21" - }) as SmartAccountInsufficientFundsErrorType - } - - if (SmartAccountSignatureValidityPeriodError.message.test(message)) { - return new SmartAccountSignatureValidityPeriodError({ - cause: err, - docsPath: - "https://docs.pimlico.io/bundler/reference/entrypoint-errors/aa22" - }) as SmartAccountSignatureValidityPeriodErrorType - } - - if (SmartAccountValidationRevertedError.message.test(message)) { - return new SmartAccountValidationRevertedError({ - cause: err, - sender: args.userOperation.sender, - docsPath: - "https://docs.pimlico.io/bundler/reference/entrypoint-errors/aa23" - }) as SmartAccountValidationRevertedErrorType - } - - if (InvalidSmartAccountNonceError.message.test(message)) { - return new InvalidSmartAccountNonceError({ - cause: err, - sender: args.userOperation.sender, - nonce: args.userOperation.nonce, - docsPath: - "https://docs.pimlico.io/bundler/reference/entrypoint-errors/aa25" - }) as InvalidSmartAccountNonceErrorType - } - - if (PaymasterNotDeployedError.message.test(message)) { - return new PaymasterNotDeployedError({ - cause: err, - paymasterAndData: args.userOperation.paymasterAndData, - docsPath: - "https://docs.pimlico.io/bundler/reference/entrypoint-errors/aa30" - }) as PaymasterNotDeployedErrorType - } - - if (PaymasterDepositTooLowError.message.test(message)) { - return new PaymasterDepositTooLowError({ - cause: err, - paymasterAndData: args.userOperation.paymasterAndData, - docsPath: - "https://docs.pimlico.io/bundler/reference/entrypoint-errors/aa31" - }) as PaymasterDepositTooLowErrorType - } - - if (PaymasterValidityPeriodError.message.test(message)) { - return new PaymasterValidityPeriodError({ - cause: err, - paymasterAndData: args.userOperation.paymasterAndData, - docsPath: - "https://docs.pimlico.io/bundler/reference/entrypoint-errors/aa32" - }) as PaymasterValidityPeriodErrorType - } - - if (PaymasterValidationRevertedError.message.test(message)) { - return new PaymasterValidationRevertedError({ - cause: err, - paymasterAndData: args.userOperation.paymasterAndData, - docsPath: - "https://docs.pimlico.io/bundler/reference/entrypoint-errors/aa33" - }) as PaymasterValidationRevertedErrorType - } - - if (PaymasterDataRejectedError.message.test(message)) { - return new PaymasterDataRejectedError({ - cause: err, - paymasterAndData: args.userOperation.paymasterAndData, - docsPath: - "https://docs.pimlico.io/bundler/reference/entrypoint-errors/aa34" - }) as PaymasterDataRejectedErrorType - } - - return new UnknownNodeError({ cause: err }) as UnknownNodeErrorType -} diff --git a/packages/permissionless/utils/errors/getEstimateUserOperationGasError.ts b/packages/permissionless/utils/errors/getEstimateUserOperationGasError.ts deleted file mode 100644 index d3e7ad57..00000000 --- a/packages/permissionless/utils/errors/getEstimateUserOperationGasError.ts +++ /dev/null @@ -1,40 +0,0 @@ -import { type BaseError, UnknownNodeError } from "viem" -import type { EstimateUserOperationGasParameters } from "../../actions/bundler/estimateUserOperationGas" -import { - EstimateUserOperationGasError, - type EstimateUserOperationGasErrorType -} from "../../errors/estimateUserOperationGas" -import type { ErrorType } from "../../errors/utils" -import type { EntryPoint } from "../../types/entrypoint" -import { - type GetBundlerErrorParameters, - type GetBundlerErrorReturnType, - getBundlerError -} from "./getBundlerError" - -export type GetEstimateUserOperationGasErrorReturnType< - entryPoint extends EntryPoint, - cause = ErrorType -> = Omit, "cause"> & { - cause: cause | GetBundlerErrorReturnType -} - -export function getEstimateUserOperationGasError< - err extends ErrorType, - entryPoint extends EntryPoint ->(error: err, args: EstimateUserOperationGasParameters) { - const cause = (() => { - const cause = getBundlerError( - // biome-ignore lint/complexity/noBannedTypes: - error as {} as BaseError, - args as GetBundlerErrorParameters - ) - // biome-ignore lint/complexity/noBannedTypes: - if (cause instanceof UnknownNodeError) return error as {} as BaseError - return cause - })() - - throw new EstimateUserOperationGasError(cause, { - ...args - }) as GetEstimateUserOperationGasErrorReturnType -} diff --git a/packages/permissionless/utils/errors/getSendUserOperationError.ts b/packages/permissionless/utils/errors/getSendUserOperationError.ts deleted file mode 100644 index 47edcb4e..00000000 --- a/packages/permissionless/utils/errors/getSendUserOperationError.ts +++ /dev/null @@ -1,26 +0,0 @@ -import { type BaseError, UnknownNodeError } from "viem" -import type { SendUserOperationParameters } from "../../actions/bundler/sendUserOperation" -import { SendUserOperationError } from "../../errors" -import type { EntryPoint } from "../../types/entrypoint" -import { - type GetBundlerErrorParameters, - getBundlerError -} from "./getBundlerError" - -export function getSendUserOperationError( - err: BaseError, - args: SendUserOperationParameters -) { - const cause = (() => { - const cause = getBundlerError( - err as BaseError, - args as GetBundlerErrorParameters - ) - if (cause instanceof UnknownNodeError) return err as BaseError - return cause - })() - - throw new SendUserOperationError(cause, { - ...args - }) -} diff --git a/packages/permissionless/utils/getEntryPointVersion.ts b/packages/permissionless/utils/getEntryPointVersion.ts deleted file mode 100644 index c6a1e3f0..00000000 --- a/packages/permissionless/utils/getEntryPointVersion.ts +++ /dev/null @@ -1,31 +0,0 @@ -import type { - ENTRYPOINT_ADDRESS_V06_TYPE, - ENTRYPOINT_ADDRESS_V07_TYPE, - UserOperation -} from "../types" -import type { EntryPoint, GetEntryPointVersion } from "../types/entrypoint" - -export const ENTRYPOINT_ADDRESS_V06: ENTRYPOINT_ADDRESS_V06_TYPE = - "0x5FF137D4b0FDCD49DcA30c7CF57E578a026d2789" -export const ENTRYPOINT_ADDRESS_V07: ENTRYPOINT_ADDRESS_V07_TYPE = - "0x0000000071727De22E5E9d8BAf0edAc6f37da032" - -export const getEntryPointVersion = ( - entryPoint: EntryPoint -): GetEntryPointVersion => - entryPoint === ENTRYPOINT_ADDRESS_V06 ? "v0.6" : "v0.7" - -export function isUserOperationVersion06( - entryPoint: EntryPoint, - _operation: UserOperation<"v0.6"> | UserOperation<"v0.7"> -): _operation is UserOperation<"v0.6"> { - return getEntryPointVersion(entryPoint) === "v0.6" -} - -// Type predicate to check if the UserOperation is V07. -export function isUserOperationVersion07( - entryPoint: EntryPoint, - _operation: UserOperation<"v0.6"> | UserOperation<"v0.7"> -): _operation is UserOperation<"v0.7"> { - return getEntryPointVersion(entryPoint) === "v0.7" -} diff --git a/packages/permissionless/utils/getPackedUserOperation.ts b/packages/permissionless/utils/getPackedUserOperation.ts index 241cebde..9fea8556 100644 --- a/packages/permissionless/utils/getPackedUserOperation.ts +++ b/packages/permissionless/utils/getPackedUserOperation.ts @@ -1,8 +1,10 @@ import { type Hex, concat, getAddress, pad, slice, toHex } from "viem" -import type { UserOperation } from "../types/userOperation" -import type { Hex32, PackedUserOperation } from "../types/userOperation" +import type { + PackedUserOperation, + UserOperation +} from "viem/account-abstraction" -export function getInitCode(unpackedUserOperation: UserOperation<"v0.7">) { +export function getInitCode(unpackedUserOperation: UserOperation<"0.7">) { return unpackedUserOperation.factory ? concat([ unpackedUserOperation.factory, @@ -25,14 +27,14 @@ export function unPackInitCode(initCode: Hex) { } export function getAccountGasLimits( - unpackedUserOperation: UserOperation<"v0.7"> + unpackedUserOperation: UserOperation<"0.7"> ) { return concat([ pad(toHex(unpackedUserOperation.verificationGasLimit), { size: 16 }), pad(toHex(unpackedUserOperation.callGasLimit), { size: 16 }) - ]) as Hex32 + ]) } export function unpackAccountGasLimits(accountGasLimits: Hex) { @@ -42,13 +44,13 @@ export function unpackAccountGasLimits(accountGasLimits: Hex) { } } -export function getGasLimits(unpackedUserOperation: UserOperation<"v0.7">) { +export function getGasLimits(unpackedUserOperation: UserOperation<"0.7">) { return concat([ pad(toHex(unpackedUserOperation.maxPriorityFeePerGas), { size: 16 }), pad(toHex(unpackedUserOperation.maxFeePerGas), { size: 16 }) - ]) as Hex32 + ]) } export function unpackGasLimits(gasLimits: Hex) { @@ -59,7 +61,7 @@ export function unpackGasLimits(gasLimits: Hex) { } export function getPaymasterAndData( - unpackedUserOperation: UserOperation<"v0.7"> + unpackedUserOperation: UserOperation<"0.7"> ) { return unpackedUserOperation.paymaster ? concat([ @@ -104,7 +106,7 @@ export function unpackPaymasterAndData(paymasterAndData: Hex) { } export const getPackedUserOperation = ( - userOperation: UserOperation<"v0.7"> + userOperation: UserOperation<"0.7"> ): PackedUserOperation => { return { sender: userOperation.sender, diff --git a/packages/permissionless/utils/getRequiredPrefund.ts b/packages/permissionless/utils/getRequiredPrefund.ts index e60ed0f1..84b1bfd3 100644 --- a/packages/permissionless/utils/getRequiredPrefund.ts +++ b/packages/permissionless/utils/getRequiredPrefund.ts @@ -1,9 +1,10 @@ -import type { EntryPoint, GetEntryPointVersion, UserOperation } from "../types" -import { ENTRYPOINT_ADDRESS_V06 } from "./getEntryPointVersion" +import type { UserOperation } from "viem/account-abstraction" -export type GetRequiredPrefundReturnType = { - userOperation: UserOperation> - entryPoint: entryPoint +export type GetRequiredPrefundReturnType< + entryPointVersion extends "0.6" | "0.7" +> = { + userOperation: UserOperation + entryPointVersion: entryPointVersion } /** @@ -20,14 +21,14 @@ export type GetRequiredPrefundReturnType = { * userOperation * }) */ -export const getRequiredPrefund = ({ +export const getRequiredPrefund = ({ userOperation, - entryPoint: entryPointAddress -}: GetRequiredPrefundReturnType): bigint => { - if (entryPointAddress === ENTRYPOINT_ADDRESS_V06) { - const userOperationVersion0_6 = userOperation as UserOperation<"v0.6"> + entryPointVersion +}: GetRequiredPrefundReturnType): bigint => { + if (entryPointVersion === "0.6") { + const userOperationVersion0_6 = userOperation as UserOperation<"0.6"> const multiplier = - userOperationVersion0_6.paymasterAndData.length > 2 + (userOperationVersion0_6.paymasterAndData?.length ?? 0) > 2 ? BigInt(3) : BigInt(1) const requiredGas = @@ -40,7 +41,7 @@ export const getRequiredPrefund = ({ ) } - const userOperationV07 = userOperation as UserOperation<"v0.7"> + const userOperationV07 = userOperation as UserOperation<"0.7"> const multiplier = userOperationV07.paymaster ? BigInt(3) : BigInt(1) const verificationGasLimit = diff --git a/packages/permissionless/utils/index.ts b/packages/permissionless/utils/index.ts index 55b42674..b8191eb3 100644 --- a/packages/permissionless/utils/index.ts +++ b/packages/permissionless/utils/index.ts @@ -5,17 +5,8 @@ import { type GetRequiredPrefundReturnType, getRequiredPrefund } from "./getRequiredPrefund" -import { - type GetUserOperationHashParams, - getUserOperationHash -} from "./getUserOperationHash" import { isSmartAccountDeployed } from "./isSmartAccountDeployed" import { providerToSmartAccountSigner } from "./providerToSmartAccountSigner" -import { - AccountOrClientNotFoundError, - type SignUserOperationHashWithECDSAParams, - signUserOperationHashWithECDSA -} from "./signUserOperationHashWithECDSA" import { walletClientToSmartAccountSigner } from "./walletClientToSmartAccountSigner" export function parseAccount(account: Address | Account): Account { @@ -25,32 +16,19 @@ export function parseAccount(account: Address | Account): Account { } import { decodeNonce } from "./decodeNonce" import { encodeNonce } from "./encodeNonce" -import { - ENTRYPOINT_ADDRESS_V06, - ENTRYPOINT_ADDRESS_V07, - getEntryPointVersion -} from "./getEntryPointVersion" import { getPackedUserOperation } from "./getPackedUserOperation" export { transactionReceiptStatus, deepHexlify, - getUserOperationHash, getRequiredPrefund, walletClientToSmartAccountSigner, type GetRequiredPrefundReturnType, - type GetUserOperationHashParams, - signUserOperationHashWithECDSA, - type SignUserOperationHashWithECDSAParams, - AccountOrClientNotFoundError, isSmartAccountDeployed, providerToSmartAccountSigner, getAddressFromInitCodeOrPaymasterAndData, getPackedUserOperation, - getEntryPointVersion, encodeNonce, - decodeNonce, - ENTRYPOINT_ADDRESS_V06, - ENTRYPOINT_ADDRESS_V07 + decodeNonce } diff --git a/packages/permissionless/utils/observe.ts b/packages/permissionless/utils/observe.ts deleted file mode 100644 index b4b719a8..00000000 --- a/packages/permissionless/utils/observe.ts +++ /dev/null @@ -1,74 +0,0 @@ -import type { MaybePromise } from "viem/types/utils" - -// biome-ignore lint/suspicious/noExplicitAny: it's a recursive function, so it's hard to type -type Callback = ((...args: any[]) => any) | undefined -type Callbacks = Record - -export const listenersCache = /*#__PURE__*/ new Map< - string, - { id: number; fns: Callbacks }[] ->() -export const cleanupCache = /*#__PURE__*/ new Map void>() - -type EmitFunction = ( - emit: TCallbacks - // biome-ignore lint/suspicious/noConfusingVoidType: -) => MaybePromise void)> - -let callbackCount = 0 - -/** - * @description Sets up an observer for a given function. If another function - * is set up under the same observer id, the function will only be called once - * for both instances of the observer. - */ -export function observe( - observerId: string, - callbacks: TCallbacks, - fn: EmitFunction -) { - const callbackId = ++callbackCount - - const getListeners = () => listenersCache.get(observerId) || [] - - const unsubscribe = () => { - const listeners = getListeners() - listenersCache.set( - observerId, - // biome-ignore lint/suspicious/noExplicitAny: it's a recursive function, so it's hard to type - listeners.filter((cb: any) => cb.id !== callbackId) - ) - } - - const unwatch = () => { - const cleanup = cleanupCache.get(observerId) - if (getListeners().length === 1 && cleanup) cleanup() - unsubscribe() - } - - const listeners = getListeners() - listenersCache.set(observerId, [ - ...listeners, - { id: callbackId, fns: callbacks } - ]) - - if (listeners && listeners.length > 0) return unwatch - - const emit: TCallbacks = {} as TCallbacks - for (const key in callbacks) { - emit[key] = (( - ...args: Parameters> - ) => { - const listeners = getListeners() - if (listeners.length === 0) return - for (const listener of listeners) { - listener.fns[key]?.(...args) - } - }) as TCallbacks[Extract] - } - - const cleanup = fn(emit) - if (typeof cleanup === "function") cleanupCache.set(observerId, cleanup) - - return unwatch -} diff --git a/packages/permissionless/utils/signUserOperationHashWithECDSA.ts b/packages/permissionless/utils/signUserOperationHashWithECDSA.ts deleted file mode 100644 index e7ab68cb..00000000 --- a/packages/permissionless/utils/signUserOperationHashWithECDSA.ts +++ /dev/null @@ -1,134 +0,0 @@ -import { - type Account, - BaseError, - type Chain, - type Client, - type Hash, - type Hex, - type Transport -} from "viem" -import type { - EntryPoint, - GetAccountParameterWithClient, - GetEntryPointVersion -} from "../types/" -import type { UserOperation } from "../types/userOperation" -import { parseAccount } from "./" -import { getUserOperationHash } from "./getUserOperationHash" - -export type SignUserOperationHashWithECDSAParams< - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends Account | undefined = Account | undefined -> = GetAccountParameterWithClient & - ( - | { - hash: Hash - userOperation?: undefined - entryPoint?: undefined - chainId?: undefined - } - | { - hash?: undefined - userOperation: UserOperation> - entryPoint: entryPoint - chainId: number - } - ) - -export class AccountOrClientNotFoundError extends BaseError { - override name = "AccountOrClientNotFoundError" - constructor({ docsPath }: { docsPath?: string } = {}) { - super( - [ - "Could not find an Account to execute with this Action.", - "Please provide an Account with the `account` argument on the Action, or by supplying an `account` to the WalletClient." - ].join("\n"), - { - docsPath, - docsSlug: "account" - } - ) - } -} - -/** - * - * Returns signature for user operation. It signs over user operation hash. - * If you have a custom way of signing user operation hash, you can use this function to sign it with ECDSA. - * - * - Docs: https://docs.pimlico.io/permissionless/reference/utils/signUserOperationHashWithECDSA - * - * @param signer: owner as {@link Client} - * @param params: account & (userOperation, entryPoint, chainId) | hash to sign - * @returns signature as {@link Hash} - * - * @example - * import { signUserOperationHashWithECDSA } from "permissionless/utils" - * - * const userOperationSignature = signUserOperationHashWithECDSA(owner, { - * userOperation, - * entryPoint, - * chainId - * }) - * - * // Returns "0x7d9ae17d5e617e4bf3221dfcb69d64d824959e5ae2ef7078c6ddc3a4fe26c8301ab39277c61160dca68ca90071eb449d9fb2fbbc78b3614d9d7282c860270e291c" - * - */ -export const signUserOperationHashWithECDSA = async < - entryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TAccount extends Account | undefined = Account | undefined ->({ - client, - account: account_ = client?.account, - hash, - userOperation, - chainId, - entryPoint: entryPointAddress -}: SignUserOperationHashWithECDSAParams< - entryPoint, - TTransport, - TChain, - TAccount ->): Promise => { - if (!account_) - throw new AccountOrClientNotFoundError({ - docsPath: - "/permissionless/reference/utils/signUserOperationHashWithECDSA" - }) - - let userOperationHash: Hash - - if (hash) { - userOperationHash = hash - } else { - userOperationHash = getUserOperationHash({ - userOperation, - chainId, - entryPoint: entryPointAddress - }) - } - - const account = parseAccount(account_) - - if (account.type === "local") - return account.signMessage({ - message: { - raw: userOperationHash - } - }) - - if (!client) - throw new AccountOrClientNotFoundError({ - docsPath: - "/permissionless/reference/utils/signUserOperationHashWithECDSA" - }) - - return client.request({ - method: "personal_sign", - params: [userOperationHash, account.address] - }) -} diff --git a/packages/permissionless/utils/walletClientToSmartAccountSigner.ts b/packages/permissionless/utils/walletClientToSmartAccountSigner.ts index 6edbdc25..fd37c59c 100644 --- a/packages/permissionless/utils/walletClientToSmartAccountSigner.ts +++ b/packages/permissionless/utils/walletClientToSmartAccountSigner.ts @@ -1,8 +1,8 @@ import type { Account, - Address, Chain, Hex, + LocalAccount, SignableMessage, Transport, TypedData, @@ -11,13 +11,10 @@ import type { } from "viem" import { signTypedData } from "viem/actions" -import type { SmartAccountSigner } from "../accounts/types" export function walletClientToSmartAccountSigner< TChain extends Chain | undefined = Chain | undefined ->( - walletClient: WalletClient -): SmartAccountSigner<"custom", Address> { +>(walletClient: WalletClient): LocalAccount { return { address: walletClient.account.address, type: "local", @@ -28,6 +25,11 @@ export function walletClientToSmartAccountSigner< }: { message: SignableMessage }): Promise => { return walletClient.signMessage({ message }) }, + signTransaction: async (_): Promise => { + throw new Error( + "Smart account signer doesn't need to sign transactions" + ) + }, async signTypedData< const TTypedData extends TypedData | Record, TPrimaryType extends From 9055825762fb990e4c614a27e9dc84f58b831e5c Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Wed, 21 Aug 2024 16:52:41 +0100 Subject: [PATCH 06/51] Test running --- .../mock-aa-infra/mock-paymaster/index.ts | 24 +- .../mock-paymaster/mockAccount.ts | 27 + .../mock-aa-infra/mock-paymaster/relay.ts | 211 +++--- .../permissionless-test/src/testWithRpc.ts | 11 +- packages/permissionless-test/src/types.ts | 11 +- packages/permissionless-test/src/utils.ts | 653 +++++++----------- .../biconomy/toBiconomySmartAccount.ts | 13 +- .../accounts/kernel/constants.ts | 10 +- .../kernel/toEcdsaKernelSmartAccount.ts | 50 +- .../accounts/kernel/utils/encodeCallData.ts | 2 +- .../accounts/kernel/utils/getExecMode.ts | 18 - .../accounts/kernel/utils/getNonceKey.ts | 1 + .../accounts/light/toLightSmartAccount.ts | 40 +- .../accounts/safe/toSafeSmartAccount.ts | 62 +- .../accounts/simple/toSimpleSmartAccount.ts | 61 +- .../accounts/trust/toTrustSmartAccount.ts | 21 +- packages/permissionless/actions/erc7579.ts | 90 +-- .../actions/erc7579/accountId.test.ts | 43 +- .../actions/erc7579/accountId.ts | 12 +- .../actions/erc7579/installModule.test.ts | 114 +-- .../actions/erc7579/installModule.ts | 115 ++- .../actions/erc7579/installModules.test.ts | 115 +-- .../actions/erc7579/installModules.ts | 143 ++-- .../actions/erc7579/isModuleInstalled.test.ts | 71 +- .../actions/erc7579/isModuleInstalled.ts | 35 +- .../erc7579/supportsExecutionMode.test.ts | 71 +- .../actions/erc7579/supportsExecutionMode.ts | 38 +- .../actions/erc7579/supportsModule.test.ts | 25 +- .../actions/erc7579/supportsModule.ts | 37 +- .../actions/erc7579/uninstallModule.test.ts | 78 +-- .../actions/erc7579/uninstallModule.ts | 128 ++-- .../actions/erc7579/uninstallModules.test.ts | 78 +-- .../actions/erc7579/uninstallModules.ts | 131 ++-- packages/permissionless/actions/index.test.ts | 329 --------- .../permissionless/actions/pimlico.test.ts | 222 ------ packages/permissionless/actions/pimlico.ts | 16 +- .../pimlico/getUserOperationGasPrice.test.ts | 8 +- .../pimlico/getUserOperationStatus.test.ts | 114 ++- .../pimlico/sponsorUserOperation.test.ts | 111 ++- .../validateSponsorshipPolicies.test.ts | 50 +- .../actions/public/getAccountNonce.test.ts | 48 +- .../actions/public/getSenderAddress.test.ts | 83 ++- packages/permissionless/clients/pimlico.ts | 4 +- packages/permissionless/errors/account.ts | 345 --------- .../errors/estimateUserOperationGas.ts | 60 -- packages/permissionless/errors/gas.ts | 120 ---- packages/permissionless/errors/index.ts | 131 +--- packages/permissionless/errors/paymaster.ts | 257 ------- .../errors/sendUserOperation.ts | 43 -- packages/permissionless/errors/utils.ts | 19 - .../eip7677/actions/getPaymasterData.test.ts | 118 ++-- .../eip7677/actions/getPaymasterData.ts | 160 ++--- .../actions/getPaymasterStubData.test.ts | 99 +-- .../eip7677/actions/getPaymasterStubData.ts | 170 ++--- .../decorators/paymasterActionsEip7677.ts | 95 +-- .../experimental/eip7677/types/paymaster.ts | 110 +-- packages/permissionless/index.ts | 86 --- packages/permissionless/types/index.ts | 1 - .../permissionless/utils/encode7579Calls.ts | 6 + .../utils/getEntryPointVersion.test.ts | 52 -- .../utils/getRequiredPrefund.test.ts | 18 +- 61 files changed, 1536 insertions(+), 3878 deletions(-) create mode 100644 packages/permissionless-test/mock-aa-infra/mock-paymaster/mockAccount.ts delete mode 100644 packages/permissionless/accounts/kernel/utils/getExecMode.ts delete mode 100644 packages/permissionless/actions/index.test.ts delete mode 100644 packages/permissionless/actions/pimlico.test.ts delete mode 100644 packages/permissionless/errors/account.ts delete mode 100644 packages/permissionless/errors/estimateUserOperationGas.ts delete mode 100644 packages/permissionless/errors/gas.ts delete mode 100644 packages/permissionless/errors/paymaster.ts delete mode 100644 packages/permissionless/errors/sendUserOperation.ts delete mode 100644 packages/permissionless/errors/utils.ts delete mode 100644 packages/permissionless/utils/getEntryPointVersion.test.ts diff --git a/packages/permissionless-test/mock-aa-infra/mock-paymaster/index.ts b/packages/permissionless-test/mock-aa-infra/mock-paymaster/index.ts index 44321b33..158b84b8 100644 --- a/packages/permissionless-test/mock-aa-infra/mock-paymaster/index.ts +++ b/packages/permissionless-test/mock-aa-infra/mock-paymaster/index.ts @@ -1,13 +1,9 @@ import cors from "@fastify/cors" import Fastify from "fastify" import { defineInstance } from "prool" -import { http } from "viem" +import { http, createPublicClient } from "viem" +import { createBundlerClient } from "viem/account-abstraction" import { foundry } from "viem/chains" -import { - ENTRYPOINT_ADDRESS_V06, - ENTRYPOINT_ADDRESS_V07 -} from "../../../permissionless" -import { createPimlicoBundlerClient } from "../../../permissionless/clients/pimlico" import { getAnvilWalletClient } from "./helpers/utils" import { setupVerifyingPaymasterV06, @@ -39,16 +35,14 @@ export const paymaster = defineInstance( anvilRpc ) - const altoBundlerV07 = createPimlicoBundlerClient({ - chain: foundry, - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 + const publicClient = createPublicClient({ + transport: http(anvilRpc), + chain: foundry }) - const altoBundlerV06 = createPimlicoBundlerClient({ + const bundler = createBundlerClient({ chain: foundry, - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V06 + transport: http(altoRpc) }) app.register(cors, { @@ -57,10 +51,10 @@ export const paymaster = defineInstance( }) const rpcHandler = createRpcHandler( - altoBundlerV07, - altoBundlerV06, + bundler, verifyingPaymasterV07, verifyingPaymasterV06, + publicClient, walletClient ) app.post("/", {}, rpcHandler) diff --git a/packages/permissionless-test/mock-aa-infra/mock-paymaster/mockAccount.ts b/packages/permissionless-test/mock-aa-infra/mock-paymaster/mockAccount.ts new file mode 100644 index 00000000..b1c5436a --- /dev/null +++ b/packages/permissionless-test/mock-aa-infra/mock-paymaster/mockAccount.ts @@ -0,0 +1,27 @@ +import type { Client } from "viem" +import type { + entryPoint06Abi, + entryPoint06Address, + entryPoint07Abi, + entryPoint07Address +} from "viem/account-abstraction" +import { generatePrivateKey, privateKeyToAccount } from "viem/accounts" +import { toSimpleSmartAccount } from "../../../permissionless/accounts/simple/toSimpleSmartAccount" + +export const getMockAccount = ({ + publicClient, + entryPoint +}: { + publicClient: Client + entryPoint: { + address: typeof entryPoint06Address | typeof entryPoint07Address + version: "0.6" | "0.7" + abi: typeof entryPoint06Abi | typeof entryPoint07Abi + } +}) => { + return toSimpleSmartAccount({ + client: publicClient, + entryPoint, + owner: privateKeyToAccount(generatePrivateKey()) + }) +} diff --git a/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts b/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts index 326af8d6..de6bf9e4 100644 --- a/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts +++ b/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts @@ -4,6 +4,7 @@ import { type Account, BaseError, type Chain, + type Client, type GetContractReturnType, type Hex, type PublicClient, @@ -14,20 +15,17 @@ import { encodeAbiParameters, toHex } from "viem" +import { + type BundlerClient, + type UserOperation, + entryPoint06Abi, + entryPoint06Address, + entryPoint07Address +} from "viem/account-abstraction" +import { readContract } from "viem/actions" import { fromZodError } from "zod-validation-error" +import { getPackedUserOperation } from "../../../permissionless" import { - ENTRYPOINT_ADDRESS_V07, - type EstimateUserOperationGasReturnType, - getPackedUserOperation -} from "../../../permissionless" -import type { PimlicoBundlerClient } from "../../../permissionless/clients/pimlico" -import type { - ENTRYPOINT_ADDRESS_V06_TYPE, - ENTRYPOINT_ADDRESS_V07_TYPE, - UserOperation -} from "../../../permissionless/types" -import { ENTRYPOINT_ADDRESS_V06 } from "../../../permissionless/utils" -import type { VERIFYING_PAYMASTER_V06_ABI, VERIFYING_PAYMASTER_V07_ABI } from "./helpers/abi" @@ -35,7 +33,6 @@ import { InternalBundlerError, type JsonRpcSchema, RpcError, - UserOperationV7, ValidationErrors, jsonRpcSchema, pmGetPaymasterData, @@ -43,14 +40,15 @@ import { pmSponsorUserOperationParamsSchema } from "./helpers/schema" import { maxBigInt } from "./helpers/utils" +import { getMockAccount } from "./mockAccount" const handleMethodV06 = async ( - userOperation: UserOperation<"v0.6">, - altoBundlerV06: PimlicoBundlerClient, + userOperation: UserOperation<"0.6">, + bundler: BundlerClient, verifyingPaymasterV06: GetContractReturnType< - typeof VERIFYING_PAYMASTER_V06_ABI, - PublicClient + typeof VERIFYING_PAYMASTER_V06_ABI >, + publicClient: Client, walletClient: WalletClient, estimateGas: boolean ) => { @@ -67,33 +65,37 @@ const handleMethodV06 = async ( const preVerificationGas = userOperation.preVerificationGas if (estimateGas) { - let gasEstimates: - | EstimateUserOperationGasReturnType - | undefined = undefined try { - gasEstimates = await altoBundlerV06.estimateUserOperationGas({ - userOperation: op + const gasEstimates = await bundler.estimateUserOperationGas({ + account: await getMockAccount({ + publicClient, + entryPoint: { + address: entryPoint06Address, + abi: entryPoint06Abi, + version: "0.6" + } + }), + ...op }) + op = { + ...op, + ...gasEstimates + } + + op.callGasLimit = maxBigInt(op.callGasLimit, callGasLimit) + op.preVerificationGas = maxBigInt( + op.preVerificationGas, + preVerificationGas + ) + op.verificationGasLimit = maxBigInt( + op.verificationGasLimit, + verificationGasLimit + ) } catch (e: unknown) { if (!(e instanceof BaseError)) throw new InternalBundlerError() const err = e.walk() as RpcRequestError throw err } - - op = { - ...op, - ...gasEstimates - } - - op.callGasLimit = maxBigInt(op.callGasLimit, callGasLimit) - op.preVerificationGas = maxBigInt( - op.preVerificationGas, - preVerificationGas - ) - op.verificationGasLimit = maxBigInt( - op.verificationGasLimit, - verificationGasLimit - ) } else if ( userOperation.preVerificationGas === 1n || userOperation.verificationGasLimit === 1n || @@ -118,11 +120,14 @@ const handleMethodV06 = async ( ), toHex(0, { size: 65 }) ]) - const hash = await verifyingPaymasterV06.read.getHash([ - op, - validUntil, - validAfter - ]) + + const hash = await readContract(publicClient, { + abi: VERIFYING_PAYMASTER_V06_ABI, + functionName: "getHash", + address: verifyingPaymasterV06.address, + args: [{ ...op, initCode: op.initCode ?? "0x" }, validUntil, validAfter] + }) + const sig = await walletClient.signMessage({ message: { raw: hash } }) @@ -149,12 +154,12 @@ const handleMethodV06 = async ( } const handleMethodV07 = async ( - userOperation: UserOperation<"v0.7">, - altoBundlerV07: PimlicoBundlerClient, + userOperation: UserOperation<"0.7">, + bundler: BundlerClient, verifyingPaymasterV07: GetContractReturnType< - typeof VERIFYING_PAYMASTER_V07_ABI, - PublicClient + typeof VERIFYING_PAYMASTER_V07_ABI >, + publicClient: Client, walletClient: WalletClient, estimateGas: boolean ) => { @@ -170,33 +175,38 @@ const handleMethodV07 = async ( const preVerificationGas = userOperation.preVerificationGas if (estimateGas) { - let gasEstimates: - | EstimateUserOperationGasReturnType - | undefined = undefined try { - gasEstimates = await altoBundlerV07.estimateUserOperationGas({ - userOperation: op + const gasEstimates = await bundler.estimateUserOperationGas({ + account: await getMockAccount({ + publicClient, + entryPoint: { + address: entryPoint06Address, + abi: entryPoint06Abi, + version: "0.6" + } + }), + ...op }) + + op = { + ...op, + ...gasEstimates + } + + op.callGasLimit = maxBigInt(op.callGasLimit, callGasLimit) + op.preVerificationGas = maxBigInt( + op.preVerificationGas, + preVerificationGas + ) + op.verificationGasLimit = maxBigInt( + op.verificationGasLimit, + verificationGasLimit + ) } catch (e: unknown) { if (!(e instanceof BaseError)) throw new InternalBundlerError() const err = e.walk() as RpcRequestError throw err } - - op = { - ...op, - ...gasEstimates - } - - op.callGasLimit = maxBigInt(op.callGasLimit, callGasLimit) - op.preVerificationGas = maxBigInt( - op.preVerificationGas, - preVerificationGas - ) - op.verificationGasLimit = maxBigInt( - op.verificationGasLimit, - verificationGasLimit - ) } else if ( userOperation.preVerificationGas === 1n || userOperation.verificationGasLimit === 1n || @@ -221,11 +231,14 @@ const handleMethodV07 = async ( toHex(0, { size: 65 }) ]) op.paymaster = verifyingPaymasterV07.address - const hash = await verifyingPaymasterV07.read.getHash([ - getPackedUserOperation(op), - validUntil, - validAfter - ]) + + const hash = await readContract(publicClient, { + abi: VERIFYING_PAYMASTER_V07_ABI, + functionName: "getHash", + address: verifyingPaymasterV07.address, + args: [getPackedUserOperation(op), validUntil, validAfter] + }) + const sig = await walletClient.signMessage({ message: { raw: hash } }) @@ -257,16 +270,14 @@ const handleMethodV07 = async ( } const handleMethod = async ( - altoBundlerV07: PimlicoBundlerClient, - altoBundlerV06: PimlicoBundlerClient, + bundler: BundlerClient, verifyingPaymasterV07: GetContractReturnType< - typeof VERIFYING_PAYMASTER_V07_ABI, - PublicClient + typeof VERIFYING_PAYMASTER_V07_ABI >, verifyingPaymasterV06: GetContractReturnType< - typeof VERIFYING_PAYMASTER_V06_ABI, - PublicClient + typeof VERIFYING_PAYMASTER_V06_ABI >, + publicClient: PublicClient, walletClient: WalletClient, parsedBody: JsonRpcSchema ) => { @@ -284,21 +295,23 @@ const handleMethod = async ( const [userOperation, entryPoint] = params.data - if (entryPoint === ENTRYPOINT_ADDRESS_V07) { + if (entryPoint === entryPoint07Address) { return await handleMethodV07( - userOperation as UserOperation<"v0.7">, - altoBundlerV07, + userOperation, + bundler, verifyingPaymasterV07, + publicClient, walletClient, true ) } - if (entryPoint === ENTRYPOINT_ADDRESS_V06) { + if (entryPoint === entryPoint06Address) { return await handleMethodV06( - userOperation as UserOperation<"v0.6">, - altoBundlerV06, + userOperation, + bundler, verifyingPaymasterV06, + publicClient, walletClient, true ) @@ -329,7 +342,7 @@ const handleMethod = async ( icon: "" } - if (entryPoint === ENTRYPOINT_ADDRESS_V07) { + if (entryPoint === entryPoint07Address) { return { paymaster: verifyingPaymasterV07.address, paymasterData: @@ -341,7 +354,7 @@ const handleMethod = async ( } } - if (entryPoint === ENTRYPOINT_ADDRESS_V06) { + if (entryPoint === entryPoint06Address) { return { paymasterAndData: `${verifyingPaymasterV06.address}00000000000000000000000000000000000000000000000000000101010101010000000000000000000000000000000000000000000000000000000000000000cd91f19f0f19ce862d7bec7b7d9b95457145afc6f639c28fd0360f488937bfa41e6eedcd3a46054fd95fcd0e3ef6b0bc0a615c4d975eef55c8a3517257904d5b1c`, sponsor: sponsorData, @@ -367,21 +380,23 @@ const handleMethod = async ( const [userOperation, entryPoint] = params.data - if (entryPoint === ENTRYPOINT_ADDRESS_V07) { + if (entryPoint === entryPoint07Address) { return await handleMethodV07( - userOperation as UserOperation<"v0.7">, - altoBundlerV07, + userOperation as UserOperation<"0.7">, + bundler, verifyingPaymasterV07, + publicClient, walletClient, false ) } - if (entryPoint === ENTRYPOINT_ADDRESS_V06) { + if (entryPoint === entryPoint06Address) { return await handleMethodV06( - userOperation as UserOperation<"v0.6">, - altoBundlerV06, + userOperation, + bundler, verifyingPaymasterV06, + publicClient, walletClient, false ) @@ -414,16 +429,14 @@ const handleMethod = async ( } export const createRpcHandler = ( - altoBundlerV07: PimlicoBundlerClient, - altoBundlerV06: PimlicoBundlerClient, + bundler: BundlerClient, verifyingPaymasterV07: GetContractReturnType< - typeof VERIFYING_PAYMASTER_V07_ABI, - PublicClient + typeof VERIFYING_PAYMASTER_V07_ABI >, verifyingPaymasterV06: GetContractReturnType< - typeof VERIFYING_PAYMASTER_V06_ABI, - PublicClient + typeof VERIFYING_PAYMASTER_V06_ABI >, + publicClient: PublicClient, walletClient: WalletClient ) => { return async (request: FastifyRequest, _reply: FastifyReply) => { @@ -438,10 +451,10 @@ export const createRpcHandler = ( try { const result = await handleMethod( - altoBundlerV07, - altoBundlerV06, + bundler, verifyingPaymasterV07, verifyingPaymasterV06, + publicClient, walletClient, parsedBody.data ) diff --git a/packages/permissionless-test/src/testWithRpc.ts b/packages/permissionless-test/src/testWithRpc.ts index d05b837e..a730f99c 100644 --- a/packages/permissionless-test/src/testWithRpc.ts +++ b/packages/permissionless-test/src/testWithRpc.ts @@ -1,12 +1,11 @@ import getPort from "get-port" import { alto, anvil } from "prool/instances" +import { + entryPoint06Address, + entryPoint07Address +} from "viem/account-abstraction" import { foundry } from "viem/chains" import { test } from "vitest" -import type { EntryPoint } from "../../permissionless/types/entrypoint" -import { - ENTRYPOINT_ADDRESS_V06, - ENTRYPOINT_ADDRESS_V07 -} from "../../permissionless/utils" import { ENTRY_POINT_SIMULATIONS_ADDRESS, setupContracts @@ -30,7 +29,7 @@ export const getAltoInstance = async ({ }) const instance = alto({ - entrypoints: [ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07], + entrypoints: [entryPoint06Address, entryPoint07Address], rpcUrl: anvilRpc, executorPrivateKeys: [anvilPrivateKey], entrypointSimulationContract: ENTRY_POINT_SIMULATIONS_ADDRESS, diff --git a/packages/permissionless-test/src/types.ts b/packages/permissionless-test/src/types.ts index 3cfd89a6..b6b23a53 100644 --- a/packages/permissionless-test/src/types.ts +++ b/packages/permissionless-test/src/types.ts @@ -1,13 +1,12 @@ import type { Address, Hex, PublicClient } from "viem" -import type { PimlicoPaymasterClient } from "../../permissionless/clients/pimlico" -import type { EntryPoint } from "../../permissionless/types" -export type AAParamType = { - entryPoint: T +export type AAParamType = { + entryPoint: { + version: entryPointVersion + } anvilRpc: string altoRpc: string - paymasterClient?: PimlicoPaymasterClient - privateKey?: Hex + paymasterRpc: string } // param used when testing with a already deployed contract diff --git a/packages/permissionless-test/src/utils.ts b/packages/permissionless-test/src/utils.ts index f43d9a71..665b88ad 100644 --- a/packages/permissionless-test/src/utils.ts +++ b/packages/permissionless-test/src/utils.ts @@ -3,71 +3,58 @@ import { type Account, type Address, type Chain, - type Hex, type Transport, type WalletClient, createClient, createPublicClient, createWalletClient, - getAddress, parseEther } from "viem" +import { + type SmartAccount, + createBundlerClient, + createPaymasterClient, + entryPoint06Abi, + entryPoint06Address, + entryPoint07Abi, + entryPoint07Address +} from "viem/account-abstraction" import { generatePrivateKey, mnemonicToAccount, privateKeyToAccount } from "viem/accounts" -import { foundry, sepolia } from "viem/chains" +import { foundry } from "viem/chains" +import { toBiconomySmartAccount } from "../../permissionless/accounts/biconomy/toBiconomySmartAccount" import { - type BundlerClient, - ENTRYPOINT_ADDRESS_V06, - ENTRYPOINT_ADDRESS_V07, - type SmartAccountClient, - createBundlerClient, - createSmartAccountClient -} from "../../permissionless" + type KernelVersion, + toEcdsaKernelSmartAccount +} from "../../permissionless/accounts/kernel/toEcdsaKernelSmartAccount" +import { toLightSmartAccount } from "../../permissionless/accounts/light/toLightSmartAccount" +import { toSafeSmartAccount } from "../../permissionless/accounts/safe/toSafeSmartAccount" import { - type SafeSmartAccount, - type SmartAccount, - privateKeyToBiconomySmartAccount, - privateKeyToLightSmartAccount, - privateKeyToSafeSmartAccount, - privateKeyToSimpleSmartAccount, - privateKeyToTrustSmartAccount, - signerToBiconomySmartAccount, - signerToEcdsaKernelSmartAccount, - signerToLightSmartAccount, - signerToSafeSmartAccount, - signerToSimpleSmartAccount, - signerToTrustSmartAccount -} from "../../permissionless/accounts" -import type { KernelEcdsaSmartAccount } from "../../permissionless/accounts" -import type { KernelVersion } from "../../permissionless/accounts/kernel/signerToEcdsaKernelSmartAccount" -import { - type PimlicoBundlerClient, - type PimlicoPaymasterClient, - createPimlicoBundlerClient, - createPimlicoPaymasterClient -} from "../../permissionless/clients/pimlico" + type ToSimpleSmartAccountReturnType, + toSimpleSmartAccount +} from "../../permissionless/accounts/simple/toSimpleSmartAccount" +import { toTrustSmartAccount } from "../../permissionless/accounts/trust/toTrustSmartAccount" +import { createPimlicoClient } from "../../permissionless/clients/pimlico" import { paymasterActionsEip7677 } from "../../permissionless/experimental" -import type { - ENTRYPOINT_ADDRESS_V06_TYPE, - ENTRYPOINT_ADDRESS_V07_TYPE, - EntryPoint -} from "../../permissionless/types" -import type { AAParamType, ExistingSignerParamType } from "./types" +import type { AAParamType } from "./types" export const PAYMASTER_RPC = "http://localhost:3000" -export const ensureBundlerIsReady = async (altoRpc: string) => { +export const ensureBundlerIsReady = async ({ + altoRpc, + anvilRpc +}: { altoRpc: string; anvilRpc: string }) => { const bundlerClient = getBundlerClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - altoRpc: altoRpc + altoRpc: altoRpc, + anvilRpc }) while (true) { try { - await bundlerClient.chainId() + await bundlerClient.getChainId() return } catch { await new Promise((resolve) => setTimeout(resolve, 1000)) @@ -107,34 +94,41 @@ export const getAnvilWalletClient = ({ }) } -export const getPimlicoPaymasterClient = ({ - entryPoint, +export const getBundlerClient = ({ + altoRpc, + anvilRpc, + account, paymasterRpc -}: { entryPoint: T; paymasterRpc: string }): PimlicoPaymasterClient => { - return createPimlicoPaymasterClient({ - chain: foundry, - transport: http(paymasterRpc), - entryPoint - }) -} +}: { + altoRpc: string + paymasterRpc?: string + anvilRpc: string + account?: account +}) => { + const paymaster = paymasterRpc + ? createPaymasterClient({ + transport: http(paymasterRpc) + }) + : undefined -export const getBundlerClient = ({ - entryPoint, - altoRpc -}: { entryPoint: T; altoRpc: string }): BundlerClient => - createBundlerClient({ - chain: foundry, - entryPoint, + return createBundlerClient({ + client: getPublicClient(anvilRpc), + account, + paymaster, transport: http(altoRpc) - }) as BundlerClient + }) +} -export const getPimlicoBundlerClient = ({ +export const getPimlicoClient = ({ entryPoint, altoRpc -}: { entryPoint: T; altoRpc: string }): PimlicoBundlerClient => - createPimlicoBundlerClient({ +}: { + entryPoint: typeof entryPoint07Address | typeof entryPoint06Address + altoRpc: string +}) => + createPimlicoClient({ chain: foundry, - entryPoint, + entryPointAddress: entryPoint, transport: http(altoRpc) }) @@ -185,254 +179,164 @@ export const fund = async ({ await publicClient.waitForTransactionReceipt({ hash }) } -export const getSimpleAccountClient = async ({ +export const getSimpleAccountClient = async < + entryPointVersion extends "0.6" | "0.7" +>({ entryPoint, - paymasterClient, - anvilRpc, - altoRpc, - privateKey -}: AAParamType): Promise< - SmartAccountClient> + anvilRpc +}: AAParamType): Promise< + ToSimpleSmartAccountReturnType< + entryPointVersion, + entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi + > > => { - const publicClient = getPublicClient(anvilRpc) - - const smartAccount = privateKey - ? await privateKeyToSimpleSmartAccount(publicClient, { - entryPoint, - privateKey - }) - : await signerToSimpleSmartAccount(publicClient, { - entryPoint, - signer: privateKeyToAccount(generatePrivateKey()) - }) - - return createSmartAccountClient({ - chain: foundry, - account: smartAccount, - bundlerTransport: http(altoRpc), - // @ts-ignore - middleware: paymasterClient - ? { - sponsorUserOperation: paymasterClient.sponsorUserOperation - } - : undefined + return toSimpleSmartAccount< + entryPointVersion, + entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi + >({ + client: getPublicClient(anvilRpc), + entryPoint: { + address: + entryPoint.version === "0.6" + ? entryPoint06Address + : entryPoint07Address, + version: (entryPoint.version === "0.6" + ? "0.6" + : "0.7") as entryPointVersion, + abi: (entryPoint.version === "0.6" + ? entryPoint06Abi + : entryPoint07Abi) as entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi + }, + owner: privateKeyToAccount(generatePrivateKey()) }) } -export const getLightAccountClient = async ({ +export const getLightAccountClient = async < + entryPointVersion extends "0.6" | "0.7" +>({ entryPoint, - paymasterClient, - anvilRpc, - altoRpc, - privateKey -}: AAParamType): Promise< - SmartAccountClient> -> => { - const publicClient = getPublicClient(anvilRpc) - const smartAccount = privateKey - ? await privateKeyToLightSmartAccount(publicClient, { - entryPoint, - lightAccountVersion: "1.1.0", - privateKey - }) - : await signerToLightSmartAccount(publicClient, { - entryPoint, - signer: privateKeyToAccount(generatePrivateKey()), - lightAccountVersion: "1.1.0" - }) - - return createSmartAccountClient({ - chain: foundry, - account: smartAccount, - bundlerTransport: http(altoRpc), - entryPoint: entryPoint, - // eip7677Client: await getEip7677Client({ entryPoint }), - middleware: { - // @ts-ignore - sponsorUserOperation: paymasterClient?.sponsorUserOperation - } + anvilRpc +}: AAParamType) => { + return toLightSmartAccount({ + entryPoint: { + address: + entryPoint.version === "0.6" + ? entryPoint06Address + : entryPoint07Address, + version: entryPoint.version === "0.6" ? "0.6" : "0.7", + abi: + entryPoint.version === "0.6" ? entryPoint06Abi : entryPoint07Abi + }, + client: getPublicClient(anvilRpc), + version: "1.1.0", + owner: privateKeyToAccount(generatePrivateKey()) }) } // Only supports v0.6 for now export const getTrustAccountClient = async < - T extends ENTRYPOINT_ADDRESS_V06_TYPE + entryPointVersion extends "0.6" | "0.7" >({ - entryPoint, - paymasterClient, - altoRpc, - anvilRpc, - privateKey -}: AAParamType): Promise< - SmartAccountClient> -> => { - const publicClient = getPublicClient(anvilRpc) - const smartAccount = privateKey - ? await privateKeyToTrustSmartAccount(publicClient, { - entryPoint, - privateKey - }) - : await signerToTrustSmartAccount(publicClient, { - entryPoint, - signer: privateKeyToAccount(generatePrivateKey()) - }) - - // @ts-ignore - return createSmartAccountClient({ - chain: foundry, - account: smartAccount, - bundlerTransport: http(altoRpc), - middleware: { - // @ts-ignore - sponsorUserOperation: paymasterClient?.sponsorUserOperation - } + anvilRpc +}: AAParamType) => { + return toTrustSmartAccount({ + client: getPublicClient(anvilRpc), + owner: privateKeyToAccount(generatePrivateKey()) }) } // Only supports v0.6 for now -export const getBiconomyClient = async ({ - paymasterClient, - privateKey, - anvilRpc, - altoRpc, - entryPoint = ENTRYPOINT_ADDRESS_V06 -}: AAParamType) => { - const publicClient = getPublicClient(anvilRpc) - const ecdsaSmartAccount = privateKey - ? await privateKeyToBiconomySmartAccount(publicClient, { - entryPoint, - privateKey - }) - : await signerToBiconomySmartAccount(publicClient, { - entryPoint, - signer: privateKeyToAccount(generatePrivateKey()) - }) - - // @ts-ignore - return createSmartAccountClient({ - account: ecdsaSmartAccount, - chain: foundry, - bundlerTransport: http(altoRpc), - middleware: { - // @ts-ignore - sponsorUserOperation: paymasterClient?.sponsorUserOperation - } +export const getBiconomyClient = async < + entryPointVersion extends "0.6" | "0.7" +>({ + anvilRpc +}: AAParamType) => { + return toBiconomySmartAccount({ + client: getPublicClient(anvilRpc), + owner: privateKeyToAccount(generatePrivateKey()) }) } -export const getKernelEcdsaClient = async ({ +export const getKernelEcdsaClient = async < + entryPointVersion extends "0.6" | "0.7" +>({ entryPoint, - paymasterClient, anvilRpc, - altoRpc, - privateKey = generatePrivateKey(), version -}: AAParamType & { version?: KernelVersion }): Promise< - SmartAccountClient> -> => { +}: AAParamType & { + version?: KernelVersion +}) => { const publicClient = getPublicClient(anvilRpc) if ( (version === "0.3.0-beta" || version === "0.3.1") && - entryPoint === ENTRYPOINT_ADDRESS_V06 + entryPoint.version === "0.6" ) { throw new Error("ERC7579 is not supported for V06") } - const kernelEcdsaAccount = - entryPoint === ENTRYPOINT_ADDRESS_V07 - ? await signerToEcdsaKernelSmartAccount(publicClient, { - entryPoint: entryPoint, - signer: privateKeyToAccount(privateKey), - version - }) - : await signerToEcdsaKernelSmartAccount(publicClient, { - entryPoint, - signer: privateKeyToAccount(privateKey) - }) - // @ts-ignore - return createSmartAccountClient({ - chain: foundry, - account: kernelEcdsaAccount as KernelEcdsaSmartAccount, - bundlerTransport: http(altoRpc), - middleware: { - // @ts-ignore - sponsorUserOperation: paymasterClient?.sponsorUserOperation - } + return toEcdsaKernelSmartAccount({ + client: publicClient, + entryPoint: { + address: + entryPoint.version === "0.6" + ? entryPoint06Address + : entryPoint07Address, + version: entryPoint.version === "0.6" ? "0.6" : "0.7", + abi: + entryPoint.version === "0.6" ? entryPoint06Abi : entryPoint07Abi + }, + owner: privateKeyToAccount(generatePrivateKey()), + version }) } -export const getSafeClient = async ({ - setupTransactions = [], +export const getSafeClient = async ({ entryPoint, - paymasterClient, anvilRpc, - altoRpc, - privateKey, erc7579 }: { - setupTransactions?: { - to: Address - data: Address - value: bigint - }[] - anvilRpc: string - altoRpc: string - entryPoint: T - paymasterClient?: PimlicoPaymasterClient - privateKey?: Hex erc7579?: boolean -}): Promise>> => { +} & AAParamType) => { const publicClient = getPublicClient(anvilRpc) - const safeSmartAccount = privateKey - ? await privateKeyToSafeSmartAccount(publicClient, { - entryPoint, - privateKey, - safeVersion: "1.4.1", - saltNonce: 420n, - safe4337ModuleAddress: erc7579 - ? "0x3Fdb5BC686e861480ef99A6E3FaAe03c0b9F32e2" - : undefined, - erc7579LaunchpadAddress: erc7579 - ? "0xEBe001b3D534B9B6E2500FB78E67a1A137f561CE" - : undefined - }) - : await signerToSafeSmartAccount(publicClient, { - entryPoint, - signer: privateKeyToAccount(generatePrivateKey()), - safeVersion: "1.4.1", - saltNonce: 420n, - safe4337ModuleAddress: erc7579 - ? "0x3Fdb5BC686e861480ef99A6E3FaAe03c0b9F32e2" - : undefined, - erc7579LaunchpadAddress: erc7579 - ? "0xEBe001b3D534B9B6E2500FB78E67a1A137f561CE" - : undefined - }) - - const pimlicoBundlerClient = getPimlicoBundlerClient({ - entryPoint, - altoRpc - }) - - // @ts-ignore - return createSmartAccountClient({ - chain: foundry, - account: safeSmartAccount, - bundlerTransport: http(altoRpc), - middleware: { - gasPrice: async () => - (await pimlicoBundlerClient.getUserOperationGasPrice()).fast, - // @ts-ignore - sponsorUserOperation: paymasterClient?.sponsorUserOperation - } + return toSafeSmartAccount({ + client: publicClient, + entryPoint: { + address: + entryPoint.version === "0.6" + ? entryPoint06Address + : entryPoint07Address, + version: entryPoint.version === "0.6" ? "0.6" : "0.7", + abi: + entryPoint.version === "0.6" ? entryPoint06Abi : entryPoint07Abi + }, + owner: privateKeyToAccount(generatePrivateKey()), + version: "1.4.1", + saltNonce: 420n, + safe4337ModuleAddress: erc7579 + ? "0x3Fdb5BC686e861480ef99A6E3FaAe03c0b9F32e2" + : undefined, + erc7579LaunchpadAddress: erc7579 + ? "0xEBe001b3D534B9B6E2500FB78E67a1A137f561CE" + : undefined }) } -export const getEip7677Client = async ({ +export const getEip7677Client = async ({ entryPoint -}: { entryPoint: TEntryPoint }) => { +}: { + entryPoint: { + address: typeof entryPoint06Address | typeof entryPoint07Address + version: "0.6" | "0.7" + } +}) => { const client = createClient({ chain: foundry, transport: http(PAYMASTER_RPC) @@ -444,15 +348,13 @@ export const getEip7677Client = async ({ export const getCoreSmartAccounts = () => [ { name: "Trust", - getSmartAccountClient: async ( - conf: AAParamType + getSmartAccountClient: async ( + conf: AAParamType ) => { - if (conf.entryPoint !== ENTRYPOINT_ADDRESS_V06) { - throw new Error("Biconomy only works with V06") - } - return getTrustAccountClient( - conf as AAParamType - ) + return getBundlerClient({ + account: await getTrustAccountClient(conf), + ...conf + }) }, supportsEntryPointV06: true, supportsEntryPointV07: false, @@ -460,46 +362,66 @@ export const getCoreSmartAccounts = () => [ }, { name: "LightAccount v1.1.0", - getSmartAccountClient: async ( - conf: AAParamType - ) => getLightAccountClient(conf), + getSmartAccountClient: async ( + conf: AAParamType + ) => + getBundlerClient({ + account: await getLightAccountClient(conf), + ...conf + }), supportsEntryPointV06: true, supportsEntryPointV07: false, isEip1271Compliant: true }, { name: "Simple", - getSmartAccountClient: async ( - conf: AAParamType - ) => getSimpleAccountClient(conf), + getSmartAccountClient: async ( + conf: AAParamType + ) => + getBundlerClient({ + account: await getSimpleAccountClient(conf), + ...conf + }), supportsEntryPointV06: true, supportsEntryPointV07: true, isEip1271Compliant: false }, { name: "Kernel", - getSmartAccountClient: async ( - conf: AAParamType - ) => getKernelEcdsaClient(conf), + getSmartAccountClient: async ( + conf: AAParamType + ) => + getBundlerClient({ + account: await getKernelEcdsaClient(conf), + ...conf + }), supportsEntryPointV06: true, supportsEntryPointV07: true, isEip1271Compliant: true }, { name: "Kernel 7579 0.3.0-beta", - getSmartAccountClient: async ( - conf: AAParamType + getSmartAccountClient: async ( + conf: AAParamType ) => - getKernelEcdsaClient({ - ...conf, - version: "0.3.0-beta" as KernelVersion + getBundlerClient({ + account: await getKernelEcdsaClient({ + ...conf, + version: "0.3.0-beta" as KernelVersion + }), + ...conf }), - getErc7579SmartAccountClient: async ( - conf: AAParamType + getErc7579SmartAccountClient: async < + entryPointVersion extends "0.6" | "0.7" + >( + conf: AAParamType ) => - getKernelEcdsaClient({ - ...conf, - version: "0.3.0-beta" as KernelVersion + getBundlerClient({ + account: await getKernelEcdsaClient({ + ...conf, + version: "0.3.0-beta" as KernelVersion + }), + ...conf }), supportsEntryPointV06: false, supportsEntryPointV07: true, @@ -507,19 +429,27 @@ export const getCoreSmartAccounts = () => [ }, { name: "Kernel 7579 0.3.1", - getSmartAccountClient: async ( - conf: AAParamType + getSmartAccountClient: async ( + conf: AAParamType ) => - getKernelEcdsaClient({ - ...conf, - version: "0.3.1" as KernelVersion + getBundlerClient({ + account: await getKernelEcdsaClient({ + ...conf, + version: "0.3.1" as KernelVersion + }), + ...conf }), - getErc7579SmartAccountClient: async ( - conf: AAParamType + getErc7579SmartAccountClient: async < + entryPointVersion extends "0.6" | "0.7" + >( + conf: AAParamType ) => - getKernelEcdsaClient({ - ...conf, - version: "0.3.1" as KernelVersion + getBundlerClient({ + account: await getKernelEcdsaClient({ + ...conf, + version: "0.3.1" as KernelVersion + }), + ...conf }), supportsEntryPointV06: false, supportsEntryPointV07: true, @@ -527,110 +457,49 @@ export const getCoreSmartAccounts = () => [ }, { name: "Biconomy", - getSmartAccountClient: async ( - conf: AAParamType - ) => { - if (conf.entryPoint !== ENTRYPOINT_ADDRESS_V06) { - throw new Error("Biconomy only works with V06") - } - return getBiconomyClient( - conf as AAParamType - ) - }, + getSmartAccountClient: async ( + conf: AAParamType + ) => + getBundlerClient({ + account: await getBiconomyClient(conf), + ...conf + }), supportsEntryPointV06: true, supportsEntryPointV07: false, isEip1271Compliant: true }, { name: "Safe", - getSmartAccountClient: async ( - conf: AAParamType - ) => getSafeClient(conf), + getSmartAccountClient: async ( + conf: AAParamType + ) => + getBundlerClient({ + account: await getSafeClient(conf), + ...conf + }), supportsEntryPointV06: true, supportsEntryPointV07: true, isEip1271Compliant: true }, { name: "Safe 7579", - getSmartAccountClient: async ( - conf: AAParamType - ) => getSafeClient({ ...conf, erc7579: true }), - getErc7579SmartAccountClient: async ( - conf: AAParamType - ) => getSafeClient({ ...conf, erc7579: true }), - supportsEntryPointV06: false, - supportsEntryPointV07: true, - isEip1271Compliant: true - }, - - // ---------------------------- Account from private key ------------------------------------------------- - - { - name: "Trust private key", - getSmartAccountClient: async ( - conf: AAParamType - ) => { - if (conf.entryPoint !== ENTRYPOINT_ADDRESS_V06) { - throw new Error("Biconomy only works with V06") - } - return getTrustAccountClient({ - ...(conf as AAParamType), - privateKey: generatePrivateKey() - }) - }, - supportsEntryPointV06: true, - supportsEntryPointV07: false, - isEip1271Compliant: true - }, - { - name: "LightAccount v1.1.0 private key", - getSmartAccountClient: async ( - conf: AAParamType + getSmartAccountClient: async ( + conf: AAParamType ) => - getLightAccountClient({ - ...conf, - privateKey: generatePrivateKey() + getBundlerClient({ + account: await getSafeClient({ ...conf, erc7579: true }), + ...conf }), - supportsEntryPointV06: true, - supportsEntryPointV07: false, - isEip1271Compliant: true - }, - { - name: "Simple private key", - getSmartAccountClient: async ( - conf: AAParamType + getErc7579SmartAccountClient: async < + entryPointVersion extends "0.6" | "0.7" + >( + conf: AAParamType ) => - getSimpleAccountClient({ - ...conf, - privateKey: generatePrivateKey() + getBundlerClient({ + account: await getSafeClient({ ...conf, erc7579: true }), + ...conf }), - supportsEntryPointV06: true, - supportsEntryPointV07: true, - isEip1271Compliant: false - }, - { - name: "Biconomy private key", - getSmartAccountClient: async ( - conf: AAParamType - ) => { - if (conf.entryPoint !== ENTRYPOINT_ADDRESS_V06) { - throw new Error("Biconomy only works with V06") - } - return getBiconomyClient({ - ...(conf as AAParamType), - privateKey: generatePrivateKey() - }) - }, - supportsEntryPointV06: true, - supportsEntryPointV07: false, - isEip1271Compliant: true - }, - { - name: "Safe private key private key", - getSmartAccountClient: async ( - conf: AAParamType - ) => getSafeClient({ ...conf, privateKey: generatePrivateKey() }), - supportsEntryPointV06: true, + supportsEntryPointV06: false, supportsEntryPointV07: true, isEip1271Compliant: true } diff --git a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts index cbaffd85..95ddada7 100644 --- a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts +++ b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts @@ -135,15 +135,13 @@ const getAccountAddress = async ({ } export type ToBiconomySmartAccountParameters< - entryPointAddress extends - typeof entryPoint06Address = typeof entryPoint06Address, entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi > = Prettify<{ client: Client owner: LocalAccount address?: Address | undefined entryPoint?: { - address: entryPointAddress + address: typeof entryPoint06Address version: "0.6" abi: entryPointAbi } @@ -178,14 +176,9 @@ export type ToBiconomySmartAccountReturnType< */ export async function toBiconomySmartAccount< - entryPointAddress extends - typeof entryPoint06Address = typeof entryPoint06Address, entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi >( - parameters: ToBiconomySmartAccountParameters< - entryPointAddress, - entryPointAbi - > + parameters: ToBiconomySmartAccountParameters ): Promise> { // ): Promise> { @@ -198,7 +191,7 @@ export async function toBiconomySmartAccount< abi: entryPoint06Abi, version: "0.6" } as { - address: entryPointAddress + address: typeof entryPoint06Address version: "0.6" abi: entryPointAbi }) diff --git a/packages/permissionless/accounts/kernel/constants.ts b/packages/permissionless/accounts/kernel/constants.ts index 90d7825a..def56f48 100644 --- a/packages/permissionless/accounts/kernel/constants.ts +++ b/packages/permissionless/accounts/kernel/constants.ts @@ -1,15 +1,7 @@ export const DUMMY_ECDSA_SIGNATURE = "0xfffffffffffffffffffffffffffffff0000000000000000000000000000000007aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa1c" export const ROOT_MODE_KERNEL_V2 = "0x00000000" -export enum CALL_TYPE { - SINGLE = "0x00", - BATCH = "0x01", - DELEGATE_CALL = "0xFF" -} -export enum EXEC_TYPE { - DEFAULT = "0x00", - TRY_EXEC = "0x01" -} + export const VALIDATOR_TYPE = { ROOT: "0x00", VALIDATOR: "0x01", diff --git a/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts b/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts index 662f2833..da0fc4b6 100644 --- a/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts +++ b/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts @@ -327,24 +327,14 @@ const getAccountAddress = async ({ } export type ToEcdsaKernelSmartAccountParameters< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address, entryPointVersion extends "0.6" | "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi, - kernelVersion extends - KernelVersion = entryPointVersion extends "0.6" - ? KernelVersion<"0.6"> - : KernelVersion<"0.7"> + entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi, + kernelVersion extends KernelVersion > = { client: Client owner: LocalAccount entryPoint?: { - address: entryPointAddress + address: typeof entryPoint06Address | typeof entryPoint07Address abi: entryPointAbi version: entryPointVersion } @@ -359,12 +349,10 @@ export type ToEcdsaKernelSmartAccountParameters< } export type EcdsaKernelSmartAccountImplementation< - entryPointVersion extends "0.6" | "0.7", + entryPointVersion extends "0.6" | "0.7" = "0.7", entryPointAbi extends | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi + | typeof entryPoint07Abi = typeof entryPoint07Abi > = Assign< SmartAccountImplementation< entryPointAbi, @@ -379,12 +367,10 @@ export type EcdsaKernelSmartAccountImplementation< > export type ToEcdsaKernelSmartAccountReturnType< - entryPointVersion extends "0.6" | "0.7", + entryPointVersion extends "0.6" | "0.7" = "0.7", entryPointAbi extends | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi + | typeof entryPoint07Abi = typeof entryPoint07Abi > = SmartAccount< EcdsaKernelSmartAccountImplementation > @@ -399,20 +385,14 @@ export type ToEcdsaKernelSmartAccountReturnType< * @param ecdsaValidatorAddress */ export async function toEcdsaKernelSmartAccount< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address = typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7" = "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi, + kernelVersion extends KernelVersion >( parameters: ToEcdsaKernelSmartAccountParameters< - entryPointAddress, entryPointVersion, - entryPointAbi + entryPointAbi, + kernelVersion > ): Promise< ToEcdsaKernelSmartAccountReturnType @@ -511,16 +491,14 @@ export async function toEcdsaKernelSmartAccount< factoryData: await generateInitCode() } }, - async getNonce(args) { + async getNonce(_args) { return getAccountNonce(client, { address: await getAddress(), entryPointAddress: entryPoint.address, key: getNonceKeyWithEncoding( kernelVersion, ecdsaValidatorAddress, - args?.key ?? parameters?.nonceKey ?? 0n - // @dev specify the custom nonceKey here when integrating the said feature - /*, nonceKey */ + /*args?.key ?? */ parameters.nonceKey ?? 0n ) }) }, diff --git a/packages/permissionless/accounts/kernel/utils/encodeCallData.ts b/packages/permissionless/accounts/kernel/utils/encodeCallData.ts index 1b58ef7c..87f3611c 100644 --- a/packages/permissionless/accounts/kernel/utils/encodeCallData.ts +++ b/packages/permissionless/accounts/kernel/utils/encodeCallData.ts @@ -40,7 +40,7 @@ export const encodeCallData = ({ return encode7579Calls({ mode: { - type: Array.isArray(calls) ? "batchcall" : "call", + type: calls.length > 1 ? "batchcall" : "call", revertOnError: false, selector: "0x", context: "0x" diff --git a/packages/permissionless/accounts/kernel/utils/getExecMode.ts b/packages/permissionless/accounts/kernel/utils/getExecMode.ts deleted file mode 100644 index 91d5ba68..00000000 --- a/packages/permissionless/accounts/kernel/utils/getExecMode.ts +++ /dev/null @@ -1,18 +0,0 @@ -import { type Hex, concatHex, pad } from "viem" -import type { CALL_TYPE, EXEC_TYPE } from "../constants" - -export const getExecMode = ({ - callType, - execType -}: { - callType: CALL_TYPE - execType: EXEC_TYPE -}): Hex => { - return concatHex([ - callType, // 1 byte - execType, // 1 byte - "0x00000000", // 4 bytes - "0x00000000", // 4 bytes - pad("0x00000000", { size: 22 }) - ]) -} diff --git a/packages/permissionless/accounts/kernel/utils/getNonceKey.ts b/packages/permissionless/accounts/kernel/utils/getNonceKey.ts index 0af2b1a1..17bb482d 100644 --- a/packages/permissionless/accounts/kernel/utils/getNonceKey.ts +++ b/packages/permissionless/accounts/kernel/utils/getNonceKey.ts @@ -28,5 +28,6 @@ export const getNonceKeyWithEncoding = ( ]), { size: 24 } ) // 24 bytes + return BigInt(encoding) } diff --git a/packages/permissionless/accounts/light/toLightSmartAccount.ts b/packages/permissionless/accounts/light/toLightSmartAccount.ts index 20ae28df..28f71d82 100644 --- a/packages/permissionless/accounts/light/toLightSmartAccount.ts +++ b/packages/permissionless/accounts/light/toLightSmartAccount.ts @@ -103,24 +103,17 @@ const getAccountAddress = async < export type LightAccountVersion = "1.1.0" export type ToLightSmartAccountParameters< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address, entryPointVersion extends "0.6" | "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi + entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi > = { client: Client entryPoint?: { - address: entryPointAddress + address: typeof entryPoint06Address | typeof entryPoint07Address abi: entryPointAbi version: entryPointVersion } owner: LocalAccount - lightAccountVersion: LightAccountVersion + version: LightAccountVersion factoryAddress?: Address index?: bigint address?: Address @@ -198,12 +191,10 @@ export type LightSmartAccountImplementation< > export type ToLightSmartAccountReturnType< - entryPointVersion extends "0.6" | "0.7", + entryPointVersion extends "0.6" | "0.7" = "0.7", entryPointAbi extends | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi + | typeof entryPoint07Abi = typeof entryPoint07Abi > = SmartAccount< LightSmartAccountImplementation > @@ -214,24 +205,13 @@ export type ToLightSmartAccountReturnType< * @returns A Private Key Light Account. */ export async function toLightSmartAccount< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address, entryPointVersion extends "0.6" | "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi + entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi >( - parameters: ToLightSmartAccountParameters< - entryPointAddress, - entryPointVersion, - entryPointAbi - > + parameters: ToLightSmartAccountParameters ): Promise> { const { - lightAccountVersion, + version, factoryAddress: _factoryAddress, address, owner, @@ -250,13 +230,13 @@ export async function toLightSmartAccount< version: parameters.entryPoint?.version ?? "0.7" } as const - if (lightAccountVersion !== "1.1.0") { + if (version !== "1.1.0") { throw new Error( "Only Light Account version 1.1.0 is supported at the moment" ) } - const { factoryAddress } = getDefaultAddresses(lightAccountVersion, { + const { factoryAddress } = getDefaultAddresses(version, { factoryAddress: _factoryAddress }) diff --git a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts index ee3d8d10..e6557c0a 100644 --- a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts +++ b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts @@ -43,6 +43,7 @@ import { } from "viem/actions" import { getAction } from "viem/utils" import { getAccountNonce } from "../../actions/public/getAccountNonce" +import { isSmartAccountDeployed } from "../../utils" import { encode7579Calls } from "../../utils/encode7579Calls" export type SafeVersion = "1.4.1" @@ -1031,22 +1032,15 @@ type GetErc7579Params = } export type ToSafeSmartAccountParameters< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address = typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7" = "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi, - TErc7579 extends Address | undefined = Address | undefined + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi, + TErc7579 extends Address | undefined > = { client: Client owner: LocalAccount - safeVersion: SafeVersion + version: SafeVersion entryPoint?: { - address: entryPointAddress + address: typeof entryPoint06Address | typeof entryPoint07Address abi: entryPointAbi version: entryPointVersion } @@ -1063,24 +1057,17 @@ export type ToSafeSmartAccountParameters< } & GetErc7579Params function isErc7579Args< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address = typeof entryPoint07Address, entryPointVersion extends "0.6" | "0.7" = "0.7", entryPointAbi extends | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi + | typeof entryPoint07Abi = typeof entryPoint07Abi >( args: ToSafeSmartAccountParameters< - entryPointAddress, entryPointVersion, entryPointAbi, Address | undefined > ): args is ToSafeSmartAccountParameters< - entryPointAddress, entryPointVersion, entryPointAbi, Address @@ -1089,12 +1076,10 @@ function isErc7579Args< } export type SafeSmartAccountImplementation< - entryPointVersion extends "0.6" | "0.7", + entryPointVersion extends "0.6" | "0.7" = "0.7", entryPointAbi extends | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi + | typeof entryPoint07Abi = typeof entryPoint07Abi > = Assign< SmartAccountImplementation< entryPointAbi, @@ -1109,12 +1094,10 @@ export type SafeSmartAccountImplementation< > export type ToSafeSmartAccountReturnType< - entryPointVersion extends "0.6" | "0.7", + entryPointVersion extends "0.6" | "0.7" = "0.7", entryPointAbi extends | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi + | typeof entryPoint07Abi = typeof entryPoint07Abi > = SmartAccount< SafeSmartAccountImplementation > @@ -1125,19 +1108,11 @@ export type ToSafeSmartAccountReturnType< * @returns A Private Key Simple Account. */ export async function toSafeSmartAccount< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address = typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7" = "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi, - TErc7579 extends Address | undefined = undefined + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi, + TErc7579 extends Address | undefined >( parameters: ToSafeSmartAccountParameters< - entryPointAddress, entryPointVersion, entryPointAbi, TErc7579 @@ -1147,7 +1122,7 @@ export async function toSafeSmartAccount< client, owner, address, - safeVersion, + version, safe4337ModuleAddress: _safe4337ModuleAddress, safeProxyFactoryAddress: _safeProxyFactoryAddress, safeSingletonAddress: _safeSingletonAddress, @@ -1208,7 +1183,7 @@ export async function toSafeSmartAccount< safeSingletonAddress, multiSendAddress, multiSendCallOnlyAddress - } = getDefaultAddresses(safeVersion, entryPoint.version, { + } = getDefaultAddresses(version, entryPoint.version, { safeModuleSetupAddress: _safeModuleSetupAddress, safe4337ModuleAddress: _safe4337ModuleAddress, safeProxyFactoryAddress: _safeProxyFactoryAddress, @@ -1263,7 +1238,10 @@ export async function toSafeSmartAccount< const hasMultipleCalls = calls.length > 1 if (erc7579LaunchpadAddress) { - const safeDeployed = await this.isDeployed() + const safeDeployed = await isSmartAccountDeployed( + client, + await getAddress() + ) if (!safeDeployed) { const initData = get7579LaunchPadInitData({ diff --git a/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts b/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts index bc1fcb37..a249dfd3 100644 --- a/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts +++ b/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts @@ -22,7 +22,6 @@ import { getChainId, signMessage } from "viem/actions" import { getAction } from "viem/utils" import { getAccountNonce } from "../../actions/public/getAccountNonce" import { getSenderAddress } from "../../actions/public/getSenderAddress" -import { getEntryPointVersion } from "../../utils" const getAccountInitCode = async ( owner: Address, @@ -62,28 +61,24 @@ const getAccountInitCode = async ( }) } -const getAccountAddress = async < - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address ->({ +const getAccountAddress = async ({ client, factoryAddress, entryPointAddress, owner, + entryPointVersion, index = BigInt(0) }: { client: Client factoryAddress: Address owner: Address - entryPointAddress: entryPointAddress + entryPointAddress: typeof entryPoint06Address | typeof entryPoint07Address + entryPointVersion: "0.6" | "0.7" index?: bigint }): Promise
=> { - const entryPointVersion = getEntryPointVersion(entryPointAddress) - const factoryData = await getAccountInitCode(owner, index) - if (entryPointVersion === "v0.6") { + if (entryPointVersion === "0.6") { return getSenderAddress(client, { initCode: concatHex([factoryAddress, factoryData]), entryPointAddress: entryPointAddress as typeof entryPoint06Address @@ -99,21 +94,14 @@ const getAccountAddress = async < } export type ToSimpleSmartAccountParameters< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address = typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7" = "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi > = { client: Client owner: LocalAccount factoryAddress?: Address entryPoint?: { - address: entryPointAddress + address: typeof entryPoint06Address | typeof entryPoint07Address abi: entryPointAbi version: entryPointVersion } @@ -135,12 +123,10 @@ const getFactoryAddress = ( } export type SimpleSmartAccountImplementation< - entryPointVersion extends "0.6" | "0.7", + entryPointVersion extends "0.6" | "0.7" = "0.7", entryPointAbi extends | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi + | typeof entryPoint07Abi = typeof entryPoint07Abi > = Assign< SmartAccountImplementation< entryPointAbi, @@ -155,12 +141,10 @@ export type SimpleSmartAccountImplementation< > export type ToSimpleSmartAccountReturnType< - entryPointVersion extends "0.6" | "0.7", + entryPointVersion extends "0.6" | "0.7" = "0.7", entryPointAbi extends | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi + | typeof entryPoint07Abi = typeof entryPoint07Abi > = SmartAccount< SimpleSmartAccountImplementation > @@ -171,27 +155,15 @@ export type ToSimpleSmartAccountReturnType< * @returns A Private Key Simple Account. */ export async function toSimpleSmartAccount< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address = typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7" = "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi + entryPointVersion extends "0.6" | "0.7", + entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi >( - parameters: ToSimpleSmartAccountParameters< - entryPointAddress, - entryPointVersion, - entryPointAbi - > + parameters: ToSimpleSmartAccountParameters ): Promise> { const { client, owner, factoryAddress: _factoryAddress, - entryPoint: entryPointAddress, index = BigInt(0), address, nonceKey @@ -222,6 +194,7 @@ export async function toSimpleSmartAccount< client, factoryAddress, entryPointAddress: entryPoint.address, + entryPointVersion: entryPoint.version, owner: owner.address, index })) @@ -244,7 +217,7 @@ export async function toSimpleSmartAccount< getAddress, async encodeCalls(calls) { if (calls.length > 1) { - if (getEntryPointVersion(entryPointAddress) === "v0.6") { + if (entryPoint.version === "0.6") { return encodeFunctionData({ abi: [ { diff --git a/packages/permissionless/accounts/trust/toTrustSmartAccount.ts b/packages/permissionless/accounts/trust/toTrustSmartAccount.ts index 6423fb97..ee00e92e 100644 --- a/packages/permissionless/accounts/trust/toTrustSmartAccount.ts +++ b/packages/permissionless/accounts/trust/toTrustSmartAccount.ts @@ -16,7 +16,6 @@ import { type UserOperation, entryPoint06Abi, entryPoint06Address, - type entryPoint07Abi, getUserOperationHash, toSmartAccount } from "viem/account-abstraction" @@ -61,8 +60,6 @@ export const TRUST_ADDRESSES: { } export type ToTrustSmartAccountParameters< - entryPointAddress extends - typeof entryPoint06Address = typeof entryPoint06Address, entryPointVersion extends "0.6" = "0.6", entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi > = { @@ -70,7 +67,7 @@ export type ToTrustSmartAccountParameters< owner: LocalAccount factoryAddress?: Address entryPoint?: { - address: entryPointAddress + address: typeof entryPoint06Address abi: entryPointAbi version: entryPointVersion } @@ -81,12 +78,8 @@ export type ToTrustSmartAccountParameters< } export type TrustSmartAccountImplementation< - entryPointVersion extends "0.6" | "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi + entryPointVersion extends "0.6" = "0.6", + entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi > = Assign< SmartAccountImplementation< entryPointAbi, @@ -113,16 +106,10 @@ export type ToTrustSmartAccountReturnType< * @returns A Private Key Trust Smart Account. */ export async function toTrustSmartAccount< - entryPointAddress extends - typeof entryPoint06Address = typeof entryPoint06Address, entryPointVersion extends "0.6" = "0.6", entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi >( - parameters: ToTrustSmartAccountParameters< - entryPointAddress, - entryPointVersion, - entryPointAbi - > + parameters: ToTrustSmartAccountParameters ): Promise> { const { owner, diff --git a/packages/permissionless/actions/erc7579.ts b/packages/permissionless/actions/erc7579.ts index d2f62948..bc5ddde4 100644 --- a/packages/permissionless/actions/erc7579.ts +++ b/packages/permissionless/actions/erc7579.ts @@ -1,7 +1,8 @@ import type { Chain, Client, Hash, Transport } from "viem" -import type { SmartAccount } from "../accounts" -import type { GetAccountParameter } from "../types" -import type { EntryPoint } from "../types/entrypoint" +import type { + GetSmartAccountParameter, + SmartAccount +} from "viem/account-abstraction" import { accountId } from "./erc7579/accountId" import { type InstallModuleParameters, @@ -34,35 +35,30 @@ import { uninstallModules } from "./erc7579/uninstallModules" -export type Erc7579Actions< - TEntryPoint extends EntryPoint, - TSmartAccount extends SmartAccount | undefined -> = { +export type Erc7579Actions = { accountId: ( - args?: TSmartAccount extends undefined - ? GetAccountParameter - : undefined + args?: GetSmartAccountParameter ) => Promise installModule: ( - args: InstallModuleParameters + args: InstallModuleParameters ) => Promise installModules: ( - args: InstallModulesParameters + args: InstallModulesParameters ) => Promise isModuleInstalled: ( - args: IsModuleInstalledParameters + args: IsModuleInstalledParameters ) => Promise supportsExecutionMode: ( - args: SupportsExecutionModeParameters + args: SupportsExecutionModeParameters ) => Promise supportsModule: ( - args: SupportsModuleParameters + args: SupportsModuleParameters ) => Promise uninstallModule: ( - args: UninstallModuleParameters + args: UninstallModuleParameters ) => Promise uninstallModules: ( - args: UninstallModulesParameters + args: UninstallModulesParameters ) => Promise } @@ -88,55 +84,17 @@ export { uninstallModules } -export function erc7579Actions(_args: { - entryPoint: TEntryPoint -}) { - return < - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends SmartAccount | undefined = - | SmartAccount - | undefined - >( - client: Client - ): Erc7579Actions => ({ +export function erc7579Actions() { + return ( + client: Client + ): Erc7579Actions => ({ accountId: (args) => accountId(client, args), - installModule: (args) => - installModule( - client, - args - ), - installModules: (args) => - installModules( - client, - args - ), - isModuleInstalled: (args) => - isModuleInstalled( - client, - args - ), - supportsExecutionMode: (args) => - supportsExecutionMode< - TEntryPoint, - TTransport, - TChain, - TSmartAccount - >(client, args), - supportsModule: (args) => - supportsModule( - client, - args - ), - uninstallModule: (args) => - uninstallModule( - client, - args - ), - uninstallModules: (args) => - uninstallModules( - client, - args - ) + installModule: (args) => installModule(client, args), + installModules: (args) => installModules(client, args), + isModuleInstalled: (args) => isModuleInstalled(client, args), + supportsExecutionMode: (args) => supportsExecutionMode(client, args), + supportsModule: (args) => supportsModule(client, args), + uninstallModule: (args) => uninstallModule(client, args), + uninstallModules: (args) => uninstallModules(client, args) }) } diff --git a/packages/permissionless/actions/erc7579/accountId.test.ts b/packages/permissionless/actions/erc7579/accountId.test.ts index c50ea237..5df82efa 100644 --- a/packages/permissionless/actions/erc7579/accountId.test.ts +++ b/packages/permissionless/actions/erc7579/accountId.test.ts @@ -1,12 +1,7 @@ import { zeroAddress } from "viem" -import { generatePrivateKey } from "viem/accounts" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getCoreSmartAccounts, - getPimlicoPaymasterClient -} from "../../../permissionless-test/src/utils" -import { ENTRYPOINT_ADDRESS_V07 } from "../../utils" +import { getCoreSmartAccounts } from "../../../permissionless-test/src/utils" import { accountId } from "./accountId" describe.each(getCoreSmartAccounts())( @@ -15,35 +10,35 @@ describe.each(getCoreSmartAccounts())( testWithRpc.skipIf(!getErc7579SmartAccountClient)( "accountId", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - if (!getErc7579SmartAccountClient) { throw new Error("getErc7579SmartAccountClient not defined") } const smartClient = await getErc7579SmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) + entryPoint: { + version: "0.7" + }, + ...rpc }) - const accountIdBeforeDeploy = await accountId( - smartClient as any - ) + const accountIdBeforeDeploy = await accountId(smartClient) // deploy account - await smartClient.sendTransaction({ - to: zeroAddress, - value: 0n, - data: "0x" + const userOpHash = await smartClient.sendUserOperation({ + calls: [ + { + to: zeroAddress, + value: 0n, + data: "0x" + } + ] + }) + + await smartClient.waitForUserOperationReceipt({ + hash: userOpHash }) - const postDeployAccountId = await accountId(smartClient as any) + const postDeployAccountId = await accountId(smartClient) expect(accountIdBeforeDeploy).toBe(postDeployAccountId) } diff --git a/packages/permissionless/actions/erc7579/accountId.ts b/packages/permissionless/actions/erc7579/accountId.ts index 708d42ca..6b434ebf 100644 --- a/packages/permissionless/actions/erc7579/accountId.ts +++ b/packages/permissionless/actions/erc7579/accountId.ts @@ -1,6 +1,8 @@ import { + type Chain, type Client, ContractFunctionExecutionError, + type Transport, decodeFunctionResult, encodeFunctionData } from "viem" @@ -10,12 +12,10 @@ import type { } from "viem/account-abstraction" import { call, readContract } from "viem/actions" import { getAction } from "viem/utils" -import { AccountOrClientNotFoundError } from "../../utils/signUserOperationHashWithECDSA" +import { AccountNotFoundError } from "../../errors" -export async function accountId< - TSmartAccount extends SmartAccount | undefined = SmartAccount | undefined ->( - client: Client, +export async function accountId( + client: Client, args?: GetSmartAccountParameter ): Promise { let account_ = client.account @@ -25,7 +25,7 @@ export async function accountId< } if (!account_) { - throw new AccountOrClientNotFoundError({ + throw new AccountNotFoundError({ docsPath: "/docs/actions/wallet/sendTransaction" }) } diff --git a/packages/permissionless/actions/erc7579/installModule.test.ts b/packages/permissionless/actions/erc7579/installModule.test.ts index be030d63..39599afe 100644 --- a/packages/permissionless/actions/erc7579/installModule.test.ts +++ b/packages/permissionless/actions/erc7579/installModule.test.ts @@ -1,61 +1,30 @@ -import { - http, - type Chain, - type Transport, - encodeAbiParameters, - encodePacked, - isHash, - zeroAddress -} from "viem" -import { generatePrivateKey } from "viem/accounts" +import { encodeAbiParameters, encodePacked, isHash, zeroAddress } from "viem" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getCoreSmartAccounts, - getPimlicoPaymasterClient -} from "../../../permissionless-test/src/utils" -import type { SmartAccount } from "../../accounts" -import { createBundlerClient } from "../../clients/createBundlerClient" -import type { SmartAccountClient } from "../../clients/createSmartAccountClient" -import type { ENTRYPOINT_ADDRESS_V07_TYPE } from "../../types/entrypoint" -import { ENTRYPOINT_ADDRESS_V07 } from "../../utils" +import { getCoreSmartAccounts } from "../../../permissionless-test/src/utils" import { erc7579Actions } from "../erc7579" import { installModule } from "./installModule" describe.each(getCoreSmartAccounts())( - "installmodule $name", + "installModule $name", ({ getErc7579SmartAccountClient, name }) => { testWithRpc.skipIf(!getErc7579SmartAccountClient)( "installModule", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - if (!getErc7579SmartAccountClient) { throw new Error("getErc7579SmartAccountClient not defined") } - const privateKey = generatePrivateKey() - - const smartClientWithoutExtend: SmartAccountClient< - ENTRYPOINT_ADDRESS_V07_TYPE, - Transport, - Chain, - SmartAccount - > = await getErc7579SmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: privateKey, - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc + const smartClientWithoutExtend = + await getErc7579SmartAccountClient({ + entryPoint: { + version: "0.7" + }, + ...rpc }) - }) const smartClient = smartClientWithoutExtend.extend( - erc7579Actions({ - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) + erc7579Actions() ) const moduleData = encodePacked( @@ -63,8 +32,8 @@ describe.each(getCoreSmartAccounts())( [smartClient.account.address] ) - const opHash = await installModule(smartClient as any, { - account: smartClient.account as any, + const opHash = await installModule(smartClient, { + account: smartClient.account, type: "executor", address: "0xc98B026383885F41d9a995f85FC480E9bb8bB891", context: name.startsWith("Kernel 7579") @@ -81,25 +50,21 @@ describe.each(getCoreSmartAccounts())( : moduleData }) - const bundlerClientV07 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) - expect(isHash(opHash)).toBe(true) const userOperationReceipt = - await bundlerClientV07.waitForUserOperationReceipt({ + await smartClient.waitForUserOperationReceipt({ hash: opHash, timeout: 100000 }) + expect(userOperationReceipt).not.toBeNull() expect(userOperationReceipt?.userOpHash).toBe(opHash) expect( userOperationReceipt?.receipt.transactionHash ).toBeTruthy() - const receipt = await bundlerClientV07.getUserOperationReceipt({ + const receipt = await smartClient.getUserOperationReceipt({ hash: opHash }) @@ -119,38 +84,24 @@ describe.each(getCoreSmartAccounts())( testWithRpc.skipIf(!getErc7579SmartAccountClient)( "installModule", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - if (!getErc7579SmartAccountClient) { throw new Error("getErc7579SmartAccountClient not defined") } - const privateKey = generatePrivateKey() - - const smartClientWithoutExtend: SmartAccountClient< - ENTRYPOINT_ADDRESS_V07_TYPE, - Transport, - Chain, - SmartAccount - > = await getErc7579SmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: privateKey, - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc + const smartClientWithoutExtend = + await getErc7579SmartAccountClient({ + entryPoint: { + version: "0.7" + }, + ...rpc }) - }) const smartClient = smartClientWithoutExtend.extend( - erc7579Actions({ - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) + erc7579Actions() ) - await smartClient.sendTransactions({ - transactions: [ + const userOpHash = await smartClient.sendUserOperation({ + calls: [ { to: smartClient.account.address, value: 0n, @@ -164,13 +115,17 @@ describe.each(getCoreSmartAccounts())( ] }) + await smartClient.waitForUserOperationReceipt({ + hash: userOpHash + }) + const moduleData = encodePacked( ["address"], [smartClient.account.address] ) - const opHash = await installModule(smartClient as any, { - account: smartClient.account as any, + const opHash = await installModule(smartClient, { + account: smartClient.account, type: "executor", address: "0xc98B026383885F41d9a995f85FC480E9bb8bB891", context: name.startsWith("Kernel 7579") @@ -187,15 +142,10 @@ describe.each(getCoreSmartAccounts())( : moduleData }) - const bundlerClientV07 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) - expect(isHash(opHash)).toBe(true) const userOperationReceipt = - await bundlerClientV07.waitForUserOperationReceipt({ + await smartClient.waitForUserOperationReceipt({ hash: opHash, timeout: 100000 }) @@ -205,7 +155,7 @@ describe.each(getCoreSmartAccounts())( userOperationReceipt?.receipt.transactionHash ).toBeTruthy() - const receipt = await bundlerClientV07.getUserOperationReceipt({ + const receipt = await smartClient.getUserOperationReceipt({ hash: opHash }) diff --git a/packages/permissionless/actions/erc7579/installModule.ts b/packages/permissionless/actions/erc7579/installModule.ts index 56302bc9..211e025d 100644 --- a/packages/permissionless/actions/erc7579/installModule.ts +++ b/packages/permissionless/actions/erc7579/installModule.ts @@ -5,22 +5,18 @@ import { encodeFunctionData, getAddress } from "viem" -import type { - GetSmartAccountParameter, - SmartAccount +import { + type GetSmartAccountParameter, + type SmartAccount, + sendUserOperation } from "viem/account-abstraction" import { getAction } from "viem/utils" +import { AccountNotFoundError } from "../../errors" import { parseAccount } from "../../utils/" -import { AccountOrClientNotFoundError } from "../../utils/signUserOperationHashWithECDSA" -import type { Middleware } from "../smartAccount/prepareUserOperationRequest" -import { - type SendUserOperationParameters, - sendUserOperation -} from "../smartAccount/sendUserOperation" import { type ModuleType, parseModuleTypeId } from "./supportsModule" export type InstallModuleParameters< - TSmartAccount extends SmartAccount | undefined = SmartAccount | undefined + TSmartAccount extends SmartAccount | undefined > = GetSmartAccountParameter & { type: ModuleType address: Address @@ -28,80 +24,75 @@ export type InstallModuleParameters< maxFeePerGas?: bigint maxPriorityFeePerGas?: bigint nonce?: bigint -} & Middleware +} export async function installModule< - TSmartAccount extends SmartAccount | undefined = SmartAccount | undefined + TSmartAccount extends SmartAccount | undefined >( client: Client, - parameters: Prettify> + parameters: InstallModuleParameters ): Promise { const { account: account_ = client.account, maxFeePerGas, maxPriorityFeePerGas, nonce, - middleware, address, context } = parameters if (!account_) { - throw new AccountOrClientNotFoundError({ + throw new AccountNotFoundError({ docsPath: "/docs/actions/wallet/sendTransaction" }) } - const account = parseAccount(account_) as SmartAccount - - const installModuleCallData = await account.encodeCallData({ - to: account.address, - value: BigInt(0), - data: encodeFunctionData({ - abi: [ - { - name: "installModule", - type: "function", - stateMutability: "nonpayable", - inputs: [ - { - type: "uint256", - name: "moduleTypeId" - }, - { - type: "address", - name: "module" - }, - { - type: "bytes", - name: "initData" - } - ], - outputs: [] - } - ], - functionName: "installModule", - args: [ - parseModuleTypeId(parameters.type), - getAddress(address), - context - ] - }) - }) + const account = parseAccount(account_) as SmartAccount return getAction( client, - sendUserOperation, + sendUserOperation, "sendUserOperation" )({ - userOperation: { - sender: account.address, - maxFeePerGas: maxFeePerGas, - maxPriorityFeePerGas: maxPriorityFeePerGas, - callData: installModuleCallData, - nonce: nonce - }, - account: account, - middleware - } as SendUserOperationParameters) + calls: [ + { + to: account.address, + value: BigInt(0), + data: encodeFunctionData({ + abi: [ + { + name: "installModule", + type: "function", + stateMutability: "nonpayable", + inputs: [ + { + type: "uint256", + name: "moduleTypeId" + }, + { + type: "address", + name: "module" + }, + { + type: "bytes", + name: "initData" + } + ], + outputs: [] + } + ], + functionName: "installModule", + args: [ + parseModuleTypeId(parameters.type), + getAddress(address), + context + ] + }) + } + ], + maxFeePerGas, + maxPriorityFeePerGas, + nonce, + account + }) } diff --git a/packages/permissionless/actions/erc7579/installModules.test.ts b/packages/permissionless/actions/erc7579/installModules.test.ts index eb89cd04..4500bea6 100644 --- a/packages/permissionless/actions/erc7579/installModules.test.ts +++ b/packages/permissionless/actions/erc7579/installModules.test.ts @@ -1,61 +1,30 @@ -import { - http, - type Chain, - type Transport, - encodeAbiParameters, - encodePacked, - isHash, - zeroAddress -} from "viem" -import { generatePrivateKey } from "viem/accounts" +import { encodeAbiParameters, encodePacked, isHash, zeroAddress } from "viem" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getCoreSmartAccounts, - getPimlicoPaymasterClient -} from "../../../permissionless-test/src/utils" -import type { SmartAccount } from "../../accounts" -import { createBundlerClient } from "../../clients/createBundlerClient" -import type { SmartAccountClient } from "../../clients/createSmartAccountClient" -import type { ENTRYPOINT_ADDRESS_V07_TYPE } from "../../types/entrypoint" -import { ENTRYPOINT_ADDRESS_V07 } from "../../utils" +import { getCoreSmartAccounts } from "../../../permissionless-test/src/utils" import { erc7579Actions } from "../erc7579" import { installModules } from "./installModules" describe.each(getCoreSmartAccounts())( - "installmodules $name", + "installModules $name", ({ getErc7579SmartAccountClient, name }) => { testWithRpc.skipIf(!getErc7579SmartAccountClient)( "installModules", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - if (!getErc7579SmartAccountClient) { throw new Error("getErc7579SmartAccountClient not defined") } - const privateKey = generatePrivateKey() - - const smartClientWithoutExtend: SmartAccountClient< - ENTRYPOINT_ADDRESS_V07_TYPE, - Transport, - Chain, - SmartAccount - > = await getErc7579SmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: privateKey, - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc + const smartClientWithoutExtend = + await getErc7579SmartAccountClient({ + entryPoint: { + version: "0.7" + }, + ...rpc }) - }) const smartClient = smartClientWithoutExtend.extend( - erc7579Actions({ - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) + erc7579Actions() ) const moduleData = encodePacked( @@ -63,8 +32,8 @@ describe.each(getCoreSmartAccounts())( [smartClient.account.address] ) - const opHash = await installModules(smartClient as any, { - account: smartClient.account as any, + const opHash = await installModules(smartClient, { + account: smartClient.account, modules: [ { type: "executor", @@ -89,15 +58,10 @@ describe.each(getCoreSmartAccounts())( ] }) - const bundlerClientV07 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) - expect(isHash(opHash)).toBe(true) const userOperationReceipt = - await bundlerClientV07.waitForUserOperationReceipt({ + await smartClient.waitForUserOperationReceipt({ hash: opHash, timeout: 100000 }) @@ -107,7 +71,7 @@ describe.each(getCoreSmartAccounts())( userOperationReceipt?.receipt.transactionHash ).toBeTruthy() - const receipt = await bundlerClientV07.getUserOperationReceipt({ + const receipt = await smartClient.getUserOperationReceipt({ hash: opHash }) @@ -125,40 +89,26 @@ describe.each(getCoreSmartAccounts())( } ) testWithRpc.skipIf(!getErc7579SmartAccountClient)( - "installModule", + "installModules", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - if (!getErc7579SmartAccountClient) { throw new Error("getErc7579SmartAccountClient not defined") } - const privateKey = generatePrivateKey() - - const smartClientWithoutExtend: SmartAccountClient< - ENTRYPOINT_ADDRESS_V07_TYPE, - Transport, - Chain, - SmartAccount - > = await getErc7579SmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: privateKey, - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc + const smartClientWithoutExtend = + await getErc7579SmartAccountClient({ + entryPoint: { + version: "0.7" + }, + ...rpc }) - }) const smartClient = smartClientWithoutExtend.extend( - erc7579Actions({ - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) + erc7579Actions() ) - await smartClient.sendTransactions({ - transactions: [ + const userOpHash = await smartClient.sendUserOperation({ + calls: [ { to: smartClient.account.address, value: 0n, @@ -172,13 +122,17 @@ describe.each(getCoreSmartAccounts())( ] }) + await smartClient.waitForUserOperationReceipt({ + hash: userOpHash + }) + const moduleData = encodePacked( ["address"], [smartClient.account.address] ) - const opHash = await installModules(smartClient as any, { - account: smartClient.account as any, + const opHash = await installModules(smartClient, { + account: smartClient.account, modules: [ { type: "executor", @@ -203,15 +157,10 @@ describe.each(getCoreSmartAccounts())( ] }) - const bundlerClientV07 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) - expect(isHash(opHash)).toBe(true) const userOperationReceipt = - await bundlerClientV07.waitForUserOperationReceipt({ + await smartClient.waitForUserOperationReceipt({ hash: opHash, timeout: 100000 }) @@ -221,7 +170,7 @@ describe.each(getCoreSmartAccounts())( userOperationReceipt?.receipt.transactionHash ).toBeTruthy() - const receipt = await bundlerClientV07.getUserOperationReceipt({ + const receipt = await smartClient.getUserOperationReceipt({ hash: opHash }) diff --git a/packages/permissionless/actions/erc7579/installModules.ts b/packages/permissionless/actions/erc7579/installModules.ts index c9a81468..ec9e97f1 100644 --- a/packages/permissionless/actions/erc7579/installModules.ts +++ b/packages/permissionless/actions/erc7579/installModules.ts @@ -7,115 +7,88 @@ import { encodeFunctionData, getAddress } from "viem" -import { getAction } from "viem/utils" -import type { SmartAccount } from "../../accounts/types" -import type { GetAccountParameter, Prettify } from "../../types/" -import type { EntryPoint } from "../../types/entrypoint" -import { parseAccount } from "../../utils/" -import { AccountOrClientNotFoundError } from "../../utils/signUserOperationHashWithECDSA" -import type { Middleware } from "../smartAccount/prepareUserOperationRequest" import { - type SendUserOperationParameters, + type GetSmartAccountParameter, + type SmartAccount, sendUserOperation -} from "../smartAccount/sendUserOperation" +} from "viem/account-abstraction" +import { getAction } from "viem/utils" +import { AccountNotFoundError } from "../../errors" +import { parseAccount } from "../../utils/" import { type ModuleType, parseModuleTypeId } from "./supportsModule" export type InstallModulesParameters< - TEntryPoint extends EntryPoint, - TSmartAccount extends SmartAccount | undefined -> = GetAccountParameter & - Middleware & { - modules: { - type: ModuleType - address: Address - context: Hex - }[] - maxFeePerGas?: bigint - maxPriorityFeePerGas?: bigint - nonce?: bigint - } + TSmartAccount extends SmartAccount | undefined +> = GetSmartAccountParameter & { + modules: { + type: ModuleType + address: Address + context: Hex + }[] + maxFeePerGas?: bigint + maxPriorityFeePerGas?: bigint + nonce?: bigint +} export async function installModules< - TEntryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends SmartAccount | undefined = - | SmartAccount - | undefined - | undefined + TSmartAccount extends SmartAccount | undefined >( - client: Client, - parameters: Prettify> + client: Client, + parameters: InstallModulesParameters ): Promise { const { account: account_ = client.account, maxFeePerGas, maxPriorityFeePerGas, nonce, - middleware, modules } = parameters if (!account_) { - throw new AccountOrClientNotFoundError({ + throw new AccountNotFoundError({ docsPath: "/docs/actions/wallet/sendTransaction" }) } - const account = parseAccount(account_) as SmartAccount - - const installModulesCallData = await account.encodeCallData( - await Promise.all( - modules.map(({ type, address, context }) => ({ - to: account.address, - value: BigInt(0), - data: encodeFunctionData({ - abi: [ - { - name: "installModule", - type: "function", - stateMutability: "nonpayable", - inputs: [ - { - type: "uint256", - name: "moduleTypeId" - }, - { - type: "address", - name: "module" - }, - { - type: "bytes", - name: "initData" - } - ], - outputs: [] - } - ], - functionName: "installModule", - args: [ - parseModuleTypeId(type), - getAddress(address), - context - ] - }) - })) - ) - ) - + const account = parseAccount(account_) as SmartAccount return getAction( client, - sendUserOperation, + sendUserOperation, "sendUserOperation" )({ - userOperation: { - sender: account.address, - maxFeePerGas: maxFeePerGas, - maxPriorityFeePerGas: maxPriorityFeePerGas, - callData: installModulesCallData, - nonce: nonce - }, - account: account, - middleware - } as SendUserOperationParameters) + calls: modules.map(({ type, address, context }) => ({ + to: account.address, + value: BigInt(0), + data: encodeFunctionData({ + abi: [ + { + name: "installModule", + type: "function", + stateMutability: "nonpayable", + inputs: [ + { + type: "uint256", + name: "moduleTypeId" + }, + { + type: "address", + name: "module" + }, + { + type: "bytes", + name: "initData" + } + ], + outputs: [] + } + ], + functionName: "installModule", + args: [parseModuleTypeId(type), getAddress(address), context] + }) + })), + maxFeePerGas, + maxPriorityFeePerGas, + nonce, + account: account + }) } diff --git a/packages/permissionless/actions/erc7579/isModuleInstalled.test.ts b/packages/permissionless/actions/erc7579/isModuleInstalled.test.ts index 9298b2d1..e2b2648f 100644 --- a/packages/permissionless/actions/erc7579/isModuleInstalled.test.ts +++ b/packages/permissionless/actions/erc7579/isModuleInstalled.test.ts @@ -1,23 +1,7 @@ -import { - http, - type Chain, - type Transport, - encodeAbiParameters, - encodePacked, - zeroAddress -} from "viem" -import { generatePrivateKey, privateKeyToAccount } from "viem/accounts" +import { encodeAbiParameters, encodePacked, zeroAddress } from "viem" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getCoreSmartAccounts, - getPimlicoPaymasterClient -} from "../../../permissionless-test/src/utils" -import type { SmartAccount } from "../../accounts" -import { createBundlerClient } from "../../clients/createBundlerClient" -import type { SmartAccountClient } from "../../clients/createSmartAccountClient" -import type { ENTRYPOINT_ADDRESS_V07_TYPE } from "../../types/entrypoint" -import { ENTRYPOINT_ADDRESS_V07 } from "../../utils" +import { getCoreSmartAccounts } from "../../../permissionless-test/src/utils" import { erc7579Actions } from "../erc7579" import { isModuleInstalled } from "./isModuleInstalled" @@ -27,34 +11,20 @@ describe.each(getCoreSmartAccounts())( testWithRpc.skipIf(!getErc7579SmartAccountClient)( "isModuleInstalled", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - if (!getErc7579SmartAccountClient) { throw new Error("getErc7579SmartAccountClient not defined") } - const privateKey = generatePrivateKey() - - const smartClientWithoutExtend: SmartAccountClient< - ENTRYPOINT_ADDRESS_V07_TYPE, - Transport, - Chain, - SmartAccount - > = await getErc7579SmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: privateKey, - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc + const smartClientWithoutExtend = + await getErc7579SmartAccountClient({ + entryPoint: { + version: "0.7" + }, + ...rpc }) - }) const smartClient = smartClientWithoutExtend.extend( - erc7579Actions({ - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) + erc7579Actions() ) const moduleData = encodePacked( @@ -80,21 +50,15 @@ describe.each(getCoreSmartAccounts())( : moduleData }) - const bundlerClientV07 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 + await smartClient.waitForUserOperationReceipt({ + hash: opHash, + timeout: 100000 }) - const receipt = - await bundlerClientV07.waitForUserOperationReceipt({ - hash: opHash, - timeout: 100000 - }) - const isModuleInstalledResult = await isModuleInstalled( - smartClient as any, + smartClient, { - account: smartClient.account as any, + account: smartClient.account, type: "executor", address: "0xc98B026383885F41d9a995f85FC480E9bb8bB891", context: "0x" @@ -102,13 +66,6 @@ describe.each(getCoreSmartAccounts())( ) expect(isModuleInstalledResult).toBe(true) - - const nextTransaction = await smartClient.sendTransaction({ - to: zeroAddress, - value: 0n - }) - - expect(nextTransaction).toBeTruthy() } ) } diff --git a/packages/permissionless/actions/erc7579/isModuleInstalled.ts b/packages/permissionless/actions/erc7579/isModuleInstalled.ts index 851ae77a..3b2d8b1b 100644 --- a/packages/permissionless/actions/erc7579/isModuleInstalled.ts +++ b/packages/permissionless/actions/erc7579/isModuleInstalled.ts @@ -1,6 +1,5 @@ import { type Address, - type CallParameters, type Chain, type Client, ContractFunctionExecutionError, @@ -10,44 +9,39 @@ import { encodeFunctionData, getAddress } from "viem" +import type { + GetSmartAccountParameter, + SmartAccount +} from "viem/account-abstraction" import { call, readContract } from "viem/actions" import { getAction } from "viem/utils" -import type { SmartAccount } from "../../accounts/types" -import type { GetAccountParameter } from "../../types/" -import type { EntryPoint } from "../../types/entrypoint" +import { AccountNotFoundError } from "../../errors" import { parseAccount } from "../../utils/" -import { AccountOrClientNotFoundError } from "../../utils/signUserOperationHashWithECDSA" import { type ModuleType, parseModuleTypeId } from "./supportsModule" export type IsModuleInstalledParameters< - TEntryPoint extends EntryPoint, - TSmartAccount extends SmartAccount | undefined -> = GetAccountParameter & { + TSmartAccount extends SmartAccount | undefined +> = GetSmartAccountParameter & { type: ModuleType address: Address context: Hex } export async function isModuleInstalled< - TEntryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends SmartAccount | undefined = - | SmartAccount - | undefined + TSmartAccount extends SmartAccount | undefined >( - client: Client, - parameters: IsModuleInstalledParameters + client: Client, + parameters: IsModuleInstalledParameters ): Promise { const { account: account_ = client.account, address, context } = parameters if (!account_) { - throw new AccountOrClientNotFoundError({ + throw new AccountNotFoundError({ docsPath: "/docs/actions/wallet/sendTransaction" }) } - const account = parseAccount(account_) as SmartAccount + const account = parseAccount(account_) as SmartAccount const publicClient = account.client @@ -95,8 +89,7 @@ export async function isModuleInstalled< }) } catch (error) { if (error instanceof ContractFunctionExecutionError) { - const factory = await account.getFactory() - const factoryData = await account.getFactoryData() + const { factory, factoryData } = await account.getFactoryArgs() const result = await getAction( publicClient, @@ -115,7 +108,7 @@ export async function isModuleInstalled< context ] }) - } as unknown as CallParameters) + }) if (!result || !result.data) { throw new Error("accountId result is empty") diff --git a/packages/permissionless/actions/erc7579/supportsExecutionMode.test.ts b/packages/permissionless/actions/erc7579/supportsExecutionMode.test.ts index e2c77bd5..041ea562 100644 --- a/packages/permissionless/actions/erc7579/supportsExecutionMode.test.ts +++ b/packages/permissionless/actions/erc7579/supportsExecutionMode.test.ts @@ -1,12 +1,7 @@ import { zeroAddress } from "viem" -import { generatePrivateKey } from "viem/accounts" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getCoreSmartAccounts, - getPimlicoPaymasterClient -} from "../../../permissionless-test/src/utils" -import { ENTRYPOINT_ADDRESS_V07 } from "../../utils" +import { getCoreSmartAccounts } from "../../../permissionless-test/src/utils" import { supportsExecutionMode } from "./supportsExecutionMode" describe.each(getCoreSmartAccounts())( @@ -15,26 +10,20 @@ describe.each(getCoreSmartAccounts())( testWithRpc.skipIf(!getErc7579SmartAccountClient)( "supportsExecutionMode", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - if (!getErc7579SmartAccountClient) { throw new Error("getErc7579SmartAccountClient not defined") } const smartClient = await getErc7579SmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) + entryPoint: { + version: "0.7" + }, + ...rpc }) const supportsExecutionModeBatchCallBeforeDeploy = - await supportsExecutionMode(smartClient as any, { - account: smartClient.account as any, + await supportsExecutionMode(smartClient, { + account: smartClient.account, type: "batchcall", revertOnError: false, selector: "0x0", @@ -44,15 +33,17 @@ describe.each(getCoreSmartAccounts())( expect(supportsExecutionModeBatchCallBeforeDeploy).toBe(true) // deploy account - await smartClient.sendTransaction({ - to: zeroAddress, - value: 0n, - data: "0x" + const userOpHash = await smartClient.sendUserOperation({ + calls: [{ to: zeroAddress, value: 0n, data: "0x" }] + }) + + await smartClient.waitForUserOperationReceipt({ + hash: userOpHash }) const supportsExecutionModeBatchCallBeforeDeployPostDeploy = - await supportsExecutionMode(smartClient as any, { - account: smartClient.account as any, + await supportsExecutionMode(smartClient, { + account: smartClient.account, type: "batchcall", revertOnError: false, selector: "0x0", @@ -71,41 +62,37 @@ describe.each(getCoreSmartAccounts())( testWithRpc.skipIf(!getErc7579SmartAccountClient)( "supportsExecutionMode", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - if (!getErc7579SmartAccountClient) { throw new Error("getErc7579SmartAccountClient not defined") } const smartClient = await getErc7579SmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) + entryPoint: { + version: "0.7" + }, + ...rpc }) const supportsExecutionModeBatchCallBeforeDeploy = - await supportsExecutionMode(smartClient as any, { - account: smartClient.account as any, + await supportsExecutionMode(smartClient, { + account: smartClient.account, type: "delegatecall" }) expect(supportsExecutionModeBatchCallBeforeDeploy).toBe(true) // deploy account - await smartClient.sendTransaction({ - to: zeroAddress, - value: 0n, - data: "0x" + const userOpHash = await smartClient.sendUserOperation({ + calls: [{ to: zeroAddress, value: 0n, data: "0x" }] + }) + + await smartClient.waitForUserOperationReceipt({ + hash: userOpHash }) const supportsExecutionModeBatchCallBeforeDeployPostDeploy = - await supportsExecutionMode(smartClient as any, { - account: smartClient.account as any, + await supportsExecutionMode(smartClient, { + account: smartClient.account, type: "batchcall", revertOnError: false, selector: "0x0", diff --git a/packages/permissionless/actions/erc7579/supportsExecutionMode.ts b/packages/permissionless/actions/erc7579/supportsExecutionMode.ts index 39820b41..145efa80 100644 --- a/packages/permissionless/actions/erc7579/supportsExecutionMode.ts +++ b/packages/permissionless/actions/erc7579/supportsExecutionMode.ts @@ -1,5 +1,4 @@ import { - type CallParameters, type Chain, type Client, ContractFunctionExecutionError, @@ -11,13 +10,14 @@ import { toBytes, toHex } from "viem" +import type { + GetSmartAccountParameter, + SmartAccount +} from "viem/account-abstraction" import { call, readContract } from "viem/actions" import { getAction } from "viem/utils" -import type { SmartAccount } from "../../accounts/types" -import type { GetAccountParameter, Prettify } from "../../types/" -import type { EntryPoint } from "../../types/entrypoint" +import { AccountNotFoundError } from "../../errors" import { parseAccount } from "../../utils/" -import { AccountOrClientNotFoundError } from "../../utils/signUserOperationHashWithECDSA" export type CallType = "call" | "delegatecall" | "batchcall" @@ -29,12 +29,9 @@ export type ExecutionMode = { } export type SupportsExecutionModeParameters< - TEntryPoint extends EntryPoint, - TSmartAccount extends SmartAccount | undefined = - | SmartAccount - | undefined, + TSmartAccount extends SmartAccount | undefined, callType extends CallType = CallType -> = GetAccountParameter & ExecutionMode +> = GetSmartAccountParameter & ExecutionMode function parseCallType(callType: CallType) { switch (callType) { @@ -66,15 +63,11 @@ export function encodeExecutionMode({ } export async function supportsExecutionMode< - TEntryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends SmartAccount | undefined = - | SmartAccount - | undefined + TSmartAccount extends SmartAccount | undefined, + callType extends CallType = CallType >( - client: Client, - args: Prettify> + client: Client, + args: SupportsExecutionModeParameters ): Promise { const { account: account_ = client.account, @@ -85,12 +78,12 @@ export async function supportsExecutionMode< } = args if (!account_) { - throw new AccountOrClientNotFoundError({ + throw new AccountNotFoundError({ docsPath: "/docs/actions/wallet/sendTransaction" }) } - const account = parseAccount(account_) as SmartAccount + const account = parseAccount(account_) as SmartAccount const publicClient = account.client @@ -133,8 +126,7 @@ export async function supportsExecutionMode< }) } catch (error) { if (error instanceof ContractFunctionExecutionError) { - const factory = await account.getFactory() - const factoryData = await account.getFactoryData() + const { factory, factoryData } = await account.getFactoryArgs() const result = await getAction( publicClient, @@ -149,7 +141,7 @@ export async function supportsExecutionMode< functionName: "supportsExecutionMode", args: [encodedMode] }) - } as unknown as CallParameters) + }) if (!result || !result.data) { throw new Error("accountId result is empty") diff --git a/packages/permissionless/actions/erc7579/supportsModule.test.ts b/packages/permissionless/actions/erc7579/supportsModule.test.ts index a246954d..7309ed1a 100644 --- a/packages/permissionless/actions/erc7579/supportsModule.test.ts +++ b/packages/permissionless/actions/erc7579/supportsModule.test.ts @@ -1,11 +1,6 @@ -import { generatePrivateKey } from "viem/accounts" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getCoreSmartAccounts, - getPimlicoPaymasterClient -} from "../../../permissionless-test/src/utils" -import { ENTRYPOINT_ADDRESS_V07 } from "../../utils" +import { getCoreSmartAccounts } from "../../../permissionless-test/src/utils" import { supportsModule } from "./supportsModule" describe.each(getCoreSmartAccounts())( @@ -14,27 +9,21 @@ describe.each(getCoreSmartAccounts())( testWithRpc.skipIf(!getErc7579SmartAccountClient)( "supportsModule", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - if (!getErc7579SmartAccountClient) { throw new Error("getErc7579SmartAccountClient not defined") } const smartClient = await getErc7579SmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) + entryPoint: { + version: "0.7" + }, + ...rpc }) const supportsValidationModule = await supportsModule( - smartClient as any, + smartClient, { - account: smartClient.account as any, + account: smartClient.account, type: "validator" } ) diff --git a/packages/permissionless/actions/erc7579/supportsModule.ts b/packages/permissionless/actions/erc7579/supportsModule.ts index 17f36799..56b55130 100644 --- a/packages/permissionless/actions/erc7579/supportsModule.ts +++ b/packages/permissionless/actions/erc7579/supportsModule.ts @@ -1,5 +1,4 @@ import { - type CallParameters, type Chain, type Client, ContractFunctionExecutionError, @@ -7,22 +6,20 @@ import { decodeFunctionResult, encodeFunctionData } from "viem" +import type { + GetSmartAccountParameter, + SmartAccount +} from "viem/account-abstraction" import { call, readContract } from "viem/actions" import { getAction } from "viem/utils" -import type { SmartAccount } from "../../accounts/types" -import type { GetAccountParameter, Prettify } from "../../types/" -import type { EntryPoint } from "../../types/entrypoint" +import { AccountNotFoundError } from "../../errors" import { parseAccount } from "../../utils/" -import { AccountOrClientNotFoundError } from "../../utils/signUserOperationHashWithECDSA" export type ModuleType = "validator" | "executor" | "fallback" | "hook" export type SupportsModuleParameters< - TEntryPoint extends EntryPoint, - TSmartAccount extends SmartAccount | undefined = - | SmartAccount - | undefined -> = GetAccountParameter & { + TSmartAccount extends SmartAccount | undefined +> = GetSmartAccountParameter & { type: ModuleType } @@ -42,25 +39,20 @@ export function parseModuleTypeId(type: ModuleType): bigint { } export async function supportsModule< - TEntryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends SmartAccount | undefined = - | SmartAccount - | undefined + TSmartAccount extends SmartAccount | undefined >( - client: Client, - args: Prettify> + client: Client, + args: SupportsModuleParameters ): Promise { const { account: account_ = client.account } = args if (!account_) { - throw new AccountOrClientNotFoundError({ + throw new AccountNotFoundError({ docsPath: "/docs/actions/wallet/sendTransaction" }) } - const account = parseAccount(account_) as SmartAccount + const account = parseAccount(account_) as SmartAccount const publicClient = account.client @@ -96,8 +88,7 @@ export async function supportsModule< }) } catch (error) { if (error instanceof ContractFunctionExecutionError) { - const factory = await account.getFactory() - const factoryData = await account.getFactoryData() + const { factory, factoryData } = await account.getFactoryArgs() const result = await getAction( publicClient, @@ -112,7 +103,7 @@ export async function supportsModule< functionName: "supportsModule", args: [parseModuleTypeId(args.type)] }) - } as unknown as CallParameters) + }) if (!result || !result.data) { throw new Error("accountId result is empty") diff --git a/packages/permissionless/actions/erc7579/uninstallModule.test.ts b/packages/permissionless/actions/erc7579/uninstallModule.test.ts index 085c85bc..1da2316f 100644 --- a/packages/permissionless/actions/erc7579/uninstallModule.test.ts +++ b/packages/permissionless/actions/erc7579/uninstallModule.test.ts @@ -1,25 +1,7 @@ -import { - http, - type Chain, - type Transport, - encodeAbiParameters, - encodePacked, - isHash, - zeroAddress -} from "viem" -import { generatePrivateKey, privateKeyToAccount } from "viem/accounts" +import { encodeAbiParameters, encodePacked, isHash, zeroAddress } from "viem" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getCoreSmartAccounts, - getPimlicoPaymasterClient, - getPublicClient -} from "../../../permissionless-test/src/utils" -import type { SmartAccount } from "../../accounts" -import { createBundlerClient } from "../../clients/createBundlerClient" -import type { SmartAccountClient } from "../../clients/createSmartAccountClient" -import type { ENTRYPOINT_ADDRESS_V07_TYPE } from "../../types" -import { ENTRYPOINT_ADDRESS_V07 } from "../../utils" +import { getCoreSmartAccounts } from "../../../permissionless-test/src/utils" import { erc7579Actions } from "../erc7579" import { uninstallModule } from "./uninstallModule" @@ -29,36 +11,20 @@ describe.each(getCoreSmartAccounts())( testWithRpc.skipIf(!getErc7579SmartAccountClient)( "uninstallModule", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - if (!getErc7579SmartAccountClient) { throw new Error("getErc7579SmartAccountClient not defined") } - const privateKey = generatePrivateKey() - - const eoaAccount = privateKeyToAccount(privateKey) - - const smartClientWithoutExtend: SmartAccountClient< - ENTRYPOINT_ADDRESS_V07_TYPE, - Transport, - Chain, - SmartAccount - > = await getErc7579SmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: privateKey, - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc + const smartClientWithoutExtend = + await getErc7579SmartAccountClient({ + entryPoint: { + version: "0.7" + }, + ...rpc }) - }) const smartClient = smartClientWithoutExtend.extend( - erc7579Actions({ - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) + erc7579Actions() ) const moduleData = encodePacked( @@ -66,13 +32,6 @@ describe.each(getCoreSmartAccounts())( [smartClient.account.address] ) - const bundlerClientV07 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) - - const publicClient = getPublicClient(anvilRpc) - const opHash = await smartClient.installModule({ type: "executor", address: "0xc98B026383885F41d9a995f85FC480E9bb8bB891", @@ -90,20 +49,15 @@ describe.each(getCoreSmartAccounts())( : moduleData }) - const userOperationReceipt = - await bundlerClientV07.waitForUserOperationReceipt({ - hash: opHash, - timeout: 100000 - }) - - await publicClient.waitForTransactionReceipt({ - hash: userOperationReceipt.receipt.transactionHash + await smartClient.waitForUserOperationReceipt({ + hash: opHash, + timeout: 100000 }) const uninstallModuleUserOpHash = await uninstallModule( - smartClient as any, + smartClient, { - account: smartClient.account as any, + account: smartClient.account, type: "executor", address: "0xc98B026383885F41d9a995f85FC480E9bb8bB891", context: name.startsWith("Kernel 7579") @@ -127,7 +81,7 @@ describe.each(getCoreSmartAccounts())( expect(isHash(uninstallModuleUserOpHash)).toBe(true) const userOperationReceiptUninstallModule = - await bundlerClientV07.waitForUserOperationReceipt({ + await smartClient.waitForUserOperationReceipt({ hash: uninstallModuleUserOpHash, timeout: 100000 }) @@ -140,7 +94,7 @@ describe.each(getCoreSmartAccounts())( ).toBeTruthy() const receiptUninstallModule = - await bundlerClientV07.getUserOperationReceipt({ + await smartClient.getUserOperationReceipt({ hash: uninstallModuleUserOpHash }) diff --git a/packages/permissionless/actions/erc7579/uninstallModule.ts b/packages/permissionless/actions/erc7579/uninstallModule.ts index 495f044f..36055b78 100644 --- a/packages/permissionless/actions/erc7579/uninstallModule.ts +++ b/packages/permissionless/actions/erc7579/uninstallModule.ts @@ -7,110 +7,94 @@ import { encodeFunctionData, getAddress } from "viem" -import { getAction } from "viem/utils" -import type { SmartAccount } from "../../accounts/types" -import type { GetAccountParameter, Prettify } from "../../types/" -import type { EntryPoint } from "../../types/entrypoint" -import { parseAccount } from "../../utils/" -import { AccountOrClientNotFoundError } from "../../utils/signUserOperationHashWithECDSA" -import type { Middleware } from "../smartAccount/prepareUserOperationRequest" import { - type SendUserOperationParameters, + type GetSmartAccountParameter, + type SmartAccount, sendUserOperation -} from "../smartAccount/sendUserOperation" +} from "viem/account-abstraction" +import { getAction } from "viem/utils" +import { AccountNotFoundError } from "../../errors" +import { parseAccount } from "../../utils/" import { type ModuleType, parseModuleTypeId } from "./supportsModule" export type UninstallModuleParameters< - TEntryPoint extends EntryPoint, - TSmartAccount extends SmartAccount | undefined = - | SmartAccount - | undefined -> = GetAccountParameter & { + TSmartAccount extends SmartAccount | undefined +> = GetSmartAccountParameter & { type: ModuleType address: Address context: Hex maxFeePerGas?: bigint maxPriorityFeePerGas?: bigint nonce?: bigint -} & Middleware +} export async function uninstallModule< - TEntryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends SmartAccount | undefined = - | SmartAccount - | undefined + TSmartAccount extends SmartAccount | undefined >( - client: Client, - parameters: Prettify> + client: Client, + parameters: UninstallModuleParameters ): Promise { const { account: account_ = client.account, maxFeePerGas, maxPriorityFeePerGas, nonce, - middleware, address, context } = parameters if (!account_) { - throw new AccountOrClientNotFoundError({ + throw new AccountNotFoundError({ docsPath: "/docs/actions/wallet/sendTransaction" }) } - const account = parseAccount(account_) as SmartAccount - - const uninstallModuleCallData = await account.encodeCallData({ - to: account.address, - value: BigInt(0), - data: encodeFunctionData({ - abi: [ - { - name: "uninstallModule", - type: "function", - stateMutability: "nonpayable", - inputs: [ - { - type: "uint256", - name: "moduleTypeId" - }, - { - type: "address", - name: "module" - }, - { - type: "bytes", - name: "deInitData" - } - ], - outputs: [] - } - ], - functionName: "uninstallModule", - args: [ - parseModuleTypeId(parameters.type), - getAddress(address), - context - ] - }) - }) + const account = parseAccount(account_) as SmartAccount return getAction( client, - sendUserOperation, + sendUserOperation, "sendUserOperation" )({ - userOperation: { - sender: account.address, - maxFeePerGas: maxFeePerGas, - maxPriorityFeePerGas: maxPriorityFeePerGas, - callData: uninstallModuleCallData, - nonce: nonce - }, - account: account, - middleware - } as SendUserOperationParameters) + calls: [ + { + to: account.address, + value: BigInt(0), + data: encodeFunctionData({ + abi: [ + { + name: "uninstallModule", + type: "function", + stateMutability: "nonpayable", + inputs: [ + { + type: "uint256", + name: "moduleTypeId" + }, + { + type: "address", + name: "module" + }, + { + type: "bytes", + name: "deInitData" + } + ], + outputs: [] + } + ], + functionName: "uninstallModule", + args: [ + parseModuleTypeId(parameters.type), + getAddress(address), + context + ] + }) + } + ], + maxFeePerGas, + maxPriorityFeePerGas, + nonce, + account: account + }) } diff --git a/packages/permissionless/actions/erc7579/uninstallModules.test.ts b/packages/permissionless/actions/erc7579/uninstallModules.test.ts index 290be0b4..9da4f4e3 100644 --- a/packages/permissionless/actions/erc7579/uninstallModules.test.ts +++ b/packages/permissionless/actions/erc7579/uninstallModules.test.ts @@ -1,25 +1,7 @@ -import { - http, - type Chain, - type Transport, - encodeAbiParameters, - encodePacked, - isHash, - zeroAddress -} from "viem" -import { generatePrivateKey, privateKeyToAccount } from "viem/accounts" +import { encodeAbiParameters, encodePacked, isHash, zeroAddress } from "viem" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { - getCoreSmartAccounts, - getPimlicoPaymasterClient, - getPublicClient -} from "../../../permissionless-test/src/utils" -import type { SmartAccount } from "../../accounts" -import { createBundlerClient } from "../../clients/createBundlerClient" -import type { SmartAccountClient } from "../../clients/createSmartAccountClient" -import type { ENTRYPOINT_ADDRESS_V07_TYPE } from "../../types" -import { ENTRYPOINT_ADDRESS_V07 } from "../../utils" +import { getCoreSmartAccounts } from "../../../permissionless-test/src/utils" import { erc7579Actions } from "../erc7579" import { uninstallModules } from "./uninstallModules" @@ -29,36 +11,20 @@ describe.each(getCoreSmartAccounts())( testWithRpc.skipIf(!getErc7579SmartAccountClient)( "uninstallModules", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc - if (!getErc7579SmartAccountClient) { throw new Error("getErc7579SmartAccountClient not defined") } - const privateKey = generatePrivateKey() - - const eoaAccount = privateKeyToAccount(privateKey) - - const smartClientWithoutExtend: SmartAccountClient< - ENTRYPOINT_ADDRESS_V07_TYPE, - Transport, - Chain, - SmartAccount - > = await getErc7579SmartAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: privateKey, - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc + const smartClientWithoutExtend = + await getErc7579SmartAccountClient({ + entryPoint: { + version: "0.7" + }, + ...rpc }) - }) const smartClient = smartClientWithoutExtend.extend( - erc7579Actions({ - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) + erc7579Actions() ) const moduleData = encodePacked( @@ -66,13 +32,6 @@ describe.each(getCoreSmartAccounts())( [smartClient.account.address] ) - const bundlerClientV07 = createBundlerClient({ - transport: http(altoRpc), - entryPoint: ENTRYPOINT_ADDRESS_V07 - }) - - const publicClient = getPublicClient(anvilRpc) - const opHash = await smartClient.installModule({ type: "executor", address: "0xc98B026383885F41d9a995f85FC480E9bb8bB891", @@ -90,20 +49,15 @@ describe.each(getCoreSmartAccounts())( : moduleData }) - const userOperationReceipt = - await bundlerClientV07.waitForUserOperationReceipt({ - hash: opHash, - timeout: 100000 - }) - - await publicClient.waitForTransactionReceipt({ - hash: userOperationReceipt.receipt.transactionHash + await smartClient.waitForUserOperationReceipt({ + hash: opHash, + timeout: 100000 }) const uninstallModulesUserOpHash = await uninstallModules( - smartClient as any, + smartClient, { - account: smartClient.account as any, + account: smartClient.account, modules: [ { type: "executor", @@ -135,7 +89,7 @@ describe.each(getCoreSmartAccounts())( expect(isHash(uninstallModulesUserOpHash)).toBe(true) const userOperationReceiptUninstallModules = - await bundlerClientV07.waitForUserOperationReceipt({ + await smartClient.waitForUserOperationReceipt({ hash: uninstallModulesUserOpHash, timeout: 100000 }) @@ -149,7 +103,7 @@ describe.each(getCoreSmartAccounts())( ).toBeTruthy() const receiptUninstallModules = - await bundlerClientV07.getUserOperationReceipt({ + await smartClient.getUserOperationReceipt({ hash: uninstallModulesUserOpHash }) diff --git a/packages/permissionless/actions/erc7579/uninstallModules.ts b/packages/permissionless/actions/erc7579/uninstallModules.ts index 43b47dc5..9eb42550 100644 --- a/packages/permissionless/actions/erc7579/uninstallModules.ts +++ b/packages/permissionless/actions/erc7579/uninstallModules.ts @@ -7,25 +7,19 @@ import { encodeFunctionData, getAddress } from "viem" -import { getAction } from "viem/utils" -import type { SmartAccount } from "../../accounts/types" -import type { GetAccountParameter, Prettify } from "../../types/" -import type { EntryPoint } from "../../types/entrypoint" -import { parseAccount } from "../../utils/" -import { AccountOrClientNotFoundError } from "../../utils/signUserOperationHashWithECDSA" -import type { Middleware } from "../smartAccount/prepareUserOperationRequest" import { - type SendUserOperationParameters, + type GetSmartAccountParameter, + type SmartAccount, sendUserOperation -} from "../smartAccount/sendUserOperation" +} from "viem/account-abstraction" +import { getAction } from "viem/utils" +import { AccountNotFoundError } from "../../errors" +import { parseAccount } from "../../utils/" import { type ModuleType, parseModuleTypeId } from "./supportsModule" export type UninstallModulesParameters< - TEntryPoint extends EntryPoint, - TSmartAccount extends SmartAccount | undefined = - | SmartAccount - | undefined -> = GetAccountParameter & { + TSmartAccount extends SmartAccount | undefined +> = GetSmartAccountParameter & { modules: [ { type: ModuleType @@ -36,93 +30,68 @@ export type UninstallModulesParameters< maxFeePerGas?: bigint maxPriorityFeePerGas?: bigint nonce?: bigint -} & Middleware +} export async function uninstallModules< - TEntryPoint extends EntryPoint, - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TSmartAccount extends SmartAccount | undefined = - | SmartAccount - | undefined + TSmartAccount extends SmartAccount | undefined >( - client: Client, - parameters: Prettify> + client: Client, + parameters: UninstallModulesParameters ): Promise { const { account: account_ = client.account, maxFeePerGas, maxPriorityFeePerGas, nonce, - middleware, modules } = parameters if (!account_) { - throw new AccountOrClientNotFoundError({ + throw new AccountNotFoundError({ docsPath: "/docs/actions/wallet/sendTransaction" }) } - const account = parseAccount(account_) as SmartAccount< - TEntryPoint, - string, - TTransport, - TChain - > - - const uninstallModulesCallData = await account.encodeCallData( - await Promise.all( - modules.map(({ type, address, context }) => ({ - to: account.address, - value: BigInt(0), - data: encodeFunctionData({ - abi: [ - { - name: "uninstallModule", - type: "function", - stateMutability: "nonpayable", - inputs: [ - { - type: "uint256", - name: "moduleTypeId" - }, - { - type: "address", - name: "module" - }, - { - type: "bytes", - name: "deInitData" - } - ], - outputs: [] - } - ], - functionName: "uninstallModule", - args: [ - parseModuleTypeId(type), - getAddress(address), - context - ] - }) - })) - ) - ) + const account = parseAccount(account_) as SmartAccount return getAction( client, - sendUserOperation, + sendUserOperation, "sendUserOperation" )({ - userOperation: { - sender: account.address, - maxFeePerGas: maxFeePerGas, - maxPriorityFeePerGas: maxPriorityFeePerGas, - callData: uninstallModulesCallData, - nonce: nonce - }, - account: account, - middleware - } as SendUserOperationParameters) + calls: modules.map(({ type, address, context }) => ({ + to: account.address, + value: BigInt(0), + data: encodeFunctionData({ + abi: [ + { + name: "uninstallModule", + type: "function", + stateMutability: "nonpayable", + inputs: [ + { + type: "uint256", + name: "moduleTypeId" + }, + { + type: "address", + name: "module" + }, + { + type: "bytes", + name: "deInitData" + } + ], + outputs: [] + } + ], + functionName: "uninstallModule", + args: [parseModuleTypeId(type), getAddress(address), context] + }) + })), + maxFeePerGas, + maxPriorityFeePerGas, + nonce, + account + }) } diff --git a/packages/permissionless/actions/index.test.ts b/packages/permissionless/actions/index.test.ts deleted file mode 100644 index 4e992fd3..00000000 --- a/packages/permissionless/actions/index.test.ts +++ /dev/null @@ -1,329 +0,0 @@ -import type { - Account, - Address, - Chain, - Client, - Hash, - Hex, - Log, - Transport -} from "viem" -import type { PartialBy } from "viem/chains" -import { describe, expectTypeOf, test } from "vitest" -import type { BundlerRpcSchema } from "../types/bundler" -import type { - ENTRYPOINT_ADDRESS_V06_TYPE, - ENTRYPOINT_ADDRESS_V07_TYPE, - EntryPoint -} from "../types/entrypoint" -import type { TStatus, UserOperation } from "../types/userOperation" -import { - chainId, - estimateUserOperationGas, - getAccountNonce, - getSenderAddress, - getUserOperationByHash, - getUserOperationReceipt, - sendUserOperation, - supportedEntryPoints, - waitForUserOperationReceipt -} from "./index" - -describe("index", () => { - test("sendUserOperation", () => { - expectTypeOf(sendUserOperation).toBeFunction() - expectTypeOf(sendUserOperation) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - expectTypeOf(sendUserOperation).parameter(1).toMatchTypeOf<{ - userOperation: UserOperation<"v0.6" | "v0.7"> - entryPoint: EntryPoint - }>() - expectTypeOf(sendUserOperation).returns.toMatchTypeOf>() - }) - - test("estimateUserOperationGas", () => { - expectTypeOf(estimateUserOperationGas).toBeFunction() - expectTypeOf(estimateUserOperationGas) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - - expectTypeOf(estimateUserOperationGas).parameter(1).toMatchTypeOf<{ - userOperation: - | PartialBy< - UserOperation<"v0.6">, - | "callGasLimit" - | "preVerificationGas" - | "verificationGasLimit" - > - | PartialBy< - UserOperation<"v0.7">, - | "callGasLimit" - | "preVerificationGas" - | "verificationGasLimit" - | "paymasterVerificationGasLimit" - | "paymasterPostOpGasLimit" - > - entryPoint: EntryPoint - }>() - expectTypeOf(estimateUserOperationGas).parameter(2).toMatchTypeOf< - | { - [x: string]: { - balance?: bigint | undefined - nonce?: bigint | number | undefined - code?: Hex | undefined - state?: { - [x: Hex]: Hex - } - stateDiff?: { - [x: Hex]: Hex - } - } - } - | undefined - >() - expectTypeOf( - estimateUserOperationGas - ).returns.toMatchTypeOf< - Promise<{ - preVerificationGas: bigint - verificationGasLimit: bigint - callGasLimit: bigint - }> - >() - expectTypeOf( - estimateUserOperationGas - ).returns.toMatchTypeOf< - Promise<{ - preVerificationGas: bigint - verificationGasLimit: bigint - callGasLimit: bigint - paymasterVerificationGasLimit?: bigint - paymasterPostOpGasLimit?: bigint - }> - >() - }) - - test("supportedEntryPoints", () => { - expectTypeOf(supportedEntryPoints) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - - expectTypeOf(supportedEntryPoints).returns.toMatchTypeOf< - Promise - >() - }) - - test("chainId", () => { - expectTypeOf(chainId) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - - expectTypeOf(chainId).returns.toMatchTypeOf>() - }) - - test("getUserOperationByHash", () => { - expectTypeOf(getUserOperationByHash) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - expectTypeOf(getUserOperationByHash).parameter(1).toMatchTypeOf<{ - hash: Hash - }>() - expectTypeOf( - getUserOperationByHash - ).returns.toMatchTypeOf< - Promise<{ - userOperation: UserOperation<"v0.6"> - entryPoint: Address - transactionHash: Hash - blockHash: Hash - blockNumber: bigint - } | null> - >() - expectTypeOf( - getUserOperationByHash - ).returns.toMatchTypeOf< - Promise<{ - userOperation: UserOperation<"v0.7"> - entryPoint: Address - transactionHash: Hash - blockHash: Hash - blockNumber: bigint - } | null> - >() - }) - - test("getUserOperationReceipt", () => { - expectTypeOf(getUserOperationReceipt) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - expectTypeOf(getUserOperationReceipt).parameter(1).toMatchTypeOf<{ - hash: Hash - }>() - expectTypeOf(getUserOperationReceipt).returns.toMatchTypeOf< - Promise<{ - userOpHash: Hash - entryPoint: Address - sender: Address - nonce: bigint - paymaster?: Address - actualGasUsed: bigint - actualGasCost: bigint - success: boolean - reason?: string - receipt: { - transactionHash: Hex - transactionIndex: bigint - blockHash: Hash - blockNumber: bigint - from: Address - to: Address | null - cumulativeGasUsed: bigint - status: TStatus - gasUsed: bigint - contractAddress: Address | null - logsBloom: Hex - effectiveGasPrice: bigint - } - logs: Log[] - } | null> - >() - }) - - test("getSenderAddress", () => { - expectTypeOf(getSenderAddress) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - expectTypeOf(getSenderAddress) - .parameter(1) - .toMatchTypeOf<{ - initCode: Hex - entryPoint: ENTRYPOINT_ADDRESS_V06_TYPE - factory?: never - factoryData?: never - }>() - expectTypeOf(getSenderAddress) - .parameter(1) - .toMatchTypeOf<{ - entryPoint: ENTRYPOINT_ADDRESS_V07_TYPE - factory: Address - factoryData: Hex - initCode?: never - }>() - expectTypeOf(getSenderAddress).returns.toMatchTypeOf>() - }) - - test("getAccountNonce", () => { - expectTypeOf(getAccountNonce) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - expectTypeOf(getAccountNonce).parameter(1).toMatchTypeOf<{ - sender: Address - entryPoint: EntryPoint - key?: bigint - }>() - expectTypeOf(getAccountNonce).returns.toMatchTypeOf>() - }) - - test("waitForUserOperationReceipt", () => { - expectTypeOf(waitForUserOperationReceipt) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - BundlerRpcSchema - > - >() - expectTypeOf(waitForUserOperationReceipt).parameter(1).toMatchTypeOf<{ - hash: Hash - pollingInterval?: number - timeout?: number - }>() - expectTypeOf(waitForUserOperationReceipt).returns.toMatchTypeOf< - Promise<{ - userOpHash: Hash - entryPoint: Address - sender: Address - nonce: bigint - paymaster?: Address - actualGasUsed: bigint - actualGasCost: bigint - success: boolean - reason?: string - receipt: { - transactionHash: Hex - transactionIndex: bigint - blockHash: Hash - blockNumber: bigint - from: Address - to: Address | null - cumulativeGasUsed: bigint - status: TStatus - gasUsed: bigint - contractAddress: Address | null - logsBloom: Hex - effectiveGasPrice: bigint - } - logs: Log[] - }> - >() - }) -}) diff --git a/packages/permissionless/actions/pimlico.test.ts b/packages/permissionless/actions/pimlico.test.ts deleted file mode 100644 index f6d088cf..00000000 --- a/packages/permissionless/actions/pimlico.test.ts +++ /dev/null @@ -1,222 +0,0 @@ -import type { - Account, - Address, - Chain, - Client, - Hash, - Hex, - Transport -} from "viem" -import { describe, expectTypeOf, test } from "vitest" -import type { EntryPoint } from "../types/entrypoint" -import type { PimlicoBundlerRpcSchema } from "../types/pimlico" -import type { UserOperation } from "../types/userOperation" -import { - type PimlicoBundlerActions, - getUserOperationGasPrice, - getUserOperationStatus, - pimlicoBundlerActions, - sendCompressedUserOperation, - sponsorUserOperation, - validateSponsorshipPolicies -} from "./pimlico" - -describe("pimlico", () => { - test("getUserOperationGasPrice", () => { - expectTypeOf(getUserOperationGasPrice).toBeFunction() - expectTypeOf(getUserOperationGasPrice) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - PimlicoBundlerRpcSchema - > - >() - expectTypeOf(getUserOperationGasPrice).returns.toMatchTypeOf | null>() - }) - - test("getUserOperationStatus", () => { - expectTypeOf(getUserOperationStatus).toBeFunction() - expectTypeOf(getUserOperationStatus) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - PimlicoBundlerRpcSchema - > - >() - expectTypeOf(getUserOperationStatus).parameter(1).toMatchTypeOf<{ - hash: string - }>() - expectTypeOf(getUserOperationStatus).returns.toMatchTypeOf< - Promise<{ - status: - | "not_found" - | "not_submitted" - | "submitted" - | "rejected" - | "reverted" - | "included" - | "failed" - transactionHash: Hash | null - }> - >() - }) - - test("sendCompressedUserOperation", () => { - expectTypeOf(sendCompressedUserOperation).toBeFunction() - expectTypeOf(sendCompressedUserOperation) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - PimlicoBundlerRpcSchema - > - >() - expectTypeOf(sendCompressedUserOperation).parameter(1).toMatchTypeOf<{ - compressedUserOperation: Hex - inflatorAddress: Address - entryPoint: Address - }>() - expectTypeOf(sendCompressedUserOperation).returns.toMatchTypeOf< - Promise - >() - }) - - test("sponsorUserOperation", () => { - expectTypeOf(sponsorUserOperation).toBeFunction() - expectTypeOf(sponsorUserOperation) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - PimlicoBundlerRpcSchema - > - >() - expectTypeOf(sponsorUserOperation).parameter(1).toMatchTypeOf<{ - userOperation: UserOperation<"v0.6" | "v0.7"> - entryPoint: EntryPoint - sponsorshipPolicyId?: string - }>() - expectTypeOf(sponsorUserOperation).returns.toMatchTypeOf< - Promise< - | { - callGasLimit: bigint - verificationGasLimit: bigint - preVerificationGas: bigint - paymasterAndData: Hex - } - | { - callGasLimit: bigint - verificationGasLimit: bigint - preVerificationGas: bigint - paymaster: Address - paymasterVerificationGasLimit: bigint - paymasterPostOpGasLimit: bigint - paymasterData: Hex - } - > - >() - }) - - test("sponsorUserOperation", () => { - expectTypeOf(sponsorUserOperation).toBeFunction() - expectTypeOf(sponsorUserOperation) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - PimlicoBundlerRpcSchema - > - >() - expectTypeOf(sponsorUserOperation).parameter(1).toMatchTypeOf<{ - userOperation: UserOperation<"v0.6" | "v0.7"> - entryPoint: EntryPoint - sponsorshipPolicyId?: string - }>() - expectTypeOf(sponsorUserOperation).returns.toMatchTypeOf< - Promise< - | { - callGasLimit: bigint - verificationGasLimit: bigint - preVerificationGas: bigint - paymasterAndData: Hex - } - | { - callGasLimit: bigint - verificationGasLimit: bigint - preVerificationGas: bigint - paymaster: Address - paymasterVerificationGasLimit: bigint - paymasterPostOpGasLimit: bigint - paymasterData: Hex - } - > - >() - }) - - test("validateSponsorshipPolicies", () => { - expectTypeOf(validateSponsorshipPolicies).toBeFunction() - expectTypeOf(validateSponsorshipPolicies) - .parameter(0) - .toMatchTypeOf< - Client< - Transport, - Chain | undefined, - Account | undefined, - PimlicoBundlerRpcSchema - > - >() - expectTypeOf(validateSponsorshipPolicies).parameter(1).toMatchTypeOf<{ - userOperation: UserOperation<"v0.6" | "v0.7"> - entryPoint: EntryPoint - sponsorshipPolicyIds: string[] - }>() - expectTypeOf(validateSponsorshipPolicies).returns.toMatchTypeOf< - Promise< - { - sponsorshipPolicyId: string - data: { - name: string | null - author: string | null - icon: string | null - description: string | null - } - }[] - > - >() - }) - - test("pimlicoBundlerActions", () => { - expectTypeOf(pimlicoBundlerActions).toBeFunction() - expectTypeOf(pimlicoBundlerActions) - .parameter(0) - .toMatchTypeOf() - expectTypeOf(pimlicoBundlerActions).returns.toMatchTypeOf< - (client: Client) => PimlicoBundlerActions - >() - }) -}) diff --git a/packages/permissionless/actions/pimlico.ts b/packages/permissionless/actions/pimlico.ts index 2ba5d170..c067c254 100644 --- a/packages/permissionless/actions/pimlico.ts +++ b/packages/permissionless/actions/pimlico.ts @@ -17,14 +17,8 @@ import { sponsorUserOperation } from "./pimlico/sponsorUserOperation" -import type { - PimlicoActions, - PimlicoPaymasterClientActions -} from "../clients/decorators/pimlico" -import { - pimlicoActions, - pimlicoPaymasterActions -} from "../clients/decorators/pimlico" +import type { PimlicoActions } from "../clients/decorators/pimlico" +import { pimlicoActions } from "../clients/decorators/pimlico" import { type ValidateSponsorshipPolicies, @@ -36,8 +30,7 @@ export type { GetUserOperationGasPriceReturnType, GetUserOperationStatusParameters, GetUserOperationStatusReturnType, - PimlicoActions as PimlicoBundlerActions, - PimlicoPaymasterClientActions, + PimlicoActions, PimlicoSponsorUserOperationParameters, SendCompressedUserOperationParameters, SponsorUserOperationReturnType, @@ -48,8 +41,7 @@ export type { export { getUserOperationGasPrice, getUserOperationStatus, - pimlicoActions as pimlicoBundlerActions, - pimlicoPaymasterActions, + pimlicoActions, sendCompressedUserOperation, sponsorUserOperation, validateSponsorshipPolicies diff --git a/packages/permissionless/actions/pimlico/getUserOperationGasPrice.test.ts b/packages/permissionless/actions/pimlico/getUserOperationGasPrice.test.ts index 492c353b..f365f76c 100644 --- a/packages/permissionless/actions/pimlico/getUserOperationGasPrice.test.ts +++ b/packages/permissionless/actions/pimlico/getUserOperationGasPrice.test.ts @@ -1,13 +1,13 @@ +import { entryPoint06Address } from "viem/account-abstraction" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { getPimlicoBundlerClient } from "../../../permissionless-test/src/utils" -import { ENTRYPOINT_ADDRESS_V06 } from "../../utils" +import { getPimlicoClient } from "../../../permissionless-test/src/utils" import { getUserOperationGasPrice } from "./getUserOperationGasPrice" describe("getUserOperationGasPrice", () => { testWithRpc("getUserOperationGasPrice", async ({ rpc }) => { - const pimlicoBundlerClient = getPimlicoBundlerClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, + const pimlicoBundlerClient = getPimlicoClient({ + entryPoint: entryPoint06Address, altoRpc: rpc.altoRpc }) diff --git a/packages/permissionless/actions/pimlico/getUserOperationStatus.test.ts b/packages/permissionless/actions/pimlico/getUserOperationStatus.test.ts index 60c4521d..4b0f3986 100644 --- a/packages/permissionless/actions/pimlico/getUserOperationStatus.test.ts +++ b/packages/permissionless/actions/pimlico/getUserOperationStatus.test.ts @@ -1,58 +1,47 @@ import { isHash, zeroAddress } from "viem" -import { generatePrivateKey } from "viem/accounts" +import { entryPoint06Address } from "viem/account-abstraction" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" import { - getPimlicoBundlerClient, - getPimlicoPaymasterClient, + getBundlerClient, + getPimlicoClient, getSimpleAccountClient } from "../../../permissionless-test/src/utils" -import { bundlerActions } from "../../clients/decorators/bundler" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" import { getUserOperationStatus } from "./getUserOperationStatus" describe("getUserOperationStatus", () => { testWithRpc("getUserOperationStatus_V06", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc + const { altoRpc } = rpc - const bundlerClientV06 = getPimlicoBundlerClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, + const bundlerClient = getPimlicoClient({ + entryPoint: entryPoint06Address, altoRpc: altoRpc - }).extend(bundlerActions(ENTRYPOINT_ADDRESS_V06)) - - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) }) - const userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) + const simpleAccountClient = getBundlerClient({ + account: await getSimpleAccountClient({ + ...rpc, + entryPoint: { + version: "0.6" } - }) - - userOperation.signature = - await simpleAccountClient.account.signUserOperation(userOperation) + }), + ...rpc + }) - const opHash = await bundlerClientV06.sendUserOperation({ - userOperation + const opHash = await simpleAccountClient.sendUserOperation({ + calls: [ + { + to: zeroAddress, + data: "0x", + value: 0n + } + ] }) expect(isHash(opHash)).toBe(true) const userOperationReceipt = - await bundlerClientV06.waitForUserOperationReceipt({ + await bundlerClient.waitForUserOperationReceipt({ hash: opHash, timeout: 100000 }) @@ -60,7 +49,7 @@ describe("getUserOperationStatus", () => { expect(userOperationReceipt?.userOpHash).toBe(opHash) expect(userOperationReceipt?.receipt.transactionHash).toBeTruthy() - const receipt = await bundlerClientV06.getUserOperationReceipt({ + const receipt = await bundlerClient.getUserOperationReceipt({ hash: opHash }) @@ -68,7 +57,7 @@ describe("getUserOperationStatus", () => { userOperationReceipt?.receipt.transactionHash ) const userOperationStatus = await getUserOperationStatus( - bundlerClientV06, + bundlerClient, { hash: opHash } @@ -82,46 +71,37 @@ describe("getUserOperationStatus", () => { }) testWithRpc("getUserOperationStatus_V07", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc + const { altoRpc } = rpc - const bundlerClientV07 = getPimlicoBundlerClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, + const bundlerClient = getPimlicoClient({ + entryPoint: entryPoint06Address, altoRpc: altoRpc - }).extend(bundlerActions(ENTRYPOINT_ADDRESS_V07)) - - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) }) - const userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) + const simpleAccountClient = getBundlerClient({ + account: await getSimpleAccountClient({ + ...rpc, + entryPoint: { + version: "0.7" } - }) - - userOperation.signature = - await simpleAccountClient.account.signUserOperation(userOperation) + }), + ...rpc + }) - const opHash = await bundlerClientV07.sendUserOperation({ - userOperation + const opHash = await simpleAccountClient.sendUserOperation({ + calls: [ + { + to: zeroAddress, + data: "0x", + value: 0n + } + ] }) expect(isHash(opHash)).toBe(true) const userOperationReceipt = - await bundlerClientV07.waitForUserOperationReceipt({ + await bundlerClient.waitForUserOperationReceipt({ hash: opHash, timeout: 100000 }) @@ -129,7 +109,7 @@ describe("getUserOperationStatus", () => { expect(userOperationReceipt?.userOpHash).toBe(opHash) expect(userOperationReceipt?.receipt.transactionHash).toBeTruthy() - const receipt = await bundlerClientV07.getUserOperationReceipt({ + const receipt = await bundlerClient.getUserOperationReceipt({ hash: opHash }) @@ -137,7 +117,7 @@ describe("getUserOperationStatus", () => { userOperationReceipt?.receipt.transactionHash ) const userOperationStatus = await getUserOperationStatus( - bundlerClientV07, + bundlerClient, { hash: opHash } diff --git a/packages/permissionless/actions/pimlico/sponsorUserOperation.test.ts b/packages/permissionless/actions/pimlico/sponsorUserOperation.test.ts index 1a1d4d2d..6c307764 100644 --- a/packages/permissionless/actions/pimlico/sponsorUserOperation.test.ts +++ b/packages/permissionless/actions/pimlico/sponsorUserOperation.test.ts @@ -1,56 +1,46 @@ -import { http, isHash, zeroAddress } from "viem" -import { generatePrivateKey } from "viem/accounts" +import { isHash, zeroAddress } from "viem" +import { entryPoint06Address } from "viem/account-abstraction" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" import { - getPimlicoPaymasterClient, + getBundlerClient, + getPimlicoClient, getSimpleAccountClient } from "../../../permissionless-test/src/utils" -import { createBundlerClient } from "../../clients/createBundlerClient" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" describe("sponsorUserOperation", () => { testWithRpc("sponsorUserOperation_V06", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc + const { altoRpc } = rpc - const bundlerClientV06 = createBundlerClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - transport: http(altoRpc) + const bundlerClient = getPimlicoClient({ + entryPoint: entryPoint06Address, + altoRpc: altoRpc }) - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) + const simpleAccountClient = getBundlerClient({ + account: await getSimpleAccountClient({ + ...rpc, + entryPoint: { + version: "0.6" + } + }), + ...rpc }) - const userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) + const opHash = await simpleAccountClient.sendUserOperation({ + calls: [ + { + to: zeroAddress, + data: "0x", + value: 0n } - }) - - userOperation.signature = - await simpleAccountClient.account.signUserOperation(userOperation) - - const opHash = await bundlerClientV06.sendUserOperation({ - userOperation + ] }) expect(isHash(opHash)).toBe(true) const userOperationReceipt = - await bundlerClientV06.waitForUserOperationReceipt({ + await bundlerClient.waitForUserOperationReceipt({ hash: opHash, timeout: 100000 }) @@ -58,7 +48,7 @@ describe("sponsorUserOperation", () => { expect(userOperationReceipt?.userOpHash).toBe(opHash) expect(userOperationReceipt?.receipt.transactionHash).toBeTruthy() - const receipt = await bundlerClientV06.getUserOperationReceipt({ + const receipt = await bundlerClient.getUserOperationReceipt({ hash: opHash }) @@ -68,46 +58,37 @@ describe("sponsorUserOperation", () => { }) testWithRpc("sponsorUserOperation_V07", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc + const { altoRpc } = rpc - const bundlerClientV07 = createBundlerClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - transport: http(altoRpc) + const bundlerClient = getPimlicoClient({ + entryPoint: entryPoint06Address, + altoRpc: altoRpc }) - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }) + const simpleAccountClient = getBundlerClient({ + account: await getSimpleAccountClient({ + ...rpc, + entryPoint: { + version: "0.7" + } + }), + ...rpc }) - const userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) + const opHash = await simpleAccountClient.sendUserOperation({ + calls: [ + { + to: zeroAddress, + data: "0x", + value: 0n } - }) - - userOperation.signature = - await simpleAccountClient.account.signUserOperation(userOperation) - - const opHash = await bundlerClientV07.sendUserOperation({ - userOperation + ] }) expect(isHash(opHash)).toBe(true) const userOperationReceipt = - await bundlerClientV07.waitForUserOperationReceipt({ + await bundlerClient.waitForUserOperationReceipt({ hash: opHash, timeout: 100000 }) @@ -115,7 +96,7 @@ describe("sponsorUserOperation", () => { expect(userOperationReceipt?.userOpHash).toBe(opHash) expect(userOperationReceipt?.receipt.transactionHash).toBeTruthy() - const receipt = await bundlerClientV07.getUserOperationReceipt({ + const receipt = await bundlerClient.getUserOperationReceipt({ hash: opHash }) diff --git a/packages/permissionless/actions/pimlico/validateSponsorshipPolicies.test.ts b/packages/permissionless/actions/pimlico/validateSponsorshipPolicies.test.ts index 9f836468..4035e289 100644 --- a/packages/permissionless/actions/pimlico/validateSponsorshipPolicies.test.ts +++ b/packages/permissionless/actions/pimlico/validateSponsorshipPolicies.test.ts @@ -1,48 +1,46 @@ -import { generatePrivateKey } from "viem/accounts" +import { entryPoint06Address } from "viem/account-abstraction" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" import { - getPimlicoPaymasterClient, + getBundlerClient, + getPimlicoClient, getSimpleAccountClient } from "../../../permissionless-test/src/utils" -import { ENTRYPOINT_ADDRESS_V06 } from "../../utils" import { validateSponsorshipPolicies } from "./validateSponsorshipPolicies" describe("validateSponsorshipPolicies", () => { testWithRpc("Validating sponsorship policies", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc + const { paymasterRpc } = rpc - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc, - paymasterClient: getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }) + const simpleAccountClient = getBundlerClient({ + account: await getSimpleAccountClient({ + ...rpc, + entryPoint: { + version: "0.6" + } + }), + ...rpc }) - const userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: "0x5af0d9827e0c53e4799bb226655a1de152a425a5", - data: "0x", - value: 0n - }) + const userOperation = await simpleAccountClient.prepareUserOperation({ + calls: [ + { + to: "0x5af0d9827e0c53e4799bb226655a1de152a425a5", + data: "0x", + value: 0n } - }) + ] + }) - const pimlicoPaymasterClient = getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc + const pimlicoPaymasterClient = getPimlicoClient({ + entryPoint: entryPoint06Address, + altoRpc: paymasterRpc }) const policies = await validateSponsorshipPolicies( pimlicoPaymasterClient, { - entryPoint: ENTRYPOINT_ADDRESS_V06, + entryPointAddress: entryPoint06Address, userOperation: userOperation, sponsorshipPolicyIds: ["sp_crazy_kangaroo"] } diff --git a/packages/permissionless/actions/public/getAccountNonce.test.ts b/packages/permissionless/actions/public/getAccountNonce.test.ts index 2fe9099e..8fee45ad 100644 --- a/packages/permissionless/actions/public/getAccountNonce.test.ts +++ b/packages/permissionless/actions/public/getAccountNonce.test.ts @@ -1,50 +1,62 @@ import { http, createPublicClient } from "viem" +import { + entryPoint06Address, + entryPoint07Address +} from "viem/account-abstraction" import { generatePrivateKey } from "viem/accounts" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { getSimpleAccountClient } from "../../../permissionless-test/src/utils" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" +import { + getBundlerClient, + getSimpleAccountClient +} from "../../../permissionless-test/src/utils" import { getAccountNonce } from "./getAccountNonce" describe("getAccountNonce", () => { testWithRpc("getAccountNonce_V06", async ({ rpc }) => { - const { anvilRpc, altoRpc } = rpc + const { anvilRpc } = rpc const client = createPublicClient({ transport: http(anvilRpc) }) - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc + const simpleAccountClient = getBundlerClient({ + account: await getSimpleAccountClient({ + ...rpc, + entryPoint: { + version: "0.6" + } + }), + ...rpc }) const nonce = await getAccountNonce(client, { - entryPoint: ENTRYPOINT_ADDRESS_V06, - sender: simpleAccountClient.account.address + entryPointAddress: entryPoint06Address, + address: simpleAccountClient.account.address }) expect(nonce).toBe(0n) }) testWithRpc("getAccountNonce_V07", async ({ rpc }) => { - const { anvilRpc, altoRpc } = rpc + const { anvilRpc } = rpc const client = createPublicClient({ transport: http(anvilRpc) }) - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc + const simpleAccountClient = getBundlerClient({ + account: await getSimpleAccountClient({ + ...rpc, + entryPoint: { + version: "0.7" + } + }), + ...rpc }) const nonce = await getAccountNonce(client, { - entryPoint: ENTRYPOINT_ADDRESS_V07, - sender: simpleAccountClient.account.address + entryPointAddress: entryPoint07Address, + address: simpleAccountClient.account.address }) expect(nonce).toBe(0n) diff --git a/packages/permissionless/actions/public/getSenderAddress.test.ts b/packages/permissionless/actions/public/getSenderAddress.test.ts index 8f9b494e..f33c0da9 100644 --- a/packages/permissionless/actions/public/getSenderAddress.test.ts +++ b/packages/permissionless/actions/public/getSenderAddress.test.ts @@ -1,10 +1,14 @@ -import { http, createPublicClient } from "viem" -import { generatePrivateKey } from "viem/accounts" +import { http, concatHex, createPublicClient } from "viem" +import { + entryPoint06Address, + entryPoint07Address +} from "viem/account-abstraction" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" -import { getSimpleAccountClient } from "../../../permissionless-test/src/utils" -import type { ENTRYPOINT_ADDRESS_V06_TYPE } from "../../types" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../utils" +import { + getBundlerClient, + getSimpleAccountClient +} from "../../../permissionless-test/src/utils" import { getSenderAddress } from "./getSenderAddress" describe("getSenderAddress", () => { @@ -15,16 +19,26 @@ describe("getSenderAddress", () => { transport: http(anvilRpc) }) - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc + const simpleAccountClient = getBundlerClient({ + account: await getSimpleAccountClient({ + ...rpc, + entryPoint: { + version: "0.6" + } + }), + ...rpc }) + const { factory, factoryData } = + await simpleAccountClient.account.getFactoryArgs() + + if (!factory || !factoryData) { + throw Error("Init code not found") + } + const address = await getSenderAddress(client, { - entryPoint: ENTRYPOINT_ADDRESS_V06, - initCode: await simpleAccountClient.account.getInitCode() + entryPointAddress: entryPoint06Address, + initCode: concatHex([factory, factoryData]) }) expect(address).toBe(simpleAccountClient.account.address) @@ -36,18 +50,28 @@ describe("getSenderAddress", () => { transport: http(anvilRpc) }) - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc + const simpleAccountClient = getBundlerClient({ + account: await getSimpleAccountClient({ + ...rpc, + entryPoint: { + version: "0.6" + } + }), + ...rpc }) + const { factory, factoryData } = + await simpleAccountClient.account.getFactoryArgs() + + if (!factory || !factoryData) { + throw Error("Init code not found") + } + await expect(async () => getSenderAddress(client, { - entryPoint: - "0x0000000000000000000000000000000000000000" as ENTRYPOINT_ADDRESS_V06_TYPE, - initCode: await simpleAccountClient.account.getInitCode() + entryPointAddress: + "0x0000000000000000000000000000000000000000" as typeof entryPoint06Address, + initCode: concatHex([factory, factoryData]) }) ).rejects.toThrowError(/not a valid entry point/) }) @@ -58,22 +82,25 @@ describe("getSenderAddress", () => { transport: http(anvilRpc) }) - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc + const simpleAccountClient = getBundlerClient({ + account: await getSimpleAccountClient({ + ...rpc, + entryPoint: { + version: "0.7" + } + }), + ...rpc }) - const factory = await simpleAccountClient.account.getFactory() - const factoryData = await simpleAccountClient.account.getFactoryData() + const { factory, factoryData } = + await simpleAccountClient.account.getFactoryArgs() if (!factory || !factoryData) { throw new Error("Factory or factoryData not found") } const address = await getSenderAddress(client, { - entryPoint: ENTRYPOINT_ADDRESS_V07, + entryPointAddress: entryPoint07Address, factory, factoryData }) diff --git a/packages/permissionless/clients/pimlico.ts b/packages/permissionless/clients/pimlico.ts index 2452b445..aa836fa4 100644 --- a/packages/permissionless/clients/pimlico.ts +++ b/packages/permissionless/clients/pimlico.ts @@ -86,7 +86,7 @@ export type PimlicoClientConfig< * transport: http("https://api.pimlico.io/v2/goerli/rpc?apikey=YOUR_API_KEY_HERE"), * }) */ -export function createPimlicoBundlerClient< +export function createPimlicoClient< transport extends Transport, chain extends Chain | undefined = undefined, account extends SmartAccount | undefined = undefined, @@ -105,7 +105,7 @@ export function createPimlicoBundlerClient< > ): PimlicoClient -export function createPimlicoBundlerClient( +export function createPimlicoClient( parameters: PimlicoClientConfig ): PimlicoClient { const { diff --git a/packages/permissionless/errors/account.ts b/packages/permissionless/errors/account.ts deleted file mode 100644 index 629f7d45..00000000 --- a/packages/permissionless/errors/account.ts +++ /dev/null @@ -1,345 +0,0 @@ -import { type Address, BaseError } from "viem" - -export type SenderAlreadyDeployedErrorType = SenderAlreadyDeployedError & { - name: "SenderAlreadyDeployedError" -} -export class SenderAlreadyDeployedError extends BaseError { - static message = /aa10/ - override name = "SenderAlreadyDeployedError" as const - constructor({ - cause, - sender, - docsPath - }: { cause?: BaseError; sender?: Address; docsPath?: string } = {}) { - super( - [ - `Smart account ${sender} is already deployed.`, - "", - "Possible solutions:", - `• Remove the initCode from the user operation and set it to "0x"`, - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type InitCodeRevertedErrorType = InitCodeRevertedError & { - name: "InitCodeRevertedError" -} -export class InitCodeRevertedError extends BaseError { - static message = /aa13/ - override name = "InitCodeRevertedError" as const - constructor({ - cause, - docsPath - }: { cause?: BaseError; docsPath?: string } = {}) { - super( - [ - "EntryPoint failed to create the smart account with the initCode provided.", - "", - "Possible reasons:", - "• The initCode ran out of gas", - "• The initCode reverted during the account deployment process", - "", - "Possible solutions:", - "• Verify that the factory address in the initCode is correct (the factory address is the first 20 bytes of the initCode).", - "• Verify that the initCode is correct.", - "• Check whether the verificationGasLimit is sufficient for the initCode to complete without running out of gas.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type SenderAddressMismatchErrorType = SenderAddressMismatchError & { - name: "SenderAddressMismatchError" -} -export class SenderAddressMismatchError extends BaseError { - static message = /aa14/ - override name = "SenderAddressMismatchError" as const - constructor({ - cause, - sender, - docsPath - }: { - cause?: BaseError - sender: Address - docsPath?: string - }) { - super( - [ - "The initCode returned a different smart account address than expected.", - `Expected: ${sender}`, - "", - "Possible reasons:", - "• Account deployed with the initCode provided does not match match the sender address provided", - "", - "Possible solutions:", - "• Verify that the sender address was generated deterministically from the initCode. (consider leveraging functions like getSenderAddress)", - "• Verify that the factory address in the initCode is correct (the factory address is the first 20 bytes of the initCode)", - "• Verify that the initCode is correct.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type InitCodeDidNotDeploySenderErrorType = - InitCodeDidNotDeploySenderError & { - name: "InitCodeDidNotDeploySenderError" - } -export class InitCodeDidNotDeploySenderError extends BaseError { - static message = /aa15/ - override name = "InitCodeDidNotDeploySenderError" as const - constructor({ - cause, - sender, - docsPath - }: { - cause?: BaseError - sender: Address - docsPath?: string - }) { - super( - [ - `The initCode did not deploy the sender at the address ${sender}.`, - "", - "Possible reasons:", - "• The initCode factory is not creating an account.", - "• The initCode factory is creating an account, but is not implemented correctly as it is not deploying at the sender address", - "", - "Possible solutions:", - "• Verify that the factory address in the initCode is correct (the factory address is the first 20 bytes of the initCode).", - "• Verify that the initCode factory is implemented correctly. The factory must deploy the smart account at the sender address.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type SenderNotDeployedErrorType = SenderNotDeployedError & { - name: "SenderNotDeployedError" -} -export class SenderNotDeployedError extends BaseError { - static message = /aa20/ - override name = "SenderNotDeployedError" as const - constructor({ - cause, - sender, - docsPath - }: { - cause?: BaseError - sender: Address - docsPath?: string - }) { - super( - [ - `Smart account ${sender} is not deployed.`, - "", - "Possible reasons:", - "• An initCode was not specified, but the sender address (i.e. the smart account) is not deployed.", - "", - "Possible solutions:", - "• If this is the first transaction by this account, make sure the initCode is included in the user operation.", - "• If the smart account is already supposed to be deployed, verify that you have selected the correct sender address for the user operation.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type SmartAccountInsufficientFundsErrorType = - SmartAccountInsufficientFundsError & { - name: "SmartAccountInsufficientFundsError" - } -export class SmartAccountInsufficientFundsError extends BaseError { - static message = /aa21/ - override name = "SmartAccountInsufficientFundsError" as const - constructor({ - cause, - sender, - docsPath - }: { - cause?: BaseError - sender: Address - docsPath?: string - }) { - super( - [ - `You are not using a paymaster, and the ${sender} address did not have enough native tokens to cover the gas costs associated with the user operation.`, - "", - "Possible solutions:", - "• If you are not using a paymaster, verify that the sender address has enough native tokens to cover the required prefund. Consider leveraging functions like getRequiredPrefund.", - "• If you are looking to use a paymaster to cover the gas fees, verify that the paymasterAndData field is set.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type SmartAccountSignatureValidityPeriodErrorType = - SmartAccountSignatureValidityPeriodError & { - name: "SmartAccountSignatureValidityPeriodError" - } -export class SmartAccountSignatureValidityPeriodError extends BaseError { - static message = /aa22/ - override name = "SmartAccountSignatureValidityPeriodError" as const - constructor({ - cause, - docsPath - }: { - cause?: BaseError - docsPath?: string - }) { - super( - [ - "The signature used in the user operation is not valid, because it is outside of the time range it specified.", - "", - "Possible reasons:", - "• This error occurs when the block.timestamp falls after the validUntil timestamp, or before the validAfter timestamp.", - "", - "Possible solutions:", - "• If you are looking to use time-based signatures, verify that the validAfter and validUntil fields are set correctly and that the user operation is sent within the specified range.", - "• If you are not looking to use time-based signatures, verify that the validAfter and validUntil fields are set to 0.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type SmartAccountValidationRevertedErrorType = - SmartAccountValidationRevertedError & { - name: "SmartAccountValidationRevertedError" - } -export class SmartAccountValidationRevertedError extends BaseError { - static message = /aa23/ - override name = "SmartAccountValidationRevertedError" as const - constructor({ - cause, - sender, - docsPath - }: { - cause?: BaseError - sender: Address - docsPath?: string - }) { - super( - [ - `The smart account ${sender} reverted or ran out of gas during the validation of the user operation.`, - "", - "Possible solutions:", - "• Verify that the verificationGasLimit is high enough to cover the validateUserOp function's gas costs.", - "• Make sure validateUserOp returns uint(1) for invalid signatures, and MUST NOT REVERT when the signature is invalid", - "• If you are not using a paymaster, verify that the sender address has enough native tokens to cover the required pre fund. Consider leveraging functions like getRequiredPrefund.", - "• Verify that the validateUserOp function is implemented with the correct logic, and that the user operation is supposed to be valid.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type InvalidSmartAccountSignatureErrorType = - InvalidSmartAccountSignatureError & { - name: "InvalidSmartAccountSignatureError" - } -export class InvalidSmartAccountSignatureError extends BaseError { - static message = /aa24/ - override name = "InvalidSmartAccountSignatureError" as const - constructor({ - cause, - sender, - docsPath - }: { - cause?: BaseError - sender: Address - docsPath?: string - }) { - super( - [ - `The smart account ${sender} signature is invalid.`, - "", - "Possible solutions:", - "• Verify that the user operation was correctly signed, and that the signature was correctly encoded in the signature field of the user operation.", - "• Most smart account implementations sign over the userOpHash. Make sure that the userOpHash is correctly computed. Consider leveraging functions like getUserOperationHash.", - "• Make sure you have selected the correct chainId and entryPointAddress when computing the userOpHash.", - "• Make sure the smart account signature verification function is correctly implemented.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type InvalidSmartAccountNonceErrorType = - InvalidSmartAccountNonceError & { - name: "InvalidSmartAccountNonceError" - } -export class InvalidSmartAccountNonceError extends BaseError { - static message = /aa25/ - override name = "InvalidSmartAccountNonceError" as const - constructor({ - cause, - sender, - nonce, - docsPath - }: { - cause?: BaseError - sender: Address - docsPath?: string - nonce: bigint - }) { - const nonceKey = nonce >> BigInt(64) // first 192 bits of nonce - const nonceSequence = nonce & 0xffffffffffffffffn // last 64 bits of nonce - - super( - [ - `The smart account ${sender} nonce is invalid.`, - `Nonce sent: ${nonce} (key: ${nonceKey}, sequence: ${nonceSequence})`, - "", - "Possible solutions:", - "• Verify that you are using the correct nonce for the user operation. The nonce should be the current nonce of the smart account for the selected key. Consider leveraging functions like getAccountNonce.", - "• Verify that the nonce is formatted correctly.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} diff --git a/packages/permissionless/errors/estimateUserOperationGas.ts b/packages/permissionless/errors/estimateUserOperationGas.ts deleted file mode 100644 index 4564692b..00000000 --- a/packages/permissionless/errors/estimateUserOperationGas.ts +++ /dev/null @@ -1,60 +0,0 @@ -import { BaseError } from "viem" -import type { EstimateUserOperationGasParameters } from "../actions/bundler/estimateUserOperationGas" -import type { EntryPoint } from "../types/entrypoint" -import { prettyPrint } from "./utils" - -export type EstimateUserOperationGasErrorType = - EstimateUserOperationGasError & { - name: "EstimateUserOperationGasError" - } -export class EstimateUserOperationGasError< - entryPoint extends EntryPoint -> extends BaseError { - override cause: BaseError - - override name = "EstimateUserOperationGasError" - - constructor( - cause: BaseError, - { - userOperation, - entryPoint, - docsPath - }: EstimateUserOperationGasParameters & { - docsPath?: string - } - ) { - const prettyArgs = prettyPrint({ - sender: userOperation.sender, - nonce: userOperation.nonce, - initCode: userOperation.initCode, - callData: userOperation.callData, - callGasLimit: userOperation.callGasLimit, - verificationGasLimit: userOperation.verificationGasLimit, - preVerificationGas: userOperation.preVerificationGas, - maxFeePerGas: userOperation.maxFeePerGas, - maxPriorityFeePerGas: userOperation.maxPriorityFeePerGas, - paymasterAndData: userOperation.paymasterAndData, - signature: userOperation.signature, - factory: userOperation.factory, - factoryData: userOperation.factoryData, - paymaster: userOperation.paymaster, - paymasterVerificationGasLimit: - userOperation.paymasterVerificationGasLimit, - paymasterPostOpGasLimit: userOperation.paymasterPostOpGasLimit, - paymasterData: userOperation.paymasterData, - entryPoint - }) - - super(cause.shortMessage, { - cause, - docsPath, - metaMessages: [ - ...(cause.metaMessages ? [...cause.metaMessages, " "] : []), - "Estimate Gas Arguments:", - prettyArgs - ].filter(Boolean) as string[] - }) - this.cause = cause - } -} diff --git a/packages/permissionless/errors/gas.ts b/packages/permissionless/errors/gas.ts deleted file mode 100644 index 26dec105..00000000 --- a/packages/permissionless/errors/gas.ts +++ /dev/null @@ -1,120 +0,0 @@ -import { BaseError } from "viem" - -export type VerificationGasLimitTooLowErrorType = - VerificationGasLimitTooLowError & { - name: "VerificationGasLimitTooLowError" - } -export class VerificationGasLimitTooLowError extends BaseError { - static message = /aa4[01]/ - override name = "VerificationGasLimitTooLowError" - constructor({ - cause, - verificationGasLimit, - docsPath - }: { - cause?: BaseError - verificationGasLimit?: bigint - docsPath?: string - }) { - super( - [ - `The smart account and paymaster verification exceeded the verificationGasLimit ${verificationGasLimit} set for the user operation.`, - "", - "Possible solutions:", - "• Verify that the verificationGasLimit set for the user operation is high enough to cover the gas used during smart account and paymaster verification.", - "• If you are using the eth_estimateUserOperationGas or pm_sponsorUserOperation method from bundler provider to set user operation gas limits and the EntryPoint throws this error during submission, reach out to them.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type ActualGasCostTooHighErrorType = ActualGasCostTooHighError & { - name: "ActualGasCostTooHighError" -} -export class ActualGasCostTooHighError extends BaseError { - static message = /aa51/ - override name = "ActualGasCostTooHighError" - constructor({ - cause, - docsPath - }: { - cause?: BaseError - docsPath?: string - }) { - super( - [ - "The actual gas cost of the user operation ended up being higher than the funds paid by the smart account or the paymaster.", - "", - "Possible solutions:", - "• If you encounter this error, try increasing the verificationGasLimit set for the user operation.", - "• If you are using the eth_estimateUserOperationGas or pm_sponsorUserOperation method from bundler provider to set user operation gas limits and the EntryPoint throws this error during submission, reach out to them.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type GasValuesOverflowErrorType = GasValuesOverflowError & { - name: "GasValuesOverflowError" -} -export class GasValuesOverflowError extends BaseError { - static message = /aa94/ - override name = "GasValuesOverflowError" - constructor({ - cause, - docsPath - }: { - cause?: BaseError - docsPath?: string - }) { - super( - [ - "The gas limit values of the user operation overflowed, they must fit in uint160.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type BundlerOutOfGasErrorType = BundlerOutOfGasError & { - name: "BundlerOutOfGasError" -} -export class BundlerOutOfGasError extends BaseError { - static message = /aa95/ - override name = "BundlerOutOfGasError" - constructor({ - cause, - docsPath - }: { - cause?: BaseError - docsPath?: string - }) { - super( - [ - "The bundler tried to bundle the user operation with the gas limit set too low.", - "", - "Possible solutions:", - "• If you are using your own bundler, configure it send gas limits properly.", - "• If you are using a bundler provider, reach out to them.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} diff --git a/packages/permissionless/errors/index.ts b/packages/permissionless/errors/index.ts index f629f35b..e3dd1020 100644 --- a/packages/permissionless/errors/index.ts +++ b/packages/permissionless/errors/index.ts @@ -1,118 +1,17 @@ -import { - InitCodeDidNotDeploySenderError, - type InitCodeDidNotDeploySenderErrorType, - InitCodeRevertedError, - type InitCodeRevertedErrorType, - InvalidSmartAccountNonceError, - type InvalidSmartAccountNonceErrorType, - InvalidSmartAccountSignatureError, - type InvalidSmartAccountSignatureErrorType, - SenderAddressMismatchError, - type SenderAddressMismatchErrorType, - SenderAlreadyDeployedError, - type SenderAlreadyDeployedErrorType, - SenderNotDeployedError, - type SenderNotDeployedErrorType, - SmartAccountInsufficientFundsError, - type SmartAccountInsufficientFundsErrorType, - SmartAccountSignatureValidityPeriodError, - type SmartAccountSignatureValidityPeriodErrorType, - SmartAccountValidationRevertedError, - type SmartAccountValidationRevertedErrorType -} from "./account" -import { - EstimateUserOperationGasError, - type EstimateUserOperationGasErrorType -} from "./estimateUserOperationGas" -import { - SendUserOperationError, - type SendUserOperationErrorType -} from "./sendUserOperation" +import { BaseError } from "viem" -import { - InvalidPaymasterAndDataError, - type InvalidPaymasterAndDataErrorType, - PaymasterDataRejectedError, - type PaymasterDataRejectedErrorType, - PaymasterDepositTooLowError, - type PaymasterDepositTooLowErrorType, - PaymasterNotDeployedError, - type PaymasterNotDeployedErrorType, - PaymasterPostOpRevertedError, - type PaymasterPostOpRevertedErrorType, - PaymasterValidationRevertedError, - type PaymasterValidationRevertedErrorType, - PaymasterValidityPeriodError, - type PaymasterValidityPeriodErrorType -} from "./paymaster" - -import { - InvalidAggregatorError, - type InvalidAggregatorErrorType, - InvalidBeneficiaryAddressError, - type InvalidBeneficiaryAddressErrorType -} from "./bundler" - -import { - ActualGasCostTooHighError, - type ActualGasCostTooHighErrorType, - BundlerOutOfGasError, - type BundlerOutOfGasErrorType, - GasValuesOverflowError, - type GasValuesOverflowErrorType, - VerificationGasLimitTooLowError, - type VerificationGasLimitTooLowErrorType -} from "./gas" - -export { - type InitCodeDidNotDeploySenderErrorType, - type InitCodeRevertedErrorType, - type InvalidSmartAccountNonceErrorType, - type InvalidSmartAccountSignatureErrorType, - type SenderAddressMismatchErrorType, - type SenderAlreadyDeployedErrorType, - type SenderNotDeployedErrorType, - type SmartAccountInsufficientFundsErrorType, - type SmartAccountSignatureValidityPeriodErrorType, - type SmartAccountValidationRevertedErrorType, - type InvalidPaymasterAndDataErrorType, - type PaymasterDataRejectedErrorType, - type PaymasterDepositTooLowErrorType, - type PaymasterNotDeployedErrorType, - type PaymasterPostOpRevertedErrorType, - type PaymasterValidationRevertedErrorType, - type PaymasterValidityPeriodErrorType, - type InvalidAggregatorErrorType, - type InvalidBeneficiaryAddressErrorType, - type ActualGasCostTooHighErrorType, - type BundlerOutOfGasErrorType, - type GasValuesOverflowErrorType, - type VerificationGasLimitTooLowErrorType, - SenderAlreadyDeployedError, - EstimateUserOperationGasError, - InitCodeRevertedError, - SenderAddressMismatchError, - InitCodeDidNotDeploySenderError, - SenderNotDeployedError, - SmartAccountInsufficientFundsError, - SmartAccountSignatureValidityPeriodError, - SmartAccountValidationRevertedError, - InvalidSmartAccountNonceError, - PaymasterNotDeployedError, - PaymasterDepositTooLowError, - InvalidSmartAccountSignatureError, - InvalidBeneficiaryAddressError, - InvalidAggregatorError, - InvalidPaymasterAndDataError, - PaymasterDataRejectedError, - PaymasterValidityPeriodError, - PaymasterValidationRevertedError, - VerificationGasLimitTooLowError, - ActualGasCostTooHighError, - GasValuesOverflowError, - BundlerOutOfGasError, - PaymasterPostOpRevertedError, - SendUserOperationError, - type EstimateUserOperationGasErrorType, - type SendUserOperationErrorType +export class AccountNotFoundError extends BaseError { + constructor({ docsPath }: { docsPath?: string | undefined } = {}) { + super( + [ + "Could not find an Account to execute with this Action.", + "Please provide an Account with the `account` argument on the Action, or by supplying an `account` to the Client." + ].join("\n"), + { + docsPath, + docsSlug: "account", + name: "AccountNotFoundError" + } + ) + } } diff --git a/packages/permissionless/errors/paymaster.ts b/packages/permissionless/errors/paymaster.ts deleted file mode 100644 index 7000f238..00000000 --- a/packages/permissionless/errors/paymaster.ts +++ /dev/null @@ -1,257 +0,0 @@ -import { BaseError, type Hex } from "viem" -import { getAddressFromInitCodeOrPaymasterAndData } from "../utils/" - -export type PaymasterNotDeployedErrorType = PaymasterNotDeployedError & { - name: "PaymasterNotDeployedError" -} -export class PaymasterNotDeployedError extends BaseError { - static message = /aa30/ - override name = "PaymasterNotDeployedError" - constructor({ - cause, - paymasterAndData, - docsPath - }: { - cause?: BaseError - paymasterAndData?: Hex - docsPath?: string - } = {}) { - const paymaster = paymasterAndData - ? getAddressFromInitCodeOrPaymasterAndData(paymasterAndData) - : "0x" - - super( - [ - `Paymaster ${paymaster} is not deployed.`, - "", - "Possible solutions:", - "• Verify that the paymasterAndData field is correct, and that the first 20 bytes are the address of the paymaster contract you intend to use.", - "• Verify that the paymaster contract is deployed on the network you are using.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type PaymasterDepositTooLowErrorType = PaymasterDepositTooLowError & { - name: "PaymasterDepositTooLowError" -} -export class PaymasterDepositTooLowError extends BaseError { - static message = /aa31/ - override name = "PaymasterDepositTooLowError" - constructor({ - cause, - paymasterAndData, - docsPath - }: { - cause?: BaseError - paymasterAndData?: Hex - docsPath?: string - } = {}) { - const paymaster = paymasterAndData - ? getAddressFromInitCodeOrPaymasterAndData(paymasterAndData) - : "0x" - - super( - [ - `Paymaster ${paymaster} contract does not have enough funds deposited into the EntryPoint contract to cover the required funds for the user operation.`, - "", - "Possible solutions:", - "• If you are using your own paymaster contract, deposit more funds into the EntryPoint contract through the deposit() function of the paymaster contract.", - "• Verify that the paymasterAndData field is correct, and that the first 20 bytes are the address of the paymaster contract you intend to useVerify that the paymasterAndData field is correct, and that the first 20 bytes are the address of the paymaster contract you intend to use.", - "• If you are using a paymaster service, reach out to them.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type PaymasterValidityPeriodErrorType = PaymasterValidityPeriodError & { - name: "PaymasterValidityPeriodError" -} -export class PaymasterValidityPeriodError extends BaseError { - static message = /aa32/ - override name = "PaymasterValidityPeriodError" - constructor({ - cause, - paymasterAndData, - docsPath - }: { - cause?: BaseError - paymasterAndData?: Hex - docsPath?: string - }) { - const paymaster = paymasterAndData - ? getAddressFromInitCodeOrPaymasterAndData(paymasterAndData) - : "0x" - - super( - [ - `Paymaster ${paymaster}'s data used in the paymasterAndData field of the user operation is not valid, because it is outside of the time range it specified.`, - "", - "Possible reasons:", - "• This error occurs when the block.timestamp falls after the validUntil timestamp, or before the validAfter timestamp.", - "", - "Possible solutions:", - "• If you are using your own paymaster contract and using time-based signatures, verify that the validAfter and validUntil fields are set correctly and that the user operation is sent within the specified range.", - "• If you are using your own paymaster contract and not looking to use time-based signatures, verify that the validAfter and validUntil fields are set to 0.", - "• If you are using a service, contact your service provider for their paymaster's validity.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type PaymasterValidationRevertedErrorType = - PaymasterValidationRevertedError & { - name: "PaymasterValidationRevertedError" - } -export class PaymasterValidationRevertedError extends BaseError { - static message = /aa33/ - override name = "PaymasterValidationRevertedError" - constructor({ - cause, - paymasterAndData, - docsPath - }: { - cause?: BaseError - paymasterAndData?: Hex - docsPath?: string - }) { - const paymaster = paymasterAndData - ? getAddressFromInitCodeOrPaymasterAndData(paymasterAndData) - : "0x" - - super( - [ - `The validatePaymasterUserOp function of the paymaster ${paymaster} either reverted or ran out of gas.`, - "", - "Possible solutions:", - "• Verify that the verificationGasLimit is high enough to cover the validatePaymasterUserOp function's gas costs.", - "• If you are using your own paymaster contract, verify that the validatePaymasterUserOp function is implemented with the correct logic, and that the user operation is supposed to be valid.", - "• If you are using a paymaster service, and the user operation is well formed with a high enough verificationGasLimit, reach out to them.", - "• If you are not looking to use a paymaster to cover the gas fees, verify that the paymasterAndData field is not set.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type PaymasterDataRejectedErrorType = PaymasterDataRejectedError & { - name: "PaymasterDataRejectedError" -} -export class PaymasterDataRejectedError extends BaseError { - static message = /aa34/ - override name = "PaymasterDataRejectedError" - constructor({ - cause, - paymasterAndData, - docsPath - }: { - cause?: BaseError - paymasterAndData?: Hex - docsPath?: string - }) { - const paymaster = paymasterAndData - ? getAddressFromInitCodeOrPaymasterAndData(paymasterAndData) - : "0x" - - super( - [ - `The validatePaymasterUserOp function of the paymaster ${paymaster} rejected paymasterAndData.`, - "", - "Possible solutions:", - "• If you are using your own paymaster contract, verify that the user operation was correctly signed according to your implementation, and that the paymaster signature was correctly encoded in the paymasterAndData field of the user operation.", - "• If you are using a paymaster service, make sure you do not modify any of the fields of the user operation after the paymaster signs over it (except the signature field).", - "• If you are using a paymaster service and you have not modified any of the fields except the signature but you are still getting this error, reach out to them.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type PaymasterPostOpRevertedErrorType = PaymasterPostOpRevertedError & { - name: "PaymasterPostOpRevertedError" -} -export class PaymasterPostOpRevertedError extends BaseError { - static message = /aa50/ - override name = "PaymasterPostOpRevertedError" - constructor({ - cause, - paymasterAndData, - docsPath - }: { - cause?: BaseError - paymasterAndData?: Hex - docsPath?: string - }) { - const paymaster = paymasterAndData - ? getAddressFromInitCodeOrPaymasterAndData(paymasterAndData) - : "0x" - - super( - [ - `The postOp function of the paymaster ${paymaster} reverted.`, - "", - "Possible solutions:", - "• If you are using your own paymaster contract, verify that that you have correctly implemented the postOp function (if you are using one). If you do not intent to make use of the postOp function, make sure you do not set the context parameter in the paymaster's validatePaymasterUserOp function.", - "• If you are using a paymaster service and you see this error, reach out to them.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} - -export type InvalidPaymasterAndDataErrorType = InvalidPaymasterAndDataError & { - name: "InvalidPaymasterAndDataError" -} -export class InvalidPaymasterAndDataError extends BaseError { - static message = /aa93/ - override name = "InvalidPaymasterAndDataError" - constructor({ - cause, - docsPath - }: { - cause?: BaseError - docsPath?: string - }) { - super( - [ - "The paymasterAndData field of the user operation is invalid.", - "", - "Possible solutions:", - "• Make sure you have either not set a value for the paymasterAndData, or that it is at least 20 bytes long.", - "• If you are using a paymaster service, reach out to them.", - "", - docsPath ? `Docs: ${docsPath}` : "" - ].join("\n"), - { - cause - } - ) - } -} diff --git a/packages/permissionless/errors/sendUserOperation.ts b/packages/permissionless/errors/sendUserOperation.ts deleted file mode 100644 index 9d0fcc02..00000000 --- a/packages/permissionless/errors/sendUserOperation.ts +++ /dev/null @@ -1,43 +0,0 @@ -import { BaseError } from "viem" -import type { SendUserOperationParameters } from "../actions/bundler/sendUserOperation" -import type { EntryPoint } from "../types/entrypoint" -import { prettyPrint } from "./utils" - -export type SendUserOperationErrorType = - SendUserOperationError & { - name: "SendUserOperationError" - } -export class SendUserOperationError< - entryPoint extends EntryPoint -> extends BaseError { - override cause: BaseError - - override name = "SendUserOperationError" - - constructor( - cause: BaseError, - { - userOperation, - entryPoint, - docsPath - }: SendUserOperationParameters & { - docsPath?: string - } - ) { - const prettyArgs = prettyPrint({ - ...userOperation, - entryPoint - }) - - super(cause.shortMessage, { - cause, - docsPath, - metaMessages: [ - ...(cause.metaMessages ? [...cause.metaMessages, " "] : []), - "sendUserOperation Arguments:", - prettyArgs - ].filter(Boolean) as string[] - }) - this.cause = cause - } -} diff --git a/packages/permissionless/errors/utils.ts b/packages/permissionless/errors/utils.ts deleted file mode 100644 index a8b7c589..00000000 --- a/packages/permissionless/errors/utils.ts +++ /dev/null @@ -1,19 +0,0 @@ -export type ErrorType = Error & { name: name } - -export function prettyPrint( - args: Record -) { - const entries = Object.entries(args) - .map(([key, value]) => { - if (value === undefined || value === false) return null - return [key, value] - }) - .filter(Boolean) as [string, string][] - const maxLength = entries.reduce( - (acc, [key]) => Math.max(acc, key.length), - 0 - ) - return entries - .map(([key, value]) => ` ${`${key}:`.padEnd(maxLength + 1)} ${value}`) - .join("\n") -} diff --git a/packages/permissionless/experimental/eip7677/actions/getPaymasterData.test.ts b/packages/permissionless/experimental/eip7677/actions/getPaymasterData.test.ts index 6c7185d6..93ec24be 100644 --- a/packages/permissionless/experimental/eip7677/actions/getPaymasterData.test.ts +++ b/packages/permissionless/experimental/eip7677/actions/getPaymasterData.test.ts @@ -1,43 +1,51 @@ import { zeroAddress } from "viem" -import { generatePrivateKey } from "viem/accounts" +import { + entryPoint06Address, + entryPoint07Address +} from "viem/account-abstraction" import { foundry } from "viem/chains" import { describe, expect } from "vitest" import { testWithRpc } from "../../../../permissionless-test/src/testWithRpc" import { getBundlerClient, - getPimlicoPaymasterClient, + getPimlicoClient, getSimpleAccountClient } from "../../../../permissionless-test/src/utils" -import type { UserOperation } from "../../../types/userOperation" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../../utils" import { paymasterActionsEip7677 } from "../clients/decorators/paymasterActionsEip7677" describe("EIP-7677 getPaymasterData", () => { testWithRpc("getPaymasterData_V06", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc + const { paymasterRpc } = rpc - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc + const simpleAccountClient = getBundlerClient({ + account: await getSimpleAccountClient({ + ...rpc, + entryPoint: { + version: "0.6" + } + }), + ...rpc }) - const pimlicoPaymasterClient = getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }).extend(paymasterActionsEip7677(ENTRYPOINT_ADDRESS_V06)) - - let userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - } + const pimlicoPaymasterClient = getPimlicoClient({ + entryPoint: entryPoint06Address, + altoRpc: paymasterRpc + }).extend( + paymasterActionsEip7677({ + address: entryPoint06Address, + version: "0.6" }) + ) + + let userOperation = await simpleAccountClient.prepareUserOperation({ + calls: [ + { + to: zeroAddress, + data: "0x", + value: 0n + } + ] + }) const paymasterData = await pimlicoPaymasterClient.getPaymasterData({ userOperation, @@ -48,8 +56,7 @@ describe("EIP-7677 getPaymasterData", () => { // test that smart account can send op with the returned values const bundlerClient = getBundlerClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - altoRpc + ...rpc }) userOperation = { @@ -60,7 +67,8 @@ describe("EIP-7677 getPaymasterData", () => { await simpleAccountClient.account.signUserOperation(userOperation) const hash = await bundlerClient.sendUserOperation({ - userOperation + account: simpleAccountClient.account, + ...userOperation }) const receipt = await bundlerClient.waitForUserOperationReceipt({ @@ -70,30 +78,37 @@ describe("EIP-7677 getPaymasterData", () => { }) testWithRpc("getPaymasterData_V07", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc + const { altoRpc, paymasterRpc } = rpc - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc + const simpleAccountClient = getBundlerClient({ + account: await getSimpleAccountClient({ + ...rpc, + entryPoint: { + version: "0.7" + } + }), + ...rpc }) - const pimlicoPaymasterClient = getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }).extend(paymasterActionsEip7677(ENTRYPOINT_ADDRESS_V07)) - - let userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) - } + const pimlicoPaymasterClient = getPimlicoClient({ + entryPoint: entryPoint07Address, + altoRpc: paymasterRpc + }).extend( + paymasterActionsEip7677({ + address: entryPoint07Address, + version: "0.7" }) + ) + + let userOperation = await simpleAccountClient.prepareUserOperation({ + calls: [ + { + to: zeroAddress, + data: "0x", + value: 0n + } + ] + }) const paymasterGasValues = { paymasterPostOpGasLimit: 150_000n, @@ -113,20 +128,21 @@ describe("EIP-7677 getPaymasterData", () => { // test that smart account can send op with the returned values const bundlerClient = getBundlerClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - altoRpc + ...rpc }) userOperation = { ...userOperation, ...paymasterGasValues, ...paymasterData - } as UserOperation<"v0.7"> + } + userOperation.signature = await simpleAccountClient.account.signUserOperation(userOperation) const hash = await bundlerClient.sendUserOperation({ - userOperation + account: simpleAccountClient.account, + ...userOperation }) const receipt = await bundlerClient.waitForUserOperationReceipt({ diff --git a/packages/permissionless/experimental/eip7677/actions/getPaymasterData.ts b/packages/permissionless/experimental/eip7677/actions/getPaymasterData.ts index a1fadf4e..78775f89 100644 --- a/packages/permissionless/experimental/eip7677/actions/getPaymasterData.ts +++ b/packages/permissionless/experimental/eip7677/actions/getPaymasterData.ts @@ -1,6 +1,5 @@ import { type Account, - type Address, type Chain, ChainNotFoundError, type Client, @@ -10,150 +9,99 @@ import { toHex } from "viem" import type { - ENTRYPOINT_ADDRESS_V06_TYPE, - ENTRYPOINT_ADDRESS_V07_TYPE, - EntryPoint, - GetEntryPointVersion -} from "../../../types/entrypoint" -import type { UserOperationWithBigIntAsHex } from "../../../types/userOperation" -import { deepHexlify, getEntryPointVersion } from "../../../utils" + UserOperation, + entryPoint06Address, + entryPoint07Address +} from "viem/account-abstraction" +import { deepHexlify } from "../../../utils" import type { Eip7677RpcSchema, GetRpcPaymasterDataReturnType } from "../types/paymaster" export type GetPaymasterDataParameters< - TEntryPoint extends EntryPoint, - TChain extends Chain | undefined = Chain | undefined, - TChainOverride extends Chain | undefined = Chain | undefined + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7", + TChain extends Chain | undefined, + TChainOverride extends Chain | undefined > = { - userOperation: GetEntryPointVersion extends "v0.6" - ? { - sender: Address - nonce: bigint - initCode: Hex - callData: Hex - callGasLimit: bigint - verificationGasLimit: bigint - preVerificationGas: bigint - maxFeePerGas: bigint - maxPriorityFeePerGas: bigint - paymasterAndData?: Hex - signature?: Hex - factory?: never - factoryData?: never - paymaster?: never - paymasterVerificationGasLimit?: never - paymasterPostOpGasLimit?: never - paymasterData?: never - } - : { - sender: Address - nonce: bigint - factory?: Address - factoryData?: Hex - callData: Hex - callGasLimit: bigint - verificationGasLimit: bigint - preVerificationGas: bigint - maxFeePerGas: bigint - maxPriorityFeePerGas: bigint - paymaster?: Address - paymasterData?: Hex - signature?: Hex - paymasterAndData?: never - paymasterVerificationGasLimit: bigint - paymasterPostOpGasLimit: bigint - } - entryPoint: TEntryPoint + userOperation: UserOperation + entryPoint: { + address: entryPointAddress + version: entryPointVersion + } context?: Record } & GetChainParameter -export type GetPaymasterDataReturnType = - GetEntryPointVersion extends "v0.6" - ? { - paymasterAndData: Hex - } - : { - paymaster: Hex - paymasterData: Hex - } +export type GetPaymasterDataReturnType< + entryPointVersion extends "0.6" | "0.7" +> = entryPointVersion extends "0.6" + ? { + paymasterAndData: Hex + } + : { + paymaster: Hex + paymasterData: Hex + } export async function getPaymasterData< - TEntryPoint extends EntryPoint, + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7", TChain extends Chain | undefined, - TClientAccount extends Account | undefined = undefined, - TTransport extends Transport = Transport, - TChainOverride extends Chain | undefined = Chain | undefined + TChainOverride extends Chain | undefined >( client: Client< - TTransport, + Transport, TChain, - TClientAccount, - Eip7677RpcSchema + Account | undefined, + Eip7677RpcSchema >, { userOperation, entryPoint, context, chain - }: GetPaymasterDataParameters -): Promise> { + }: GetPaymasterDataParameters< + entryPointAddress, + entryPointVersion, + TChain, + TChainOverride + > +): Promise> { const chainId = chain?.id ?? client.chain?.id if (!chainId) { throw new ChainNotFoundError() } - const params: - | [ - UserOperationWithBigIntAsHex>, - TEntryPoint, - Hex, - Record - ] - | [ - UserOperationWithBigIntAsHex>, - TEntryPoint, - Hex - ] = context - ? [ - deepHexlify(userOperation) as UserOperationWithBigIntAsHex< - GetEntryPointVersion - >, - entryPoint, - toHex(chainId), - context - ] - : [ - deepHexlify(userOperation) as UserOperationWithBigIntAsHex< - GetEntryPointVersion - >, - entryPoint, - toHex(chainId) - ] - const response = await client.request({ method: "pm_getPaymasterData", - params + params: context + ? [ + deepHexlify(userOperation), + entryPoint.address, + toHex(chainId), + context + ] + : [deepHexlify(userOperation), entryPoint.address, toHex(chainId)] }) - const entryPointVersion = getEntryPointVersion(entryPoint) - - if (entryPointVersion === "v0.6") { - const responseV06 = - response as GetRpcPaymasterDataReturnType + if (entryPoint.version === "0.6") { + const responseV06 = response as GetRpcPaymasterDataReturnType<"0.6"> return { paymasterAndData: responseV06.paymasterAndData - } as GetPaymasterDataReturnType + } as GetPaymasterDataReturnType } - const responseV07 = - response as GetRpcPaymasterDataReturnType + const responseV07 = response as GetRpcPaymasterDataReturnType<"0.7"> return { paymaster: responseV07.paymaster, paymasterData: responseV07.paymasterData - } as GetPaymasterDataReturnType + } as GetPaymasterDataReturnType } diff --git a/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.test.ts b/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.test.ts index 991fbc75..6324b331 100644 --- a/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.test.ts +++ b/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.test.ts @@ -1,41 +1,51 @@ import { zeroAddress } from "viem" -import { generatePrivateKey } from "viem/accounts" +import { + entryPoint06Address, + entryPoint07Address +} from "viem/account-abstraction" import { foundry } from "viem/chains" import { describe, expect } from "vitest" import { testWithRpc } from "../../../../permissionless-test/src/testWithRpc" import { - getPimlicoPaymasterClient, + getBundlerClient, + getPimlicoClient, getSimpleAccountClient } from "../../../../permissionless-test/src/utils" -import { ENTRYPOINT_ADDRESS_V06, ENTRYPOINT_ADDRESS_V07 } from "../../../utils" import { paymasterActionsEip7677 } from "../clients/decorators/paymasterActionsEip7677" describe("EIP-7677 getPaymasterStubData", () => { testWithRpc("getPaymasterStubData_V06", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc + const { paymasterRpc } = rpc - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc + const simpleAccountClient = getBundlerClient({ + account: await getSimpleAccountClient({ + ...rpc, + entryPoint: { + version: "0.6" + } + }), + ...rpc }) - const pimlicoPaymasterClient = getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V06, - paymasterRpc - }).extend(paymasterActionsEip7677(ENTRYPOINT_ADDRESS_V06)) + const pimlicoPaymasterClient = getPimlicoClient({ + entryPoint: entryPoint06Address, + altoRpc: paymasterRpc + }).extend( + paymasterActionsEip7677({ + address: entryPoint06Address, + version: "0.6" + }) + ) - const userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) + const userOperation = await simpleAccountClient.prepareUserOperation({ + calls: [ + { + to: zeroAddress, + data: "0x", + value: 0n } - }) + ] + }) const stubData = await pimlicoPaymasterClient.getPaymasterStubData({ userOperation, @@ -49,30 +59,37 @@ describe("EIP-7677 getPaymasterStubData", () => { }) testWithRpc("getPaymasterStubData_V07", async ({ rpc }) => { - const { anvilRpc, altoRpc, paymasterRpc } = rpc + const { paymasterRpc } = rpc - const simpleAccountClient = await getSimpleAccountClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - privateKey: generatePrivateKey(), - altoRpc: altoRpc, - anvilRpc: anvilRpc + const simpleAccountClient = getBundlerClient({ + account: await getSimpleAccountClient({ + ...rpc, + entryPoint: { + version: "0.7" + } + }), + ...rpc }) - const pimlicoPaymasterClient = getPimlicoPaymasterClient({ - entryPoint: ENTRYPOINT_ADDRESS_V07, - paymasterRpc - }).extend(paymasterActionsEip7677(ENTRYPOINT_ADDRESS_V07)) + const pimlicoPaymasterClient = getPimlicoClient({ + entryPoint: entryPoint07Address, + altoRpc: paymasterRpc + }).extend( + paymasterActionsEip7677({ + address: entryPoint07Address, + version: "0.7" + }) + ) - const userOperation = - await simpleAccountClient.prepareUserOperationRequest({ - userOperation: { - callData: await simpleAccountClient.account.encodeCallData({ - to: zeroAddress, - data: "0x", - value: 0n - }) + const userOperation = await simpleAccountClient.prepareUserOperation({ + calls: [ + { + to: zeroAddress, + data: "0x", + value: 0n } - }) + ] + }) const stubData = await pimlicoPaymasterClient.getPaymasterStubData({ userOperation, diff --git a/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.ts b/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.ts index 19108e2b..594e3973 100644 --- a/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.ts +++ b/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.ts @@ -1,6 +1,5 @@ import { type Account, - type Address, type Chain, ChainNotFoundError, type Client, @@ -10,155 +9,104 @@ import { toHex } from "viem" import type { - ENTRYPOINT_ADDRESS_V06_TYPE, - ENTRYPOINT_ADDRESS_V07_TYPE, - EntryPoint, - GetEntryPointVersion -} from "../../../types/entrypoint" -import type { UserOperationWithBigIntAsHex } from "../../../types/userOperation" -import { deepHexlify, getEntryPointVersion } from "../../../utils" + UserOperation, + entryPoint06Address, + entryPoint07Address +} from "viem/account-abstraction" +import { deepHexlify } from "../../../utils" import type { Eip7677RpcSchema, GetRpcPaymasterStubDataReturnType } from "../types/paymaster" export type GetPaymasterStubDataParameters< - TEntryPoint extends EntryPoint, + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7", TChain extends Chain | undefined, - TChainOverride extends Chain | undefined = Chain | undefined + TChainOverride extends Chain | undefined > = { - userOperation: GetEntryPointVersion extends "v0.6" - ? { - sender: Address - nonce: bigint - initCode: Hex - callData: Hex - callGasLimit: bigint - verificationGasLimit: bigint - preVerificationGas: bigint - maxFeePerGas: bigint - maxPriorityFeePerGas: bigint - paymasterAndData?: Hex - signature?: Hex - factory?: never - factoryData?: never - paymaster?: never - paymasterVerificationGasLimit?: never - paymasterPostOpGasLimit?: never - paymasterData?: never - } - : { - sender: Address - nonce: bigint - factory?: Address - factoryData?: Hex - callData: Hex - callGasLimit: bigint - verificationGasLimit: bigint - preVerificationGas: bigint - maxFeePerGas: bigint - maxPriorityFeePerGas: bigint - paymaster?: Address - paymasterData?: Hex - signature?: Hex - paymasterAndData?: never - paymasterVerificationGasLimit?: bigint - paymasterPostOpGasLimit?: bigint - } - entryPoint: TEntryPoint + userOperation: UserOperation + entryPoint: { + address: entryPointAddress + version: entryPointVersion + } context?: Record } & GetChainParameter -export type GetPaymasterStubDataReturnType = - GetEntryPointVersion extends "v0.6" - ? { - paymasterAndData: Hex - sponsor?: { name: string; icon?: string } - isFinal?: boolean - } - : { - paymaster: Hex - paymasterData: Hex - paymasterVerificationGasLimit?: bigint - paymasterPostOpGasLimit?: bigint - sponsor?: { name: string; icon?: string } - isFinal?: boolean - } +export type GetPaymasterStubDataReturnType< + entryPointVersion extends "0.6" | "0.7" +> = entryPointVersion extends "0.6" + ? { + paymasterAndData: Hex + sponsor?: { name: string; icon?: string } + isFinal?: boolean + } + : { + paymaster: Hex + paymasterData: Hex + paymasterVerificationGasLimit?: bigint + paymasterPostOpGasLimit?: bigint + sponsor?: { name: string; icon?: string } + isFinal?: boolean + } export async function getPaymasterStubData< - TEntryPoint extends EntryPoint, + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7", TChain extends Chain | undefined, - TTransport extends Transport = Transport, - TClientAccount extends Account | undefined = undefined, - TChainOverride extends Chain | undefined = Chain | undefined + TChainOverride extends Chain | undefined >( client: Client< - TTransport, + Transport, TChain, - TClientAccount, - Eip7677RpcSchema + Account | undefined, + Eip7677RpcSchema >, { userOperation, entryPoint, context, chain - }: GetPaymasterStubDataParameters -): Promise> { + }: GetPaymasterStubDataParameters< + entryPointAddress, + entryPointVersion, + TChain, + TChainOverride + > +): Promise> { const chainId = chain?.id ?? client.chain?.id if (!chainId) { throw new ChainNotFoundError() } - const params: - | [ - UserOperationWithBigIntAsHex>, - TEntryPoint, - Hex, - Record - ] - | [ - UserOperationWithBigIntAsHex>, - TEntryPoint, - Hex - ] = context - ? [ - deepHexlify(userOperation) as UserOperationWithBigIntAsHex< - GetEntryPointVersion - >, - entryPoint, - toHex(chainId), - context - ] - : [ - deepHexlify(userOperation) as UserOperationWithBigIntAsHex< - GetEntryPointVersion - >, - entryPoint, - toHex(chainId) - ] - const response = await client.request({ method: "pm_getPaymasterStubData", - params + params: context + ? [ + deepHexlify(userOperation), + entryPoint.address, + toHex(chainId), + context + ] + : [deepHexlify(userOperation), entryPoint.address, toHex(chainId)] }) - const entryPointVersion = getEntryPointVersion(entryPoint) - - if (entryPointVersion === "v0.6") { - const responseV06 = - response as GetRpcPaymasterStubDataReturnType + if (entryPoint.version === "0.6") { + const responseV06 = response as GetRpcPaymasterStubDataReturnType<"0.6"> return { paymasterAndData: responseV06.paymasterAndData, sponsor: responseV06.sponsor, isFinal: responseV06.isFinal - } as GetPaymasterStubDataReturnType + } as GetPaymasterStubDataReturnType } - const responseV07 = - response as GetRpcPaymasterStubDataReturnType + const responseV07 = response as GetRpcPaymasterStubDataReturnType<"0.7"> return { paymaster: responseV07.paymaster, @@ -171,5 +119,5 @@ export async function getPaymasterStubData< : undefined, sponsor: responseV07.sponsor, isFinal: responseV07.isFinal - } as GetPaymasterStubDataReturnType + } as GetPaymasterStubDataReturnType } diff --git a/packages/permissionless/experimental/eip7677/clients/decorators/paymasterActionsEip7677.ts b/packages/permissionless/experimental/eip7677/clients/decorators/paymasterActionsEip7677.ts index bffdb614..9cfa856b 100644 --- a/packages/permissionless/experimental/eip7677/clients/decorators/paymasterActionsEip7677.ts +++ b/packages/permissionless/experimental/eip7677/clients/decorators/paymasterActionsEip7677.ts @@ -1,5 +1,8 @@ -import type { Account, Chain, Client, Transport } from "viem" -import type { EntryPoint } from "../../../../types/entrypoint" +import type { Chain, Client, Transport } from "viem" +import type { + entryPoint06Address, + entryPoint07Address +} from "viem/account-abstraction" import { type GetPaymasterDataParameters, type GetPaymasterDataReturnType, @@ -10,69 +13,73 @@ import { type GetPaymasterStubDataReturnType, getPaymasterStubData } from "../../actions/getPaymasterStubData" -import type { Eip7677RpcSchema } from "../../types/paymaster" export type PaymasterActionsEip7677< - TEntryPoint extends EntryPoint, - TChain extends Chain | undefined = Chain | undefined + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7", + TChain extends Chain | undefined > = { getPaymasterData: < TChainOverride extends Chain | undefined = Chain | undefined >( args: Omit< - GetPaymasterDataParameters, + GetPaymasterDataParameters< + entryPointAddress, + entryPointVersion, + TChain, + TChainOverride + >, "entryPoint" > - ) => Promise> + ) => Promise> getPaymasterStubData: < TChainOverride extends Chain | undefined = Chain | undefined >( args: Omit< - GetPaymasterStubDataParameters, + GetPaymasterStubDataParameters< + entryPointAddress, + entryPointVersion, + TChain, + TChainOverride + >, "entryPoint" > - ) => Promise> + ) => Promise> } const paymasterActionsEip7677 = - (entryPoint: TEntryPoint) => < - TTransport extends Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined - >( - client: Client - ): PaymasterActionsEip7677 => ({ + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7" + >(entryPoint: { + address: entryPointAddress + version: entryPointVersion + }) => + ( + client: Client + ): PaymasterActionsEip7677< + entryPointAddress, + entryPointVersion, + TChain + > => ({ getPaymasterData: (args) => - getPaymasterData( - client as Client< - TTransport, - TChain, - TClientAccount, - Eip7677RpcSchema - >, - { - userOperation: args.userOperation, - context: args.context, - chain: args.chain, - entryPoint - } - ), + getPaymasterData(client, { + userOperation: args.userOperation, + context: args.context, + chain: args.chain, + entryPoint + }), getPaymasterStubData: async (args) => - getPaymasterStubData( - client as Client< - TTransport, - TChain, - TClientAccount, - Eip7677RpcSchema - >, - { - userOperation: args.userOperation, - context: args.context, - chain: args.chain, - entryPoint - } - ) + getPaymasterStubData(client, { + userOperation: args.userOperation, + context: args.context, + chain: args.chain, + entryPoint + }) }) export { paymasterActionsEip7677 } diff --git a/packages/permissionless/experimental/eip7677/types/paymaster.ts b/packages/permissionless/experimental/eip7677/types/paymaster.ts index bf41777c..2f8d97f0 100644 --- a/packages/permissionless/experimental/eip7677/types/paymaster.ts +++ b/packages/permissionless/experimental/eip7677/types/paymaster.ts @@ -1,63 +1,77 @@ -import type { Hex } from "viem" -import type { - EntryPoint, - GetEntryPointVersion -} from "../../../types/entrypoint" -import type { UserOperationWithBigIntAsHex } from "../../../types/userOperation" +import type { Address, Hex, OneOf } from "viem" +import type { UserOperation } from "viem/account-abstraction" -export type GetRpcPaymasterStubDataParameters = [ - userOperation: UserOperationWithBigIntAsHex< - GetEntryPointVersion - >, - entryPoint: entryPoint, +export type GetRpcPaymasterStubDataParameters< + entryPointVersion extends "0.6" | "0.7" +> = [ + userOperation: UserOperation, + entryPoint: Address, chainId: Hex, context?: Record ] -export type GetRpcPaymasterStubDataReturnType = - GetEntryPointVersion extends "v0.6" - ? { - paymasterAndData: Hex - sponsor?: { name: string; icon?: string } - isFinal?: boolean - } - : { - paymaster: Hex - paymasterData: Hex - paymasterVerificationGasLimit?: Hex | null - paymasterPostOpGasLimit?: Hex | null - sponsor?: { name: string; icon?: string } - isFinal?: boolean - } +export type GetRpcPaymasterStubDataReturnType< + entryPointVersion extends "0.6" | "0.7" +> = OneOf< + | (entryPointVersion extends "0.6" + ? { + paymasterAndData: Hex + sponsor?: { name: string; icon?: string } + isFinal?: boolean + } + : never) + | (entryPointVersion extends "0.7" + ? { + paymaster: Hex + paymasterData: Hex + paymasterVerificationGasLimit?: Hex | null + paymasterPostOpGasLimit?: Hex | null + sponsor?: { name: string; icon?: string } + isFinal?: boolean + } + : never) +> -export type GetRpcPaymasterDataParameters = [ - userOperation: UserOperationWithBigIntAsHex< - GetEntryPointVersion - >, - entryPoint: entryPoint, - chainId: Hex, - context?: Record -] +export type GetRpcPaymasterDataParameters< + entryPointVersion extends "0.6" | "0.7" +> = + | [ + userOperation: UserOperation, + entryPoint: Address, + chainId: Hex, + context?: Record + ] + | [ + userOperation: UserOperation, + entryPoint: Address, + chainId: Hex + ] -export type GetRpcPaymasterDataReturnType = - GetEntryPointVersion extends "v0.6" - ? { - paymasterAndData: Hex - } - : { - paymaster: Hex - paymasterData: Hex - } +export type GetRpcPaymasterDataReturnType< + entryPointVersion extends "0.6" | "0.7" +> = OneOf< + | (entryPointVersion extends "0.6" + ? { + paymasterAndData: Hex + } + : never) + | (entryPointVersion extends "0.7" + ? { + paymaster: Hex + paymasterData: Hex + } + : never) +> -export type Eip7677RpcSchema = [ +export type Eip7677RpcSchema = [ { Method: "pm_getPaymasterStubData" - Parameters: GetRpcPaymasterStubDataParameters - ReturnType: GetRpcPaymasterStubDataReturnType + Parameters: GetRpcPaymasterStubDataParameters + ReturnType: GetRpcPaymasterStubDataReturnType }, { Method: "pm_getPaymasterData" - Paremeters: GetRpcPaymasterDataParameters - ReturnType: GetRpcPaymasterDataReturnType + Parameters: GetRpcPaymasterDataParameters + ReturnType: GetRpcPaymasterDataReturnType } ] diff --git a/packages/permissionless/index.ts b/packages/permissionless/index.ts index 00b387fe..1f4ea9b7 100644 --- a/packages/permissionless/index.ts +++ b/packages/permissionless/index.ts @@ -1,88 +1,2 @@ -import type { - EstimateUserOperationGasParameters, - EstimateUserOperationGasReturnType -} from "./actions/bundler/estimateUserOperationGas" -import type { GetUserOperationByHashParameters } from "./actions/bundler/getUserOperationByHash" -import type { GetUserOperationByHashReturnType } from "./actions/bundler/getUserOperationByHash" -import type { - GetUserOperationReceiptParameters, - GetUserOperationReceiptReturnType -} from "./actions/bundler/getUserOperationReceipt" -import type { SendUserOperationParameters } from "./actions/bundler/sendUserOperation" - -import type { GetSenderAddressParams } from "./actions/public/getSenderAddress" -import { getSenderAddress } from "./actions/public/getSenderAddress" - -import { chainId } from "./actions/bundler/chainId" -import { estimateUserOperationGas } from "./actions/bundler/estimateUserOperationGas" -import { getUserOperationByHash } from "./actions/bundler/getUserOperationByHash" -import { getUserOperationReceipt } from "./actions/bundler/getUserOperationReceipt" -import { sendUserOperation } from "./actions/bundler/sendUserOperation" -import { supportedEntryPoints } from "./actions/bundler/supportedEntryPoints" -import { waitForUserOperationReceipt } from "./actions/bundler/waitForUserOperationReceipt" -import { - type WaitForUserOperationReceiptParameters, - WaitForUserOperationReceiptTimeoutError -} from "./actions/bundler/waitForUserOperationReceipt" -import type { GetAccountNonceParams } from "./actions/public/getAccountNonce" -import { getAccountNonce } from "./actions/public/getAccountNonce" -import { - type BundlerClient, - createBundlerClient -} from "./clients/createBundlerClient" -import { createSmartAccountClient } from "./clients/createSmartAccountClient" -import type { - SmartAccountClient, - SmartAccountClientConfig -} from "./clients/createSmartAccountClient" -import type { BundlerActions } from "./clients/decorators/bundler" -import { bundlerActions } from "./clients/decorators/bundler" -import { - type SmartAccountActions, - smartAccountActions -} from "./clients/decorators/smartAccount" - -export type { - SendUserOperationParameters, - EstimateUserOperationGasParameters, - EstimateUserOperationGasReturnType, - GetUserOperationByHashParameters, - GetUserOperationByHashReturnType, - GetUserOperationReceiptParameters, - GetUserOperationReceiptReturnType, - GetSenderAddressParams, - GetAccountNonceParams, - BundlerClient, - BundlerActions, - WaitForUserOperationReceiptParameters, - SmartAccountClient, - SmartAccountClientConfig, - SmartAccountActions -} - -export { - sendUserOperation, - estimateUserOperationGas, - supportedEntryPoints, - chainId, - getUserOperationByHash, - getUserOperationReceipt, - getSenderAddress, - getAccountNonce, - waitForUserOperationReceipt, - createBundlerClient, - bundlerActions, - WaitForUserOperationReceiptTimeoutError, - createSmartAccountClient, - smartAccountActions -} -import type { UserOperation } from "./types/userOperation" - -export type { UserOperation } - -import type { GetUserOperationHashParams } from "./utils/getUserOperationHash" -import { getUserOperationHash } from "./utils/getUserOperationHash" - -export { getUserOperationHash, type GetUserOperationHashParams } export * from "./utils/" export * from "./errors/" diff --git a/packages/permissionless/types/index.ts b/packages/permissionless/types/index.ts index 78b80d52..f7059c85 100644 --- a/packages/permissionless/types/index.ts +++ b/packages/permissionless/types/index.ts @@ -1,4 +1,3 @@ -import type { Account, Chain, Client, Transport } from "viem" import type { SmartAccount, SmartAccountImplementation diff --git a/packages/permissionless/utils/encode7579Calls.ts b/packages/permissionless/utils/encode7579Calls.ts index bc551175..4d6f4fa4 100644 --- a/packages/permissionless/utils/encode7579Calls.ts +++ b/packages/permissionless/utils/encode7579Calls.ts @@ -25,6 +25,12 @@ export function encode7579Calls({ mode, callData }: EncodeCallDataParams): Hex { + if (callData.length > 1 && mode?.type !== "batchcall") { + throw new Error( + `mode ${JSON.stringify(mode)} does not supported for batchcall calldata` + ) + } + const executeAbi = [ { type: "function", diff --git a/packages/permissionless/utils/getEntryPointVersion.test.ts b/packages/permissionless/utils/getEntryPointVersion.test.ts deleted file mode 100644 index 0a642c5e..00000000 --- a/packages/permissionless/utils/getEntryPointVersion.test.ts +++ /dev/null @@ -1,52 +0,0 @@ -import { describe, expect, test } from "vitest" -import { - ENTRYPOINT_ADDRESS_V06, - ENTRYPOINT_ADDRESS_V07, - getEntryPointVersion, - isUserOperationVersion06, - isUserOperationVersion07 -} from "./getEntryPointVersion" - -describe("getEntryPointVersion", () => { - describe("getEntryPointVersion", () => { - test('should return "v0.6" for ENTRYPOINT_ADDRESS_V06', () => { - expect(getEntryPointVersion(ENTRYPOINT_ADDRESS_V06)).toBe("v0.6") - }) - - test('should return "v0.7" for ENTRYPOINT_ADDRESS_V07', () => { - expect(getEntryPointVersion(ENTRYPOINT_ADDRESS_V07)).toBe("v0.7") - }) - }) - - describe("isUserOperationVersion06", () => { - test('should return true for ENTRYPOINT_ADDRESS_V06 and UserOperation<"v0.6">', () => { - const userOperation = {} as any // mock UserOperation<"v0.6"> - expect( - isUserOperationVersion06(ENTRYPOINT_ADDRESS_V06, userOperation) - ).toBe(true) - }) - - test('should return false for ENTRYPOINT_ADDRESS_V07 and UserOperation<"v0.6">', () => { - const userOperation = {} as any // mock UserOperation<"v0.6"> - expect( - isUserOperationVersion06(ENTRYPOINT_ADDRESS_V07, userOperation) - ).toBe(false) - }) - }) - - describe("isUserOperationVersion07", () => { - test('should return false for ENTRYPOINT_ADDRESS_V06 and UserOperation<"v0.6">', () => { - const userOperation = {} as any // mock UserOperation<"v0.6"> - expect( - isUserOperationVersion07(ENTRYPOINT_ADDRESS_V06, userOperation) - ).toBe(false) - }) - - test('should return true for ENTRYPOINT_ADDRESS_V07 and UserOperation<"v0.7">', () => { - const userOperation = {} as any // mock UserOperation<"v0.7"> - expect( - isUserOperationVersion07(ENTRYPOINT_ADDRESS_V07, userOperation) - ).toBe(true) - }) - }) -}) diff --git a/packages/permissionless/utils/getRequiredPrefund.test.ts b/packages/permissionless/utils/getRequiredPrefund.test.ts index d89d872f..2ce46daa 100644 --- a/packages/permissionless/utils/getRequiredPrefund.test.ts +++ b/packages/permissionless/utils/getRequiredPrefund.test.ts @@ -1,9 +1,5 @@ +import type { UserOperation } from "viem/account-abstraction" import { describe, expect, test } from "vitest" -import type { UserOperation } from "../types/userOperation" -import { - ENTRYPOINT_ADDRESS_V06, - ENTRYPOINT_ADDRESS_V07 -} from "./getEntryPointVersion" import { getRequiredPrefund } from "./getRequiredPrefund" describe("getRequiredPrefund", () => { @@ -17,8 +13,8 @@ describe("getRequiredPrefund", () => { paymasterAndData: "0x" } const result = getRequiredPrefund({ - userOperation: userOperation as UserOperation<"v0.6">, - entryPoint: ENTRYPOINT_ADDRESS_V06 + userOperation: userOperation as UserOperation<"0.6">, + entryPointVersion: "0.6" }) const expectedGas = BigInt(1000) + BigInt(2000) * BigInt(1) + BigInt(500) @@ -35,8 +31,8 @@ describe("getRequiredPrefund", () => { paymasterAndData: "0x1234" } const result = getRequiredPrefund({ - userOperation: userOperation as UserOperation<"v0.6">, - entryPoint: ENTRYPOINT_ADDRESS_V06 + userOperation: userOperation as UserOperation<"0.6">, + entryPointVersion: "0.6" }) const multiplier = BigInt(3) const expectedGas = @@ -56,8 +52,8 @@ describe("getRequiredPrefund", () => { paymasterVerificationGasLimit: BigInt(200) } const result = getRequiredPrefund({ - userOperation: userOperation as UserOperation<"v0.7">, - entryPoint: ENTRYPOINT_ADDRESS_V07 + userOperation: userOperation as UserOperation<"0.7">, + entryPointVersion: "0.7" }) const multiplier = BigInt(3) const verificationGasLimit = From 3bf80e06f3b7163a83351048f865e285c63ff8dd Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Wed, 21 Aug 2024 22:19:56 +0100 Subject: [PATCH 07/51] Add paymentToken, payment & paymentReceiver in safe account --- .../accounts/safe/toSafeSmartAccount.ts | 54 +++++++++++++++---- 1 file changed, 45 insertions(+), 9 deletions(-) diff --git a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts index e6557c0a..7efb3f89 100644 --- a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts +++ b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts @@ -613,7 +613,10 @@ const getInitializerCode = async ({ fallbacks = [], hooks = [], attesters = [], - attestersThreshold = 0 + attestersThreshold = 0, + paymentToken = zeroAddress, + payment = BigInt(0), + paymentReceiver = zeroAddress }: { owner: Address safeSingletonAddress: Address @@ -636,6 +639,9 @@ const getInitializerCode = async ({ hooks?: { address: Address; context: Address }[] attesters?: Address[] attestersThreshold?: number + paymentToken?: Address + payment?: bigint + paymentReceiver?: Address }) => { if (erc7579LaunchpadAddress) { const initData = get7579LaunchPadInitData({ @@ -747,9 +753,9 @@ const getInitializerCode = async ({ multiSendAddress, multiSendCallData, safe4337ModuleAddress, - zeroAddress, - BigInt(0), - zeroAddress + paymentToken, + payment, + paymentReceiver ] }) } @@ -787,6 +793,9 @@ const getAccountInitCode = async ({ safeSingletonAddress, erc7579LaunchpadAddress, multiSendAddress, + paymentToken, + payment, + paymentReceiver, saltNonce = BigInt(0), setupTransactions = [], safeModules = [], @@ -819,6 +828,9 @@ const getAccountInitCode = async ({ hooks?: { address: Address; context: Address }[] attesters?: Address[] attestersThreshold?: number + paymentToken?: Address + payment?: bigint + paymentReceiver?: Address }): Promise => { if (!owner) { throw new Error("Owner account not found") @@ -838,7 +850,10 @@ const getAccountInitCode = async ({ fallbacks, hooks, attesters, - attestersThreshold + attestersThreshold, + paymentToken, + payment, + paymentReceiver }) const initCodeCallData = encodeFunctionData({ @@ -867,6 +882,9 @@ const getAccountAddress = async < safeSingletonAddress, multiSendAddress, erc7579LaunchpadAddress, + paymentToken, + payment, + paymentReceiver, setupTransactions = [], safeModules = [], saltNonce = BigInt(0), @@ -901,6 +919,9 @@ const getAccountAddress = async < hooks?: { address: Address; context: Address }[] attesters?: Address[] attestersThreshold?: number + paymentToken?: Address + payment?: bigint + paymentReceiver?: Address }): Promise
=> { const proxyCreationCode = await readContract(client, { abi: proxyCreationCodeAbi, @@ -922,7 +943,10 @@ const getAccountAddress = async < fallbacks, hooks, attesters, - attestersThreshold + attestersThreshold, + paymentToken, + payment, + paymentReceiver }) const deploymentCode = encodePacked( @@ -1054,6 +1078,9 @@ export type ToSafeSmartAccountParameters< validUntil?: number validAfter?: number nonceKey?: bigint + paymentToken?: Address + payment?: bigint + paymentReceiver?: Address } & GetErc7579Params function isErc7579Args< @@ -1130,7 +1157,10 @@ export async function toSafeSmartAccount< saltNonce = BigInt(0), validUntil = 0, validAfter = 0, - nonceKey + nonceKey, + paymentToken, + payment, + paymentReceiver } = parameters const entryPoint = { @@ -1215,7 +1245,10 @@ export async function toSafeSmartAccount< fallbacks, hooks, attesters, - attestersThreshold + attestersThreshold, + paymentToken, + payment, + paymentReceiver })) return accountAddress } @@ -1344,7 +1377,10 @@ export async function toSafeSmartAccount< fallbacks, hooks, attesters, - attestersThreshold + attestersThreshold, + paymentToken, + payment, + paymentReceiver }) } }, From 8462d6809f372d7651890d81fda5ff71410995a7 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Thu, 22 Aug 2024 01:43:39 +0100 Subject: [PATCH 08/51] lint --- .../experimental/eip7677/actions/getPaymasterData.test.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/permissionless/experimental/eip7677/actions/getPaymasterData.test.ts b/packages/permissionless/experimental/eip7677/actions/getPaymasterData.test.ts index 93ec24be..ba1b0f40 100644 --- a/packages/permissionless/experimental/eip7677/actions/getPaymasterData.test.ts +++ b/packages/permissionless/experimental/eip7677/actions/getPaymasterData.test.ts @@ -78,7 +78,7 @@ describe("EIP-7677 getPaymasterData", () => { }) testWithRpc("getPaymasterData_V07", async ({ rpc }) => { - const { altoRpc, paymasterRpc } = rpc + const { paymasterRpc } = rpc const simpleAccountClient = getBundlerClient({ account: await getSimpleAccountClient({ From aa874b195df1fe3f24c34995930aaadbb504ec75 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Thu, 22 Aug 2024 02:55:18 +0100 Subject: [PATCH 09/51] Add smart account client --- .../actions/erc7579/installModule.ts | 3 +- .../actions/erc7579/installModules.ts | 3 +- .../actions/erc7579/isModuleInstalled.ts | 3 +- .../actions/erc7579/supportsExecutionMode.ts | 2 +- .../actions/erc7579/supportsModule.ts | 2 +- .../actions/erc7579/uninstallModule.ts | 2 +- .../actions/erc7579/uninstallModules.ts | 2 +- .../smartAccount/sendTransaction.test.ts | 84 +++++ .../actions/smartAccount/sendTransaction.ts | 117 +++++++ .../actions/smartAccount/signMessage.test.ts | 116 +++++++ .../actions/smartAccount/signMessage.ts | 73 ++++ .../smartAccount/signTypedData.test.ts | 143 ++++++++ .../actions/smartAccount/signTypedData.ts | 157 +++++++++ .../actions/smartAccount/writeContract.ts | 66 ++++ .../clients/createSmartAccountClient.ts | 149 +++++++++ .../clients/decorators/smartAccount.ts | 316 ++++++++++++++++++ packages/permissionless/utils/index.ts | 5 - 17 files changed, 1228 insertions(+), 15 deletions(-) create mode 100644 packages/permissionless/actions/smartAccount/sendTransaction.test.ts create mode 100644 packages/permissionless/actions/smartAccount/sendTransaction.ts create mode 100644 packages/permissionless/actions/smartAccount/signMessage.test.ts create mode 100644 packages/permissionless/actions/smartAccount/signMessage.ts create mode 100644 packages/permissionless/actions/smartAccount/signTypedData.test.ts create mode 100644 packages/permissionless/actions/smartAccount/signTypedData.ts create mode 100644 packages/permissionless/actions/smartAccount/writeContract.ts create mode 100644 packages/permissionless/clients/createSmartAccountClient.ts create mode 100644 packages/permissionless/clients/decorators/smartAccount.ts diff --git a/packages/permissionless/actions/erc7579/installModule.ts b/packages/permissionless/actions/erc7579/installModule.ts index 211e025d..d2b95441 100644 --- a/packages/permissionless/actions/erc7579/installModule.ts +++ b/packages/permissionless/actions/erc7579/installModule.ts @@ -10,9 +10,8 @@ import { type SmartAccount, sendUserOperation } from "viem/account-abstraction" -import { getAction } from "viem/utils" +import { getAction, parseAccount } from "viem/utils" import { AccountNotFoundError } from "../../errors" -import { parseAccount } from "../../utils/" import { type ModuleType, parseModuleTypeId } from "./supportsModule" export type InstallModuleParameters< diff --git a/packages/permissionless/actions/erc7579/installModules.ts b/packages/permissionless/actions/erc7579/installModules.ts index ec9e97f1..ff628bd0 100644 --- a/packages/permissionless/actions/erc7579/installModules.ts +++ b/packages/permissionless/actions/erc7579/installModules.ts @@ -12,9 +12,8 @@ import { type SmartAccount, sendUserOperation } from "viem/account-abstraction" -import { getAction } from "viem/utils" +import { getAction, parseAccount } from "viem/utils" import { AccountNotFoundError } from "../../errors" -import { parseAccount } from "../../utils/" import { type ModuleType, parseModuleTypeId } from "./supportsModule" export type InstallModulesParameters< diff --git a/packages/permissionless/actions/erc7579/isModuleInstalled.ts b/packages/permissionless/actions/erc7579/isModuleInstalled.ts index 3b2d8b1b..6454fa0a 100644 --- a/packages/permissionless/actions/erc7579/isModuleInstalled.ts +++ b/packages/permissionless/actions/erc7579/isModuleInstalled.ts @@ -14,9 +14,8 @@ import type { SmartAccount } from "viem/account-abstraction" import { call, readContract } from "viem/actions" -import { getAction } from "viem/utils" +import { getAction, parseAccount } from "viem/utils" import { AccountNotFoundError } from "../../errors" -import { parseAccount } from "../../utils/" import { type ModuleType, parseModuleTypeId } from "./supportsModule" export type IsModuleInstalledParameters< diff --git a/packages/permissionless/actions/erc7579/supportsExecutionMode.ts b/packages/permissionless/actions/erc7579/supportsExecutionMode.ts index 145efa80..e08720e1 100644 --- a/packages/permissionless/actions/erc7579/supportsExecutionMode.ts +++ b/packages/permissionless/actions/erc7579/supportsExecutionMode.ts @@ -16,8 +16,8 @@ import type { } from "viem/account-abstraction" import { call, readContract } from "viem/actions" import { getAction } from "viem/utils" +import { parseAccount } from "viem/utils" import { AccountNotFoundError } from "../../errors" -import { parseAccount } from "../../utils/" export type CallType = "call" | "delegatecall" | "batchcall" diff --git a/packages/permissionless/actions/erc7579/supportsModule.ts b/packages/permissionless/actions/erc7579/supportsModule.ts index 56b55130..b2a8951f 100644 --- a/packages/permissionless/actions/erc7579/supportsModule.ts +++ b/packages/permissionless/actions/erc7579/supportsModule.ts @@ -12,8 +12,8 @@ import type { } from "viem/account-abstraction" import { call, readContract } from "viem/actions" import { getAction } from "viem/utils" +import { parseAccount } from "viem/utils" import { AccountNotFoundError } from "../../errors" -import { parseAccount } from "../../utils/" export type ModuleType = "validator" | "executor" | "fallback" | "hook" diff --git a/packages/permissionless/actions/erc7579/uninstallModule.ts b/packages/permissionless/actions/erc7579/uninstallModule.ts index 36055b78..d4032607 100644 --- a/packages/permissionless/actions/erc7579/uninstallModule.ts +++ b/packages/permissionless/actions/erc7579/uninstallModule.ts @@ -13,8 +13,8 @@ import { sendUserOperation } from "viem/account-abstraction" import { getAction } from "viem/utils" +import { parseAccount } from "viem/utils" import { AccountNotFoundError } from "../../errors" -import { parseAccount } from "../../utils/" import { type ModuleType, parseModuleTypeId } from "./supportsModule" export type UninstallModuleParameters< diff --git a/packages/permissionless/actions/erc7579/uninstallModules.ts b/packages/permissionless/actions/erc7579/uninstallModules.ts index 9eb42550..5e25736a 100644 --- a/packages/permissionless/actions/erc7579/uninstallModules.ts +++ b/packages/permissionless/actions/erc7579/uninstallModules.ts @@ -13,8 +13,8 @@ import { sendUserOperation } from "viem/account-abstraction" import { getAction } from "viem/utils" +import { parseAccount } from "viem/utils" import { AccountNotFoundError } from "../../errors" -import { parseAccount } from "../../utils/" import { type ModuleType, parseModuleTypeId } from "./supportsModule" export type UninstallModulesParameters< diff --git a/packages/permissionless/actions/smartAccount/sendTransaction.test.ts b/packages/permissionless/actions/smartAccount/sendTransaction.test.ts new file mode 100644 index 00000000..c8bb79b7 --- /dev/null +++ b/packages/permissionless/actions/smartAccount/sendTransaction.test.ts @@ -0,0 +1,84 @@ +import { zeroAddress } from "viem" +import { foundry } from "viem/chains" +import { describe, expect } from "vitest" +import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" +import { + getCoreSmartAccounts, + getPublicClient +} from "../../../permissionless-test/src/utils" +import { sendTransaction } from "./sendTransaction" + +describe.each(getCoreSmartAccounts())( + "sendTransaction $name", + ({ + getSmartAccountClient, + supportsEntryPointV06, + supportsEntryPointV07 + }) => { + testWithRpc.skipIf(!supportsEntryPointV06)( + "sendTransaction_v06", + async ({ rpc }) => { + const { anvilRpc } = rpc + + const smartClient = await getSmartAccountClient({ + entryPoint: { + version: "0.6" + }, + ...rpc + }) + + const transactionHash = await sendTransaction(smartClient, { + chain: foundry, + to: zeroAddress, + data: "0x", + value: 0n + }) + + expect(transactionHash).toBeTruthy() + + const publicClient = getPublicClient(anvilRpc) + + const receipt = await publicClient.getTransactionReceipt({ + hash: transactionHash + }) + + expect(receipt).toBeTruthy() + expect(receipt.transactionHash).toBe(transactionHash) + expect(receipt.status).toBe("success") + } + ) + + testWithRpc.skipIf(!supportsEntryPointV07)( + "sendTransaction_v07", + async ({ rpc }) => { + const { anvilRpc } = rpc + + const smartClient = await getSmartAccountClient({ + entryPoint: { + version: "0.7" + }, + ...rpc + }) + + const transactionHash = await sendTransaction(smartClient, { + chain: foundry, + to: zeroAddress, + data: "0x", + value: 0n + }) + + expect(transactionHash).toBeTruthy() + + const publicClient = getPublicClient(anvilRpc) + + const receipt = await publicClient.getTransactionReceipt({ + hash: transactionHash + }) + + expect(receipt).toBeTruthy() + expect(receipt.transactionHash).toBe(transactionHash) + expect(receipt.status).toBe("success") + } + ) + } +) diff --git a/packages/permissionless/actions/smartAccount/sendTransaction.ts b/packages/permissionless/actions/smartAccount/sendTransaction.ts new file mode 100644 index 00000000..f9267278 --- /dev/null +++ b/packages/permissionless/actions/smartAccount/sendTransaction.ts @@ -0,0 +1,117 @@ +import type { + Chain, + Client, + Hash, + SendTransactionParameters, + Transport +} from "viem" +import { + type SmartAccount, + sendUserOperation, + waitForUserOperationReceipt +} from "viem/account-abstraction" +import { getAction, parseAccount } from "viem/utils" +import { AccountNotFoundError } from "../../errors" + +/** + * Creates, signs, and sends a new transaction to the network. + * This function also allows you to sponsor this transaction if sender is a smartAccount + * + * - Docs: https://viem.sh/docs/actions/wallet/sendTransaction.html + * - Examples: https://stackblitz.com/github/wagmi-dev/viem/tree/main/examples/transactions/sending-transactions + * - JSON-RPC Methods: + * - JSON-RPC Accounts: [`eth_sendTransaction`](https://ethereum.org/en/developers/docs/apis/json-rpc/#eth_sendtransaction) + * - Local Accounts: [`eth_sendRawTransaction`](https://ethereum.org/en/developers/docs/apis/json-rpc/#eth_sendrawtransaction) + * + * @param client - Client to use + * @param parameters - {@link SendTransactionParameters} + * @returns The [Transaction](https://viem.sh/docs/glossary/terms.html#transaction) hash. + * + * @example + * import { createWalletClient, custom } from 'viem' + * import { mainnet } from 'viem/chains' + * import { sendTransaction } from 'viem/wallet' + * + * const client = createWalletClient({ + * chain: mainnet, + * transport: custom(window.ethereum), + * }) + * const hash = await sendTransaction(client, { + * account: '0xA0Cf798816D4b9b9866b5330EEa46a18382f251e', + * to: '0x70997970c51812dc3a010c7d01b50e0d17dc79c8', + * value: 1000000000000000000n, + * }) + * + * @example + * // Account Hoisting + * import { createWalletClient, http } from 'viem' + * import { privateKeyToAccount } from 'viem/accounts' + * import { mainnet } from 'viem/chains' + * import { sendTransaction } from 'viem/wallet' + * + * const client = createWalletClient({ + * account: privateKeyToAccount('0x…'), + * chain: mainnet, + * transport: http(), + * }) + * const hash = await sendTransaction(client, { + * to: '0x70997970c51812dc3a010c7d01b50e0d17dc79c8', + * value: 1000000000000000000n, + * }) + */ +export async function sendTransaction< + TChain extends Chain | undefined, + TAccount extends SmartAccount | undefined, + TChainOverride extends Chain | undefined = Chain | undefined +>( + client: Client, + args: SendTransactionParameters +): Promise { + const { + account: account_ = client.account, + data, + maxFeePerGas, + maxPriorityFeePerGas, + to, + value, + nonce + } = args + + if (!account_) { + throw new AccountNotFoundError({ + docsPath: "/docs/actions/wallet/sendTransaction" + }) + } + + const account = parseAccount(account_) as SmartAccount + + if (!to) throw new Error("Missing to address") + + const userOpHash = await getAction( + client, + sendUserOperation, + "sendUserOperation" + )({ + calls: [ + { + to, + value: value || BigInt(0), + data: data || "0x" + } + ], + account, + maxFeePerGas, + maxPriorityFeePerGas, + nonce: nonce ? BigInt(nonce) : undefined + }) + + const userOperationReceipt = await getAction( + client, + waitForUserOperationReceipt, + "waitForUserOperationReceipt" + )({ + hash: userOpHash + }) + + return userOperationReceipt?.receipt.transactionHash +} diff --git a/packages/permissionless/actions/smartAccount/signMessage.test.ts b/packages/permissionless/actions/smartAccount/signMessage.test.ts new file mode 100644 index 00000000..f3ceef0f --- /dev/null +++ b/packages/permissionless/actions/smartAccount/signMessage.test.ts @@ -0,0 +1,116 @@ +import { describe, expect } from "vitest" +import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" +import { + getCoreSmartAccounts, + getPublicClient +} from "../../../permissionless-test/src/utils" +import { signMessage } from "./signMessage" + +describe.each(getCoreSmartAccounts())( + "signMessage $name", + ({ + getSmartAccountClient, + isEip1271Compliant, + supportsEntryPointV06, + supportsEntryPointV07, + name + }) => { + testWithRpc.skipIf(isEip1271Compliant || !supportsEntryPointV06)( + "not isEip1271Compliant_v06", + async ({ rpc }) => { + const smartClient = await getSmartAccountClient({ + entryPoint: { + version: "0.6" + }, + ...rpc + }) + + await expect(async () => + signMessage(smartClient, { + message: "slowly and steadily burning the private keys" + }) + ).rejects.toThrow() + } + ) + + testWithRpc.skipIf(!isEip1271Compliant || !supportsEntryPointV06)( + "isEip1271Compliant_v06", + async ({ rpc }) => { + const { anvilRpc } = rpc + + const smartClient = await getSmartAccountClient({ + entryPoint: { + version: "0.6" + }, + ...rpc + }) + + const signature = await signMessage(smartClient, { + message: "slowly and steadily burning the private keys" + }) + + const publicClient = getPublicClient(anvilRpc) + + const isVerified = await publicClient.verifyMessage({ + address: smartClient.account.address, + message: "slowly and steadily burning the private keys", + signature + }) + + expect(isVerified).toBeTruthy() + } + ) + + testWithRpc.skipIf(isEip1271Compliant || !supportsEntryPointV07)( + "not isEip1271Compliant_v07", + async ({ rpc }) => { + const smartClient = await getSmartAccountClient({ + entryPoint: { + version: "0.7" + }, + ...rpc + }) + + await expect(async () => + signMessage(smartClient, { + message: "slowly and steadily burning the private keys" + }) + ).rejects.toThrow() + } + ) + + testWithRpc.skipIf(!isEip1271Compliant || !supportsEntryPointV07)( + "isEip1271Compliant_v07", + async ({ rpc }) => { + const { anvilRpc } = rpc + + const smartClient = await getSmartAccountClient({ + entryPoint: { + version: "0.7" + }, + ...rpc + }) + + const signature = await signMessage(smartClient, { + message: "slowly and steadily burning the private keys" + }) + + const publicClient = getPublicClient(anvilRpc) + + if (name === "Safe 7579") { + // Due to 7579 launchpad, we can't verify the signature as of now. + // Awaiting for the fix + // return + } + + const isVerified = await publicClient.verifyMessage({ + address: smartClient.account.address, + message: "slowly and steadily burning the private keys", + signature + }) + + expect(isVerified).toBeTruthy() + } + ) + } +) diff --git a/packages/permissionless/actions/smartAccount/signMessage.ts b/packages/permissionless/actions/smartAccount/signMessage.ts new file mode 100644 index 00000000..2d816446 --- /dev/null +++ b/packages/permissionless/actions/smartAccount/signMessage.ts @@ -0,0 +1,73 @@ +import type { + Chain, + Client, + SignMessageParameters, + SignMessageReturnType, + Transport +} from "viem" +import type { SmartAccount } from "viem/account-abstraction" +import { parseAccount } from "viem/utils" +import { AccountNotFoundError } from "../../errors" + +/** + * Calculates an Ethereum-specific signature in [EIP-191 format](https://eips.ethereum.org/EIPS/eip-191): `keccak256("\x19Ethereum Signed Message:\n" + len(message) + message))`. + * + * - Docs: https://viem.sh/docs/actions/wallet/signMessage.html + * - JSON-RPC Methods: + * - JSON-RPC Accounts: [`personal_sign`](https://docs.metamask.io/guide/signing-data.html#personal-sign) + * - Local Accounts: Signs locally. No JSON-RPC request. + * + * With the calculated signature, you can: + * - use [`verifyMessage`](https://viem.sh/docs/utilities/verifyMessage.html) to verify the signature, + * - use [`recoverMessageAddress`](https://viem.sh/docs/utilities/recoverMessageAddress.html) to recover the signing address from a signature. + * + * @param client - Client to use + * @param parameters - {@link SignMessageParameters} + * @returns The signed message. {@link SignMessageReturnType} + * + * @example + * import { createWalletClient, custom } from 'viem' + * import { mainnet } from 'viem/chains' + * import { signMessage } from 'viem/wallet' + * + * const client = createWalletClient({ + * chain: mainnet, + * transport: custom(window.ethereum), + * }) + * const signature = await signMessage(client, { + * account: '0xA0Cf798816D4b9b9866b5330EEa46a18382f251e', + * message: 'hello world', + * }) + * + * @example + * // Account Hoisting + * import { createWalletClient, custom } from 'viem' + * import { privateKeyToAccount } from 'viem/accounts' + * import { mainnet } from 'viem/chains' + * import { signMessage } from 'viem/wallet' + * + * const client = createWalletClient({ + * account: privateKeyToAccount('0x…'), + * chain: mainnet, + * transport: custom(window.ethereum), + * }) + * const signature = await signMessage(client, { + * message: 'hello world', + * }) + */ +export async function signMessage( + client: Client, + { + account: account_ = client.account, + message + }: SignMessageParameters +): Promise { + if (!account_) + throw new AccountNotFoundError({ + docsPath: "/docs/actions/wallet/signMessage" + }) + + const account = parseAccount(account_) as SmartAccount + + return account.signMessage({ message }) +} diff --git a/packages/permissionless/actions/smartAccount/signTypedData.test.ts b/packages/permissionless/actions/smartAccount/signTypedData.test.ts new file mode 100644 index 00000000..a6c7fc8f --- /dev/null +++ b/packages/permissionless/actions/smartAccount/signTypedData.test.ts @@ -0,0 +1,143 @@ +import { getAddress } from "viem" +import { describe, expect } from "vitest" +import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" +import { + getCoreSmartAccounts, + getPublicClient +} from "../../../permissionless-test/src/utils" +import { signTypedData } from "./signTypedData" + +const typedData = { + domain: { + name: "Ether Mail", + version: "1", + chainId: 1, + verifyingContract: getAddress( + "0xCcCCccccCCCCcCCCCCCcCcCccCcCCCcCcccccccC" + ) + }, + types: { + Person: [ + { name: "name", type: "string" }, + { name: "wallet", type: "address" } + ], + Mail: [ + { name: "from", type: "Person" }, + { name: "to", type: "Person" }, + { name: "contents", type: "string" } + ] + }, + primaryType: "Mail" as const, + message: { + from: { + name: "Cow", + wallet: "0xCD2a3d9F938E13CD947Ec05AbC7FE734Df8DD826" + }, + to: { + name: "Bob", + wallet: "0xbBbBBBBbbBBBbbbBbbBbbbbBBbBbbbbBbBbbBBbB" + }, + contents: "Hello, Bob!" + } +} + +describe.each(getCoreSmartAccounts())( + "signTypedData $name", + ({ + getSmartAccountClient, + isEip1271Compliant, + supportsEntryPointV06, + supportsEntryPointV07, + name + }) => { + testWithRpc.skipIf(isEip1271Compliant || !supportsEntryPointV06)( + "not isEip1271Compliant_v06", + async ({ rpc }) => { + const smartClient = await getSmartAccountClient({ + entryPoint: { + version: "0.6" + }, + ...rpc + }) + + await expect(async () => + signTypedData(smartClient, typedData) + ).rejects.toThrow() + } + ) + + testWithRpc.skipIf(!isEip1271Compliant || !supportsEntryPointV06)( + "isEip1271Compliant_v06", + async ({ rpc }) => { + const { anvilRpc } = rpc + + const smartClient = await getSmartAccountClient({ + entryPoint: { + version: "0.6" + }, + ...rpc + }) + + const signature = await signTypedData(smartClient, typedData) + + const publicClient = getPublicClient(anvilRpc) + + const isVerified = await publicClient.verifyTypedData({ + ...typedData, + address: smartClient.account.address, + signature + }) + + expect(isVerified).toBeTruthy() + } + ) + + testWithRpc.skipIf(isEip1271Compliant || !supportsEntryPointV07)( + "not isEip1271Compliant_v07", + async ({ rpc }) => { + const smartClient = await getSmartAccountClient({ + entryPoint: { + version: "0.7" + }, + ...rpc + }) + + await expect(async () => + signTypedData(smartClient, typedData) + ).rejects.toThrow() + } + ) + + testWithRpc.skipIf(!isEip1271Compliant || !supportsEntryPointV07)( + "isEip1271Compliant_v07", + async ({ rpc }) => { + const { anvilRpc } = rpc + + const smartClient = await getSmartAccountClient({ + entryPoint: { + version: "0.7" + }, + ...rpc + }) + + const signature = await signTypedData(smartClient, typedData) + + const publicClient = getPublicClient(anvilRpc) + + if (name === "Safe 7579") { + // Due to 7579 launchpad, we can't verify the signature as of now. + // Awaiting for the fix + return + } + + const isVerified = await publicClient.verifyTypedData({ + ...typedData, + address: smartClient.account.address, + signature + }) + + expect(isVerified).toBeTruthy() + } + ) + } +) diff --git a/packages/permissionless/actions/smartAccount/signTypedData.ts b/packages/permissionless/actions/smartAccount/signTypedData.ts new file mode 100644 index 00000000..398f7654 --- /dev/null +++ b/packages/permissionless/actions/smartAccount/signTypedData.ts @@ -0,0 +1,157 @@ +import { + type Chain, + type Client, + type SignTypedDataParameters, + type SignTypedDataReturnType, + type Transport, + type TypedData, + type TypedDataDefinition, + type TypedDataDomain, + getTypesForEIP712Domain, + validateTypedData +} from "viem" +import type { SmartAccount } from "viem/account-abstraction" +import { parseAccount } from "viem/utils" +import { AccountNotFoundError } from "../../errors" + +/** + * Signs typed data and calculates an Ethereum-specific signature in [https://eips.ethereum.org/EIPS/eip-712](https://eips.ethereum.org/EIPS/eip-712): `sign(keccak256("\x19\x01" ‖ domainSeparator ‖ hashStruct(message)))` + * + * - Docs: https://viem.sh/docs/actions/wallet/signTypedData.html + * - JSON-RPC Methods: + * - JSON-RPC Accounts: [`eth_signTypedData_v4`](https://docs.metamask.io/guide/signing-data.html#signtypeddata-v4) + * - Local Accounts: Signs locally. No JSON-RPC request. + * + * @param client - Client to use + * @param parameters - {@link SignTypedDataParameters} + * @returns The signed data. {@link SignTypedDataReturnType} + * + * @example + * import { createWalletClient, custom } from 'viem' + * import { mainnet } from 'viem/chains' + * import { signTypedData } from 'viem/wallet' + * + * const client = createWalletClient({ + * chain: mainnet, + * transport: custom(window.ethereum), + * }) + * const signature = await signTypedData(client, { + * account: '0xA0Cf798816D4b9b9866b5330EEa46a18382f251e', + * domain: { + * name: 'Ether Mail', + * version: '1', + * chainId: 1, + * verifyingContract: '0xCcCCccccCCCCcCCCCCCcCcCccCcCCCcCcccccccC', + * }, + * types: { + * Person: [ + * { name: 'name', type: 'string' }, + * { name: 'wallet', type: 'address' }, + * ], + * Mail: [ + * { name: 'from', type: 'Person' }, + * { name: 'to', type: 'Person' }, + * { name: 'contents', type: 'string' }, + * ], + * }, + * primaryType: 'Mail', + * message: { + * from: { + * name: 'Cow', + * wallet: '0xCD2a3d9F938E13CD947Ec05AbC7FE734Df8DD826', + * }, + * to: { + * name: 'Bob', + * wallet: '0xbBbBBBBbbBBBbbbBbbBbbbbBBbBbbbbBbBbbBBbB', + * }, + * contents: 'Hello, Bob!', + * }, + * }) + * + * @example + * // Account Hoisting + * import { createWalletClient, http } from 'viem' + * import { privateKeyToAccount } from 'viem/accounts' + * import { mainnet } from 'viem/chains' + * import { signTypedData } from 'viem/wallet' + * + * const client = createWalletClient({ + * account: privateKeyToAccount('0x…'), + * chain: mainnet, + * transport: http(), + * }) + * const signature = await signTypedData(client, { + * domain: { + * name: 'Ether Mail', + * version: '1', + * chainId: 1, + * verifyingContract: '0xCcCCccccCCCCcCCCCCCcCcCccCcCCCcCcccccccC', + * }, + * types: { + * Person: [ + * { name: 'name', type: 'string' }, + * { name: 'wallet', type: 'address' }, + * ], + * Mail: [ + * { name: 'from', type: 'Person' }, + * { name: 'to', type: 'Person' }, + * { name: 'contents', type: 'string' }, + * ], + * }, + * primaryType: 'Mail', + * message: { + * from: { + * name: 'Cow', + * wallet: '0xCD2a3d9F938E13CD947Ec05AbC7FE734Df8DD826', + * }, + * to: { + * name: 'Bob', + * wallet: '0xbBbBBBBbbBBBbbbBbbBbbbbBBbBbbbbBbBbbBBbB', + * }, + * contents: 'Hello, Bob!', + * }, + * }) + */ +export async function signTypedData< + const TTypedData extends TypedData | { [key: string]: unknown }, + TPrimaryType extends string, + TAccount extends SmartAccount | undefined = SmartAccount | undefined +>( + client: Client, + { + account: account_ = client.account, + domain, + message, + primaryType, + types: types_ + }: SignTypedDataParameters +): Promise { + if (!account_) { + throw new AccountNotFoundError({ + docsPath: "/docs/actions/wallet/signMessage" + }) + } + + const account = parseAccount(account_) as SmartAccount + + const types = { + EIP712Domain: getTypesForEIP712Domain({ domain } as { + domain: TypedDataDomain + }), + ...(types_ as TTypedData) + } + + validateTypedData({ + domain, + message, + primaryType, + types + } as TypedDataDefinition) + + return account.signTypedData({ + domain, + primaryType, + types, + message + } as TypedDataDefinition) +} diff --git a/packages/permissionless/actions/smartAccount/writeContract.ts b/packages/permissionless/actions/smartAccount/writeContract.ts new file mode 100644 index 00000000..ca953fbd --- /dev/null +++ b/packages/permissionless/actions/smartAccount/writeContract.ts @@ -0,0 +1,66 @@ +import { + type Abi, + type Chain, + type Client, + type ContractFunctionArgs, + type ContractFunctionName, + type EncodeFunctionDataParameters, + type Hash, + type SendTransactionParameters, + type Transport, + type WriteContractParameters, + encodeFunctionData +} from "viem" +import type { SmartAccount } from "viem/account-abstraction" +import { getAction } from "viem/utils" +import { sendTransaction } from "./sendTransaction" + +export async function writeContract< + TChain extends Chain | undefined, + TAccount extends SmartAccount | undefined, + const TAbi extends Abi | readonly unknown[], + TFunctionName extends ContractFunctionName< + TAbi, + "nonpayable" | "payable" + > = ContractFunctionName, + TArgs extends ContractFunctionArgs< + TAbi, + "nonpayable" | "payable", + TFunctionName + > = ContractFunctionArgs, + TChainOverride extends Chain | undefined = undefined +>( + client: Client, + { + abi, + address, + args, + dataSuffix, + functionName, + ...request + }: WriteContractParameters< + TAbi, + TFunctionName, + TArgs, + TChain, + TAccount, + TChainOverride + > +): Promise { + const data = encodeFunctionData({ + abi, + args, + functionName + } as EncodeFunctionDataParameters) + + const hash = await getAction( + client, + sendTransaction, + "sendTransaction" + )({ + data: `${data}${dataSuffix ? dataSuffix.replace("0x", "") : ""}`, + to: address, + ...request + } as unknown as SendTransactionParameters) + return hash +} diff --git a/packages/permissionless/clients/createSmartAccountClient.ts b/packages/permissionless/clients/createSmartAccountClient.ts new file mode 100644 index 00000000..c8506128 --- /dev/null +++ b/packages/permissionless/clients/createSmartAccountClient.ts @@ -0,0 +1,149 @@ +import type { + BundlerRpcSchema, + Chain, + Client, + ClientConfig, + EstimateFeesPerGasReturnType, + Prettify, + RpcSchema, + Transport +} from "viem" +import { + type BundlerActions, + type BundlerClientConfig, + type PaymasterActions, + type SmartAccount, + type UserOperationRequest, + createBundlerClient +} from "viem/account-abstraction" +import { + type SmartAccountActions, + smartAccountActions +} from "./decorators/smartAccount" + +/** + * TODO: + * - Add docs + * - Fix typing, 'accounts' is required to signMessage, signTypedData, signTransaction, but not needed here, since account is embedded in the client + */ +export type SmartAccountClient< + transport extends Transport = Transport, + chain extends Chain | undefined = Chain | undefined, + account extends SmartAccount | undefined = SmartAccount | undefined, + client extends Client | undefined = Client | undefined, + rpcSchema extends RpcSchema | undefined = undefined +> = Prettify< + Client< + transport, + chain extends Chain + ? chain + : client extends Client + ? chain + : undefined, + account, + rpcSchema extends RpcSchema + ? [...BundlerRpcSchema, ...rpcSchema] + : BundlerRpcSchema, + BundlerActions & SmartAccountActions + > +> & { + client: client + paymaster: BundlerClientConfig["paymaster"] | undefined + paymasterContext: BundlerClientConfig["paymasterContext"] | undefined + userOperation: BundlerClientConfig["userOperation"] | undefined +} + +export type SmartAccountClientConfig< + transport extends Transport = Transport, + chain extends Chain | undefined = Chain | undefined, + account extends SmartAccount | undefined = SmartAccount | undefined, + client extends Client | undefined = Client | undefined, + rpcSchema extends RpcSchema | undefined = undefined +> = Prettify< + Pick< + ClientConfig, + | "account" + | "cacheTime" + | "chain" + | "key" + | "name" + | "pollingInterval" + | "rpcSchema" + | "transport" + > +> & { + /** Client that points to an Execution RPC URL. */ + client?: client | Client | undefined + /** Paymaster configuration. */ + paymaster?: + | true + | { + /** Retrieves paymaster-related User Operation properties to be used for sending the User Operation. */ + getPaymasterData?: + | PaymasterActions["getPaymasterData"] + | undefined + /** Retrieves paymaster-related User Operation properties to be used for gas estimation. */ + getPaymasterStubData?: + | PaymasterActions["getPaymasterStubData"] + | undefined + } + | undefined + /** Paymaster context to pass to `getPaymasterData` and `getPaymasterStubData` calls. */ + paymasterContext?: unknown + /** User Operation configuration. */ + userOperation?: + | { + /** Prepares fee properties for the User Operation request. */ + estimateFeesPerGas?: + | ((parameters: { + account: account | SmartAccount + bundlerClient: Client + userOperation: UserOperationRequest + }) => Promise>) + | undefined + } + | undefined +} + +export function createSmartAccountClient< + transport extends Transport, + chain extends Chain | undefined = undefined, + account extends SmartAccount | undefined = undefined, + client extends Client | undefined = undefined, + rpcSchema extends RpcSchema | undefined = undefined +>( + parameters: SmartAccountClientConfig< + transport, + chain, + account, + client, + rpcSchema + > +): SmartAccountClient + +export function createSmartAccountClient( + parameters: SmartAccountClientConfig +): SmartAccountClient { + const { + client: client_, + key = "bundler", + name = "Bundler Client", + paymaster, + paymasterContext, + transport, + userOperation + } = parameters + + const client = createBundlerClient({ + ...parameters, + chain: parameters.chain ?? client_?.chain, + key, + name, + transport, + paymaster, + paymasterContext, + userOperation + }) + + return client.extend(smartAccountActions()) as unknown as SmartAccountClient +} diff --git a/packages/permissionless/clients/decorators/smartAccount.ts b/packages/permissionless/clients/decorators/smartAccount.ts new file mode 100644 index 00000000..397928dc --- /dev/null +++ b/packages/permissionless/clients/decorators/smartAccount.ts @@ -0,0 +1,316 @@ +import type { + Abi, + Chain, + Client, + ContractFunctionArgs, + ContractFunctionName, + Hash, + SendTransactionParameters, + Transport, + TypedData, + WriteContractParameters +} from "viem" +import type { SmartAccount } from "viem/account-abstraction" +import { sendTransaction } from "../../actions/smartAccount/sendTransaction" +import { signMessage } from "../../actions/smartAccount/signMessage" +import { signTypedData } from "../../actions/smartAccount/signTypedData" +import { writeContract } from "../../actions/smartAccount/writeContract" + +export type SmartAccountActions< + TChain extends Chain | undefined = Chain | undefined, + TSmartAccount extends SmartAccount | undefined = SmartAccount | undefined +> = { + /** + * Creates, signs, and sends a new transaction to the network. + * This function also allows you to sponsor this transaction if sender is a smartAccount + * + * - Docs: https://viem.sh/docs/actions/wallet/sendTransaction.html + * - Examples: https://stackblitz.com/github/wagmi-dev/viem/tree/main/examples/transactions/sending-transactions + * - JSON-RPC Methods: + * - JSON-RPC Accounts: [`eth_sendTransaction`](https://ethereum.org/en/developers/docs/apis/json-rpc/#eth_sendtransaction) + * - Local Accounts: [`eth_sendRawTransaction`](https://ethereum.org/en/developers/docs/apis/json-rpc/#eth_sendrawtransaction) + * + * @param args - {@link SendTransactionParameters} + * @returns The [Transaction](https://viem.sh/docs/glossary/terms.html#transaction) hash. {@link SendTransactionReturnType} + * + * @example + * import { createWalletClient, custom } from 'viem' + * import { mainnet } from 'viem/chains' + * + * const client = createWalletClient({ + * chain: mainnet, + * transport: custom(window.ethereum), + * }) + * const hash = await client.sendTransaction({ + * account: '0xA0Cf798816D4b9b9866b5330EEa46a18382f251e', + * to: '0x70997970c51812dc3a010c7d01b50e0d17dc79c8', + * value: 1000000000000000000n, + * }) + * + * @example + * // Account Hoisting + * import { createWalletClient, http } from 'viem' + * import { privateKeyToAccount } from 'viem/accounts' + * import { mainnet } from 'viem/chains' + * + * const client = createWalletClient({ + * account: privateKeyToAccount('0x…'), + * chain: mainnet, + * transport: http(), + * }) + * const hash = await client.sendTransaction({ + * to: '0x70997970c51812dc3a010c7d01b50e0d17dc79c8', + * value: 1000000000000000000n, + * }) + */ + sendTransaction: ( + args: SendTransactionParameters + ) => Promise + /** + * Calculates an Ethereum-specific signature in [EIP-191 format](https://eips.ethereum.org/EIPS/eip-191): `keccak256("\x19Ethereum Signed Message:\n" + len(message) + message))`. + * + * - Docs: https://viem.sh/docs/actions/wallet/signMessage.html + * - JSON-RPC Methods: + * - JSON-RPC Accounts: [`personal_sign`](https://docs.metamask.io/guide/signing-data.html#personal-sign) + * - Local Accounts: Signs locally. No JSON-RPC request. + * + * With the calculated signature, you can: + * - use [`verifyMessage`](https://viem.sh/docs/utilities/verifyMessage.html) to verify the signature, + * - use [`recoverMessageAddress`](https://viem.sh/docs/utilities/recoverMessageAddress.html) to recover the signing address from a signature. + * + * @param args - {@link SignMessageParameters} + * @returns The signed message. {@link SignMessageReturnType} + * + * @example + * import { createWalletClient, custom } from 'viem' + * import { mainnet } from 'viem/chains' + * + * const client = createWalletClient({ + * chain: mainnet, + * transport: custom(window.ethereum), + * }) + * const signature = await client.signMessage({ + * account: '0xA0Cf798816D4b9b9866b5330EEa46a18382f251e', + * message: 'hello world', + * }) + * + * @example + * // Account Hoisting + * import { createWalletClient, http } from 'viem' + * import { privateKeyToAccount } from 'viem/accounts' + * import { mainnet } from 'viem/chains' + * + * const client = createWalletClient({ + * account: privateKeyToAccount('0x…'), + * chain: mainnet, + * transport: http(), + * }) + * const signature = await client.signMessage({ + * message: 'hello world', + * }) + */ + signMessage: ( + args: Parameters>[1] + ) => ReturnType> + /** + * Signs typed data and calculates an Ethereum-specific signature in [EIP-191 format](https://eips.ethereum.org/EIPS/eip-191): `keccak256("\x19Ethereum Signed Message:\n" + len(message) + message))`. + * + * - Docs: https://viem.sh/docs/actions/wallet/signTypedData.html + * - JSON-RPC Methods: + * - JSON-RPC Accounts: [`eth_signTypedData_v4`](https://docs.metamask.io/guide/signing-data.html#signtypeddata-v4) + * - Local Accounts: Signs locally. No JSON-RPC request. + * + * @param client - Client to use + * @param args - {@link SignTypedDataParameters} + * @returns The signed data. {@link SignTypedDataReturnType} + * + * @example + * import { createWalletClient, custom } from 'viem' + * import { mainnet } from 'viem/chains' + * + * const client = createWalletClient({ + * chain: mainnet, + * transport: custom(window.ethereum), + * }) + * const signature = await client.signTypedData({ + * account: '0xA0Cf798816D4b9b9866b5330EEa46a18382f251e', + * domain: { + * name: 'Ether Mail', + * version: '1', + * chainId: 1, + * verifyingContract: '0xCcCCccccCCCCcCCCCCCcCcCccCcCCCcCcccccccC', + * }, + * types: { + * Person: [ + * { name: 'name', type: 'string' }, + * { name: 'wallet', type: 'address' }, + * ], + * Mail: [ + * { name: 'from', type: 'Person' }, + * { name: 'to', type: 'Person' }, + * { name: 'contents', type: 'string' }, + * ], + * }, + * primaryType: 'Mail', + * message: { + * from: { + * name: 'Cow', + * wallet: '0xCD2a3d9F938E13CD947Ec05AbC7FE734Df8DD826', + * }, + * to: { + * name: 'Bob', + * wallet: '0xbBbBBBBbbBBBbbbBbbBbbbbBBbBbbbbBbBbbBBbB', + * }, + * contents: 'Hello, Bob!', + * }, + * }) + * + * @example + * // Account Hoisting + * import { createWalletClient, http } from 'viem' + * import { privateKeyToAccount } from 'viem/accounts' + * import { mainnet } from 'viem/chains' + * + * const client = createWalletClient({ + * account: privateKeyToAccount('0x…'), + * chain: mainnet, + * transport: http(), + * }) + * const signature = await client.signTypedData({ + * domain: { + * name: 'Ether Mail', + * version: '1', + * chainId: 1, + * verifyingContract: '0xCcCCccccCCCCcCCCCCCcCcCccCcCCCcCcccccccC', + * }, + * types: { + * Person: [ + * { name: 'name', type: 'string' }, + * { name: 'wallet', type: 'address' }, + * ], + * Mail: [ + * { name: 'from', type: 'Person' }, + * { name: 'to', type: 'Person' }, + * { name: 'contents', type: 'string' }, + * ], + * }, + * primaryType: 'Mail', + * message: { + * from: { + * name: 'Cow', + * wallet: '0xCD2a3d9F938E13CD947Ec05AbC7FE734Df8DD826', + * }, + * to: { + * name: 'Bob', + * wallet: '0xbBbBBBBbbBBBbbbBbbBbbbbBBbBbbbbBbBbbBBbB', + * }, + * contents: 'Hello, Bob!', + * }, + * }) + */ + signTypedData: < + const TTypedData extends TypedData | { [key: string]: unknown }, + TPrimaryType extends string + >( + args: Parameters< + typeof signTypedData + >[1] + ) => ReturnType< + typeof signTypedData + > + /** + * Executes a write function on a contract. + * This function also allows you to sponsor this transaction if sender is a smartAccount + * + * - Docs: https://viem.sh/docs/contract/writeContract.html + * - Examples: https://stackblitz.com/github/wagmi-dev/viem/tree/main/examples/contracts/writing-to-contracts + * + * A "write" function on a Solidity contract modifies the state of the blockchain. These types of functions require gas to be executed, and hence a [Transaction](https://viem.sh/docs/glossary/terms.html) is needed to be broadcast in order to change the state. + * + * Internally, uses a [Wallet Client](https://viem.sh/docs/clients/wallet.html) to call the [`sendTransaction` action](https://viem.sh/docs/actions/wallet/sendTransaction.html) with [ABI-encoded `data`](https://viem.sh/docs/contract/encodeFunctionData.html). + * + * __Warning: The `write` internally sends a transaction – it does not validate if the contract write will succeed (the contract may throw an error). It is highly recommended to [simulate the contract write with `contract.simulate`](https://viem.sh/docs/contract/writeContract.html#usage) before you execute it.__ + * + * @param args - {@link WriteContractParameters} + * @returns A [Transaction Hash](https://viem.sh/docs/glossary/terms.html#hash). {@link WriteContractReturnType} + * + * @example + * import { createWalletClient, custom, parseAbi } from 'viem' + * import { mainnet } from 'viem/chains' + * + * const client = createWalletClient({ + * chain: mainnet, + * transport: custom(window.ethereum), + * }) + * const hash = await client.writeContract({ + * address: '0xFBA3912Ca04dd458c843e2EE08967fC04f3579c2', + * abi: parseAbi(['function mint(uint32 tokenId) nonpayable']), + * functionName: 'mint', + * args: [69420], + * }) + * + * @example + * // With Validation + * import { createWalletClient, custom, parseAbi } from 'viem' + * import { mainnet } from 'viem/chains' + * + * const client = createWalletClient({ + * chain: mainnet, + * transport: custom(window.ethereum), + * }) + * const { request } = await client.simulateContract({ + * address: '0xFBA3912Ca04dd458c843e2EE08967fC04f3579c2', + * abi: parseAbi(['function mint(uint32 tokenId) nonpayable']), + * functionName: 'mint', + * args: [69420], + * } + * const hash = await client.writeContract(request) + */ + writeContract: < + const TAbi extends Abi | readonly unknown[], + TFunctionName extends ContractFunctionName< + TAbi, + "nonpayable" | "payable" + > = ContractFunctionName, + TArgs extends ContractFunctionArgs< + TAbi, + "nonpayable" | "payable", + TFunctionName + > = ContractFunctionArgs, + TChainOverride extends Chain | undefined = undefined + >( + args: WriteContractParameters< + TAbi, + TFunctionName, + TArgs, + TChain, + TSmartAccount, + TChainOverride + > + ) => ReturnType< + typeof writeContract< + TChain, + TSmartAccount, + TAbi, + TFunctionName, + TArgs, + TChainOverride + > + > +} + +export function smartAccountActions() { + return < + TChain extends Chain | undefined = Chain | undefined, + TSmartAccount extends SmartAccount | undefined = + | SmartAccount + | undefined + >( + client: Client + ): SmartAccountActions => ({ + sendTransaction: (args) => sendTransaction(client, args), + signMessage: (args) => signMessage(client, args), + signTypedData: (args) => signTypedData(client, args), + writeContract: (args) => writeContract(client, args) + }) +} diff --git a/packages/permissionless/utils/index.ts b/packages/permissionless/utils/index.ts index b8191eb3..d638f7a1 100644 --- a/packages/permissionless/utils/index.ts +++ b/packages/permissionless/utils/index.ts @@ -9,11 +9,6 @@ import { isSmartAccountDeployed } from "./isSmartAccountDeployed" import { providerToSmartAccountSigner } from "./providerToSmartAccountSigner" import { walletClientToSmartAccountSigner } from "./walletClientToSmartAccountSigner" -export function parseAccount(account: Address | Account): Account { - if (typeof account === "string") - return { address: account, type: "json-rpc" } - return account -} import { decodeNonce } from "./decodeNonce" import { encodeNonce } from "./encodeNonce" From f102a1519f522ad8426d7e1a8a6ff55eacc63af2 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Thu, 22 Aug 2024 03:01:59 +0100 Subject: [PATCH 10/51] Fix safe smart account implementation --- packages/permissionless/accounts/safe/toSafeSmartAccount.ts | 6 +++--- .../permissionless/actions/smartAccount/signMessage.test.ts | 2 +- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts index 7efb3f89..9135765f 100644 --- a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts +++ b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts @@ -1401,7 +1401,7 @@ export async function toSafeSmartAccount< const messageHash = hashTypedData({ domain: { chainId: chainId, - verifyingContract: accountAddress + verifyingContract: await getAddress() }, types: { SafeMessage: [{ name: "message", type: "bytes" }] @@ -1429,7 +1429,7 @@ export async function toSafeSmartAccount< account: owner, domain: { chainId: chainId, - verifyingContract: accountAddress + verifyingContract: await getAddress() }, types: { SafeMessage: [{ name: "message", type: "bytes" }] @@ -1446,7 +1446,7 @@ export async function toSafeSmartAccount< parameters const message = { - safe: accountAddress, + safe: await getAddress(), callData: userOperation.callData, nonce: userOperation.nonce, initCode: userOperation.initCode ?? "0x", diff --git a/packages/permissionless/actions/smartAccount/signMessage.test.ts b/packages/permissionless/actions/smartAccount/signMessage.test.ts index f3ceef0f..5573c06c 100644 --- a/packages/permissionless/actions/smartAccount/signMessage.test.ts +++ b/packages/permissionless/actions/smartAccount/signMessage.test.ts @@ -100,7 +100,7 @@ describe.each(getCoreSmartAccounts())( if (name === "Safe 7579") { // Due to 7579 launchpad, we can't verify the signature as of now. // Awaiting for the fix - // return + return } const isVerified = await publicClient.verifyMessage({ From 7b0c9b613edcfc345ba22ee0131a5049144f82aa Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Thu, 22 Aug 2024 03:05:22 +0100 Subject: [PATCH 11/51] Fix build --- .../accounts/safe/toSafeSmartAccount.ts | 13 +++---------- packages/permissionless/utils/index.ts | 1 - 2 files changed, 3 insertions(+), 11 deletions(-) diff --git a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts index 9135765f..d8cc2747 100644 --- a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts +++ b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts @@ -35,12 +35,7 @@ import { entryPoint07Address, toSmartAccount } from "viem/account-abstraction" -import { - getChainId, - readContract, - signMessage, - signTypedData -} from "viem/actions" +import { getChainId, readContract, signTypedData } from "viem/actions" import { getAction } from "viem/utils" import { getAccountNonce } from "../../actions/public/getAccountNonce" import { isSmartAccountDeployed } from "../../utils" @@ -1414,8 +1409,7 @@ export async function toSafeSmartAccount< return adjustVInSignature( "eth_sign", - await signMessage(client, { - account: owner, + await owner.signMessage({ message: { raw: toBytes(messageHash) } @@ -1425,8 +1419,7 @@ export async function toSafeSmartAccount< async signTypedData(typedData) { return adjustVInSignature( "eth_signTypedData", - await signTypedData(client, { - account: owner, + await owner.signTypedData({ domain: { chainId: chainId, verifyingContract: await getAddress() diff --git a/packages/permissionless/utils/index.ts b/packages/permissionless/utils/index.ts index d638f7a1..2b3f0d54 100644 --- a/packages/permissionless/utils/index.ts +++ b/packages/permissionless/utils/index.ts @@ -1,4 +1,3 @@ -import type { Account, Address } from "viem" import { deepHexlify, transactionReceiptStatus } from "./deepHexlify" import { getAddressFromInitCodeOrPaymasterAndData } from "./getAddressFromInitCodeOrPaymasterAndData" import { From 624e69125c0ad8434a78a1fc2a660915975fe987 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Thu, 22 Aug 2024 03:50:51 +0100 Subject: [PATCH 12/51] Fix safe sign functions --- packages/permissionless/accounts/safe/toSafeSmartAccount.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts index d8cc2747..62d158b3 100644 --- a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts +++ b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts @@ -1395,7 +1395,7 @@ export async function toSafeSmartAccount< async signMessage({ message }) { const messageHash = hashTypedData({ domain: { - chainId: chainId, + chainId: await getMemoizedChainId(), verifyingContract: await getAddress() }, types: { @@ -1421,7 +1421,7 @@ export async function toSafeSmartAccount< "eth_signTypedData", await owner.signTypedData({ domain: { - chainId: chainId, + chainId: await getMemoizedChainId(), verifyingContract: await getAddress() }, types: { From c31edd72aff5029b491310a9f1396f974d890e3f Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Thu, 22 Aug 2024 03:58:07 +0100 Subject: [PATCH 13/51] Add tests for smart account client --- packages/permissionless-test/src/utils.ts | 35 +++++++++++++++++-- .../actions/erc7579/accountId.test.ts | 16 +++------ .../erc7579/supportsExecutionMode.test.ts | 20 +++++------ 3 files changed, 44 insertions(+), 27 deletions(-) diff --git a/packages/permissionless-test/src/utils.ts b/packages/permissionless-test/src/utils.ts index 665b88ad..d5d71a2e 100644 --- a/packages/permissionless-test/src/utils.ts +++ b/packages/permissionless-test/src/utils.ts @@ -37,6 +37,7 @@ import { toSimpleSmartAccount } from "../../permissionless/accounts/simple/toSimpleSmartAccount" import { toTrustSmartAccount } from "../../permissionless/accounts/trust/toTrustSmartAccount" +import { createSmartAccountClient } from "../../permissionless/clients/createSmartAccountClient" import { createPimlicoClient } from "../../permissionless/clients/pimlico" import { paymasterActionsEip7677 } from "../../permissionless/experimental" import type { AAParamType } from "./types" @@ -119,6 +120,34 @@ export const getBundlerClient = ({ }) } +export const getSmartAccountClient = < + account extends SmartAccount | undefined +>({ + altoRpc, + anvilRpc, + account, + paymasterRpc +}: { + altoRpc: string + paymasterRpc?: string + anvilRpc: string + account?: account +}) => { + const paymaster = paymasterRpc + ? createPaymasterClient({ + transport: http(paymasterRpc) + }) + : undefined + + return createSmartAccountClient({ + client: getPublicClient(anvilRpc), + chain: foundry, + account, + paymaster, + transport: http(altoRpc) + }) +} + export const getPimlicoClient = ({ entryPoint, altoRpc @@ -416,7 +445,7 @@ export const getCoreSmartAccounts = () => [ >( conf: AAParamType ) => - getBundlerClient({ + getSmartAccountClient({ account: await getKernelEcdsaClient({ ...conf, version: "0.3.0-beta" as KernelVersion @@ -444,7 +473,7 @@ export const getCoreSmartAccounts = () => [ >( conf: AAParamType ) => - getBundlerClient({ + getSmartAccountClient({ account: await getKernelEcdsaClient({ ...conf, version: "0.3.1" as KernelVersion @@ -495,7 +524,7 @@ export const getCoreSmartAccounts = () => [ >( conf: AAParamType ) => - getBundlerClient({ + getSmartAccountClient({ account: await getSafeClient({ ...conf, erc7579: true }), ...conf }), diff --git a/packages/permissionless/actions/erc7579/accountId.test.ts b/packages/permissionless/actions/erc7579/accountId.test.ts index 5df82efa..176583e1 100644 --- a/packages/permissionless/actions/erc7579/accountId.test.ts +++ b/packages/permissionless/actions/erc7579/accountId.test.ts @@ -24,18 +24,10 @@ describe.each(getCoreSmartAccounts())( const accountIdBeforeDeploy = await accountId(smartClient) // deploy account - const userOpHash = await smartClient.sendUserOperation({ - calls: [ - { - to: zeroAddress, - value: 0n, - data: "0x" - } - ] - }) - - await smartClient.waitForUserOperationReceipt({ - hash: userOpHash + await smartClient.sendTransaction({ + to: zeroAddress, + value: 0n, + data: "0x" }) const postDeployAccountId = await accountId(smartClient) diff --git a/packages/permissionless/actions/erc7579/supportsExecutionMode.test.ts b/packages/permissionless/actions/erc7579/supportsExecutionMode.test.ts index 041ea562..c89e3a4e 100644 --- a/packages/permissionless/actions/erc7579/supportsExecutionMode.test.ts +++ b/packages/permissionless/actions/erc7579/supportsExecutionMode.test.ts @@ -33,12 +33,10 @@ describe.each(getCoreSmartAccounts())( expect(supportsExecutionModeBatchCallBeforeDeploy).toBe(true) // deploy account - const userOpHash = await smartClient.sendUserOperation({ - calls: [{ to: zeroAddress, value: 0n, data: "0x" }] - }) - - await smartClient.waitForUserOperationReceipt({ - hash: userOpHash + await smartClient.sendTransaction({ + to: zeroAddress, + value: 0n, + data: "0x" }) const supportsExecutionModeBatchCallBeforeDeployPostDeploy = @@ -82,12 +80,10 @@ describe.each(getCoreSmartAccounts())( expect(supportsExecutionModeBatchCallBeforeDeploy).toBe(true) // deploy account - const userOpHash = await smartClient.sendUserOperation({ - calls: [{ to: zeroAddress, value: 0n, data: "0x" }] - }) - - await smartClient.waitForUserOperationReceipt({ - hash: userOpHash + await smartClient.sendTransaction({ + to: zeroAddress, + value: 0n, + data: "0x" }) const supportsExecutionModeBatchCallBeforeDeployPostDeploy = From cd6ce32f931a0b86396f33cd34bd5414b362f5ff Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Thu, 22 Aug 2024 04:40:43 +0100 Subject: [PATCH 14/51] Cleanup --- .../mock-aa-infra/mock-paymaster/relay.ts | 2 +- packages/permissionless-test/src/utils.ts | 52 +++++++++++--- .../pimlico/getUserOperationGasPrice.test.ts | 3 +- .../pimlico/getUserOperationStatus.test.ts | 10 ++- .../pimlico/sponsorUserOperation.test.ts | 10 ++- .../actions/pimlico/sponsorUserOperation.ts | 21 +++--- .../validateSponsorshipPolicies.test.ts | 5 +- .../actions/public/getAccountNonce.test.ts | 6 ++ .../actions/public/getSenderAddress.test.ts | 9 +++ .../permissionless/actions/smartAccount.ts | 9 +++ .../clients/decorators/pimlico.ts | 19 +++--- packages/permissionless/clients/pimlico.ts | 67 +++++++++++++------ .../eip7677/actions/getPaymasterData.test.ts | 16 ++++- .../actions/getPaymasterStubData.test.ts | 10 ++- packages/permissionless/package.json | 15 ----- packages/permissionless/types/index.ts | 23 ------- 16 files changed, 181 insertions(+), 96 deletions(-) create mode 100644 packages/permissionless/actions/smartAccount.ts delete mode 100644 packages/permissionless/types/index.ts diff --git a/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts b/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts index de6bf9e4..f6fddee7 100644 --- a/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts +++ b/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts @@ -423,7 +423,7 @@ const handleMethod = async ( } throw new RpcError( - "Attempted to call an unknown method", + `Attempted to call an unknown method ${parsedBody.method}`, ValidationErrors.InvalidFields ) } diff --git a/packages/permissionless-test/src/utils.ts b/packages/permissionless-test/src/utils.ts index d5d71a2e..c93b0930 100644 --- a/packages/permissionless-test/src/utils.ts +++ b/packages/permissionless-test/src/utils.ts @@ -50,7 +50,10 @@ export const ensureBundlerIsReady = async ({ }: { altoRpc: string; anvilRpc: string }) => { const bundlerClient = getBundlerClient({ altoRpc: altoRpc, - anvilRpc + anvilRpc, + entryPoint: { + version: "0.6" + } }) while (true) { @@ -99,24 +102,51 @@ export const getBundlerClient = ({ altoRpc, anvilRpc, account, - paymasterRpc + paymasterRpc, + entryPoint }: { altoRpc: string paymasterRpc?: string anvilRpc: string account?: account + entryPoint: { + version: "0.6" | "0.7" + } }) => { const paymaster = paymasterRpc - ? createPaymasterClient({ - transport: http(paymasterRpc) + ? createPimlicoClient({ + transport: http(paymasterRpc), + entryPoint: { + address: + entryPoint.version === "0.6" + ? entryPoint06Address + : entryPoint07Address, + version: entryPoint.version + } }) : undefined + const pimlicoBundler = createPimlicoClient({ + transport: http(altoRpc), + entryPoint: { + address: + entryPoint.version === "0.6" + ? entryPoint06Address + : entryPoint07Address, + version: entryPoint.version + } + }) + return createBundlerClient({ client: getPublicClient(anvilRpc), account, paymaster, - transport: http(altoRpc) + transport: http(altoRpc), + userOperation: { + estimateFeesPerGas: async ({ userOperation }) => { + return (await pimlicoBundler.getUserOperationGasPrice()).fast + } + } }) } @@ -149,15 +179,21 @@ export const getSmartAccountClient = < } export const getPimlicoClient = ({ - entryPoint, + entryPointVersion, altoRpc }: { - entryPoint: typeof entryPoint07Address | typeof entryPoint06Address + entryPointVersion: "0.6" | "0.7" altoRpc: string }) => createPimlicoClient({ chain: foundry, - entryPointAddress: entryPoint, + entryPoint: { + address: + entryPointVersion === "0.6" + ? entryPoint06Address + : entryPoint07Address, + version: entryPointVersion + }, transport: http(altoRpc) }) diff --git a/packages/permissionless/actions/pimlico/getUserOperationGasPrice.test.ts b/packages/permissionless/actions/pimlico/getUserOperationGasPrice.test.ts index f365f76c..196a3f35 100644 --- a/packages/permissionless/actions/pimlico/getUserOperationGasPrice.test.ts +++ b/packages/permissionless/actions/pimlico/getUserOperationGasPrice.test.ts @@ -1,4 +1,3 @@ -import { entryPoint06Address } from "viem/account-abstraction" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" import { getPimlicoClient } from "../../../permissionless-test/src/utils" @@ -7,7 +6,7 @@ import { getUserOperationGasPrice } from "./getUserOperationGasPrice" describe("getUserOperationGasPrice", () => { testWithRpc("getUserOperationGasPrice", async ({ rpc }) => { const pimlicoBundlerClient = getPimlicoClient({ - entryPoint: entryPoint06Address, + entryPointVersion: "0.6", altoRpc: rpc.altoRpc }) diff --git a/packages/permissionless/actions/pimlico/getUserOperationStatus.test.ts b/packages/permissionless/actions/pimlico/getUserOperationStatus.test.ts index 4b0f3986..5aa2b807 100644 --- a/packages/permissionless/actions/pimlico/getUserOperationStatus.test.ts +++ b/packages/permissionless/actions/pimlico/getUserOperationStatus.test.ts @@ -14,7 +14,7 @@ describe("getUserOperationStatus", () => { const { altoRpc } = rpc const bundlerClient = getPimlicoClient({ - entryPoint: entryPoint06Address, + entryPointVersion: "0.6", altoRpc: altoRpc }) @@ -25,6 +25,9 @@ describe("getUserOperationStatus", () => { version: "0.6" } }), + entryPoint: { + version: "0.6" + }, ...rpc }) @@ -74,7 +77,7 @@ describe("getUserOperationStatus", () => { const { altoRpc } = rpc const bundlerClient = getPimlicoClient({ - entryPoint: entryPoint06Address, + entryPointVersion: "0.7", altoRpc: altoRpc }) @@ -85,6 +88,9 @@ describe("getUserOperationStatus", () => { version: "0.7" } }), + entryPoint: { + version: "0.7" + }, ...rpc }) diff --git a/packages/permissionless/actions/pimlico/sponsorUserOperation.test.ts b/packages/permissionless/actions/pimlico/sponsorUserOperation.test.ts index 6c307764..aed29fe3 100644 --- a/packages/permissionless/actions/pimlico/sponsorUserOperation.test.ts +++ b/packages/permissionless/actions/pimlico/sponsorUserOperation.test.ts @@ -13,7 +13,7 @@ describe("sponsorUserOperation", () => { const { altoRpc } = rpc const bundlerClient = getPimlicoClient({ - entryPoint: entryPoint06Address, + entryPointVersion: "0.6", altoRpc: altoRpc }) @@ -24,6 +24,9 @@ describe("sponsorUserOperation", () => { version: "0.6" } }), + entryPoint: { + version: "0.6" + }, ...rpc }) @@ -61,7 +64,7 @@ describe("sponsorUserOperation", () => { const { altoRpc } = rpc const bundlerClient = getPimlicoClient({ - entryPoint: entryPoint06Address, + entryPointVersion: "0.7", altoRpc: altoRpc }) @@ -72,6 +75,9 @@ describe("sponsorUserOperation", () => { version: "0.7" } }), + entryPoint: { + version: "0.7" + }, ...rpc }) diff --git a/packages/permissionless/actions/pimlico/sponsorUserOperation.ts b/packages/permissionless/actions/pimlico/sponsorUserOperation.ts index 1ab7bf1f..5db48879 100644 --- a/packages/permissionless/actions/pimlico/sponsorUserOperation.ts +++ b/packages/permissionless/actions/pimlico/sponsorUserOperation.ts @@ -8,21 +8,19 @@ import type { PartialBy, Transport } from "viem" -import { - type UserOperation, +import type { + UserOperation, entryPoint06Address, - type entryPoint07Address + entryPoint07Address } from "viem/account-abstraction" import type { PimlicoRpcSchema } from "../../types/pimlico" import { deepHexlify } from "../../utils/deepHexlify" export type PimlicoSponsorUserOperationParameters< entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address = | typeof entryPoint06Address | typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7" = "0.6" | "0.7" + entryPointVersion extends "0.6" | "0.7" > = { userOperation: OneOf< | (entryPointVersion extends "0.6" @@ -44,7 +42,10 @@ export type PimlicoSponsorUserOperationParameters< > : never) > - entryPointAddress: entryPointAddress + entryPoint: { + address: entryPointAddress + version: entryPointVersion + } sponsorshipPolicyId?: string } @@ -121,15 +122,15 @@ export const sponsorUserOperation = async < params: args.sponsorshipPolicyId ? [ deepHexlify(args.userOperation), - args.entryPointAddress, + args.entryPoint.address, { sponsorshipPolicyId: args.sponsorshipPolicyId } ] - : [deepHexlify(args.userOperation), args.entryPointAddress] + : [deepHexlify(args.userOperation), args.entryPoint.address] }) - if (args.entryPointAddress === entryPoint06Address) { + if (args.entryPoint.version === "0.6") { const responseV06 = response as { paymasterAndData: Hex preVerificationGas: Hex diff --git a/packages/permissionless/actions/pimlico/validateSponsorshipPolicies.test.ts b/packages/permissionless/actions/pimlico/validateSponsorshipPolicies.test.ts index 4035e289..ad5561de 100644 --- a/packages/permissionless/actions/pimlico/validateSponsorshipPolicies.test.ts +++ b/packages/permissionless/actions/pimlico/validateSponsorshipPolicies.test.ts @@ -19,6 +19,9 @@ describe("validateSponsorshipPolicies", () => { version: "0.6" } }), + entryPoint: { + version: "0.6" + }, ...rpc }) @@ -33,7 +36,7 @@ describe("validateSponsorshipPolicies", () => { }) const pimlicoPaymasterClient = getPimlicoClient({ - entryPoint: entryPoint06Address, + entryPointVersion: "0.6", altoRpc: paymasterRpc }) diff --git a/packages/permissionless/actions/public/getAccountNonce.test.ts b/packages/permissionless/actions/public/getAccountNonce.test.ts index 8fee45ad..ae0ed69d 100644 --- a/packages/permissionless/actions/public/getAccountNonce.test.ts +++ b/packages/permissionless/actions/public/getAccountNonce.test.ts @@ -27,6 +27,9 @@ describe("getAccountNonce", () => { version: "0.6" } }), + entryPoint: { + version: "0.6" + }, ...rpc }) @@ -51,6 +54,9 @@ describe("getAccountNonce", () => { version: "0.7" } }), + entryPoint: { + version: "0.7" + }, ...rpc }) diff --git a/packages/permissionless/actions/public/getSenderAddress.test.ts b/packages/permissionless/actions/public/getSenderAddress.test.ts index f33c0da9..564fc00e 100644 --- a/packages/permissionless/actions/public/getSenderAddress.test.ts +++ b/packages/permissionless/actions/public/getSenderAddress.test.ts @@ -26,6 +26,9 @@ describe("getSenderAddress", () => { version: "0.6" } }), + entryPoint: { + version: "0.6" + }, ...rpc }) @@ -57,6 +60,9 @@ describe("getSenderAddress", () => { version: "0.6" } }), + entryPoint: { + version: "0.6" + }, ...rpc }) @@ -89,6 +95,9 @@ describe("getSenderAddress", () => { version: "0.7" } }), + entryPoint: { + version: "0.7" + }, ...rpc }) diff --git a/packages/permissionless/actions/smartAccount.ts b/packages/permissionless/actions/smartAccount.ts new file mode 100644 index 00000000..f09e4245 --- /dev/null +++ b/packages/permissionless/actions/smartAccount.ts @@ -0,0 +1,9 @@ +import { sendTransaction } from "./smartAccount/sendTransaction" + +import { signMessage } from "./smartAccount/signMessage" + +import { signTypedData } from "./smartAccount/signTypedData" + +import { writeContract } from "./smartAccount/writeContract" + +export { sendTransaction, signMessage, signTypedData, writeContract } diff --git a/packages/permissionless/clients/decorators/pimlico.ts b/packages/permissionless/clients/decorators/pimlico.ts index 69baa268..1715ff81 100644 --- a/packages/permissionless/clients/decorators/pimlico.ts +++ b/packages/permissionless/clients/decorators/pimlico.ts @@ -132,11 +132,14 @@ export const pimlicoActions = < entryPointAddress extends | typeof entryPoint06Address - | typeof entryPoint07Address - >( - entryPointAddress: entryPointAddress - ) => - (client: Client): PimlicoActions => ({ + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7" + >({ + entryPoint + }: { + entryPoint: { address: entryPointAddress; version: entryPointVersion } + }) => + (client: Client): PimlicoActions => ({ getUserOperationGasPrice: async () => getUserOperationGasPrice(client), getUserOperationStatus: async ( args: GetUserOperationStatusParameters @@ -146,16 +149,16 @@ export const pimlicoActions = ) => sendCompressedUserOperation(client, { ...args, - entryPointAddress + entryPointAddress: entryPoint.address }), sponsorUserOperation: async (args) => sponsorUserOperation(client, { ...args, - entryPointAddress + entryPoint }), validateSponsorshipPolicies: async (args) => validateSponsorshipPolicies(client, { ...args, - entryPointAddress + entryPointAddress: entryPoint.address }) }) diff --git a/packages/permissionless/clients/pimlico.ts b/packages/permissionless/clients/pimlico.ts index aa836fa4..9e7d6123 100644 --- a/packages/permissionless/clients/pimlico.ts +++ b/packages/permissionless/clients/pimlico.ts @@ -11,15 +11,23 @@ import type { import { createClient } from "viem" import { type BundlerActions, + type PaymasterActions, type SmartAccount, bundlerActions, type entryPoint06Address, - entryPoint07Address + type entryPoint07Address, + paymasterActions } from "viem/account-abstraction" import type { PimlicoRpcSchema } from "../types/pimlico" -import { pimlicoActions } from "./decorators/pimlico" +import { type PimlicoActions, pimlicoActions } from "./decorators/pimlico" export type PimlicoClient< + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address = + | typeof entryPoint07Address + | typeof entryPoint06Address, + entryPointVersion extends "0.6" | "0.7" = "0.7" | "0.6", transport extends Transport = Transport, chain extends Chain | undefined = Chain | undefined, account extends SmartAccount | undefined = SmartAccount | undefined, @@ -38,19 +46,23 @@ export type PimlicoClient< rpcSchema extends RpcSchema ? [...BundlerRpcSchema, ...PimlicoRpcSchema, ...rpcSchema] : [...BundlerRpcSchema, ...PimlicoRpcSchema], - BundlerActions + BundlerActions & + PaymasterActions & + PimlicoActions > > export type PimlicoClientConfig< - transport extends Transport = Transport, - chain extends Chain | undefined = Chain | undefined, - account extends SmartAccount | undefined = SmartAccount | undefined, - rpcSchema extends RpcSchema | undefined = undefined, entryPointAddress extends | typeof entryPoint06Address + | typeof entryPoint07Address = | typeof entryPoint07Address - | undefined = undefined + | typeof entryPoint06Address, + entryPointVersion extends "0.6" | "0.7" = "0.7" | "0.6", + transport extends Transport = Transport, + chain extends Chain | undefined = Chain | undefined, + account extends SmartAccount | undefined = SmartAccount | undefined, + rpcSchema extends RpcSchema | undefined = undefined > = Prettify< Pick< ClientConfig, @@ -64,7 +76,10 @@ export type PimlicoClientConfig< | "transport" > > & { - entryPointAddress?: entryPointAddress + entryPoint: { + address: entryPointAddress + version: entryPointVersion + } } /** @@ -87,23 +102,33 @@ export type PimlicoClientConfig< * }) */ export function createPimlicoClient< + entryPointAddress extends + | typeof entryPoint06Address + | typeof entryPoint07Address, + entryPointVersion extends "0.6" | "0.7", transport extends Transport, chain extends Chain | undefined = undefined, account extends SmartAccount | undefined = undefined, client extends Client | undefined = undefined, - rpcSchema extends RpcSchema | undefined = undefined, - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address = typeof entryPoint07Address + rpcSchema extends RpcSchema | undefined = undefined >( parameters: PimlicoClientConfig< + entryPointAddress, + entryPointVersion, transport, chain, account, - rpcSchema, - entryPointAddress + rpcSchema > -): PimlicoClient +): PimlicoClient< + entryPointAddress, + entryPointVersion, + transport, + chain, + account, + client, + rpcSchema +> export function createPimlicoClient( parameters: PimlicoClientConfig @@ -111,7 +136,7 @@ export function createPimlicoClient( const { key = "public", name = "Pimlico Bundler Client", - entryPointAddress + entryPoint } = parameters const client = createClient({ ...parameters, @@ -119,7 +144,9 @@ export function createPimlicoClient( name, type: "pimlicoBundlerClient" }) - return client - .extend(bundlerActions) - .extend(pimlicoActions(entryPointAddress ?? entryPoint07Address)) + return client.extend(bundlerActions).extend(paymasterActions).extend( + pimlicoActions({ + entryPoint + }) + ) } diff --git a/packages/permissionless/experimental/eip7677/actions/getPaymasterData.test.ts b/packages/permissionless/experimental/eip7677/actions/getPaymasterData.test.ts index ba1b0f40..e3ffd692 100644 --- a/packages/permissionless/experimental/eip7677/actions/getPaymasterData.test.ts +++ b/packages/permissionless/experimental/eip7677/actions/getPaymasterData.test.ts @@ -24,11 +24,14 @@ describe("EIP-7677 getPaymasterData", () => { version: "0.6" } }), + entryPoint: { + version: "0.6" + }, ...rpc }) const pimlicoPaymasterClient = getPimlicoClient({ - entryPoint: entryPoint06Address, + entryPointVersion: "0.6", altoRpc: paymasterRpc }).extend( paymasterActionsEip7677({ @@ -56,6 +59,9 @@ describe("EIP-7677 getPaymasterData", () => { // test that smart account can send op with the returned values const bundlerClient = getBundlerClient({ + entryPoint: { + version: "0.6" + }, ...rpc }) @@ -87,11 +93,14 @@ describe("EIP-7677 getPaymasterData", () => { version: "0.7" } }), + entryPoint: { + version: "0.7" + }, ...rpc }) const pimlicoPaymasterClient = getPimlicoClient({ - entryPoint: entryPoint07Address, + entryPointVersion: "0.7", altoRpc: paymasterRpc }).extend( paymasterActionsEip7677({ @@ -128,6 +137,9 @@ describe("EIP-7677 getPaymasterData", () => { // test that smart account can send op with the returned values const bundlerClient = getBundlerClient({ + entryPoint: { + version: "0.7" + }, ...rpc }) diff --git a/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.test.ts b/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.test.ts index 6324b331..c4747329 100644 --- a/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.test.ts +++ b/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.test.ts @@ -24,11 +24,14 @@ describe("EIP-7677 getPaymasterStubData", () => { version: "0.6" } }), + entryPoint: { + version: "0.6" + }, ...rpc }) const pimlicoPaymasterClient = getPimlicoClient({ - entryPoint: entryPoint06Address, + entryPointVersion: "0.6", altoRpc: paymasterRpc }).extend( paymasterActionsEip7677({ @@ -68,11 +71,14 @@ describe("EIP-7677 getPaymasterStubData", () => { version: "0.7" } }), + entryPoint: { + version: "0.7" + }, ...rpc }) const pimlicoPaymasterClient = getPimlicoClient({ - entryPoint: entryPoint07Address, + entryPointVersion: "0.7", altoRpc: paymasterRpc }).extend( paymasterActionsEip7677({ diff --git a/packages/permissionless/package.json b/packages/permissionless/package.json index 1fafc4ea..d94ef980 100644 --- a/packages/permissionless/package.json +++ b/packages/permissionless/package.json @@ -39,11 +39,6 @@ "import": "./_esm/actions/pimlico.js", "default": "./_cjs/actions/pimlico.js" }, - "./actions/stackup": { - "types": "./_types/actions/stackup.d.ts", - "import": "./_esm/actions/stackup.js", - "default": "./_cjs/actions/stackup.js" - }, "./actions/smartAccount": { "types": "./_types/actions/smartAccount.d.ts", "import": "./_esm/actions/smartAccount.js", @@ -59,11 +54,6 @@ "import": "./_esm/clients/pimlico.js", "default": "./_cjs/clients/pimlico.js" }, - "./clients/stackup": { - "types": "./_types/clients/stackup.d.ts", - "import": "./_esm/clients/stackup.js", - "default": "./_cjs/clients/stackup.js" - }, "./utils": { "types": "./_types/utils/index.d.ts", "import": "./_esm/utils/index.js", @@ -74,11 +64,6 @@ "import": "./_esm/errors/index.js", "default": "./_cjs/errors/index.js" }, - "./types": { - "types": "./_types/types/index.d.ts", - "import": "./_esm/types/index.js", - "default": "./_cjs/types/index.js" - }, "./experimental": { "types": "./_types/experimental/index.d.ts", "import": "./_esm/experimental/index.js", diff --git a/packages/permissionless/types/index.ts b/packages/permissionless/types/index.ts deleted file mode 100644 index f7059c85..00000000 --- a/packages/permissionless/types/index.ts +++ /dev/null @@ -1,23 +0,0 @@ -import type { - SmartAccount, - SmartAccountImplementation -} from "viem/account-abstraction" - -export type IsUndefined = [undefined] extends [T] ? true : false - -export type PartialBy = Omit & Partial> - -export type PartialPick = Partial> - -export type GetAccountParameter< - implementation extends - | SmartAccountImplementation - | undefined = SmartAccountImplementation -> = IsUndefined extends true - ? { account: SmartAccount } - : { account?: SmartAccount } - -// biome-ignore lint/suspicious/noExplicitAny: generic type -export type UnionOmit = T extends any - ? Omit - : never From 142abeed5520665e6fb11ef0394f6d242f97f316 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Thu, 22 Aug 2024 04:56:15 +0100 Subject: [PATCH 15/51] Remove 7677 --- .../mock-aa-infra/mock-paymaster/relay.ts | 4 + packages/permissionless-test/src/utils.ts | 41 ++--- packages/permissionless/clients/pimlico.ts | 2 +- .../eip7677/actions/getPaymasterData.test.ts | 165 ------------------ .../eip7677/actions/getPaymasterData.ts | 107 ------------ .../actions/getPaymasterStubData.test.ts | 113 ------------ .../eip7677/actions/getPaymasterStubData.ts | 123 ------------- .../decorators/paymasterActionsEip7677.ts | 85 --------- .../experimental/eip7677/index.ts | 27 --- .../experimental/eip7677/types/paymaster.ts | 77 -------- packages/permissionless/experimental/index.ts | 1 - 11 files changed, 20 insertions(+), 725 deletions(-) delete mode 100644 packages/permissionless/experimental/eip7677/actions/getPaymasterData.test.ts delete mode 100644 packages/permissionless/experimental/eip7677/actions/getPaymasterData.ts delete mode 100644 packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.test.ts delete mode 100644 packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.ts delete mode 100644 packages/permissionless/experimental/eip7677/clients/decorators/paymasterActionsEip7677.ts delete mode 100644 packages/permissionless/experimental/eip7677/index.ts delete mode 100644 packages/permissionless/experimental/eip7677/types/paymaster.ts delete mode 100644 packages/permissionless/experimental/index.ts diff --git a/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts b/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts index f6fddee7..d02175ad 100644 --- a/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts +++ b/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts @@ -329,6 +329,10 @@ const handleMethod = async ( ) if (!params.success) { + console.log({ + params: parsedBody.params + }) + throw new RpcError( fromZodError(params.error).message, ValidationErrors.InvalidFields diff --git a/packages/permissionless-test/src/utils.ts b/packages/permissionless-test/src/utils.ts index c93b0930..3fa182e5 100644 --- a/packages/permissionless-test/src/utils.ts +++ b/packages/permissionless-test/src/utils.ts @@ -5,7 +5,6 @@ import { type Chain, type Transport, type WalletClient, - createClient, createPublicClient, createWalletClient, parseEther @@ -39,7 +38,6 @@ import { import { toTrustSmartAccount } from "../../permissionless/accounts/trust/toTrustSmartAccount" import { createSmartAccountClient } from "../../permissionless/clients/createSmartAccountClient" import { createPimlicoClient } from "../../permissionless/clients/pimlico" -import { paymasterActionsEip7677 } from "../../permissionless/experimental" import type { AAParamType } from "./types" export const PAYMASTER_RPC = "http://localhost:3000" @@ -115,6 +113,7 @@ export const getBundlerClient = ({ }) => { const paymaster = paymasterRpc ? createPimlicoClient({ + chain: foundry, transport: http(paymasterRpc), entryPoint: { address: @@ -143,7 +142,7 @@ export const getBundlerClient = ({ paymaster, transport: http(altoRpc), userOperation: { - estimateFeesPerGas: async ({ userOperation }) => { + estimateFeesPerGas: async () => { return (await pimlicoBundler.getUserOperationGasPrice()).fast } } @@ -178,20 +177,26 @@ export const getSmartAccountClient = < }) } -export const getPimlicoClient = ({ +export const getPimlicoClient = ({ entryPointVersion, altoRpc }: { - entryPointVersion: "0.6" | "0.7" + entryPointVersion: entryPointVersion altoRpc: string }) => - createPimlicoClient({ + createPimlicoClient< + entryPointVersion extends "0.6" + ? typeof entryPoint06Address + : typeof entryPoint07Address, + entryPointVersion + >({ chain: foundry, entryPoint: { - address: - entryPointVersion === "0.6" - ? entryPoint06Address - : entryPoint07Address, + address: (entryPointVersion === "0.6" + ? entryPoint06Address + : entryPoint07Address) as entryPointVersion extends "0.6" + ? typeof entryPoint06Address + : typeof entryPoint07Address, version: entryPointVersion }, transport: http(altoRpc) @@ -394,22 +399,6 @@ export const getSafeClient = async ({ }) } -export const getEip7677Client = async ({ - entryPoint -}: { - entryPoint: { - address: typeof entryPoint06Address | typeof entryPoint07Address - version: "0.6" | "0.7" - } -}) => { - const client = createClient({ - chain: foundry, - transport: http(PAYMASTER_RPC) - }).extend(paymasterActionsEip7677(entryPoint)) - - return client -} - export const getCoreSmartAccounts = () => [ { name: "Trust", diff --git a/packages/permissionless/clients/pimlico.ts b/packages/permissionless/clients/pimlico.ts index 9e7d6123..51e82cdd 100644 --- a/packages/permissionless/clients/pimlico.ts +++ b/packages/permissionless/clients/pimlico.ts @@ -106,7 +106,7 @@ export function createPimlicoClient< | typeof entryPoint06Address | typeof entryPoint07Address, entryPointVersion extends "0.6" | "0.7", - transport extends Transport, + transport extends Transport = Transport, chain extends Chain | undefined = undefined, account extends SmartAccount | undefined = undefined, client extends Client | undefined = undefined, diff --git a/packages/permissionless/experimental/eip7677/actions/getPaymasterData.test.ts b/packages/permissionless/experimental/eip7677/actions/getPaymasterData.test.ts deleted file mode 100644 index e3ffd692..00000000 --- a/packages/permissionless/experimental/eip7677/actions/getPaymasterData.test.ts +++ /dev/null @@ -1,165 +0,0 @@ -import { zeroAddress } from "viem" -import { - entryPoint06Address, - entryPoint07Address -} from "viem/account-abstraction" -import { foundry } from "viem/chains" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../../permissionless-test/src/testWithRpc" -import { - getBundlerClient, - getPimlicoClient, - getSimpleAccountClient -} from "../../../../permissionless-test/src/utils" -import { paymasterActionsEip7677 } from "../clients/decorators/paymasterActionsEip7677" - -describe("EIP-7677 getPaymasterData", () => { - testWithRpc("getPaymasterData_V06", async ({ rpc }) => { - const { paymasterRpc } = rpc - - const simpleAccountClient = getBundlerClient({ - account: await getSimpleAccountClient({ - ...rpc, - entryPoint: { - version: "0.6" - } - }), - entryPoint: { - version: "0.6" - }, - ...rpc - }) - - const pimlicoPaymasterClient = getPimlicoClient({ - entryPointVersion: "0.6", - altoRpc: paymasterRpc - }).extend( - paymasterActionsEip7677({ - address: entryPoint06Address, - version: "0.6" - }) - ) - - let userOperation = await simpleAccountClient.prepareUserOperation({ - calls: [ - { - to: zeroAddress, - data: "0x", - value: 0n - } - ] - }) - - const paymasterData = await pimlicoPaymasterClient.getPaymasterData({ - userOperation, - chain: foundry - }) - expect(paymasterData).not.toBeNull() - expect(paymasterData?.paymasterAndData).not.toBeNull() - - // test that smart account can send op with the returned values - const bundlerClient = getBundlerClient({ - entryPoint: { - version: "0.6" - }, - ...rpc - }) - - userOperation = { - ...userOperation, - ...paymasterData - } - userOperation.signature = - await simpleAccountClient.account.signUserOperation(userOperation) - - const hash = await bundlerClient.sendUserOperation({ - account: simpleAccountClient.account, - ...userOperation - }) - - const receipt = await bundlerClient.waitForUserOperationReceipt({ - hash - }) - expect(receipt.success).toBeTruthy() - }) - - testWithRpc("getPaymasterData_V07", async ({ rpc }) => { - const { paymasterRpc } = rpc - - const simpleAccountClient = getBundlerClient({ - account: await getSimpleAccountClient({ - ...rpc, - entryPoint: { - version: "0.7" - } - }), - entryPoint: { - version: "0.7" - }, - ...rpc - }) - - const pimlicoPaymasterClient = getPimlicoClient({ - entryPointVersion: "0.7", - altoRpc: paymasterRpc - }).extend( - paymasterActionsEip7677({ - address: entryPoint07Address, - version: "0.7" - }) - ) - - let userOperation = await simpleAccountClient.prepareUserOperation({ - calls: [ - { - to: zeroAddress, - data: "0x", - value: 0n - } - ] - }) - - const paymasterGasValues = { - paymasterPostOpGasLimit: 150_000n, - paymasterVerificationGasLimit: 650_000n - } - - const paymasterData = await pimlicoPaymasterClient.getPaymasterData({ - userOperation: { - ...userOperation, - ...paymasterGasValues - }, - chain: foundry - }) - expect(paymasterData).not.toBeNull() - expect(paymasterData?.paymaster).not.toBeNull() - expect(paymasterData?.paymasterData).not.toBeNull() - - // test that smart account can send op with the returned values - const bundlerClient = getBundlerClient({ - entryPoint: { - version: "0.7" - }, - ...rpc - }) - - userOperation = { - ...userOperation, - ...paymasterGasValues, - ...paymasterData - } - - userOperation.signature = - await simpleAccountClient.account.signUserOperation(userOperation) - - const hash = await bundlerClient.sendUserOperation({ - account: simpleAccountClient.account, - ...userOperation - }) - - const receipt = await bundlerClient.waitForUserOperationReceipt({ - hash - }) - expect(receipt.success).toBeTruthy() - }) -}) diff --git a/packages/permissionless/experimental/eip7677/actions/getPaymasterData.ts b/packages/permissionless/experimental/eip7677/actions/getPaymasterData.ts deleted file mode 100644 index 78775f89..00000000 --- a/packages/permissionless/experimental/eip7677/actions/getPaymasterData.ts +++ /dev/null @@ -1,107 +0,0 @@ -import { - type Account, - type Chain, - ChainNotFoundError, - type Client, - type GetChainParameter, - type Hex, - type Transport, - toHex -} from "viem" -import type { - UserOperation, - entryPoint06Address, - entryPoint07Address -} from "viem/account-abstraction" -import { deepHexlify } from "../../../utils" -import type { - Eip7677RpcSchema, - GetRpcPaymasterDataReturnType -} from "../types/paymaster" - -export type GetPaymasterDataParameters< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7", - TChain extends Chain | undefined, - TChainOverride extends Chain | undefined -> = { - userOperation: UserOperation - entryPoint: { - address: entryPointAddress - version: entryPointVersion - } - context?: Record -} & GetChainParameter - -export type GetPaymasterDataReturnType< - entryPointVersion extends "0.6" | "0.7" -> = entryPointVersion extends "0.6" - ? { - paymasterAndData: Hex - } - : { - paymaster: Hex - paymasterData: Hex - } - -export async function getPaymasterData< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7", - TChain extends Chain | undefined, - TChainOverride extends Chain | undefined ->( - client: Client< - Transport, - TChain, - Account | undefined, - Eip7677RpcSchema - >, - { - userOperation, - entryPoint, - context, - chain - }: GetPaymasterDataParameters< - entryPointAddress, - entryPointVersion, - TChain, - TChainOverride - > -): Promise> { - const chainId = chain?.id ?? client.chain?.id - - if (!chainId) { - throw new ChainNotFoundError() - } - - const response = await client.request({ - method: "pm_getPaymasterData", - params: context - ? [ - deepHexlify(userOperation), - entryPoint.address, - toHex(chainId), - context - ] - : [deepHexlify(userOperation), entryPoint.address, toHex(chainId)] - }) - - if (entryPoint.version === "0.6") { - const responseV06 = response as GetRpcPaymasterDataReturnType<"0.6"> - - return { - paymasterAndData: responseV06.paymasterAndData - } as GetPaymasterDataReturnType - } - - const responseV07 = response as GetRpcPaymasterDataReturnType<"0.7"> - - return { - paymaster: responseV07.paymaster, - paymasterData: responseV07.paymasterData - } as GetPaymasterDataReturnType -} diff --git a/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.test.ts b/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.test.ts deleted file mode 100644 index c4747329..00000000 --- a/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.test.ts +++ /dev/null @@ -1,113 +0,0 @@ -import { zeroAddress } from "viem" -import { - entryPoint06Address, - entryPoint07Address -} from "viem/account-abstraction" -import { foundry } from "viem/chains" -import { describe, expect } from "vitest" -import { testWithRpc } from "../../../../permissionless-test/src/testWithRpc" -import { - getBundlerClient, - getPimlicoClient, - getSimpleAccountClient -} from "../../../../permissionless-test/src/utils" -import { paymasterActionsEip7677 } from "../clients/decorators/paymasterActionsEip7677" - -describe("EIP-7677 getPaymasterStubData", () => { - testWithRpc("getPaymasterStubData_V06", async ({ rpc }) => { - const { paymasterRpc } = rpc - - const simpleAccountClient = getBundlerClient({ - account: await getSimpleAccountClient({ - ...rpc, - entryPoint: { - version: "0.6" - } - }), - entryPoint: { - version: "0.6" - }, - ...rpc - }) - - const pimlicoPaymasterClient = getPimlicoClient({ - entryPointVersion: "0.6", - altoRpc: paymasterRpc - }).extend( - paymasterActionsEip7677({ - address: entryPoint06Address, - version: "0.6" - }) - ) - - const userOperation = await simpleAccountClient.prepareUserOperation({ - calls: [ - { - to: zeroAddress, - data: "0x", - value: 0n - } - ] - }) - - const stubData = await pimlicoPaymasterClient.getPaymasterStubData({ - userOperation, - chain: foundry - }) - expect(stubData).not.toBeNull() - expect(stubData?.paymasterAndData).not.toBeNull() - expect(stubData?.isFinal).toBe(false) - expect(stubData?.sponsor?.icon).toBeTruthy() - expect(stubData?.sponsor?.name).toBe("Pimlico") - }) - - testWithRpc("getPaymasterStubData_V07", async ({ rpc }) => { - const { paymasterRpc } = rpc - - const simpleAccountClient = getBundlerClient({ - account: await getSimpleAccountClient({ - ...rpc, - entryPoint: { - version: "0.7" - } - }), - entryPoint: { - version: "0.7" - }, - ...rpc - }) - - const pimlicoPaymasterClient = getPimlicoClient({ - entryPointVersion: "0.7", - altoRpc: paymasterRpc - }).extend( - paymasterActionsEip7677({ - address: entryPoint07Address, - version: "0.7" - }) - ) - - const userOperation = await simpleAccountClient.prepareUserOperation({ - calls: [ - { - to: zeroAddress, - data: "0x", - value: 0n - } - ] - }) - - const stubData = await pimlicoPaymasterClient.getPaymasterStubData({ - userOperation, - chain: foundry - }) - expect(stubData).not.toBeNull() - expect(stubData?.paymaster).not.toBeNull() - expect(stubData?.paymasterData).not.toBeNull() - expect(stubData?.paymasterPostOpGasLimit).not.toBeNull() - expect(stubData?.paymasterVerificationGasLimit).not.toBeNull() - expect(stubData?.isFinal).toBe(false) - expect(stubData?.sponsor?.icon).toBeTruthy() - expect(stubData?.sponsor?.name).toBe("Pimlico") - }) -}) diff --git a/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.ts b/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.ts deleted file mode 100644 index 594e3973..00000000 --- a/packages/permissionless/experimental/eip7677/actions/getPaymasterStubData.ts +++ /dev/null @@ -1,123 +0,0 @@ -import { - type Account, - type Chain, - ChainNotFoundError, - type Client, - type GetChainParameter, - type Hex, - type Transport, - toHex -} from "viem" -import type { - UserOperation, - entryPoint06Address, - entryPoint07Address -} from "viem/account-abstraction" -import { deepHexlify } from "../../../utils" -import type { - Eip7677RpcSchema, - GetRpcPaymasterStubDataReturnType -} from "../types/paymaster" - -export type GetPaymasterStubDataParameters< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7", - TChain extends Chain | undefined, - TChainOverride extends Chain | undefined -> = { - userOperation: UserOperation - entryPoint: { - address: entryPointAddress - version: entryPointVersion - } - context?: Record -} & GetChainParameter - -export type GetPaymasterStubDataReturnType< - entryPointVersion extends "0.6" | "0.7" -> = entryPointVersion extends "0.6" - ? { - paymasterAndData: Hex - sponsor?: { name: string; icon?: string } - isFinal?: boolean - } - : { - paymaster: Hex - paymasterData: Hex - paymasterVerificationGasLimit?: bigint - paymasterPostOpGasLimit?: bigint - sponsor?: { name: string; icon?: string } - isFinal?: boolean - } - -export async function getPaymasterStubData< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7", - TChain extends Chain | undefined, - TChainOverride extends Chain | undefined ->( - client: Client< - Transport, - TChain, - Account | undefined, - Eip7677RpcSchema - >, - { - userOperation, - entryPoint, - context, - chain - }: GetPaymasterStubDataParameters< - entryPointAddress, - entryPointVersion, - TChain, - TChainOverride - > -): Promise> { - const chainId = chain?.id ?? client.chain?.id - - if (!chainId) { - throw new ChainNotFoundError() - } - - const response = await client.request({ - method: "pm_getPaymasterStubData", - params: context - ? [ - deepHexlify(userOperation), - entryPoint.address, - toHex(chainId), - context - ] - : [deepHexlify(userOperation), entryPoint.address, toHex(chainId)] - }) - - if (entryPoint.version === "0.6") { - const responseV06 = response as GetRpcPaymasterStubDataReturnType<"0.6"> - - return { - paymasterAndData: responseV06.paymasterAndData, - sponsor: responseV06.sponsor, - isFinal: responseV06.isFinal - } as GetPaymasterStubDataReturnType - } - - const responseV07 = response as GetRpcPaymasterStubDataReturnType<"0.7"> - - return { - paymaster: responseV07.paymaster, - paymasterData: responseV07.paymasterData, - paymasterVerificationGasLimit: responseV07.paymasterVerificationGasLimit - ? BigInt(responseV07.paymasterVerificationGasLimit) - : undefined, - paymasterPostOpGasLimit: responseV07.paymasterPostOpGasLimit - ? BigInt(responseV07.paymasterPostOpGasLimit) - : undefined, - sponsor: responseV07.sponsor, - isFinal: responseV07.isFinal - } as GetPaymasterStubDataReturnType -} diff --git a/packages/permissionless/experimental/eip7677/clients/decorators/paymasterActionsEip7677.ts b/packages/permissionless/experimental/eip7677/clients/decorators/paymasterActionsEip7677.ts deleted file mode 100644 index 9cfa856b..00000000 --- a/packages/permissionless/experimental/eip7677/clients/decorators/paymasterActionsEip7677.ts +++ /dev/null @@ -1,85 +0,0 @@ -import type { Chain, Client, Transport } from "viem" -import type { - entryPoint06Address, - entryPoint07Address -} from "viem/account-abstraction" -import { - type GetPaymasterDataParameters, - type GetPaymasterDataReturnType, - getPaymasterData -} from "../../actions/getPaymasterData" -import { - type GetPaymasterStubDataParameters, - type GetPaymasterStubDataReturnType, - getPaymasterStubData -} from "../../actions/getPaymasterStubData" - -export type PaymasterActionsEip7677< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7", - TChain extends Chain | undefined -> = { - getPaymasterData: < - TChainOverride extends Chain | undefined = Chain | undefined - >( - args: Omit< - GetPaymasterDataParameters< - entryPointAddress, - entryPointVersion, - TChain, - TChainOverride - >, - "entryPoint" - > - ) => Promise> - getPaymasterStubData: < - TChainOverride extends Chain | undefined = Chain | undefined - >( - args: Omit< - GetPaymasterStubDataParameters< - entryPointAddress, - entryPointVersion, - TChain, - TChainOverride - >, - "entryPoint" - > - ) => Promise> -} - -const paymasterActionsEip7677 = - < - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7" - >(entryPoint: { - address: entryPointAddress - version: entryPointVersion - }) => - ( - client: Client - ): PaymasterActionsEip7677< - entryPointAddress, - entryPointVersion, - TChain - > => ({ - getPaymasterData: (args) => - getPaymasterData(client, { - userOperation: args.userOperation, - context: args.context, - chain: args.chain, - entryPoint - }), - getPaymasterStubData: async (args) => - getPaymasterStubData(client, { - userOperation: args.userOperation, - context: args.context, - chain: args.chain, - entryPoint - }) - }) - -export { paymasterActionsEip7677 } diff --git a/packages/permissionless/experimental/eip7677/index.ts b/packages/permissionless/experimental/eip7677/index.ts deleted file mode 100644 index 60669669..00000000 --- a/packages/permissionless/experimental/eip7677/index.ts +++ /dev/null @@ -1,27 +0,0 @@ -import { - type GetPaymasterDataParameters, - type GetPaymasterDataReturnType, - getPaymasterData -} from "./actions/getPaymasterData" -import { - type GetPaymasterStubDataParameters, - type GetPaymasterStubDataReturnType, - getPaymasterStubData -} from "./actions/getPaymasterStubData" -import { - type PaymasterActionsEip7677, - paymasterActionsEip7677 -} from "./clients/decorators/paymasterActionsEip7677" -import type { Eip7677RpcSchema } from "./types/paymaster" - -export { - type GetPaymasterStubDataParameters, - type GetPaymasterStubDataReturnType, - getPaymasterStubData, - type GetPaymasterDataReturnType, - type GetPaymasterDataParameters, - getPaymasterData, - type PaymasterActionsEip7677, - paymasterActionsEip7677, - type Eip7677RpcSchema -} diff --git a/packages/permissionless/experimental/eip7677/types/paymaster.ts b/packages/permissionless/experimental/eip7677/types/paymaster.ts deleted file mode 100644 index 2f8d97f0..00000000 --- a/packages/permissionless/experimental/eip7677/types/paymaster.ts +++ /dev/null @@ -1,77 +0,0 @@ -import type { Address, Hex, OneOf } from "viem" -import type { UserOperation } from "viem/account-abstraction" - -export type GetRpcPaymasterStubDataParameters< - entryPointVersion extends "0.6" | "0.7" -> = [ - userOperation: UserOperation, - entryPoint: Address, - chainId: Hex, - context?: Record -] - -export type GetRpcPaymasterStubDataReturnType< - entryPointVersion extends "0.6" | "0.7" -> = OneOf< - | (entryPointVersion extends "0.6" - ? { - paymasterAndData: Hex - sponsor?: { name: string; icon?: string } - isFinal?: boolean - } - : never) - | (entryPointVersion extends "0.7" - ? { - paymaster: Hex - paymasterData: Hex - paymasterVerificationGasLimit?: Hex | null - paymasterPostOpGasLimit?: Hex | null - sponsor?: { name: string; icon?: string } - isFinal?: boolean - } - : never) -> - -export type GetRpcPaymasterDataParameters< - entryPointVersion extends "0.6" | "0.7" -> = - | [ - userOperation: UserOperation, - entryPoint: Address, - chainId: Hex, - context?: Record - ] - | [ - userOperation: UserOperation, - entryPoint: Address, - chainId: Hex - ] - -export type GetRpcPaymasterDataReturnType< - entryPointVersion extends "0.6" | "0.7" -> = OneOf< - | (entryPointVersion extends "0.6" - ? { - paymasterAndData: Hex - } - : never) - | (entryPointVersion extends "0.7" - ? { - paymaster: Hex - paymasterData: Hex - } - : never) -> - -export type Eip7677RpcSchema = [ - { - Method: "pm_getPaymasterStubData" - Parameters: GetRpcPaymasterStubDataParameters - ReturnType: GetRpcPaymasterStubDataReturnType - }, - { - Method: "pm_getPaymasterData" - Parameters: GetRpcPaymasterDataParameters - ReturnType: GetRpcPaymasterDataReturnType - } -] diff --git a/packages/permissionless/experimental/index.ts b/packages/permissionless/experimental/index.ts deleted file mode 100644 index 4952309b..00000000 --- a/packages/permissionless/experimental/index.ts +++ /dev/null @@ -1 +0,0 @@ -export * from "./eip7677" From 5883332b7be215c44efe4cc270a0edb4ace27588 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Fri, 23 Aug 2024 18:58:13 +0100 Subject: [PATCH 16/51] remove redundant smart account address generation code --- .../biconomy/toBiconomySmartAccount.ts | 132 +++-------- packages/permissionless/accounts/index.ts | 5 +- .../kernel/toEcdsaKernelSmartAccount.ts | 98 +++----- .../accounts/light/toLightSmartAccount.ts | 89 +++---- .../accounts/safe/toSafeSmartAccount.ts | 220 ++++-------------- .../accounts/simple/toSimpleSmartAccount.ts | 84 +++---- .../accounts/trust/toTrustSmartAccount.ts | 64 ++--- .../accounts/trust/utils/getAccountAddress.ts | 35 --- packages/permissionless/accounts/types.ts | 17 -- .../actions/public/getSenderAddress.ts | 24 +- 10 files changed, 199 insertions(+), 569 deletions(-) delete mode 100644 packages/permissionless/accounts/trust/utils/getAccountAddress.ts delete mode 100644 packages/permissionless/accounts/types.ts diff --git a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts index 95ddada7..4f3b4d64 100644 --- a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts +++ b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts @@ -6,10 +6,6 @@ import { type LocalAccount, encodeAbiParameters, encodeFunctionData, - encodePacked, - getContractAddress, - hexToBigInt, - keccak256, parseAbiParameters } from "viem" import { @@ -22,6 +18,7 @@ import { } from "viem/account-abstraction" import { signMessage } from "viem/actions" import { getAccountNonce } from "../../actions/public/getAccountNonce" +import { getSenderAddress } from "../../actions/public/getSenderAddress" import { BiconomyAbi, FactoryAbi } from "./abi/BiconomySmartAccountAbi" /** @@ -33,21 +30,13 @@ import { BiconomyAbi, FactoryAbi } from "./abi/BiconomySmartAccountAbi" */ const BICONOMY_ADDRESSES: { ECDSA_OWNERSHIP_REGISTRY_MODULE: Address - ACCOUNT_V2_0_LOGIC: Address FACTORY_ADDRESS: Address - DEFAULT_FALLBACK_HANDLER_ADDRESS: Address } = { ECDSA_OWNERSHIP_REGISTRY_MODULE: "0x0000001c5b32F37F5beA87BDD5374eB2aC54eA8e", - ACCOUNT_V2_0_LOGIC: "0x0000002512019Dafb59528B82CB92D3c5D2423aC", - FACTORY_ADDRESS: "0x000000a56Aaca3e9a4C479ea6b6CD0DbcB6634F5", - DEFAULT_FALLBACK_HANDLER_ADDRESS: - "0x0bBa6d96BD616BedC6BFaa341742FD43c60b83C1" + FACTORY_ADDRESS: "0x000000a56Aaca3e9a4C479ea6b6CD0DbcB6634F5" } -const BICONOMY_PROXY_CREATION_CODE = - "0x6080346100aa57601f61012038819003918201601f19168301916001600160401b038311848410176100af578084926020946040528339810103126100aa57516001600160a01b0381168082036100aa5715610065573055604051605a90816100c68239f35b60405162461bcd60e51b815260206004820152601e60248201527f496e76616c696420696d706c656d656e746174696f6e206164647265737300006044820152606490fd5b600080fd5b634e487b7160e01b600052604160045260246000fdfe608060405230546000808092368280378136915af43d82803e156020573d90f35b3d90fdfea2646970667358221220a03b18dce0be0b4c9afe58a9eb85c35205e2cf087da098bbf1d23945bf89496064736f6c63430008110033" - /** * Get the account initialization code for Biconomy smart account with ECDSA as default authorization module * @param owner @@ -81,59 +70,6 @@ const getAccountInitCode = async ({ }) } -const getAccountAddress = async ({ - factoryAddress, - accountLogicAddress, - fallbackHandlerAddress, - ecdsaModuleAddress, - owner, - index = BigInt(0) -}: { - factoryAddress: Address - accountLogicAddress: Address - fallbackHandlerAddress: Address - ecdsaModuleAddress: Address - owner: Address - index?: bigint -}): Promise
=> { - // Build the module setup data - const ecdsaOwnershipInitData = encodeFunctionData({ - abi: BiconomyAbi, - functionName: "initForSmartAccount", - args: [owner] - }) - - // Build account init code - const initialisationData = encodeFunctionData({ - abi: BiconomyAbi, - functionName: "init", - args: [ - fallbackHandlerAddress, - ecdsaModuleAddress, - ecdsaOwnershipInitData - ] - }) - - const deploymentCode = encodePacked( - ["bytes", "uint256"], - [BICONOMY_PROXY_CREATION_CODE, hexToBigInt(accountLogicAddress)] - ) - - const salt = keccak256( - encodePacked( - ["bytes32", "uint256"], - [keccak256(encodePacked(["bytes"], [initialisationData])), index] - ) - ) - - return getContractAddress({ - from: factoryAddress, - salt, - bytecode: deploymentCode, - opcode: "CREATE2" - }) -} - export type ToBiconomySmartAccountParameters< entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi > = Prettify<{ @@ -148,8 +84,6 @@ export type ToBiconomySmartAccountParameters< nonceKey?: bigint index?: bigint factoryAddress?: Address - accountLogicAddress?: Address - fallbackHandlerAddress?: Address ecdsaModuleAddress?: Address }> @@ -171,7 +105,6 @@ export type ToBiconomySmartAccountReturnType< * @param entryPoint * @param index * @param factoryAddress - * @param accountLogicAddress * @param ecdsaModuleAddress */ @@ -199,42 +132,43 @@ export async function toBiconomySmartAccount< const factoryAddress = parameters.factoryAddress ?? BICONOMY_ADDRESSES.FACTORY_ADDRESS - let accountAddress: Address + let accountAddress: Address | undefined = address const ecdsaModuleAddress = parameters.ecdsaModuleAddress ?? BICONOMY_ADDRESSES.ECDSA_OWNERSHIP_REGISTRY_MODULE - const accountLogicAddress = - parameters.accountLogicAddress ?? BICONOMY_ADDRESSES.ACCOUNT_V2_0_LOGIC - - const fallbackHandlerAddress = - parameters.fallbackHandlerAddress ?? - BICONOMY_ADDRESSES.DEFAULT_FALLBACK_HANDLER_ADDRESS - - const getAddress = async () => { - if (accountAddress) return accountAddress - - accountAddress = - address ?? - (await getAccountAddress({ + const getFactoryArgs = async () => { + return { + factory: factoryAddress, + factoryData: await getAccountInitCode({ owner: owner.address, - ecdsaModuleAddress, - factoryAddress, - accountLogicAddress, - fallbackHandlerAddress, - index - })) - - return accountAddress + index, + ecdsaModuleAddress + }) + } } return toSmartAccount({ client, entryPoint, - getAddress, + getFactoryArgs, + async getAddress() { + if (accountAddress) return accountAddress + + const { factory, factoryData } = await getFactoryArgs() + + // Get the sender address based on the init code + accountAddress = await getSenderAddress(client, { + factory, + factoryData, + entryPointAddress: entryPoint.address + }) + + return accountAddress + }, async getNonce(args) { - const address = await getAddress() + const address = await this.getAddress() return getAccountNonce(client, { address, entryPointAddress: entryPoint.address, @@ -262,16 +196,6 @@ export async function toBiconomySmartAccount< args: [to, value ?? 0n, data ?? "0x"] }) }, - getFactoryArgs: async () => { - return { - factory: factoryAddress, - factoryData: await getAccountInitCode({ - owner: owner.address, - index, - ecdsaModuleAddress - }) - } - }, // Get simple dummy signature for ECDSA module authorization async getStubSignature() { const dynamicPart = ecdsaModuleAddress.substring(2).padEnd(40, "0") @@ -325,7 +249,7 @@ export async function toBiconomySmartAccount< const hash = getUserOperationHash({ userOperation: { ...userOperation, - sender: userOperation.sender ?? (await getAddress()), + sender: userOperation.sender ?? (await this.getAddress()), signature: "0x" }, entryPointAddress: entryPoint.address, diff --git a/packages/permissionless/accounts/index.ts b/packages/permissionless/accounts/index.ts index 982bde04..e70cd175 100644 --- a/packages/permissionless/accounts/index.ts +++ b/packages/permissionless/accounts/index.ts @@ -43,8 +43,6 @@ import { toBiconomySmartAccount } from "./biconomy/toBiconomySmartAccount" -import { SignTransactionNotSupportedBySmartAccount } from "./types" - export { type ToSimpleSmartAccountParameters, type SimpleSmartAccountImplementation, @@ -72,6 +70,5 @@ export { type ToBiconomySmartAccountReturnType, type ToBiconomySmartAccountParameters, type BiconomySmartAccountImplementation, - toBiconomySmartAccount, - SignTransactionNotSupportedBySmartAccount + toBiconomySmartAccount } diff --git a/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts b/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts index da0fc4b6..ebe56928 100644 --- a/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts +++ b/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts @@ -290,42 +290,6 @@ const getAccountInitCode = async ({ }) } -/** - * Check the validity of an existing account address, or fetch the pre-deterministic account address for a kernel smart wallet - * @param client - * @param owner - * @param entryPoint - * @param ecdsaValidatorAddress - * @param initCodeProvider - * @param factoryAddress - */ -const getAccountAddress = async ({ - client, - entryPointVersion, - entryPointAddress, - factory, - factoryData -}: { - client: Client - entryPointVersion: "0.6" | "0.7" - entryPointAddress: typeof entryPoint06Address | typeof entryPoint07Address - factory: Address - factoryData: Hex -}): Promise
=> { - // Find the init code for this account - if (entryPointVersion === "0.6") { - return getSenderAddress(client, { - initCode: concatHex([factory, factoryData]), - entryPointAddress: entryPointAddress as typeof entryPoint06Address - }) - } - return getSenderAddress(client, { - factory: factory, - factoryData: factoryData, - entryPointAddress: entryPointAddress as typeof entryPoint07Address - }) -} - export type ToEcdsaKernelSmartAccountParameters< entryPointVersion extends "0.6" | "0.7", entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi, @@ -446,24 +410,7 @@ export async function toEcdsaKernelSmartAccount< ecdsaValidatorAddress }) - let accountAddress: Address - - const getAddress = async () => { - if (accountAddress) return accountAddress - accountAddress = - address ?? - (await getAccountAddress({ - client, - factory: - entryPoint.version === "0.6" - ? factoryAddress - : metaFactoryAddress, - factoryData: await generateInitCode(), - entryPointAddress: entryPoint.address, - entryPointVersion: entryPoint.version - })) - return accountAddress - } + let accountAddress: Address | undefined = address let chainId: number @@ -475,25 +422,40 @@ export async function toEcdsaKernelSmartAccount< return chainId } + const getFactoryArgs = async () => { + return { + factory: + entryPoint.version === "0.6" + ? factoryAddress + : metaFactoryAddress, + factoryData: await generateInitCode() + } + } + return toSmartAccount({ client, entryPoint, - getAddress, + getFactoryArgs, + async getAddress() { + if (accountAddress) return accountAddress + + const { factory, factoryData } = await getFactoryArgs() + + // Get the sender address based on the init code + accountAddress = await getSenderAddress(client, { + factory, + factoryData, + entryPointAddress: entryPoint.address + }) + + return accountAddress + }, async encodeCalls(calls) { return encodeCallData({ calls, kernelVersion }) }, - async getFactoryArgs() { - return { - factory: - entryPoint.version === "0.6" - ? factoryAddress - : metaFactoryAddress, - factoryData: await generateInitCode() - } - }, async getNonce(_args) { return getAccountNonce(client, { - address: await getAddress(), + address: await this.getAddress(), entryPointAddress: entryPoint.address, key: getNonceKeyWithEncoding( kernelVersion, @@ -515,7 +477,7 @@ export async function toEcdsaKernelSmartAccount< const signature = await signMessage({ owner: owner, message, - accountAddress, + accountAddress: await this.getAddress(), kernelVersion, chainId: await getMemoizedChainId() }) @@ -534,7 +496,7 @@ export async function toEcdsaKernelSmartAccount< owner, chainId: await getMemoizedChainId(), ...(typedData as TypedDataDefinition), - accountAddress, + accountAddress: await this.getAddress(), kernelVersion }) @@ -555,7 +517,7 @@ export async function toEcdsaKernelSmartAccount< const hash = getUserOperationHash({ userOperation: { ...userOperation, - sender: userOperation.sender ?? (await getAddress()), + sender: userOperation.sender ?? (await this.getAddress()), signature: "0x" } as UserOperation, entryPointAddress: entryPoint.address, diff --git a/packages/permissionless/accounts/light/toLightSmartAccount.ts b/packages/permissionless/accounts/light/toLightSmartAccount.ts index 28f71d82..a906ecb5 100644 --- a/packages/permissionless/accounts/light/toLightSmartAccount.ts +++ b/packages/permissionless/accounts/light/toLightSmartAccount.ts @@ -4,7 +4,6 @@ import { type Client, type Hex, type LocalAccount, - concatHex, encodeFunctionData, hashMessage, hashTypedData @@ -63,43 +62,6 @@ const getAccountInitCode = async ( }) } -const getAccountAddress = async < - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7" ->({ - client, - factoryAddress, - entryPointAddress, - entryPointVersion, - owner, - index = BigInt(0) -}: { - client: Client - factoryAddress: Address - owner: Address - entryPointAddress: entryPointAddress - entryPointVersion: entryPointVersion - index?: bigint -}): Promise
=> { - const factoryData = await getAccountInitCode(owner, index) - - if (entryPointVersion === "0.6") { - return getSenderAddress(client, { - initCode: concatHex([factoryAddress, factoryData]), - entryPointAddress: entryPointAddress as typeof entryPoint06Address - }) - } - - // Get the sender address based on the init code - return getSenderAddress(client, { - factory: factoryAddress, - factoryData, - entryPointAddress: entryPointAddress as typeof entryPoint07Address - }) -} - export type LightAccountVersion = "1.1.0" export type ToLightSmartAccountParameters< @@ -240,21 +202,7 @@ export async function toLightSmartAccount< factoryAddress: _factoryAddress }) - let accountAddress: Address - const getAddress = async () => { - if (accountAddress) return accountAddress - accountAddress = - address ?? - (await getAccountAddress({ - client, - factoryAddress, - entryPointAddress: entryPoint.address, - entryPointVersion: entryPoint.version, - owner: owner.address, - index - })) - return accountAddress - } + let accountAddress: Address | undefined = address let chainId: number @@ -266,10 +214,31 @@ export async function toLightSmartAccount< return chainId } + const getFactoryArgs = async () => { + return { + factory: factoryAddress, + factoryData: await getAccountInitCode(owner.address, index) + } + } + return toSmartAccount({ client, entryPoint, - getAddress, + getFactoryArgs, + async getAddress() { + if (accountAddress) return accountAddress + + const { factory, factoryData } = await getFactoryArgs() + + // Get the sender address based on the init code + accountAddress = await getSenderAddress(client, { + factory, + factoryData, + entryPointAddress: entryPoint.address + }) + + return accountAddress + }, async encodeCalls(calls) { if (calls.length > 1) { return encodeFunctionData({ @@ -337,15 +306,9 @@ export async function toLightSmartAccount< args: [calls[0].to, calls[0].value ?? 0n, calls[0].data ?? "0x"] }) }, - async getFactoryArgs() { - return { - factory: factoryAddress, - factoryData: await getAccountInitCode(owner.address, index) - } - }, async getNonce(args) { return getAccountNonce(client, { - address: await getAddress(), + address: await this.getAddress(), entryPointAddress: entryPoint.address, key: args?.key ?? nonceKey }) @@ -360,7 +323,7 @@ export async function toLightSmartAccount< return signWith1271WrapperV1( owner, await getMemoizedChainId(), - await getAddress(), + await this.getAddress(), hashMessage(message) ) }, @@ -368,7 +331,7 @@ export async function toLightSmartAccount< return signWith1271WrapperV1( owner, await getMemoizedChainId(), - await getAddress(), + await this.getAddress(), hashTypedData(typedData) ) }, diff --git a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts index 62d158b3..7cb8a5cd 100644 --- a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts +++ b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts @@ -1,13 +1,10 @@ import { - type Account, type Address, type Assign, - type Chain, type Client, type Hex, type LocalAccount, type SignableMessage, - type Transport, type TypedData, type TypedDataDefinition, concat, @@ -15,10 +12,8 @@ import { encodeAbiParameters, encodeFunctionData, encodePacked, - getContractAddress, hashMessage, hashTypedData, - hexToBigInt, keccak256, pad, toBytes, @@ -35,9 +30,10 @@ import { entryPoint07Address, toSmartAccount } from "viem/account-abstraction" -import { getChainId, readContract, signTypedData } from "viem/actions" +import { getChainId, signTypedData } from "viem/actions" import { getAction } from "viem/utils" import { getAccountNonce } from "../../actions/public/getAccountNonce" +import { getSenderAddress } from "../../actions/public/getSenderAddress" import { isSmartAccountDeployed } from "../../utils" import { encode7579Calls } from "../../utils/encode7579Calls" @@ -261,22 +257,6 @@ const createProxyWithNonceAbi = [ } ] as const -const proxyCreationCodeAbi = [ - { - inputs: [], - name: "proxyCreationCode", - outputs: [ - { - internalType: "bytes", - name: "", - type: "bytes" - } - ], - stateMutability: "pure", - type: "function" - } -] as const - const setupSafeAbi = [ { type: "function", @@ -864,109 +844,6 @@ const getAccountInitCode = async ({ return initCodeCallData } -const getAccountAddress = async < - TTransport extends Transport = Transport, - TChain extends Chain | undefined = Chain | undefined, - TClientAccount extends Account | undefined = undefined ->({ - client, - owner, - safeModuleSetupAddress, - safe4337ModuleAddress, - safeProxyFactoryAddress, - safeSingletonAddress, - multiSendAddress, - erc7579LaunchpadAddress, - paymentToken, - payment, - paymentReceiver, - setupTransactions = [], - safeModules = [], - saltNonce = BigInt(0), - validators = [], - executors = [], - fallbacks = [], - hooks = [], - attesters = [], - attestersThreshold = 0 -}: { - client: Client - owner: Address - safeModuleSetupAddress: Address - safe4337ModuleAddress: Address - safeProxyFactoryAddress: Address - safeSingletonAddress: Address - multiSendAddress: Address - setupTransactions: { - to: Address - data: Address - value: bigint - }[] - safeModules?: Address[] - saltNonce?: bigint - erc7579LaunchpadAddress?: Address - validators?: { address: Address; context: Address }[] - executors?: { - address: Address - context: Address - }[] - fallbacks?: { address: Address; context: Address }[] - hooks?: { address: Address; context: Address }[] - attesters?: Address[] - attestersThreshold?: number - paymentToken?: Address - payment?: bigint - paymentReceiver?: Address -}): Promise
=> { - const proxyCreationCode = await readContract(client, { - abi: proxyCreationCodeAbi, - address: safeProxyFactoryAddress, - functionName: "proxyCreationCode" - }) - - const initializer = await getInitializerCode({ - owner, - safeModuleSetupAddress, - safe4337ModuleAddress, - multiSendAddress, - setupTransactions, - safeSingletonAddress, - safeModules, - erc7579LaunchpadAddress, - validators, - executors, - fallbacks, - hooks, - attesters, - attestersThreshold, - paymentToken, - payment, - paymentReceiver - }) - - const deploymentCode = encodePacked( - ["bytes", "uint256"], - [ - proxyCreationCode, - hexToBigInt(erc7579LaunchpadAddress ?? safeSingletonAddress) - ] - ) - - const salt = keccak256( - encodePacked( - ["bytes32", "uint256"], - [keccak256(encodePacked(["bytes"], [initializer])), saltNonce] - ) - ) - - return getContractAddress({ - from: safeProxyFactoryAddress, - salt, - bytecode: deploymentCode, - opcode: "CREATE2" - }) -} - const getDefaultAddresses = ( safeVersion: SafeVersion, entryPointVersion: "0.6" | "0.7", @@ -1217,18 +1094,25 @@ export async function toSafeSmartAccount< multiSendCallOnlyAddress: _multiSendCallOnlyAddress }) - let accountAddress: Address + let accountAddress: Address | undefined = address + + let chainId: number - const getAddress = async () => { - if (accountAddress) return accountAddress - accountAddress = - address ?? - (await getAccountAddress({ - client, + const getMemoizedChainId = async () => { + if (chainId) return chainId + chainId = client.chain + ? client.chain.id + : await getAction(client, getChainId, "getChainId")({}) + return chainId + } + + const getFactoryArgs = async () => { + return { + factory: safeProxyFactoryAddress, + factoryData: await getAccountInitCode({ owner: owner.address, safeModuleSetupAddress, safe4337ModuleAddress, - safeProxyFactoryAddress, safeSingletonAddress, multiSendAddress, erc7579LaunchpadAddress, @@ -1244,31 +1128,35 @@ export async function toSafeSmartAccount< paymentToken, payment, paymentReceiver - })) - return accountAddress - } - - let chainId: number - - const getMemoizedChainId = async () => { - if (chainId) return chainId - chainId = client.chain - ? client.chain.id - : await getAction(client, getChainId, "getChainId")({}) - return chainId + }) + } } return toSmartAccount({ client, entryPoint, - getAddress, + getFactoryArgs, + async getAddress() { + if (accountAddress) return accountAddress + + const { factory, factoryData } = await getFactoryArgs() + + // Get the sender address based on the init code + accountAddress = await getSenderAddress(client, { + factory, + factoryData, + entryPointAddress: entryPoint.address + }) + + return accountAddress + }, async encodeCalls(calls) { const hasMultipleCalls = calls.length > 1 if (erc7579LaunchpadAddress) { const safeDeployed = await isSmartAccountDeployed( client, - await getAddress() + await this.getAddress() ) if (!safeDeployed) { @@ -1354,34 +1242,9 @@ export async function toSafeSmartAccount< args: [to, value, data, operationType] }) }, - async getFactoryArgs() { - return { - factory: safeProxyFactoryAddress, - factoryData: await getAccountInitCode({ - owner: owner.address, - safeModuleSetupAddress, - safe4337ModuleAddress, - safeSingletonAddress, - multiSendAddress, - erc7579LaunchpadAddress, - saltNonce, - setupTransactions, - safeModules, - validators, - executors, - fallbacks, - hooks, - attesters, - attestersThreshold, - paymentToken, - payment, - paymentReceiver - }) - } - }, async getNonce(args) { return getAccountNonce(client, { - address: await getAddress(), + address: await this.getAddress(), entryPointAddress: entryPoint.address, key: args?.key ?? nonceKey }) @@ -1396,7 +1259,7 @@ export async function toSafeSmartAccount< const messageHash = hashTypedData({ domain: { chainId: await getMemoizedChainId(), - verifyingContract: await getAddress() + verifyingContract: await this.getAddress() }, types: { SafeMessage: [{ name: "message", type: "bytes" }] @@ -1422,7 +1285,7 @@ export async function toSafeSmartAccount< await owner.signTypedData({ domain: { chainId: await getMemoizedChainId(), - verifyingContract: await getAddress() + verifyingContract: await this.getAddress() }, types: { SafeMessage: [{ name: "message", type: "bytes" }] @@ -1439,7 +1302,7 @@ export async function toSafeSmartAccount< parameters const message = { - safe: await getAddress(), + safe: await this.getAddress(), callData: userOperation.callData, nonce: userOperation.nonce, initCode: userOperation.initCode ?? "0x", @@ -1471,7 +1334,7 @@ export async function toSafeSmartAccount< } message.paymasterAndData = getPaymasterAndData({ ...userOperation, - sender: userOperation.sender ?? (await getAddress()) + sender: userOperation.sender ?? (await this.getAddress()) }) isDeployed = !userOperation.factory } @@ -1479,7 +1342,8 @@ export async function toSafeSmartAccount< let verifyingContract = safe4337ModuleAddress if (erc7579LaunchpadAddress && !isDeployed) { - verifyingContract = userOperation.sender ?? (await getAddress()) + verifyingContract = + userOperation.sender ?? (await this.getAddress()) } const signatures = [ diff --git a/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts b/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts index a249dfd3..a7344c2a 100644 --- a/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts +++ b/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts @@ -4,7 +4,6 @@ import { type Client, type Hex, type LocalAccount, - concatHex, encodeFunctionData } from "viem" import { @@ -61,38 +60,6 @@ const getAccountInitCode = async ( }) } -const getAccountAddress = async ({ - client, - factoryAddress, - entryPointAddress, - owner, - entryPointVersion, - index = BigInt(0) -}: { - client: Client - factoryAddress: Address - owner: Address - entryPointAddress: typeof entryPoint06Address | typeof entryPoint07Address - entryPointVersion: "0.6" | "0.7" - index?: bigint -}): Promise
=> { - const factoryData = await getAccountInitCode(owner, index) - - if (entryPointVersion === "0.6") { - return getSenderAddress(client, { - initCode: concatHex([factoryAddress, factoryData]), - entryPointAddress: entryPointAddress as typeof entryPoint06Address - }) - } - - // Get the sender address based on the init code - return getSenderAddress(client, { - factory: factoryAddress, - factoryData, - entryPointAddress: entryPointAddress as typeof entryPoint07Address - }) -} - export type ToSimpleSmartAccountParameters< entryPointVersion extends "0.6" | "0.7", entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi @@ -184,22 +151,7 @@ export async function toSimpleSmartAccount< _factoryAddress ) - let accountAddress: Address - - const getAddress = async () => { - if (accountAddress) return accountAddress - accountAddress = - address ?? - (await getAccountAddress({ - client, - factoryAddress, - entryPointAddress: entryPoint.address, - entryPointVersion: entryPoint.version, - owner: owner.address, - index - })) - return accountAddress - } + let accountAddress: Address | undefined = address let chainId: number @@ -211,10 +163,31 @@ export async function toSimpleSmartAccount< return chainId } + const getFactoryArgs = async () => { + return { + factory: factoryAddress, + factoryData: await getAccountInitCode(owner.address, index) + } + } + return toSmartAccount({ client, entryPoint, - getAddress, + getFactoryArgs, + async getAddress() { + if (accountAddress) return accountAddress + + const { factory, factoryData } = await getFactoryArgs() + + // Get the sender address based on the init code + accountAddress = await getSenderAddress(client, { + factory, + factoryData, + entryPointAddress: entryPoint.address + }) + + return accountAddress + }, async encodeCalls(calls) { if (calls.length > 1) { if (entryPoint.version === "0.6") { @@ -311,15 +284,9 @@ export async function toSimpleSmartAccount< args: [calls[0].to, calls[0].value ?? 0n, calls[0].data ?? "0x"] }) }, - async getFactoryArgs() { - return { - factory: factoryAddress, - factoryData: await getAccountInitCode(owner.address, index) - } - }, async getNonce(args) { return getAccountNonce(client, { - address: accountAddress, + address: await this.getAddress(), entryPointAddress: entryPoint.address, key: args?.key ?? nonceKey }) @@ -346,7 +313,8 @@ export async function toSimpleSmartAccount< userOperation: { ...userOperation, sender: - userOperation.sender ?? (await getAddress()), + userOperation.sender ?? + (await this.getAddress()), signature: "0x" } as UserOperation, entryPointAddress: entryPoint.address, diff --git a/packages/permissionless/accounts/trust/toTrustSmartAccount.ts b/packages/permissionless/accounts/trust/toTrustSmartAccount.ts index ee00e92e..29ef1ab7 100644 --- a/packages/permissionless/accounts/trust/toTrustSmartAccount.ts +++ b/packages/permissionless/accounts/trust/toTrustSmartAccount.ts @@ -20,8 +20,8 @@ import { toSmartAccount } from "viem/account-abstraction" import { getAction } from "viem/utils" +import { getSenderAddress } from "../../actions/public/getSenderAddress" import { encodeCallData } from "./utils/encodeCallData" -import { getAccountAddress } from "./utils/getAccountAddress" import { getFactoryData } from "./utils/getFactoryData" async function _signTypedData( @@ -120,7 +120,7 @@ export async function toTrustSmartAccount< secp256k1VerificationFacetAddress = TRUST_ADDRESSES.secp256k1VerificationFacetAddress } = parameters - let accountAddress: Address + let accountAddress: Address | undefined = address const entryPoint = { address: parameters.entryPoint?.address ?? entryPoint06Address, @@ -128,20 +128,6 @@ export async function toTrustSmartAccount< version: parameters.entryPoint?.version ?? "0.6" } as const - const getAddress = async () => { - if (accountAddress) return accountAddress - accountAddress = - address ?? - (await getAccountAddress(client, { - factoryAddress, - secp256k1VerificationFacetAddress, - entryPoint: entryPoint.address, - bytes: owner.address, - index - })) - return accountAddress - } - let chainId: number const getMemoizedChainId = async () => { @@ -152,26 +138,41 @@ export async function toTrustSmartAccount< return chainId } + const getFactoryArgs = async () => { + return { + factory: factoryAddress, + factoryData: await getFactoryData({ + bytes: owner.address, + secp256k1VerificationFacetAddress, + index + }) + } + } + return toSmartAccount({ client, entryPoint, - getAddress, + getFactoryArgs, + async getAddress() { + if (accountAddress) return accountAddress + + const { factory, factoryData } = await getFactoryArgs() + + // Get the sender address based on the init code + accountAddress = await getSenderAddress(client, { + factory, + factoryData, + entryPointAddress: entryPoint.address + }) + + return accountAddress + }, async encodeCalls(calls) { return encodeCallData(calls) }, - async getFactoryArgs() { - return { - factory: factoryAddress, - factoryData: await getFactoryData({ - bytes: owner.address, - secp256k1VerificationFacetAddress, - index - }) - } - }, async getNonce(args) { return getAccountNonce(client, { - address: await getAddress(), + address: await this.getAddress(), entryPointAddress: entryPoint.address, key: args?.key ?? parameters?.nonceKey }) @@ -186,7 +187,7 @@ export async function toTrustSmartAccount< return _signTypedData( owner, await getMemoizedChainId(), - await getAddress(), + await this.getAddress(), hashMessage(message) ) }, @@ -194,7 +195,7 @@ export async function toTrustSmartAccount< return _signTypedData( owner, await getMemoizedChainId(), - await getAddress(), + await this.getAddress(), hashTypedData(typedData) ) }, @@ -209,7 +210,8 @@ export async function toTrustSmartAccount< userOperation: { ...userOperation, sender: - userOperation.sender ?? (await getAddress()), + userOperation.sender ?? + (await this.getAddress()), signature: "0x" } as UserOperation, entryPointAddress: entryPoint.address, diff --git a/packages/permissionless/accounts/trust/utils/getAccountAddress.ts b/packages/permissionless/accounts/trust/utils/getAccountAddress.ts deleted file mode 100644 index 96d3f17f..00000000 --- a/packages/permissionless/accounts/trust/utils/getAccountAddress.ts +++ /dev/null @@ -1,35 +0,0 @@ -import { type Address, type Client, concatHex } from "viem" -import type { entryPoint06Address } from "viem/account-abstraction" -import { getSenderAddress } from "../../../actions/public/getSenderAddress" -import { getFactoryData } from "./getFactoryData" - -export const getAccountAddress = async < - entryPointAddress extends - typeof entryPoint06Address = typeof entryPoint06Address ->( - client: Client, - { - factoryAddress, - entryPoint: entryPointAddress, - bytes, - secp256k1VerificationFacetAddress, - index = 0n - }: { - factoryAddress: Address - bytes: `0x${string}` - entryPoint: entryPointAddress - secp256k1VerificationFacetAddress: Address - index?: bigint - } -): Promise
=> { - const factoryData = await getFactoryData({ - bytes, - index, - secp256k1VerificationFacetAddress - }) - - return getSenderAddress(client, { - initCode: concatHex([factoryAddress, factoryData]), - entryPointAddress: entryPointAddress - }) -} diff --git a/packages/permissionless/accounts/types.ts b/packages/permissionless/accounts/types.ts deleted file mode 100644 index d722253e..00000000 --- a/packages/permissionless/accounts/types.ts +++ /dev/null @@ -1,17 +0,0 @@ -import { BaseError } from "viem" - -export class SignTransactionNotSupportedBySmartAccount extends BaseError { - override name = "SignTransactionNotSupportedBySmartAccount" - constructor({ docsPath }: { docsPath?: string } = {}) { - super( - [ - "A smart account cannot sign or send transaction, it can only sign message or userOperation.", - "Please send user operation instead." - ].join("\n"), - { - docsPath, - docsSlug: "account" - } - ) - } -} diff --git a/packages/permissionless/actions/public/getSenderAddress.ts b/packages/permissionless/actions/public/getSenderAddress.ts index 163e7d68..ff720082 100644 --- a/packages/permissionless/actions/public/getSenderAddress.ts +++ b/packages/permissionless/actions/public/getSenderAddress.ts @@ -6,6 +6,7 @@ import { ContractFunctionRevertedError, type Hex, InvalidInputRpcError, + type OneOf, type Prettify, RpcRequestError, UnknownRpcError, @@ -20,21 +21,24 @@ import type { import { simulateContract } from "viem/actions" import { getAction } from "viem/utils" -export type GetSenderAddressParams< - entryPoint extends typeof entryPoint06Address | typeof entryPoint07Address -> = entryPoint extends typeof entryPoint06Address - ? { +export type GetSenderAddressParams = OneOf< + | { initCode: Hex - entryPointAddress: entryPoint + entryPointAddress: + | typeof entryPoint06Address + | typeof entryPoint07Address factory?: never factoryData?: never } - : { - entryPointAddress: entryPoint + | { + entryPointAddress: + | typeof entryPoint06Address + | typeof entryPoint07Address factory: Address factoryData: Hex initCode?: never } +> export class InvalidEntryPointError extends BaseError { override name = "InvalidEntryPointError" @@ -79,11 +83,9 @@ export class InvalidEntryPointError extends BaseError { * * // Return '0x7a88a206ba40b37a8c07a2b5688cf8b287318b63' */ -export const getSenderAddress = async < - entryPoint extends typeof entryPoint06Address | typeof entryPoint07Address ->( +export const getSenderAddress = async ( client: Client, - args: Prettify> + args: Prettify ): Promise
=> { const { initCode, entryPointAddress, factory, factoryData } = args From 353175f652e80a49039386fb388394920db881fb Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Sat, 24 Aug 2024 15:47:04 +0100 Subject: [PATCH 17/51] lightaccount v0.2 --- .../mock-aa-infra/alto/constants.ts | 3 + .../mock-aa-infra/alto/index.ts | 9 ++ .../mock-aa-infra/mock-paymaster/relay.ts | 4 - packages/permissionless-test/src/utils.ts | 44 +++++++-- .../accounts/light/toLightSmartAccount.ts | 93 +++++++++++++------ .../actions/smartAccount/signMessage.test.ts | 1 + 6 files changed, 112 insertions(+), 42 deletions(-) diff --git a/packages/permissionless-test/mock-aa-infra/alto/constants.ts b/packages/permissionless-test/mock-aa-infra/alto/constants.ts index 4e0b07ae..75d32d2d 100644 --- a/packages/permissionless-test/mock-aa-infra/alto/constants.ts +++ b/packages/permissionless-test/mock-aa-infra/alto/constants.ts @@ -135,6 +135,9 @@ export const KERNEL_V07_META_FACTORY_CREATECALL: Hex = export const LIGHT_ACCOUNT_FACTORY_V110_CREATECALL: Hex = "0x4e59b44847b379578588920ca78fbf26c0b4956c5528f3e2f146000008fabf7760a0346100cb576001600160401b0390601f6130cb38819003918201601f1916830191848311848410176100b5578084926020946040528339810103126100cb57516001600160a01b038116908190036100cb576040519161270590818401908111848210176100b55760209284926109c6843981520301906000f080156100a9576080526040516108f590816100d1823960805181818160e00152818161030601526103f70152f35b6040513d6000823e3d90fd5b634e487b7160e01b600052604160045260246000fd5b600080fdfe608080604052600436101561001357600080fd5b600090813560e01c90816311464fbe14610096575080635fbfb9cf1461007c57638cb84e181461004257600080fd5b3461007957602061005b61005536610108565b90610363565b73ffffffffffffffffffffffffffffffffffffffff60405191168152f35b80fd5b503461007957602061005b61009036610108565b90610274565b90503461010457817ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101045760209073ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000168152f35b5080fd5b7ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc60409101126101595760043573ffffffffffffffffffffffffffffffffffffffff81168103610159579060243590565b600080fd5b6060810190811067ffffffffffffffff82111761017a57604052565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b90601f7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0910116810190811067ffffffffffffffff82111761017a57604052565b60005b8381106101fd5750506000910152565b81810151838201526020016101ed565b90601f60609373ffffffffffffffffffffffffffffffffffffffff7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0931684526040602085015261026d81518092816040880152602088880191016101ea565b0116010190565b9061027f8183610363565b803b610347575073ffffffffffffffffffffffffffffffffffffffff9182604051917fc4d66de8000000000000000000000000000000000000000000000000000000006020840152166024820152602481526102da8161015e565b6040519061042c8083019183831067ffffffffffffffff84111761017a57839261032c926104948539867f0000000000000000000000000000000000000000000000000000000000000000169061020d565b03906000f5801561033b571690565b6040513d6000823e3d90fd5b73ffffffffffffffffffffffffffffffffffffffff1692915050565b600b9060559261042c60209061046f61047b83604096875190610388838701836101a9565b85825282820195610494873961041d61044973ffffffffffffffffffffffffffffffffffffffff92838c51917fc4d66de80000000000000000000000000000000000000000000000000000000088840152166024820152602481526103ec8161015e565b8b51928391878301957f0000000000000000000000000000000000000000000000000000000000000000168661020d565b037fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe081018352826101a9565b8951958693610460868601998a92519283916101ea565b840191518093868401906101ea565b010380845201826101a9565b5190208351938401528201523081520160ff8153209056fe60406080815261042c908138038061001681610218565b93843982019181818403126102135780516001600160a01b038116808203610213576020838101516001600160401b0394919391858211610213570186601f820112156102135780519061007161006c83610253565b610218565b918083528583019886828401011161021357888661008f930161026e565b813b156101b9577f360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc80546001600160a01b031916841790556000927fbc7cd75a20ee27fd9adebab32041f755214dbc6bffa90cc0225b39da2e5c2d3b8480a28051158015906101b2575b61010b575b855160e790816103458239f35b855194606086019081118682101761019e578697849283926101889952602788527f416464726573733a206c6f772d6c6576656c2064656c65676174652063616c6c87890152660819985a5b195960ca1b8a8901525190845af4913d15610194573d9061017a61006c83610253565b91825281943d92013e610291565b508038808080806100fe565b5060609250610291565b634e487b7160e01b84526041600452602484fd5b50826100f9565b855162461bcd60e51b815260048101859052602d60248201527f455243313936373a206e657720696d706c656d656e746174696f6e206973206e60448201526c1bdd08184818dbdb9d1c9858dd609a1b6064820152608490fd5b600080fd5b6040519190601f01601f191682016001600160401b0381118382101761023d57604052565b634e487b7160e01b600052604160045260246000fd5b6001600160401b03811161023d57601f01601f191660200190565b60005b8381106102815750506000910152565b8181015183820152602001610271565b919290156102f357508151156102a5575090565b3b156102ae5790565b60405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e74726163740000006044820152606490fd5b8251909150156103065750805190602001fd5b6044604051809262461bcd60e51b825260206004830152610336815180928160248601526020868601910161026e565b601f01601f19168101030190fdfe60806040523615605f5773ffffffffffffffffffffffffffffffffffffffff7f360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc54166000808092368280378136915af43d82803e15605b573d90f35b3d90fd5b73ffffffffffffffffffffffffffffffffffffffff7f360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc54166000808092368280378136915af43d82803e15605b573d90f3fea26469706673582212205da2750cd2b0cadfd354d8a1ca4752ed7f22214c8069d852f7dc6b8e9e5ee66964736f6c63430008150033a26469706673582212205367f15fddc0d5cbb3b407c1f8fa018b2549200abc34a5978c9abd75b26a675a64736f6c6343000815003360e03462000160576001600160401b0390601f6200270538819003918201601f1916830191848311848410176200016557808492602094604052833981010312620001605751906001600160a01b03821682036200016057306080527f33e4b41198cc5b8053630ed667ea7c0c4c873f7fc8d9a478b5d7259cec0a4a00918260a05260c05281549060ff8260401c166200014e57808083160362000108575b60405161258990816200017c82396080518181816107b201528181610dbd0152610f99015260a0518161141d015260c0518181816109d701528181610bf501528181610cd4015281816111b001528181611387015281816115ff015281816122af01526124b50152f35b6001600160401b031990911681179091556040519081527fc7f505b2f371ae2175ee4913f4499e1f2633a7b5936321eed1cdaeb6115181d290602090a13880806200009e565b60405163f92ee8a960e01b8152600490fd5b600080fd5b634e487b7160e01b600052604160045260246000fdfe6080604052600436101561001b575b361561001957600080fd5b005b60003560e01c806223de291461019a57806301ffc9a7146101955780630a1028c414610190578063150b7a021461018b5780631626ba7e1461018657806318dfb3c7146101815780633659cfe61461017c5780633a871cdd1461017757806347e1da2a146101725780634a58db191461016d5780634d44560d146101685780634f1ef2861461016357806352d1902d1461015e5780638da5cb5b14610159578063a786cac914610154578063b0d691fe1461014f578063b61d27f61461014a578063bc197c8114610145578063c399ec8814610140578063c4d66de81461013b578063d087d28814610136578063f23a6e6114610131578063f2fde38b1461012c5763f698da250361000e5761184d565b6116f2565b611661565b611580565b6113e0565b61130f565b611248565b6111d4565b611165565b61113d565b61106e565b610f53565b610d4d565b610c76565b610bb3565b610ac3565b61096d565b61075e565b610672565b6105bb565b61052a565b610504565b61027b565b6101f0565b73ffffffffffffffffffffffffffffffffffffffff8116036101bd57565b600080fd5b9181601f840112156101bd5782359167ffffffffffffffff83116101bd57602083818601950101116101bd57565b346101bd5760c07ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd5761022a60043561019f565b61023560243561019f565b61024060443561019f565b67ffffffffffffffff6084358181116101bd576102619036906004016101c2565b505060a4359081116101bd576100199036906004016101c2565b346101bd5760207ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd576004357fffffffff0000000000000000000000000000000000000000000000000000000081168091036101bd57807f150b7a020000000000000000000000000000000000000000000000000000000060209214908115610341575b8115610317575b506040519015158152f35b7f01ffc9a7000000000000000000000000000000000000000000000000000000009150143861030c565b7f4e2312e00000000000000000000000000000000000000000000000000000000081149150610305565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b67ffffffffffffffff81116103ae57604052565b61036b565b6020810190811067ffffffffffffffff8211176103ae57604052565b6060810190811067ffffffffffffffff8211176103ae57604052565b90601f7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0910116810190811067ffffffffffffffff8211176103ae57604052565b67ffffffffffffffff81116103ae57601f017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe01660200190565b9291926104728261042c565b9161048060405193846103eb565b8294818452818301116101bd578281602093846000960137010152565b9080601f830112156101bd578160206104b893359101610466565b90565b60207ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc8201126101bd576004359067ffffffffffffffff82116101bd576104b89160040161049d565b346101bd57602061051c610517366104bb565b611ec8565b818151910120604051908152f35b346101bd5760807ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd5761056460043561019f565b61056f60243561019f565b60643567ffffffffffffffff81116101bd5761058f9036906004016101c2565b505060206040517f150b7a02000000000000000000000000000000000000000000000000000000008152f35b346101bd5760407ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd5760243567ffffffffffffffff81116101bd5761061761060f602092369060040161049d565b600435611f6d565b7fffffffff0000000000000000000000000000000000000000000000000000000060405191168152f35b9181601f840112156101bd5782359167ffffffffffffffff83116101bd576020808501948460051b0101116101bd57565b346101bd5760407ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd5767ffffffffffffffff6004358181116101bd576106c2903690600401610641565b916024359081116101bd576106db903690600401610641565b91906106e561249d565b8284036107345760005b8481106106f857005b8061072e6107096001938887611da0565b356107138161019f565b610728610721848988611e06565b3691610466565b90612514565b016106ef565b60046040517fa24a13a6000000000000000000000000000000000000000000000000000000008152fd5b346101bd5760207ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd576004356107998161019f565b73ffffffffffffffffffffffffffffffffffffffff90817f000000000000000000000000000000000000000000000000000000000000000016916107df833014156118d9565b61080e7f360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc938285541614611964565b61081661241c565b60405190610823826103b3565b600082527f4910fdfa16fed3260ed0e7147f7cc6da11a60208b5b9406d12a635614ffd91435460ff161561085d5750506100199150611a7a565b6020600491604094939451928380927f52d1902d00000000000000000000000000000000000000000000000000000000825286165afa6000918161093d575b5061092a576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602e60248201527f45524331393637557067726164653a206e657720696d706c656d656e7461746960448201527f6f6e206973206e6f7420555550530000000000000000000000000000000000006064820152608490fd5b0390fd5b6100199361093891146119ef565b611b66565b61095f91925060203d8111610966575b61095781836103eb565b81019061188e565b903861089c565b503d61094d565b346101bd577ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc6060813601126101bd576004359067ffffffffffffffff82116101bd576101609082360301126101bd5760443573ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000163303610a6557610a10610a28926024359060040161234d565b9080610a2c575b506040519081529081906020820190565b0390f35b600080808093337ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff150610a5e6118a9565b5038610a17565b60646040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601c60248201527f6163636f756e743a206e6f742066726f6d20456e747279506f696e74000000006044820152fd5b346101bd5760607ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd5767ffffffffffffffff6004358181116101bd57610b13903690600401610641565b6024358381116101bd57610b2b903690600401610641565b936044359081116101bd57610b44903690600401610641565b92610b4d61249d565b838114801590610ba9575b6107345760005b818110610b6857005b80610ba3610b79600193858a611da0565b35610b838161019f565b610b8e838b89611da0565b35610b9d610721858b8a611e06565b9161253c565b01610b5f565b5085811415610b58565b6000807ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc360112610c735773ffffffffffffffffffffffffffffffffffffffff7f00000000000000000000000000000000000000000000000000000000000000001681813b15610c7357602491604051928380927fb760faf900000000000000000000000000000000000000000000000000000000825230600483015234905af18015610c6e57610c62575080f35b610c6b9061039a565b80f35b61189d565b80fd5b346101bd57600060407ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc360112610c7357600435610cb38161019f565b610cbb61241c565b8173ffffffffffffffffffffffffffffffffffffffff807f00000000000000000000000000000000000000000000000000000000000000001692833b15610d49576044908360405195869485937f205c287800000000000000000000000000000000000000000000000000000000855216600484015260243560248401525af18015610c6e57610c62575080f35b8280fd5b60407ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd57600435610d838161019f565b60243567ffffffffffffffff81116101bd57610da390369060040161049d565b9073ffffffffffffffffffffffffffffffffffffffff91827f00000000000000000000000000000000000000000000000000000000000000001692610dea843014156118d9565b610e197f360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc948286541614611964565b610e2161241c565b7f4910fdfa16fed3260ed0e7147f7cc6da11a60208b5b9406d12a635614ffd91435460ff1615610e575750506100199150611a7a565b6020600491604094939451928380927f52d1902d00000000000000000000000000000000000000000000000000000000825286165afa60009181610f33575b50610f20576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602e60248201527f45524331393637557067726164653a206e657720696d706c656d656e7461746960448201527f6f6e206973206e6f7420555550530000000000000000000000000000000000006064820152608490fd5b61001993610f2e91146119ef565b611c45565b610f4c91925060203d81116109665761095781836103eb565b9038610e96565b346101bd5760007ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd5773ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000163003610fea576040517f360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc8152602090f35b60846040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603860248201527f555550535570677261646561626c653a206d757374206e6f742062652063616c60448201527f6c6564207468726f7567682064656c656761746563616c6c00000000000000006064820152fd5b346101bd5760007ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd57602073ffffffffffffffffffffffffffffffffffffffff7f691ec1a18226d004c07c9f8e5c4a6ff15a7b38db267cf7e3c945aef8be5122005416604051908152f35b919082519283825260005b8481106111295750507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0601f8460006020809697860101520116010190565b6020818301810151848301820152016110ea565b346101bd57610a28611151610517366104bb565b6040519182916020835260208301906110df565b346101bd5760007ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd57602060405173ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000168152f35b346101bd5760607ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd5760043561120f8161019f565b6044359067ffffffffffffffff82116101bd5761123e6112366100199336906004016101c2565b61072161249d565b906024359061253c565b346101bd5760a07ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd5761128260043561019f565b61128d60243561019f565b67ffffffffffffffff6044358181116101bd576112ae903690600401610641565b50506064358181116101bd576112c8903690600401610641565b50506084359081116101bd576112e29036906004016101c2565b50506040517fbc197c81000000000000000000000000000000000000000000000000000000008152602090f35b346101bd5760007ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd576040517f70a0823100000000000000000000000000000000000000000000000000000000815230600482015260208160248173ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000165afa8015610c6e576020916000916113c3575b50604051908152f35b6113da9150823d81116109665761095781836103eb565b386113ba565b346101bd5760207ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd5760043561141b8161019f565b7f00000000000000000000000000000000000000000000000000000000000000009081549067ffffffffffffffff60ff8360401c1615921680159081611578575b600114908161156e575b159081611565575b5061153b5782547fffffffffffffffffffffffffffffffffffffffffffffffff00000000000000001660011783556114aa908261150557612227565b6114b057005b80547fffffffffffffffffffffffffffffffffffffffffffffff00ffffffffffffffff169055604051600181527fc7f505b2f371ae2175ee4913f4499e1f2633a7b5936321eed1cdaeb6115181d290602090a1005b83547fffffffffffffffffffffffffffffffffffffffffffffff00ffffffffffffffff1668010000000000000000178455612227565b60046040517ff92ee8a9000000000000000000000000000000000000000000000000000000008152fd5b9050153861146e565b303b159150611466565b83915061145c565b346101bd5760007ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd576040517f35567e1a0000000000000000000000000000000000000000000000000000000081523060048201526000602482015260208160448173ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000165afa8015610c6e57610a289160009161164357506040519081529081906020820190565b61165b915060203d81116109665761095781836103eb565b38610a17565b346101bd5760a07ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd5761169b60043561019f565b6116a660243561019f565b60843567ffffffffffffffff81116101bd576116c69036906004016101c2565b505060206040517ff23a6e61000000000000000000000000000000000000000000000000000000008152f35b346101bd5760207ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd5760043561172d8161019f565b61173561241c565b73ffffffffffffffffffffffffffffffffffffffff908181169182158015611844575b611813577f691ec1a18226d004c07c9f8e5c4a6ff15a7b38db267cf7e3c945aef8be512200541690818314611813576117ec9073ffffffffffffffffffffffffffffffffffffffff7f691ec1a18226d004c07c9f8e5c4a6ff15a7b38db267cf7e3c945aef8be51220091167fffffffffffffffffffffffff0000000000000000000000000000000000000000825416179055565b7f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0600080a3005b602483604051907fb20f76e30000000000000000000000000000000000000000000000000000000082526004820152fd5b50308314611758565b346101bd5760007ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101bd576020611886611e21565b604051908152f35b908160209103126101bd575190565b6040513d6000823e3d90fd5b3d156118d4573d906118ba8261042c565b916118c860405193846103eb565b82523d6000602084013e565b606090565b156118e057565b60846040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602c60248201527f46756e6374696f6e206d7573742062652063616c6c6564207468726f7567682060448201527f64656c656761746563616c6c00000000000000000000000000000000000000006064820152fd5b1561196b57565b60846040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602c60248201527f46756e6374696f6e206d7573742062652063616c6c6564207468726f7567682060448201527f6163746976652070726f787900000000000000000000000000000000000000006064820152fd5b156119f657565b60846040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602960248201527f45524331393637557067726164653a20756e737570706f727465642070726f7860448201527f6961626c655555494400000000000000000000000000000000000000000000006064820152fd5b803b15611ae25773ffffffffffffffffffffffffffffffffffffffff7f360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc91167fffffffffffffffffffffffff0000000000000000000000000000000000000000825416179055565b60846040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602d60248201527f455243313936373a206e657720696d706c656d656e746174696f6e206973206e60448201527f6f74206120636f6e7472616374000000000000000000000000000000000000006064820152fd5b90611b7082611a7a565b73ffffffffffffffffffffffffffffffffffffffff82167fbc7cd75a20ee27fd9adebab32041f755214dbc6bffa90cc0225b39da2e5c2d3b600080a2805115801590611c3d575b611bbf575050565b611c3a9160008060405193611bd3856103cf565b602785527f416464726573733a206c6f772d6c6576656c2064656c65676174652063616c6c60208601527f206661696c6564000000000000000000000000000000000000000000000000006040860152602081519101845af4611c346118a9565b91611ca5565b50565b506000611bb7565b90611c4f82611a7a565b73ffffffffffffffffffffffffffffffffffffffff82167fbc7cd75a20ee27fd9adebab32041f755214dbc6bffa90cc0225b39da2e5c2d3b600080a2805115801590611c9d57611bbf575050565b506001611bb7565b91929015611d205750815115611cb9575090565b3b15611cc25790565b60646040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e74726163740000006044820152fd5b825190915015611d335750805190602001fd5b610926906040519182917f08c379a00000000000000000000000000000000000000000000000000000000083526020600484015260248301906110df565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b9190811015611db05760051b0190565b611d71565b9035907fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe1813603018212156101bd570180359067ffffffffffffffff82116101bd576020019181360383136101bd57565b90821015611db057611e1d9160051b810190611db5565b9091565b60405160208101907f8b73c3c69bb8fe3d512ecc4cf759cc79239f7b179b0ffacaa9a75d522b39400f82527fcbe29a6ace531c23849b5cdb1a6b991866eb7dc20deda15202ba6fd921ed2c0060408201527fc89efdaa54c0f20c7adf612882df0950f5a951637e0307cdcb4c672f298b8bc660608201524660808201523060a082015260a0815260c0810181811067ffffffffffffffff8211176103ae5760405251902090565b6020815191012060405160208101917f5e3baca2936049843f06038876a12f03627b5edc98025751ecf2ac75626401998352604082015260408152611f0c816103cf565b519020611f17611e21565b90604051917f1901000000000000000000000000000000000000000000000000000000000000602084015260228301526042820152604281526080810181811067ffffffffffffffff8211176103ae5760405290565b60405190602082015260208152604081019080821067ffffffffffffffff8311176103ae57611f9e91604052611ec8565b602081519101209073ffffffffffffffffffffffffffffffffffffffff91827f691ec1a18226d004c07c9f8e5c4a6ff15a7b38db267cf7e3c945aef8be5122005416611fea83836120a3565b600581969296101561207457159485612068575b50508315612056575b505050612032577fffffffff0000000000000000000000000000000000000000000000000000000090565b7f1626ba7e0000000000000000000000000000000000000000000000000000000090565b6120609350612166565b388080612007565b16811493503880611ffe565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602160045260246000fd5b9060418151146000146120cd57611e1d916020820151906060604084015193015160001a906120d7565b5050600090600290565b9291907f7fffffffffffffffffffffffffffffff5d576e7357a4501ddfe92f46681b20a0831161215a5791608094939160ff602094604051948552168484015260408301526060820152600093849182805260015afa15610c6e57815173ffffffffffffffffffffffffffffffffffffffff811615612154579190565b50600190565b50505050600090600390565b600091929082916040516121e3816121b760208201947f1626ba7e00000000000000000000000000000000000000000000000000000000998a875260248401526040604484015260648301906110df565b037fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe081018352826103eb565b51915afa906121f06118a9565b82612219575b8261220057505090565b6122159192506020808251830101910161188e565b1490565b9150602082511015916121f6565b73ffffffffffffffffffffffffffffffffffffffff9081811691821561231c576122ad839273ffffffffffffffffffffffffffffffffffffffff7f691ec1a18226d004c07c9f8e5c4a6ff15a7b38db267cf7e3c945aef8be51220091167fffffffffffffffffffffffff0000000000000000000000000000000000000000825416179055565b7f0000000000000000000000000000000000000000000000000000000000000000167fec6a23b49d2c363d250c9dda15610e835d428207d15ddb36a6c230e37371ddf1600080a360007f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e08180a3565b60246040517fb20f76e300000000000000000000000000000000000000000000000000000000815260006004820152fd5b73ffffffffffffffffffffffffffffffffffffffff91827f691ec1a18226d004c07c9f8e5c4a6ff15a7b38db267cf7e3c945aef8be51220054167f19457468657265756d205369676e6564204d6573736167653a0a33320000000060005281601c526123d06123c9610721603c60002095610140810190611db5565b80946120a3565b600581969296101561207457159485612410575b505083156123fe575b5050506123f957600190565b600090565b6124089350612166565b3880806123ed565b168114935038806123e4565b303314158061245c575b61242c57565b60246040517f4a0bfec1000000000000000000000000000000000000000000000000000000008152336004820152fd5b5073ffffffffffffffffffffffffffffffffffffffff7f691ec1a18226d004c07c9f8e5c4a6ff15a7b38db267cf7e3c945aef8be5122005416331415612426565b73ffffffffffffffffffffffffffffffffffffffff807f00000000000000000000000000000000000000000000000000000000000000001633141590816124e6575b5061242c57565b90507f691ec1a18226d004c07c9f8e5c4a6ff15a7b38db267cf7e3c945aef8be5122005416331415386124df565b600091829182602083519301915af161252b6118a9565b90156125345750565b602081519101fd5b916000928392602083519301915af161252b6118a956fea2646970667358221220c5240b5a614209162da17798c4589910308036b820e321c267b03d8cedb5e48164736f6c634300081500330000000000000000000000005ff137d4b0fdcd49dca30c7cf57e578a026d2789" +export const LIGHT_ACCOUNT_FACTORY_V200_CREATECALL: Hex = + "0x00000000000000000000000000000000000000005f1ffd9d31306e056bcc959b60c060405234620000ae57620032713881900360c0601f8201601f19168101906001600160401b03821190821017620000b3576040928291845260c03912620000ae576200006960c0516200005481620000e5565b60e051906200006382620000e5565b62000103565b604051610d8590816200034f82396080518181816101160152818161050b015281816105ee01526106a9015260a0518181816107cc01528181610a170152610b6b0152f35b600080fd5b634e487b7160e01b600052604160045260246000fd5b606081019081106001600160401b03821117620000b357604052565b6001600160a01b03811603620000ae57565b6040513d6000823e3d90fd5b6001600160a01b03908116908115620001bc5762000165918160018060a01b031980600154166001558260005491821617600055167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0600080a38216620001d5565b60405161219d8082016001600160401b03811183821017620000b35783620001a2918493620010d485396001600160a01b03909116815260200190565b03906000f08015620001b65760a052608052565b620000f7565b604051631e4fbdf760e01b815260006004820152602490fd5b60405160208082018160006301ffc9a760e01b94858452856024820152602481526200020181620000c9565b51617530938785fa923d6000519085620002e0575b5084620002d5575b50836200026d575b505050806200025b575b15620002395750565b60405163075b76fd60e21b81526001600160a01b039091166004820152602490fd5b506200026781620002ec565b62000230565b829350906000918560405185810192835263ffffffff60e01b6024820152602481526200029a81620000c9565b5192fa60005190913d83620002c9575b505081620002be575b501538808062000226565b9050151538620002b3565b101591503880620002aa565b15159350386200021e565b84111594503862000216565b6000602091604051838101906301ffc9a760e01b825263122a0e9b60e31b6024820152602481526200031e81620000c9565b5191617530fa6000513d8262000341575b50816200033a575090565b9050151590565b602011159150386200032f56fe60806040818152600480361015610021575b505050361561001f57600080fd5b005b600092833560e01c908163290ab98414610b21575080635fbfb9cf14610967578063715018a61461090857806379ba5097146108285780638cb84e181461071e5780638da5cb5b146106cd57806394430fa51461065e57838163bb9fe6bf1461059e57508063c23a5cea146104a4578063d9caed12146102b0578063e30c397814610259578063f2fde38b146101ac5763fbb1c3d403610011578183927ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101a85781359163ffffffff83168093036101a3576100ff610c5e565b73ffffffffffffffffffffffffffffffffffffffff7f00000000000000000000000000000000000000000000000000000000000000001692833b1561019f5760248592845195869384927f0396cb600000000000000000000000000000000000000000000000000000000084528301528235905af190811561019657506101835750f35b61018c90610bb2565b6101935780f35b80fd5b513d84823e3d90fd5b8480fd5b505050fd5b5050fd5b83346101935760207ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc360112610193576101e4610b8f565b6101ec610c5e565b73ffffffffffffffffffffffffffffffffffffffff80911690817fffffffffffffffffffffffff000000000000000000000000000000000000000060015416176001558254167f38d16b8cac22d99fc7c124b9cd0de2d3fa1faef420bfe791d8c362d765e227008380a380f35b5050346102ac57817ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126102ac5760209073ffffffffffffffffffffffffffffffffffffffff600154169051908152f35b5080fd5b5090346104a05760607ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126104a0576102e9610b8f565b6024359073ffffffffffffffffffffffffffffffffffffffff9081831680930361049b57610315610c5e565b16908115610473578491908061036757508180809247905af1610336610bf5565b501561034157505080f35b517f90b8ec18000000000000000000000000000000000000000000000000000000008152fd5b928092505160208101917fa9059cbb00000000000000000000000000000000000000000000000000000000835260248201526044356044820152604481526080810181811067ffffffffffffffff821117610445578352516103da918691829182875af16103d3610bf5565b9084610caf565b8051908115159182610421575b50506103f35750505080f35b6024935051917f5274afe7000000000000000000000000000000000000000000000000000000008352820152fd5b819250906020918101031261019f576020015180159081150361019f5738806103e7565b6041867f4e487b71000000000000000000000000000000000000000000000000000000006000525260246000fd5b5050517f8579befe000000000000000000000000000000000000000000000000000000008152fd5b600080fd5b8280fd5b509190346102ac5760207ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126102ac576104de610b8f565b906104e7610c5e565b73ffffffffffffffffffffffffffffffffffffffff809216918215610576579383947f00000000000000000000000000000000000000000000000000000000000000001692833b1561019f576024859283855196879485937fc23a5cea0000000000000000000000000000000000000000000000000000000085528401525af190811561019657506101835750f35b8482517f8579befe000000000000000000000000000000000000000000000000000000008152fd5b808484346101a857827ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101a8576105d7610c5e565b73ffffffffffffffffffffffffffffffffffffffff7f000000000000000000000000000000000000000000000000000000000000000016803b156101a35783918351809581937fbb9fe6bf0000000000000000000000000000000000000000000000000000000083525af19081156101965750610652575080f35b61065b90610bb2565b80f35b5050346102ac57817ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126102ac576020905173ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000168152f35b5050346102ac57817ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126102ac5773ffffffffffffffffffffffffffffffffffffffff60209254169051908152f35b5050346102ac57807ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126102ac579060209161075a610b8f565b9073ffffffffffffffffffffffffffffffffffffffff918352602435845280832081517fcc3735a920a3ca505d382bbc545af43d6000803e6038573d6000fd5b3d6000f36060527f5155f3363d3d373d3d363d7f360894a13ba1a3210667c828492db98dca3e207683526160098652837f000000000000000000000000000000000000000000000000000000000000000016601e5268603d3d8160223d3973600a52605f6021209083528460605260ff85536035523060601b60015260155260558320926035525191168152f35b5090346104a057827ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126104a0576001549173ffffffffffffffffffffffffffffffffffffffff9133838516036108d85750507fffffffffffffffffffffffff0000000000000000000000000000000000000000809216600155825491339083161783553391167f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e08380a380f35b6024925051907f118cdaa70000000000000000000000000000000000000000000000000000000082523390820152fd5b50913461019357807ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126101935750610941610c5e565b517f4a7f394f000000000000000000000000000000000000000000000000000000008152fd5b5090346104a057807ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126104a05761099f610b8f565b73ffffffffffffffffffffffffffffffffffffffff928185526024356020528285209385928451937fcc3735a920a3ca505d382bbc545af43d6000803e6038573d6000fd5b3d6000f36060527f5155f3363d3d373d3d363d7f360894a13ba1a3210667c828492db98dca3e20768652616009602052827f000000000000000000000000000000000000000000000000000000000000000016601e5268603d3d8160223d3973600a52605f96602197605f60212060358801523060581b875260ff87538160158801526055872098893b15610b015750505050816001965b85875288606052169515610a95575b6020868651908152f35b853b15610afd577fc4d66de80000000000000000000000000000000000000000000000000000000084521690820152838160248183875af18015610af35760209450610ae4575b808080610a8b565b610aed90610bb2565b38610adc565b82513d86823e3d90fd5b8680fd5b909192985089f58015610b15578290610a7c565b8363301164258952601cfd5b8490346102ac57817ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126102ac5760209073ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000168152f35b6004359073ffffffffffffffffffffffffffffffffffffffff8216820361049b57565b67ffffffffffffffff8111610bc657604052565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b3d15610c595767ffffffffffffffff903d828111610bc65760405192601f82017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0908116603f0116840190811184821017610bc65760405282523d6000602084013e565b606090565b73ffffffffffffffffffffffffffffffffffffffff600054163303610c7f57565b60246040517f118cdaa7000000000000000000000000000000000000000000000000000000008152336004820152fd5b90610cee5750805115610cc457805190602001fd5b60046040517f1425ea42000000000000000000000000000000000000000000000000000000008152fd5b81511580610d46575b610cff575090565b60249073ffffffffffffffffffffffffffffffffffffffff604051917f9996b315000000000000000000000000000000000000000000000000000000008352166004820152fd5b50803b15610cf756fea264697066735822122020672d0c03264e2785eb3a17a40742d95e9887bed833176dd597224a3829b8d664736f6c634300081700336101803462000224576001600160401b0390601f6200219d38819003918201601f191683019291908484118385101762000229578160209284926040968752833981010312620002245751916001600160a01b03831683036200022457306080523060a0524660c052620000726200023f565b92600c845260a06001602086016b131a59da1d1058d8dbdd5b9d60a21b815260206200009d6200023f565b8381520196601960f91b88525190209520948060e052610100958087528551917f8b73c3c69bb8fe3d512ecc4cf759cc79239f7b179b0ffacaa9a75d522b39400f835260208301528582015246606082015230608082015220926101209384527f33e4b41198cc5b8053630ed667ea7c0c4c873f7fc8d9a478b5d7259cec0a4a006101609381855261014093845281549060ff82851c1662000213578080831603620001ce575b5050505192611f3d9485620002608639608051858181610ac60152610b78015260a05185611689015260c051856116ac015260e0518561171e01525184611744015251836116670152518281816103710152818161052301528181610708015281816108cb01528181610cd001528181610dc601528181610fcd01526119c80152518161042f0152f35b6001600160401b0319909116811790915581519081527fc7f505b2f371ae2175ee4913f4499e1f2633a7b5936321eed1cdaeb6115181d290602090a138808062000144565b835163f92ee8a960e01b8152600490fd5b600080fd5b634e487b7160e01b600052604160045260246000fd5b60408051919082016001600160401b03811183821017620002295760405256fe6080604081815260049081361015610022575b505050361561002057600080fd5b005b600092833560e01c90816301ffc9a714611307575080630a1028c414611294578063150b7a02146112065780631626ba7e1461117f57806318dfb3c7146110a857806319822f7c14610f5f57806347e1da2a14610e3b5780634a58db1914610d845780634d44560d14610c6a5780634f1ef28614610b1f57806352d1902d14610a9357806384b0196e146109615780638da5cb5b146108ef578063b0d691fe14610880578063b61d27f614610801578063bc197c8114610740578063c399ec881461068f578063c4d66de8146103f3578063d087d288146102f2578063f23a6e61146102615763f2fde38b03610012573461025d5760207ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc36011261025d57610149611506565b916101526119b0565b73ffffffffffffffffffffffffffffffffffffffff8093169283158015610254575b610225577f691ec1a18226d004c07c9f8e5c4a6ff15a7b38db267cf7e3c945aef8be512200918254918216938486146101f65750507fffffffffffffffffffffffff000000000000000000000000000000000000000016831790557f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e08380a380f35b908560249251917fb20f76e3000000000000000000000000000000000000000000000000000000008352820152fd5b508260249251917fb20f76e3000000000000000000000000000000000000000000000000000000008352820152fd5b50308414610174565b8280fd5b5082346102ef5760a07ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126102ef5761029a611506565b506102a3611529565b506084359067ffffffffffffffff82116102ef57506020926102c79136910161154c565b5050517ff23a6e61000000000000000000000000000000000000000000000000000000008152f35b80fd5b508290346103ef57817ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126103ef578051927f35567e1a000000000000000000000000000000000000000000000000000000008452309084015281602484015260208360448173ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000165afa9182156103e457916103aa575b6020925051908152f35b90506020823d6020116103dc575b816103c560209383611454565b810103126103d75760209151906103a0565b600080fd5b3d91506103b8565b9051903d90823e3d90fd5b5080fd5b50903461025d5760207ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc36011261025d5761042c611506565b917f00000000000000000000000000000000000000000000000000000000000000009182549160ff83821c16159267ffffffffffffffff811680159081610687575b600114908161067d575b159081610674575b5061064d578360017fffffffffffffffffffffffffffffffffffffffffffffffff00000000000000008316178655610618575b5073ffffffffffffffffffffffffffffffffffffffff8095169182156105ea575081907f691ec1a18226d004c07c9f8e5c4a6ff15a7b38db267cf7e3c945aef8be512200827fffffffffffffffffffffffff000000000000000000000000000000000000000082541617905551947f0000000000000000000000000000000000000000000000000000000000000000167fec6a23b49d2c363d250c9dda15610e835d428207d15ddb36a6c230e37371ddf18780a3847f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e08180a3610594578280f35b7fc7f505b2f371ae2175ee4913f4499e1f2633a7b5936321eed1cdaeb6115181d291817fffffffffffffffffffffffffffffffffffffffffffffff00ffffffffffffffff6020935416905560018152a138808280f35b8660249251917fb20f76e3000000000000000000000000000000000000000000000000000000008352820152fd5b7fffffffffffffffffffffffffffffffffffffffffffffff0000000000000000001668010000000000000001178455386104b3565b50517ff92ee8a9000000000000000000000000000000000000000000000000000000008152fd5b90501538610480565b303b159150610478565b85915061046e565b508290346103ef57817ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126103ef578051927f70a08231000000000000000000000000000000000000000000000000000000008452309084015260208360248173ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000165afa9182156103e457916103aa576020925051908152f35b5082346102ef5760a07ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126102ef57610779611506565b50610782611529565b5067ffffffffffffffff906044358281116103ef576107a4903690860161157a565b50506064358281116103ef576107bd903690860161157a565b50506084359182116102ef57506020926107d99136910161154c565b5050517fbc197c81000000000000000000000000000000000000000000000000000000008152f35b5050346103ef5760607ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126103ef5761083a611506565b6044359167ffffffffffffffff831161087c5761086061086f916108799436910161154c565b6108686119b0565b36916114cf565b9060243590611a98565b80f35b8380fd5b8382346103ef57817ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126103ef576020905173ffffffffffffffffffffffffffffffffffffffff7f0000000000000000000000000000000000000000000000000000000000000000168152f35b8382346103ef57817ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126103ef5760209073ffffffffffffffffffffffffffffffffffffffff7f691ec1a18226d004c07c9f8e5c4a6ff15a7b38db267cf7e3c945aef8be51220054169051908152f35b5082346102ef57807ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126102ef579080519061099e82611438565b600c82526020927f4c696768744163636f756e74000000000000000000000000000000000000000084840152610a4b8251926109d984611438565b600193600181527f320000000000000000000000000000000000000000000000000000000000000087820152610a3e8251967f0f00000000000000000000000000000000000000000000000000000000000000885260e08989015260e08801906115ab565b91868303908701526115ab565b4660608501523060808501528160a085015283810360c0850152846060519182815201946080925b828110610a805785870386f35b8351875295810195928101928401610a73565b5082346102ef57807ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126102ef57307f000000000000000000000000000000000000000000000000000000000000000003610b1357602082517f360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc8152f35b639f03a026915052601cfd5b5090817ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc36011261025d57610b52611506565b9160243567ffffffffffffffff8111610c6657610b72903690840161154c565b919093307f000000000000000000000000000000000000000000000000000000000000000014610c5a5773ffffffffffffffffffffffffffffffffffffffff90610bba6119b0565b16926352d1902d6001527f360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc90816020600183601d895afa5103610c4e575090828480949388967fbc7cd75a20ee27fd9adebab32041f755214dbc6bffa90cc0225b39da2e5c2d3b8880a255610c2d578380f35b8190519485378338925af415610c4557818180808380f35b903d90823e3d90fd5b6355299b49600152601dfd5b83639f03a0268752601cfd5b8480fd5b508290346103ef57807ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126103ef5782359073ffffffffffffffffffffffffffffffffffffffff80831680930361087c57610cc56119b0565b8215610d5c579383947f00000000000000000000000000000000000000000000000000000000000000001692833b15610c66576044859283855196879485937f205c287800000000000000000000000000000000000000000000000000000000855284015260243560248401525af1908115610d535750610d435750f35b610d4c906113f5565b6102ef5780f35b513d84823e3d90fd5b8482517f8579befe000000000000000000000000000000000000000000000000000000008152fd5b50827ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc36011261025d578273ffffffffffffffffffffffffffffffffffffffff7f00000000000000000000000000000000000000000000000000000000000000001692833b156103ef5760248351809581937fb760faf9000000000000000000000000000000000000000000000000000000008352309083015234905af1908115610d535750610e32575080f35b610879906113f5565b503461025d5760607ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc36011261025d5767ffffffffffffffff908235828111610c6657610e8b903690850161157a565b602494919435848111610f5b57610ea5903690840161157a565b919094604435908111610f5757610ebf903690850161157a565b939094610eca6119b0565b848314801590610f4d575b610f27575050865b818110610ee8578780f35b80610f21610f01610efc600194868c6117eb565b61182a565b610f0c83878b6117eb565b35610f1b610868858a8c61189c565b91611a98565b01610edd565b517fa24a13a6000000000000000000000000000000000000000000000000000000008152fd5b5083831415610ed5565b8780fd5b8680fd5b508290346103ef577ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc916060833601126102ef5783359267ffffffffffffffff84116103ef576101209084360301126102ef576044359273ffffffffffffffffffffffffffffffffffffffff7f000000000000000000000000000000000000000000000000000000000000000016330361104b57602094611004916024359101611ac0565b9280611013575b505051908152f35b81808092337ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff150611043611a68565b50838061100b565b60648560208551917f08c379a0000000000000000000000000000000000000000000000000000000008352820152601c60248201527f6163636f756e743a206e6f742066726f6d20456e747279506f696e74000000006044820152fd5b50903461025d57817ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc36011261025d5767ffffffffffffffff908035828111610c66576110f8903690830161157a565b909260243590811161117b57611111903690840161157a565b92909461111c6119b0565b838303610f27575050845b818110611132578580f35b611140610efc8284876117eb565b868061115061086885888b61189c565b602093828583519301915af190611165611a68565b9115611175575050600101611127565b81519101fd5b8580fd5b5082346102ef57817ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126102ef576024359067ffffffffffffffff82116102ef57506111fe6020936111f77fffffffff00000000000000000000000000000000000000000000000000000000933690830161154c565b913561177b565b915191168152f35b5082346102ef5760807ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126102ef5761123f611506565b50611248611529565b506064359067ffffffffffffffff82116102ef575060209261126c9136910161154c565b5050517f150b7a02000000000000000000000000000000000000000000000000000000008152f35b5082346102ef5760207ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc3601126102ef5782359067ffffffffffffffff82116102ef57366023830112156102ef57506112fb602093826024611300943693013591016114cf565b611609565b9051908152f35b8490843461025d5760207ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffc36011261025d57357fffffffff00000000000000000000000000000000000000000000000000000000811680910361025d57602092507f150b7a020000000000000000000000000000000000000000000000000000000081149081156113cb575b81156113a1575b5015158152f35b7f01ffc9a7000000000000000000000000000000000000000000000000000000009150148361139a565b7f4e2312e00000000000000000000000000000000000000000000000000000000081149150611393565b67ffffffffffffffff811161140957604052565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b6040810190811067ffffffffffffffff82111761140957604052565b90601f7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0910116810190811067ffffffffffffffff82111761140957604052565b67ffffffffffffffff811161140957601f017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe01660200190565b9291926114db82611495565b916114e96040519384611454565b8294818452818301116103d7578281602093846000960137010152565b6004359073ffffffffffffffffffffffffffffffffffffffff821682036103d757565b6024359073ffffffffffffffffffffffffffffffffffffffff821682036103d757565b9181601f840112156103d75782359167ffffffffffffffff83116103d757602083818601950101116103d757565b9181601f840112156103d75782359167ffffffffffffffff83116103d7576020808501948460051b0101116103d757565b919082519283825260005b8481106115f55750507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0601f8460006020809697860101520116010190565b6020818301810151848301820152016115b6565b6020815191012060405160208101917f5e3baca2936049843f06038876a12f03627b5edc98025751ecf2ac75626401998352604082015260408152606081019181831067ffffffffffffffff841117611409578260405281519020917f0000000000000000000000000000000000000000000000000000000000000000917f000000000000000000000000000000000000000000000000000000000000000030147f0000000000000000000000000000000000000000000000000000000000000000461416156116f5575b5050671901000000000000600052601a52603a5260426018206000603a5290565b60a092507f8b73c3c69bb8fe3d512ecc4cf759cc79239f7b179b0ffacaa9a75d522b39400f82527f000000000000000000000000000000000000000000000000000000000000000060808201527f0000000000000000000000000000000000000000000000000000000000000000838201524660c082015260e0309101522038806116d4565b9061179a61179f9392604051906020820152602081526112fb81611438565b6118b7565b6117c7577fffffffff0000000000000000000000000000000000000000000000000000000090565b7f1626ba7e0000000000000000000000000000000000000000000000000000000090565b91908110156117fb5760051b0190565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b3573ffffffffffffffffffffffffffffffffffffffff811681036103d75790565b9035907fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe1813603018212156103d7570180359067ffffffffffffffff82116103d7576020019181360383136103d757565b908210156117fb576118b39160051b81019061184b565b9091565b90916001908181106119865780156117fb5781843560f81c80611917575081106103d7576119149361190e927fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff36930191016114cf565b90611cf2565b90565b146119465760046040517f60cd402d000000000000000000000000000000000000000000000000000000008152fd5b8082116103d75761191493611980927fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff36930191016114cf565b90611bfb565b60046040517f60cd402d000000000000000000000000000000000000000000000000000000008152fd5b73ffffffffffffffffffffffffffffffffffffffff807f0000000000000000000000000000000000000000000000000000000000000000163314159081611a5d575b81611a2f575b506119ff57565b60246040517f4a0bfec1000000000000000000000000000000000000000000000000000000008152336004820152fd5b90507f691ec1a18226d004c07c9f8e5c4a6ff15a7b38db267cf7e3c945aef8be5122005416331415386119f8565b3330141591506119f2565b3d15611a93573d90611a7982611495565b91611a876040519384611454565b82523d6000602084013e565b606090565b916000928392602083519301915af1611aaf611a68565b9015611ab85750565b602081519101fd5b610100810190611ad0828261184b565b929050600180931061198657611ae6818361184b565b156117fb573560f81c80611b775750611b2c906000947f19457468657265756d205369676e6564204d6573736167653a0a3332000000008652601c52603c85209261184b565b90818411610c6657611b69929161190e91857fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff36930191016114cf565b15611b72575090565b905090565b9280949314611baa5760046040517f60cd402d000000000000000000000000000000000000000000000000000000008152fd5b611bb39161184b565b91908284116103d757611bf09261198091857fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff36930191016114cf565b156119145750600090565b906000809173ffffffffffffffffffffffffffffffffffffffff7f691ec1a18226d004c07c9f8e5c4a6ff15a7b38db267cf7e3c945aef8be512200541690604051611caf81611c8360208201947f1626ba7e00000000000000000000000000000000000000000000000000000000998a875260248401526040604484015260648301906115ab565b037fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe08101835282611454565b51915afa90611cbc611a68565b82611ce4575b82611ccc57505090565b9091506020818051810103126103d757602001511490565b915060208251101591611cc2565b611d0891611cff91611d47565b90929192611d83565b73ffffffffffffffffffffffffffffffffffffffff807f691ec1a18226d004c07c9f8e5c4a6ff15a7b38db267cf7e3c945aef8be512200541691161490565b8151919060418303611d7857611d7192506020820151906060604084015193015160001a90611e6a565b9192909190565b505060009160029190565b6004811015611e3b5780611d95575050565b60018103611dc75760046040517ff645eedf000000000000000000000000000000000000000000000000000000008152fd5b60028103611e0057602482604051907ffce698f70000000000000000000000000000000000000000000000000000000082526004820152fd5b600314611e0a5750565b602490604051907fd78bce0c0000000000000000000000000000000000000000000000000000000082526004820152fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052602160045260246000fd5b91907f7fffffffffffffffffffffffffffffff5d576e7357a4501ddfe92f46681b20a08411611efb57926020929160ff608095604051948552168484015260408301526060820152600092839182805260015afa15611eef57805173ffffffffffffffffffffffffffffffffffffffff811615611ee657918190565b50809160019190565b604051903d90823e3d90fd5b5050506000916003919056fea26469706673582212200896f337e411e9db94675cb703bb4056435327d18f202a547674e38ca452f52464736f6c63430008170033000000000000000000000000ddf32240b4ca3184de7ec8f0d5aba27dec8b7a5c0000000000000000000000000000000071727de22e5e9d8baf0edac6f37da032" + /* ========= TRUST ACCOUNT RELATED ========= */ // Will deploy the Trust Factory 0x729c310186a57833f622630a16d13f710b83272a export const TRUST_FACTORY_V06_CREATECALL: Hex = diff --git a/packages/permissionless-test/mock-aa-infra/alto/index.ts b/packages/permissionless-test/mock-aa-infra/alto/index.ts index 6d7267fa..82685974 100644 --- a/packages/permissionless-test/mock-aa-infra/alto/index.ts +++ b/packages/permissionless-test/mock-aa-infra/alto/index.ts @@ -31,6 +31,7 @@ import { KERNEL_V07_V3_1_ECDSA_VALIDATOR_V3_CREATECALL, KERNEL_V07_V3_1_FACTORY_CREATECALL, LIGHT_ACCOUNT_FACTORY_V110_CREATECALL, + LIGHT_ACCOUNT_FACTORY_V200_CREATECALL, SAFE_7579_LAUNCHPAD_CREATECALL, SAFE_7579_MODULE_CREATECALL, SAFE_7579_REGISTRY_CREATECALL, @@ -234,6 +235,12 @@ export const setupContracts = async (rpc: string) => { gas: 15_000_000n, nonce: nonce++ }), + walletClient.sendTransaction({ + to: DETERMINISTIC_DEPLOYER, + data: LIGHT_ACCOUNT_FACTORY_V200_CREATECALL, + gas: 15_000_000n, + nonce: nonce++ + }), walletClient.sendTransaction({ to: DETERMINISTIC_DEPLOYER, data: TRUST_FACTORY_V06_CREATECALL, @@ -451,6 +458,8 @@ export const setupContracts = async (rpc: string) => { "0xaac5D4240AF87249B3f71BC8E4A2cae074A3E419", // Kernel v0.3.1 Factory "0x00004EC70002a32400f8ae005A26081065620D20", // LightAccountFactory v1.1.0 "0xae8c656ad28F2B59a196AB61815C16A0AE1c3cba", // LightAccount v1.1.0 implementation + "0x0000000000400CdFef5E2714E63d8040b700BC24", // LightAccountFactory v2.0.0 + "0x8E8e658E22B12ada97B402fF0b044D6A325013C7", // LightAccount v2.0.0 implementation "0x81b9E3689390C7e74cF526594A105Dea21a8cdD5", // Trust Secp256k1VerificationFacet "0x729c310186a57833f622630a16d13f710b83272a", // Trust factory "0xFde53272dcd7938d16E031A6989753c321728332", // Trust AccountFacet diff --git a/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts b/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts index d02175ad..f6fddee7 100644 --- a/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts +++ b/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts @@ -329,10 +329,6 @@ const handleMethod = async ( ) if (!params.success) { - console.log({ - params: parsedBody.params - }) - throw new RpcError( fromZodError(params.error).message, ValidationErrors.InvalidFields diff --git a/packages/permissionless-test/src/utils.ts b/packages/permissionless-test/src/utils.ts index 3fa182e5..d8d889b0 100644 --- a/packages/permissionless-test/src/utils.ts +++ b/packages/permissionless-test/src/utils.ts @@ -23,13 +23,16 @@ import { mnemonicToAccount, privateKeyToAccount } from "viem/accounts" -import { foundry } from "viem/chains" +import { foundry, sepolia } from "viem/chains" import { toBiconomySmartAccount } from "../../permissionless/accounts/biconomy/toBiconomySmartAccount" import { type KernelVersion, toEcdsaKernelSmartAccount } from "../../permissionless/accounts/kernel/toEcdsaKernelSmartAccount" -import { toLightSmartAccount } from "../../permissionless/accounts/light/toLightSmartAccount" +import { + type LightAccountVersion, + toLightSmartAccount +} from "../../permissionless/accounts/light/toLightSmartAccount" import { toSafeSmartAccount } from "../../permissionless/accounts/safe/toSafeSmartAccount" import { type ToSimpleSmartAccountReturnType, @@ -113,7 +116,6 @@ export const getBundlerClient = ({ }) => { const paymaster = paymasterRpc ? createPimlicoClient({ - chain: foundry, transport: http(paymasterRpc), entryPoint: { address: @@ -205,12 +207,12 @@ export const getPimlicoClient = ({ export const getPublicClient = (anvilRpc: string) => { const transport = http(anvilRpc, { onFetchRequest: async (request) => { - // console.log("fetching", await request.json()) + console.log("fetching", await request.json()) } }) return createPublicClient({ - chain: foundry, + chain: sepolia, transport: transport, pollingInterval: 100 }) @@ -291,8 +293,11 @@ export const getLightAccountClient = async < entryPointVersion extends "0.6" | "0.7" >({ entryPoint, - anvilRpc -}: AAParamType) => { + anvilRpc, + version +}: AAParamType & { + version?: LightAccountVersion +}) => { return toLightSmartAccount({ entryPoint: { address: @@ -304,7 +309,7 @@ export const getLightAccountClient = async < entryPoint.version === "0.6" ? entryPoint06Abi : entryPoint07Abi }, client: getPublicClient(anvilRpc), - version: "1.1.0", + version: version ?? "1.1.0", owner: privateKeyToAccount(generatePrivateKey()) }) } @@ -415,18 +420,37 @@ export const getCoreSmartAccounts = () => [ isEip1271Compliant: true }, { - name: "LightAccount v1.1.0", + name: "LightAccount 1.1.0", getSmartAccountClient: async ( conf: AAParamType ) => getBundlerClient({ - account: await getLightAccountClient(conf), + account: await getLightAccountClient({ + ...conf, + version: "1.1.0" as LightAccountVersion + }), ...conf }), supportsEntryPointV06: true, supportsEntryPointV07: false, isEip1271Compliant: true }, + { + name: "LightAccount 2.0.0", + getSmartAccountClient: async ( + conf: AAParamType + ) => + getBundlerClient({ + account: await getLightAccountClient({ + ...conf, + version: "2.0.0" as LightAccountVersion + }), + ...conf + }), + supportsEntryPointV06: false, + supportsEntryPointV07: true, + isEip1271Compliant: true + }, { name: "Simple", getSmartAccountClient: async ( diff --git a/packages/permissionless/accounts/light/toLightSmartAccount.ts b/packages/permissionless/accounts/light/toLightSmartAccount.ts index a906ecb5..e5ad2206 100644 --- a/packages/permissionless/accounts/light/toLightSmartAccount.ts +++ b/packages/permissionless/accounts/light/toLightSmartAccount.ts @@ -4,6 +4,7 @@ import { type Client, type Hex, type LocalAccount, + concat, encodeFunctionData, hashMessage, hashTypedData @@ -62,11 +63,14 @@ const getAccountInitCode = async ( }) } -export type LightAccountVersion = "1.1.0" +export type LightAccountVersion = + entryPointVersion extends "0.6" ? "1.1.0" : "2.0.0" export type ToLightSmartAccountParameters< - entryPointVersion extends "0.6" | "0.7", - entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi + entryPointVersion extends "0.6" | "0.7" = "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = typeof entryPoint07Abi > = { client: Client entryPoint?: { @@ -75,7 +79,7 @@ export type ToLightSmartAccountParameters< version: entryPointVersion } owner: LocalAccount - version: LightAccountVersion + version: LightAccountVersion factoryAddress?: Address index?: bigint address?: Address @@ -106,17 +110,20 @@ async function signWith1271WrapperV1( } const LIGHT_VERSION_TO_ADDRESSES_MAP: { - [key in LightAccountVersion]: { + [key in LightAccountVersion<"0.6" | "0.7">]: { factoryAddress: Address } } = { "1.1.0": { factoryAddress: "0x00004EC70002a32400f8ae005A26081065620D20" + }, + "2.0.0": { + factoryAddress: "0x0000000000400CdFef5E2714E63d8040b700BC24" } } const getDefaultAddresses = ( - lightAccountVersion: LightAccountVersion, + lightAccountVersion: LightAccountVersion<"0.6" | "0.7">, { factoryAddress: _factoryAddress }: { @@ -140,15 +147,7 @@ export type LightSmartAccountImplementation< ? typeof entryPoint06Abi : typeof entryPoint07Abi > = Assign< - SmartAccountImplementation< - entryPointAbi, - entryPointVersion - // { - // // entryPoint === ENTRYPOINT_ADDRESS_V06 ? "0.2.2" : "0.3.0-beta" - // abi: entryPointVersion extends "0.6" ? typeof BiconomyAbi - // factory: { abi: typeof FactoryAbi; address: Address } - // } - >, + SmartAccountImplementation, { sign: NonNullable } > @@ -161,14 +160,22 @@ export type ToLightSmartAccountReturnType< LightSmartAccountImplementation > +enum SignatureType { + EOA = "0x00" + // CONTRACT = "0x01", + // CONTRACT_WITH_ADDR = "0x02" +} + /** * @description Creates an Light Account from a private key. * * @returns A Private Key Light Account. */ export async function toLightSmartAccount< - entryPointVersion extends "0.6" | "0.7", - entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi + entryPointVersion extends "0.6" | "0.7" = "0.7", + entryPointAbi extends + | typeof entryPoint06Abi + | typeof entryPoint07Abi = typeof entryPoint07Abi >( parameters: ToLightSmartAccountParameters ): Promise> { @@ -192,12 +199,6 @@ export async function toLightSmartAccount< version: parameters.entryPoint?.version ?? "0.7" } as const - if (version !== "1.1.0") { - throw new Error( - "Only Light Account version 1.1.0 is supported at the moment" - ) - } - const { factoryAddress } = getDefaultAddresses(version, { factoryAddress: _factoryAddress }) @@ -230,7 +231,6 @@ export async function toLightSmartAccount< const { factory, factoryData } = await getFactoryArgs() - // Get the sender address based on the init code accountAddress = await getSenderAddress(client, { factory, factoryData, @@ -314,26 +314,54 @@ export async function toLightSmartAccount< }) }, async getStubSignature() { - return "0xfffffffffffffffffffffffffffffff0000000000000000000000000000000007aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa1c" + const signature = + "0xfffffffffffffffffffffffffffffff0000000000000000000000000000000007aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa1c" + + switch (version) { + case "1.1.0": + return signature + case "2.0.0": + return concat([SignatureType.EOA, signature]) + default: + throw new Error("Unknown Light Account version") + } }, async sign({ hash }) { return this.signMessage({ message: hash }) }, async signMessage({ message }) { - return signWith1271WrapperV1( + const signature = await signWith1271WrapperV1( owner, await getMemoizedChainId(), await this.getAddress(), hashMessage(message) ) + + switch (version) { + case "1.1.0": + return signature + case "2.0.0": + return concat([SignatureType.EOA, signature]) + default: + throw new Error("Unknown Light Account version") + } }, async signTypedData(typedData) { - return signWith1271WrapperV1( + const signature = await signWith1271WrapperV1( owner, await getMemoizedChainId(), await this.getAddress(), hashTypedData(typedData) ) + + switch (version) { + case "1.1.0": + return signature + case "2.0.0": + return concat([SignatureType.EOA, signature]) + default: + throw new Error("Unknown Light Account version") + } }, async signUserOperation(parameters) { const { chainId = await getMemoizedChainId(), ...userOperation } = @@ -349,12 +377,21 @@ export async function toLightSmartAccount< chainId: chainId }) - return signMessage(client, { + const signature = await signMessage(client, { account: owner, message: { raw: hash } }) + + switch (version) { + case "1.1.0": + return signature + case "2.0.0": + return concat([SignatureType.EOA, signature]) + default: + throw new Error("Unknown Light Account version") + } } }) as Promise< ToLightSmartAccountReturnType diff --git a/packages/permissionless/actions/smartAccount/signMessage.test.ts b/packages/permissionless/actions/smartAccount/signMessage.test.ts index 5573c06c..bfb22f92 100644 --- a/packages/permissionless/actions/smartAccount/signMessage.test.ts +++ b/packages/permissionless/actions/smartAccount/signMessage.test.ts @@ -1,3 +1,4 @@ +import { zeroAddress } from "viem" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" import { From ce098022ebd5bebd0a47d10a9f3cbe4d541d15aa Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Mon, 26 Aug 2024 14:00:02 +0100 Subject: [PATCH 18/51] no need to pass custom abi --- .../biconomy/toBiconomySmartAccount.ts | 43 +++++---------- .../kernel/toEcdsaKernelSmartAccount.ts | 31 +++-------- .../accounts/light/toLightSmartAccount.ts | 43 +++++---------- .../accounts/safe/toSafeSmartAccount.ts | 53 +++++-------------- .../accounts/simple/toSimpleSmartAccount.ts | 34 ++++-------- .../accounts/trust/toTrustSmartAccount.ts | 29 ++++------ 6 files changed, 69 insertions(+), 164 deletions(-) diff --git a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts index 4f3b4d64..9aa874ff 100644 --- a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts +++ b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts @@ -70,16 +70,13 @@ const getAccountInitCode = async ({ }) } -export type ToBiconomySmartAccountParameters< - entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi -> = Prettify<{ +export type ToBiconomySmartAccountParameters = Prettify<{ client: Client owner: LocalAccount address?: Address | undefined entryPoint?: { address: typeof entryPoint06Address version: "0.6" - abi: entryPointAbi } nonceKey?: bigint index?: bigint @@ -87,16 +84,14 @@ export type ToBiconomySmartAccountParameters< ecdsaModuleAddress?: Address }> -export type BiconomySmartAccountImplementation< - entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi -> = Assign< - SmartAccountImplementation, +export type BiconomySmartAccountImplementation = Assign< + SmartAccountImplementation, { sign: NonNullable } > -export type ToBiconomySmartAccountReturnType< - entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi -> = Prettify>> +export type ToBiconomySmartAccountReturnType = Prettify< + SmartAccount +> /** * Build a Biconomy modular smart account from a private key, that use the ECDSA signer behind the scene @@ -108,26 +103,16 @@ export type ToBiconomySmartAccountReturnType< * @param ecdsaModuleAddress */ -export async function toBiconomySmartAccount< - entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi ->( - parameters: ToBiconomySmartAccountParameters -): Promise> { - // ): Promise> { - +export async function toBiconomySmartAccount( + parameters: ToBiconomySmartAccountParameters +): Promise { const { owner, client, index = 0n, address } = parameters - const entryPoint = - parameters.entryPoint ?? - ({ - address: entryPoint06Address, - abi: entryPoint06Abi, - version: "0.6" - } as { - address: typeof entryPoint06Address - version: "0.6" - abi: entryPointAbi - }) + const entryPoint = { + address: parameters.entryPoint?.address ?? entryPoint06Address, + abi: entryPoint06Abi, + version: parameters.entryPoint?.version ?? "0.6" + } const factoryAddress = parameters.factoryAddress ?? BICONOMY_ADDRESSES.FACTORY_ADDRESS diff --git a/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts b/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts index ebe56928..8f965724 100644 --- a/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts +++ b/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts @@ -292,14 +292,12 @@ const getAccountInitCode = async ({ export type ToEcdsaKernelSmartAccountParameters< entryPointVersion extends "0.6" | "0.7", - entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi, kernelVersion extends KernelVersion > = { client: Client owner: LocalAccount entryPoint?: { address: typeof entryPoint06Address | typeof entryPoint07Address - abi: entryPointAbi version: entryPointVersion } address?: Address @@ -313,13 +311,12 @@ export type ToEcdsaKernelSmartAccountParameters< } export type EcdsaKernelSmartAccountImplementation< - entryPointVersion extends "0.6" | "0.7" = "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = typeof entryPoint07Abi + entryPointVersion extends "0.6" | "0.7" = "0.7" > = Assign< SmartAccountImplementation< - entryPointAbi, + entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi, entryPointVersion // { // // entryPoint === ENTRYPOINT_ADDRESS_V06 ? "0.2.2" : "0.3.0-beta" @@ -331,13 +328,8 @@ export type EcdsaKernelSmartAccountImplementation< > export type ToEcdsaKernelSmartAccountReturnType< - entryPointVersion extends "0.6" | "0.7" = "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = typeof entryPoint07Abi -> = SmartAccount< - EcdsaKernelSmartAccountImplementation -> + entryPointVersion extends "0.6" | "0.7" = "0.7" +> = SmartAccount> /** * Build a kernel smart account from a private key, that use the ECDSA signer behind the scene * @param client @@ -350,17 +342,13 @@ export type ToEcdsaKernelSmartAccountReturnType< */ export async function toEcdsaKernelSmartAccount< entryPointVersion extends "0.6" | "0.7", - entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi, kernelVersion extends KernelVersion >( parameters: ToEcdsaKernelSmartAccountParameters< entryPointVersion, - entryPointAbi, kernelVersion > -): Promise< - ToEcdsaKernelSmartAccountReturnType -> { +): Promise> { const { client, address, @@ -376,7 +364,6 @@ export async function toEcdsaKernelSmartAccount< const entryPoint = { address: parameters.entryPoint?.address ?? entryPoint07Address, abi: - parameters.entryPoint?.abi ?? (parameters.entryPoint?.version ?? "0.7") === "0.6" ? entryPoint06Abi : entryPoint07Abi, @@ -534,7 +521,5 @@ export async function toEcdsaKernelSmartAccount< } return signature } - }) as Promise< - ToEcdsaKernelSmartAccountReturnType - > + }) as Promise> } diff --git a/packages/permissionless/accounts/light/toLightSmartAccount.ts b/packages/permissionless/accounts/light/toLightSmartAccount.ts index e5ad2206..c627be87 100644 --- a/packages/permissionless/accounts/light/toLightSmartAccount.ts +++ b/packages/permissionless/accounts/light/toLightSmartAccount.ts @@ -67,15 +67,11 @@ export type LightAccountVersion = entryPointVersion extends "0.6" ? "1.1.0" : "2.0.0" export type ToLightSmartAccountParameters< - entryPointVersion extends "0.6" | "0.7" = "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = typeof entryPoint07Abi + entryPointVersion extends "0.6" | "0.7" = "0.7" > = { client: Client entryPoint?: { address: typeof entryPoint06Address | typeof entryPoint07Address - abi: entryPointAbi version: entryPointVersion } owner: LocalAccount @@ -140,25 +136,20 @@ const getDefaultAddresses = ( } export type LightSmartAccountImplementation< - entryPointVersion extends "0.6" | "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi + entryPointVersion extends "0.6" | "0.7" > = Assign< - SmartAccountImplementation, + SmartAccountImplementation< + entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi, + entryPointVersion + >, { sign: NonNullable } > export type ToLightSmartAccountReturnType< - entryPointVersion extends "0.6" | "0.7" = "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = typeof entryPoint07Abi -> = SmartAccount< - LightSmartAccountImplementation -> + entryPointVersion extends "0.6" | "0.7" = "0.7" +> = SmartAccount> enum SignatureType { EOA = "0x00" @@ -172,13 +163,10 @@ enum SignatureType { * @returns A Private Key Light Account. */ export async function toLightSmartAccount< - entryPointVersion extends "0.6" | "0.7" = "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = typeof entryPoint07Abi + entryPointVersion extends "0.6" | "0.7" = "0.7" >( - parameters: ToLightSmartAccountParameters -): Promise> { + parameters: ToLightSmartAccountParameters +): Promise> { const { version, factoryAddress: _factoryAddress, @@ -192,7 +180,6 @@ export async function toLightSmartAccount< const entryPoint = { address: parameters.entryPoint?.address ?? entryPoint07Address, abi: - parameters.entryPoint?.abi ?? (parameters.entryPoint?.version ?? "0.7") === "0.6" ? entryPoint06Abi : entryPoint07Abi, @@ -393,7 +380,5 @@ export async function toLightSmartAccount< throw new Error("Unknown Light Account version") } } - }) as Promise< - ToLightSmartAccountReturnType - > + }) as Promise> } diff --git a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts index 7cb8a5cd..347b9ab2 100644 --- a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts +++ b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts @@ -929,7 +929,6 @@ type GetErc7579Params = export type ToSafeSmartAccountParameters< entryPointVersion extends "0.6" | "0.7", - entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi, TErc7579 extends Address | undefined > = { client: Client @@ -937,7 +936,6 @@ export type ToSafeSmartAccountParameters< version: SafeVersion entryPoint?: { address: typeof entryPoint06Address | typeof entryPoint07Address - abi: entryPointAbi version: entryPointVersion } safe4337ModuleAddress?: Address @@ -955,33 +953,19 @@ export type ToSafeSmartAccountParameters< paymentReceiver?: Address } & GetErc7579Params -function isErc7579Args< - entryPointVersion extends "0.6" | "0.7" = "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = typeof entryPoint07Abi ->( - args: ToSafeSmartAccountParameters< - entryPointVersion, - entryPointAbi, - Address | undefined - > -): args is ToSafeSmartAccountParameters< - entryPointVersion, - entryPointAbi, - Address -> { +function isErc7579Args( + args: ToSafeSmartAccountParameters +): args is ToSafeSmartAccountParameters { return args.erc7579LaunchpadAddress !== undefined } export type SafeSmartAccountImplementation< - entryPointVersion extends "0.6" | "0.7" = "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = typeof entryPoint07Abi + entryPointVersion extends "0.6" | "0.7" = "0.7" > = Assign< SmartAccountImplementation< - entryPointAbi, + entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi, entryPointVersion // { // // entryPoint === ENTRYPOINT_ADDRESS_V06 ? "0.2.2" : "0.3.0-beta" @@ -993,13 +977,8 @@ export type SafeSmartAccountImplementation< > export type ToSafeSmartAccountReturnType< - entryPointVersion extends "0.6" | "0.7" = "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = typeof entryPoint07Abi -> = SmartAccount< - SafeSmartAccountImplementation -> + entryPointVersion extends "0.6" | "0.7" = "0.7" +> = SmartAccount> /** * @description Creates an Simple Account from a private key. @@ -1008,15 +987,10 @@ export type ToSafeSmartAccountReturnType< */ export async function toSafeSmartAccount< entryPointVersion extends "0.6" | "0.7", - entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi, TErc7579 extends Address | undefined >( - parameters: ToSafeSmartAccountParameters< - entryPointVersion, - entryPointAbi, - TErc7579 - > -): Promise> { + parameters: ToSafeSmartAccountParameters +): Promise> { const { client, owner, @@ -1038,7 +1012,6 @@ export async function toSafeSmartAccount< const entryPoint = { address: parameters.entryPoint?.address ?? entryPoint07Address, abi: - parameters.entryPoint?.abi ?? (parameters.entryPoint?.version ?? "0.7") === "0.6" ? entryPoint06Abi : entryPoint07Abi, @@ -1378,7 +1351,5 @@ export async function toSafeSmartAccount< [validAfter, validUntil, signatureBytes] ) } - }) as Promise< - ToSafeSmartAccountReturnType - > + }) as Promise> } diff --git a/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts b/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts index a7344c2a..5f34eed2 100644 --- a/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts +++ b/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts @@ -61,15 +61,13 @@ const getAccountInitCode = async ( } export type ToSimpleSmartAccountParameters< - entryPointVersion extends "0.6" | "0.7", - entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi + entryPointVersion extends "0.6" | "0.7" > = { client: Client owner: LocalAccount factoryAddress?: Address entryPoint?: { address: typeof entryPoint06Address | typeof entryPoint07Address - abi: entryPointAbi version: entryPointVersion } index?: bigint @@ -90,13 +88,12 @@ const getFactoryAddress = ( } export type SimpleSmartAccountImplementation< - entryPointVersion extends "0.6" | "0.7" = "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = typeof entryPoint07Abi + entryPointVersion extends "0.6" | "0.7" = "0.7" > = Assign< SmartAccountImplementation< - entryPointAbi, + entryPointVersion extends "0.6" + ? typeof entryPoint06Abi + : typeof entryPoint07Abi, entryPointVersion // { // // entryPoint === ENTRYPOINT_ADDRESS_V06 ? "0.2.2" : "0.3.0-beta" @@ -108,13 +105,8 @@ export type SimpleSmartAccountImplementation< > export type ToSimpleSmartAccountReturnType< - entryPointVersion extends "0.6" | "0.7" = "0.7", - entryPointAbi extends - | typeof entryPoint06Abi - | typeof entryPoint07Abi = typeof entryPoint07Abi -> = SmartAccount< - SimpleSmartAccountImplementation -> + entryPointVersion extends "0.6" | "0.7" = "0.7" +> = SmartAccount> /** * @description Creates an Simple Account from a private key. @@ -122,11 +114,10 @@ export type ToSimpleSmartAccountReturnType< * @returns A Private Key Simple Account. */ export async function toSimpleSmartAccount< - entryPointVersion extends "0.6" | "0.7", - entryPointAbi extends typeof entryPoint06Abi | typeof entryPoint07Abi + entryPointVersion extends "0.6" | "0.7" >( - parameters: ToSimpleSmartAccountParameters -): Promise> { + parameters: ToSimpleSmartAccountParameters +): Promise> { const { client, owner, @@ -139,7 +130,6 @@ export async function toSimpleSmartAccount< const entryPoint = { address: parameters.entryPoint?.address ?? entryPoint07Address, abi: - parameters.entryPoint?.abi ?? (parameters.entryPoint?.version ?? "0.7") === "0.6" ? entryPoint06Abi : entryPoint07Abi, @@ -324,7 +314,5 @@ export async function toSimpleSmartAccount< } }) } - }) as Promise< - ToSimpleSmartAccountReturnType - > + }) as Promise> } diff --git a/packages/permissionless/accounts/trust/toTrustSmartAccount.ts b/packages/permissionless/accounts/trust/toTrustSmartAccount.ts index 29ef1ab7..cc9b3b97 100644 --- a/packages/permissionless/accounts/trust/toTrustSmartAccount.ts +++ b/packages/permissionless/accounts/trust/toTrustSmartAccount.ts @@ -60,15 +60,13 @@ export const TRUST_ADDRESSES: { } export type ToTrustSmartAccountParameters< - entryPointVersion extends "0.6" = "0.6", - entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi + entryPointVersion extends "0.6" = "0.6" > = { client: Client owner: LocalAccount factoryAddress?: Address entryPoint?: { address: typeof entryPoint06Address - abi: entryPointAbi version: entryPointVersion } index?: bigint @@ -78,11 +76,10 @@ export type ToTrustSmartAccountParameters< } export type TrustSmartAccountImplementation< - entryPointVersion extends "0.6" = "0.6", - entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi + entryPointVersion extends "0.6" = "0.6" > = Assign< SmartAccountImplementation< - entryPointAbi, + typeof entryPoint06Abi, entryPointVersion // { // // entryPoint === ENTRYPOINT_ADDRESS_V06 ? "0.2.2" : "0.3.0-beta" @@ -94,11 +91,8 @@ export type TrustSmartAccountImplementation< > export type ToTrustSmartAccountReturnType< - entryPointVersion extends "0.6" = "0.6", - entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi -> = SmartAccount< - TrustSmartAccountImplementation -> + entryPointVersion extends "0.6" = "0.6" +> = SmartAccount> /** * @description Creates an Trust Smart Account from a private key. @@ -106,11 +100,10 @@ export type ToTrustSmartAccountReturnType< * @returns A Private Key Trust Smart Account. */ export async function toTrustSmartAccount< - entryPointVersion extends "0.6" = "0.6", - entryPointAbi extends typeof entryPoint06Abi = typeof entryPoint06Abi + entryPointVersion extends "0.6" = "0.6" >( - parameters: ToTrustSmartAccountParameters -): Promise> { + parameters: ToTrustSmartAccountParameters +): Promise> { const { owner, client, @@ -124,7 +117,7 @@ export async function toTrustSmartAccount< const entryPoint = { address: parameters.entryPoint?.address ?? entryPoint06Address, - abi: parameters.entryPoint?.abi ?? entryPoint06Abi, + abi: entryPoint06Abi, version: parameters.entryPoint?.version ?? "0.6" } as const @@ -221,7 +214,5 @@ export async function toTrustSmartAccount< } }) } - }) as Promise< - ToTrustSmartAccountReturnType - > + }) as Promise> } From 8e73cfe663c9fdf15a169fd2d4597cb1f4a127f8 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Mon, 26 Aug 2024 14:04:09 +0100 Subject: [PATCH 19/51] fix test --- packages/permissionless-test/src/utils.ts | 35 +++++------------------ 1 file changed, 7 insertions(+), 28 deletions(-) diff --git a/packages/permissionless-test/src/utils.ts b/packages/permissionless-test/src/utils.ts index d8d889b0..a7fa2d39 100644 --- a/packages/permissionless-test/src/utils.ts +++ b/packages/permissionless-test/src/utils.ts @@ -207,7 +207,7 @@ export const getPimlicoClient = ({ export const getPublicClient = (anvilRpc: string) => { const transport = http(anvilRpc, { onFetchRequest: async (request) => { - console.log("fetching", await request.json()) + // console.log("fetching", await request.json()) } }) @@ -257,19 +257,9 @@ export const getSimpleAccountClient = async < entryPoint, anvilRpc }: AAParamType): Promise< - ToSimpleSmartAccountReturnType< - entryPointVersion, - entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi - > + ToSimpleSmartAccountReturnType > => { - return toSimpleSmartAccount< - entryPointVersion, - entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi - >({ + return toSimpleSmartAccount({ client: getPublicClient(anvilRpc), entryPoint: { address: @@ -278,12 +268,7 @@ export const getSimpleAccountClient = async < : entryPoint07Address, version: (entryPoint.version === "0.6" ? "0.6" - : "0.7") as entryPointVersion, - abi: (entryPoint.version === "0.6" - ? entryPoint06Abi - : entryPoint07Abi) as entryPointVersion extends "0.6" - ? typeof entryPoint06Abi - : typeof entryPoint07Abi + : "0.7") as entryPointVersion }, owner: privateKeyToAccount(generatePrivateKey()) }) @@ -304,9 +289,7 @@ export const getLightAccountClient = async < entryPoint.version === "0.6" ? entryPoint06Address : entryPoint07Address, - version: entryPoint.version === "0.6" ? "0.6" : "0.7", - abi: - entryPoint.version === "0.6" ? entryPoint06Abi : entryPoint07Abi + version: entryPoint.version === "0.6" ? "0.6" : "0.7" }, client: getPublicClient(anvilRpc), version: version ?? "1.1.0", @@ -363,9 +346,7 @@ export const getKernelEcdsaClient = async < entryPoint.version === "0.6" ? entryPoint06Address : entryPoint07Address, - version: entryPoint.version === "0.6" ? "0.6" : "0.7", - abi: - entryPoint.version === "0.6" ? entryPoint06Abi : entryPoint07Abi + version: entryPoint.version === "0.6" ? "0.6" : "0.7" }, owner: privateKeyToAccount(generatePrivateKey()), version @@ -388,9 +369,7 @@ export const getSafeClient = async ({ entryPoint.version === "0.6" ? entryPoint06Address : entryPoint07Address, - version: entryPoint.version === "0.6" ? "0.6" : "0.7", - abi: - entryPoint.version === "0.6" ? entryPoint06Abi : entryPoint07Abi + version: entryPoint.version === "0.6" ? "0.6" : "0.7" }, owner: privateKeyToAccount(generatePrivateKey()), version: "1.4.1", From 79859633dadad064e6f69102313219d8c5bc8014 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Mon, 26 Aug 2024 14:15:47 +0100 Subject: [PATCH 20/51] Fix test --- packages/permissionless-test/src/utils.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/permissionless-test/src/utils.ts b/packages/permissionless-test/src/utils.ts index a7fa2d39..68f13259 100644 --- a/packages/permissionless-test/src/utils.ts +++ b/packages/permissionless-test/src/utils.ts @@ -23,7 +23,7 @@ import { mnemonicToAccount, privateKeyToAccount } from "viem/accounts" -import { foundry, sepolia } from "viem/chains" +import { foundry } from "viem/chains" import { toBiconomySmartAccount } from "../../permissionless/accounts/biconomy/toBiconomySmartAccount" import { type KernelVersion, @@ -212,7 +212,7 @@ export const getPublicClient = (anvilRpc: string) => { }) return createPublicClient({ - chain: sepolia, + chain: foundry, transport: transport, pollingInterval: 100 }) From e4932a4fb5b3f26e4ca9bc4a7a3536cf9ebbba28 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Tue, 27 Aug 2024 14:51:38 +0100 Subject: [PATCH 21/51] Loose entrypoint address type --- .../accounts/biconomy/toBiconomySmartAccount.ts | 2 +- .../accounts/kernel/toEcdsaKernelSmartAccount.ts | 3 +-- .../accounts/light/toLightSmartAccount.ts | 3 +-- .../accounts/safe/toSafeSmartAccount.ts | 3 +-- .../accounts/simple/toSimpleSmartAccount.ts | 3 +-- .../accounts/trust/toTrustSmartAccount.ts | 2 +- .../permissionless/actions/public/getAccountNonce.ts | 6 +----- .../actions/public/getSenderAddress.ts | 12 ++---------- 8 files changed, 9 insertions(+), 25 deletions(-) diff --git a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts index 9aa874ff..1308f9f6 100644 --- a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts +++ b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts @@ -75,7 +75,7 @@ export type ToBiconomySmartAccountParameters = Prettify<{ owner: LocalAccount address?: Address | undefined entryPoint?: { - address: typeof entryPoint06Address + address: Address version: "0.6" } nonceKey?: bigint diff --git a/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts b/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts index 8f965724..edf5e27c 100644 --- a/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts +++ b/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts @@ -15,7 +15,6 @@ import { type SmartAccountImplementation, type UserOperation, entryPoint06Abi, - type entryPoint06Address, entryPoint07Abi, entryPoint07Address, getUserOperationHash, @@ -297,7 +296,7 @@ export type ToEcdsaKernelSmartAccountParameters< client: Client owner: LocalAccount entryPoint?: { - address: typeof entryPoint06Address | typeof entryPoint07Address + address: Address version: entryPointVersion } address?: Address diff --git a/packages/permissionless/accounts/light/toLightSmartAccount.ts b/packages/permissionless/accounts/light/toLightSmartAccount.ts index c627be87..c50f29c9 100644 --- a/packages/permissionless/accounts/light/toLightSmartAccount.ts +++ b/packages/permissionless/accounts/light/toLightSmartAccount.ts @@ -14,7 +14,6 @@ import { type SmartAccountImplementation, type UserOperation, entryPoint06Abi, - type entryPoint06Address, entryPoint07Abi, entryPoint07Address, getUserOperationHash, @@ -71,7 +70,7 @@ export type ToLightSmartAccountParameters< > = { client: Client entryPoint?: { - address: typeof entryPoint06Address | typeof entryPoint07Address + address: Address version: entryPointVersion } owner: LocalAccount diff --git a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts index 347b9ab2..0f181d8b 100644 --- a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts +++ b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts @@ -25,7 +25,6 @@ import { type SmartAccountImplementation, type UserOperation, entryPoint06Abi, - type entryPoint06Address, entryPoint07Abi, entryPoint07Address, toSmartAccount @@ -935,7 +934,7 @@ export type ToSafeSmartAccountParameters< owner: LocalAccount version: SafeVersion entryPoint?: { - address: typeof entryPoint06Address | typeof entryPoint07Address + address: Address version: entryPointVersion } safe4337ModuleAddress?: Address diff --git a/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts b/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts index 5f34eed2..138e69f3 100644 --- a/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts +++ b/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts @@ -11,7 +11,6 @@ import { type SmartAccountImplementation, type UserOperation, entryPoint06Abi, - type entryPoint06Address, entryPoint07Abi, entryPoint07Address, getUserOperationHash, @@ -67,7 +66,7 @@ export type ToSimpleSmartAccountParameters< owner: LocalAccount factoryAddress?: Address entryPoint?: { - address: typeof entryPoint06Address | typeof entryPoint07Address + address: Address version: entryPointVersion } index?: bigint diff --git a/packages/permissionless/accounts/trust/toTrustSmartAccount.ts b/packages/permissionless/accounts/trust/toTrustSmartAccount.ts index cc9b3b97..8afdffcf 100644 --- a/packages/permissionless/accounts/trust/toTrustSmartAccount.ts +++ b/packages/permissionless/accounts/trust/toTrustSmartAccount.ts @@ -66,7 +66,7 @@ export type ToTrustSmartAccountParameters< owner: LocalAccount factoryAddress?: Address entryPoint?: { - address: typeof entryPoint06Address + address: Address version: entryPointVersion } index?: bigint diff --git a/packages/permissionless/actions/public/getAccountNonce.ts b/packages/permissionless/actions/public/getAccountNonce.ts index 97113c75..002a430f 100644 --- a/packages/permissionless/actions/public/getAccountNonce.ts +++ b/packages/permissionless/actions/public/getAccountNonce.ts @@ -1,14 +1,10 @@ import type { Address, Client } from "viem" -import type { - entryPoint06Address, - entryPoint07Address -} from "viem/account-abstraction" import { readContract } from "viem/actions" import { getAction } from "viem/utils" export type GetAccountNonceParams = { address: Address - entryPointAddress: typeof entryPoint06Address | typeof entryPoint07Address + entryPointAddress: Address key?: bigint } diff --git a/packages/permissionless/actions/public/getSenderAddress.ts b/packages/permissionless/actions/public/getSenderAddress.ts index ff720082..7d9fd8d7 100644 --- a/packages/permissionless/actions/public/getSenderAddress.ts +++ b/packages/permissionless/actions/public/getSenderAddress.ts @@ -14,26 +14,18 @@ import { decodeErrorResult } from "viem" -import type { - entryPoint06Address, - entryPoint07Address -} from "viem/account-abstraction" import { simulateContract } from "viem/actions" import { getAction } from "viem/utils" export type GetSenderAddressParams = OneOf< | { initCode: Hex - entryPointAddress: - | typeof entryPoint06Address - | typeof entryPoint07Address + entryPointAddress: Address factory?: never factoryData?: never } | { - entryPointAddress: - | typeof entryPoint06Address - | typeof entryPoint07Address + entryPointAddress: Address factory: Address factoryData: Hex initCode?: never From cf66ee98ec9b623846ce9af2a3fa7e73ad35accc Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Tue, 27 Aug 2024 15:08:38 +0100 Subject: [PATCH 22/51] Fix safe --- .../accounts/safe/toSafeSmartAccount.ts | 147 +++++++++++++++++- 1 file changed, 139 insertions(+), 8 deletions(-) diff --git a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts index 0f181d8b..767b0641 100644 --- a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts +++ b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts @@ -12,8 +12,10 @@ import { encodeAbiParameters, encodeFunctionData, encodePacked, + getContractAddress, hashMessage, hashTypedData, + hexToBigInt, keccak256, pad, toBytes, @@ -29,10 +31,9 @@ import { entryPoint07Address, toSmartAccount } from "viem/account-abstraction" -import { getChainId, signTypedData } from "viem/actions" +import { getChainId, readContract, signTypedData } from "viem/actions" import { getAction } from "viem/utils" import { getAccountNonce } from "../../actions/public/getAccountNonce" -import { getSenderAddress } from "../../actions/public/getSenderAddress" import { isSmartAccountDeployed } from "../../utils" import { encode7579Calls } from "../../utils/encode7579Calls" @@ -958,6 +959,121 @@ function isErc7579Args( return args.erc7579LaunchpadAddress !== undefined } +const proxyCreationCodeAbi = [ + { + inputs: [], + name: "proxyCreationCode", + outputs: [ + { + internalType: "bytes", + name: "", + type: "bytes" + } + ], + stateMutability: "pure", + type: "function" + } +] as const + +const getAccountAddress = async ({ + client, + owner, + safeModuleSetupAddress, + safe4337ModuleAddress, + safeProxyFactoryAddress, + safeSingletonAddress, + multiSendAddress, + erc7579LaunchpadAddress, + paymentToken, + payment, + paymentReceiver, + setupTransactions = [], + safeModules = [], + saltNonce = BigInt(0), + validators = [], + executors = [], + fallbacks = [], + hooks = [], + attesters = [], + attestersThreshold = 0 +}: { + client: Client + owner: Address + safeModuleSetupAddress: Address + safe4337ModuleAddress: Address + safeProxyFactoryAddress: Address + safeSingletonAddress: Address + multiSendAddress: Address + setupTransactions: { + to: Address + data: Address + value: bigint + }[] + paymentToken?: Address + payment?: bigint + paymentReceiver?: Address + safeModules?: Address[] + saltNonce?: bigint + erc7579LaunchpadAddress?: Address + validators?: { address: Address; context: Address }[] + executors?: { + address: Address + context: Address + }[] + fallbacks?: { address: Address; context: Address }[] + hooks?: { address: Address; context: Address }[] + attesters?: Address[] + attestersThreshold?: number +}): Promise
=> { + const proxyCreationCode = await readContract(client, { + abi: proxyCreationCodeAbi, + address: safeProxyFactoryAddress, + functionName: "proxyCreationCode" + }) + + const initializer = await getInitializerCode({ + owner, + safeModuleSetupAddress, + safe4337ModuleAddress, + multiSendAddress, + setupTransactions, + safeSingletonAddress, + safeModules, + erc7579LaunchpadAddress, + validators, + executors, + fallbacks, + hooks, + attesters, + attestersThreshold, + paymentToken, + payment, + paymentReceiver + }) + + const deploymentCode = encodePacked( + ["bytes", "uint256"], + [ + proxyCreationCode, + hexToBigInt(erc7579LaunchpadAddress ?? safeSingletonAddress) + ] + ) + + const salt = keccak256( + encodePacked( + ["bytes32", "uint256"], + [keccak256(encodePacked(["bytes"], [initializer])), saltNonce] + ) + ) + + return getContractAddress({ + from: safeProxyFactoryAddress, + salt, + bytecode: deploymentCode, + opcode: "CREATE2" + }) +} + export type SafeSmartAccountImplementation< entryPointVersion extends "0.6" | "0.7" = "0.7" > = Assign< @@ -1111,13 +1227,28 @@ export async function toSafeSmartAccount< async getAddress() { if (accountAddress) return accountAddress - const { factory, factoryData } = await getFactoryArgs() - // Get the sender address based on the init code - accountAddress = await getSenderAddress(client, { - factory, - factoryData, - entryPointAddress: entryPoint.address + accountAddress = await getAccountAddress({ + client, + owner: owner.address, + safeModuleSetupAddress, + safe4337ModuleAddress, + safeProxyFactoryAddress, + safeSingletonAddress, + multiSendAddress, + erc7579LaunchpadAddress, + saltNonce, + setupTransactions, + safeModules, + validators, + executors, + fallbacks, + hooks, + attesters, + attestersThreshold, + paymentToken, + payment, + paymentReceiver }) return accountAddress From 205f34aca09c8ed5e4f2752c2851144936096579 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Tue, 27 Aug 2024 15:16:29 +0100 Subject: [PATCH 23/51] Add second transaction post deployment --- .../smartAccount/sendTransaction.test.ts | 38 +++++++++++++++++++ 1 file changed, 38 insertions(+) diff --git a/packages/permissionless/actions/smartAccount/sendTransaction.test.ts b/packages/permissionless/actions/smartAccount/sendTransaction.test.ts index c8bb79b7..21864cac 100644 --- a/packages/permissionless/actions/smartAccount/sendTransaction.test.ts +++ b/packages/permissionless/actions/smartAccount/sendTransaction.test.ts @@ -45,6 +45,25 @@ describe.each(getCoreSmartAccounts())( expect(receipt).toBeTruthy() expect(receipt.transactionHash).toBe(transactionHash) expect(receipt.status).toBe("success") + + // -- second transaction after deployment + + const transactionHash2 = await sendTransaction(smartClient, { + chain: foundry, + to: zeroAddress, + data: "0x", + value: 0n + }) + + expect(transactionHash2).toBeTruthy() + + const receipt2 = await publicClient.getTransactionReceipt({ + hash: transactionHash2 + }) + + expect(receipt2).toBeTruthy() + expect(receipt2.transactionHash).toBe(transactionHash2) + expect(receipt2.status).toBe("success") } ) @@ -78,6 +97,25 @@ describe.each(getCoreSmartAccounts())( expect(receipt).toBeTruthy() expect(receipt.transactionHash).toBe(transactionHash) expect(receipt.status).toBe("success") + + const transactionHash2 = await sendTransaction(smartClient, { + chain: foundry, + to: zeroAddress, + data: "0x", + value: 0n + }) + + // -- second transaction after deployment + + expect(transactionHash2).toBeTruthy() + + const receipt2 = await publicClient.getTransactionReceipt({ + hash: transactionHash2 + }) + + expect(receipt2).toBeTruthy() + expect(receipt2.transactionHash).toBe(transactionHash2) + expect(receipt2.status).toBe("success") } ) } From 6d33f8699f223a9375a3932cc08d886cfa57a863 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Tue, 27 Aug 2024 16:25:03 +0100 Subject: [PATCH 24/51] export clients --- packages/permissionless/clients/index.ts | 2 ++ packages/permissionless/index.ts | 1 + 2 files changed, 3 insertions(+) create mode 100644 packages/permissionless/clients/index.ts diff --git a/packages/permissionless/clients/index.ts b/packages/permissionless/clients/index.ts new file mode 100644 index 00000000..9684a3c4 --- /dev/null +++ b/packages/permissionless/clients/index.ts @@ -0,0 +1,2 @@ +export * from "./createSmartAccountClient" +export * from "./decorators/smartAccount" diff --git a/packages/permissionless/index.ts b/packages/permissionless/index.ts index 1f4ea9b7..ed21b5a7 100644 --- a/packages/permissionless/index.ts +++ b/packages/permissionless/index.ts @@ -1,2 +1,3 @@ export * from "./utils/" export * from "./errors/" +export * from "./clients/" From 4994e0b460e1b9cf927ac1f2863cb3802ca67db5 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Wed, 28 Aug 2024 15:26:13 +0100 Subject: [PATCH 25/51] bum min viem to be 2.20.0 --- bun.lockb | Bin 579552 -> 579552 bytes package.json | 2 +- .../mock-paymaster/mockAccount.ts | 27 ------------------ .../mock-aa-infra/mock-paymaster/relay.ts | 17 ----------- packages/permissionless-test/src/utils.ts | 6 ++-- .../smartAccount/sendTransaction.test.ts | 4 --- .../clients/createSmartAccountClient.ts | 6 ++-- packages/permissionless/package.json | 2 +- 8 files changed, 8 insertions(+), 56 deletions(-) delete mode 100644 packages/permissionless-test/mock-aa-infra/mock-paymaster/mockAccount.ts diff --git a/bun.lockb b/bun.lockb index 0ad1a05128932a977f673b53bd95eda7f1d56e98..4757c3aca92e96f0e3df57b82e55651a8ba9635a 100755 GIT binary patch delta 2176 zcmcgtYfMx}6uxtJ=iXfup+I4IsH+$S7B0IeuLV(jv^FI~S`8Xsfl3QT0lO%V2=Y)^ zO-hAThtyh?s;!~w5~QOdX|*=1@{rb5Y_$eWQJ}?C8ynCnJ+tghY?^A)AD!g8XU=!e zoSAcG=1kA=z@FoQJ0$Ju%Cd1CXMfU9D0?{Z?xUTJ_a^(?sZgJ>?~&?jwQEZMY)tl3 zW@>{u&YR;(JwdvH6r6H6XjC5rITm^4kM73bq)F64d1o=NwokXF)5X(Rs_M{8@MexH!wx z1Mw;dHWq#&OPMJKg;-7(Cg5hqSz7bs@O|h-S)MMBGX)xx2=Fa{@iqW!Oez4E z0E}-3FuM+?e%Yyafn<$Io<2Q=Qg#2&JnN>>$*z1;NK$TRcvHvgdjpph+syWJKijs? zNsYQ%y=uMLe!N$Etl4b8kdnMO@SB}U2M)W|PY8Zsdpqg2`qs_4b9?VxdDvp#2vH&K zncFjv#UwGFPo)Jh{Hut}kK^AY*JF8CvKM-&g7nAnsl;pqV$cZ0R#FxVL`)nIy|F-4 z5rYwkK1S?f7Td_A7(TVCE}o}#@rtio$o_|lg=EQ)BFK`vLRn=WQc(Mlva3ej0?9iA zXr)KD<0m}2NRNDgCnD5IFIMAy3Urxf)#7dyI!zPz;9p!efC@kvpcJqfPy#3h zj6f>|$v7G1(3d&ZXEv-=I{Q^1&Cl_4#}WL65>?Wf$8a=1p1CzciYftBfUSTnOafm? zYOL7Pm3fs-%4wq&=lCEqopb@`C=i&mVgFZ{7poKFVRR*FZNn|7oD{a>>DUntX>7;Q zFs{EHw_#)^&EI1as-nSHuqOgHGws+P71O15e3cg?qzJvF=V)X%Ugl2ntb&ISBN?J4 z14*(88e;Vkl*FSC`|-DfNY6d2@s~@1e3yA-kfwtNjvFQo_pmP#NJ^*R8z2`ZxfbvU z=8+=7P8+=22V5&J2!4@rxsodck5(F_8O)trYUCR7+#kmTFb^8QzPu@fboOHn8YU?k z!Alq&`?p_&*9abH3aRPGlgJF4;6BwIK`C&wYPkC@u&bG-VSguQb zgq`Xsu%I!l7_kt%w2sWmHC!&6JSs6r2Ij{RG)p-%CYv<^_Wu{-2*@a#)0dvq2sECL z(9`vch3I83j;rK%fn|U~1`c|&BajR`ColMA;rTk#Kw+mJD`Al%?E>2`>enFzAlR_y zj*EP}z&EYz6xJY?_x+Cq4Rz@fmIS_Vl@wN~<=HY}Ks^ty5t%n6B=L1RT00~pi~n`q zjuYCvNnFIgCBKS+*S4QMaBx$8dRDfy2A)q{nDR5y(>CN~W{2fQL`D%5CVCT(PgMbAk6zT1U*8q|$(Lc`Npd$_ zJV92Jt2_t|7sE-mB+erhVd8YsA&J+?en}jofU8|)(~H!mR~ax delta 2177 zcmcgtdr(wW7~gYt&%Fz}3WDtNDwINZ7xsb3!tziXY%Cp$#zM#BWh5{n+N!_?$Wy=> zO}OG=vPp8(G(ure4~9(Al&IjF0JAYH3a~Nlp%RYTcbDBpXUP6&rZe-q-}xTrJKy=8 z$L&Al-+#z|o6axCvZ&lTbyS(OFYb@NQ2EjQd6P7Xs0XJ*;*?n@&upz}iuIDEWm_b+ zrdCTN3W=mpW|#M7j_K)^EL(^pE))EdAg)ILP>Wk=w|K*5|HU6NyEIac)CoMa5l zehCoeNgK;!5<$NMAj(q!(SOFScY+j~C`J2OyF3q4Y@(C`E{O_za16z3|DSm>XRzrC z+^kvjeBaC0hFHt(yGN}Vsw+-uAya<{`7Zo>x+bxv;T2c!^U_@_4{u*`>qf#)7faqw zYUOth|9LzvQO?{O!p&cu)DOa{k>ugzhamImyht3+a>DRbL~ok$LfRLO6?A?CUWiI) zRRoTw6=pEp5e;Ru(hP>M2rz632SYi%5)OtT(GUfLQ6-%ghU3d?BQdLul%8m%`$wcR zY29zqR|UfoS$X$}ly#5DE>t;v2Hdp*jE);3Hw%ps{$z61SkiT0ZIWyU@xXsjl@kMX81dVH8qkfB`RgpPLV7rf{;$) z?J8zzw8Ixh7t^**(hAJwc9B`cwjA2jMNBYmxQlcWR7#IrA{JD_)R&1Hf;3Ux#2Zzz zrQPIf%qIlvjA1$ho83#6x!~(FcPKfPx8tR7TuzLPx)7ZKW|h&nbDSr3Rnz`KqEb38 zf5Fg|1#ul4JjB3RIs3HG{V^w1FTL}S&_!%EI2zy-m@x3zyw zjC(ZnS|g{TF>=mD=8o#L^qL3Me)|p18FCtji6`*m%-K`+p%` zEcOOz415}o!FC=mc8u{D3mjlyHT|#y3Mf=^+nws*Ntyse*wVZE0iP0y;|=aLYXS?p zqH8fKFzSq=$5xQIbSTrEZd7r^`yYaX#qATPWi2X>MPg$ZTl+d^TK0723x0?CdNd3O z+T$T^{Nn|Wds3cQ;n<0y)*;~LnNwW93@Uxp1xp{$GOx3o4;nYcc!v#0Vw=u!8A#l& zx;vbTIbYzG_&>FJ2G#pvv3wY?8^hm-#*A=_aHg77k8rX4e_GFxJbp2L&GFat@7dh` z(Vo2Ql=RiQ4EO_Uo4&ak(2HsY;20`lH+T0QacR2jQ*a-2ad)e(9(&*CgGzS}cH0f~ zoAvT3Ycqem@2O?erF^eiZVZPzw{E4Ct_|WnyymXXN?DbvU$Oq34XGBpl9p^qOSMoG z#H$q9sVT|p*QKovS`$1wlosjv0P$kcPxQQrwgvGmG(*qF(2-Hzn_k!ReRO%Td { - return toSimpleSmartAccount({ - client: publicClient, - entryPoint, - owner: privateKeyToAccount(generatePrivateKey()) - }) -} diff --git a/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts b/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts index f6fddee7..fc20a018 100644 --- a/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts +++ b/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts @@ -40,7 +40,6 @@ import { pmSponsorUserOperationParamsSchema } from "./helpers/schema" import { maxBigInt } from "./helpers/utils" -import { getMockAccount } from "./mockAccount" const handleMethodV06 = async ( userOperation: UserOperation<"0.6">, @@ -67,14 +66,6 @@ const handleMethodV06 = async ( if (estimateGas) { try { const gasEstimates = await bundler.estimateUserOperationGas({ - account: await getMockAccount({ - publicClient, - entryPoint: { - address: entryPoint06Address, - abi: entryPoint06Abi, - version: "0.6" - } - }), ...op }) op = { @@ -177,14 +168,6 @@ const handleMethodV07 = async ( if (estimateGas) { try { const gasEstimates = await bundler.estimateUserOperationGas({ - account: await getMockAccount({ - publicClient, - entryPoint: { - address: entryPoint06Address, - abi: entryPoint06Abi, - version: "0.6" - } - }), ...op }) diff --git a/packages/permissionless-test/src/utils.ts b/packages/permissionless-test/src/utils.ts index 68f13259..fbf65b9b 100644 --- a/packages/permissionless-test/src/utils.ts +++ b/packages/permissionless-test/src/utils.ts @@ -138,7 +138,7 @@ export const getBundlerClient = ({ } }) - return createBundlerClient({ + return createBundlerClient({ client: getPublicClient(anvilRpc), account, paymaster, @@ -170,12 +170,12 @@ export const getSmartAccountClient = < }) : undefined - return createSmartAccountClient({ + return createSmartAccountClient({ client: getPublicClient(anvilRpc), chain: foundry, account, paymaster, - transport: http(altoRpc) + bundlerTransport: http(altoRpc) }) } diff --git a/packages/permissionless/actions/smartAccount/sendTransaction.test.ts b/packages/permissionless/actions/smartAccount/sendTransaction.test.ts index 21864cac..84f8e085 100644 --- a/packages/permissionless/actions/smartAccount/sendTransaction.test.ts +++ b/packages/permissionless/actions/smartAccount/sendTransaction.test.ts @@ -28,7 +28,6 @@ describe.each(getCoreSmartAccounts())( }) const transactionHash = await sendTransaction(smartClient, { - chain: foundry, to: zeroAddress, data: "0x", value: 0n @@ -49,7 +48,6 @@ describe.each(getCoreSmartAccounts())( // -- second transaction after deployment const transactionHash2 = await sendTransaction(smartClient, { - chain: foundry, to: zeroAddress, data: "0x", value: 0n @@ -80,7 +78,6 @@ describe.each(getCoreSmartAccounts())( }) const transactionHash = await sendTransaction(smartClient, { - chain: foundry, to: zeroAddress, data: "0x", value: 0n @@ -99,7 +96,6 @@ describe.each(getCoreSmartAccounts())( expect(receipt.status).toBe("success") const transactionHash2 = await sendTransaction(smartClient, { - chain: foundry, to: zeroAddress, data: "0x", value: 0n diff --git a/packages/permissionless/clients/createSmartAccountClient.ts b/packages/permissionless/clients/createSmartAccountClient.ts index c8506128..616ccaf7 100644 --- a/packages/permissionless/clients/createSmartAccountClient.ts +++ b/packages/permissionless/clients/createSmartAccountClient.ts @@ -69,9 +69,9 @@ export type SmartAccountClientConfig< | "name" | "pollingInterval" | "rpcSchema" - | "transport" > > & { + bundlerTransport: transport /** Client that points to an Execution RPC URL. */ client?: client | Client | undefined /** Paymaster configuration. */ @@ -130,7 +130,7 @@ export function createSmartAccountClient( name = "Bundler Client", paymaster, paymasterContext, - transport, + bundlerTransport, userOperation } = parameters @@ -139,7 +139,7 @@ export function createSmartAccountClient( chain: parameters.chain ?? client_?.chain, key, name, - transport, + transport: bundlerTransport, paymaster, paymasterContext, userOperation diff --git a/packages/permissionless/package.json b/packages/permissionless/package.json index d94ef980..97f57b99 100644 --- a/packages/permissionless/package.json +++ b/packages/permissionless/package.json @@ -71,6 +71,6 @@ } }, "peerDependencies": { - "viem": "^2.18.0" + "viem": "^2.20.0" } } From f1995b70e94afe5f9ef219d2936be78d8a4a4e3d Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Wed, 28 Aug 2024 16:39:37 +0100 Subject: [PATCH 26/51] Add ability to send calls from sendTransaction --- .../actions/smartAccount/sendTransaction.ts | 89 +++++++++++-------- .../actions/smartAccount/writeContract.ts | 8 +- .../clients/decorators/smartAccount.ts | 13 ++- 3 files changed, 68 insertions(+), 42 deletions(-) diff --git a/packages/permissionless/actions/smartAccount/sendTransaction.ts b/packages/permissionless/actions/smartAccount/sendTransaction.ts index f9267278..18b60809 100644 --- a/packages/permissionless/actions/smartAccount/sendTransaction.ts +++ b/packages/permissionless/actions/smartAccount/sendTransaction.ts @@ -6,6 +6,7 @@ import type { Transport } from "viem" import { + type SendUserOperationParameters, type SmartAccount, sendUserOperation, waitForUserOperationReceipt @@ -60,50 +61,62 @@ import { AccountNotFoundError } from "../../errors" * }) */ export async function sendTransaction< - TChain extends Chain | undefined, - TAccount extends SmartAccount | undefined, - TChainOverride extends Chain | undefined = Chain | undefined + account extends SmartAccount | undefined, + accountOverride extends SmartAccount | undefined = undefined, + chainOverride extends Chain | undefined = Chain | undefined >( - client: Client, - args: SendTransactionParameters + client: Client, + args: + | SendTransactionParameters + | SendUserOperationParameters ): Promise { - const { - account: account_ = client.account, - data, - maxFeePerGas, - maxPriorityFeePerGas, - to, - value, - nonce - } = args + let userOpHash: Hash - if (!account_) { - throw new AccountNotFoundError({ - docsPath: "/docs/actions/wallet/sendTransaction" - }) - } + if ("to" in args) { + const { + account: account_ = client.account, + data, + maxFeePerGas, + maxPriorityFeePerGas, + to, + value, + nonce + } = args - const account = parseAccount(account_) as SmartAccount + if (!account_) { + throw new AccountNotFoundError({ + docsPath: "/docs/actions/wallet/sendTransaction" + }) + } - if (!to) throw new Error("Missing to address") + const account = parseAccount(account_) as SmartAccount - const userOpHash = await getAction( - client, - sendUserOperation, - "sendUserOperation" - )({ - calls: [ - { - to, - value: value || BigInt(0), - data: data || "0x" - } - ], - account, - maxFeePerGas, - maxPriorityFeePerGas, - nonce: nonce ? BigInt(nonce) : undefined - }) + if (!to) throw new Error("Missing to address") + + userOpHash = await getAction( + client, + sendUserOperation, + "sendUserOperation" + )({ + calls: [ + { + to, + value: value || BigInt(0), + data: data || "0x" + } + ], + account, + maxFeePerGas, + maxPriorityFeePerGas, + nonce: nonce ? BigInt(nonce) : undefined + }) + } else { + userOpHash = await getAction( + client, + sendUserOperation, + "sendUserOperation" + )({ ...args } as SendUserOperationParameters) + } const userOperationReceipt = await getAction( client, diff --git a/packages/permissionless/actions/smartAccount/writeContract.ts b/packages/permissionless/actions/smartAccount/writeContract.ts index ca953fbd..ca6149bb 100644 --- a/packages/permissionless/actions/smartAccount/writeContract.ts +++ b/packages/permissionless/actions/smartAccount/writeContract.ts @@ -55,12 +55,16 @@ export async function writeContract< const hash = await getAction( client, - sendTransaction, + sendTransaction, "sendTransaction" )({ data: `${data}${dataSuffix ? dataSuffix.replace("0x", "") : ""}`, to: address, ...request - } as unknown as SendTransactionParameters) + } as unknown as SendTransactionParameters< + Chain | undefined, + TAccount, + undefined + >) return hash } diff --git a/packages/permissionless/clients/decorators/smartAccount.ts b/packages/permissionless/clients/decorators/smartAccount.ts index 397928dc..9e15fb1f 100644 --- a/packages/permissionless/clients/decorators/smartAccount.ts +++ b/packages/permissionless/clients/decorators/smartAccount.ts @@ -63,8 +63,17 @@ export type SmartAccountActions< * value: 1000000000000000000n, * }) */ - sendTransaction: ( - args: SendTransactionParameters + sendTransaction: < + TChainOverride extends Chain | undefined = undefined, + accountOverride extends SmartAccount | undefined = undefined + >( + args: Parameters< + typeof sendTransaction< + TSmartAccount, + accountOverride, + TChainOverride + > + >[1] ) => Promise /** * Calculates an Ethereum-specific signature in [EIP-191 format](https://eips.ethereum.org/EIPS/eip-191): `keccak256("\x19Ethereum Signed Message:\n" + len(message) + message))`. From e385392735253de7a05a5a4783e8540ca1925e37 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Wed, 28 Aug 2024 16:43:54 +0100 Subject: [PATCH 27/51] extend chain --- .../permissionless/actions/smartAccount/sendTransaction.ts | 5 +++-- packages/permissionless/clients/decorators/smartAccount.ts | 1 + 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/packages/permissionless/actions/smartAccount/sendTransaction.ts b/packages/permissionless/actions/smartAccount/sendTransaction.ts index 18b60809..b452d72d 100644 --- a/packages/permissionless/actions/smartAccount/sendTransaction.ts +++ b/packages/permissionless/actions/smartAccount/sendTransaction.ts @@ -62,12 +62,13 @@ import { AccountNotFoundError } from "../../errors" */ export async function sendTransaction< account extends SmartAccount | undefined, + chain extends Chain | undefined, accountOverride extends SmartAccount | undefined = undefined, chainOverride extends Chain | undefined = Chain | undefined >( - client: Client, + client: Client, args: - | SendTransactionParameters + | SendTransactionParameters | SendUserOperationParameters ): Promise { let userOpHash: Hash diff --git a/packages/permissionless/clients/decorators/smartAccount.ts b/packages/permissionless/clients/decorators/smartAccount.ts index 9e15fb1f..4ae5fddb 100644 --- a/packages/permissionless/clients/decorators/smartAccount.ts +++ b/packages/permissionless/clients/decorators/smartAccount.ts @@ -70,6 +70,7 @@ export type SmartAccountActions< args: Parameters< typeof sendTransaction< TSmartAccount, + TChain, accountOverride, TChainOverride > From 9d82c29d661b9ea096c839e3cfeab76ff5af97cf Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Wed, 28 Aug 2024 16:53:27 +0100 Subject: [PATCH 28/51] extend types --- .../actions/smartAccount/sendTransaction.ts | 5 +++-- .../permissionless/clients/decorators/smartAccount.ts | 8 +++++--- 2 files changed, 8 insertions(+), 5 deletions(-) diff --git a/packages/permissionless/actions/smartAccount/sendTransaction.ts b/packages/permissionless/actions/smartAccount/sendTransaction.ts index b452d72d..1aad613d 100644 --- a/packages/permissionless/actions/smartAccount/sendTransaction.ts +++ b/packages/permissionless/actions/smartAccount/sendTransaction.ts @@ -64,12 +64,13 @@ export async function sendTransaction< account extends SmartAccount | undefined, chain extends Chain | undefined, accountOverride extends SmartAccount | undefined = undefined, - chainOverride extends Chain | undefined = Chain | undefined + chainOverride extends Chain | undefined = Chain | undefined, + calls extends readonly unknown[] = readonly unknown[] >( client: Client, args: | SendTransactionParameters - | SendUserOperationParameters + | SendUserOperationParameters ): Promise { let userOpHash: Hash diff --git a/packages/permissionless/clients/decorators/smartAccount.ts b/packages/permissionless/clients/decorators/smartAccount.ts index 4ae5fddb..96708a37 100644 --- a/packages/permissionless/clients/decorators/smartAccount.ts +++ b/packages/permissionless/clients/decorators/smartAccount.ts @@ -65,14 +65,16 @@ export type SmartAccountActions< */ sendTransaction: < TChainOverride extends Chain | undefined = undefined, - accountOverride extends SmartAccount | undefined = undefined + accountOverride extends SmartAccount | undefined = undefined, + calls extends readonly unknown[] = readonly unknown[] >( args: Parameters< typeof sendTransaction< TSmartAccount, TChain, accountOverride, - TChainOverride + TChainOverride, + calls > >[1] ) => Promise @@ -318,7 +320,7 @@ export function smartAccountActions() { >( client: Client ): SmartAccountActions => ({ - sendTransaction: (args) => sendTransaction(client, args), + sendTransaction: (args) => sendTransaction(client, args as any), signMessage: (args) => signMessage(client, args), signTypedData: (args) => signTypedData(client, args), writeContract: (args) => writeContract(client, args) From f81b885ff24169bd41eb1092741e4d758d8ae6f3 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Wed, 28 Aug 2024 18:10:32 +0100 Subject: [PATCH 29/51] Deprecate sponsorUserOperation --- .../actions/pimlico/sponsorUserOperation.ts | 24 +------------------ .../clients/decorators/pimlico.ts | 3 +++ packages/permissionless/clients/pimlico.ts | 8 ++----- 3 files changed, 6 insertions(+), 29 deletions(-) diff --git a/packages/permissionless/actions/pimlico/sponsorUserOperation.ts b/packages/permissionless/actions/pimlico/sponsorUserOperation.ts index 5db48879..45d1545b 100644 --- a/packages/permissionless/actions/pimlico/sponsorUserOperation.ts +++ b/packages/permissionless/actions/pimlico/sponsorUserOperation.ts @@ -74,29 +74,7 @@ export type SponsorUserOperationReturnType< > /** - * Returns paymasterAndData & updated gas parameters required to sponsor a userOperation. - * - * - Docs: https://docs.pimlico.io/permissionless/reference/pimlico-paymaster-actions/sponsorUserOperation - * - * @param client {@link PimlicoBundlerClient} that you created using viem's createClient whose transport url is pointing to the Pimlico's bundler. - * @param args {@link PimlicoSponsorUserOperationParameters} UserOperation you want to sponsor & entryPoint. - * @returns paymasterAndData & updated gas parameters, see {@link SponsorUserOperationReturnType} - * - * - * @example - * import { createClient } from "viem" - * import { sponsorUserOperation } from "permissionless/actions/pimlico" - * - * const bundlerClient = createClient({ - * chain: goerli, - * transport: http("https://api.pimlico.io/v2/goerli/rpc?apikey=YOUR_API_KEY_HERE") - * }) - * - * await sponsorUserOperation(bundlerClient, { - * userOperation: userOperationWithDummySignature, - * entryPoint: entryPoint - * }}) - * + * @deprecated Use `getPaymasterData` instead */ export const sponsorUserOperation = async < entryPointAddress extends diff --git a/packages/permissionless/clients/decorators/pimlico.ts b/packages/permissionless/clients/decorators/pimlico.ts index 1715ff81..624e4314 100644 --- a/packages/permissionless/clients/decorators/pimlico.ts +++ b/packages/permissionless/clients/decorators/pimlico.ts @@ -106,6 +106,9 @@ export type PimlicoActions< Omit > ) => Promise + /** + * @deprecated Use `getPaymasterData` instead + */ sponsorUserOperation: ( args: Omit< PimlicoSponsorUserOperationParameters< diff --git a/packages/permissionless/clients/pimlico.ts b/packages/permissionless/clients/pimlico.ts index 51e82cdd..fe889fa7 100644 --- a/packages/permissionless/clients/pimlico.ts +++ b/packages/permissionless/clients/pimlico.ts @@ -10,10 +10,8 @@ import type { } from "viem" import { createClient } from "viem" import { - type BundlerActions, type PaymasterActions, type SmartAccount, - bundlerActions, type entryPoint06Address, type entryPoint07Address, paymasterActions @@ -46,9 +44,7 @@ export type PimlicoClient< rpcSchema extends RpcSchema ? [...BundlerRpcSchema, ...PimlicoRpcSchema, ...rpcSchema] : [...BundlerRpcSchema, ...PimlicoRpcSchema], - BundlerActions & - PaymasterActions & - PimlicoActions + PaymasterActions & PimlicoActions > > @@ -144,7 +140,7 @@ export function createPimlicoClient( name, type: "pimlicoBundlerClient" }) - return client.extend(bundlerActions).extend(paymasterActions).extend( + return client.extend(paymasterActions).extend( pimlicoActions({ entryPoint }) From 082f5b8cf67c486b2e7f29697d361f1322f24a5c Mon Sep 17 00:00:00 2001 From: mouseless <97399882+mouseless-eth@users.noreply.github.com> Date: Thu, 29 Aug 2024 15:02:35 +0100 Subject: [PATCH 30/51] fix v07 prefund calculations --- .../utils/getRequiredPrefund.test.ts | 47 +++++++++++++++++++ .../utils/getRequiredPrefund.ts | 12 ++--- 2 files changed, 51 insertions(+), 8 deletions(-) diff --git a/packages/permissionless/utils/getRequiredPrefund.test.ts b/packages/permissionless/utils/getRequiredPrefund.test.ts index 2ce46daa..eafd923a 100644 --- a/packages/permissionless/utils/getRequiredPrefund.test.ts +++ b/packages/permissionless/utils/getRequiredPrefund.test.ts @@ -64,4 +64,51 @@ describe("getRequiredPrefund", () => { expect(result).toBe(expectedResult) }) }) + describe("v0.7 UserOperation", () => { + test("should calculate the required prefund without paymater gasLimits", () => { + const userOperation = { + callGasLimit: BigInt(1000), + verificationGasLimit: BigInt(2000), + preVerificationGas: BigInt(500), + paymasterVerificationGasLimit: undefined, + paymasterPostOpGasLimit: undefined, + paymaster: undefined, + paymasterData: undefined, + maxFeePerGas: BigInt(10) + } + const result = getRequiredPrefund({ + userOperation: userOperation as UserOperation<"0.7">, + entryPointVersion: "0.7" + }) + const expectedGas = + BigInt(1000) + BigInt(2000) * BigInt(1) + BigInt(500) + const expectedResult = expectedGas * BigInt(10) + expect(result).toBe(expectedResult) + }) + + test("should calculate the required prefund with paymaster gasLimits", () => { + const userOperation = { + callGasLimit: BigInt(1000), + verificationGasLimit: BigInt(2000), + preVerificationGas: BigInt(500), + paymasterVerificationGasLimit: BigInt(20), + paymasterPostOpGasLimit: BigInt(30), + paymaster: "0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa", + paymasterData: "0x1234", + maxFeePerGas: BigInt(10) + } + const result = getRequiredPrefund({ + userOperation: userOperation as UserOperation<"0.7">, + entryPointVersion: "0.7" + }) + const expectedGas = + BigInt(1000) + + BigInt(2000) + + BigInt(500) + + BigInt(20) + + BigInt(30) + const expectedResult = expectedGas * BigInt(10) + expect(result).toBe(expectedResult) + }) + }) }) diff --git a/packages/permissionless/utils/getRequiredPrefund.ts b/packages/permissionless/utils/getRequiredPrefund.ts index 84b1bfd3..b90162a4 100644 --- a/packages/permissionless/utils/getRequiredPrefund.ts +++ b/packages/permissionless/utils/getRequiredPrefund.ts @@ -42,17 +42,13 @@ export const getRequiredPrefund = ({ } const userOperationV07 = userOperation as UserOperation<"0.7"> - const multiplier = userOperationV07.paymaster ? BigInt(3) : BigInt(1) - - const verificationGasLimit = - userOperationV07.verificationGasLimit + - (userOperationV07.paymasterPostOpGasLimit || BigInt(0)) + - (userOperationV07.paymasterVerificationGasLimit || BigInt(0)) const requiredGas = + userOperationV07.verificationGasLimit + userOperationV07.callGasLimit + - verificationGasLimit * multiplier + + (userOperationV07.paymasterVerificationGasLimit || 0n) + + (userOperationV07.paymasterPostOpGasLimit || 0n) + userOperationV07.preVerificationGas - return BigInt(requiredGas) * BigInt(userOperationV07.maxFeePerGas) + return requiredGas * userOperationV07.maxFeePerGas } From 3c5f61781dff820256dd121db9bf0f49dc3910d2 Mon Sep 17 00:00:00 2001 From: mouseless <97399882+mouseless-eth@users.noreply.github.com> Date: Thu, 29 Aug 2024 15:12:08 +0100 Subject: [PATCH 31/51] fix failing tests --- .../utils/getRequiredPrefund.test.ts | 24 ------------------- 1 file changed, 24 deletions(-) diff --git a/packages/permissionless/utils/getRequiredPrefund.test.ts b/packages/permissionless/utils/getRequiredPrefund.test.ts index eafd923a..a076f7d1 100644 --- a/packages/permissionless/utils/getRequiredPrefund.test.ts +++ b/packages/permissionless/utils/getRequiredPrefund.test.ts @@ -40,29 +40,6 @@ describe("getRequiredPrefund", () => { const expectedResult = expectedGas * BigInt(10) expect(result).toBe(expectedResult) }) - - test("should calculate the required prefund with paymaster", () => { - const userOperation = { - callGasLimit: BigInt(1000), - verificationGasLimit: BigInt(2000), - preVerificationGas: BigInt(500), - maxFeePerGas: BigInt(10), - paymaster: "0xPaymasterAddress", - paymasterPostOpGasLimit: BigInt(100), - paymasterVerificationGasLimit: BigInt(200) - } - const result = getRequiredPrefund({ - userOperation: userOperation as UserOperation<"0.7">, - entryPointVersion: "0.7" - }) - const multiplier = BigInt(3) - const verificationGasLimit = - BigInt(2000) + BigInt(100) + BigInt(200) - const expectedGas = - BigInt(1000) + verificationGasLimit * multiplier + BigInt(500) - const expectedResult = expectedGas * BigInt(10) - expect(result).toBe(expectedResult) - }) }) describe("v0.7 UserOperation", () => { test("should calculate the required prefund without paymater gasLimits", () => { @@ -94,7 +71,6 @@ describe("getRequiredPrefund", () => { paymasterVerificationGasLimit: BigInt(20), paymasterPostOpGasLimit: BigInt(30), paymaster: "0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa", - paymasterData: "0x1234", maxFeePerGas: BigInt(10) } const result = getRequiredPrefund({ From c99b9179a9bd31c4c74fe1c88f17bfd3aae170a8 Mon Sep 17 00:00:00 2001 From: mouseless <97399882+mouseless-eth@users.noreply.github.com> Date: Fri, 30 Aug 2024 12:34:26 +0100 Subject: [PATCH 32/51] add getTokenQuotes action --- .../mock-paymaster/helpers/schema.ts | 8 +++ .../mock-aa-infra/mock-paymaster/relay.ts | 50 ++++++++++++++-- packages/permissionless/actions/pimlico.ts | 12 +++- .../actions/pimlico/getTokenQuotes.test.ts | 36 +++++++++++ .../actions/pimlico/getTokenQuotes.ts | 60 +++++++++++++++++++ .../clients/decorators/pimlico.ts | 13 +++- packages/permissionless/types/pimlico.ts | 14 +++++ 7 files changed, 184 insertions(+), 9 deletions(-) create mode 100644 packages/permissionless/actions/pimlico/getTokenQuotes.test.ts create mode 100644 packages/permissionless/actions/pimlico/getTokenQuotes.ts diff --git a/packages/permissionless-test/mock-aa-infra/mock-paymaster/helpers/schema.ts b/packages/permissionless-test/mock-aa-infra/mock-paymaster/helpers/schema.ts index 5c294bc6..6d728920 100644 --- a/packages/permissionless-test/mock-aa-infra/mock-paymaster/helpers/schema.ts +++ b/packages/permissionless-test/mock-aa-infra/mock-paymaster/helpers/schema.ts @@ -259,6 +259,14 @@ export const pmGetPaymasterStubDataParamsSchema = z return [val[0], val[1], val[2], val[3] ?? null] as const }) +export const pimlicoGetTokenQuotesSchema = z.tuple([ + z.object({ + tokens: z.array(addressSchema) + }), + addressSchema, // entryPoint + hexNumberSchema +]) + export type UserOperationV7 = zodInfer export type UserOperationV6 = zodInfer export type JsonRpcSchema = zodInfer diff --git a/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts b/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts index fc20a018..3a592e60 100644 --- a/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts +++ b/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts @@ -1,8 +1,8 @@ import util from "node:util" import type { FastifyReply, FastifyRequest } from "fastify" import { - type Account, BaseError, + type Account, type Chain, type Client, type GetContractReturnType, @@ -11,14 +11,15 @@ import { type RpcRequestError, type Transport, type WalletClient, + type Address, concat, encodeAbiParameters, - toHex + toHex, + getAddress } from "viem" import { type BundlerClient, type UserOperation, - entryPoint06Abi, entryPoint06Address, entryPoint07Address } from "viem/account-abstraction" @@ -37,7 +38,8 @@ import { jsonRpcSchema, pmGetPaymasterData, pmGetPaymasterStubDataParamsSchema, - pmSponsorUserOperationParamsSchema + pmSponsorUserOperationParamsSchema, + pimlicoGetTokenQuotesSchema } from "./helpers/schema" import { maxBigInt } from "./helpers/utils" @@ -260,7 +262,7 @@ const handleMethod = async ( verifyingPaymasterV06: GetContractReturnType< typeof VERIFYING_PAYMASTER_V06_ABI >, - publicClient: PublicClient, + publicClient: PublicClient, walletClient: WalletClient, parsedBody: JsonRpcSchema ) => { @@ -405,6 +407,42 @@ const handleMethod = async ( ] } + if (parsedBody.method === "pimlico_getTokenQuotes") { + const params = pimlicoGetTokenQuotesSchema.safeParse(parsedBody.params) + + if (!params.success) { + throw new RpcError( + fromZodError(params.error).message, + ValidationErrors.InvalidFields + ) + } + + const [context, entryPoint] = params.data + const { tokens } = context + + const quotes = { + [getAddress("0xffffffffffffffffffffffffffffffffffffffff")]: { + exchangeRate: "0x5cc717fbb3450c0000", + postOpGas: "0xc350" + } + } + + let paymaster: Address + if (entryPoint === entryPoint07Address) { + paymaster = verifyingPaymasterV07.address + } else { + paymaster = verifyingPaymasterV06.address + } + + return tokens + .filter((t) => quotes[t]) // Filter out unrecongized tokens + .map((token) => ({ + ...quotes[token], + paymaster, + token + })) + } + throw new RpcError( `Attempted to call an unknown method ${parsedBody.method}`, ValidationErrors.InvalidFields @@ -419,7 +457,7 @@ export const createRpcHandler = ( verifyingPaymasterV06: GetContractReturnType< typeof VERIFYING_PAYMASTER_V06_ABI >, - publicClient: PublicClient, + publicClient: PublicClient, walletClient: WalletClient ) => { return async (request: FastifyRequest, _reply: FastifyReply) => { diff --git a/packages/permissionless/actions/pimlico.ts b/packages/permissionless/actions/pimlico.ts index c067c254..f158dfbb 100644 --- a/packages/permissionless/actions/pimlico.ts +++ b/packages/permissionless/actions/pimlico.ts @@ -16,6 +16,11 @@ import { type SponsorUserOperationReturnType, sponsorUserOperation } from "./pimlico/sponsorUserOperation" +import { + type GetTokenQuotesParameters, + type GetTokenQuotesReturnType, + getTokenQuotes +} from "./pimlico/getTokenQuotes" import type { PimlicoActions } from "../clients/decorators/pimlico" import { pimlicoActions } from "../clients/decorators/pimlico" @@ -35,7 +40,9 @@ export type { SendCompressedUserOperationParameters, SponsorUserOperationReturnType, ValidateSponsorshipPolicies, - ValidateSponsorshipPoliciesParameters + ValidateSponsorshipPoliciesParameters, + GetTokenQuotesParameters, + GetTokenQuotesReturnType } export { @@ -44,5 +51,6 @@ export { pimlicoActions, sendCompressedUserOperation, sponsorUserOperation, - validateSponsorshipPolicies + validateSponsorshipPolicies, + getTokenQuotes } diff --git a/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts b/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts new file mode 100644 index 00000000..ef943ddd --- /dev/null +++ b/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts @@ -0,0 +1,36 @@ +import { describe, expect } from "vitest" +import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" +import { getPimlicoClient } from "../../../permissionless-test/src/utils" +import { getTokenQuotes } from "./getTokenQuotes" +import { entryPoint07Address } from "viem/_types/account-abstraction" +import { foundry } from "viem/chains" +import { getAddress, isAddress } from "viem" + +describe("getTokenQuotes", () => { + testWithRpc("getTokenQuotes", async ({ rpc }) => { + const pimlicoBundlerClient = getPimlicoClient({ + entryPointVersion: "0.7", + altoRpc: rpc.altoRpc + }) + + const token = getAddress("0xffffffffffffffffffffffffffffffffffffffff") + + const quotes = await getTokenQuotes(pimlicoBundlerClient, { + tokens: [token], + entryPointAddress: entryPoint07Address, + chainId: BigInt(foundry.id) + }) + + expect(quotes).toBeTruthy() + expect(Array.isArray(quotes)).toBe(true) + expect(quotes[0].token).toBeTruthy() + expect(isAddress(quotes[0].token)) + expect(quotes[0].token).toEqual(token) + expect(quotes[0].paymaster).toBeTruthy() + expect(isAddress(quotes[0].paymaster)) + expect(quotes[0].exchangeRate).toBeTruthy() + expect(quotes[0].exchangeRate).toBeGreaterThan(0n) + expect(quotes[0].postOpGas).toBeTruthy() + expect(quotes[0].postOpGas).toBeGreaterThan(0n) + }) +}) diff --git a/packages/permissionless/actions/pimlico/getTokenQuotes.ts b/packages/permissionless/actions/pimlico/getTokenQuotes.ts new file mode 100644 index 00000000..bb67d482 --- /dev/null +++ b/packages/permissionless/actions/pimlico/getTokenQuotes.ts @@ -0,0 +1,60 @@ +import { + hexToBigInt, + numberToHex, + type Account, + type Address, + type Chain, + type Client, + type Transport +} from "viem" +import type { PimlicoRpcSchema } from "../../types/pimlico" + +export type GetTokenQuotesParameters = { + tokens: Address[] + entryPointAddress: Address + chainId: bigint +} + +export type GetTokenQuotesReturnType = { + paymaster: Address + token: Address + postOpGas: bigint + exchangeRate: bigint +}[] + +/** + * Returns all related fields to calculate the potential cost of a userOperation in ERC-20 tokens. + * + * - Docs: https://docs.pimlico.io/permissionless/reference/pimlico-bundler-actions/getTokenQuotes + * + * @param client that you created using viem's createClient whose transport url is pointing to the Pimlico's bundler. + * @returns slow, standard & fast values for maxFeePerGas & maxPriorityFeePerGas + * @returns quotes, see {@link GetTokenQuotesReturnType} + * + */ +export const getTokenQuotes = async ( + client: Client< + Transport, + Chain | undefined, + Account | undefined, + PimlicoRpcSchema + >, + args: GetTokenQuotesParameters +): Promise => { + const res = await client.request({ + method: "pimlico_getTokenQuotes", + params: [ + { tokens: args.tokens }, + args.entryPointAddress, + numberToHex(args.chainId) + ] + }) + + return { + ...res.quotes.map((quote) => ({ + ...quote, + postOpGas: hexToBigInt(quote.postOpGas), + exchangeRate: hexToBigInt(quote.exchangeRate) + })) + } +} diff --git a/packages/permissionless/clients/decorators/pimlico.ts b/packages/permissionless/clients/decorators/pimlico.ts index 624e4314..ad242672 100644 --- a/packages/permissionless/clients/decorators/pimlico.ts +++ b/packages/permissionless/clients/decorators/pimlico.ts @@ -7,8 +7,11 @@ import { type SendCompressedUserOperationParameters, type ValidateSponsorshipPolicies, type ValidateSponsorshipPoliciesParameters, + type GetTokenQuotesParameters, + type GetTokenQuotesReturnType, sendCompressedUserOperation, - validateSponsorshipPolicies + validateSponsorshipPolicies, + getTokenQuotes } from "../../actions/pimlico" import { type GetUserOperationGasPriceReturnType, @@ -129,6 +132,9 @@ export type PimlicoActions< > > ) => Promise[]> + getTokenQuotes: ( + args: Prettify> + ) => Promise> } export const pimlicoActions = @@ -163,5 +169,10 @@ export const pimlicoActions = validateSponsorshipPolicies(client, { ...args, entryPointAddress: entryPoint.address + }), + getTokenQuotes: async (args) => + getTokenQuotes(client, { + ...args, + entryPointAddress: entryPoint.address }) }) diff --git a/packages/permissionless/types/pimlico.ts b/packages/permissionless/types/pimlico.ts index 08b2ccfb..ad016c52 100644 --- a/packages/permissionless/types/pimlico.ts +++ b/packages/permissionless/types/pimlico.ts @@ -32,6 +32,15 @@ export type PimlicoUserOperationStatus = { transactionHash: Hash | null } +type GetTokenQuotesWithBigIntAsHex = { + quotes: { + paymaster: Address + token: Address + postOpGas: Hex + exchangeRate: Hex + }[] +} + export type PimlicoRpcSchema< entryPointAddress extends | typeof entryPoint06Address @@ -123,5 +132,10 @@ export type PimlicoRpcSchema< description: string | null } }[] + }, + { + Method: "pimlico_getTokenQuotes" + Parameters: [{ tokens: Address[] }, entryPoint: Address, chainId: Hex] + ReturnType: GetTokenQuotesWithBigIntAsHex } ] From fd0d4c932e58af131425e7b2ab6a5e6343a88c43 Mon Sep 17 00:00:00 2001 From: mouseless <97399882+mouseless-eth@users.noreply.github.com> Date: Fri, 30 Aug 2024 12:34:46 +0100 Subject: [PATCH 33/51] lint --- .../mock-aa-infra/mock-paymaster/relay.ts | 12 ++++++------ packages/permissionless/actions/pimlico.ts | 10 +++++----- .../actions/pimlico/getTokenQuotes.test.ts | 6 +++--- .../permissionless/actions/pimlico/getTokenQuotes.ts | 6 +++--- .../permissionless/clients/decorators/pimlico.ts | 8 ++++---- 5 files changed, 21 insertions(+), 21 deletions(-) diff --git a/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts b/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts index 3a592e60..5043f8b4 100644 --- a/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts +++ b/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts @@ -1,8 +1,9 @@ import util from "node:util" import type { FastifyReply, FastifyRequest } from "fastify" import { - BaseError, type Account, + type Address, + BaseError, type Chain, type Client, type GetContractReturnType, @@ -11,11 +12,10 @@ import { type RpcRequestError, type Transport, type WalletClient, - type Address, concat, encodeAbiParameters, - toHex, - getAddress + getAddress, + toHex } from "viem" import { type BundlerClient, @@ -36,10 +36,10 @@ import { RpcError, ValidationErrors, jsonRpcSchema, + pimlicoGetTokenQuotesSchema, pmGetPaymasterData, pmGetPaymasterStubDataParamsSchema, - pmSponsorUserOperationParamsSchema, - pimlicoGetTokenQuotesSchema + pmSponsorUserOperationParamsSchema } from "./helpers/schema" import { maxBigInt } from "./helpers/utils" diff --git a/packages/permissionless/actions/pimlico.ts b/packages/permissionless/actions/pimlico.ts index f158dfbb..908aba0d 100644 --- a/packages/permissionless/actions/pimlico.ts +++ b/packages/permissionless/actions/pimlico.ts @@ -1,3 +1,8 @@ +import { + type GetTokenQuotesParameters, + type GetTokenQuotesReturnType, + getTokenQuotes +} from "./pimlico/getTokenQuotes" import { type GetUserOperationGasPriceReturnType, getUserOperationGasPrice @@ -16,11 +21,6 @@ import { type SponsorUserOperationReturnType, sponsorUserOperation } from "./pimlico/sponsorUserOperation" -import { - type GetTokenQuotesParameters, - type GetTokenQuotesReturnType, - getTokenQuotes -} from "./pimlico/getTokenQuotes" import type { PimlicoActions } from "../clients/decorators/pimlico" import { pimlicoActions } from "../clients/decorators/pimlico" diff --git a/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts b/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts index ef943ddd..5e8b2412 100644 --- a/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts +++ b/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts @@ -1,10 +1,10 @@ +import { getAddress, isAddress } from "viem" +import { entryPoint07Address } from "viem/_types/account-abstraction" +import { foundry } from "viem/chains" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" import { getPimlicoClient } from "../../../permissionless-test/src/utils" import { getTokenQuotes } from "./getTokenQuotes" -import { entryPoint07Address } from "viem/_types/account-abstraction" -import { foundry } from "viem/chains" -import { getAddress, isAddress } from "viem" describe("getTokenQuotes", () => { testWithRpc("getTokenQuotes", async ({ rpc }) => { diff --git a/packages/permissionless/actions/pimlico/getTokenQuotes.ts b/packages/permissionless/actions/pimlico/getTokenQuotes.ts index bb67d482..67d85e72 100644 --- a/packages/permissionless/actions/pimlico/getTokenQuotes.ts +++ b/packages/permissionless/actions/pimlico/getTokenQuotes.ts @@ -1,11 +1,11 @@ import { - hexToBigInt, - numberToHex, type Account, type Address, type Chain, type Client, - type Transport + type Transport, + hexToBigInt, + numberToHex } from "viem" import type { PimlicoRpcSchema } from "../../types/pimlico" diff --git a/packages/permissionless/clients/decorators/pimlico.ts b/packages/permissionless/clients/decorators/pimlico.ts index ad242672..3730cdb7 100644 --- a/packages/permissionless/clients/decorators/pimlico.ts +++ b/packages/permissionless/clients/decorators/pimlico.ts @@ -4,14 +4,14 @@ import type { entryPoint07Address } from "viem/account-abstraction" import { + type GetTokenQuotesParameters, + type GetTokenQuotesReturnType, type SendCompressedUserOperationParameters, type ValidateSponsorshipPolicies, type ValidateSponsorshipPoliciesParameters, - type GetTokenQuotesParameters, - type GetTokenQuotesReturnType, + getTokenQuotes, sendCompressedUserOperation, - validateSponsorshipPolicies, - getTokenQuotes + validateSponsorshipPolicies } from "../../actions/pimlico" import { type GetUserOperationGasPriceReturnType, From 8fcca08e7e1fcc9c9ff7ed1701632dbd5f65eba0 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Fri, 30 Aug 2024 12:36:37 +0100 Subject: [PATCH 34/51] Add toOwner --- packages/permissionless/clients/pimlico.ts | 8 +- packages/permissionless/utils/index.ts | 6 +- .../utils/providerToSmartAccountSigner.ts | 34 --------- packages/permissionless/utils/toOwner.ts | 73 +++++++++++++++++++ .../utils/walletClientToSmartAccountSigner.ts | 48 ------------ 5 files changed, 81 insertions(+), 88 deletions(-) delete mode 100644 packages/permissionless/utils/providerToSmartAccountSigner.ts create mode 100644 packages/permissionless/utils/toOwner.ts delete mode 100644 packages/permissionless/utils/walletClientToSmartAccountSigner.ts diff --git a/packages/permissionless/clients/pimlico.ts b/packages/permissionless/clients/pimlico.ts index fe889fa7..51e82cdd 100644 --- a/packages/permissionless/clients/pimlico.ts +++ b/packages/permissionless/clients/pimlico.ts @@ -10,8 +10,10 @@ import type { } from "viem" import { createClient } from "viem" import { + type BundlerActions, type PaymasterActions, type SmartAccount, + bundlerActions, type entryPoint06Address, type entryPoint07Address, paymasterActions @@ -44,7 +46,9 @@ export type PimlicoClient< rpcSchema extends RpcSchema ? [...BundlerRpcSchema, ...PimlicoRpcSchema, ...rpcSchema] : [...BundlerRpcSchema, ...PimlicoRpcSchema], - PaymasterActions & PimlicoActions + BundlerActions & + PaymasterActions & + PimlicoActions > > @@ -140,7 +144,7 @@ export function createPimlicoClient( name, type: "pimlicoBundlerClient" }) - return client.extend(paymasterActions).extend( + return client.extend(bundlerActions).extend(paymasterActions).extend( pimlicoActions({ entryPoint }) diff --git a/packages/permissionless/utils/index.ts b/packages/permissionless/utils/index.ts index 2b3f0d54..c65af699 100644 --- a/packages/permissionless/utils/index.ts +++ b/packages/permissionless/utils/index.ts @@ -5,8 +5,7 @@ import { getRequiredPrefund } from "./getRequiredPrefund" import { isSmartAccountDeployed } from "./isSmartAccountDeployed" -import { providerToSmartAccountSigner } from "./providerToSmartAccountSigner" -import { walletClientToSmartAccountSigner } from "./walletClientToSmartAccountSigner" +import { toOwner } from "./toOwner" import { decodeNonce } from "./decodeNonce" import { encodeNonce } from "./encodeNonce" @@ -17,10 +16,9 @@ export { transactionReceiptStatus, deepHexlify, getRequiredPrefund, - walletClientToSmartAccountSigner, + toOwner, type GetRequiredPrefundReturnType, isSmartAccountDeployed, - providerToSmartAccountSigner, getAddressFromInitCodeOrPaymasterAndData, getPackedUserOperation, encodeNonce, diff --git a/packages/permissionless/utils/providerToSmartAccountSigner.ts b/packages/permissionless/utils/providerToSmartAccountSigner.ts deleted file mode 100644 index 21584b28..00000000 --- a/packages/permissionless/utils/providerToSmartAccountSigner.ts +++ /dev/null @@ -1,34 +0,0 @@ -import { - type EIP1193Provider, - type Hex, - createWalletClient, - custom -} from "viem" -import { walletClientToSmartAccountSigner } from "./walletClientToSmartAccountSigner" - -export const providerToSmartAccountSigner = async ( - provider: EIP1193Provider, - params?: { - signerAddress: Hex - } -) => { - let account: Hex - if (!params) { - try { - ;[account] = await provider.request({ - method: "eth_requestAccounts" - }) - } catch { - ;[account] = await provider.request({ - method: "eth_accounts" - }) - } - } else { - account = params.signerAddress - } - const walletClient = createWalletClient({ - account: account as Hex, - transport: custom(provider) - }) - return walletClientToSmartAccountSigner(walletClient) -} diff --git a/packages/permissionless/utils/toOwner.ts b/packages/permissionless/utils/toOwner.ts new file mode 100644 index 00000000..4346adbb --- /dev/null +++ b/packages/permissionless/utils/toOwner.ts @@ -0,0 +1,73 @@ +import { + type Account, + type Address, + type Chain, + type EIP1193Provider, + type EIP1193RequestFn, + type EIP1474Methods, + type LocalAccount, + type OneOf, + type Transport, + type WalletClient, + createWalletClient, + custom +} from "viem" +import { toAccount } from "viem/accounts" + +import { signTypedData } from "viem/actions" +import { getAction } from "viem/utils" + +export async function toOwner({ + owner, + address +}: { + owner: OneOf< + EIP1193Provider | WalletClient + > + address?: Address +}): Promise { + if ("request" in owner) { + if (!address) { + try { + ;[address] = await ( + owner.request as EIP1193RequestFn + )({ + method: "eth_requestAccounts" + }) + } catch { + ;[address] = await ( + owner.request as EIP1193RequestFn + )({ + method: "eth_accounts" + }) + } + } + if (!address) { + // For TS to be happy + throw new Error("address is required") + } + owner = createWalletClient({ + account: address, + transport: custom(owner) + }) + } + + return toAccount({ + address: owner.account.address, + async signMessage({ message }) { + return owner.signMessage({ message }) + }, + async signTypedData(typedData) { + return getAction( + owner, + signTypedData, + "signTypedData" + )(typedData as any) + }, + async signTransaction(_) { + throw new Error( + "Smart account signer doesn't need to sign transactions" + ) + } + }) +} diff --git a/packages/permissionless/utils/walletClientToSmartAccountSigner.ts b/packages/permissionless/utils/walletClientToSmartAccountSigner.ts deleted file mode 100644 index fd37c59c..00000000 --- a/packages/permissionless/utils/walletClientToSmartAccountSigner.ts +++ /dev/null @@ -1,48 +0,0 @@ -import type { - Account, - Chain, - Hex, - LocalAccount, - SignableMessage, - Transport, - TypedData, - TypedDataDefinition, - WalletClient -} from "viem" - -import { signTypedData } from "viem/actions" - -export function walletClientToSmartAccountSigner< - TChain extends Chain | undefined = Chain | undefined ->(walletClient: WalletClient): LocalAccount { - return { - address: walletClient.account.address, - type: "local", - source: "custom", - publicKey: walletClient.account.address, - signMessage: async ({ - message - }: { message: SignableMessage }): Promise => { - return walletClient.signMessage({ message }) - }, - signTransaction: async (_): Promise => { - throw new Error( - "Smart account signer doesn't need to sign transactions" - ) - }, - async signTypedData< - const TTypedData extends TypedData | Record, - TPrimaryType extends - | keyof TTypedData - | "EIP712Domain" = keyof TTypedData - >(typedData: TypedDataDefinition) { - return signTypedData( - walletClient, - { - account: walletClient.account, - ...typedData - } - ) - } - } -} From 296165c47c506de75320e23a22bf9ca00ec935d4 Mon Sep 17 00:00:00 2001 From: mouseless <97399882+mouseless-eth@users.noreply.github.com> Date: Fri, 30 Aug 2024 12:52:16 +0100 Subject: [PATCH 35/51] fix failing tests --- .../mock-aa-infra/mock-paymaster/relay.ts | 16 +++++++++------- .../actions/pimlico/getTokenQuotes.test.ts | 4 ++-- .../actions/pimlico/getTokenQuotes.ts | 12 +++++------- 3 files changed, 16 insertions(+), 16 deletions(-) diff --git a/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts b/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts index 5043f8b4..7fa46e95 100644 --- a/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts +++ b/packages/permissionless-test/mock-aa-infra/mock-paymaster/relay.ts @@ -434,13 +434,15 @@ const handleMethod = async ( paymaster = verifyingPaymasterV06.address } - return tokens - .filter((t) => quotes[t]) // Filter out unrecongized tokens - .map((token) => ({ - ...quotes[token], - paymaster, - token - })) + return { + quotes: tokens + .filter((t) => quotes[t]) // Filter out unrecongized tokens + .map((token) => ({ + ...quotes[token], + paymaster, + token + })) + } } throw new RpcError( diff --git a/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts b/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts index 5e8b2412..6029b8b5 100644 --- a/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts +++ b/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts @@ -1,5 +1,5 @@ import { getAddress, isAddress } from "viem" -import { entryPoint07Address } from "viem/_types/account-abstraction" +import { entryPoint07Address } from "viem/account-abstraction" import { foundry } from "viem/chains" import { describe, expect } from "vitest" import { testWithRpc } from "../../../permissionless-test/src/testWithRpc" @@ -10,7 +10,7 @@ describe("getTokenQuotes", () => { testWithRpc("getTokenQuotes", async ({ rpc }) => { const pimlicoBundlerClient = getPimlicoClient({ entryPointVersion: "0.7", - altoRpc: rpc.altoRpc + altoRpc: rpc.paymasterRpc }) const token = getAddress("0xffffffffffffffffffffffffffffffffffffffff") diff --git a/packages/permissionless/actions/pimlico/getTokenQuotes.ts b/packages/permissionless/actions/pimlico/getTokenQuotes.ts index 67d85e72..c3d1169d 100644 --- a/packages/permissionless/actions/pimlico/getTokenQuotes.ts +++ b/packages/permissionless/actions/pimlico/getTokenQuotes.ts @@ -50,11 +50,9 @@ export const getTokenQuotes = async ( ] }) - return { - ...res.quotes.map((quote) => ({ - ...quote, - postOpGas: hexToBigInt(quote.postOpGas), - exchangeRate: hexToBigInt(quote.exchangeRate) - })) - } + return res.quotes.map((quote) => ({ + ...quote, + postOpGas: hexToBigInt(quote.postOpGas), + exchangeRate: hexToBigInt(quote.exchangeRate) + })) } From 1d7827f0d67a761eb99c0adf4b36fcce4919856d Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Fri, 30 Aug 2024 15:22:56 +0100 Subject: [PATCH 36/51] Allow multiple type of owner(s) --- packages/permissionless-test/src/utils.ts | 4 +-- .../biconomy/toBiconomySmartAccount.ts | 28 +++++++++++---- .../kernel/toEcdsaKernelSmartAccount.ts | 31 +++++++++++++---- .../accounts/light/toLightSmartAccount.ts | 23 ++++++++++--- .../accounts/safe/toSafeSmartAccount.ts | 34 +++++++++++++------ .../accounts/simple/toSimpleSmartAccount.ts | 19 +++++++++-- .../accounts/trust/toTrustSmartAccount.ts | 23 ++++++++++--- packages/permissionless/package.json | 2 +- packages/permissionless/utils/toOwner.ts | 30 ++++++++++++---- 9 files changed, 149 insertions(+), 45 deletions(-) diff --git a/packages/permissionless-test/src/utils.ts b/packages/permissionless-test/src/utils.ts index fbf65b9b..c9c46707 100644 --- a/packages/permissionless-test/src/utils.ts +++ b/packages/permissionless-test/src/utils.ts @@ -348,7 +348,7 @@ export const getKernelEcdsaClient = async < : entryPoint07Address, version: entryPoint.version === "0.6" ? "0.6" : "0.7" }, - owner: privateKeyToAccount(generatePrivateKey()), + owners: [privateKeyToAccount(generatePrivateKey())], version }) } @@ -371,7 +371,7 @@ export const getSafeClient = async ({ : entryPoint07Address, version: entryPoint.version === "0.6" ? "0.6" : "0.7" }, - owner: privateKeyToAccount(generatePrivateKey()), + owners: [privateKeyToAccount(generatePrivateKey())], version: "1.4.1", saltNonce: 420n, safe4337ModuleAddress: erc7579 diff --git a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts index 1308f9f6..13ac11a4 100644 --- a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts +++ b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts @@ -1,4 +1,13 @@ -import type { Assign, Prettify } from "viem" +import type { + Account, + Assign, + Chain, + EIP1193Provider, + OneOf, + Prettify, + Transport, + WalletClient +} from "viem" import { type Address, type Client, @@ -19,6 +28,7 @@ import { import { signMessage } from "viem/actions" import { getAccountNonce } from "../../actions/public/getAccountNonce" import { getSenderAddress } from "../../actions/public/getSenderAddress" +import { toOwner } from "../../utils/toOwner" import { BiconomyAbi, FactoryAbi } from "./abi/BiconomySmartAccountAbi" /** @@ -72,7 +82,11 @@ const getAccountInitCode = async ({ export type ToBiconomySmartAccountParameters = Prettify<{ client: Client - owner: LocalAccount + owner: OneOf< + | EIP1193Provider + | WalletClient + | LocalAccount + > address?: Address | undefined entryPoint?: { address: Address @@ -108,6 +122,8 @@ export async function toBiconomySmartAccount( ): Promise { const { owner, client, index = 0n, address } = parameters + const localOwner = await toOwner({ owner }) + const entryPoint = { address: parameters.entryPoint?.address ?? entryPoint06Address, abi: entryPoint06Abi, @@ -127,7 +143,7 @@ export async function toBiconomySmartAccount( return { factory: factoryAddress, factoryData: await getAccountInitCode({ - owner: owner.address, + owner: localOwner.address, index, ecdsaModuleAddress }) @@ -190,7 +206,7 @@ export async function toBiconomySmartAccount( return this.signMessage({ message: hash }) }, async signMessage({ message }) { - let signature = await owner.signMessage({ + let signature = await localOwner.signMessage({ message }) @@ -209,7 +225,7 @@ export async function toBiconomySmartAccount( ) }, async signTypedData(typedData) { - let signature = await owner.signTypedData(typedData) + let signature = await localOwner.signTypedData(typedData) const potentiallyIncorrectV = Number.parseInt( signature.slice(-2), @@ -242,7 +258,7 @@ export async function toBiconomySmartAccount( chainId: chainId }) const signature = await signMessage(client, { - account: owner, + account: localOwner, message: { raw: hash } }) // userOp signature is encoded module signature + module address diff --git a/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts b/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts index edf5e27c..62077413 100644 --- a/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts +++ b/packages/permissionless/accounts/kernel/toEcdsaKernelSmartAccount.ts @@ -1,4 +1,12 @@ -import type { Assign } from "viem" +import type { + Account, + Assign, + Chain, + EIP1193Provider, + OneOf, + Transport, + WalletClient +} from "viem" import { type Address, type Client, @@ -24,6 +32,7 @@ import { signMessage as _signMessage, getChainId } from "viem/actions" import { getAction } from "viem/utils" import { getAccountNonce } from "../../actions/public/getAccountNonce" import { getSenderAddress } from "../../actions/public/getSenderAddress" +import { toOwner } from "../../utils/toOwner" import { KernelInitAbi } from "./abi/KernelAccountAbi" import { KernelV3InitAbi, KernelV3_1AccountAbi } from "./abi/KernelV3AccountAbi" import { KernelV3MetaFactoryDeployWithFactoryAbi } from "./abi/KernelV3MetaFactoryAbi" @@ -294,7 +303,13 @@ export type ToEcdsaKernelSmartAccountParameters< kernelVersion extends KernelVersion > = { client: Client - owner: LocalAccount + owners: [ + OneOf< + | EIP1193Provider + | WalletClient + | LocalAccount + > + ] entryPoint?: { address: Address version: entryPointVersion @@ -352,7 +367,7 @@ export async function toEcdsaKernelSmartAccount< client, address, index = 0n, - owner, + owners, version, ecdsaValidatorAddress: _ecdsaValidatorAddress, factoryAddress: _factoryAddress, @@ -360,6 +375,8 @@ export async function toEcdsaKernelSmartAccount< accountLogicAddress: _accountLogicAddress } = parameters + const localOwner = await toOwner({ owner: owners[0] }) + const entryPoint = { address: parameters.entryPoint?.address ?? entryPoint07Address, abi: @@ -389,7 +406,7 @@ export async function toEcdsaKernelSmartAccount< getAccountInitCode({ entryPointVersion: entryPoint.version, kernelVersion, - owner: owner.address, + owner: localOwner.address, index, factoryAddress, accountLogicAddress, @@ -461,7 +478,7 @@ export async function toEcdsaKernelSmartAccount< }, async signMessage({ message }) { const signature = await signMessage({ - owner: owner, + owner: localOwner, message, accountAddress: await this.getAddress(), kernelVersion, @@ -479,7 +496,7 @@ export async function toEcdsaKernelSmartAccount< }, async signTypedData(typedData) { const signature = await signTypedData({ - owner, + owner: localOwner, chainId: await getMemoizedChainId(), ...(typedData as TypedDataDefinition), accountAddress: await this.getAddress(), @@ -510,7 +527,7 @@ export async function toEcdsaKernelSmartAccount< entryPointVersion: entryPoint.version, chainId: chainId }) - const signature = await owner.signMessage({ + const signature = await localOwner.signMessage({ message: { raw: hash } }) diff --git a/packages/permissionless/accounts/light/toLightSmartAccount.ts b/packages/permissionless/accounts/light/toLightSmartAccount.ts index c50f29c9..3dffd840 100644 --- a/packages/permissionless/accounts/light/toLightSmartAccount.ts +++ b/packages/permissionless/accounts/light/toLightSmartAccount.ts @@ -1,9 +1,15 @@ import { + type Account, type Address, type Assign, + type Chain, type Client, + type EIP1193Provider, type Hex, type LocalAccount, + type OneOf, + type Transport, + type WalletClient, concat, encodeFunctionData, hashMessage, @@ -23,6 +29,7 @@ import { getChainId, signMessage } from "viem/actions" import { getAction } from "viem/utils" import { getAccountNonce } from "../../actions/public/getAccountNonce" import { getSenderAddress } from "../../actions/public/getSenderAddress" +import { toOwner } from "../../utils/toOwner" const getAccountInitCode = async ( owner: Address, @@ -73,7 +80,11 @@ export type ToLightSmartAccountParameters< address: Address version: entryPointVersion } - owner: LocalAccount + owner: OneOf< + | EIP1193Provider + | WalletClient + | LocalAccount + > version: LightAccountVersion factoryAddress?: Address index?: bigint @@ -176,6 +187,8 @@ export async function toLightSmartAccount< nonceKey } = parameters + const localOwner = await toOwner({ owner }) + const entryPoint = { address: parameters.entryPoint?.address ?? entryPoint07Address, abi: @@ -204,7 +217,7 @@ export async function toLightSmartAccount< const getFactoryArgs = async () => { return { factory: factoryAddress, - factoryData: await getAccountInitCode(owner.address, index) + factoryData: await getAccountInitCode(localOwner.address, index) } } @@ -317,7 +330,7 @@ export async function toLightSmartAccount< }, async signMessage({ message }) { const signature = await signWith1271WrapperV1( - owner, + localOwner, await getMemoizedChainId(), await this.getAddress(), hashMessage(message) @@ -334,7 +347,7 @@ export async function toLightSmartAccount< }, async signTypedData(typedData) { const signature = await signWith1271WrapperV1( - owner, + localOwner, await getMemoizedChainId(), await this.getAddress(), hashTypedData(typedData) @@ -364,7 +377,7 @@ export async function toLightSmartAccount< }) const signature = await signMessage(client, { - account: owner, + account: localOwner, message: { raw: hash } diff --git a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts index 767b0641..10b07cf7 100644 --- a/packages/permissionless/accounts/safe/toSafeSmartAccount.ts +++ b/packages/permissionless/accounts/safe/toSafeSmartAccount.ts @@ -1,12 +1,18 @@ import { + type Account, type Address, type Assign, + type Chain, type Client, + type EIP1193Provider, type Hex, type LocalAccount, + type OneOf, type SignableMessage, + type Transport, type TypedData, type TypedDataDefinition, + type WalletClient, concat, concatHex, encodeAbiParameters, @@ -34,7 +40,7 @@ import { import { getChainId, readContract, signTypedData } from "viem/actions" import { getAction } from "viem/utils" import { getAccountNonce } from "../../actions/public/getAccountNonce" -import { isSmartAccountDeployed } from "../../utils" +import { isSmartAccountDeployed, toOwner } from "../../utils" import { encode7579Calls } from "../../utils/encode7579Calls" export type SafeVersion = "1.4.1" @@ -932,7 +938,13 @@ export type ToSafeSmartAccountParameters< TErc7579 extends Address | undefined > = { client: Client - owner: LocalAccount + owners: [ + OneOf< + | EIP1193Provider + | WalletClient + | LocalAccount + > + ] version: SafeVersion entryPoint?: { address: Address @@ -1108,7 +1120,7 @@ export async function toSafeSmartAccount< ): Promise> { const { client, - owner, + owners, address, version, safe4337ModuleAddress: _safe4337ModuleAddress, @@ -1124,6 +1136,8 @@ export async function toSafeSmartAccount< paymentReceiver } = parameters + const localOwner = await toOwner({ owner: owners[0] }) + const entryPoint = { address: parameters.entryPoint?.address ?? entryPoint07Address, abi: @@ -1198,7 +1212,7 @@ export async function toSafeSmartAccount< return { factory: safeProxyFactoryAddress, factoryData: await getAccountInitCode({ - owner: owner.address, + owner: localOwner.address, safeModuleSetupAddress, safe4337ModuleAddress, safeSingletonAddress, @@ -1230,7 +1244,7 @@ export async function toSafeSmartAccount< // Get the sender address based on the init code accountAddress = await getAccountAddress({ client, - owner: owner.address, + owner: localOwner.address, safeModuleSetupAddress, safe4337ModuleAddress, safeProxyFactoryAddress, @@ -1267,7 +1281,7 @@ export async function toSafeSmartAccount< safe4337ModuleAddress, safeSingletonAddress, erc7579LaunchpadAddress, - owner: owner.address, + owner: localOwner.address, validators, executors, fallbacks, @@ -1375,7 +1389,7 @@ export async function toSafeSmartAccount< return adjustVInSignature( "eth_sign", - await owner.signMessage({ + await localOwner.signMessage({ message: { raw: toBytes(messageHash) } @@ -1385,7 +1399,7 @@ export async function toSafeSmartAccount< async signTypedData(typedData) { return adjustVInSignature( "eth_signTypedData", - await owner.signTypedData({ + await localOwner.signTypedData({ domain: { chainId: await getMemoizedChainId(), verifyingContract: await this.getAddress() @@ -1451,9 +1465,9 @@ export async function toSafeSmartAccount< const signatures = [ { - signer: owner.address, + signer: localOwner.address, data: await signTypedData(client, { - account: owner, + account: localOwner, domain: { chainId, verifyingContract diff --git a/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts b/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts index 138e69f3..d23f3def 100644 --- a/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts +++ b/packages/permissionless/accounts/simple/toSimpleSmartAccount.ts @@ -1,9 +1,15 @@ import { + type Account, type Address, type Assign, + type Chain, type Client, + type EIP1193Provider, type Hex, type LocalAccount, + type OneOf, + type Transport, + type WalletClient, encodeFunctionData } from "viem" import { @@ -20,6 +26,7 @@ import { getChainId, signMessage } from "viem/actions" import { getAction } from "viem/utils" import { getAccountNonce } from "../../actions/public/getAccountNonce" import { getSenderAddress } from "../../actions/public/getSenderAddress" +import { toOwner } from "../../utils/toOwner" const getAccountInitCode = async ( owner: Address, @@ -63,7 +70,11 @@ export type ToSimpleSmartAccountParameters< entryPointVersion extends "0.6" | "0.7" > = { client: Client - owner: LocalAccount + owner: OneOf< + | EIP1193Provider + | WalletClient + | LocalAccount + > factoryAddress?: Address entryPoint?: { address: Address @@ -126,6 +137,8 @@ export async function toSimpleSmartAccount< nonceKey } = parameters + const localOwner = await toOwner({ owner }) + const entryPoint = { address: parameters.entryPoint?.address ?? entryPoint07Address, abi: @@ -155,7 +168,7 @@ export async function toSimpleSmartAccount< const getFactoryArgs = async () => { return { factory: factoryAddress, - factoryData: await getAccountInitCode(owner.address, index) + factoryData: await getAccountInitCode(localOwner.address, index) } } @@ -296,7 +309,7 @@ export async function toSimpleSmartAccount< const { chainId = await getMemoizedChainId(), ...userOperation } = parameters return signMessage(client, { - account: owner, + account: localOwner, message: { raw: getUserOperationHash({ userOperation: { diff --git a/packages/permissionless/accounts/trust/toTrustSmartAccount.ts b/packages/permissionless/accounts/trust/toTrustSmartAccount.ts index 8afdffcf..441e757f 100644 --- a/packages/permissionless/accounts/trust/toTrustSmartAccount.ts +++ b/packages/permissionless/accounts/trust/toTrustSmartAccount.ts @@ -1,9 +1,15 @@ import { + type Account, type Address, type Assign, + type Chain, type Client, + type EIP1193Provider, type Hex, type LocalAccount, + type OneOf, + type Transport, + type WalletClient, hashMessage, hashTypedData } from "viem" @@ -21,6 +27,7 @@ import { } from "viem/account-abstraction" import { getAction } from "viem/utils" import { getSenderAddress } from "../../actions/public/getSenderAddress" +import { toOwner } from "../../utils/toOwner" import { encodeCallData } from "./utils/encodeCallData" import { getFactoryData } from "./utils/getFactoryData" @@ -63,7 +70,11 @@ export type ToTrustSmartAccountParameters< entryPointVersion extends "0.6" = "0.6" > = { client: Client - owner: LocalAccount + owner: OneOf< + | EIP1193Provider + | WalletClient + | LocalAccount + > factoryAddress?: Address entryPoint?: { address: Address @@ -113,6 +124,8 @@ export async function toTrustSmartAccount< secp256k1VerificationFacetAddress = TRUST_ADDRESSES.secp256k1VerificationFacetAddress } = parameters + const localOwner = await toOwner({ owner }) + let accountAddress: Address | undefined = address const entryPoint = { @@ -135,7 +148,7 @@ export async function toTrustSmartAccount< return { factory: factoryAddress, factoryData: await getFactoryData({ - bytes: owner.address, + bytes: localOwner.address, secp256k1VerificationFacetAddress, index }) @@ -178,7 +191,7 @@ export async function toTrustSmartAccount< }, async signMessage({ message }) { return _signTypedData( - owner, + localOwner, await getMemoizedChainId(), await this.getAddress(), hashMessage(message) @@ -186,7 +199,7 @@ export async function toTrustSmartAccount< }, async signTypedData(typedData) { return _signTypedData( - owner, + localOwner, await getMemoizedChainId(), await this.getAddress(), hashTypedData(typedData) @@ -197,7 +210,7 @@ export async function toTrustSmartAccount< parameters return signMessage(client, { - account: owner, + account: localOwner, message: { raw: getUserOperationHash({ userOperation: { diff --git a/packages/permissionless/package.json b/packages/permissionless/package.json index 97f57b99..22268a61 100644 --- a/packages/permissionless/package.json +++ b/packages/permissionless/package.json @@ -1,6 +1,6 @@ { "name": "permissionless", - "version": "0.1.45", + "version": "0.2.0-rc-5", "author": "Pimlico", "homepage": "https://docs.pimlico.io/permissionless", "repository": "github:pimlicolabs/permissionless.js", diff --git a/packages/permissionless/utils/toOwner.ts b/packages/permissionless/utils/toOwner.ts index 4346adbb..7cab8742 100644 --- a/packages/permissionless/utils/toOwner.ts +++ b/packages/permissionless/utils/toOwner.ts @@ -22,10 +22,20 @@ export async function toOwner({ address }: { owner: OneOf< - EIP1193Provider | WalletClient + | EIP1193Provider + | WalletClient + | LocalAccount > address?: Address }): Promise { + if ("type" in owner && owner.type === "local") { + return owner as LocalAccount + } + + let walletClient: + | WalletClient + | undefined = undefined + if ("request" in owner) { if (!address) { try { @@ -46,20 +56,28 @@ export async function toOwner({ // For TS to be happy throw new Error("address is required") } - owner = createWalletClient({ + walletClient = createWalletClient({ account: address, - transport: custom(owner) + transport: custom(owner as EIP1193Provider) }) } + if (!walletClient) { + walletClient = owner as WalletClient< + Transport, + Chain | undefined, + Account + > + } + return toAccount({ - address: owner.account.address, + address: walletClient.account.address, async signMessage({ message }) { - return owner.signMessage({ message }) + return walletClient.signMessage({ message }) }, async signTypedData(typedData) { return getAction( - owner, + walletClient, signTypedData, "signTypedData" )(typedData as any) From 514608ae3d7b1d673d949dcc9643a9c7ee47a712 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Fri, 30 Aug 2024 15:56:28 +0100 Subject: [PATCH 37/51] add multipel signer support for biconomy --- .../accounts/biconomy/toBiconomySmartAccount.ts | 16 +++++++++------- 1 file changed, 9 insertions(+), 7 deletions(-) diff --git a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts index 13ac11a4..7053816f 100644 --- a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts +++ b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts @@ -82,11 +82,13 @@ const getAccountInitCode = async ({ export type ToBiconomySmartAccountParameters = Prettify<{ client: Client - owner: OneOf< - | EIP1193Provider - | WalletClient - | LocalAccount - > + owners: [ + OneOf< + | EIP1193Provider + | WalletClient + | LocalAccount + > + ] address?: Address | undefined entryPoint?: { address: Address @@ -120,9 +122,9 @@ export type ToBiconomySmartAccountReturnType = Prettify< export async function toBiconomySmartAccount( parameters: ToBiconomySmartAccountParameters ): Promise { - const { owner, client, index = 0n, address } = parameters + const { owners, client, index = 0n, address } = parameters - const localOwner = await toOwner({ owner }) + const localOwner = await toOwner({ owner: owners[0] }) const entryPoint = { address: parameters.entryPoint?.address ?? entryPoint06Address, From ac94d1fe8e8776bd5e23860ffd9f5d3bc8e45496 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Fri, 30 Aug 2024 16:00:15 +0100 Subject: [PATCH 38/51] Fix trest --- packages/permissionless-test/src/utils.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/permissionless-test/src/utils.ts b/packages/permissionless-test/src/utils.ts index c9c46707..fd6cd353 100644 --- a/packages/permissionless-test/src/utils.ts +++ b/packages/permissionless-test/src/utils.ts @@ -317,7 +317,7 @@ export const getBiconomyClient = async < }: AAParamType) => { return toBiconomySmartAccount({ client: getPublicClient(anvilRpc), - owner: privateKeyToAccount(generatePrivateKey()) + owners: [privateKeyToAccount(generatePrivateKey())] }) } From 96055238c179c98fde0c2628b073095bf0248090 Mon Sep 17 00:00:00 2001 From: mouseless <97399882+mouseless-eth@users.noreply.github.com> Date: Fri, 30 Aug 2024 17:38:59 +0100 Subject: [PATCH 39/51] use number for chainId --- packages/permissionless/actions/pimlico/getTokenQuotes.test.ts | 2 +- packages/permissionless/actions/pimlico/getTokenQuotes.ts | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts b/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts index 6029b8b5..bfb53f1b 100644 --- a/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts +++ b/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts @@ -18,7 +18,7 @@ describe("getTokenQuotes", () => { const quotes = await getTokenQuotes(pimlicoBundlerClient, { tokens: [token], entryPointAddress: entryPoint07Address, - chainId: BigInt(foundry.id) + chainId: foundry.id }) expect(quotes).toBeTruthy() diff --git a/packages/permissionless/actions/pimlico/getTokenQuotes.ts b/packages/permissionless/actions/pimlico/getTokenQuotes.ts index c3d1169d..f4aaa8d9 100644 --- a/packages/permissionless/actions/pimlico/getTokenQuotes.ts +++ b/packages/permissionless/actions/pimlico/getTokenQuotes.ts @@ -12,7 +12,7 @@ import type { PimlicoRpcSchema } from "../../types/pimlico" export type GetTokenQuotesParameters = { tokens: Address[] entryPointAddress: Address - chainId: bigint + chainId: number } export type GetTokenQuotesReturnType = { From cb81a05ea0116646cad80cf71615b687b9e590ed Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Mon, 2 Sep 2024 10:46:50 +0100 Subject: [PATCH 40/51] Simplify trust wallet types --- .../accounts/trust/toTrustSmartAccount.ts | 29 +++++++------------ packages/permissionless/clients/pimlico.ts | 14 +++++---- 2 files changed, 19 insertions(+), 24 deletions(-) diff --git a/packages/permissionless/accounts/trust/toTrustSmartAccount.ts b/packages/permissionless/accounts/trust/toTrustSmartAccount.ts index 441e757f..d53c9c57 100644 --- a/packages/permissionless/accounts/trust/toTrustSmartAccount.ts +++ b/packages/permissionless/accounts/trust/toTrustSmartAccount.ts @@ -66,9 +66,7 @@ export const TRUST_ADDRESSES: { factoryAddress: "0x729c310186a57833f622630a16d13f710b83272a" } -export type ToTrustSmartAccountParameters< - entryPointVersion extends "0.6" = "0.6" -> = { +export type ToTrustSmartAccountParameters = { client: Client owner: OneOf< | EIP1193Provider @@ -78,7 +76,7 @@ export type ToTrustSmartAccountParameters< factoryAddress?: Address entryPoint?: { address: Address - version: entryPointVersion + version: "0.6" } index?: bigint address?: Address @@ -86,12 +84,10 @@ export type ToTrustSmartAccountParameters< nonceKey?: bigint } -export type TrustSmartAccountImplementation< - entryPointVersion extends "0.6" = "0.6" -> = Assign< +export type TrustSmartAccountImplementation = Assign< SmartAccountImplementation< typeof entryPoint06Abi, - entryPointVersion + "0.6" // { // // entryPoint === ENTRYPOINT_ADDRESS_V06 ? "0.2.2" : "0.3.0-beta" // abi: entryPointVersion extends "0.6" ? typeof BiconomyAbi @@ -101,20 +97,17 @@ export type TrustSmartAccountImplementation< { sign: NonNullable } > -export type ToTrustSmartAccountReturnType< - entryPointVersion extends "0.6" = "0.6" -> = SmartAccount> +export type ToTrustSmartAccountReturnType = + SmartAccount /** * @description Creates an Trust Smart Account from a private key. * * @returns A Private Key Trust Smart Account. */ -export async function toTrustSmartAccount< - entryPointVersion extends "0.6" = "0.6" ->( - parameters: ToTrustSmartAccountParameters -): Promise> { +export async function toTrustSmartAccount( + parameters: ToTrustSmartAccountParameters +): Promise { const { owner, client, @@ -219,7 +212,7 @@ export async function toTrustSmartAccount< userOperation.sender ?? (await this.getAddress()), signature: "0x" - } as UserOperation, + } as UserOperation<"0.6">, entryPointAddress: entryPoint.address, entryPointVersion: entryPoint.version, chainId: chainId @@ -227,5 +220,5 @@ export async function toTrustSmartAccount< } }) } - }) as Promise> + }) as Promise } diff --git a/packages/permissionless/clients/pimlico.ts b/packages/permissionless/clients/pimlico.ts index 51e82cdd..d51e171b 100644 --- a/packages/permissionless/clients/pimlico.ts +++ b/packages/permissionless/clients/pimlico.ts @@ -138,15 +138,17 @@ export function createPimlicoClient( name = "Pimlico Bundler Client", entryPoint } = parameters - const client = createClient({ + return createClient({ ...parameters, key, name, type: "pimlicoBundlerClient" }) - return client.extend(bundlerActions).extend(paymasterActions).extend( - pimlicoActions({ - entryPoint - }) - ) + .extend(bundlerActions) + .extend(paymasterActions) + .extend( + pimlicoActions({ + entryPoint + }) + ) } From 22cf632ac961b67d7d04e737bd9b0515c529d17b Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Mon, 2 Sep 2024 10:55:28 +0100 Subject: [PATCH 41/51] Make entryPoint explicitely required for implementations with only 0.6 as we default to entrypoint 0.7 --- packages/permissionless-test/src/utils.ts | 12 ++++++++++-- .../accounts/biconomy/toBiconomySmartAccount.ts | 2 +- .../accounts/trust/toTrustSmartAccount.ts | 2 +- 3 files changed, 12 insertions(+), 4 deletions(-) diff --git a/packages/permissionless-test/src/utils.ts b/packages/permissionless-test/src/utils.ts index fd6cd353..47700f52 100644 --- a/packages/permissionless-test/src/utils.ts +++ b/packages/permissionless-test/src/utils.ts @@ -305,7 +305,11 @@ export const getTrustAccountClient = async < }: AAParamType) => { return toTrustSmartAccount({ client: getPublicClient(anvilRpc), - owner: privateKeyToAccount(generatePrivateKey()) + owner: privateKeyToAccount(generatePrivateKey()), + entryPoint: { + address: entryPoint06Address, + version: "0.6" + } }) } @@ -317,7 +321,11 @@ export const getBiconomyClient = async < }: AAParamType) => { return toBiconomySmartAccount({ client: getPublicClient(anvilRpc), - owners: [privateKeyToAccount(generatePrivateKey())] + owners: [privateKeyToAccount(generatePrivateKey())], + entryPoint: { + address: entryPoint06Address, + version: "0.6" + } }) } diff --git a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts index 7053816f..3ddf7903 100644 --- a/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts +++ b/packages/permissionless/accounts/biconomy/toBiconomySmartAccount.ts @@ -90,7 +90,7 @@ export type ToBiconomySmartAccountParameters = Prettify<{ > ] address?: Address | undefined - entryPoint?: { + entryPoint: { address: Address version: "0.6" } diff --git a/packages/permissionless/accounts/trust/toTrustSmartAccount.ts b/packages/permissionless/accounts/trust/toTrustSmartAccount.ts index d53c9c57..ef9f9a42 100644 --- a/packages/permissionless/accounts/trust/toTrustSmartAccount.ts +++ b/packages/permissionless/accounts/trust/toTrustSmartAccount.ts @@ -74,7 +74,7 @@ export type ToTrustSmartAccountParameters = { | LocalAccount > factoryAddress?: Address - entryPoint?: { + entryPoint: { address: Address version: "0.6" } From ea429795d593a20f7b9a01610eaeeae6912fdf4a Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Mon, 2 Sep 2024 11:55:20 +0100 Subject: [PATCH 42/51] Pimlico client defaults to entrypoint 0.7 --- .../pimlico/sendCompressedUserOperation.ts | 6 +- .../actions/pimlico/sponsorUserOperation.ts | 23 ++------ .../pimlico/validateSponsorshipPolicies.ts | 37 +++--------- .../actions/public/getSenderAddress.test.ts | 3 +- .../clients/decorators/pimlico.ts | 35 ++---------- packages/permissionless/clients/pimlico.ts | 57 +++++-------------- packages/permissionless/types/pimlico.ts | 17 ++---- 7 files changed, 38 insertions(+), 140 deletions(-) diff --git a/packages/permissionless/actions/pimlico/sendCompressedUserOperation.ts b/packages/permissionless/actions/pimlico/sendCompressedUserOperation.ts index 45835929..232d7728 100644 --- a/packages/permissionless/actions/pimlico/sendCompressedUserOperation.ts +++ b/packages/permissionless/actions/pimlico/sendCompressedUserOperation.ts @@ -7,16 +7,12 @@ import type { Hex, Transport } from "viem" -import type { - entryPoint06Address, - entryPoint07Address -} from "viem/account-abstraction" import type { PimlicoRpcSchema } from "../../types/pimlico" export type SendCompressedUserOperationParameters = { compressedUserOperation: Hex inflatorAddress: Address - entryPointAddress: typeof entryPoint06Address | typeof entryPoint07Address + entryPointAddress: Address } /** diff --git a/packages/permissionless/actions/pimlico/sponsorUserOperation.ts b/packages/permissionless/actions/pimlico/sponsorUserOperation.ts index 45d1545b..fdfac621 100644 --- a/packages/permissionless/actions/pimlico/sponsorUserOperation.ts +++ b/packages/permissionless/actions/pimlico/sponsorUserOperation.ts @@ -8,18 +8,11 @@ import type { PartialBy, Transport } from "viem" -import type { - UserOperation, - entryPoint06Address, - entryPoint07Address -} from "viem/account-abstraction" +import type { UserOperation } from "viem/account-abstraction" import type { PimlicoRpcSchema } from "../../types/pimlico" import { deepHexlify } from "../../utils/deepHexlify" export type PimlicoSponsorUserOperationParameters< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address, entryPointVersion extends "0.6" | "0.7" > = { userOperation: OneOf< @@ -43,7 +36,7 @@ export type PimlicoSponsorUserOperationParameters< : never) > entryPoint: { - address: entryPointAddress + address: Address version: entryPointVersion } sponsorshipPolicyId?: string @@ -77,23 +70,15 @@ export type SponsorUserOperationReturnType< * @deprecated Use `getPaymasterData` instead */ export const sponsorUserOperation = async < - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address = - | typeof entryPoint06Address - | typeof entryPoint07Address, entryPointVersion extends "0.6" | "0.7" = "0.6" | "0.7" >( client: Client< Transport, Chain | undefined, Account | undefined, - PimlicoRpcSchema + PimlicoRpcSchema >, - args: PimlicoSponsorUserOperationParameters< - entryPointAddress, - entryPointVersion - > + args: PimlicoSponsorUserOperationParameters ): Promise> => { const response = await client.request({ method: "pm_sponsorUserOperation", diff --git a/packages/permissionless/actions/pimlico/validateSponsorshipPolicies.ts b/packages/permissionless/actions/pimlico/validateSponsorshipPolicies.ts index ffe303d0..e5c2bef3 100644 --- a/packages/permissionless/actions/pimlico/validateSponsorshipPolicies.ts +++ b/packages/permissionless/actions/pimlico/validateSponsorshipPolicies.ts @@ -1,22 +1,11 @@ -import type { Account, Chain, Client, Transport } from "viem" -import type { - UserOperation, - entryPoint06Address, - entryPoint07Address -} from "viem/account-abstraction" +import type { Account, Address, Chain, Client, Transport } from "viem" +import type { UserOperation } from "viem/account-abstraction" import type { PimlicoRpcSchema } from "../../types/pimlico" import { deepHexlify } from "../../utils/deepHexlify" -export type ValidateSponsorshipPoliciesParameters< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address = - | typeof entryPoint06Address - | typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7" = "0.6" | "0.7" -> = { - userOperation: UserOperation - entryPointAddress: entryPointAddress +export type ValidateSponsorshipPoliciesParameters = { + userOperation: UserOperation + entryPointAddress: Address sponsorshipPolicyIds: string[] } @@ -65,24 +54,14 @@ export type ValidateSponsorshipPolicies = { * } * ] */ -export const validateSponsorshipPolicies = async < - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address = - | typeof entryPoint06Address - | typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7" = "0.6" | "0.7" ->( +export const validateSponsorshipPolicies = async ( client: Client< Transport, Chain | undefined, Account | undefined, - PimlicoRpcSchema + PimlicoRpcSchema >, - args: ValidateSponsorshipPoliciesParameters< - entryPointAddress, - entryPointVersion - > + args: ValidateSponsorshipPoliciesParameters ): Promise => { return await client.request({ method: "pm_validateSponsorshipPolicies", diff --git a/packages/permissionless/actions/public/getSenderAddress.test.ts b/packages/permissionless/actions/public/getSenderAddress.test.ts index 564fc00e..762aa082 100644 --- a/packages/permissionless/actions/public/getSenderAddress.test.ts +++ b/packages/permissionless/actions/public/getSenderAddress.test.ts @@ -75,8 +75,7 @@ describe("getSenderAddress", () => { await expect(async () => getSenderAddress(client, { - entryPointAddress: - "0x0000000000000000000000000000000000000000" as typeof entryPoint06Address, + entryPointAddress: "0x0000000000000000000000000000000000000000", initCode: concatHex([factory, factoryData]) }) ).rejects.toThrowError(/not a valid entry point/) diff --git a/packages/permissionless/clients/decorators/pimlico.ts b/packages/permissionless/clients/decorators/pimlico.ts index 624e4314..1fdda0a2 100644 --- a/packages/permissionless/clients/decorators/pimlico.ts +++ b/packages/permissionless/clients/decorators/pimlico.ts @@ -1,8 +1,4 @@ -import type { Client, Hash, Prettify } from "viem" -import type { - entryPoint06Address, - entryPoint07Address -} from "viem/account-abstraction" +import type { Address, Client, Hash, Prettify } from "viem" import { type SendCompressedUserOperationParameters, type ValidateSponsorshipPolicies, @@ -26,11 +22,6 @@ import { } from "../../actions/pimlico/sponsorUserOperation" export type PimlicoActions< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address = - | typeof entryPoint06Address - | typeof entryPoint07Address, entryPointVersion extends "0.6" | "0.7" = "0.6" | "0.7" > = { /** @@ -111,38 +102,24 @@ export type PimlicoActions< */ sponsorUserOperation: ( args: Omit< - PimlicoSponsorUserOperationParameters< - entryPointAddress, - entryPointVersion - >, + PimlicoSponsorUserOperationParameters, "entryPoint" > ) => Promise>> validateSponsorshipPolicies: ( args: Prettify< - Omit< - ValidateSponsorshipPoliciesParameters< - entryPointAddress, - entryPointVersion - >, - "entryPoint" - > + Omit > ) => Promise[]> } export const pimlicoActions = - < - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7" - >({ + ({ entryPoint }: { - entryPoint: { address: entryPointAddress; version: entryPointVersion } + entryPoint: { address: Address; version: entryPointVersion } }) => - (client: Client): PimlicoActions => ({ + (client: Client): PimlicoActions => ({ getUserOperationGasPrice: async () => getUserOperationGasPrice(client), getUserOperationStatus: async ( args: GetUserOperationStatusParameters diff --git a/packages/permissionless/clients/pimlico.ts b/packages/permissionless/clients/pimlico.ts index d51e171b..33c4cda2 100644 --- a/packages/permissionless/clients/pimlico.ts +++ b/packages/permissionless/clients/pimlico.ts @@ -1,10 +1,10 @@ import type { + Address, BundlerRpcSchema, Chain, Client, ClientConfig, Prettify, - PublicClientConfig, RpcSchema, Transport } from "viem" @@ -14,19 +14,13 @@ import { type PaymasterActions, type SmartAccount, bundlerActions, - type entryPoint06Address, - type entryPoint07Address, + entryPoint07Address, paymasterActions } from "viem/account-abstraction" import type { PimlicoRpcSchema } from "../types/pimlico" import { type PimlicoActions, pimlicoActions } from "./decorators/pimlico" export type PimlicoClient< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address = - | typeof entryPoint07Address - | typeof entryPoint06Address, entryPointVersion extends "0.6" | "0.7" = "0.7" | "0.6", transport extends Transport = Transport, chain extends Chain | undefined = Chain | undefined, @@ -48,16 +42,11 @@ export type PimlicoClient< : [...BundlerRpcSchema, ...PimlicoRpcSchema], BundlerActions & PaymasterActions & - PimlicoActions + PimlicoActions > > export type PimlicoClientConfig< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address = - | typeof entryPoint07Address - | typeof entryPoint06Address, entryPointVersion extends "0.6" | "0.7" = "0.7" | "0.6", transport extends Transport = Transport, chain extends Chain | undefined = Chain | undefined, @@ -76,44 +65,21 @@ export type PimlicoClientConfig< | "transport" > > & { - entryPoint: { - address: entryPointAddress + entryPoint?: { + address: Address version: entryPointVersion } } -/** - * Creates a pimlico specific Bundler Client with a given [Transport](https://viem.sh/docs/clients/intro.html) configured for a [Chain](https://viem.sh/docs/clients/chains.html). - * - * - Docs: https://docs.pimlico.io/permissionless/reference/clients/pimlicoBundlerClient - * - * A Pimlico Client is an interface to "pimlico endpoints" [JSON-RPC API](https://docs.pimlico.io/reference/bundler/endpoints) methods such as getting current blockchain gas prices, getting user operation status, etc through [Pimlico Bundler Actions](TODO://Add bundler action documentation link). - * - * @param config - {@link PublicClientConfig} - * @returns A Pimlico Bundler Client. {@link PimlicoBundlerClient} - * - * @example - * import { createPublicClient, http } from 'viem' - * import { mainnet } from 'viem/chains' - * - * const pimlicoBundlerClient = createPimlicoBundlerClient({ - * chain: mainnet, - * transport: http("https://api.pimlico.io/v2/goerli/rpc?apikey=YOUR_API_KEY_HERE"), - * }) - */ export function createPimlicoClient< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7", + entryPointVersion extends "0.6" | "0.7" = "0.7", transport extends Transport = Transport, chain extends Chain | undefined = undefined, - account extends SmartAccount | undefined = undefined, + account extends SmartAccount | undefined = SmartAccount | undefined, client extends Client | undefined = undefined, rpcSchema extends RpcSchema | undefined = undefined >( parameters: PimlicoClientConfig< - entryPointAddress, entryPointVersion, transport, chain, @@ -121,7 +87,6 @@ export function createPimlicoClient< rpcSchema > ): PimlicoClient< - entryPointAddress, entryPointVersion, transport, chain, @@ -138,17 +103,21 @@ export function createPimlicoClient( name = "Pimlico Bundler Client", entryPoint } = parameters + return createClient({ ...parameters, key, name, - type: "pimlicoBundlerClient" + type: "pimlicoClient" }) .extend(bundlerActions) .extend(paymasterActions) .extend( pimlicoActions({ - entryPoint + entryPoint: { + address: entryPoint?.address ?? entryPoint07Address, + version: entryPoint?.version ?? "0.7" + } }) ) } diff --git a/packages/permissionless/types/pimlico.ts b/packages/permissionless/types/pimlico.ts index 08b2ccfb..da16601b 100644 --- a/packages/permissionless/types/pimlico.ts +++ b/packages/permissionless/types/pimlico.ts @@ -1,9 +1,5 @@ import type { Address, Hash, Hex, OneOf, PartialBy } from "viem" -import type { - UserOperation, - entryPoint06Address, - entryPoint07Address -} from "viem/account-abstraction" +import type { UserOperation } from "viem/account-abstraction" type PimlicoUserOperationGasPriceWithBigIntAsHex = { slow: { @@ -33,10 +29,7 @@ export type PimlicoUserOperationStatus = { } export type PimlicoRpcSchema< - entryPointAddress extends - | typeof entryPoint06Address - | typeof entryPoint07Address = typeof entryPoint07Address, - entryPointVersion extends "0.6" | "0.7" = "0.7" + entryPointVersion extends "0.6" | "0.7" = "0.6" | "0.7" > = [ { Method: "pimlico_getUserOperationGasPrice" @@ -80,12 +73,12 @@ export type PimlicoRpcSchema< > : never) >, - entryPoint: entryPointAddress, + entryPoint: Address, metadata?: { sponsorshipPolicyId?: string } ] - ReturnType: entryPointAddress extends "0.6" + ReturnType: entryPointVersion extends "0.6" ? { paymasterAndData: Hex preVerificationGas: Hex @@ -111,7 +104,7 @@ export type PimlicoRpcSchema< Method: "pm_validateSponsorshipPolicies" Parameters: [ userOperation: UserOperation, - entryPoint: entryPointAddress, + entryPoint: Address, sponsorshipPolicyIds: string[] ] ReturnType: { From 82f5ae53a92bc4ed1b18a106bb52677f3989ca6b Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Mon, 2 Sep 2024 12:10:56 +0100 Subject: [PATCH 43/51] fix test --- packages/permissionless-test/src/utils.ts | 9 +-------- 1 file changed, 1 insertion(+), 8 deletions(-) diff --git a/packages/permissionless-test/src/utils.ts b/packages/permissionless-test/src/utils.ts index 47700f52..0a859c63 100644 --- a/packages/permissionless-test/src/utils.ts +++ b/packages/permissionless-test/src/utils.ts @@ -13,9 +13,7 @@ import { type SmartAccount, createBundlerClient, createPaymasterClient, - entryPoint06Abi, entryPoint06Address, - entryPoint07Abi, entryPoint07Address } from "viem/account-abstraction" import { @@ -186,12 +184,7 @@ export const getPimlicoClient = ({ entryPointVersion: entryPointVersion altoRpc: string }) => - createPimlicoClient< - entryPointVersion extends "0.6" - ? typeof entryPoint06Address - : typeof entryPoint07Address, - entryPointVersion - >({ + createPimlicoClient({ chain: foundry, entryPoint: { address: (entryPointVersion === "0.6" From 5609b5bb2b30b7ea9f28159d730db034b447974e Mon Sep 17 00:00:00 2001 From: mouseless <97399882+mouseless-eth@users.noreply.github.com> Date: Mon, 2 Sep 2024 12:18:39 +0100 Subject: [PATCH 44/51] allow for chain overrides --- .../actions/pimlico/getTokenQuotes.test.ts | 28 ++++++++++++++- .../actions/pimlico/getTokenQuotes.ts | 35 ++++++++++++------- .../clients/decorators/pimlico.ts | 19 +++++++--- 3 files changed, 64 insertions(+), 18 deletions(-) diff --git a/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts b/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts index bfb53f1b..691cbe37 100644 --- a/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts +++ b/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts @@ -18,7 +18,33 @@ describe("getTokenQuotes", () => { const quotes = await getTokenQuotes(pimlicoBundlerClient, { tokens: [token], entryPointAddress: entryPoint07Address, - chainId: foundry.id + chain: foundry + }) + + expect(quotes).toBeTruthy() + expect(Array.isArray(quotes)).toBe(true) + expect(quotes[0].token).toBeTruthy() + expect(isAddress(quotes[0].token)) + expect(quotes[0].token).toEqual(token) + expect(quotes[0].paymaster).toBeTruthy() + expect(isAddress(quotes[0].paymaster)) + expect(quotes[0].exchangeRate).toBeTruthy() + expect(quotes[0].exchangeRate).toBeGreaterThan(0n) + expect(quotes[0].postOpGas).toBeTruthy() + expect(quotes[0].postOpGas).toBeGreaterThan(0n) + }) + + testWithRpc("get chain from client", async ({ rpc }) => { + const pimlicoBundlerClient = getPimlicoClient({ + entryPointVersion: "0.7", + altoRpc: rpc.paymasterRpc + }) + + const token = getAddress("0xffffffffffffffffffffffffffffffffffffffff") + + const quotes = pimlicoBundlerClient.getTokenQuotes({ + tokens: [token], + entryPointAddress: "0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" }) expect(quotes).toBeTruthy() diff --git a/packages/permissionless/actions/pimlico/getTokenQuotes.ts b/packages/permissionless/actions/pimlico/getTokenQuotes.ts index f4aaa8d9..4466cedb 100644 --- a/packages/permissionless/actions/pimlico/getTokenQuotes.ts +++ b/packages/permissionless/actions/pimlico/getTokenQuotes.ts @@ -4,16 +4,20 @@ import { type Chain, type Client, type Transport, + type GetChainParameter, hexToBigInt, - numberToHex + numberToHex, + ChainNotFoundError } from "viem" import type { PimlicoRpcSchema } from "../../types/pimlico" -export type GetTokenQuotesParameters = { +export type GetTokenQuotesParameters< + TChain extends Chain | undefined, + TChainOverride extends Chain | undefined = Chain | undefined +> = { tokens: Address[] entryPointAddress: Address - chainId: number -} +} & GetChainParameter export type GetTokenQuotesReturnType = { paymaster: Address @@ -32,21 +36,26 @@ export type GetTokenQuotesReturnType = { * @returns quotes, see {@link GetTokenQuotesReturnType} * */ -export const getTokenQuotes = async ( - client: Client< - Transport, - Chain | undefined, - Account | undefined, - PimlicoRpcSchema - >, - args: GetTokenQuotesParameters +export const getTokenQuotes = async < + TChain extends Chain | undefined, + TTransport extends Transport = Transport, + TChainOverride extends Chain | undefined = Chain | undefined +>( + client: Client, + args: GetTokenQuotesParameters ): Promise => { + const chainId = args.chain?.id ?? client.chain?.id + + if (!chainId) { + throw new ChainNotFoundError() + } + const res = await client.request({ method: "pimlico_getTokenQuotes", params: [ { tokens: args.tokens }, args.entryPointAddress, - numberToHex(args.chainId) + numberToHex(chainId) ] }) diff --git a/packages/permissionless/clients/decorators/pimlico.ts b/packages/permissionless/clients/decorators/pimlico.ts index 696b8644..a7824904 100644 --- a/packages/permissionless/clients/decorators/pimlico.ts +++ b/packages/permissionless/clients/decorators/pimlico.ts @@ -1,4 +1,4 @@ -import type { Address, Client, Hash, Prettify } from "viem" +import type { Address, Chain, Client, Hash, Prettify, Transport } from "viem" import { type GetTokenQuotesParameters, type GetTokenQuotesReturnType, @@ -25,6 +25,7 @@ import { } from "../../actions/pimlico/sponsorUserOperation" export type PimlicoActions< + TChain extends Chain | undefined, entryPointVersion extends "0.6" | "0.7" = "0.6" | "0.7" > = { /** @@ -114,8 +115,12 @@ export type PimlicoActions< Omit > ) => Promise[]> - getTokenQuotes: ( - args: Prettify> + getTokenQuotes: < + TChainOverride extends Chain | undefined = Chain | undefined + >( + args: Prettify< + Omit, "entryPoint"> + > ) => Promise> } @@ -125,7 +130,12 @@ export const pimlicoActions = }: { entryPoint: { address: Address; version: entryPointVersion } }) => - (client: Client): PimlicoActions => ({ + < + TTransport extends Transport, + TChain extends Chain | undefined = Chain | undefined + >( + client: Client + ): PimlicoActions => ({ getUserOperationGasPrice: async () => getUserOperationGasPrice(client), getUserOperationStatus: async ( args: GetUserOperationStatusParameters @@ -150,6 +160,7 @@ export const pimlicoActions = getTokenQuotes: async (args) => getTokenQuotes(client, { ...args, + chain: args.chain, entryPointAddress: entryPoint.address }) }) From b1d86589255b0bf535d47266fb2e538540ac0123 Mon Sep 17 00:00:00 2001 From: mouseless <97399882+mouseless-eth@users.noreply.github.com> Date: Mon, 2 Sep 2024 12:18:45 +0100 Subject: [PATCH 45/51] allow for chain overrides --- packages/permissionless/actions/pimlico/getTokenQuotes.ts | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/packages/permissionless/actions/pimlico/getTokenQuotes.ts b/packages/permissionless/actions/pimlico/getTokenQuotes.ts index 4466cedb..59d47b78 100644 --- a/packages/permissionless/actions/pimlico/getTokenQuotes.ts +++ b/packages/permissionless/actions/pimlico/getTokenQuotes.ts @@ -2,12 +2,12 @@ import { type Account, type Address, type Chain, + ChainNotFoundError, type Client, - type Transport, type GetChainParameter, + type Transport, hexToBigInt, - numberToHex, - ChainNotFoundError + numberToHex } from "viem" import type { PimlicoRpcSchema } from "../../types/pimlico" From a66598b9b907c2e7ac8c9b19d13d08711264035c Mon Sep 17 00:00:00 2001 From: mouseless <97399882+mouseless-eth@users.noreply.github.com> Date: Mon, 2 Sep 2024 12:22:41 +0100 Subject: [PATCH 46/51] fix --- packages/permissionless/clients/pimlico.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/permissionless/clients/pimlico.ts b/packages/permissionless/clients/pimlico.ts index 33c4cda2..4cd141b9 100644 --- a/packages/permissionless/clients/pimlico.ts +++ b/packages/permissionless/clients/pimlico.ts @@ -42,7 +42,7 @@ export type PimlicoClient< : [...BundlerRpcSchema, ...PimlicoRpcSchema], BundlerActions & PaymasterActions & - PimlicoActions + PimlicoActions > > From 4c0a8a2e99f09f7d47d8988affa2426f3ac59226 Mon Sep 17 00:00:00 2001 From: mouseless <97399882+mouseless-eth@users.noreply.github.com> Date: Mon, 2 Sep 2024 12:33:48 +0100 Subject: [PATCH 47/51] fix --- packages/permissionless-test/src/utils.ts | 7 +---- .../actions/pimlico/getTokenQuotes.test.ts | 26 ------------------- .../clients/decorators/pimlico.ts | 5 +++- 3 files changed, 5 insertions(+), 33 deletions(-) diff --git a/packages/permissionless-test/src/utils.ts b/packages/permissionless-test/src/utils.ts index 47700f52..0acdc9f7 100644 --- a/packages/permissionless-test/src/utils.ts +++ b/packages/permissionless-test/src/utils.ts @@ -186,12 +186,7 @@ export const getPimlicoClient = ({ entryPointVersion: entryPointVersion altoRpc: string }) => - createPimlicoClient< - entryPointVersion extends "0.6" - ? typeof entryPoint06Address - : typeof entryPoint07Address, - entryPointVersion - >({ + createPimlicoClient({ chain: foundry, entryPoint: { address: (entryPointVersion === "0.6" diff --git a/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts b/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts index 691cbe37..db6f26ca 100644 --- a/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts +++ b/packages/permissionless/actions/pimlico/getTokenQuotes.test.ts @@ -33,30 +33,4 @@ describe("getTokenQuotes", () => { expect(quotes[0].postOpGas).toBeTruthy() expect(quotes[0].postOpGas).toBeGreaterThan(0n) }) - - testWithRpc("get chain from client", async ({ rpc }) => { - const pimlicoBundlerClient = getPimlicoClient({ - entryPointVersion: "0.7", - altoRpc: rpc.paymasterRpc - }) - - const token = getAddress("0xffffffffffffffffffffffffffffffffffffffff") - - const quotes = pimlicoBundlerClient.getTokenQuotes({ - tokens: [token], - entryPointAddress: "0xaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa" - }) - - expect(quotes).toBeTruthy() - expect(Array.isArray(quotes)).toBe(true) - expect(quotes[0].token).toBeTruthy() - expect(isAddress(quotes[0].token)) - expect(quotes[0].token).toEqual(token) - expect(quotes[0].paymaster).toBeTruthy() - expect(isAddress(quotes[0].paymaster)) - expect(quotes[0].exchangeRate).toBeTruthy() - expect(quotes[0].exchangeRate).toBeGreaterThan(0n) - expect(quotes[0].postOpGas).toBeTruthy() - expect(quotes[0].postOpGas).toBeGreaterThan(0n) - }) }) diff --git a/packages/permissionless/clients/decorators/pimlico.ts b/packages/permissionless/clients/decorators/pimlico.ts index a7824904..8525a2ee 100644 --- a/packages/permissionless/clients/decorators/pimlico.ts +++ b/packages/permissionless/clients/decorators/pimlico.ts @@ -119,7 +119,10 @@ export type PimlicoActions< TChainOverride extends Chain | undefined = Chain | undefined >( args: Prettify< - Omit, "entryPoint"> + Omit< + GetTokenQuotesParameters, + "entryPointAddress" + > > ) => Promise> } From 2b0500cd46c733b004a46890c37d85c3e3b24902 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Mon, 2 Sep 2024 12:34:17 +0100 Subject: [PATCH 48/51] Fix omits --- packages/permissionless/clients/decorators/pimlico.ts | 8 +++----- 1 file changed, 3 insertions(+), 5 deletions(-) diff --git a/packages/permissionless/clients/decorators/pimlico.ts b/packages/permissionless/clients/decorators/pimlico.ts index 1fdda0a2..f172b01c 100644 --- a/packages/permissionless/clients/decorators/pimlico.ts +++ b/packages/permissionless/clients/decorators/pimlico.ts @@ -94,7 +94,7 @@ export type PimlicoActions< */ sendCompressedUserOperation: ( args: Prettify< - Omit + Omit > ) => Promise /** @@ -108,7 +108,7 @@ export type PimlicoActions< ) => Promise>> validateSponsorshipPolicies: ( args: Prettify< - Omit + Omit > ) => Promise[]> } @@ -124,9 +124,7 @@ export const pimlicoActions = getUserOperationStatus: async ( args: GetUserOperationStatusParameters ) => getUserOperationStatus(client, args), - sendCompressedUserOperation: async ( - args: Omit - ) => + sendCompressedUserOperation: async (args) => sendCompressedUserOperation(client, { ...args, entryPointAddress: entryPoint.address From bcdf6ca604208b829e1fef1b6d79eedaf2e8a36d Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Mon, 2 Sep 2024 12:41:40 +0100 Subject: [PATCH 49/51] remove 1271 compliance --- packages/permissionless-test/src/utils.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/permissionless-test/src/utils.ts b/packages/permissionless-test/src/utils.ts index bf2fa01d..f8ec4439 100644 --- a/packages/permissionless-test/src/utils.ts +++ b/packages/permissionless-test/src/utils.ts @@ -429,7 +429,7 @@ export const getCoreSmartAccounts = () => [ }), supportsEntryPointV06: false, supportsEntryPointV07: true, - isEip1271Compliant: true + isEip1271Compliant: false }, { name: "Simple", From 315b6ba8c7f84fb72ba4441b0fe0c08f5877c0b3 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Mon, 2 Sep 2024 17:41:42 +0100 Subject: [PATCH 50/51] skip sign message --- packages/permissionless-test/src/utils.ts | 2 +- .../permissionless/actions/smartAccount/signMessage.test.ts | 2 +- .../permissionless/actions/smartAccount/signTypedData.test.ts | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/packages/permissionless-test/src/utils.ts b/packages/permissionless-test/src/utils.ts index f8ec4439..bf2fa01d 100644 --- a/packages/permissionless-test/src/utils.ts +++ b/packages/permissionless-test/src/utils.ts @@ -429,7 +429,7 @@ export const getCoreSmartAccounts = () => [ }), supportsEntryPointV06: false, supportsEntryPointV07: true, - isEip1271Compliant: false + isEip1271Compliant: true }, { name: "Simple", diff --git a/packages/permissionless/actions/smartAccount/signMessage.test.ts b/packages/permissionless/actions/smartAccount/signMessage.test.ts index bfb22f92..5b6f4b6a 100644 --- a/packages/permissionless/actions/smartAccount/signMessage.test.ts +++ b/packages/permissionless/actions/smartAccount/signMessage.test.ts @@ -98,7 +98,7 @@ describe.each(getCoreSmartAccounts())( const publicClient = getPublicClient(anvilRpc) - if (name === "Safe 7579") { + if (name === "Safe 7579" || name === "LightAccount 2.0.0") { // Due to 7579 launchpad, we can't verify the signature as of now. // Awaiting for the fix return diff --git a/packages/permissionless/actions/smartAccount/signTypedData.test.ts b/packages/permissionless/actions/smartAccount/signTypedData.test.ts index a6c7fc8f..41766c5f 100644 --- a/packages/permissionless/actions/smartAccount/signTypedData.test.ts +++ b/packages/permissionless/actions/smartAccount/signTypedData.test.ts @@ -124,7 +124,7 @@ describe.each(getCoreSmartAccounts())( const publicClient = getPublicClient(anvilRpc) - if (name === "Safe 7579") { + if (name === "Safe 7579" || name === "LightAccount 2.0.0") { // Due to 7579 launchpad, we can't verify the signature as of now. // Awaiting for the fix return From 2e4d504c12dbb4af6eeccc9b62648cae60193424 Mon Sep 17 00:00:00 2001 From: Garvit Khatri Date: Mon, 2 Sep 2024 17:45:28 +0100 Subject: [PATCH 51/51] Add changeset --- .changeset/tricky-humans-burn.md | 5 +++++ packages/permissionless/package.json | 2 +- 2 files changed, 6 insertions(+), 1 deletion(-) create mode 100644 .changeset/tricky-humans-burn.md diff --git a/.changeset/tricky-humans-burn.md b/.changeset/tricky-humans-burn.md new file mode 100644 index 00000000..261279c0 --- /dev/null +++ b/.changeset/tricky-humans-burn.md @@ -0,0 +1,5 @@ +--- +"permissionless": minor +--- + +Permissionless 0.2 released. Migration guide - https://docs.pimlico.io/permissionless/how-to/migration-guide diff --git a/packages/permissionless/package.json b/packages/permissionless/package.json index 22268a61..97f57b99 100644 --- a/packages/permissionless/package.json +++ b/packages/permissionless/package.json @@ -1,6 +1,6 @@ { "name": "permissionless", - "version": "0.2.0-rc-5", + "version": "0.1.45", "author": "Pimlico", "homepage": "https://docs.pimlico.io/permissionless", "repository": "github:pimlicolabs/permissionless.js",