@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.
139 lines • 16.7 kB
JavaScript
/**
* Persona loading and file management
*/
import * as fs from 'fs/promises';
import * as path from 'path';
import { ensureDirectory, generateUniqueId } from '../utils/filesystem.js';
import { SecureYamlParser } from '../security/secureYamlParser.js';
import { SecurityError } from '../errors/SecurityError.js';
import { logger } from '../utils/logger.js';
export class PersonaLoader {
personasDir;
constructor(personasDir) {
this.personasDir = personasDir;
}
/**
* Load all personas from the personas directory
*/
async loadAll(getCurrentUser) {
// Ensure directory exists
await ensureDirectory(this.personasDir);
const personas = new Map();
try {
const files = await fs.readdir(this.personasDir);
const markdownFiles = files.filter(file => file.endsWith('.md'));
for (const file of markdownFiles) {
try {
const persona = await this.loadPersona(file, getCurrentUser);
if (persona) {
personas.set(file, persona);
logger.debug(`Loaded persona: ${persona.metadata.name} (${persona.unique_id})`);
}
}
catch (error) {
logger.error(`Error loading persona ${file}: ${error}`);
}
}
}
catch (error) {
logger.error(`Error reading personas directory: ${error}`);
}
return personas;
}
/**
* Load a single persona from file
*/
async loadPersona(filename, getCurrentUser) {
try {
const filePath = path.join(this.personasDir, filename);
const fileContent = await fs.readFile(filePath, 'utf-8');
// Use secure YAML parser instead of direct gray-matter
let parsed;
try {
parsed = SecureYamlParser.safeMatter(fileContent);
}
catch (error) {
if (error instanceof SecurityError) {
logger.error(`Security threat detected in persona ${filename}: ${error.message}`);
return null;
}
throw error;
}
const metadata = parsed.data;
const content = parsed.content;
if (!metadata.name) {
metadata.name = path.basename(filename, '.md');
}
// Generate unique ID if not present
let uniqueId = metadata.unique_id;
if (!uniqueId) {
const authorForId = metadata.author || getCurrentUser() || undefined;
uniqueId = generateUniqueId(metadata.name, authorForId);
logger.debug(`Generated unique ID for ${metadata.name}: ${uniqueId}`);
}
// Set default values for metadata fields
this.setDefaultMetadata(metadata);
const persona = {
metadata,
content,
filename,
unique_id: uniqueId,
};
return persona;
}
catch (error) {
logger.error(`Error loading persona ${filename}: ${error}`);
return null;
}
}
/**
* Save a persona to file
*/
async savePersona(persona) {
const filePath = path.join(this.personasDir, persona.filename);
// Use secure YAML stringification
const secureParser = SecureYamlParser.createSecureMatterParser();
const fileContent = secureParser.stringify(persona.content, persona.metadata);
await fs.writeFile(filePath, fileContent, 'utf-8');
}
/**
* Delete a persona file
*/
async deletePersona(filename) {
const filePath = path.join(this.personasDir, filename);
await fs.unlink(filePath);
}
/**
* Check if a persona file exists
*/
async personaExists(filename) {
try {
const filePath = path.join(this.personasDir, filename);
await fs.access(filePath);
return true;
}
catch {
return false;
}
}
/**
* Set default metadata values
*/
setDefaultMetadata(metadata) {
if (!metadata.category)
metadata.category = 'general';
if (!metadata.age_rating)
metadata.age_rating = 'all';
if (!metadata.content_flags)
metadata.content_flags = [];
if (metadata.ai_generated === undefined)
metadata.ai_generated = false;
if (!metadata.generation_method)
metadata.generation_method = 'human';
if (!metadata.price)
metadata.price = 'free';
if (!metadata.license)
metadata.license = 'CC-BY-SA-4.0';
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PersonaLoader.js","sourceRoot":"","sources":["../../src/persona/PersonaLoader.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Persona loading and file management\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport matter from 'gray-matter';\nimport { Persona, PersonaMetadata } from '../types/persona.js';\nimport { ensureDirectory, generateUniqueId } from '../utils/filesystem.js';\nimport { SecureYamlParser } from '../security/secureYamlParser.js';\nimport { SecurityError } from '../errors/SecurityError.js';\nimport { logger } from '../utils/logger.js';\n\nexport class PersonaLoader {\n  private personasDir: string;\n  \n  constructor(personasDir: string) {\n    this.personasDir = personasDir;\n  }\n  \n  /**\n   * Load all personas from the personas directory\n   */\n  async loadAll(getCurrentUser: () => string | null): Promise<Map<string, Persona>> {\n    // Ensure directory exists\n    await ensureDirectory(this.personasDir);\n    \n    const personas = new Map<string, Persona>();\n    \n    try {\n      const files = await fs.readdir(this.personasDir);\n      const markdownFiles = files.filter(file => file.endsWith('.md'));\n      \n      for (const file of markdownFiles) {\n        try {\n          const persona = await this.loadPersona(file, getCurrentUser);\n          if (persona) {\n            personas.set(file, persona);\n            logger.debug(`Loaded persona: ${persona.metadata.name} (${persona.unique_id})`);\n          }\n        } catch (error) {\n          logger.error(`Error loading persona ${file}: ${error}`);\n        }\n      }\n    } catch (error) {\n      logger.error(`Error reading personas directory: ${error}`);\n    }\n    \n    return personas;\n  }\n  \n  /**\n   * Load a single persona from file\n   */\n  async loadPersona(filename: string, getCurrentUser: () => string | null): Promise<Persona | null> {\n    try {\n      const filePath = path.join(this.personasDir, filename);\n      const fileContent = await fs.readFile(filePath, 'utf-8');\n      \n      // Use secure YAML parser instead of direct gray-matter\n      let parsed;\n      try {\n        parsed = SecureYamlParser.safeMatter(fileContent);\n      } catch (error) {\n        if (error instanceof SecurityError) {\n          logger.error(`Security threat detected in persona ${filename}: ${error.message}`);\n          return null;\n        }\n        throw error;\n      }\n      \n      const metadata = parsed.data as PersonaMetadata;\n      const content = parsed.content;\n      \n      if (!metadata.name) {\n        metadata.name = path.basename(filename, '.md');\n      }\n      \n      // Generate unique ID if not present\n      let uniqueId = metadata.unique_id;\n      if (!uniqueId) {\n        const authorForId = metadata.author || getCurrentUser() || undefined;\n        uniqueId = generateUniqueId(metadata.name, authorForId);\n        logger.debug(`Generated unique ID for ${metadata.name}: ${uniqueId}`);\n      }\n      \n      // Set default values for metadata fields\n      this.setDefaultMetadata(metadata);\n      \n      const persona: Persona = {\n        metadata,\n        content,\n        filename,\n        unique_id: uniqueId,\n      };\n      \n      return persona;\n    } catch (error) {\n      logger.error(`Error loading persona ${filename}: ${error}`);\n      return null;\n    }\n  }\n  \n  /**\n   * Save a persona to file\n   */\n  async savePersona(persona: Persona): Promise<void> {\n    const filePath = path.join(this.personasDir, persona.filename);\n    \n    // Use secure YAML stringification\n    const secureParser = SecureYamlParser.createSecureMatterParser();\n    const fileContent = secureParser.stringify(persona.content, persona.metadata);\n    \n    await fs.writeFile(filePath, fileContent, 'utf-8');\n  }\n  \n  /**\n   * Delete a persona file\n   */\n  async deletePersona(filename: string): Promise<void> {\n    const filePath = path.join(this.personasDir, filename);\n    await fs.unlink(filePath);\n  }\n  \n  /**\n   * Check if a persona file exists\n   */\n  async personaExists(filename: string): Promise<boolean> {\n    try {\n      const filePath = path.join(this.personasDir, filename);\n      await fs.access(filePath);\n      return true;\n    } catch {\n      return false;\n    }\n  }\n  \n  /**\n   * Set default metadata values\n   */\n  private setDefaultMetadata(metadata: PersonaMetadata): void {\n    if (!metadata.category) metadata.category = 'general';\n    if (!metadata.age_rating) metadata.age_rating = 'all';\n    if (!metadata.content_flags) metadata.content_flags = [];\n    if (metadata.ai_generated === undefined) metadata.ai_generated = false;\n    if (!metadata.generation_method) metadata.generation_method = 'human';\n    if (!metadata.price) metadata.price = 'free';\n    if (!metadata.license) metadata.license = 'CC-BY-SA-4.0';\n  }\n}"]}