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>
39 lines
2.0 KiB
JavaScript
39 lines
2.0 KiB
JavaScript
(function (global, factory) {
|
|
typeof exports === 'object' && typeof module !== 'undefined' ? factory(exports, require('react'), require('use-sync-external-store/shim/with-selector'), require('zustand/vanilla')) :
|
|
typeof define === 'function' && define.amd ? define(['exports', 'react', 'use-sync-external-store/shim/with-selector', 'zustand/vanilla'], factory) :
|
|
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, factory(global.zustandTraditional = {}, global.React, global.useSyncExternalStoreShimWithSelector, global.zustandVanilla));
|
|
})(this, (function (exports, ReactExports, useSyncExternalStoreExports, vanilla) { 'use strict';
|
|
|
|
var useDebugValue = ReactExports.useDebugValue;
|
|
var useSyncExternalStoreWithSelector = useSyncExternalStoreExports.useSyncExternalStoreWithSelector;
|
|
var identity = function identity(arg) {
|
|
return arg;
|
|
};
|
|
function useStoreWithEqualityFn(api, selector, equalityFn) {
|
|
if (selector === void 0) {
|
|
selector = identity;
|
|
}
|
|
var slice = useSyncExternalStoreWithSelector(api.subscribe, api.getState, api.getServerState || api.getInitialState, selector, equalityFn);
|
|
useDebugValue(slice);
|
|
return slice;
|
|
}
|
|
var createWithEqualityFnImpl = function createWithEqualityFnImpl(createState, defaultEqualityFn) {
|
|
var api = vanilla.createStore(createState);
|
|
var useBoundStoreWithEqualityFn = function useBoundStoreWithEqualityFn(selector, equalityFn) {
|
|
if (equalityFn === void 0) {
|
|
equalityFn = defaultEqualityFn;
|
|
}
|
|
return useStoreWithEqualityFn(api, selector, equalityFn);
|
|
};
|
|
Object.assign(useBoundStoreWithEqualityFn, api);
|
|
return useBoundStoreWithEqualityFn;
|
|
};
|
|
var createWithEqualityFn = function createWithEqualityFn(createState, defaultEqualityFn) {
|
|
return createState ? createWithEqualityFnImpl(createState, defaultEqualityFn) : createWithEqualityFnImpl;
|
|
};
|
|
|
|
exports.createWithEqualityFn = createWithEqualityFn;
|
|
exports.useStoreWithEqualityFn = useStoreWithEqualityFn;
|
|
|
|
}));
|