eden2query 0.3.6 → 0.3.8

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,6 +1,6 @@
1
1
  # eden2query
2
2
 
3
- Type-safe [Eden Treaty](https://elysiajs.com/eden/overview) to [React Query](https://tanstack.com/query) helpers. Two functions, zero runtime dependencies.
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
4
 
5
5
  ## Install
6
6
 
@@ -8,138 +8,410 @@ Type-safe [Eden Treaty](https://elysiajs.com/eden/overview) to [React Query](htt
8
8
  bun add eden2query @elysiajs/eden @tanstack/react-query
9
9
  ```
10
10
 
11
- ## Usage
11
+ ## Quick Start
12
12
 
13
- **Queries (GET)** pass a thunk that calls the Treaty GET:
13
+ ### 1. Define your Elysia API
14
14
 
15
15
  ```ts
16
- const getOptions = treatyQueryOptions(
17
- () => client.api.resource.get({ query: { q: "hello" } }),
18
- { queryKey: ["resource"], refetchInterval: 1000 },
19
- );
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
+ });
20
45
  ```
21
46
 
22
- **Mutations with variables (POST, PUT, etc.)** pass a function that receives the mutation input, or a bound Treaty method:
47
+ ### 2. Create Treaty Client
23
48
 
24
49
  ```ts
25
- // Wrapper when Treaty uses (body, options)
26
- const postOptions = treatyMutationOptions(
27
- (vars: { body: { name: string }; query: { q: string } }) =>
28
- client.api.resource.post(vars.body, { query: vars.query }),
29
- { onSuccess: () => console.log("Success") },
30
- );
31
-
32
- // Parameterised routes — bind params first
33
- const putOptions = treatyMutationOptions(
34
- client.api.resource({ id: "dummy" }).put,
35
- { onSettled: () => console.log("Settled") },
36
- );
50
+ import { treaty } from "@elysiajs/eden";
51
+
52
+ const client = treaty<typeof app>("http://localhost:3000");
37
53
  ```
38
54
 
39
- **Mutations without variables (e.g. DELETE with fixed params)** — pass a no-argument function; `mutate()` is then called with no args:
55
+ ### 3. Create Type-Safe Hooks
40
56
 
41
57
  ```ts
42
- const deleteOptions = treatyMutationOptions(
43
- () => client.api.resource({ id: "dummy" }).delete(),
44
- { onMutate: () => console.log("Mutate") },
45
- );
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
+ });
46
113
  ```
47
114
 
48
- Use with React Query as usual:
115
+ ### 4. Use in Components
49
116
 
50
117
  ```tsx
51
- const { data } = useQuery(getOptions);
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
+ ```
52
171
 
53
- const postMutation = useMutation(postOptions);
54
- postMutation.mutate({ body: { name: "World" }, query: { q: "hello" } });
172
+ ## API Reference
55
173
 
56
- const putMutation = useMutation(putOptions);
57
- putMutation.mutate({ name: "World" });
174
+ ### `treatyQueryHook`
58
175
 
59
- const deleteMutation = useMutation(deleteOptions);
60
- deleteMutation.mutate(); // no arguments
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
61
184
  ```
62
185
 
63
- Prefetch and other helpers work as usual:
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`)
64
189
 
190
+ **Returns:** A hook function that accepts a request object and optional query options
191
+
192
+ **Example:**
65
193
  ```ts
66
- const queryClient = new QueryClient();
67
- queryClient.prefetchQuery(getOptions);
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
+ });
68
224
  ```
69
225
 
70
- ### Hook builders: `treatyQueryHook` / `treatyMutationHook`
226
+ ### `treatyMutationHook`
227
+
228
+ Creates a type-safe React Query hook for mutations (POST, PUT, DELETE, etc.).
71
229
 
72
- Create reusable, pre-configured hooks for a Treaty endpoint. The first parameter is the **Treaty function**, and the second optional parameter is **`useOptions`** — a hook that returns default options (so you can use React hooks like `useQueryClient()`).
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
+ ```
73
237
 
74
- **Mutation hook** — pass the mutation function and optional `useOptions`:
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`)
75
241
 
242
+ **Returns:** A hook function that accepts optional mutation options
243
+
244
+ **Example:**
76
245
  ```ts
