@claudecam/server 0.1.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 (52) hide show
  1. package/dist/db/index.js +68 -0
  2. package/dist/db/index.js.map +1 -0
  3. package/dist/db/queries.js +658 -0
  4. package/dist/db/queries.js.map +1 -0
  5. package/dist/db/schema.sql +259 -0
  6. package/dist/index.js +128 -0
  7. package/dist/index.js.map +1 -0
  8. package/dist/routes/agents.js +68 -0
  9. package/dist/routes/agents.js.map +1 -0
  10. package/dist/routes/correlation-audit.js +31 -0
  11. package/dist/routes/correlation-audit.js.map +1 -0
  12. package/dist/routes/events.js +81 -0
  13. package/dist/routes/events.js.map +1 -0
  14. package/dist/routes/files.js +24 -0
  15. package/dist/routes/files.js.map +1 -0
  16. package/dist/routes/parse-prd.js +38 -0
  17. package/dist/routes/parse-prd.js.map +1 -0
  18. package/dist/routes/projects.js +96 -0
  19. package/dist/routes/projects.js.map +1 -0
  20. package/dist/routes/registry.js +88 -0
  21. package/dist/routes/registry.js.map +1 -0
  22. package/dist/routes/session-groups.js +182 -0
  23. package/dist/routes/session-groups.js.map +1 -0
  24. package/dist/routes/sessions.js +109 -0
  25. package/dist/routes/sessions.js.map +1 -0
  26. package/dist/routes/sprints.js +58 -0
  27. package/dist/routes/sprints.js.map +1 -0
  28. package/dist/routes/stats.js +63 -0
  29. package/dist/routes/stats.js.map +1 -0
  30. package/dist/routes/stream.js +21 -0
  31. package/dist/routes/stream.js.map +1 -0
  32. package/dist/routes/tasks.js +198 -0
  33. package/dist/routes/tasks.js.map +1 -0
  34. package/dist/services/correlation-engine.js +577 -0
  35. package/dist/services/correlation-engine.js.map +1 -0
  36. package/dist/services/event-processor.js +857 -0
  37. package/dist/services/event-processor.js.map +1 -0
  38. package/dist/services/prd-parser.js +142 -0
  39. package/dist/services/prd-parser.js.map +1 -0
  40. package/dist/services/project-manager.js +351 -0
  41. package/dist/services/project-manager.js.map +1 -0
  42. package/dist/services/project-router.js +56 -0
  43. package/dist/services/project-router.js.map +1 -0
  44. package/dist/services/session-manager.js +76 -0
  45. package/dist/services/session-manager.js.map +1 -0
  46. package/dist/services/sse-manager.js +115 -0
  47. package/dist/services/sse-manager.js.map +1 -0
  48. package/dist/services/string-similarity.js +256 -0
  49. package/dist/services/string-similarity.js.map +1 -0
  50. package/dist/services/task-completion.js +251 -0
  51. package/dist/services/task-completion.js.map +1 -0
  52. package/package.json +59 -0
