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>
77 lines
1.9 KiB
JavaScript
77 lines
1.9 KiB
JavaScript
import { join, dirname, basename, isAbsolute, resolve } from 'pathe';
|
|
|
|
class SnapshotManager {
|
|
constructor(options) {
|
|
this.options = options;
|
|
this.clear();
|
|
}
|
|
summary = void 0;
|
|
extension = ".snap";
|
|
clear() {
|
|
this.summary = emptySummary(this.options);
|
|
}
|
|
add(result) {
|
|
addSnapshotResult(this.summary, result);
|
|
}
|
|
resolvePath(testPath) {
|
|
const resolver = this.options.resolveSnapshotPath || (() => {
|
|
return join(
|
|
join(dirname(testPath), "__snapshots__"),
|
|
`${basename(testPath)}${this.extension}`
|
|
);
|
|
});
|
|
const path = resolver(testPath, this.extension);
|
|
return path;
|
|
}
|
|
resolveRawPath(testPath, rawPath) {
|
|
return isAbsolute(rawPath) ? rawPath : resolve(dirname(testPath), rawPath);
|
|
}
|
|
}
|
|
function emptySummary(options) {
|
|
const summary = {
|
|
added: 0,
|
|
failure: false,
|
|
filesAdded: 0,
|
|
filesRemoved: 0,
|
|
filesRemovedList: [],
|
|
filesUnmatched: 0,
|
|
filesUpdated: 0,
|
|
matched: 0,
|
|
total: 0,
|
|
unchecked: 0,
|
|
uncheckedKeysByFile: [],
|
|
unmatched: 0,
|
|
updated: 0,
|
|
didUpdate: options.updateSnapshot === "all"
|
|
};
|
|
return summary;
|
|
}
|
|
function addSnapshotResult(summary, result) {
|
|
if (result.added) {
|
|
summary.filesAdded++;
|
|
}
|
|
if (result.fileDeleted) {
|
|
summary.filesRemoved++;
|
|
}
|
|
if (result.unmatched) {
|
|
summary.filesUnmatched++;
|
|
}
|
|
if (result.updated) {
|
|
summary.filesUpdated++;
|
|
}
|
|
summary.added += result.added;
|
|
summary.matched += result.matched;
|
|
summary.unchecked += result.unchecked;
|
|
if (result.uncheckedKeys && result.uncheckedKeys.length > 0) {
|
|
summary.uncheckedKeysByFile.push({
|
|
filePath: result.filepath,
|
|
keys: result.uncheckedKeys
|
|
});
|
|
}
|
|
summary.unmatched += result.unmatched;
|
|
summary.updated += result.updated;
|
|
summary.total += result.added + result.matched + result.unmatched + result.updated;
|
|
}
|
|
|
|
export { SnapshotManager, addSnapshotResult, emptySummary };
|