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>
61 lines
2.2 KiB
JavaScript
61 lines
2.2 KiB
JavaScript
import {STANDARD_STREAMS_ALIASES} from '../utils/standard-stream.js';
|
|
import {normalizeIpcStdioArray} from '../ipc/array.js';
|
|
import {isFullVerbose} from '../verbose/values.js';
|
|
|
|
// Add support for `stdin`/`stdout`/`stderr` as an alias for `stdio`.
|
|
// Also normalize the `stdio` option.
|
|
export const normalizeStdioOption = ({stdio, ipc, buffer, ...options}, verboseInfo, isSync) => {
|
|
const stdioArray = getStdioArray(stdio, options).map((stdioOption, fdNumber) => addDefaultValue(stdioOption, fdNumber));
|
|
return isSync
|
|
? normalizeStdioSync(stdioArray, buffer, verboseInfo)
|
|
: normalizeIpcStdioArray(stdioArray, ipc);
|
|
};
|
|
|
|
const getStdioArray = (stdio, options) => {
|
|
if (stdio === undefined) {
|
|
return STANDARD_STREAMS_ALIASES.map(alias => options[alias]);
|
|
}
|
|
|
|
if (hasAlias(options)) {
|
|
throw new Error(`It's not possible to provide \`stdio\` in combination with one of ${STANDARD_STREAMS_ALIASES.map(alias => `\`${alias}\``).join(', ')}`);
|
|
}
|
|
|
|
if (typeof stdio === 'string') {
|
|
return [stdio, stdio, stdio];
|
|
}
|
|
|
|
if (!Array.isArray(stdio)) {
|
|
throw new TypeError(`Expected \`stdio\` to be of type \`string\` or \`Array\`, got \`${typeof stdio}\``);
|
|
}
|
|
|
|
const length = Math.max(stdio.length, STANDARD_STREAMS_ALIASES.length);
|
|
return Array.from({length}, (_, fdNumber) => stdio[fdNumber]);
|
|
};
|
|
|
|
const hasAlias = options => STANDARD_STREAMS_ALIASES.some(alias => options[alias] !== undefined);
|
|
|
|
const addDefaultValue = (stdioOption, fdNumber) => {
|
|
if (Array.isArray(stdioOption)) {
|
|
return stdioOption.map(item => addDefaultValue(item, fdNumber));
|
|
}
|
|
|
|
if (stdioOption === null || stdioOption === undefined) {
|
|
return fdNumber >= STANDARD_STREAMS_ALIASES.length ? 'ignore' : 'pipe';
|
|
}
|
|
|
|
return stdioOption;
|
|
};
|
|
|
|
// Using `buffer: false` with synchronous methods implies `stdout`/`stderr`: `ignore`.
|
|
// Unless the output is needed, e.g. due to `verbose: 'full'` or to redirecting to a file.
|
|
const normalizeStdioSync = (stdioArray, buffer, verboseInfo) => stdioArray.map((stdioOption, fdNumber) =>
|
|
!buffer[fdNumber]
|
|
&& fdNumber !== 0
|
|
&& !isFullVerbose(verboseInfo, fdNumber)
|
|
&& isOutputPipeOnly(stdioOption)
|
|
? 'ignore'
|
|
: stdioOption);
|
|
|
|
const isOutputPipeOnly = stdioOption => stdioOption === 'pipe'
|
|
|| (Array.isArray(stdioOption) && stdioOption.every(item => item === 'pipe'));
|