@agent-native/core 0.9.1 → 0.10.0

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 (209) hide show
  1. package/README.md +4 -4
  2. package/dist/agent/engine/builder-engine.d.ts.map +1 -1
  3. package/dist/agent/engine/builder-engine.js +5 -4
  4. package/dist/agent/engine/builder-engine.js.map +1 -1
  5. package/dist/agent/engine/registry.d.ts +6 -3
  6. package/dist/agent/engine/registry.d.ts.map +1 -1
  7. package/dist/agent/engine/registry.js +8 -17
  8. package/dist/agent/engine/registry.js.map +1 -1
  9. package/dist/agent/production-agent.d.ts +1 -1
  10. package/dist/agent/production-agent.d.ts.map +1 -1
  11. package/dist/agent/production-agent.js +28 -11
  12. package/dist/agent/production-agent.js.map +1 -1
  13. package/dist/agent/run-manager.d.ts.map +1 -1
  14. package/dist/agent/run-manager.js +12 -3
  15. package/dist/agent/run-manager.js.map +1 -1
  16. package/dist/agent/thread-data-builder.d.ts +12 -0
  17. package/dist/agent/thread-data-builder.d.ts.map +1 -1
  18. package/dist/agent/thread-data-builder.js +96 -0
  19. package/dist/agent/thread-data-builder.js.map +1 -1
  20. package/dist/cli/create.d.ts.map +1 -1
  21. package/dist/cli/create.js +16 -10
  22. package/dist/cli/create.js.map +1 -1
  23. package/dist/client/AgentPanel.d.ts.map +1 -1
  24. package/dist/client/AgentPanel.js +6 -20
  25. package/dist/client/AgentPanel.js.map +1 -1
  26. package/dist/client/AssistantChat.d.ts.map +1 -1
  27. package/dist/client/AssistantChat.js +113 -28
  28. package/dist/client/AssistantChat.js.map +1 -1
  29. package/dist/client/agent-chat-adapter.d.ts.map +1 -1
  30. package/dist/client/agent-chat-adapter.js +21 -7
  31. package/dist/client/agent-chat-adapter.js.map +1 -1
  32. package/dist/client/agent-sidebar-state.d.ts +3 -0
  33. package/dist/client/agent-sidebar-state.d.ts.map +1 -0
  34. package/dist/client/agent-sidebar-state.js +24 -0
  35. package/dist/client/agent-sidebar-state.js.map +1 -0
  36. package/dist/client/analytics.d.ts +25 -0
  37. package/dist/client/analytics.d.ts.map +1 -1
  38. package/dist/client/analytics.js +40 -0
  39. package/dist/client/analytics.js.map +1 -1
  40. package/dist/client/components/ui/tooltip.d.ts +2 -1
  41. package/dist/client/components/ui/tooltip.d.ts.map +1 -1
  42. package/dist/client/components/ui/tooltip.js +9 -2
  43. package/dist/client/components/ui/tooltip.js.map +1 -1
  44. package/dist/client/composer/ComposerPlusMenu.d.ts.map +1 -1
  45. package/dist/client/composer/ComposerPlusMenu.js +41 -8
  46. package/dist/client/composer/ComposerPlusMenu.js.map +1 -1
  47. package/dist/client/composer/PromptComposer.d.ts.map +1 -1
  48. package/dist/client/composer/PromptComposer.js +30 -0
  49. package/dist/client/composer/PromptComposer.js.map +1 -1
  50. package/dist/client/composer/TiptapComposer.d.ts.map +1 -1
  51. package/dist/client/composer/TiptapComposer.js +26 -1
  52. package/dist/client/composer/TiptapComposer.js.map +1 -1
  53. package/dist/client/dev-overlay/DevOverlay.d.ts.map +1 -1
  54. package/dist/client/dev-overlay/DevOverlay.js +4 -4
  55. package/dist/client/dev-overlay/DevOverlay.js.map +1 -1
  56. package/dist/client/error-format.d.ts.map +1 -1
  57. package/dist/client/error-format.js +6 -0
  58. package/dist/client/error-format.js.map +1 -1
  59. package/dist/client/extensions/EmbeddedExtension.d.ts.map +1 -1
  60. package/dist/client/extensions/EmbeddedExtension.js +14 -4
  61. package/dist/client/extensions/EmbeddedExtension.js.map +1 -1
  62. package/dist/client/extensions/ExtensionEditor.d.ts.map +1 -1
  63. package/dist/client/extensions/ExtensionEditor.js +6 -6
  64. package/dist/client/extensions/ExtensionEditor.js.map +1 -1
  65. package/dist/client/extensions/ExtensionSlot.d.ts.map +1 -1
  66. package/dist/client/extensions/ExtensionSlot.js +2 -2
  67. package/dist/client/extensions/ExtensionSlot.js.map +1 -1
  68. package/dist/client/extensions/ExtensionViewer.d.ts.map +1 -1
  69. package/dist/client/extensions/ExtensionViewer.js +39 -19
  70. package/dist/client/extensions/ExtensionViewer.js.map +1 -1
  71. package/dist/client/extensions/ExtensionsSidebarSection.d.ts.map +1 -1
  72. package/dist/client/extensions/ExtensionsSidebarSection.js +52 -52
  73. package/dist/client/extensions/ExtensionsSidebarSection.js.map +1 -1
  74. package/dist/client/index.d.ts +1 -1
  75. package/dist/client/index.d.ts.map +1 -1
  76. package/dist/client/index.js +1 -1
  77. package/dist/client/index.js.map +1 -1
  78. package/dist/client/notifications/NotificationsBell.d.ts.map +1 -1
  79. package/dist/client/notifications/NotificationsBell.js +42 -6
  80. package/dist/client/notifications/NotificationsBell.js.map +1 -1
  81. package/dist/client/org/InvitationBanner.d.ts.map +1 -1
  82. package/dist/client/org/InvitationBanner.js +5 -5
  83. package/dist/client/org/InvitationBanner.js.map +1 -1
  84. package/dist/client/org/TeamPage.d.ts.map +1 -1
  85. package/dist/client/org/TeamPage.js +3 -2
  86. package/dist/client/org/TeamPage.js.map +1 -1
  87. package/dist/client/resources/ResourcesPanel.d.ts.map +1 -1
  88. package/dist/client/resources/ResourcesPanel.js +41 -7
  89. package/dist/client/resources/ResourcesPanel.js.map +1 -1
  90. package/dist/client/resources/use-mcp-servers.d.ts +2 -0
  91. package/dist/client/resources/use-mcp-servers.d.ts.map +1 -1
  92. package/dist/client/resources/use-mcp-servers.js +59 -3
  93. package/dist/client/resources/use-mcp-servers.js.map +1 -1
  94. package/dist/client/settings/SecretsSection.d.ts.map +1 -1
  95. package/dist/client/settings/SecretsSection.js +9 -0
  96. package/dist/client/settings/SecretsSection.js.map +1 -1
  97. package/dist/client/settings/SettingsPanel.d.ts.map +1 -1
  98. package/dist/client/settings/SettingsPanel.js +12 -10
  99. package/dist/client/settings/SettingsPanel.js.map +1 -1
  100. package/dist/client/settings/VoiceTranscriptionSection.d.ts.map +1 -1
  101. package/dist/client/settings/VoiceTranscriptionSection.js +13 -30
  102. package/dist/client/settings/VoiceTranscriptionSection.js.map +1 -1
  103. package/dist/client/settings/useBuilderStatus.d.ts.map +1 -1
  104. package/dist/client/settings/useBuilderStatus.js +27 -1
  105. package/dist/client/settings/useBuilderStatus.js.map +1 -1
  106. package/dist/client/sharing/ShareButton.d.ts +4 -0
  107. package/dist/client/sharing/ShareButton.d.ts.map +1 -1
  108. package/dist/client/sharing/ShareButton.js +5 -1
  109. package/dist/client/sharing/ShareButton.js.map +1 -1
  110. package/dist/client/sse-event-processor.d.ts +1 -1
  111. package/dist/client/sse-event-processor.d.ts.map +1 -1
  112. package/dist/client/sse-event-processor.js +14 -7
  113. package/dist/client/sse-event-processor.js.map +1 -1
  114. package/dist/client/use-db-sync.d.ts.map +1 -1
  115. package/dist/client/use-db-sync.js +100 -19
  116. package/dist/client/use-db-sync.js.map +1 -1
  117. package/dist/deploy/build.d.ts.map +1 -1
  118. package/dist/deploy/build.js +5 -0
  119. package/dist/deploy/build.js.map +1 -1
  120. package/dist/deploy/route-discovery.d.ts.map +1 -1
  121. package/dist/deploy/route-discovery.js +1 -0
  122. package/dist/deploy/route-discovery.js.map +1 -1
  123. package/dist/deploy/workspace-core.d.ts +1 -1
  124. package/dist/deploy/workspace-core.d.ts.map +1 -1
  125. package/dist/deploy/workspace-core.js +1 -0
  126. package/dist/deploy/workspace-core.js.map +1 -1
  127. package/dist/extensions/actions.d.ts.map +1 -1
  128. package/dist/extensions/actions.js +17 -3
  129. package/dist/extensions/actions.js.map +1 -1
  130. package/dist/extensions/routes.js +1 -1
  131. package/dist/extensions/routes.js.map +1 -1
  132. package/dist/extensions/schema.d.ts +14 -14
  133. package/dist/extensions/schema.d.ts.map +1 -1
  134. package/dist/extensions/schema.js +4 -4
  135. package/dist/extensions/schema.js.map +1 -1
  136. package/dist/extensions/store.d.ts.map +1 -1
  137. package/dist/extensions/store.js +23 -0
  138. package/dist/extensions/store.js.map +1 -1
  139. package/dist/extensions/theme.d.ts +8 -1
  140. package/dist/extensions/theme.d.ts.map +1 -1
  141. package/dist/extensions/theme.js +43 -34
  142. package/dist/extensions/theme.js.map +1 -1
  143. package/dist/mcp-client/routes.d.ts +1 -0
  144. package/dist/mcp-client/routes.d.ts.map +1 -1
  145. package/dist/mcp-client/routes.js +28 -1
  146. package/dist/mcp-client/routes.js.map +1 -1
  147. package/dist/server/agent-chat-plugin.d.ts.map +1 -1
  148. package/dist/server/agent-chat-plugin.js +77 -102
  149. package/dist/server/agent-chat-plugin.js.map +1 -1
  150. package/dist/server/auth.d.ts.map +1 -1
  151. package/dist/server/auth.js +33 -0
  152. package/dist/server/auth.js.map +1 -1
  153. package/dist/server/builder-browser.d.ts.map +1 -1
  154. package/dist/server/builder-browser.js +169 -68
  155. package/dist/server/builder-browser.js.map +1 -1
  156. package/dist/server/credential-provider.d.ts +2 -2
  157. package/dist/server/credential-provider.d.ts.map +1 -1
  158. package/dist/server/credential-provider.js +31 -12
  159. package/dist/server/credential-provider.js.map +1 -1
  160. package/dist/server/framework-request-handler.d.ts.map +1 -1
  161. package/dist/server/framework-request-handler.js +31 -0
  162. package/dist/server/framework-request-handler.js.map +1 -1
  163. package/dist/server/google-realtime-session.d.ts.map +1 -1
  164. package/dist/server/google-realtime-session.js +19 -6
  165. package/dist/server/google-realtime-session.js.map +1 -1
  166. package/dist/server/index.d.ts +2 -0
  167. package/dist/server/index.d.ts.map +1 -1
  168. package/dist/server/index.js +2 -0
  169. package/dist/server/index.js.map +1 -1
  170. package/dist/server/onboarding-html.d.ts.map +1 -1
  171. package/dist/server/onboarding-html.js +45 -6
  172. package/dist/server/onboarding-html.js.map +1 -1
  173. package/dist/server/request-context.d.ts +17 -0
  174. package/dist/server/request-context.d.ts.map +1 -1
  175. package/dist/server/request-context.js +40 -1
  176. package/dist/server/request-context.js.map +1 -1
  177. package/dist/server/sentry-plugin.d.ts +11 -0
  178. package/dist/server/sentry-plugin.d.ts.map +1 -0
  179. package/dist/server/sentry-plugin.js +116 -0
  180. package/dist/server/sentry-plugin.js.map +1 -0
  181. package/dist/server/sentry.d.ts +92 -0
  182. package/dist/server/sentry.d.ts.map +1 -0
  183. package/dist/server/sentry.js +287 -0
  184. package/dist/server/sentry.js.map +1 -0
  185. package/dist/server/transcribe-voice.d.ts +2 -4
  186. package/dist/server/transcribe-voice.d.ts.map +1 -1
  187. package/dist/server/transcribe-voice.js +4 -16
  188. package/dist/server/transcribe-voice.js.map +1 -1
  189. package/dist/server/voice-providers-status.d.ts.map +1 -1
  190. package/dist/server/voice-providers-status.js +19 -35
  191. package/dist/server/voice-providers-status.js.map +1 -1
  192. package/dist/styles/agent-native.css +15 -0
  193. package/docs/content/cloneable-saas.md +7 -9
  194. package/docs/content/deployment.md +6 -2
  195. package/docs/content/dispatch.md +1 -1
  196. package/docs/content/extensions.md +177 -142
  197. package/docs/content/faq.md +2 -2
  198. package/docs/content/getting-started.md +13 -11
  199. package/docs/content/multi-app-workspace.md +2 -2
  200. package/docs/content/observability.md +47 -0
  201. package/docs/content/pure-agent-apps.md +1 -1
  202. package/docs/content/template-clips.md +3 -3
  203. package/docs/content/template-design.md +3 -3
  204. package/docs/content/template-dispatch.md +1 -1
  205. package/docs/content/template-forms.md +1 -1
  206. package/docs/content/template-mail.md +1 -1
  207. package/docs/content/what-is-agent-native.md +4 -4
  208. package/docs/content/workspace.md +1 -1
  209. package/package.json +1 -1
