@agent-native/core 0.7.12 → 0.7.13

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 (302) hide show
  1. package/README.md +1 -1
  2. package/dist/agent/engine/ai-sdk-engine.d.ts.map +1 -1
  3. package/dist/agent/engine/ai-sdk-engine.js +26 -8
  4. package/dist/agent/engine/ai-sdk-engine.js.map +1 -1
  5. package/dist/agent/engine/builder-engine.d.ts +19 -0
  6. package/dist/agent/engine/builder-engine.d.ts.map +1 -0
  7. package/dist/agent/engine/builder-engine.js +430 -0
  8. package/dist/agent/engine/builder-engine.js.map +1 -0
  9. package/dist/agent/engine/builtin.d.ts.map +1 -1
  10. package/dist/agent/engine/builtin.js +26 -10
  11. package/dist/agent/engine/builtin.js.map +1 -1
  12. package/dist/agent/engine/index.d.ts +1 -1
  13. package/dist/agent/engine/index.d.ts.map +1 -1
  14. package/dist/agent/engine/index.js +1 -1
  15. package/dist/agent/engine/index.js.map +1 -1
  16. package/dist/agent/engine/registry.d.ts +20 -1
  17. package/dist/agent/engine/registry.d.ts.map +1 -1
  18. package/dist/agent/engine/registry.js +49 -1
  19. package/dist/agent/engine/registry.js.map +1 -1
  20. package/dist/agent/engine/types.d.ts +30 -0
  21. package/dist/agent/engine/types.d.ts.map +1 -1
  22. package/dist/agent/engine/types.js +19 -1
  23. package/dist/agent/engine/types.js.map +1 -1
  24. package/dist/agent/production-agent.d.ts.map +1 -1
  25. package/dist/agent/production-agent.js +65 -7
  26. package/dist/agent/production-agent.js.map +1 -1
  27. package/dist/agent/run-manager.d.ts.map +1 -1
  28. package/dist/agent/run-manager.js +11 -1
  29. package/dist/agent/run-manager.js.map +1 -1
  30. package/dist/agent/thread-data-builder.d.ts +4 -0
  31. package/dist/agent/thread-data-builder.d.ts.map +1 -1
  32. package/dist/agent/thread-data-builder.js +1 -0
  33. package/dist/agent/thread-data-builder.js.map +1 -1
  34. package/dist/agent/types.d.ts +8 -0
  35. package/dist/agent/types.d.ts.map +1 -1
  36. package/dist/checkpoints/service.d.ts +1 -0
  37. package/dist/checkpoints/service.d.ts.map +1 -1
  38. package/dist/checkpoints/service.js +26 -2
  39. package/dist/checkpoints/service.js.map +1 -1
  40. package/dist/cli/create.d.ts +30 -0
  41. package/dist/cli/create.d.ts.map +1 -1
  42. package/dist/cli/create.js +25 -13
  43. package/dist/cli/create.js.map +1 -1
  44. package/dist/client/AgentPanel.js +1 -1
  45. package/dist/client/AgentPanel.js.map +1 -1
  46. package/dist/client/AssistantChat.d.ts.map +1 -1
  47. package/dist/client/AssistantChat.js +49 -10
  48. package/dist/client/AssistantChat.js.map +1 -1
  49. package/dist/client/ConnectBuilderCard.d.ts +1 -7
  50. package/dist/client/ConnectBuilderCard.d.ts.map +1 -1
  51. package/dist/client/ConnectBuilderCard.js +30 -132
  52. package/dist/client/ConnectBuilderCard.js.map +1 -1
  53. package/dist/client/MultiTabAssistantChat.d.ts.map +1 -1
  54. package/dist/client/MultiTabAssistantChat.js +30 -9
  55. package/dist/client/MultiTabAssistantChat.js.map +1 -1
  56. package/dist/client/analytics.d.ts +5 -8
  57. package/dist/client/analytics.d.ts.map +1 -1
  58. package/dist/client/analytics.js +53 -11
  59. package/dist/client/analytics.js.map +1 -1
  60. package/dist/client/builder-mark.d.ts +9 -0
  61. package/dist/client/builder-mark.d.ts.map +1 -0
  62. package/dist/client/builder-mark.js +10 -0
  63. package/dist/client/builder-mark.js.map +1 -0
  64. package/dist/client/components/ui/popover.d.ts +8 -0
  65. package/dist/client/components/ui/popover.d.ts.map +1 -0
  66. package/dist/client/components/ui/popover.js +11 -0
  67. package/dist/client/components/ui/popover.js.map +1 -0
  68. package/dist/client/composer/ComposerPlusMenu.d.ts +2 -0
  69. package/dist/client/composer/ComposerPlusMenu.d.ts.map +1 -0
  70. package/dist/client/composer/ComposerPlusMenu.js +244 -0
  71. package/dist/client/composer/ComposerPlusMenu.js.map +1 -0
  72. package/dist/client/composer/TiptapComposer.d.ts.map +1 -1
  73. package/dist/client/composer/TiptapComposer.js +9 -5
  74. package/dist/client/composer/TiptapComposer.js.map +1 -1
  75. package/dist/client/composer/useVoiceDictation.d.ts.map +1 -1
  76. package/dist/client/composer/useVoiceDictation.js +4 -2
  77. package/dist/client/composer/useVoiceDictation.js.map +1 -1
  78. package/dist/client/error-format.d.ts +2 -0
  79. package/dist/client/error-format.d.ts.map +1 -0
  80. package/dist/client/error-format.js +31 -0
  81. package/dist/client/error-format.js.map +1 -0
  82. package/dist/client/index.d.ts +3 -1
  83. package/dist/client/index.d.ts.map +1 -1
  84. package/dist/client/index.js +3 -1
  85. package/dist/client/index.js.map +1 -1
  86. package/dist/client/observability/ObservabilityDashboard.d.ts +5 -0
  87. package/dist/client/observability/ObservabilityDashboard.d.ts.map +1 -0
  88. package/dist/client/observability/ObservabilityDashboard.js +169 -0
  89. package/dist/client/observability/ObservabilityDashboard.js.map +1 -0
  90. package/dist/client/observability/ThumbsFeedback.d.ts +8 -0
  91. package/dist/client/observability/ThumbsFeedback.d.ts.map +1 -0
  92. package/dist/client/observability/ThumbsFeedback.js +64 -0
  93. package/dist/client/observability/ThumbsFeedback.js.map +1 -0
  94. package/dist/client/observability/index.d.ts +4 -0
  95. package/dist/client/observability/index.d.ts.map +1 -0
  96. package/dist/client/observability/index.js +4 -0
  97. package/dist/client/observability/index.js.map +1 -0
  98. package/dist/client/observability/useObservability.d.ts +128 -0
  99. package/dist/client/observability/useObservability.d.ts.map +1 -0
  100. package/dist/client/observability/useObservability.js +109 -0
  101. package/dist/client/observability/useObservability.js.map +1 -0
  102. package/dist/client/onboarding/OnboardingPanel.d.ts.map +1 -1
  103. package/dist/client/onboarding/OnboardingPanel.js +34 -92
  104. package/dist/client/onboarding/OnboardingPanel.js.map +1 -1
  105. package/dist/client/org/RequireActiveOrg.d.ts +33 -0
  106. package/dist/client/org/RequireActiveOrg.d.ts.map +1 -0
  107. package/dist/client/org/RequireActiveOrg.js +63 -0
  108. package/dist/client/org/RequireActiveOrg.js.map +1 -0
  109. package/dist/client/org/hooks.d.ts.map +1 -1
  110. package/dist/client/org/hooks.js +50 -15
  111. package/dist/client/org/hooks.js.map +1 -1
  112. package/dist/client/org/index.d.ts +1 -0
  113. package/dist/client/org/index.d.ts.map +1 -1
  114. package/dist/client/org/index.js +1 -0
  115. package/dist/client/org/index.js.map +1 -1
  116. package/dist/client/resources/ResourcesPanel.js +3 -3
  117. package/dist/client/resources/ResourcesPanel.js.map +1 -1
  118. package/dist/client/settings/AutomationsSection.js +1 -1
  119. package/dist/client/settings/AutomationsSection.js.map +1 -1
  120. package/dist/client/settings/BrowserSection.js +1 -1
  121. package/dist/client/settings/BrowserSection.js.map +1 -1
  122. package/dist/client/settings/SettingsPanel.d.ts.map +1 -1
  123. package/dist/client/settings/SettingsPanel.js +112 -12
  124. package/dist/client/settings/SettingsPanel.js.map +1 -1
  125. package/dist/client/settings/VoiceTranscriptionSection.d.ts.map +1 -1
  126. package/dist/client/settings/VoiceTranscriptionSection.js +10 -4
  127. package/dist/client/settings/VoiceTranscriptionSection.js.map +1 -1
  128. package/dist/client/settings/useBuilderStatus.d.ts +26 -0
  129. package/dist/client/settings/useBuilderStatus.d.ts.map +1 -1
  130. package/dist/client/settings/useBuilderStatus.js +128 -4
  131. package/dist/client/settings/useBuilderStatus.js.map +1 -1
  132. package/dist/client/sse-event-processor.d.ts +2 -0
  133. package/dist/client/sse-event-processor.d.ts.map +1 -1
  134. package/dist/client/sse-event-processor.js +6 -2
  135. package/dist/client/sse-event-processor.js.map +1 -1
  136. package/dist/client/transcription/BuilderTranscriptionCta.d.ts +9 -0
  137. package/dist/client/transcription/BuilderTranscriptionCta.d.ts.map +1 -0
  138. package/dist/client/transcription/BuilderTranscriptionCta.js +18 -0
  139. package/dist/client/transcription/BuilderTranscriptionCta.js.map +1 -0
  140. package/dist/client/transcription/use-live-transcription.d.ts +29 -0
  141. package/dist/client/transcription/use-live-transcription.d.ts.map +1 -0
  142. package/dist/client/transcription/use-live-transcription.js +156 -0
  143. package/dist/client/transcription/use-live-transcription.js.map +1 -0
  144. package/dist/client/use-builder-enabled.d.ts +17 -0
  145. package/dist/client/use-builder-enabled.d.ts.map +1 -0
  146. package/dist/client/use-builder-enabled.js +36 -0
  147. package/dist/client/use-builder-enabled.js.map +1 -0
  148. package/dist/client/use-db-sync.d.ts.map +1 -1
  149. package/dist/client/use-db-sync.js +4 -2
  150. package/dist/client/use-db-sync.js.map +1 -1
  151. package/dist/client/useProductionAgent.d.ts.map +1 -1
  152. package/dist/client/useProductionAgent.js +3 -1
  153. package/dist/client/useProductionAgent.js.map +1 -1
  154. package/dist/db/migrations.d.ts +9 -0
  155. package/dist/db/migrations.d.ts.map +1 -1
  156. package/dist/db/migrations.js +75 -10
  157. package/dist/db/migrations.js.map +1 -1
  158. package/dist/file-upload/builder.d.ts.map +1 -1
  159. package/dist/file-upload/builder.js +11 -4
  160. package/dist/file-upload/builder.js.map +1 -1
  161. package/dist/jobs/tools.d.ts.map +1 -1
  162. package/dist/jobs/tools.js +137 -161
  163. package/dist/jobs/tools.js.map +1 -1
  164. package/dist/notifications/actions.d.ts +2 -2
  165. package/dist/notifications/actions.d.ts.map +1 -1
  166. package/dist/notifications/actions.js +77 -69
  167. package/dist/notifications/actions.js.map +1 -1
  168. package/dist/observability/evals.d.ts +22 -0
  169. package/dist/observability/evals.d.ts.map +1 -0
  170. package/dist/observability/evals.js +371 -0
  171. package/dist/observability/evals.js.map +1 -0
  172. package/dist/observability/experiments.d.ts +24 -0
  173. package/dist/observability/experiments.d.ts.map +1 -0
  174. package/dist/observability/experiments.js +274 -0
  175. package/dist/observability/experiments.js.map +1 -0
  176. package/dist/observability/feedback.d.ts +14 -0
  177. package/dist/observability/feedback.d.ts.map +1 -0
  178. package/dist/observability/feedback.js +256 -0
  179. package/dist/observability/feedback.js.map +1 -0
  180. package/dist/observability/index.d.ts +6 -0
  181. package/dist/observability/index.d.ts.map +1 -0
  182. package/dist/observability/index.js +5 -0
  183. package/dist/observability/index.js.map +1 -0
  184. package/dist/observability/plugin.d.ts +2 -0
  185. package/dist/observability/plugin.d.ts.map +1 -0
  186. package/dist/observability/plugin.js +12 -0
  187. package/dist/observability/plugin.js.map +1 -0
  188. package/dist/observability/routes.d.ts +68 -0
  189. package/dist/observability/routes.d.ts.map +1 -0
  190. package/dist/observability/routes.js +301 -0
  191. package/dist/observability/routes.js.map +1 -0
  192. package/dist/observability/store.d.ts +77 -0
  193. package/dist/observability/store.d.ts.map +1 -0
  194. package/dist/observability/store.js +976 -0
  195. package/dist/observability/store.js.map +1 -0
  196. package/dist/observability/traces.d.ts +37 -0
  197. package/dist/observability/traces.d.ts.map +1 -0
  198. package/dist/observability/traces.js +182 -0
  199. package/dist/observability/traces.js.map +1 -0
  200. package/dist/observability/types.d.ts +159 -0
  201. package/dist/observability/types.d.ts.map +1 -0
  202. package/dist/observability/types.js +16 -0
  203. package/dist/observability/types.js.map +1 -0
  204. package/dist/onboarding/default-steps.d.ts.map +1 -1
  205. package/dist/onboarding/default-steps.js +6 -5
  206. package/dist/onboarding/default-steps.js.map +1 -1
  207. package/dist/onboarding/types.d.ts +10 -1
  208. package/dist/onboarding/types.d.ts.map +1 -1
  209. package/dist/org/context.d.ts +8 -1
  210. package/dist/org/context.d.ts.map +1 -1
  211. package/dist/org/context.js +163 -6
  212. package/dist/org/context.js.map +1 -1
  213. package/dist/org/handlers.d.ts.map +1 -1
  214. package/dist/org/handlers.js +49 -30
  215. package/dist/org/handlers.js.map +1 -1
  216. package/dist/progress/actions.d.ts +3 -0
  217. package/dist/progress/actions.d.ts.map +1 -1
  218. package/dist/progress/actions.js +86 -110
  219. package/dist/progress/actions.js.map +1 -1
  220. package/dist/progress/routes.d.ts +1 -1
  221. package/dist/progress/routes.js +1 -1
  222. package/dist/scripts/agent-engines/list-agent-engines.js +1 -1
  223. package/dist/scripts/agent-engines/list-agent-engines.js.map +1 -1
  224. package/dist/scripts/agent-engines/manage-agent-engine.d.ts +10 -0
  225. package/dist/scripts/agent-engines/manage-agent-engine.d.ts.map +1 -0
  226. package/dist/scripts/agent-engines/manage-agent-engine.js +47 -0
  227. package/dist/scripts/agent-engines/manage-agent-engine.js.map +1 -0
  228. package/dist/scripts/agent-engines/set-agent-engine.js +2 -2
  229. package/dist/scripts/agent-engines/set-agent-engine.js.map +1 -1
  230. package/dist/server/agent-chat-plugin.d.ts +39 -0
  231. package/dist/server/agent-chat-plugin.d.ts.map +1 -1
  232. package/dist/server/agent-chat-plugin.js +707 -443
  233. package/dist/server/agent-chat-plugin.js.map +1 -1
  234. package/dist/server/agent-teams.js +1 -1
  235. package/dist/server/agent-teams.js.map +1 -1
  236. package/dist/server/analytics.d.ts +5 -6
  237. package/dist/server/analytics.d.ts.map +1 -1
  238. package/dist/server/analytics.js +6 -14
  239. package/dist/server/analytics.js.map +1 -1
  240. package/dist/server/app-name.d.ts +5 -2
  241. package/dist/server/app-name.d.ts.map +1 -1
  242. package/dist/server/app-name.js +14 -3
  243. package/dist/server/app-name.js.map +1 -1
  244. package/dist/server/app-url.d.ts.map +1 -1
  245. package/dist/server/app-url.js +10 -1
  246. package/dist/server/app-url.js.map +1 -1
  247. package/dist/server/auth.d.ts +2 -0
  248. package/dist/server/auth.d.ts.map +1 -1
  249. package/dist/server/auth.js +153 -2
  250. package/dist/server/auth.js.map +1 -1
  251. package/dist/server/better-auth-instance.d.ts +2 -0
  252. package/dist/server/better-auth-instance.d.ts.map +1 -1
  253. package/dist/server/better-auth-instance.js +4 -0
  254. package/dist/server/better-auth-instance.js.map +1 -1
  255. package/dist/server/builder-browser.d.ts +59 -1
  256. package/dist/server/builder-browser.d.ts.map +1 -1
  257. package/dist/server/builder-browser.js +127 -11
  258. package/dist/server/builder-browser.js.map +1 -1
  259. package/dist/server/core-routes-plugin.d.ts.map +1 -1
  260. package/dist/server/core-routes-plugin.js +208 -6
  261. package/dist/server/core-routes-plugin.js.map +1 -1
  262. package/dist/server/credential-provider.d.ts +7 -0
  263. package/dist/server/credential-provider.d.ts.map +1 -1
  264. package/dist/server/credential-provider.js +10 -0
  265. package/dist/server/credential-provider.js.map +1 -1
  266. package/dist/server/onboarding-html.d.ts.map +1 -1
  267. package/dist/server/onboarding-html.js +29 -4
  268. package/dist/server/onboarding-html.js.map +1 -1
  269. package/dist/server/poll.d.ts.map +1 -1
  270. package/dist/server/poll.js +46 -5
  271. package/dist/server/poll.js.map +1 -1
  272. package/dist/server/ssr-handler.d.ts.map +1 -1
  273. package/dist/server/ssr-handler.js +2 -1
  274. package/dist/server/ssr-handler.js.map +1 -1
  275. package/dist/server/transcribe-voice.d.ts.map +1 -1
  276. package/dist/server/transcribe-voice.js +44 -5
  277. package/dist/server/transcribe-voice.js.map +1 -1
  278. package/dist/styles/agent-native.css +11 -2
  279. package/dist/templates/default/.agents/skills/progress/SKILL.md +14 -12
  280. package/dist/templates/default/app/root.tsx +7 -0
  281. package/dist/transcription/builder-transcription.d.ts +27 -0
  282. package/dist/transcription/builder-transcription.d.ts.map +1 -0
  283. package/dist/transcription/builder-transcription.js +41 -0
  284. package/dist/transcription/builder-transcription.js.map +1 -0
  285. package/dist/triggers/actions.d.ts +3 -0
  286. package/dist/triggers/actions.d.ts.map +1 -1
  287. package/dist/triggers/actions.js +189 -213
  288. package/dist/triggers/actions.js.map +1 -1
  289. package/docs/content/agent-mentions.md +1 -1
  290. package/docs/content/automations.md +22 -19
  291. package/docs/content/cloneable-saas.md +2 -2
  292. package/docs/content/deployment.md +21 -61
  293. package/docs/content/getting-started.md +1 -1
  294. package/docs/content/key-concepts.md +1 -1
  295. package/docs/content/{enterprise-workspace.md → multi-app-workspace.md} +3 -3
  296. package/docs/content/multi-tenancy.md +1 -1
  297. package/docs/content/progress.md +11 -11
  298. package/docs/content/template-dispatch.md +3 -3
  299. package/docs/content/workspace-management.md +1 -1
  300. package/package.json +9 -2
  301. package/src/templates/default/.agents/skills/progress/SKILL.md +14 -12
  302. package/src/templates/default/app/root.tsx +7 -0
