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>
68 lines
1.7 KiB
JavaScript
68 lines
1.7 KiB
JavaScript
'use strict'
|
|
|
|
let Document = require('./document')
|
|
let LazyResult = require('./lazy-result')
|
|
let NoWorkResult = require('./no-work-result')
|
|
let Root = require('./root')
|
|
|
|
class Processor {
|
|
constructor(plugins = []) {
|
|
this.version = '8.5.8'
|
|
this.plugins = this.normalize(plugins)
|
|
}
|
|
|
|
normalize(plugins) {
|
|
let normalized = []
|
|
for (let i of plugins) {
|
|
if (i.postcss === true) {
|
|
i = i()
|
|
} else if (i.postcss) {
|
|
i = i.postcss
|
|
}
|
|
|
|
if (typeof i === 'object' && Array.isArray(i.plugins)) {
|
|
normalized = normalized.concat(i.plugins)
|
|
} else if (typeof i === 'object' && i.postcssPlugin) {
|
|
normalized.push(i)
|
|
} else if (typeof i === 'function') {
|
|
normalized.push(i)
|
|
} else if (typeof i === 'object' && (i.parse || i.stringify)) {
|
|
if (process.env.NODE_ENV !== 'production') {
|
|
throw new Error(
|
|
'PostCSS syntaxes cannot be used as plugins. Instead, please use ' +
|
|
'one of the syntax/parser/stringifier options as outlined ' +
|
|
'in your PostCSS runner documentation.'
|
|
)
|
|
}
|
|
} else {
|
|
throw new Error(i + ' is not a PostCSS plugin')
|
|
}
|
|
}
|
|
return normalized
|
|
}
|
|
|
|
process(css, opts = {}) {
|
|
if (
|
|
!this.plugins.length &&
|
|
!opts.parser &&
|
|
!opts.stringifier &&
|
|
!opts.syntax
|
|
) {
|
|
return new NoWorkResult(this, css, opts)
|
|
} else {
|
|
return new LazyResult(this, css, opts)
|
|
}
|
|
}
|
|
|
|
use(plugin) {
|
|
this.plugins = this.plugins.concat(this.normalize([plugin]))
|
|
return this
|
|
}
|
|
}
|
|
|
|
module.exports = Processor
|
|
Processor.default = Processor
|
|
|
|
Root.registerProcessor(Processor)
|
|
Document.registerProcessor(Processor)
|