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>
40 lines
1.1 KiB
JavaScript
40 lines
1.1 KiB
JavaScript
/**
|
|
* @import {Join} from 'mdast-util-to-markdown'
|
|
*/
|
|
|
|
import {formatCodeAsIndented} from './util/format-code-as-indented.js'
|
|
import {formatHeadingAsSetext} from './util/format-heading-as-setext.js'
|
|
|
|
/** @type {Array<Join>} */
|
|
export const join = [joinDefaults]
|
|
|
|
/** @type {Join} */
|
|
function joinDefaults(left, right, parent, state) {
|
|
// Indented code after list or another indented code.
|
|
if (
|
|
right.type === 'code' &&
|
|
formatCodeAsIndented(right, state) &&
|
|
(left.type === 'list' ||
|
|
(left.type === right.type && formatCodeAsIndented(left, state)))
|
|
) {
|
|
return false
|
|
}
|
|
|
|
// Join children of a list or an item.
|
|
// In which case, `parent` has a `spread` field.
|
|
if ('spread' in parent && typeof parent.spread === 'boolean') {
|
|
if (
|
|
left.type === 'paragraph' &&
|
|
// Two paragraphs.
|
|
(left.type === right.type ||
|
|
right.type === 'definition' ||
|
|
// Paragraph followed by a setext heading.
|
|
(right.type === 'heading' && formatHeadingAsSetext(right, state)))
|
|
) {
|
|
return
|
|
}
|
|
|
|
return parent.spread ? 1 : 0
|
|
}
|
|
}
|