@@ -0,0 +1,976 @@
1
+ /**
2
+ * SQL persistence for the agent observability system.
3
+ *
4
+ * Creates and manages tables for traces, feedback, evals, experiments,
5
+ * and satisfaction scores. Follows the same raw-SQL pattern as
6
+ * run-store.ts and usage/store.ts — framework tables use getDbExec()
7
+ * rather than Drizzle ORM (which is for template-level schemas).
8
+ */
9
+ import { getDbExec, intType, isPostgres } from "../db/client.js";
10
+ import { isDuplicateColumnError } from "../db/migrations.js";
11
+ function safeJsonParse(value, fallback) {
12
+ if (!value)
13
+ return fallback;
14
+ try {
15
+ return JSON.parse(String(value));
16
+ }
17
+ catch {
18
+ return fallback;
19
+ }
20
+ }
21
+ // Tables whose rows are owned by an end user — drives the boot-time
22
+ // user_id ALTER loop and the per-user composite indexes below. Every
23
+ // new user-owned observability table must be added here so the upgrade
24
+ // path and the per-user query plan stay in sync.
25
+ const USER_SCOPED_TABLES = [
26
+ "agent_trace_spans",
27
+ "agent_trace_summaries",
28
+ "agent_satisfaction_scores",
29
+ "agent_evals",
30
+ "agent_feedback",
31
+ ];
32
+ /**
33
+ * Append an `AND user_id = ?` clause when a userId filter is requested.
34
+ * Returns the fully-bound WHERE clause + args ready to splice into the
35
+ * caller's SQL. Centralizes the pattern so tests can assert one shape.
36
+ */
37
+ function withUserFilter(baseWhere, baseArgs, userId) {
38
+ if (userId == null)
39
+ return { where: baseWhere, args: baseArgs };
40
+ return {
41
+ where: `${baseWhere} AND user_id = ?`,
42
+ args: [...baseArgs, userId],
43
+ };
44
+ }
45
+ let _initPromise;
46
+ export async function ensureObservabilityTables() {
47
+ if (!_initPromise) {
48
+ _initPromise = (async () => {
49
+ const client = getDbExec();
50
+ await client.execute(`
51
+ CREATE TABLE IF NOT EXISTS agent_trace_spans (
52
+ id TEXT PRIMARY KEY,
53
+ run_id TEXT NOT NULL,
54
+ thread_id TEXT,
55
+ user_id TEXT,
56
+ parent_span_id TEXT,
57
+ span_type TEXT NOT NULL,
58
+ name TEXT NOT NULL,
59
+ input_tokens ${intType()} NOT NULL DEFAULT 0,
60
+ output_tokens ${intType()} NOT NULL DEFAULT 0,
61
+ cache_read_tokens ${intType()} NOT NULL DEFAULT 0,
62
+ cache_write_tokens ${intType()} NOT NULL DEFAULT 0,
63
+ cost_cents_x100 ${intType()} NOT NULL DEFAULT 0,
64
+ duration_ms ${intType()} NOT NULL DEFAULT 0,
65
+ status TEXT NOT NULL DEFAULT 'success',
66
+ error_message TEXT,
67
+ metadata TEXT,
68
+ created_at ${intType()} NOT NULL
69
+ )
70
+ `);
71
+ await client.execute(`
72
+ CREATE TABLE IF NOT EXISTS agent_trace_summaries (
73
+ run_id TEXT PRIMARY KEY,
74
+ thread_id TEXT,
75
+ user_id TEXT,
76
+ total_spans ${intType()} NOT NULL DEFAULT 0,
77
+ llm_calls ${intType()} NOT NULL DEFAULT 0,
78
+ tool_calls ${intType()} NOT NULL DEFAULT 0,
79
+ successful_tools ${intType()} NOT NULL DEFAULT 0,
80
+ failed_tools ${intType()} NOT NULL DEFAULT 0,
81
+ total_duration_ms ${intType()} NOT NULL DEFAULT 0,
82
+ total_cost_cents_x100 ${intType()} NOT NULL DEFAULT 0,
83
+ total_input_tokens ${intType()} NOT NULL DEFAULT 0,
84
+ total_output_tokens ${intType()} NOT NULL DEFAULT 0,
85
+ model TEXT NOT NULL DEFAULT '',
86
+ created_at ${intType()} NOT NULL
87
+ )
88
+ `);
89
+ await client.execute(`
90
+ CREATE TABLE IF NOT EXISTS agent_feedback (
91
+ id TEXT PRIMARY KEY,
92
+ run_id TEXT,
93
+ thread_id TEXT,
94
+ message_seq ${intType()},
95
+ feedback_type TEXT NOT NULL,
96
+ value TEXT NOT NULL DEFAULT '',
97
+ user_id TEXT,
98
+ created_at ${intType()} NOT NULL
99
+ )
100
+ `);
101
+ await client.execute(`
102
+ CREATE TABLE IF NOT EXISTS agent_satisfaction_scores (
103
+ id TEXT PRIMARY KEY,
104
+ thread_id TEXT NOT NULL,
105
+ user_id TEXT,
106
+ frustration_score REAL NOT NULL DEFAULT 0,
107
+ rephrasing_score REAL NOT NULL DEFAULT 0,
108
+ abandonment_score REAL NOT NULL DEFAULT 0,
109
+ sentiment_score REAL NOT NULL DEFAULT 0,
110
+ length_trend_score REAL NOT NULL DEFAULT 0,
111
+ computed_at ${intType()} NOT NULL
112
+ )
113
+ `);
114
+ await client.execute(`
115
+ CREATE TABLE IF NOT EXISTS agent_evals (
116
+ id TEXT PRIMARY KEY,
117
+ run_id TEXT NOT NULL,
118
+ thread_id TEXT,
119
+ user_id TEXT,
120
+ eval_type TEXT NOT NULL,
121
+ criteria TEXT NOT NULL,
122
+ score REAL NOT NULL DEFAULT 0,
123
+ reasoning TEXT,
124
+ metadata TEXT,
125
+ created_at ${intType()} NOT NULL
126
+ )
127
+ `);
128
+ await client.execute(`
129
+ CREATE TABLE IF NOT EXISTS agent_eval_datasets (
130
+ id TEXT PRIMARY KEY,
131
+ name TEXT NOT NULL,
132
+ description TEXT NOT NULL DEFAULT '',
133
+ entries TEXT NOT NULL DEFAULT '[]',
134
+ created_at ${intType()} NOT NULL,
135
+ updated_at ${intType()} NOT NULL
136
+ )
137
+ `);
138
+ await client.execute(`
139
+ CREATE TABLE IF NOT EXISTS agent_experiments (
140
+ id TEXT PRIMARY KEY,
141
+ name TEXT NOT NULL,
142
+ status TEXT NOT NULL DEFAULT 'draft',
143
+ variants TEXT NOT NULL DEFAULT '[]',
144
+ metrics TEXT NOT NULL DEFAULT '[]',
145
+ assignment_level TEXT NOT NULL DEFAULT 'user',
146
+ started_at ${intType()},
147
+ ended_at ${intType()},
148
+ created_at ${intType()} NOT NULL
149
+ )
150
+ `);
151
+ await client.execute(`
152
+ CREATE TABLE IF NOT EXISTS agent_experiment_assignments (
153
+ experiment_id TEXT NOT NULL,
154
+ user_id TEXT NOT NULL,
155
+ variant_id TEXT NOT NULL,
156
+ assigned_at ${intType()} NOT NULL,
157
+ PRIMARY KEY (experiment_id, user_id)
158
+ )
159
+ `);
160
+ await client.execute(`
161
+ CREATE TABLE IF NOT EXISTS agent_experiment_results (
162
+ id TEXT PRIMARY KEY,
163
+ experiment_id TEXT NOT NULL,
164
+ variant_id TEXT NOT NULL,
165
+ metric TEXT NOT NULL,
166
+ value REAL NOT NULL DEFAULT 0,
167
+ sample_size ${intType()} NOT NULL DEFAULT 0,
168
+ confidence_low REAL NOT NULL DEFAULT 0,
169
+ confidence_high REAL NOT NULL DEFAULT 0,
170
+ computed_at ${intType()} NOT NULL
171
+ )
172
+ `);
173
+ // Idempotent column upgrades for DBs created before per-user
174
+ // isolation. SQLite has no `ADD COLUMN IF NOT EXISTS`; Postgres
175
+ // surfaces "column ... already exists". `isDuplicateColumnError`
176
+ // (from db/migrations.ts) recognizes both shapes.
177
+ for (const table of USER_SCOPED_TABLES) {
178
+ try {
179
+ await client.execute(`ALTER TABLE ${table} ADD COLUMN user_id TEXT`);
180
+ }
181
+ catch (err) {
182
+ if (isDuplicateColumnError(err))
183
+ continue;
184
+ throw err;
185
+ }
186
+ }
187
+ // Indexes for common query patterns
188
+ const indexes = [
189
+ `CREATE INDEX IF NOT EXISTS idx_trace_spans_run ON agent_trace_spans (run_id)`,
190
+ `CREATE INDEX IF NOT EXISTS idx_trace_spans_thread ON agent_trace_spans (thread_id)`,
191
+ `CREATE INDEX IF NOT EXISTS idx_trace_spans_created ON agent_trace_spans (created_at)`,
192
+ `CREATE INDEX IF NOT EXISTS idx_trace_summaries_created ON agent_trace_summaries (created_at)`,
193
+ `CREATE INDEX IF NOT EXISTS idx_trace_summaries_user ON agent_trace_summaries (user_id, created_at)`,
194
+ `CREATE INDEX IF NOT EXISTS idx_trace_spans_user ON agent_trace_spans (user_id)`,
195
+ `CREATE INDEX IF NOT EXISTS idx_feedback_thread ON agent_feedback (thread_id)`,
196
+ `CREATE INDEX IF NOT EXISTS idx_feedback_created ON agent_feedback (created_at)`,
197
+ `CREATE INDEX IF NOT EXISTS idx_feedback_user ON agent_feedback (user_id, created_at)`,
198
+ `CREATE INDEX IF NOT EXISTS idx_satisfaction_thread ON agent_satisfaction_scores (thread_id)`,
199
+ `CREATE INDEX IF NOT EXISTS idx_satisfaction_user ON agent_satisfaction_scores (user_id, computed_at)`,
200
+ `CREATE INDEX IF NOT EXISTS idx_evals_run ON agent_evals (run_id)`,
201
+ `CREATE INDEX IF NOT EXISTS idx_evals_created ON agent_evals (created_at)`,
202
+ `CREATE INDEX IF NOT EXISTS idx_evals_user ON agent_evals (user_id, created_at)`,
203
+ `CREATE INDEX IF NOT EXISTS idx_experiment_results_exp ON agent_experiment_results (experiment_id)`,
204
+ ];
205
+ for (const sql of indexes) {
206
+ try {
207
+ await client.execute(sql);
208
+ }
209
+ catch {
210
+ // Index might already exist
211
+ }
212
+ }
213
+ })().catch((err) => {
214
+ _initPromise = undefined;
215
+ throw err;
216
+ });
217
+ }
218
+ return _initPromise;
219
+ }
220
+ // ─── Trace span CRUD ─────────────────────────────────────────────────
221
+ export async function insertTraceSpan(span) {
222
+ await ensureObservabilityTables();
223
+ const client = getDbExec();
224
+ await client.execute({
225
+ sql: `INSERT INTO agent_trace_spans
226
+ (id, run_id, thread_id, user_id, parent_span_id, span_type, name,
227
+ input_tokens, output_tokens, cache_read_tokens, cache_write_tokens,
228
+ cost_cents_x100, duration_ms, status, error_message, metadata, created_at)
229
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,
230
+ args: [
231
+ span.id,
232
+ span.runId,
233
+ span.threadId,
234
+ span.userId,
235
+ span.parentSpanId,
236
+ span.spanType,
237
+ span.name,
238
+ span.inputTokens,
239
+ span.outputTokens,
240
+ span.cacheReadTokens,
241
+ span.cacheWriteTokens,
242
+ span.costCentsX100,
243
+ span.durationMs,
244
+ span.status,
245
+ span.errorMessage,
246
+ span.metadata ? JSON.stringify(span.metadata) : null,
247
+ span.createdAt,
248
+ ],
249
+ });
250
+ }
251
+ export async function upsertTraceSummary(summary) {
252
+ await ensureObservabilityTables();
253
+ const client = getDbExec();
254
+ // user_id is intentionally NOT updated on conflict — once a run's
255
+ // owner is recorded it shouldn't change under us.
256
+ if (isPostgres()) {
257
+ await client.execute({
258
+ sql: `INSERT INTO agent_trace_summaries
259
+ (run_id, thread_id, user_id, total_spans, llm_calls, tool_calls,
260
+ successful_tools, failed_tools, total_duration_ms,
261
+ total_cost_cents_x100, total_input_tokens, total_output_tokens,
262
+ model, created_at)
263
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
264
+ ON CONFLICT (run_id) DO UPDATE SET
265
+ total_spans = EXCLUDED.total_spans,
266
+ llm_calls = EXCLUDED.llm_calls,
267
+ tool_calls = EXCLUDED.tool_calls,
268
+ successful_tools = EXCLUDED.successful_tools,
269
+ failed_tools = EXCLUDED.failed_tools,
270
+ total_duration_ms = EXCLUDED.total_duration_ms,
271
+ total_cost_cents_x100 = EXCLUDED.total_cost_cents_x100,
272
+ total_input_tokens = EXCLUDED.total_input_tokens,
273
+ total_output_tokens = EXCLUDED.total_output_tokens,
274
+ model = EXCLUDED.model`,
275
+ args: [
276
+ summary.runId,
277
+ summary.threadId,
278
+ summary.userId,
279
+ summary.totalSpans,
280
+ summary.llmCalls,
281
+ summary.toolCalls,
282
+ summary.successfulTools,
283
+ summary.failedTools,
284
+ summary.totalDurationMs,
285
+ summary.totalCostCentsX100,
286
+ summary.totalInputTokens,
287
+ summary.totalOutputTokens,
288
+ summary.model,
289
+ summary.createdAt,
290
+ ],
291
+ });
292
+ }
293
+ else {
294
+ await client.execute({
295
+ sql: `INSERT INTO agent_trace_summaries
296
+ (run_id, thread_id, user_id, total_spans, llm_calls, tool_calls,
297
+ successful_tools, failed_tools, total_duration_ms,
298
+ total_cost_cents_x100, total_input_tokens, total_output_tokens,
299
+ model, created_at)
300
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
301
+ ON CONFLICT (run_id) DO UPDATE SET
302
+ total_spans = EXCLUDED.total_spans,
303
+ llm_calls = EXCLUDED.llm_calls,
304
+ tool_calls = EXCLUDED.tool_calls,
305
+ successful_tools = EXCLUDED.successful_tools,
306
+ failed_tools = EXCLUDED.failed_tools,
307
+ total_duration_ms = EXCLUDED.total_duration_ms,
308
+ total_cost_cents_x100 = EXCLUDED.total_cost_cents_x100,
309
+ total_input_tokens = EXCLUDED.total_input_tokens,
310
+ total_output_tokens = EXCLUDED.total_output_tokens,
311
+ model = EXCLUDED.model`,
312
+ args: [
313
+ summary.runId,
314
+ summary.threadId,
315
+ summary.userId,
316
+ summary.totalSpans,
317
+ summary.llmCalls,
318
+ summary.toolCalls,
319
+ summary.successfulTools,
320
+ summary.failedTools,
321
+ summary.totalDurationMs,
322
+ summary.totalCostCentsX100,
323
+ summary.totalInputTokens,
324
+ summary.totalOutputTokens,
325
+ summary.model,
326
+ summary.createdAt,
327
+ ],
328
+ });
329
+ }
330
+ }
331
+ export async function getTraceSpansForRun(runId, opts = {}) {
332
+ await ensureObservabilityTables();
333
+ const client = getDbExec();
334
+ const { where, args } = withUserFilter("run_id = ?", [runId], opts.userId);
335
+ const { rows } = await client.execute({
336
+ sql: `SELECT * FROM agent_trace_spans WHERE ${where} ORDER BY created_at ASC`,
337
+ args,
338
+ });
339
+ return rows.map(rowToTraceSpan);
340
+ }
341
+ export async function getTraceSummaries(opts) {
342
+ await ensureObservabilityTables();
343
+ const client = getDbExec();
344
+ const sinceMs = opts.sinceMs ?? 0;
345
+ const limit = opts.limit ?? 100;
346
+ const { where, args } = withUserFilter("created_at >= ?", [sinceMs], opts.userId);
347
+ const { rows } = await client.execute({
348
+ sql: `SELECT * FROM agent_trace_summaries
349
+ WHERE ${where}
350
+ ORDER BY created_at DESC
351
+ LIMIT ?`,
352
+ args: [...args, limit],
353
+ });
354
+ return rows.map(rowToTraceSummary);
355
+ }
356
+ export async function getTraceSummary(runId, opts = {}) {
357
+ await ensureObservabilityTables();
358
+ const client = getDbExec();
359
+ const { where, args } = withUserFilter("run_id = ?", [runId], opts.userId);
360
+ const { rows } = await client.execute({
361
+ sql: `SELECT * FROM agent_trace_summaries WHERE ${where}`,
362
+ args,
363
+ });
364
+ if (rows.length === 0)
365
+ return null;
366
+ return rowToTraceSummary(rows[0]);
367
+ }
368
+ // ─── Feedback CRUD ───────────────────────────────────────────────────
369
+ export async function insertFeedback(entry) {
370
+ await ensureObservabilityTables();
371
+ const client = getDbExec();
372
+ await client.execute({
373
+ sql: `INSERT INTO agent_feedback
374
+ (id, run_id, thread_id, message_seq, feedback_type, value, user_id, created_at)
375
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?)`,
376
+ args: [
377
+ entry.id,
378
+ entry.runId,
379
+ entry.threadId,
380
+ entry.messageSeq,
381
+ entry.feedbackType,
382
+ entry.value,
383
+ entry.userId,
384
+ entry.createdAt,
385
+ ],
386
+ });
387
+ }
388
+ export async function getFeedback(opts) {
389
+ await ensureObservabilityTables();
390
+ const client = getDbExec();
391
+ const conditions = [];
392
+ const args = [];
393
+ if (opts.threadId) {
394
+ conditions.push("thread_id = ?");
395
+ args.push(opts.threadId);
396
+ }
397
+ if (opts.sinceMs) {
398
+ conditions.push("created_at >= ?");
399
+ args.push(opts.sinceMs);
400
+ }
401
+ if (opts.feedbackType) {
402
+ conditions.push("feedback_type = ?");
403
+ args.push(opts.feedbackType);
404
+ }
405
+ if (opts.userId) {
406
+ conditions.push("user_id = ?");
407
+ args.push(opts.userId);
408
+ }
409
+ const where = conditions.length > 0 ? `WHERE ${conditions.join(" AND ")}` : "";
410
+ const limit = opts.limit ?? 100;
411
+ const { rows } = await client.execute({
412
+ sql: `SELECT * FROM agent_feedback ${where} ORDER BY created_at DESC LIMIT ?`,
413
+ args: [...args, limit],
414
+ });
415
+ return rows.map(rowToFeedback);
416
+ }
417
+ export async function getFeedbackStats(sinceMs, opts = {}) {
418
+ await ensureObservabilityTables();
419
+ const client = getDbExec();
420
+ const { where, args } = withUserFilter("created_at >= ?", [sinceMs], opts.userId);
421
+ const { rows } = await client.execute({
422
+ sql: `SELECT feedback_type, value, COUNT(*) as cnt
423
+ FROM agent_feedback WHERE ${where}
424
+ GROUP BY feedback_type, value`,
425
+ args,
426
+ });
427
+ let total = 0;
428
+ let thumbsUp = 0;
429
+ let thumbsDown = 0;
430
+ const categories = {};
431
+ for (const row of rows) {
432
+ const cnt = Number(row.cnt);
433
+ total += cnt;
434
+ if (row.feedback_type === "thumbs_up")
435
+ thumbsUp += cnt;
436
+ else if (row.feedback_type === "thumbs_down")
437
+ thumbsDown += cnt;
438
+ else if (row.feedback_type === "category")
439
+ categories[String(row.value)] = cnt;
440
+ }
441
+ return { total, thumbsUp, thumbsDown, categories };
442
+ }
443
+ // ─── Satisfaction scores CRUD ────────────────────────────────────────
444
+ export async function upsertSatisfactionScore(score) {
445
+ await ensureObservabilityTables();
446
+ const client = getDbExec();
447
+ if (isPostgres()) {
448
+ await client.execute({
449
+ sql: `INSERT INTO agent_satisfaction_scores
450
+ (id, thread_id, user_id, frustration_score, rephrasing_score,
451
+ abandonment_score, sentiment_score, length_trend_score, computed_at)
452
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)
453
+ ON CONFLICT (id) DO UPDATE SET
454
+ frustration_score = EXCLUDED.frustration_score,
455
+ rephrasing_score = EXCLUDED.rephrasing_score,
456
+ abandonment_score = EXCLUDED.abandonment_score,
457
+ sentiment_score = EXCLUDED.sentiment_score,
458
+ length_trend_score = EXCLUDED.length_trend_score,
459
+ computed_at = EXCLUDED.computed_at`,
460
+ args: [
461
+ score.id,
462
+ score.threadId,
463
+ score.userId,
464
+ score.frustrationScore,
465
+ score.rephrasingScore,
466
+ score.abandonmentScore,
467
+ score.sentimentScore,
468
+ score.lengthTrendScore,
469
+ score.computedAt,
470
+ ],
471
+ });
472
+ }
473
+ else {
474
+ await client.execute({
475
+ sql: `INSERT INTO agent_satisfaction_scores
476
+ (id, thread_id, user_id, frustration_score, rephrasing_score,
477
+ abandonment_score, sentiment_score, length_trend_score, computed_at)
478
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)
479
+ ON CONFLICT (id) DO UPDATE SET
480
+ frustration_score = EXCLUDED.frustration_score,
481
+ rephrasing_score = EXCLUDED.rephrasing_score,
482
+ abandonment_score = EXCLUDED.abandonment_score,
483
+ sentiment_score = EXCLUDED.sentiment_score,
484
+ length_trend_score = EXCLUDED.length_trend_score,
485
+ computed_at = EXCLUDED.computed_at`,
486
+ args: [
487
+ score.id,
488
+ score.threadId,
489
+ score.userId,
490
+ score.frustrationScore,
491
+ score.rephrasingScore,
492
+ score.abandonmentScore,
493
+ score.sentimentScore,
494
+ score.lengthTrendScore,
495
+ score.computedAt,
496
+ ],
497
+ });
498
+ }
499
+ }
500
+ export async function getSatisfactionScores(opts) {
501
+ await ensureObservabilityTables();
502
+ const client = getDbExec();
503
+ const conditions = [];
504
+ const args = [];
505
+ if (opts.sinceMs) {
506
+ conditions.push("computed_at >= ?");
507
+ args.push(opts.sinceMs);
508
+ }
509
+ if (opts.minFrustration != null) {
510
+ conditions.push("frustration_score >= ?");
511
+ args.push(opts.minFrustration);
512
+ }
513
+ if (opts.userId) {
514
+ conditions.push("user_id = ?");
515
+ args.push(opts.userId);
516
+ }
517
+ const where = conditions.length > 0 ? `WHERE ${conditions.join(" AND ")}` : "";
518
+ const { rows } = await client.execute({
519
+ sql: `SELECT * FROM agent_satisfaction_scores ${where}
520
+ ORDER BY computed_at DESC LIMIT ?`,
521
+ args: [...args, opts.limit ?? 100],
522
+ });
523
+ return rows.map(rowToSatisfaction);
524
+ }
525
+ // ─── Evals CRUD ──────────────────────────────────────────────────────
526
+ export async function insertEvalResult(result) {
527
+ await ensureObservabilityTables();
528
+ const client = getDbExec();
529
+ await client.execute({
530
+ sql: `INSERT INTO agent_evals
531
+ (id, run_id, thread_id, user_id, eval_type, criteria, score, reasoning, metadata, created_at)
532
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)`,
533
+ args: [
534
+ result.id,
535
+ result.runId,
536
+ result.threadId,
537
+ result.userId,
538
+ result.evalType,
539
+ result.criteria,
540
+ result.score,
541
+ result.reasoning,
542
+ result.metadata ? JSON.stringify(result.metadata) : null,
543
+ result.createdAt,
544
+ ],
545
+ });
546
+ }
547
+ export async function getEvalsForRun(runId, opts = {}) {
548
+ await ensureObservabilityTables();
549
+ const client = getDbExec();
550
+ const { where, args } = withUserFilter("run_id = ?", [runId], opts.userId);
551
+ const { rows } = await client.execute({
552
+ sql: `SELECT * FROM agent_evals WHERE ${where} ORDER BY created_at ASC`,
553
+ args,
554
+ });
555
+ return rows.map(rowToEval);
556
+ }
557
+ export async function getEvalStats(sinceMs, opts = {}) {
558
+ await ensureObservabilityTables();
559
+ const client = getDbExec();
560
+ const { where, args } = withUserFilter("created_at >= ?", [sinceMs], opts.userId);
561
+ const { rows: totalRows } = await client.execute({
562
+ sql: `SELECT COUNT(*) as cnt, AVG(score) as avg_score
563
+ FROM agent_evals WHERE ${where}`,
564
+ args,
565
+ });
566
+ const t = (totalRows[0] ?? {});
567
+ const { rows: criteriaRows } = await client.execute({
568
+ sql: `SELECT criteria, AVG(score) as avg_score, COUNT(*) as cnt
569
+ FROM agent_evals WHERE ${where}
570
+ GROUP BY criteria ORDER BY cnt DESC`,
571
+ args,
572
+ });
573
+ return {
574
+ totalEvals: Number(t.cnt ?? 0),
575
+ avgScore: Number(t.avg_score ?? 0),
576
+ byCriteria: criteriaRows.map((r) => ({
577
+ criteria: String(r.criteria),
578
+ avgScore: Number(r.avg_score ?? 0),
579
+ count: Number(r.cnt ?? 0),
580
+ })),
581
+ };
582
+ }
583
+ // ─── Eval datasets CRUD ──────────────────────────────────────────────
584
+ export async function insertEvalDataset(dataset) {
585
+ await ensureObservabilityTables();
586
+ const client = getDbExec();
587
+ await client.execute({
588
+ sql: `INSERT INTO agent_eval_datasets
589
+ (id, name, description, entries, created_at, updated_at)
590
+ VALUES (?, ?, ?, ?, ?, ?)`,
591
+ args: [
592
+ dataset.id,
593
+ dataset.name,
594
+ dataset.description,
595
+ JSON.stringify(dataset.entries),
596
+ dataset.createdAt,
597
+ dataset.updatedAt,
598
+ ],
599
+ });
600
+ }
601
+ export async function listEvalDatasets() {
602
+ await ensureObservabilityTables();
603
+ const client = getDbExec();
604
+ const { rows } = await client.execute(`SELECT * FROM agent_eval_datasets ORDER BY updated_at DESC`);
605
+ return rows.map(rowToDataset);
606
+ }
607
+ export async function getEvalDataset(id) {
608
+ await ensureObservabilityTables();
609
+ const client = getDbExec();
610
+ const { rows } = await client.execute({
611
+ sql: `SELECT * FROM agent_eval_datasets WHERE id = ?`,
612
+ args: [id],
613
+ });
614
+ if (rows.length === 0)
615
+ return null;
616
+ return rowToDataset(rows[0]);
617
+ }
618
+ export async function updateEvalDataset(id, updates) {
619
+ await ensureObservabilityTables();
620
+ const client = getDbExec();
621
+ const sets = [];
622
+ const args = [];
623
+ if (updates.name !== undefined) {
624
+ sets.push("name = ?");
625
+ args.push(updates.name);
626
+ }
627
+ if (updates.description !== undefined) {
628
+ sets.push("description = ?");
629
+ args.push(updates.description);
630
+ }
631
+ if (updates.entries !== undefined) {
632
+ sets.push("entries = ?");
633
+ args.push(JSON.stringify(updates.entries));
634
+ }
635
+ sets.push("updated_at = ?");
636
+ args.push(Date.now());
637
+ args.push(id);
638
+ await client.execute({
639
+ sql: `UPDATE agent_eval_datasets SET ${sets.join(", ")} WHERE id = ?`,
640
+ args,
641
+ });
642
+ }
643
+ // ─── Experiments CRUD ────────────────────────────────────────────────
644
+ export async function insertExperiment(exp) {
645
+ await ensureObservabilityTables();
646
+ const client = getDbExec();
647
+ await client.execute({
648
+ sql: `INSERT INTO agent_experiments
649
+ (id, name, status, variants, metrics, assignment_level,
650
+ started_at, ended_at, created_at)
651
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)`,
652
+ args: [
653
+ exp.id,
654
+ exp.name,
655
+ exp.status,
656
+ JSON.stringify(exp.variants),
657
+ JSON.stringify(exp.metrics),
658
+ exp.assignmentLevel,
659
+ exp.startedAt,
660
+ exp.endedAt,
661
+ exp.createdAt,
662
+ ],
663
+ });
664
+ }
665
+ export async function updateExperiment(id, updates) {
666
+ await ensureObservabilityTables();
667
+ const client = getDbExec();
668
+ const sets = [];
669
+ const args = [];
670
+ if (updates.name !== undefined) {
671
+ sets.push("name = ?");
672
+ args.push(updates.name);
673
+ }
674
+ if (updates.status !== undefined) {
675
+ sets.push("status = ?");
676
+ args.push(updates.status);
677
+ if (updates.status === "running" && !updates.endedAt) {
678
+ sets.push("started_at = ?");
679
+ args.push(Date.now());
680
+ }
681
+ }
682
+ if (updates.variants !== undefined) {
683
+ sets.push("variants = ?");
684
+ args.push(JSON.stringify(updates.variants));
685
+ }
686
+ if (updates.metrics !== undefined) {
687
+ sets.push("metrics = ?");
688
+ args.push(JSON.stringify(updates.metrics));
689
+ }
690
+ if (updates.endedAt !== undefined) {
691
+ sets.push("ended_at = ?");
692
+ args.push(updates.endedAt);
693
+ }
694
+ if (sets.length === 0)
695
+ return;
696
+ args.push(id);
697
+ await client.execute({
698
+ sql: `UPDATE agent_experiments SET ${sets.join(", ")} WHERE id = ?`,
699
+ args,
700
+ });
701
+ }
702
+ export async function listExperiments() {
703
+ await ensureObservabilityTables();
704
+ const client = getDbExec();
705
+ const { rows } = await client.execute(`SELECT * FROM agent_experiments ORDER BY created_at DESC`);
706
+ return rows.map(rowToExperiment);
707
+ }
708
+ export async function getExperiment(id) {
709
+ await ensureObservabilityTables();
710
+ const client = getDbExec();
711
+ const { rows } = await client.execute({
712
+ sql: `SELECT * FROM agent_experiments WHERE id = ?`,
713
+ args: [id],
714
+ });
715
+ if (rows.length === 0)
716
+ return null;
717
+ return rowToExperiment(rows[0]);
718
+ }
719
+ // ─── Experiment assignments CRUD ────────────────────────────────────
720
+ export async function upsertAssignment(assignment) {
721
+ await ensureObservabilityTables();
722
+ const client = getDbExec();
723
+ if (isPostgres()) {
724
+ await client.execute({
725
+ sql: `INSERT INTO agent_experiment_assignments
726
+ (experiment_id, user_id, variant_id, assigned_at)
727
+ VALUES (?, ?, ?, ?)
728
+ ON CONFLICT (experiment_id, user_id) DO UPDATE SET
729
+ variant_id = EXCLUDED.variant_id,
730
+ assigned_at = EXCLUDED.assigned_at`,
731
+ args: [
732
+ assignment.experimentId,
733
+ assignment.userId,
734
+ assignment.variantId,
735
+ assignment.assignedAt,
736
+ ],
737
+ });
738
+ }
739
+ else {
740
+ await client.execute({
741
+ sql: `INSERT OR REPLACE INTO agent_experiment_assignments
742
+ (experiment_id, user_id, variant_id, assigned_at)
743
+ VALUES (?, ?, ?, ?)`,
744
+ args: [
745
+ assignment.experimentId,
746
+ assignment.userId,
747
+ assignment.variantId,
748
+ assignment.assignedAt,
749
+ ],
750
+ });
751
+ }
752
+ }
753
+ export async function getAssignment(experimentId, userId) {
754
+ await ensureObservabilityTables();
755
+ const client = getDbExec();
756
+ const { rows } = await client.execute({
757
+ sql: `SELECT * FROM agent_experiment_assignments
758
+ WHERE experiment_id = ? AND user_id = ?`,
759
+ args: [experimentId, userId],
760
+ });
761
+ if (rows.length === 0)
762
+ return null;
763
+ const r = rows[0];
764
+ return {
765
+ experimentId: r.experiment_id,
766
+ userId: r.user_id,
767
+ variantId: r.variant_id,
768
+ assignedAt: Number(r.assigned_at),
769
+ };
770
+ }
771
+ // ─── Experiment results CRUD ─────────────────────────────────────────
772
+ export async function insertExperimentResult(result) {
773
+ await ensureObservabilityTables();
774
+ const client = getDbExec();
775
+ await client.execute({
776
+ sql: `INSERT INTO agent_experiment_results
777
+ (id, experiment_id, variant_id, metric, value,
778
+ sample_size, confidence_low, confidence_high, computed_at)
779
+ VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)`,
780
+ args: [
781
+ result.id,
782
+ result.experimentId,
783
+ result.variantId,
784
+ result.metric,
785
+ result.value,
786
+ result.sampleSize,
787
+ result.confidenceLow,
788
+ result.confidenceHigh,
789
+ result.computedAt,
790
+ ],
791
+ });
792
+ }
793
+ export async function getExperimentResults(experimentId) {
794
+ await ensureObservabilityTables();
795
+ const client = getDbExec();
796
+ const { rows } = await client.execute({
797
+ sql: `SELECT * FROM agent_experiment_results
798
+ WHERE experiment_id = ?
799
+ ORDER BY computed_at DESC`,
800
+ args: [experimentId],
801
+ });
802
+ return rows.map(rowToExperimentResult);
803
+ }
804
+ // ─── Aggregate queries for dashboard ─────────────────────────────────
805
+ export async function getObservabilityOverview(sinceMs, opts = {}) {
806
+ await ensureObservabilityTables();
807
+ const client = getDbExec();
808
+ // Three of the four sub-queries time-key on `created_at`; satisfaction
809
+ // uses `computed_at`. Each gets its own `withUserFilter` invocation so
810
+ // the args array isn't aliased across calls (some drivers mutate args
811
+ // for prepared-statement caching).
812
+ const created = withUserFilter("created_at >= ?", [sinceMs], opts.userId);
813
+ const computed = withUserFilter("computed_at >= ?", [sinceMs], opts.userId);
814
+ const [tracesResult, satisfactionResult, feedbackResult, evalsResult] = await Promise.all([
815
+ client.execute({
816
+ sql: `SELECT
817
+ COUNT(*) as total_runs,
818
+ COALESCE(SUM(total_cost_cents_x100), 0) as total_cost,
819
+ COALESCE(AVG(total_duration_ms), 0) as avg_duration,
820
+ COALESCE(SUM(successful_tools), 0) as success_tools,
821
+ COALESCE(SUM(tool_calls), 0) as total_tools
822
+ FROM agent_trace_summaries WHERE ${created.where}`,
823
+ args: created.args,
824
+ }),
825
+ client.execute({
826
+ sql: `SELECT COALESCE(AVG(frustration_score), 0) as avg_frustration
827
+ FROM agent_satisfaction_scores WHERE ${computed.where}`,
828
+ args: computed.args,
829
+ }),
830
+ client.execute({
831
+ sql: `SELECT
832
+ COALESCE(SUM(CASE WHEN feedback_type = 'thumbs_up' THEN 1 ELSE 0 END), 0) as up,
833
+ COALESCE(SUM(CASE WHEN feedback_type IN ('thumbs_up', 'thumbs_down') THEN 1 ELSE 0 END), 0) as total
834
+ FROM agent_feedback WHERE ${created.where}`,
835
+ args: created.args,
836
+ }),
837
+ client.execute({
838
+ sql: `SELECT COALESCE(AVG(score), 0) as avg_score
839
+ FROM agent_evals WHERE ${created.where}`,
840
+ args: created.args,
841
+ }),
842
+ ]);
843
+ const t = (tracesResult.rows[0] ?? {});
844
+ const s = (satisfactionResult.rows[0] ?? {});
845
+ const f = (feedbackResult.rows[0] ?? {});
846
+ const e = (evalsResult.rows[0] ?? {});
847
+ const totalTools = Number(t.total_tools ?? 0);
848
+ const successTools = Number(t.success_tools ?? 0);
849
+ const feedbackTotal = Number(f.total ?? 0);
850
+ const feedbackUp = Number(f.up ?? 0);
851
+ return {
852
+ totalRuns: Number(t.total_runs ?? 0),
853
+ totalCostCents: Number(t.total_cost ?? 0) / 100,
854
+ avgDurationMs: Number(t.avg_duration ?? 0),
855
+ toolSuccessRate: totalTools > 0 ? successTools / totalTools : 1,
856
+ avgFrustrationScore: Number(s.avg_frustration ?? 0),
857
+ thumbsUpRate: feedbackTotal > 0 ? feedbackUp / feedbackTotal : 0,
858
+ avgEvalScore: Number(e.avg_score ?? 0),
859
+ };
860
+ }
861
+ // ─── Row mappers ─────────────────────────────────────────────────────
862
+ function rowToTraceSpan(row) {
863
+ return {
864
+ id: String(row.id),
865
+ runId: String(row.run_id),
866
+ threadId: row.thread_id ? String(row.thread_id) : null,
867
+ userId: row.user_id ? String(row.user_id) : null,
868
+ parentSpanId: row.parent_span_id ? String(row.parent_span_id) : null,
869
+ spanType: row.span_type,
870
+ name: String(row.name),
871
+ inputTokens: Number(row.input_tokens ?? 0),
872
+ outputTokens: Number(row.output_tokens ?? 0),
873
+ cacheReadTokens: Number(row.cache_read_tokens ?? 0),
874
+ cacheWriteTokens: Number(row.cache_write_tokens ?? 0),
875
+ costCentsX100: Number(row.cost_cents_x100 ?? 0),
876
+ durationMs: Number(row.duration_ms ?? 0),
877
+ status: row.status,
878
+ errorMessage: row.error_message ? String(row.error_message) : null,
879
+ metadata: safeJsonParse(row.metadata, null),
880
+ createdAt: Number(row.created_at),
881
+ };
882
+ }
883
+ function rowToTraceSummary(row) {
884
+ return {
885
+ runId: String(row.run_id),
886
+ threadId: row.thread_id ? String(row.thread_id) : null,
887
+ userId: row.user_id ? String(row.user_id) : null,
888
+ totalSpans: Number(row.total_spans ?? 0),
889
+ llmCalls: Number(row.llm_calls ?? 0),
890
+ toolCalls: Number(row.tool_calls ?? 0),
891
+ successfulTools: Number(row.successful_tools ?? 0),
892
+ failedTools: Number(row.failed_tools ?? 0),
893
+ totalDurationMs: Number(row.total_duration_ms ?? 0),
894
+ totalCostCentsX100: Number(row.total_cost_cents_x100 ?? 0),
895
+ totalInputTokens: Number(row.total_input_tokens ?? 0),
896
+ totalOutputTokens: Number(row.total_output_tokens ?? 0),
897
+ model: String(row.model ?? ""),
898
+ createdAt: Number(row.created_at),
899
+ };
900
+ }
901
+ function rowToFeedback(row) {
902
+ return {
903
+ id: String(row.id),
904
+ runId: row.run_id ? String(row.run_id) : null,
905
+ threadId: row.thread_id ? String(row.thread_id) : null,
906
+ messageSeq: row.message_seq != null ? Number(row.message_seq) : null,
907
+ feedbackType: row.feedback_type,
908
+ value: String(row.value ?? ""),
909
+ userId: row.user_id ? String(row.user_id) : null,
910
+ createdAt: Number(row.created_at),
911
+ };
912
+ }
913
+ function rowToSatisfaction(row) {
914
+ return {
915
+ id: String(row.id),
916
+ threadId: String(row.thread_id),
917
+ userId: row.user_id ? String(row.user_id) : null,
918
+ frustrationScore: Number(row.frustration_score ?? 0),
919
+ rephrasingScore: Number(row.rephrasing_score ?? 0),
920
+ abandonmentScore: Number(row.abandonment_score ?? 0),
921
+ sentimentScore: Number(row.sentiment_score ?? 0),
922
+ lengthTrendScore: Number(row.length_trend_score ?? 0),
923
+ computedAt: Number(row.computed_at),
924
+ };
925
+ }
926
+ function rowToEval(row) {
927
+ return {
928
+ id: String(row.id),
929
+ runId: String(row.run_id),
930
+ threadId: row.thread_id ? String(row.thread_id) : null,
931
+ userId: row.user_id ? String(row.user_id) : null,
932
+ evalType: row.eval_type,
933
+ criteria: String(row.criteria),
934
+ score: Number(row.score ?? 0),
935
+ reasoning: row.reasoning ? String(row.reasoning) : null,
936
+ metadata: safeJsonParse(row.metadata, null),
937
+ createdAt: Number(row.created_at),
938
+ };
939
+ }
940
+ function rowToDataset(row) {
941
+ return {
942
+ id: String(row.id),
943
+ name: String(row.name),
944
+ description: String(row.description ?? ""),
945
+ entries: safeJsonParse(row.entries, []),
946
+ createdAt: Number(row.created_at),
947
+ updatedAt: Number(row.updated_at),
948
+ };
949
+ }
950
+ function rowToExperiment(row) {
951
+ return {
952
+ id: String(row.id),
953
+ name: String(row.name),
954
+ status: row.status,
955
+ variants: safeJsonParse(row.variants, []),
956
+ metrics: safeJsonParse(row.metrics, []),
957
+ assignmentLevel: row.assignment_level ?? "user",
958
+ startedAt: row.started_at ? Number(row.started_at) : null,
959
+ endedAt: row.ended_at ? Number(row.ended_at) : null,
960
+ createdAt: Number(row.created_at),
961
+ };
962
+ }
963
+ function rowToExperimentResult(row) {
964
+ return {
965
+ id: String(row.id),
966
+ experimentId: String(row.experiment_id),
967
+ variantId: String(row.variant_id),
968
+ metric: String(row.metric),
969
+ value: Number(row.value ?? 0),
970
+ sampleSize: Number(row.sample_size ?? 0),
971
+ confidenceLow: Number(row.confidence_low ?? 0),
972
+ confidenceHigh: Number(row.confidence_high ?? 0),
973
+ computedAt: Number(row.computed_at),
974
+ };
975
+ }
976
+ //# sourceMappingURL=store.js.map