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,80 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.startDaemon = startDaemon;
7
+ exports.stopDaemon = stopDaemon;
8
+ exports.getDaemonStatus = getDaemonStatus;
9
+ const child_process_1 = require("child_process");
10
+ const path_1 = __importDefault(require("path"));
11
+ const fs_1 = __importDefault(require("fs"));
12
+ const config_1 = require("../src/config");
13
+ const DAEMON_NAME = 'claudectx';
14
+ function startDaemon(workerPath) {
15
+ try {
16
+ // Try to use pm2 if available
17
+ (0, child_process_1.execSync)('which pm2', { stdio: 'pipe' });
18
+ (0, child_process_1.execSync)(`pm2 start ${workerPath} --name ${DAEMON_NAME} --interpreter node`, { stdio: 'inherit' });
19
+ (0, child_process_1.execSync)('pm2 save', { stdio: 'inherit' });
20
+ console.log('Daemon started with pm2');
21
+ }
22
+ catch {
23
+ // pm2 not available — use nohup
24
+ const logPath = path_1.default.join(config_1.CONFIG.logsDir, 'worker.log');
25
+ fs_1.default.mkdirSync(config_1.CONFIG.logsDir, { recursive: true });
26
+ const child = (0, child_process_1.spawn)('node', [workerPath], {
27
+ detached: true,
28
+ stdio: ['ignore', 'ignore', 'ignore'],
29
+ env: { ...process.env }
30
+ });
31
+ child.unref();
32
+ console.log(`Daemon started (PID: ${child.pid}), logs at ${logPath}`);
33
+ // Save PID
34
+ const pidFile = path_1.default.join(config_1.CONFIG.dataDir, 'worker.pid');
35
+ fs_1.default.writeFileSync(pidFile, String(child.pid), 'utf8');
36
+ }
37
+ }
38
+ function stopDaemon() {
39
+ try {
40
+ (0, child_process_1.execSync)('which pm2', { stdio: 'pipe' });
41
+ (0, child_process_1.execSync)(`pm2 stop ${DAEMON_NAME}`, { stdio: 'inherit' });
42
+ console.log('Daemon stopped');
43
+ }
44
+ catch {
45
+ // Try PID file
46
+ const pidFile = path_1.default.join(config_1.CONFIG.dataDir, 'worker.pid');
47
+ if (fs_1.default.existsSync(pidFile)) {
48
+ const pid = fs_1.default.readFileSync(pidFile, 'utf8').trim();
49
+ try {
50
+ process.kill(parseInt(pid), 'SIGTERM');
51
+ fs_1.default.unlinkSync(pidFile);
52
+ console.log('Daemon stopped (PID:', pid, ')');
53
+ }
54
+ catch {
55
+ console.log('Could not stop daemon — may not be running');
56
+ }
57
+ }
58
+ }
59
+ }
60
+ function getDaemonStatus() {
61
+ try {
62
+ (0, child_process_1.execSync)('which pm2', { stdio: 'pipe' });
63
+ const out = (0, child_process_1.execSync)(`pm2 describe ${DAEMON_NAME}`, { encoding: 'utf8', stdio: ['pipe', 'pipe', 'pipe'] });
64
+ return out.includes('online') ? 'running' : 'stopped';
65
+ }
66
+ catch {
67
+ const pidFile = path_1.default.join(config_1.CONFIG.dataDir, 'worker.pid');
68
+ if (!fs_1.default.existsSync(pidFile))
69
+ return 'stopped';
70
+ const pid = parseInt(fs_1.default.readFileSync(pidFile, 'utf8').trim());
71
+ try {
72
+ process.kill(pid, 0);
73
+ return 'running';
74
+ }
75
+ catch {
76
+ return 'stopped';
77
+ }
78
+ }
79
+ }
80
+ //# sourceMappingURL=daemon.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"daemon.js","sourceRoot":"","sources":["../../installer/daemon.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export declare function patchClaudeSettings(): void;
2
+ export declare function removeClaudeHooks(): void;
@@ -0,0 +1,83 @@
1
+ "use strict";
2
+ var __importDefault = (this && this.__importDefault) || function (mod) {
3
+ return (mod && mod.__esModule) ? mod : { "default": mod };
4
+ };
5
+ Object.defineProperty(exports, "__esModule", { value: true });
6
+ exports.patchClaudeSettings = patchClaudeSettings;
7
+ exports.removeClaudeHooks = removeClaudeHooks;
8
+ const fs_1 = __importDefault(require("fs"));
9
+ const path_1 = __importDefault(require("path"));
10
+ const config_1 = require("../src/config");
11
+ function patchClaudeSettings() {
12
+ const settingsPath = config_1.CONFIG.claudeSettingsPath;
13
+ const claudeDir = path_1.default.dirname(settingsPath);
14
+ // Ensure ~/.claude directory exists
15
+ if (!fs_1.default.existsSync(claudeDir)) {
16
+ fs_1.default.mkdirSync(claudeDir, { recursive: true });
17
+ }
18
+ // Read existing settings or start fresh
19
+ let existing = {};
20
+ if (fs_1.default.existsSync(settingsPath)) {
21
+ try {
22
+ existing = JSON.parse(fs_1.default.readFileSync(settingsPath, 'utf8'));
23
+ }
24
+ catch {
25
+ console.warn('Could not parse existing settings.json, will overwrite hooks section');
26
+ }
27
+ }
28
+ // Load our hooks config
29
+ const hooksConfig = JSON.parse(fs_1.default.readFileSync(path_1.default.join(__dirname, 'hooks-settings.json'), 'utf8'));
30
+ // Merge hooks (deep merge by event type)
31
+ if (!existing.hooks) {
32
+ existing.hooks = {};
33
+ }
34
+ for (const [event, eventHooks] of Object.entries(hooksConfig.hooks)) {
35
+ if (!existing.hooks[event]) {
36
+ existing.hooks[event] = eventHooks;
37
+ }
38
+ else {
39
+ // Add our hooks if not already present
40
+ const existingHooks = existing.hooks[event];
41
+ const ourHooks = eventHooks;
42
+ for (const ourHook of ourHooks) {
43
+ const alreadyExists = existingHooks.some((h) => JSON.stringify(h) === JSON.stringify(ourHook));
44
+ if (!alreadyExists) {
45
+ existingHooks.push(ourHook);
46
+ }
47
+ }
48
+ }
49
+ }
50
+ fs_1.default.writeFileSync(settingsPath, JSON.stringify(existing, null, 2), 'utf8');
51
+ console.log('Hooks patched into', settingsPath);
52
+ }
53
+ function removeClaudeHooks() {
54
+ const settingsPath = config_1.CONFIG.claudeSettingsPath;
55
+ if (!fs_1.default.existsSync(settingsPath))
56
+ return;
57
+ try {
58
+ const existing = JSON.parse(fs_1.default.readFileSync(settingsPath, 'utf8'));
59
+ if (!existing.hooks)
60
+ return;
61
+ // Remove our hooks from each event
62
+ for (const event of Object.keys(existing.hooks)) {
63
+ if (Array.isArray(existing.hooks[event])) {
64
+ existing.hooks[event] = existing.hooks[event].filter((h) => {
65
+ if (h.hooks) {
66
+ h.hooks = h.hooks.filter((inner) => !inner.command?.includes('~/.claudectx/hooks/'));
67
+ return h.hooks.length > 0;
68
+ }
69
+ return !h.command?.includes('~/.claudectx/hooks/');
70
+ });
71
+ if (existing.hooks[event].length === 0) {
72
+ delete existing.hooks[event];
73
+ }
74
+ }
75
+ }
76
+ fs_1.default.writeFileSync(settingsPath, JSON.stringify(existing, null, 2), 'utf8');
77
+ console.log('Hooks removed from', settingsPath);
78
+ }
79
+ catch (err) {
80
+ console.error('Could not remove hooks:', err);
81
+ }
82
+ }
83
+ //# sourceMappingURL=patch-settings.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"patch-settings.js","sourceRoot":"","sources":["../../installer/patch-settings.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,29 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ const express_1 = require("express");
4
+ const memory_consolidator_1 = require("../services/memory-consolidator");
5
+ const router = (0, express_1.Router)();
6
+ /**
7
+ * POST /api/consolidate/:projectId
8
+ * Trigger memory consolidation for a project
9
+ */
10
+ router.post('/:projectId', async (req, res) => {
11
+ try {
12
+ const { projectId } = req.params;
13
+ console.log(`[API] Starting memory consolidation for project ${projectId}`);
14
+ const result = await memory_consolidator_1.consolidator.consolidateProject(projectId);
15
+ res.json({
16
+ success: true,
17
+ result
18
+ });
19
+ }
20
+ catch (error) {
21
+ console.error('[API] Consolidation failed:', error);
22
+ res.status(500).json({
23
+ success: false,
24
+ error: error.message
25
+ });
26
+ }
27
+ });
28
+ exports.default = router;
29
+ //# sourceMappingURL=consolidate.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"consolidate.js","sourceRoot":"","sources":["../../../src/api/consolidate.ts"],"names":[],"mappings":";;AAAA,qCAAmD;AACnD,yEAA8D;AAE9D,MAAM,MAAM,GAAW,IAAA,gBAAM,GAAE,CAAA;AAE/B;;;GAGG;AACH,MAAM,CAAC,IAAI,CAAC,aAAa,EAAE,KAAK,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE;IAC5C,IAAI,CAAC;QACH,MAAM,EAAE,SAAS,EAAE,GAAG,GAAG,CAAC,MAAM,CAAA;QAEhC,OAAO,CAAC,GAAG,CAAC,mDAAmD,SAAS,EAAE,CAAC,CAAA;QAC3E,MAAM,MAAM,GAAG,MAAM,kCAAY,CAAC,kBAAkB,CAAC,SAAS,CAAC,CAAA;QAE/D,GAAG,CAAC,IAAI,CAAC;YACP,OAAO,EAAE,IAAI;YACb,MAAM;SACP,CAAC,CAAA;IACJ,CAAC;IAAC,OAAO,KAAU,EAAE,CAAC;QACpB,OAAO,CAAC,KAAK,CAAC,6BAA6B,EAAE,KAAK,CAAC,CAAA;QACnD,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 contextRouter: import("express").Router;
@@ -0,0 +1,26 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.contextRouter = void 0;
4
+ const express_1 = require("express");
5
+ const context_builder_1 = require("../services/context-builder");
6
+ const config_1 = require("../config");
7
+ exports.contextRouter = (0, express_1.Router)();
8
+ async function handleContext(req, res) {
9
+ try {
10
+ const cwd = req.query.cwd || req.body?.cwd;
11
+ const n = parseInt(req.query.n || req.body?.n || String(config_1.CONFIG.defaultContextSessions));
12
+ const session_id = req.query.session_id || req.body?.session_id;
13
+ if (!cwd) {
14
+ res.json({ markdown: '' });
15
+ return;
16
+ }
17
+ const markdown = await (0, context_builder_1.buildContextMarkdown)(cwd, n);
18
+ res.json({ markdown, session_id });
19
+ }
20
+ catch (err) {
21
+ res.json({ markdown: '' });
22
+ }
23
+ }
24
+ exports.contextRouter.get('/', handleContext);
25
+ exports.contextRouter.post('/', handleContext);
26
+ //# sourceMappingURL=context.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context.js","sourceRoot":"","sources":["../../../src/api/context.ts"],"names":[],"mappings":";;;AAAA,qCAAgC;AAChC,iEAAkE;AAClE,sCAAkC;AAErB,QAAA,aAAa,GAA6B,IAAA,gBAAM,GAAE,CAAA;AAE/D,KAAK,UAAU,aAAa,CAAC,GAAQ,EAAE,GAAQ;IAC7C,IAAI,CAAC;QACH,MAAM,GAAG,GAAG,GAAG,CAAC,KAAK,CAAC,GAAG,IAAI,GAAG,CAAC,IAAI,EAAE,GAAG,CAAA;QAC1C,MAAM,CAAC,GAAG,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,IAAI,GAAG,CAAC,IAAI,EAAE,CAAC,IAAI,MAAM,CAAC,eAAM,CAAC,sBAAsB,CAAC,CAAC,CAAA;QACvF,MAAM,UAAU,GAAG,GAAG,CAAC,KAAK,CAAC,UAAU,IAAI,GAAG,CAAC,IAAI,EAAE,UAAU,CAAA;QAE/D,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,GAAG,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,EAAE,EAAE,CAAC,CAAA;YAC1B,OAAM;QACR,CAAC;QAED,MAAM,QAAQ,GAAG,MAAM,IAAA,sCAAoB,EAAC,GAAa,EAAE,CAAC,CAAC,CAAA;QAC7D,GAAG,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,UAAU,EAAE,CAAC,CAAA;IACpC,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,GAAG,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,EAAE,EAAE,CAAC,CAAA;IAC5B,CAAC;AACH,CAAC;AAED,qBAAa,CAAC,GAAG,CAAC,GAAG,EAAE,aAAa,CAAC,CAAA;AACrC,qBAAa,CAAC,IAAI,CAAC,GAAG,EAAE,aAAa,CAAC,CAAA"}
@@ -0,0 +1,2 @@
1
+ import { type Router as RouterType } from 'express';
2
+ export declare const forceEndSessionRouter: RouterType;
@@ -0,0 +1,60 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.forceEndSessionRouter = void 0;
4
+ const express_1 = require("express");
5
+ const queries_1 = require("../db/queries");
6
+ const summarization_queue_1 = require("../services/summarization-queue");
7
+ const broadcast_1 = require("../ws/broadcast");
8
+ const logger_1 = require("../services/logger");
9
+ exports.forceEndSessionRouter = (0, express_1.Router)();
10
+ exports.forceEndSessionRouter.post('/:sessionId', async (req, res) => {
11
+ const { sessionId } = req.params;
12
+ try {
13
+ const session = queries_1.queries.getSession(sessionId);
14
+ if (!session) {
15
+ return res.status(404).json({ error: 'Session not found' });
16
+ }
17
+ // Allow force end if session is active OR if summary is still in progress
18
+ const summaryInProgress = session.summary_status && session.summary_status.toLowerCase() === 'in_progress';
19
+ if (session.status === 'completed' && !summaryInProgress) {
20
+ return res.status(400).json({ error: 'Session already completed with summary' });
21
+ }
22
+ console.log(`[ForceEndSession] Forcing session end: ${sessionId.slice(0, 8)}`);
23
+ // Mark session as completed
24
+ const now = Math.floor(Date.now() / 1000);
25
+ queries_1.queries.updateSession(sessionId, {
26
+ ended_at: session.ended_at || session.last_activity || now,
27
+ status: 'completed'
28
+ });
29
+ // Queue for AI summarization if we have a transcript
30
+ if (session.transcript_path) {
31
+ console.log(`[ForceEndSession] Queuing summarization for: ${sessionId.slice(0, 8)}`);
32
+ summarization_queue_1.summarizationQueue.enqueue({
33
+ sessionId: session.id,
34
+ transcriptPath: session.transcript_path,
35
+ projectId: session.project_id,
36
+ priority: 'high' // Use high priority for manual force end
37
+ });
38
+ }
39
+ else {
40
+ console.log(`[ForceEndSession] No transcript path, skipping summarization`);
41
+ }
42
+ // Broadcast session end event
43
+ (0, broadcast_1.broadcast)({
44
+ type: 'session_end',
45
+ session_id: sessionId,
46
+ reason: 'manual_force_end'
47
+ });
48
+ logger_1.logger.info('ForceEndSession', `Session ${sessionId} manually ended and queued for summarization`);
49
+ res.json({
50
+ success: true,
51
+ message: 'Session marked as completed and queued for summarization',
52
+ session_id: sessionId
53
+ });
54
+ }
55
+ catch (err) {
56
+ console.error('[ForceEndSession] Error:', err);
57
+ res.status(500).json({ error: 'Failed to end session' });
58
+ }
59
+ });
60
+ //# sourceMappingURL=force-end-session.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"force-end-session.js","sourceRoot":"","sources":["../../../src/api/force-end-session.ts"],"names":[],"mappings":";;;AAAA,qCAA2D;AAC3D,2CAAuC;AACvC,yEAAoE;AACpE,+CAA2C;AAC3C,+CAA2C;AAE9B,QAAA,qBAAqB,GAAe,IAAA,gBAAM,GAAE,CAAA;AAEzD,6BAAqB,CAAC,IAAI,CAAC,aAAa,EAAE,KAAK,EAAE,GAAG,EAAE,GAAG,EAAE,EAAE;IAC3D,MAAM,EAAE,SAAS,EAAE,GAAG,GAAG,CAAC,MAAM,CAAA;IAEhC,IAAI,CAAC;QACH,MAAM,OAAO,GAAG,iBAAO,CAAC,UAAU,CAAC,SAAS,CAAC,CAAA;QAE7C,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,OAAO,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,mBAAmB,EAAE,CAAC,CAAA;QAC7D,CAAC;QAED,0EAA0E;QAC1E,MAAM,iBAAiB,GAAG,OAAO,CAAC,cAAc,IAAI,OAAO,CAAC,cAAc,CAAC,WAAW,EAAE,KAAK,aAAa,CAAA;QAC1G,IAAI,OAAO,CAAC,MAAM,KAAK,WAAW,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzD,OAAO,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,wCAAwC,EAAE,CAAC,CAAA;QAClF,CAAC;QAED,OAAO,CAAC,GAAG,CAAC,0CAA0C,SAAS,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,CAAC,CAAA;QAE9E,4BAA4B;QAC5B,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,CAAA;QACzC,iBAAO,CAAC,aAAa,CAAC,SAAS,EAAE;YAC/B,QAAQ,EAAE,OAAO,CAAC,QAAQ,IAAI,OAAO,CAAC,aAAa,IAAI,GAAG;YAC1D,MAAM,EAAE,WAAW;SACpB,CAAC,CAAA;QAEF,qDAAqD;QACrD,IAAI,OAAO,CAAC,eAAe,EAAE,CAAC;YAC5B,OAAO,CAAC,GAAG,CAAC,gDAAgD,SAAS,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,CAAC,CAAA;YACpF,wCAAkB,CAAC,OAAO,CAAC;gBACzB,SAAS,EAAE,OAAO,CAAC,EAAE;gBACrB,cAAc,EAAE,OAAO,CAAC,eAAe;gBACvC,SAAS,EAAE,OAAO,CAAC,UAAU;gBAC7B,QAAQ,EAAE,MAAM,CAAC,yCAAyC;aAC3D,CAAC,CAAA;QACJ,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,GAAG,CAAC,8DAA8D,CAAC,CAAA;QAC7E,CAAC;QAED,8BAA8B;QAC9B,IAAA,qBAAS,EAAC;YACR,IAAI,EAAE,aAAa;YACnB,UAAU,EAAE,SAAS;YACrB,MAAM,EAAE,kBAAkB;SAC3B,CAAC,CAAA;QAEF,eAAM,CAAC,IAAI,CAAC,iBAAiB,EAAE,WAAW,SAAS,8CAA8C,CAAC,CAAA;QAElG,GAAG,CAAC,IAAI,CAAC;YACP,OAAO,EAAE,IAAI;YACb,OAAO,EAAE,0DAA0D;YACnE,UAAU,EAAE,SAAS;SACtB,CAAC,CAAA;IACJ,CAAC;IAAC,OAAO,GAAG,EAAE,CAAC;QACb,OAAO,CAAC,KAAK,CAAC,0BAA0B,EAAE,GAAG,CAAC,CAAA;QAC9C,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,uBAAuB,EAAE,CAAC,CAAA;IAC1D,CAAC;AACH,CAAC,CAAC,CAAA"}
@@ -0,0 +1 @@
1
+ export declare const healthRouter: import("express").Router;
@@ -0,0 +1,27 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.healthRouter = void 0;
4
+ const express_1 = require("express");
5
+ const config_1 = require("../config");
6
+ const client_1 = require("../db/client");
7
+ const queue_1 = require("../services/queue");
8
+ exports.healthRouter = (0, express_1.Router)();
9
+ const startTime = Date.now();
10
+ exports.healthRouter.get('/', (_req, res) => {
11
+ let dbConnected = false;
12
+ try {
13
+ (0, client_1.getDB)().prepare('SELECT 1').get();
14
+ dbConnected = true;
15
+ }
16
+ catch { }
17
+ res.json({
18
+ status: 'ok',
19
+ version: '1.0.0',
20
+ db: dbConnected ? 'connected' : 'error',
21
+ api_key: !!config_1.CONFIG.apiKey,
22
+ summaries_enabled: !config_1.CONFIG.disableSummaries,
23
+ uptime: Math.floor((Date.now() - startTime) / 1000),
24
+ queue_size: (0, queue_1.getQueueSize)()
25
+ });
26
+ });
27
+ //# sourceMappingURL=health.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"health.js","sourceRoot":"","sources":["../../../src/api/health.ts"],"names":[],"mappings":";;;AAAA,qCAAgC;AAChC,sCAAkC;AAClC,yCAAoC;AACpC,6CAAgD;AAEnC,QAAA,YAAY,GAA6B,IAAA,gBAAM,GAAE,CAAA;AAC9D,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAA;AAE5B,oBAAY,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,EAAE,GAAG,EAAE,EAAE;IAClC,IAAI,WAAW,GAAG,KAAK,CAAA;IACvB,IAAI,CAAC;QACH,IAAA,cAAK,GAAE,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,GAAG,EAAE,CAAA;QACjC,WAAW,GAAG,IAAI,CAAA;IACpB,CAAC;IAAC,MAAM,CAAC,CAAA,CAAC;IAEV,GAAG,CAAC,IAAI,CAAC;QACP,MAAM,EAAE,IAAI;QACZ,OAAO,EAAE,OAAO;QAChB,EAAE,EAAE,WAAW,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,OAAO;QACvC,OAAO,EAAE,CAAC,CAAC,eAAM,CAAC,MAAM;QACxB,iBAAiB,EAAE,CAAC,eAAM,CAAC,gBAAgB;QAC3C,MAAM,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,SAAS,CAAC,GAAG,IAAI,CAAC;QACnD,UAAU,EAAE,IAAA,oBAAY,GAAE;KAC3B,CAAC,CAAA;AACJ,CAAC,CAAC,CAAA"}
@@ -0,0 +1,2 @@
1
+ import { type Router as RouterType } from 'express';
2
+ export declare const hookRouter: RouterType;
@@ -0,0 +1,187 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.hookRouter = void 0;
4
+ const express_1 = require("express");
5
+ const project_detector_1 = require("../services/project-detector");
6
+ const queue_1 = require("../services/queue");
7
+ const summarizer_1 = require("../services/summarizer");
8
+ const broadcast_1 = require("../ws/broadcast");
9
+ const queries_1 = require("../db/queries");
10
+ const summarization_queue_1 = require("../services/summarization-queue");
11
+ const logger_1 = require("../services/logger");
12
+ exports.hookRouter = (0, express_1.Router)();
13
+ exports.hookRouter.post('/', async (req, res) => {
14
+ console.log('[Hook] Received event:', req.body.event, 'session:', req.body.session_id?.slice(0, 8));
15
+ res.json({ ok: true }); // Always respond immediately
16
+ const { event, session_id, cwd, ...data } = req.body;
17
+ if (!session_id || !cwd) {
18
+ console.log('[Hook] Missing session_id or cwd, ignoring');
19
+ return;
20
+ }
21
+ try {
22
+ const project = await (0, project_detector_1.detectProject)(cwd);
23
+ console.log('[Hook] Project detected:', project.name, 'id:', project.id.slice(0, 8));
24
+ // Ensure session exists for all events (not just SessionStart)
25
+ const existingSession = queries_1.queries.getSession(session_id);
26
+ if (!existingSession) {
27
+ console.log('[Hook] Creating new session:', session_id.slice(0, 8));
28
+ queries_1.queries.upsertSession({
29
+ id: session_id,
30
+ project_id: project.id,
31
+ started_at: Math.floor(Date.now() / 1000),
32
+ status: 'active'
33
+ });
34
+ }
35
+ switch (event) {
36
+ case 'SessionStart': {
37
+ console.log('[Hook] SessionStart:', session_id.slice(0, 8));
38
+ const existingSession = queries_1.queries.getSession(session_id);
39
+ // If resuming a completed session, reactivate it
40
+ if (existingSession && existingSession.status === 'completed') {
41
+ console.log('[Hook] Resuming completed session - reactivating');
42
+ queries_1.queries.updateSession(session_id, {
43
+ status: 'active',
44
+ ended_at: null, // Clear ended_at since session is active again
45
+ last_activity: Math.floor(Date.now() / 1000)
46
+ });
47
+ }
48
+ else {
49
+ // New session or already active
50
+ queries_1.queries.upsertSession({
51
+ id: session_id,
52
+ project_id: project.id,
53
+ started_at: Math.floor(Date.now() / 1000),
54
+ status: 'active'
55
+ });
56
+ }
57
+ (0, broadcast_1.broadcast)({ type: 'session_start', session_id, project });
58
+ break;
59
+ }
60
+ case 'SessionEnd': {
61
+ console.log('[Hook] SessionEnd:', session_id.slice(0, 8), 'transcript:', data.transcript_path);
62
+ queries_1.queries.updateSession(session_id, {
63
+ ended_at: Math.floor(Date.now() / 1000),
64
+ status: 'completed',
65
+ transcript_path: data.transcript_path || null
66
+ });
67
+ console.log('[Hook] Session marked as completed');
68
+ if (data.transcript_path) {
69
+ logger_1.logger.info('Hook', `SessionEnd received for ${session_id}`, { projectId: project.id });
70
+ console.log('[Hook] Queuing summarization for:', session_id.slice(0, 8));
71
+ summarization_queue_1.summarizationQueue.enqueue({
72
+ sessionId: session_id,
73
+ transcriptPath: data.transcript_path,
74
+ projectId: project.id,
75
+ priority: 'normal'
76
+ });
77
+ }
78
+ (0, broadcast_1.broadcast)({ type: 'session_end', session_id });
79
+ console.log('[Hook] Broadcast session_end event');
80
+ break;
81
+ }
82
+ case 'PostToolUse': {
83
+ console.log('[Hook] PostToolUse:', data.tool_name, 'file:', data.file_path);
84
+ const obs = {
85
+ session_id,
86
+ project_id: project.id,
87
+ event_type: 'tool_call',
88
+ tool_name: data.tool_name,
89
+ file_path: data.file_path || null,
90
+ content: data.command
91
+ ? `${data.tool_name}: ${data.command}`
92
+ : `${data.tool_name}${data.file_path ? ': ' + data.file_path : ''}`,
93
+ metadata: JSON.stringify({ success: data.success })
94
+ };
95
+ queries_1.queries.insertObservation(obs);
96
+ queries_1.queries.incrementTurnStats(session_id, 'tool_calls');
97
+ if (data.file_path) {
98
+ queries_1.queries.addFileTouched(session_id, data.file_path);
99
+ }
100
+ (0, broadcast_1.broadcast)({ type: 'tool_use', session_id, tool_name: data.tool_name, file_path: data.file_path });
101
+ break;
102
+ }
103
+ case 'UserPromptSubmit': {
104
+ console.log('[Hook] UserPromptSubmit:', data.prompt_preview?.slice(0, 50));
105
+ // Log user prompt as observation
106
+ const obs = {
107
+ session_id,
108
+ project_id: project.id,
109
+ event_type: 'user_message',
110
+ tool_name: null,
111
+ file_path: null,
112
+ content: data.prompt_preview || data.prompt || '',
113
+ metadata: JSON.stringify({})
114
+ };
115
+ queries_1.queries.insertObservation(obs);
116
+ queries_1.queries.incrementTurnStats(session_id, 'turns');
117
+ // Update session last_activity timestamp
118
+ queries_1.queries.updateSession(session_id, {
119
+ last_activity: Math.floor(Date.now() / 1000)
120
+ });
121
+ console.log('[Hook] Updated last_activity for session:', session_id.slice(0, 8));
122
+ (0, broadcast_1.broadcast)({ type: 'user_prompt', session_id, preview: data.prompt_preview });
123
+ break;
124
+ }
125
+ case 'Stop': {
126
+ console.log('[Hook] Stop event - session:', session_id.slice(0, 8));
127
+ // Log assistant response as observation
128
+ const obs = {
129
+ session_id,
130
+ project_id: project.id,
131
+ event_type: 'assistant_message',
132
+ tool_name: null,
133
+ file_path: null,
134
+ content: data.message_preview || '',
135
+ metadata: JSON.stringify({})
136
+ };
137
+ queries_1.queries.insertObservation(obs);
138
+ const now = Math.floor(Date.now() / 1000);
139
+ // End the session immediately on Stop event
140
+ const session = queries_1.queries.getSession(session_id);
141
+ if (session && session.status === 'active') {
142
+ console.log('[Hook] Ending session on Stop event:', session_id.slice(0, 8));
143
+ queries_1.queries.updateSession(session_id, {
144
+ ended_at: now,
145
+ status: 'completed',
146
+ last_activity: now
147
+ });
148
+ // Queue for summarization if transcript exists
149
+ if (session.transcript_path) {
150
+ console.log('[Hook] Queuing summarization for:', session_id.slice(0, 8));
151
+ summarization_queue_1.summarizationQueue.enqueue({
152
+ sessionId: session_id,
153
+ transcriptPath: session.transcript_path,
154
+ projectId: project.id,
155
+ priority: 'normal'
156
+ });
157
+ }
158
+ (0, broadcast_1.broadcast)({ type: 'session_end', session_id });
159
+ }
160
+ else {
161
+ // Just update last_activity if already completed
162
+ queries_1.queries.updateSession(session_id, {
163
+ last_activity: now
164
+ });
165
+ }
166
+ (0, broadcast_1.broadcast)({ type: 'stop', session_id, preview: data.message_preview });
167
+ break;
168
+ }
169
+ case 'PreCompact': {
170
+ console.log('[Hook] PreCompact:', session_id.slice(0, 8));
171
+ queries_1.queries.updateSession(session_id, { status: 'compacted' });
172
+ if (data.transcript_path) {
173
+ (0, queue_1.enqueue)(() => (0, summarizer_1.snapshotSession)(session_id, data.transcript_path, project.id));
174
+ }
175
+ (0, broadcast_1.broadcast)({ type: 'pre_compact', session_id });
176
+ break;
177
+ }
178
+ default: {
179
+ console.log('[Hook] Unknown event:', event);
180
+ }
181
+ }
182
+ }
183
+ catch (err) {
184
+ console.error('[Hook] Processing error:', err);
185
+ }
186
+ });
187
+ //# sourceMappingURL=hook.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"hook.js","sourceRoot":"","sources":["../../../src/api/hook.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import { Router } from 'express';
2
+ export declare const logsRouter: Router;
@@ -0,0 +1,66 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.logsRouter = void 0;
4
+ const express_1 = require("express");
5
+ const child_process_1 = require("child_process");
6
+ exports.logsRouter = (0, express_1.Router)();
7
+ const LOG_FILE = '/tmp/memctx.log';
8
+ // Get logs with optional filtering
9
+ exports.logsRouter.get('/', (req, res) => {
10
+ try {
11
+ const lines = parseInt(req.query.lines || '100');
12
+ const level = req.query.level; // info, error, warn, etc.
13
+ const search = req.query.search;
14
+ const minutes = parseInt(req.query.minutes || '0');
15
+ let command = `tail -n ${lines} ${LOG_FILE}`;
16
+ // If minutes filter is set, use time-based filtering
17
+ if (minutes > 0) {
18
+ const cutoffTime = new Date(Date.now() - minutes * 60 * 1000).toISOString();
19
+ command = `tail -n 10000 ${LOG_FILE} | grep -E "\\[${cutoffTime.slice(0, 16)}" || tail -n ${lines} ${LOG_FILE}`;
20
+ }
21
+ const tail = (0, child_process_1.spawn)('sh', ['-c', command]);
22
+ let output = '';
23
+ tail.stdout.on('data', (data) => {
24
+ output += data.toString();
25
+ });
26
+ tail.stderr.on('data', (data) => {
27
+ console.error('Tail error:', data.toString());
28
+ });
29
+ tail.on('close', () => {
30
+ let logs = output.split('\n').filter(line => line.trim());
31
+ // Apply level filter
32
+ if (level) {
33
+ logs = logs.filter(line => line.toLowerCase().includes(`[${level.toLowerCase()}]`));
34
+ }
35
+ // Apply search filter
36
+ if (search) {
37
+ logs = logs.filter(line => line.toLowerCase().includes(search.toLowerCase()));
38
+ }
39
+ res.json({ logs, total: logs.length });
40
+ });
41
+ }
42
+ catch (err) {
43
+ res.status(500).json({ error: String(err) });
44
+ }
45
+ });
46
+ // Stream logs in real-time (SSE)
47
+ exports.logsRouter.get('/stream', (req, res) => {
48
+ res.setHeader('Content-Type', 'text/event-stream');
49
+ res.setHeader('Cache-Control', 'no-cache');
50
+ res.setHeader('Connection', 'keep-alive');
51
+ const tail = (0, child_process_1.spawn)('tail', ['-f', '-n', '50', LOG_FILE]);
52
+ tail.stdout.on('data', (data) => {
53
+ const lines = data.toString().split('\n').filter(line => line.trim());
54
+ lines.forEach(line => {
55
+ res.write(`data: ${JSON.stringify({ log: line, timestamp: Date.now() })}\n\n`);
56
+ });
57
+ });
58
+ tail.stderr.on('data', (data) => {
59
+ console.error('Tail stream error:', data.toString());
60
+ });
61
+ req.on('close', () => {
62
+ tail.kill();
63
+ res.end();
64
+ });
65
+ });
66
+ //# sourceMappingURL=logs.js.map