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>
134 lines
4.3 KiB
JavaScript
134 lines
4.3 KiB
JavaScript
import { Observable } from './Observable';
|
|
import { Subscription, EMPTY_SUBSCRIPTION } from './Subscription';
|
|
import { ObjectUnsubscribedError } from './util/ObjectUnsubscribedError';
|
|
import { arrRemove } from './util/arrRemove';
|
|
import { errorContext } from './util/errorContext';
|
|
export class Subject extends Observable {
|
|
constructor() {
|
|
super();
|
|
this.closed = false;
|
|
this.currentObservers = null;
|
|
this.observers = [];
|
|
this.isStopped = false;
|
|
this.hasError = false;
|
|
this.thrownError = null;
|
|
}
|
|
lift(operator) {
|
|
const subject = new AnonymousSubject(this, this);
|
|
subject.operator = operator;
|
|
return subject;
|
|
}
|
|
_throwIfClosed() {
|
|
if (this.closed) {
|
|
throw new ObjectUnsubscribedError();
|
|
}
|
|
}
|
|
next(value) {
|
|
errorContext(() => {
|
|
this._throwIfClosed();
|
|
if (!this.isStopped) {
|
|
if (!this.currentObservers) {
|
|
this.currentObservers = Array.from(this.observers);
|
|
}
|
|
for (const observer of this.currentObservers) {
|
|
observer.next(value);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
error(err) {
|
|
errorContext(() => {
|
|
this._throwIfClosed();
|
|
if (!this.isStopped) {
|
|
this.hasError = this.isStopped = true;
|
|
this.thrownError = err;
|
|
const { observers } = this;
|
|
while (observers.length) {
|
|
observers.shift().error(err);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
complete() {
|
|
errorContext(() => {
|
|
this._throwIfClosed();
|
|
if (!this.isStopped) {
|
|
this.isStopped = true;
|
|
const { observers } = this;
|
|
while (observers.length) {
|
|
observers.shift().complete();
|
|
}
|
|
}
|
|
});
|
|
}
|
|
unsubscribe() {
|
|
this.isStopped = this.closed = true;
|
|
this.observers = this.currentObservers = null;
|
|
}
|
|
get observed() {
|
|
var _a;
|
|
return ((_a = this.observers) === null || _a === void 0 ? void 0 : _a.length) > 0;
|
|
}
|
|
_trySubscribe(subscriber) {
|
|
this._throwIfClosed();
|
|
return super._trySubscribe(subscriber);
|
|
}
|
|
_subscribe(subscriber) {
|
|
this._throwIfClosed();
|
|
this._checkFinalizedStatuses(subscriber);
|
|
return this._innerSubscribe(subscriber);
|
|
}
|
|
_innerSubscribe(subscriber) {
|
|
const { hasError, isStopped, observers } = this;
|
|
if (hasError || isStopped) {
|
|
return EMPTY_SUBSCRIPTION;
|
|
}
|
|
this.currentObservers = null;
|
|
observers.push(subscriber);
|
|
return new Subscription(() => {
|
|
this.currentObservers = null;
|
|
arrRemove(observers, subscriber);
|
|
});
|
|
}
|
|
_checkFinalizedStatuses(subscriber) {
|
|
const { hasError, thrownError, isStopped } = this;
|
|
if (hasError) {
|
|
subscriber.error(thrownError);
|
|
}
|
|
else if (isStopped) {
|
|
subscriber.complete();
|
|
}
|
|
}
|
|
asObservable() {
|
|
const observable = new Observable();
|
|
observable.source = this;
|
|
return observable;
|
|
}
|
|
}
|
|
Subject.create = (destination, source) => {
|
|
return new AnonymousSubject(destination, source);
|
|
};
|
|
export class AnonymousSubject extends Subject {
|
|
constructor(destination, source) {
|
|
super();
|
|
this.destination = destination;
|
|
this.source = source;
|
|
}
|
|
next(value) {
|
|
var _a, _b;
|
|
(_b = (_a = this.destination) === null || _a === void 0 ? void 0 : _a.next) === null || _b === void 0 ? void 0 : _b.call(_a, value);
|
|
}
|
|
error(err) {
|
|
var _a, _b;
|
|
(_b = (_a = this.destination) === null || _a === void 0 ? void 0 : _a.error) === null || _b === void 0 ? void 0 : _b.call(_a, err);
|
|
}
|
|
complete() {
|
|
var _a, _b;
|
|
(_b = (_a = this.destination) === null || _a === void 0 ? void 0 : _a.complete) === null || _b === void 0 ? void 0 : _b.call(_a);
|
|
}
|
|
_subscribe(subscriber) {
|
|
var _a, _b;
|
|
return (_b = (_a = this.source) === null || _a === void 0 ? void 0 : _a.subscribe(subscriber)) !== null && _b !== void 0 ? _b : EMPTY_SUBSCRIPTION;
|
|
}
|
|
}
|
|
//# sourceMappingURL=Subject.js.map
|