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>
123 lines
4.0 KiB
JavaScript
123 lines
4.0 KiB
JavaScript
"use strict";
|
|
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
};
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
exports.normalizeFgArgs = void 0;
|
|
exports.foregroundChild = foregroundChild;
|
|
const child_process_1 = require("child_process");
|
|
const cross_spawn_1 = __importDefault(require("cross-spawn"));
|
|
const signal_exit_1 = require("signal-exit");
|
|
const proxy_signals_js_1 = require("./proxy-signals.js");
|
|
const watchdog_js_1 = require("./watchdog.js");
|
|
/* c8 ignore start */
|
|
const spawn = process?.platform === 'win32' ? cross_spawn_1.default : child_process_1.spawn;
|
|
/**
|
|
* Normalizes the arguments passed to `foregroundChild`.
|
|
*
|
|
* Exposed for testing.
|
|
*
|
|
* @internal
|
|
*/
|
|
const normalizeFgArgs = (fgArgs) => {
|
|
let [program, args = [], spawnOpts = {}, cleanup = () => { }] = fgArgs;
|
|
if (typeof args === 'function') {
|
|
cleanup = args;
|
|
spawnOpts = {};
|
|
args = [];
|
|
}
|
|
else if (!!args && typeof args === 'object' && !Array.isArray(args)) {
|
|
if (typeof spawnOpts === 'function')
|
|
cleanup = spawnOpts;
|
|
spawnOpts = args;
|
|
args = [];
|
|
}
|
|
else if (typeof spawnOpts === 'function') {
|
|
cleanup = spawnOpts;
|
|
spawnOpts = {};
|
|
}
|
|
if (Array.isArray(program)) {
|
|
const [pp, ...pa] = program;
|
|
program = pp;
|
|
args = pa;
|
|
}
|
|
return [program, args, { ...spawnOpts }, cleanup];
|
|
};
|
|
exports.normalizeFgArgs = normalizeFgArgs;
|
|
function foregroundChild(...fgArgs) {
|
|
const [program, args, spawnOpts, cleanup] = (0, exports.normalizeFgArgs)(fgArgs);
|
|
spawnOpts.stdio = [0, 1, 2];
|
|
if (process.send) {
|
|
spawnOpts.stdio.push('ipc');
|
|
}
|
|
const child = spawn(program, args, spawnOpts);
|
|
const childHangup = () => {
|
|
try {
|
|
child.kill('SIGHUP');
|
|
/* c8 ignore start */
|
|
}
|
|
catch (_) {
|
|
// SIGHUP is weird on windows
|
|
child.kill('SIGTERM');
|
|
}
|
|
/* c8 ignore stop */
|
|
};
|
|
const removeOnExit = (0, signal_exit_1.onExit)(childHangup);
|
|
(0, proxy_signals_js_1.proxySignals)(child);
|
|
const dog = (0, watchdog_js_1.watchdog)(child);
|
|
let done = false;
|
|
child.on('close', async (code, signal) => {
|
|
/* c8 ignore start */
|
|
if (done)
|
|
return;
|
|
/* c8 ignore stop */
|
|
done = true;
|
|
const result = cleanup(code, signal, {
|
|
watchdogPid: dog.pid,
|
|
});
|
|
const res = isPromise(result) ? await result : result;
|
|
removeOnExit();
|
|
if (res === false)
|
|
return;
|
|
else if (typeof res === 'string') {
|
|
signal = res;
|
|
code = null;
|
|
}
|
|
else if (typeof res === 'number') {
|
|
code = res;
|
|
signal = null;
|
|
}
|
|
if (signal) {
|
|
// If there is nothing else keeping the event loop alive,
|
|
// then there's a race between a graceful exit and getting
|
|
// the signal to this process. Put this timeout here to
|
|
// make sure we're still alive to get the signal, and thus
|
|
// exit with the intended signal code.
|
|
/* istanbul ignore next */
|
|
setTimeout(() => { }, 2000);
|
|
try {
|
|
process.kill(process.pid, signal);
|
|
/* c8 ignore start */
|
|
}
|
|
catch (_) {
|
|
process.kill(process.pid, 'SIGTERM');
|
|
}
|
|
/* c8 ignore stop */
|
|
}
|
|
else {
|
|
process.exit(code || 0);
|
|
}
|
|
});
|
|
if (process.send) {
|
|
process.removeAllListeners('message');
|
|
child.on('message', (message, sendHandle) => {
|
|
process.send?.(message, sendHandle);
|
|
});
|
|
process.on('message', (message, sendHandle) => {
|
|
child.send(message, sendHandle);
|
|
});
|
|
}
|
|
return child;
|
|
}
|
|
const isPromise = (o) => !!o && typeof o === 'object' && typeof o.then === 'function';
|
|
//# sourceMappingURL=index.js.map
|