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.

181 lines 21.6 kB
"use strict"; /** * JSON Validator Engine * * Validates JSON structure and optionally schema. * * @module engines/json-validator * @author Haiec * @license MIT */ Object.defineProperty(exports, "__esModule", { value: true }); exports.JSONValidatorEngine = void 0; class JSONValidatorEngine { constructor(_config) { this._config = _config; this.LIMITATIONS = [ 'Basic JSON structure validation', 'Simple schema validation (type checking only)', 'Does not validate semantic correctness', 'Limited repair capabilities' ]; this.METHODOLOGY = 'Parses JSON content and validates structure. Performs basic schema ' + 'validation if schema is provided. Attempts simple repairs for common issues.'; } async validate(content, schema) { let parsed = null; let valid = false; let repaired = false; let repairMethod; const issues = []; // Try to parse try { parsed = JSON.parse(content); valid = true; } catch (e) { // Try to repair const repairResult = this.attemptRepair(content); if (repairResult.success) { parsed = repairResult.parsed; valid = true; repaired = true; repairMethod = repairResult.method; issues.push(`Repaired: ${repairMethod}`); } else { issues.push(`Parse error: ${e.message}`); } } // Schema validation let schemaValid = true; const schemaErrors = []; if (valid && schema && parsed) { const schemaResult = this.validateSchema(parsed, schema); schemaValid = schemaResult.valid; schemaErrors.push(...schemaResult.errors); } // Structure analysis const structure = this.analyzeStructure(parsed); structure.issues.push(...issues); return { valid, parsed, schema, schemaValid, schemaErrors, repaired, repairMethod, structure, limitations: this.LIMITATIONS, methodology: this.METHODOLOGY }; } attemptRepair(content) { // Try common repairs const repairs = [ { name: 'trailing comma removal', fix: (s) => s.replace(/,\s*([\]}])/g, '$1') }, { name: 'single to double quotes', fix: (s) => s.replace(/'/g, '"') }, { name: 'unquoted keys', fix: (s) => s.replace(/(\{|,)\s*([a-zA-Z_][a-zA-Z0-9_]*)\s*:/g, '$1"$2":') }, { name: 'extract JSON from text', fix: (s) => { const match = s.match(/\{[\s\S]*\}|\[[\s\S]*\]/); return match ? match[0] : s; } } ]; for (const repair of repairs) { try { const fixed = repair.fix(content); const parsed = JSON.parse(fixed); return { success: true, parsed, method: repair.name }; } catch { continue; } } return { success: false }; } validateSchema(data, schema) { const errors = []; if (typeof schema !== 'object' || schema === null) { return { valid: true, errors: [] }; } const schemaObj = schema; if (typeof data !== 'object' || data === null) { errors.push('Data is not an object'); return { valid: false, errors }; } const dataObj = data; // Simple type checking for (const [key, expectedType] of Object.entries(schemaObj)) { if (!(key in dataObj)) { errors.push(`Missing required field: ${key}`); continue; } const actualType = typeof dataObj[key]; if (expectedType === 'array') { if (!Array.isArray(dataObj[key])) { errors.push(`Field ${key}: expected array, got ${actualType}`); } } else if (actualType !== expectedType) { errors.push(`Field ${key}: expected ${expectedType}, got ${actualType}`); } } return { valid: errors.length === 0, errors }; } analyzeStructure(data) { if (data === null || data === undefined) { return { depth: 0, keyCount: 0, issues: ['No valid data to analyze'] }; } const depth = this.calculateDepth(data); const keyCount = this.countKeys(data); const issues = []; if (depth > 10) { issues.push('Deep nesting detected (>10 levels)'); } if (keyCount > 1000) { issues.push('Large object detected (>1000 keys)'); } return { depth, keyCount, issues }; } calculateDepth(data, current = 0) { if (typeof data !== 'object' || data === null) { return current; } if (Array.isArray(data)) { return Math.max(current, ...data.map(item => this.calculateDepth(item, current + 1))); } const values = Object.values(data); if (values.length === 0) return current; return Math.max(...values.map(v => this.calculateDepth(v, current + 1))); } countKeys(data) { if (typeof data !== 'object' || data === null) { return 0; } if (Array.isArray(data)) { return data.reduce((sum, item) => sum + this.countKeys(item), 0); } const obj = data; let count = Object.keys(obj).length; for (const value of Object.values(obj)) { count += this.countKeys(value); } return count; } } exports.JSONValidatorEngine = JSONValidatorEngine; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/engines/json-validator/index.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * JSON Validator Engine\n * \n * Validates JSON structure and optionally schema.\n * \n * @module engines/json-validator\n * @author Haiec\n * @license MIT\n */\n\nimport { Config } from '../../types/config';\nimport { JSONResult } from '../../types/results';\n\nexport class JSONValidatorEngine {\n  private readonly LIMITATIONS = [\n    'Basic JSON structure validation',\n    'Simple schema validation (type checking only)',\n    'Does not validate semantic correctness',\n    'Limited repair capabilities'\n  ];\n  \n  private readonly METHODOLOGY = \n    'Parses JSON content and validates structure. Performs basic schema ' +\n    'validation if schema is provided. Attempts simple repairs for common issues.';\n  \n  constructor(private _config: Config) {}\n  \n  async validate(content: string, schema?: unknown): Promise<JSONResult> {\n    let parsed: unknown = null;\n    let valid = false;\n    let repaired = false;\n    let repairMethod: string | undefined;\n    const issues: string[] = [];\n    \n    // Try to parse\n    try {\n      parsed = JSON.parse(content);\n      valid = true;\n    } catch (e) {\n      // Try to repair\n      const repairResult = this.attemptRepair(content);\n      if (repairResult.success) {\n        parsed = repairResult.parsed;\n        valid = true;\n        repaired = true;\n        repairMethod = repairResult.method;\n        issues.push(`Repaired: ${repairMethod}`);\n      } else {\n        issues.push(`Parse error: ${(e as Error).message}`);\n      }\n    }\n    \n    // Schema validation\n    let schemaValid = true;\n    const schemaErrors: string[] = [];\n    \n    if (valid && schema && parsed) {\n      const schemaResult = this.validateSchema(parsed, schema);\n      schemaValid = schemaResult.valid;\n      schemaErrors.push(...schemaResult.errors);\n    }\n    \n    // Structure analysis\n    const structure = this.analyzeStructure(parsed);\n    structure.issues.push(...issues);\n    \n    return {\n      valid,\n      parsed,\n      schema,\n      schemaValid,\n      schemaErrors,\n      repaired,\n      repairMethod,\n      structure,\n      limitations: this.LIMITATIONS,\n      methodology: this.METHODOLOGY\n    };\n  }\n  \n  private attemptRepair(content: string): { success: boolean; parsed?: unknown; method?: string } {\n    // Try common repairs\n    const repairs = [\n      {\n        name: 'trailing comma removal',\n        fix: (s: string) => s.replace(/,\\s*([\\]}])/g, '$1')\n      },\n      {\n        name: 'single to double quotes',\n        fix: (s: string) => s.replace(/'/g, '\"')\n      },\n      {\n        name: 'unquoted keys',\n        fix: (s: string) => s.replace(/(\\{|,)\\s*([a-zA-Z_][a-zA-Z0-9_]*)\\s*:/g, '$1\"$2\":')\n      },\n      {\n        name: 'extract JSON from text',\n        fix: (s: string) => {\n          const match = s.match(/\\{[\\s\\S]*\\}|\\[[\\s\\S]*\\]/);\n          return match ? match[0] : s;\n        }\n      }\n    ];\n    \n    for (const repair of repairs) {\n      try {\n        const fixed = repair.fix(content);\n        const parsed = JSON.parse(fixed);\n        return { success: true, parsed, method: repair.name };\n      } catch {\n        continue;\n      }\n    }\n    \n    return { success: false };\n  }\n  \n  private validateSchema(\n    data: unknown, \n    schema: unknown\n  ): { valid: boolean; errors: string[] } {\n    const errors: string[] = [];\n    \n    if (typeof schema !== 'object' || schema === null) {\n      return { valid: true, errors: [] };\n    }\n    \n    const schemaObj = schema as Record<string, unknown>;\n    \n    if (typeof data !== 'object' || data === null) {\n      errors.push('Data is not an object');\n      return { valid: false, errors };\n    }\n    \n    const dataObj = data as Record<string, unknown>;\n    \n    // Simple type checking\n    for (const [key, expectedType] of Object.entries(schemaObj)) {\n      if (!(key in dataObj)) {\n        errors.push(`Missing required field: ${key}`);\n        continue;\n      }\n      \n      const actualType = typeof dataObj[key];\n      \n      if (expectedType === 'array') {\n        if (!Array.isArray(dataObj[key])) {\n          errors.push(`Field ${key}: expected array, got ${actualType}`);\n        }\n      } else if (actualType !== expectedType) {\n        errors.push(`Field ${key}: expected ${expectedType}, got ${actualType}`);\n      }\n    }\n    \n    return { valid: errors.length === 0, errors };\n  }\n  \n  private analyzeStructure(data: unknown): { depth: number; keyCount: number; issues: string[] } {\n    if (data === null || data === undefined) {\n      return { depth: 0, keyCount: 0, issues: ['No valid data to analyze'] };\n    }\n    \n    const depth = this.calculateDepth(data);\n    const keyCount = this.countKeys(data);\n    const issues: string[] = [];\n    \n    if (depth > 10) {\n      issues.push('Deep nesting detected (>10 levels)');\n    }\n    \n    if (keyCount > 1000) {\n      issues.push('Large object detected (>1000 keys)');\n    }\n    \n    return { depth, keyCount, issues };\n  }\n  \n  private calculateDepth(data: unknown, current = 0): number {\n    if (typeof data !== 'object' || data === null) {\n      return current;\n    }\n    \n    if (Array.isArray(data)) {\n      return Math.max(current, ...data.map(item => this.calculateDepth(item, current + 1)));\n    }\n    \n    const values = Object.values(data as Record<string, unknown>);\n    if (values.length === 0) return current;\n    \n    return Math.max(...values.map(v => this.calculateDepth(v, current + 1)));\n  }\n  \n  private countKeys(data: unknown): number {\n    if (typeof data !== 'object' || data === null) {\n      return 0;\n    }\n    \n    if (Array.isArray(data)) {\n      return data.reduce((sum, item) => sum + this.countKeys(item), 0);\n    }\n    \n    const obj = data as Record<string, unknown>;\n    let count = Object.keys(obj).length;\n    \n    for (const value of Object.values(obj)) {\n      count += this.countKeys(value);\n    }\n    \n    return count;\n  }\n}\n"]}