@mickdarling/dollhousemcp
Version:
DollhouseMCP - A Model Context Protocol (MCP) server that enables dynamic AI persona management from markdown files, allowing Claude and other compatible AI assistants to activate and switch between different behavioral personas.
158 lines • 23.2 kB
JavaScript
// Default configuration
export const DEFAULT_INDICATOR_CONFIG = {
enabled: true,
style: 'full',
showEmoji: true,
showName: true,
showVersion: true,
showAuthor: true,
showCategory: false,
separator: ' | ',
emoji: '🎭',
bracketStyle: 'square'
};
// Predefined styles
export const INDICATOR_STYLES = {
full: '[{emoji} {name} v{version} by {author}]',
minimal: '{emoji} {name}',
compact: '[{name} v{version}]',
custom: '{customFormat}'
};
// Bracket mappings
export const BRACKETS = {
square: { open: '[', close: ']' },
round: { open: '(', close: ')' },
curly: { open: '{', close: '}' },
angle: { open: '<', close: '>' },
none: { open: '', close: '' }
};
/**
* Load indicator configuration from environment variables or use defaults.
* Environment variables take precedence over default values.
*
* @returns {IndicatorConfig} The loaded configuration
* @example
* ```typescript
* // Set environment variables before loading
* process.env.DOLLHOUSE_INDICATOR_STYLE = 'minimal';
* process.env.DOLLHOUSE_INDICATOR_EMOJI = '🤖';
*
* const config = loadIndicatorConfig();
* // config.style === 'minimal'
* // config.emoji === '🤖'
* ```
*/
export function loadIndicatorConfig() {
const config = { ...DEFAULT_INDICATOR_CONFIG };
// Check environment variables for overrides
if (process.env.DOLLHOUSE_INDICATOR_ENABLED !== undefined) {
config.enabled = process.env.DOLLHOUSE_INDICATOR_ENABLED === 'true';
}
if (process.env.DOLLHOUSE_INDICATOR_STYLE) {
const style = process.env.DOLLHOUSE_INDICATOR_STYLE;
if (['full', 'minimal', 'compact', 'custom'].includes(style)) {
config.style = style;
}
}
if (process.env.DOLLHOUSE_INDICATOR_FORMAT) {
config.customFormat = process.env.DOLLHOUSE_INDICATOR_FORMAT;
config.style = 'custom';
}
if (process.env.DOLLHOUSE_INDICATOR_EMOJI) {
config.emoji = process.env.DOLLHOUSE_INDICATOR_EMOJI;
}
if (process.env.DOLLHOUSE_INDICATOR_BRACKETS) {
const bracketStyle = process.env.DOLLHOUSE_INDICATOR_BRACKETS;
if (['square', 'round', 'curly', 'angle', 'none'].includes(bracketStyle)) {
config.bracketStyle = bracketStyle;
}
}
// Parse show flags from environment
if (process.env.DOLLHOUSE_INDICATOR_SHOW_VERSION !== undefined) {
config.showVersion = process.env.DOLLHOUSE_INDICATOR_SHOW_VERSION === 'true';
}
if (process.env.DOLLHOUSE_INDICATOR_SHOW_AUTHOR !== undefined) {
config.showAuthor = process.env.DOLLHOUSE_INDICATOR_SHOW_AUTHOR === 'true';
}
if (process.env.DOLLHOUSE_INDICATOR_SHOW_CATEGORY !== undefined) {
config.showCategory = process.env.DOLLHOUSE_INDICATOR_SHOW_CATEGORY === 'true';
}
return config;
}
/**
* Validate custom format template for valid placeholders
*/
export function validateCustomFormat(format) {
const validPlaceholders = ['{emoji}', '{name}', '{version}', '{author}', '{category}'];
// Length limit added to prevent ReDoS attacks
const placeholderRegex = /\{[^}]{0,50}\}/g; // Limited to 50 chars for placeholder names
const foundPlaceholders = format.match(placeholderRegex) || [];
for (const placeholder of foundPlaceholders) {
if (!validPlaceholders.includes(placeholder)) {
return {
valid: false,
error: `Invalid placeholder: ${placeholder}. Valid placeholders are: ${validPlaceholders.join(', ')}`
};
}
}
return { valid: true };
}
/**
* Format the indicator based on configuration and persona metadata
*/
export function formatIndicator(config, metadata) {
if (!config.enabled) {
return '';
}
// Get the format template based on style
let template = INDICATOR_STYLES[config.style];
if (config.style === 'custom' && config.customFormat) {
// Validate custom format
const validation = validateCustomFormat(config.customFormat);
if (!validation.valid) {
// Fall back to full style if custom format is invalid
template = INDICATOR_STYLES.full;
}
else {
template = config.customFormat;
}
}
// Replace placeholders with values or empty strings
let result = template
.replace('{emoji}', config.showEmoji ? config.emoji : '')
.replace('{name}', config.showName ? metadata.name : '')
.replace('{version}', config.showVersion && metadata.version ? metadata.version : '')
.replace('{author}', config.showAuthor && metadata.author ? metadata.author : '')
.replace('{category}', config.showCategory && metadata.category ? metadata.category : '');
// Clean up the format string
// Remove "v" if no version follows it
if (!config.showVersion || !metadata.version) {
result = result.replace(/\sv(?=\s|]|\)|>|}|$)/, '');
}
// Remove "by" if no author follows it
if (!config.showAuthor || !metadata.author) {
result = result.replace(/\sby(?=\s|]|\)|>|}|$)/, '');
}
// Clean up extra spaces
result = result.replace(/\s+/g, ' ').trim();
// Apply brackets based on the template format (only if template doesn't already have them)
if (result && config.style !== 'custom') {
// Check if the template already includes brackets
const templateHasBrackets = template.includes('[') || template.includes(']') ||
template.includes('(') || template.includes(')') ||
template.includes('{') || template.includes('}') ||
template.includes('<') || template.includes('>');
if (!templateHasBrackets) {
const brackets = BRACKETS[config.bracketStyle];
if (brackets.open || brackets.close) {
result = `${brackets.open}${result}${brackets.close}`;
}
}
}
// Add separator if we have content
if (result) {
result += config.separator;
}
return result;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"indicator-config.js","sourceRoot":"","sources":["../../src/config/indicator-config.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Configuration interface for the persona active indicator system.\n * Controls how persona information is displayed in AI responses.\n * \n * @interface IndicatorConfig\n * @example\n * ```typescript\n * const config: IndicatorConfig = {\n *   enabled: true,\n *   style: 'full',\n *   showEmoji: true,\n *   showName: true,\n *   showVersion: true,\n *   showAuthor: true,\n *   showCategory: false,\n *   separator: ' | ',\n *   emoji: '🎭',\n *   bracketStyle: 'square'\n * };\n * ```\n */\nexport interface IndicatorConfig {\n  /** Whether to show the indicator at all */\n  enabled: boolean;\n  \n  /** Format style: 'full', 'minimal', 'compact', 'custom' */\n  style: 'full' | 'minimal' | 'compact' | 'custom';\n  \n  /** \n   * Custom format template (used when style is 'custom')\n   * Available placeholders: {emoji}, {name}, {version}, {author}, {category}\n   */\n  customFormat?: string;\n  \n  /** Whether to include specific elements */\n  showEmoji: boolean;\n  showName: boolean;\n  showVersion: boolean;\n  showAuthor: boolean;\n  showCategory: boolean;\n  \n  /** Separator between indicator and response */\n  separator: string;\n  \n  /** Emoji to use (defaults to 🎭) */\n  emoji: string;\n  \n  /** Bracket style: 'square', 'round', 'curly', 'angle', 'none' */\n  bracketStyle: 'square' | 'round' | 'curly' | 'angle' | 'none';\n}\n\n// Default configuration\nexport const DEFAULT_INDICATOR_CONFIG: IndicatorConfig = {\n  enabled: true,\n  style: 'full',\n  showEmoji: true,\n  showName: true,\n  showVersion: true,\n  showAuthor: true,\n  showCategory: false,\n  separator: ' | ',\n  emoji: '🎭',\n  bracketStyle: 'square'\n};\n\n// Predefined styles\nexport const INDICATOR_STYLES = {\n  full: '[{emoji} {name} v{version} by {author}]',\n  minimal: '{emoji} {name}',\n  compact: '[{name} v{version}]',\n  custom: '{customFormat}'\n};\n\n// Bracket mappings\nexport const BRACKETS = {\n  square: { open: '[', close: ']' },\n  round: { open: '(', close: ')' },\n  curly: { open: '{', close: '}' },\n  angle: { open: '<', close: '>' },\n  none: { open: '', close: '' }\n};\n\n/**\n * Load indicator configuration from environment variables or use defaults.\n * Environment variables take precedence over default values.\n * \n * @returns {IndicatorConfig} The loaded configuration\n * @example\n * ```typescript\n * // Set environment variables before loading\n * process.env.DOLLHOUSE_INDICATOR_STYLE = 'minimal';\n * process.env.DOLLHOUSE_INDICATOR_EMOJI = '🤖';\n * \n * const config = loadIndicatorConfig();\n * // config.style === 'minimal'\n * // config.emoji === '🤖'\n * ```\n */\nexport function loadIndicatorConfig(): IndicatorConfig {\n  const config = { ...DEFAULT_INDICATOR_CONFIG };\n  \n  // Check environment variables for overrides\n  if (process.env.DOLLHOUSE_INDICATOR_ENABLED !== undefined) {\n    config.enabled = process.env.DOLLHOUSE_INDICATOR_ENABLED === 'true';\n  }\n  \n  if (process.env.DOLLHOUSE_INDICATOR_STYLE) {\n    const style = process.env.DOLLHOUSE_INDICATOR_STYLE;\n    if (['full', 'minimal', 'compact', 'custom'].includes(style)) {\n      config.style = style as IndicatorConfig['style'];\n    }\n  }\n  \n  if (process.env.DOLLHOUSE_INDICATOR_FORMAT) {\n    config.customFormat = process.env.DOLLHOUSE_INDICATOR_FORMAT;\n    config.style = 'custom';\n  }\n  \n  if (process.env.DOLLHOUSE_INDICATOR_EMOJI) {\n    config.emoji = process.env.DOLLHOUSE_INDICATOR_EMOJI;\n  }\n  \n  if (process.env.DOLLHOUSE_INDICATOR_BRACKETS) {\n    const bracketStyle = process.env.DOLLHOUSE_INDICATOR_BRACKETS;\n    if (['square', 'round', 'curly', 'angle', 'none'].includes(bracketStyle)) {\n      config.bracketStyle = bracketStyle as IndicatorConfig['bracketStyle'];\n    }\n  }\n  \n  // Parse show flags from environment\n  if (process.env.DOLLHOUSE_INDICATOR_SHOW_VERSION !== undefined) {\n    config.showVersion = process.env.DOLLHOUSE_INDICATOR_SHOW_VERSION === 'true';\n  }\n  \n  if (process.env.DOLLHOUSE_INDICATOR_SHOW_AUTHOR !== undefined) {\n    config.showAuthor = process.env.DOLLHOUSE_INDICATOR_SHOW_AUTHOR === 'true';\n  }\n  \n  if (process.env.DOLLHOUSE_INDICATOR_SHOW_CATEGORY !== undefined) {\n    config.showCategory = process.env.DOLLHOUSE_INDICATOR_SHOW_CATEGORY === 'true';\n  }\n  \n  return config;\n}\n\n/**\n * Validate custom format template for valid placeholders\n */\nexport function validateCustomFormat(format: string): { valid: boolean; error?: string } {\n  const validPlaceholders = ['{emoji}', '{name}', '{version}', '{author}', '{category}'];\n  // Length limit added to prevent ReDoS attacks\n  const placeholderRegex = /\\{[^}]{0,50}\\}/g;  // Limited to 50 chars for placeholder names\n  const foundPlaceholders = format.match(placeholderRegex) || [];\n  \n  for (const placeholder of foundPlaceholders) {\n    if (!validPlaceholders.includes(placeholder)) {\n      return {\n        valid: false,\n        error: `Invalid placeholder: ${placeholder}. Valid placeholders are: ${validPlaceholders.join(', ')}`\n      };\n    }\n  }\n  \n  return { valid: true };\n}\n\n/**\n * Format the indicator based on configuration and persona metadata\n */\nexport function formatIndicator(\n  config: IndicatorConfig,\n  metadata: {\n    name: string;\n    version?: string;\n    author?: string;\n    category?: string;\n  }\n): string {\n  if (!config.enabled) {\n    return '';\n  }\n  \n  // Get the format template based on style\n  let template = INDICATOR_STYLES[config.style];\n  if (config.style === 'custom' && config.customFormat) {\n    // Validate custom format\n    const validation = validateCustomFormat(config.customFormat);\n    if (!validation.valid) {\n      // Fall back to full style if custom format is invalid\n      template = INDICATOR_STYLES.full;\n    } else {\n      template = config.customFormat;\n    }\n  }\n  \n  // Replace placeholders with values or empty strings\n  let result = template\n    .replace('{emoji}', config.showEmoji ? config.emoji : '')\n    .replace('{name}', config.showName ? metadata.name : '')\n    .replace('{version}', config.showVersion && metadata.version ? metadata.version : '')\n    .replace('{author}', config.showAuthor && metadata.author ? metadata.author : '')\n    .replace('{category}', config.showCategory && metadata.category ? metadata.category : '');\n  \n  // Clean up the format string\n  // Remove \"v\" if no version follows it\n  if (!config.showVersion || !metadata.version) {\n    result = result.replace(/\\sv(?=\\s|]|\\)|>|}|$)/, '');\n  }\n  // Remove \"by\" if no author follows it\n  if (!config.showAuthor || !metadata.author) {\n    result = result.replace(/\\sby(?=\\s|]|\\)|>|}|$)/, '');\n  }\n  // Clean up extra spaces\n  result = result.replace(/\\s+/g, ' ').trim();\n  \n  // Apply brackets based on the template format (only if template doesn't already have them)\n  if (result && config.style !== 'custom') {\n    // Check if the template already includes brackets\n    const templateHasBrackets = template.includes('[') || template.includes(']') || \n                               template.includes('(') || template.includes(')') ||\n                               template.includes('{') || template.includes('}') ||\n                               template.includes('<') || template.includes('>');\n    \n    if (!templateHasBrackets) {\n      const brackets = BRACKETS[config.bracketStyle];\n      if (brackets.open || brackets.close) {\n        result = `${brackets.open}${result}${brackets.close}`;\n      }\n    }\n  }\n  \n  // Add separator if we have content\n  if (result) {\n    result += config.separator;\n  }\n  \n  return result;\n}"]}