@amodalai/runtime 0.2.10 → 0.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (139) hide show
  1. package/dist/src/__fixtures__/e2e.test.js +2 -2
  2. package/dist/src/__fixtures__/e2e.test.js.map +1 -1
  3. package/dist/src/__fixtures__/smoke.test.js +0 -88
  4. package/dist/src/__fixtures__/smoke.test.js.map +1 -1
  5. package/dist/src/__tests__/studio-integration.test.js +298 -0
  6. package/dist/src/__tests__/studio-integration.test.js.map +1 -0
  7. package/dist/src/agent/agent-types.d.ts +4 -0
  8. package/dist/src/agent/feedback-store.d.ts +11 -10
  9. package/dist/src/agent/feedback-store.js +147 -75
  10. package/dist/src/agent/feedback-store.js.map +1 -1
  11. package/dist/src/agent/local-server.js +30 -111
  12. package/dist/src/agent/local-server.js.map +1 -1
  13. package/dist/src/agent/local-server.test.js +17 -1
  14. package/dist/src/agent/local-server.test.js.map +1 -1
  15. package/dist/src/agent/routes/context.d.ts +24 -0
  16. package/dist/src/agent/routes/context.js +30 -0
  17. package/dist/src/agent/routes/context.js.map +1 -0
  18. package/dist/src/agent/routes/feedback.js +28 -56
  19. package/dist/src/agent/routes/feedback.js.map +1 -1
  20. package/dist/src/api/create-agent.js +8 -4
  21. package/dist/src/api/create-agent.js.map +1 -1
  22. package/dist/src/api/types.d.ts +1 -1
  23. package/dist/src/channels/channel-session-mapper.js +1 -1
  24. package/dist/src/channels/channel-session-mapper.js.map +1 -1
  25. package/dist/src/config.d.ts +2 -2
  26. package/dist/src/config.js +2 -1
  27. package/dist/src/config.js.map +1 -1
  28. package/dist/src/config.test.js +1 -1
  29. package/dist/src/config.test.js.map +1 -1
  30. package/dist/src/errors.d.ts +2 -2
  31. package/dist/src/errors.js +2 -2
  32. package/dist/src/index.d.ts +0 -3
  33. package/dist/src/index.js +0 -3
  34. package/dist/src/index.js.map +1 -1
  35. package/dist/src/session/drizzle-session-store.d.ts +4 -6
  36. package/dist/src/session/drizzle-session-store.js +15 -5
  37. package/dist/src/session/drizzle-session-store.js.map +1 -1
  38. package/dist/src/session/manager.js +1 -1
  39. package/dist/src/session/manager.test.js +7 -5
  40. package/dist/src/session/manager.test.js.map +1 -1
  41. package/dist/src/session/postgres-session-store.d.ts +3 -24
  42. package/dist/src/session/postgres-session-store.js +9 -128
  43. package/dist/src/session/postgres-session-store.js.map +1 -1
  44. package/dist/src/session/session-builder.d.ts +0 -4
  45. package/dist/src/session/session-builder.js +2 -9
  46. package/dist/src/session/session-builder.js.map +1 -1
  47. package/dist/src/session/session-builder.test.js +0 -25
  48. package/dist/src/session/session-builder.test.js.map +1 -1
  49. package/dist/src/session/session-store-selector.d.ts +11 -26
  50. package/dist/src/session/session-store-selector.js +3 -48
  51. package/dist/src/session/session-store-selector.js.map +1 -1
  52. package/dist/src/session/session-store-selector.test.js +5 -57
  53. package/dist/src/session/session-store-selector.test.js.map +1 -1
  54. package/dist/src/session/store.d.ts +8 -14
  55. package/dist/src/session/store.js +8 -10
  56. package/dist/src/session/store.js.map +1 -1
  57. package/dist/src/session/store.test.js +6 -126
  58. package/dist/src/session/store.test.js.map +1 -1
  59. package/dist/src/session/tool-context-factory.js +1 -1
  60. package/dist/src/session/tool-context-factory.js.map +1 -1
  61. package/dist/src/stores/drizzle-store-backend.d.ts +5 -0
  62. package/dist/src/stores/drizzle-store-backend.js +23 -3
  63. package/dist/src/stores/drizzle-store-backend.js.map +1 -1
  64. package/dist/src/stores/drizzle-store-backend.test.js +10 -58
  65. package/dist/src/stores/drizzle-store-backend.test.js.map +1 -1
  66. package/dist/src/stores/index.d.ts +0 -2
  67. package/dist/src/stores/index.js +0 -1
  68. package/dist/src/stores/index.js.map +1 -1
  69. package/dist/src/stores/postgres-store-backend.d.ts +5 -15
  70. package/dist/src/stores/postgres-store-backend.js +14 -72
  71. package/dist/src/stores/postgres-store-backend.js.map +1 -1
  72. package/dist/tsconfig.tsbuildinfo +1 -1
  73. package/package.json +4 -6
  74. package/dist/src/agent/automation-bridge.d.ts +0 -33
  75. package/dist/src/agent/automation-bridge.js +0 -50
  76. package/dist/src/agent/automation-bridge.js.map +0 -1
  77. package/dist/src/agent/automation-bridge.test.d.ts +0 -6
  78. package/dist/src/agent/automation-bridge.test.js +0 -130
  79. package/dist/src/agent/automation-bridge.test.js.map +0 -1
  80. package/dist/src/agent/eval-store.d.ts +0 -50
  81. package/dist/src/agent/eval-store.js +0 -137
  82. package/dist/src/agent/eval-store.js.map +0 -1
  83. package/dist/src/agent/proactive/delivery-router.d.ts +0 -68
  84. package/dist/src/agent/proactive/delivery-router.js +0 -337
  85. package/dist/src/agent/proactive/delivery-router.js.map +0 -1
  86. package/dist/src/agent/proactive/delivery-router.test.js +0 -455
  87. package/dist/src/agent/proactive/delivery-router.test.js.map +0 -1
  88. package/dist/src/agent/proactive/delivery.d.ts +0 -21
  89. package/dist/src/agent/proactive/delivery.js +0 -68
  90. package/dist/src/agent/proactive/delivery.js.map +0 -1
  91. package/dist/src/agent/proactive/delivery.test.d.ts +0 -6
  92. package/dist/src/agent/proactive/delivery.test.js +0 -65
  93. package/dist/src/agent/proactive/delivery.test.js.map +0 -1
  94. package/dist/src/agent/proactive/proactive-runner.d.ts +0 -129
  95. package/dist/src/agent/proactive/proactive-runner.js +0 -301
  96. package/dist/src/agent/proactive/proactive-runner.js.map +0 -1
  97. package/dist/src/agent/proactive/proactive-runner.test.d.ts +0 -6
  98. package/dist/src/agent/proactive/proactive-runner.test.js +0 -250
  99. package/dist/src/agent/proactive/proactive-runner.test.js.map +0 -1
  100. package/dist/src/agent/routes/admin-chat-abort.test.d.ts +0 -6
  101. package/dist/src/agent/routes/admin-chat-abort.test.js +0 -207
  102. package/dist/src/agent/routes/admin-chat-abort.test.js.map +0 -1
  103. package/dist/src/agent/routes/admin-chat.d.ts +0 -28
  104. package/dist/src/agent/routes/admin-chat.js +0 -110
  105. package/dist/src/agent/routes/admin-chat.js.map +0 -1
  106. package/dist/src/agent/routes/automations.d.ts +0 -19
  107. package/dist/src/agent/routes/automations.js +0 -86
  108. package/dist/src/agent/routes/automations.js.map +0 -1
  109. package/dist/src/agent/routes/automations.test.d.ts +0 -6
  110. package/dist/src/agent/routes/automations.test.js +0 -117
  111. package/dist/src/agent/routes/automations.test.js.map +0 -1
  112. package/dist/src/agent/routes/evals.d.ts +0 -17
  113. package/dist/src/agent/routes/evals.js +0 -389
  114. package/dist/src/agent/routes/evals.js.map +0 -1
  115. package/dist/src/agent/routes/webhooks.d.ts +0 -17
  116. package/dist/src/agent/routes/webhooks.js +0 -63
  117. package/dist/src/agent/routes/webhooks.js.map +0 -1
  118. package/dist/src/agent/routes/webhooks.test.d.ts +0 -6
  119. package/dist/src/agent/routes/webhooks.test.js +0 -100
  120. package/dist/src/agent/routes/webhooks.test.js.map +0 -1
  121. package/dist/src/session/pglite-session-store.d.ts +0 -23
  122. package/dist/src/session/pglite-session-store.js +0 -92
  123. package/dist/src/session/pglite-session-store.js.map +0 -1
  124. package/dist/src/stores/pglite-store-backend.d.ts +0 -39
  125. package/dist/src/stores/pglite-store-backend.js +0 -128
  126. package/dist/src/stores/pglite-store-backend.js.map +0 -1
  127. package/dist/src/stores/pglite-store-backend.test.d.ts +0 -6
  128. package/dist/src/stores/pglite-store-backend.test.js +0 -150
  129. package/dist/src/stores/pglite-store-backend.test.js.map +0 -1
  130. package/dist/src/stores/schema.d.ts +0 -593
  131. package/dist/src/stores/schema.js +0 -75
  132. package/dist/src/stores/schema.js.map +0 -1
  133. package/dist/src/tools/admin-file-tools.d.ts +0 -42
  134. package/dist/src/tools/admin-file-tools.js +0 -714
  135. package/dist/src/tools/admin-file-tools.js.map +0 -1
  136. package/dist/src/tools/admin-file-tools.test.d.ts +0 -6
  137. package/dist/src/tools/admin-file-tools.test.js +0 -523
  138. package/dist/src/tools/admin-file-tools.test.js.map +0 -1
  139. /package/dist/src/{agent/proactive/delivery-router.test.d.ts → __tests__/studio-integration.test.d.ts} +0 -0
