eden2query 0.3.8 → 0.3.10

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -1,418 +1,418 @@
1
- # eden2query
2
-
3
- Type-safe [Eden Treaty](https://elysiajs.com/eden/overview) to [React Query](https://tanstack.com/query) helpers. Create fully type-safe React Query hooks from your Elysia API with zero runtime overhead.
4
-
5
- ## Install
6
-
7
- ```bash
8
- bun add eden2query @elysiajs/eden @tanstack/react-query
9
- ```
10
-
11
- ## Quick Start
12
-
13
- ### 1. Define your Elysia API
14
-
15
- ```ts
16
- import Elysia, { t } from "elysia";
17
-
18
- const app = new Elysia({ prefix: "/api" })
19
- .get("/users/:id", ({ params, query }) => {
20
- return { user: { id: params.id, name: query.name } };
21
- }, {
22
- params: t.Object({ id: t.String() }),
23
- query: t.Object({ name: t.String() }),
24
- response: { 200: t.Object({ user: t.Object({ id: t.String(), name: t.String() }) }) },
25
- })
26
- .post("/users", ({ body }) => {
27
- return { id: "123", ...body };
28
- }, {
29
- body: t.Object({ name: t.String(), email: t.String() }),
30
- response: { 200: t.Object({ id: t.String(), name: t.String(), email: t.String() }) },
31
- })
32
- .put("/users/:id", ({ params, body }) => {
33
- return { id: params.id, ...body };
34
- }, {
35
- params: t.Object({ id: t.String() }),
36
- body: t.Object({ name: t.String() }),
37
- response: { 200: t.Object({ id: t.String(), name: t.String() }) },
38
- })
39
- .delete("/users/:id", ({ params }) => {
40
- return { deleted: params.id };
41
- }, {
42
- params: t.Object({ id: t.String() }),
43
- response: { 200: t.Object({ deleted: t.String() }) },
44
- });
45
- ```
46
-
47
- ### 2. Create Treaty Client
48
-
49
- ```ts
50
- import { treaty } from "@elysiajs/eden";
51
-
52
- const client = treaty<typeof app>("http://localhost:3000");
53
- ```
54
-
55
- ### 3. Create Type-Safe Hooks
56
-
57
- ```ts
58
- import { treatyQueryHook, treatyMutationHook } from "eden2query";
59
- import { useQueryClient } from "@tanstack/react-query";
60
-
61
- // GET - Query Hook (with params)
62
- function getUser(params: { id: string }) {
63
- return client.api.users({ id: params.id }).get;
64
- }
65
-
66
- const useGetUser = treatyQueryHook(getUser, (request) => {
67
- return {
68
- queryKey: ["users", request.params.id],
69
- };
70
- });
71
-
72
- // POST - Mutation Hook
73
- function createUser() {
74
- return client.api.users.post;
75
- }
76
-
77
- const useCreateUser = treatyMutationHook(createUser, () => {
78
- const queryClient = useQueryClient();
79
- return {
80
- onSuccess: () => {
81
- queryClient.invalidateQueries({ queryKey: ["users"] });
82
- },
83
- };
84
- });
85
-
86
- // PUT - Mutation Hook with Params
87
- function updateUser(params: { id: string }) {
88
- return client.api.users({ id: params.id }).put;
89
- }
90
-
91
- const useUpdateUser = treatyMutationHook(updateUser, () => {
92
- const queryClient = useQueryClient();
93
- return {
94
- onSuccess: () => {
95
- queryClient.invalidateQueries({ queryKey: ["users"] });
96
- },
97
- };
98
- });
99
-
100
- // DELETE - Mutation Hook with Params (no body)
101
- function deleteUser(params: { id: string }) {
102
- return client.api.users({ id: params.id }).delete;
103
- }
104
-
105
- const useDeleteUser = treatyMutationHook(deleteUser, () => {
106
- const queryClient = useQueryClient();
107
- return {
108
- onSuccess: () => {
109
- queryClient.invalidateQueries({ queryKey: ["users"] });
110
- },
111
- };
112
- });
113
- ```
114
-
115
- ### 4. Use in Components
116
-
117
- ```tsx
118
- function UserProfile({ userId }: { userId: string }) {
119
- // Query - automatically typed
120
- const { data, isLoading } = useGetUser({
121
- params: { id: userId },
122
- query: { name: "John" },
123
- });
124
-
125
- // Mutations - fully typed
126
- const createMutation = useCreateUser({
127
- onSuccess: (data) => {
128
- console.log("Created:", data);
129
- },
130
- });
131
-
132
- const updateMutation = useUpdateUser({
133
- onSuccess: (data) => {
134
- console.log("Updated:", data);
135
- },
136
- });
137
-
138
- const deleteMutation = useDeleteUser({
139
- onSuccess: (data) => {
140
- console.log("Deleted:", data);
141
- },
142
- });
143
-
144
- return (
145
- <div>
146
- {isLoading ? "Loading..." : <div>{data?.user.name}</div>}
147
-
148
- <button onClick={() => createMutation.mutate({
149
- body: { name: "New User", email: "user@example.com" },
150
- })}>
151
- Create User
152
- </button>
153
-
154
- <button onClick={() => updateMutation.mutate({
155
- params: { id: userId },
156
- body: { name: "Updated Name" },
157
- })}>
158
- Update User
159
- </button>
160
-
161
- <button onClick={() => deleteMutation.mutate({
162
- params: { id: userId },
163
- // body is not required for DELETE mutations
164
- })}>
165
- Delete User
166
- </button>
167
- </div>
168
- );
169
- }
170
- ```
171
-
172
- ## API Reference
173
-
174
- ### `treatyQueryHook`
175
-
176
- Creates a type-safe React Query hook for GET requests.
177
-
178
- **Signature:**
179
- ```ts
180
- function treatyQueryHook<TParams, TOption extends object, TResponse extends Record<number, unknown>>(
181
- fn: TreatyQueryFunctionFactory<TParams, TOption, TResponse>,
182
- useOptions: (request: TreatyQueryRequest<TParams, TOption>) => Omit<UseQueryOptions<TResponse>, "queryFn">
183
- ): (request: TreatyQueryRequest<TParams, TOption>, options?: UseQueryOptions) => UseQueryResult
184
- ```
185
-
186
- **Parameters:**
187
- - `fn`: A factory function that accepts params (if route has params) or no params (if route doesn't have params), and returns a Treaty query function
188
- - `useOptions`: A function that receives the request and returns query options (including `queryKey`)
189
-
190
- **Returns:** A hook function that accepts a request object and optional query options
191
-
192
- **Example:**
193
- ```ts
194
- // With params
195
- function getUser(params: { id: string }) {
196
- return client.api.users({ id: params.id }).get;
197
- }
198
-
199
- const useGetUser = treatyQueryHook(getUser, (request) => ({
200
- queryKey: ["users", request.params.id], // params is required when route has params
201
- staleTime: 5000,
202
- }));
203
-
204
- // Usage
205
- const { data } = useGetUser({
206
- params: { id: "123" }, // Required when route has params
207
- query: { name: "John" },
208
- });
209
-
210
- // Without params
211
- function getUsers() {
212
- return client.api.users.get;
213
- }
214
-
215
- const useGetUsers = treatyQueryHook(getUsers, (request) => ({
216
- queryKey: ["users"],
217
- }));
218
-
219
- // Usage - params are absent from the type
220
- const { data } = useGetUsers({
221
- query: { page: "1" },
222
- // params is not part of the request type
223
- });
224
- ```
225
-
226
- ### `treatyMutationHook`
227
-
228
- Creates a type-safe React Query hook for mutations (POST, PUT, DELETE, etc.).
229
-
230
- **Signature:**
231
- ```ts
232
- function treatyMutationHook<TParams, TBody, TOption, TResponse extends Record<number, unknown>>(
233
- fn: TreatyMutationFunctionFactory<TParams, TBody, TOption, TResponse>,
234
- useOptions: () => Omit<UseMutationOptions<TParams, TBody, TOption, TResponse>, "mutationFn">
235
- ): (options?: UseMutationOptions) => UseMutationResult
236
- ```
237
-
238
- **Parameters:**
239
- - `fn`: A factory function that accepts params (if route has params) or no params (if route doesn't have params), and returns a Treaty mutation function
240
- - `useOptions`: A hook function that returns default mutation options (can use React hooks like `useQueryClient`)
241
-
242
- **Returns:** A hook function that accepts optional mutation options
243
-
244
- **Example:**
245
- ```ts
246
- // Without params, with body
247
- function createUser() {
248
- return client.api.users.post;
249
- }
250
-
251
- const useCreateUser = treatyMutationHook(createUser, () => {
252
- const queryClient = useQueryClient();
253
- return {
254
- onSuccess: () => {
255
- queryClient.invalidateQueries({ queryKey: ["users"] });
256
- },
257
- };
258
- });
259
-
260
- // Usage
261
- const mutation = useCreateUser({
262
- onSuccess: (data) => console.log(data),
263
- });
264
-
265
- mutation.mutate({
266
- body: { name: "John", email: "john@example.com" },
267
- // params is not part of the request type
268
- });
269
-
270
- // With params and body
271
- function updateUser(params: { id: string }) {
272
- return client.api.users({ id: params.id }).put;
273
- }
274
-
275
- const useUpdateUser = treatyMutationHook(updateUser, () => ({
276
- onSuccess: () => console.log("Updated"),
277
- }));
278
-
279
- useUpdateUser().mutate({
280
- params: { id: "123" },
281
- body: { name: "Updated Name" },
282
- });
283
-
284
- // With params, without body
285
- function deleteUser(params: { id: string }) {
286
- return client.api.users({ id: params.id }).delete;
287
- }
288
-
289
- const useDeleteUser = treatyMutationHook(deleteUser, () => ({
290
- onSuccess: () => console.log("Deleted"),
291
- }));
292
-
293
- useDeleteUser().mutate({
294
- params: { id: "123" },
295
- // body is not part of the request type
296
- });
297
- ```
298
-
299
- ## Type Safety
300
-
301
- All types are inferred end-to-end from your Elysia route definitions:
302
-
303
- - **Request types**: Automatically inferred from route params, query, and body schemas
304
- - `params` and `body` are **conditionally included** in request types - they're absent (not optional) when the route doesn't require them
305
- - This means you don't need to pass `undefined` or `unknown` values - the fields simply don't exist in the type
306
- - **Response types**: Extracted from your response schema definitions
307
- - **Error types**: Typed based on your error response schemas
308
- - **Query keys**: Type-safe based on your request structure
309
-
310
- ## Patterns
311
-
312
- ### Routes with Params
313
-
314
- For routes like `/users/:id`, params are **required** and included in the request type:
315
-
316
- ```ts
317
- function getUser(params: { id: string }) {
318
- return client.api.users({ id: params.id }).get;
319
- }
320
-
321
- const useGetUser = treatyQueryHook(getUser, (request) => ({
322
- queryKey: ["users", request.params.id], // params is required here
323
- }));
324
-
325
- // Must provide params
326
- useGetUser({
327
- params: { id: "123" }, // ✅ Required - params is part of the request type
328
- query: { name: "John" },
329
- });
330
- ```
331
-
332
- ### Routes without Params
333
-
334
- For routes without params, params are **absent** from the request type (not optional):
335
-
336
- ```ts
337
- function getUsers() {
338
- return client.api.users.get;
339
- }
340
-
341
- const useGetUsers = treatyQueryHook(getUsers, (request) => ({
342
- queryKey: ["users"],
343
- }));
344
-
345
- // Params are absent from the type - you cannot include them
346
- useGetUsers({
347
- query: { page: "1" },
348
- // params is not part of the request type at all
349
- });
350
- ```
351
-
352
- ### Mutations with Body (no params)
353
-
354
- POST mutations without route params include a `body` field:
355
-
356
- ```ts
357
- function createUser() {
358
- return client.api.users.post;
359
- }
360
-
361
- const useCreateUser = treatyMutationHook(createUser, () => ({}));
362
-
363
- const mutation = useCreateUser();
364
- mutation.mutate({
365
- body: { name: "John", email: "john@example.com" },
366
- // params is not part of the request type
367
- });
368
- ```
369
-
370
- ### Mutations with Params and Body
371
-
372
- PUT mutations can have both params and body:
373
-
374
- ```ts
375
- function updateUser(params: { id: string }) {
376
- return client.api.users({ id: params.id }).put;
377
- }
378
-
379
- const useUpdateUser = treatyMutationHook(updateUser, () => ({}));
380
-
381
- const mutation = useUpdateUser();
382
- mutation.mutate({
383
- params: { id: "123" }, // Required - params is part of the request type
384
- body: { name: "Updated Name" }, // Required - body is part of the request type
385
- });
386
- ```
387
-
388
- ### Mutations with Only Params (no body)
389
-
390
- DELETE mutations typically only have params, and body is **absent** from the request type:
391
-
392
- ```ts
393
- function deleteUser(params: { id: string }) {
394
- return client.api.users({ id: params.id }).delete;
395
- }
396
-
397
- const useDeleteUser = treatyMutationHook(deleteUser, () => ({}));
398
-
399
- const mutation = useDeleteUser();
400
- mutation.mutate({
401
- params: { id: "123" }, // Required - params is part of the request type
402
- // body is not part of the request type at all
403
- });
404
- ```
405
-
406
- ## Options Merging
407
-
408
- Options are merged in this order (later options override earlier ones):
409
-
410
- 1. Base options from `useOptions` hook
411
- 2. Options passed to the hook factory
412
- 3. Options passed when calling the hook
413
-
414
- Callbacks (`onSuccess`, `onError`, etc.) are **chained** - all callbacks run in order.
415
-
416
- ## License
417
-
418
- MIT
1
+ # eden2query
2
+
3
+ Type-safe [Eden Treaty](https://elysiajs.com/eden/overview) to [React Query](https://tanstack.com/query) helpers. Create fully type-safe React Query hooks from your Elysia API with zero runtime overhead.
4
+
5
+ ## Install
6
+
7
+ ```bash
8
+ bun add eden2query @elysiajs/eden @tanstack/react-query
9
+ ```
10
+
11
+ ## Quick Start
12
+
13
+ ### 1. Define your Elysia API
14
+
15
+ ```ts
16
+ import Elysia, { t } from "elysia";
17
+
18
+ const app = new Elysia({ prefix: "/api" })
19
+ .get("/users/:id", ({ params, query }) => {
20
+ return { user: { id: params.id, name: query.name } };
21
+ }, {
22
+ params: t.Object({ id: t.String() }),
23
+ query: t.Object({ name: t.String() }),
24
+ response: { 200: t.Object({ user: t.Object({ id: t.String(), name: t.String() }) }) },
25
+ })
26
+ .post("/users", ({ body }) => {
27
+ return { id: "123", ...body };
28
+ }, {
29
+ body: t.Object({ name: t.String(), email: t.String() }),
30
+ response: { 200: t.Object({ id: t.String(), name: t.String(), email: t.String() }) },
31
+ })
32
+ .put("/users/:id", ({ params, body }) => {
33
+ return { id: params.id, ...body };
34
+ }, {
35
+ params: t.Object({ id: t.String() }),
36
+ body: t.Object({ name: t.String() }),
37
+ response: { 200: t.Object({ id: t.String(), name: t.String() }) },
38
+ })
39
+ .delete("/users/:id", ({ params }) => {
40
+ return { deleted: params.id };
41
+ }, {
42
+ params: t.Object({ id: t.String() }),
43
+ response: { 200: t.Object({ deleted: t.String() }) },
44
+ });
45
+ ```
46
+
47
+ ### 2. Create Treaty Client
48
+
49
+ ```ts
50
+ import { treaty } from "@elysiajs/eden";
51
+
52
+ const client = treaty<typeof app>("http://localhost:3000");
53
+ ```
54
+
55
+ ### 3. Create Type-Safe Hooks
56
+
57
+ ```ts
58
+ import { treatyQueryHook, treatyMutationHook } from "eden2query";
59
+ import { useQueryClient } from "@tanstack/react-query";
60
+
61
+ // GET - Query Hook (with params)
62
+ function getUser(params: { id: string }) {
63
+ return client.api.users({ id: params.id }).get;
64
+ }
65
+
66
+ const useGetUser = treatyQueryHook(getUser, (request) => {
67
+ return {
68
+ queryKey: ["users", request.params.id],
69
+ };
70
+ });
71
+
72
+ // POST - Mutation Hook
73
+ function createUser() {
74
+ return client.api.users.post;
75
+ }
76
+
77
+ const useCreateUser = treatyMutationHook(createUser, () => {
78
+ const queryClient = useQueryClient();
79
+ return {
80
+ onSuccess: () => {
81
+ queryClient.invalidateQueries({ queryKey: ["users"] });
82
+ },
83
+ };
84
+ });
85
+
86
+ // PUT - Mutation Hook with Params
87
+ function updateUser(params: { id: string }) {
88
+ return client.api.users({ id: params.id }).put;
89
+ }
90
+
91
+ const useUpdateUser = treatyMutationHook(updateUser, () => {
92
+ const queryClient = useQueryClient();
93
+ return {
94
+ onSuccess: () => {
95
+ queryClient.invalidateQueries({ queryKey: ["users"] });
96
+ },
97
+ };
98
+ });
99
+
100
+ // DELETE - Mutation Hook with Params (no body)
101
+ function deleteUser(params: { id: string }) {
102
+ return client.api.users({ id: params.id }).delete;
103
+ }
104
+
105
+ const useDeleteUser = treatyMutationHook(deleteUser, () => {
106
+ const queryClient = useQueryClient();
107
+ return {
108
+ onSuccess: () => {
109
+ queryClient.invalidateQueries({ queryKey: ["users"] });
110
+ },
111
+ };
112
+ });
113
+ ```
114
+
115
+ ### 4. Use in Components
116
+
117
+ ```tsx
118
+ function UserProfile({ userId }: { userId: string }) {
119
+ // Query - automatically typed
120
+ const { data, isLoading } = useGetUser({
121
+ params: { id: userId },
122
+ query: { name: "John" },
123
+ });
124
+
125
+ // Mutations - fully typed
126
+ const createMutation = useCreateUser({
127
+ onSuccess: (data) => {
128
+ console.log("Created:", data);
129
+ },
130
+ });
131
+
132
+ const updateMutation = useUpdateUser({
133
+ onSuccess: (data) => {
134
+ console.log("Updated:", data);
135
+ },
136
+ });
137
+
138
+ const deleteMutation = useDeleteUser({
139
+ onSuccess: (data) => {
140
+ console.log("Deleted:", data);
141
+ },
142
+ });
143
+
144
+ return (
145
+ <div>
146
+ {isLoading ? "Loading..." : <div>{data?.user.name}</div>}
147
+
148
+ <button onClick={() => createMutation.mutate({
149
+ body: { name: "New User", email: "user@example.com" },
150
+ })}>
151
+ Create User
152
+ </button>
153
+
154
+ <button onClick={() => updateMutation.mutate({
155
+ params: { id: userId },
156
+ body: { name: "Updated Name" },
157
+ })}>
158
+ Update User
159
+ </button>
160
+
161
+ <button onClick={() => deleteMutation.mutate({
162
+ params: { id: userId },
163
+ // body is not required for DELETE mutations
164
+ })}>
165
+ Delete User
166
+ </button>
167
+ </div>
168
+ );
169
+ }
170
+ ```
171
+
172
+ ## API Reference
173
+
174
+ ### `treatyQueryHook`
175
+
176
+ Creates a type-safe React Query hook for GET requests.
177
+
178
+ **Signature:**
179
+ ```ts
180
+ function treatyQueryHook<TParams, TOption extends object, TResponse extends Record<number, unknown>>(
181
+ fn: TreatyQueryFunctionFactory<TParams, TOption, TResponse>,
182
+ useOptions: (request: TreatyQueryRequest<TParams, TOption>) => Omit<UseQueryOptions<TResponse>, "queryFn">
183
+ ): (request: TreatyQueryRequest<TParams, TOption>, options?: UseQueryOptions) => UseQueryResult
184
+ ```
185
+
186
+ **Parameters:**
187
+ - `fn`: A factory function that accepts params (if route has params) or no params (if route doesn't have params), and returns a Treaty query function
188
+ - `useOptions`: A function that receives the request and returns query options (including `queryKey`)
189
+
190
+ **Returns:** A hook function that accepts a request object and optional query options
191
+
192
+ **Example:**
193
+ ```ts
194
+ // With params
195
+ function getUser(params: { id: string }) {
196
+ return client.api.users({ id: params.id }).get;
197
+ }
198
+
199
+ const useGetUser = treatyQueryHook(getUser, (request) => ({
200
+ queryKey: ["users", request.params.id], // params is required when route has params
201
+ staleTime: 5000,
202
+ }));
203
+
204
+ // Usage
205
+ const { data } = useGetUser({
206
+ params: { id: "123" }, // Required when route has params
207
+ query: { name: "John" },
208
+ });
209
+
210
+ // Without params
211
+ function getUsers() {
212
+ return client.api.users.get;
213
+ }
214
+
215
+ const useGetUsers = treatyQueryHook(getUsers, (request) => ({
216
+ queryKey: ["users"],
217
+ }));
218
+
219
+ // Usage - params are absent from the type
220
+ const { data } = useGetUsers({
221
+ query: { page: "1" },
222
+ // params is not part of the request type
223
+ });
224
+ ```
225
+
226
+ ### `treatyMutationHook`
227
+
228
+ Creates a type-safe React Query hook for mutations (POST, PUT, DELETE, etc.).
229
+
230
+ **Signature:**
231
+ ```ts
232
+ function treatyMutationHook<TParams, TBody, TOption, TResponse extends Record<number, unknown>>(
233
+ fn: TreatyMutationFunctionFactory<TParams, TBody, TOption, TResponse>,
234
+ useOptions: () => Omit<UseMutationOptions<TParams, TBody, TOption, TResponse>, "mutationFn">
235
+ ): (options?: UseMutationOptions) => UseMutationResult
236
+ ```
237
+
238
+ **Parameters:**
239
+ - `fn`: A factory function that accepts params (if route has params) or no params (if route doesn't have params), and returns a Treaty mutation function
240
+ - `useOptions`: A hook function that returns default mutation options (can use React hooks like `useQueryClient`)
241
+
242
+ **Returns:** A hook function that accepts optional mutation options
243
+
244
+ **Example:**
245
+ ```ts
246
+ // Without params, with body
247
+ function createUser() {
248
+ return client.api.users.post;
249
+ }
250
+
251
+ const useCreateUser = treatyMutationHook(createUser, () => {
252
+ const queryClient = useQueryClient();
253
+ return {
254
+ onSuccess: () => {
255
+ queryClient.invalidateQueries({ queryKey: ["users"] });
256
+ },
257
+ };
258
+ });
259
+
260
+ // Usage
261
+ const mutation = useCreateUser({
262
+ onSuccess: (data) => console.log(data),
263
+ });
264
+
265
+ mutation.mutate({
266
+ body: { name: "John", email: "john@example.com" },
267
+ // params is not part of the request type
268
+ });
269
+
270
+ // With params and body
271
+ function updateUser(params: { id: string }) {
272
+ return client.api.users({ id: params.id }).put;
273
+ }
274
+
275
+ const useUpdateUser = treatyMutationHook(updateUser, () => ({
276
+ onSuccess: () => console.log("Updated"),
277
+ }));
278
+
279
+ useUpdateUser().mutate({
280
+ params: { id: "123" },
281
+ body: { name: "Updated Name" },
282
+ });
283
+
284
+ // With params, without body
285
+ function deleteUser(params: { id: string }) {
286
+ return client.api.users({ id: params.id }).delete;
287
+ }
288
+
289
+ const useDeleteUser = treatyMutationHook(deleteUser, () => ({
290
+ onSuccess: () => console.log("Deleted"),
291
+ }));
292
+
293
+ useDeleteUser().mutate({
294
+ params: { id: "123" },
295
+ // body is not part of the request type
296
+ });
297
+ ```
298
+
299
+ ## Type Safety
300
+
301
+ All types are inferred end-to-end from your Elysia route definitions:
302
+
303
+ - **Request types**: Automatically inferred from route params, query, and body schemas
304
+ - `params` and `body` are **conditionally included** in request types - they're absent (not optional) when the route doesn't require them
305
+ - This means you don't need to pass `undefined` or `unknown` values - the fields simply don't exist in the type
306
+ - **Response types**: Extracted from your response schema definitions
307
+ - **Error types**: Typed based on your error response schemas
308
+ - **Query keys**: Type-safe based on your request structure
309
+
310
+ ## Patterns
311
+
312
+ ### Routes with Params
313
+
314
+ For routes like `/users/:id`, params are **required** and included in the request type:
315
+
316
+ ```ts
317
+ function getUser(params: { id: string }) {
318
+ return client.api.users({ id: params.id }).get;
319
+ }
320
+
321
+ const useGetUser = treatyQueryHook(getUser, (request) => ({
322
+ queryKey: ["users", request.params.id], // params is required here
323
+ }));
324
+
325
+ // Must provide params
326
+ useGetUser({
327
+ params: { id: "123" }, // ✅ Required - params is part of the request type
328
+ query: { name: "John" },
329
+ });
330
+ ```
331
+
332
+ ### Routes without Params
333
+
334
+ For routes without params, params are **absent** from the request type (not optional):
335
+
336
+ ```ts
337
+ function getUsers() {
338
+ return client.api.users.get;
339
+ }
340
+
341
+ const useGetUsers = treatyQueryHook(getUsers, (request) => ({
342
+ queryKey: ["users"],
343
+ }));
344
+
345
+ // Params are absent from the type - you cannot include them
346
+ useGetUsers({
347
+ query: { page: "1" },
348
+ // params is not part of the request type at all
349
+ });
350
+ ```
351
+
352
+ ### Mutations with Body (no params)
353
+
354
+ POST mutations without route params include a `body` field:
355
+
356
+ ```ts
357
+ function createUser() {
358
+ return client.api.users.post;
359
+ }
360
+
361
+ const useCreateUser = treatyMutationHook(createUser, () => ({}));
362
+
363
+ const mutation = useCreateUser();
364
+ mutation.mutate({
365
+ body: { name: "John", email: "john@example.com" },
366
+ // params is not part of the request type
367
+ });
368
+ ```
369
+
370
+ ### Mutations with Params and Body
371
+
372
+ PUT mutations can have both params and body:
373
+
374
+ ```ts
375
+ function updateUser(params: { id: string }) {
376
+ return client.api.users({ id: params.id }).put;
377
+ }
378
+
379
+ const useUpdateUser = treatyMutationHook(updateUser, () => ({}));
380
+
381
+ const mutation = useUpdateUser();
382
+ mutation.mutate({
383
+ params: { id: "123" }, // Required - params is part of the request type
384
+ body: { name: "Updated Name" }, // Required - body is part of the request type
385
+ });
386
+ ```
387
+
388
+ ### Mutations with Only Params (no body)
389
+
390
+ DELETE mutations typically only have params, and body is **absent** from the request type:
391
+
392
+ ```ts
393
+ function deleteUser(params: { id: string }) {
394
+ return client.api.users({ id: params.id }).delete;
395
+ }
396
+
397
+ const useDeleteUser = treatyMutationHook(deleteUser, () => ({}));
398
+
399
+ const mutation = useDeleteUser();
400
+ mutation.mutate({
401
+ params: { id: "123" }, // Required - params is part of the request type
402
+ // body is not part of the request type at all
403
+ });
404
+ ```
405
+
406
+ ## Options Merging
407
+
408
+ Options are merged in this order (later options override earlier ones):
409
+
410
+ 1. Base options from `useOptions` hook
411
+ 2. Options passed to the hook factory
412
+ 3. Options passed when calling the hook
413
+
414
+ Callbacks (`onSuccess`, `onError`, etc.) are **chained** - all callbacks run in order.
415
+
416
+ ## License
417
+
418
+ MIT
package/dist/index.d.ts CHANGED
@@ -1,6 +1,6 @@
1
1
  import { Treaty } from "@elysiajs/eden";
2
2
  import { UseMutationOptions, UseMutationResult, UseQueryOptions, UseQueryResult } from "@tanstack/react-query";
3
- type IsEmpty<T> = [T] extends [never | undefined | void] ? true : [unknown] extends [T] ? [T] extends [unknown] ? true : false : false;
3
+ type IsEmpty<T> = [T] extends [never | undefined | void] ? true : [unknown] extends [T] ? [T] extends [unknown] ? true : false : [T] extends [object] ? [keyof T] extends [never] ? true : false : false;
4
4
  type TreatyMutationFunctionFactory<
5
5
  TParams,
6
6
  TBody,
@@ -16,9 +16,13 @@ type TreatyMutationRequest<
16
16
  TParams,
17
17
  TBody,
18
18
  TOption
19
- > = TOption & (IsEmpty<TParams> extends true ? {} : {
19
+ > = TOption & (IsEmpty<TParams> extends true ? {
20
+ params?: TParams;
21
+ } : {
20
22
  params: TParams;
21
- }) & (IsEmpty<TBody> extends true ? {} : {
23
+ }) & (IsEmpty<TBody> extends true ? {
24
+ body?: TBody;
25
+ } : {
22
26
  body: TBody;
23
27
  });
24
28
  type TreatyData<TResponse extends Record<number, unknown>> = Treaty.TreatyResponse<TResponse>["data"];
@@ -53,7 +57,9 @@ type TreatyQueryFunctionFactory<
53
57
  type TreatyQueryRequest<
54
58
  TParams,
55
59
  TOption extends object
56
- > = TOption & (IsEmpty<TParams> extends true ? {} : {
60
+ > = TOption & (IsEmpty<TParams> extends true ? {
61
+ params?: TParams;
62
+ } : {
57
63
  params: TParams;
58
64
  });
59
65
  type TreatyQueryOptions<TResponse extends Record<number, unknown>> = UseQueryOptions<TreatyData<TResponse>, TreatyError<TResponse>>;
package/dist/index.js CHANGED
@@ -48,7 +48,7 @@ function treatyMutationHook(fn, useOptions) {
48
48
  ...mergeMutationOptions(baseOptions, options),
49
49
  mutationFn: async (request) => {
50
50
  const mutationFn = "params" in request && request.params !== undefined ? fn(request.params) : fn();
51
- const body = "body" in request && request.body !== undefined ? request.body : undefined;
51
+ const body = "body" in request && request.body !== undefined ? request.body : {};
52
52
  const { data, error } = await mutationFn(body, request);
53
53
  if (error)
54
54
  throw error;
@@ -78,5 +78,5 @@ export {
78
78
  treatyMutationHook
79
79
  };
80
80
 
81
- //# debugId=89C48BF36DA4F8BA64756E2164756E21
82
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src\\treaty.ts", "src\\utils.ts"],
  "sourcesContent": [
    "import type { Treaty } from \"@elysiajs/eden\";\r\nimport {\r\n  useMutation,\r\n  useQuery,\r\n  type QueryKey,\r\n  type UseMutationOptions,\r\n  type UseMutationResult,\r\n  type UseQueryOptions,\r\n  type UseQueryResult,\r\n} from \"@tanstack/react-query\";\r\nimport { mergeMutationOptions } from \"./utils\";\r\n\r\n// Helper type to check if a type represents \"no params\" (absent/empty)\r\n// Checks for never, undefined, void, and also handles unknown when it represents an unconstrained type\r\ntype IsEmpty<T> = \r\n  // Check for never, undefined, void\r\n  [T] extends [never | undefined | void] \r\n    ? true\r\n    // Check if T is unknown (unconstrained type parameter)\r\n    : [unknown] extends [T]\r\n      ? [T] extends [unknown]\r\n        ? true\r\n        : false\r\n      : false;\r\n\r\ntype TreatyMutationFunctionFactory<\r\n  TParams,\r\n  TBody,\r\n  TOption,\r\n  TResponse extends Record<number, unknown>,\r\n> = IsEmpty<TParams> extends true\r\n  ? () => TreatyMutationFunction<TBody, TOption, TResponse> \r\n  : (params: TParams) => TreatyMutationFunction<TBody, TOption, TResponse>;\r\n\r\ntype TreatyMutationFunction<\r\n  TBody,\r\n  TOption,\r\n  TResponse extends Record<number, unknown>,\r\n> = (\r\n  body: TBody,\r\n  options: TOption,\r\n) => Promise<Treaty.TreatyResponse<TResponse>>;\r\n\r\ntype TreatyMutationRequest<\r\n  TParams,\r\n  TBody,\r\n  TOption,\r\n> = TOption &\r\n  (IsEmpty<TParams> extends true ? {} : { params: TParams }) &\r\n  (IsEmpty<TBody> extends true ? {} : { body: TBody });\r\n\r\ntype TreatyData<TResponse extends Record<number, unknown>> =\r\n  Treaty.TreatyResponse<TResponse>[\"data\"];\r\n\r\ntype TreatyError<TResponse extends Record<number, unknown>> =\r\n  Treaty.TreatyResponse<TResponse>[\"error\"];\r\n\r\ntype TreatyMutationOptions<\r\n  TParams,\r\n  TBody,\r\n  TOption,\r\n  TResponse extends Record<number, unknown>,\r\n> = UseMutationOptions<\r\n  TreatyData<TResponse>,\r\n  TreatyError<TResponse>,\r\n  TreatyMutationRequest<TParams, TBody, TOption>\r\n>;\r\n\r\ntype TreatyMutationHook<\r\n  TParams,\r\n  TBody,\r\n  TOption,\r\n  TResponse extends Record<number, unknown>,\r\n> = (\r\n  options?: Omit<\r\n    TreatyMutationOptions<TParams, TBody, TOption, TResponse>,\r\n    \"mutationFn\"\r\n  >,\r\n) => UseMutationResult<\r\n  TreatyData<TResponse>,\r\n  TreatyError<TResponse>,\r\n  TreatyMutationRequest<TParams, TBody, TOption>\r\n>;\r\n\r\nexport function treatyMutationHook<\r\n  TParams,\r\n  TBody,\r\n  TOption,\r\n  TResponse extends Record<number, unknown>,\r\n>(\r\n  fn: TreatyMutationFunctionFactory<TParams, TBody, TOption, TResponse>,\r\n  useOptions?: () => Omit<\r\n    TreatyMutationOptions<TParams, TBody, TOption, TResponse>,\r\n    \"mutationFn\"\r\n  >,\r\n): TreatyMutationHook<TParams, TBody, TOption, TResponse> {\r\n  return (options) => {\r\n    const baseOptions = useOptions?.() ?? {};\r\n    return useMutation({\r\n      ...mergeMutationOptions(baseOptions, options),\r\n      mutationFn: async (request) => {\r\n        // Handle optional params: if params exists, pass it; otherwise call fn() without args\r\n        const mutationFn =\r\n          \"params\" in request && request.params !== undefined\r\n            ? (fn as (params: TParams) => TreatyMutationFunction<TBody, TOption, TResponse>)(request.params)\r\n            : (fn as () => TreatyMutationFunction<TBody, TOption, TResponse>)();\r\n        \r\n        // Handle optional body: if body exists, pass it; otherwise pass undefined\r\n        const body = \"body\" in request && request.body !== undefined ? request.body : (undefined as TBody);\r\n        const { data, error } = await mutationFn(body, request);\r\n        if (error) throw error;\r\n        return data;\r\n      },\r\n    });\r\n  };\r\n}\r\n\r\ntype TreatyQueryFunction<\r\n  TOption extends object,\r\n  TResponse extends Record<number, unknown>,\r\n> = (options: TOption) => Promise<Treaty.TreatyResponse<TResponse>>;\r\n\r\ntype TreatyQueryFunctionFactory<\r\n  TParams,\r\n  TOption extends object,\r\n  TResponse extends Record<number, unknown>,\r\n> = IsEmpty<TParams> extends true\r\n  ? () => TreatyQueryFunction<TOption, TResponse>\r\n  : (params: TParams) => TreatyQueryFunction<TOption, TResponse>;\r\n\r\ntype TreatyQueryRequest<\r\n  TParams,\r\n  TOption extends object,\r\n> = TOption &\r\n  (IsEmpty<TParams> extends true ? {} : { params: TParams });\r\n\r\ntype TreatyQueryOptions<TResponse extends Record<number, unknown>> =\r\n  UseQueryOptions<TreatyData<TResponse>, TreatyError<TResponse>>;\r\n\r\ntype TreatyQueryHook<\r\n  TParams,\r\n  TOption extends object,\r\n  TResponse extends Record<number, unknown>,\r\n> = (\r\n  request: TreatyQueryRequest<TParams, TOption>,\r\n  options?: Omit<TreatyQueryOptions<TResponse>, \"queryKey\" | \"queryFn\">,\r\n) => UseQueryResult<TreatyData<TResponse>, TreatyError<TResponse>>;\r\n\r\nexport function treatyQueryHook<\r\n  TParams,\r\n  TOption extends object,\r\n  TResponse extends Record<number, unknown>,\r\n>(\r\n  fn: TreatyQueryFunctionFactory<TParams, TOption, TResponse>,\r\n  useOptions: (\r\n    request: TreatyQueryRequest<TParams, TOption>,\r\n  ) => Omit<TreatyQueryOptions<TResponse>, \"queryFn\">,\r\n): TreatyQueryHook<TParams, TOption, TResponse> {\r\n  return (request, options) => {\r\n    // Handle optional params: if params exists, pass it; otherwise call fn() without args\r\n    const queryFn =\r\n      \"params\" in request && request.params !== undefined\r\n        ? (fn as (params: TParams) => TreatyQueryFunction<TOption, TResponse>)(request.params)\r\n        : (fn as () => TreatyQueryFunction<TOption, TResponse>)();\r\n    const baseOptions = useOptions(request);\r\n    return useQuery({\r\n      ...options,\r\n      ...baseOptions,\r\n      queryFn: async () => {\r\n        const { data, error } = await queryFn(request);\r\n        if (error) throw error;\r\n        return data;\r\n      },\r\n    });\r\n  };\r\n}\r\n",
    "import type {\r\n  QueryKey,\r\n  UseMutationOptions,\r\n  UseQueryOptions,\r\n} from \"@tanstack/react-query\";\r\n\r\nexport function mergeMutationOptions<TData, TError, TVariables, TContext>(\r\n  base: UseMutationOptions<TData, TError, TVariables, TContext>,\r\n  options?: Partial<UseMutationOptions<TData, TError, TVariables, TContext>>,\r\n): UseMutationOptions<TData, TError, TVariables, TContext> {\r\n  if (!options) return base;\r\n  const { onSuccess, onError, onSettled, onMutate, ...rest } = options;\r\n  return {\r\n    ...base,\r\n    ...rest,\r\n    ...(onMutate !== undefined && {\r\n      onMutate: async (variables, context) => {\r\n        const prev = await base.onMutate?.(variables, context);\r\n        const next = await options.onMutate?.(variables, context);\r\n        return (next !== undefined ? next : prev) as TContext;\r\n      },\r\n    }),\r\n    ...(onSuccess !== undefined && {\r\n      onSuccess: async (data, variables, onMutateResult, context) => {\r\n        await base.onSuccess?.(data, variables, onMutateResult, context);\r\n        await options.onSuccess?.(data, variables, onMutateResult, context);\r\n      },\r\n    }),\r\n    ...(onError !== undefined && {\r\n      onError: async (error, variables, onMutateResult, context) => {\r\n        await base.onError?.(error, variables, onMutateResult, context);\r\n        await options.onError?.(error, variables, onMutateResult, context);\r\n      },\r\n    }),\r\n    ...(onSettled !== undefined && {\r\n      onSettled: async (data, error, variables, onMutateResult, context) => {\r\n        await base.onSettled?.(data, error, variables, onMutateResult, context);\r\n        await options.onSettled?.(\r\n          data,\r\n          error,\r\n          variables,\r\n          onMutateResult,\r\n          context,\r\n        );\r\n      },\r\n    }),\r\n  };\r\n}\r\n"
  ],
  "mappings": ";AACA;AAAA;AAAA;AAAA;;;ACKO,SAAS,oBAAyD,CACvE,MACA,SACyD;AAAA,EACzD,IAAI,CAAC;AAAA,IAAS,OAAO;AAAA,EACrB,QAAQ,WAAW,SAAS,WAAW,aAAa,SAAS;AAAA,EAC7D,OAAO;AAAA,OACF;AAAA,OACA;AAAA,OACC,aAAa,aAAa;AAAA,MAC5B,UAAU,OAAO,WAAW,YAAY;AAAA,QACtC,MAAM,OAAO,MAAM,KAAK,WAAW,WAAW,OAAO;AAAA,QACrD,MAAM,OAAO,MAAM,QAAQ,WAAW,WAAW,OAAO;AAAA,QACxD,OAAQ,SAAS,YAAY,OAAO;AAAA;AAAA,IAExC;AAAA,OACI,cAAc,aAAa;AAAA,MAC7B,WAAW,OAAO,MAAM,WAAW,gBAAgB,YAAY;AAAA,QAC7D,MAAM,KAAK,YAAY,MAAM,WAAW,gBAAgB,OAAO;AAAA,QAC/D,MAAM,QAAQ,YAAY,MAAM,WAAW,gBAAgB,OAAO;AAAA;AAAA,IAEtE;AAAA,OACI,YAAY,aAAa;AAAA,MAC3B,SAAS,OAAO,OAAO,WAAW,gBAAgB,YAAY;AAAA,QAC5D,MAAM,KAAK,UAAU,OAAO,WAAW,gBAAgB,OAAO;AAAA,QAC9D,MAAM,QAAQ,UAAU,OAAO,WAAW,gBAAgB,OAAO;AAAA;AAAA,IAErE;AAAA,OACI,cAAc,aAAa;AAAA,MAC7B,WAAW,OAAO,MAAM,OAAO,WAAW,gBAAgB,YAAY;AAAA,QACpE,MAAM,KAAK,YAAY,MAAM,OAAO,WAAW,gBAAgB,OAAO;AAAA,QACtE,MAAM,QAAQ,YACZ,MACA,OACA,WACA,gBACA,OACF;AAAA;AAAA,IAEJ;AAAA,EACF;AAAA;;;ADsCK,SAAS,kBAKf,CACC,IACA,YAIwD;AAAA,EACxD,OAAO,CAAC,YAAY;AAAA,IAClB,MAAM,cAAc,aAAa,KAAK,CAAC;AAAA,IACvC,OAAO,YAAY;AAAA,SACd,qBAAqB,aAAa,OAAO;AAAA,MAC5C,YAAY,OAAO,YAAY;AAAA,QAE7B,MAAM,aACJ,YAAY,WAAW,QAAQ,WAAW,YACrC,GAA8E,QAAQ,MAAM,IAC5F,GAA+D;AAAA,QAGtE,MAAM,OAAO,UAAU,WAAW,QAAQ,SAAS,YAAY,QAAQ,OAAQ;AAAA,QAC/E,QAAQ,MAAM,UAAU,MAAM,WAAW,MAAM,OAAO;AAAA,QACtD,IAAI;AAAA,UAAO,MAAM;AAAA,QACjB,OAAO;AAAA;AAAA,IAEX,CAAC;AAAA;AAAA;AAmCE,SAAS,eAIf,CACC,IACA,YAG8C;AAAA,EAC9C,OAAO,CAAC,SAAS,YAAY;AAAA,IAE3B,MAAM,UACJ,YAAY,WAAW,QAAQ,WAAW,YACrC,GAAoE,QAAQ,MAAM,IAClF,GAAqD;AAAA,IAC5D,MAAM,cAAc,WAAW,OAAO;AAAA,IACtC,OAAO,SAAS;AAAA,SACX;AAAA,SACA;AAAA,MACH,SAAS,YAAY;AAAA,QACnB,QAAQ,MAAM,UAAU,MAAM,QAAQ,OAAO;AAAA,QAC7C,IAAI;AAAA,UAAO,MAAM;AAAA,QACjB,OAAO;AAAA;AAAA,IAEX,CAAC;AAAA;AAAA;",
  "debugId": "89C48BF36DA4F8BA64756E2164756E21",
  "names": []
}
81
+ //# debugId=D2B04DF79F0D08F564756E2164756E21
82
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src\\treaty.ts", "src\\utils.ts"],
  "sourcesContent": [
    "import type { Treaty } from \"@elysiajs/eden\";\r\nimport {\r\n  useMutation,\r\n  useQuery,\r\n  type QueryKey,\r\n  type UseMutationOptions,\r\n  type UseMutationResult,\r\n  type UseQueryOptions,\r\n  type UseQueryResult,\r\n} from \"@tanstack/react-query\";\r\nimport { mergeMutationOptions } from \"./utils\";\r\n\r\n// Helper type to check if a type represents \"no params\" or \"no body\" (absent/empty)\r\n// Checks for never, undefined, void, unknown (unconstrained), and empty object {}\r\ntype IsEmpty<T> =\r\n  // Check for never, undefined, void\r\n  [T] extends [never | undefined | void]\r\n    ? true\r\n    // Check if T is unknown (unconstrained type parameter)\r\n    : [unknown] extends [T]\r\n      ? [T] extends [unknown]\r\n        ? true\r\n        : false\r\n      // Check if T is an empty object {}\r\n      : [T] extends [object]\r\n        ? [keyof T] extends [never]\r\n          ? true\r\n          : false\r\n        : false;\r\n\r\ntype TreatyMutationFunctionFactory<\r\n  TParams,\r\n  TBody,\r\n  TOption,\r\n  TResponse extends Record<number, unknown>,\r\n> = IsEmpty<TParams> extends true\r\n  ? () => TreatyMutationFunction<TBody, TOption, TResponse> \r\n  : (params: TParams) => TreatyMutationFunction<TBody, TOption, TResponse>;\r\n\r\ntype TreatyMutationFunction<\r\n  TBody,\r\n  TOption,\r\n  TResponse extends Record<number, unknown>,\r\n> = (\r\n  body: TBody,\r\n  options: TOption,\r\n) => Promise<Treaty.TreatyResponse<TResponse>>;\r\n\r\ntype TreatyMutationRequest<\r\n  TParams,\r\n  TBody,\r\n  TOption,\r\n> = TOption &\r\n  (IsEmpty<TParams> extends true ? { params?: TParams } : { params: TParams }) &\r\n  (IsEmpty<TBody> extends true ? { body?: TBody } : { body: TBody });\r\n\r\ntype TreatyData<TResponse extends Record<number, unknown>> =\r\n  Treaty.TreatyResponse<TResponse>[\"data\"];\r\n\r\ntype TreatyError<TResponse extends Record<number, unknown>> =\r\n  Treaty.TreatyResponse<TResponse>[\"error\"];\r\n\r\ntype TreatyMutationOptions<\r\n  TParams,\r\n  TBody,\r\n  TOption,\r\n  TResponse extends Record<number, unknown>,\r\n> = UseMutationOptions<\r\n  TreatyData<TResponse>,\r\n  TreatyError<TResponse>,\r\n  TreatyMutationRequest<TParams, TBody, TOption>\r\n>;\r\n\r\ntype TreatyMutationHook<\r\n  TParams,\r\n  TBody,\r\n  TOption,\r\n  TResponse extends Record<number, unknown>,\r\n> = (\r\n  options?: Omit<\r\n    TreatyMutationOptions<TParams, TBody, TOption, TResponse>,\r\n    \"mutationFn\"\r\n  >,\r\n) => UseMutationResult<\r\n  TreatyData<TResponse>,\r\n  TreatyError<TResponse>,\r\n  TreatyMutationRequest<TParams, TBody, TOption>\r\n>;\r\n\r\nexport function treatyMutationHook<\r\n  TParams,\r\n  TBody,\r\n  TOption,\r\n  TResponse extends Record<number, unknown>,\r\n>(\r\n  fn: TreatyMutationFunctionFactory<TParams, TBody, TOption, TResponse>,\r\n  useOptions?: () => Omit<\r\n    TreatyMutationOptions<TParams, TBody, TOption, TResponse>,\r\n    \"mutationFn\"\r\n  >,\r\n): TreatyMutationHook<TParams, TBody, TOption, TResponse> {\r\n  return (options) => {\r\n    const baseOptions = useOptions?.() ?? {};\r\n    return useMutation({\r\n      ...mergeMutationOptions(baseOptions, options),\r\n      mutationFn: async (request) => {\r\n        // Handle optional params: if params exists, pass it; otherwise call fn() without args\r\n        const mutationFn =\r\n          \"params\" in request && request.params !== undefined\r\n            ? (fn as (params: TParams) => TreatyMutationFunction<TBody, TOption, TResponse>)(request.params)\r\n            : (fn as () => TreatyMutationFunction<TBody, TOption, TResponse>)();\r\n        \r\n        // Handle optional body: if body exists, pass it; otherwise default to empty object\r\n        const body = \"body\" in request && request.body !== undefined ? request.body : ({} as TBody);\r\n        const { data, error } = await mutationFn(body, request);\r\n        if (error) throw error;\r\n        return data;\r\n      },\r\n    });\r\n  };\r\n}\r\n\r\ntype TreatyQueryFunction<\r\n  TOption extends object,\r\n  TResponse extends Record<number, unknown>,\r\n> = (options: TOption) => Promise<Treaty.TreatyResponse<TResponse>>;\r\n\r\ntype TreatyQueryFunctionFactory<\r\n  TParams,\r\n  TOption extends object,\r\n  TResponse extends Record<number, unknown>,\r\n> = IsEmpty<TParams> extends true\r\n  ? () => TreatyQueryFunction<TOption, TResponse>\r\n  : (params: TParams) => TreatyQueryFunction<TOption, TResponse>;\r\n\r\ntype TreatyQueryRequest<\r\n  TParams,\r\n  TOption extends object,\r\n> = TOption &\r\n  (IsEmpty<TParams> extends true ? { params?: TParams } : { params: TParams });\r\n\r\ntype TreatyQueryOptions<TResponse extends Record<number, unknown>> =\r\n  UseQueryOptions<TreatyData<TResponse>, TreatyError<TResponse>>;\r\n\r\ntype TreatyQueryHook<\r\n  TParams,\r\n  TOption extends object,\r\n  TResponse extends Record<number, unknown>,\r\n> = (\r\n  request: TreatyQueryRequest<TParams, TOption>,\r\n  options?: Omit<TreatyQueryOptions<TResponse>, \"queryKey\" | \"queryFn\">,\r\n) => UseQueryResult<TreatyData<TResponse>, TreatyError<TResponse>>;\r\n\r\nexport function treatyQueryHook<\r\n  TParams,\r\n  TOption extends object,\r\n  TResponse extends Record<number, unknown>,\r\n>(\r\n  fn: TreatyQueryFunctionFactory<TParams, TOption, TResponse>,\r\n  useOptions: (\r\n    request: TreatyQueryRequest<TParams, TOption>,\r\n  ) => Omit<TreatyQueryOptions<TResponse>, \"queryFn\">,\r\n): TreatyQueryHook<TParams, TOption, TResponse> {\r\n  return (request, options) => {\r\n    // Handle optional params: if params exists, pass it; otherwise call fn() without args\r\n    const queryFn =\r\n      \"params\" in request && request.params !== undefined\r\n        ? (fn as (params: TParams) => TreatyQueryFunction<TOption, TResponse>)(request.params)\r\n        : (fn as () => TreatyQueryFunction<TOption, TResponse>)();\r\n    const baseOptions = useOptions(request);\r\n    return useQuery({\r\n      ...options,\r\n      ...baseOptions,\r\n      queryFn: async () => {\r\n        const { data, error } = await queryFn(request);\r\n        if (error) throw error;\r\n        return data;\r\n      },\r\n    });\r\n  };\r\n}\r\n",
    "import type {\r\n  QueryKey,\r\n  UseMutationOptions,\r\n  UseQueryOptions,\r\n} from \"@tanstack/react-query\";\r\n\r\nexport function mergeMutationOptions<TData, TError, TVariables, TContext>(\r\n  base: UseMutationOptions<TData, TError, TVariables, TContext>,\r\n  options?: Partial<UseMutationOptions<TData, TError, TVariables, TContext>>,\r\n): UseMutationOptions<TData, TError, TVariables, TContext> {\r\n  if (!options) return base;\r\n  const { onSuccess, onError, onSettled, onMutate, ...rest } = options;\r\n  return {\r\n    ...base,\r\n    ...rest,\r\n    ...(onMutate !== undefined && {\r\n      onMutate: async (variables, context) => {\r\n        const prev = await base.onMutate?.(variables, context);\r\n        const next = await options.onMutate?.(variables, context);\r\n        return (next !== undefined ? next : prev) as TContext;\r\n      },\r\n    }),\r\n    ...(onSuccess !== undefined && {\r\n      onSuccess: async (data, variables, onMutateResult, context) => {\r\n        await base.onSuccess?.(data, variables, onMutateResult, context);\r\n        await options.onSuccess?.(data, variables, onMutateResult, context);\r\n      },\r\n    }),\r\n    ...(onError !== undefined && {\r\n      onError: async (error, variables, onMutateResult, context) => {\r\n        await base.onError?.(error, variables, onMutateResult, context);\r\n        await options.onError?.(error, variables, onMutateResult, context);\r\n      },\r\n    }),\r\n    ...(onSettled !== undefined && {\r\n      onSettled: async (data, error, variables, onMutateResult, context) => {\r\n        await base.onSettled?.(data, error, variables, onMutateResult, context);\r\n        await options.onSettled?.(\r\n          data,\r\n          error,\r\n          variables,\r\n          onMutateResult,\r\n          context,\r\n        );\r\n      },\r\n    }),\r\n  };\r\n}\r\n"
  ],
  "mappings": ";AACA;AAAA;AAAA;AAAA;;;ACKO,SAAS,oBAAyD,CACvE,MACA,SACyD;AAAA,EACzD,IAAI,CAAC;AAAA,IAAS,OAAO;AAAA,EACrB,QAAQ,WAAW,SAAS,WAAW,aAAa,SAAS;AAAA,EAC7D,OAAO;AAAA,OACF;AAAA,OACA;AAAA,OACC,aAAa,aAAa;AAAA,MAC5B,UAAU,OAAO,WAAW,YAAY;AAAA,QACtC,MAAM,OAAO,MAAM,KAAK,WAAW,WAAW,OAAO;AAAA,QACrD,MAAM,OAAO,MAAM,QAAQ,WAAW,WAAW,OAAO;AAAA,QACxD,OAAQ,SAAS,YAAY,OAAO;AAAA;AAAA,IAExC;AAAA,OACI,cAAc,aAAa;AAAA,MAC7B,WAAW,OAAO,MAAM,WAAW,gBAAgB,YAAY;AAAA,QAC7D,MAAM,KAAK,YAAY,MAAM,WAAW,gBAAgB,OAAO;AAAA,QAC/D,MAAM,QAAQ,YAAY,MAAM,WAAW,gBAAgB,OAAO;AAAA;AAAA,IAEtE;AAAA,OACI,YAAY,aAAa;AAAA,MAC3B,SAAS,OAAO,OAAO,WAAW,gBAAgB,YAAY;AAAA,QAC5D,MAAM,KAAK,UAAU,OAAO,WAAW,gBAAgB,OAAO;AAAA,QAC9D,MAAM,QAAQ,UAAU,OAAO,WAAW,gBAAgB,OAAO;AAAA;AAAA,IAErE;AAAA,OACI,cAAc,aAAa;AAAA,MAC7B,WAAW,OAAO,MAAM,OAAO,WAAW,gBAAgB,YAAY;AAAA,QACpE,MAAM,KAAK,YAAY,MAAM,OAAO,WAAW,gBAAgB,OAAO;AAAA,QACtE,MAAM,QAAQ,YACZ,MACA,OACA,WACA,gBACA,OACF;AAAA;AAAA,IAEJ;AAAA,EACF;AAAA;;;AD2CK,SAAS,kBAKf,CACC,IACA,YAIwD;AAAA,EACxD,OAAO,CAAC,YAAY;AAAA,IAClB,MAAM,cAAc,aAAa,KAAK,CAAC;AAAA,IACvC,OAAO,YAAY;AAAA,SACd,qBAAqB,aAAa,OAAO;AAAA,MAC5C,YAAY,OAAO,YAAY;AAAA,QAE7B,MAAM,aACJ,YAAY,WAAW,QAAQ,WAAW,YACrC,GAA8E,QAAQ,MAAM,IAC5F,GAA+D;AAAA,QAGtE,MAAM,OAAO,UAAU,WAAW,QAAQ,SAAS,YAAY,QAAQ,OAAQ,CAAC;AAAA,QAChF,QAAQ,MAAM,UAAU,MAAM,WAAW,MAAM,OAAO;AAAA,QACtD,IAAI;AAAA,UAAO,MAAM;AAAA,QACjB,OAAO;AAAA;AAAA,IAEX,CAAC;AAAA;AAAA;AAmCE,SAAS,eAIf,CACC,IACA,YAG8C;AAAA,EAC9C,OAAO,CAAC,SAAS,YAAY;AAAA,IAE3B,MAAM,UACJ,YAAY,WAAW,QAAQ,WAAW,YACrC,GAAoE,QAAQ,MAAM,IAClF,GAAqD;AAAA,IAC5D,MAAM,cAAc,WAAW,OAAO;AAAA,IACtC,OAAO,SAAS;AAAA,SACX;AAAA,SACA;AAAA,MACH,SAAS,YAAY;AAAA,QACnB,QAAQ,MAAM,UAAU,MAAM,QAAQ,OAAO;AAAA,QAC7C,IAAI;AAAA,UAAO,MAAM;AAAA,QACjB,OAAO;AAAA;AAAA,IAEX,CAAC;AAAA;AAAA;",
  "debugId": "D2B04DF79F0D08F564756E2164756E21",
  "names": []
}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "eden2query",
3
3
  "description": "Type-safe Eden Treaty to React Query helpers",
4
- "version": "0.3.8",
4
+ "version": "0.3.10",
5
5
  "type": "module",
6
6
  "files": [
7
7
  "dist"
@@ -39,7 +39,7 @@
39
39
  "prepublishOnly": "bun run build"
40
40
  },
41
41
  "peerDependencies": {
42
- "@elysiajs/eden": "^1.4.6",
42
+ "@elysiajs/eden": "^1.4.8",
43
43
  "@tanstack/react-query": "^5.90.20",
44
44
  "react": "^19.2.4",
45
45
  "typescript": "^5"
@@ -48,7 +48,7 @@
48
48
  "@types/bun": "latest",
49
49
  "@types/react": "^19.2.14",
50
50
  "bunup": "^0.16.22",
51
- "elysia": "^1.4.22",
51
+ "elysia": "^1.4.25",
52
52
  "prettier": "^3.4.2"
53
53
  }
54
54
  }