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
JavaScript
;
/**
* 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":"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","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"]}