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