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.
- package/LICENSE +21 -0
- package/README.md +289 -0
- package/dist/bin/claudectx.d.ts +2 -0
- package/dist/bin/claudectx.js +304 -0
- package/dist/bin/claudectx.js.map +1 -0
- package/dist/installer/daemon.d.ts +3 -0
- package/dist/installer/daemon.js +80 -0
- package/dist/installer/daemon.js.map +1 -0
- package/dist/installer/patch-settings.d.ts +2 -0
- package/dist/installer/patch-settings.js +83 -0
- package/dist/installer/patch-settings.js.map +1 -0
- package/dist/src/api/consolidate.d.ts +3 -0
- package/dist/src/api/consolidate.js +29 -0
- package/dist/src/api/consolidate.js.map +1 -0
- package/dist/src/api/context.d.ts +1 -0
- package/dist/src/api/context.js +26 -0
- package/dist/src/api/context.js.map +1 -0
- package/dist/src/api/force-end-session.d.ts +2 -0
- package/dist/src/api/force-end-session.js +60 -0
- package/dist/src/api/force-end-session.js.map +1 -0
- package/dist/src/api/health.d.ts +1 -0
- package/dist/src/api/health.js +27 -0
- package/dist/src/api/health.js.map +1 -0
- package/dist/src/api/hook.d.ts +2 -0
- package/dist/src/api/hook.js +187 -0
- package/dist/src/api/hook.js.map +1 -0
- package/dist/src/api/logs.d.ts +2 -0
- package/dist/src/api/logs.js +66 -0
- package/dist/src/api/logs.js.map +1 -0
- package/dist/src/api/memory.d.ts +2 -0
- package/dist/src/api/memory.js +93 -0
- package/dist/src/api/memory.js.map +1 -0
- package/dist/src/api/metrics.d.ts +3 -0
- package/dist/src/api/metrics.js +58 -0
- package/dist/src/api/metrics.js.map +1 -0
- package/dist/src/api/observations.d.ts +1 -0
- package/dist/src/api/observations.js +31 -0
- package/dist/src/api/observations.js.map +1 -0
- package/dist/src/api/projects.d.ts +1 -0
- package/dist/src/api/projects.js +29 -0
- package/dist/src/api/projects.js.map +1 -0
- package/dist/src/api/resync.d.ts +3 -0
- package/dist/src/api/resync.js +188 -0
- package/dist/src/api/resync.js.map +1 -0
- package/dist/src/api/search.d.ts +1 -0
- package/dist/src/api/search.js +36 -0
- package/dist/src/api/search.js.map +1 -0
- package/dist/src/api/sessions.d.ts +1 -0
- package/dist/src/api/sessions.js +137 -0
- package/dist/src/api/sessions.js.map +1 -0
- package/dist/src/api/settings.d.ts +2 -0
- package/dist/src/api/settings.js +90 -0
- package/dist/src/api/settings.js.map +1 -0
- package/dist/src/api/tags.d.ts +1 -0
- package/dist/src/api/tags.js +89 -0
- package/dist/src/api/tags.js.map +1 -0
- package/dist/src/config.d.ts +17 -0
- package/dist/src/config.js +39 -0
- package/dist/src/config.js.map +1 -0
- package/dist/src/db/client.d.ts +3 -0
- package/dist/src/db/client.js +38 -0
- package/dist/src/db/client.js.map +1 -0
- package/dist/src/db/migrate.d.ts +1 -0
- package/dist/src/db/migrate.js +56 -0
- package/dist/src/db/migrate.js.map +1 -0
- package/dist/src/db/migrations/001_add_memory_tables.sql +149 -0
- package/dist/src/db/migrations/002_add_project_id_to_memory.sql +25 -0
- package/dist/src/db/migrations/003_enhance_sessions_schema.sql +27 -0
- package/dist/src/db/migrations/004_add_bookmarks.sql +5 -0
- package/dist/src/db/migrations/005_add_tags.sql +21 -0
- package/dist/src/db/migrations/006_add_notes.sql +2 -0
- package/dist/src/db/migrations/007_add_archived.sql +2 -0
- package/dist/src/db/queries.d.ts +104 -0
- package/dist/src/db/queries.js +432 -0
- package/dist/src/db/queries.js.map +1 -0
- package/dist/src/db/schema.d.ts +1 -0
- package/dist/src/db/schema.js +81 -0
- package/dist/src/db/schema.js.map +1 -0
- package/dist/src/hooks/post-tool-use.d.ts +1 -0
- package/dist/src/hooks/post-tool-use.js +23 -0
- package/dist/src/hooks/post-tool-use.js.map +1 -0
- package/dist/src/hooks/pre-compact.d.ts +1 -0
- package/dist/src/hooks/pre-compact.js +18 -0
- package/dist/src/hooks/pre-compact.js.map +1 -0
- package/dist/src/hooks/session-end.d.ts +1 -0
- package/dist/src/hooks/session-end.js +20 -0
- package/dist/src/hooks/session-end.js.map +1 -0
- package/dist/src/hooks/session-start.d.ts +1 -0
- package/dist/src/hooks/session-start.js +32 -0
- package/dist/src/hooks/session-start.js.map +1 -0
- package/dist/src/hooks/stop.d.ts +1 -0
- package/dist/src/hooks/stop.js +22 -0
- package/dist/src/hooks/stop.js.map +1 -0
- package/dist/src/hooks/user-prompt-submit.d.ts +1 -0
- package/dist/src/hooks/user-prompt-submit.js +18 -0
- package/dist/src/hooks/user-prompt-submit.js.map +1 -0
- package/dist/src/hooks/utils.d.ts +3 -0
- package/dist/src/hooks/utils.js +96 -0
- package/dist/src/hooks/utils.js.map +1 -0
- package/dist/src/index.d.ts +1 -0
- package/dist/src/index.js +92 -0
- package/dist/src/index.js.map +1 -0
- package/dist/src/services/auto-summarizer.d.ts +5 -0
- package/dist/src/services/auto-summarizer.js +50 -0
- package/dist/src/services/auto-summarizer.js.map +1 -0
- package/dist/src/services/claude-md-updater.d.ts +1 -0
- package/dist/src/services/claude-md-updater.js +43 -0
- package/dist/src/services/claude-md-updater.js.map +1 -0
- package/dist/src/services/context-builder.d.ts +1 -0
- package/dist/src/services/context-builder.js +97 -0
- package/dist/src/services/context-builder.js.map +1 -0
- package/dist/src/services/fuzzy-task-matcher.d.ts +37 -0
- package/dist/src/services/fuzzy-task-matcher.js +96 -0
- package/dist/src/services/fuzzy-task-matcher.js.map +1 -0
- package/dist/src/services/logger.d.ts +20 -0
- package/dist/src/services/logger.js +43 -0
- package/dist/src/services/logger.js.map +1 -0
- package/dist/src/services/memory-consolidator.d.ts +32 -0
- package/dist/src/services/memory-consolidator.js +192 -0
- package/dist/src/services/memory-consolidator.js.map +1 -0
- package/dist/src/services/memory-decay.d.ts +16 -0
- package/dist/src/services/memory-decay.js +79 -0
- package/dist/src/services/memory-decay.js.map +1 -0
- package/dist/src/services/metrics.d.ts +58 -0
- package/dist/src/services/metrics.js +100 -0
- package/dist/src/services/metrics.js.map +1 -0
- package/dist/src/services/project-detector.d.ts +5 -0
- package/dist/src/services/project-detector.js +43 -0
- package/dist/src/services/project-detector.js.map +1 -0
- package/dist/src/services/queue.d.ts +2 -0
- package/dist/src/services/queue.js +16 -0
- package/dist/src/services/queue.js.map +1 -0
- package/dist/src/services/session-timeout.d.ts +1 -0
- package/dist/src/services/session-timeout.js +50 -0
- package/dist/src/services/session-timeout.js.map +1 -0
- package/dist/src/services/summarization-queue.d.ts +43 -0
- package/dist/src/services/summarization-queue.js +150 -0
- package/dist/src/services/summarization-queue.js.map +1 -0
- package/dist/src/services/summarizer.d.ts +2 -0
- package/dist/src/services/summarizer.js +239 -0
- package/dist/src/services/summarizer.js.map +1 -0
- package/dist/src/services/transcript-reader.d.ts +9 -0
- package/dist/src/services/transcript-reader.js +50 -0
- package/dist/src/services/transcript-reader.js.map +1 -0
- package/dist/src/services/watcher.d.ts +1 -0
- package/dist/src/services/watcher.js +34 -0
- package/dist/src/services/watcher.js.map +1 -0
- package/dist/src/ws/broadcast.d.ts +3 -0
- package/dist/src/ws/broadcast.js +24 -0
- package/dist/src/ws/broadcast.js.map +1 -0
- 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,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,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,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,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,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
|