77
- function myMutationFn(variables: { name: string; q: string }) {
78
- return client.api.resource.post(
79
- { name: variables.name },
80
- { query: { q: variables.q } }
81
- );
246
+ // Without params, with body
247
+ function createUser() {
248
+ return client.api.users.post;
82
249
  }
83
250
 
84
- const useMutationHook = treatyMutationHook(myMutationFn, () => {
251
+ const useCreateUser = treatyMutationHook(createUser, () => {
85
252
  const queryClient = useQueryClient();
86
253
  return {
87
254
  onSuccess: () => {
88
- queryClient.invalidateQueries({ queryKey: ["resource"] });
255
+ queryClient.invalidateQueries({ queryKey: ["users"] });
89
256
  },
90
257
  };
91
258
  });
92
259
 
93
- // In a component
94
- const mutationResult = useMutationHook({
95
- onSuccess: (data, variables) => {
96
- console.log(data, variables);
97
- },
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" },
98
282
  });
99
- mutationResult.mutate({
100
- name: "World",
101
- q: "hello",
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
102
296
  });
103
297
  ```
104
298
 
105
- **Query hook** — pass the query function and optional `useOptions` that receives the `request` and returns options including `queryKey`:
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:
106
315
 
107
316
  ```ts
108
- function myQueryFn(request: { q: string }) {
109
- return client.api.resource.get({ query: { q: request.q } });
317
+ function getUser(params: { id: string }) {
318
+ return client.api.users({ id: params.id }).get;
110
319
  }
111
320
 
112
- const useQueryHook = treatyQueryHook(myQueryFn, (request) => ({
113
- queryKey: ["resource", request] as const,
114
- enabled: true,
321
+ const useGetUser = treatyQueryHook(getUser, (request) => ({
322
+ queryKey: ["users", request.params.id], // params is required here
115
323
  }));
116
324
 
117
- // In a component
118
- const queryResult = useQueryHook({ q: "hello" });
325
+ // Must provide params
326
+ useGetUser({
327
+ params: { id: "123" }, // ✅ Required - params is part of the request type
328
+ query: { name: "John" },
329
+ });
119
330
  ```
120
331
 
121
- Hooks accept the same options as `useQuery` / `useMutation`; call-time options are merged with the base options from `useOptions`. Callbacks are chained: base options run first, then `useOptions` callbacks, then call-site callbacks.
332
+ ### Routes without Params
122
333
 
123
- ## API
334
+ For routes without params, params are **absent** from the request type (not optional):
124
335
 
125
- **`treatyQueryOptions(fn, queryOptions)`** — wraps a Treaty GET call into `queryOptions`. `fn` is a **thunk** (zero-argument function) that calls the Treaty GET, e.g. `() => client.api.resource.get({ query: { ... } })`. The second argument accepts all `queryOptions` fields except `queryFn`. Extracts `data` from the response and throws on `error`.
336
+ ```ts
337
+ function getUsers() {
338
+ return client.api.users.get;
339
+ }
126
340
 
127
- **`treatyMutationOptions(fn, mutationOptions?)`** wraps a Treaty mutation into `mutationOptions`. Two shapes:
341
+ const useGetUsers = treatyQueryHook(getUsers, (request) => ({
342
+ queryKey: ["users"],
343
+ }));
128
344
 
129
- - **No-arg:** `fn` is `() => Promise<...>`. Use when the mutation has no call-time input (e.g. DELETE with fixed path params). `mutate()` is called with no arguments.
130
- - **With variables:** `fn` is `(vars) => Promise<...>`. Use for POST/PUT etc. `mutate(vars)` receives a single object (e.g. `{ body, query }` or the Treaty method's first argument). You can pass the Treaty method directly when its signature matches (e.g. `client.api.resource({ id }).put`).
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
+ }
131
360
 
132
- The optional second argument accepts all `mutationOptions` fields except `mutationFn` (e.g. `onSuccess`, `onSettled`, `onMutate`).
361
+ const useCreateUser = treatyMutationHook(createUser, () => ({}));
133
362
 
134
- **`treatyMutationHook(mutationFn, useOptions?)`** — returns a hook that wraps `useMutation` with the given Treaty mutation function. **`mutationFn`** is the Treaty mutation function (e.g. `myMutationFn`). **`useOptions`** is optional; it's a hook that returns default mutation options (so you can use `useQueryClient()` for invalidation). The returned hook accepts the same options as `useMutation`; callbacks are chained (base → `useOptions` → call-site).
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
+ ```
135
405
 
136
- **`treatyQueryHook(queryFn, useOptions?)`** — returns a hook that wraps `useQuery` with the given Treaty query function. **`queryFn`** is the Treaty query function (e.g. `myQueryFn`). **`useOptions`** is optional; it's a function `(request) => ...` that receives the query request and returns default query options including `queryKey` (so you can use React hooks like `useQueryClient()`). The returned hook is called with `(request, options?)`; callbacks are chained (base → `useOptions` → call-site).
406
+ ## Options Merging
137
407
 
138
- **`InferTreatyQueryOptions<T>`** full `UseQueryOptions` type for a given Treaty query.
408
+ Options are merged in this order (later options override earlier ones):
139
409
 
140
- **`InferTreatyMutationOptions<TVariables, TResponse>`** full `UseMutationOptions` type for a given Treaty mutation.
410
+ 1. Base options from `useOptions` hook
411
+ 2. Options passed to the hook factory
412
+ 3. Options passed when calling the hook
141
413
 
142
- Data, error, and input types are inferred end-to-end from your Elysia route definitions.
414
+ Callbacks (`onSuccess`, `onError`, etc.) are **chained** - all callbacks run in order.
143
415
 
144
416
  ## License
145
417
 
package/dist/index.d.ts CHANGED
@@ -1,48 +1,70 @@
1
1
  import { Treaty } from "@elysiajs/eden";
2
- import { QueryKey, UseMutationOptions, UseMutationResult, UseQueryOptions, UseQueryResult } from "@tanstack/react-query";
3
- type TreatyFunction = (...request: any[]) => Promise<Treaty.TreatyResponse<Record<number, unknown>>>;
4
- type InferTreatyData<TFn extends TreatyFunction> = Awaited<ReturnType<TFn>>["data"];
5
- type InferTreatyError<TFn extends TreatyFunction> = Awaited<ReturnType<TFn>>["error"];
6
- type InferTreatyRequest<TFn extends TreatyFunction> = Parameters<TFn>[0];
7
- type InferTreatyMutationOptions<TFn extends TreatyFunction> = UseMutationOptions<InferTreatyData<TFn>, InferTreatyError<TFn>, InferTreatyRequest<TFn>>;
8
- type InferTreatyQueryOptions<
9
- TFn extends TreatyFunction,
10
- TQueryKey extends QueryKey = unknown[]
11
- > = UseQueryOptions<InferTreatyData<TFn>, InferTreatyError<TFn>, InferTreatyData<TFn>, TQueryKey>;
12
- declare function treatyMutationOptions<TFn extends TreatyFunction>(options: {
13
- mutationFn: TFn;
14
- } & Omit<InferTreatyMutationOptions<TFn>, "mutationFn">): InferTreatyMutationOptions<TFn>;
15
- declare function treatyQueryOptions<
16
- TFn extends TreatyFunction,
17
- const TQueryKey extends QueryKey
18
- >(options: {
19
- queryKey: TQueryKey;
20
- queryFn: TFn;
21
- } & Omit<InferTreatyQueryOptions<TFn, TQueryKey>, "queryKey" | "queryFn">): InferTreatyQueryOptions<TFn, TQueryKey>;
22
- type UseTreatyMutationResult<TFn extends TreatyFunction> = UseMutationResult<InferTreatyData<TFn>, InferTreatyError<TFn>, InferTreatyRequest<TFn>>;
23
- type UseTreatyQueryResult<
24
- TFn extends TreatyFunction,
25
- TQueryKey extends QueryKey = unknown[]
26
- > = UseQueryResult<InferTreatyData<TFn>, InferTreatyError<TFn>>;
27
- type TreatyMutationHookOptions<TFn extends TreatyFunction> = Omit<InferTreatyMutationOptions<TFn>, "mutationFn">;
28
- type TreatyQueryHookOptions<
29
- TFn extends TreatyFunction,
30
- TQueryKey extends QueryKey = unknown[]
31
- > = Omit<InferTreatyQueryOptions<TFn, TQueryKey>, "queryKey" | "queryFn">;
32
- type TreatyQueryHookUseOptions<
33
- TFn extends TreatyFunction,
34
- TQueryKey extends QueryKey = unknown[]
35
- > = {
36
- queryKey: TQueryKey;
37
- } & TreatyQueryHookOptions<TFn, TQueryKey>;
38
- type TreatyMutationHook<TFn extends TreatyFunction> = (options?: TreatyMutationHookOptions<TFn>) => UseTreatyMutationResult<TFn>;
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;
4
+ type TreatyMutationFunctionFactory<
5
+ TParams,
6
+ TBody,
7
+ TOption,
8
+ TResponse extends Record<number, unknown>
9
+ > = IsEmpty<TParams> extends true ? () => TreatyMutationFunction<TBody, TOption, TResponse> : (params: TParams) => TreatyMutationFunction<TBody, TOption, TResponse>;
10
+ type TreatyMutationFunction<
11
+ TBody,
12
+ TOption,
13
+ TResponse extends Record<number, unknown>
14
+ > = (body: TBody, options: TOption) => Promise<Treaty.TreatyResponse<TResponse>>;
15
+ type TreatyMutationRequest<
16
+ TParams,
17
+ TBody,
18
+ TOption
19
+ > = TOption & (IsEmpty<TParams> extends true ? {} : {
20
+ params: TParams;
21
+ }) & (IsEmpty<TBody> extends true ? {} : {
22
+ body: TBody;
23
+ });
24
+ type TreatyData<TResponse extends Record<number, unknown>> = Treaty.TreatyResponse<TResponse>["data"];
25
+ type TreatyError<TResponse extends Record<number, unknown>> = Treaty.TreatyResponse<TResponse>["error"];
26
+ type TreatyMutationOptions<
27
+ TParams,
28
+ TBody,
29
+ TOption,
30
+ TResponse extends Record<number, unknown>
31
+ > = UseMutationOptions<TreatyData<TResponse>, TreatyError<TResponse>, TreatyMutationRequest<TParams, TBody, TOption>>;
32
+ type TreatyMutationHook<
33
+ TParams,
34
+ TBody,
35
+ TOption,
36
+ TResponse extends Record<number, unknown>
37
+ > = (options?: Omit<TreatyMutationOptions<TParams, TBody, TOption, TResponse>, "mutationFn">) => UseMutationResult<TreatyData<TResponse>, TreatyError<TResponse>, TreatyMutationRequest<TParams, TBody, TOption>>;
38
+ declare function treatyMutationHook<
39
+ TParams,
40
+ TBody,
41
+ TOption,
42
+ TResponse extends Record<number, unknown>
43
+ >(fn: TreatyMutationFunctionFactory<TParams, TBody, TOption, TResponse>, useOptions?: () => Omit<TreatyMutationOptions<TParams, TBody, TOption, TResponse>, "mutationFn">): TreatyMutationHook<TParams, TBody, TOption, TResponse>;
44
+ type TreatyQueryFunction<
45
+ TOption extends object,
46
+ TResponse extends Record<number, unknown>
47
+ > = (options: TOption) => Promise<Treaty.TreatyResponse<TResponse>>;
48
+ type TreatyQueryFunctionFactory<
49
+ TParams,
50
+ TOption extends object,
51
+ TResponse extends Record<number, unknown>
52
+ > = IsEmpty<TParams> extends true ? () => TreatyQueryFunction<TOption, TResponse> : (params: TParams) => TreatyQueryFunction<TOption, TResponse>;
53
+ type TreatyQueryRequest<
54
+ TParams,
55
+ TOption extends object
56
+ > = TOption & (IsEmpty<TParams> extends true ? {} : {
57
+ params: TParams;
58
+ });
59
+ type TreatyQueryOptions<TResponse extends Record<number, unknown>> = UseQueryOptions<TreatyData<TResponse>, TreatyError<TResponse>>;
39
60
  type TreatyQueryHook<
40
- TFn extends TreatyFunction,
41
- TQueryKey extends QueryKey = unknown[]
42
- > = (request: InferTreatyRequest<TFn>, options?: TreatyQueryHookOptions<TFn, TQueryKey>) => UseTreatyQueryResult<TFn, TQueryKey>;
43
- declare function treatyMutationHook<TFn extends TreatyFunction>(mutationFn: TFn, useOptions?: () => TreatyMutationHookOptions<TFn>): TreatyMutationHook<TFn>;
61
+ TParams,
62
+ TOption extends object,
63
+ TResponse extends Record<number, unknown>
64
+ > = (request: TreatyQueryRequest<TParams, TOption>, options?: Omit<TreatyQueryOptions<TResponse>, "queryKey" | "queryFn">) => UseQueryResult<TreatyData<TResponse>, TreatyError<TResponse>>;
44
65
  declare function treatyQueryHook<
45
- TFn extends TreatyFunction,
46
- TQueryKey extends QueryKey = QueryKey
47
- >(queryFn: TFn, useOptions?: (request: InferTreatyRequest<TFn>) => TreatyQueryHookUseOptions<TFn, TQueryKey>): TreatyQueryHook<TFn, TQueryKey>;
48
- export { treatyQueryOptions, treatyQueryHook, treatyMutationOptions, treatyMutationHook, InferTreatyQueryOptions, InferTreatyMutationOptions };
66
+ TParams,
67
+ TOption extends object,
68
+ TResponse extends Record<number, unknown>
69
+ >(fn: TreatyQueryFunctionFactory<TParams, TOption, TResponse>, useOptions: (request: TreatyQueryRequest<TParams, TOption>) => Omit<TreatyQueryOptions<TResponse>, "queryFn">): TreatyQueryHook<TParams, TOption, TResponse>;
70
+ export { treatyQueryHook, treatyMutationHook };
package/dist/index.js CHANGED
@@ -1,5 +1,8 @@
1
1
  // src/treaty.ts
2
- import { mutationOptions, queryOptions, useMutation, useQuery } from "@tanstack/react-query";
2
+ import {
3
+ useMutation,
4
+ useQuery
5
+ } from "@tanstack/react-query";
3
6
 
4
7
  // src/utils.ts
5
8
  function mergeMutationOptions(base, options) {
@@ -36,69 +39,44 @@ function mergeMutationOptions(base, options) {
36
39
  }
37
40
  };
38
41
  }
39
- function mergeQueryOptions(base, options) {
40
- if (!options)
41
- return base;
42
- return {
43
- ...base,
44
- ...options
45
- };
46
- }
47
42
 
48
43
  // src/treaty.ts
49
- function treatyMutationOptions(options) {
50
- const { mutationFn, ...rest } = options;
51
- return mutationOptions({
52
- ...rest,
53
- mutationFn: async (variables) => {
54
- const response = await mutationFn(variables);
55
- const { data, error } = response;
56
- if (error)
57
- throw error;
58
- return data;
59
- }
60
- });
61
- }
62
- function treatyQueryOptions(options) {
63
- const { queryFn, queryKey, ...rest } = options;
64
- return queryOptions({
65
- ...rest,
66
- queryKey,
67
- queryFn: async () => {
68
- const response = await queryFn();
69
- const { data, error } = response;
70
- if (error)
71
- throw error;
72
- return data;
73
- }
74
- });
75
- }
76
- function treatyMutationHook(mutationFn, useOptions) {
44
+ function treatyMutationHook(fn, useOptions) {
77
45
  return (options) => {
78
- const baseOptions = treatyMutationOptions({ mutationFn });
79
- const defaultOptions = useOptions?.();
80
- const merged = mergeMutationOptions(mergeMutationOptions(baseOptions, defaultOptions), options);
81
- return useMutation(merged);
46
+ const baseOptions = useOptions?.() ?? {};
47
+ return useMutation({
48
+ ...mergeMutationOptions(baseOptions, options),
49
+ mutationFn: async (request) => {
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;
52
+ const { data, error } = await mutationFn(body, request);
53
+ if (error)
54
+ throw error;
55
+ return data;
56
+ }
57
+ });
82
58
  };
83
59
  }
84
- function treatyQueryHook(queryFn, useOptions) {
60
+ function treatyQueryHook(fn, useOptions) {
85
61
  return (request, options) => {
86
- const defaultOptions = useOptions?.(request);
87
- const baseOptions = treatyQueryOptions({
88
- queryKey: defaultOptions?.queryKey ?? ["resource", request],
89
- queryFn: () => queryFn(request),
90
- ...defaultOptions
62
+ const queryFn = "params" in request && request.params !== undefined ? fn(request.params) : fn();
63
+ const baseOptions = useOptions(request);
64
+ return useQuery({
65
+ ...options,
66
+ ...baseOptions,
67
+ queryFn: async () => {
68
+ const { data, error } = await queryFn(request);
69
+ if (error)
70
+ throw error;
71
+ return data;
72
+ }
91
73
  });
92
- const merged = mergeQueryOptions(baseOptions, options);
93
- return useQuery(merged);
94
74
  };
95
75
  }
96
76
  export {
97
- treatyQueryOptions,
98
77
  treatyQueryHook,
99
- treatyMutationOptions,
100
78
  treatyMutationHook
101
79
  };
102
80
 
103
- //# debugId=3974E80E199B9F8064756E2164756E21
104
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["src\\treaty.ts", "src\\utils.ts"],
  "sourcesContent": [
    "import type { Treaty } from \"@elysiajs/eden\";\r\nimport { mutationOptions, queryOptions, useMutation, useQuery, type QueryKey, type UseMutationOptions, type UseMutationResult, type UseQueryOptions, type UseQueryResult } from \"@tanstack/react-query\";\r\nimport { mergeMutationOptions, mergeQueryOptions } from \"./utils\";\r\n\r\ntype TreatyFunction = (...request: any[]) => \r\n  Promise<Treaty.TreatyResponse<Record<number, unknown>>>;\r\n\r\nexport type InferTreatyData<\r\n  TFn extends TreatyFunction,\r\n> = Awaited<ReturnType<TFn>>[\"data\"];\r\n\r\nexport type InferTreatyError<\r\n  TFn extends TreatyFunction,\r\n> = Awaited<ReturnType<TFn>>[\"error\"];\r\n\r\nexport type InferTreatyRequest<\r\n  TFn extends TreatyFunction,\r\n> = Parameters<TFn>[0];\r\n\r\nexport type InferTreatyMutationOptions<\r\n  TFn extends TreatyFunction,\r\n> = UseMutationOptions<\r\n  InferTreatyData<TFn>,\r\n  InferTreatyError<TFn>,\r\n  InferTreatyRequest<TFn>\r\n>;\r\n\r\nexport type InferTreatyQueryOptions<\r\n  TFn extends TreatyFunction,\r\n  TQueryKey extends QueryKey = unknown[],\r\n> = UseQueryOptions<\r\n  InferTreatyData<TFn>,\r\n  InferTreatyError<TFn>,\r\n  InferTreatyData<TFn>,\r\n  TQueryKey\r\n>;\r\n\r\nexport function treatyMutationOptions<TFn extends TreatyFunction>(\r\n  options: {\r\n    mutationFn: TFn;\r\n  } & Omit<\r\n    InferTreatyMutationOptions<TFn>,\r\n    \"mutationFn\"\r\n  >\r\n): InferTreatyMutationOptions<TFn> {\r\n  type TData = InferTreatyData<TFn>;\r\n  type TError = InferTreatyError<TFn>;\r\n  type TVariables = InferTreatyRequest<TFn>;\r\n  const { mutationFn, ...rest } = options;\r\n  return mutationOptions<TData, TError, TVariables>({\r\n    ...rest,\r\n    mutationFn: async (variables) => {\r\n      const response = await mutationFn(variables);\r\n      const { data, error } = response;\r\n      if (error) throw error;\r\n      return data;\r\n    },\r\n  });\r\n}\r\n\r\nexport function treatyQueryOptions<\r\n  TFn extends TreatyFunction,\r\n  const TQueryKey extends QueryKey,\r\n>(\r\n  options: {\r\n    queryKey: TQueryKey;\r\n    queryFn: TFn;\r\n  } & Omit<\r\n    InferTreatyQueryOptions<TFn, TQueryKey>,\r\n    \"queryKey\" | \"queryFn\"\r\n  >\r\n): InferTreatyQueryOptions<TFn, TQueryKey> {\r\n  type TData = InferTreatyData<TFn>;\r\n  type TError = InferTreatyError<TFn>;\r\n  const { queryFn, queryKey, ...rest } = options;\r\n  return queryOptions<TData, TError, TData, TQueryKey>({\r\n    ...rest,\r\n    queryKey,\r\n    queryFn: async () => {\r\n      const response = await queryFn();\r\n      const { data, error } = response;\r\n      if (error) throw error;\r\n      return data;\r\n    },\r\n  });\r\n}\r\n\r\nexport type UseTreatyMutationResult<TFn extends TreatyFunction> = UseMutationResult<\r\n  InferTreatyData<TFn>,\r\n  InferTreatyError<TFn>,\r\n  InferTreatyRequest<TFn>\r\n>;\r\n\r\nexport type UseTreatyQueryResult<\r\n  TFn extends TreatyFunction,\r\n  TQueryKey extends QueryKey = unknown[],\r\n> = UseQueryResult<\r\n  InferTreatyData<TFn>,\r\n  InferTreatyError<TFn>\r\n>;\r\n\r\nexport type TreatyMutationHookOptions<TFn extends TreatyFunction> = Omit<\r\n  InferTreatyMutationOptions<TFn>,\r\n  \"mutationFn\"\r\n>;\r\n\r\nexport type TreatyQueryHookOptions<\r\n  TFn extends TreatyFunction,\r\n  TQueryKey extends QueryKey = unknown[],\r\n> = Omit<\r\n  InferTreatyQueryOptions<TFn, TQueryKey>,\r\n  \"queryKey\" | \"queryFn\"\r\n>;\r\n\r\nexport type TreatyQueryHookUseOptions<\r\n  TFn extends TreatyFunction,\r\n  TQueryKey extends QueryKey = unknown[],\r\n> = {\r\n  queryKey: TQueryKey;\r\n} & TreatyQueryHookOptions<TFn, TQueryKey>;\r\n\r\nexport type TreatyMutationHook<TFn extends TreatyFunction> = (\r\n  options?: TreatyMutationHookOptions<TFn>\r\n) => UseTreatyMutationResult<TFn>;\r\n\r\nexport type TreatyQueryHook<\r\n  TFn extends TreatyFunction,\r\n  TQueryKey extends QueryKey = unknown[],\r\n> = (\r\n  request: InferTreatyRequest<TFn>,\r\n  options?: TreatyQueryHookOptions<TFn, TQueryKey>\r\n) => UseTreatyQueryResult<TFn, TQueryKey>;\r\n\r\nexport function treatyMutationHook<TFn extends TreatyFunction>(\r\n  mutationFn: TFn,\r\n  useOptions?: () => TreatyMutationHookOptions<TFn>\r\n): TreatyMutationHook<TFn> {\r\n  return (options) => {\r\n    const baseOptions = treatyMutationOptions({ mutationFn });\r\n    const defaultOptions = useOptions?.();\r\n    const merged = mergeMutationOptions(\r\n      mergeMutationOptions(baseOptions, defaultOptions),\r\n      options\r\n    );\r\n    return useMutation(merged);\r\n  };\r\n}\r\n\r\nexport function treatyQueryHook<\r\n  TFn extends TreatyFunction,\r\n  TQueryKey extends QueryKey = QueryKey,\r\n>(\r\n  queryFn: TFn,\r\n  useOptions?: (\r\n    request: InferTreatyRequest<TFn>\r\n  ) => TreatyQueryHookUseOptions<TFn, TQueryKey>\r\n): TreatyQueryHook<TFn, TQueryKey> {\r\n  return (request, options) => {\r\n    const defaultOptions = useOptions?.(request);\r\n    const baseOptions = treatyQueryOptions({\r\n      queryKey: (defaultOptions?.queryKey ?? ([\"resource\", request] as unknown)) as TQueryKey,\r\n      queryFn: (() => queryFn(request)) as TFn,\r\n      ...defaultOptions,\r\n    });\r\n    const merged = mergeQueryOptions(\r\n      baseOptions,\r\n      options as UseQueryOptions<\r\n        InferTreatyData<TFn>,\r\n        InferTreatyError<TFn>,\r\n        InferTreatyData<TFn>,\r\n        TQueryKey\r\n      > | undefined\r\n    );\r\n    return useQuery(merged);\r\n  };\r\n}",
    "import type { QueryKey, UseMutationOptions, UseQueryOptions } 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?.(data, error, variables, onMutateResult, context);\r\n      },\r\n    }),\r\n  };\r\n}\r\n\r\nexport function mergeQueryOptions<TData, TError, TQueryKey extends QueryKey>(\r\n  base: UseQueryOptions<TData, TError, TData, TQueryKey>,\r\n  options?: Partial<UseQueryOptions<TData, TError, TData, TQueryKey>>\r\n): UseQueryOptions<TData, TError, TData, TQueryKey> {\r\n  if (!options) return base;\r\n  return {\r\n    ...base,\r\n    ...options,\r\n  };\r\n}\r\n"
  ],
  "mappings": "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",
  "debugId": "3974E80E199B9F8064756E2164756E21",
  "names": []
}
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": []
}
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.6",
4
+ "version": "0.3.8",
5
5
  "type": "module",
6
6
  "files": [
7
7
  "dist"