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>
142 lines
3.2 KiB
JavaScript
142 lines
3.2 KiB
JavaScript
/**
|
|
* @import {Chunk, Code, Encoding, Value} from 'micromark-util-types'
|
|
*/
|
|
|
|
/**
|
|
* @callback Preprocessor
|
|
* Preprocess a value.
|
|
* @param {Value} value
|
|
* Value.
|
|
* @param {Encoding | null | undefined} [encoding]
|
|
* Encoding when `value` is a typed array (optional).
|
|
* @param {boolean | null | undefined} [end=false]
|
|
* Whether this is the last chunk (default: `false`).
|
|
* @returns {Array<Chunk>}
|
|
* Chunks.
|
|
*/
|
|
|
|
import {codes, constants} from 'micromark-util-symbol'
|
|
|
|
const search = /[\0\t\n\r]/g
|
|
|
|
/**
|
|
* @returns {Preprocessor}
|
|
* Preprocess a value.
|
|
*/
|
|
export function preprocess() {
|
|
let column = 1
|
|
let buffer = ''
|
|
/** @type {boolean | undefined} */
|
|
let start = true
|
|
/** @type {boolean | undefined} */
|
|
let atCarriageReturn
|
|
|
|
return preprocessor
|
|
|
|
/** @type {Preprocessor} */
|
|
// eslint-disable-next-line complexity
|
|
function preprocessor(value, encoding, end) {
|
|
/** @type {Array<Chunk>} */
|
|
const chunks = []
|
|
/** @type {RegExpMatchArray | null} */
|
|
let match
|
|
/** @type {number} */
|
|
let next
|
|
/** @type {number} */
|
|
let startPosition
|
|
/** @type {number} */
|
|
let endPosition
|
|
/** @type {Code} */
|
|
let code
|
|
|
|
value =
|
|
buffer +
|
|
(typeof value === 'string'
|
|
? value.toString()
|
|
: new TextDecoder(encoding || undefined).decode(value))
|
|
|
|
startPosition = 0
|
|
buffer = ''
|
|
|
|
if (start) {
|
|
// To do: `markdown-rs` actually parses BOMs (byte order mark).
|
|
if (value.charCodeAt(0) === codes.byteOrderMarker) {
|
|
startPosition++
|
|
}
|
|
|
|
start = undefined
|
|
}
|
|
|
|
while (startPosition < value.length) {
|
|
search.lastIndex = startPosition
|
|
match = search.exec(value)
|
|
endPosition =
|
|
match && match.index !== undefined ? match.index : value.length
|
|
code = value.charCodeAt(endPosition)
|
|
|
|
if (!match) {
|
|
buffer = value.slice(startPosition)
|
|
break
|
|
}
|
|
|
|
if (
|
|
code === codes.lf &&
|
|
startPosition === endPosition &&
|
|
atCarriageReturn
|
|
) {
|
|
chunks.push(codes.carriageReturnLineFeed)
|
|
atCarriageReturn = undefined
|
|
} else {
|
|
if (atCarriageReturn) {
|
|
chunks.push(codes.carriageReturn)
|
|
atCarriageReturn = undefined
|
|
}
|
|
|
|
if (startPosition < endPosition) {
|
|
chunks.push(value.slice(startPosition, endPosition))
|
|
column += endPosition - startPosition
|
|
}
|
|
|
|
switch (code) {
|
|
case codes.nul: {
|
|
chunks.push(codes.replacementCharacter)
|
|
column++
|
|
|
|
break
|
|
}
|
|
|
|
case codes.ht: {
|
|
next = Math.ceil(column / constants.tabSize) * constants.tabSize
|
|
chunks.push(codes.horizontalTab)
|
|
while (column++ < next) chunks.push(codes.virtualSpace)
|
|
|
|
break
|
|
}
|
|
|
|
case codes.lf: {
|
|
chunks.push(codes.lineFeed)
|
|
column = 1
|
|
|
|
break
|
|
}
|
|
|
|
default: {
|
|
atCarriageReturn = true
|
|
column = 1
|
|
}
|
|
}
|
|
}
|
|
|
|
startPosition = endPosition + 1
|
|
}
|
|
|
|
if (end) {
|
|
if (atCarriageReturn) chunks.push(codes.carriageReturn)
|
|
if (buffer) chunks.push(buffer)
|
|
chunks.push(codes.eof)
|
|
}
|
|
|
|
return chunks
|
|
}
|
|
}
|