Skip to content

Commit

Permalink
feat: allowlist module installation and uninstallation
Browse files Browse the repository at this point in the history
  • Loading branch information
Zer0dot committed Dec 18, 2024
1 parent 2b4b01a commit afccbd9
Show file tree
Hide file tree
Showing 2 changed files with 178 additions and 1 deletion.
80 changes: 79 additions & 1 deletion account-kit/smart-contracts/src/ma-v2/client/client.test.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { custom, parseEther, publicActions } from "viem";
import { custom, parseEther, publicActions, zeroAddress } from "viem";

import {
LocalAccountSigner,
Expand All @@ -16,6 +16,8 @@ import { setBalance } from "viem/actions";
import { accounts } from "~test/constants.js";
import { getDefaultSingleSignerValidationModuleAddress } from "../modules/utils.js";
import { SingleSignerValidationModule } from "../modules/single-signer-validation/module.js";
import { allowlistModule } from "../modules/allowlist-module/module.js";
import { HookType } from "../actions/common/types.js";

describe("MA v2 Tests", async () => {
const instance = local070Instance;
Expand Down Expand Up @@ -188,6 +190,82 @@ describe("MA v2 Tests", async () => {
).rejects.toThrowError();
});

it("installs allowlist module, then uninstalls", async () => {
let provider = (await givenConnectedProvider({ signer })).extend(
installValidationActions
);

await setBalance(client, {
address: provider.getAddress(),
value: parseEther("2"),
});

const hookInstallData = allowlistModule.encodeOnInstallData({
entityId: 1,
inputs: [
{
target,
hasSelectorAllowlist: false,
hasERC20SpendLimit: false,
erc20SpendLimit: 0n,
selectors: [],
},
],
});

const installResult = await provider.installValidation({
validationConfig: {
moduleAddress: zeroAddress,
entityId: 0,
isGlobal: true,
isSignatureValidation: true,
isUserOpValidation: true,
},
selectors: [],
installData: "0x",
hooks: [
{
hookConfig: {
address: addresses.allowlistModule,
entityId: 0, // uint32
hookType: HookType.VALIDATION,
hasPreHooks: true,
hasPostHooks: true,
},
initData: hookInstallData,
},
],
});

await expect(
provider.waitForUserOperationTransaction(installResult)
).resolves.not.toThrowError();

const hookUninstallData = allowlistModule.encodeOnUninstallData({
entityId: 0,
inputs: [
{
target,
hasSelectorAllowlist: false,
hasERC20SpendLimit: false,
erc20SpendLimit: 0n,
selectors: [],
},
],
});

const uninstallResult = await provider.uninstallValidation({
moduleAddress: zeroAddress,
entityId: 0,
uninstallData: "0x",
hookUninstallDatas: [hookUninstallData],
});

await expect(
provider.waitForUserOperationTransaction(uninstallResult)
).resolves.not.toThrowError();
});

const givenConnectedProvider = async ({
signer,
accountAddress,
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,99 @@
import { encodeAbiParameters, type Address, type Hex } from "viem";

import { allowlistModuleAbi } from "./abis/allowlistModuleAbi.js";

const addresses: Record<number | "default", Address> = {
default: "0xE46ca4a98c485caEE2Abb6ef5116292B8c78a868",
};

const meta = {
name: "AllowlistModule",
version: "alpha.1",
addresses,
};

export const allowlistModule = {
meta,
abi: allowlistModuleAbi,
encodeOnInstallData: (args: {
entityId: number;
inputs: Array<{
target: Address;
hasSelectorAllowlist: boolean;
hasERC20SpendLimit: boolean;
erc20SpendLimit: bigint;
selectors: Array<Hex>;
}>;
}): Hex => {
const { entityId, inputs } = args;
return encodeAbiParameters(
[
{ type: "uint32" },
{
type: "tuple[]",
components: [
{ type: "address" },
{ type: "bool" },
{ type: "bool" },
{ type: "uint256" },
{ type: "bytes4[]" },
],
},
],
[
entityId,
inputs.map(
(input) =>
[
input.target,
input.hasSelectorAllowlist,
input.hasERC20SpendLimit,
input.erc20SpendLimit,
input.selectors,
] as const
),
]
);
},

encodeOnUninstallData: (args: {
entityId: number;
inputs: Array<{
target: Address;
hasSelectorAllowlist: boolean;
hasERC20SpendLimit: boolean;
erc20SpendLimit: bigint;
selectors: Array<Hex>;
}>;
}): Hex => {
const { entityId, inputs } = args;
return encodeAbiParameters(
[
{ type: "uint32" },
{
type: "tuple[]",
components: [
{ type: "address" },
{ type: "bool" },
{ type: "bool" },
{ type: "uint256" },
{ type: "bytes4[]" },
],
},
],
[
entityId,
inputs.map(
(input) =>
[
input.target,
input.hasSelectorAllowlist,
input.hasERC20SpendLimit,
input.erc20SpendLimit,
input.selectors,
] as const
),
]
);
},
};

0 comments on commit afccbd9

Please sign in to comment.