@@ -1,5 +1,25 @@
1
1
  import { useEffect, useRef, useState } from "react";
2
2
  import { agentNativePath } from "./api-path.js";
3
+ const POLL_ABORT_MIN_MS = 10_000;
4
+ function getPollAbortMs(interval) {
5
+ return Math.max(POLL_ABORT_MIN_MS, interval * 4);
6
+ }
7
+ async function fetchPollJson(pollUrl, since, interval) {
8
+ const controller = typeof AbortController === "undefined" ? null : new AbortController();
9
+ const timeout = controller
10
+ ? setTimeout(() => controller.abort(), getPollAbortMs(interval))
11
+ : null;
12
+ try {
13
+ const res = await fetch(`${pollUrl}?since=${since}`, controller ? { signal: controller.signal } : undefined);
14
+ if (!res.ok)
15
+ throw new Error("HTTP " + res.status);
16
+ return res.json();
17
+ }
18
+ finally {
19
+ if (timeout)
20
+ clearTimeout(timeout);
21
+ }
22
+ }
3
23
  /**
4
24
  * Hook that polls /_agent-native/poll for DB change events and invalidates
5
25
  * react-query caches when changes are detected.
@@ -28,14 +48,23 @@ export function useDbSync(options = {}) {
28
48
  let versionRef = 0;
29
49
  let timer = null;
30
50
  let stopped = false;
31
- async function poll() {
51
+ let inFlight = false;
52
+ function schedulePoll() {
32
53
  if (stopped)
33
54
  return;
55
+ if (timer)
56
+ clearTimeout(timer);
57
+ timer = setTimeout(() => {
58
+ timer = null;
59
+ void poll();
60
+ }, interval);
61
+ }
62
+ async function poll() {
63
+ if (stopped || inFlight)
64
+ return;
65
+ inFlight = true;
34
66
  try {
35
- const res = await fetch(`${pollUrl}?since=${versionRef}`);
36
- if (!res.ok)
37
- throw new Error("HTTP " + res.status);
38
- const data = await res.json();
67
+ const data = await fetchPollJson(pollUrl, versionRef, interval);
39
68
  const { version, events } = data;
40
69
  if (events.length > 0 && queryClient) {
41
70
  const ignore = ignoreSourceRef.current;
@@ -51,6 +80,8 @@ export function useDbSync(options = {}) {
51
80
  // (agent or HTTP) auto-refresh the UI — regardless of how the
52
81
  // template configured queryKeys / onEvent.
53
82
  queryClient.invalidateQueries({ queryKey: ["action"] });
83
+ queryClient.invalidateQueries({ queryKey: ["extension"] });
84
+ queryClient.invalidateQueries({ queryKey: ["extensions"] });
54
85
  queryClient.invalidateQueries({ queryKey: ["tool"] });
55
86
  queryClient.invalidateQueries({ queryKey: ["tools"] });
56
87
  }
@@ -66,16 +97,36 @@ export function useDbSync(options = {}) {
66
97
  catch {
67
98
  // Network error — will retry on next interval
68
99
  }
69
- if (!stopped) {
70
- timer = setTimeout(poll, interval);
100
+ finally {
101
+ inFlight = false;
102
+ schedulePoll();
103
+ }
104
+ }
105
+ function pollNow() {
106
+ if (typeof document !== "undefined" &&
107
+ document.visibilityState === "hidden") {
108
+ return;
71
109
  }
110
+ if (timer) {
111
+ clearTimeout(timer);
112
+ timer = null;
113
+ }
114
+ void poll();
115
+ }
116
+ function handleVisibilityChange() {
117
+ if (document.visibilityState === "visible")
118
+ pollNow();
72
119
  }
73
120
  // Initial poll immediately
74
- poll();
121
+ void poll();
122
+ window.addEventListener("focus", pollNow);
123
+ document.addEventListener("visibilitychange", handleVisibilityChange);
75
124
  return () => {
76
125
  stopped = true;
77
126
  if (timer)
78
127
  clearTimeout(timer);
128
+ window.removeEventListener("focus", pollNow);
129
+ document.removeEventListener("visibilitychange", handleVisibilityChange);
79
130
  };
80
131
  }, [pollUrl, queryClient, interval]);
81
132
  }
@@ -107,30 +158,60 @@ export function useScreenRefreshKey(options = {}) {
107
158
  let versionRef = 0;
108
159
  let timer = null;
109
160
  let stopped = false;
110
- async function poll() {
161
+ let inFlight = false;
162
+ function schedulePoll() {
111
163
  if (stopped)
112
164
  return;
165
+ if (timer)
166
+ clearTimeout(timer);
167
+ timer = setTimeout(() => {
168
+ timer = null;
169
+ void poll();
170
+ }, interval);
171
+ }
172
+ async function poll() {
173
+ if (stopped || inFlight)
174
+ return;
175
+ inFlight = true;
113
176
  try {
114
- const res = await fetch(`${pollUrl}?since=${versionRef}`);
115
- if (res.ok) {
116
- const data = (await res.json());
117
- if (data.events?.some((e) => e.source === "screen-refresh")) {
118
- setKey((k) => k + 1);
119
- }
120
- versionRef = Math.max(versionRef, data.version);
177
+ const data = await fetchPollJson(pollUrl, versionRef, interval);
178
+ if (data.events?.some((e) => e.source === "screen-refresh")) {
179
+ setKey((k) => k + 1);
121
180
  }
181
+ versionRef = Math.max(versionRef, data.version);
122
182
  }
123
183
  catch {
124
184
  // Network error — retry on next interval.
125
185
  }
126
- if (!stopped)
127
- timer = setTimeout(poll, interval);
186
+ finally {
187
+ inFlight = false;
188
+ schedulePoll();
189
+ }
190
+ }
191
+ function pollNow() {
192
+ if (typeof document !== "undefined" &&
193
+ document.visibilityState === "hidden") {
194
+ return;
195
+ }
196
+ if (timer) {
197
+ clearTimeout(timer);
198
+ timer = null;
199
+ }
200
+ void poll();
201
+ }
202
+ function handleVisibilityChange() {
203
+ if (document.visibilityState === "visible")
204
+ pollNow();
128
205
  }
129
- poll();
206
+ void poll();
207
+ window.addEventListener("focus", pollNow);
208
+ document.addEventListener("visibilitychange", handleVisibilityChange);
130
209
  return () => {
131
210
  stopped = true;
132
211
  if (timer)
133
212
  clearTimeout(timer);
213
+ window.removeEventListener("focus", pollNow);
214
+ document.removeEventListener("visibilitychange", handleVisibilityChange);
134
215
  };
135
216
  }, [pollUrl, interval]);
136
217
  return key;
@@ -1 +1 @@
1
- {"version":3,"file":"use-db-sync.js","sourceRoot":"","sources":["../../src/client/use-db-sync.ts"],"names":[],"mappings":"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","sourcesContent":["import { useEffect, useRef, useState } from \"react\";\nimport { agentNativePath } from \"./api-path.js\";\n\ninterface QueryClient {\n invalidateQueries(opts?: { queryKey?: string[] }): void;\n}\n\n/**\n * Hook that polls /_agent-native/poll for DB change events and invalidates\n * react-query caches when changes are detected.\n *\n * Works in all deployment environments (serverless, edge, long-lived server).\n *\n * @param options.queryClient - The react-query QueryClient instance\n * @param options.queryKeys - Array of query key prefixes to invalidate on change.\n * Default: [\"data\"]\n * @param options.pollUrl - Poll endpoint URL. Default: \"/_agent-native/poll\"\n * @param options.onEvent - Optional callback for each change event\n * @param options.interval - Poll interval in ms. Default: 2000\n * @param options.ignoreSource - Skip events whose `requestSource` matches this\n * value. Use a per-tab ID so the UI ignores its own writes while still\n * picking up changes from other tabs, agents, and scripts.\n */\nexport function useDbSync(\n options: {\n queryClient?: QueryClient;\n queryKeys?: string[];\n pollUrl?: string;\n /** @deprecated Use pollUrl instead */\n eventsUrl?: string;\n onEvent?: (data: any) => void;\n interval?: number;\n ignoreSource?: string;\n } = {},\n): void {\n const {\n queryClient,\n queryKeys = [\"data\"],\n pollUrl = agentNativePath(options.eventsUrl ?? \"/_agent-native/poll\"),\n interval = 2000,\n } = options;\n\n const onEventRef = useRef(options.onEvent);\n onEventRef.current = options.onEvent;\n\n const keysRef = useRef(queryKeys);\n keysRef.current = queryKeys;\n\n const ignoreSourceRef = useRef(options.ignoreSource);\n ignoreSourceRef.current = options.ignoreSource;\n\n useEffect(() => {\n let versionRef = 0;\n let timer: ReturnType<typeof setTimeout> | null = null;\n let stopped = false;\n\n async function poll() {\n if (stopped) return;\n try {\n const res = await fetch(`${pollUrl}?since=${versionRef}`);\n if (!res.ok) throw new Error(\"HTTP \" + res.status);\n const data = await res.json();\n const { version, events } = data as {\n version: number;\n events: Array<{ source: string; type: string; key?: string }>;\n };\n\n if (events.length > 0 && queryClient) {\n const ignore = ignoreSourceRef.current;\n const relevant = ignore\n ? events.filter((e: any) => e.requestSource !== ignore)\n : events;\n\n if (relevant.length > 0) {\n for (const key of keysRef.current) {\n queryClient.invalidateQueries({ queryKey: [key] });\n }\n\n // Framework-level invalidation: always invalidate framework query\n // keys on any non-own change event so that mutating actions\n // (agent or HTTP) auto-refresh the UI — regardless of how the\n // template configured queryKeys / onEvent.\n queryClient.invalidateQueries({ queryKey: [\"action\"] });\n queryClient.invalidateQueries({ queryKey: [\"tool\"] });\n queryClient.invalidateQueries({ queryKey: [\"tools\"] });\n }\n\n // Always forward all events to onEvent — templates can decide\n for (const evt of events) {\n onEventRef.current?.(evt);\n }\n }\n\n // Never decrease — protects against serverless instances with\n // slightly different version counters.\n versionRef = Math.max(versionRef, version);\n } catch {\n // Network error — will retry on next interval\n }\n if (!stopped) {\n timer = setTimeout(poll, interval);\n }\n }\n\n // Initial poll immediately\n poll();\n\n return () => {\n stopped = true;\n if (timer) clearTimeout(timer);\n };\n }, [pollUrl, queryClient, interval]);\n}\n\n/** @deprecated Use useDbSync instead */\nexport const useFileWatcher = useDbSync;\n\n/**\n * Subscribe to `refresh-screen` events from the agent. Returns an integer\n * that increments every time the agent invokes the framework's `refresh-screen`\n * tool. Apply it as a React `key` on the main content wrapper (the part\n * OUTSIDE the agent chat sidebar) so that region remounts and re-fetches its\n * data while the chat, sidebar, and any other persistent chrome keep their\n * in-flight state.\n *\n * Usage in a template's root:\n *\n * const screenKey = useScreenRefreshKey();\n * return (\n * <AppLayout>\n * <div key={screenKey}>\n * <Outlet />\n * </div>\n * </AppLayout>\n * );\n */\nexport function useScreenRefreshKey(\n options: { pollUrl?: string; interval?: number } = {},\n): number {\n const {\n pollUrl = agentNativePath(options.pollUrl ?? \"/_agent-native/poll\"),\n interval = 2000,\n } = options;\n const [key, setKey] = useState(0);\n\n useEffect(() => {\n let versionRef = 0;\n let timer: ReturnType<typeof setTimeout> | null = null;\n let stopped = false;\n\n async function poll() {\n if (stopped) return;\n try {\n const res = await fetch(`${pollUrl}?since=${versionRef}`);\n if (res.ok) {\n const data = (await res.json()) as {\n version: number;\n events: Array<{ source: string }>;\n };\n if (data.events?.some((e) => e.source === \"screen-refresh\")) {\n setKey((k) => k + 1);\n }\n versionRef = Math.max(versionRef, data.version);\n }\n } catch {\n // Network error — retry on next interval.\n }\n if (!stopped) timer = setTimeout(poll, interval);\n }\n\n poll();\n return () => {\n stopped = true;\n if (timer) clearTimeout(timer);\n };\n }, [pollUrl, interval]);\n\n return key;\n}\n"]}
1
+ {"version":3,"file":"use-db-sync.js","sourceRoot":"","sources":["../../src/client/use-db-sync.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,OAAO,CAAC;AACpD,OAAO,EAAE,eAAe,EAAE,MAAM,eAAe,CAAC;AAMhD,MAAM,iBAAiB,GAAG,MAAM,CAAC;AAEjC,SAAS,cAAc,CAAC,QAAgB;IACtC,OAAO,IAAI,CAAC,GAAG,CAAC,iBAAiB,EAAE,QAAQ,GAAG,CAAC,CAAC,CAAC;AACnD,CAAC;AAED,KAAK,UAAU,aAAa,CAC1B,OAAe,EACf,KAAa,EACb,QAAgB;IAEhB,MAAM,UAAU,GACd,OAAO,eAAe,KAAK,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,eAAe,EAAE,CAAC;IACxE,MAAM,OAAO,GAAG,UAAU;QACxB,CAAC,CAAC,UAAU,CAAC,GAAG,EAAE,CAAC,UAAU,CAAC,KAAK,EAAE,EAAE,cAAc,CAAC,QAAQ,CAAC,CAAC;QAChE,CAAC,CAAC,IAAI,CAAC;IAET,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,MAAM,KAAK,CACrB,GAAG,OAAO,UAAU,KAAK,EAAE,EAC3B,UAAU,CAAC,CAAC,CAAC,EAAE,MAAM,EAAE,UAAU,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC,SAAS,CACvD,CAAC;QACF,IAAI,CAAC,GAAG,CAAC,EAAE;YAAE,MAAM,IAAI,KAAK,CAAC,OAAO,GAAG,GAAG,CAAC,MAAM,CAAC,CAAC;QACnD,OAAO,GAAG,CAAC,IAAI,EAAE,CAAC;IACpB,CAAC;YAAS,CAAC;QACT,IAAI,OAAO;YAAE,YAAY,CAAC,OAAO,CAAC,CAAC;IACrC,CAAC;AACH,CAAC;AAED;;;;;;;;;;;;;;;GAeG;AACH,MAAM,UAAU,SAAS,CACvB,UASI,EAAE;IAEN,MAAM,EACJ,WAAW,EACX,SAAS,GAAG,CAAC,MAAM,CAAC,EACpB,OAAO,GAAG,eAAe,CAAC,OAAO,CAAC,SAAS,IAAI,qBAAqB,CAAC,EACrE,QAAQ,GAAG,IAAI,GAChB,GAAG,OAAO,CAAC;IAEZ,MAAM,UAAU,GAAG,MAAM,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;IAC3C,UAAU,CAAC,OAAO,GAAG,OAAO,CAAC,OAAO,CAAC;IAErC,MAAM,OAAO,GAAG,MAAM,CAAC,SAAS,CAAC,CAAC;IAClC,OAAO,CAAC,OAAO,GAAG,SAAS,CAAC;IAE5B,MAAM,eAAe,GAAG,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC,CAAC;IACrD,eAAe,CAAC,OAAO,GAAG,OAAO,CAAC,YAAY,CAAC;IAE/C,SAAS,CAAC,GAAG,EAAE;QACb,IAAI,UAAU,GAAG,CAAC,CAAC;QACnB,IAAI,KAAK,GAAyC,IAAI,CAAC;QACvD,IAAI,OAAO,GAAG,KAAK,CAAC;QACpB,IAAI,QAAQ,GAAG,KAAK,CAAC;QAErB,SAAS,YAAY;YACnB,IAAI,OAAO;gBAAE,OAAO;YACpB,IAAI,KAAK;gBAAE,YAAY,CAAC,KAAK,CAAC,CAAC;YAC/B,KAAK,GAAG,UAAU,CAAC,GAAG,EAAE;gBACtB,KAAK,GAAG,IAAI,CAAC;gBACb,KAAK,IAAI,EAAE,CAAC;YACd,CAAC,EAAE,QAAQ,CAAC,CAAC;QACf,CAAC;QAED,KAAK,UAAU,IAAI;YACjB,IAAI,OAAO,IAAI,QAAQ;gBAAE,OAAO;YAChC,QAAQ,GAAG,IAAI,CAAC;YAChB,IAAI,CAAC;gBACH,MAAM,IAAI,GAAG,MAAM,aAAa,CAQ7B,OAAO,EAAE,UAAU,EAAE,QAAQ,CAAC,CAAC;gBAClC,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,GAAG,IAQ3B,CAAC;gBAEF,IAAI,MAAM,CAAC,MAAM,GAAG,CAAC,IAAI,WAAW,EAAE,CAAC;oBACrC,MAAM,MAAM,GAAG,eAAe,CAAC,OAAO,CAAC;oBACvC,MAAM,QAAQ,GAAG,MAAM;wBACrB,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,aAAa,KAAK,MAAM,CAAC;wBACvD,CAAC,CAAC,MAAM,CAAC;oBAEX,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;wBACxB,KAAK,MAAM,GAAG,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC;4BAClC,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;wBACrD,CAAC;wBAED,kEAAkE;wBAClE,4DAA4D;wBAC5D,8DAA8D;wBAC9D,2CAA2C;wBAC3C,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;wBACxD,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;wBAC3D,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,CAAC,YAAY,CAAC,EAAE,CAAC,CAAC;wBAC5D,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;wBACtD,WAAW,CAAC,iBAAiB,CAAC,EAAE,QAAQ,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;oBACzD,CAAC;oBAED,8DAA8D;oBAC9D,KAAK,MAAM,GAAG,IAAI,MAAM,EAAE,CAAC;wBACzB,UAAU,CAAC,OAAO,EAAE,CAAC,GAAG,CAAC,CAAC;oBAC5B,CAAC;gBACH,CAAC;gBAED,8DAA8D;gBAC9D,uCAAuC;gBACvC,UAAU,GAAG,IAAI,CAAC,GAAG,CAAC,UAAU,EAAE,OAAO,CAAC,CAAC;YAC7C,CAAC;YAAC,MAAM,CAAC;gBACP,8CAA8C;YAChD,CAAC;oBAAS,CAAC;gBACT,QAAQ,GAAG,KAAK,CAAC;gBACjB,YAAY,EAAE,CAAC;YACjB,CAAC;QACH,CAAC;QAED,SAAS,OAAO;YACd,IACE,OAAO,QAAQ,KAAK,WAAW;gBAC/B,QAAQ,CAAC,eAAe,KAAK,QAAQ,EACrC,CAAC;gBACD,OAAO;YACT,CAAC;YACD,IAAI,KAAK,EAAE,CAAC;gBACV,YAAY,CAAC,KAAK,CAAC,CAAC;gBACpB,KAAK,GAAG,IAAI,CAAC;YACf,CAAC;YACD,KAAK,IAAI,EAAE,CAAC;QACd,CAAC;QAED,SAAS,sBAAsB;YAC7B,IAAI,QAAQ,CAAC,eAAe,KAAK,SAAS;gBAAE,OAAO,EAAE,CAAC;QACxD,CAAC;QAED,2BAA2B;QAC3B,KAAK,IAAI,EAAE,CAAC;QACZ,MAAM,CAAC,gBAAgB,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;QAC1C,QAAQ,CAAC,gBAAgB,CAAC,kBAAkB,EAAE,sBAAsB,CAAC,CAAC;QAEtE,OAAO,GAAG,EAAE;YACV,OAAO,GAAG,IAAI,CAAC;YACf,IAAI,KAAK;gBAAE,YAAY,CAAC,KAAK,CAAC,CAAC;YAC/B,MAAM,CAAC,mBAAmB,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;YAC7C,QAAQ,CAAC,mBAAmB,CAAC,kBAAkB,EAAE,sBAAsB,CAAC,CAAC;QAC3E,CAAC,CAAC;IACJ,CAAC,EAAE,CAAC,OAAO,EAAE,WAAW,EAAE,QAAQ,CAAC,CAAC,CAAC;AACvC,CAAC;AAED,wCAAwC;AACxC,MAAM,CAAC,MAAM,cAAc,GAAG,SAAS,CAAC;AAExC;;;;;;;;;;;;;;;;;;GAkBG;AACH,MAAM,UAAU,mBAAmB,CACjC,UAAmD,EAAE;IAErD,MAAM,EACJ,OAAO,GAAG,eAAe,CAAC,OAAO,CAAC,OAAO,IAAI,qBAAqB,CAAC,EACnE,QAAQ,GAAG,IAAI,GAChB,GAAG,OAAO,CAAC;IACZ,MAAM,CAAC,GAAG,EAAE,MAAM,CAAC,GAAG,QAAQ,CAAC,CAAC,CAAC,CAAC;IAElC,SAAS,CAAC,GAAG,EAAE;QACb,IAAI,UAAU,GAAG,CAAC,CAAC;QACnB,IAAI,KAAK,GAAyC,IAAI,CAAC;QACvD,IAAI,OAAO,GAAG,KAAK,CAAC;QACpB,IAAI,QAAQ,GAAG,KAAK,CAAC;QAErB,SAAS,YAAY;YACnB,IAAI,OAAO;gBAAE,OAAO;YACpB,IAAI,KAAK;gBAAE,YAAY,CAAC,KAAK,CAAC,CAAC;YAC/B,KAAK,GAAG,UAAU,CAAC,GAAG,EAAE;gBACtB,KAAK,GAAG,IAAI,CAAC;gBACb,KAAK,IAAI,EAAE,CAAC;YACd,CAAC,EAAE,QAAQ,CAAC,CAAC;QACf,CAAC;QAED,KAAK,UAAU,IAAI;YACjB,IAAI,OAAO,IAAI,QAAQ;gBAAE,OAAO;YAChC,QAAQ,GAAG,IAAI,CAAC;YAChB,IAAI,CAAC;gBACH,MAAM,IAAI,GAAG,MAAM,aAAa,CAG7B,OAAO,EAAE,UAAU,EAAE,QAAQ,CAAC,CAAC;gBAClC,IAAI,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,gBAAgB,CAAC,EAAE,CAAC;oBAC5D,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;gBACvB,CAAC;gBACD,UAAU,GAAG,IAAI,CAAC,GAAG,CAAC,UAAU,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;YAClD,CAAC;YAAC,MAAM,CAAC;gBACP,0CAA0C;YAC5C,CAAC;oBAAS,CAAC;gBACT,QAAQ,GAAG,KAAK,CAAC;gBACjB,YAAY,EAAE,CAAC;YACjB,CAAC;QACH,CAAC;QAED,SAAS,OAAO;YACd,IACE,OAAO,QAAQ,KAAK,WAAW;gBAC/B,QAAQ,CAAC,eAAe,KAAK,QAAQ,EACrC,CAAC;gBACD,OAAO;YACT,CAAC;YACD,IAAI,KAAK,EAAE,CAAC;gBACV,YAAY,CAAC,KAAK,CAAC,CAAC;gBACpB,KAAK,GAAG,IAAI,CAAC;YACf,CAAC;YACD,KAAK,IAAI,EAAE,CAAC;QACd,CAAC;QAED,SAAS,sBAAsB;YAC7B,IAAI,QAAQ,CAAC,eAAe,KAAK,SAAS;gBAAE,OAAO,EAAE,CAAC;QACxD,CAAC;QAED,KAAK,IAAI,EAAE,CAAC;QACZ,MAAM,CAAC,gBAAgB,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;QAC1C,QAAQ,CAAC,gBAAgB,CAAC,kBAAkB,EAAE,sBAAsB,CAAC,CAAC;QAEtE,OAAO,GAAG,EAAE;YACV,OAAO,GAAG,IAAI,CAAC;YACf,IAAI,KAAK;gBAAE,YAAY,CAAC,KAAK,CAAC,CAAC;YAC/B,MAAM,CAAC,mBAAmB,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC;YAC7C,QAAQ,CAAC,mBAAmB,CAAC,kBAAkB,EAAE,sBAAsB,CAAC,CAAC;QAC3E,CAAC,CAAC;IACJ,CAAC,EAAE,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC,CAAC;IAExB,OAAO,GAAG,CAAC;AACb,CAAC","sourcesContent":["import { useEffect, useRef, useState } from \"react\";\nimport { agentNativePath } from \"./api-path.js\";\n\ninterface QueryClient {\n invalidateQueries(opts?: { queryKey?: string[] }): void;\n}\n\nconst POLL_ABORT_MIN_MS = 10_000;\n\nfunction getPollAbortMs(interval: number): number {\n return Math.max(POLL_ABORT_MIN_MS, interval * 4);\n}\n\nasync function fetchPollJson<T>(\n pollUrl: string,\n since: number,\n interval: number,\n): Promise<T> {\n const controller =\n typeof AbortController === \"undefined\" ? null : new AbortController();\n const timeout = controller\n ? setTimeout(() => controller.abort(), getPollAbortMs(interval))\n : null;\n\n try {\n const res = await fetch(\n `${pollUrl}?since=${since}`,\n controller ? { signal: controller.signal } : undefined,\n );\n if (!res.ok) throw new Error(\"HTTP \" + res.status);\n return res.json();\n } finally {\n if (timeout) clearTimeout(timeout);\n }\n}\n\n/**\n * Hook that polls /_agent-native/poll for DB change events and invalidates\n * react-query caches when changes are detected.\n *\n * Works in all deployment environments (serverless, edge, long-lived server).\n *\n * @param options.queryClient - The react-query QueryClient instance\n * @param options.queryKeys - Array of query key prefixes to invalidate on change.\n * Default: [\"data\"]\n * @param options.pollUrl - Poll endpoint URL. Default: \"/_agent-native/poll\"\n * @param options.onEvent - Optional callback for each change event\n * @param options.interval - Poll interval in ms. Default: 2000\n * @param options.ignoreSource - Skip events whose `requestSource` matches this\n * value. Use a per-tab ID so the UI ignores its own writes while still\n * picking up changes from other tabs, agents, and scripts.\n */\nexport function useDbSync(\n options: {\n queryClient?: QueryClient;\n queryKeys?: string[];\n pollUrl?: string;\n /** @deprecated Use pollUrl instead */\n eventsUrl?: string;\n onEvent?: (data: any) => void;\n interval?: number;\n ignoreSource?: string;\n } = {},\n): void {\n const {\n queryClient,\n queryKeys = [\"data\"],\n pollUrl = agentNativePath(options.eventsUrl ?? \"/_agent-native/poll\"),\n interval = 2000,\n } = options;\n\n const onEventRef = useRef(options.onEvent);\n onEventRef.current = options.onEvent;\n\n const keysRef = useRef(queryKeys);\n keysRef.current = queryKeys;\n\n const ignoreSourceRef = useRef(options.ignoreSource);\n ignoreSourceRef.current = options.ignoreSource;\n\n useEffect(() => {\n let versionRef = 0;\n let timer: ReturnType<typeof setTimeout> | null = null;\n let stopped = false;\n let inFlight = false;\n\n function schedulePoll() {\n if (stopped) return;\n if (timer) clearTimeout(timer);\n timer = setTimeout(() => {\n timer = null;\n void poll();\n }, interval);\n }\n\n async function poll() {\n if (stopped || inFlight) return;\n inFlight = true;\n try {\n const data = await fetchPollJson<{\n version: number;\n events: Array<{\n source: string;\n type: string;\n key?: string;\n requestSource?: string;\n }>;\n }>(pollUrl, versionRef, interval);\n const { version, events } = data as {\n version: number;\n events: Array<{\n source: string;\n type: string;\n key?: string;\n requestSource?: string;\n }>;\n };\n\n if (events.length > 0 && queryClient) {\n const ignore = ignoreSourceRef.current;\n const relevant = ignore\n ? events.filter((e: any) => e.requestSource !== ignore)\n : events;\n\n if (relevant.length > 0) {\n for (const key of keysRef.current) {\n queryClient.invalidateQueries({ queryKey: [key] });\n }\n\n // Framework-level invalidation: always invalidate framework query\n // keys on any non-own change event so that mutating actions\n // (agent or HTTP) auto-refresh the UI — regardless of how the\n // template configured queryKeys / onEvent.\n queryClient.invalidateQueries({ queryKey: [\"action\"] });\n queryClient.invalidateQueries({ queryKey: [\"extension\"] });\n queryClient.invalidateQueries({ queryKey: [\"extensions\"] });\n queryClient.invalidateQueries({ queryKey: [\"tool\"] });\n queryClient.invalidateQueries({ queryKey: [\"tools\"] });\n }\n\n // Always forward all events to onEvent — templates can decide\n for (const evt of events) {\n onEventRef.current?.(evt);\n }\n }\n\n // Never decrease — protects against serverless instances with\n // slightly different version counters.\n versionRef = Math.max(versionRef, version);\n } catch {\n // Network error — will retry on next interval\n } finally {\n inFlight = false;\n schedulePoll();\n }\n }\n\n function pollNow() {\n if (\n typeof document !== \"undefined\" &&\n document.visibilityState === \"hidden\"\n ) {\n return;\n }\n if (timer) {\n clearTimeout(timer);\n timer = null;\n }\n void poll();\n }\n\n function handleVisibilityChange() {\n if (document.visibilityState === \"visible\") pollNow();\n }\n\n // Initial poll immediately\n void poll();\n window.addEventListener(\"focus\", pollNow);\n document.addEventListener(\"visibilitychange\", handleVisibilityChange);\n\n return () => {\n stopped = true;\n if (timer) clearTimeout(timer);\n window.removeEventListener(\"focus\", pollNow);\n document.removeEventListener(\"visibilitychange\", handleVisibilityChange);\n };\n }, [pollUrl, queryClient, interval]);\n}\n\n/** @deprecated Use useDbSync instead */\nexport const useFileWatcher = useDbSync;\n\n/**\n * Subscribe to `refresh-screen` events from the agent. Returns an integer\n * that increments every time the agent invokes the framework's `refresh-screen`\n * tool. Apply it as a React `key` on the main content wrapper (the part\n * OUTSIDE the agent chat sidebar) so that region remounts and re-fetches its\n * data while the chat, sidebar, and any other persistent chrome keep their\n * in-flight state.\n *\n * Usage in a template's root:\n *\n * const screenKey = useScreenRefreshKey();\n * return (\n * <AppLayout>\n * <div key={screenKey}>\n * <Outlet />\n * </div>\n * </AppLayout>\n * );\n */\nexport function useScreenRefreshKey(\n options: { pollUrl?: string; interval?: number } = {},\n): number {\n const {\n pollUrl = agentNativePath(options.pollUrl ?? \"/_agent-native/poll\"),\n interval = 2000,\n } = options;\n const [key, setKey] = useState(0);\n\n useEffect(() => {\n let versionRef = 0;\n let timer: ReturnType<typeof setTimeout> | null = null;\n let stopped = false;\n let inFlight = false;\n\n function schedulePoll() {\n if (stopped) return;\n if (timer) clearTimeout(timer);\n timer = setTimeout(() => {\n timer = null;\n void poll();\n }, interval);\n }\n\n async function poll() {\n if (stopped || inFlight) return;\n inFlight = true;\n try {\n const data = await fetchPollJson<{\n version: number;\n events: Array<{ source: string }>;\n }>(pollUrl, versionRef, interval);\n if (data.events?.some((e) => e.source === \"screen-refresh\")) {\n setKey((k) => k + 1);\n }\n versionRef = Math.max(versionRef, data.version);\n } catch {\n // Network error — retry on next interval.\n } finally {\n inFlight = false;\n schedulePoll();\n }\n }\n\n function pollNow() {\n if (\n typeof document !== \"undefined\" &&\n document.visibilityState === \"hidden\"\n ) {\n return;\n }\n if (timer) {\n clearTimeout(timer);\n timer = null;\n }\n void poll();\n }\n\n function handleVisibilityChange() {\n if (document.visibilityState === \"visible\") pollNow();\n }\n\n void poll();\n window.addEventListener(\"focus\", pollNow);\n document.addEventListener(\"visibilitychange\", handleVisibilityChange);\n\n return () => {\n stopped = true;\n if (timer) clearTimeout(timer);\n window.removeEventListener(\"focus\", pollNow);\n document.removeEventListener(\"visibilitychange\", handleVisibilityChange);\n };\n }, [pollUrl, interval]);\n\n return key;\n}\n"]}
@@ -1 +1 @@
1
- {"version":3,"file":"build.d.ts","sourceRoot":"","sources":["../../src/deploy/build.ts"],"names":[],"mappings":";AAEA;;;;;;;;;;;;GAYG;AAOH,OAAO,EAML,KAAK,eAAe,EACpB,KAAK,gBAAgB,EACtB,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EAEL,KAAK,oBAAoB,EAC1B,MAAM,qBAAqB,CAAC;AAyB7B;;;;;;;;GAQG;AACH,wBAAgB,mBAAmB,CACjC,MAAM,EAAE,eAAe,EAAE,EACzB,WAAW,EAAE,MAAM,EAAE,EACrB,kBAAkB,GAAE,MAAM,EAAO,EACjC,OAAO,GAAE,gBAAgB,EAAO,EAChC,aAAa,GAAE,oBAAoB,GAAG,IAAW,GAChD,MAAM,CAkVR;AAiXD,wBAAgB,mBAAmB,IAAI,MAAM,EAAE,CAE9C"}
1
+ {"version":3,"file":"build.d.ts","sourceRoot":"","sources":["../../src/deploy/build.ts"],"names":[],"mappings":";AAEA;;;;;;;;;;;;GAYG;AAOH,OAAO,EAML,KAAK,eAAe,EACpB,KAAK,gBAAgB,EACtB,MAAM,sBAAsB,CAAC;AAC9B,OAAO,EAEL,KAAK,oBAAoB,EAC1B,MAAM,qBAAqB,CAAC;AA8B7B;;;;;;;;GAQG;AACH,wBAAgB,mBAAmB,CACjC,MAAM,EAAE,eAAe,EAAE,EACzB,WAAW,EAAE,MAAM,EAAE,EACrB,kBAAkB,GAAE,MAAM,EAAO,EACjC,OAAO,GAAE,gBAAgB,EAAO,EAChC,aAAa,GAAE,oBAAoB,GAAG,IAAW,GAChD,MAAM,CAkVR;AAiXD,wBAAgB,mBAAmB,IAAI,MAAM,EAAE,CAE9C"}
@@ -35,6 +35,11 @@ function normalizeConfiguredAppBasePath() {
35
35
  /** Plugins that require Node.js runtime and cannot run on edge/serverless */
36
36
  const NODE_ONLY_PLUGINS = new Set([
37
37
  "terminal", // PTY requires child_process
38
+ // @sentry/node ships node:fs / node:async_hooks bindings that don't load
39
+ // on workerd / Cloudflare Workers. Templates running on edge presets can
40
+ // mount their own edge-compatible Sentry wrapper if they want server
41
+ // observability there; the framework default is the Node SDK.
42
+ "sentry",
38
43
  ]);
39
44
  function isNodeOnlyPlugin(filePath) {
40
45
  const basename = path.basename(filePath, path.extname(filePath));