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.

165 lines 15 kB
"use strict"; /** * Plugin Registry System * * Extensible rule system for custom verification logic * * @module plugins/registry */ Object.defineProperty(exports, "__esModule", { value: true }); exports.PluginRegistry = void 0; exports.getPluginRegistry = getPluginRegistry; exports.resetPluginRegistry = resetPluginRegistry; /** * Plugin registry */ class PluginRegistry { constructor() { this.plugins = new Map(); } /** * Register a plugin */ register(plugin) { if (this.plugins.has(plugin.id)) { throw new Error(`Plugin ${plugin.id} is already registered`); } // Validate plugin this.validatePlugin(plugin); this.plugins.set(plugin.id, plugin); } /** * Validate plugin structure */ validatePlugin(plugin) { if (!plugin.id || typeof plugin.id !== 'string') { throw new Error('Plugin must have a valid id'); } if (!plugin.name || typeof plugin.name !== 'string') { throw new Error('Plugin must have a valid name'); } if (typeof plugin.execute !== 'function') { throw new Error('Plugin must have an execute function'); } } /** * Unregister a plugin */ unregister(pluginId) { return this.plugins.delete(pluginId); } /** * Get a plugin */ get(pluginId) { return this.plugins.get(pluginId); } /** * Get all plugins */ getAll() { return Array.from(this.plugins.values()); } /** * Get enabled plugins */ getEnabled() { return this.getAll() .filter(p => p.enabled) .sort((a, b) => (b.priority || 0) - (a.priority || 0)); } /** * Get plugins by category */ getByCategory(category) { return this.getAll().filter(p => p.category === category); } /** * Enable a plugin */ enable(pluginId) { const plugin = this.plugins.get(pluginId); if (plugin) { plugin.enabled = true; } } /** * Disable a plugin */ disable(pluginId) { const plugin = this.plugins.get(pluginId); if (plugin) { plugin.enabled = false; } } /** * Execute all enabled plugins */ async executeAll(context) { const enabledPlugins = this.getEnabled(); const results = []; for (const plugin of enabledPlugins) { try { const result = await plugin.execute(context); results.push(result); } catch (error) { console.error(`Plugin ${plugin.id} failed:`, error); // Continue with other plugins } } return results; } /** * Execute specific plugins */ async execute(pluginIds, context) { const results = []; for (const id of pluginIds) { const plugin = this.plugins.get(id); if (plugin && plugin.enabled) { try { const result = await plugin.execute(context); results.push(result); } catch (error) { console.error(`Plugin ${id} failed:`, error); } } } return results; } /** * Clear all plugins */ clear() { this.plugins.clear(); } /** * Get plugin count */ count() { return this.plugins.size; } } exports.PluginRegistry = PluginRegistry; /** * Global plugin registry */ let globalRegistry = null; /** * Get global plugin registry */ function getPluginRegistry() { if (!globalRegistry) { globalRegistry = new PluginRegistry(); } return globalRegistry; } /** * Reset global registry */ function resetPluginRegistry() { globalRegistry = null; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"registry.js","sourceRoot":"","sources":["../../src/plugins/registry.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Plugin Registry System\n * \n * Extensible rule system for custom verification logic\n * \n * @module plugins/registry\n */\n\n/**\n * Plugin context provided to rules\n */\nexport interface PluginContext {\n  content: string;\n  prompt?: string;\n  config?: any;\n  metadata?: Record<string, any>;\n}\n\n/**\n * Plugin result\n */\nexport interface PluginResult {\n  findings: Array<{\n    category: string;\n    severity: string;\n    message: string;\n    metadata?: any;\n  }>;\n  score?: number;\n  metadata?: Record<string, any>;\n}\n\n/**\n * Plugin function signature\n */\nexport type PluginFunction = (context: PluginContext) => PluginResult | Promise<PluginResult>;\n\n/**\n * Plugin definition\n */\nexport interface Plugin {\n  id: string;\n  name: string;\n  version: string;\n  description?: string;\n  author?: string;\n  category: 'security' | 'privacy' | 'safety' | 'quality' | 'custom';\n  enabled: boolean;\n  priority?: number; // Higher priority runs first\n  execute: PluginFunction;\n}\n\n/**\n * Plugin registry\n */\nexport class PluginRegistry {\n  private plugins: Map<string, Plugin> = new Map();\n  \n  /**\n   * Register a plugin\n   */\n  public register(plugin: Plugin): void {\n    if (this.plugins.has(plugin.id)) {\n      throw new Error(`Plugin ${plugin.id} is already registered`);\n    }\n    \n    // Validate plugin\n    this.validatePlugin(plugin);\n    \n    this.plugins.set(plugin.id, plugin);\n  }\n  \n  /**\n   * Validate plugin structure\n   */\n  private validatePlugin(plugin: Plugin): void {\n    if (!plugin.id || typeof plugin.id !== 'string') {\n      throw new Error('Plugin must have a valid id');\n    }\n    \n    if (!plugin.name || typeof plugin.name !== 'string') {\n      throw new Error('Plugin must have a valid name');\n    }\n    \n    if (typeof plugin.execute !== 'function') {\n      throw new Error('Plugin must have an execute function');\n    }\n  }\n  \n  /**\n   * Unregister a plugin\n   */\n  public unregister(pluginId: string): boolean {\n    return this.plugins.delete(pluginId);\n  }\n  \n  /**\n   * Get a plugin\n   */\n  public get(pluginId: string): Plugin | undefined {\n    return this.plugins.get(pluginId);\n  }\n  \n  /**\n   * Get all plugins\n   */\n  public getAll(): Plugin[] {\n    return Array.from(this.plugins.values());\n  }\n  \n  /**\n   * Get enabled plugins\n   */\n  public getEnabled(): Plugin[] {\n    return this.getAll()\n      .filter(p => p.enabled)\n      .sort((a, b) => (b.priority || 0) - (a.priority || 0));\n  }\n  \n  /**\n   * Get plugins by category\n   */\n  public getByCategory(category: Plugin['category']): Plugin[] {\n    return this.getAll().filter(p => p.category === category);\n  }\n  \n  /**\n   * Enable a plugin\n   */\n  public enable(pluginId: string): void {\n    const plugin = this.plugins.get(pluginId);\n    if (plugin) {\n      plugin.enabled = true;\n    }\n  }\n  \n  /**\n   * Disable a plugin\n   */\n  public disable(pluginId: string): void {\n    const plugin = this.plugins.get(pluginId);\n    if (plugin) {\n      plugin.enabled = false;\n    }\n  }\n  \n  /**\n   * Execute all enabled plugins\n   */\n  public async executeAll(context: PluginContext): Promise<PluginResult[]> {\n    const enabledPlugins = this.getEnabled();\n    const results: PluginResult[] = [];\n    \n    for (const plugin of enabledPlugins) {\n      try {\n        const result = await plugin.execute(context);\n        results.push(result);\n      } catch (error) {\n        console.error(`Plugin ${plugin.id} failed:`, error);\n        // Continue with other plugins\n      }\n    }\n    \n    return results;\n  }\n  \n  /**\n   * Execute specific plugins\n   */\n  public async execute(pluginIds: string[], context: PluginContext): Promise<PluginResult[]> {\n    const results: PluginResult[] = [];\n    \n    for (const id of pluginIds) {\n      const plugin = this.plugins.get(id);\n      if (plugin && plugin.enabled) {\n        try {\n          const result = await plugin.execute(context);\n          results.push(result);\n        } catch (error) {\n          console.error(`Plugin ${id} failed:`, error);\n        }\n      }\n    }\n    \n    return results;\n  }\n  \n  /**\n   * Clear all plugins\n   */\n  public clear(): void {\n    this.plugins.clear();\n  }\n  \n  /**\n   * Get plugin count\n   */\n  public count(): number {\n    return this.plugins.size;\n  }\n}\n\n/**\n * Global plugin registry\n */\nlet globalRegistry: PluginRegistry | null = null;\n\n/**\n * Get global plugin registry\n */\nexport function getPluginRegistry(): PluginRegistry {\n  if (!globalRegistry) {\n    globalRegistry = new PluginRegistry();\n  }\n  return globalRegistry;\n}\n\n/**\n * Reset global registry\n */\nexport function resetPluginRegistry(): void {\n  globalRegistry = null;\n}\n"]}