@fragno-dev/core 0.1.7 → 0.1.9

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.
Files changed (183) hide show
  1. package/.turbo/turbo-build.log +131 -64
  2. package/CHANGELOG.md +19 -0
  3. package/dist/api/api.d.ts +38 -2
  4. package/dist/api/api.d.ts.map +1 -0
  5. package/dist/api/api.js +9 -2
  6. package/dist/api/api.js.map +1 -0
  7. package/dist/api/bind-services.d.ts +6 -0
  8. package/dist/api/bind-services.d.ts.map +1 -0
  9. package/dist/api/bind-services.js +20 -0
  10. package/dist/api/bind-services.js.map +1 -0
  11. package/dist/api/error.d.ts +26 -0
  12. package/dist/api/error.d.ts.map +1 -0
  13. package/dist/{api-DngJDcmO.js → api/error.js} +2 -8
  14. package/dist/api/error.js.map +1 -0
  15. package/dist/api/fragment-definition-builder.d.ts +313 -0
  16. package/dist/api/fragment-definition-builder.d.ts.map +1 -0
  17. package/dist/api/fragment-definition-builder.js +326 -0
  18. package/dist/api/fragment-definition-builder.js.map +1 -0
  19. package/dist/api/fragment-instantiator.d.ts +216 -0
  20. package/dist/api/fragment-instantiator.d.ts.map +1 -0
  21. package/dist/api/fragment-instantiator.js +487 -0
  22. package/dist/api/fragment-instantiator.js.map +1 -0
  23. package/dist/api/fragno-response.d.ts +30 -0
  24. package/dist/api/fragno-response.d.ts.map +1 -0
  25. package/dist/api/fragno-response.js +73 -0
  26. package/dist/api/fragno-response.js.map +1 -0
  27. package/dist/api/internal/path.d.ts +50 -0
  28. package/dist/api/internal/path.d.ts.map +1 -0
  29. package/dist/api/internal/path.js +76 -0
  30. package/dist/api/internal/path.js.map +1 -0
  31. package/dist/api/internal/response-stream.d.ts +43 -0
  32. package/dist/api/internal/response-stream.d.ts.map +1 -0
  33. package/dist/api/internal/response-stream.js +81 -0
  34. package/dist/api/internal/response-stream.js.map +1 -0
  35. package/dist/api/internal/route.js +10 -0
  36. package/dist/api/internal/route.js.map +1 -0
  37. package/dist/api/mutable-request-state.d.ts +82 -0
  38. package/dist/api/mutable-request-state.d.ts.map +1 -0
  39. package/dist/api/mutable-request-state.js +97 -0
  40. package/dist/api/mutable-request-state.js.map +1 -0
  41. package/dist/api/request-context-storage.d.ts +42 -0
  42. package/dist/api/request-context-storage.d.ts.map +1 -0
  43. package/dist/api/request-context-storage.js +43 -0
  44. package/dist/api/request-context-storage.js.map +1 -0
  45. package/dist/api/request-input-context.d.ts +89 -0
  46. package/dist/api/request-input-context.d.ts.map +1 -0
  47. package/dist/api/request-input-context.js +118 -0
  48. package/dist/api/request-input-context.js.map +1 -0
  49. package/dist/api/request-middleware.d.ts +50 -0
  50. package/dist/api/request-middleware.d.ts.map +1 -0
  51. package/dist/api/request-middleware.js +83 -0
  52. package/dist/api/request-middleware.js.map +1 -0
  53. package/dist/api/request-output-context.d.ts +41 -0
  54. package/dist/api/request-output-context.d.ts.map +1 -0
  55. package/dist/api/request-output-context.js +119 -0
  56. package/dist/api/request-output-context.js.map +1 -0
  57. package/dist/api/route-handler-input-options.d.ts +21 -0
  58. package/dist/api/route-handler-input-options.d.ts.map +1 -0
  59. package/dist/api/route.d.ts +54 -3
  60. package/dist/api/route.d.ts.map +1 -0
  61. package/dist/api/route.js +29 -2
  62. package/dist/api/route.js.map +1 -0
  63. package/dist/api/shared-types.d.ts +47 -0
  64. package/dist/api/shared-types.d.ts.map +1 -0
  65. package/dist/api/shared-types.js +1 -0
  66. package/dist/client/client-error.d.ts +60 -0
  67. package/dist/client/client-error.d.ts.map +1 -0
  68. package/dist/client/client-error.js +92 -0
  69. package/dist/client/client-error.js.map +1 -0
  70. package/dist/client/client.d.ts +210 -4
  71. package/dist/client/client.d.ts.map +1 -0
  72. package/dist/client/client.js +397 -6
  73. package/dist/client/client.js.map +1 -0
  74. package/dist/client/client.svelte.d.ts +5 -3
  75. package/dist/client/client.svelte.d.ts.map +1 -1
  76. package/dist/client/client.svelte.js +1 -5
  77. package/dist/client/client.svelte.js.map +1 -1
  78. package/dist/client/internal/fetcher-merge.js +36 -0
  79. package/dist/client/internal/fetcher-merge.js.map +1 -0
  80. package/dist/client/internal/ndjson-streaming.js +139 -0
  81. package/dist/client/internal/ndjson-streaming.js.map +1 -0
  82. package/dist/client/react.d.ts +5 -3
  83. package/dist/client/react.d.ts.map +1 -1
  84. package/dist/client/react.js +3 -5
  85. package/dist/client/react.js.map +1 -1
  86. package/dist/client/solid.d.ts +5 -3
  87. package/dist/client/solid.d.ts.map +1 -1
  88. package/dist/client/solid.js +2 -5
  89. package/dist/client/solid.js.map +1 -1
  90. package/dist/client/vanilla.d.ts +5 -3
  91. package/dist/client/vanilla.d.ts.map +1 -1
  92. package/dist/client/vanilla.js +2 -43
  93. package/dist/client/vanilla.js.map +1 -1
  94. package/dist/client/vue.d.ts +5 -3
  95. package/dist/client/vue.d.ts.map +1 -1
  96. package/dist/client/vue.js +1 -5
  97. package/dist/client/vue.js.map +1 -1
  98. package/dist/http/http-status.d.ts +26 -0
  99. package/dist/http/http-status.d.ts.map +1 -0
  100. package/dist/integrations/react-ssr.js +1 -1
  101. package/dist/internal/symbols.d.ts +9 -0
  102. package/dist/internal/symbols.d.ts.map +1 -0
  103. package/dist/internal/symbols.js +10 -0
  104. package/dist/internal/symbols.js.map +1 -0
  105. package/dist/mod-client.d.ts +36 -0
  106. package/dist/mod-client.d.ts.map +1 -0
  107. package/dist/mod-client.js +21 -0
  108. package/dist/mod-client.js.map +1 -0
  109. package/dist/mod.d.ts +7 -4
  110. package/dist/mod.js +4 -6
  111. package/dist/request/request.d.ts +4 -0
  112. package/dist/request/request.js +5 -0
  113. package/dist/test/test.d.ts +62 -35
  114. package/dist/test/test.d.ts.map +1 -1
  115. package/dist/test/test.js +75 -40
  116. package/dist/test/test.js.map +1 -1
  117. package/dist/util/async.js +40 -0
  118. package/dist/util/async.js.map +1 -0
  119. package/dist/util/content-type.js +49 -0
  120. package/dist/util/content-type.js.map +1 -0
  121. package/dist/util/nanostores.js +31 -0
  122. package/dist/util/nanostores.js.map +1 -0
  123. package/dist/{ssr-BByDVfFD.js → util/ssr.js} +2 -2
  124. package/dist/util/ssr.js.map +1 -0
  125. package/dist/util/types-util.d.ts +8 -0
  126. package/dist/util/types-util.d.ts.map +1 -0
  127. package/package.json +19 -12
  128. package/src/api/api.ts +41 -6
  129. package/src/api/bind-services.ts +42 -0
  130. package/src/api/fragment-definition-builder.extend.test.ts +810 -0
  131. package/src/api/fragment-definition-builder.test.ts +499 -0
  132. package/src/api/fragment-definition-builder.ts +1088 -0
  133. package/src/api/fragment-instantiator.test.ts +1488 -0
  134. package/src/api/fragment-instantiator.ts +1053 -0
  135. package/src/api/fragment-services.test.ts +727 -0
  136. package/src/api/request-context-storage.ts +64 -0
  137. package/src/api/request-middleware.test.ts +301 -225
  138. package/src/api/route.test.ts +87 -1
  139. package/src/api/route.ts +345 -24
  140. package/src/api/shared-types.ts +43 -0
  141. package/src/client/client-builder.test.ts +23 -23
  142. package/src/client/client.ssr.test.ts +3 -3
  143. package/src/client/client.svelte.test.ts +15 -15
  144. package/src/client/client.test.ts +22 -22
  145. package/src/client/client.ts +72 -12
  146. package/src/client/internal/fetcher-merge.ts +1 -1
  147. package/src/client/react.test.ts +2 -2
  148. package/src/client/solid.test.ts +2 -2
  149. package/src/client/vanilla.test.ts +2 -2
  150. package/src/client/vue.test.ts +2 -2
  151. package/src/internal/symbols.ts +5 -0
  152. package/src/mod-client.ts +59 -0
  153. package/src/mod.ts +26 -9
  154. package/src/request/request.ts +8 -0
  155. package/src/test/test.test.ts +200 -381
  156. package/src/test/test.ts +190 -117
  157. package/tsdown.config.ts +8 -5
  158. package/dist/api/fragment-builder.d.ts +0 -4
  159. package/dist/api/fragment-builder.js +0 -3
  160. package/dist/api/fragment-instantiation.d.ts +0 -4
  161. package/dist/api/fragment-instantiation.js +0 -6
  162. package/dist/api-BWN97TOr.d.ts +0 -377
  163. package/dist/api-BWN97TOr.d.ts.map +0 -1
  164. package/dist/api-DngJDcmO.js.map +0 -1
  165. package/dist/client-C5LsYHEI.js +0 -782
  166. package/dist/client-C5LsYHEI.js.map +0 -1
  167. package/dist/fragment-builder-DOnCVBqc.js +0 -47
  168. package/dist/fragment-builder-DOnCVBqc.js.map +0 -1
  169. package/dist/fragment-builder-MGr68GNb.d.ts +0 -409
  170. package/dist/fragment-builder-MGr68GNb.d.ts.map +0 -1
  171. package/dist/fragment-instantiation-C4wvwl6V.js +0 -446
  172. package/dist/fragment-instantiation-C4wvwl6V.js.map +0 -1
  173. package/dist/request-output-context-CdIjwmEN.js +0 -320
  174. package/dist/request-output-context-CdIjwmEN.js.map +0 -1
  175. package/dist/route-Bl9Zr1Yv.d.ts +0 -26
  176. package/dist/route-Bl9Zr1Yv.d.ts.map +0 -1
  177. package/dist/route-C5Uryylh.js +0 -21
  178. package/dist/route-C5Uryylh.js.map +0 -1
  179. package/dist/ssr-BByDVfFD.js.map +0 -1
  180. package/src/api/fragment-builder.ts +0 -80
  181. package/src/api/fragment-instantiation.test.ts +0 -460
  182. package/src/api/fragment-instantiation.ts +0 -499
  183. package/src/api/fragment.test.ts +0 -537
@@ -1,7 +1,398 @@
1
- import "../api-DngJDcmO.js";
2
- import "../request-output-context-CdIjwmEN.js";
3
- import "../route-C5Uryylh.js";
4
- import { a as isGetHook, d as FragnoClientFetchAbortError, f as FragnoClientFetchError, i as getCacheKey, l as FragnoClientApiError, m as FragnoClientUnknownApiError, n as buildUrl, o as isMutatorHook, p as FragnoClientFetchNetworkError, r as createClientBuilder, s as isStore, t as ClientBuilder, u as FragnoClientError } from "../client-C5LsYHEI.js";
5
- import "../ssr-BByDVfFD.js";
1
+ import { resolveRouteFactories } from "../api/route.js";
2
+ import { getMountRoute } from "../api/internal/route.js";
3
+ import { RequestInputContext } from "../api/request-input-context.js";
4
+ import { RequestOutputContext } from "../api/request-output-context.js";
5
+ import { buildPath, extractPathParams } from "../api/internal/path.js";
6
+ import { FragnoClientApiError, FragnoClientError, FragnoClientFetchAbortError, FragnoClientFetchError, FragnoClientFetchNetworkError, FragnoClientUnknownApiError } from "./client-error.js";
7
+ import { parseContentType } from "../util/content-type.js";
8
+ import { handleNdjsonStreamingFirstItem } from "./internal/ndjson-streaming.js";
9
+ import { SSR_ENABLED, addStore, getInitialData } from "../util/ssr.js";
10
+ import { unwrapObject } from "../util/nanostores.js";
11
+ import { mergeFetcherConfigs } from "./internal/fetcher-merge.js";
12
+ import { computed, task } from "nanostores";
13
+ import { nanoquery } from "@nanostores/query";
6
14
 
