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>
73 lines
2.2 KiB
JavaScript
73 lines
2.2 KiB
JavaScript
// Following http://www.w3.org/TR/css3-selectors/#nth-child-pseudo
|
|
// Whitespace as per https://www.w3.org/TR/selectors-3/#lex is " \t\r\n\f"
|
|
const whitespace = new Set([9, 10, 12, 13, 32]);
|
|
const ZERO = "0".charCodeAt(0);
|
|
const NINE = "9".charCodeAt(0);
|
|
/**
|
|
* Parses an expression.
|
|
*
|
|
* @throws An `Error` if parsing fails.
|
|
* @returns An array containing the integer step size and the integer offset of the nth rule.
|
|
* @example nthCheck.parse("2n+3"); // returns [2, 3]
|
|
*/
|
|
export function parse(formula) {
|
|
formula = formula.trim().toLowerCase();
|
|
if (formula === "even") {
|
|
return [2, 0];
|
|
}
|
|
else if (formula === "odd") {
|
|
return [2, 1];
|
|
}
|
|
// Parse [ ['-'|'+']? INTEGER? {N} [ S* ['-'|'+'] S* INTEGER ]?
|
|
let idx = 0;
|
|
let a = 0;
|
|
let sign = readSign();
|
|
let number = readNumber();
|
|
if (idx < formula.length && formula.charAt(idx) === "n") {
|
|
idx++;
|
|
a = sign * (number !== null && number !== void 0 ? number : 1);
|
|
skipWhitespace();
|
|
if (idx < formula.length) {
|
|
sign = readSign();
|
|
skipWhitespace();
|
|
number = readNumber();
|
|
}
|
|
else {
|
|
sign = number = 0;
|
|
}
|
|
}
|
|
// Throw if there is anything else
|
|
if (number === null || idx < formula.length) {
|
|
throw new Error(`n-th rule couldn't be parsed ('${formula}')`);
|
|
}
|
|
return [a, sign * number];
|
|
function readSign() {
|
|
if (formula.charAt(idx) === "-") {
|
|
idx++;
|
|
return -1;
|
|
}
|
|
if (formula.charAt(idx) === "+") {
|
|
idx++;
|
|
}
|
|
return 1;
|
|
}
|
|
function readNumber() {
|
|
const start = idx;
|
|
let value = 0;
|
|
while (idx < formula.length &&
|
|
formula.charCodeAt(idx) >= ZERO &&
|
|
formula.charCodeAt(idx) <= NINE) {
|
|
value = value * 10 + (formula.charCodeAt(idx) - ZERO);
|
|
idx++;
|
|
}
|
|
// Return `null` if we didn't read anything.
|
|
return idx === start ? null : value;
|
|
}
|
|
function skipWhitespace() {
|
|
while (idx < formula.length &&
|
|
whitespace.has(formula.charCodeAt(idx))) {
|
|
idx++;
|
|
}
|
|
}
|
|
}
|
|
//# sourceMappingURL=parse.js.map
|