memctx 1.0.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 (151) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +289 -0
  3. package/dist/bin/claudectx.d.ts +2 -0
  4. package/dist/bin/claudectx.js +304 -0
  5. package/dist/bin/claudectx.js.map +1 -0
  6. package/dist/installer/daemon.d.ts +3 -0
  7. package/dist/installer/daemon.js +80 -0
  8. package/dist/installer/daemon.js.map +1 -0
  9. package/dist/installer/patch-settings.d.ts +2 -0
  10. package/dist/installer/patch-settings.js +83 -0
  11. package/dist/installer/patch-settings.js.map +1 -0
  12. package/dist/src/api/consolidate.d.ts +3 -0
  13. package/dist/src/api/consolidate.js +29 -0
  14. package/dist/src/api/consolidate.js.map +1 -0
  15. package/dist/src/api/context.d.ts +1 -0
  16. package/dist/src/api/context.js +26 -0
  17. package/dist/src/api/context.js.map +1 -0
  18. package/dist/src/api/force-end-session.d.ts +2 -0
  19. package/dist/src/api/force-end-session.js +60 -0
  20. package/dist/src/api/force-end-session.js.map +1 -0
  21. package/dist/src/api/health.d.ts +1 -0
  22. package/dist/src/api/health.js +27 -0
  23. package/dist/src/api/health.js.map +1 -0
  24. package/dist/src/api/hook.d.ts +2 -0
  25. package/dist/src/api/hook.js +187 -0
  26. package/dist/src/api/hook.js.map +1 -0
  27. package/dist/src/api/logs.d.ts +2 -0
  28. package/dist/src/api/logs.js +66 -0
  29. package/dist/src/api/logs.js.map +1 -0
  30. package/dist/src/api/memory.d.ts +2 -0
  31. package/dist/src/api/memory.js +93 -0
  32. package/dist/src/api/memory.js.map +1 -0
  33. package/dist/src/api/metrics.d.ts +3 -0
  34. package/dist/src/api/metrics.js +58 -0
  35. package/dist/src/api/metrics.js.map +1 -0
  36. package/dist/src/api/observations.d.ts +1 -0
  37. package/dist/src/api/observations.js +31 -0
  38. package/dist/src/api/observations.js.map +1 -0
  39. package/dist/src/api/projects.d.ts +1 -0
  40. package/dist/src/api/projects.js +29 -0
  41. package/dist/src/api/projects.js.map +1 -0
  42. package/dist/src/api/resync.d.ts +3 -0
  43. package/dist/src/api/resync.js +188 -0
  44. package/dist/src/api/resync.js.map +1 -0
  45. package/dist/src/api/search.d.ts +1 -0
  46. package/dist/src/api/search.js +36 -0
  47. package/dist/src/api/search.js.map +1 -0
  48. package/dist/src/api/sessions.d.ts +1 -0
  49. package/dist/src/api/sessions.js +137 -0
  50. package/dist/src/api/sessions.js.map +1 -0
  51. package/dist/src/api/settings.d.ts +2 -0
  52. package/dist/src/api/settings.js +90 -0
  53. package/dist/src/api/settings.js.map +1 -0
  54. package/dist/src/api/tags.d.ts +1 -0
  55. package/dist/src/api/tags.js +89 -0
  56. package/dist/src/api/tags.js.map +1 -0
  57. package/dist/src/config.d.ts +17 -0
  58. package/dist/src/config.js +39 -0
  59. package/dist/src/config.js.map +1 -0
  60. package/dist/src/db/client.d.ts +3 -0
  61. package/dist/src/db/client.js +38 -0
  62. package/dist/src/db/client.js.map +1 -0
  63. package/dist/src/db/migrate.d.ts +1 -0
  64. package/dist/src/db/migrate.js +56 -0
  65. package/dist/src/db/migrate.js.map +1 -0
  66. package/dist/src/db/migrations/001_add_memory_tables.sql +149 -0
  67. package/dist/src/db/migrations/002_add_project_id_to_memory.sql +25 -0
  68. package/dist/src/db/migrations/003_enhance_sessions_schema.sql +27 -0
  69. package/dist/src/db/migrations/004_add_bookmarks.sql +5 -0
  70. package/dist/src/db/migrations/005_add_tags.sql +21 -0
  71. package/dist/src/db/migrations/006_add_notes.sql +2 -0
  72. package/dist/src/db/migrations/007_add_archived.sql +2 -0
  73. package/dist/src/db/queries.d.ts +104 -0
  74. package/dist/src/db/queries.js +432 -0
  75. package/dist/src/db/queries.js.map +1 -0
  76. package/dist/src/db/schema.d.ts +1 -0
  77. package/dist/src/db/schema.js +81 -0
  78. package/dist/src/db/schema.js.map +1 -0
  79. package/dist/src/hooks/post-tool-use.d.ts +1 -0
  80. package/dist/src/hooks/post-tool-use.js +23 -0
  81. package/dist/src/hooks/post-tool-use.js.map +1 -0
  82. package/dist/src/hooks/pre-compact.d.ts +1 -0
  83. package/dist/src/hooks/pre-compact.js +18 -0
  84. package/dist/src/hooks/pre-compact.js.map +1 -0
  85. package/dist/src/hooks/session-end.d.ts +1 -0
  86. package/dist/src/hooks/session-end.js +20 -0
  87. package/dist/src/hooks/session-end.js.map +1 -0
  88. package/dist/src/hooks/session-start.d.ts +1 -0
  89. package/dist/src/hooks/session-start.js +32 -0
  90. package/dist/src/hooks/session-start.js.map +1 -0
  91. package/dist/src/hooks/stop.d.ts +1 -0
  92. package/dist/src/hooks/stop.js +22 -0
  93. package/dist/src/hooks/stop.js.map +1 -0
  94. package/dist/src/hooks/user-prompt-submit.d.ts +1 -0
  95. package/dist/src/hooks/user-prompt-submit.js +18 -0
  96. package/dist/src/hooks/user-prompt-submit.js.map +1 -0
  97. package/dist/src/hooks/utils.d.ts +3 -0
  98. package/dist/src/hooks/utils.js +96 -0
  99. package/dist/src/hooks/utils.js.map +1 -0
  100. package/dist/src/index.d.ts +1 -0
  101. package/dist/src/index.js +92 -0
  102. package/dist/src/index.js.map +1 -0
  103. package/dist/src/services/auto-summarizer.d.ts +5 -0
  104. package/dist/src/services/auto-summarizer.js +50 -0
  105. package/dist/src/services/auto-summarizer.js.map +1 -0
  106. package/dist/src/services/claude-md-updater.d.ts +1 -0
  107. package/dist/src/services/claude-md-updater.js +43 -0
  108. package/dist/src/services/claude-md-updater.js.map +1 -0
  109. package/dist/src/services/context-builder.d.ts +1 -0
  110. package/dist/src/services/context-builder.js +97 -0
  111. package/dist/src/services/context-builder.js.map +1 -0
  112. package/dist/src/services/fuzzy-task-matcher.d.ts +37 -0
  113. package/dist/src/services/fuzzy-task-matcher.js +96 -0
  114. package/dist/src/services/fuzzy-task-matcher.js.map +1 -0
  115. package/dist/src/services/logger.d.ts +20 -0
  116. package/dist/src/services/logger.js +43 -0
  117. package/dist/src/services/logger.js.map +1 -0
  118. package/dist/src/services/memory-consolidator.d.ts +32 -0
  119. package/dist/src/services/memory-consolidator.js +192 -0
  120. package/dist/src/services/memory-consolidator.js.map +1 -0
  121. package/dist/src/services/memory-decay.d.ts +16 -0
  122. package/dist/src/services/memory-decay.js +79 -0
  123. package/dist/src/services/memory-decay.js.map +1 -0
  124. package/dist/src/services/metrics.d.ts +58 -0
  125. package/dist/src/services/metrics.js +100 -0
  126. package/dist/src/services/metrics.js.map +1 -0
  127. package/dist/src/services/project-detector.d.ts +5 -0
  128. package/dist/src/services/project-detector.js +43 -0
  129. package/dist/src/services/project-detector.js.map +1 -0
  130. package/dist/src/services/queue.d.ts +2 -0
  131. package/dist/src/services/queue.js +16 -0
  132. package/dist/src/services/queue.js.map +1 -0
  133. package/dist/src/services/session-timeout.d.ts +1 -0
  134. package/dist/src/services/session-timeout.js +50 -0
  135. package/dist/src/services/session-timeout.js.map +1 -0
  136. package/dist/src/services/summarization-queue.d.ts +43 -0
  137. package/dist/src/services/summarization-queue.js +150 -0
  138. package/dist/src/services/summarization-queue.js.map +1 -0
  139. package/dist/src/services/summarizer.d.ts +2 -0
  140. package/dist/src/services/summarizer.js +239 -0
  141. package/dist/src/services/summarizer.js.map +1 -0
  142. package/dist/src/services/transcript-reader.d.ts +9 -0
  143. package/dist/src/services/transcript-reader.js +50 -0
  144. package/dist/src/services/transcript-reader.js.map +1 -0
  145. package/dist/src/services/watcher.d.ts +1 -0
  146. package/dist/src/services/watcher.js +34 -0
  147. package/dist/src/services/watcher.js.map +1 -0
  148. package/dist/src/ws/broadcast.d.ts +3 -0
  149. package/dist/src/ws/broadcast.js +24 -0
  150. package/dist/src/ws/broadcast.js.map +1 -0
  151. package/package.json +66 -0