7
- export { ClientBuilder, FragnoClientApiError, FragnoClientError, FragnoClientFetchAbortError, FragnoClientFetchError, FragnoClientFetchNetworkError, FragnoClientUnknownApiError, buildUrl, createClientBuilder, getCacheKey, isGetHook, isMutatorHook, isStore };
15
+ //#region src/client/client.ts
16
+ /**
17
+ * Symbols used to identify hook types
18
+ */
19
+ const GET_HOOK_SYMBOL = Symbol("fragno-get-hook");
20
+ const MUTATOR_HOOK_SYMBOL = Symbol("fragno-mutator-hook");
21
+ const STORE_SYMBOL = Symbol("fragno-store");
22
+ /**
23
+ * @internal
24
+ */
25
+ function buildUrl(config, params) {
26
+ const { baseUrl = "", mountRoute, path } = config;
27
+ const { pathParams, queryParams } = params ?? {};
28
+ const normalizedPathParams = unwrapObject(pathParams);
29
+ const normalizedQueryParams = unwrapObject(queryParams) ?? {};
30
+ const filteredQueryParams = Object.fromEntries(Object.entries(normalizedQueryParams).filter(([_, value]) => value !== void 0));
31
+ const searchParams = new URLSearchParams(filteredQueryParams);
32
+ return `${baseUrl}${mountRoute}${buildPath(path, normalizedPathParams ?? {})}${searchParams.toString() ? `?${searchParams.toString()}` : ""}`;
33
+ }
34
+ /**
35
+ * This method returns an array, which can be passed directly to nanostores.
36
+ *
37
+ * The returned array is always: path, pathParams (In order they appear in the path), queryParams (In alphabetical order)
38
+ * Missing pathParams are replaced with "<missing>".
39
+ * Atoms with undefined values are wrapped in computed atoms that map undefined to "" to avoid nanoquery treating the key as incomplete.
40
+ * @param path
41
+ * @param params
42
+ * @returns
43
+ * @internal
44
+ */
45
+ function getCacheKey(method, path, params) {
46
+ if (!params) return [method, path];
47
+ const { pathParams, queryParams } = params;
48
+ const pathParamValues = extractPathParams(path).map((name) => pathParams?.[name] ?? "<missing>");
49
+ const queryParamValues = queryParams ? Object.keys(queryParams).sort().map((key) => {
50
+ const value = queryParams[key];
51
+ if (value && typeof value === "object" && "get" in value) return computed(value, (v) => v ?? "");
52
+ return value ?? "";
53
+ }) : [];
54
+ return [
55
+ method,
56
+ path,
57
+ ...pathParamValues,
58
+ ...queryParamValues
59
+ ];
60
+ }
61
+ function isStreamingResponse(response) {
62
+ const contentType = parseContentType(response.headers.get("content-type"));
63
+ if (!contentType) return false;
64
+ if (!(response.headers.get("transfer-encoding") === "chunked")) return false;
65
+ if (contentType.subtype === "octet-stream") return "octet-stream";
66
+ if (contentType.subtype === "x-ndjson") return "ndjson";
67
+ return false;
68
+ }
69
+ /**
70
+ * @internal
71
+ */
72
+ function isGetHook(hook) {
73
+ return typeof hook === "object" && hook !== null && GET_HOOK_SYMBOL in hook && hook[GET_HOOK_SYMBOL] === true;
74
+ }
75
+ /**
76
+ * @internal
77
+ */
78
+ function isMutatorHook(hook) {
79
+ return typeof hook === "object" && hook !== null && MUTATOR_HOOK_SYMBOL in hook && hook[MUTATOR_HOOK_SYMBOL] === true;
80
+ }
81
+ /**
82
+ * @internal
83
+ */
84
+ function isStore(obj) {
85
+ return typeof obj === "object" && obj !== null && STORE_SYMBOL in obj && obj[STORE_SYMBOL] === true;
86
+ }
87
+ var ClientBuilder = class {
88
+ #publicConfig;
89
+ #fragmentConfig;
90
+ #fetcherConfig;
91
+ #cache = /* @__PURE__ */ new Map();
92
+ #createFetcherStore;
93
+ #createMutatorStore;
94
+ #invalidateKeys;
95
+ constructor(publicConfig, fragmentConfig) {
96
+ this.#publicConfig = publicConfig;
97
+ this.#fragmentConfig = fragmentConfig;
98
+ this.#fetcherConfig = publicConfig.fetcherConfig;
99
+ const [createFetcherStore, createMutatorStore, { invalidateKeys }] = nanoquery({ cache: this.#cache });
100
+ this.#createFetcherStore = createFetcherStore;
101
+ this.#createMutatorStore = createMutatorStore;
102
+ this.#invalidateKeys = invalidateKeys;
103
+ }
104
+ get cacheEntries() {
105
+ return Object.fromEntries(this.#cache.entries());
106
+ }
107
+ createStore(obj) {
108
+ return {
109
+ obj,
110
+ [STORE_SYMBOL]: true
111
+ };
112
+ }
113
+ /**
114
+ * Build a URL for a custom backend call using the configured baseUrl and mountRoute.
115
+ * Useful for fragment authors who need to make custom fetch calls.
116
+ */
117
+ buildUrl(path, params) {
118
+ return buildUrl({
119
+ baseUrl: this.#publicConfig.baseUrl ?? "",
120
+ mountRoute: getMountRoute(this.#fragmentConfig),
121
+ path
122
+ }, {
123
+ pathParams: params?.path,
124
+ queryParams: params?.query
125
+ });
126
+ }
127
+ /**
128
+ * Get the configured fetcher function for custom backend calls.
129
+ * Returns fetch with merged options applied.
130
+ */
131
+ getFetcher() {
132
+ return {
133
+ fetcher: this.#getFetcher(),
134
+ defaultOptions: this.#getFetcherOptions()
135
+ };
136
+ }
137
+ #getFetcher() {
138
+ if (this.#fetcherConfig?.type === "function") return this.#fetcherConfig.fetcher;
139
+ return fetch;
140
+ }
141
+ #getFetcherOptions() {
142
+ if (this.#fetcherConfig?.type === "options") return this.#fetcherConfig.options;
143
+ }
144
+ createHook(path, options) {
145
+ const route = this.#fragmentConfig.routes.find((r) => r.path === path && r.method === "GET" && r.outputSchema !== void 0);
146
+ if (!route) throw new Error(`Route '${path}' not found or is not a GET route with an output schema.`);
147
+ return this.#createRouteQueryHook(route, options);
148
+ }
149
+ createMutator(method, path, onInvalidate) {
150
+ const route = this.#fragmentConfig.routes.find((r) => r.method !== "GET" && r.path === path && r.method === method);
151
+ if (!route) throw new Error(`Route '${path}' not found or is a GET route with an input and output schema.`);
152
+ return this.#createRouteQueryMutator(route, onInvalidate);
153
+ }
154
+ #createRouteQueryHook(route, options = {}) {
155
+ if (route.method !== "GET") throw new Error(`Only GET routes are supported for hooks. Route '${route.path}' is a ${route.method} route.`);
156
+ if (!route.outputSchema) throw new Error(`Output schema is required for GET routes. Route '${route.path}' has no output schema.`);
157
+ const baseUrl = this.#publicConfig.baseUrl ?? "";
158
+ const mountRoute = getMountRoute(this.#fragmentConfig);
159
+ const fetcher = this.#getFetcher();
160
+ const fetcherOptions = this.#getFetcherOptions();
161
+ async function callServerSideHandler(params) {
162
+ const { pathParams, queryParams } = params ?? {};
163
+ const normalizedPathParams = unwrapObject(pathParams);
164
+ const normalizedQueryParams = unwrapObject(queryParams) ?? {};
165
+ const filteredQueryParams = Object.fromEntries(Object.entries(normalizedQueryParams).filter(([_, value]) => value !== void 0));
166
+ const searchParams = new URLSearchParams(filteredQueryParams);
167
+ return await route.handler(RequestInputContext.fromSSRContext({
168
+ method: route.method,
169
+ path: route.path,
170
+ pathParams: normalizedPathParams,
171
+ searchParams
172
+ }), new RequestOutputContext(route.outputSchema));
173
+ }
174
+ async function executeQuery(params) {
175
+ const { pathParams, queryParams } = params ?? {};
176
+ if (typeof window === "undefined") return task(async () => callServerSideHandler({
177
+ pathParams,
178
+ queryParams
179
+ }));
180
+ const url = buildUrl({
181
+ baseUrl,
182
+ mountRoute,
183
+ path: route.path
184
+ }, {
185
+ pathParams,
186
+ queryParams
187
+ });
188
+ let response;
189
+ try {
190
+ response = fetcherOptions ? await fetcher(url, fetcherOptions) : await fetcher(url);
191
+ } catch (error) {
192
+ throw FragnoClientFetchError.fromUnknownFetchError(error);
193
+ }
194
+ if (!response.ok) throw await FragnoClientApiError.fromResponse(response);
195
+ return response;
196
+ }
197
+ return {
198
+ route,
199
+ store: (args) => {
200
+ const { path, query } = args ?? {};
201
+ const key = getCacheKey(route.method, route.path, {
202
+ pathParams: path,
203
+ queryParams: query
204
+ });
205
+ const store = this.#createFetcherStore(key, {
206
+ fetcher: async () => {
207
+ if (SSR_ENABLED) {
208
+ const initialData = getInitialData(key.map((d) => typeof d === "string" ? d : d.get()).join(""));
209
+ if (initialData) return initialData;
210
+ }
211
+ const response = await executeQuery({
212
+ pathParams: path,
213
+ queryParams: query
214
+ });
215
+ const isStreaming = isStreamingResponse(response);
216
+ if (!isStreaming) return response.json();
217
+ if (typeof window === "undefined") return [];
218
+ if (isStreaming === "ndjson") {
219
+ const { firstItem } = await handleNdjsonStreamingFirstItem(response, {
220
+ setData: (value) => {
221
+ store.set({
222
+ ...store.get(),
223
+ loading: !(Array.isArray(value) && value.length > 0),
224
+ data: value
225
+ });
226
+ },
227
+ setError: (value) => {
228
+ store.set({
229
+ ...store.get(),
230
+ error: value
231
+ });
232
+ }
233
+ });
234
+ return [firstItem];
235
+ }
236
+ if (isStreaming === "octet-stream") throw new Error("Octet-stream streaming is not supported.");
237
+ throw new Error("Unreachable");
238
+ },
239
+ onErrorRetry: options?.onErrorRetry,
240
+ dedupeTime: Infinity
241
+ });
242
+ if (typeof window === "undefined") addStore(store);
243
+ return store;
244
+ },
245
+ query: async (args) => {
246
+ const { path, query } = args ?? {};
247
+ const response = await executeQuery({
248
+ pathParams: path,
249
+ queryParams: query
250
+ });
251
+ const isStreaming = isStreamingResponse(response);
252
+ if (!isStreaming) return await response.json();
253
+ if (isStreaming === "ndjson") {
254
+ const { streamingPromise } = await handleNdjsonStreamingFirstItem(response);
255
+ return await streamingPromise;
256
+ }
257
+ if (isStreaming === "octet-stream") throw new Error("Octet-stream streaming is not supported.");
258
+ throw new Error("Unreachable");
259
+ },
260
+ [GET_HOOK_SYMBOL]: true
261
+ };
262
+ }
263
+ #createRouteQueryMutator(route, onInvalidate = (invalidate, params) => invalidate("GET", route.path, params)) {
264
+ const method = route.method;
265
+ const baseUrl = this.#publicConfig.baseUrl ?? "";
266
+ const mountRoute = getMountRoute(this.#fragmentConfig);
267
+ const fetcher = this.#getFetcher();
268
+ const fetcherOptions = this.#getFetcherOptions();
269
+ async function executeMutateQuery({ body, path, query }) {
270
+ if (typeof window === "undefined") return task(async () => route.handler(RequestInputContext.fromSSRContext({
271
+ inputSchema: route.inputSchema,
272
+ method,
273
+ path: route.path,
274
+ pathParams: path ?? {},
275
+ searchParams: new URLSearchParams(query),
276
+ body
277
+ }), new RequestOutputContext(route.outputSchema)));
278
+ const url = buildUrl({
279
+ baseUrl,
280
+ mountRoute,
281
+ path: route.path
282
+ }, {
283
+ pathParams: path,
284
+ queryParams: query
285
+ });
286
+ let response;
287
+ try {
288
+ response = await fetcher(url, {
289
+ ...fetcherOptions,
290
+ method,
291
+ body: body !== void 0 ? JSON.stringify(body) : void 0
292
+ });
293
+ } catch (error) {
294
+ throw FragnoClientFetchError.fromUnknownFetchError(error);
295
+ }
296
+ if (!response.ok) throw await FragnoClientApiError.fromResponse(response);
297
+ return response;
298
+ }
299
+ const mutatorStore = this.#createMutatorStore(async ({ data }) => {
300
+ if (typeof window === "undefined") {}
301
+ const { body, path, query } = data;
302
+ if (typeof body === "undefined" && route.inputSchema !== void 0) throw new Error("Body is required.");
303
+ const response = await executeMutateQuery({
304
+ body,
305
+ path,
306
+ query
307
+ });
308
+ onInvalidate(this.#invalidate.bind(this), {
309
+ pathParams: path ?? {},
310
+ queryParams: query
311
+ });
312
+ if (response.status === 201 || response.status === 204) return;
313
+ const isStreaming = isStreamingResponse(response);
314
+ if (!isStreaming) return response.json();
315
+ if (typeof window === "undefined") return [];
316
+ if (isStreaming === "ndjson") {
317
+ const { firstItem } = await handleNdjsonStreamingFirstItem(response, {
318
+ setData: (value) => {
319
+ mutatorStore.set({
320
+ ...mutatorStore.get(),
321
+ loading: !(Array.isArray(value) && value.length > 0),
322
+ data: value
323
+ });
324
+ },
325
+ setError: (value) => {
326
+ mutatorStore.set({
327
+ ...mutatorStore.get(),
328
+ error: value
329
+ });
330
+ }
331
+ });
332
+ return [firstItem];
333
+ }
334
+ if (isStreaming === "octet-stream") throw new Error("Octet-stream streaming is not supported.");
335
+ throw new Error("Unreachable");
336
+ }, { onError: (error) => {
337
+ console.error("Error in mutatorStore", error);
338
+ } });
339
+ const mutateQuery = (async (data) => {
340
+ const { body, path, query } = data;
341
+ if (typeof body === "undefined" && route.inputSchema !== void 0) throw new Error("Body is required for mutateQuery");
342
+ const response = await executeMutateQuery({
343
+ body,
344
+ path,
345
+ query
346
+ });
347
+ if (response.status === 201 || response.status === 204) return;
348
+ const isStreaming = isStreamingResponse(response);
349
+ if (!isStreaming) return response.json();
350
+ if (isStreaming === "ndjson") {
351
+ const { streamingPromise } = await handleNdjsonStreamingFirstItem(response);
352
+ return await streamingPromise;
353
+ }
354
+ if (isStreaming === "octet-stream") throw new Error("Octet-stream streaming is not supported for mutations");
355
+ throw new Error("Unreachable");
356
+ });
357
+ return {
358
+ route,
359
+ mutateQuery,
360
+ mutatorStore,
361
+ [MUTATOR_HOOK_SYMBOL]: true
362
+ };
363
+ }
364
+ #invalidate(method, path, params) {
365
+ const prefix = getCacheKey(method, path, {
366
+ pathParams: params?.pathParams,
367
+ queryParams: params?.queryParams
368
+ }).map((k) => typeof k === "string" ? k : k.get()).join("");
369
+ this.#invalidateKeys((key) => key.startsWith(prefix));
370
+ }
371
+ };
372
+ /**
373
+ * Create a client builder for fragments using the new fragment definition API.
374
+ * This is the same as createClientBuilder but works with FragmentDefinition.
375
+ */
376
+ function createClientBuilder(definition, publicConfig, routesOrFactories, authorFetcherConfig) {
377
+ const routes = resolveRouteFactories({
378
+ config: {},
379
+ deps: {},
380
+ services: {},
381
+ serviceDeps: {}
382
+ }, routesOrFactories);
383
+ const fragmentConfig = {
384
+ name: definition.name,
385
+ routes
386
+ };
387
+ const mountRoute = publicConfig.mountRoute ?? `/${definition.name}`;
388
+ const mergedFetcherConfig = mergeFetcherConfigs(authorFetcherConfig, publicConfig.fetcherConfig);
389
+ return new ClientBuilder({
390
+ ...publicConfig,
391
+ mountRoute,
392
+ fetcherConfig: mergedFetcherConfig
393
+ }, fragmentConfig);
394
+ }
395
+
396
+ //#endregion
397
+ export { ClientBuilder, FragnoClientApiError, FragnoClientError, FragnoClientFetchAbortError, FragnoClientFetchError, FragnoClientFetchNetworkError, FragnoClientUnknownApiError, buildUrl, createClientBuilder, getCacheKey, isGetHook, isMutatorHook, isStore };
398
+ //# sourceMappingURL=client.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"client.js","names":["#publicConfig","#fragmentConfig","#fetcherConfig","#cache","#createFetcherStore","#createMutatorStore","#invalidateKeys","#getFetcher","#getFetcherOptions","#createRouteQueryHook","#createRouteQueryMutator","response: Response","mutatorStore: FragnoClientMutatorData<\n NonGetHTTPMethod,\n TPath,\n TInputSchema,\n TOutputSchema,\n TErrorCode,\n TQueryParameters\n >[\"mutatorStore\"]","#invalidate","fragmentConfig: FragnoFragmentSharedConfig<FlattenRouteFactories<TRoutesOrFactories>>"],"sources":["../../src/client/client.ts"],"sourcesContent":["import { nanoquery, type FetcherStore, type MutatorStore } from \"@nanostores/query\";\nimport type { StandardSchemaV1 } from \"@standard-schema/spec\";\nimport { computed, task, type ReadableAtom, type Store } from \"nanostores\";\nimport type {\n FragnoRouteConfig,\n HTTPMethod,\n NonGetHTTPMethod,\n RequestThisContext,\n} from \"../api/api\";\nimport {\n buildPath,\n extractPathParams,\n type ExtractPathParams,\n type ExtractPathParamsOrWiden,\n type MaybeExtractPathParamsOrWiden,\n} from \"../api/internal/path\";\nimport { getMountRoute } from \"../api/internal/route\";\nimport { RequestInputContext } from \"../api/request-input-context\";\nimport { RequestOutputContext } from \"../api/request-output-context\";\nimport type {\n FetcherConfig,\n FragnoFragmentSharedConfig,\n FragnoPublicClientConfig,\n FragnoPublicConfig,\n} from \"../api/shared-types\";\nimport { FragnoClientApiError, FragnoClientError, FragnoClientFetchError } from \"./client-error\";\nimport type { InferOr } from \"../util/types-util\";\nimport { parseContentType } from \"../util/content-type\";\nimport {\n handleNdjsonStreamingFirstItem,\n type NdjsonStreamingStore,\n} from \"./internal/ndjson-streaming\";\nimport { addStore, getInitialData, SSR_ENABLED } from \"../util/ssr\";\nimport { unwrapObject } from \"../util/nanostores\";\nimport type { FragmentDefinition } from \"../api/fragment-definition-builder\";\nimport {\n type AnyRouteOrFactory,\n type FlattenRouteFactories,\n resolveRouteFactories,\n} from \"../api/route\";\nimport { mergeFetcherConfigs } from \"./internal/fetcher-merge\";\n\n/**\n * Symbols used to identify hook types\n */\nconst GET_HOOK_SYMBOL = Symbol(\"fragno-get-hook\");\nconst MUTATOR_HOOK_SYMBOL = Symbol(\"fragno-mutator-hook\");\nconst STORE_SYMBOL = Symbol(\"fragno-store\");\n\n/**\n * Extract only GET routes from a library config's routes array\n * @internal\n */\nexport type ExtractGetRoutes<\n T extends readonly FragnoRouteConfig<\n HTTPMethod,\n string,\n StandardSchemaV1 | undefined,\n StandardSchemaV1 | undefined,\n string,\n string\n >[],\n> = {\n [K in keyof T]: T[K] extends FragnoRouteConfig<\n infer Method,\n infer Path,\n infer Input,\n infer Output,\n infer ErrorCode,\n infer QueryParams\n >\n ? Method extends \"GET\"\n ? FragnoRouteConfig<Method, Path, Input, Output, ErrorCode, QueryParams>\n : never\n : never;\n}[number][];\n\n/**\n * Extract the path from a route configuration for a given method\n * @internal\n */\nexport type ExtractRoutePath<\n T extends readonly FragnoRouteConfig<\n HTTPMethod,\n string,\n StandardSchemaV1 | undefined,\n StandardSchemaV1 | undefined,\n string,\n string\n >[],\n TExpectedMethod extends HTTPMethod = HTTPMethod,\n> = {\n [K in keyof T]: T[K] extends FragnoRouteConfig<\n infer Method,\n infer Path,\n StandardSchemaV1 | undefined,\n StandardSchemaV1 | undefined,\n string,\n string\n >\n ? Method extends TExpectedMethod\n ? Path\n : never\n : never;\n}[number];\n\n/**\n * @internal\n */\nexport type ExtractGetRoutePaths<\n T extends readonly FragnoRouteConfig<\n HTTPMethod,\n string,\n StandardSchemaV1 | undefined,\n StandardSchemaV1 | undefined,\n string,\n string\n >[],\n> = ExtractRoutePath<T, \"GET\">;\n\n/**\n * @internal\n */\nexport type ExtractNonGetRoutePaths<\n T extends readonly FragnoRouteConfig<\n HTTPMethod,\n string,\n StandardSchemaV1 | undefined,\n StandardSchemaV1 | undefined,\n string,\n string\n >[],\n> = ExtractRoutePath<T, NonGetHTTPMethod>;\n\n/**\n * Extract the route configuration type(s) for a given path from a routes array.\n * Optionally narrow by HTTP method via the third type parameter.\n *\n * Defaults to extracting all methods for the matching path, producing a union\n * if multiple methods exist for the same path.\n * @internal\n */\nexport type ExtractRouteByPath<\n TRoutes extends readonly FragnoRouteConfig<\n HTTPMethod,\n string,\n StandardSchemaV1 | undefined,\n StandardSchemaV1 | undefined,\n string,\n string\n >[],\n TPath extends string,\n TMethod extends HTTPMethod = HTTPMethod,\n> = {\n [K in keyof TRoutes]: TRoutes[K] extends FragnoRouteConfig<\n infer M,\n TPath,\n infer Input,\n infer Output,\n infer ErrorCode,\n infer QueryParams\n >\n ? M extends TMethod\n ? FragnoRouteConfig<M, TPath, Input, Output, ErrorCode, QueryParams>\n : never\n : never;\n}[number];\n\n/**\n * Extract the output schema type for a specific route path from a routes array\n * @internal\n */\nexport type ExtractOutputSchemaForPath<\n TRoutes extends readonly FragnoRouteConfig<\n HTTPMethod,\n string,\n StandardSchemaV1 | undefined,\n StandardSchemaV1 | undefined\n >[],\n TPath extends string,\n> = {\n [K in keyof TRoutes]: TRoutes[K] extends FragnoRouteConfig<\n infer Method,\n TPath,\n StandardSchemaV1 | undefined,\n infer Output\n >\n ? Method extends \"GET\"\n ? Output\n : never\n : never;\n}[number];\n\n/**\n * Check if a path exists as a GET route in the routes array\n * @internal\n */\nexport type IsValidGetRoutePath<\n TRoutes extends readonly FragnoRouteConfig<\n HTTPMethod,\n string,\n StandardSchemaV1 | undefined,\n StandardSchemaV1 | undefined,\n string,\n string\n >[],\n TPath extends string,\n> = TPath extends ExtractGetRoutePaths<TRoutes> ? true : false;\n\n/**\n * Utility type to ensure only valid GET route paths can be used\n * @internal\n */\nexport type ValidateGetRoutePath<\n TRoutes extends readonly FragnoRouteConfig<\n HTTPMethod,\n string,\n StandardSchemaV1 | undefined,\n StandardSchemaV1 | undefined,\n string,\n string\n >[],\n TPath extends string,\n> =\n TPath extends ExtractGetRoutePaths<TRoutes>\n ? TPath\n : `Error: Path '${TPath}' is not a valid GET route. Available GET routes: ${ExtractGetRoutePaths<TRoutes>}`;\n\n/**\n * Helper type to check if a routes array has any GET routes\n * @internal\n */\nexport type HasGetRoutes<\n T extends readonly FragnoRouteConfig<\n HTTPMethod,\n string,\n StandardSchemaV1 | undefined,\n StandardSchemaV1 | undefined,\n string,\n string\n >[],\n> = ExtractGetRoutePaths<T> extends never ? false : true;\n\n/**\n * @internal\n */\nexport type ObjectContainingStoreField<T extends object> = T extends Store\n ? T\n : {\n [K in keyof T]: T[K] extends Store ? { [P in K]: T[P] } & Partial<Omit<T, K>> : never;\n }[keyof T] extends never\n ? never\n : T;\n\n/**\n * @internal\n */\nexport type FragnoStoreData<T extends object> = {\n obj: T;\n [STORE_SYMBOL]: true;\n};\n\nexport type FragnoClientHookData<\n TMethod extends HTTPMethod,\n TPath extends string,\n TOutputSchema extends StandardSchemaV1,\n TErrorCode extends string,\n TQueryParameters extends string,\n> = {\n route: FragnoRouteConfig<\n TMethod,\n TPath,\n StandardSchemaV1 | undefined,\n TOutputSchema,\n TErrorCode,\n TQueryParameters\n >;\n query(args?: {\n path?: MaybeExtractPathParamsOrWiden<TPath, string>;\n query?: Record<TQueryParameters, string | undefined>;\n }): Promise<StandardSchemaV1.InferOutput<TOutputSchema>>;\n store(args?: {\n path?: MaybeExtractPathParamsOrWiden<TPath, string | ReadableAtom<string>>;\n query?: Record<TQueryParameters, string | undefined | ReadableAtom<string | undefined>>;\n }): FetcherStore<StandardSchemaV1.InferOutput<TOutputSchema>, FragnoClientError<TErrorCode>>;\n [GET_HOOK_SYMBOL]: true;\n} & {\n // Phantom field that preserves the specific TOutputSchema type parameter\n // in the structural type. This makes the type covariant, allowing more\n // specific schema types (like z.ZodString) to be assigned to variables\n // typed with more general schema types (like StandardSchemaV1<any, any>)\n readonly _outputSchema?: TOutputSchema;\n};\n\nexport type FragnoClientMutatorData<\n TMethod extends NonGetHTTPMethod,\n TPath extends string,\n TInputSchema extends StandardSchemaV1 | undefined,\n TOutputSchema extends StandardSchemaV1 | undefined,\n TErrorCode extends string,\n TQueryParameters extends string,\n> = {\n route: FragnoRouteConfig<\n TMethod,\n TPath,\n TInputSchema,\n TOutputSchema,\n TErrorCode,\n TQueryParameters\n >;\n\n mutateQuery(args?: {\n body?: InferOr<TInputSchema, undefined>;\n path?: MaybeExtractPathParamsOrWiden<TPath, string>;\n query?: Record<TQueryParameters, string | undefined>;\n }): Promise<InferOr<TOutputSchema, undefined>>;\n\n mutatorStore: MutatorStore<\n {\n body?: InferOr<TInputSchema, undefined>;\n path?: MaybeExtractPathParamsOrWiden<TPath, string | ReadableAtom<string>>;\n query?: Record<TQueryParameters, string | undefined | ReadableAtom<string | undefined>>;\n },\n InferOr<TOutputSchema, undefined>,\n FragnoClientError<TErrorCode>\n >;\n [MUTATOR_HOOK_SYMBOL]: true;\n} & {\n readonly _inputSchema?: TInputSchema;\n readonly _outputSchema?: TOutputSchema;\n};\n\n/**\n * @internal\n */\nexport function buildUrl<TPath extends string>(\n config: {\n baseUrl?: string;\n mountRoute: string;\n path: TPath;\n },\n params: {\n pathParams?: Record<string, string | ReadableAtom<string>>;\n queryParams?: Record<string, string | undefined | ReadableAtom<string | undefined>>;\n },\n): string {\n const { baseUrl = \"\", mountRoute, path } = config;\n const { pathParams, queryParams } = params ?? {};\n\n const normalizedPathParams = unwrapObject(pathParams) as ExtractPathParams<TPath, string>;\n const normalizedQueryParams = unwrapObject(queryParams) ?? {};\n\n // Filter out undefined values to prevent URLSearchParams from converting them to string \"undefined\"\n const filteredQueryParams = Object.fromEntries(\n Object.entries(normalizedQueryParams).filter(([_, value]) => value !== undefined),\n ) as Record<string, string>;\n\n const searchParams = new URLSearchParams(filteredQueryParams);\n const builtPath = buildPath(path, normalizedPathParams ?? {});\n const search = searchParams.toString() ? `?${searchParams.toString()}` : \"\";\n return `${baseUrl}${mountRoute}${builtPath}${search}`;\n}\n\n/**\n * This method returns an array, which can be passed directly to nanostores.\n *\n * The returned array is always: path, pathParams (In order they appear in the path), queryParams (In alphabetical order)\n * Missing pathParams are replaced with \"<missing>\".\n * Atoms with undefined values are wrapped in computed atoms that map undefined to \"\" to avoid nanoquery treating the key as incomplete.\n * @param path\n * @param params\n * @returns\n * @internal\n */\nexport function getCacheKey<TMethod extends HTTPMethod, TPath extends string>(\n method: TMethod,\n path: TPath,\n params?: {\n pathParams?: Record<string, string | ReadableAtom<string>>;\n queryParams?: Record<string, string | undefined | ReadableAtom<string | undefined>>;\n },\n): (string | ReadableAtom<string>)[] {\n if (!params) {\n return [method, path];\n }\n\n const { pathParams, queryParams } = params;\n\n const pathParamNames = extractPathParams(path);\n const pathParamValues = pathParamNames.map((name) => pathParams?.[name] ?? \"<missing>\");\n\n const queryParamValues = queryParams\n ? Object.keys(queryParams)\n .sort()\n .map((key) => {\n const value = queryParams[key];\n // If it's an atom, wrap it to convert undefined to \"\"\n if (value && typeof value === \"object\" && \"get\" in value) {\n return computed(value as ReadableAtom<string | undefined>, (v) => v ?? \"\");\n }\n // Plain string value (or undefined)\n return value ?? \"\";\n })\n : [];\n\n return [method, path, ...pathParamValues, ...queryParamValues];\n}\n\nfunction isStreamingResponse(response: Response): false | \"ndjson\" | \"octet-stream\" {\n const contentType = parseContentType(response.headers.get(\"content-type\"));\n\n if (!contentType) {\n // Always assume 'normal' JSON by default.\n return false;\n }\n\n const isChunked = response.headers.get(\"transfer-encoding\") === \"chunked\";\n\n if (!isChunked) {\n return false;\n }\n\n if (contentType.subtype === \"octet-stream\") {\n // TODO(Wilco): This is not actually supported properly\n return \"octet-stream\";\n }\n\n if (contentType.subtype === \"x-ndjson\") {\n return \"ndjson\";\n }\n\n return false;\n}\n\n// Type guard to check if a hook is a GET hook\n/**\n * @internal\n */\nexport function isGetHook<\n TPath extends string,\n TOutputSchema extends StandardSchemaV1,\n TErrorCode extends string,\n TQueryParameters extends string,\n>(\n hook: unknown,\n): hook is FragnoClientHookData<\"GET\", TPath, TOutputSchema, TErrorCode, TQueryParameters> {\n return (\n typeof hook === \"object\" &&\n hook !== null &&\n GET_HOOK_SYMBOL in hook &&\n hook[GET_HOOK_SYMBOL] === true\n );\n}\n\n// Type guard to check if a hook is a mutator\n/**\n * @internal\n */\nexport function isMutatorHook<\n TMethod extends NonGetHTTPMethod,\n TPath extends string,\n TInputSchema extends StandardSchemaV1 | undefined,\n TOutputSchema extends StandardSchemaV1 | undefined,\n TErrorCode extends string,\n TQueryParameters extends string,\n>(\n hook: unknown,\n): hook is FragnoClientMutatorData<\n TMethod,\n TPath,\n TInputSchema,\n TOutputSchema,\n TErrorCode,\n TQueryParameters\n> {\n return (\n typeof hook === \"object\" &&\n hook !== null &&\n MUTATOR_HOOK_SYMBOL in hook &&\n hook[MUTATOR_HOOK_SYMBOL] === true\n );\n}\n\n/**\n * @internal\n */\nexport function isStore<TStore extends Store>(obj: unknown): obj is FragnoStoreData<TStore> {\n return (\n typeof obj === \"object\" && obj !== null && STORE_SYMBOL in obj && obj[STORE_SYMBOL] === true\n );\n}\n\ntype OnErrorRetryFn = (opts: {\n error: unknown;\n key: string;\n retryCount: number;\n}) => number | undefined;\n\nexport type CreateHookOptions = {\n /**\n * A function that will be called when an error occurs. Implements an exponential backoff strategy\n * when left undefined. When null, retries will be disabled. The number returned (> 0) by the\n * callback will determine in how many ms to retry next.\n */\n onErrorRetry?: OnErrorRetryFn | null;\n};\n\ntype OnInvalidateFn<TPath extends string> = (\n invalidate: <TInnerPath extends string>(\n method: HTTPMethod,\n path: TInnerPath,\n params: {\n pathParams?: MaybeExtractPathParamsOrWiden<TInnerPath, string>;\n queryParams?: Record<string, string>;\n },\n ) => void,\n params: {\n pathParams: MaybeExtractPathParamsOrWiden<TPath, string>;\n queryParams?: Record<string, string>;\n },\n) => void;\n\n/**\n * @internal\n */\nexport type CacheLine = {\n data: unknown;\n error: unknown;\n retryCount: number;\n created: number;\n expires: number;\n};\n\nexport class ClientBuilder<\n TRoutes extends readonly FragnoRouteConfig<\n HTTPMethod,\n string,\n StandardSchemaV1 | undefined,\n StandardSchemaV1 | undefined,\n string,\n string\n >[],\n TFragmentConfig extends FragnoFragmentSharedConfig<TRoutes>,\n> {\n #publicConfig: FragnoPublicClientConfig;\n #fragmentConfig: TFragmentConfig;\n #fetcherConfig?: FetcherConfig;\n\n #cache = new Map<string, CacheLine>();\n\n #createFetcherStore;\n #createMutatorStore;\n #invalidateKeys;\n\n constructor(publicConfig: FragnoPublicClientConfig, fragmentConfig: TFragmentConfig) {\n this.#publicConfig = publicConfig;\n this.#fragmentConfig = fragmentConfig;\n this.#fetcherConfig = publicConfig.fetcherConfig;\n\n const [createFetcherStore, createMutatorStore, { invalidateKeys }] = nanoquery({\n cache: this.#cache,\n });\n this.#createFetcherStore = createFetcherStore;\n this.#createMutatorStore = createMutatorStore;\n this.#invalidateKeys = invalidateKeys;\n }\n\n get cacheEntries(): Readonly<Record<string, CacheLine>> {\n return Object.fromEntries(this.#cache.entries());\n }\n\n createStore<const T extends object>(obj: T): FragnoStoreData<T> {\n return { obj: obj, [STORE_SYMBOL]: true };\n }\n\n /**\n * Build a URL for a custom backend call using the configured baseUrl and mountRoute.\n * Useful for fragment authors who need to make custom fetch calls.\n */\n buildUrl<TPath extends string>(\n path: TPath,\n params?: {\n path?: MaybeExtractPathParamsOrWiden<TPath, string>;\n query?: Record<string, string>;\n },\n ): string {\n const baseUrl = this.#publicConfig.baseUrl ?? \"\";\n const mountRoute = getMountRoute(this.#fragmentConfig);\n\n return buildUrl(\n { baseUrl, mountRoute, path },\n { pathParams: params?.path, queryParams: params?.query },\n );\n }\n\n /**\n * Get the configured fetcher function for custom backend calls.\n * Returns fetch with merged options applied.\n */\n getFetcher(): {\n fetcher: typeof fetch;\n defaultOptions: RequestInit | undefined;\n } {\n return {\n fetcher: this.#getFetcher(),\n defaultOptions: this.#getFetcherOptions(),\n };\n }\n\n #getFetcher(): typeof fetch {\n if (this.#fetcherConfig?.type === \"function\") {\n return this.#fetcherConfig.fetcher;\n }\n return fetch;\n }\n\n #getFetcherOptions(): RequestInit | undefined {\n if (this.#fetcherConfig?.type === \"options\") {\n return this.#fetcherConfig.options;\n }\n return undefined;\n }\n\n createHook<TPath extends ExtractGetRoutePaths<TFragmentConfig[\"routes\"]>>(\n path: ValidateGetRoutePath<TFragmentConfig[\"routes\"], TPath>,\n options?: CreateHookOptions,\n ): FragnoClientHookData<\n \"GET\",\n TPath,\n NonNullable<ExtractRouteByPath<TFragmentConfig[\"routes\"], TPath>[\"outputSchema\"]>,\n NonNullable<ExtractRouteByPath<TFragmentConfig[\"routes\"], TPath>[\"errorCodes\"]>[number],\n NonNullable<ExtractRouteByPath<TFragmentConfig[\"routes\"], TPath>[\"queryParameters\"]>[number]\n > {\n const route = this.#fragmentConfig.routes.find(\n (\n r,\n ): r is FragnoRouteConfig<\n \"GET\",\n TPath,\n StandardSchemaV1 | undefined,\n StandardSchemaV1,\n string,\n string\n > => r.path === path && r.method === \"GET\" && r.outputSchema !== undefined,\n );\n\n if (!route) {\n throw new Error(`Route '${path}' not found or is not a GET route with an output schema.`);\n }\n\n return this.#createRouteQueryHook(route, options);\n }\n\n createMutator<TPath extends ExtractNonGetRoutePaths<TFragmentConfig[\"routes\"]>>(\n method: NonGetHTTPMethod,\n path: TPath,\n onInvalidate?: OnInvalidateFn<TPath>,\n ): FragnoClientMutatorData<\n NonGetHTTPMethod, // TODO: This can be any Method, but should be related to TPath\n TPath,\n ExtractRouteByPath<TFragmentConfig[\"routes\"], TPath>[\"inputSchema\"],\n ExtractRouteByPath<TFragmentConfig[\"routes\"], TPath>[\"outputSchema\"],\n NonNullable<ExtractRouteByPath<TFragmentConfig[\"routes\"], TPath>[\"errorCodes\"]>[number],\n NonNullable<ExtractRouteByPath<TFragmentConfig[\"routes\"], TPath>[\"queryParameters\"]>[number]\n > {\n type TRoute = ExtractRouteByPath<TFragmentConfig[\"routes\"], TPath>;\n\n const route = this.#fragmentConfig.routes.find(\n (\n r,\n ): r is FragnoRouteConfig<\n NonGetHTTPMethod,\n TPath,\n TRoute[\"inputSchema\"],\n TRoute[\"outputSchema\"],\n string,\n string\n > => r.method !== \"GET\" && r.path === path && r.method === method,\n );\n\n if (!route) {\n throw new Error(\n `Route '${path}' not found or is a GET route with an input and output schema.`,\n );\n }\n\n return this.#createRouteQueryMutator(route, onInvalidate);\n }\n\n #createRouteQueryHook<\n TPath extends string,\n TInputSchema extends StandardSchemaV1 | undefined,\n TOutputSchema extends StandardSchemaV1,\n TErrorCode extends string,\n TQueryParameters extends string,\n >(\n route: FragnoRouteConfig<\n \"GET\",\n TPath,\n TInputSchema,\n TOutputSchema,\n TErrorCode,\n TQueryParameters\n >,\n options: CreateHookOptions = {},\n ): FragnoClientHookData<\"GET\", TPath, TOutputSchema, TErrorCode, TQueryParameters> {\n if (route.method !== \"GET\") {\n throw new Error(\n `Only GET routes are supported for hooks. Route '${route.path}' is a ${route.method} route.`,\n );\n }\n\n if (!route.outputSchema) {\n throw new Error(\n `Output schema is required for GET routes. Route '${route.path}' has no output schema.`,\n );\n }\n\n const baseUrl = this.#publicConfig.baseUrl ?? \"\";\n const mountRoute = getMountRoute(this.#fragmentConfig);\n const fetcher = this.#getFetcher();\n const fetcherOptions = this.#getFetcherOptions();\n\n async function callServerSideHandler(params: {\n pathParams?: Record<string, string | ReadableAtom<string>>;\n queryParams?: Record<string, string | undefined | ReadableAtom<string | undefined>>;\n }): Promise<Response> {\n const { pathParams, queryParams } = params ?? {};\n\n const normalizedPathParams = unwrapObject(pathParams) as ExtractPathParams<TPath, string>;\n const normalizedQueryParams = unwrapObject(queryParams) ?? {};\n\n // Filter out undefined values to prevent URLSearchParams from converting them to string \"undefined\"\n const filteredQueryParams = Object.fromEntries(\n Object.entries(normalizedQueryParams).filter(([_, value]) => value !== undefined),\n ) as Record<string, string>;\n\n const searchParams = new URLSearchParams(filteredQueryParams);\n\n const result = await route.handler(\n RequestInputContext.fromSSRContext({\n method: route.method,\n path: route.path,\n pathParams: normalizedPathParams,\n searchParams,\n }),\n new RequestOutputContext(route.outputSchema),\n );\n\n return result;\n }\n\n async function executeQuery(params?: {\n pathParams?: Record<string, string | ReadableAtom<string>>;\n queryParams?: Record<string, string | undefined | ReadableAtom<string | undefined>>;\n }): Promise<Response> {\n const { pathParams, queryParams } = params ?? {};\n\n if (typeof window === \"undefined\") {\n return task(async () => callServerSideHandler({ pathParams, queryParams }));\n }\n\n const url = buildUrl({ baseUrl, mountRoute, path: route.path }, { pathParams, queryParams });\n\n let response: Response;\n try {\n response = fetcherOptions ? await fetcher(url, fetcherOptions) : await fetcher(url);\n } catch (error) {\n throw FragnoClientFetchError.fromUnknownFetchError(error);\n }\n\n if (!response.ok) {\n throw await FragnoClientApiError.fromResponse<TErrorCode>(response);\n }\n\n return response;\n }\n\n return {\n route,\n store: (args) => {\n const { path, query } = args ?? {};\n\n const key = getCacheKey(route.method, route.path, {\n pathParams: path,\n queryParams: query,\n });\n\n const store = this.#createFetcherStore<\n StandardSchemaV1.InferOutput<TOutputSchema>,\n FragnoClientError<TErrorCode>\n >(key, {\n fetcher: async (): Promise<StandardSchemaV1.InferOutput<TOutputSchema>> => {\n if (SSR_ENABLED) {\n const initialData = getInitialData(\n key.map((d) => (typeof d === \"string\" ? d : d.get())).join(\"\"),\n );\n\n if (initialData) {\n return initialData;\n }\n }\n\n const response = await executeQuery({ pathParams: path, queryParams: query });\n const isStreaming = isStreamingResponse(response);\n\n if (!isStreaming) {\n return response.json() as Promise<StandardSchemaV1.InferOutput<TOutputSchema>>;\n }\n\n if (typeof window === \"undefined\") {\n return [];\n }\n\n if (isStreaming === \"ndjson\") {\n const storeAdapter: NdjsonStreamingStore<TOutputSchema, TErrorCode> = {\n setData: (value) => {\n store.set({\n ...store.get(),\n loading: !(Array.isArray(value) && value.length > 0),\n data: value as InferOr<TOutputSchema, undefined>,\n });\n },\n setError: (value) => {\n store.set({\n ...store.get(),\n error: value,\n });\n },\n };\n\n // Start streaming in background and return first item\n const { firstItem } = await handleNdjsonStreamingFirstItem(response, storeAdapter);\n return [firstItem];\n }\n\n if (isStreaming === \"octet-stream\") {\n // TODO(Wilco): Implement this\n throw new Error(\"Octet-stream streaming is not supported.\");\n }\n\n throw new Error(\"Unreachable\");\n },\n\n onErrorRetry: options?.onErrorRetry,\n dedupeTime: Infinity,\n });\n\n if (typeof window === \"undefined\") {\n addStore(store);\n }\n\n return store;\n },\n query: async (args) => {\n const { path, query } = args ?? {};\n\n const response = await executeQuery({ pathParams: path, queryParams: query });\n\n const isStreaming = isStreamingResponse(response);\n\n if (!isStreaming) {\n return (await response.json()) as StandardSchemaV1.InferOutput<TOutputSchema>;\n }\n\n if (isStreaming === \"ndjson\") {\n const { streamingPromise } = await handleNdjsonStreamingFirstItem(response);\n // Resolves once the stream is done\n return await streamingPromise;\n }\n\n if (isStreaming === \"octet-stream\") {\n // TODO(Wilco): Implement this\n throw new Error(\"Octet-stream streaming is not supported.\");\n }\n\n throw new Error(\"Unreachable\");\n },\n [GET_HOOK_SYMBOL]: true,\n };\n }\n\n #createRouteQueryMutator<\n TPath extends string,\n TInputSchema extends StandardSchemaV1 | undefined,\n TOutputSchema extends StandardSchemaV1 | undefined,\n TErrorCode extends string,\n TQueryParameters extends string,\n >(\n route: FragnoRouteConfig<\n NonGetHTTPMethod,\n TPath,\n TInputSchema,\n TOutputSchema,\n TErrorCode,\n TQueryParameters\n >,\n onInvalidate: OnInvalidateFn<TPath> = (invalidate, params) =>\n invalidate(\"GET\", route.path, params),\n ): FragnoClientMutatorData<\n NonGetHTTPMethod,\n TPath,\n TInputSchema,\n TOutputSchema,\n TErrorCode,\n TQueryParameters\n > {\n const method = route.method;\n\n const baseUrl = this.#publicConfig.baseUrl ?? \"\";\n const mountRoute = getMountRoute(this.#fragmentConfig);\n const fetcher = this.#getFetcher();\n const fetcherOptions = this.#getFetcherOptions();\n\n async function executeMutateQuery({\n body,\n path,\n query,\n }: {\n body?: InferOr<TInputSchema, undefined>;\n path?: ExtractPathParamsOrWiden<TPath, string>;\n query?: Record<string, string>;\n }): Promise<Response> {\n if (typeof window === \"undefined\") {\n return task(async () =>\n route.handler(\n RequestInputContext.fromSSRContext({\n inputSchema: route.inputSchema,\n method,\n path: route.path,\n pathParams: (path ?? {}) as ExtractPathParams<TPath, string>,\n searchParams: new URLSearchParams(query),\n body,\n }),\n new RequestOutputContext(route.outputSchema),\n ),\n );\n }\n\n const url = buildUrl(\n { baseUrl, mountRoute, path: route.path },\n { pathParams: path, queryParams: query },\n );\n\n let response: Response;\n try {\n const requestOptions: RequestInit = {\n ...fetcherOptions,\n method,\n body: body !== undefined ? JSON.stringify(body) : undefined,\n };\n response = await fetcher(url, requestOptions);\n } catch (error) {\n throw FragnoClientFetchError.fromUnknownFetchError(error);\n }\n\n if (!response.ok) {\n throw await FragnoClientApiError.fromResponse<TErrorCode>(response);\n }\n\n return response;\n }\n\n const mutatorStore: FragnoClientMutatorData<\n NonGetHTTPMethod,\n TPath,\n TInputSchema,\n TOutputSchema,\n TErrorCode,\n TQueryParameters\n >[\"mutatorStore\"] = this.#createMutatorStore(\n async ({ data }) => {\n if (typeof window === \"undefined\") {\n // TODO(Wilco): Handle server-side rendering.\n }\n\n const { body, path, query } = data as {\n body?: InferOr<TInputSchema, undefined>;\n path?: ExtractPathParamsOrWiden<TPath, string>;\n query?: Record<string, string>;\n };\n\n if (typeof body === \"undefined\" && route.inputSchema !== undefined) {\n throw new Error(\"Body is required.\");\n }\n\n const response = await executeMutateQuery({ body, path, query });\n\n onInvalidate(this.#invalidate.bind(this), {\n pathParams: (path ?? {}) as MaybeExtractPathParamsOrWiden<TPath, string>,\n queryParams: query,\n });\n\n if (response.status === 201 || response.status === 204) {\n return undefined;\n }\n\n const isStreaming = isStreamingResponse(response);\n\n if (!isStreaming) {\n return response.json();\n }\n\n if (typeof window === \"undefined\") {\n return [];\n }\n\n if (isStreaming === \"ndjson\") {\n const storeAdapter: NdjsonStreamingStore<NonNullable<TOutputSchema>, TErrorCode> = {\n setData: (value) => {\n mutatorStore.set({\n ...mutatorStore.get(),\n loading: !(Array.isArray(value) && value.length > 0),\n data: value as InferOr<TOutputSchema, undefined>,\n });\n },\n setError: (value) => {\n mutatorStore.set({\n ...mutatorStore.get(),\n error: value,\n });\n },\n };\n\n // Start streaming in background and return first item\n const { firstItem } = await handleNdjsonStreamingFirstItem(response, storeAdapter);\n\n // Return the first item immediately. The streaming will continue in the background\n return [firstItem];\n }\n\n if (isStreaming === \"octet-stream\") {\n // TODO(Wilco): Implement this\n throw new Error(\"Octet-stream streaming is not supported.\");\n }\n\n throw new Error(\"Unreachable\");\n },\n {\n onError: (error) => {\n console.error(\"Error in mutatorStore\", error);\n },\n },\n );\n\n const mutateQuery = (async (data) => {\n // TypeScript infers the fields to not exist, even though they might\n const { body, path, query } = data as {\n body?: InferOr<TInputSchema, undefined>;\n path?: ExtractPathParamsOrWiden<TPath, string>;\n query?: Record<string, string>;\n };\n\n if (typeof body === \"undefined\" && route.inputSchema !== undefined) {\n throw new Error(\"Body is required for mutateQuery\");\n }\n\n const response = await executeMutateQuery({ body, path, query });\n\n if (response.status === 201 || response.status === 204) {\n return undefined;\n }\n\n const isStreaming = isStreamingResponse(response);\n\n if (!isStreaming) {\n return response.json();\n }\n\n if (isStreaming === \"ndjson\") {\n const { streamingPromise } = await handleNdjsonStreamingFirstItem(response);\n // Resolves once the stream is done, i.e. we block until done\n return await streamingPromise;\n }\n\n if (isStreaming === \"octet-stream\") {\n throw new Error(\"Octet-stream streaming is not supported for mutations\");\n }\n\n throw new Error(\"Unreachable\");\n }) satisfies FragnoClientMutatorData<\n NonGetHTTPMethod,\n TPath,\n TInputSchema,\n TOutputSchema,\n TErrorCode,\n TQueryParameters\n >[\"mutateQuery\"];\n\n return {\n route,\n mutateQuery,\n mutatorStore,\n [MUTATOR_HOOK_SYMBOL]: true,\n };\n }\n\n #invalidate<TPath extends string>(\n method: HTTPMethod,\n path: TPath,\n params: {\n pathParams?: MaybeExtractPathParamsOrWiden<TPath, string>;\n queryParams?: Record<string, string>;\n },\n ) {\n const prefixArray = getCacheKey(method, path, {\n pathParams: params?.pathParams,\n queryParams: params?.queryParams,\n });\n\n const prefix = prefixArray.map((k) => (typeof k === \"string\" ? k : k.get())).join(\"\");\n\n this.#invalidateKeys((key) => key.startsWith(prefix));\n }\n}\n\n/**\n * Create a client builder for fragments using the new fragment definition API.\n * This is the same as createClientBuilder but works with FragmentDefinition.\n */\nexport function createClientBuilder<\n TConfig,\n TOptions extends FragnoPublicConfig,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies,\n TPrivateServices,\n TServiceThisContext extends RequestThisContext,\n THandlerThisContext extends RequestThisContext,\n TRequestStorage,\n const TRoutesOrFactories extends readonly AnyRouteOrFactory[],\n>(\n definition: FragmentDefinition<\n TConfig,\n TOptions,\n TDeps,\n TBaseServices,\n TServices,\n TServiceDependencies,\n TPrivateServices,\n TServiceThisContext,\n THandlerThisContext,\n TRequestStorage\n >,\n publicConfig: FragnoPublicClientConfig,\n routesOrFactories: TRoutesOrFactories,\n authorFetcherConfig?: FetcherConfig,\n): ClientBuilder<\n FlattenRouteFactories<TRoutesOrFactories>,\n FragnoFragmentSharedConfig<FlattenRouteFactories<TRoutesOrFactories>>\n> {\n // For client-side, we resolve route factories with dummy context\n // This will be removed by the bundle plugin anyway\n const dummyContext = {\n config: {} as TConfig,\n deps: {} as TDeps,\n services: {} as TBaseServices & TServices,\n serviceDeps: {},\n };\n\n const routes = resolveRouteFactories(dummyContext, routesOrFactories);\n\n const fragmentConfig: FragnoFragmentSharedConfig<FlattenRouteFactories<TRoutesOrFactories>> = {\n name: definition.name,\n routes,\n };\n\n const mountRoute = publicConfig.mountRoute ?? `/${definition.name}`;\n const mergedFetcherConfig = mergeFetcherConfigs(authorFetcherConfig, publicConfig.fetcherConfig);\n const fullPublicConfig = {\n ...publicConfig,\n mountRoute,\n fetcherConfig: mergedFetcherConfig,\n };\n\n return new ClientBuilder(fullPublicConfig, fragmentConfig);\n}\n\nexport * from \"./client-error\";\nexport type { FetcherConfig, FragnoPublicClientConfig } from \"../api/shared-types\";\nexport type { FragnoFragmentSharedConfig } from \"../api/fragment-instantiator\";\n"],"mappings":";;;;;;;;;;;;;;;;;;AA6CA,MAAM,kBAAkB,OAAO,kBAAkB;AACjD,MAAM,sBAAsB,OAAO,sBAAsB;AACzD,MAAM,eAAe,OAAO,eAAe;;;;AAgS3C,SAAgB,SACd,QAKA,QAIQ;CACR,MAAM,EAAE,UAAU,IAAI,YAAY,SAAS;CAC3C,MAAM,EAAE,YAAY,gBAAgB,UAAU,EAAE;CAEhD,MAAM,uBAAuB,aAAa,WAAW;CACrD,MAAM,wBAAwB,aAAa,YAAY,IAAI,EAAE;CAG7D,MAAM,sBAAsB,OAAO,YACjC,OAAO,QAAQ,sBAAsB,CAAC,QAAQ,CAAC,GAAG,WAAW,UAAU,OAAU,CAClF;CAED,MAAM,eAAe,IAAI,gBAAgB,oBAAoB;AAG7D,QAAO,GAAG,UAAU,aAFF,UAAU,MAAM,wBAAwB,EAAE,CAAC,GAC9C,aAAa,UAAU,GAAG,IAAI,aAAa,UAAU,KAAK;;;;;;;;;;;;;AAe3E,SAAgB,YACd,QACA,MACA,QAImC;AACnC,KAAI,CAAC,OACH,QAAO,CAAC,QAAQ,KAAK;CAGvB,MAAM,EAAE,YAAY,gBAAgB;CAGpC,MAAM,kBADiB,kBAAkB,KAAK,CACP,KAAK,SAAS,aAAa,SAAS,YAAY;CAEvF,MAAM,mBAAmB,cACrB,OAAO,KAAK,YAAY,CACrB,MAAM,CACN,KAAK,QAAQ;EACZ,MAAM,QAAQ,YAAY;AAE1B,MAAI,SAAS,OAAO,UAAU,YAAY,SAAS,MACjD,QAAO,SAAS,QAA4C,MAAM,KAAK,GAAG;AAG5E,SAAO,SAAS;GAChB,GACJ,EAAE;AAEN,QAAO;EAAC;EAAQ;EAAM,GAAG;EAAiB,GAAG;EAAiB;;AAGhE,SAAS,oBAAoB,UAAuD;CAClF,MAAM,cAAc,iBAAiB,SAAS,QAAQ,IAAI,eAAe,CAAC;AAE1E,KAAI,CAAC,YAEH,QAAO;AAKT,KAAI,EAFc,SAAS,QAAQ,IAAI,oBAAoB,KAAK,WAG9D,QAAO;AAGT,KAAI,YAAY,YAAY,eAE1B,QAAO;AAGT,KAAI,YAAY,YAAY,WAC1B,QAAO;AAGT,QAAO;;;;;AAOT,SAAgB,UAMd,MACyF;AACzF,QACE,OAAO,SAAS,YAChB,SAAS,QACT,mBAAmB,QACnB,KAAK,qBAAqB;;;;;AAQ9B,SAAgB,cAQd,MAQA;AACA,QACE,OAAO,SAAS,YAChB,SAAS,QACT,uBAAuB,QACvB,KAAK,yBAAyB;;;;;AAOlC,SAAgB,QAA8B,KAA8C;AAC1F,QACE,OAAO,QAAQ,YAAY,QAAQ,QAAQ,gBAAgB,OAAO,IAAI,kBAAkB;;AA6C5F,IAAa,gBAAb,MAUE;CACA;CACA;CACA;CAEA,yBAAS,IAAI,KAAwB;CAErC;CACA;CACA;CAEA,YAAY,cAAwC,gBAAiC;AACnF,QAAKA,eAAgB;AACrB,QAAKC,iBAAkB;AACvB,QAAKC,gBAAiB,aAAa;EAEnC,MAAM,CAAC,oBAAoB,oBAAoB,EAAE,oBAAoB,UAAU,EAC7E,OAAO,MAAKC,OACb,CAAC;AACF,QAAKC,qBAAsB;AAC3B,QAAKC,qBAAsB;AAC3B,QAAKC,iBAAkB;;CAGzB,IAAI,eAAoD;AACtD,SAAO,OAAO,YAAY,MAAKH,MAAO,SAAS,CAAC;;CAGlD,YAAoC,KAA4B;AAC9D,SAAO;GAAO;IAAM,eAAe;GAAM;;;;;;CAO3C,SACE,MACA,QAIQ;AAIR,SAAO,SACL;GAAE,SAJY,MAAKH,aAAc,WAAW;GAIjC,YAHM,cAAc,MAAKC,eAAgB;GAG7B;GAAM,EAC7B;GAAE,YAAY,QAAQ;GAAM,aAAa,QAAQ;GAAO,CACzD;;;;;;CAOH,aAGE;AACA,SAAO;GACL,SAAS,MAAKM,YAAa;GAC3B,gBAAgB,MAAKC,mBAAoB;GAC1C;;CAGH,cAA4B;AAC1B,MAAI,MAAKN,eAAgB,SAAS,WAChC,QAAO,MAAKA,cAAe;AAE7B,SAAO;;CAGT,qBAA8C;AAC5C,MAAI,MAAKA,eAAgB,SAAS,UAChC,QAAO,MAAKA,cAAe;;CAK/B,WACE,MACA,SAOA;EACA,MAAM,QAAQ,MAAKD,eAAgB,OAAO,MAEtC,MAQG,EAAE,SAAS,QAAQ,EAAE,WAAW,SAAS,EAAE,iBAAiB,OAClE;AAED,MAAI,CAAC,MACH,OAAM,IAAI,MAAM,UAAU,KAAK,0DAA0D;AAG3F,SAAO,MAAKQ,qBAAsB,OAAO,QAAQ;;CAGnD,cACE,QACA,MACA,cAQA;EAGA,MAAM,QAAQ,MAAKR,eAAgB,OAAO,MAEtC,MAQG,EAAE,WAAW,SAAS,EAAE,SAAS,QAAQ,EAAE,WAAW,OAC5D;AAED,MAAI,CAAC,MACH,OAAM,IAAI,MACR,UAAU,KAAK,gEAChB;AAGH,SAAO,MAAKS,wBAAyB,OAAO,aAAa;;CAG3D,sBAOE,OAQA,UAA6B,EAAE,EACkD;AACjF,MAAI,MAAM,WAAW,MACnB,OAAM,IAAI,MACR,mDAAmD,MAAM,KAAK,SAAS,MAAM,OAAO,SACrF;AAGH,MAAI,CAAC,MAAM,aACT,OAAM,IAAI,MACR,oDAAoD,MAAM,KAAK,yBAChE;EAGH,MAAM,UAAU,MAAKV,aAAc,WAAW;EAC9C,MAAM,aAAa,cAAc,MAAKC,eAAgB;EACtD,MAAM,UAAU,MAAKM,YAAa;EAClC,MAAM,iBAAiB,MAAKC,mBAAoB;EAEhD,eAAe,sBAAsB,QAGf;GACpB,MAAM,EAAE,YAAY,gBAAgB,UAAU,EAAE;GAEhD,MAAM,uBAAuB,aAAa,WAAW;GACrD,MAAM,wBAAwB,aAAa,YAAY,IAAI,EAAE;GAG7D,MAAM,sBAAsB,OAAO,YACjC,OAAO,QAAQ,sBAAsB,CAAC,QAAQ,CAAC,GAAG,WAAW,UAAU,OAAU,CAClF;GAED,MAAM,eAAe,IAAI,gBAAgB,oBAAoB;AAY7D,UAVe,MAAM,MAAM,QACzB,oBAAoB,eAAe;IACjC,QAAQ,MAAM;IACd,MAAM,MAAM;IACZ,YAAY;IACZ;IACD,CAAC,EACF,IAAI,qBAAqB,MAAM,aAAa,CAC7C;;EAKH,eAAe,aAAa,QAGN;GACpB,MAAM,EAAE,YAAY,gBAAgB,UAAU,EAAE;AAEhD,OAAI,OAAO,WAAW,YACpB,QAAO,KAAK,YAAY,sBAAsB;IAAE;IAAY;IAAa,CAAC,CAAC;GAG7E,MAAM,MAAM,SAAS;IAAE;IAAS;IAAY,MAAM,MAAM;IAAM,EAAE;IAAE;IAAY;IAAa,CAAC;GAE5F,IAAIG;AACJ,OAAI;AACF,eAAW,iBAAiB,MAAM,QAAQ,KAAK,eAAe,GAAG,MAAM,QAAQ,IAAI;YAC5E,OAAO;AACd,UAAM,uBAAuB,sBAAsB,MAAM;;AAG3D,OAAI,CAAC,SAAS,GACZ,OAAM,MAAM,qBAAqB,aAAyB,SAAS;AAGrE,UAAO;;AAGT,SAAO;GACL;GACA,QAAQ,SAAS;IACf,MAAM,EAAE,MAAM,UAAU,QAAQ,EAAE;IAElC,MAAM,MAAM,YAAY,MAAM,QAAQ,MAAM,MAAM;KAChD,YAAY;KACZ,aAAa;KACd,CAAC;IAEF,MAAM,QAAQ,MAAKP,mBAGjB,KAAK;KACL,SAAS,YAAkE;AACzE,UAAI,aAAa;OACf,MAAM,cAAc,eAClB,IAAI,KAAK,MAAO,OAAO,MAAM,WAAW,IAAI,EAAE,KAAK,CAAE,CAAC,KAAK,GAAG,CAC/D;AAED,WAAI,YACF,QAAO;;MAIX,MAAM,WAAW,MAAM,aAAa;OAAE,YAAY;OAAM,aAAa;OAAO,CAAC;MAC7E,MAAM,cAAc,oBAAoB,SAAS;AAEjD,UAAI,CAAC,YACH,QAAO,SAAS,MAAM;AAGxB,UAAI,OAAO,WAAW,YACpB,QAAO,EAAE;AAGX,UAAI,gBAAgB,UAAU;OAkB5B,MAAM,EAAE,cAAc,MAAM,+BAA+B,UAjBW;QACpE,UAAU,UAAU;AAClB,eAAM,IAAI;UACR,GAAG,MAAM,KAAK;UACd,SAAS,EAAE,MAAM,QAAQ,MAAM,IAAI,MAAM,SAAS;UAClD,MAAM;UACP,CAAC;;QAEJ,WAAW,UAAU;AACnB,eAAM,IAAI;UACR,GAAG,MAAM,KAAK;UACd,OAAO;UACR,CAAC;;QAEL,CAGiF;AAClF,cAAO,CAAC,UAAU;;AAGpB,UAAI,gBAAgB,eAElB,OAAM,IAAI,MAAM,2CAA2C;AAG7D,YAAM,IAAI,MAAM,cAAc;;KAGhC,cAAc,SAAS;KACvB,YAAY;KACb,CAAC;AAEF,QAAI,OAAO,WAAW,YACpB,UAAS,MAAM;AAGjB,WAAO;;GAET,OAAO,OAAO,SAAS;IACrB,MAAM,EAAE,MAAM,UAAU,QAAQ,EAAE;IAElC,MAAM,WAAW,MAAM,aAAa;KAAE,YAAY;KAAM,aAAa;KAAO,CAAC;IAE7E,MAAM,cAAc,oBAAoB,SAAS;AAEjD,QAAI,CAAC,YACH,QAAQ,MAAM,SAAS,MAAM;AAG/B,QAAI,gBAAgB,UAAU;KAC5B,MAAM,EAAE,qBAAqB,MAAM,+BAA+B,SAAS;AAE3E,YAAO,MAAM;;AAGf,QAAI,gBAAgB,eAElB,OAAM,IAAI,MAAM,2CAA2C;AAG7D,UAAM,IAAI,MAAM,cAAc;;IAE/B,kBAAkB;GACpB;;CAGH,yBAOE,OAQA,gBAAuC,YAAY,WACjD,WAAW,OAAO,MAAM,MAAM,OAAO,EAQvC;EACA,MAAM,SAAS,MAAM;EAErB,MAAM,UAAU,MAAKJ,aAAc,WAAW;EAC9C,MAAM,aAAa,cAAc,MAAKC,eAAgB;EACtD,MAAM,UAAU,MAAKM,YAAa;EAClC,MAAM,iBAAiB,MAAKC,mBAAoB;EAEhD,eAAe,mBAAmB,EAChC,MACA,MACA,SAKoB;AACpB,OAAI,OAAO,WAAW,YACpB,QAAO,KAAK,YACV,MAAM,QACJ,oBAAoB,eAAe;IACjC,aAAa,MAAM;IACnB;IACA,MAAM,MAAM;IACZ,YAAa,QAAQ,EAAE;IACvB,cAAc,IAAI,gBAAgB,MAAM;IACxC;IACD,CAAC,EACF,IAAI,qBAAqB,MAAM,aAAa,CAC7C,CACF;GAGH,MAAM,MAAM,SACV;IAAE;IAAS;IAAY,MAAM,MAAM;IAAM,EACzC;IAAE,YAAY;IAAM,aAAa;IAAO,CACzC;GAED,IAAIG;AACJ,OAAI;AAMF,eAAW,MAAM,QAAQ,KALW;KAClC,GAAG;KACH;KACA,MAAM,SAAS,SAAY,KAAK,UAAU,KAAK,GAAG;KACnD,CAC4C;YACtC,OAAO;AACd,UAAM,uBAAuB,sBAAsB,MAAM;;AAG3D,OAAI,CAAC,SAAS,GACZ,OAAM,MAAM,qBAAqB,aAAyB,SAAS;AAGrE,UAAO;;EAGT,MAAMC,eAOc,MAAKP,mBACvB,OAAO,EAAE,WAAW;AAClB,OAAI,OAAO,WAAW,aAAa;GAInC,MAAM,EAAE,MAAM,MAAM,UAAU;AAM9B,OAAI,OAAO,SAAS,eAAe,MAAM,gBAAgB,OACvD,OAAM,IAAI,MAAM,oBAAoB;GAGtC,MAAM,WAAW,MAAM,mBAAmB;IAAE;IAAM;IAAM;IAAO,CAAC;AAEhE,gBAAa,MAAKQ,WAAY,KAAK,KAAK,EAAE;IACxC,YAAa,QAAQ,EAAE;IACvB,aAAa;IACd,CAAC;AAEF,OAAI,SAAS,WAAW,OAAO,SAAS,WAAW,IACjD;GAGF,MAAM,cAAc,oBAAoB,SAAS;AAEjD,OAAI,CAAC,YACH,QAAO,SAAS,MAAM;AAGxB,OAAI,OAAO,WAAW,YACpB,QAAO,EAAE;AAGX,OAAI,gBAAgB,UAAU;IAkB5B,MAAM,EAAE,cAAc,MAAM,+BAA+B,UAjBwB;KACjF,UAAU,UAAU;AAClB,mBAAa,IAAI;OACf,GAAG,aAAa,KAAK;OACrB,SAAS,EAAE,MAAM,QAAQ,MAAM,IAAI,MAAM,SAAS;OAClD,MAAM;OACP,CAAC;;KAEJ,WAAW,UAAU;AACnB,mBAAa,IAAI;OACf,GAAG,aAAa,KAAK;OACrB,OAAO;OACR,CAAC;;KAEL,CAGiF;AAGlF,WAAO,CAAC,UAAU;;AAGpB,OAAI,gBAAgB,eAElB,OAAM,IAAI,MAAM,2CAA2C;AAG7D,SAAM,IAAI,MAAM,cAAc;KAEhC,EACE,UAAU,UAAU;AAClB,WAAQ,MAAM,yBAAyB,MAAM;KAEhD,CACF;EAED,MAAM,eAAe,OAAO,SAAS;GAEnC,MAAM,EAAE,MAAM,MAAM,UAAU;AAM9B,OAAI,OAAO,SAAS,eAAe,MAAM,gBAAgB,OACvD,OAAM,IAAI,MAAM,mCAAmC;GAGrD,MAAM,WAAW,MAAM,mBAAmB;IAAE;IAAM;IAAM;IAAO,CAAC;AAEhE,OAAI,SAAS,WAAW,OAAO,SAAS,WAAW,IACjD;GAGF,MAAM,cAAc,oBAAoB,SAAS;AAEjD,OAAI,CAAC,YACH,QAAO,SAAS,MAAM;AAGxB,OAAI,gBAAgB,UAAU;IAC5B,MAAM,EAAE,qBAAqB,MAAM,+BAA+B,SAAS;AAE3E,WAAO,MAAM;;AAGf,OAAI,gBAAgB,eAClB,OAAM,IAAI,MAAM,wDAAwD;AAG1E,SAAM,IAAI,MAAM,cAAc;;AAUhC,SAAO;GACL;GACA;GACA;IACC,sBAAsB;GACxB;;CAGH,YACE,QACA,MACA,QAIA;EAMA,MAAM,SALc,YAAY,QAAQ,MAAM;GAC5C,YAAY,QAAQ;GACpB,aAAa,QAAQ;GACtB,CAAC,CAEyB,KAAK,MAAO,OAAO,MAAM,WAAW,IAAI,EAAE,KAAK,CAAE,CAAC,KAAK,GAAG;AAErF,QAAKP,gBAAiB,QAAQ,IAAI,WAAW,OAAO,CAAC;;;;;;;AAQzD,SAAgB,oBAad,YAYA,cACA,mBACA,qBAIA;CAUA,MAAM,SAAS,sBAPM;EACnB,QAAQ,EAAE;EACV,MAAM,EAAE;EACR,UAAU,EAAE;EACZ,aAAa,EAAE;EAChB,EAEkD,kBAAkB;CAErE,MAAMQ,iBAAwF;EAC5F,MAAM,WAAW;EACjB;EACD;CAED,MAAM,aAAa,aAAa,cAAc,IAAI,WAAW;CAC7D,MAAM,sBAAsB,oBAAoB,qBAAqB,aAAa,cAAc;AAOhG,QAAO,IAAI,cANc;EACvB,GAAG;EACH;EACA,eAAe;EAChB,EAE0C,eAAe"}
@@ -1,6 +1,8 @@
1
- import { _ as QueryParamsHint, g as MaybeExtractPathParamsOrWiden, r as NonGetHTTPMethod, u as InferOr } from "../api-BWN97TOr.js";
2
- import "../route-Bl9Zr1Yv.js";
3
- import { C as FragnoClientMutatorData, I as FragnoClientError, S as FragnoClientHookData, w as FragnoStoreData } from "../fragment-builder-MGr68GNb.js";
1
+ import { MaybeExtractPathParamsOrWiden, QueryParamsHint } from "../api/internal/path.js";
2
+ import { InferOr } from "../util/types-util.js";
3
+ import { NonGetHTTPMethod } from "../api/api.js";
4
+ import { FragnoClientError } from "./client-error.js";
5
+ import { FragnoClientHookData, FragnoClientMutatorData, FragnoStoreData } from "./client.js";
4
6
  import { ReadableAtom } from "nanostores";
