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