Implements Phases 1-8 of the TFTSR implementation plan. Rust backend (Tauri 2.x, src-tauri/): - Multi-provider AI: OpenAI-compatible, Anthropic, Gemini, Mistral, Ollama - PII detection engine: 11 regex patterns with overlap resolution - SQLCipher AES-256 encrypted database with 10 versioned migrations - 28 Tauri IPC commands for triage, analysis, document, and system ops - Ollama: hardware probe, model recommendations, pull/delete with events - RCA and blameless post-mortem Markdown document generators - PDF export via printpdf - Audit log: SHA-256 hash of every external data send - Integration stubs for Confluence, ServiceNow, Azure DevOps (v0.2) Frontend (React 18 + TypeScript + Vite, src/): - 9 pages: full triage workflow NewIssue→LogUpload→Triage→Resolution→RCA→Postmortem→History+Settings - 7 components: ChatWindow, TriageProgress, PiiDiffViewer, DocEditor, HardwareReport, ModelSelector, UI primitives - 3 Zustand stores: session, settings (persisted), history - Type-safe tauriCommands.ts matching Rust backend types exactly - 8 IT domain system prompts (Linux, Windows, Network, K8s, DB, Virt, HW, Obs) DevOps: - .woodpecker/test.yml: rustfmt, clippy, cargo test, tsc, vitest on every push - .woodpecker/release.yml: linux/amd64 + linux/arm64 builds, Gogs release upload Verified: - cargo check: zero errors - tsc --noEmit: zero errors - vitest run: 13/13 unit tests passing Co-Authored-By: Claude Sonnet 4.6 (1M context) <noreply@anthropic.com>
46 lines
1.4 KiB
JavaScript
46 lines
1.4 KiB
JavaScript
// this spawns a child process that listens for SIGHUP when the
|
|
// parent process exits, and after 200ms, sends a SIGKILL to the
|
|
// child, in case it did not terminate.
|
|
import { spawn } from 'child_process';
|
|
const watchdogCode = String.raw `
|
|
const pid = parseInt(process.argv[1], 10)
|
|
process.title = 'node (foreground-child watchdog pid=' + pid + ')'
|
|
if (!isNaN(pid)) {
|
|
let barked = false
|
|
// keepalive
|
|
const interval = setInterval(() => {}, 60000)
|
|
const bark = () => {
|
|
clearInterval(interval)
|
|
if (barked) return
|
|
barked = true
|
|
process.removeListener('SIGHUP', bark)
|
|
setTimeout(() => {
|
|
try {
|
|
process.kill(pid, 'SIGKILL')
|
|
setTimeout(() => process.exit(), 200)
|
|
} catch (_) {}
|
|
}, 500)
|
|
})
|
|
process.on('SIGHUP', bark)
|
|
}
|
|
`;
|
|
/**
|
|
* Pass in a ChildProcess, and this will spawn a watchdog process that
|
|
* will make sure it exits if the parent does, thus preventing any
|
|
* dangling detached zombie processes.
|
|
*
|
|
* If the child ends before the parent, then the watchdog will terminate.
|
|
*/
|
|
export const watchdog = (child) => {
|
|
let dogExited = false;
|
|
const dog = spawn(process.execPath, ['-e', watchdogCode, String(child.pid)], {
|
|
stdio: 'ignore',
|
|
});
|
|
dog.on('exit', () => (dogExited = true));
|
|
child.on('exit', () => {
|
|
if (!dogExited)
|
|
dog.kill('SIGKILL');
|
|
});
|
|
return dog;
|
|
};
|
|
//# sourceMappingURL=watchdog.js.map
|