@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.
- package/README.md +1 -1
- package/dist/agent/engine/ai-sdk-engine.d.ts.map +1 -1
- package/dist/agent/engine/ai-sdk-engine.js +26 -8
- package/dist/agent/engine/ai-sdk-engine.js.map +1 -1
- package/dist/agent/engine/builder-engine.d.ts +19 -0
- package/dist/agent/engine/builder-engine.d.ts.map +1 -0
- package/dist/agent/engine/builder-engine.js +430 -0
- package/dist/agent/engine/builder-engine.js.map +1 -0
- package/dist/agent/engine/builtin.d.ts.map +1 -1
- package/dist/agent/engine/builtin.js +26 -10
- package/dist/agent/engine/builtin.js.map +1 -1
- package/dist/agent/engine/index.d.ts +1 -1
- package/dist/agent/engine/index.d.ts.map +1 -1
- package/dist/agent/engine/index.js +1 -1
- package/dist/agent/engine/index.js.map +1 -1
- package/dist/agent/engine/registry.d.ts +20 -1
- package/dist/agent/engine/registry.d.ts.map +1 -1
- package/dist/agent/engine/registry.js +49 -1
- package/dist/agent/engine/registry.js.map +1 -1
- package/dist/agent/engine/types.d.ts +30 -0
- package/dist/agent/engine/types.d.ts.map +1 -1
- package/dist/agent/engine/types.js +19 -1
- package/dist/agent/engine/types.js.map +1 -1
- package/dist/agent/production-agent.d.ts.map +1 -1
- package/dist/agent/production-agent.js +65 -7
- package/dist/agent/production-agent.js.map +1 -1
- package/dist/agent/run-manager.d.ts.map +1 -1
- package/dist/agent/run-manager.js +11 -1
- package/dist/agent/run-manager.js.map +1 -1
- package/dist/agent/thread-data-builder.d.ts +4 -0
- package/dist/agent/thread-data-builder.d.ts.map +1 -1
- package/dist/agent/thread-data-builder.js +1 -0
- package/dist/agent/thread-data-builder.js.map +1 -1
- package/dist/agent/types.d.ts +8 -0
- package/dist/agent/types.d.ts.map +1 -1
- package/dist/checkpoints/service.d.ts +1 -0
- package/dist/checkpoints/service.d.ts.map +1 -1
- package/dist/checkpoints/service.js +26 -2
- package/dist/checkpoints/service.js.map +1 -1
- package/dist/cli/create.d.ts +30 -0
- package/dist/cli/create.d.ts.map +1 -1
- package/dist/cli/create.js +25 -13
- package/dist/cli/create.js.map +1 -1
- package/dist/client/AgentPanel.js +1 -1
- package/dist/client/AgentPanel.js.map +1 -1
- package/dist/client/AssistantChat.d.ts.map +1 -1
- package/dist/client/AssistantChat.js +49 -10
- package/dist/client/AssistantChat.js.map +1 -1
- package/dist/client/ConnectBuilderCard.d.ts +1 -7
- package/dist/client/ConnectBuilderCard.d.ts.map +1 -1
- package/dist/client/ConnectBuilderCard.js +30 -132
- package/dist/client/ConnectBuilderCard.js.map +1 -1
- package/dist/client/MultiTabAssistantChat.d.ts.map +1 -1
- package/dist/client/MultiTabAssistantChat.js +30 -9
- package/dist/client/MultiTabAssistantChat.js.map +1 -1
- package/dist/client/analytics.d.ts +5 -8
- package/dist/client/analytics.d.ts.map +1 -1
- package/dist/client/analytics.js +53 -11
- package/dist/client/analytics.js.map +1 -1
- package/dist/client/builder-mark.d.ts +9 -0
- package/dist/client/builder-mark.d.ts.map +1 -0
- package/dist/client/builder-mark.js +10 -0
- package/dist/client/builder-mark.js.map +1 -0
- package/dist/client/components/ui/popover.d.ts +8 -0
- package/dist/client/components/ui/popover.d.ts.map +1 -0
- package/dist/client/components/ui/popover.js +11 -0
- package/dist/client/components/ui/popover.js.map +1 -0
- package/dist/client/composer/ComposerPlusMenu.d.ts +2 -0
- package/dist/client/composer/ComposerPlusMenu.d.ts.map +1 -0
- package/dist/client/composer/ComposerPlusMenu.js +244 -0
- package/dist/client/composer/ComposerPlusMenu.js.map +1 -0
- package/dist/client/composer/TiptapComposer.d.ts.map +1 -1
- package/dist/client/composer/TiptapComposer.js +9 -5
- package/dist/client/composer/TiptapComposer.js.map +1 -1
- package/dist/client/composer/useVoiceDictation.d.ts.map +1 -1
- package/dist/client/composer/useVoiceDictation.js +4 -2
- package/dist/client/composer/useVoiceDictation.js.map +1 -1
- package/dist/client/error-format.d.ts +2 -0
- package/dist/client/error-format.d.ts.map +1 -0
- package/dist/client/error-format.js +31 -0
- package/dist/client/error-format.js.map +1 -0
- package/dist/client/index.d.ts +3 -1
- package/dist/client/index.d.ts.map +1 -1
- package/dist/client/index.js +3 -1
- package/dist/client/index.js.map +1 -1
- package/dist/client/observability/ObservabilityDashboard.d.ts +5 -0
- package/dist/client/observability/ObservabilityDashboard.d.ts.map +1 -0
- package/dist/client/observability/ObservabilityDashboard.js +169 -0
- package/dist/client/observability/ObservabilityDashboard.js.map +1 -0
- package/dist/client/observability/ThumbsFeedback.d.ts +8 -0
- package/dist/client/observability/ThumbsFeedback.d.ts.map +1 -0
- package/dist/client/observability/ThumbsFeedback.js +64 -0
- package/dist/client/observability/ThumbsFeedback.js.map +1 -0
- package/dist/client/observability/index.d.ts +4 -0
- package/dist/client/observability/index.d.ts.map +1 -0
- package/dist/client/observability/index.js +4 -0
- package/dist/client/observability/index.js.map +1 -0
- package/dist/client/observability/useObservability.d.ts +128 -0
- package/dist/client/observability/useObservability.d.ts.map +1 -0
- package/dist/client/observability/useObservability.js +109 -0
- package/dist/client/observability/useObservability.js.map +1 -0
- package/dist/client/onboarding/OnboardingPanel.d.ts.map +1 -1
- package/dist/client/onboarding/OnboardingPanel.js +34 -92
- package/dist/client/onboarding/OnboardingPanel.js.map +1 -1
- package/dist/client/org/RequireActiveOrg.d.ts +33 -0
- package/dist/client/org/RequireActiveOrg.d.ts.map +1 -0
- package/dist/client/org/RequireActiveOrg.js +63 -0
- package/dist/client/org/RequireActiveOrg.js.map +1 -0
- package/dist/client/org/hooks.d.ts.map +1 -1
- package/dist/client/org/hooks.js +50 -15
- package/dist/client/org/hooks.js.map +1 -1
- package/dist/client/org/index.d.ts +1 -0
- package/dist/client/org/index.d.ts.map +1 -1
- package/dist/client/org/index.js +1 -0
- package/dist/client/org/index.js.map +1 -1
- package/dist/client/resources/ResourcesPanel.js +3 -3
- package/dist/client/resources/ResourcesPanel.js.map +1 -1
- package/dist/client/settings/AutomationsSection.js +1 -1
- package/dist/client/settings/AutomationsSection.js.map +1 -1
- package/dist/client/settings/BrowserSection.js +1 -1
- package/dist/client/settings/BrowserSection.js.map +1 -1
- package/dist/client/settings/SettingsPanel.d.ts.map +1 -1
- package/dist/client/settings/SettingsPanel.js +112 -12
- package/dist/client/settings/SettingsPanel.js.map +1 -1
- package/dist/client/settings/VoiceTranscriptionSection.d.ts.map +1 -1
- package/dist/client/settings/VoiceTranscriptionSection.js +10 -4
- package/dist/client/settings/VoiceTranscriptionSection.js.map +1 -1
- package/dist/client/settings/useBuilderStatus.d.ts +26 -0
- package/dist/client/settings/useBuilderStatus.d.ts.map +1 -1
- package/dist/client/settings/useBuilderStatus.js +128 -4
- package/dist/client/settings/useBuilderStatus.js.map +1 -1
- package/dist/client/sse-event-processor.d.ts +2 -0
- package/dist/client/sse-event-processor.d.ts.map +1 -1
- package/dist/client/sse-event-processor.js +6 -2
- package/dist/client/sse-event-processor.js.map +1 -1
- package/dist/client/transcription/BuilderTranscriptionCta.d.ts +9 -0
- package/dist/client/transcription/BuilderTranscriptionCta.d.ts.map +1 -0
- package/dist/client/transcription/BuilderTranscriptionCta.js +18 -0
- package/dist/client/transcription/BuilderTranscriptionCta.js.map +1 -0
- package/dist/client/transcription/use-live-transcription.d.ts +29 -0
- package/dist/client/transcription/use-live-transcription.d.ts.map +1 -0
- package/dist/client/transcription/use-live-transcription.js +156 -0
- package/dist/client/transcription/use-live-transcription.js.map +1 -0
- package/dist/client/use-builder-enabled.d.ts +17 -0
- package/dist/client/use-builder-enabled.d.ts.map +1 -0
- package/dist/client/use-builder-enabled.js +36 -0
- package/dist/client/use-builder-enabled.js.map +1 -0
- package/dist/client/use-db-sync.d.ts.map +1 -1
- package/dist/client/use-db-sync.js +4 -2
- package/dist/client/use-db-sync.js.map +1 -1
- package/dist/client/useProductionAgent.d.ts.map +1 -1
- package/dist/client/useProductionAgent.js +3 -1
- package/dist/client/useProductionAgent.js.map +1 -1
- package/dist/db/migrations.d.ts +9 -0
- package/dist/db/migrations.d.ts.map +1 -1
- package/dist/db/migrations.js +75 -10
- package/dist/db/migrations.js.map +1 -1
- package/dist/file-upload/builder.d.ts.map +1 -1
- package/dist/file-upload/builder.js +11 -4
- package/dist/file-upload/builder.js.map +1 -1
- package/dist/jobs/tools.d.ts.map +1 -1
- package/dist/jobs/tools.js +137 -161
- package/dist/jobs/tools.js.map +1 -1
- package/dist/notifications/actions.d.ts +2 -2
- package/dist/notifications/actions.d.ts.map +1 -1
- package/dist/notifications/actions.js +77 -69
- package/dist/notifications/actions.js.map +1 -1
- package/dist/observability/evals.d.ts +22 -0
- package/dist/observability/evals.d.ts.map +1 -0
- package/dist/observability/evals.js +371 -0
- package/dist/observability/evals.js.map +1 -0
- package/dist/observability/experiments.d.ts +24 -0
- package/dist/observability/experiments.d.ts.map +1 -0
- package/dist/observability/experiments.js +274 -0
- package/dist/observability/experiments.js.map +1 -0
- package/dist/observability/feedback.d.ts +14 -0
- package/dist/observability/feedback.d.ts.map +1 -0
- package/dist/observability/feedback.js +256 -0
- package/dist/observability/feedback.js.map +1 -0
- package/dist/observability/index.d.ts +6 -0
- package/dist/observability/index.d.ts.map +1 -0
- package/dist/observability/index.js +5 -0
- package/dist/observability/index.js.map +1 -0
- package/dist/observability/plugin.d.ts +2 -0
- package/dist/observability/plugin.d.ts.map +1 -0
- package/dist/observability/plugin.js +12 -0
- package/dist/observability/plugin.js.map +1 -0
- package/dist/observability/routes.d.ts +68 -0
- package/dist/observability/routes.d.ts.map +1 -0
- package/dist/observability/routes.js +301 -0
- package/dist/observability/routes.js.map +1 -0
- package/dist/observability/store.d.ts +77 -0
- package/dist/observability/store.d.ts.map +1 -0
- package/dist/observability/store.js +976 -0
- package/dist/observability/store.js.map +1 -0
- package/dist/observability/traces.d.ts +37 -0
- package/dist/observability/traces.d.ts.map +1 -0
- package/dist/observability/traces.js +182 -0
- package/dist/observability/traces.js.map +1 -0
- package/dist/observability/types.d.ts +159 -0
- package/dist/observability/types.d.ts.map +1 -0
- package/dist/observability/types.js +16 -0
- package/dist/observability/types.js.map +1 -0
- package/dist/onboarding/default-steps.d.ts.map +1 -1
- package/dist/onboarding/default-steps.js +6 -5
- package/dist/onboarding/default-steps.js.map +1 -1
- package/dist/onboarding/types.d.ts +10 -1
- package/dist/onboarding/types.d.ts.map +1 -1
- package/dist/org/context.d.ts +8 -1
- package/dist/org/context.d.ts.map +1 -1
- package/dist/org/context.js +163 -6
- package/dist/org/context.js.map +1 -1
- package/dist/org/handlers.d.ts.map +1 -1
- package/dist/org/handlers.js +49 -30
- package/dist/org/handlers.js.map +1 -1
- package/dist/progress/actions.d.ts +3 -0
- package/dist/progress/actions.d.ts.map +1 -1
- package/dist/progress/actions.js +86 -110
- package/dist/progress/actions.js.map +1 -1
- package/dist/progress/routes.d.ts +1 -1
- package/dist/progress/routes.js +1 -1
- package/dist/scripts/agent-engines/list-agent-engines.js +1 -1
- package/dist/scripts/agent-engines/list-agent-engines.js.map +1 -1
- package/dist/scripts/agent-engines/manage-agent-engine.d.ts +10 -0
- package/dist/scripts/agent-engines/manage-agent-engine.d.ts.map +1 -0
- package/dist/scripts/agent-engines/manage-agent-engine.js +47 -0
- package/dist/scripts/agent-engines/manage-agent-engine.js.map +1 -0
- package/dist/scripts/agent-engines/set-agent-engine.js +2 -2
- package/dist/scripts/agent-engines/set-agent-engine.js.map +1 -1
- package/dist/server/agent-chat-plugin.d.ts +39 -0
- package/dist/server/agent-chat-plugin.d.ts.map +1 -1
- package/dist/server/agent-chat-plugin.js +707 -443
- package/dist/server/agent-chat-plugin.js.map +1 -1
- package/dist/server/agent-teams.js +1 -1
- package/dist/server/agent-teams.js.map +1 -1
- package/dist/server/analytics.d.ts +5 -6
- package/dist/server/analytics.d.ts.map +1 -1
- package/dist/server/analytics.js +6 -14
- package/dist/server/analytics.js.map +1 -1
- package/dist/server/app-name.d.ts +5 -2
- package/dist/server/app-name.d.ts.map +1 -1
- package/dist/server/app-name.js +14 -3
- package/dist/server/app-name.js.map +1 -1
- package/dist/server/app-url.d.ts.map +1 -1
- package/dist/server/app-url.js +10 -1
- package/dist/server/app-url.js.map +1 -1
- package/dist/server/auth.d.ts +2 -0
- package/dist/server/auth.d.ts.map +1 -1
- package/dist/server/auth.js +153 -2
- package/dist/server/auth.js.map +1 -1
- package/dist/server/better-auth-instance.d.ts +2 -0
- package/dist/server/better-auth-instance.d.ts.map +1 -1
- package/dist/server/better-auth-instance.js +4 -0
- package/dist/server/better-auth-instance.js.map +1 -1
- package/dist/server/builder-browser.d.ts +59 -1
- package/dist/server/builder-browser.d.ts.map +1 -1
- package/dist/server/builder-browser.js +127 -11
- package/dist/server/builder-browser.js.map +1 -1
- package/dist/server/core-routes-plugin.d.ts.map +1 -1
- package/dist/server/core-routes-plugin.js +208 -6
- package/dist/server/core-routes-plugin.js.map +1 -1
- package/dist/server/credential-provider.d.ts +7 -0
- package/dist/server/credential-provider.d.ts.map +1 -1
- package/dist/server/credential-provider.js +10 -0
- package/dist/server/credential-provider.js.map +1 -1
- package/dist/server/onboarding-html.d.ts.map +1 -1
- package/dist/server/onboarding-html.js +29 -4
- package/dist/server/onboarding-html.js.map +1 -1
- package/dist/server/poll.d.ts.map +1 -1
- package/dist/server/poll.js +46 -5
- package/dist/server/poll.js.map +1 -1
- package/dist/server/ssr-handler.d.ts.map +1 -1
- package/dist/server/ssr-handler.js +2 -1
- package/dist/server/ssr-handler.js.map +1 -1
- package/dist/server/transcribe-voice.d.ts.map +1 -1
- package/dist/server/transcribe-voice.js +44 -5
- package/dist/server/transcribe-voice.js.map +1 -1
- package/dist/styles/agent-native.css +11 -2
- package/dist/templates/default/.agents/skills/progress/SKILL.md +14 -12
- package/dist/templates/default/app/root.tsx +7 -0
- package/dist/transcription/builder-transcription.d.ts +27 -0
- package/dist/transcription/builder-transcription.d.ts.map +1 -0
- package/dist/transcription/builder-transcription.js +41 -0
- package/dist/transcription/builder-transcription.js.map +1 -0
- package/dist/triggers/actions.d.ts +3 -0
- package/dist/triggers/actions.d.ts.map +1 -1
- package/dist/triggers/actions.js +189 -213
- package/dist/triggers/actions.js.map +1 -1
- package/docs/content/agent-mentions.md +1 -1
- package/docs/content/automations.md +22 -19
- package/docs/content/cloneable-saas.md +2 -2
- package/docs/content/deployment.md +21 -61
- package/docs/content/getting-started.md +1 -1
- package/docs/content/key-concepts.md +1 -1
- package/docs/content/{enterprise-workspace.md → multi-app-workspace.md} +3 -3
- package/docs/content/multi-tenancy.md +1 -1
- package/docs/content/progress.md +11 -11
- package/docs/content/template-dispatch.md +3 -3
- package/docs/content/workspace-management.md +1 -1
- package/package.json +9 -2
- package/src/templates/default/.agents/skills/progress/SKILL.md +14 -12
- 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
|