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>
69 lines
1.2 KiB
JavaScript
69 lines
1.2 KiB
JavaScript
const tape = require('tape')
|
|
const tee = require('./')
|
|
const { Readable } = require('streamx')
|
|
|
|
tape('throttled by eachother', function (t) {
|
|
const r = new Readable()
|
|
|
|
for (let i = 0; i < 1000; i++) {
|
|
r.push(Buffer.alloc(1000))
|
|
}
|
|
|
|
const [a, b] = tee(r)
|
|
|
|
let aTicks = 0
|
|
|
|
a.on('data', function (data) {
|
|
aTicks++
|
|
})
|
|
|
|
setTimeout(() => b.read(), 100)
|
|
|
|
setTimeout(() => {
|
|
t.same(aTicks, 18)
|
|
t.end()
|
|
}, 200)
|
|
})
|
|
|
|
tape('does not premature destroy', function (t) {
|
|
const r = new Readable()
|
|
|
|
const [a, b] = tee(r)
|
|
|
|
r.push('a')
|
|
r.push('b')
|
|
r.push('c')
|
|
r.push(null)
|
|
|
|
setTimeout(() => {
|
|
const aSeen = []
|
|
const bSeen = []
|
|
|
|
a.on('data', function (data) {
|
|
aSeen.push(data)
|
|
})
|
|
a.on('end', function () {
|
|
aSeen.push(null)
|
|
})
|
|
|
|
b.on('data', function (data) {
|
|
bSeen.push(data)
|
|
})
|
|
b.on('end', function () {
|
|
bSeen.push(null)
|
|
})
|
|
|
|
let missing = 2
|
|
a.on('close', onclose)
|
|
b.on('close', onclose)
|
|
|
|
function onclose () {
|
|
if (--missing === 0) {
|
|
t.same(aSeen, ['a', 'b', 'c', null])
|
|
t.same(bSeen, ['a', 'b', 'c', null])
|
|
t.end()
|
|
}
|
|
}
|
|
}, 200)
|
|
})
|