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>
159 lines
4.6 KiB
JavaScript
159 lines
4.6 KiB
JavaScript
'use strict'
|
|
|
|
const { InvalidArgumentError, MaxOriginsReachedError } = require('../core/errors')
|
|
const { kClients, kRunning, kClose, kDestroy, kDispatch, kUrl } = require('../core/symbols')
|
|
const DispatcherBase = require('./dispatcher-base')
|
|
const Pool = require('./pool')
|
|
const Client = require('./client')
|
|
const util = require('../core/util')
|
|
|
|
const kOnConnect = Symbol('onConnect')
|
|
const kOnDisconnect = Symbol('onDisconnect')
|
|
const kOnConnectionError = Symbol('onConnectionError')
|
|
const kOnDrain = Symbol('onDrain')
|
|
const kFactory = Symbol('factory')
|
|
const kOptions = Symbol('options')
|
|
const kOrigins = Symbol('origins')
|
|
|
|
function defaultFactory (origin, opts) {
|
|
return opts && opts.connections === 1
|
|
? new Client(origin, opts)
|
|
: new Pool(origin, opts)
|
|
}
|
|
|
|
class Agent extends DispatcherBase {
|
|
constructor ({ factory = defaultFactory, maxOrigins = Infinity, connect, ...options } = {}) {
|
|
if (typeof factory !== 'function') {
|
|
throw new InvalidArgumentError('factory must be a function.')
|
|
}
|
|
|
|
if (connect != null && typeof connect !== 'function' && typeof connect !== 'object') {
|
|
throw new InvalidArgumentError('connect must be a function or an object')
|
|
}
|
|
|
|
if (typeof maxOrigins !== 'number' || Number.isNaN(maxOrigins) || maxOrigins <= 0) {
|
|
throw new InvalidArgumentError('maxOrigins must be a number greater than 0')
|
|
}
|
|
|
|
super()
|
|
|
|
if (connect && typeof connect !== 'function') {
|
|
connect = { ...connect }
|
|
}
|
|
|
|
this[kOptions] = { ...util.deepClone(options), maxOrigins, connect }
|
|
this[kFactory] = factory
|
|
this[kClients] = new Map()
|
|
this[kOrigins] = new Set()
|
|
|
|
this[kOnDrain] = (origin, targets) => {
|
|
this.emit('drain', origin, [this, ...targets])
|
|
}
|
|
|
|
this[kOnConnect] = (origin, targets) => {
|
|
this.emit('connect', origin, [this, ...targets])
|
|
}
|
|
|
|
this[kOnDisconnect] = (origin, targets, err) => {
|
|
this.emit('disconnect', origin, [this, ...targets], err)
|
|
}
|
|
|
|
this[kOnConnectionError] = (origin, targets, err) => {
|
|
this.emit('connectionError', origin, [this, ...targets], err)
|
|
}
|
|
}
|
|
|
|
get [kRunning] () {
|
|
let ret = 0
|
|
for (const { dispatcher } of this[kClients].values()) {
|
|
ret += dispatcher[kRunning]
|
|
}
|
|
return ret
|
|
}
|
|
|
|
[kDispatch] (opts, handler) {
|
|
let key
|
|
if (opts.origin && (typeof opts.origin === 'string' || opts.origin instanceof URL)) {
|
|
key = String(opts.origin)
|
|
} else {
|
|
throw new InvalidArgumentError('opts.origin must be a non-empty string or URL.')
|
|
}
|
|
|
|
if (this[kOrigins].size >= this[kOptions].maxOrigins && !this[kOrigins].has(key)) {
|
|
throw new MaxOriginsReachedError()
|
|
}
|
|
|
|
const result = this[kClients].get(key)
|
|
let dispatcher = result && result.dispatcher
|
|
if (!dispatcher) {
|
|
const closeClientIfUnused = (connected) => {
|
|
const result = this[kClients].get(key)
|
|
if (result) {
|
|
if (connected) result.count -= 1
|
|
if (result.count <= 0) {
|
|
this[kClients].delete(key)
|
|
if (!result.dispatcher.destroyed) {
|
|
result.dispatcher.close()
|
|
}
|
|
}
|
|
this[kOrigins].delete(key)
|
|
}
|
|
}
|
|
dispatcher = this[kFactory](opts.origin, this[kOptions])
|
|
.on('drain', this[kOnDrain])
|
|
.on('connect', (origin, targets) => {
|
|
const result = this[kClients].get(key)
|
|
if (result) {
|
|
result.count += 1
|
|
}
|
|
this[kOnConnect](origin, targets)
|
|
})
|
|
.on('disconnect', (origin, targets, err) => {
|
|
closeClientIfUnused(true)
|
|
this[kOnDisconnect](origin, targets, err)
|
|
})
|
|
.on('connectionError', (origin, targets, err) => {
|
|
closeClientIfUnused(false)
|
|
this[kOnConnectionError](origin, targets, err)
|
|
})
|
|
|
|
this[kClients].set(key, { count: 0, dispatcher })
|
|
this[kOrigins].add(key)
|
|
}
|
|
|
|
return dispatcher.dispatch(opts, handler)
|
|
}
|
|
|
|
[kClose] () {
|
|
const closePromises = []
|
|
for (const { dispatcher } of this[kClients].values()) {
|
|
closePromises.push(dispatcher.close())
|
|
}
|
|
this[kClients].clear()
|
|
|
|
return Promise.all(closePromises)
|
|
}
|
|
|
|
[kDestroy] (err) {
|
|
const destroyPromises = []
|
|
for (const { dispatcher } of this[kClients].values()) {
|
|
destroyPromises.push(dispatcher.destroy(err))
|
|
}
|
|
this[kClients].clear()
|
|
|
|
return Promise.all(destroyPromises)
|
|
}
|
|
|
|
get stats () {
|
|
const allClientStats = {}
|
|
for (const { dispatcher } of this[kClients].values()) {
|
|
if (dispatcher.stats) {
|
|
allClientStats[dispatcher[kUrl].origin] = dispatcher.stats
|
|
}
|
|
}
|
|
return allClientStats
|
|
}
|
|
}
|
|
|
|
module.exports = Agent
|