@@ -0,0 +1 @@
1
+ {"version":3,"file":"logs.js","sourceRoot":"","sources":["../../../src/api/logs.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import type { Router as ExpressRouter } from 'express';
2
+ export declare const memoryRouter: ExpressRouter;
@@ -0,0 +1,93 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.memoryRouter = void 0;
4
+ const express_1 = require("express");
5
+ const queries_1 = require("../db/queries");
6
+ exports.memoryRouter = (0, express_1.Router)();
7
+ exports.memoryRouter.get('/', (req, res) => {
8
+ try {
9
+ const projectId = req.query.project_id;
10
+ if (!projectId) {
11
+ return res.status(400).json({ error: 'project_id is required' });
12
+ }
13
+ const preferences = queries_1.queries.getPreferences(projectId);
14
+ const knowledge = queries_1.queries.getKnowledge(undefined, 50, projectId);
15
+ const patterns = queries_1.queries.getPatterns(undefined, 50, projectId);
16
+ const tasks = queries_1.queries.getTasks(undefined, projectId);
17
+ const contacts = queries_1.queries.getContacts(projectId);
18
+ res.json({
19
+ preferences,
20
+ knowledge,
21
+ patterns,
22
+ tasks,
23
+ contacts,
24
+ stats: {
25
+ total_preferences: preferences.length,
26
+ total_knowledge: knowledge.length,
27
+ total_patterns: patterns.length,
28
+ total_tasks: tasks.length,
29
+ total_contacts: contacts.length
30
+ }
31
+ });
32
+ }
33
+ catch (err) {
34
+ console.error('Memory fetch error:', err);
35
+ res.status(500).json({ error: 'Failed to fetch memory' });
36
+ }
37
+ });
38
+ exports.memoryRouter.get('/preferences', (req, res) => {
39
+ try {
40
+ const projectId = req.query.project_id;
41
+ const prefs = queries_1.queries.getPreferences(projectId);
42
+ res.json(prefs);
43
+ }
44
+ catch (err) {
45
+ res.status(500).json({ error: 'Failed to fetch preferences' });
46
+ }
47
+ });
48
+ exports.memoryRouter.get('/knowledge', (req, res) => {
49
+ try {
50
+ const category = req.query.category;
51
+ const projectId = req.query.project_id;
52
+ const limit = parseInt(req.query.limit) || 50;
53
+ const knowledge = queries_1.queries.getKnowledge(category, limit, projectId);
54
+ res.json(knowledge);
55
+ }
56
+ catch (err) {
57
+ res.status(500).json({ error: 'Failed to fetch knowledge' });
58
+ }
59
+ });
60
+ exports.memoryRouter.get('/patterns', (req, res) => {
61
+ try {
62
+ const type = req.query.type;
63
+ const projectId = req.query.project_id;
64
+ const limit = parseInt(req.query.limit) || 50;
65
+ const patterns = queries_1.queries.getPatterns(type, limit, projectId);
66
+ res.json(patterns);
67
+ }
68
+ catch (err) {
69
+ res.status(500).json({ error: 'Failed to fetch patterns' });
70
+ }
71
+ });
72
+ exports.memoryRouter.get('/tasks', (req, res) => {
73
+ try {
74
+ const status = req.query.status;
75
+ const projectId = req.query.project_id;
76
+ const tasks = queries_1.queries.getTasks(status, projectId);
77
+ res.json(tasks);
78
+ }
79
+ catch (err) {
80
+ res.status(500).json({ error: 'Failed to fetch tasks' });
81
+ }
82
+ });
83
+ exports.memoryRouter.get('/contacts', (req, res) => {
84
+ try {
85
+ const projectId = req.query.project_id;
86
+ const contacts = queries_1.queries.getContacts(projectId);
87
+ res.json(contacts);
88
+ }
89
+ catch (err) {
90
+ res.status(500).json({ error: 'Failed to fetch contacts' });
91
+ }
92
+ });
93
+ //# sourceMappingURL=memory.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"memory.js","sourceRoot":"","sources":["../../../src/api/memory.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,3 @@
1
+ import { Router } from 'express';
2
+ declare const router: Router;
3
+ export default router;
@@ -0,0 +1,58 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const express_1 = require("express");
4
+ const metrics_1 = require("../services/metrics");
5
+ const summarization_queue_1 = require("../services/summarization-queue");
6
+ const queries_1 = require("../db/queries");
7
+ const router = (0, express_1.Router)();
8
+ /**
9
+ * GET /api/metrics
10
+ * Get system metrics
11
+ */
12
+ router.get('/', async (req, res) => {
13
+ try {
14
+ // Update queue metrics
15
+ const queueStats = summarization_queue_1.summarizationQueue.getStats();
16
+ metrics_1.metricsTracker.updateQueueCounts(queueStats.high, queueStats.normal, queueStats.low);
17
+ // Update session metrics
18
+ const allSessions = queries_1.queries.getSessions({});
19
+ const activeSessions = allSessions.filter((s) => s.status === 'active');
20
+ const completedSessions = allSessions.filter((s) => s.status === 'completed');
21
+ metrics_1.metricsTracker.updateSessionCounts(allSessions.length, activeSessions.length, completedSessions.length);
22
+ // Get metrics
23
+ const metrics = metrics_1.metricsTracker.getMetrics();
24
+ res.json({
25
+ success: true,
26
+ metrics,
27
+ timestamp: Date.now()
28
+ });
29
+ }
30
+ catch (error) {
31
+ console.error('[API] Failed to get metrics:', error);
32
+ res.status(500).json({
33
+ success: false,
34
+ error: error.message
35
+ });
36
+ }
37
+ });
38
+ /**
39
+ * POST /api/metrics/reset
40
+ * Reset metrics
41
+ */
42
+ router.post('/reset', (req, res) => {
43
+ try {
44
+ metrics_1.metricsTracker.reset();
45
+ res.json({
46
+ success: true,
47
+ message: 'Metrics reset'
48
+ });
49
+ }
50
+ catch (error) {
51
+ res.status(500).json({
52
+ success: false,
53
+ error: error.message
54
+ });
55
+ }
56
+ });
57
+ exports.default = router;
58
+ //# sourceMappingURL=metrics.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"metrics.js","sourceRoot":"","sources":["../../../src/api/metrics.ts"],"names":[],"mappings":";;AAAA,qCAAmD;AACnD,iDAAoD;AACpD,yEAAoE;AACpE,2CAAuC;AAEvC,MAAM,MAAM,GAAW,IAAA,gBAAM,GAAE,CAAA;AAE/B;;;GAGG;AACH,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,EAAE,GAAY,EAAE,GAAa,EAAE,EAAE;IACpD,IAAI,CAAC;QACH,uBAAuB;QACvB,MAAM,UAAU,GAAG,wCAAkB,CAAC,QAAQ,EAAE,CAAA;QAChD,wBAAc,CAAC,iBAAiB,CAAC,UAAU,CAAC,IAAI,EAAE,UAAU,CAAC,MAAM,EAAE,UAAU,CAAC,GAAG,CAAC,CAAA;QAEpF,yBAAyB;QACzB,MAAM,WAAW,GAAG,iBAAO,CAAC,WAAW,CAAC,EAAE,CAAC,CAAA;QAC3C,MAAM,cAAc,GAAG,WAAW,CAAC,MAAM,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,QAAQ,CAAC,CAAA;QAC5E,MAAM,iBAAiB,GAAG,WAAW,CAAC,MAAM,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,WAAW,CAAC,CAAA;QAClF,wBAAc,CAAC,mBAAmB,CAAC,WAAW,CAAC,MAAM,EAAE,cAAc,CAAC,MAAM,EAAE,iBAAiB,CAAC,MAAM,CAAC,CAAA;QAEvG,cAAc;QACd,MAAM,OAAO,GAAG,wBAAc,CAAC,UAAU,EAAE,CAAA;QAE3C,GAAG,CAAC,IAAI,CAAC;YACP,OAAO,EAAE,IAAI;YACb,OAAO;YACP,SAAS,EAAE,IAAI,CAAC,GAAG,EAAE;SACtB,CAAC,CAAA;IACJ,CAAC;IAAC,OAAO,KAAU,EAAE,CAAC;QACpB,OAAO,CAAC,KAAK,CAAC,8BAA8B,EAAE,KAAK,CAAC,CAAA;QACpD,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC;YACnB,OAAO,EAAE,KAAK;YACd,KAAK,EAAE,KAAK,CAAC,OAAO;SACrB,CAAC,CAAA;IACJ,CAAC;AACH,CAAC,CAAC,CAAA;AAEF;;;GAGG;AACH,MAAM,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,GAAY,EAAE,GAAa,EAAE,EAAE;IACpD,IAAI,CAAC;QACH,wBAAc,CAAC,KAAK,EAAE,CAAA;QACtB,GAAG,CAAC,IAAI,CAAC;YACP,OAAO,EAAE,IAAI;YACb,OAAO,EAAE,eAAe;SACzB,CAAC,CAAA;IACJ,CAAC;IAAC,OAAO,KAAU,EAAE,CAAC;QACpB,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC;YACnB,OAAO,EAAE,KAAK;YACd,KAAK,EAAE,KAAK,CAAC,OAAO;SACrB,CAAC,CAAA;IACJ,CAAC;AACH,CAAC,CAAC,CAAA;AAEF,kBAAe,MAAM,CAAA"}
@@ -0,0 +1 @@
1
+ export declare const observationsRouter: import("express").Router;
@@ -0,0 +1,31 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.observationsRouter = void 0;
4
+ const express_1 = require("express");
5
+ const client_1 = require("../db/client");
6
+ exports.observationsRouter = (0, express_1.Router)();
7
+ exports.observationsRouter.get('/', (req, res) => {
8
+ try {
9
+ const db = (0, client_1.getDB)();
10
+ const session_id = req.query.session_id;
11
+ const limit = parseInt(req.query.limit || '100');
12
+ const offset = parseInt(req.query.offset || '0');
13
+ let rows;
14
+ if (session_id) {
15
+ rows = db.prepare(`
16
+ SELECT * FROM observations WHERE session_id = ?
17
+ ORDER BY created_at ASC LIMIT ? OFFSET ?
18
+ `).all(session_id, limit, offset);
19
+ }
20
+ else {
21
+ rows = db.prepare(`
22
+ SELECT * FROM observations ORDER BY created_at DESC LIMIT ? OFFSET ?
23
+ `).all(limit, offset);
24
+ }
25
+ res.json(rows);
26
+ }
27
+ catch (err) {
28
+ res.status(500).json({ error: String(err) });
29
+ }
30
+ });
31
+ //# sourceMappingURL=observations.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"observations.js","sourceRoot":"","sources":["../../../src/api/observations.ts"],"names":[],"mappings":";;;AAAA,qCAAgC;AAChC,yCAAoC;AAEvB,QAAA,kBAAkB,GAA6B,IAAA,gBAAM,GAAE,CAAA;AAEpE,0BAAkB,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;IACvC,IAAI,CAAC;QACH,MAAM,EAAE,GAAG,IAAA,cAAK,GAAE,CAAA;QAClB,MAAM,UAAU,GAAG,GAAG,CAAC,KAAK,CAAC,UAAoB,CAAA;QACjD,MAAM,KAAK,GAAG,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,KAAe,IAAI,KAAK,CAAC,CAAA;QAC1D,MAAM,MAAM,GAAG,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,MAAgB,IAAI,GAAG,CAAC,CAAA;QAE1D,IAAI,IAAW,CAAA;QACf,IAAI,UAAU,EAAE,CAAC;YACf,IAAI,GAAG,EAAE,CAAC,OAAO,CAAC;;;OAGjB,CAAC,CAAC,GAAG,CAAC,UAAU,EAAE,KAAK,EAAE,MAAM,CAAU,CAAA;QAC5C,CAAC;aAAM,CAAC;YACN,IAAI,GAAG,EAAE,CAAC,OAAO,CAAC;;OAEjB,CAAC,CAAC,GAAG,CAAC,KAAK,EAAE,MAAM,CAAU,CAAA;QAChC,CAAC;QAED,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAA;IAChB,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;IAC9C,CAAC;AACH,CAAC,CAAC,CAAA"}
@@ -0,0 +1 @@
1
+ export declare const projectsRouter: import("express").Router;
@@ -0,0 +1,29 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.projectsRouter = void 0;
4
+ const express_1 = require("express");
5
+ const queries_1 = require("../db/queries");
6
+ exports.projectsRouter = (0, express_1.Router)();
7
+ exports.projectsRouter.get('/', (_req, res) => {
8
+ try {
9
+ const projects = queries_1.queries.getAllProjects();
10
+ res.json(projects);
11
+ }
12
+ catch (err) {
13
+ res.status(500).json({ error: String(err) });
14
+ }
15
+ });
16
+ exports.projectsRouter.get('/:id', (req, res) => {
17
+ try {
18
+ const project = queries_1.queries.getProjectWithSessions(req.params.id);
19
+ if (!project) {
20
+ res.status(404).json({ error: 'Project not found' });
21
+ return;
22
+ }
23
+ res.json(project);
24
+ }
25
+ catch (err) {
26
+ res.status(500).json({ error: String(err) });
27
+ }
28
+ });
29
+ //# sourceMappingURL=projects.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"projects.js","sourceRoot":"","sources":["../../../src/api/projects.ts"],"names":[],"mappings":";;;AAAA,qCAAgC;AAChC,2CAAuC;AAE1B,QAAA,cAAc,GAA6B,IAAA,gBAAM,GAAE,CAAA;AAEhE,sBAAc,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,EAAE,GAAG,EAAE,EAAE;IACpC,IAAI,CAAC;QACH,MAAM,QAAQ,GAAG,iBAAO,CAAC,cAAc,EAAE,CAAA;QACzC,GAAG,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAA;IACpB,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;IAC9C,CAAC;AACH,CAAC,CAAC,CAAA;AAEF,sBAAc,CAAC,GAAG,CAAC,MAAM,EAAE,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;IACtC,IAAI,CAAC;QACH,MAAM,OAAO,GAAG,iBAAO,CAAC,sBAAsB,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,CAAA;QAC7D,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,mBAAmB,EAAE,CAAC,CAAA;YACpD,OAAM;QACR,CAAC;QACD,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC,CAAA;IACnB,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;IAC9C,CAAC;AACH,CAAC,CAAC,CAAA"}
@@ -0,0 +1,3 @@
1
+ import { Router } from 'express';
2
+ declare const router: Router;
3
+ export default router;
@@ -0,0 +1,188 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const express_1 = require("express");
4
+ const queries_1 = require("../db/queries");
5
+ const summarization_queue_1 = require("../services/summarization-queue");
6
+ const logger_1 = require("../services/logger");
7
+ const memory_consolidator_1 = require("../services/memory-consolidator");
8
+ const router = (0, express_1.Router)();
9
+ /**
10
+ * POST /api/resync/all
11
+ * Resync all projects
12
+ * IMPORTANT: Must be defined BEFORE /:projectId route
13
+ */
14
+ router.post('/all', async (req, res) => {
15
+ console.log('[Resync /all] Endpoint hit!');
16
+ try {
17
+ const { force } = req.query;
18
+ console.log('[Resync /all] Force param:', force, 'type:', typeof force);
19
+ logger_1.logger.info('Resync', 'Starting resync for all projects', { force });
20
+ const projects = queries_1.queries.getAllProjects();
21
+ console.log('[Resync /all] getAllProjects returned:', projects.length, 'projects');
22
+ let totalQueued = 0;
23
+ let totalSkipped = 0;
24
+ let totalSessions = 0;
25
+ for (const project of projects) {
26
+ const sessions = queries_1.queries.getSessions({ project_id: project.id, limit: 1000 });
27
+ console.log(`[Resync] Project ${project.name}: ${sessions.length} sessions`);
28
+ totalSessions += sessions.length;
29
+ for (const session of sessions) {
30
+ console.log(`[Resync] Session ${session.id.slice(0, 8)}: status=${session.status}, transcript=${!!session.transcript_path}, summary=${!!session.summary_title}`);
31
+ if (session.status === 'active' || !session.transcript_path) {
32
+ console.log(`[Resync] Skipping ${session.id.slice(0, 8)}: active or no transcript`);
33
+ totalSkipped++;
34
+ continue;
35
+ }
36
+ if (session.summary_title && force !== 'true') {
37
+ console.log(`[Resync] Skipping ${session.id.slice(0, 8)}: has summary and force=${force}`);
38
+ totalSkipped++;
39
+ continue;
40
+ }
41
+ console.log(`[Resync] Queuing ${session.id.slice(0, 8)} for summarization`);
42
+ summarization_queue_1.summarizationQueue.enqueue({
43
+ sessionId: session.id,
44
+ transcriptPath: session.transcript_path,
45
+ projectId: project.id,
46
+ priority: 'low'
47
+ });
48
+ totalQueued++;
49
+ }
50
+ }
51
+ console.log(`[Resync] Complete: ${totalSessions} total, ${totalQueued} queued, ${totalSkipped} skipped`);
52
+ res.json({
53
+ success: true,
54
+ result: {
55
+ projects: projects.length,
56
+ queued: totalQueued,
57
+ skipped: totalSkipped,
58
+ message: `Queued ${totalQueued} sessions across ${projects.length} projects`
59
+ }
60
+ });
61
+ }
62
+ catch (error) {
63
+ logger_1.logger.error('Resync', 'Global resync failed', { error: error.message });
64
+ res.status(500).json({
65
+ success: false,
66
+ error: error.message
67
+ });
68
+ }
69
+ });
70
+ /**
71
+ * POST /api/resync/session/:sessionId
72
+ * Resync a single session
73
+ */
74
+ router.post('/session/:sessionId', async (req, res) => {
75
+ try {
76
+ const { sessionId } = req.params;
77
+ const session = queries_1.queries.getSession(sessionId);
78
+ if (!session) {
79
+ return res.status(404).json({
80
+ success: false,
81
+ error: 'Session not found'
82
+ });
83
+ }
84
+ if (!session.transcript_path) {
85
+ return res.status(400).json({
86
+ success: false,
87
+ error: 'Session has no transcript'
88
+ });
89
+ }
90
+ if (session.status === 'active') {
91
+ return res.status(400).json({
92
+ success: false,
93
+ error: 'Cannot resync active session'
94
+ });
95
+ }
96
+ // Queue for summarization with high priority
97
+ summarization_queue_1.summarizationQueue.enqueue({
98
+ sessionId: session.id,
99
+ transcriptPath: session.transcript_path,
100
+ projectId: session.project_id,
101
+ priority: 'high'
102
+ });
103
+ res.json({
104
+ success: true,
105
+ message: 'Session queued for resync'
106
+ });
107
+ }
108
+ catch (error) {
109
+ logger_1.logger.error('Resync', 'Session resync failed', { error: error.message });
110
+ res.status(500).json({
111
+ success: false,
112
+ error: error.message
113
+ });
114
+ }
115
+ });
116
+ /**
117
+ * POST /api/resync/:projectId
118
+ * Resync all sessions for a project
119
+ * - Regenerate summaries for sessions with transcripts
120
+ * - Extract memory from all summaries
121
+ * - Consolidate memory
122
+ */
123
+ router.post('/:projectId', async (req, res) => {
124
+ try {
125
+ const { projectId } = req.params;
126
+ const { force } = req.query; // force=true to regenerate existing summaries
127
+ logger_1.logger.info('Resync', `Starting resync for project ${projectId}`, { force });
128
+ // Get all sessions for project
129
+ const sessions = queries_1.queries.getSessions({ project_id: projectId, limit: 1000 });
130
+ let queued = 0;
131
+ let skipped = 0;
132
+ for (const session of sessions) {
133
+ // Skip active sessions
134
+ if (session.status === 'active') {
135
+ skipped++;
136
+ continue;
137
+ }
138
+ // Skip sessions without transcripts
139
+ if (!session.transcript_path) {
140
+ skipped++;
141
+ continue;
142
+ }
143
+ // Skip sessions that already have summaries (unless force=true)
144
+ if (session.summary_title && force !== 'true') {
145
+ skipped++;
146
+ continue;
147
+ }
148
+ // Queue for summarization
149
+ summarization_queue_1.summarizationQueue.enqueue({
150
+ sessionId: session.id,
151
+ transcriptPath: session.transcript_path,
152
+ projectId: projectId,
153
+ priority: 'low' // Use low priority for bulk resyncs
154
+ });
155
+ queued++;
156
+ }
157
+ // Consolidate memory after all summaries are done
158
+ // Note: This runs immediately, but summaries are queued
159
+ // In production, you'd want to wait for queue to empty
160
+ setTimeout(async () => {
161
+ try {
162
+ await memory_consolidator_1.consolidator.consolidateProject(projectId);
163
+ logger_1.logger.info('Resync', `Memory consolidation complete for project ${projectId}`);
164
+ }
165
+ catch (error) {
166
+ logger_1.logger.error('Resync', `Memory consolidation failed for project ${projectId}`, { error });
167
+ }
168
+ }, 5000); // Wait 5s for summaries to start processing
169
+ res.json({
170
+ success: true,
171
+ result: {
172
+ total: sessions.length,
173
+ queued,
174
+ skipped,
175
+ message: `Queued ${queued} sessions for resync. Skipped ${skipped} sessions.`
176
+ }
177
+ });
178
+ }
179
+ catch (error) {
180
+ logger_1.logger.error('Resync', 'Resync failed', { error: error.message });
181
+ res.status(500).json({
182
+ success: false,
183
+ error: error.message
184
+ });
185
+ }
186
+ });
187
+ exports.default = router;
188
+ //# sourceMappingURL=resync.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"resync.js","sourceRoot":"","sources":["../../../src/api/resync.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export declare const searchRouter: import("express").Router;
@@ -0,0 +1,36 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.searchRouter = void 0;
4
+ const express_1 = require("express");
5
+ const queries_1 = require("../db/queries");
6
+ exports.searchRouter = (0, express_1.Router)();
7
+ exports.searchRouter.post('/', (req, res) => {
8
+ try {
9
+ const { query, project_id } = req.body;
10
+ if (!query || query.trim().length < 2) {
11
+ res.json({ results: [] });
12
+ return;
13
+ }
14
+ // Escape FTS5 special characters
15
+ const safeQuery = query.trim().replace(/['"*]/g, ' ');
16
+ const results = queries_1.queries.searchObservations(safeQuery, project_id);
17
+ res.json({
18
+ results: results.map(r => ({
19
+ observation_id: r.id,
20
+ content: r.content,
21
+ session_id: r.session_id,
22
+ project_id: r.project_id,
23
+ session_title: r.session_title,
24
+ project_name: r.project_name,
25
+ event_type: r.event_type,
26
+ created_at: r.created_at,
27
+ relevance_rank: r.relevance_rank
28
+ }))
29
+ });
30
+ }
31
+ catch (err) {
32
+ // FTS5 can throw on malformed queries — return empty
33
+ res.json({ results: [], error: String(err) });
34
+ }
35
+ });
36
+ //# sourceMappingURL=search.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"search.js","sourceRoot":"","sources":["../../../src/api/search.ts"],"names":[],"mappings":";;;AAAA,qCAAgC;AAChC,2CAAuC;AAE1B,QAAA,YAAY,GAA6B,IAAA,gBAAM,GAAE,CAAA;AAE9D,oBAAY,CAAC,IAAI,CAAC,GAAG,EAAE,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;IAClC,IAAI,CAAC;QACH,MAAM,EAAE,KAAK,EAAE,UAAU,EAAE,GAAG,GAAG,CAAC,IAAI,CAAA;QACtC,IAAI,CAAC,KAAK,IAAI,KAAK,CAAC,IAAI,EAAE,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACtC,GAAG,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC,CAAA;YACzB,OAAM;QACR,CAAC;QAED,iCAAiC;QACjC,MAAM,SAAS,GAAG,KAAK,CAAC,IAAI,EAAE,CAAC,OAAO,CAAC,QAAQ,EAAE,GAAG,CAAC,CAAA;QACrD,MAAM,OAAO,GAAG,iBAAO,CAAC,kBAAkB,CAAC,SAAS,EAAE,UAAU,CAAC,CAAA;QAEjE,GAAG,CAAC,IAAI,CAAC;YACP,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;gBACzB,cAAc,EAAE,CAAC,CAAC,EAAE;gBACpB,OAAO,EAAE,CAAC,CAAC,OAAO;gBAClB,UAAU,EAAE,CAAC,CAAC,UAAU;gBACxB,UAAU,EAAE,CAAC,CAAC,UAAU;gBACxB,aAAa,EAAE,CAAC,CAAC,aAAa;gBAC9B,YAAY,EAAE,CAAC,CAAC,YAAY;gBAC5B,UAAU,EAAE,CAAC,CAAC,UAAU;gBACxB,UAAU,EAAE,CAAC,CAAC,UAAU;gBACxB,cAAc,EAAE,CAAC,CAAC,cAAc;aACjC,CAAC,CAAC;SACJ,CAAC,CAAA;IACJ,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,qDAAqD;QACrD,GAAG,CAAC,IAAI,CAAC,EAAE,OAAO,EAAE,EAAE,EAAE,KAAK,EAAE,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAA;IAC/C,CAAC;AACH,CAAC,CAAC,CAAA"}
@@ -0,0 +1 @@
1
+ export declare const sessionsRouter: import("express").Router;
@@ -0,0 +1,137 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.sessionsRouter = void 0;
4
+ const express_1 = require("express");
5
+ const queries_1 = require("../db/queries");
6
+ exports.sessionsRouter = (0, express_1.Router)();
7
+ exports.sessionsRouter.get('/', (req, res) => {
8
+ try {
9
+ const sessions = queries_1.queries.getSessions({
10
+ project_id: req.query.project_id,
11
+ limit: parseInt(req.query.limit || '20'),
12
+ offset: parseInt(req.query.offset || '0'),
13
+ status: req.query.status,
14
+ });
15
+ // Parse JSON fields
16
+ const result = sessions.map(s => ({
17
+ ...s,
18
+ summary_what_we_did: tryParse(s.summary_what_we_did),
19
+ summary_decisions: tryParse(s.summary_decisions),
20
+ summary_files_changed: tryParse(s.summary_files_changed),
21
+ summary_next_steps: tryParse(s.summary_next_steps),
22
+ summary_gotchas: tryParse(s.summary_gotchas),
23
+ summary_tech_notes: tryParse(s.summary_tech_notes),
24
+ files_touched: tryParse(s.files_touched),
25
+ tools_used: tryParse(s.tools_used),
26
+ }));
27
+ res.json(result);
28
+ }
29
+ catch (err) {
30
+ res.status(500).json({ error: String(err) });
31
+ }
32
+ });
33
+ exports.sessionsRouter.get('/:id', (req, res) => {
34
+ try {
35
+ const session = queries_1.queries.getSessionWithObservations(req.params.id);
36
+ if (!session) {
37
+ res.status(404).json({ error: 'Session not found' });
38
+ return;
39
+ }
40
+ // Get tags for this session
41
+ const tags = queries_1.queries.getSessionTags(req.params.id);
42
+ res.json({
43
+ ...session,
44
+ summary_what_we_did: tryParse(session.summary_what_we_did),
45
+ summary_decisions: tryParse(session.summary_decisions),
46
+ summary_files_changed: tryParse(session.summary_files_changed),
47
+ summary_next_steps: tryParse(session.summary_next_steps),
48
+ summary_gotchas: tryParse(session.summary_gotchas),
49
+ summary_tech_notes: tryParse(session.summary_tech_notes),
50
+ files_touched: tryParse(session.files_touched),
51
+ tools_used: tryParse(session.tools_used),
52
+ tags,
53
+ });
54
+ }
55
+ catch (err) {
56
+ res.status(500).json({ error: String(err) });
57
+ }
58
+ });
59
+ exports.sessionsRouter.delete('/:id', (req, res) => {
60
+ try {
61
+ const session = queries_1.queries.getSession(req.params.id);
62
+ if (!session) {
63
+ res.status(404).json({ error: 'Session not found' });
64
+ return;
65
+ }
66
+ // Delete session and all related data (observations, memory)
67
+ queries_1.queries.deleteSession(req.params.id);
68
+ console.log(`[API] Deleted session: ${req.params.id.slice(0, 8)}`);
69
+ res.json({ success: true, message: 'Session deleted' });
70
+ }
71
+ catch (err) {
72
+ console.error('[API] Error deleting session:', err);
73
+ res.status(500).json({ error: String(err) });
74
+ }
75
+ });
76
+ exports.sessionsRouter.post('/:id/bookmark', (req, res) => {
77
+ try {
78
+ const { bookmarked } = req.body;
79
+ const session = queries_1.queries.getSession(req.params.id);
80
+ if (!session) {
81
+ res.status(404).json({ error: 'Session not found' });
82
+ return;
83
+ }
84
+ queries_1.queries.updateSessionBookmark(req.params.id, bookmarked ? 1 : 0);
85
+ console.log(`[API] ${bookmarked ? 'Bookmarked' : 'Unbookmarked'} session: ${req.params.id.slice(0, 8)}`);
86
+ res.json({ success: true });
87
+ }
88
+ catch (err) {
89
+ console.error('[API] Error updating bookmark:', err);
90
+ res.status(500).json({ error: String(err) });
91
+ }
92
+ });
93
+ exports.sessionsRouter.post('/:id/notes', (req, res) => {
94
+ try {
95
+ const { notes } = req.body;
96
+ const session = queries_1.queries.getSession(req.params.id);
97
+ if (!session) {
98
+ res.status(404).json({ error: 'Session not found' });
99
+ return;
100
+ }
101
+ queries_1.queries.updateSessionNotes(req.params.id, notes || '');
102
+ console.log(`[API] Updated notes for session: ${req.params.id.slice(0, 8)}`);
103
+ res.json({ success: true });
104
+ }
105
+ catch (err) {
106
+ console.error('[API] Error updating notes:', err);
107
+ res.status(500).json({ error: String(err) });
108
+ }
109
+ });
110
+ exports.sessionsRouter.post('/:id/archive', (req, res) => {
111
+ try {
112
+ const { archived } = req.body;
113
+ const session = queries_1.queries.getSession(req.params.id);
114
+ if (!session) {
115
+ res.status(404).json({ error: 'Session not found' });
116
+ return;
117
+ }
118
+ queries_1.queries.updateSessionArchived(req.params.id, archived ? 1 : 0);
119
+ console.log(`[API] ${archived ? 'Archived' : 'Unarchived'} session: ${req.params.id.slice(0, 8)}`);
120
+ res.json({ success: true });
121
+ }
122
+ catch (err) {
123
+ console.error('[API] Error updating archive status:', err);
124
+ res.status(500).json({ error: String(err) });
125
+ }
126
+ });
127
+ function tryParse(val) {
128
+ if (!val)
129
+ return null;
130
+ try {
131
+ return JSON.parse(val);
132
+ }
133
+ catch {
134
+ return val;
135
+ }
136
+ }
137
+ //# sourceMappingURL=sessions.js.map