UNPKG

llmverify

Version:

AI Output Verification Toolkit — Local-first LLM safety, hallucination detection, PII redaction, prompt injection defense, and runtime monitoring. Zero telemetry. OWASP LLM Top 10 aligned.

290 lines 30.5 kB
"use strict"; /** * Audit Trail System * * Compliance-ready audit logging for verification operations * * @module logging/audit */ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || (function () { var ownKeys = function(o) { ownKeys = Object.getOwnPropertyNames || function (o) { var ar = []; for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k; return ar; }; return ownKeys(o); }; return function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]); __setModuleDefault(result, mod); return result; }; })(); Object.defineProperty(exports, "__esModule", { value: true }); exports.AuditLogger = void 0; exports.getAuditLogger = getAuditLogger; exports.setAuditLogger = setAuditLogger; const fs = __importStar(require("fs")); const path = __importStar(require("path")); const os = __importStar(require("os")); const crypto = __importStar(require("crypto")); /** * Default audit configuration */ const DEFAULT_AUDIT_CONFIG = { enabled: true, auditDir: path.join(os.homedir(), '.llmverify', 'audit'), includeContentHash: true, includeUserInfo: false, maxFileSize: 10 * 1024 * 1024, // 10MB maxFiles: 50 // Keep more audit files }; /** * Audit logger class */ class AuditLogger { constructor(config) { this.config = { ...DEFAULT_AUDIT_CONFIG, ...config }; this.ensureAuditDirectory(); } /** * Ensure audit directory exists */ ensureAuditDirectory() { if (this.config.enabled && this.config.auditDir) { try { fs.mkdirSync(this.config.auditDir, { recursive: true }); } catch (error) { console.error('Failed to create audit directory:', error); this.config.enabled = false; } } } /** * Get audit file path */ getAuditFilePath() { const date = new Date().toISOString().split('T')[0]; return path.join(this.config.auditDir, `audit-${date}.jsonl`); } /** * Generate content hash */ hashContent(content) { return crypto.createHash('sha256').update(content).digest('hex').substring(0, 16); } /** * Write audit entry */ log(entry) { if (!this.config.enabled) return; try { const fullEntry = { timestamp: new Date().toISOString(), ...entry }; const auditFile = this.getAuditFilePath(); const line = JSON.stringify(fullEntry) + '\n'; fs.appendFileSync(auditFile, line, 'utf-8'); // Rotate if needed this.rotateIfNeeded(auditFile); } catch (error) { if (process.env.NODE_ENV === 'development') { console.error('Failed to write audit entry:', error); } } } /** * Log verification operation */ logVerification(params) { this.log({ requestId: params.requestId, operation: 'verify', input: { contentLength: params.content.length, contentHash: this.config.includeContentHash ? this.hashContent(params.content) : '', hasPrompt: !!params.prompt }, output: { riskLevel: params.riskLevel, findingsCount: params.findingsCount, blocked: params.blocked }, metadata: { version: require('../../package.json').version, duration: params.duration, enginesUsed: params.enginesUsed, configTier: params.configTier }, user: this.config.includeUserInfo ? { id: params.userId, ip: params.userIp } : undefined }); } /** * Rotate audit files */ rotateIfNeeded(auditFile) { try { const stats = fs.statSync(auditFile); if (stats.size > this.config.maxFileSize) { const timestamp = Date.now(); const rotatedFile = auditFile.replace('.jsonl', `.${timestamp}.jsonl`); fs.renameSync(auditFile, rotatedFile); this.cleanupOldAudits(); } } catch (error) { // Ignore rotation errors } } /** * Clean up old audit files */ cleanupOldAudits() { if (!this.config.auditDir) return; try { const files = fs.readdirSync(this.config.auditDir) .filter(f => f.startsWith('audit-') && f.endsWith('.jsonl')) .map(f => ({ name: f, path: path.join(this.config.auditDir, f), time: fs.statSync(path.join(this.config.auditDir, f)).mtime.getTime() })) .sort((a, b) => b.time - a.time); if (files.length > this.config.maxFiles) { files.slice(this.config.maxFiles).forEach(file => { try { fs.unlinkSync(file.path); } catch (error) { // Ignore deletion errors } }); } } catch (error) { // Ignore cleanup errors } } /** * Read audit entries */ readAudit(date) { if (!this.config.auditDir) return []; const dateStr = date || new Date().toISOString().split('T')[0]; const auditFile = path.join(this.config.auditDir, `audit-${dateStr}.jsonl`); if (!fs.existsSync(auditFile)) return []; try { const content = fs.readFileSync(auditFile, 'utf-8'); return content .split('\n') .filter(line => line.trim()) .map(line => JSON.parse(line)); } catch (error) { console.error('Failed to read audit:', error); return []; } } /** * Get audit statistics */ getStats(date) { const entries = this.readAudit(date); const stats = { totalOperations: entries.length, byOperation: {}, blockedCount: 0, avgDuration: 0, riskDistribution: {} }; let totalDuration = 0; entries.forEach(entry => { // Count by operation stats.byOperation[entry.operation] = (stats.byOperation[entry.operation] || 0) + 1; // Count blocked if (entry.output.blocked) stats.blockedCount++; // Sum duration totalDuration += entry.metadata.duration; // Risk distribution const risk = entry.output.riskLevel; stats.riskDistribution[risk] = (stats.riskDistribution[risk] || 0) + 1; }); if (entries.length > 0) { stats.avgDuration = totalDuration / entries.length; } return stats; } /** * Export audit trail for compliance */ exportAuditTrail(startDate, endDate, outputPath) { const start = new Date(startDate); const end = new Date(endDate); const allEntries = []; // Collect all entries in date range for (let d = new Date(start); d <= end; d.setDate(d.getDate() + 1)) { const dateStr = d.toISOString().split('T')[0]; const entries = this.readAudit(dateStr); allEntries.push(...entries); } // Write to output file const report = { exportDate: new Date().toISOString(), dateRange: { start: startDate, end: endDate }, totalEntries: allEntries.length, entries: allEntries }; fs.writeFileSync(outputPath, JSON.stringify(report, null, 2), 'utf-8'); } } exports.AuditLogger = AuditLogger; /** * Global audit logger */ let globalAuditLogger = null; /** * Get global audit logger */ function getAuditLogger(config) { if (!globalAuditLogger) { globalAuditLogger = new AuditLogger(config); } return globalAuditLogger; } /** * Set global audit logger */ function setAuditLogger(logger) { globalAuditLogger = logger; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audit.js","sourceRoot":"","sources":["../../src/logging/audit.ts"],"names":[],"mappings":";AAAA;;;;;;GAMG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA+TH,wCAKC;AAKD,wCAEC;AAzUD,uCAAyB;AACzB,2CAA6B;AAC7B,uCAAyB;AACzB,+CAAiC;AA2CjC;;GAEG;AACH,MAAM,oBAAoB,GAAgB;IACxC,OAAO,EAAE,IAAI;IACb,QAAQ,EAAE,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,YAAY,EAAE,OAAO,CAAC;IACxD,kBAAkB,EAAE,IAAI;IACxB,eAAe,EAAE,KAAK;IACtB,WAAW,EAAE,EAAE,GAAG,IAAI,GAAG,IAAI,EAAE,OAAO;IACtC,QAAQ,EAAE,EAAE,CAAC,wBAAwB;CACtC,CAAC;AAEF;;GAEG;AACH,MAAa,WAAW;IAGtB,YAAY,MAA6B;QACvC,IAAI,CAAC,MAAM,GAAG,EAAE,GAAG,oBAAoB,EAAE,GAAG,MAAM,EAAE,CAAC;QACrD,IAAI,CAAC,oBAAoB,EAAE,CAAC;IAC9B,CAAC;IAED;;OAEG;IACK,oBAAoB;QAC1B,IAAI,IAAI,CAAC,MAAM,CAAC,OAAO,IAAI,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC;YAChD,IAAI,CAAC;gBACH,EAAE,CAAC,SAAS,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,CAAC;YAC1D,CAAC;YAAC,OAAO,KAAK,EAAE,CAAC;gBACf,OAAO,CAAC,KAAK,CAAC,mCAAmC,EAAE,KAAK,CAAC,CAAC;gBAC1D,IAAI,CAAC,MAAM,CAAC,OAAO,GAAG,KAAK,CAAC;YAC9B,CAAC;QACH,CAAC;IACH,CAAC;IAED;;OAEG;IACK,gBAAgB;QACtB,MAAM,IAAI,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;QACpD,OAAO,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,QAAS,EAAE,SAAS,IAAI,QAAQ,CAAC,CAAC;IACjE,CAAC;IAED;;OAEG;IACK,WAAW,CAAC,OAAe;QACjC,OAAO,MAAM,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,SAAS,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IACpF,CAAC;IAED;;OAEG;IACI,GAAG,CAAC,KAAoC;QAC7C,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO;YAAE,OAAO;QAEjC,IAAI,CAAC;YACH,MAAM,SAAS,GAAe;gBAC5B,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;gBACnC,GAAG,KAAK;aACT,CAAC;YAEF,MAAM,SAAS,GAAG,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC1C,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,GAAG,IAAI,CAAC;YAE9C,EAAE,CAAC,cAAc,CAAC,SAAS,EAAE,IAAI,EAAE,OAAO,CAAC,CAAC;YAE5C,mBAAmB;YACnB,IAAI,CAAC,cAAc,CAAC,SAAS,CAAC,CAAC;QACjC,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,IAAI,OAAO,CAAC,GAAG,CAAC,QAAQ,KAAK,aAAa,EAAE,CAAC;gBAC3C,OAAO,CAAC,KAAK,CAAC,8BAA8B,EAAE,KAAK,CAAC,CAAC;YACvD,CAAC;QACH,CAAC;IACH,CAAC;IAED;;OAEG;IACI,eAAe,CAAC,MAYtB;QACC,IAAI,CAAC,GAAG,CAAC;YACP,SAAS,EAAE,MAAM,CAAC,SAAS;YAC3B,SAAS,EAAE,QAAQ;YACnB,KAAK,EAAE;gBACL,aAAa,EAAE,MAAM,CAAC,OAAO,CAAC,MAAM;gBACpC,WAAW,EAAE,IAAI,CAAC,MAAM,CAAC,kBAAkB,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE;gBACnF,SAAS,EAAE,CAAC,CAAC,MAAM,CAAC,MAAM;aAC3B;YACD,MAAM,EAAE;gBACN,SAAS,EAAE,MAAM,CAAC,SAAS;gBAC3B,aAAa,EAAE,MAAM,CAAC,aAAa;gBACnC,OAAO,EAAE,MAAM,CAAC,OAAO;aACxB;YACD,QAAQ,EAAE;gBACR,OAAO,EAAE,OAAO,CAAC,oBAAoB,CAAC,CAAC,OAAO;gBAC9C,QAAQ,EAAE,MAAM,CAAC,QAAQ;gBACzB,WAAW,EAAE,MAAM,CAAC,WAAW;gBAC/B,UAAU,EAAE,MAAM,CAAC,UAAU;aAC9B;YACD,IAAI,EAAE,IAAI,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC,CAAC;gBAClC,EAAE,EAAE,MAAM,CAAC,MAAM;gBACjB,EAAE,EAAE,MAAM,CAAC,MAAM;aAClB,CAAC,CAAC,CAAC,SAAS;SACd,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACK,cAAc,CAAC,SAAiB;QACtC,IAAI,CAAC;YACH,MAAM,KAAK,GAAG,EAAE,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC;YAErC,IAAI,KAAK,CAAC,IAAI,GAAG,IAAI,CAAC,MAAM,CAAC,WAAY,EAAE,CAAC;gBAC1C,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;gBAC7B,MAAM,WAAW,GAAG,SAAS,CAAC,OAAO,CAAC,QAAQ,EAAE,IAAI,SAAS,QAAQ,CAAC,CAAC;gBACvE,EAAE,CAAC,UAAU,CAAC,SAAS,EAAE,WAAW,CAAC,CAAC;gBAEtC,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC1B,CAAC;QACH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,yBAAyB;QAC3B,CAAC;IACH,CAAC;IAED;;OAEG;IACK,gBAAgB;QACtB,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ;YAAE,OAAO;QAElC,IAAI,CAAC;YACH,MAAM,KAAK,GAAG,EAAE,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,CAAC;iBAC/C,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,QAAQ,CAAC,QAAQ,CAAC,CAAC;iBAC3D,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;gBACT,IAAI,EAAE,CAAC;gBACP,IAAI,EAAE,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,QAAS,EAAE,CAAC,CAAC;gBACzC,IAAI,EAAE,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,QAAS,EAAE,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,EAAE;aACvE,CAAC,CAAC;iBACF,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,GAAG,CAAC,CAAC,IAAI,CAAC,CAAC;YAEnC,IAAI,KAAK,CAAC,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,QAAS,EAAE,CAAC;gBACzC,KAAK,CAAC,KAAK,CAAC,IAAI,CAAC,MAAM,CAAC,QAAS,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;oBAChD,IAAI,CAAC;wBACH,EAAE,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;oBAC3B,CAAC;oBAAC,OAAO,KAAK,EAAE,CAAC;wBACf,yBAAyB;oBAC3B,CAAC;gBACH,CAAC,CAAC,CAAC;YACL,CAAC;QACH,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,wBAAwB;QAC1B,CAAC;IACH,CAAC;IAED;;OAEG;IACI,SAAS,CAAC,IAAa;QAC5B,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ;YAAE,OAAO,EAAE,CAAC;QAErC,MAAM,OAAO,GAAG,IAAI,IAAI,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;QAC/D,MAAM,SAAS,GAAG,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,SAAS,OAAO,QAAQ,CAAC,CAAC;QAE5E,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,SAAS,CAAC;YAAE,OAAO,EAAE,CAAC;QAEzC,IAAI,CAAC;YACH,MAAM,OAAO,GAAG,EAAE,CAAC,YAAY,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;YACpD,OAAO,OAAO;iBACX,KAAK,CAAC,IAAI,CAAC;iBACX,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC;iBAC3B,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAe,CAAC,CAAC;QACjD,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,uBAAuB,EAAE,KAAK,CAAC,CAAC;YAC9C,OAAO,EAAE,CAAC;QACZ,CAAC;IACH,CAAC;IAED;;OAEG;IACI,QAAQ,CAAC,IAAa;QAO3B,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;QAErC,MAAM,KAAK,GAAG;YACZ,eAAe,EAAE,OAAO,CAAC,MAAM;YAC/B,WAAW,EAAE,EAA4B;YACzC,YAAY,EAAE,CAAC;YACf,WAAW,EAAE,CAAC;YACd,gBAAgB,EAAE,EAA4B;SAC/C,CAAC;QAEF,IAAI,aAAa,GAAG,CAAC,CAAC;QAEtB,OAAO,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE;YACtB,qBAAqB;YACrB,KAAK,CAAC,WAAW,CAAC,KAAK,CAAC,SAAS,CAAC,GAAG,CAAC,KAAK,CAAC,WAAW,CAAC,KAAK,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC;YAEnF,gBAAgB;YAChB,IAAI,KAAK,CAAC,MAAM,CAAC,OAAO;gBAAE,KAAK,CAAC,YAAY,EAAE,CAAC;YAE/C,eAAe;YACf,aAAa,IAAI,KAAK,CAAC,QAAQ,CAAC,QAAQ,CAAC;YAEzC,oBAAoB;YACpB,MAAM,IAAI,GAAG,KAAK,CAAC,MAAM,CAAC,SAAS,CAAC;YACpC,KAAK,CAAC,gBAAgB,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,gBAAgB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC;QACzE,CAAC,CAAC,CAAC;QAEH,IAAI,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACvB,KAAK,CAAC,WAAW,GAAG,aAAa,GAAG,OAAO,CAAC,MAAM,CAAC;QACrD,CAAC;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAED;;OAEG;IACI,gBAAgB,CAAC,SAAiB,EAAE,OAAe,EAAE,UAAkB;QAC5E,MAAM,KAAK,GAAG,IAAI,IAAI,CAAC,SAAS,CAAC,CAAC;QAClC,MAAM,GAAG,GAAG,IAAI,IAAI,CAAC,OAAO,CAAC,CAAC;QAC9B,MAAM,UAAU,GAAiB,EAAE,CAAC;QAEpC,oCAAoC;QACpC,KAAK,IAAI,CAAC,GAAG,IAAI,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC,IAAI,GAAG,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC,EAAE,CAAC;YACnE,MAAM,OAAO,GAAG,CAAC,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;YAC9C,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;YACxC,UAAU,CAAC,IAAI,CAAC,GAAG,OAAO,CAAC,CAAC;QAC9B,CAAC;QAED,uBAAuB;QACvB,MAAM,MAAM,GAAG;YACb,UAAU,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;YACpC,SAAS,EAAE,EAAE,KAAK,EAAE,SAAS,EAAE,GAAG,EAAE,OAAO,EAAE;YAC7C,YAAY,EAAE,UAAU,CAAC,MAAM;YAC/B,OAAO,EAAE,UAAU;SACpB,CAAC;QAEF,EAAE,CAAC,aAAa,CAAC,UAAU,EAAE,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,OAAO,CAAC,CAAC;IACzE,CAAC;CACF;AAtPD,kCAsPC;AAED;;GAEG;AACH,IAAI,iBAAiB,GAAuB,IAAI,CAAC;AAEjD;;GAEG;AACH,SAAgB,cAAc,CAAC,MAA6B;IAC1D,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACvB,iBAAiB,GAAG,IAAI,WAAW,CAAC,MAAM,CAAC,CAAC;IAC9C,CAAC;IACD,OAAO,iBAAiB,CAAC;AAC3B,CAAC;AAED;;GAEG;AACH,SAAgB,cAAc,CAAC,MAAmB;IAChD,iBAAiB,GAAG,MAAM,CAAC;AAC7B,CAAC","sourcesContent":["/**\n * Audit Trail System\n * \n * Compliance-ready audit logging for verification operations\n * \n * @module logging/audit\n */\n\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport * as os from 'os';\nimport * as crypto from 'crypto';\n\n/**\n * Audit entry structure\n */\nexport interface AuditEntry {\n  timestamp: string;\n  requestId: string;\n  operation: 'verify' | 'classify' | 'check-input' | 'check-pii' | 'plugin-execute';\n  input: {\n    contentLength: number;\n    contentHash: string;\n    hasPrompt: boolean;\n  };\n  output: {\n    riskLevel: string;\n    findingsCount: number;\n    blocked: boolean;\n  };\n  metadata: {\n    version: string;\n    duration: number;\n    enginesUsed: string[];\n    configTier: string;\n  };\n  user?: {\n    id?: string;\n    ip?: string;\n  };\n}\n\n/**\n * Audit configuration\n */\nexport interface AuditConfig {\n  enabled: boolean;\n  auditDir?: string;\n  includeContentHash?: boolean;\n  includeUserInfo?: boolean;\n  maxFileSize?: number;\n  maxFiles?: number;\n}\n\n/**\n * Default audit configuration\n */\nconst DEFAULT_AUDIT_CONFIG: AuditConfig = {\n  enabled: true,\n  auditDir: path.join(os.homedir(), '.llmverify', 'audit'),\n  includeContentHash: true,\n  includeUserInfo: false,\n  maxFileSize: 10 * 1024 * 1024, // 10MB\n  maxFiles: 50 // Keep more audit files\n};\n\n/**\n * Audit logger class\n */\nexport class AuditLogger {\n  private config: AuditConfig;\n  \n  constructor(config?: Partial<AuditConfig>) {\n    this.config = { ...DEFAULT_AUDIT_CONFIG, ...config };\n    this.ensureAuditDirectory();\n  }\n  \n  /**\n   * Ensure audit directory exists\n   */\n  private ensureAuditDirectory(): void {\n    if (this.config.enabled && this.config.auditDir) {\n      try {\n        fs.mkdirSync(this.config.auditDir, { recursive: true });\n      } catch (error) {\n        console.error('Failed to create audit directory:', error);\n        this.config.enabled = false;\n      }\n    }\n  }\n  \n  /**\n   * Get audit file path\n   */\n  private getAuditFilePath(): string {\n    const date = new Date().toISOString().split('T')[0];\n    return path.join(this.config.auditDir!, `audit-${date}.jsonl`);\n  }\n  \n  /**\n   * Generate content hash\n   */\n  private hashContent(content: string): string {\n    return crypto.createHash('sha256').update(content).digest('hex').substring(0, 16);\n  }\n  \n  /**\n   * Write audit entry\n   */\n  public log(entry: Omit<AuditEntry, 'timestamp'>): void {\n    if (!this.config.enabled) return;\n    \n    try {\n      const fullEntry: AuditEntry = {\n        timestamp: new Date().toISOString(),\n        ...entry\n      };\n      \n      const auditFile = this.getAuditFilePath();\n      const line = JSON.stringify(fullEntry) + '\\n';\n      \n      fs.appendFileSync(auditFile, line, 'utf-8');\n      \n      // Rotate if needed\n      this.rotateIfNeeded(auditFile);\n    } catch (error) {\n      if (process.env.NODE_ENV === 'development') {\n        console.error('Failed to write audit entry:', error);\n      }\n    }\n  }\n  \n  /**\n   * Log verification operation\n   */\n  public logVerification(params: {\n    requestId: string;\n    content: string;\n    prompt?: string;\n    riskLevel: string;\n    findingsCount: number;\n    blocked: boolean;\n    duration: number;\n    enginesUsed: string[];\n    configTier: string;\n    userId?: string;\n    userIp?: string;\n  }): void {\n    this.log({\n      requestId: params.requestId,\n      operation: 'verify',\n      input: {\n        contentLength: params.content.length,\n        contentHash: this.config.includeContentHash ? this.hashContent(params.content) : '',\n        hasPrompt: !!params.prompt\n      },\n      output: {\n        riskLevel: params.riskLevel,\n        findingsCount: params.findingsCount,\n        blocked: params.blocked\n      },\n      metadata: {\n        version: require('../../package.json').version,\n        duration: params.duration,\n        enginesUsed: params.enginesUsed,\n        configTier: params.configTier\n      },\n      user: this.config.includeUserInfo ? {\n        id: params.userId,\n        ip: params.userIp\n      } : undefined\n    });\n  }\n  \n  /**\n   * Rotate audit files\n   */\n  private rotateIfNeeded(auditFile: string): void {\n    try {\n      const stats = fs.statSync(auditFile);\n      \n      if (stats.size > this.config.maxFileSize!) {\n        const timestamp = Date.now();\n        const rotatedFile = auditFile.replace('.jsonl', `.${timestamp}.jsonl`);\n        fs.renameSync(auditFile, rotatedFile);\n        \n        this.cleanupOldAudits();\n      }\n    } catch (error) {\n      // Ignore rotation errors\n    }\n  }\n  \n  /**\n   * Clean up old audit files\n   */\n  private cleanupOldAudits(): void {\n    if (!this.config.auditDir) return;\n    \n    try {\n      const files = fs.readdirSync(this.config.auditDir)\n        .filter(f => f.startsWith('audit-') && f.endsWith('.jsonl'))\n        .map(f => ({\n          name: f,\n          path: path.join(this.config.auditDir!, f),\n          time: fs.statSync(path.join(this.config.auditDir!, f)).mtime.getTime()\n        }))\n        .sort((a, b) => b.time - a.time);\n      \n      if (files.length > this.config.maxFiles!) {\n        files.slice(this.config.maxFiles!).forEach(file => {\n          try {\n            fs.unlinkSync(file.path);\n          } catch (error) {\n            // Ignore deletion errors\n          }\n        });\n      }\n    } catch (error) {\n      // Ignore cleanup errors\n    }\n  }\n  \n  /**\n   * Read audit entries\n   */\n  public readAudit(date?: string): AuditEntry[] {\n    if (!this.config.auditDir) return [];\n    \n    const dateStr = date || new Date().toISOString().split('T')[0];\n    const auditFile = path.join(this.config.auditDir, `audit-${dateStr}.jsonl`);\n    \n    if (!fs.existsSync(auditFile)) return [];\n    \n    try {\n      const content = fs.readFileSync(auditFile, 'utf-8');\n      return content\n        .split('\\n')\n        .filter(line => line.trim())\n        .map(line => JSON.parse(line) as AuditEntry);\n    } catch (error) {\n      console.error('Failed to read audit:', error);\n      return [];\n    }\n  }\n  \n  /**\n   * Get audit statistics\n   */\n  public getStats(date?: string): {\n    totalOperations: number;\n    byOperation: Record<string, number>;\n    blockedCount: number;\n    avgDuration: number;\n    riskDistribution: Record<string, number>;\n  } {\n    const entries = this.readAudit(date);\n    \n    const stats = {\n      totalOperations: entries.length,\n      byOperation: {} as Record<string, number>,\n      blockedCount: 0,\n      avgDuration: 0,\n      riskDistribution: {} as Record<string, number>\n    };\n    \n    let totalDuration = 0;\n    \n    entries.forEach(entry => {\n      // Count by operation\n      stats.byOperation[entry.operation] = (stats.byOperation[entry.operation] || 0) + 1;\n      \n      // Count blocked\n      if (entry.output.blocked) stats.blockedCount++;\n      \n      // Sum duration\n      totalDuration += entry.metadata.duration;\n      \n      // Risk distribution\n      const risk = entry.output.riskLevel;\n      stats.riskDistribution[risk] = (stats.riskDistribution[risk] || 0) + 1;\n    });\n    \n    if (entries.length > 0) {\n      stats.avgDuration = totalDuration / entries.length;\n    }\n    \n    return stats;\n  }\n  \n  /**\n   * Export audit trail for compliance\n   */\n  public exportAuditTrail(startDate: string, endDate: string, outputPath: string): void {\n    const start = new Date(startDate);\n    const end = new Date(endDate);\n    const allEntries: AuditEntry[] = [];\n    \n    // Collect all entries in date range\n    for (let d = new Date(start); d <= end; d.setDate(d.getDate() + 1)) {\n      const dateStr = d.toISOString().split('T')[0];\n      const entries = this.readAudit(dateStr);\n      allEntries.push(...entries);\n    }\n    \n    // Write to output file\n    const report = {\n      exportDate: new Date().toISOString(),\n      dateRange: { start: startDate, end: endDate },\n      totalEntries: allEntries.length,\n      entries: allEntries\n    };\n    \n    fs.writeFileSync(outputPath, JSON.stringify(report, null, 2), 'utf-8');\n  }\n}\n\n/**\n * Global audit logger\n */\nlet globalAuditLogger: AuditLogger | null = null;\n\n/**\n * Get global audit logger\n */\nexport function getAuditLogger(config?: Partial<AuditConfig>): AuditLogger {\n  if (!globalAuditLogger) {\n    globalAuditLogger = new AuditLogger(config);\n  }\n  return globalAuditLogger;\n}\n\n/**\n * Set global audit logger\n */\nexport function setAuditLogger(logger: AuditLogger): void {\n  globalAuditLogger = logger;\n}\n"]}