@@ -0,0 +1,38 @@
1
+ import { Router } from 'express';
2
+ import { parsePrdRequestSchema } from '@claudecam/shared';
3
+ import { parsePrd } from '../services/prd-parser.js';
4
+ export const parsePrdRouter = Router();
5
+ // POST /api/parse-prd - Parse a PRD without creating a project
6
+ parsePrdRouter.post('/', (req, res) => {
7
+ try {
8
+ const parsed = parsePrdRequestSchema.safeParse(req.body);
9
+ if (!parsed.success) {
10
+ res.status(400).json({ error: 'Invalid request', details: parsed.error.issues });
11
+ return;
12
+ }
13
+ const { content, method } = parsed.data;
14
+ const result = parsePrd(content, method);
15
+ res.json({
16
+ sections: result.sections.map(s => ({
17
+ title: s.title,
18
+ level: s.level,
19
+ taskCount: 0,
20
+ })),
21
+ suggested_tasks: result.suggestedTasks.map(t => ({
22
+ title: t.title,
23
+ description: t.description,
24
+ priority: t.priority,
25
+ complexity: t.complexity,
26
+ dependsOn: t.dependsOn,
27
+ prdSection: t.prdSection,
28
+ prdLineStart: t.prdLineStart,
29
+ prdLineEnd: t.prdLineEnd,
30
+ })),
31
+ suggested_sprints: result.suggestedSprints,
32
+ });
33
+ }
34
+ catch (err) {
35
+ res.status(500).json({ error: 'Internal server error' });
36
+ }
37
+ });
38
+ //# sourceMappingURL=parse-prd.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"parse-prd.js","sourceRoot":"","sources":["../../src/routes/parse-prd.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAEjC,OAAO,EAAE,qBAAqB,EAAE,MAAM,mBAAmB,CAAC;AAC1D,OAAO,EAAE,QAAQ,EAAE,MAAM,2BAA2B,CAAC;AAErD,MAAM,CAAC,MAAM,cAAc,GAAe,MAAM,EAAE,CAAC;AAEnD,+DAA+D;AAC/D,cAAc,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;IACvD,IAAI,CAAC;QACH,MAAM,MAAM,GAAG,qBAAqB,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACzD,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;YACpB,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,iBAAiB,EAAE,OAAO,EAAE,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,CAAC;YACjF,OAAO;QACT,CAAC;QAED,MAAM,EAAE,OAAO,EAAE,MAAM,EAAE,GAAG,MAAM,CAAC,IAAI,CAAC;QACxC,MAAM,MAAM,GAAG,QAAQ,CAAC,OAAO,EAAE,MAAM,CAAC,CAAC;QAEzC,GAAG,CAAC,IAAI,CAAC;YACP,QAAQ,EAAE,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;gBAClC,KAAK,EAAE,CAAC,CAAC,KAAK;gBACd,KAAK,EAAE,CAAC,CAAC,KAAK;gBACd,SAAS,EAAE,CAAC;aACb,CAAC,CAAC;YACH,eAAe,EAAE,MAAM,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;gBAC/C,KAAK,EAAE,CAAC,CAAC,KAAK;gBACd,WAAW,EAAE,CAAC,CAAC,WAAW;gBAC1B,QAAQ,EAAE,CAAC,CAAC,QAAQ;gBACpB,UAAU,EAAE,CAAC,CAAC,UAAU;gBACxB,SAAS,EAAE,CAAC,CAAC,SAAS;gBACtB,UAAU,EAAE,CAAC,CAAC,UAAU;gBACxB,YAAY,EAAE,CAAC,CAAC,YAAY;gBAC5B,UAAU,EAAE,CAAC,CAAC,UAAU;aACzB,CAAC,CAAC;YACH,iBAAiB,EAAE,MAAM,CAAC,gBAAgB;SAC3C,CAAC,CAAC;IACL,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,uBAAuB,EAAE,CAAC,CAAC;IAC3D,CAAC;AACH,CAAC,CAAC,CAAC"}
@@ -0,0 +1,96 @@
1
+ import { Router } from 'express';
2
+ import { createProjectRequestSchema } from '@claudecam/shared';
3
+ import { createProject, getProject, listProjects, deleteProject, listSprints, } from '../services/project-manager.js';
4
+ export const projectsRouter = Router();
5
+ // POST /api/projects - Create a project from a PRD
6
+ projectsRouter.post('/', (req, res) => {
7
+ try {
8
+ const parsed = createProjectRequestSchema.safeParse(req.body);
9
+ if (!parsed.success) {
10
+ res.status(400).json({ error: 'Invalid request', details: parsed.error.issues });
11
+ return;
12
+ }
13
+ const { name, prd_content, parse_method } = parsed.data;
14
+ const result = createProject(name, prd_content, parse_method);
15
+ res.status(201).json({
16
+ project: {
17
+ id: result.project.id,
18
+ name: result.project.name,
19
+ totalTasks: result.tasks.length,
20
+ sprints: result.sprints.map(s => ({
21
+ id: s.id,
22
+ name: s.name,
23
+ taskCount: s.totalTasks,
24
+ })),
25
+ },
26
+ tasks_preview: result.tasks.slice(0, 20).map(t => ({
27
+ id: t.id,
28
+ title: t.title,
29
+ priority: t.priority,
30
+ complexity: t.complexity,
31
+ })),
32
+ });
33
+ }
34
+ catch (err) {
35
+ res.status(500).json({ error: 'Internal server error' });
36
+ }
37
+ });
38
+ // GET /api/projects - List all projects
39
+ projectsRouter.get('/', (_req, res) => {
40
+ try {
41
+ const projects = listProjects();
42
+ res.json({ projects });
43
+ }
44
+ catch (err) {
45
+ res.status(500).json({ error: 'Internal server error' });
46
+ }
47
+ });
48
+ // GET /api/projects/:id - Get project details with stats
49
+ projectsRouter.get('/:id', (req, res) => {
50
+ try {
51
+ const projectId = String(req.params['id']);
52
+ const project = getProject(projectId);
53
+ if (!project) {
54
+ res.status(404).json({ error: 'Project not found' });
55
+ return;
56
+ }
57
+ const sprints = listSprints(project.id);
58
+ const currentSprint = sprints.find(s => s.id === project.currentSprintId);
59
+ const completionPercent = project.totalTasks > 0
60
+ ? Math.round((project.completedTasks / project.totalTasks) * 1000) / 10
61
+ : 0;
62
+ res.json({
63
+ project: {
64
+ ...project,
65
+ completionPercent,
66
+ currentSprint: currentSprint ? {
67
+ id: currentSprint.id,
68
+ name: currentSprint.name,
69
+ totalTasks: currentSprint.totalTasks,
70
+ completedTasks: currentSprint.completedTasks,
71
+ completionPercent: currentSprint.totalTasks > 0
72
+ ? Math.round((currentSprint.completedTasks / currentSprint.totalTasks) * 1000) / 10
73
+ : 0,
74
+ } : undefined,
75
+ },
76
+ });
77
+ }
78
+ catch (err) {
79
+ res.status(500).json({ error: 'Internal server error' });
80
+ }
81
+ });
82
+ // DELETE /api/projects/:id
83
+ projectsRouter.delete('/:id', (req, res) => {
84
+ try {
85
+ const deleted = deleteProject(String(req.params['id']));
86
+ if (!deleted) {
87
+ res.status(404).json({ error: 'Project not found' });
88
+ return;
89
+ }
90
+ res.json({ ok: true });
91
+ }
92
+ catch (err) {
93
+ res.status(500).json({ error: 'Internal server error' });
94
+ }
95
+ });
96
+ //# sourceMappingURL=projects.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"projects.js","sourceRoot":"","sources":["../../src/routes/projects.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,88 @@
1
+ import { Router } from "express";
2
+ import { projectRegistryQueries, projectQueries } from "../db/queries.js";
3
+ export const registryRouter = Router();
4
+ // POST /api/registry - Register a working directory to a project
5
+ registryRouter.post("/", (req, res) => {
6
+ try {
7
+ const { working_directory, project_id, prd_path } = req.body;
8
+ if (!working_directory || !project_id) {
9
+ res
10
+ .status(400)
11
+ .json({ error: "working_directory and project_id are required" });
12
+ return;
13
+ }
14
+ // Verify project exists
15
+ const project = projectQueries.getById().get(project_id);
16
+ if (!project) {
17
+ res.status(404).json({ error: "Project not found" });
18
+ return;
19
+ }
20
+ projectRegistryQueries
21
+ .register()
22
+ .run(working_directory, project_id, prd_path || null, 0);
23
+ const registered = projectRegistryQueries
24
+ .getByWorkingDir()
25
+ .get(working_directory);
26
+ res.status(201).json({ registry: registered });
27
+ }
28
+ catch {
29
+ res.status(500).json({ error: "Failed to register project" });
30
+ }
31
+ });
32
+ // GET /api/registry/lookup?dir=<path> - Look up project by working directory
33
+ registryRouter.get("/lookup", (req, res) => {
34
+ try {
35
+ const dir = req.query["dir"];
36
+ if (!dir) {
37
+ res.status(400).json({ error: "dir query parameter required" });
38
+ return;
39
+ }
40
+ const registry = projectRegistryQueries
41
+ .getByWorkingDir()
42
+ .get(dir);
43
+ if (!registry) {
44
+ // Try prefix match
45
+ const prefix = projectRegistryQueries
46
+ .getByWorkingDirPrefix()
47
+ .get(dir);
48
+ if (prefix) {
49
+ res.json({ registry: prefix });
50
+ return;
51
+ }
52
+ res.status(404).json({ error: "Directory not registered" });
53
+ return;
54
+ }
55
+ res.json({ registry });
56
+ }
57
+ catch {
58
+ res.status(500).json({ error: "Failed to lookup registry" });
59
+ }
60
+ });
61
+ // GET /api/registry - List all registrations
62
+ registryRouter.get("/", (_req, res) => {
63
+ try {
64
+ const registrations = projectRegistryQueries
65
+ .getAll()
66
+ .all();
67
+ res.json({ registrations });
68
+ }
69
+ catch {
70
+ res.status(500).json({ error: "Failed to list registrations" });
71
+ }
72
+ });
73
+ // DELETE /api/registry?dir=<path> - Remove a registration
74
+ registryRouter.delete("/", (req, res) => {
75
+ try {
76
+ const dir = req.query["dir"];
77
+ if (!dir) {
78
+ res.status(400).json({ error: "dir query parameter required" });
79
+ return;
80
+ }
81
+ projectRegistryQueries.delete().run(dir);
82
+ res.json({ success: true });
83
+ }
84
+ catch {
85
+ res.status(500).json({ error: "Failed to delete registration" });
86
+ }
87
+ });
88
+ //# sourceMappingURL=registry.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"registry.js","sourceRoot":"","sources":["../../src/routes/registry.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,182 @@
1
+ import { Router } from 'express';
2
+ import { sessionGroupQueries, sessionGroupMemberQueries, agentQueries } from '../db/queries.js';
3
+ import { getDb } from '../db/index.js';
4
+ import { DEFAULT_EVENT_LIMIT } from '@cam/shared';
5
+ export const sessionGroupsRouter = Router();
6
+ function mapGroup(row) {
7
+ return {
8
+ id: row.id,
9
+ name: row.name ?? undefined,
10
+ mainSessionId: row.main_session_id,
11
+ createdAt: row.created_at,
12
+ };
13
+ }
14
+ function mapMember(row) {
15
+ return {
16
+ groupId: row.group_id,
17
+ sessionId: row.session_id,
18
+ agentName: row.agent_name ?? undefined,
19
+ agentType: row.agent_type ?? undefined,
20
+ joinedAt: row.joined_at,
21
+ sessionStatus: row.session_status,
22
+ startedAt: row.started_at,
23
+ endedAt: row.ended_at ?? undefined,
24
+ agentCount: row.agent_count,
25
+ eventCount: row.event_count,
26
+ workingDirectory: row.working_directory,
27
+ };
28
+ }
29
+ // GET /api/session-groups - List all groups
30
+ sessionGroupsRouter.get('/', (_req, res) => {
31
+ try {
32
+ const rows = sessionGroupQueries.getAll().all();
33
+ const groups = rows.map(row => {
34
+ const members = sessionGroupMemberQueries.getByGroup().all(row.id);
35
+ return {
36
+ ...mapGroup(row),
37
+ memberCount: members.length,
38
+ members: members.map(mapMember),
39
+ };
40
+ });
41
+ res.json({ groups });
42
+ }
43
+ catch {
44
+ res.status(500).json({ error: 'Internal server error' });
45
+ }
46
+ });
47
+ // GET /api/session-groups/active - Get currently active group
48
+ sessionGroupsRouter.get('/active', (_req, res) => {
49
+ try {
50
+ const row = sessionGroupQueries.getActiveGroup().get();
51
+ if (!row) {
52
+ res.status(404).json({ error: 'No active session group found' });
53
+ return;
54
+ }
55
+ const members = sessionGroupMemberQueries.getByGroup().all(row.id);
56
+ const group = {
57
+ ...mapGroup(row),
58
+ memberCount: members.length,
59
+ members: members.map(mapMember),
60
+ };
61
+ res.json({ group });
62
+ }
63
+ catch {
64
+ res.status(500).json({ error: 'Internal server error' });
65
+ }
66
+ });
67
+ // GET /api/session-groups/:id - Group details with members
68
+ sessionGroupsRouter.get('/:id', (req, res) => {
69
+ try {
70
+ const groupId = req.params['id'];
71
+ const row = sessionGroupQueries.getById().get(groupId);
72
+ if (!row) {
73
+ res.status(404).json({ error: 'Session group not found' });
74
+ return;
75
+ }
76
+ const members = sessionGroupMemberQueries.getByGroup().all(groupId);
77
+ const group = {
78
+ ...mapGroup(row),
79
+ memberCount: members.length,
80
+ members: members.map(mapMember),
81
+ };
82
+ res.json({ group });
83
+ }
84
+ catch {
85
+ res.status(500).json({ error: 'Internal server error' });
86
+ }
87
+ });
88
+ // GET /api/session-groups/:id/events - Events from ALL members (merged timeline)
89
+ sessionGroupsRouter.get('/:id/events', (req, res) => {
90
+ try {
91
+ const groupId = req.params['id'];
92
+ const limit = parseInt(req.query['limit']) || DEFAULT_EVENT_LIMIT;
93
+ const offset = parseInt(req.query['offset']) || 0;
94
+ const category = req.query['category'];
95
+ // Get all session IDs in the group
96
+ const memberRows = sessionGroupMemberQueries.getAllSessionIdsInGroup().all(groupId);
97
+ if (memberRows.length === 0) {
98
+ res.status(404).json({ error: 'Session group not found or has no members' });
99
+ return;
100
+ }
101
+ const sessionIds = memberRows.map(m => m['session_id']);
102
+ // Query events from all sessions, merged by timestamp
103
+ // We build the query dynamically since we need IN (?, ?, ...)
104
+ const db = getDb();
105
+ const placeholders = sessionIds.map(() => '?').join(', ');
106
+ let query;
107
+ let params;
108
+ if (category) {
109
+ query = `
110
+ SELECT * FROM events
111
+ WHERE session_id IN (${placeholders}) AND category = ?
112
+ ORDER BY timestamp DESC
113
+ LIMIT ? OFFSET ?
114
+ `;
115
+ params = [...sessionIds, category, limit, offset];
116
+ }
117
+ else {
118
+ query = `
119
+ SELECT * FROM events
120
+ WHERE session_id IN (${placeholders})
121
+ ORDER BY timestamp DESC
122
+ LIMIT ? OFFSET ?
123
+ `;
124
+ params = [...sessionIds, limit, offset];
125
+ }
126
+ const rows = db.prepare(query).all(...params);
127
+ const events = rows.map(row => ({
128
+ id: row['id'],
129
+ sessionId: row['session_id'],
130
+ agentId: row['agent_id'],
131
+ timestamp: row['timestamp'],
132
+ hookType: row['hook_type'],
133
+ category: row['category'],
134
+ tool: row['tool'] ?? undefined,
135
+ filePath: row['file_path'] ?? undefined,
136
+ input: row['input'] ?? undefined,
137
+ output: row['output'] ?? undefined,
138
+ error: row['error'] ?? undefined,
139
+ duration: row['duration'] ?? undefined,
140
+ metadata: row['metadata'] ? JSON.parse(row['metadata']) : undefined,
141
+ }));
142
+ res.json({ events, sessionIds });
143
+ }
144
+ catch {
145
+ res.status(500).json({ error: 'Internal server error' });
146
+ }
147
+ });
148
+ // GET /api/session-groups/:id/agents - All agents across all sessions in group
149
+ sessionGroupsRouter.get('/:id/agents', (req, res) => {
150
+ try {
151
+ const groupId = req.params['id'];
152
+ const memberRows = sessionGroupMemberQueries.getAllSessionIdsInGroup().all(groupId);
153
+ if (memberRows.length === 0) {
154
+ res.status(404).json({ error: 'Session group not found or has no members' });
155
+ return;
156
+ }
157
+ const agents = [];
158
+ for (const member of memberRows) {
159
+ const sessionId = member['session_id'];
160
+ const sessionAgents = agentQueries.getBySession().all(sessionId);
161
+ for (const agent of sessionAgents) {
162
+ agents.push({
163
+ id: agent['id'],
164
+ sessionId: agent['session_id'],
165
+ name: agent['name'],
166
+ type: agent['type'],
167
+ status: agent['status'],
168
+ firstSeenAt: agent['first_seen_at'],
169
+ lastActivityAt: agent['last_activity_at'],
170
+ currentTask: agent['current_task'] ?? undefined,
171
+ toolCallCount: agent['tool_call_count'],
172
+ errorCount: agent['error_count'],
173
+ });
174
+ }
175
+ }
176
+ res.json({ agents });
177
+ }
178
+ catch {
179
+ res.status(500).json({ error: 'Internal server error' });
180
+ }
181
+ });
182
+ //# sourceMappingURL=session-groups.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"session-groups.js","sourceRoot":"","sources":["../../src/routes/session-groups.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAEjC,OAAO,EAAE,mBAAmB,EAAE,yBAAyB,EAAE,YAAY,EAAE,MAAM,kBAAkB,CAAC;AAChG,OAAO,EAAE,KAAK,EAAE,MAAM,gBAAgB,CAAC;AACvC,OAAO,EAAE,mBAAmB,EAAE,MAAM,aAAa,CAAC;AAElD,MAAM,CAAC,MAAM,mBAAmB,GAAG,MAAM,EAAE,CAAC;AAuB5C,SAAS,QAAQ,CAAC,GAAa;IAC7B,OAAO;QACL,EAAE,EAAE,GAAG,CAAC,EAAE;QACV,IAAI,EAAE,GAAG,CAAC,IAAI,IAAI,SAAS;QAC3B,aAAa,EAAE,GAAG,CAAC,eAAe;QAClC,SAAS,EAAE,GAAG,CAAC,UAAU;KAC1B,CAAC;AACJ,CAAC;AAED,SAAS,SAAS,CAAC,GAAc;IAC/B,OAAO;QACL,OAAO,EAAE,GAAG,CAAC,QAAQ;QACrB,SAAS,EAAE,GAAG,CAAC,UAAU;QACzB,SAAS,EAAE,GAAG,CAAC,UAAU,IAAI,SAAS;QACtC,SAAS,EAAE,GAAG,CAAC,UAAU,IAAI,SAAS;QACtC,QAAQ,EAAE,GAAG,CAAC,SAAS;QACvB,aAAa,EAAE,GAAG,CAAC,cAAc;QACjC,SAAS,EAAE,GAAG,CAAC,UAAU;QACzB,OAAO,EAAE,GAAG,CAAC,QAAQ,IAAI,SAAS;QAClC,UAAU,EAAE,GAAG,CAAC,WAAW;QAC3B,UAAU,EAAE,GAAG,CAAC,WAAW;QAC3B,gBAAgB,EAAE,GAAG,CAAC,iBAAiB;KACxC,CAAC;AACJ,CAAC;AAED,4CAA4C;AAC5C,mBAAmB,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAa,EAAE,GAAa,EAAE,EAAE;IAC5D,IAAI,CAAC;QACH,MAAM,IAAI,GAAG,mBAAmB,CAAC,MAAM,EAAE,CAAC,GAAG,EAAgB,CAAC;QAC9D,MAAM,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE;YAC5B,MAAM,OAAO,GAAG,yBAAyB,CAAC,UAAU,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAgB,CAAC;YAClF,OAAO;gBACL,GAAG,QAAQ,CAAC,GAAG,CAAC;gBAChB,WAAW,EAAE,OAAO,CAAC,MAAM;gBAC3B,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,SAAS,CAAC;aAChC,CAAC;QACJ,CAAC,CAAC,CAAC;QACH,GAAG,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC;IACvB,CAAC;IAAC,MAAM,CAAC;QACP,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,uBAAuB,EAAE,CAAC,CAAC;IAC3D,CAAC;AACH,CAAC,CAAC,CAAC;AAEH,8DAA8D;AAC9D,mBAAmB,CAAC,GAAG,CAAC,SAAS,EAAE,CAAC,IAAa,EAAE,GAAa,EAAE,EAAE;IAClE,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,mBAAmB,CAAC,cAAc,EAAE,CAAC,GAAG,EAA0B,CAAC;QAC/E,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,+BAA+B,EAAE,CAAC,CAAC;YACjE,OAAO;QACT,CAAC;QAED,MAAM,OAAO,GAAG,yBAAyB,CAAC,UAAU,EAAE,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAgB,CAAC;QAClF,MAAM,KAAK,GAAG;YACZ,GAAG,QAAQ,CAAC,GAAG,CAAC;YAChB,WAAW,EAAE,OAAO,CAAC,MAAM;YAC3B,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,SAAS,CAAC;SAChC,CAAC;QAEF,GAAG,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;IACtB,CAAC;IAAC,MAAM,CAAC;QACP,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,uBAAuB,EAAE,CAAC,CAAC;IAC3D,CAAC;AACH,CAAC,CAAC,CAAC;AAEH,2DAA2D;AAC3D,mBAAmB,CAAC,GAAG,CAAC,MAAM,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;IAC9D,IAAI,CAAC;QACH,MAAM,OAAO,GAAG,GAAG,CAAC,MAAM,CAAC,IAAI,CAAE,CAAC;QAClC,MAAM,GAAG,GAAG,mBAAmB,CAAC,OAAO,EAAE,CAAC,GAAG,CAAC,OAAO,CAAyB,CAAC;QAC/E,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,yBAAyB,EAAE,CAAC,CAAC;YAC3D,OAAO;QACT,CAAC;QAED,MAAM,OAAO,GAAG,yBAAyB,CAAC,UAAU,EAAE,CAAC,GAAG,CAAC,OAAO,CAAgB,CAAC;QACnF,MAAM,KAAK,GAAG;YACZ,GAAG,QAAQ,CAAC,GAAG,CAAC;YAChB,WAAW,EAAE,OAAO,CAAC,MAAM;YAC3B,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,SAAS,CAAC;SAChC,CAAC;QAEF,GAAG,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;IACtB,CAAC;IAAC,MAAM,CAAC;QACP,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,uBAAuB,EAAE,CAAC,CAAC;IAC3D,CAAC;AACH,CAAC,CAAC,CAAC;AAEH,iFAAiF;AACjF,mBAAmB,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;IACrE,IAAI,CAAC;QACH,MAAM,OAAO,GAAG,GAAG,CAAC,MAAM,CAAC,IAAI,CAAE,CAAC;QAClC,MAAM,KAAK,GAAG,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,OAAO,CAAW,CAAC,IAAI,mBAAmB,CAAC;QAC5E,MAAM,MAAM,GAAG,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,QAAQ,CAAW,CAAC,IAAI,CAAC,CAAC;QAC5D,MAAM,QAAQ,GAAG,GAAG,CAAC,KAAK,CAAC,UAAU,CAAuB,CAAC;QAE7D,mCAAmC;QACnC,MAAM,UAAU,GAAG,yBAAyB,CAAC,uBAAuB,EAAE,CAAC,GAAG,CAAC,OAAO,CAAmC,CAAC;QACtH,IAAI,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC5B,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,2CAA2C,EAAE,CAAC,CAAC;YAC7E,OAAO;QACT,CAAC;QAED,MAAM,UAAU,GAAG,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,YAAY,CAAW,CAAC,CAAC;QAElE,sDAAsD;QACtD,8DAA8D;QAC9D,MAAM,EAAE,GAAG,KAAK,EAAE,CAAC;QAEnB,MAAM,YAAY,GAAG,UAAU,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAC1D,IAAI,KAAa,CAAC;QAClB,IAAI,MAAiB,CAAC;QAEtB,IAAI,QAAQ,EAAE,CAAC;YACb,KAAK,GAAG;;+BAEiB,YAAY;;;OAGpC,CAAC;YACF,MAAM,GAAG,CAAC,GAAG,UAAU,EAAE,QAAQ,EAAE,KAAK,EAAE,MAAM,CAAC,CAAC;QACpD,CAAC;aAAM,CAAC;YACN,KAAK,GAAG;;+BAEiB,YAAY;;;OAGpC,CAAC;YACF,MAAM,GAAG,CAAC,GAAG,UAAU,EAAE,KAAK,EAAE,MAAM,CAAC,CAAC;QAC1C,CAAC;QAED,MAAM,IAAI,GAAG,EAAE,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,GAAG,CAAC,GAAG,MAAM,CAAmC,CAAC;QAEhF,MAAM,MAAM,GAAG,IAAI,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YAC9B,EAAE,EAAE,GAAG,CAAC,IAAI,CAAC;YACb,SAAS,EAAE,GAAG,CAAC,YAAY,CAAC;YAC5B,OAAO,EAAE,GAAG,CAAC,UAAU,CAAC;YACxB,SAAS,EAAE,GAAG,CAAC,WAAW,CAAC;YAC3B,QAAQ,EAAE,GAAG,CAAC,WAAW,CAAC;YAC1B,QAAQ,EAAE,GAAG,CAAC,UAAU,CAAC;YACzB,IAAI,EAAE,GAAG,CAAC,MAAM,CAAC,IAAI,SAAS;YAC9B,QAAQ,EAAE,GAAG,CAAC,WAAW,CAAC,IAAI,SAAS;YACvC,KAAK,EAAE,GAAG,CAAC,OAAO,CAAC,IAAI,SAAS;YAChC,MAAM,EAAE,GAAG,CAAC,QAAQ,CAAC,IAAI,SAAS;YAClC,KAAK,EAAE,GAAG,CAAC,OAAO,CAAC,IAAI,SAAS;YAChC,QAAQ,EAAE,GAAG,CAAC,UAAU,CAAC,IAAI,SAAS;YACtC,QAAQ,EAAE,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,UAAU,CAAW,CAAC,CAAC,CAAC,CAAC,SAAS;SAC9E,CAAC,CAAC,CAAC;QAEJ,GAAG,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,UAAU,EAAE,CAAC,CAAC;IACnC,CAAC;IAAC,MAAM,CAAC;QACP,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,uBAAuB,EAAE,CAAC,CAAC;IAC3D,CAAC;AACH,CAAC,CAAC,CAAC;AAEH,+EAA+E;AAC/E,mBAAmB,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;IACrE,IAAI,CAAC;QACH,MAAM,OAAO,GAAG,GAAG,CAAC,MAAM,CAAC,IAAI,CAAE,CAAC;QAElC,MAAM,UAAU,GAAG,yBAAyB,CAAC,uBAAuB,EAAE,CAAC,GAAG,CAAC,OAAO,CAAmC,CAAC;QACtH,IAAI,UAAU,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC5B,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,2CAA2C,EAAE,CAAC,CAAC;YAC7E,OAAO;QACT,CAAC;QAED,MAAM,MAAM,GAAmC,EAAE,CAAC;QAClD,KAAK,MAAM,MAAM,IAAI,UAAU,EAAE,CAAC;YAChC,MAAM,SAAS,GAAG,MAAM,CAAC,YAAY,CAAW,CAAC;YACjD,MAAM,aAAa,GAAG,YAAY,CAAC,YAAY,EAAE,CAAC,GAAG,CAAC,SAAS,CAAmC,CAAC;YACnG,KAAK,MAAM,KAAK,IAAI,aAAa,EAAE,CAAC;gBAClC,MAAM,CAAC,IAAI,CAAC;oBACV,EAAE,EAAE,KAAK,CAAC,IAAI,CAAC;oBACf,SAAS,EAAE,KAAK,CAAC,YAAY,CAAC;oBAC9B,IAAI,EAAE,KAAK,CAAC,MAAM,CAAC;oBACnB,IAAI,EAAE,KAAK,CAAC,MAAM,CAAC;oBACnB,MAAM,EAAE,KAAK,CAAC,QAAQ,CAAC;oBACvB,WAAW,EAAE,KAAK,CAAC,eAAe,CAAC;oBACnC,cAAc,EAAE,KAAK,CAAC,kBAAkB,CAAC;oBACzC,WAAW,EAAE,KAAK,CAAC,cAAc,CAAC,IAAI,SAAS;oBAC/C,aAAa,EAAE,KAAK,CAAC,iBAAiB,CAAC;oBACvC,UAAU,EAAE,KAAK,CAAC,aAAa,CAAC;iBACjC,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAED,GAAG,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC;IACvB,CAAC;IAAC,MAAM,CAAC;QACP,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,uBAAuB,EAAE,CAAC,CAAC;IAC3D,CAAC;AACH,CAAC,CAAC,CAAC"}
@@ -0,0 +1,109 @@
1
+ import { Router } from 'express';
2
+ import { DEFAULT_SESSION_LIMIT } from '@claudecam/shared';
3
+ import { listSessions, deleteSession, getSessionWithDetails, listSessionsByProject } from '../services/session-manager.js';
4
+ import { sessionQueries, agentQueries } from '../db/queries.js';
5
+ export const sessionsRouter = Router();
6
+ // GET /api/sessions
7
+ sessionsRouter.get('/', (req, res) => {
8
+ try {
9
+ const status = req.query['status'];
10
+ const projectId = req.query['project_id'];
11
+ const limit = parseInt(req.query['limit']) || DEFAULT_SESSION_LIMIT;
12
+ const offset = parseInt(req.query['offset']) || 0;
13
+ let sessions;
14
+ if (projectId) {
15
+ sessions = listSessionsByProject(projectId, { limit, offset });
16
+ }
17
+ else {
18
+ sessions = listSessions({ status, limit, offset });
19
+ }
20
+ res.json({ sessions });
21
+ }
22
+ catch (err) {
23
+ res.status(500).json({ error: 'Internal server error' });
24
+ }
25
+ });
26
+ // GET /api/sessions/:id
27
+ sessionsRouter.get('/:id', (req, res) => {
28
+ try {
29
+ const session = getSessionWithDetails(String(req.params['id']));
30
+ if (!session) {
31
+ res.status(404).json({ error: 'Session not found' });
32
+ return;
33
+ }
34
+ res.json({ session });
35
+ }
36
+ catch (err) {
37
+ res.status(500).json({ error: 'Internal server error' });
38
+ }
39
+ });
40
+ // PATCH /api/sessions/:id/name - Rename a session
41
+ sessionsRouter.patch('/:id/name', (req, res) => {
42
+ try {
43
+ const sessionId = String(req.params['id']);
44
+ const { name } = req.body;
45
+ if (!name || typeof name !== 'string' || name.trim().length === 0) {
46
+ res.status(400).json({ error: 'Name is required' });
47
+ return;
48
+ }
49
+ const trimmedName = name.trim().slice(0, 100);
50
+ const row = sessionQueries.getById().get(sessionId);
51
+ if (!row) {
52
+ res.status(404).json({ error: 'Session not found' });
53
+ return;
54
+ }
55
+ const meta = row['metadata'] ? JSON.parse(row['metadata']) : {};
56
+ meta.name = trimmedName;
57
+ meta.nameSource = 'user';
58
+ sessionQueries.updateMetadata().run(JSON.stringify(meta), sessionId);
59
+ res.json({ ok: true, name: trimmedName });
60
+ }
61
+ catch (err) {
62
+ res.status(500).json({ error: 'Internal server error' });
63
+ }
64
+ });
65
+ // PATCH /api/sessions/:id/close - Manually close an active session
66
+ sessionsRouter.patch('/:id/close', (req, res) => {
67
+ try {
68
+ const sessionId = String(req.params['id']);
69
+ const row = sessionQueries.getById().get(sessionId);
70
+ if (!row) {
71
+ res.status(404).json({ error: 'Session not found' });
72
+ return;
73
+ }
74
+ if (row['status'] !== 'active') {
75
+ res.json({ ok: true, status: row['status'] });
76
+ return;
77
+ }
78
+ const now = new Date().toISOString();
79
+ // Mark all active/idle agents as completed
80
+ const agents = agentQueries.getBySession().all(sessionId);
81
+ for (const agent of agents) {
82
+ const s = agent['status'];
83
+ if (s === 'active' || s === 'idle') {
84
+ agentQueries.updateStatus().run('completed', now, agent['id'], sessionId);
85
+ }
86
+ }
87
+ // Mark session as completed
88
+ sessionQueries.updateStatus().run('completed', now, sessionId);
89
+ res.json({ ok: true, status: 'completed' });
90
+ }
91
+ catch (err) {
92
+ res.status(500).json({ error: 'Internal server error' });
93
+ }
94
+ });
95
+ // DELETE /api/sessions/:id
96
+ sessionsRouter.delete('/:id', (req, res) => {
97
+ try {
98
+ const deleted = deleteSession(String(req.params['id']));
99
+ if (!deleted) {
100
+ res.status(404).json({ error: 'Session not found' });
101
+ return;
102
+ }
103
+ res.json({ ok: true });
104
+ }
105
+ catch (err) {
106
+ res.status(500).json({ error: 'Internal server error' });
107
+ }
108
+ });
109
+ //# sourceMappingURL=sessions.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sessions.js","sourceRoot":"","sources":["../../src/routes/sessions.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,58 @@
1
+ import { Router } from 'express';
2
+ import { createSprintRequestSchema } from '@claudecam/shared';
3
+ import { listSprints, createSprint, updateSprint, getProject, } from '../services/project-manager.js';
4
+ export const sprintsRouter = Router();
5
+ // GET /api/projects/:id/sprints
6
+ sprintsRouter.get('/:id/sprints', (req, res) => {
7
+ try {
8
+ const projectId = String(req.params['id']);
9
+ const project = getProject(projectId);
10
+ if (!project) {
11
+ res.status(404).json({ error: 'Project not found' });
12
+ return;
13
+ }
14
+ const sprints = listSprints(projectId);
15
+ res.json({ sprints });
16
+ }
17
+ catch (err) {
18
+ res.status(500).json({ error: 'Internal server error' });
19
+ }
20
+ });
21
+ // POST /api/projects/:id/sprints
22
+ sprintsRouter.post('/:id/sprints', (req, res) => {
23
+ try {
24
+ const projectId = String(req.params['id']);
25
+ const project = getProject(projectId);
26
+ if (!project) {
27
+ res.status(404).json({ error: 'Project not found' });
28
+ return;
29
+ }
30
+ const parsed = createSprintRequestSchema.safeParse(req.body);
31
+ if (!parsed.success) {
32
+ res.status(400).json({ error: 'Invalid request', details: parsed.error.issues });
33
+ return;
34
+ }
35
+ const sprint = createSprint(projectId, parsed.data.name, parsed.data.task_ids);
36
+ res.status(201).json({ sprint });
37
+ }
38
+ catch (err) {
39
+ res.status(500).json({ error: 'Internal server error' });
40
+ }
41
+ });
42
+ // PATCH /api/projects/:projectId/sprints/:sprintId
43
+ sprintsRouter.patch('/:projectId/sprints/:sprintId', (req, res) => {
44
+ try {
45
+ const sprintId = String(req.params['sprintId']);
46
+ const updates = req.body;
47
+ const sprint = updateSprint(sprintId, updates);
48
+ if (!sprint) {
49
+ res.status(404).json({ error: 'Sprint not found' });
50
+ return;
51
+ }
52
+ res.json({ sprint });
53
+ }
54
+ catch (err) {
55
+ res.status(500).json({ error: 'Internal server error' });
56
+ }
57
+ });
58
+ //# sourceMappingURL=sprints.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"sprints.js","sourceRoot":"","sources":["../../src/routes/sprints.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,MAAM,EAAE,MAAM,SAAS,CAAC;AAEjC,OAAO,EAAE,yBAAyB,EAAE,MAAM,mBAAmB,CAAC;AAC9D,OAAO,EACL,WAAW,EACX,YAAY,EACZ,YAAY,EACZ,UAAU,GACX,MAAM,gCAAgC,CAAC;AAExC,MAAM,CAAC,MAAM,aAAa,GAAG,MAAM,EAAE,CAAC;AAEtC,gCAAgC;AAChC,aAAa,CAAC,GAAG,CAAC,cAAc,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;IAChE,IAAI,CAAC;QACH,MAAM,SAAS,GAAG,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC;QAC3C,MAAM,OAAO,GAAG,UAAU,CAAC,SAAS,CAAC,CAAC;QACtC,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,mBAAmB,EAAE,CAAC,CAAC;YACrD,OAAO;QACT,CAAC;QAED,MAAM,OAAO,GAAG,WAAW,CAAC,SAAS,CAAC,CAAC;QACvC,GAAG,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,CAAC,CAAC;IACxB,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,uBAAuB,EAAE,CAAC,CAAC;IAC3D,CAAC;AACH,CAAC,CAAC,CAAC;AAEH,iCAAiC;AACjC,aAAa,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;IACjE,IAAI,CAAC;QACH,MAAM,SAAS,GAAG,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC;QAC3C,MAAM,OAAO,GAAG,UAAU,CAAC,SAAS,CAAC,CAAC;QACtC,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,mBAAmB,EAAE,CAAC,CAAC;YACrD,OAAO;QACT,CAAC;QAED,MAAM,MAAM,GAAG,yBAAyB,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAC7D,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;YACpB,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,iBAAiB,EAAE,OAAO,EAAE,MAAM,CAAC,KAAK,CAAC,MAAM,EAAE,CAAC,CAAC;YACjF,OAAO;QACT,CAAC;QAED,MAAM,MAAM,GAAG,YAAY,CAAC,SAAS,EAAE,MAAM,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAC/E,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC;IACnC,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,uBAAuB,EAAE,CAAC,CAAC;IAC3D,CAAC;AACH,CAAC,CAAC,CAAC;AAEH,mDAAmD;AACnD,aAAa,CAAC,KAAK,CAAC,+BAA+B,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;IACnF,IAAI,CAAC;QACH,MAAM,QAAQ,GAAG,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC;QAChD,MAAM,OAAO,GAAG,GAAG,CAAC,IAInB,CAAC;QAEF,MAAM,MAAM,GAAG,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;QAC/C,IAAI,CAAC,MAAM,EAAE,CAAC;YACZ,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,kBAAkB,EAAE,CAAC,CAAC;YACpD,OAAO;QACT,CAAC;QAED,GAAG,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC;IACvB,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,uBAAuB,EAAE,CAAC,CAAC;IAC3D,CAAC;AACH,CAAC,CAAC,CAAC"}