@@ -0,0 +1,298 @@
1
+ /**
2
+ * @license
3
+ * Copyright 2026 Amodal Labs, Inc.
4
+ * SPDX-License-Identifier: MIT
5
+ */
6
+ /**
7
+ * Cross-service smoke tests for the studio-standalone architecture.
8
+ *
9
+ * These tests verify:
10
+ * 1. Runtime isolation — studio/admin routes are NOT served by the runtime
11
+ * 2. /api/context returns service URLs based on config
12
+ * 3. Context router works correctly in isolation
13
+ */
14
+ import { describe, it, expect, vi, beforeEach, afterAll } from 'vitest';
15
+ import { mkdtempSync, rmSync } from 'node:fs';
16
+ import { join } from 'node:path';
17
+ import { tmpdir } from 'node:os';
18
+ // ---------------------------------------------------------------------------
19
+ // Shared temp dir for tests that need a repo path
20
+ // ---------------------------------------------------------------------------
21
+ const TEST_REPO = mkdtempSync(join(tmpdir(), 'amodal-studio-integ-'));
22
+ afterAll(() => {
23
+ rmSync(TEST_REPO, { recursive: true, force: true });
24
+ });
25
+ // ---------------------------------------------------------------------------
26
+ // Mocks — same pattern as local-server.test.ts
27
+ // ---------------------------------------------------------------------------
28
+ const { mockLoadRepo, mockSetupSession, mockPrepareExploreConfig, mockPlanModeManager } = vi.hoisted(() => ({
29
+ mockLoadRepo: vi.fn(),
30
+ mockSetupSession: vi.fn(),
31
+ mockPrepareExploreConfig: vi.fn(),
32
+ mockPlanModeManager: vi.fn(),
33
+ }));
34
+ // Mock @amodalai/db so tests don't need a real Postgres connection
35
+ vi.mock('@amodalai/db', () => ({
36
+ getDb: vi.fn(() => ({})),
37
+ ensureSchema: vi.fn(async () => { }),
38
+ closeDb: vi.fn(async () => { }),
39
+ agentSessions: {},
40
+ channelSessions: {},
41
+ storeDocuments: {},
42
+ storeDocumentVersions: {},
43
+ feedback: {},
44
+ evalRuns: {},
45
+ studioDrafts: {},
46
+ notifyStoreUpdated: vi.fn(async () => { }),
47
+ notifySessionUpdated: vi.fn(async () => { }),
48
+ notifyFeedbackCreated: vi.fn(async () => { }),
49
+ }));
50
+ vi.mock('@amodalai/core', async (importOriginal) => {
51
+ const actual = await importOriginal();
52
+ return {
53
+ ...actual,
54
+ loadRepo: mockLoadRepo,
55
+ setupSession: mockSetupSession,
56
+ prepareExploreConfig: mockPrepareExploreConfig,
57
+ PlanModeManager: mockPlanModeManager,
58
+ buildConnectionsMap: vi.fn(() => ({})),
59
+ buildDefaultPrompt: vi.fn(() => 'You are test agent.'),
60
+ generateFieldGuidance: vi.fn(() => ''),
61
+ generateAlternativeLookupGuidance: vi.fn(() => ''),
62
+ getModelContextWindow: vi.fn(() => 200_000),
63
+ };
64
+ });
65
+ const MOCK_REPO = {
66
+ source: 'local',
67
+ origin: TEST_REPO,
68
+ config: {
69
+ name: 'test-studio',
70
+ version: '1.0.0',
71
+ models: { main: { provider: 'anthropic', model: 'claude-sonnet-4-20250514' } },
72
+ },
73
+ connections: new Map(),
74
+ skills: [],
75
+ agents: {},
76
+ automations: [],
77
+ knowledge: [],
78
+ evals: [],
79
+ tools: [],
80
+ stores: [],
81
+ };
82
+ function applyMockImplementations() {
83
+ mockLoadRepo.mockResolvedValue(MOCK_REPO);
84
+ mockSetupSession.mockReturnValue({
85
+ repo: { connections: new Map(), skills: [], automations: [], knowledge: [], evals: [], tools: [], stores: [] },
86
+ scrubTracker: {},
87
+ fieldScrubber: {},
88
+ outputGuard: {},
89
+ actionGate: { evaluate: vi.fn() },
90
+ contextCompiler: {},
91
+ compiledContext: { systemPrompt: 'test', tokenUsage: { total: 100000, used: 500, remaining: 99500, sectionBreakdown: {} }, sections: [] },
92
+ exploreContext: { systemPrompt: 'explore', tokenUsage: { total: 100000, used: 300, remaining: 99700, sectionBreakdown: {} }, sections: [] },
93
+ outputPipeline: { process: vi.fn(), createStreamProcessor: vi.fn() },
94
+ telemetry: { logScrub: vi.fn(), logGuard: vi.fn(), logGate: vi.fn() },
95
+ connectionsMap: {},
96
+ sessionId: 'test',
97
+ isDelegated: false,
98
+ });
99
+ mockPrepareExploreConfig.mockReturnValue({
100
+ systemPrompt: 'explore',
101
+ model: { provider: 'anthropic', model: 'test' },
102
+ connectionsMap: {},
103
+ readOnly: true,
104
+ maxTurns: 10,
105
+ maxDepth: 2,
106
+ });
107
+ mockPlanModeManager.mockImplementation(() => ({
108
+ isActive: vi.fn(() => false),
109
+ enter: vi.fn(),
110
+ exit: vi.fn(),
111
+ getPlanningReminder: vi.fn(() => null),
112
+ getApprovedPlanContext: vi.fn(() => null),
113
+ }));
114
+ }
115
+ // ---------------------------------------------------------------------------
116
+ // 1. Runtime isolation tests
117
+ // ---------------------------------------------------------------------------
118
+ describe('runtime isolation — studio/admin routes are absent', () => {
119
+ beforeEach(() => {
120
+ vi.clearAllMocks();
121
+ applyMockImplementations();
122
+ });
123
+ it('GET /api/studio/drafts returns 404 (runtime does not serve studio routes)', async () => {
124
+ const { createLocalServer } = await import('../agent/local-server.js');
125
+ const server = await createLocalServer({ repoPath: TEST_REPO, port: 0 });
126
+ const { default: request } = await import('supertest');
127
+ const res = await request(server.app).get('/api/studio/drafts');
128
+ expect(res.status).toBe(404);
129
+ await server.stop();
130
+ });
131
+ it('POST /admin-chat returns 404 (admin chat endpoint is gone)', async () => {
132
+ const { createLocalServer } = await import('../agent/local-server.js');
133
+ const server = await createLocalServer({ repoPath: TEST_REPO, port: 0 });
134
+ const { default: request } = await import('supertest');
135
+ const res = await request(server.app)
136
+ .post('/admin-chat')
137
+ .send({ message: 'hello' });
138
+ expect(res.status).toBe(404);
139
+ await server.stop();
140
+ });
141
+ it('POST /api/studio/publish returns 404', async () => {
142
+ const { createLocalServer } = await import('../agent/local-server.js');
143
+ const server = await createLocalServer({ repoPath: TEST_REPO, port: 0 });
144
+ const { default: request } = await import('supertest');
145
+ const res = await request(server.app)
146
+ .post('/api/studio/publish')
147
+ .send({ commitMessage: 'test' });
148
+ expect(res.status).toBe(404);
149
+ await server.stop();
150
+ });
151
+ it('GET /api/studio/workspace returns 404', async () => {
152
+ const { createLocalServer } = await import('../agent/local-server.js');
153
+ const server = await createLocalServer({ repoPath: TEST_REPO, port: 0 });
154
+ const { default: request } = await import('supertest');
155
+ const res = await request(server.app).get('/api/studio/workspace');
156
+ expect(res.status).toBe(404);
157
+ await server.stop();
158
+ });
159
+ it('POST /api/studio/discard returns 404', async () => {
160
+ const { createLocalServer } = await import('../agent/local-server.js');
161
+ const server = await createLocalServer({ repoPath: TEST_REPO, port: 0 });
162
+ const { default: request } = await import('supertest');
163
+ const res = await request(server.app).post('/api/studio/discard');
164
+ expect(res.status).toBe(404);
165
+ await server.stop();
166
+ });
167
+ it('POST /api/studio/preview returns 404', async () => {
168
+ const { createLocalServer } = await import('../agent/local-server.js');
169
+ const server = await createLocalServer({ repoPath: TEST_REPO, port: 0 });
170
+ const { default: request } = await import('supertest');
171
+ const res = await request(server.app).post('/api/studio/preview');
172
+ expect(res.status).toBe(404);
173
+ await server.stop();
174
+ });
175
+ it('runtime still serves its own routes (health check)', async () => {
176
+ const { createLocalServer } = await import('../agent/local-server.js');
177
+ const server = await createLocalServer({ repoPath: TEST_REPO, port: 0 });
178
+ const { default: request } = await import('supertest');
179
+ const res = await request(server.app).get('/health');
180
+ expect(res.status).toBe(200);
181
+ expect(res.body).toMatchObject({ status: 'ok', mode: 'repo' });
182
+ await server.stop();
183
+ });
184
+ });
185
+ // ---------------------------------------------------------------------------
186
+ // 2. /api/context endpoint tests
187
+ // ---------------------------------------------------------------------------
188
+ describe('/api/context returns service URLs from config', () => {
189
+ beforeEach(() => {
190
+ vi.clearAllMocks();
191
+ applyMockImplementations();
192
+ });
193
+ it('returns studioUrl and adminAgentUrl when configured', async () => {
194
+ const { createLocalServer } = await import('../agent/local-server.js');
195
+ const server = await createLocalServer({
196
+ repoPath: TEST_REPO,
197
+ port: 0,
198
+ studioUrl: 'http://localhost:3848',
199
+ adminAgentUrl: 'http://localhost:3849',
200
+ });
201
+ const { default: request } = await import('supertest');
202
+ const res = await request(server.app).get('/api/context');
203
+ expect(res.status).toBe(200);
204
+ expect(res.body).toEqual({
205
+ studioUrl: 'http://localhost:3848',
206
+ adminAgentUrl: 'http://localhost:3849',
207
+ });
208
+ await server.stop();
209
+ });
210
+ it('returns null when service URLs are not configured', async () => {
211
+ const { createLocalServer } = await import('../agent/local-server.js');
212
+ const originalStudio = process.env['STUDIO_URL'];
213
+ const originalAdmin = process.env['ADMIN_AGENT_URL'];
214
+ delete process.env['STUDIO_URL'];
215
+ delete process.env['ADMIN_AGENT_URL'];
216
+ try {
217
+ const server = await createLocalServer({
218
+ repoPath: TEST_REPO,
219
+ port: 0,
220
+ });
221
+ const { default: request } = await import('supertest');
222
+ const res = await request(server.app).get('/api/context');
223
+ expect(res.status).toBe(200);
224
+ expect(res.body).toEqual({
225
+ studioUrl: null,
226
+ adminAgentUrl: null,
227
+ });
228
+ await server.stop();
229
+ }
230
+ finally {
231
+ if (originalStudio !== undefined)
232
+ process.env['STUDIO_URL'] = originalStudio;
233
+ if (originalAdmin !== undefined)
234
+ process.env['ADMIN_AGENT_URL'] = originalAdmin;
235
+ }
236
+ });
237
+ it('falls back to STUDIO_URL env var when config option is not set', async () => {
238
+ const { createLocalServer } = await import('../agent/local-server.js');
239
+ process.env['STUDIO_URL'] = 'http://env-studio:3848';
240
+ process.env['ADMIN_AGENT_URL'] = 'http://env-admin:3849';
241
+ try {
242
+ const server = await createLocalServer({
243
+ repoPath: TEST_REPO,
244
+ port: 0,
245
+ });
246
+ const { default: request } = await import('supertest');
247
+ const res = await request(server.app).get('/api/context');
248
+ expect(res.status).toBe(200);
249
+ expect(res.body).toEqual({
250
+ studioUrl: 'http://env-studio:3848',
251
+ adminAgentUrl: 'http://env-admin:3849',
252
+ });
253
+ await server.stop();
254
+ }
255
+ finally {
256
+ delete process.env['STUDIO_URL'];
257
+ delete process.env['ADMIN_AGENT_URL'];
258
+ }
259
+ });
260
+ });
261
+ // ---------------------------------------------------------------------------
262
+ // 3. Context router unit tests (isolated, no local-server overhead)
263
+ // ---------------------------------------------------------------------------
264
+ describe('createContextRouter — unit', () => {
265
+ it('serves the configured URLs at /api/context', async () => {
266
+ const express = await import('express');
267
+ const { createContextRouter } = await import('../agent/routes/context.js');
268
+ const { default: request } = await import('supertest');
269
+ const app = express.default();
270
+ app.use(createContextRouter({
271
+ studioUrl: 'http://studio:3848',
272
+ adminAgentUrl: 'http://admin:3849',
273
+ }));
274
+ const res = await request(app).get('/api/context');
275
+ expect(res.status).toBe(200);
276
+ expect(res.body).toEqual({
277
+ studioUrl: 'http://studio:3848',
278
+ adminAgentUrl: 'http://admin:3849',
279
+ });
280
+ });
281
+ it('returns nulls when no URLs configured', async () => {
282
+ const express = await import('express');
283
+ const { createContextRouter } = await import('../agent/routes/context.js');
284
+ const { default: request } = await import('supertest');
285
+ const app = express.default();
286
+ app.use(createContextRouter({
287
+ studioUrl: null,
288
+ adminAgentUrl: null,
289
+ }));
290
+ const res = await request(app).get('/api/context');
291
+ expect(res.status).toBe(200);
292
+ expect(res.body).toEqual({
293
+ studioUrl: null,
294
+ adminAgentUrl: null,
295
+ });
296
+ });
297
+ });
298
+ //# sourceMappingURL=studio-integration.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"studio-integration.test.js","sourceRoot":"","sources":["../../../src/__tests__/studio-integration.test.ts"],"names":[],"mappings":"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"}
@@ -64,6 +64,10 @@ export interface LocalServerConfig {
64
64
  * Self-hosted deployments can plug in their own auth.
65
65
  */
66
66
  roleProvider?: import('../role-provider.js').RoleProvider;
67
+ /** URL of the Studio service. Defaults to `process.env.STUDIO_URL` or null. */
68
+ studioUrl?: string;
69
+ /** URL of the admin agent service. Defaults to `process.env.ADMIN_AGENT_URL` or null. */
70
+ adminAgentUrl?: string;
67
71
  }
