Skip to content

Commit 0ede013

Browse files
committed
test(sdk): erc721mint action
1 parent 40df682 commit 0ede013

File tree

3 files changed

+133
-4
lines changed

3 files changed

+133
-4
lines changed
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,51 @@
1-
import { isAddress, zeroAddress } from 'viem';
2-
import { describe, expect, test } from 'vitest';
3-
import { defaultOptions } from '../../test/helpers';
1+
import { mockErc721Abi } from '@boostxyz/evm';
2+
import { loadFixture } from '@nomicfoundation/hardhat-network-helpers';
3+
import {
4+
encodeAbiParameters,
5+
encodeFunctionData,
6+
isAddress,
7+
parseEther,
8+
toFunctionSelector,
9+
zeroAddress,
10+
} from 'viem';
11+
import { beforeAll, beforeEach, describe, expect, test } from 'vitest';
12+
import type { MockERC721 } from '../../test/MockERC721';
13+
import { accounts } from '../../test/accounts';
14+
import {
15+
type Fixtures,
16+
defaultOptions,
17+
deployFixtures,
18+
fundErc721,
19+
} from '../../test/helpers';
420
import { ERC721MintAction } from './ERC721MintAction';
521

6-
describe('ERC721MintAction', () => {
22+
let fixtures: Fixtures, erc721: MockERC721;
23+
24+
beforeAll(async () => {
25+
fixtures = await loadFixture(deployFixtures);
26+
});
27+
28+
const mintSelector = toFunctionSelector('function mint(address to)');
29+
30+
function nonPayableAction(fixtures: Fixtures, erc721: MockERC721) {
31+
return function nonPayableAction() {
32+
return fixtures.registry.clone(
33+
crypto.randomUUID(),
34+
new fixtures.bases.ERC721MintAction(defaultOptions, {
35+
chainId: BigInt(31_337),
36+
target: erc721.assertValidAddress(),
37+
selector: mintSelector,
38+
value: 0n,
39+
}),
40+
);
41+
};
42+
}
43+
44+
describe('ContractAction', () => {
45+
beforeEach(async () => {
46+
erc721 = await loadFixture(fundErc721(defaultOptions));
47+
});
48+
749
test('can successfully be deployed', async () => {
850
const action = new ERC721MintAction(defaultOptions, {
951
chainId: BigInt(31_337),
@@ -14,4 +56,56 @@ describe('ERC721MintAction', () => {
1456
await action.deploy();
1557
expect(isAddress(action.assertValidAddress())).toBe(true);
1658
});
59+
60+
test('can read chain id', async () => {
61+
const action = await loadFixture(nonPayableAction(fixtures, erc721));
62+
expect(await action.chainId()).toBe(BigInt(31_337));
63+
});
64+
65+
test('can read target', async () => {
66+
const action = await loadFixture(nonPayableAction(fixtures, erc721));
67+
expect((await action.target()).toLowerCase()).toBe(
68+
erc721.assertValidAddress().toLowerCase(),
69+
);
70+
});
71+
72+
test('can read selector', async () => {
73+
const action = await loadFixture(nonPayableAction(fixtures, erc721));
74+
expect(await action.selector()).toBe(mintSelector);
75+
});
76+
77+
test('can read value', async () => {
78+
const action = await loadFixture(nonPayableAction(fixtures, erc721));
79+
expect(await action.value()).toBe(parseEther('0.1'));
80+
});
81+
82+
test('prepare will properly encode execution payload', async () => {
83+
const action = await loadFixture(nonPayableAction(fixtures, erc721));
84+
const { account } = accounts.at(1)!;
85+
const payload = await action.prepare(
86+
encodeAbiParameters([{ type: 'address', name: 'address' }], [account]),
87+
);
88+
expect(payload).toBe(
89+
encodeFunctionData({
90+
abi: mockErc721Abi,
91+
functionName: 'mint',
92+
args: [account],
93+
}),
94+
);
95+
});
96+
97+
test('nonpayable execute', async () => {
98+
const action = await loadFixture(nonPayableAction(fixtures, erc721));
99+
const { account } = accounts.at(1)!;
100+
const [success] = await action.execute(
101+
encodeAbiParameters(
102+
[
103+
{ type: 'address', name: 'to' },
104+
{ type: 'uint256', name: 'amount' },
105+
],
106+
[account, parseEther('100')],
107+
),
108+
);
109+
expect(success).toBe(true);
110+
});
17111
});

packages/sdk/src/Actions/ERC721MintAction.ts

+14
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,7 @@ import {
66
readErc721MintActionGetComponentInterface,
77
readErc721MintActionPrepare,
88
readErc721MintActionSupportsInterface,
9+
readErc721MintActionValidated,
910
simulateErc721MintActionExecute,
1011
simulateErc721MintActionValidate,
1112
writeErc721MintActionExecute,
@@ -26,6 +27,19 @@ export class ERC721MintAction extends ContractAction {
2627
public static override base = import.meta.env.VITE_ERC721_MINT_ACTION_BASE;
2728
public static override registryType: RegistryType = RegistryType.ACTION;
2829

30+
public async validated(
31+
token: bigint,
32+
params?: ReadParams<typeof erc721MintActionAbi, 'validated'>,
33+
) {
34+
return readErc721MintActionValidated(this._config, {
35+
address: this.assertValidAddress(),
36+
...this.optionallyAttachAccount(),
37+
// biome-ignore lint/suspicious/noExplicitAny: Accept any shape of valid wagmi/viem parameters, wagmi does the same thing internally
38+
...(params as any),
39+
args: [token],
40+
});
41+
}
42+
2943
public override async execute(
3044
data: Hex,
3145
params?: WriteParams<typeof erc721MintActionAbi, 'execute'>,

packages/sdk/test/helpers.ts

+21
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import {
22
readMockErc20BalanceOf,
3+
readMockErc721BalanceOf,
34
readMockErc1155BalanceOf,
45
writeMockErc20Approve,
56
writeMockErc1155SetApprovalForAll,
@@ -320,6 +321,26 @@ export function fundErc20(
320321
};
321322
}
322323

324+
export function fundErc721(
325+
options: DeployableTestOptions,
326+
erc721?: MockERC721,
327+
funded: Address[] = [],
328+
amount: bigint = parseEther('100'),
329+
) {
330+
return async function fundErc721() {
331+
if (!erc721) erc721 = await freshERC721();
332+
for (const address of [options.account.address, ...(funded ?? [])]) {
333+
await erc721.mint(address);
334+
const balance = await readMockErc721BalanceOf(options.config, {
335+
address: erc721.assertValidAddress(),
336+
args: [address],
337+
});
338+
if (amount !== balance) throw new Error(`Balance did not match`);
339+
}
340+
return erc721;
341+
};
342+
}
343+
323344
export function fundPoints(
324345
options: DeployableTestOptions,
325346
points?: MockPoints,

0 commit comments

Comments
 (0)