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.

182 lines 20.6 kB
"use strict"; /** * Plugin API * * High-level API for creating and using plugins * * @module plugins/api */ Object.defineProperty(exports, "__esModule", { value: true }); exports.createPlugin = createPlugin; exports.use = use; exports.createBlacklistPlugin = createBlacklistPlugin; exports.createRegexPlugin = createRegexPlugin; exports.createLengthValidatorPlugin = createLengthValidatorPlugin; exports.createKeywordDetectorPlugin = createKeywordDetectorPlugin; const registry_1 = require("./registry"); /** * Helper function to create a plugin */ function createPlugin(config) { return { id: config.id, name: config.name, version: config.version || '1.0.0', description: config.description, author: config.author, category: config.category || 'custom', enabled: config.enabled !== false, priority: config.priority || 0, execute: config.execute }; } /** * Use a plugin (register and enable) */ function use(plugin) { const registry = (0, registry_1.getPluginRegistry)(); registry.register(plugin); } /** * Create a blacklist plugin */ function createBlacklistPlugin(blacklist, options) { return createPlugin({ id: options?.id || 'blacklist', name: options?.name || 'Blacklist Filter', description: 'Detects blacklisted words', category: 'security', execute: (context) => { const findings = []; const content = options?.caseSensitive ? context.content : context.content.toLowerCase(); for (const word of blacklist) { const searchWord = options?.caseSensitive ? word : word.toLowerCase(); if (content.includes(searchWord)) { findings.push({ category: 'security', severity: 'medium', message: `Blacklisted term detected: ${word}` }); } } return { findings, score: findings.length > 0 ? 0.5 : 0 }; } }); } /** * Create a regex pattern plugin */ function createRegexPlugin(patterns, options) { return createPlugin({ id: options?.id || 'regex-patterns', name: options?.name || 'Regex Patterns', description: 'Detects custom regex patterns', category: 'custom', execute: (context) => { const findings = []; for (const { pattern, message, severity } of patterns) { try { const matches = context.content.match(pattern); if (matches) { findings.push({ category: 'custom', severity: severity || 'medium', message: message }); } } catch (error) { console.error('Regex pattern error:', error); } } return { findings, score: findings.length > 0 ? 0.5 : 0 }; } }); } /** * Create a length validator plugin */ function createLengthValidatorPlugin(config, options) { return createPlugin({ id: options?.id || 'length-validator', name: options?.name || 'Length Validator', description: 'Validates content length', category: 'quality', execute: (context) => { const findings = []; const length = context.content.length; if (config.min && length < config.min) { findings.push({ category: 'quality', severity: 'low', message: `Content too short: ${length} < ${config.min}`, metadata: { length, min: config.min } }); } if (config.max && length > config.max) { findings.push({ category: 'quality', severity: 'medium', message: `Content too long: ${length} > ${config.max}`, metadata: { length, max: config.max } }); } return { findings, score: findings.length > 0 ? 0.3 : 0 }; } }); } /** * Create a keyword detector plugin */ function createKeywordDetectorPlugin(keywords, options) { return createPlugin({ id: options?.id || 'keyword-detector', name: options?.name || 'Keyword Detector', description: 'Detects required and forbidden keywords', category: 'quality', execute: (context) => { const findings = []; const content = options?.caseSensitive ? context.content : context.content.toLowerCase(); // Check required keywords if (keywords.required) { for (const keyword of keywords.required) { const searchKeyword = options?.caseSensitive ? keyword : keyword.toLowerCase(); if (!content.includes(searchKeyword)) { findings.push({ category: 'quality', severity: 'medium', message: `Required keyword missing: ${keyword}` }); } } } // Check forbidden keywords if (keywords.forbidden) { for (const keyword of keywords.forbidden) { const searchKeyword = options?.caseSensitive ? keyword : keyword.toLowerCase(); if (content.includes(searchKeyword)) { findings.push({ category: 'security', severity: 'high', message: `Forbidden keyword detected: ${keyword}` }); } } } return { findings, score: findings.length > 0 ? 0.6 : 0 }; } }); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"api.js","sourceRoot":"","sources":["../../src/plugins/api.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Plugin API\n * \n * High-level API for creating and using plugins\n * \n * @module plugins/api\n */\n\nimport { Plugin, PluginFunction, getPluginRegistry } from './registry';\n\n/**\n * Helper function to create a plugin\n */\nexport function createPlugin(config: {\n  id: string;\n  name: string;\n  version?: string;\n  description?: string;\n  author?: string;\n  category?: Plugin['category'];\n  enabled?: boolean;\n  priority?: number;\n  execute: PluginFunction;\n}): Plugin {\n  return {\n    id: config.id,\n    name: config.name,\n    version: config.version || '1.0.0',\n    description: config.description,\n    author: config.author,\n    category: config.category || 'custom',\n    enabled: config.enabled !== false,\n    priority: config.priority || 0,\n    execute: config.execute\n  };\n}\n\n/**\n * Use a plugin (register and enable)\n */\nexport function use(plugin: Plugin): void {\n  const registry = getPluginRegistry();\n  registry.register(plugin);\n}\n\n/**\n * Create a blacklist plugin\n */\nexport function createBlacklistPlugin(blacklist: string[], options?: {\n  id?: string;\n  name?: string;\n  caseSensitive?: boolean;\n}): Plugin {\n  return createPlugin({\n    id: options?.id || 'blacklist',\n    name: options?.name || 'Blacklist Filter',\n    description: 'Detects blacklisted words',\n    category: 'security',\n    execute: (context) => {\n      const findings: any[] = [];\n      const content = options?.caseSensitive ? context.content : context.content.toLowerCase();\n      \n      for (const word of blacklist) {\n        const searchWord = options?.caseSensitive ? word : word.toLowerCase();\n        if (content.includes(searchWord)) {\n          findings.push({\n            category: 'security',\n            severity: 'medium',\n            message: `Blacklisted term detected: ${word}`\n          });\n        }\n      }\n      \n      return {\n        findings,\n        score: findings.length > 0 ? 0.5 : 0\n      };\n    }\n  });\n}\n\n/**\n * Create a regex pattern plugin\n */\nexport function createRegexPlugin(patterns: Array<{\n  pattern: RegExp;\n  message: string;\n  severity?: string;\n}>, options?: {\n  id?: string;\n  name?: string;\n}): Plugin {\n  return createPlugin({\n    id: options?.id || 'regex-patterns',\n    name: options?.name || 'Regex Patterns',\n    description: 'Detects custom regex patterns',\n    category: 'custom',\n    execute: (context) => {\n      const findings: any[] = [];\n      \n      for (const { pattern, message, severity } of patterns) {\n        try {\n          const matches = context.content.match(pattern);\n          if (matches) {\n            findings.push({\n              category: 'custom',\n              severity: severity || 'medium',\n              message: message\n            });\n          }\n        } catch (error) {\n          console.error('Regex pattern error:', error);\n        }\n      }\n      \n      return {\n        findings,\n        score: findings.length > 0 ? 0.5 : 0\n      };\n    }\n  });\n}\n\n/**\n * Create a length validator plugin\n */\nexport function createLengthValidatorPlugin(config: {\n  min?: number;\n  max?: number;\n}, options?: {\n  id?: string;\n  name?: string;\n}): Plugin {\n  return createPlugin({\n    id: options?.id || 'length-validator',\n    name: options?.name || 'Length Validator',\n    description: 'Validates content length',\n    category: 'quality',\n    execute: (context) => {\n      const findings: any[] = [];\n      const length = context.content.length;\n      \n      if (config.min && length < config.min) {\n        findings.push({\n          category: 'quality',\n          severity: 'low',\n          message: `Content too short: ${length} < ${config.min}`,\n          metadata: { length, min: config.min }\n        });\n      }\n      \n      if (config.max && length > config.max) {\n        findings.push({\n          category: 'quality',\n          severity: 'medium',\n          message: `Content too long: ${length} > ${config.max}`,\n          metadata: { length, max: config.max }\n        });\n      }\n      \n      return {\n        findings,\n        score: findings.length > 0 ? 0.3 : 0\n      };\n    }\n  });\n}\n\n/**\n * Create a keyword detector plugin\n */\nexport function createKeywordDetectorPlugin(keywords: {\n  required?: string[];\n  forbidden?: string[];\n}, options?: {\n  id?: string;\n  name?: string;\n  caseSensitive?: boolean;\n}): Plugin {\n  return createPlugin({\n    id: options?.id || 'keyword-detector',\n    name: options?.name || 'Keyword Detector',\n    description: 'Detects required and forbidden keywords',\n    category: 'quality',\n    execute: (context) => {\n      const findings: any[] = [];\n      const content = options?.caseSensitive ? context.content : context.content.toLowerCase();\n      \n      // Check required keywords\n      if (keywords.required) {\n        for (const keyword of keywords.required) {\n          const searchKeyword = options?.caseSensitive ? keyword : keyword.toLowerCase();\n          if (!content.includes(searchKeyword)) {\n            findings.push({\n              category: 'quality',\n              severity: 'medium',\n              message: `Required keyword missing: ${keyword}`\n            });\n          }\n        }\n      }\n      \n      // Check forbidden keywords\n      if (keywords.forbidden) {\n        for (const keyword of keywords.forbidden) {\n          const searchKeyword = options?.caseSensitive ? keyword : keyword.toLowerCase();\n          if (content.includes(searchKeyword)) {\n            findings.push({\n              category: 'security',\n              severity: 'high',\n              message: `Forbidden keyword detected: ${keyword}`\n            });\n          }\n        }\n      }\n      \n      return {\n        findings,\n        score: findings.length > 0 ? 0.6 : 0\n      };\n    }\n  });\n}\n"]}