68
72
  export interface AgentSession {
69
73
  id: string;
@@ -23,17 +23,18 @@ export interface FeedbackSummary {
23
23
  recentDown: FeedbackEntry[];
24
24
  }
25
25
  /**
26
- * Persists user feedback (thumbs up/down) to disk.
27
- * Files stored in .amodal/feedback/ under the repo root.
26
+ * Persists user feedback (thumbs up/down) to Postgres.
28
27
  */
29
28
  export declare class FeedbackStore {
30
- private readonly dir;
31
- constructor(repoPath: string);
32
- private ensureDir;
33
- private resolvePath;
34
- save(entry: Omit<FeedbackEntry, 'id' | 'timestamp'>): FeedbackEntry;
35
- list(limit?: number): FeedbackEntry[];
36
- summary(): FeedbackSummary;
29
+ private readonly agentId;
30
+ private readonly db;
31
+ constructor({ agentId }: {
32
+ agentId: string;
33
+ });
34
+ save(entry: Omit<FeedbackEntry, 'id' | 'timestamp'>): Promise<FeedbackEntry>;
35
+ list(limit?: number): Promise<FeedbackEntry[]>;
36
+ summary(): Promise<FeedbackSummary>;
37
37
  /** Mark feedback entries as reviewed so they're excluded from the next synthesis. */
38
- markReviewed(ids: string[]): void;
38
+ markReviewed(ids: string[]): Promise<void>;
39
+ private rowToEntry;
39
40
  }
@@ -3,96 +3,168 @@
3
3
  * Copyright 2026 Amodal Labs, Inc.
4
4
  * SPDX-License-Identifier: MIT
5
5
  */
6
- import { existsSync, mkdirSync, readFileSync, readdirSync, writeFileSync } from 'node:fs';
7
- import { join, resolve } from 'node:path';
6
+ /**
7
+ * Postgres-backed feedback store.
8
+ *
9
+ * Persists user feedback (thumbs up/down) to the shared Postgres
10
+ * database via Drizzle ORM. Replaces the previous file-based JSON
11
+ * implementation.
12
+ */
13
+ import { eq, desc, inArray, count, sql } from 'drizzle-orm';
8
14
  import { randomUUID } from 'node:crypto';
15
+ import { getDb, feedback, notifyFeedbackCreated } from '@amodalai/db';
16
+ import { StoreError } from '../errors.js';
17
+ import { log } from '../logger.js';
9
18
  /**
10
- * Persists user feedback (thumbs up/down) to disk.
11
- * Files stored in .amodal/feedback/ under the repo root.
19
+ * Persists user feedback (thumbs up/down) to Postgres.
12
20
  */
13
21
  export class FeedbackStore {
14
- dir;
15
- constructor(repoPath) {
16
- this.dir = join(repoPath, '.amodal', 'feedback');
17
- }
18
- ensureDir() {
19
- if (!existsSync(this.dir)) {
20
- mkdirSync(this.dir, { recursive: true });
21
- }
22
- }
23
- resolvePath(id) {
24
- const trimmed = id.trim();
25
- if (trimmed.length === 0 || trimmed.length > 128)
26
- return null;
27
- if (!/^[A-Za-z0-9_-]+$/.test(trimmed))
28
- return null;
29
- const resolved = resolve(this.dir, `${trimmed}.json`);
30
- if (!resolved.startsWith(resolve(this.dir) + '/'))
31
- return null;
32
- return resolved;
22
+ agentId;
23
+ db;
24
+ constructor({ agentId }) {
25
+ this.agentId = agentId;
26
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-type-assertion -- getDb returns Db which extends NodePgDatabase
27
+ this.db = getDb();
33
28
  }
34
- save(entry) {
29
+ async save(entry) {
30
+ const id = randomUUID();
31
+ const now = new Date();
35
32
  const full = {
36
33
  ...entry,
37
- id: randomUUID(),
38
- timestamp: new Date().toISOString(),
34
+ id,
35
+ timestamp: now.toISOString(),
39
36
  };
40
- const file = this.resolvePath(full.id);
41
- if (!file)
42
- throw new Error('Invalid feedback ID');
43
- this.ensureDir();
44
- writeFileSync(file, JSON.stringify(full, null, 2));
37
+ try {
38
+ await this.db.insert(feedback).values({
39
+ id: full.id,
40
+ agentId: this.agentId,
41
+ sessionId: full.sessionId,
42
+ messageId: full.messageId,
43
+ rating: full.rating,
44
+ comment: full.comment ?? null,
45
+ query: full.query,
46
+ response: full.response,
47
+ toolCalls: full.toolCalls ?? null,
48
+ model: full.model ?? null,
49
+ createdAt: now,
50
+ });
51
+ }
52
+ catch (err) {
53
+ throw new StoreError('Failed to save feedback', {
54
+ store: 'feedback',
55
+ operation: 'save',
56
+ cause: err,
57
+ context: { agentId: this.agentId, feedbackId: id },
58
+ });
59
+ }
60
+ // Best-effort NOTIFY — don't fail the write.
61
+ try {
62
+ await notifyFeedbackCreated(this.db, {
63
+ feedbackId: id,
64
+ agentId: this.agentId,
65
+ sessionId: full.sessionId,
66
+ });
67
+ }
68
+ catch (err) {
69
+ log.warn('feedback_notify_failed', {
70
+ feedbackId: id,
71
+ error: err instanceof Error ? err.message : String(err),
72
+ });
73
+ }
45
74
  return full;
46
75
  }
47
- list(limit = 100) {
48
- if (!existsSync(this.dir))
49
- return [];
50
- const files = readdirSync(this.dir).filter((f) => f.endsWith('.json'));
51
- const entries = [];
52
- for (const file of files) {
53
- try {
54
- const raw = JSON.parse(readFileSync(join(this.dir, file), 'utf-8'));
55
- entries.push(raw); // eslint-disable-line @typescript-eslint/no-unsafe-type-assertion -- trusted local file
56
- }
57
- catch {
58
- // Skip corrupt files
59
- }
76
+ async list(limit = 100) {
77
+ try {
78
+ const rows = await this.db
79
+ .select()
80
+ .from(feedback)
81
+ .where(eq(feedback.agentId, this.agentId))
82
+ .orderBy(desc(feedback.createdAt))
83
+ .limit(limit);
84
+ return rows.map((r) => this.rowToEntry(r));
85
+ }
86
+ catch (err) {
87
+ throw new StoreError('Failed to list feedback', {
88
+ store: 'feedback',
89
+ operation: 'list',
90
+ cause: err,
91
+ context: { agentId: this.agentId },
92
+ });
60
93
  }
61
- return entries
62
- .sort((a, b) => new Date(b.timestamp).getTime() - new Date(a.timestamp).getTime())
63
- .slice(0, limit);
64
94
  }
65
- summary() {
66
- const all = this.list(1000);
67
- const thumbsUp = all.filter((e) => e.rating === 'up').length;
68
- const thumbsDown = all.filter((e) => e.rating === 'down').length;
69
- const unreviewedDown = all.filter((e) => e.rating === 'down' && !e.reviewedAt);
70
- const recentDown = unreviewedDown.slice(0, 20);
71
- return {
72
- total: all.length,
73
- thumbsUp,
74
- thumbsDown,
75
- recentDown,
76
- };
95
+ async summary() {
96
+ try {
97
+ const counts = await this.db
98
+ .select({ rating: feedback.rating, count: count() })
99
+ .from(feedback)
100
+ .where(eq(feedback.agentId, this.agentId))
101
+ .groupBy(feedback.rating);
102
+ let thumbsUp = 0;
103
+ let thumbsDown = 0;
104
+ for (const row of counts) {
105
+ if (row.rating === 'up')
106
+ thumbsUp = Number(row.count);
107
+ else if (row.rating === 'down')
108
+ thumbsDown = Number(row.count);
109
+ }
110
+ const unreviewedDown = await this.db
111
+ .select()
112
+ .from(feedback)
113
+ .where(sql `${feedback.agentId} = ${this.agentId} AND ${feedback.rating} = 'down' AND ${feedback.reviewedAt} IS NULL`)
114
+ .orderBy(desc(feedback.createdAt))
115
+ .limit(20);
116
+ return {
117
+ total: thumbsUp + thumbsDown,
118
+ thumbsUp,
119
+ thumbsDown,
120
+ recentDown: unreviewedDown.map((r) => this.rowToEntry(r)),
121
+ };
122
+ }
123
+ catch (err) {
124
+ throw new StoreError('Failed to summarize feedback', {
125
+ store: 'feedback',
126
+ operation: 'summary',
127
+ cause: err,
128
+ context: { agentId: this.agentId },
129
+ });
130
+ }
77
131
  }
78
132
  /** Mark feedback entries as reviewed so they're excluded from the next synthesis. */
79
- markReviewed(ids) {
80
- const now = new Date().toISOString();
81
- for (const id of ids) {
82
- const file = this.resolvePath(id);
83
- if (!file || !existsSync(file))
84
- continue;
85
- try {
86
- const raw = JSON.parse(readFileSync(file, 'utf-8'));
87
- // eslint-disable-next-line @typescript-eslint/no-unsafe-type-assertion -- trusted local file
88
- const entry = raw;
89
- entry.reviewedAt = now;
90
- writeFileSync(file, JSON.stringify(entry, null, 2));
91
- }
92
- catch {
93
- // Skip
94
- }
133
+ async markReviewed(ids) {
134
+ if (ids.length === 0)
135
+ return;
136
+ try {
137
+ await this.db
138
+ .update(feedback)
139
+ .set({ reviewedAt: new Date() })
140
+ .where(inArray(feedback.id, ids));
141
+ }
142
+ catch (err) {
143
+ throw new StoreError('Failed to mark feedback reviewed', {
144
+ store: 'feedback',
145
+ operation: 'markReviewed',
146
+ cause: err,
147
+ context: { agentId: this.agentId, ids },
148
+ });
95
149
  }
96
150
  }
151
+ rowToEntry(row) {
152
+ return {
153
+ id: row.id,
154
+ sessionId: row.sessionId,
155
+ messageId: row.messageId,
156
+ // eslint-disable-next-line @typescript-eslint/no-unsafe-type-assertion -- DB constraint ensures 'up' | 'down'
157
+ rating: row.rating,
158
+ comment: row.comment ?? undefined,
159
+ query: row.query,
160
+ response: row.response,
161
+ toolCalls: Array.isArray(row.toolCalls)
162
+ ? row.toolCalls.filter((v) => typeof v === 'string')
163
+ : undefined,
164
+ model: row.model ?? undefined,
165
+ timestamp: row.createdAt.toISOString(),
166
+ reviewedAt: row.reviewedAt?.toISOString(),
167
+ };
168
+ }
97
169
  }
98
170
  //# sourceMappingURL=feedback-store.js.map
@@ -1 +1 @@
1
- {"version":3,"file":"feedback-store.js","sourceRoot":"","sources":["../../../src/agent/feedback-store.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAEH,OAAO,EAAC,UAAU,EAAE,SAAS,EAAE,YAAY,EAAE,WAAW,EAAE,aAAa,EAAC,MAAM,SAAS,CAAC;AACxF,OAAO,EAAC,IAAI,EAAE,OAAO,EAAC,MAAM,WAAW,CAAC;AACxC,OAAO,EAAC,UAAU,EAAC,MAAM,aAAa,CAAC;AAuBvC;;;GAGG;AACH,MAAM,OAAO,aAAa;IACP,GAAG,CAAS;IAE7B,YAAY,QAAgB;QAC1B,IAAI,CAAC,GAAG,GAAG,IAAI,CAAC,QAAQ,EAAE,SAAS,EAAE,UAAU,CAAC,CAAC;IACnD,CAAC;IAEO,SAAS;QACf,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC;YAC1B,SAAS,CAAC,IAAI,CAAC,GAAG,EAAE,EAAC,SAAS,EAAE,IAAI,EAAC,CAAC,CAAC;QACzC,CAAC;IACH,CAAC;IAEO,WAAW,CAAC,EAAU;QAC5B,MAAM,OAAO,GAAG,EAAE,CAAC,IAAI,EAAE,CAAC;QAC1B,IAAI,OAAO,CAAC,MAAM,KAAK,CAAC,IAAI,OAAO,CAAC,MAAM,GAAG,GAAG;YAAE,OAAO,IAAI,CAAC;QAC9D,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,OAAO,CAAC;YAAE,OAAO,IAAI,CAAC;QACnD,MAAM,QAAQ,GAAG,OAAO,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,OAAO,OAAO,CAAC,CAAC;QACtD,IAAI,CAAC,QAAQ,CAAC,UAAU,CAAC,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,GAAG,CAAC;YAAE,OAAO,IAAI,CAAC;QAC/D,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED,IAAI,CAAC,KAA8C;QACjD,MAAM,IAAI,GAAkB;YAC1B,GAAG,KAAK;YACR,EAAE,EAAE,UAAU,EAAE;YAChB,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;SACpC,CAAC;QACF,MAAM,IAAI,GAAG,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QACvC,IAAI,CAAC,IAAI;YAAE,MAAM,IAAI,KAAK,CAAC,qBAAqB,CAAC,CAAC;QAClD,IAAI,CAAC,SAAS,EAAE,CAAC;QACjB,aAAa,CAAC,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QACnD,OAAO,IAAI,CAAC;IACd,CAAC;IAED,IAAI,CAAC,KAAK,GAAG,GAAG;QACd,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,GAAG,CAAC;YAAE,OAAO,EAAE,CAAC;QACrC,MAAM,KAAK,GAAG,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC;QACvE,MAAM,OAAO,GAAoB,EAAE,CAAC;QAEpC,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;YACzB,IAAI,CAAC;gBACH,MAAM,GAAG,GAAY,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,IAAI,CAAC,EAAE,OAAO,CAAC,CAAC,CAAC;gBAC7E,OAAO,CAAC,IAAI,CAAC,GAAoB,CAAC,CAAC,CAAC,wFAAwF;YAC9H,CAAC;YAAC,MAAM,CAAC;gBACP,qBAAqB;YACvB,CAAC;QACH,CAAC;QAED,OAAO,OAAO;aACX,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,IAAI,IAAI,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,OAAO,EAAE,GAAG,IAAI,IAAI,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,OAAO,EAAE,CAAC;aACjF,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC;IACrB,CAAC;IAED,OAAO;QACL,MAAM,GAAG,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC5B,MAAM,QAAQ,GAAG,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,IAAI,CAAC,CAAC,MAAM,CAAC;QAC7D,MAAM,UAAU,GAAG,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,MAAM,CAAC,CAAC,MAAM,CAAC;QACjE,MAAM,cAAc,GAAG,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,MAAM,IAAI,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC;QAC/E,MAAM,UAAU,GAAG,cAAc,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;QAE/C,OAAO;YACL,KAAK,EAAE,GAAG,CAAC,MAAM;YACjB,QAAQ;YACR,UAAU;YACV,UAAU;SACX,CAAC;IACJ,CAAC;IAED,qFAAqF;IACrF,YAAY,CAAC,GAAa;QACxB,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC;QACrC,KAAK,MAAM,EAAE,IAAI,GAAG,EAAE,CAAC;YACrB,MAAM,IAAI,GAAG,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;YAClC,IAAI,CAAC,IAAI,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC;gBAAE,SAAS;YACzC,IAAI,CAAC;gBACH,MAAM,GAAG,GAAY,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC,CAAC;gBAC7D,6FAA6F;gBAC7F,MAAM,KAAK,GAAG,GAAoB,CAAC;gBACnC,KAAK,CAAC,UAAU,GAAG,GAAG,CAAC;gBACvB,aAAa,CAAC,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;YACtD,CAAC;YAAC,MAAM,CAAC;gBACP,OAAO;YACT,CAAC;QACH,CAAC;IACH,CAAC;CACF"}
1
+ {"version":3,"file":"feedback-store.js","sourceRoot":"","sources":["../../../src/agent/feedback-store.ts"],"names":[],"mappings":"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"}