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>
71 lines
2.1 KiB
JavaScript
71 lines
2.1 KiB
JavaScript
var fs = require('fs');
|
|
var path = require('path');
|
|
var test = require('tape');
|
|
var resolve = require('../');
|
|
|
|
test('$NODE_PATH', function (t) {
|
|
t.plan(8);
|
|
|
|
var isDir = function (dir, cb) {
|
|
if (dir === '/node_path' || dir === 'node_path/x') {
|
|
return cb(null, true);
|
|
}
|
|
fs.stat(dir, function (err, stat) {
|
|
if (!err) {
|
|
return cb(null, stat.isDirectory());
|
|
}
|
|
if (err.code === 'ENOENT' || err.code === 'ENOTDIR') return cb(null, false);
|
|
return cb(err);
|
|
});
|
|
};
|
|
|
|
resolve('aaa', {
|
|
paths: [
|
|
path.join(__dirname, '/node_path/x'),
|
|
path.join(__dirname, '/node_path/y')
|
|
],
|
|
basedir: __dirname,
|
|
isDirectory: isDir
|
|
}, function (err, res) {
|
|
t.error(err);
|
|
t.equal(res, path.join(__dirname, '/node_path/x/aaa/index.js'), 'aaa resolves');
|
|
});
|
|
|
|
resolve('bbb', {
|
|
paths: [
|
|
path.join(__dirname, '/node_path/x'),
|
|
path.join(__dirname, '/node_path/y')
|
|
],
|
|
basedir: __dirname,
|
|
isDirectory: isDir
|
|
}, function (err, res) {
|
|
t.error(err);
|
|
t.equal(res, path.join(__dirname, '/node_path/y/bbb/index.js'), 'bbb resolves');
|
|
});
|
|
|
|
resolve('ccc', {
|
|
paths: [
|
|
path.join(__dirname, '/node_path/x'),
|
|
path.join(__dirname, '/node_path/y')
|
|
],
|
|
basedir: __dirname,
|
|
isDirectory: isDir
|
|
}, function (err, res) {
|
|
t.error(err);
|
|
t.equal(res, path.join(__dirname, '/node_path/x/ccc/index.js'), 'ccc resolves');
|
|
});
|
|
|
|
// ensure that relative paths still resolve against the regular `node_modules` correctly
|
|
resolve('tap', {
|
|
paths: [
|
|
'node_path'
|
|
],
|
|
basedir: path.join(__dirname, 'node_path/x'),
|
|
isDirectory: isDir
|
|
}, function (err, res) {
|
|
var root = require('tap/package.json').main; // eslint-disable-line global-require
|
|
t.error(err);
|
|
t.equal(res, path.resolve(__dirname, '..', 'node_modules/tap', root), 'tap resolves');
|
|
});
|
|
});
|