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>
82 lines
1.8 KiB
JavaScript
82 lines
1.8 KiB
JavaScript
var test = require('tape')
|
|
var randomBytes = require('./')
|
|
var MAX_BYTES = 65536
|
|
var MAX_UINT32 = 4294967295
|
|
|
|
test('sync', function (t) {
|
|
t.plan(9)
|
|
t.equals(randomBytes(0).length, 0, 'len: ' + 0)
|
|
t.equals(randomBytes(3).length, 3, 'len: ' + 3)
|
|
t.equals(randomBytes(30).length, 30, 'len: ' + 30)
|
|
t.equals(randomBytes(300).length, 300, 'len: ' + 300)
|
|
t.equals(randomBytes(17 + MAX_BYTES).length, 17 + MAX_BYTES, 'len: ' + 17 + MAX_BYTES)
|
|
t.equals(randomBytes(MAX_BYTES * 100).length, MAX_BYTES * 100, 'len: ' + MAX_BYTES * 100)
|
|
t.throws(function () {
|
|
randomBytes(MAX_UINT32 + 1)
|
|
})
|
|
t.throws(function () {
|
|
t.equals(randomBytes(-1))
|
|
})
|
|
t.throws(function () {
|
|
t.equals(randomBytes('hello'))
|
|
})
|
|
})
|
|
|
|
test('async', function (t) {
|
|
t.plan(9)
|
|
|
|
randomBytes(0, function (err, resp) {
|
|
if (err) throw err
|
|
|
|
t.equals(resp.length, 0, 'len: ' + 0)
|
|
})
|
|
|
|
randomBytes(3, function (err, resp) {
|
|
if (err) throw err
|
|
|
|
t.equals(resp.length, 3, 'len: ' + 3)
|
|
})
|
|
|
|
randomBytes(30, function (err, resp) {
|
|
if (err) throw err
|
|
|
|
t.equals(resp.length, 30, 'len: ' + 30)
|
|
})
|
|
|
|
randomBytes(300, function (err, resp) {
|
|
if (err) throw err
|
|
|
|
t.equals(resp.length, 300, 'len: ' + 300)
|
|
})
|
|
|
|
randomBytes(17 + MAX_BYTES, function (err, resp) {
|
|
if (err) throw err
|
|
|
|
t.equals(resp.length, 17 + MAX_BYTES, 'len: ' + 17 + MAX_BYTES)
|
|
})
|
|
|
|
randomBytes(MAX_BYTES * 100, function (err, resp) {
|
|
if (err) throw err
|
|
|
|
t.equals(resp.length, MAX_BYTES * 100, 'len: ' + MAX_BYTES * 100)
|
|
})
|
|
|
|
t.throws(function () {
|
|
randomBytes(MAX_UINT32 + 1, function () {
|
|
t.ok(false, 'should not get here')
|
|
})
|
|
})
|
|
|
|
t.throws(function () {
|
|
randomBytes(-1, function () {
|
|
t.ok(false, 'should not get here')
|
|
})
|
|
})
|
|
|
|
t.throws(function () {
|
|
randomBytes('hello', function () {
|
|
t.ok(false, 'should not get here')
|
|
})
|
|
})
|
|
})
|