Merge remote-tracking branch 'origin/main' into fix/chroma-mcp-spawn-storm
# Conflicts: # src/services/worker-service.ts # tests/infrastructure/process-manager.test.ts
This commit is contained in:
@@ -10,7 +10,7 @@
|
|||||||
"plugins": [
|
"plugins": [
|
||||||
{
|
{
|
||||||
"name": "claude-mem",
|
"name": "claude-mem",
|
||||||
"version": "10.0.1",
|
"version": "10.0.2",
|
||||||
"source": "./plugin",
|
"source": "./plugin",
|
||||||
"description": "Persistent memory system for Claude Code - context compression across sessions"
|
"description": "Persistent memory system for Claude Code - context compression across sessions"
|
||||||
}
|
}
|
||||||
|
|||||||
+21
-12
@@ -2,6 +2,27 @@
|
|||||||
|
|
||||||
All notable changes to claude-mem.
|
All notable changes to claude-mem.
|
||||||
|
|
||||||
|
## [v10.0.2] - 2026-02-11
|
||||||
|
|
||||||
|
## Bug Fixes
|
||||||
|
|
||||||
|
- **Prevent daemon silent death from SIGHUP + unhandled errors** — Worker process could silently die when receiving SIGHUP signals or encountering unhandled errors, leaving hooks without a backend. Now properly handles these signals and prevents silent crashes.
|
||||||
|
- **Hook resilience and worker lifecycle improvements** — Comprehensive fixes for hook command error classification, addressing issues #957, #923, #984, #987, and #1042. Hooks now correctly distinguish between worker unavailability errors and other failures.
|
||||||
|
- **Clarify TypeError order dependency in error classifier** — Fixed error classification logic to properly handle TypeError ordering edge cases.
|
||||||
|
|
||||||
|
## New Features
|
||||||
|
|
||||||
|
- **Project-scoped statusline counter utility** — Added `statusline-counts.js` for tracking observation counts per project in the Claude Code status line.
|
||||||
|
|
||||||
|
## Internal
|
||||||
|
|
||||||
|
- Added test coverage for hook command error classification and process manager
|
||||||
|
- Worker service and MCP server lifecycle improvements
|
||||||
|
- Process manager enhancements for better cross-platform stability
|
||||||
|
|
||||||
|
### Contributors
|
||||||
|
- @rodboev — Hook resilience and worker lifecycle fixes (PR #1056)
|
||||||
|
|
||||||
## [v10.0.1] - 2026-02-11
|
## [v10.0.1] - 2026-02-11
|
||||||
|
|
||||||
## What's Changed
|
## What's Changed
|
||||||
@@ -1494,15 +1515,3 @@ Set in ~/.claude-mem/settings.json:
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
## [v8.0.1] - 2025-12-23
|
|
||||||
|
|
||||||
## 🎨 UI Improvements
|
|
||||||
|
|
||||||
- **Header Redesign**: Moved documentation and X (Twitter) links from settings modal to main header for better accessibility
|
|
||||||
- **Removed Product Hunt Badge**: Cleaned up header layout by removing the Product Hunt badge
|
|
||||||
- **Icon Reorganization**: Reordered header icons for improved UX flow (Docs → X → Discord → GitHub)
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
🤖 Generated with [Claude Code](https://claude.com/claude-code)
|
|
||||||
|
|
||||||
|
|||||||
+1
-1
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "claude-mem",
|
"name": "claude-mem",
|
||||||
"version": "10.0.1",
|
"version": "10.0.2",
|
||||||
"description": "Memory compression system for Claude Code - persist context across sessions",
|
"description": "Memory compression system for Claude Code - persist context across sessions",
|
||||||
"keywords": [
|
"keywords": [
|
||||||
"claude",
|
"claude",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "claude-mem",
|
"name": "claude-mem",
|
||||||
"version": "10.0.1",
|
"version": "10.0.2",
|
||||||
"description": "Persistent memory system for Claude Code - seamlessly preserve context across sessions",
|
"description": "Persistent memory system for Claude Code - seamlessly preserve context across sessions",
|
||||||
"author": {
|
"author": {
|
||||||
"name": "Alex Newman"
|
"name": "Alex Newman"
|
||||||
|
|||||||
+1
-1
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "claude-mem-plugin",
|
"name": "claude-mem-plugin",
|
||||||
"version": "10.0.1",
|
"version": "10.0.2",
|
||||||
"private": true,
|
"private": true,
|
||||||
"description": "Runtime dependencies for claude-mem bundled hooks",
|
"description": "Runtime dependencies for claude-mem bundled hooks",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
|
|||||||
File diff suppressed because one or more lines are too long
Executable
+61
@@ -0,0 +1,61 @@
|
|||||||
|
#!/usr/bin/env bun
|
||||||
|
/**
|
||||||
|
* Statusline Counts — lightweight project-scoped observation counter
|
||||||
|
*
|
||||||
|
* Returns JSON with observation and prompt counts for the given project,
|
||||||
|
* suitable for integration into Claude Code's statusLineCommand.
|
||||||
|
*
|
||||||
|
* Usage:
|
||||||
|
* bun statusline-counts.js <cwd>
|
||||||
|
* bun statusline-counts.js /home/user/my-project
|
||||||
|
*
|
||||||
|
* Output (JSON, stdout):
|
||||||
|
* {"observations": 42, "prompts": 15, "project": "my-project"}
|
||||||
|
*
|
||||||
|
* The project name is derived from basename(cwd). Observations are counted
|
||||||
|
* with a WHERE project = ? filter so only the current project's data is
|
||||||
|
* returned — preventing inflated counts from cross-project observations.
|
||||||
|
*
|
||||||
|
* Performance: ~10ms typical (direct SQLite read, no HTTP, no worker dependency)
|
||||||
|
*/
|
||||||
|
import { Database } from "bun:sqlite";
|
||||||
|
import { existsSync, readFileSync } from "fs";
|
||||||
|
import { homedir } from "os";
|
||||||
|
import { join, basename } from "path";
|
||||||
|
|
||||||
|
const cwd = process.argv[2] || process.env.CLAUDE_CWD || process.cwd();
|
||||||
|
const project = basename(cwd);
|
||||||
|
|
||||||
|
try {
|
||||||
|
// Resolve data directory: env var → settings.json → default
|
||||||
|
let dataDir = process.env.CLAUDE_MEM_DATA_DIR || join(homedir(), ".claude-mem");
|
||||||
|
if (!process.env.CLAUDE_MEM_DATA_DIR) {
|
||||||
|
const settingsPath = join(dataDir, "settings.json");
|
||||||
|
if (existsSync(settingsPath)) {
|
||||||
|
try {
|
||||||
|
const settings = JSON.parse(readFileSync(settingsPath, "utf-8"));
|
||||||
|
if (settings.CLAUDE_MEM_DATA_DIR) dataDir = settings.CLAUDE_MEM_DATA_DIR;
|
||||||
|
} catch { /* use default */ }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const dbPath = join(dataDir, "claude-mem.db");
|
||||||
|
if (!existsSync(dbPath)) {
|
||||||
|
console.log(JSON.stringify({ observations: 0, prompts: 0, project }));
|
||||||
|
process.exit(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
const db = new Database(dbPath, { readonly: true });
|
||||||
|
|
||||||
|
const obs = db.query("SELECT COUNT(*) as c FROM observations WHERE project = ?").get(project);
|
||||||
|
// user_prompts links to projects through sdk_sessions.content_session_id
|
||||||
|
const prompts = db.query(
|
||||||
|
`SELECT COUNT(*) as c FROM user_prompts up
|
||||||
|
JOIN sdk_sessions s ON s.content_session_id = up.content_session_id
|
||||||
|
WHERE s.project = ?`
|
||||||
|
).get(project);
|
||||||
|
console.log(JSON.stringify({ observations: obs.c, prompts: prompts.c, project }));
|
||||||
|
db.close();
|
||||||
|
} catch (e) {
|
||||||
|
console.log(JSON.stringify({ observations: 0, prompts: 0, project, error: e.message }));
|
||||||
|
}
|
||||||
+191
-191
File diff suppressed because one or more lines are too long
+28
-13
@@ -9,6 +9,7 @@ import type { EventHandler, NormalizedHookInput, HookResult } from '../types.js'
|
|||||||
import { ensureWorkerRunning, getWorkerPort } from '../../shared/worker-utils.js';
|
import { ensureWorkerRunning, getWorkerPort } from '../../shared/worker-utils.js';
|
||||||
import { getProjectContext } from '../../utils/project-name.js';
|
import { getProjectContext } from '../../utils/project-name.js';
|
||||||
import { HOOK_EXIT_CODES } from '../../shared/hook-constants.js';
|
import { HOOK_EXIT_CODES } from '../../shared/hook-constants.js';
|
||||||
|
import { logger } from '../../utils/logger.js';
|
||||||
|
|
||||||
export const contextHandler: EventHandler = {
|
export const contextHandler: EventHandler = {
|
||||||
async execute(input: NormalizedHookInput): Promise<HookResult> {
|
async execute(input: NormalizedHookInput): Promise<HookResult> {
|
||||||
@@ -35,20 +36,34 @@ export const contextHandler: EventHandler = {
|
|||||||
|
|
||||||
// Note: Removed AbortSignal.timeout due to Windows Bun cleanup issue (libuv assertion)
|
// Note: Removed AbortSignal.timeout due to Windows Bun cleanup issue (libuv assertion)
|
||||||
// Worker service has its own timeouts, so client-side timeout is redundant
|
// Worker service has its own timeouts, so client-side timeout is redundant
|
||||||
const response = await fetch(url);
|
try {
|
||||||
|
const response = await fetch(url);
|
||||||
|
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
throw new Error(`Context generation failed: ${response.status}`);
|
// Log but don't throw — context fetch failure should not block session start
|
||||||
}
|
logger.warn('HOOK', 'Context generation failed, returning empty', { status: response.status });
|
||||||
|
return {
|
||||||
const result = await response.text();
|
hookSpecificOutput: { hookEventName: 'SessionStart', additionalContext: '' },
|
||||||
const additionalContext = result.trim();
|
exitCode: HOOK_EXIT_CODES.SUCCESS
|
||||||
|
};
|
||||||
return {
|
|
||||||
hookSpecificOutput: {
|
|
||||||
hookEventName: 'SessionStart',
|
|
||||||
additionalContext
|
|
||||||
}
|
}
|
||||||
};
|
|
||||||
|
const result = await response.text();
|
||||||
|
const additionalContext = result.trim();
|
||||||
|
|
||||||
|
return {
|
||||||
|
hookSpecificOutput: {
|
||||||
|
hookEventName: 'SessionStart',
|
||||||
|
additionalContext
|
||||||
|
}
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
// Worker unreachable — return empty context gracefully
|
||||||
|
logger.warn('HOOK', 'Context fetch error, returning empty', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
return {
|
||||||
|
hookSpecificOutput: { hookEventName: 'SessionStart', additionalContext: '' },
|
||||||
|
exitCode: HOOK_EXIT_CODES.SUCCESS
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -39,25 +39,33 @@ export const fileEditHandler: EventHandler = {
|
|||||||
|
|
||||||
// Send to worker as an observation with file edit metadata
|
// Send to worker as an observation with file edit metadata
|
||||||
// The observation handler on the worker will process this appropriately
|
// The observation handler on the worker will process this appropriately
|
||||||
const response = await fetch(`http://127.0.0.1:${port}/api/sessions/observations`, {
|
try {
|
||||||
method: 'POST',
|
const response = await fetch(`http://127.0.0.1:${port}/api/sessions/observations`, {
|
||||||
headers: { 'Content-Type': 'application/json' },
|
method: 'POST',
|
||||||
body: JSON.stringify({
|
headers: { 'Content-Type': 'application/json' },
|
||||||
contentSessionId: sessionId,
|
body: JSON.stringify({
|
||||||
tool_name: 'write_file',
|
contentSessionId: sessionId,
|
||||||
tool_input: { filePath, edits },
|
tool_name: 'write_file',
|
||||||
tool_response: { success: true },
|
tool_input: { filePath, edits },
|
||||||
cwd
|
tool_response: { success: true },
|
||||||
})
|
cwd
|
||||||
// Note: Removed signal to avoid Windows Bun cleanup issue (libuv assertion)
|
})
|
||||||
});
|
// Note: Removed signal to avoid Windows Bun cleanup issue (libuv assertion)
|
||||||
|
});
|
||||||
|
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
throw new Error(`File edit observation storage failed: ${response.status}`);
|
// Log but don't throw — file edit observation failure should not block editing
|
||||||
|
logger.warn('HOOK', 'File edit observation storage failed, skipping', { status: response.status, filePath });
|
||||||
|
return { continue: true, suppressOutput: true, exitCode: HOOK_EXIT_CODES.SUCCESS };
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.debug('HOOK', 'File edit observation sent successfully', { filePath });
|
||||||
|
} catch (error) {
|
||||||
|
// Worker unreachable — skip file edit observation gracefully
|
||||||
|
logger.warn('HOOK', 'File edit observation fetch error, skipping', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
return { continue: true, suppressOutput: true, exitCode: HOOK_EXIT_CODES.SUCCESS };
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.debug('HOOK', 'File edit observation sent successfully', { filePath });
|
|
||||||
|
|
||||||
return { continue: true, suppressOutput: true };
|
return { continue: true, suppressOutput: true };
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -5,6 +5,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import type { EventHandler } from '../types.js';
|
import type { EventHandler } from '../types.js';
|
||||||
|
import { HOOK_EXIT_CODES } from '../../shared/hook-constants.js';
|
||||||
import { contextHandler } from './context.js';
|
import { contextHandler } from './context.js';
|
||||||
import { sessionInitHandler } from './session-init.js';
|
import { sessionInitHandler } from './session-init.js';
|
||||||
import { observationHandler } from './observation.js';
|
import { observationHandler } from './observation.js';
|
||||||
@@ -35,14 +36,22 @@ const handlers: Record<EventType, EventHandler> = {
|
|||||||
/**
|
/**
|
||||||
* Get the event handler for a given event type.
|
* Get the event handler for a given event type.
|
||||||
*
|
*
|
||||||
|
* Returns a no-op handler for unknown event types instead of throwing (fix #984).
|
||||||
|
* Claude Code may send new event types that the plugin doesn't handle yet —
|
||||||
|
* throwing would surface as a BLOCKING_ERROR to the user.
|
||||||
|
*
|
||||||
* @param eventType The type of event to handle
|
* @param eventType The type of event to handle
|
||||||
* @returns The appropriate EventHandler
|
* @returns The appropriate EventHandler, or a no-op handler for unknown types
|
||||||
* @throws Error if event type is not recognized
|
|
||||||
*/
|
*/
|
||||||
export function getEventHandler(eventType: EventType): EventHandler {
|
export function getEventHandler(eventType: string): EventHandler {
|
||||||
const handler = handlers[eventType];
|
const handler = handlers[eventType as EventType];
|
||||||
if (!handler) {
|
if (!handler) {
|
||||||
throw new Error(`Unknown event type: ${eventType}`);
|
console.error(`[claude-mem] Unknown event type: ${eventType}, returning no-op`);
|
||||||
|
return {
|
||||||
|
async execute() {
|
||||||
|
return { continue: true, suppressOutput: true, exitCode: HOOK_EXIT_CODES.SUCCESS };
|
||||||
|
}
|
||||||
|
};
|
||||||
}
|
}
|
||||||
return handler;
|
return handler;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -48,25 +48,33 @@ export const observationHandler: EventHandler = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Send to worker - worker handles privacy check and database operations
|
// Send to worker - worker handles privacy check and database operations
|
||||||
const response = await fetch(`http://127.0.0.1:${port}/api/sessions/observations`, {
|
try {
|
||||||
method: 'POST',
|
const response = await fetch(`http://127.0.0.1:${port}/api/sessions/observations`, {
|
||||||
headers: { 'Content-Type': 'application/json' },
|
method: 'POST',
|
||||||
body: JSON.stringify({
|
headers: { 'Content-Type': 'application/json' },
|
||||||
contentSessionId: sessionId,
|
body: JSON.stringify({
|
||||||
tool_name: toolName,
|
contentSessionId: sessionId,
|
||||||
tool_input: toolInput,
|
tool_name: toolName,
|
||||||
tool_response: toolResponse,
|
tool_input: toolInput,
|
||||||
cwd
|
tool_response: toolResponse,
|
||||||
})
|
cwd
|
||||||
// Note: Removed signal to avoid Windows Bun cleanup issue (libuv assertion)
|
})
|
||||||
});
|
// Note: Removed signal to avoid Windows Bun cleanup issue (libuv assertion)
|
||||||
|
});
|
||||||
|
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
throw new Error(`Observation storage failed: ${response.status}`);
|
// Log but don't throw — observation storage failure should not block tool use
|
||||||
|
logger.warn('HOOK', 'Observation storage failed, skipping', { status: response.status, toolName });
|
||||||
|
return { continue: true, suppressOutput: true, exitCode: HOOK_EXIT_CODES.SUCCESS };
|
||||||
|
}
|
||||||
|
|
||||||
|
logger.debug('HOOK', 'Observation sent successfully', { toolName });
|
||||||
|
} catch (error) {
|
||||||
|
// Worker unreachable — skip observation gracefully
|
||||||
|
logger.warn('HOOK', 'Observation fetch error, skipping', { error: error instanceof Error ? error.message : String(error) });
|
||||||
|
return { continue: true, suppressOutput: true, exitCode: HOOK_EXIT_CODES.SUCCESS };
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.debug('HOOK', 'Observation sent successfully', { toolName });
|
|
||||||
|
|
||||||
return { continue: true, suppressOutput: true };
|
return { continue: true, suppressOutput: true };
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -16,7 +16,11 @@ import { logger } from '../../utils/logger.js';
|
|||||||
export const sessionCompleteHandler: EventHandler = {
|
export const sessionCompleteHandler: EventHandler = {
|
||||||
async execute(input: NormalizedHookInput): Promise<HookResult> {
|
async execute(input: NormalizedHookInput): Promise<HookResult> {
|
||||||
// Ensure worker is running
|
// Ensure worker is running
|
||||||
await ensureWorkerRunning();
|
const workerReady = await ensureWorkerRunning();
|
||||||
|
if (!workerReady) {
|
||||||
|
// Worker not available — skip session completion gracefully
|
||||||
|
return { continue: true, suppressOutput: true };
|
||||||
|
}
|
||||||
|
|
||||||
const { sessionId } = input;
|
const { sessionId } = input;
|
||||||
const port = getWorkerPort();
|
const port = getWorkerPort();
|
||||||
|
|||||||
@@ -13,37 +13,46 @@ import { HOOK_EXIT_CODES } from '../../shared/hook-constants.js';
|
|||||||
export const userMessageHandler: EventHandler = {
|
export const userMessageHandler: EventHandler = {
|
||||||
async execute(input: NormalizedHookInput): Promise<HookResult> {
|
async execute(input: NormalizedHookInput): Promise<HookResult> {
|
||||||
// Ensure worker is running
|
// Ensure worker is running
|
||||||
await ensureWorkerRunning();
|
const workerReady = await ensureWorkerRunning();
|
||||||
|
if (!workerReady) {
|
||||||
|
// Worker not available — skip user message gracefully
|
||||||
|
return { exitCode: HOOK_EXIT_CODES.SUCCESS };
|
||||||
|
}
|
||||||
|
|
||||||
const port = getWorkerPort();
|
const port = getWorkerPort();
|
||||||
const project = basename(input.cwd ?? process.cwd());
|
const project = basename(input.cwd ?? process.cwd());
|
||||||
|
|
||||||
// Fetch formatted context directly from worker API
|
// Fetch formatted context directly from worker API
|
||||||
// Note: Removed AbortSignal.timeout to avoid Windows Bun cleanup issue (libuv assertion)
|
// Note: Removed AbortSignal.timeout to avoid Windows Bun cleanup issue (libuv assertion)
|
||||||
const response = await fetch(
|
try {
|
||||||
`http://127.0.0.1:${port}/api/context/inject?project=${encodeURIComponent(project)}&colors=true`,
|
const response = await fetch(
|
||||||
{ method: 'GET' }
|
`http://127.0.0.1:${port}/api/context/inject?project=${encodeURIComponent(project)}&colors=true`,
|
||||||
);
|
{ method: 'GET' }
|
||||||
|
);
|
||||||
|
|
||||||
if (!response.ok) {
|
if (!response.ok) {
|
||||||
// Don't throw - context fetch failure should not block the user's prompt
|
// Don't throw - context fetch failure should not block the user's prompt
|
||||||
return { exitCode: HOOK_EXIT_CODES.SUCCESS };
|
return { exitCode: HOOK_EXIT_CODES.SUCCESS };
|
||||||
|
}
|
||||||
|
|
||||||
|
const output = await response.text();
|
||||||
|
|
||||||
|
// Write to stderr for user visibility
|
||||||
|
// Note: Using process.stderr.write instead of console.error to avoid
|
||||||
|
// Claude Code treating this as a hook error. The actual hook output
|
||||||
|
// goes to stdout via hook-command.ts JSON serialization.
|
||||||
|
process.stderr.write(
|
||||||
|
"\n\n" + String.fromCodePoint(0x1F4DD) + " Claude-Mem Context Loaded\n\n" +
|
||||||
|
output +
|
||||||
|
"\n\n" + String.fromCodePoint(0x1F4A1) + " Wrap any message with <private> ... </private> to prevent storing sensitive information.\n" +
|
||||||
|
"\n" + String.fromCodePoint(0x1F4AC) + " Community https://discord.gg/J4wttp9vDu" +
|
||||||
|
`\n` + String.fromCodePoint(0x1F4FA) + ` Watch live in browser http://localhost:${port}/\n`
|
||||||
|
);
|
||||||
|
} catch (error) {
|
||||||
|
// Worker unreachable — skip user message gracefully
|
||||||
|
// User message context error is non-critical — skip gracefully
|
||||||
}
|
}
|
||||||
|
|
||||||
const output = await response.text();
|
|
||||||
|
|
||||||
// Write to stderr for user visibility
|
|
||||||
// Note: Using process.stderr.write instead of console.error to avoid
|
|
||||||
// Claude Code treating this as a hook error. The actual hook output
|
|
||||||
// goes to stdout via hook-command.ts JSON serialization.
|
|
||||||
process.stderr.write(
|
|
||||||
"\n\n" + String.fromCodePoint(0x1F4DD) + " Claude-Mem Context Loaded\n\n" +
|
|
||||||
output +
|
|
||||||
"\n\n" + String.fromCodePoint(0x1F4A1) + " Wrap any message with <private> ... </private> to prevent storing sensitive information.\n" +
|
|
||||||
"\n" + String.fromCodePoint(0x1F4AC) + " Community https://discord.gg/J4wttp9vDu" +
|
|
||||||
`\n` + String.fromCodePoint(0x1F4FA) + ` Watch live in browser http://localhost:${port}/\n`
|
|
||||||
);
|
|
||||||
|
|
||||||
return { exitCode: HOOK_EXIT_CODES.SUCCESS };
|
return { exitCode: HOOK_EXIT_CODES.SUCCESS };
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
+66
-2
@@ -8,6 +8,62 @@ export interface HookCommandOptions {
|
|||||||
skipExit?: boolean;
|
skipExit?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Classify whether an error indicates the worker is unavailable (graceful degradation)
|
||||||
|
* vs a handler/client bug (blocking error that developers need to see).
|
||||||
|
*
|
||||||
|
* Exit 0 (graceful degradation):
|
||||||
|
* - Transport failures: ECONNREFUSED, ECONNRESET, EPIPE, ETIMEDOUT, fetch failed
|
||||||
|
* - Timeout errors: timed out, timeout
|
||||||
|
* - Server errors: HTTP 5xx status codes
|
||||||
|
*
|
||||||
|
* Exit 2 (blocking error — handler/client bug):
|
||||||
|
* - HTTP 4xx status codes (bad request, not found, validation error)
|
||||||
|
* - Programming errors (TypeError, ReferenceError, SyntaxError)
|
||||||
|
* - All other unexpected errors
|
||||||
|
*/
|
||||||
|
export function isWorkerUnavailableError(error: unknown): boolean {
|
||||||
|
const message = error instanceof Error ? error.message : String(error);
|
||||||
|
const lower = message.toLowerCase();
|
||||||
|
|
||||||
|
// Transport failures — worker unreachable
|
||||||
|
const transportPatterns = [
|
||||||
|
'econnrefused',
|
||||||
|
'econnreset',
|
||||||
|
'epipe',
|
||||||
|
'etimedout',
|
||||||
|
'enotfound',
|
||||||
|
'econnaborted',
|
||||||
|
'enetunreach',
|
||||||
|
'ehostunreach',
|
||||||
|
'fetch failed',
|
||||||
|
'unable to connect',
|
||||||
|
'socket hang up',
|
||||||
|
];
|
||||||
|
if (transportPatterns.some(p => lower.includes(p))) return true;
|
||||||
|
|
||||||
|
// Timeout errors — worker didn't respond in time
|
||||||
|
if (lower.includes('timed out') || lower.includes('timeout')) return true;
|
||||||
|
|
||||||
|
// HTTP 5xx server errors — worker has internal problems
|
||||||
|
if (/failed:\s*5\d{2}/.test(message) || /status[:\s]+5\d{2}/.test(message)) return true;
|
||||||
|
|
||||||
|
// HTTP 429 (rate limit) — treat as transient unavailability, not a bug
|
||||||
|
if (/failed:\s*429/.test(message) || /status[:\s]+429/.test(message)) return true;
|
||||||
|
|
||||||
|
// HTTP 4xx client errors — our bug, NOT worker unavailability
|
||||||
|
if (/failed:\s*4\d{2}/.test(message) || /status[:\s]+4\d{2}/.test(message)) return false;
|
||||||
|
|
||||||
|
// Programming errors — code bugs, not worker unavailability
|
||||||
|
// Note: TypeError('fetch failed') already handled by transport patterns above
|
||||||
|
if (error instanceof TypeError || error instanceof ReferenceError || error instanceof SyntaxError) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Default: treat unknown errors as blocking (conservative — surface bugs)
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
export async function hookCommand(platform: string, event: string, options: HookCommandOptions = {}): Promise<number> {
|
export async function hookCommand(platform: string, event: string, options: HookCommandOptions = {}): Promise<number> {
|
||||||
try {
|
try {
|
||||||
const adapter = getPlatformAdapter(platform);
|
const adapter = getPlatformAdapter(platform);
|
||||||
@@ -26,9 +82,17 @@ export async function hookCommand(platform: string, event: string, options: Hook
|
|||||||
}
|
}
|
||||||
return exitCode;
|
return exitCode;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
if (isWorkerUnavailableError(error)) {
|
||||||
|
// Worker unavailable — degrade gracefully, don't block the user
|
||||||
|
console.error(`[claude-mem] Worker unavailable, skipping hook: ${error instanceof Error ? error.message : error}`);
|
||||||
|
if (!options.skipExit) {
|
||||||
|
process.exit(HOOK_EXIT_CODES.SUCCESS); // = 0 (graceful)
|
||||||
|
}
|
||||||
|
return HOOK_EXIT_CODES.SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handler/client bug — show as blocking error so developers see it
|
||||||
console.error(`Hook error: ${error}`);
|
console.error(`Hook error: ${error}`);
|
||||||
// Use exit code 2 (blocking error) so users see the error message
|
|
||||||
// Exit code 1 only shows in verbose mode per Claude Code docs
|
|
||||||
if (!options.skipExit) {
|
if (!options.skipExit) {
|
||||||
process.exit(HOOK_EXIT_CODES.BLOCKING_ERROR); // = 2
|
process.exit(HOOK_EXIT_CODES.BLOCKING_ERROR); // = 2
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -77,7 +77,11 @@ export function removePidFile(): void {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get platform-adjusted timeout (Windows socket cleanup is slower)
|
* Get platform-adjusted timeout for worker-side socket operations (2.0x on Windows).
|
||||||
|
*
|
||||||
|
* Note: Two platform multiplier functions exist intentionally:
|
||||||
|
* - getTimeout() in hook-constants.ts uses 1.5x for hook-side operations (fast path)
|
||||||
|
* - getPlatformTimeout() here uses 2.0x for worker-side socket operations (slower path)
|
||||||
*/
|
*/
|
||||||
export function getPlatformTimeout(baseMs: number): number {
|
export function getPlatformTimeout(baseMs: number): number {
|
||||||
const WINDOWS_MULTIPLIER = 2.0;
|
const WINDOWS_MULTIPLIER = 2.0;
|
||||||
@@ -451,7 +455,27 @@ export function spawnDaemon(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Unix: standard detached spawn
|
// Unix: Use setsid to create a new session, fully detaching from the
|
||||||
|
// controlling terminal. This prevents SIGHUP from reaching the daemon
|
||||||
|
// even if the in-process SIGHUP handler somehow fails (belt-and-suspenders).
|
||||||
|
// Fall back to standard detached spawn if setsid is not available.
|
||||||
|
const setsidPath = '/usr/bin/setsid';
|
||||||
|
if (existsSync(setsidPath)) {
|
||||||
|
const child = spawn(setsidPath, [process.execPath, scriptPath, '--daemon'], {
|
||||||
|
detached: true,
|
||||||
|
stdio: 'ignore',
|
||||||
|
env
|
||||||
|
});
|
||||||
|
|
||||||
|
if (child.pid === undefined) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
child.unref();
|
||||||
|
return child.pid;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fallback: standard detached spawn (macOS, systems without setsid)
|
||||||
const child = spawn(process.execPath, [scriptPath, '--daemon'], {
|
const child = spawn(process.execPath, [scriptPath, '--daemon'], {
|
||||||
detached: true,
|
detached: true,
|
||||||
stdio: 'ignore',
|
stdio: 'ignore',
|
||||||
@@ -467,6 +491,56 @@ export function spawnDaemon(
|
|||||||
return child.pid;
|
return child.pid;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Check if a process with the given PID is alive.
|
||||||
|
*
|
||||||
|
* Uses the process.kill(pid, 0) idiom: signal 0 doesn't send a signal,
|
||||||
|
* it just checks if the process exists and is reachable.
|
||||||
|
*
|
||||||
|
* EPERM is treated as "alive" because it means the process exists but
|
||||||
|
* belongs to a different user/session (common in multi-user setups).
|
||||||
|
* PID 0 (Windows WMIC sentinel for unknown PID) is treated as alive.
|
||||||
|
*/
|
||||||
|
export function isProcessAlive(pid: number): boolean {
|
||||||
|
// PID 0 is the Windows WMIC sentinel value — process was spawned but PID unknown
|
||||||
|
if (pid === 0) return true;
|
||||||
|
|
||||||
|
// Invalid PIDs are not alive
|
||||||
|
if (!Number.isInteger(pid) || pid < 0) return false;
|
||||||
|
|
||||||
|
try {
|
||||||
|
process.kill(pid, 0);
|
||||||
|
return true;
|
||||||
|
} catch (error: unknown) {
|
||||||
|
const code = (error as NodeJS.ErrnoException).code;
|
||||||
|
// EPERM = process exists but different user/session — treat as alive
|
||||||
|
if (code === 'EPERM') return true;
|
||||||
|
// ESRCH = no such process — it's dead
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read the PID file and remove it if the recorded process is dead (stale).
|
||||||
|
*
|
||||||
|
* This is a cheap operation: one filesystem read + one signal-0 check.
|
||||||
|
* Called at the top of ensureWorkerStarted() to clean up after WSL2
|
||||||
|
* hibernate, OOM kills, or other ungraceful worker deaths.
|
||||||
|
*/
|
||||||
|
export function cleanStalePidFile(): void {
|
||||||
|
const pidInfo = readPidFile();
|
||||||
|
if (!pidInfo) return;
|
||||||
|
|
||||||
|
if (!isProcessAlive(pidInfo.pid)) {
|
||||||
|
logger.info('SYSTEM', 'Removing stale PID file (worker process is dead)', {
|
||||||
|
pid: pidInfo.pid,
|
||||||
|
port: pidInfo.port,
|
||||||
|
startedAt: pidInfo.startedAt
|
||||||
|
});
|
||||||
|
removePidFile();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create signal handler factory for graceful shutdown
|
* Create signal handler factory for graceful shutdown
|
||||||
* Returns a handler function that can be passed to process.on('SIGTERM') etc.
|
* Returns a handler function that can be passed to process.on('SIGTERM') etc.
|
||||||
|
|||||||
@@ -0,0 +1,65 @@
|
|||||||
|
import { DEFAULT_CONFIG_PATH, DEFAULT_STATE_PATH, expandHomePath, loadTranscriptWatchConfig, writeSampleConfig } from './config.js';
|
||||||
|
import { TranscriptWatcher } from './watcher.js';
|
||||||
|
|
||||||
|
function getArgValue(args: string[], name: string): string | null {
|
||||||
|
const index = args.indexOf(name);
|
||||||
|
if (index === -1) return null;
|
||||||
|
return args[index + 1] ?? null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function runTranscriptCommand(subcommand: string | undefined, args: string[]): Promise<number> {
|
||||||
|
switch (subcommand) {
|
||||||
|
case 'init': {
|
||||||
|
const configPath = getArgValue(args, '--config') ?? DEFAULT_CONFIG_PATH;
|
||||||
|
writeSampleConfig(configPath);
|
||||||
|
console.log(`Created sample config: ${expandHomePath(configPath)}`);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
case 'watch': {
|
||||||
|
const configPath = getArgValue(args, '--config') ?? DEFAULT_CONFIG_PATH;
|
||||||
|
let config;
|
||||||
|
try {
|
||||||
|
config = loadTranscriptWatchConfig(configPath);
|
||||||
|
} catch (error) {
|
||||||
|
if (error instanceof Error && error.message.includes('not found')) {
|
||||||
|
writeSampleConfig(configPath);
|
||||||
|
console.log(`Created sample config: ${expandHomePath(configPath)}`);
|
||||||
|
config = loadTranscriptWatchConfig(configPath);
|
||||||
|
} else {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const statePath = expandHomePath(config.stateFile ?? DEFAULT_STATE_PATH);
|
||||||
|
const watcher = new TranscriptWatcher(config, statePath);
|
||||||
|
await watcher.start();
|
||||||
|
console.log('Transcript watcher running. Press Ctrl+C to stop.');
|
||||||
|
|
||||||
|
const shutdown = () => {
|
||||||
|
watcher.stop();
|
||||||
|
process.exit(0);
|
||||||
|
};
|
||||||
|
process.on('SIGINT', shutdown);
|
||||||
|
process.on('SIGTERM', shutdown);
|
||||||
|
return await new Promise(() => undefined);
|
||||||
|
}
|
||||||
|
case 'validate': {
|
||||||
|
const configPath = getArgValue(args, '--config') ?? DEFAULT_CONFIG_PATH;
|
||||||
|
try {
|
||||||
|
loadTranscriptWatchConfig(configPath);
|
||||||
|
} catch (error) {
|
||||||
|
if (error instanceof Error && error.message.includes('not found')) {
|
||||||
|
writeSampleConfig(configPath);
|
||||||
|
console.log(`Created sample config: ${expandHomePath(configPath)}`);
|
||||||
|
loadTranscriptWatchConfig(configPath);
|
||||||
|
} else {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
console.log(`Config OK: ${expandHomePath(configPath)}`);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
console.log('Usage: claude-mem transcript <init|watch|validate> [--config <path>]');
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,137 @@
|
|||||||
|
import { existsSync, readFileSync, writeFileSync, mkdirSync } from 'fs';
|
||||||
|
import { homedir } from 'os';
|
||||||
|
import { join, dirname } from 'path';
|
||||||
|
import type { TranscriptSchema, TranscriptWatchConfig } from './types.js';
|
||||||
|
|
||||||
|
export const DEFAULT_CONFIG_PATH = join(homedir(), '.claude-mem', 'transcript-watch.json');
|
||||||
|
export const DEFAULT_STATE_PATH = join(homedir(), '.claude-mem', 'transcript-watch-state.json');
|
||||||
|
|
||||||
|
const CODEX_SAMPLE_SCHEMA: TranscriptSchema = {
|
||||||
|
name: 'codex',
|
||||||
|
version: '0.2',
|
||||||
|
description: 'Schema for Codex session JSONL files under ~/.codex/sessions.',
|
||||||
|
events: [
|
||||||
|
{
|
||||||
|
name: 'session-meta',
|
||||||
|
match: { path: 'type', equals: 'session_meta' },
|
||||||
|
action: 'session_context',
|
||||||
|
fields: {
|
||||||
|
sessionId: 'payload.id',
|
||||||
|
cwd: 'payload.cwd'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'turn-context',
|
||||||
|
match: { path: 'type', equals: 'turn_context' },
|
||||||
|
action: 'session_context',
|
||||||
|
fields: {
|
||||||
|
cwd: 'payload.cwd'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'user-message',
|
||||||
|
match: { path: 'payload.type', equals: 'user_message' },
|
||||||
|
action: 'session_init',
|
||||||
|
fields: {
|
||||||
|
prompt: 'payload.message'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'assistant-message',
|
||||||
|
match: { path: 'payload.type', equals: 'agent_message' },
|
||||||
|
action: 'assistant_message',
|
||||||
|
fields: {
|
||||||
|
message: 'payload.message'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'tool-use',
|
||||||
|
match: { path: 'payload.type', in: ['function_call', 'custom_tool_call', 'web_search_call'] },
|
||||||
|
action: 'tool_use',
|
||||||
|
fields: {
|
||||||
|
toolId: 'payload.call_id',
|
||||||
|
toolName: {
|
||||||
|
coalesce: [
|
||||||
|
'payload.name',
|
||||||
|
{ value: 'web_search' }
|
||||||
|
]
|
||||||
|
},
|
||||||
|
toolInput: {
|
||||||
|
coalesce: [
|
||||||
|
'payload.arguments',
|
||||||
|
'payload.input',
|
||||||
|
'payload.action'
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'tool-result',
|
||||||
|
match: { path: 'payload.type', in: ['function_call_output', 'custom_tool_call_output'] },
|
||||||
|
action: 'tool_result',
|
||||||
|
fields: {
|
||||||
|
toolId: 'payload.call_id',
|
||||||
|
toolResponse: 'payload.output'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'session-end',
|
||||||
|
match: { path: 'payload.type', equals: 'turn_aborted' },
|
||||||
|
action: 'session_end'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
};
|
||||||
|
|
||||||
|
export const SAMPLE_CONFIG: TranscriptWatchConfig = {
|
||||||
|
version: 1,
|
||||||
|
schemas: {
|
||||||
|
codex: CODEX_SAMPLE_SCHEMA
|
||||||
|
},
|
||||||
|
watches: [
|
||||||
|
{
|
||||||
|
name: 'codex',
|
||||||
|
path: '~/.codex/sessions/**/*.jsonl',
|
||||||
|
schema: 'codex',
|
||||||
|
startAtEnd: true,
|
||||||
|
context: {
|
||||||
|
mode: 'agents',
|
||||||
|
path: '~/.codex/AGENTS.md',
|
||||||
|
updateOn: ['session_start', 'session_end']
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
stateFile: DEFAULT_STATE_PATH
|
||||||
|
};
|
||||||
|
|
||||||
|
export function expandHomePath(inputPath: string): string {
|
||||||
|
if (!inputPath) return inputPath;
|
||||||
|
if (inputPath.startsWith('~')) {
|
||||||
|
return join(homedir(), inputPath.slice(1));
|
||||||
|
}
|
||||||
|
return inputPath;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function loadTranscriptWatchConfig(path = DEFAULT_CONFIG_PATH): TranscriptWatchConfig {
|
||||||
|
const resolvedPath = expandHomePath(path);
|
||||||
|
if (!existsSync(resolvedPath)) {
|
||||||
|
throw new Error(`Transcript watch config not found: ${resolvedPath}`);
|
||||||
|
}
|
||||||
|
const raw = readFileSync(resolvedPath, 'utf-8');
|
||||||
|
const parsed = JSON.parse(raw) as TranscriptWatchConfig;
|
||||||
|
if (!parsed.version || !parsed.watches) {
|
||||||
|
throw new Error(`Invalid transcript watch config: ${resolvedPath}`);
|
||||||
|
}
|
||||||
|
if (!parsed.stateFile) {
|
||||||
|
parsed.stateFile = DEFAULT_STATE_PATH;
|
||||||
|
}
|
||||||
|
return parsed;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function writeSampleConfig(path = DEFAULT_CONFIG_PATH): void {
|
||||||
|
const resolvedPath = expandHomePath(path);
|
||||||
|
const dir = dirname(resolvedPath);
|
||||||
|
if (!existsSync(dir)) {
|
||||||
|
mkdirSync(dir, { recursive: true });
|
||||||
|
}
|
||||||
|
writeFileSync(resolvedPath, JSON.stringify(SAMPLE_CONFIG, null, 2));
|
||||||
|
}
|
||||||
@@ -0,0 +1,151 @@
|
|||||||
|
import type { FieldSpec, MatchRule, TranscriptSchema, WatchTarget } from './types.js';
|
||||||
|
|
||||||
|
interface ResolveContext {
|
||||||
|
watch: WatchTarget;
|
||||||
|
schema: TranscriptSchema;
|
||||||
|
session?: Record<string, unknown>;
|
||||||
|
}
|
||||||
|
|
||||||
|
function parsePath(path: string): Array<string | number> {
|
||||||
|
const cleaned = path.trim().replace(/^\$\.?/, '');
|
||||||
|
if (!cleaned) return [];
|
||||||
|
|
||||||
|
const tokens: Array<string | number> = [];
|
||||||
|
const parts = cleaned.split('.');
|
||||||
|
|
||||||
|
for (const part of parts) {
|
||||||
|
const regex = /([^[\]]+)|\[(\d+)\]/g;
|
||||||
|
let match: RegExpExecArray | null;
|
||||||
|
while ((match = regex.exec(part)) !== null) {
|
||||||
|
if (match[1]) {
|
||||||
|
tokens.push(match[1]);
|
||||||
|
} else if (match[2]) {
|
||||||
|
tokens.push(parseInt(match[2], 10));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return tokens;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getValueByPath(input: unknown, path: string): unknown {
|
||||||
|
if (!path) return undefined;
|
||||||
|
const tokens = parsePath(path);
|
||||||
|
let current: any = input;
|
||||||
|
|
||||||
|
for (const token of tokens) {
|
||||||
|
if (current === null || current === undefined) return undefined;
|
||||||
|
current = current[token as any];
|
||||||
|
}
|
||||||
|
|
||||||
|
return current;
|
||||||
|
}
|
||||||
|
|
||||||
|
function isEmptyValue(value: unknown): boolean {
|
||||||
|
return value === undefined || value === null || value === '';
|
||||||
|
}
|
||||||
|
|
||||||
|
function resolveFromContext(path: string, ctx: ResolveContext): unknown {
|
||||||
|
if (path.startsWith('$watch.')) {
|
||||||
|
const key = path.slice('$watch.'.length);
|
||||||
|
return (ctx.watch as any)[key];
|
||||||
|
}
|
||||||
|
if (path.startsWith('$schema.')) {
|
||||||
|
const key = path.slice('$schema.'.length);
|
||||||
|
return (ctx.schema as any)[key];
|
||||||
|
}
|
||||||
|
if (path.startsWith('$session.')) {
|
||||||
|
const key = path.slice('$session.'.length);
|
||||||
|
return ctx.session ? (ctx.session as any)[key] : undefined;
|
||||||
|
}
|
||||||
|
if (path === '$cwd') return ctx.watch.workspace;
|
||||||
|
if (path === '$project') return ctx.watch.project;
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function resolveFieldSpec(
|
||||||
|
spec: FieldSpec | undefined,
|
||||||
|
entry: unknown,
|
||||||
|
ctx: ResolveContext
|
||||||
|
): unknown {
|
||||||
|
if (spec === undefined) return undefined;
|
||||||
|
|
||||||
|
if (typeof spec === 'string') {
|
||||||
|
const fromContext = resolveFromContext(spec, ctx);
|
||||||
|
if (fromContext !== undefined) return fromContext;
|
||||||
|
return getValueByPath(entry, spec);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (spec.coalesce && Array.isArray(spec.coalesce)) {
|
||||||
|
for (const candidate of spec.coalesce) {
|
||||||
|
const value = resolveFieldSpec(candidate, entry, ctx);
|
||||||
|
if (!isEmptyValue(value)) return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (spec.path) {
|
||||||
|
const fromContext = resolveFromContext(spec.path, ctx);
|
||||||
|
if (fromContext !== undefined) return fromContext;
|
||||||
|
const value = getValueByPath(entry, spec.path);
|
||||||
|
if (!isEmptyValue(value)) return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (spec.value !== undefined) return spec.value;
|
||||||
|
|
||||||
|
if (spec.default !== undefined) return spec.default;
|
||||||
|
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function resolveFields(
|
||||||
|
fields: Record<string, FieldSpec> | undefined,
|
||||||
|
entry: unknown,
|
||||||
|
ctx: ResolveContext
|
||||||
|
): Record<string, unknown> {
|
||||||
|
const resolved: Record<string, unknown> = {};
|
||||||
|
if (!fields) return resolved;
|
||||||
|
|
||||||
|
for (const [key, spec] of Object.entries(fields)) {
|
||||||
|
resolved[key] = resolveFieldSpec(spec, entry, ctx);
|
||||||
|
}
|
||||||
|
|
||||||
|
return resolved;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function matchesRule(
|
||||||
|
entry: unknown,
|
||||||
|
rule: MatchRule | undefined,
|
||||||
|
schema: TranscriptSchema
|
||||||
|
): boolean {
|
||||||
|
if (!rule) return true;
|
||||||
|
|
||||||
|
const path = rule.path || schema.eventTypePath || 'type';
|
||||||
|
const value = path ? getValueByPath(entry, path) : undefined;
|
||||||
|
|
||||||
|
if (rule.exists) {
|
||||||
|
if (value === undefined || value === null || value === '') return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rule.equals !== undefined) {
|
||||||
|
return value === rule.equals;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rule.in && Array.isArray(rule.in)) {
|
||||||
|
return rule.in.includes(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rule.contains !== undefined) {
|
||||||
|
return typeof value === 'string' && value.includes(rule.contains);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (rule.regex) {
|
||||||
|
try {
|
||||||
|
const regex = new RegExp(rule.regex);
|
||||||
|
return regex.test(String(value ?? ''));
|
||||||
|
} catch {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
@@ -0,0 +1,371 @@
|
|||||||
|
import { sessionInitHandler } from '../../cli/handlers/session-init.js';
|
||||||
|
import { observationHandler } from '../../cli/handlers/observation.js';
|
||||||
|
import { fileEditHandler } from '../../cli/handlers/file-edit.js';
|
||||||
|
import { sessionCompleteHandler } from '../../cli/handlers/session-complete.js';
|
||||||
|
import { ensureWorkerRunning, getWorkerPort } from '../../shared/worker-utils.js';
|
||||||
|
import { logger } from '../../utils/logger.js';
|
||||||
|
import { getProjectContext, getProjectName } from '../../utils/project-name.js';
|
||||||
|
import { writeAgentsMd } from '../../utils/agents-md-utils.js';
|
||||||
|
import { resolveFieldSpec, resolveFields, matchesRule } from './field-utils.js';
|
||||||
|
import { expandHomePath } from './config.js';
|
||||||
|
import type { TranscriptSchema, WatchTarget, SchemaEvent } from './types.js';
|
||||||
|
|
||||||
|
interface SessionState {
|
||||||
|
sessionId: string;
|
||||||
|
cwd?: string;
|
||||||
|
project?: string;
|
||||||
|
lastUserMessage?: string;
|
||||||
|
lastAssistantMessage?: string;
|
||||||
|
pendingTools: Map<string, { name?: string; input?: unknown }>;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface PendingTool {
|
||||||
|
id?: string;
|
||||||
|
name?: string;
|
||||||
|
input?: unknown;
|
||||||
|
response?: unknown;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class TranscriptEventProcessor {
|
||||||
|
private sessions = new Map<string, SessionState>();
|
||||||
|
|
||||||
|
async processEntry(
|
||||||
|
entry: unknown,
|
||||||
|
watch: WatchTarget,
|
||||||
|
schema: TranscriptSchema,
|
||||||
|
sessionIdOverride?: string | null
|
||||||
|
): Promise<void> {
|
||||||
|
for (const event of schema.events) {
|
||||||
|
if (!matchesRule(entry, event.match, schema)) continue;
|
||||||
|
await this.handleEvent(entry, watch, schema, event, sessionIdOverride ?? undefined);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private getSessionKey(watch: WatchTarget, sessionId: string): string {
|
||||||
|
return `${watch.name}:${sessionId}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private getOrCreateSession(watch: WatchTarget, sessionId: string): SessionState {
|
||||||
|
const key = this.getSessionKey(watch, sessionId);
|
||||||
|
let session = this.sessions.get(key);
|
||||||
|
if (!session) {
|
||||||
|
session = {
|
||||||
|
sessionId,
|
||||||
|
pendingTools: new Map()
|
||||||
|
};
|
||||||
|
this.sessions.set(key, session);
|
||||||
|
}
|
||||||
|
return session;
|
||||||
|
}
|
||||||
|
|
||||||
|
private resolveSessionId(
|
||||||
|
entry: unknown,
|
||||||
|
watch: WatchTarget,
|
||||||
|
schema: TranscriptSchema,
|
||||||
|
event: SchemaEvent,
|
||||||
|
sessionIdOverride?: string
|
||||||
|
): string | null {
|
||||||
|
const ctx = { watch, schema } as any;
|
||||||
|
const fieldSpec = event.fields?.sessionId ?? (schema.sessionIdPath ? { path: schema.sessionIdPath } : undefined);
|
||||||
|
const resolved = resolveFieldSpec(fieldSpec, entry, ctx);
|
||||||
|
if (typeof resolved === 'string' && resolved.trim()) return resolved;
|
||||||
|
if (typeof resolved === 'number') return String(resolved);
|
||||||
|
if (sessionIdOverride && sessionIdOverride.trim()) return sessionIdOverride;
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private resolveCwd(
|
||||||
|
entry: unknown,
|
||||||
|
watch: WatchTarget,
|
||||||
|
schema: TranscriptSchema,
|
||||||
|
event: SchemaEvent,
|
||||||
|
session: SessionState
|
||||||
|
): string | undefined {
|
||||||
|
const ctx = { watch, schema, session } as any;
|
||||||
|
const fieldSpec = event.fields?.cwd ?? (schema.cwdPath ? { path: schema.cwdPath } : undefined);
|
||||||
|
const resolved = resolveFieldSpec(fieldSpec, entry, ctx);
|
||||||
|
if (typeof resolved === 'string' && resolved.trim()) return resolved;
|
||||||
|
if (watch.workspace) return watch.workspace;
|
||||||
|
return session.cwd;
|
||||||
|
}
|
||||||
|
|
||||||
|
private resolveProject(
|
||||||
|
entry: unknown,
|
||||||
|
watch: WatchTarget,
|
||||||
|
schema: TranscriptSchema,
|
||||||
|
event: SchemaEvent,
|
||||||
|
session: SessionState
|
||||||
|
): string | undefined {
|
||||||
|
const ctx = { watch, schema, session } as any;
|
||||||
|
const fieldSpec = event.fields?.project ?? (schema.projectPath ? { path: schema.projectPath } : undefined);
|
||||||
|
const resolved = resolveFieldSpec(fieldSpec, entry, ctx);
|
||||||
|
if (typeof resolved === 'string' && resolved.trim()) return resolved;
|
||||||
|
if (watch.project) return watch.project;
|
||||||
|
if (session.cwd) return getProjectName(session.cwd);
|
||||||
|
return session.project;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async handleEvent(
|
||||||
|
entry: unknown,
|
||||||
|
watch: WatchTarget,
|
||||||
|
schema: TranscriptSchema,
|
||||||
|
event: SchemaEvent,
|
||||||
|
sessionIdOverride?: string
|
||||||
|
): Promise<void> {
|
||||||
|
const sessionId = this.resolveSessionId(entry, watch, schema, event, sessionIdOverride);
|
||||||
|
if (!sessionId) {
|
||||||
|
logger.debug('TRANSCRIPT', 'Skipping event without sessionId', { event: event.name, watch: watch.name });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const session = this.getOrCreateSession(watch, sessionId);
|
||||||
|
const cwd = this.resolveCwd(entry, watch, schema, event, session);
|
||||||
|
if (cwd) session.cwd = cwd;
|
||||||
|
const project = this.resolveProject(entry, watch, schema, event, session);
|
||||||
|
if (project) session.project = project;
|
||||||
|
|
||||||
|
const fields = resolveFields(event.fields, entry, { watch, schema, session });
|
||||||
|
|
||||||
|
switch (event.action) {
|
||||||
|
case 'session_context':
|
||||||
|
this.applySessionContext(session, fields);
|
||||||
|
break;
|
||||||
|
case 'session_init':
|
||||||
|
await this.handleSessionInit(session, fields);
|
||||||
|
if (watch.context?.updateOn?.includes('session_start')) {
|
||||||
|
await this.updateContext(session, watch);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case 'user_message':
|
||||||
|
if (typeof fields.message === 'string') session.lastUserMessage = fields.message;
|
||||||
|
if (typeof fields.prompt === 'string') session.lastUserMessage = fields.prompt;
|
||||||
|
break;
|
||||||
|
case 'assistant_message':
|
||||||
|
if (typeof fields.message === 'string') session.lastAssistantMessage = fields.message;
|
||||||
|
break;
|
||||||
|
case 'tool_use':
|
||||||
|
await this.handleToolUse(session, fields);
|
||||||
|
break;
|
||||||
|
case 'tool_result':
|
||||||
|
await this.handleToolResult(session, fields);
|
||||||
|
break;
|
||||||
|
case 'observation':
|
||||||
|
await this.sendObservation(session, fields);
|
||||||
|
break;
|
||||||
|
case 'file_edit':
|
||||||
|
await this.sendFileEdit(session, fields);
|
||||||
|
break;
|
||||||
|
case 'session_end':
|
||||||
|
await this.handleSessionEnd(session, watch);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private applySessionContext(session: SessionState, fields: Record<string, unknown>): void {
|
||||||
|
const cwd = typeof fields.cwd === 'string' ? fields.cwd : undefined;
|
||||||
|
const project = typeof fields.project === 'string' ? fields.project : undefined;
|
||||||
|
if (cwd) session.cwd = cwd;
|
||||||
|
if (project) session.project = project;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async handleSessionInit(session: SessionState, fields: Record<string, unknown>): Promise<void> {
|
||||||
|
const prompt = typeof fields.prompt === 'string' ? fields.prompt : '';
|
||||||
|
const cwd = session.cwd ?? process.cwd();
|
||||||
|
if (prompt) {
|
||||||
|
session.lastUserMessage = prompt;
|
||||||
|
}
|
||||||
|
|
||||||
|
await sessionInitHandler.execute({
|
||||||
|
sessionId: session.sessionId,
|
||||||
|
cwd,
|
||||||
|
prompt,
|
||||||
|
platform: 'transcript'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private async handleToolUse(session: SessionState, fields: Record<string, unknown>): Promise<void> {
|
||||||
|
const toolId = typeof fields.toolId === 'string' ? fields.toolId : undefined;
|
||||||
|
const toolName = typeof fields.toolName === 'string' ? fields.toolName : undefined;
|
||||||
|
const toolInput = this.maybeParseJson(fields.toolInput);
|
||||||
|
const toolResponse = this.maybeParseJson(fields.toolResponse);
|
||||||
|
|
||||||
|
const pending: PendingTool = { id: toolId, name: toolName, input: toolInput, response: toolResponse };
|
||||||
|
|
||||||
|
if (toolId) {
|
||||||
|
session.pendingTools.set(toolId, { name: pending.name, input: pending.input });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (toolName === 'apply_patch' && typeof toolInput === 'string') {
|
||||||
|
const files = this.parseApplyPatchFiles(toolInput);
|
||||||
|
for (const filePath of files) {
|
||||||
|
await this.sendFileEdit(session, {
|
||||||
|
filePath,
|
||||||
|
edits: [{ type: 'apply_patch', patch: toolInput }]
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (toolResponse !== undefined && toolName) {
|
||||||
|
await this.sendObservation(session, {
|
||||||
|
toolName,
|
||||||
|
toolInput,
|
||||||
|
toolResponse
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async handleToolResult(session: SessionState, fields: Record<string, unknown>): Promise<void> {
|
||||||
|
const toolId = typeof fields.toolId === 'string' ? fields.toolId : undefined;
|
||||||
|
const toolName = typeof fields.toolName === 'string' ? fields.toolName : undefined;
|
||||||
|
const toolResponse = this.maybeParseJson(fields.toolResponse);
|
||||||
|
|
||||||
|
let toolInput: unknown = this.maybeParseJson(fields.toolInput);
|
||||||
|
let name = toolName;
|
||||||
|
|
||||||
|
if (toolId && session.pendingTools.has(toolId)) {
|
||||||
|
const pending = session.pendingTools.get(toolId)!;
|
||||||
|
toolInput = pending.input ?? toolInput;
|
||||||
|
name = name ?? pending.name;
|
||||||
|
session.pendingTools.delete(toolId);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (name) {
|
||||||
|
await this.sendObservation(session, {
|
||||||
|
toolName: name,
|
||||||
|
toolInput,
|
||||||
|
toolResponse
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async sendObservation(session: SessionState, fields: Record<string, unknown>): Promise<void> {
|
||||||
|
const toolName = typeof fields.toolName === 'string' ? fields.toolName : undefined;
|
||||||
|
if (!toolName) return;
|
||||||
|
|
||||||
|
await observationHandler.execute({
|
||||||
|
sessionId: session.sessionId,
|
||||||
|
cwd: session.cwd ?? process.cwd(),
|
||||||
|
toolName,
|
||||||
|
toolInput: this.maybeParseJson(fields.toolInput),
|
||||||
|
toolResponse: this.maybeParseJson(fields.toolResponse),
|
||||||
|
platform: 'transcript'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private async sendFileEdit(session: SessionState, fields: Record<string, unknown>): Promise<void> {
|
||||||
|
const filePath = typeof fields.filePath === 'string' ? fields.filePath : undefined;
|
||||||
|
if (!filePath) return;
|
||||||
|
|
||||||
|
await fileEditHandler.execute({
|
||||||
|
sessionId: session.sessionId,
|
||||||
|
cwd: session.cwd ?? process.cwd(),
|
||||||
|
filePath,
|
||||||
|
edits: Array.isArray(fields.edits) ? fields.edits : undefined,
|
||||||
|
platform: 'transcript'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private maybeParseJson(value: unknown): unknown {
|
||||||
|
if (typeof value !== 'string') return value;
|
||||||
|
const trimmed = value.trim();
|
||||||
|
if (!trimmed) return value;
|
||||||
|
if (!(trimmed.startsWith('{') || trimmed.startsWith('['))) return value;
|
||||||
|
try {
|
||||||
|
return JSON.parse(trimmed);
|
||||||
|
} catch {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private parseApplyPatchFiles(patch: string): string[] {
|
||||||
|
const files: string[] = [];
|
||||||
|
const lines = patch.split('\n');
|
||||||
|
for (const line of lines) {
|
||||||
|
const trimmed = line.trim();
|
||||||
|
if (trimmed.startsWith('*** Update File: ')) {
|
||||||
|
files.push(trimmed.replace('*** Update File: ', '').trim());
|
||||||
|
} else if (trimmed.startsWith('*** Add File: ')) {
|
||||||
|
files.push(trimmed.replace('*** Add File: ', '').trim());
|
||||||
|
} else if (trimmed.startsWith('*** Delete File: ')) {
|
||||||
|
files.push(trimmed.replace('*** Delete File: ', '').trim());
|
||||||
|
} else if (trimmed.startsWith('*** Move to: ')) {
|
||||||
|
files.push(trimmed.replace('*** Move to: ', '').trim());
|
||||||
|
} else if (trimmed.startsWith('+++ ')) {
|
||||||
|
const path = trimmed.replace('+++ ', '').replace(/^b\//, '').trim();
|
||||||
|
if (path && path !== '/dev/null') files.push(path);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return Array.from(new Set(files));
|
||||||
|
}
|
||||||
|
|
||||||
|
private async handleSessionEnd(session: SessionState, watch: WatchTarget): Promise<void> {
|
||||||
|
await this.queueSummary(session);
|
||||||
|
await sessionCompleteHandler.execute({
|
||||||
|
sessionId: session.sessionId,
|
||||||
|
cwd: session.cwd ?? process.cwd(),
|
||||||
|
platform: 'transcript'
|
||||||
|
});
|
||||||
|
await this.updateContext(session, watch);
|
||||||
|
session.pendingTools.clear();
|
||||||
|
const key = this.getSessionKey(watch, session.sessionId);
|
||||||
|
this.sessions.delete(key);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async queueSummary(session: SessionState): Promise<void> {
|
||||||
|
const workerReady = await ensureWorkerRunning();
|
||||||
|
if (!workerReady) return;
|
||||||
|
|
||||||
|
const port = getWorkerPort();
|
||||||
|
const lastAssistantMessage = session.lastAssistantMessage ?? '';
|
||||||
|
|
||||||
|
try {
|
||||||
|
await fetch(`http://127.0.0.1:${port}/api/sessions/summarize`, {
|
||||||
|
method: 'POST',
|
||||||
|
headers: { 'Content-Type': 'application/json' },
|
||||||
|
body: JSON.stringify({
|
||||||
|
contentSessionId: session.sessionId,
|
||||||
|
last_assistant_message: lastAssistantMessage
|
||||||
|
})
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
logger.warn('TRANSCRIPT', 'Summary request failed', {
|
||||||
|
error: error instanceof Error ? error.message : String(error)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async updateContext(session: SessionState, watch: WatchTarget): Promise<void> {
|
||||||
|
if (!watch.context) return;
|
||||||
|
if (watch.context.mode !== 'agents') return;
|
||||||
|
|
||||||
|
const workerReady = await ensureWorkerRunning();
|
||||||
|
if (!workerReady) return;
|
||||||
|
|
||||||
|
const cwd = session.cwd ?? watch.workspace;
|
||||||
|
if (!cwd) return;
|
||||||
|
|
||||||
|
const context = getProjectContext(cwd);
|
||||||
|
const projectsParam = context.allProjects.join(',');
|
||||||
|
const port = getWorkerPort();
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch(
|
||||||
|
`http://127.0.0.1:${port}/api/context/inject?projects=${encodeURIComponent(projectsParam)}`
|
||||||
|
);
|
||||||
|
if (!response.ok) return;
|
||||||
|
|
||||||
|
const content = (await response.text()).trim();
|
||||||
|
if (!content) return;
|
||||||
|
|
||||||
|
const agentsPath = expandHomePath(watch.context.path ?? `${cwd}/AGENTS.md`);
|
||||||
|
writeAgentsMd(agentsPath, content);
|
||||||
|
logger.debug('TRANSCRIPT', 'Updated AGENTS.md context', { agentsPath, watch: watch.name });
|
||||||
|
} catch (error) {
|
||||||
|
logger.warn('TRANSCRIPT', 'Failed to update AGENTS.md context', {
|
||||||
|
error: error instanceof Error ? error.message : String(error)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,40 @@
|
|||||||
|
import { existsSync, readFileSync, writeFileSync, mkdirSync } from 'fs';
|
||||||
|
import { dirname } from 'path';
|
||||||
|
import { logger } from '../../utils/logger.js';
|
||||||
|
|
||||||
|
export interface TranscriptWatchState {
|
||||||
|
offsets: Record<string, number>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function loadWatchState(statePath: string): TranscriptWatchState {
|
||||||
|
try {
|
||||||
|
if (!existsSync(statePath)) {
|
||||||
|
return { offsets: {} };
|
||||||
|
}
|
||||||
|
const raw = readFileSync(statePath, 'utf-8');
|
||||||
|
const parsed = JSON.parse(raw) as TranscriptWatchState;
|
||||||
|
if (!parsed.offsets) return { offsets: {} };
|
||||||
|
return parsed;
|
||||||
|
} catch (error) {
|
||||||
|
logger.warn('TRANSCRIPT', 'Failed to load watch state, starting fresh', {
|
||||||
|
statePath,
|
||||||
|
error: error instanceof Error ? error.message : String(error)
|
||||||
|
});
|
||||||
|
return { offsets: {} };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export function saveWatchState(statePath: string, state: TranscriptWatchState): void {
|
||||||
|
try {
|
||||||
|
const dir = dirname(statePath);
|
||||||
|
if (!existsSync(dir)) {
|
||||||
|
mkdirSync(dir, { recursive: true });
|
||||||
|
}
|
||||||
|
writeFileSync(statePath, JSON.stringify(state, null, 2));
|
||||||
|
} catch (error) {
|
||||||
|
logger.warn('TRANSCRIPT', 'Failed to save watch state', {
|
||||||
|
statePath,
|
||||||
|
error: error instanceof Error ? error.message : String(error)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,70 @@
|
|||||||
|
export type FieldSpec =
|
||||||
|
| string
|
||||||
|
| {
|
||||||
|
path?: string;
|
||||||
|
value?: unknown;
|
||||||
|
coalesce?: FieldSpec[];
|
||||||
|
default?: unknown;
|
||||||
|
};
|
||||||
|
|
||||||
|
export interface MatchRule {
|
||||||
|
path?: string;
|
||||||
|
equals?: unknown;
|
||||||
|
in?: unknown[];
|
||||||
|
contains?: string;
|
||||||
|
exists?: boolean;
|
||||||
|
regex?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type EventAction =
|
||||||
|
| 'session_init'
|
||||||
|
| 'session_context'
|
||||||
|
| 'user_message'
|
||||||
|
| 'assistant_message'
|
||||||
|
| 'tool_use'
|
||||||
|
| 'tool_result'
|
||||||
|
| 'observation'
|
||||||
|
| 'file_edit'
|
||||||
|
| 'session_end';
|
||||||
|
|
||||||
|
export interface SchemaEvent {
|
||||||
|
name: string;
|
||||||
|
match?: MatchRule;
|
||||||
|
action: EventAction;
|
||||||
|
fields?: Record<string, FieldSpec>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface TranscriptSchema {
|
||||||
|
name: string;
|
||||||
|
version?: string;
|
||||||
|
description?: string;
|
||||||
|
eventTypePath?: string;
|
||||||
|
sessionIdPath?: string;
|
||||||
|
cwdPath?: string;
|
||||||
|
projectPath?: string;
|
||||||
|
events: SchemaEvent[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface WatchContextConfig {
|
||||||
|
mode: 'agents';
|
||||||
|
path?: string;
|
||||||
|
updateOn?: Array<'session_start' | 'session_end'>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface WatchTarget {
|
||||||
|
name: string;
|
||||||
|
path: string;
|
||||||
|
schema: string | TranscriptSchema;
|
||||||
|
workspace?: string;
|
||||||
|
project?: string;
|
||||||
|
context?: WatchContextConfig;
|
||||||
|
rescanIntervalMs?: number;
|
||||||
|
startAtEnd?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface TranscriptWatchConfig {
|
||||||
|
version: 1;
|
||||||
|
schemas?: Record<string, TranscriptSchema>;
|
||||||
|
watches: WatchTarget[];
|
||||||
|
stateFile?: string;
|
||||||
|
}
|
||||||
@@ -0,0 +1,224 @@
|
|||||||
|
import { existsSync, statSync, watch as fsWatch, createReadStream } from 'fs';
|
||||||
|
import { basename, join } from 'path';
|
||||||
|
import { globSync } from 'glob';
|
||||||
|
import { logger } from '../../utils/logger.js';
|
||||||
|
import { expandHomePath } from './config.js';
|
||||||
|
import { loadWatchState, saveWatchState, type TranscriptWatchState } from './state.js';
|
||||||
|
import type { TranscriptWatchConfig, TranscriptSchema, WatchTarget } from './types.js';
|
||||||
|
import { TranscriptEventProcessor } from './processor.js';
|
||||||
|
|
||||||
|
interface TailState {
|
||||||
|
offset: number;
|
||||||
|
partial: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
class FileTailer {
|
||||||
|
private watcher: ReturnType<typeof fsWatch> | null = null;
|
||||||
|
private tailState: TailState;
|
||||||
|
|
||||||
|
constructor(
|
||||||
|
private filePath: string,
|
||||||
|
initialOffset: number,
|
||||||
|
private onLine: (line: string) => Promise<void>,
|
||||||
|
private onOffset: (offset: number) => void
|
||||||
|
) {
|
||||||
|
this.tailState = { offset: initialOffset, partial: '' };
|
||||||
|
}
|
||||||
|
|
||||||
|
start(): void {
|
||||||
|
this.readNewData().catch(() => undefined);
|
||||||
|
this.watcher = fsWatch(this.filePath, { persistent: true }, () => {
|
||||||
|
this.readNewData().catch(() => undefined);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
close(): void {
|
||||||
|
this.watcher?.close();
|
||||||
|
this.watcher = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async readNewData(): Promise<void> {
|
||||||
|
if (!existsSync(this.filePath)) return;
|
||||||
|
|
||||||
|
let size = 0;
|
||||||
|
try {
|
||||||
|
size = statSync(this.filePath).size;
|
||||||
|
} catch {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (size < this.tailState.offset) {
|
||||||
|
this.tailState.offset = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (size === this.tailState.offset) return;
|
||||||
|
|
||||||
|
const stream = createReadStream(this.filePath, {
|
||||||
|
start: this.tailState.offset,
|
||||||
|
end: size - 1,
|
||||||
|
encoding: 'utf8'
|
||||||
|
});
|
||||||
|
|
||||||
|
let data = '';
|
||||||
|
for await (const chunk of stream) {
|
||||||
|
data += chunk as string;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.tailState.offset = size;
|
||||||
|
this.onOffset(this.tailState.offset);
|
||||||
|
|
||||||
|
const combined = this.tailState.partial + data;
|
||||||
|
const lines = combined.split('\n');
|
||||||
|
this.tailState.partial = lines.pop() ?? '';
|
||||||
|
|
||||||
|
for (const line of lines) {
|
||||||
|
const trimmed = line.trim();
|
||||||
|
if (!trimmed) continue;
|
||||||
|
await this.onLine(trimmed);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class TranscriptWatcher {
|
||||||
|
private processor = new TranscriptEventProcessor();
|
||||||
|
private tailers = new Map<string, FileTailer>();
|
||||||
|
private state: TranscriptWatchState;
|
||||||
|
private rescanTimers: Array<NodeJS.Timeout> = [];
|
||||||
|
|
||||||
|
constructor(private config: TranscriptWatchConfig, private statePath: string) {
|
||||||
|
this.state = loadWatchState(statePath);
|
||||||
|
}
|
||||||
|
|
||||||
|
async start(): Promise<void> {
|
||||||
|
for (const watch of this.config.watches) {
|
||||||
|
await this.setupWatch(watch);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
stop(): void {
|
||||||
|
for (const tailer of this.tailers.values()) {
|
||||||
|
tailer.close();
|
||||||
|
}
|
||||||
|
this.tailers.clear();
|
||||||
|
for (const timer of this.rescanTimers) {
|
||||||
|
clearInterval(timer);
|
||||||
|
}
|
||||||
|
this.rescanTimers = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
private async setupWatch(watch: WatchTarget): Promise<void> {
|
||||||
|
const schema = this.resolveSchema(watch);
|
||||||
|
if (!schema) {
|
||||||
|
logger.warn('TRANSCRIPT', 'Missing schema for watch', { watch: watch.name });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const resolvedPath = expandHomePath(watch.path);
|
||||||
|
const files = this.resolveWatchFiles(resolvedPath);
|
||||||
|
|
||||||
|
for (const filePath of files) {
|
||||||
|
await this.addTailer(filePath, watch, schema);
|
||||||
|
}
|
||||||
|
|
||||||
|
const rescanIntervalMs = watch.rescanIntervalMs ?? 5000;
|
||||||
|
const timer = setInterval(async () => {
|
||||||
|
const newFiles = this.resolveWatchFiles(resolvedPath);
|
||||||
|
for (const filePath of newFiles) {
|
||||||
|
if (!this.tailers.has(filePath)) {
|
||||||
|
await this.addTailer(filePath, watch, schema);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, rescanIntervalMs);
|
||||||
|
this.rescanTimers.push(timer);
|
||||||
|
}
|
||||||
|
|
||||||
|
private resolveSchema(watch: WatchTarget): TranscriptSchema | null {
|
||||||
|
if (typeof watch.schema === 'string') {
|
||||||
|
return this.config.schemas?.[watch.schema] ?? null;
|
||||||
|
}
|
||||||
|
return watch.schema;
|
||||||
|
}
|
||||||
|
|
||||||
|
private resolveWatchFiles(inputPath: string): string[] {
|
||||||
|
if (this.hasGlob(inputPath)) {
|
||||||
|
return globSync(inputPath, { nodir: true, absolute: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (existsSync(inputPath)) {
|
||||||
|
try {
|
||||||
|
const stat = statSync(inputPath);
|
||||||
|
if (stat.isDirectory()) {
|
||||||
|
const pattern = join(inputPath, '**', '*.jsonl');
|
||||||
|
return globSync(pattern, { nodir: true, absolute: true });
|
||||||
|
}
|
||||||
|
return [inputPath];
|
||||||
|
} catch {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
|
||||||
|
private hasGlob(inputPath: string): boolean {
|
||||||
|
return /[*?[\]{}()]/.test(inputPath);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async addTailer(filePath: string, watch: WatchTarget, schema: TranscriptSchema): Promise<void> {
|
||||||
|
if (this.tailers.has(filePath)) return;
|
||||||
|
|
||||||
|
const sessionIdOverride = this.extractSessionIdFromPath(filePath);
|
||||||
|
|
||||||
|
let offset = this.state.offsets[filePath] ?? 0;
|
||||||
|
if (offset === 0 && watch.startAtEnd) {
|
||||||
|
try {
|
||||||
|
offset = statSync(filePath).size;
|
||||||
|
} catch {
|
||||||
|
offset = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const tailer = new FileTailer(
|
||||||
|
filePath,
|
||||||
|
offset,
|
||||||
|
async (line: string) => {
|
||||||
|
await this.handleLine(line, watch, schema, filePath, sessionIdOverride);
|
||||||
|
},
|
||||||
|
(newOffset: number) => {
|
||||||
|
this.state.offsets[filePath] = newOffset;
|
||||||
|
saveWatchState(this.statePath, this.state);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
tailer.start();
|
||||||
|
this.tailers.set(filePath, tailer);
|
||||||
|
logger.info('TRANSCRIPT', 'Watching transcript file', {
|
||||||
|
file: filePath,
|
||||||
|
watch: watch.name,
|
||||||
|
schema: schema.name
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private async handleLine(
|
||||||
|
line: string,
|
||||||
|
watch: WatchTarget,
|
||||||
|
schema: TranscriptSchema,
|
||||||
|
filePath: string,
|
||||||
|
sessionIdOverride?: string | null
|
||||||
|
): Promise<void> {
|
||||||
|
try {
|
||||||
|
const entry = JSON.parse(line);
|
||||||
|
await this.processor.processEntry(entry, watch, schema, sessionIdOverride ?? undefined);
|
||||||
|
} catch (error) {
|
||||||
|
logger.debug('TRANSCRIPT', 'Failed to parse transcript line', {
|
||||||
|
watch: watch.name,
|
||||||
|
file: basename(filePath)
|
||||||
|
}, error as Error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private extractSessionIdFromPath(filePath: string): string | null {
|
||||||
|
const match = filePath.match(/[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}/i);
|
||||||
|
return match ? match[0] : null;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -14,6 +14,7 @@ import { existsSync, writeFileSync, unlinkSync, statSync } from 'fs';
|
|||||||
import { Client } from '@modelcontextprotocol/sdk/client/index.js';
|
import { Client } from '@modelcontextprotocol/sdk/client/index.js';
|
||||||
import { StdioClientTransport } from '@modelcontextprotocol/sdk/client/stdio.js';
|
import { StdioClientTransport } from '@modelcontextprotocol/sdk/client/stdio.js';
|
||||||
import { getWorkerPort, getWorkerHost } from '../shared/worker-utils.js';
|
import { getWorkerPort, getWorkerHost } from '../shared/worker-utils.js';
|
||||||
|
import { HOOK_TIMEOUTS } from '../shared/hook-constants.js';
|
||||||
import { SettingsDefaultsManager } from '../shared/SettingsDefaultsManager.js';
|
import { SettingsDefaultsManager } from '../shared/SettingsDefaultsManager.js';
|
||||||
import { logger } from '../utils/logger.js';
|
import { logger } from '../utils/logger.js';
|
||||||
|
|
||||||
@@ -67,6 +68,7 @@ import {
|
|||||||
getPlatformTimeout,
|
getPlatformTimeout,
|
||||||
cleanupOrphanedProcesses,
|
cleanupOrphanedProcesses,
|
||||||
cleanupExcessChromaProcesses,
|
cleanupExcessChromaProcesses,
|
||||||
|
cleanStalePidFile,
|
||||||
spawnDaemon,
|
spawnDaemon,
|
||||||
createSignalHandler
|
createSignalHandler
|
||||||
} from './infrastructure/ProcessManager.js';
|
} from './infrastructure/ProcessManager.js';
|
||||||
@@ -230,6 +232,22 @@ export class WorkerService {
|
|||||||
this.isShuttingDown = shutdownRef.value;
|
this.isShuttingDown = shutdownRef.value;
|
||||||
handler('SIGINT');
|
handler('SIGINT');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// SIGHUP: sent by kernel when controlling terminal closes.
|
||||||
|
// Daemon mode: ignore it (survive parent shell exit).
|
||||||
|
// Interactive mode: treat like SIGTERM (graceful shutdown).
|
||||||
|
if (process.platform !== 'win32') {
|
||||||
|
if (process.argv.includes('--daemon')) {
|
||||||
|
process.on('SIGHUP', () => {
|
||||||
|
logger.debug('SYSTEM', 'Ignoring SIGHUP in daemon mode');
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
process.on('SIGHUP', () => {
|
||||||
|
this.isShuttingDown = shutdownRef.value;
|
||||||
|
handler('SIGHUP');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -748,6 +766,9 @@ export class WorkerService {
|
|||||||
* @returns true if worker is healthy (existing or newly started), false on failure
|
* @returns true if worker is healthy (existing or newly started), false on failure
|
||||||
*/
|
*/
|
||||||
async function ensureWorkerStarted(port: number): Promise<boolean> {
|
async function ensureWorkerStarted(port: number): Promise<boolean> {
|
||||||
|
// Clean stale PID file first (cheap: 1 fs read + 1 signal-0 check)
|
||||||
|
cleanStalePidFile();
|
||||||
|
|
||||||
// Check if worker is already running and healthy
|
// Check if worker is already running and healthy
|
||||||
if (await waitForHealth(port, 1000)) {
|
if (await waitForHealth(port, 1000)) {
|
||||||
const versionCheck = await checkVersionMatch(port);
|
const versionCheck = await checkVersionMatch(port);
|
||||||
@@ -758,7 +779,7 @@ async function ensureWorkerStarted(port: number): Promise<boolean> {
|
|||||||
});
|
});
|
||||||
|
|
||||||
await httpShutdown(port);
|
await httpShutdown(port);
|
||||||
const freed = await waitForPortFree(port, getPlatformTimeout(15000));
|
const freed = await waitForPortFree(port, getPlatformTimeout(HOOK_TIMEOUTS.PORT_IN_USE_WAIT));
|
||||||
if (!freed) {
|
if (!freed) {
|
||||||
logger.error('SYSTEM', 'Port did not free up after shutdown for version mismatch restart', { port });
|
logger.error('SYSTEM', 'Port did not free up after shutdown for version mismatch restart', { port });
|
||||||
return false;
|
return false;
|
||||||
@@ -774,7 +795,7 @@ async function ensureWorkerStarted(port: number): Promise<boolean> {
|
|||||||
const portInUse = await isPortInUse(port);
|
const portInUse = await isPortInUse(port);
|
||||||
if (portInUse) {
|
if (portInUse) {
|
||||||
logger.info('SYSTEM', 'Port in use, waiting for worker to become healthy');
|
logger.info('SYSTEM', 'Port in use, waiting for worker to become healthy');
|
||||||
const healthy = await waitForHealth(port, getPlatformTimeout(15000));
|
const healthy = await waitForHealth(port, getPlatformTimeout(HOOK_TIMEOUTS.PORT_IN_USE_WAIT));
|
||||||
if (healthy) {
|
if (healthy) {
|
||||||
logger.info('SYSTEM', 'Worker is now healthy');
|
logger.info('SYSTEM', 'Worker is now healthy');
|
||||||
return true;
|
return true;
|
||||||
@@ -801,7 +822,7 @@ async function ensureWorkerStarted(port: number): Promise<boolean> {
|
|||||||
// PID file is written by the worker itself after listen() succeeds
|
// PID file is written by the worker itself after listen() succeeds
|
||||||
// This is race-free and works correctly on Windows where cmd.exe PID is useless
|
// This is race-free and works correctly on Windows where cmd.exe PID is useless
|
||||||
|
|
||||||
const healthy = await waitForHealth(port, getPlatformTimeout(30000));
|
const healthy = await waitForHealth(port, getPlatformTimeout(HOOK_TIMEOUTS.POST_SPAWN_WAIT));
|
||||||
if (!healthy) {
|
if (!healthy) {
|
||||||
removePidFile();
|
removePidFile();
|
||||||
logger.error('SYSTEM', 'Worker failed to start (health check timeout)');
|
logger.error('SYSTEM', 'Worker failed to start (health check timeout)');
|
||||||
@@ -873,7 +894,7 @@ async function main() {
|
|||||||
// PID file is written by the worker itself after listen() succeeds
|
// PID file is written by the worker itself after listen() succeeds
|
||||||
// This is race-free and works correctly on Windows where cmd.exe PID is useless
|
// This is race-free and works correctly on Windows where cmd.exe PID is useless
|
||||||
|
|
||||||
const healthy = await waitForHealth(port, getPlatformTimeout(30000));
|
const healthy = await waitForHealth(port, getPlatformTimeout(HOOK_TIMEOUTS.POST_SPAWN_WAIT));
|
||||||
if (!healthy) {
|
if (!healthy) {
|
||||||
removePidFile();
|
removePidFile();
|
||||||
logger.error('SYSTEM', 'Worker failed to restart');
|
logger.error('SYSTEM', 'Worker failed to restart');
|
||||||
@@ -968,6 +989,18 @@ async function main() {
|
|||||||
|
|
||||||
case '--daemon':
|
case '--daemon':
|
||||||
default: {
|
default: {
|
||||||
|
// Prevent daemon from dying silently on unhandled errors.
|
||||||
|
// The HTTP server can continue serving even if a background task throws.
|
||||||
|
process.on('unhandledRejection', (reason) => {
|
||||||
|
logger.error('SYSTEM', 'Unhandled rejection in daemon', {
|
||||||
|
reason: reason instanceof Error ? reason.message : String(reason)
|
||||||
|
});
|
||||||
|
});
|
||||||
|
process.on('uncaughtException', (error) => {
|
||||||
|
logger.error('SYSTEM', 'Uncaught exception in daemon', {}, error as Error);
|
||||||
|
// Don't exit — keep the HTTP server running
|
||||||
|
});
|
||||||
|
|
||||||
const worker = new WorkerService();
|
const worker = new WorkerService();
|
||||||
worker.start().catch((error) => {
|
worker.start().catch((error) => {
|
||||||
logger.failure('SYSTEM', 'Worker failed to start', {}, error as Error);
|
logger.failure('SYSTEM', 'Worker failed to start', {}, error as Error);
|
||||||
|
|||||||
@@ -1,11 +1,12 @@
|
|||||||
export const HOOK_TIMEOUTS = {
|
export const HOOK_TIMEOUTS = {
|
||||||
DEFAULT: 300000, // Standard HTTP timeout (5 min for slow systems)
|
DEFAULT: 300000, // Standard HTTP timeout (5 min for slow systems)
|
||||||
HEALTH_CHECK: 30000, // Worker health check (30s for slow systems)
|
HEALTH_CHECK: 3000, // Worker health check (3s — healthy worker responds in <100ms)
|
||||||
|
POST_SPAWN_WAIT: 5000, // Wait for daemon to start after spawn (starts in <1s on Linux)
|
||||||
|
PORT_IN_USE_WAIT: 3000, // Wait when port occupied but health failing
|
||||||
WORKER_STARTUP_WAIT: 1000,
|
WORKER_STARTUP_WAIT: 1000,
|
||||||
WORKER_STARTUP_RETRIES: 300,
|
|
||||||
PRE_RESTART_SETTLE_DELAY: 2000, // Give files time to sync before restart
|
PRE_RESTART_SETTLE_DELAY: 2000, // Give files time to sync before restart
|
||||||
POWERSHELL_COMMAND: 10000, // PowerShell process enumeration (10s - typically completes in <1s)
|
POWERSHELL_COMMAND: 10000, // PowerShell process enumeration (10s - typically completes in <1s)
|
||||||
WINDOWS_MULTIPLIER: 1.5 // Platform-specific adjustment
|
WINDOWS_MULTIPLIER: 1.5 // Platform-specific adjustment for hook-side operations
|
||||||
} as const;
|
} as const;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
+53
-14
@@ -6,7 +6,21 @@ import { SettingsDefaultsManager } from "./SettingsDefaultsManager.js";
|
|||||||
import { MARKETPLACE_ROOT } from "./paths.js";
|
import { MARKETPLACE_ROOT } from "./paths.js";
|
||||||
|
|
||||||
// Named constants for health checks
|
// Named constants for health checks
|
||||||
const HEALTH_CHECK_TIMEOUT_MS = getTimeout(HOOK_TIMEOUTS.HEALTH_CHECK);
|
// Allow env var override for users on slow systems (e.g., CLAUDE_MEM_HEALTH_TIMEOUT_MS=10000)
|
||||||
|
const HEALTH_CHECK_TIMEOUT_MS = (() => {
|
||||||
|
const envVal = process.env.CLAUDE_MEM_HEALTH_TIMEOUT_MS;
|
||||||
|
if (envVal) {
|
||||||
|
const parsed = parseInt(envVal, 10);
|
||||||
|
if (Number.isFinite(parsed) && parsed >= 500 && parsed <= 300000) {
|
||||||
|
return parsed;
|
||||||
|
}
|
||||||
|
// Invalid env var — log once and use default
|
||||||
|
logger.warn('SYSTEM', 'Invalid CLAUDE_MEM_HEALTH_TIMEOUT_MS, using default', {
|
||||||
|
value: envVal, min: 500, max: 300000
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return getTimeout(HOOK_TIMEOUTS.HEALTH_CHECK);
|
||||||
|
})();
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Fetch with a timeout using Promise.race instead of AbortSignal.
|
* Fetch with a timeout using Promise.race instead of AbortSignal.
|
||||||
@@ -89,12 +103,22 @@ async function isWorkerHealthy(): Promise<boolean> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the current plugin version from package.json
|
* Get the current plugin version from package.json.
|
||||||
|
* Returns 'unknown' on ENOENT/EBUSY (shutdown race condition, fix #1042).
|
||||||
*/
|
*/
|
||||||
function getPluginVersion(): string {
|
function getPluginVersion(): string {
|
||||||
const packageJsonPath = path.join(MARKETPLACE_ROOT, 'package.json');
|
try {
|
||||||
const packageJson = JSON.parse(readFileSync(packageJsonPath, 'utf-8'));
|
const packageJsonPath = path.join(MARKETPLACE_ROOT, 'package.json');
|
||||||
return packageJson.version;
|
const packageJson = JSON.parse(readFileSync(packageJsonPath, 'utf-8'));
|
||||||
|
return packageJson.version;
|
||||||
|
} catch (error: unknown) {
|
||||||
|
const code = (error as NodeJS.ErrnoException).code;
|
||||||
|
if (code === 'ENOENT' || code === 'EBUSY') {
|
||||||
|
logger.debug('SYSTEM', 'Could not read plugin version (shutdown race)', { code });
|
||||||
|
return 'unknown';
|
||||||
|
}
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -115,18 +139,33 @@ async function getWorkerVersion(): Promise<string> {
|
|||||||
/**
|
/**
|
||||||
* Check if worker version matches plugin version
|
* Check if worker version matches plugin version
|
||||||
* Note: Auto-restart on version mismatch is now handled in worker-service.ts start command (issue #484)
|
* Note: Auto-restart on version mismatch is now handled in worker-service.ts start command (issue #484)
|
||||||
* This function logs for informational purposes only
|
* This function logs for informational purposes only.
|
||||||
|
* Skips comparison when either version is 'unknown' (fix #1042 — avoids restart loops).
|
||||||
*/
|
*/
|
||||||
async function checkWorkerVersion(): Promise<void> {
|
async function checkWorkerVersion(): Promise<void> {
|
||||||
const pluginVersion = getPluginVersion();
|
try {
|
||||||
const workerVersion = await getWorkerVersion();
|
const pluginVersion = getPluginVersion();
|
||||||
|
|
||||||
if (pluginVersion !== workerVersion) {
|
// Skip version check if plugin version couldn't be read (shutdown race)
|
||||||
// Just log debug info - auto-restart handles the mismatch in worker-service.ts
|
if (pluginVersion === 'unknown') return;
|
||||||
logger.debug('SYSTEM', 'Version check', {
|
|
||||||
pluginVersion,
|
const workerVersion = await getWorkerVersion();
|
||||||
workerVersion,
|
|
||||||
note: 'Mismatch will be auto-restarted by worker-service start command'
|
// Skip version check if worker version is 'unknown' (avoids restart loops)
|
||||||
|
if (workerVersion === 'unknown') return;
|
||||||
|
|
||||||
|
if (pluginVersion !== workerVersion) {
|
||||||
|
// Just log debug info - auto-restart handles the mismatch in worker-service.ts
|
||||||
|
logger.debug('SYSTEM', 'Version check', {
|
||||||
|
pluginVersion,
|
||||||
|
workerVersion,
|
||||||
|
note: 'Mismatch will be auto-restarted by worker-service start command'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
// Version check is informational — don't fail the hook
|
||||||
|
logger.debug('SYSTEM', 'Version check failed', {
|
||||||
|
error: error instanceof Error ? error.message : String(error)
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,33 @@
|
|||||||
|
import { existsSync, readFileSync, writeFileSync, renameSync, mkdirSync } from 'fs';
|
||||||
|
import { dirname } from 'path';
|
||||||
|
import { replaceTaggedContent } from './claude-md-utils.js';
|
||||||
|
import { logger } from './logger.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Write AGENTS.md with claude-mem context, preserving user content outside tags.
|
||||||
|
* Uses atomic write to prevent partial writes.
|
||||||
|
*/
|
||||||
|
export function writeAgentsMd(agentsPath: string, context: string): void {
|
||||||
|
if (!agentsPath) return;
|
||||||
|
|
||||||
|
const dir = dirname(agentsPath);
|
||||||
|
if (!existsSync(dir)) {
|
||||||
|
mkdirSync(dir, { recursive: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
let existingContent = '';
|
||||||
|
if (existsSync(agentsPath)) {
|
||||||
|
existingContent = readFileSync(agentsPath, 'utf-8');
|
||||||
|
}
|
||||||
|
|
||||||
|
const contentBlock = `# Memory Context\n\n${context}`;
|
||||||
|
const finalContent = replaceTaggedContent(existingContent, contentBlock);
|
||||||
|
const tempFile = `${agentsPath}.tmp`;
|
||||||
|
|
||||||
|
try {
|
||||||
|
writeFileSync(tempFile, finalContent);
|
||||||
|
renameSync(tempFile, agentsPath);
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('AGENTS_MD', 'Failed to write AGENTS.md', { agentsPath }, error as Error);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,164 @@
|
|||||||
|
/**
|
||||||
|
* Tests for hook-command error classifier
|
||||||
|
*
|
||||||
|
* Validates that isWorkerUnavailableError correctly distinguishes between:
|
||||||
|
* - Transport failures (ECONNREFUSED, etc.) → true (graceful degradation)
|
||||||
|
* - Server errors (5xx) → true (graceful degradation)
|
||||||
|
* - Client errors (4xx) → false (handler bug, blocking)
|
||||||
|
* - Programming errors (TypeError, etc.) → false (code bug, blocking)
|
||||||
|
*/
|
||||||
|
import { describe, it, expect } from 'bun:test';
|
||||||
|
import { isWorkerUnavailableError } from '../src/cli/hook-command.js';
|
||||||
|
|
||||||
|
describe('isWorkerUnavailableError', () => {
|
||||||
|
describe('transport failures → true (graceful)', () => {
|
||||||
|
it('should classify ECONNREFUSED as worker unavailable', () => {
|
||||||
|
const error = new Error('connect ECONNREFUSED 127.0.0.1:37777');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should classify ECONNRESET as worker unavailable', () => {
|
||||||
|
const error = new Error('socket hang up ECONNRESET');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should classify EPIPE as worker unavailable', () => {
|
||||||
|
const error = new Error('write EPIPE');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should classify ETIMEDOUT as worker unavailable', () => {
|
||||||
|
const error = new Error('connect ETIMEDOUT 127.0.0.1:37777');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should classify "fetch failed" as worker unavailable', () => {
|
||||||
|
const error = new TypeError('fetch failed');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should classify "Unable to connect" as worker unavailable', () => {
|
||||||
|
const error = new Error('Unable to connect to server');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should classify ENOTFOUND as worker unavailable', () => {
|
||||||
|
const error = new Error('getaddrinfo ENOTFOUND localhost');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should classify "socket hang up" as worker unavailable', () => {
|
||||||
|
const error = new Error('socket hang up');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should classify ECONNABORTED as worker unavailable', () => {
|
||||||
|
const error = new Error('ECONNABORTED');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('timeout errors → true (graceful)', () => {
|
||||||
|
it('should classify "timed out" as worker unavailable', () => {
|
||||||
|
const error = new Error('Request timed out after 3000ms');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should classify "timeout" as worker unavailable', () => {
|
||||||
|
const error = new Error('Connection timeout');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('HTTP 5xx server errors → true (graceful)', () => {
|
||||||
|
it('should classify 500 status as worker unavailable', () => {
|
||||||
|
const error = new Error('Context generation failed: 500');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should classify 502 status as worker unavailable', () => {
|
||||||
|
const error = new Error('Observation storage failed: 502');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should classify 503 status as worker unavailable', () => {
|
||||||
|
const error = new Error('Request failed: 503');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should classify "status: 500" format as worker unavailable', () => {
|
||||||
|
const error = new Error('HTTP error status: 500');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('HTTP 429 rate limit → true (graceful)', () => {
|
||||||
|
it('should classify 429 as worker unavailable (rate limit is transient)', () => {
|
||||||
|
const error = new Error('Request failed: 429');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should classify "status: 429" format as worker unavailable', () => {
|
||||||
|
const error = new Error('HTTP error status: 429');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(true);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('HTTP 4xx client errors → false (blocking)', () => {
|
||||||
|
it('should NOT classify 400 Bad Request as worker unavailable', () => {
|
||||||
|
const error = new Error('Request failed: 400');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should NOT classify 404 Not Found as worker unavailable', () => {
|
||||||
|
const error = new Error('Observation storage failed: 404');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should NOT classify 422 Validation Error as worker unavailable', () => {
|
||||||
|
const error = new Error('Request failed: 422');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should NOT classify "status: 400" format as worker unavailable', () => {
|
||||||
|
const error = new Error('HTTP error status: 400');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(false);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('programming errors → false (blocking)', () => {
|
||||||
|
it('should NOT classify TypeError as worker unavailable', () => {
|
||||||
|
const error = new TypeError('Cannot read properties of undefined');
|
||||||
|
// Note: TypeError with "fetch failed" IS classified as unavailable (transport layer)
|
||||||
|
// But generic TypeErrors are NOT
|
||||||
|
expect(isWorkerUnavailableError(new TypeError('Cannot read properties of undefined'))).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should NOT classify ReferenceError as worker unavailable', () => {
|
||||||
|
const error = new ReferenceError('foo is not defined');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should NOT classify SyntaxError as worker unavailable', () => {
|
||||||
|
const error = new SyntaxError('Unexpected token');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(false);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('unknown errors → false (blocking, conservative)', () => {
|
||||||
|
it('should NOT classify generic Error as worker unavailable', () => {
|
||||||
|
const error = new Error('Something unexpected happened');
|
||||||
|
expect(isWorkerUnavailableError(error)).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should handle string errors', () => {
|
||||||
|
expect(isWorkerUnavailableError('ECONNREFUSED')).toBe(true);
|
||||||
|
expect(isWorkerUnavailableError('random error')).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should handle null/undefined errors', () => {
|
||||||
|
expect(isWorkerUnavailableError(null)).toBe(false);
|
||||||
|
expect(isWorkerUnavailableError(undefined)).toBe(false);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -28,18 +28,22 @@ describe('hook-constants', () => {
|
|||||||
expect(HOOK_TIMEOUTS.DEFAULT).toBe(300000);
|
expect(HOOK_TIMEOUTS.DEFAULT).toBe(300000);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should define HEALTH_CHECK timeout', () => {
|
it('should define HEALTH_CHECK timeout as 3s (reduced from 30s)', () => {
|
||||||
expect(HOOK_TIMEOUTS.HEALTH_CHECK).toBe(30000);
|
expect(HOOK_TIMEOUTS.HEALTH_CHECK).toBe(3000);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should define POST_SPAWN_WAIT as 5s', () => {
|
||||||
|
expect(HOOK_TIMEOUTS.POST_SPAWN_WAIT).toBe(5000);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should define PORT_IN_USE_WAIT as 3s', () => {
|
||||||
|
expect(HOOK_TIMEOUTS.PORT_IN_USE_WAIT).toBe(3000);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should define WORKER_STARTUP_WAIT', () => {
|
it('should define WORKER_STARTUP_WAIT', () => {
|
||||||
expect(HOOK_TIMEOUTS.WORKER_STARTUP_WAIT).toBe(1000);
|
expect(HOOK_TIMEOUTS.WORKER_STARTUP_WAIT).toBe(1000);
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should define WORKER_STARTUP_RETRIES', () => {
|
|
||||||
expect(HOOK_TIMEOUTS.WORKER_STARTUP_RETRIES).toBe(300);
|
|
||||||
});
|
|
||||||
|
|
||||||
it('should define PRE_RESTART_SETTLE_DELAY', () => {
|
it('should define PRE_RESTART_SETTLE_DELAY', () => {
|
||||||
expect(HOOK_TIMEOUTS.PRE_RESTART_SETTLE_DELAY).toBe(2000);
|
expect(HOOK_TIMEOUTS.PRE_RESTART_SETTLE_DELAY).toBe(2000);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -9,6 +9,9 @@ import {
|
|||||||
getPlatformTimeout,
|
getPlatformTimeout,
|
||||||
parseElapsedTime,
|
parseElapsedTime,
|
||||||
cleanupExcessChromaProcesses,
|
cleanupExcessChromaProcesses,
|
||||||
|
isProcessAlive,
|
||||||
|
cleanStalePidFile,
|
||||||
|
spawnDaemon,
|
||||||
type PidInfo
|
type PidInfo
|
||||||
} from '../../src/services/infrastructure/index.js';
|
} from '../../src/services/infrastructure/index.js';
|
||||||
|
|
||||||
@@ -281,4 +284,138 @@ describe('ProcessManager', () => {
|
|||||||
expect(sourceFile).toContain('.slice(maxAllowed)');
|
expect(sourceFile).toContain('.slice(maxAllowed)');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('isProcessAlive', () => {
|
||||||
|
it('should return true for the current process', () => {
|
||||||
|
expect(isProcessAlive(process.pid)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return false for a non-existent PID', () => {
|
||||||
|
// Use a very high PID that's extremely unlikely to exist
|
||||||
|
expect(isProcessAlive(2147483647)).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return true for PID 0 (Windows WMIC sentinel)', () => {
|
||||||
|
expect(isProcessAlive(0)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return false for negative PIDs', () => {
|
||||||
|
expect(isProcessAlive(-1)).toBe(false);
|
||||||
|
expect(isProcessAlive(-999)).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return false for non-integer PIDs', () => {
|
||||||
|
expect(isProcessAlive(1.5)).toBe(false);
|
||||||
|
expect(isProcessAlive(NaN)).toBe(false);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('cleanStalePidFile', () => {
|
||||||
|
it('should remove PID file when process is dead', () => {
|
||||||
|
// Write a PID file with a non-existent PID
|
||||||
|
const staleInfo: PidInfo = {
|
||||||
|
pid: 2147483647,
|
||||||
|
port: 37777,
|
||||||
|
startedAt: '2024-01-01T00:00:00.000Z'
|
||||||
|
};
|
||||||
|
writePidFile(staleInfo);
|
||||||
|
expect(existsSync(PID_FILE)).toBe(true);
|
||||||
|
|
||||||
|
cleanStalePidFile();
|
||||||
|
|
||||||
|
expect(existsSync(PID_FILE)).toBe(false);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should keep PID file when process is alive', () => {
|
||||||
|
// Write a PID file with the current process PID (definitely alive)
|
||||||
|
const liveInfo: PidInfo = {
|
||||||
|
pid: process.pid,
|
||||||
|
port: 37777,
|
||||||
|
startedAt: new Date().toISOString()
|
||||||
|
};
|
||||||
|
writePidFile(liveInfo);
|
||||||
|
|
||||||
|
cleanStalePidFile();
|
||||||
|
|
||||||
|
// PID file should still exist since process.pid is alive
|
||||||
|
expect(existsSync(PID_FILE)).toBe(true);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should do nothing when PID file does not exist', () => {
|
||||||
|
removePidFile();
|
||||||
|
expect(existsSync(PID_FILE)).toBe(false);
|
||||||
|
|
||||||
|
// Should not throw
|
||||||
|
expect(() => cleanStalePidFile()).not.toThrow();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('spawnDaemon', () => {
|
||||||
|
it('should use setsid on Linux when available', () => {
|
||||||
|
// setsid should exist at /usr/bin/setsid on Linux
|
||||||
|
if (process.platform === 'win32') return; // Skip on Windows
|
||||||
|
|
||||||
|
const setsidAvailable = existsSync('/usr/bin/setsid');
|
||||||
|
if (!setsidAvailable) return; // Skip if setsid not installed
|
||||||
|
|
||||||
|
// Spawn a daemon with a non-existent script (it will fail to start, but we can verify the spawn attempt)
|
||||||
|
// Use a harmless script path — the child will exit immediately
|
||||||
|
const pid = spawnDaemon('/dev/null', 39999);
|
||||||
|
|
||||||
|
// setsid spawn should return a PID (the setsid process itself)
|
||||||
|
expect(pid).toBeDefined();
|
||||||
|
expect(typeof pid).toBe('number');
|
||||||
|
|
||||||
|
// Clean up: kill the spawned process if it's still alive
|
||||||
|
if (pid !== undefined && pid > 0) {
|
||||||
|
try { process.kill(pid, 'SIGKILL'); } catch { /* already exited */ }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should return undefined when spawn fails on Windows path', () => {
|
||||||
|
// On non-Windows, this tests the Unix path which should succeed
|
||||||
|
// The function should not throw, only return undefined on failure
|
||||||
|
if (process.platform === 'win32') return;
|
||||||
|
|
||||||
|
// Spawning with a totally invalid script should still return a PID
|
||||||
|
// (setsid/spawn succeeds even if the child will exit immediately)
|
||||||
|
const result = spawnDaemon('/nonexistent/script.cjs', 39998);
|
||||||
|
// spawn itself should succeed (returns PID), even if child exits
|
||||||
|
expect(result).toBeDefined();
|
||||||
|
|
||||||
|
// Clean up
|
||||||
|
if (result !== undefined && result > 0) {
|
||||||
|
try { process.kill(result, 'SIGKILL'); } catch { /* already exited */ }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('SIGHUP handling', () => {
|
||||||
|
it('should have SIGHUP listeners registered (integration check)', () => {
|
||||||
|
// Verify that SIGHUP listener registration is possible on Unix
|
||||||
|
if (process.platform === 'win32') return;
|
||||||
|
|
||||||
|
// Register a test handler, verify it works, then remove it
|
||||||
|
let received = false;
|
||||||
|
const testHandler = () => { received = true; };
|
||||||
|
|
||||||
|
process.on('SIGHUP', testHandler);
|
||||||
|
expect(process.listenerCount('SIGHUP')).toBeGreaterThanOrEqual(1);
|
||||||
|
|
||||||
|
// Clean up the test handler
|
||||||
|
process.removeListener('SIGHUP', testHandler);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should ignore SIGHUP when --daemon is in process.argv', () => {
|
||||||
|
if (process.platform === 'win32') return;
|
||||||
|
|
||||||
|
// Simulate the daemon SIGHUP handler logic
|
||||||
|
const isDaemon = process.argv.includes('--daemon');
|
||||||
|
// In test context, --daemon is not in argv, so this tests the branch logic
|
||||||
|
expect(isDaemon).toBe(false);
|
||||||
|
|
||||||
|
// Verify the non-daemon path: SIGHUP should trigger shutdown (covered by registerSignalHandlers)
|
||||||
|
// This is a logic verification test — actual signal delivery is tested manually
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -0,0 +1,94 @@
|
|||||||
|
{
|
||||||
|
"version": 1,
|
||||||
|
"schemas": {
|
||||||
|
"codex": {
|
||||||
|
"name": "codex",
|
||||||
|
"version": "0.2",
|
||||||
|
"description": "Schema for Codex session JSONL files under ~/.codex/sessions.",
|
||||||
|
"events": [
|
||||||
|
{
|
||||||
|
"name": "session-meta",
|
||||||
|
"match": { "path": "type", "equals": "session_meta" },
|
||||||
|
"action": "session_context",
|
||||||
|
"fields": {
|
||||||
|
"sessionId": "payload.id",
|
||||||
|
"cwd": "payload.cwd"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "turn-context",
|
||||||
|
"match": { "path": "type", "equals": "turn_context" },
|
||||||
|
"action": "session_context",
|
||||||
|
"fields": {
|
||||||
|
"cwd": "payload.cwd"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "user-message",
|
||||||
|
"match": { "path": "payload.type", "equals": "user_message" },
|
||||||
|
"action": "session_init",
|
||||||
|
"fields": {
|
||||||
|
"prompt": "payload.message"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "assistant-message",
|
||||||
|
"match": { "path": "payload.type", "equals": "agent_message" },
|
||||||
|
"action": "assistant_message",
|
||||||
|
"fields": {
|
||||||
|
"message": "payload.message"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "tool-use",
|
||||||
|
"match": { "path": "payload.type", "in": ["function_call", "custom_tool_call", "web_search_call"] },
|
||||||
|
"action": "tool_use",
|
||||||
|
"fields": {
|
||||||
|
"toolId": "payload.call_id",
|
||||||
|
"toolName": {
|
||||||
|
"coalesce": [
|
||||||
|
"payload.name",
|
||||||
|
{ "value": "web_search" }
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"toolInput": {
|
||||||
|
"coalesce": [
|
||||||
|
"payload.arguments",
|
||||||
|
"payload.input",
|
||||||
|
"payload.action"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "tool-result",
|
||||||
|
"match": { "path": "payload.type", "in": ["function_call_output", "custom_tool_call_output"] },
|
||||||
|
"action": "tool_result",
|
||||||
|
"fields": {
|
||||||
|
"toolId": "payload.call_id",
|
||||||
|
"toolResponse": "payload.output"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "session-end",
|
||||||
|
"match": { "path": "payload.type", "equals": "turn_aborted" },
|
||||||
|
"action": "session_end"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"watches": [
|
||||||
|
{
|
||||||
|
"name": "codex",
|
||||||
|
"path": "~/.codex/sessions/**/*.jsonl",
|
||||||
|
"schema": "codex",
|
||||||
|
"startAtEnd": true,
|
||||||
|
"context": {
|
||||||
|
"mode": "agents",
|
||||||
|
"path": "~/.codex/AGENTS.md",
|
||||||
|
"updateOn": ["session_start", "session_end"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"stateFile": "~/.claude-mem/transcript-watch-state.json"
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user