5
7
  import { Readable } from "svelte/store";
6
8
  import { StandardSchemaV1 } from "@standard-schema/spec";
@@ -1 +1 @@
1
- {"version":3,"file":"client.svelte.d.ts","names":[],"sources":["../../src/client/client.svelte.ts"],"sourcesContent":[],"mappings":"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"}
1
+ {"version":3,"file":"client.svelte.d.ts","names":[],"sources":["../../src/client/client.svelte.ts"],"sourcesContent":[],"mappings":";;;;;;;;;;KAkBY,yFAGY,yFAAA;SAKlB,8BAA8B,kBAAgB,mBAAmB;EAR3D,KAAA,CAAA,EAUF,eAVkB,CAWxB,kBAXwB,EAAA,MAAA,GAYf,QAZe,CAAA,MAAA,CAAA,GAYI,YAZJ,CAAA,MAAA,CAAA,GAAA,CAAA,GAAA,GAAA,MAAA,CAAA,CAAA;CAGJ,EAAA,GAAA;EAKY,IAAA,EAO5B,QAP4B,CAOnB,OAPmB,CAOX,eAPW,EAAA,SAAA,CAAA,CAAA;EAAgB,OAAA,EAQzC,QARyC,CAAA,OAAA,CAAA;EAAmB,KAAA,EAS9D,QAT8D,CASrD,iBATqD,CASnC,YATmC,CAAA,MAAA,CAAA,CAAA,GAAA,SAAA,CAAA;CAAjE;AAGF,KASQ,mBATR,CAAA,iBAUe,gBAVf,EAAA,gBAAA,MAAA,EAAA,uBAYmB,gBAZnB,GAAA,SAAA,EAAA,wBAaoB,gBAbpB,GAAA,SAAA,EAAA,qBAAA,MAAA,EAAA,2BAAA,MAAA,CAAA,GAAA,GAAA,GAAA;EACS,MAAA,EAAA,CAAA,IAAA,EAAA;IAAmB,IAAA,CAAA,EAiBrB,OAjBqB,CAiBb,cAjBa,EAAA,SAAA,CAAA;IAFtB,IAAA,CAAA,EAoBC,6BApBD,CAqBJ,OArBI,EAAA,MAAA,GAsBK,QAtBL,CAAA,MAAA,CAAA,GAsBwB,YAtBxB,CAAA,MAAA,CAAA,GAAA,CAAA,GAAA,GAAA,MAAA,CAAA,CAAA;IAKe,KAAA,CAAA,EAmBb,eAnBa,CAoBnB,kBApBmB,EAAA,MAAA,GAqBV,QArBU,CAAA,MAAA,CAAA,GAqBS,YArBT,CAAA,MAAA,CAAA,GAAA,CAAA,GAAA,GAAA,MAAA,CAAA,CAAA;EAAR,CAAA,EAAA,GAuBT,OAvBS,CAuBD,OAvBC,CAuBO,eAvBP,EAAA,SAAA,CAAA,CAAA;EAAT,OAAA,EAwBG,QAxBH,CAAA,OAAA,GAAA,SAAA,CAAA;EACG,KAAA,EAwBF,QAxBE,CAwBO,iBAxBP,CAwByB,YAxBzB,CAAA,MAAA,CAAA,CAAA,GAAA,SAAA,CAAA;EACyB,IAAA,EAwB5B,QAxB4B,CAwBnB,OAxBmB,CAwBX,eAxBW,EAAA,SAAA,CAAA,CAAA;CAAlB;AAAT,iBAmCO,cAnCP,CAAA,CAAA,CAAA,CAAA,KAAA,EAmCgC,QAnChC,CAmCyC,CAnCzC,CAAA,CAAA,EAmC8C,YAnC9C,CAmC2D,CAnC3D,CAAA;AAAQ,iBA6CD,UA7CC,CAAA,CAAA,CAAA,CAAA,KAAA,EAAA,GAAA,GA6C0B,CA7C1B,CAAA,EA6C8B,YA7C9B,CA6C2C,CA7C3C,CAAA;AAGL,iBA2LI,iBA3Le,CAAA,UAAA,MAAA,CAAA,CAAA,IAAA,EA2L2B,eA3L3B,CA2L2C,CA3L3C,CAAA,CAAA,EA2LgD,CA3LhD;AACZ,iBAgMH,SAhMG,CAAA,UAgMiB,MAhMjB,CAAA,MAAA,EAAA,OAAA,CAAA,CAAA,CAAA,SAAA,EAiMN,CAjMM,CAAA,EAAA,QAEI,MAiMT,CAjMS,GAiML,CAjMK,CAiMH,CAjMG,CAAA,SAiMQ,oBAjMR,CAAA,KAAA,EAAA,KAAA,MAAA,EAAA,KAAA,cAAA,EAAA,KAAA,WAAA,EAAA,KAAA,iBAAA,CAAA,GAwMjB,gBAxMiB,CAAA,KAAA,EAwMO,KAxMP,EAwMc,aAxMd,EAwM6B,UAxM7B,EAwMyC,gBAxMzC,CAAA,GAyMjB,CAzMiB,CAyMf,CAzMe,CAAA,SAyMJ,uBAzMI,CAAA,KAAA,EAAA,EAAA,KAAA,MAAA,EAAA,KAAA,aAAA,EAAA,KAAA,cAAA,EAAA,KAAA,WAAA,EAAA,KAAA,iBAAA,CAAA,GAiNf,mBAjNe,CAiNK,CAjNL,EAiNQ,KAjNR,EAiNe,YAjNf,EAiN6B,aAjN7B,EAiN4C,UAjN5C,EAiNwD,gBAjNxD,CAAA,GAkNf,CAlNe,CAkNb,CAlNa,CAAA,SAkNF,eAlNE,CAAA,KAAA,UAAA,CAAA,GAmNb,SAnNa,GAoNb,CApNa,CAoNX,CApNW,CAAA,EACC"}
@@ -1,8 +1,4 @@
1
- import "../api-DngJDcmO.js";
2
- import "../request-output-context-CdIjwmEN.js";
3
- import "../route-C5Uryylh.js";
4
- import { a as isGetHook, o as isMutatorHook, s as isStore } from "../client-C5LsYHEI.js";
5
- import "../ssr-BByDVfFD.js";
1
+ import { isGetHook, isMutatorHook, isStore } from "./client.js";
6
2
  import { atom } from "nanostores";
