UNPKG

@dollhousemcp/mcp-server

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.

315 lines 40.9 kB
/** * Persona import functionality with validation */ import * as fs from 'fs/promises'; import * as path from 'path'; import matter from 'gray-matter'; import { SecureYamlParser } from '../../security/secureYamlParser.js'; import { ContentValidator } from '../../security/contentValidator.js'; import { validateFilename, validatePath, validateContentSize } from '../../security/InputValidator.js'; import { FileLockManager } from '../../security/fileLockManager.js'; import { generateUniqueId } from '../../utils/filesystem.js'; import { logger } from '../../utils/logger.js'; export class PersonaImporter { personasDir; currentUser; constructor(personasDir, currentUser) { this.personasDir = personasDir; this.currentUser = currentUser; } /** * Import a persona from various sources */ async importPersona(source, existingPersonas, overwrite = false) { try { // Determine source type let personaData = null; // Check if it's a file path if (source.startsWith('/') || source.startsWith('./') || source.endsWith('.md') || source.endsWith('.json')) { personaData = await this.importFromFile(source); } // Check if it's base64 encoded else if (this.isBase64(source)) { personaData = await this.importFromBase64(source); } // Try parsing as JSON directly else { try { const parsed = JSON.parse(source); if (this.isExportBundle(parsed)) { return this.importBundle(parsed, existingPersonas, overwrite); } else if (this.isExportedPersona(parsed)) { personaData = parsed; } } catch { // Not JSON, might be raw markdown return this.importFromMarkdown(source, existingPersonas, overwrite); } } if (!personaData) { return { success: false, message: "Could not parse import source. Please provide a file path, JSON string, or base64 encoded data." }; } // Validate and create persona return await this.createPersonaFromExport(personaData, existingPersonas, overwrite); } catch (error) { logger.error('Import error', error); return { success: false, message: `Import failed: ${error instanceof Error ? error.message : String(error)}` }; } } /** * Import from file path */ async importFromFile(filePath) { try { // Validate path const validatedPath = validatePath(filePath); const content = await fs.readFile(validatedPath, 'utf-8'); if (filePath.endsWith('.json')) { const parsed = JSON.parse(content); if (this.isExportedPersona(parsed)) { return parsed; } else if (this.isExportBundle(parsed)) { throw new Error("This is a bundle file. It will be imported as multiple personas."); } } else if (filePath.endsWith('.md')) { // Parse markdown file const { data, content: mdContent } = matter(content); const filename = path.basename(filePath); return { metadata: data, content: mdContent, filename: filename, exportedAt: new Date().toISOString() }; } } catch (error) { logger.error('File import error', error); throw error; } return null; } /** * Import from base64 string */ async importFromBase64(base64) { try { const json = Buffer.from(base64, 'base64').toString('utf-8'); const parsed = JSON.parse(json); if (this.isExportedPersona(parsed)) { return parsed; } else if (this.isExportBundle(parsed)) { throw new Error("This is a bundle. It will be imported as multiple personas."); } } catch (error) { logger.error('Base64 import error', error); } return null; } /** * Import from raw markdown content */ async importFromMarkdown(content, existingPersonas, overwrite) { try { // Validate content size validateContentSize(content, 100 * 1024); // 100KB limit // Try to parse as markdown with frontmatter const { data, content: mdContent } = matter(content); if (!data.name || !data.description) { return { success: false, message: "Invalid persona format. Must include name and description in YAML frontmatter." }; } const metadata = data; const filename = `${metadata.name.toLowerCase().replace(/\s+/g, '-')}.md`; const exportedPersona = { metadata, content: mdContent, filename, exportedAt: new Date().toISOString(), exportedBy: this.currentUser || undefined }; return await this.createPersonaFromExport(exportedPersona, existingPersonas, overwrite); } catch (error) { return { success: false, message: `Failed to parse markdown: ${error instanceof Error ? error.message : String(error)}` }; } } /** * Import a bundle of personas */ async importBundle(bundle, existingPersonas, overwrite) { const results = { success: true, imported: [], failed: [], conflicts: [] }; for (const personaData of bundle.personas) { const result = await this.createPersonaFromExport(personaData, existingPersonas, overwrite); if (result.success) { results.imported.push(personaData.metadata.name); } else { results.failed.push(`${personaData.metadata.name}: ${result.message}`); if (result.conflicts) { results.conflicts.push(...result.conflicts); } } } return { success: results.failed.length === 0, message: this.formatBundleImportResult(results), conflicts: results.conflicts.length > 0 ? results.conflicts : undefined }; } /** * Create persona from exported data */ async createPersonaFromExport(exportData, existingPersonas, overwrite) { try { // Validate metadata const metadata = await this.validateAndEnrichMetadata(exportData.metadata); // Validate content const validationResult = ContentValidator.validateAndSanitize(exportData.content); if (!validationResult.isValid && validationResult.severity === 'critical') { throw new Error(`Critical security threat detected: ${validationResult.detectedPatterns?.join(', ')}`); } const sanitizedContent = validationResult.sanitizedContent || exportData.content; // Generate safe filename let filename = validateFilename(exportData.filename || `${metadata.name.toLowerCase().replace(/\s+/g, '-')}.md`); // Check for conflicts const conflicts = this.findConflicts(metadata.name, filename, existingPersonas); if (conflicts.length > 0 && !overwrite) { return { success: false, message: `Persona already exists: ${conflicts.join(', ')}. Use overwrite=true to replace.`, conflicts }; } // Create the persona file const personaPath = path.join(this.personasDir, filename); const fileContent = matter.stringify(sanitizedContent, metadata); // Use file locking to prevent race conditions await FileLockManager.withLock(`persona:${metadata.name}`, async () => { await FileLockManager.atomicWriteFile(personaPath, fileContent); }); // Create persona object const persona = { metadata, content: sanitizedContent, filename, unique_id: metadata.unique_id }; return { success: true, message: `Successfully imported "${metadata.name}"`, persona, filename }; } catch (error) { logger.error('Create persona error', error); return { success: false, message: `Failed to create persona: ${error instanceof Error ? error.message : String(error)}` }; } } /** * Validate and enrich metadata */ async validateAndEnrichMetadata(metadata) { // Ensure required fields if (!metadata.name || !metadata.description) { throw new Error("Missing required fields: name and description"); } // Generate unique_id if missing if (!metadata.unique_id) { metadata.unique_id = generateUniqueId(metadata.name, this.currentUser || 'imported'); } // Set defaults metadata.version = metadata.version || '1.0'; metadata.author = metadata.author || this.currentUser || 'imported'; metadata.category = metadata.category || 'custom'; metadata.created_date = metadata.created_date || new Date().toISOString(); // Validate with YAML parser for security const validated = await SecureYamlParser.parse(matter.stringify('', metadata)); return validated.data; } /** * Find conflicts with existing personas */ findConflicts(name, filename, existingPersonas) { const conflicts = []; for (const [key, persona] of existingPersonas) { if (persona.metadata.name === name || persona.filename === filename) { conflicts.push(key); } } return conflicts; } /** * Check if string is base64 */ isBase64(str) { // Check length is multiple of 4 if (str.length % 4 !== 0) return false; // SECURITY FIX: Ensure base64 string is not empty // Previously: /^[A-Za-z0-9+/]*={0,2}$/ allowed empty strings // Now: Require at least one character before optional padding return /^[A-Za-z0-9+/]+={0,2}$/.test(str); } /** * Type guard for ExportedPersona */ isExportedPersona(obj) { return obj && typeof obj.metadata === 'object' && typeof obj.content === 'string' && typeof obj.filename === 'string'; } /** * Type guard for ExportBundle */ isExportBundle(obj) { return obj && typeof obj.version === 'string' && Array.isArray(obj.personas) && typeof obj.personaCount === 'number'; } /** * Format bundle import results */ formatBundleImportResult(results) { let message = `Bundle Import Summary:\n`; message += `✅ Successfully imported: ${results.imported.length} personas\n`; if (results.imported.length > 0) { message += `Imported:\n${results.imported.map((n) => ` - ${n}`).join('\n')}\n`; } if (results.failed.length > 0) { message += `\n❌ Failed: ${results.failed.length} personas\n`; message += `Errors:\n${results.failed.map((e) => ` - ${e}`).join('\n')}`; } return message; } } //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiUGVyc29uYUltcG9ydGVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3BlcnNvbmEvZXhwb3J0LWltcG9ydC9QZXJzb25hSW1wb3J0ZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxPQUFPLEtBQUssRUFBRSxNQUFNLGFBQWEsQ0FBQztBQUNsQyxPQUFPLEtBQUssSUFBSSxNQUFNLE1BQU0sQ0FBQztBQUM3QixPQUFPLE1BQU0sTUFBTSxhQUFhLENBQUM7QUFHakMsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sb0NBQW9DLENBQUM7QUFDdEUsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sb0NBQW9DLENBQUM7QUFDdEUsT0FBTyxFQUFFLGdCQUFnQixFQUFFLFlBQVksRUFBaUIsbUJBQW1CLEVBQUUsTUFBTSxrQ0FBa0MsQ0FBQztBQUN0SCxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sbUNBQW1DLENBQUM7QUFDcEUsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sMkJBQTJCLENBQUM7QUFDN0QsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLHVCQUF1QixDQUFDO0FBVS9DLE1BQU0sT0FBTyxlQUFlO0lBRWhCO0lBQ0E7SUFGVixZQUNVLFdBQW1CLEVBQ25CLFdBQTBCO1FBRDFCLGdCQUFXLEdBQVgsV0FBVyxDQUFRO1FBQ25CLGdCQUFXLEdBQVgsV0FBVyxDQUFlO0lBQ2pDLENBQUM7SUFFSjs7T0FFRztJQUNILEtBQUssQ0FBQyxhQUFhLENBQUMsTUFBYyxFQUFFLGdCQUFzQyxFQUFFLFNBQVMsR0FBRyxLQUFLO1FBQzNGLElBQUksQ0FBQztZQUNILHdCQUF3QjtZQUN4QixJQUFJLFdBQVcsR0FBMkIsSUFBSSxDQUFDO1lBRS9DLDRCQUE0QjtZQUM1QixJQUFJLE1BQU0sQ0FBQyxVQUFVLENBQUMsR0FBRyxDQUFDLElBQUksTUFBTSxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsSUFB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YUFBYSxDQUFDO1FBRTVFLElBQUksT0FBTyxDQUFDLFFBQVEsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxFQUFFLENBQUM7WUFDaEMsT0FBTyxJQUFJLGNBQWMsT0FBTyxDQUFDLFFBQVEsQ0FBQyxHQUFHLENBQUMsQ0FBQyxDQUFTLEVBQUUsRUFBRSxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUMsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQztRQUMxRixDQUFDO1FBRUQsSUFBSSxPQUFPLENBQUMsTUFBTSxDQUFDLE1BQU0sR0FBRyxDQUFDLEVBQUUsQ0FBQztZQUM5QixPQUFPLElBQUksZUFBZSxPQUFPLENBQUMsTUFBTSxDQUFDLE1BQU0sYUFBYSxDQUFDO1lBQzdELE9BQU8sSUFBSSxZQUFZLE9BQU8sQ0FBQyxNQUFNLENBQUMsR0FBRyxDQUFDLENBQUMsQ0FBUyxFQUFFLEVBQUUsQ0FBQyxPQUFPLENBQUMsRUFBRSxDQUFDLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUM7UUFDcEYsQ0FBQztRQUVELE9BQU8sT0FBTyxDQUFDO0lBQ2pCLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogUGVyc29uYSBpbXBvcnQgZnVuY3Rpb25hbGl0eSB3aXRoIHZhbGlkYXRpb25cbiAqL1xuXG5pbXBvcnQgKiBhcyBmcyBmcm9tICdmcy9wcm9taXNlcyc7XG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gJ3BhdGgnO1xuaW1wb3J0IG1hdHRlciBmcm9tICdncmF5LW1hdHRlcic7XG5pbXBvcnQgeyBQZXJzb25hLCBQZXJzb25hTWV0YWRhdGEgfSBmcm9tICcuLi8uLi90eXBlcy9wZXJzb25hLmpzJztcbmltcG9ydCB7IEV4cG9ydGVkUGVyc29uYSwgRXhwb3J0QnVuZGxlIH0gZnJvbSAnLi9QZXJzb25hRXhwb3J0ZXIuanMnO1xuaW1wb3J0IHsgU2VjdXJlWWFtbFBhcnNlciB9IGZyb20gJy4uLy4uL3NlY3VyaXR5L3NlY3VyZVlhbWxQYXJzZXIuanMnO1xuaW1wb3J0IHsgQ29udGVudFZhbGlkYXRvciB9IGZyb20gJy4uLy4uL3NlY3VyaXR5L2NvbnRlbnRWYWxpZGF0b3IuanMnO1xuaW1wb3J0IHsgdmFsaWRhdGVGaWxlbmFtZSwgdmFsaWRhdGVQYXRoLCBzYW5pdGl6ZUlucHV0LCB2YWxpZGF0ZUNvbnRlbnRTaXplIH0gZnJvbSAnLi4vLi4vc2VjdXJpdHkvSW5wdXRWYWxpZGF0b3IuanMnO1xuaW1wb3J0IHsgRmlsZUxvY2tNYW5hZ2VyIH0gZnJvbSAnLi4vLi4vc2VjdXJpdHkvZmlsZUxvY2tNYW5hZ2VyLmpzJztcbmltcG9ydCB7IGdlbmVyYXRlVW5pcXVlSWQgfSBmcm9tICcuLi8uLi91dGlscy9maWxlc3lzdGVtLmpzJztcbmltcG9ydCB7IGxvZ2dlciB9IGZyb20gJy4uLy4uL3V0aWxzL2xvZ2dlci5qcyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgSW1wb3J0UmVzdWx0IHtcbiAgc3VjY2VzczogYm9vbGVhbjtcbiAgbWVzc2FnZTogc3RyaW5nO1xuICBwZXJzb25hPzogUGVyc29uYTtcbiAgZmlsZW5hbWU/OiBzdHJpbmc7XG4gIGNvbmZsaWN0cz86IHN0cmluZ1tdO1xufVxuXG5leHBvcnQgY2xhc3MgUGVyc29uYUltcG9ydGVyIHtcbiAgY29uc3RydWN0b3IoXG4gICAgcHJpdmF0ZSBwZXJzb25hc0Rpcjogc3RyaW5nLFxuICAgIHByaXZhdGUgY3VycmVudFVzZXI6IHN0cmluZyB8IG51bGxcbiAgKSB7fVxuXG4gIC8qKlxuICAgKiBJbXBvcnQgYSBwZXJzb25hIGZyb20gdmFyaW91cyBzb3VyY2VzXG4gICAqL1xuICBhc3luYyBpbXBvcnRQZXJzb25hKHNvdXJjZTogc3RyaW5nLCBleGlzdGluZ1BlcnNvbmFzOiBNYXA8c3RyaW5nLCBQZXJzb25hPiwgb3ZlcndyaXRlID0gZmFsc2UpOiBQcm9taXNlPEltcG9ydFJlc3VsdD4ge1xuICAgIHRyeSB7XG4gICAgICAvLyBEZXRlcm1pbmUgc291cmNlIHR5cGVcbiAgICAgIGxldCBwZXJzb25hRGF0YTogRXhwb3J0ZWRQZXJzb25hIHwgbnVsbCA9IG51bGw7XG5cbiAgICAgIC8vIENoZWNrIGlmIGl0J3MgYSBmaWxlIHBhdGhcbiAgICAgIGlmIChzb3VyY2Uuc3RhcnRzV2l0aCgnLycpIHx8IHNvdXJjZS5zdGFydHNXaXRoKCcuLycpIHx8IHNvdXJjZS5lbmRzV2l0aCgnLm1kJykgfHwgc291cmNlLmVuZHNXaXRoKCcuanNvbicpKSB7XG4gICAgICAgIHBlcnNvbmFEYXRhID0gYXdhaXQgdGhpcy5pbXBvcnRGcm9tRmlsZShzb3VyY2UpO1xuICAgICAgfSBcbiAgICAgIC8vIENoZWNrIGlmIGl0J3MgYmFzZTY0IGVuY29kZWRcbiAgICAgIGVsc2UgaWYgKHRoaXMuaXNCYXNlNjQoc291cmNlKSkge1xuICAgICAgICBwZXJzb25hRGF0YSA9IGF3YWl0IHRoaXMuaW1wb3J0RnJvbUJhc2U2NChzb3VyY2UpO1xuICAgICAgfVxuICAgICAgLy8gVHJ5IHBhcnNpbmcgYXMgSlNPTiBkaXJlY3RseVxuICAgICAgZWxzZSB7XG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgY29uc3QgcGFyc2VkID0gSlNPTi5wYXJzZShzb3VyY2UpO1xuICAgICAgICAgIGlmICh0aGlzLmlzRXhwb3J0QnVuZGxlKHBhcnNlZCkpIHtcbiAgICAgICAgICAgIHJldHVybiB0aGlzLmltcG9ydEJ1bmRsZShwYXJzZWQsIGV4aXN0aW5nUGVyc29uYXMsIG92ZXJ3cml0ZSk7XG4gICAgICAgICAgfSBlbHNlIGlmICh0aGlzLmlzRXhwb3J0ZWRQZXJzb25hKHBhcnNlZCkpIHtcbiAgICAgICAgICAgIHBlcnNvbmFEYXRhID0gcGFyc2VkO1xuICAgICAgICAgIH1cbiAgICAgICAgfSBjYXRjaCB7XG4gICAgICAgICAgLy8gTm90IEpTT04sIG1pZ2h0IGJlIHJhdyBtYXJrZG93blxuICAgICAgICAgIHJldHVybiB0aGlzLmltcG9ydEZyb21NYXJrZG93bihzb3VyY2UsIGV4aXN0aW5nUGVyc29uYXMsIG92ZXJ3cml0ZSk7XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgaWYgKCFwZXJzb25hRGF0YSkge1xuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIHN1Y2Nlc3M6IGZhbHNlLFxuICAgICAgICAgIG1lc3NhZ2U6IFwiQ291bGQgbm90IHBhcnNlIGltcG9ydCBzb3VyY2UuIFBsZWFzZSBwcm92aWRlIGEgZmlsZSBwYXRoLCBKU09OIHN0cmluZywgb3IgYmFzZTY0IGVuY29kZWQgZGF0YS5cIlxuICAgICAgICB9O1xuICAgICAgfVxuXG4gICAgICAvLyBWYWxpZGF0ZSBhbmQgY3JlYXRlIHBlcnNvbmFcbiAgICAgIHJldHVybiBhd2FpdCB0aGlzLmNyZWF0ZVBlcnNvbmFGcm9tRXhwb3J0KHBlcnNvbmFEYXRhLCBleGlzdGluZ1BlcnNvbmFzLCBvdmVyd3JpdGUpO1xuXG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIGxvZ2dlci5lcnJvcignSW1wb3J0IGVycm9yJywgZXJyb3IpO1xuICAgICAgcmV0dXJuIHtcbiAgICAgICAgc3VjY2VzczogZmFsc2UsXG4gICAgICAgIG1lc3NhZ2U6IGBJbXBvcnQgZmFpbGVkOiAke2Vycm9yIGluc3RhbmNlb2YgRXJyb3IgPyBlcnJvci5tZXNzYWdlIDogU3RyaW5nKGVycm9yKX1gXG4gICAgICB9O1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBJbXBvcnQgZnJvbSBmaWxlIHBhdGhcbiAgICovXG4gIHByaXZhdGUgYXN5bmMgaW1wb3J0RnJvbUZpbGUoZmlsZVBhdGg6IHN0cmluZyk6IFByb21pc2U8RXhwb3J0ZWRQZXJzb25hIHwgbnVsbD4ge1xuICAgIHRyeSB7XG4gICAgICAvLyBWYWxpZGF0ZSBwYXRoXG4gICAgICBjb25zdCB2YWxpZGF0ZWRQYXRoID0gdmFsaWRhdGVQYXRoKGZpbGVQYXRoKTtcbiAgICAgIGNvbnN0IGNvbnRlbnQgPSBhd2FpdCBmcy5yZWFkRmlsZSh2YWxpZGF0ZWRQYXRoLCAndXRmLTgnKTtcblxuICAgICAgaWYgKGZpbGVQYXRoLmVuZHNXaXRoKCcuanNvbicpKSB7XG4gICAgICAgIGNvbnN0IHBhcnNlZCA9IEpTT04ucGFyc2UoY29udGVudCk7XG4gICAgICAgIGlmICh0aGlzLmlzRXhwb3J0ZWRQZXJzb25hKHBhcnNlZCkpIHtcbiAgICAgICAgICByZXR1cm4gcGFyc2VkO1xuICAgICAgICB9IGVsc2UgaWYgKHRoaXMuaXNFeHBvcnRCdW5kbGUocGFyc2VkKSkge1xuICAgICAgICAgIHRocm93IG5ldyBFcnJvcihcIlRoaXMgaXMgYSBidW5kbGUgZmlsZS4gSXQgd2lsbCBiZSBpbXBvcnRlZCBhcyBtdWx0aXBsZSBwZXJzb25hcy5cIik7XG4gICAgICAgIH1cbiAgICAgIH0gZWxzZSBpZiAoZmlsZVBhdGguZW5kc1dpdGgoJy5tZCcpKSB7XG4gICAgICAgIC8vIFBhcnNlIG1hcmtkb3duIGZpbGVcbiAgICAgICAgY29uc3QgeyBkYXRhLCBjb250ZW50OiBtZENvbnRlbnQgfSA9IG1hdHRlcihjb250ZW50KTtcbiAgICAgICAgY29uc3QgZmlsZW5hbWUgPSBwYXRoLmJhc2VuYW1lKGZpbGVQYXRoKTtcbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICBtZXRhZGF0YTogZGF0YSBhcyBQZXJzb25hTWV0YWRhdGEsXG4gICAgICAgICAgY29udGVudDogbWRDb250ZW50LFxuICAgICAgICAgIGZpbGVuYW1lOiBmaWxlbmFtZSxcbiAgICAgICAgICBleHBvcnRlZEF0OiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKClcbiAgICAgICAgfTtcbiAgICAgIH1cbiAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgbG9nZ2VyLmVycm9yKCdGaWxlIGltcG9ydCBlcnJvcicsIGVycm9yKTtcbiAgICAgIHRocm93IGVycm9yO1xuICAgIH1cbiAgICByZXR1cm4gbnVsbDtcbiAgfVxuXG4gIC8qKlxuICAgKiBJbXBvcnQgZnJvbSBiYXNlNjQgc3RyaW5nXG4gICAqL1xuICBwcml2YXRlIGFzeW5jIGltcG9ydEZyb21CYXNlNjQoYmFzZTY0OiBzdHJpbmcpOiBQcm9taXNlPEV4cG9ydGVkUGVyc29uYSB8IG51bGw+IHtcbiAgICB0cnkge1xuICAgICAgY29uc3QganNvbiA9IEJ1ZmZlci5mcm9tKGJhc2U2NCwgJ2Jhc2U2NCcpLnRvU3RyaW5nKCd1dGYtOCcpO1xuICAgICAgY29uc3QgcGFyc2VkID0gSlNPTi5wYXJzZShqc29uKTtcbiAgICAgIFxuICAgICAgaWYgKHRoaXMuaXNFeHBvcnRlZFBlcnNvbmEocGFyc2VkKSkge1xuICAgICAgICByZXR1cm4gcGFyc2VkO1xuICAgICAgfSBlbHNlIGlmICh0aGlzLmlzRXhwb3J0QnVuZGxlKHBhcnNlZCkpIHtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKFwiVGhpcyBpcyBhIGJ1bmRsZS4gSXQgd2lsbCBiZSBpbXBvcnRlZCBhcyBtdWx0aXBsZSBwZXJzb25hcy5cIik7XG4gICAgICB9XG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIGxvZ2dlci5lcnJvcignQmFzZTY0IGltcG9ydCBlcnJvcicsIGVycm9yKTtcbiAgICB9XG4gICAgcmV0dXJuIG51bGw7XG4gIH1cblxuICAvKipcbiAgICogSW1wb3J0IGZyb20gcmF3IG1hcmtkb3duIGNvbnRlbnRcbiAgICovXG4gIHByaXZhdGUgYXN5bmMgaW1wb3J0RnJvbU1hcmtkb3duKGNvbnRlbnQ6IHN0cmluZywgZXhpc3RpbmdQZXJzb25hczogTWFwPHN0cmluZywgUGVyc29uYT4sIG92ZXJ3cml0ZTogYm9vbGVhbik6IFByb21pc2U8SW1wb3J0UmVzdWx0PiB7XG4gICAgdHJ5IHtcbiAgICAgIC8vIFZhbGlkYXRlIGNvbnRlbnQgc2l6ZVxuICAgICAgdmFsaWRhdGVDb250ZW50U2l6ZShjb250ZW50LCAxMDAgKiAxMDI0KTsgLy8gMTAwS0IgbGltaXRcblxuICAgICAgLy8gVHJ5IHRvIHBhcnNlIGFzIG1hcmtkb3duIHdpdGggZnJvbnRtYXR0ZXJcbiAgICAgIGNvbnN0IHsgZGF0YSwgY29udGVudDogbWRDb250ZW50IH0gPSBtYXR0ZXIoY29udGVudCk7XG4gICAgICBcbiAgICAgIGlmICghZGF0YS5uYW1lIHx8ICFkYXRhLmRlc2NyaXB0aW9uKSB7XG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgc3VjY2VzczogZmFsc2UsXG4gICAgICAgICAgbWVzc2FnZTogXCJJbnZhbGlkIHBlcnNvbmEgZm9ybWF0LiBNdXN0IGluY2x1ZGUgbmFtZSBhbmQgZGVzY3JpcHRpb24gaW4gWUFNTCBmcm9udG1hdHRlci5cIlxuICAgICAgICB9O1xuICAgICAgfVxuXG4gICAgICBjb25zdCBtZXRhZGF0YSA9IGRhdGEgYXMgUGVyc29uYU1ldGFkYXRhO1xuICAgICAgY29uc3QgZmlsZW5hbWUgPSBgJHttZXRhZGF0YS5uYW1lLnRvTG93ZXJDYXNlKCkucmVwbGFjZSgvXFxzKy9nLCAnLScpfS5tZGA7XG5cbiAgICAgIGNvbnN0IGV4cG9ydGVkUGVyc29uYTogRXhwb3J0ZWRQZXJzb25hID0ge1xuICAgICAgICBtZXRhZGF0YSxcbiAgICAgICAgY29udGVudDogbWRDb250ZW50LFxuICAgICAgICBmaWxlbmFtZSxcbiAgICAgICAgZXhwb3J0ZWRBdDogbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgICAgICBleHBvcnRlZEJ5OiB0aGlzLmN1cnJlbnRVc2VyIHx8IHVuZGVmaW5lZFxuICAgICAgfTtcblxuICAgICAgcmV0dXJuIGF3YWl0IHRoaXMuY3JlYXRlUGVyc29uYUZyb21FeHBvcnQoZXhwb3J0ZWRQZXJzb25hLCBleGlzdGluZ1BlcnNvbmFzLCBvdmVyd3JpdGUpO1xuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICByZXR1cm4ge1xuICAgICAgICBzdWNjZXNzOiBmYWxzZSxcbiAgICAgICAgbWVzc2FnZTogYEZhaWxlZCB0byBwYXJzZSBtYXJrZG93bjogJHtlcnJvciBpbnN0YW5jZW9mIEVycm9yID8gZXJyb3IubWVzc2FnZSA6IFN0cmluZyhlcnJvcil9YFxuICAgICAgfTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogSW1wb3J0IGEgYnVuZGxlIG9mIHBlcnNvbmFzXG4gICAqL1xuICBwcml2YXRlIGFzeW5jIGltcG9ydEJ1bmRsZShidW5kbGU6IEV4cG9ydEJ1bmRsZSwgZXhpc3RpbmdQZXJzb25hczogTWFwPHN0cmluZywgUGVyc29uYT4sIG92ZXJ3cml0ZTogYm9vbGVhbik6IFByb21pc2U8SW1wb3J0UmVzdWx0PiB7XG4gICAgY29uc3QgcmVzdWx0cyA9IHtcbiAgICAgIHN1Y2Nlc3M6IHRydWUsXG4gICAgICBpbXBvcnRlZDogW10gYXMgc3RyaW5nW10sXG4gICAgICBmYWlsZWQ6IFtdIGFzIHN0cmluZ1tdLFxuICAgICAgY29uZmxpY3RzOiBbXSBhcyBzdHJpbmdbXVxuICAgIH07XG5cbiAgICBmb3IgKGNvbnN0IHBlcnNvbmFEYXRhIG9mIGJ1bmRsZS5wZXJzb25hcykge1xuICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgdGhpcy5jcmVhdGVQZXJzb25hRnJvbUV4cG9ydChwZXJzb25hRGF0YSwgZXhpc3RpbmdQZXJzb25hcywgb3ZlcndyaXRlKTtcbiAgICAgIFxuICAgICAgaWYgKHJlc3VsdC5zdWNjZXNzKSB7XG4gICAgICAgIHJlc3VsdHMuaW1wb3J0ZWQucHVzaChwZXJzb25hRGF0YS5tZXRhZGF0YS5uYW1lKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHJlc3VsdHMuZmFpbGVkLnB1c2goYCR7cGVyc29uYURhdGEubWV0YWRhdGEubmFtZX06ICR7cmVzdWx0Lm1lc3NhZ2V9YCk7XG4gICAgICAgIGlmIChyZXN1bHQuY29uZmxpY3RzKSB7XG4gICAgICAgICAgcmVzdWx0cy5jb25mbGljdHMucHVzaCguLi5yZXN1bHQuY29uZmxpY3RzKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cblxuICAgIHJldHVybiB7XG4gICAgICBzdWNjZXNzOiByZXN1bHRzLmZhaWxlZC5sZW5ndGggPT09IDAsXG4gICAgICBtZXNzYWdlOiB0aGlzLmZvcm1hdEJ1bmRsZUltcG9ydFJlc3VsdChyZXN1bHRzKSxcbiAgICAgIGNvbmZsaWN0czogcmVzdWx0cy5jb25mbGljdHMubGVuZ3RoID4gMCA/IHJlc3VsdHMuY29uZmxpY3RzIDogdW5kZWZpbmVkXG4gICAgfTtcbiAgfVxuXG4gIC8qKlxuICAgKiBDcmVhdGUgcGVyc29uYSBmcm9tIGV4cG9ydGVkIGRhdGFcbiAgICovXG4gIHByaXZhdGUgYXN5bmMgY3JlYXRlUGVyc29uYUZyb21FeHBvcnQoXG4gICAgZXhwb3J0RGF0YTogRXhwb3J0ZWRQZXJzb25hLCBcbiAgICBleGlzdGluZ1BlcnNvbmFzOiBNYXA8c3RyaW5nLCBQZXJzb25hPiwgXG4gICAgb3ZlcndyaXRlOiBib29sZWFuXG4gICk6IFByb21pc2U8SW1wb3J0UmVzdWx0PiB7XG4gICAgdHJ5IHtcbiAgICAgIC8vIFZhbGlkYXRlIG1ldGFkYXRhXG4gICAgICBjb25zdCBtZXRhZGF0YSA9IGF3YWl0IHRoaXMudmFsaWRhdGVBbmRFbnJpY2hNZXRhZGF0YShleHBvcnREYXRhLm1ldGFkYXRhKTtcbiAgICAgIFxuICAgICAgLy8gVmFsaWRhdGUgY29udGVudFxuICAgICAgY29uc3QgdmFsaWRhdGlvblJlc3VsdCA9IENvbnRlbnRWYWxpZGF0b3IudmFsaWRhdGVBbmRTYW5pdGl6ZShleHBvcnREYXRhLmNvbnRlbnQpO1xuICAgICAgaWYgKCF2YWxpZGF0aW9uUmVzdWx0LmlzVmFsaWQgJiYgdmFsaWRhdGlvblJlc3VsdC5zZXZlcml0eSA9PT0gJ2NyaXRpY2FsJykge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYENyaXRpY2FsIHNlY3VyaXR5IHRocmVhdCBkZXRlY3RlZDogJHt2YWxpZGF0aW9uUmVzdWx0LmRldGVjdGVkUGF0dGVybnM/LmpvaW4oJywgJyl9YCk7XG4gICAgICB9XG4gICAgICBjb25zdCBzYW5pdGl6ZWRDb250ZW50ID0gdmFsaWRhdGlvblJlc3VsdC5zYW5pdGl6ZWRDb250ZW50IHx8IGV4cG9ydERhdGEuY29udGVudDtcblxuICAgICAgLy8gR2VuZXJhdGUgc2FmZSBmaWxlbmFtZVxuICAgICAgbGV0IGZpbGVuYW1lID0gdmFsaWRhdGVGaWxlbmFtZShleHBvcnREYXRhLmZpbGVuYW1lIHx8IGAke21ldGFkYXRhLm5hbWUudG9Mb3dlckNhc2UoKS5yZXBsYWNlKC9cXHMrL2csICctJyl9Lm1kYCk7XG4gICAgICBcbiAgICAgIC8vIENoZWNrIGZvciBjb25mbGljdHNcbiAgICAgIGNvbnN0IGNvbmZsaWN0cyA9IHRoaXMuZmluZENvbmZsaWN0cyhtZXRhZGF0YS5uYW1lLCBmaWxlbmFtZSwgZXhpc3RpbmdQZXJzb25hcyk7XG4gICAgICBpZiAoY29uZmxpY3RzLmxlbmd0aCA+IDAgJiYgIW92ZXJ3cml0ZSkge1xuICAgICAgICByZXR1cm4ge1xuICAgICAgICAgIHN1Y2Nlc3M6IGZhbHNlLFxuICAgICAgICAgIG1lc3NhZ2U6IGBQZXJzb25hIGFscmVhZHkgZXhpc3RzOiAke2NvbmZsaWN0cy5qb2luKCcsICcpfS4gVXNlIG92ZXJ3cml0ZT10cnVlIHRvIHJlcGxhY2UuYCxcbiAgICAgICAgICBjb25mbGljdHNcbiAgICAgICAgfTtcbiAgICAgIH1cblxuICAgICAgLy8gQ3JlYXRlIHRoZSBwZXJzb25hIGZpbGVcbiAgICAgIGNvbnN0IHBlcnNvbmFQYXRoID0gcGF0aC5qb2luKHRoaXMucGVyc29uYXNEaXIsIGZpbGVuYW1lKTtcbiAgICAgIGNvbnN0IGZpbGVDb250ZW50ID0gbWF0dGVyLnN0cmluZ2lmeShzYW5pdGl6ZWRDb250ZW50LCBtZXRhZGF0YSk7XG4gICAgICBcbiAgICAgIC8vIFVzZSBmaWxlIGxvY2tpbmcgdG8gcHJldmVudCByYWNlIGNvbmRpdGlvbnNcbiAgICAgIGF3YWl0IEZpbGVMb2NrTWFuYWdlci53aXRoTG9jayhgcGVyc29uYToke21ldGFkYXRhLm5hbWV9YCwgYXN5bmMgKCkgPT4ge1xuICAgICAgICBhd2FpdCBGaWxlTG9ja01hbmFnZXIuYXRvbWljV3JpdGVGaWxlKHBlcnNvbmFQYXRoLCBmaWxlQ29udGVudCk7XG4gICAgICB9KTtcblxuICAgICAgLy8gQ3JlYXRlIHBlcnNvbmEgb2JqZWN0XG4gICAgICBjb25zdCBwZXJzb25hOiBQZXJzb25hID0ge1xuICAgICAgICBtZXRhZGF0YSxcbiAgICAgICAgY29udGVudDogc2FuaXRpemVkQ29udGVudCxcbiAgICAgICAgZmlsZW5hbWUsXG4gICAgICAgIHVuaXF1ZV9pZDogbWV0YWRhdGEudW5pcXVlX2lkIVxuICAgICAgfTtcblxuICAgICAgcmV0dXJuIHtcbiAgICAgICAgc3VjY2VzczogdHJ1ZSxcbiAgICAgICAgbWVzc2FnZTogYFN1Y2Nlc3NmdWxseSBpbXBvcnRlZCBcIiR7bWV0YWRhdGEubmFtZX1cImAsXG4gICAgICAgIHBlcnNvbmEsXG4gICAgICAgIGZpbGVuYW1lXG4gICAgICB9O1xuXG4gICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIGxvZ2dlci5lcnJvcignQ3JlYXRlIHBlcnNvbmEgZXJyb3InLCBlcnJvcik7XG4gICAgICByZXR1cm4ge1xuICAgICAgICBzdWNjZXNzOiBmYWxzZSxcbiAgICAgICAgbWVzc2FnZTogYEZhaWxlZCB0byBjcmVhdGUgcGVyc29uYTogJHtlcnJvciBpbnN0YW5jZW9mIEVycm9yID8gZXJyb3IubWVzc2FnZSA6IFN0cmluZyhlcnJvcil9YFxuICAgICAgfTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogVmFsaWRhdGUgYW5kIGVucmljaCBtZXRhZGF0YVxuICAgKi9cbiAgcHJpdmF0ZSBhc3luYyB2YWxpZGF0ZUFuZEVucmljaE1ldGFkYXRhKG1ldGFkYXRhOiBhbnkpOiBQcm9taXNlPFBlcnNvbmFNZXRhZGF0YT4ge1xuICAgIC8vIEVuc3VyZSByZXF1aXJlZCBmaWVsZHNcbiAgICBpZiAoIW1ldGFkYXRhLm5hbWUgfHwgIW1ldGFkYXRhLmRlc2NyaXB0aW9uKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXCJNaXNzaW5nIHJlcXVpcmVkIGZpZWxkczogbmFtZSBhbmQgZGVzY3JpcHRpb25cIik7XG4gICAgfVxuXG4gICAgLy8gR2VuZXJhdGUgdW5pcXVlX2lkIGlmIG1pc3NpbmdcbiAgICBpZiAoIW1ldGFkYXRhLnVuaXF1ZV9pZCkge1xuICAgICAgbWV0YWRhdGEudW5pcXVlX2lkID0gZ2VuZXJhdGVVbmlxdWVJZChtZXRhZGF0YS5uYW1lLCB0aGlzLmN1cnJlbnRVc2VyIHx8ICdpbXBvcnRlZCcpO1xuICAgIH1cblxuICAgIC8vIFNldCBkZWZhdWx0c1xuICAgIG1ldGFkYXRhLnZlcnNpb24gPSBtZXRhZGF0YS52ZXJzaW9uIHx8ICcxLjAnO1xuICAgIG1ldGFkYXRhLmF1dGhvciA9IG1ldGFkYXRhLmF1dGhvciB8fCB0aGlzLmN1cnJlbnRVc2VyIHx8ICdpbXBvcnRlZCc7XG4gICAgbWV0YWRhdGEuY2F0ZWdvcnkgPSBtZXRhZGF0YS5jYXRlZ29yeSB8fCAnY3VzdG9tJztcbiAgICBtZXRhZGF0YS5jcmVhdGVkX2RhdGUgPSBtZXRhZGF0YS5jcmVhdGVkX2RhdGUgfHwgbmV3IERhdGUoKS50b0lTT1N0cmluZygpO1xuXG4gICAgLy8gVmFsaWRhdGUgd2l0aCBZQU1MIHBhcnNlciBmb3Igc2VjdXJpdHlcbiAgICBjb25zdCB2YWxpZGF0ZWQgPSBhd2FpdCBTZWN1cmVZYW1sUGFyc2VyLnBhcnNlKG1hdHRlci5zdHJpbmdpZnkoJycsIG1ldGFkYXRhKSk7XG5cbiAgICByZXR1cm4gdmFsaWRhdGVkLmRhdGEgYXMgUGVyc29uYU1ldGFkYXRhO1xuICB9XG5cbiAgLyoqXG4gICAqIEZpbmQgY29uZmxpY3RzIHdpdGggZXhpc3RpbmcgcGVyc29uYXNcbiAgICovXG4gIHByaXZhdGUgZmluZENvbmZsaWN0cyhuYW1lOiBzdHJpbmcsIGZpbGVuYW1lOiBzdHJpbmcsIGV4aXN0aW5nUGVyc29uYXM6IE1hcDxzdHJpbmcsIFBlcnNvbmE+KTogc3RyaW5nW10ge1xuICAgIGNvbnN0IGNvbmZsaWN0czogc3RyaW5nW10gPSBbXTtcblxuICAgIGZvciAoY29uc3QgW2tleSwgcGVyc29uYV0gb2YgZXhpc3RpbmdQZXJzb25hcykge1xuICAgICAgaWYgKHBlcnNvbmEubWV0YWRhdGEubmFtZSA9PT0gbmFtZSB8fCBwZXJzb25hLmZpbGVuYW1lID09PSBmaWxlbmFtZSkge1xuICAgICAgICBjb25mbGljdHMucHVzaChrZXkpO1xuICAgICAgfVxuICAgIH1cblxuICAgIHJldHVybiBjb25mbGljdHM7XG4gIH1cblxuICAvKipcbiAgICogQ2hlY2sgaWYgc3RyaW5nIGlzIGJhc2U2NFxuICAgKi9cbiAgcHJpdmF0ZSBpc0Jhc2U2NChzdHI6IHN0cmluZyk6IGJvb2xlYW4ge1xuICAgIC8vIENoZWNrIGxlbmd0aCBpcyBtdWx0aXBsZSBvZiA0XG4gICAgaWYgKHN0ci5sZW5ndGggJSA0ICE9PSAwKSByZXR1cm4gZmFsc2U7XG4gICAgXG4gICAgLy8gU0VDVVJJVFkgRklYOiBFbnN1cmUgYmFzZTY0IHN0cmluZyBpcyBub3QgZW1wdHlcbiAgICAvLyBQcmV2aW91c2x5OiAvXltBLVphLXowLTkrL10qPXswLDJ9JC8gYWxsb3dlZCBlbXB0eSBzdHJpbmdzXG4gICAgLy8gTm93OiBSZXF1aXJlIGF0IGxlYXN0IG9uZSBjaGFyYWN0ZXIgYmVmb3JlIG9wdGlvbmFsIHBhZGRpbmdcbiAgICByZXR1cm4gL15bQS1aYS16MC05Ky9dKz17MCwyfSQvLnRlc3Qoc3RyKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBUeXBlIGd1YXJkIGZvciBFeHBvcnRlZFBlcnNvbmFcbiAgICovXG4gIHByaXZhdGUgaXNFeHBvcnRlZFBlcnNvbmEob2JqOiBhbnkpOiBvYmogaXMgRXhwb3J0ZWRQZXJzb25hIHtcbiAgICByZXR1cm4gb2JqICYmIFxuICAgICAgdHlwZW9mIG9iai5tZXRhZGF0YSA9PT0gJ29iamVjdCcgJiZcbiAgICAgIHR5cGVvZiBvYmouY29udGVudCA9PT0gJ3N0cmluZycgJiZcbiAgICAgIHR5cGVvZiBvYmouZmlsZW5hbWUgPT09ICdzdHJpbmcnO1xuICB9XG5cbiAgLyoqXG4gICAqIFR5cGUgZ3VhcmQgZm9yIEV4cG9ydEJ1bmRsZVxuICAgKi9cbiAgcHJpdmF0ZSBpc0V4cG9ydEJ1bmRsZShvYmo6IGFueSk6IG9iaiBpcyBFeHBvcnRCdW5kbGUge1xuICAgIHJldHVybiBvYmogJiZcbiAgICAgIHR5cGVvZiBvYmoudmVyc2lvbiA9PT0gJ3N0cmluZycgJiZcbiAgICAgIEFycmF5LmlzQXJyYXkob2JqLnBlcnNvbmFzKSAmJlxuICAgICAgdHlwZW9mIG9iai5wZXJzb25hQ291bnQgPT09ICdudW1iZXInO1xuICB9XG5cbiAgLyoqXG4gICAqIEZvcm1hdCBidW5kbGUgaW1wb3J0IHJlc3VsdHNcbiAgICovXG4gIHByaXZhdGUgZm9ybWF0QnVuZGxlSW1wb3J0UmVzdWx0KHJlc3VsdHM6IGFueSk6IHN0cmluZyB7XG4gICAgbGV0IG1lc3NhZ2UgPSBgQnVuZGxlIEltcG9ydCBTdW1tYXJ5OlxcbmA7XG4gICAgbWVzc2FnZSArPSBg4pyFIFN1Y2Nlc3NmdWxseSBpbXBvcnRlZDogJHtyZXN1bHRzLmltcG9ydGVkLmxlbmd0aH0gcGVyc29uYXNcXG5gO1xuICAgIFxuICAgIGlmIChyZXN1bHRzLmltcG9ydGVkLmxlbmd0aCA+IDApIHtcbiAgICAgIG1lc3NhZ2UgKz0gYEltcG9ydGVkOlxcbiR7cmVzdWx0cy5pbXBvcnRlZC5tYXAoKG46IHN0cmluZykgPT4gYCAgLSAke259YCkuam9pbignXFxuJyl9XFxuYDtcbiAgICB9XG5cbiAgICBpZiAocmVzdWx0cy5mYWlsZWQubGVuZ3RoID4gMCkge1xuICAgICAgbWVzc2FnZSArPSBgXFxu4p2MIEZhaWxlZDogJHtyZXN1bHRzLmZhaWxlZC5sZW5ndGh9IHBlcnNvbmFzXFxuYDtcbiAgICAgIG1lc3NhZ2UgKz0gYEVycm9yczpcXG4ke3Jlc3VsdHMuZmFpbGVkLm1hcCgoZTogc3RyaW5nKSA9PiBgICAtICR7ZX1gKS5qb2luKCdcXG4nKX1gO1xuICAgIH1cblxuICAgIHJldHVybiBtZXNzYWdlO1xuICB9XG59Il19