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>
78 lines
2.2 KiB
JavaScript
78 lines
2.2 KiB
JavaScript
'use strict';
|
|
const singleComment = Symbol('singleComment');
|
|
const multiComment = Symbol('multiComment');
|
|
const stripWithoutWhitespace = () => '';
|
|
const stripWithWhitespace = (string, start, end) => string.slice(start, end).replace(/\S/g, ' ');
|
|
|
|
const isEscaped = (jsonString, quotePosition) => {
|
|
let index = quotePosition - 1;
|
|
let backslashCount = 0;
|
|
|
|
while (jsonString[index] === '\\') {
|
|
index -= 1;
|
|
backslashCount += 1;
|
|
}
|
|
|
|
return Boolean(backslashCount % 2);
|
|
};
|
|
|
|
module.exports = (jsonString, options = {}) => {
|
|
if (typeof jsonString !== 'string') {
|
|
throw new TypeError(`Expected argument \`jsonString\` to be a \`string\`, got \`${typeof jsonString}\``);
|
|
}
|
|
|
|
const strip = options.whitespace === false ? stripWithoutWhitespace : stripWithWhitespace;
|
|
|
|
let insideString = false;
|
|
let insideComment = false;
|
|
let offset = 0;
|
|
let result = '';
|
|
|
|
for (let i = 0; i < jsonString.length; i++) {
|
|
const currentCharacter = jsonString[i];
|
|
const nextCharacter = jsonString[i + 1];
|
|
|
|
if (!insideComment && currentCharacter === '"') {
|
|
const escaped = isEscaped(jsonString, i);
|
|
if (!escaped) {
|
|
insideString = !insideString;
|
|
}
|
|
}
|
|
|
|
if (insideString) {
|
|
continue;
|
|
}
|
|
|
|
if (!insideComment && currentCharacter + nextCharacter === '//') {
|
|
result += jsonString.slice(offset, i);
|
|
offset = i;
|
|
insideComment = singleComment;
|
|
i++;
|
|
} else if (insideComment === singleComment && currentCharacter + nextCharacter === '\r\n') {
|
|
i++;
|
|
insideComment = false;
|
|
result += strip(jsonString, offset, i);
|
|
offset = i;
|
|
continue;
|
|
} else if (insideComment === singleComment && currentCharacter === '\n') {
|
|
insideComment = false;
|
|
result += strip(jsonString, offset, i);
|
|
offset = i;
|
|
} else if (!insideComment && currentCharacter + nextCharacter === '/*') {
|
|
result += jsonString.slice(offset, i);
|
|
offset = i;
|
|
insideComment = multiComment;
|
|
i++;
|
|
continue;
|
|
} else if (insideComment === multiComment && currentCharacter + nextCharacter === '*/') {
|
|
i++;
|
|
insideComment = false;
|
|
result += strip(jsonString, offset, i + 1);
|
|
offset = i + 1;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
return result + (insideComment ? strip(jsonString.slice(offset)) : jsonString.slice(offset));
|
|
};
|