7
3
  import { get, writable } from "svelte/store";
8
4
  import { onDestroy } from "svelte";
@@ -1 +1 @@
1
- {"version":3,"file":"client.svelte.js","names":["pathParams: Record<string, string | ReadableAtom<string>>","queryParams: Record<string, string | ReadableAtom<string>>"],"sources":["../../src/client/client.svelte.ts"],"sourcesContent":["import type { StandardSchemaV1 } from \"@standard-schema/spec\";\nimport { atom, type ReadableAtom } from \"nanostores\";\nimport type { NonGetHTTPMethod } from \"../api/api\";\nimport {\n isGetHook,\n isMutatorHook,\n isStore,\n type FragnoClientHookData,\n type FragnoClientMutatorData,\n type FragnoStoreData,\n} from \"./client\";\nimport type { FragnoClientError } from \"./client-error\";\nimport type { InferOr } from \"../util/types-util\";\nimport type { MaybeExtractPathParamsOrWiden, QueryParamsHint } from \"../api/internal/path\";\n\nimport { writable, type Readable, get } from \"svelte/store\";\nimport { onDestroy } from \"svelte\";\n\nexport type FragnoSvelteHook<\n _TMethod extends \"GET\",\n TPath extends string,\n TOutputSchema extends StandardSchemaV1,\n TErrorCode extends string,\n TQueryParameters extends string,\n> = (args?: {\n path?:\n | MaybeExtractPathParamsOrWiden<TPath, string | Readable<string> | ReadableAtom<string>>\n | (() => string);\n query?: QueryParamsHint<\n TQueryParameters,\n string | Readable<string> | ReadableAtom<string> | (() => string)\n >;\n}) => {\n data: Readable<InferOr<TOutputSchema, undefined>>;\n loading: Readable<boolean>;\n error: Readable<FragnoClientError<TErrorCode[number]> | undefined>;\n};\n\nexport type FragnoSvelteMutator<\n _TMethod extends NonGetHTTPMethod,\n TPath extends string,\n TInputSchema extends StandardSchemaV1 | undefined,\n TOutputSchema extends StandardSchemaV1 | undefined,\n TErrorCode extends string,\n TQueryParameters extends string,\n> = () => {\n mutate: (args: {\n body?: InferOr<TInputSchema, undefined>;\n path?: MaybeExtractPathParamsOrWiden<\n TPath,\n string | Readable<string> | ReadableAtom<string> | (() => string)\n >;\n query?: QueryParamsHint<\n TQueryParameters,\n string | Readable<string> | ReadableAtom<string> | (() => string)\n >;\n }) => Promise<InferOr<TOutputSchema, undefined>>;\n loading: Readable<boolean | undefined>;\n error: Readable<FragnoClientError<TErrorCode[number]> | undefined>;\n data: Readable<InferOr<TOutputSchema, undefined>>;\n};\n\nfunction isReadable(value: unknown): value is Readable<string> {\n return typeof value === \"object\" && value !== null && \"subscribe\" in value;\n}\n\nfunction isCallable(value: unknown): value is () => string {\n return typeof value === \"function\";\n}\n\nexport function readableToAtom<T>(value: Readable<T>): ReadableAtom<T> {\n const a = atom(get(value));\n\n value.subscribe((newVal) => {\n a.set(newVal);\n });\n\n return a;\n}\n\nexport function runeToAtom<T>(value: () => T): ReadableAtom<T> {\n const a = atom(value());\n\n $effect(() => {\n a.set(value());\n });\n\n return a;\n}\n\nfunction createSvelteHook<\n TMethod extends \"GET\",\n TPath extends string,\n TOutputSchema extends StandardSchemaV1,\n TErrorCode extends string,\n TQueryParameters extends string,\n>(\n hook: FragnoClientHookData<TMethod, TPath, TOutputSchema, TErrorCode, TQueryParameters>,\n): FragnoSvelteHook<TMethod, TPath, TOutputSchema, TErrorCode, TQueryParameters> {\n return ({ path, query } = {}) => {\n const pathParams: Record<string, string | ReadableAtom<string>> = {};\n const queryParams: Record<string, string | ReadableAtom<string>> = {};\n\n for (const [key, value] of Object.entries(path ?? {})) {\n const v = value as string | Readable<string> | ReadableAtom<string> | (() => string);\n if (isCallable(v)) {\n pathParams[key] = runeToAtom(v);\n } else {\n pathParams[key] = isReadable(v) ? readableToAtom(v) : v;\n }\n }\n\n for (const [key, value] of Object.entries(query ?? {})) {\n const v = value as string | Readable<string> | ReadableAtom<string> | (() => string);\n if (isCallable(v)) {\n queryParams[key] = runeToAtom(v);\n } else {\n queryParams[key] = isReadable(v) ? readableToAtom(v) : v;\n }\n }\n\n const store = hook.store({\n path: pathParams as MaybeExtractPathParamsOrWiden<TPath, string | ReadableAtom<string>>,\n query: queryParams,\n });\n\n const data = writable<InferOr<TOutputSchema, undefined>>(undefined);\n const loading = writable<boolean>(false);\n const error = writable<FragnoClientError<TErrorCode[number]> | undefined>(undefined);\n\n const unsubscribe = store.subscribe((updatedStoreValue) => {\n data.set(updatedStoreValue.data as InferOr<TOutputSchema, undefined>);\n loading.set(updatedStoreValue.loading);\n error.set(updatedStoreValue.error);\n });\n\n onDestroy(() => {\n unsubscribe();\n });\n\n return {\n data,\n loading,\n error,\n };\n };\n}\n\nfunction createSvelteMutator<\n TMethod extends NonGetHTTPMethod,\n TPath extends string,\n TInputSchema extends StandardSchemaV1 | undefined,\n TOutputSchema extends StandardSchemaV1 | undefined,\n TErrorCode extends string,\n TQueryParameters extends string,\n>(\n hook: FragnoClientMutatorData<\n TMethod,\n TPath,\n TInputSchema,\n TOutputSchema,\n TErrorCode,\n TQueryParameters\n >,\n): FragnoSvelteMutator<TMethod, TPath, TInputSchema, TOutputSchema, TErrorCode, TQueryParameters> {\n return () => {\n const data = writable<InferOr<TOutputSchema, undefined>>(undefined);\n const loading = writable<boolean | undefined>(undefined);\n const error = writable<FragnoClientError<TErrorCode[number]> | undefined>(undefined);\n\n // Subscribe to the mutator store and sync with our Svelte stores\n const unsubscribe = hook.mutatorStore.subscribe((storeValue) => {\n data.set(storeValue.data as InferOr<TOutputSchema, undefined>);\n loading.set(storeValue.loading);\n error.set(storeValue.error);\n });\n\n onDestroy(() => {\n unsubscribe();\n });\n\n // Create a wrapped mutate function that handles Svelte readable stores\n const mutate = async (args: {\n body?: InferOr<TInputSchema, undefined>;\n path?: MaybeExtractPathParamsOrWiden<\n TPath,\n string | Readable<string> | ReadableAtom<string> | (() => string)\n >;\n query?: QueryParamsHint<\n TQueryParameters,\n string | Readable<string> | ReadableAtom<string> | (() => string)\n >;\n }) => {\n const { body, path, query } = args;\n\n const pathParams: Record<string, string | ReadableAtom<string>> = {};\n const queryParams: Record<string, string | ReadableAtom<string>> = {};\n\n for (const [key, value] of Object.entries(path ?? {})) {\n const v = value as string | Readable<string> | ReadableAtom<string>;\n pathParams[key] = isReadable(v) ? readableToAtom(v) : v;\n }\n\n for (const [key, value] of Object.entries(query ?? {})) {\n const v = value as string | Readable<string> | ReadableAtom<string>;\n queryParams[key] = isReadable(v) ? readableToAtom(v) : v;\n }\n\n // Call the store's mutate function with normalized params\n return hook.mutatorStore.mutate({\n body,\n path: pathParams as MaybeExtractPathParamsOrWiden<TPath, string | ReadableAtom<string>>,\n query: queryParams,\n });\n };\n\n return {\n mutate,\n loading,\n error,\n data,\n };\n };\n}\n\nexport function createSvelteStore<T extends object>(hook: FragnoStoreData<T>): T {\n // Since nanostores already implement Svelte's store contract,\n // we can return the store object directly for use with $ syntax\n return hook.obj;\n}\n\nexport function useFragno<T extends Record<string, unknown>>(\n clientObj: T,\n): {\n [K in keyof T]: T[K] extends FragnoClientHookData<\n \"GET\",\n infer TPath,\n infer TOutputSchema,\n infer TErrorCode,\n infer TQueryParameters\n >\n ? FragnoSvelteHook<\"GET\", TPath, TOutputSchema, TErrorCode, TQueryParameters>\n : T[K] extends FragnoClientMutatorData<\n infer M,\n infer TPath,\n infer TInputSchema,\n infer TOutputSchema,\n infer TErrorCode,\n infer TQueryParameters\n >\n ? FragnoSvelteMutator<M, TPath, TInputSchema, TOutputSchema, TErrorCode, TQueryParameters>\n : T[K] extends FragnoStoreData<infer TStoreObj>\n ? TStoreObj\n : T[K];\n} {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const result = {} as any;\n\n for (const key in clientObj) {\n if (!Object.prototype.hasOwnProperty.call(clientObj, key)) {\n continue;\n }\n\n const hook = clientObj[key];\n if (isGetHook(hook)) {\n result[key] = createSvelteHook(hook);\n } else if (isMutatorHook(hook)) {\n result[key] = createSvelteMutator(hook);\n } else if (isStore(hook)) {\n result[key] = createSvelteStore(hook);\n } else {\n // Pass through non-hook values unchanged\n result[key] = hook;\n }\n }\n\n return result;\n}\n"],"mappings":"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"}
1
+ {"version":3,"file":"client.svelte.js","names":["pathParams: Record<string, string | ReadableAtom<string>>","queryParams: Record<string, string | ReadableAtom<string>>"],"sources":["../../src/client/client.svelte.ts"],"sourcesContent":["import type { StandardSchemaV1 } from \"@standard-schema/spec\";\nimport { atom, type ReadableAtom } from \"nanostores\";\nimport type { NonGetHTTPMethod } from \"../api/api\";\nimport {\n isGetHook,\n isMutatorHook,\n isStore,\n type FragnoClientHookData,\n type FragnoClientMutatorData,\n type FragnoStoreData,\n} from \"./client\";\nimport type { FragnoClientError } from \"./client-error\";\nimport type { InferOr } from \"../util/types-util\";\nimport type { MaybeExtractPathParamsOrWiden, QueryParamsHint } from \"../api/internal/path\";\n\nimport { writable, type Readable, get } from \"svelte/store\";\nimport { onDestroy } from \"svelte\";\n\nexport type FragnoSvelteHook<\n _TMethod extends \"GET\",\n TPath extends string,\n TOutputSchema extends StandardSchemaV1,\n TErrorCode extends string,\n TQueryParameters extends string,\n> = (args?: {\n path?:\n | MaybeExtractPathParamsOrWiden<TPath, string | Readable<string> | ReadableAtom<string>>\n | (() => string);\n query?: QueryParamsHint<\n TQueryParameters,\n string | Readable<string> | ReadableAtom<string> | (() => string)\n >;\n}) => {\n data: Readable<InferOr<TOutputSchema, undefined>>;\n loading: Readable<boolean>;\n error: Readable<FragnoClientError<TErrorCode[number]> | undefined>;\n};\n\nexport type FragnoSvelteMutator<\n _TMethod extends NonGetHTTPMethod,\n TPath extends string,\n TInputSchema extends StandardSchemaV1 | undefined,\n TOutputSchema extends StandardSchemaV1 | undefined,\n TErrorCode extends string,\n TQueryParameters extends string,\n> = () => {\n mutate: (args: {\n body?: InferOr<TInputSchema, undefined>;\n path?: MaybeExtractPathParamsOrWiden<\n TPath,\n string | Readable<string> | ReadableAtom<string> | (() => string)\n >;\n query?: QueryParamsHint<\n TQueryParameters,\n string | Readable<string> | ReadableAtom<string> | (() => string)\n >;\n }) => Promise<InferOr<TOutputSchema, undefined>>;\n loading: Readable<boolean | undefined>;\n error: Readable<FragnoClientError<TErrorCode[number]> | undefined>;\n data: Readable<InferOr<TOutputSchema, undefined>>;\n};\n\nfunction isReadable(value: unknown): value is Readable<string> {\n return typeof value === \"object\" && value !== null && \"subscribe\" in value;\n}\n\nfunction isCallable(value: unknown): value is () => string {\n return typeof value === \"function\";\n}\n\nexport function readableToAtom<T>(value: Readable<T>): ReadableAtom<T> {\n const a = atom(get(value));\n\n value.subscribe((newVal) => {\n a.set(newVal);\n });\n\n return a;\n}\n\nexport function runeToAtom<T>(value: () => T): ReadableAtom<T> {\n const a = atom(value());\n\n $effect(() => {\n a.set(value());\n });\n\n return a;\n}\n\nfunction createSvelteHook<\n TMethod extends \"GET\",\n TPath extends string,\n TOutputSchema extends StandardSchemaV1,\n TErrorCode extends string,\n TQueryParameters extends string,\n>(\n hook: FragnoClientHookData<TMethod, TPath, TOutputSchema, TErrorCode, TQueryParameters>,\n): FragnoSvelteHook<TMethod, TPath, TOutputSchema, TErrorCode, TQueryParameters> {\n return ({ path, query } = {}) => {\n const pathParams: Record<string, string | ReadableAtom<string>> = {};\n const queryParams: Record<string, string | ReadableAtom<string>> = {};\n\n for (const [key, value] of Object.entries(path ?? {})) {\n const v = value as string | Readable<string> | ReadableAtom<string> | (() => string);\n if (isCallable(v)) {\n pathParams[key] = runeToAtom(v);\n } else {\n pathParams[key] = isReadable(v) ? readableToAtom(v) : v;\n }\n }\n\n for (const [key, value] of Object.entries(query ?? {})) {\n const v = value as string | Readable<string> | ReadableAtom<string> | (() => string);\n if (isCallable(v)) {\n queryParams[key] = runeToAtom(v);\n } else {\n queryParams[key] = isReadable(v) ? readableToAtom(v) : v;\n }\n }\n\n const store = hook.store({\n path: pathParams as MaybeExtractPathParamsOrWiden<TPath, string | ReadableAtom<string>>,\n query: queryParams,\n });\n\n const data = writable<InferOr<TOutputSchema, undefined>>(undefined);\n const loading = writable<boolean>(false);\n const error = writable<FragnoClientError<TErrorCode[number]> | undefined>(undefined);\n\n const unsubscribe = store.subscribe((updatedStoreValue) => {\n data.set(updatedStoreValue.data as InferOr<TOutputSchema, undefined>);\n loading.set(updatedStoreValue.loading);\n error.set(updatedStoreValue.error);\n });\n\n onDestroy(() => {\n unsubscribe();\n });\n\n return {\n data,\n loading,\n error,\n };\n };\n}\n\nfunction createSvelteMutator<\n TMethod extends NonGetHTTPMethod,\n TPath extends string,\n TInputSchema extends StandardSchemaV1 | undefined,\n TOutputSchema extends StandardSchemaV1 | undefined,\n TErrorCode extends string,\n TQueryParameters extends string,\n>(\n hook: FragnoClientMutatorData<\n TMethod,\n TPath,\n TInputSchema,\n TOutputSchema,\n TErrorCode,\n TQueryParameters\n >,\n): FragnoSvelteMutator<TMethod, TPath, TInputSchema, TOutputSchema, TErrorCode, TQueryParameters> {\n return () => {\n const data = writable<InferOr<TOutputSchema, undefined>>(undefined);\n const loading = writable<boolean | undefined>(undefined);\n const error = writable<FragnoClientError<TErrorCode[number]> | undefined>(undefined);\n\n // Subscribe to the mutator store and sync with our Svelte stores\n const unsubscribe = hook.mutatorStore.subscribe((storeValue) => {\n data.set(storeValue.data as InferOr<TOutputSchema, undefined>);\n loading.set(storeValue.loading);\n error.set(storeValue.error);\n });\n\n onDestroy(() => {\n unsubscribe();\n });\n\n // Create a wrapped mutate function that handles Svelte readable stores\n const mutate = async (args: {\n body?: InferOr<TInputSchema, undefined>;\n path?: MaybeExtractPathParamsOrWiden<\n TPath,\n string | Readable<string> | ReadableAtom<string> | (() => string)\n >;\n query?: QueryParamsHint<\n TQueryParameters,\n string | Readable<string> | ReadableAtom<string> | (() => string)\n >;\n }) => {\n const { body, path, query } = args;\n\n const pathParams: Record<string, string | ReadableAtom<string>> = {};\n const queryParams: Record<string, string | ReadableAtom<string>> = {};\n\n for (const [key, value] of Object.entries(path ?? {})) {\n const v = value as string | Readable<string> | ReadableAtom<string>;\n pathParams[key] = isReadable(v) ? readableToAtom(v) : v;\n }\n\n for (const [key, value] of Object.entries(query ?? {})) {\n const v = value as string | Readable<string> | ReadableAtom<string>;\n queryParams[key] = isReadable(v) ? readableToAtom(v) : v;\n }\n\n // Call the store's mutate function with normalized params\n return hook.mutatorStore.mutate({\n body,\n path: pathParams as MaybeExtractPathParamsOrWiden<TPath, string | ReadableAtom<string>>,\n query: queryParams,\n });\n };\n\n return {\n mutate,\n loading,\n error,\n data,\n };\n };\n}\n\nexport function createSvelteStore<T extends object>(hook: FragnoStoreData<T>): T {\n // Since nanostores already implement Svelte's store contract,\n // we can return the store object directly for use with $ syntax\n return hook.obj;\n}\n\nexport function useFragno<T extends Record<string, unknown>>(\n clientObj: T,\n): {\n [K in keyof T]: T[K] extends FragnoClientHookData<\n \"GET\",\n infer TPath,\n infer TOutputSchema,\n infer TErrorCode,\n infer TQueryParameters\n >\n ? FragnoSvelteHook<\"GET\", TPath, TOutputSchema, TErrorCode, TQueryParameters>\n : T[K] extends FragnoClientMutatorData<\n infer M,\n infer TPath,\n infer TInputSchema,\n infer TOutputSchema,\n infer TErrorCode,\n infer TQueryParameters\n >\n ? FragnoSvelteMutator<M, TPath, TInputSchema, TOutputSchema, TErrorCode, TQueryParameters>\n : T[K] extends FragnoStoreData<infer TStoreObj>\n ? TStoreObj\n : T[K];\n} {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const result = {} as any;\n\n for (const key in clientObj) {\n if (!Object.prototype.hasOwnProperty.call(clientObj, key)) {\n continue;\n }\n\n const hook = clientObj[key];\n if (isGetHook(hook)) {\n result[key] = createSvelteHook(hook);\n } else if (isMutatorHook(hook)) {\n result[key] = createSvelteMutator(hook);\n } else if (isStore(hook)) {\n result[key] = createSvelteStore(hook);\n } else {\n // Pass through non-hook values unchanged\n result[key] = hook;\n }\n }\n\n return result;\n}\n"],"mappings":"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"}
@@ -0,0 +1,36 @@
1
+ //#region src/client/internal/fetcher-merge.ts
2
+ /**
3
+ * Merge two fetcher configurations, with user config taking precedence.
4
+ * If user provides a custom function, it takes full precedence.
5
+ * Otherwise, deep merge RequestInit options.
6
+ */
7
+ function mergeFetcherConfigs(authorConfig, userConfig) {
8
+ if (userConfig?.type === "function") return userConfig;
9
+ if (!userConfig && authorConfig?.type === "function") return authorConfig;
10
+ const authorOpts = authorConfig?.type === "options" ? authorConfig.options : {};
11
+ const userOpts = userConfig?.type === "options" ? userConfig.options : {};
12
+ if (Object.keys(authorOpts).length === 0 && Object.keys(userOpts).length === 0) return;
13
+ return {
14
+ type: "options",
15
+ options: {
16
+ ...authorOpts,
17
+ ...userOpts,
18
+ headers: mergeHeaders(authorOpts.headers, userOpts.headers)
19
+ }
20
+ };
21
+ }
22
+ /**
23
+ * Merge headers from author and user configs.
24
+ * User headers override author headers.
25
+ */
26
+ function mergeHeaders(author, user) {
27
+ if (!author && !user) return;
28
+ const merged = new Headers(author);
29
+ new Headers(user).forEach((value, key) => merged.set(key, value));
30
+ if (merged.keys().next().done) return;
31
+ return merged;
32
+ }
33
+
34
+ //#endregion
35
+ export { mergeFetcherConfigs };
36
+ //# sourceMappingURL=fetcher-merge.js.map