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" ;
5
13
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 ;
6
28
7
- export function viem ( client : ViemWalletClient ) : EVMWalletClient {
8
29
const publicClient = client . extend ( publicActions ) ;
9
30
31
+ const waitForReceipt = async ( hash : `0x${string } `) => {
32
+ const receipt = await publicClient . waitForTransactionReceipt ( { hash } ) ;
33
+ return { hash : receipt . transactionHash , status : receipt . status } ;
34
+ } ;
35
+
10
36
return {
11
37
getAddress : ( ) => client . account ?. address ?? "" ,
12
38
getChain ( ) {
@@ -16,9 +42,8 @@ export function viem(client: ViemWalletClient): EVMWalletClient {
16
42
} ;
17
43
} ,
18
44
async resolveAddress ( address : string ) {
19
- if ( / ^ 0 x [ a - f A - F 0 - 9 ] { 40 } $ / . test ( address ) ) {
45
+ if ( / ^ 0 x [ a - f A - F 0 - 9 ] { 40 } $ / . test ( address ) )
20
46
return address as `0x${string } `;
21
- }
22
47
23
48
try {
24
49
const resolvedAddress = ( await publicClient . getEnsAddress ( {
@@ -39,9 +64,7 @@ export function viem(client: ViemWalletClient): EVMWalletClient {
39
64
account : client . account ,
40
65
} ) ;
41
66
42
- return {
43
- signature,
44
- } ;
67
+ return { signature } ;
45
68
} ,
46
69
async signTypedData ( data : EVMTypedData ) {
47
70
if ( ! client . account ) throw new Error ( "No account connected" ) ;
@@ -54,68 +77,82 @@ export function viem(client: ViemWalletClient): EVMWalletClient {
54
77
account : client . account ,
55
78
} ) ;
56
79
57
- return {
58
- signature,
59
- } ;
80
+ return { signature } ;
60
81
} ,
61
82
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" ) ;
63
85
64
86
const toAddress = await this . resolveAddress ( to ) ;
65
- if ( ! client . account ) throw new Error ( "No account connected" ) ;
66
87
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)
67
99
if ( ! abi ) {
68
- const tx = await client . sendTransaction ( {
100
+ const txParams = {
69
101
account : client . account ,
70
102
to : toAddress ,
71
103
chain : client . chain ,
72
104
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 } : { } ) ,
82
106
} ;
107
+
108
+ const txHash = await sendingClient . sendTransaction ( txParams ) ;
109
+ return waitForReceipt ( txHash ) ;
83
110
}
84
111
112
+ // Contract call
85
113
if ( ! functionName ) {
86
- throw new Error ( "Function name is required" ) ;
114
+ throw new Error ( "Function name is required for contract calls " ) ;
87
115
}
88
116
89
- await publicClient . simulateContract ( {
117
+ const { request } = await publicClient . simulateContract ( {
90
118
account : client . account ,
91
119
address : toAddress ,
92
- abi,
120
+ abi : abi ,
93
121
functionName,
94
122
args,
95
123
chain : client . chain ,
96
124
} ) ;
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 ,
101
129
functionName,
102
130
args,
103
- chain : client . chain ,
104
- value,
105
131
} ) ;
106
132
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
+ }
110
147
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 ) ;
115
153
} ,
116
154
async read ( request : EVMReadRequest ) {
117
155
const { address, abi, functionName, args } = request ;
118
-
119
156
if ( ! abi ) throw new Error ( "Read request must include ABI for EVM" ) ;
120
157
121
158
const result = await publicClient . readContract ( {
@@ -125,22 +162,21 @@ export function viem(client: ViemWalletClient): EVMWalletClient {
125
162
args,
126
163
} ) ;
127
164
128
- return {
129
- value : result ,
130
- } ;
165
+ return { value : result } ;
131
166
} ,
132
167
async balanceOf ( address : string ) {
133
168
const resolvedAddress = await this . resolveAddress ( address ) ;
134
-
135
169
const balance = await publicClient . getBalance ( {
136
170
address : resolvedAddress ,
137
171
} ) ;
138
172
173
+ const chain = client . chain ?? mainnet ;
174
+
139
175
return {
140
176
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 ,
144
180
} ;
145
181
} ,
146
182
} ;
0 commit comments