Skip to content

Commit d4fad06

Browse files
committed
Add support for paymasters in zk stack networks
1 parent f32d00a commit d4fad06

File tree

8 files changed

+209
-49
lines changed

8 files changed

+209
-49
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,4 @@
1+
OPENAI_API_KEY=
2+
WALLET_PRIVATE_KEY=
3+
PAYMASTER_ADDRESS=
4+
RPC_PROVIDER_URL=
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
# Vercel AI Abstract Example
2+
3+
## Setup
4+
5+
Copy the `.env.template` and populate with your values.
6+
7+
```
8+
cp .env.template .env
9+
```
10+
11+
## Usage
12+
13+
```
14+
npx ts-node index.ts
15+
```
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
import { openai } from "@ai-sdk/openai";
2+
import { generateText } from "ai";
3+
4+
import { http } from "viem";
5+
import { createWalletClient } from "viem";
6+
import { privateKeyToAccount } from "viem/accounts";
7+
import { abstractTestnet } from "viem/chains";
8+
9+
import { getOnChainTools } from "@goat-sdk/adapter-vercel-ai";
10+
import { PEPE, USDC, erc20 } from "@goat-sdk/plugin-erc20";
11+
12+
import { sendETH } from "@goat-sdk/core";
13+
import { viem } from "@goat-sdk/wallet-viem";
14+
15+
require("dotenv").config();
16+
17+
const account = privateKeyToAccount(
18+
process.env.WALLET_PRIVATE_KEY as `0x${string}`
19+
);
20+
21+
const walletClient = createWalletClient({
22+
account: account,
23+
transport: http(process.env.RPC_PROVIDER_URL),
24+
chain: abstractTestnet,
25+
});
26+
27+
(async () => {
28+
const tools = await getOnChainTools({
29+
wallet: viem(walletClient, {
30+
defaultPaymaster: process.env.PAYMASTER_ADDRESS as `0x${string}`,
31+
}),
32+
plugins: [sendETH(), erc20({ tokens: [USDC, PEPE] })],
33+
});
34+
35+
const result = await generateText({
36+
model: openai("gpt-4o-mini"),
37+
tools: tools,
38+
maxSteps: 5,
39+
prompt: "Send 1 USDC to 0x016c0803FFC6880a9a871ba104709cDBf341A90a",
40+
});
41+
42+
console.log(result.text);
43+
})();
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
{
2+
"name": "goat-examples-vercel-ai-abstract",
3+
"version": "0.1.0",
4+
"description": "",
5+
"private": true,
6+
"scripts": {
7+
"test": "vitest run --passWithNoTests"
8+
},
9+
"author": "",
10+
"license": "MIT",
11+
"dependencies": {
12+
"@ai-sdk/openai": "^1.0.4",
13+
"@goat-sdk/adapter-vercel-ai": "workspace:*",
14+
"@goat-sdk/core": "workspace:*",
15+
"@goat-sdk/plugin-erc20": "workspace:*",
16+
"@goat-sdk/wallet-viem": "workspace:*",
17+
"ai": "^4.0.3",
18+
"dotenv": "^16.4.5",
19+
"viem": "2.21.49"
20+
}
21+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
{
2+
"extends": "../../../tsconfig.base.json",
3+
"compilerOptions": {
4+
"outDir": "dist"
5+
},
6+
"include": ["index.ts"],
7+
"exclude": ["node_modules", "dist"]
8+
}

typescript/packages/core/src/wallets/evm.ts

+6
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,12 @@ export type EVMTransaction = {
1111
args?: unknown[];
1212
value?: bigint;
1313
abi?: Abi;
14+
options?: EVMTransactionOptions;
15+
};
16+
17+
export type EVMTransactionOptions = {
18+
paymaster?: string;
19+
paymasterInput?: string;
1420
};
1521

1622
export type EVMReadRequest = {

typescript/packages/wallets/viem/src/index.ts

+85-49
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,38 @@
1-
import type { EVMReadRequest, EVMTransaction, EVMTypedData, EVMWalletClient } from "@goat-sdk/core";
2-
3-
import { publicActions } from "viem";
4-
import type { WalletClient as ViemWalletClient } from "viem";
1+
import type {
2+
EVMReadRequest,
3+
EVMTransaction,
4+
EVMTypedData,
5+
EVMWalletClient,
6+
} from "@goat-sdk/core";
7+
import {
8+
publicActions,
9+
encodeFunctionData,
10+
type WalletClient as ViemWalletClient,
11+
} from "viem";
12+
import { mainnet } from "viem/chains";
513
import { normalize } from "viem/ens";
14+
import { eip712WalletActions } from "viem/zksync";
15+
16+
export type ViemOptions = {
17+
// Only used for zkSync Stack networks
18+
defaultPaymaster?: string;
19+
defaultPaymasterInput?: string;
20+
};
21+
22+
export function viem(
23+
client: ViemWalletClient,
24+
options?: ViemOptions
25+
): EVMWalletClient {
26+
const defaultPaymaster = options?.defaultPaymaster;
27+
const defaultPaymasterInput = options?.defaultPaymasterInput;
628

7-
export function viem(client: ViemWalletClient): EVMWalletClient {
829
const publicClient = client.extend(publicActions);
930

31+
const waitForReceipt = async (hash: `0x${string}`) => {
32+
const receipt = await publicClient.waitForTransactionReceipt({ hash });
33+
return { hash: receipt.transactionHash, status: receipt.status };
34+
};
35+
1036
return {
1137
getAddress: () => client.account?.address ?? "",
1238
getChain() {
@@ -16,9 +42,8 @@ export function viem(client: ViemWalletClient): EVMWalletClient {
1642
};
1743
},
1844
async resolveAddress(address: string) {
19-
if (/^0x[a-fA-F0-9]{40}$/.test(address)) {
45+
if (/^0x[a-fA-F0-9]{40}$/.test(address))
2046
return address as `0x${string}`;
21-
}
2247

2348
try {
2449
const resolvedAddress = (await publicClient.getEnsAddress({
@@ -39,9 +64,7 @@ export function viem(client: ViemWalletClient): EVMWalletClient {
3964
account: client.account,
4065
});
4166

42-
return {
43-
signature,
44-
};
67+
return { signature };
4568
},
4669
async signTypedData(data: EVMTypedData) {
4770
if (!client.account) throw new Error("No account connected");
@@ -54,68 +77,82 @@ export function viem(client: ViemWalletClient): EVMWalletClient {
5477
account: client.account,
5578
});
5679

57-
return {
58-
signature,
59-
};
80+
return { signature };
6081
},
6182
async sendTransaction(transaction: EVMTransaction) {
62-
const { to, abi, functionName, args, value } = transaction;
83+
const { to, abi, functionName, args, value, options } = transaction;
84+
if (!client.account) throw new Error("No account connected");
6385

6486
const toAddress = await this.resolveAddress(to);
65-
if (!client.account) throw new Error("No account connected");
6687

88+
const paymaster = options?.paymaster ?? defaultPaymaster;
89+
const paymasterInput =
90+
options?.paymasterInput ?? defaultPaymasterInput;
91+
const isPaymasterTx = !!paymaster || !!paymasterInput;
92+
93+
// If paymaster params exist, extend with EIP712 actions
94+
const sendingClient = isPaymasterTx
95+
? client.extend(eip712WalletActions())
96+
: client;
97+
98+
// Simple ETH transfer (no ABI)
6799
if (!abi) {
68-
const tx = await client.sendTransaction({
100+
const txParams = {
69101
account: client.account,
70102
to: toAddress,
71103
chain: client.chain,
72104
value,
73-
});
74-
75-
const transaction = await publicClient.waitForTransactionReceipt({
76-
hash: tx,
77-
});
78-
79-
return {
80-
hash: transaction.transactionHash,
81-
status: transaction.status,
105+
...(isPaymasterTx ? { paymaster, paymasterInput } : {}),
82106
};
107+
108+
const txHash = await sendingClient.sendTransaction(txParams);
109+
return waitForReceipt(txHash);
83110
}
84111

112+
// Contract call
85113
if (!functionName) {
86-
throw new Error("Function name is required");
114+
throw new Error("Function name is required for contract calls");
87115
}
88116

89-
await publicClient.simulateContract({
117+
const { request } = await publicClient.simulateContract({
90118
account: client.account,
91119
address: toAddress,
92-
abi,
120+
abi: abi,
93121
functionName,
94122
args,
95123
chain: client.chain,
96124
});
97-
const hash = await client.writeContract({
98-
account: client.account,
99-
address: toAddress,
100-
abi,
125+
126+
// Encode the call data ourselves
127+
const data = encodeFunctionData({
128+
abi: abi,
101129
functionName,
102130
args,
103-
chain: client.chain,
104-
value,
105131
});
106132

107-
const t = await publicClient.waitForTransactionReceipt({
108-
hash: hash,
109-
});
133+
if (isPaymasterTx) {
134+
// With paymaster, we must use sendTransaction() directly
135+
const txParams = {
136+
account: client.account,
137+
chain: client.chain,
138+
to: request.address,
139+
data,
140+
value: request.value,
141+
paymaster,
142+
paymasterInput,
143+
};
144+
const txHash = await sendingClient.sendTransaction(txParams);
145+
return waitForReceipt(txHash);
146+
}
110147

111-
return {
112-
hash: t.transactionHash,
113-
status: t.status,
114-
};
148+
// Without paymaster, use writeContract which handles encoding too,
149+
// but since we already have request, let's let writeContract do its thing.
150+
// However, writeContract expects the original request format (with abi, functionName, args).
151+
const txHash = await client.writeContract(request);
152+
return waitForReceipt(txHash);
115153
},
116154
async read(request: EVMReadRequest) {
117155
const { address, abi, functionName, args } = request;
118-
119156
if (!abi) throw new Error("Read request must include ABI for EVM");
120157

121158
const result = await publicClient.readContract({
@@ -125,22 +162,21 @@ export function viem(client: ViemWalletClient): EVMWalletClient {
125162
args,
126163
});
127164

128-
return {
129-
value: result,
130-
};
165+
return { value: result };
131166
},
132167
async balanceOf(address: string) {
133168
const resolvedAddress = await this.resolveAddress(address);
134-
135169
const balance = await publicClient.getBalance({
136170
address: resolvedAddress,
137171
});
138172

173+
const chain = client.chain ?? mainnet;
174+
139175
return {
140176
value: balance,
141-
decimals: 18,
142-
symbol: "ETH",
143-
name: "Ether",
177+
decimals: chain.nativeCurrency.decimals,
178+
symbol: chain.nativeCurrency.symbol,
179+
name: chain.nativeCurrency.name,
144180
};
145181
},
146182
};

typescript/pnpm-lock.yaml

+27
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)