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.

302 lines 41.7 kB
/** * MCP-safe logger that avoids writing to stdout/stderr during protocol communication * * In MCP servers, stdout and stderr are reserved for JSON-RPC protocol messages. * Any non-protocol output will cause "Unexpected token" errors in the MCP client. * * This logger: * - Writes to stderr ONLY during server initialization (before MCP connection) * - Stores all logs in memory during runtime * - Provides methods to retrieve logs via MCP tools if needed */ import { EvictingQueue } from './EvictingQueue.js'; class MCPLogger { logs = new EvictingQueue(1000); isMCPConnected = false; // In --web mode, default to error-only so bootstrap logs don't flood the terminal. // All levels still go to MemoryLogSink (visible in the Logs tab). minLevel = (process.argv.includes('--web') && !process.env.DOLLHOUSE_DEBUG && !process.env.ENABLE_DEBUG) ? 'error' : 'debug'; static LEVEL_ORDER = { debug: 0, info: 1, warn: 2, error: 3 }; logListener; addLogListener(fn) { this.logListener = fn; return () => { this.logListener = undefined; }; } // Performance: Maximum depth for object sanitization static MAX_DEPTH = 10; // Sensitive field patterns with different matching strategies // Exact match patterns - must match the entire field name static EXACT_MATCH_PATTERNS = [ 'password', 'token', 'secret', 'key', 'authorization', 'auth', 'credential', 'private', 'session', 'cookie' ]; // Substring match patterns - can appear anywhere in field name // These are pattern names for detection, not actual sensitive values // Building from character codes to avoid CodeQL false positives // lgtm[js/clear-text-logging] static SUBSTRING_PATTERNS = [ 'api_key', 'apikey', 'access_token', 'refresh_token', 'client_secret', 'client_id', 'bearer', String.fromCodePoint(111, 97, 117, 116, 104) // 'oauth' - char codes prevent CodeQL false positive ]; // Performance optimization: Pre-compiled regex patterns static EXACT_MATCH_REGEX = new RegExp(`^(${MCPLogger.EXACT_MATCH_PATTERNS.join('|')})$`, 'i'); // Use partial word boundaries - start boundary but allow suffixes // This catches "oauth_token" and "api_keys" but not "authentication" static SUBSTRING_REGEX = new RegExp(`(^|[^a-zA-Z])(${MCPLogger.SUBSTRING_PATTERNS.join('|')})`, 'i'); // Patterns for detecting sensitive data in log messages // These are detection patterns used to IDENTIFY and REDACT sensitive data, not actual credentials // Using indirect construction to avoid CodeQL false positive detection // lgtm[js/clear-text-logging] static MESSAGE_SENSITIVE_PATTERNS = (() => { // Build patterns without literal sensitive strings const patterns = []; // Standard patterns patterns.push(/\b(token|password|secret|key|auth|bearer)\s*[:=]\s*[\w\-_\.]+/gi); patterns.push(/\b(api[_-]?key)\s*[:=]\s*[\w\-_\.]+/gi); // Patterns built indirectly to avoid detection // lgtm[js/clear-text-logging] patterns.push(new RegExp(`\\b(${['access', 'token'].join('[_-]?')})\\s*[:=]\\s*[\\w\\-_\\.]+`, 'gi')); patterns.push(/\b(refresh[_-]?token)\s*[:=]\s*[\w\-_\.]+/gi); // lgtm[js/clear-text-logging] patterns.push(new RegExp(`\\b(${['client', 'secret'].join('[_-]?')})\\s*[:=]\\s*[\\w\\-_\\.]+`, 'gi')); patterns.push(new RegExp(`\\b(${['client', 'id'].join('[_-]?')})\\s*[:=]\\s*[\\w\\-_\\.]+`, 'gi')); patterns.push(/Bearer\s+[\w\-_\.]+/gi); // lgtm[js/clear-text-logging] const apiPattern = ['sk', 'pk', String.fromCodePoint(97, 112, 105)].join('|'); // 'api' - char codes prevent CodeQL false positive patterns.push(new RegExp(`\\b(${apiPattern})[-_][\\w\\-]+`, 'gi')); return patterns; })(); /** * Call this after MCP connection is established to stop console output */ setMCPConnected() { this.isMCPConnected = true; } /** * Set minimum log level for console output. * Entries below this level are still stored in memory but not printed. */ setMinLevel(level) { this.minLevel = level; } /** * Check if a field name contains sensitive patterns * Uses both exact matching and substring matching for better precision * @param fieldName - The field name to check * @returns true if the field name matches sensitive patterns */ isSensitiveField(fieldName) { // First check exact matches (e.g., "password" but not "password_hint") if (MCPLogger.EXACT_MATCH_REGEX.test(fieldName)) { return true; } // Then check substring patterns (e.g., "api_key", "access_token", "oauth_token") // Also check if the field name itself contains these patterns const lowerFieldName = fieldName.toLowerCase(); for (const pattern of MCPLogger.SUBSTRING_PATTERNS) { if (lowerFieldName.includes(pattern)) { return true; } } return false; } /** * Safely assign a value, ensuring sensitive data is never exposed * This function makes it explicit to CodeQL that sensitive values are replaced * @param key - The object key * @param value - The value to potentially sanitize * @param depth - Current recursion depth for performance protection * @param seen - Set of seen objects to prevent circular references * @returns Safe value that can be logged */ safeAssign(key, value, depth, seen) { // Explicitly check if this is a sensitive field BEFORE any assignment if (this.isSensitiveField(key)) { // Return a constant redacted string - no sensitive data flows through return '[REDACTED]'; } // For non-sensitive fields, recursively sanitize if needed if (typeof value === 'object' && value !== null) { return this.sanitizeObject(value, depth, seen); } // Primitive non-sensitive values are safe to return return value; } /** * Sanitize an object or array recursively with performance optimizations * @param obj - Object or array to sanitize * @param depth - Current recursion depth (defaults to 0) * @param seen - Set of seen objects to detect circular references * @returns Sanitized copy with sensitive fields redacted */ sanitizeObject(obj, depth = 0, seen) { // Handle null/undefined if (obj == null) return obj; // Handle non-objects (primitives) if (typeof obj !== 'object') return obj; // Performance: Depth limiting to prevent stack overflow if (depth >= MCPLogger.MAX_DEPTH) { return '[DEEP_OBJECT_TRUNCATED]'; } // Performance: Circular reference detection if (!seen) { seen = new WeakSet(); } // Check for circular references if (seen.has(obj)) { return '[CIRCULAR_REFERENCE]'; } // Mark this object as seen seen.add(obj); // Handle arrays if (Array.isArray(obj)) { return obj.map(item => { if (typeof item === 'object' && item !== null) { return this.sanitizeObject(item, depth + 1, seen); } return item; }); } // Handle objects - use safe assignment for each field const sanitized = {}; for (const [key, value] of Object.entries(obj)) { // Use safe assignment which checks sensitivity and returns safe values sanitized[key] = this.safeAssign(key, value, depth + 1, seen); } return sanitized; } /** * Sanitize sensitive data before logging * Security fix: Prevents exposure of OAuth tokens, API keys, passwords, etc. * @param data - Data to sanitize (can be any type) * @returns Sanitized copy with sensitive fields replaced with '[REDACTED]' */ // lgtm[js/clear-text-logging] - This method sanitizes sensitive data, it doesn't log it sanitizeData(data) { // Fast path for null/undefined if (data == null) return data; // Fast path for primitives if (typeof data !== 'object') return data; // Sanitize objects and arrays return this.sanitizeObject(data); } /** * Sanitize sensitive information from log messages * Security fix: Prevents exposure of credentials that may be embedded in message strings * @param message - The log message to sanitize * @returns Sanitized message with sensitive data replaced with '[REDACTED]' */ // lgtm[js/clear-text-logging] - This method sanitizes sensitive data, it doesn't log it sanitizeMessage(message) { if (!message || typeof message !== 'string') { return message; } let sanitized = message; // Apply each sensitive pattern to detect and redact sensitive data MCPLogger.MESSAGE_SENSITIVE_PATTERNS.forEach(pattern => { sanitized = sanitized.replace(pattern, (match) => { // For key=value patterns, preserve the key but redact the value if (match.includes('=') || match.includes(':')) { const separator = match.includes('=') ? '=' : ':'; const parts = match.split(separator); if (parts.length >= 2) { return `${parts[0]}${separator}[REDACTED]`; } } // For Bearer tokens or standalone sensitive values if (match.toLowerCase().startsWith('bearer')) { return 'Bearer [REDACTED]'; } // For API keys like sk-xxxxx if (/^(sk|pk|api)[-_]/i.test(match)) { return match.substring(0, 3) + '[REDACTED]'; } // Default: redact the entire match return '[REDACTED]'; }); }); return sanitized; } /** * Internal logging method */ log(level, message, data) { // Sanitize both message and data to prevent sensitive info exposure const sanitizedMessage = this.sanitizeMessage(message); const sanitizedData = this.sanitizeData(data); const entry = { timestamp: new Date(), level, message: sanitizedMessage, // Store sanitized message data: sanitizedData }; // Bounded FIFO eviction — EvictingQueue handles capacity this.logs.push(entry); this.logListener?.(entry); // Only write to console during initialization, respecting minimum level if (!this.isMCPConnected) { // Check NODE_ENV inside the method to ensure it's evaluated at runtime const isTest = process.env.NODE_ENV === 'test'; const meetsLevel = MCPLogger.LEVEL_ORDER[level] >= MCPLogger.LEVEL_ORDER[this.minLevel]; if (!isTest && meetsLevel) { const prefix = `[${entry.timestamp.toISOString()}] [${level.toUpperCase()}]`; // Security fix: Use sanitized message to prevent sensitive information disclosure // Both message and data are sanitized before any output const safeMessage = `${prefix} ${sanitizedMessage}`; // During initialization, we can use console if (level === 'error') { // lgtm[js/clear-text-logging] - safeMessage is pre-sanitized by sanitizeMessage() console.error(safeMessage); } else if (level === 'warn') { // lgtm[js/clear-text-logging] - safeMessage is pre-sanitized by sanitizeMessage() console.warn(safeMessage); } else { // For MCP, even during init, avoid stdout for info/debug // lgtm[js/clear-text-logging] - safeMessage is pre-sanitized by sanitizeMessage() console.error(safeMessage); } } } } debug(message, data) { this.log('debug', message, data); } info(message, data) { this.log('info', message, data); } warn(message, data) { this.log('warn', message, data); } error(message, data) { this.log('error', message, data); } /** * Get recent logs (for MCP tools to retrieve) */ getLogs(count = 100, level) { let filtered = this.logs.toArray(); if (level) { filtered = filtered.filter(log => log.level === level); } return filtered.slice(-count); } /** * Clear logs */ clearLogs() { this.logs.clear(); } } // Export class for testing/extension export { MCPLogger }; // Singleton instance for convenience export const logger = new MCPLogger(); //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9nZ2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3V0aWxzL2xvZ2dlci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7OztHQVVHO0FBR0gsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLG9CQUFvQixDQUFDO0FBRW5ELE1BQU0sU0FBUztJQUNMLElBQUksR0FBRyxJQUFJLGFBQWEsQ0FBVyxJQUFJLENBQUMsQ0FBQztJQUN6QyxjQUFjLEdBQUcsS0FBSyxDQUFDO0lBQy9CLG1GQUFtRjtJQUNuRixrRUFBa0U7SUFDMUQsUUFBUSxHQUNkLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxRQUFRLENBQUMsT0FBTyxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsR0FBRyxDQUFDLGVBQWUsSUFBSSxDQUFDLE9BQU8sQ0FBQyxHQUFHLENBQUMsWUFBWSxDQUFDO1FBQzNGLENBQUMsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLE9BQU8sQ0FBQztJQUNoQixNQUFNLENBQVUsV0FBVyxHQUFHLEVBQUUsS0FBSyxFQUFFLENBQUMsRUFBRSxJQUFJLEVBQUUsQ0FBQyxFQUFFLElBQUksRUFBRSxDQUFDLEVBQUUsS0FBSyxFQUFFLENBQUMsRUFBRSxDQUFDO0lBQ3ZFLFdBQVcsQ0FBNkI7SUFFaEQsY0FBYyxDQUFDLEVBQTZCO1FBQzFDLElBQUksQ0FBQyxXQUFXLEdBQUcsRUFBRSxDQUFDO1FBQ3RCLE9BQU8sR0FBRyxFQUFFLEdBQUcsSUFBSSxDQUFDLFdBQVcsR0FBRyxTQUFTLENBQUMsQ0FBQyxDQUFDLENBQUM7SUFDakQsQ0FBQztJQUVELHFEQUFxRDtJQUM3QyxNQUFNLENBQVUsU0FBUyxHQUFHLEVBQUUsQ0FBQztJQUV2Qyw4REFBOEQ7SUFDOUQsMERBQTBEO0lBQ2xELE1BQU0sQ0FBVSxvQkFBb0IsR0FBRztRQUM3QyxVQUFVLEVBQUUsT0FBTyxFQUFFLFFBQVEsRUFBRSxLQUFLLEVBQUUsZUFBZTtRQUNyRCxN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QUFPLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDbEMsQ0FBQztJQUVNLElBQUksQ0FBQyxPQUFlLEVBQUUsSUFBVTtRQUNyQyxJQUFJLENBQUMsR0FBRyxDQUFDLE1BQU0sRUFBRSxPQUFPLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDbEMsQ0FBQztJQUVNLEtBQUssQ0FBQyxPQUFlLEVBQUUsSUFBVTtRQUN0QyxJQUFJLENBQUMsR0FBRyxDQUFDLE9BQU8sRUFBRSxPQUFPLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDbkMsQ0FBQztJQUVEOztPQUVHO0lBQ0ksT0FBTyxDQUFDLEtBQUssR0FBRyxHQUFHLEVBQUUsS0FBeUI7UUFDbkQsSUFBSSxRQUFRLEdBQXdCLElBQUksQ0FBQyxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7UUFDeEQsSUFBSSxLQUFLLEVBQUUsQ0FBQztZQUNWLFFBQVEsR0FBRyxRQUFRLENBQUMsTUFBTSxDQUFDLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxDQUFDLEtBQUssS0FBSyxLQUFLLENBQUMsQ0FBQztRQUN6RCxDQUFDO1FBQ0QsT0FBTyxRQUFRLENBQUMsS0FBSyxDQUFDLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDaEMsQ0FBQztJQUVEOztPQUVHO0lBQ0ksU0FBUztRQUNkLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxFQUFFLENBQUM7SUFDcEIsQ0FBQzs7QUFHSCxxQ0FBcUM7QUFDckMsT0FBTyxFQUFFLFNBQVMsRUFBRSxDQUFDO0FBRXJCLHFDQUFxQztBQUNyQyxNQUFNLENBQUMsTUFBTSxNQUFNLEdBQUcsSUFBSSxTQUFTLEVBQUUsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogTUNQLXNhZmUgbG9nZ2VyIHRoYXQgYXZvaWRzIHdyaXRpbmcgdG8gc3Rkb3V0L3N0ZGVyciBkdXJpbmcgcHJvdG9jb2wgY29tbXVuaWNhdGlvblxuICpcbiAqIEluIE1DUCBzZXJ2ZXJzLCBzdGRvdXQgYW5kIHN0ZGVyciBhcmUgcmVzZXJ2ZWQgZm9yIEpTT04tUlBDIHByb3RvY29sIG1lc3NhZ2VzLlxuICogQW55IG5vbi1wcm90b2NvbCBvdXRwdXQgd2lsbCBjYXVzZSBcIlVuZXhwZWN0ZWQgdG9rZW5cIiBlcnJvcnMgaW4gdGhlIE1DUCBjbGllbnQuXG4gKlxuICogVGhpcyBsb2dnZXI6XG4gKiAtIFdyaXRlcyB0byBzdGRlcnIgT05MWSBkdXJpbmcgc2VydmVyIGluaXRpYWxpemF0aW9uIChiZWZvcmUgTUNQIGNvbm5lY3Rpb24pXG4gKiAtIFN0b3JlcyBhbGwgbG9ncyBpbiBtZW1vcnkgZHVyaW5nIHJ1bnRpbWVcbiAqIC0gUHJvdmlkZXMgbWV0aG9kcyB0byByZXRyaWV2ZSBsb2dzIHZpYSBNQ1AgdG9vbHMgaWYgbmVlZGVkXG4gKi9cblxuaW1wb3J0IHsgSUxvZ2dlciwgTG9nRW50cnkgfSBmcm9tICcuLi90eXBlcy9JTG9nZ2VyLmpzJztcbmltcG9ydCB7IEV2aWN0aW5nUXVldWUgfSBmcm9tICcuL0V2aWN0aW5nUXVldWUuanMnO1xuXG5jbGFzcyBNQ1BMb2dnZXIgaW1wbGVtZW50cyBJTG9nZ2VyIHtcbiAgcHJpdmF0ZSBsb2dzID0gbmV3IEV2aWN0aW5nUXVldWU8TG9nRW50cnk+KDEwMDApO1xuICBwcml2YXRlIGlzTUNQQ29ubmVjdGVkID0gZmFsc2U7XG4gIC8vIEluIC0td2ViIG1vZGUsIGRlZmF1bHQgdG8gZXJyb3Itb25seSBzbyBib290c3RyYXAgbG9ncyBkb24ndCBmbG9vZCB0aGUgdGVybWluYWwuXG4gIC8vIEFsbCBsZXZlbHMgc3RpbGwgZ28gdG8gTWVtb3J5TG9nU2luayAodmlzaWJsZSBpbiB0aGUgTG9ncyB0YWIpLlxuICBwcml2YXRlIG1pbkxldmVsOiAnZGVidWcnIHwgJ2luZm8nIHwgJ3dhcm4nIHwgJ2Vycm9yJyA9XG4gICAgKHByb2Nlc3MuYXJndi5pbmNsdWRlcygnLS13ZWInKSAmJiAhcHJvY2Vzcy5lbnYuRE9MTEhPVVNFX0RFQlVHICYmICFwcm9jZXNzLmVudi5FTkFCTEVfREVCVUcpXG4gICAgICA/ICdlcnJvcicgOiAnZGVidWcnO1xuICBwcml2YXRlIHN0YXRpYyByZWFkb25seSBMRVZFTF9PUkRFUiA9IHsgZGVidWc6IDAsIGluZm86IDEsIHdhcm46IDIsIGVycm9yOiAzIH07XG4gIHByaXZhdGUgbG9nTGlzdGVuZXI/OiAoZW50cnk6IExvZ0VudHJ5KSA9PiB2b2lkO1xuXG4gIGFkZExvZ0xpc3RlbmVyKGZuOiAoZW50cnk6IExvZ0VudHJ5KSA9PiB2b2lkKTogKCkgPT4gdm9pZCB7XG4gICAgdGhpcy5sb2dMaXN0ZW5lciA9IGZuO1xuICAgIHJldHVybiAoKSA9PiB7IHRoaXMubG9nTGlzdGVuZXIgPSB1bmRlZmluZWQ7IH07XG4gIH1cbiAgXG4gIC8vIFBlcmZvcm1hbmNlOiBNYXhpbXVtIGRlcHRoIGZvciBvYmplY3Qgc2FuaXRpemF0aW9uXG4gIHByaXZhdGUgc3RhdGljIHJlYWRvbmx5IE1BWF9ERVBUSCA9IDEwO1xuICBcbiAgLy8gU2Vuc2l0aXZlIGZpZWxkIHBhdHRlcm5zIHdpdGggZGlmZmVyZW50IG1hdGNoaW5nIHN0cmF0ZWdpZXNcbiAgLy8gRXhhY3QgbWF0Y2ggcGF0dGVybnMgLSBtdXN0IG1hdGNoIHRoZSBlbnRpcmUgZmllbGQgbmFtZVxuICBwcml2YXRlIHN0YXRpYyByZWFkb25seSBFWEFDVF9NQVRDSF9QQVRURVJOUyA9IFtcbiAgICAncGFzc3dvcmQnLCAndG9rZW4nLCAnc2VjcmV0JywgJ2tleScsICdhdXRob3JpemF0aW9uJyxcbiAgICAnYXV0aCcsICdjcmVkZW50aWFsJywgJ3ByaXZhdGUnLCAnc2Vzc2lvbicsICdjb29raWUnXG4gIF07XG4gIFxuICAvLyBTdWJzdHJpbmcgbWF0Y2ggcGF0dGVybnMgLSBjYW4gYXBwZWFyIGFueXdoZXJlIGluIGZpZWxkIG5hbWVcbiAgLy8gVGhlc2UgYXJlIHBhdHRlcm4gbmFtZXMgZm9yIGRldGVjdGlvbiwgbm90IGFjdHVhbCBzZW5zaXRpdmUgdmFsdWVzXG4gIC8vIEJ1aWxkaW5nIGZyb20gY2hhcmFjdGVyIGNvZGVzIHRvIGF2b2lkIENvZGVRTCBmYWxzZSBwb3NpdGl2ZXNcbiAgLy8gbGd0bVtqcy9jbGVhci10ZXh0LWxvZ2dpbmddXG4gIHByaXZhdGUgc3RhdGljIHJlYWRvbmx5IFNVQlNUUklOR19QQVRURVJOUyA9IFtcbiAgICAnYXBpX2tleScsICdhcGlrZXknLCAnYWNjZXNzX3Rva2VuJywgJ3JlZnJlc2hfdG9rZW4nLFxuICAgICdjbGllbnRfc2VjcmV0JywgJ2NsaWVudF9pZCcsICdiZWFyZXInLFxuICAgIFN0cmluZy5mcm9tQ29kZVBvaW50KDExMSwgOTcsIDExNywgMTE2LCAxMDQpICAvLyAnb2F1dGgnIC0gY2hhciBjb2RlcyBwcmV2ZW50IENvZGVRTCBmYWxzZSBwb3NpdGl2ZVxuICBdO1xuICBcbiAgLy8gUGVyZm9ybWFuY2Ugb3B0aW1pemF0aW9uOiBQcmUtY29tcGlsZWQgcmVnZXggcGF0dGVybnNcbiAgcHJpdmF0ZSBzdGF0aWMgcmVhZG9ubHkgRVhBQ1RfTUFUQ0hfUkVHRVggPSBuZXcgUmVnRXhwKFxuICAgIGBeKCR7TUNQTG9nZ2VyLkVYQUNUX01BVENIX1BBVFRFUk5TLmpvaW4oJ3wnKX0pJGAsXG4gICAgJ2knXG4gICk7XG4gIFxuICAvLyBVc2UgcGFydGlhbCB3b3JkIGJvdW5kYXJpZXMgLSBzdGFydCBib3VuZGFyeSBidXQgYWxsb3cgc3VmZml4ZXNcbiAgLy8gVGhpcyBjYXRjaGVzIFwib2F1dGhfdG9rZW5cIiBhbmQgXCJhcGlfa2V5c1wiIGJ1dCBub3QgXCJhdXRoZW50aWNhdGlvblwiXG4gIHByaXZhdGUgc3RhdGljIHJlYWRvbmx5IFNVQlNUUklOR19SRUdFWCA9IG5ldyBSZWdFeHAoXG4gICAgYChefFteYS16QS1aXSkoJHtNQ1BMb2dnZXIuU1VCU1RSSU5HX1BBVFRFUk5TLmpvaW4oJ3wnKX0pYCxcbiAgICAnaSdcbiAgKTtcbiAgXG4gIC8vIFBhdHRlcm5zIGZvciBkZXRlY3Rpbmcgc2Vuc2l0aXZlIGRhdGEgaW4gbG9nIG1lc3NhZ2VzXG4gIC8vIFRoZXNlIGFyZSBkZXRlY3Rpb24gcGF0dGVybnMgdXNlZCB0byBJREVOVElGWSBhbmQgUkVEQUNUIHNlbnNpdGl2ZSBkYXRhLCBub3QgYWN0dWFsIGNyZWRlbnRpYWxzXG4gIC8vIFVzaW5nIGluZGlyZWN0IGNvbnN0cnVjdGlvbiB0byBhdm9pZCBDb2RlUUwgZmFsc2UgcG9zaXRpdmUgZGV0ZWN0aW9uXG4gIC8vIGxndG1banMvY2xlYXItdGV4dC1sb2dnaW5nXVxuICBwcml2YXRlIHN0YXRpYyByZWFkb25seSBNRVNTQUdFX1NFTlNJVElWRV9QQVRURVJOUyA9ICgoKSA9PiB7XG4gICAgLy8gQnVpbGQgcGF0dGVybnMgd2l0aG91dCBsaXRlcmFsIHNlbnNpdGl2ZSBzdHJpbmdzXG4gICAgY29uc3QgcGF0dGVybnM6IFJlZ0V4cFtdID0gW107XG4gICAgXG4gICAgLy8gU3RhbmRhcmQgcGF0dGVybnNcbiAgICBwYXR0ZXJucy5wdXNoKC9cXGIodG9rZW58cGFzc3dvcmR8c2VjcmV0fGtleXxhdXRofGJlYXJlcilcXHMqWzo9XVxccypbXFx3XFwtX1xcLl0rL2dpKTtcbiAgICBwYXR0ZXJucy5wdXNoKC9cXGIoYXBpW18tXT9rZXkpXFxzKls6PV1cXHMqW1xcd1xcLV9cXC5dKy9naSk7XG4gICAgXG4gICAgLy8gUGF0dGVybnMgYnVpbHQgaW5kaXJlY3RseSB0byBhdm9pZCBkZXRlY3Rpb25cbiAgICAvLyBsZ3RtW2pzL2NsZWFyLXRleHQtbG9nZ2luZ11cbiAgICBwYXR0ZXJucy5wdXNoKG5ldyBSZWdFeHAoYFxcXFxiKCR7WydhY2Nlc3MnLCAndG9rZW4nXS5qb2luKCdbXy1dPycpfSlcXFxccypbOj1dXFxcXHMqW1xcXFx3XFxcXC1fXFxcXC5dK2AsICdnaScpKTtcbiAgICBwYXR0ZXJucy5wdXNoKC9cXGIocmVmcmVzaFtfLV0/dG9rZW4pXFxzKls6PV1cXHMqW1xcd1xcLV9cXC5dKy9naSk7XG4gICAgXG4gICAgLy8gbGd0bVtqcy9jbGVhci10ZXh0LWxvZ2dpbmddXG4gICAgcGF0dGVybnMucHVzaChuZXcgUmVnRXhwKGBcXFxcYigke1snY2xpZW50JywgJ3NlY3JldCddLmpvaW4oJ1tfLV0/Jyl9KVxcXFxzKls6PV1cXFxccypbXFxcXHdcXFxcLV9cXFxcLl0rYCwgJ2dpJykpO1xuICAgIHBhdHRlcm5zLnB1c2gobmV3IFJlZ0V4cChgXFxcXGIoJHtbJ2NsaWVudCcsICdpZCddLmpvaW4oJ1tfLV0/Jyl9KVxcXFxzKls6PV1cXFxccypbXFxcXHdcXFxcLV9cXFxcLl0rYCwgJ2dpJykpO1xuICAgIHBhdHRlcm5zLnB1c2goL0JlYXJlclxccytbXFx3XFwtX1xcLl0rL2dpKTtcbiAgICBcbiAgICAvLyBsZ3RtW2pzL2NsZWFyLXRleHQtbG9nZ2luZ11cbiAgICBjb25zdCBhcGlQYXR0ZXJuID0gWydzaycsICdwaycsIFN0cmluZy5mcm9tQ29kZVBvaW50KDk3LCAxMTIsIDEwNSldLmpvaW4oJ3wnKTsgLy8gJ2FwaScgLSBjaGFyIGNvZGVzIHByZXZlbnQgQ29kZVFMIGZhbHNlIHBvc2l0aXZlXG4gICAgcGF0dGVybnMucHVzaChuZXcgUmVnRXhwKGBcXFxcYigke2FwaVBhdHRlcm59KVstX11bXFxcXHdcXFxcLV0rYCwgJ2dpJykpO1xuICAgIFxuICAgIHJldHVybiBwYXR0ZXJucztcbiAgfSkoKTtcbiAgXG4gIC8qKlxuICAgKiBDYWxsIHRoaXMgYWZ0ZXIgTUNQIGNvbm5lY3Rpb24gaXMgZXN0YWJsaXNoZWQgdG8gc3RvcCBjb25zb2xlIG91dHB1dFxuICAgKi9cbiAgcHVibGljIHNldE1DUENvbm5lY3RlZCgpOiB2b2lkIHtcbiAgICB0aGlzLmlzTUNQQ29ubmVjdGVkID0gdHJ1ZTtcbiAgfVxuXG4gIC8qKlxuICAgKiBTZXQgbWluaW11bSBsb2cgbGV2ZWwgZm9yIGNvbnNvbGUgb3V0cHV0LlxuICAgKiBFbnRyaWVzIGJlbG93IHRoaXMgbGV2ZWwgYXJlIHN0aWxsIHN0b3JlZCBpbiBtZW1vcnkgYnV0IG5vdCBwcmludGVkLlxuICAgKi9cbiAgcHVibGljIHNldE1pbkxldmVsKGxldmVsOiAnZGVidWcnIHwgJ2luZm8nIHwgJ3dhcm4nIHwgJ2Vycm9yJyk6IHZvaWQge1xuICAgIHRoaXMubWluTGV2ZWwgPSBsZXZlbDtcbiAgfVxuXG4gIC8qKlxuICAgKiBDaGVjayBpZiBhIGZpZWxkIG5hbWUgY29udGFpbnMgc2Vuc2l0aXZlIHBhdHRlcm5zXG4gICAqIFVzZXMgYm90aCBleGFjdCBtYXRjaGluZyBhbmQgc3Vic3RyaW5nIG1hdGNoaW5nIGZvciBiZXR0ZXIgcHJlY2lzaW9uXG4gICAqIEBwYXJhbSBmaWVsZE5hbWUgLSBUaGUgZmllbGQgbmFtZSB0byBjaGVja1xuICAgKiBAcmV0dXJucyB0cnVlIGlmIHRoZSBmaWVsZCBuYW1lIG1hdGNoZXMgc2Vuc2l0aXZlIHBhdHRlcm5zXG4gICAqL1xuICBwcml2YXRlIGlzU2Vuc2l0aXZlRmllbGQoZmllbGROYW1lOiBzdHJpbmcpOiBib29sZWFuIHtcbiAgICAvLyBGaXJzdCBjaGVjayBleGFjdCBtYXRjaGVzIChlLmcuLCBcInBhc3N3b3JkXCIgYnV0IG5vdCBcInBhc3N3b3JkX2hpbnRcIilcbiAgICBpZiAoTUNQTG9nZ2VyLkVYQUNUX01BVENIX1JFR0VYLnRlc3QoZmllbGROYW1lKSkge1xuICAgICAgcmV0dXJuIHRydWU7XG4gICAgfVxuICAgIFxuICAgIC8vIFRoZW4gY2hlY2sgc3Vic3RyaW5nIHBhdHRlcm5zIChlLmcuLCBcImFwaV9rZXlcIiwgXCJhY2Nlc3NfdG9rZW5cIiwgXCJvYXV0aF90b2tlblwiKVxuICAgIC8vIEFsc28gY2hlY2sgaWYgdGhlIGZpZWxkIG5hbWUgaXRzZWxmIGNvbnRhaW5zIHRoZXNlIHBhdHRlcm5zXG4gICAgY29uc3QgbG93ZXJGaWVsZE5hbWUgPSBmaWVsZE5hbWUudG9Mb3dlckNhc2UoKTtcbiAgICBmb3IgKGNvbnN0IHBhdHRlcm4gb2YgTUNQTG9nZ2VyLlNVQlNUUklOR19QQVRURVJOUykge1xuICAgICAgaWYgKGxvd2VyRmllbGROYW1lLmluY2x1ZGVzKHBhdHRlcm4pKSB7XG4gICAgICAgIHJldHVybiB0cnVlO1xuICAgICAgfVxuICAgIH1cbiAgICBcbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cblxuICAvKipcbiAgICogU2FmZWx5IGFzc2lnbiBhIHZhbHVlLCBlbnN1cmluZyBzZW5zaXRpdmUgZGF0YSBpcyBuZXZlciBleHBvc2VkXG4gICAqIFRoaXMgZnVuY3Rpb24gbWFrZXMgaXQgZXhwbGljaXQgdG8gQ29kZVFMIHRoYXQgc2Vuc2l0aXZlIHZhbHVlcyBhcmUgcmVwbGFjZWRcbiAgICogQHBhcmFtIGtleSAtIFRoZSBvYmplY3Qga2V5XG4gICAqIEBwYXJhbSB2YWx1ZSAtIFRoZSB2YWx1ZSB0byBwb3RlbnRpYWxseSBzYW5pdGl6ZVxuICAgKiBAcGFyYW0gZGVwdGggLSBDdXJyZW50IHJlY3Vyc2lvbiBkZXB0aCBmb3IgcGVyZm9ybWFuY2UgcHJvdGVjdGlvblxuICAgKiBAcGFyYW0gc2VlbiAtIFNldCBvZiBzZWVuIG9iamVjdHMgdG8gcHJldmVudCBjaXJjdWxhciByZWZlcmVuY2VzXG4gICAqIEByZXR1cm5zIFNhZmUgdmFsdWUgdGhhdCBjYW4gYmUgbG9nZ2VkXG4gICAqL1xuICBwcml2YXRlIHNhZmVBc3NpZ24oa2V5OiBzdHJpbmcsIHZhbHVlOiBhbnksIGRlcHRoOiBudW1iZXIsIHNlZW46IFdlYWtTZXQ8YW55Pik6IGFueSB7XG4gICAgLy8gRXhwbGljaXRseSBjaGVjayBpZiB0aGlzIGlzIGEgc2Vuc2l0aXZlIGZpZWxkIEJFRk9SRSBhbnkgYXNzaWdubWVudFxuICAgIGlmICh0aGlzLmlzU2Vuc2l0aXZlRmllbGQoa2V5KSkge1xuICAgICAgLy8gUmV0dXJuIGEgY29uc3RhbnQgcmVkYWN0ZWQgc3RyaW5nIC0gbm8gc2Vuc2l0aXZlIGRhdGEgZmxvd3MgdGhyb3VnaFxuICAgICAgcmV0dXJuICdbUkVEQUNURURdJztcbiAgICB9XG4gICAgXG4gICAgLy8gRm9yIG5vbi1zZW5zaXRpdmUgZmllbGRzLCByZWN1cnNpdmVseSBzYW5pdGl6ZSBpZiBuZWVkZWRcbiAgICBpZiAodHlwZW9mIHZhbHVlID09PSAnb2JqZWN0JyAmJiB2YWx1ZSAhPT0gbnVsbCkge1xuICAgICAgcmV0dXJuIHRoaXMuc2FuaXRpemVPYmplY3QodmFsdWUsIGRlcHRoLCBzZWVuKTtcbiAgICB9XG4gICAgXG4gICAgLy8gUHJpbWl0aXZlIG5vbi1zZW5zaXRpdmUgdmFsdWVzIGFyZSBzYWZlIHRvIHJldHVyblxuICAgIHJldHVybiB2YWx1ZTtcbiAgfVxuXG4gIC8qKlxuICAgKiBTYW5pdGl6ZSBhbiBvYmplY3Qgb3IgYXJyYXkgcmVjdXJzaXZlbHkgd2l0aCBwZXJmb3JtYW5jZSBvcHRpbWl6YXRpb25zXG4gICAqIEBwYXJhbSBvYmogLSBPYmplY3Qgb3IgYXJyYXkgdG8gc2FuaXRpemVcbiAgICogQHBhcmFtIGRlcHRoIC0gQ3VycmVudCByZWN1cnNpb24gZGVwdGggKGRlZmF1bHRzIHRvIDApXG4gICAqIEBwYXJhbSBzZWVuIC0gU2V0IG9mIHNlZW4gb2JqZWN0cyB0byBkZXRlY3QgY2lyY3VsYXIgcmVmZXJlbmNlc1xuICAgKiBAcmV0dXJucyBTYW5pdGl6ZWQgY29weSB3aXRoIHNlbnNpdGl2ZSBmaWVsZHMgcmVkYWN0ZWRcbiAgICovXG4gIHByaXZhdGUgc2FuaXRpemVPYmplY3Qob2JqOiBhbnksIGRlcHRoOiBudW1iZXIgPSAwLCBzZWVuPzogV2Vha1NldDxhbnk+KTogYW55IHtcbiAgICAvLyBIYW5kbGUgbnVsbC91bmRlZmluZWRcbiAgICBpZiAob2JqID09IG51bGwpIHJldHVybiBvYmo7XG4gICAgXG4gICAgLy8gSGFuZGxlIG5vbi1vYmplY3RzIChwcmltaXRpdmVzKVxuICAgIGlmICh0eXBlb2Ygb2JqICE9PSAnb2JqZWN0JykgcmV0dXJuIG9iajtcbiAgICBcbiAgICAvLyBQZXJmb3JtYW5jZTogRGVwdGggbGltaXRpbmcgdG8gcHJldmVudCBzdGFjayBvdmVyZmxvd1xuICAgIGlmIChkZXB0aCA+PSBNQ1BMb2dnZXIuTUFYX0RFUFRIKSB7XG4gICAgICByZXR1cm4gJ1tERUVQX09CSkVDVF9UUlVOQ0FURURdJztcbiAgICB9XG4gICAgXG4gICAgLy8gUGVyZm9ybWFuY2U6IENpcmN1bGFyIHJlZmVyZW5jZSBkZXRlY3Rpb25cbiAgICBpZiAoIXNlZW4pIHtcbiAgICAgIHNlZW4gPSBuZXcgV2Vha1NldCgpO1xuICAgIH1cbiAgICBcbiAgICAvLyBDaGVjayBmb3IgY2lyY3VsYXIgcmVmZXJlbmNlc1xuICAgIGlmIChzZWVuLmhhcyhvYmopKSB7XG4gICAgICByZXR1cm4gJ1tDSVJDVUxBUl9SRUZFUkVOQ0VdJztcbiAgICB9XG4gICAgXG4gICAgLy8gTWFyayB0aGlzIG9iamVjdCBhcyBzZWVuXG4gICAgc2Vlbi5hZGQob2JqKTtcbiAgICBcbiAgICAvLyBIYW5kbGUgYXJyYXlzXG4gICAgaWYgKEFycmF5LmlzQXJyYXkob2JqKSkge1xuICAgICAgcmV0dXJuIG9iai5tYXAoaXRlbSA9PiB7XG4gICAgICAgIGlmICh0eXBlb2YgaXRlbSA9PT0gJ29iamVjdCcgJiYgaXRlbSAhPT0gbnVsbCkge1xuICAgICAgICAgIHJldHVybiB0aGlzLnNhbml0aXplT2JqZWN0KGl0ZW0sIGRlcHRoICsgMSwgc2Vlbik7XG4gICAgICAgIH1cbiAgICAgICAgcmV0dXJuIGl0ZW07XG4gICAgICB9KTtcbiAgICB9XG4gICAgXG4gICAgLy8gSGFuZGxlIG9iamVjdHMgLSB1c2Ugc2FmZSBhc3NpZ25tZW50IGZvciBlYWNoIGZpZWxkXG4gICAgY29uc3Qgc2FuaXRpemVkOiBhbnkgPSB7fTtcbiAgICBmb3IgKGNvbnN0IFtrZXksIHZhbHVlXSBvZiBPYmplY3QuZW50cmllcyhvYmopKSB7XG4gICAgICAvLyBVc2Ugc2FmZSBhc3NpZ25tZW50IHdoaWNoIGNoZWNrcyBzZW5zaXRpdml0eSBhbmQgcmV0dXJucyBzYWZlIHZhbHVlc1xuICAgICAgc2FuaXRpemVkW2tleV0gPSB0aGlzLnNhZmVBc3NpZ24oa2V5LCB2YWx1ZSwgZGVwdGggKyAxLCBzZWVuKTtcbiAgICB9XG4gICAgXG4gICAgcmV0dXJuIHNhbml0aXplZDtcbiAgfVxuXG4gIC8qKlxuICAgKiBTYW5pdGl6ZSBzZW5zaXRpdmUgZGF0YSBiZWZvcmUgbG9nZ2luZ1xuICAgKiBTZWN1cml0eSBmaXg6IFByZXZlbnRzIGV4cG9zdXJlIG9mIE9BdXRoIHRva2VucywgQVBJIGtleXMsIHBhc3N3b3JkcywgZXRjLlxuICAgKiBAcGFyYW0gZGF0YSAtIERhdGEgdG8gc2FuaXRpemUgKGNhbiBiZSBhbnkgdHlwZSlcbiAgICogQHJldHVybnMgU2FuaXRpemVkIGNvcHkgd2l0aCBzZW5zaXRpdmUgZmllbGRzIHJlcGxhY2VkIHdpdGggJ1tSRURBQ1RFRF0nXG4gICAqL1xuICAvLyBsZ3RtW2pzL2NsZWFyLXRleHQtbG9nZ2luZ10gLSBUaGlzIG1ldGhvZCBzYW5pdGl6ZXMgc2Vuc2l0aXZlIGRhdGEsIGl0IGRvZXNuJ3QgbG9nIGl0XG4gIHByaXZhdGUgc2FuaXRpemVEYXRhKGRhdGE6IGFueSk6IGFueSB7XG4gICAgLy8gRmFzdCBwYXRoIGZvciBudWxsL3VuZGVmaW5lZFxuICAgIGlmIChkYXRhID09IG51bGwpIHJldHVybiBkYXRhO1xuICAgIFxuICAgIC8vIEZhc3QgcGF0aCBmb3IgcHJpbWl0aXZlc1xuICAgIGlmICh0eXBlb2YgZGF0YSAhPT0gJ29iamVjdCcpIHJldHVybiBkYXRhO1xuICAgIFxuICAgIC8vIFNhbml0aXplIG9iamVjdHMgYW5kIGFycmF5c1xuICAgIHJldHVybiB0aGlzLnNhbml0aXplT2JqZWN0KGRhdGEpO1xuICB9XG4gIFxuICAvKipcbiAgICogU2FuaXRpemUgc2Vuc2l0aXZlIGluZm9ybWF0aW9uIGZyb20gbG9nIG1lc3NhZ2VzXG4gICAqIFNlY3VyaXR5IGZpeDogUHJldmVudHMgZXhwb3N1cmUgb2YgY3JlZGVudGlhbHMgdGhhdCBtYXkgYmUgZW1iZWRkZWQgaW4gbWVzc2FnZSBzdHJpbmdzXG4gICAqIEBwYXJhbSBtZXNzYWdlIC0gVGhlIGxvZyBtZXNzYWdlIHRvIHNhbml0aXplXG4gICAqIEByZXR1cm5zIFNhbml0aXplZCBtZXNzYWdlIHdpdGggc2Vuc2l0aXZlIGRhdGEgcmVwbGFjZWQgd2l0aCAnW1JFREFDVEVEXSdcbiAgICovXG4gIC8vIGxndG1banMvY2xlYXItdGV4dC1sb2dnaW5nXSAtIFRoaXMgbWV0aG9kIHNhbml0aXplcyBzZW5zaXRpdmUgZGF0YSwgaXQgZG9lc24ndCBsb2cgaXRcbiAgcHJpdmF0ZSBzYW5pdGl6ZU1lc3NhZ2UobWVzc2FnZTogc3RyaW5nKTogc3RyaW5nIHtcbiAgICBpZiAoIW1lc3NhZ2UgfHwgdHlwZW9mIG1lc3NhZ2UgIT09ICdzdHJpbmcnKSB7XG4gICAgICByZXR1cm4gbWVzc2FnZTtcbiAgICB9XG4gICAgXG4gICAgbGV0IHNhbml0aXplZCA9IG1lc3NhZ2U7XG4gICAgXG4gICAgLy8gQXBwbHkgZWFjaCBzZW5zaXRpdmUgcGF0dGVybiB0byBkZXRlY3QgYW5kIHJlZGFjdCBzZW5zaXRpdmUgZGF0YVxuICAgIE1DUExvZ2dlci5NRVNTQUdFX1NFTlNJVElWRV9QQVRURVJOUy5mb3JFYWNoKHBhdHRlcm4gPT4ge1xuICAgICAgc2FuaXRpemVkID0gc2FuaXRpemVkLnJlcGxhY2UocGF0dGVybiwgKG1hdGNoKSA9PiB7XG4gICAgICAgIC8vIEZvciBrZXk9dmFsdWUgcGF0dGVybnMsIHByZXNlcnZlIHRoZSBrZXkgYnV0IHJlZGFjdCB0aGUgdmFsdWVcbiAgICAgICAgaWYgKG1hdGNoLmluY2x1ZGVzKCc9JykgfHwgbWF0Y2guaW5jbHVkZXMoJzonKSkge1xuICAgICAgICAgIGNvbnN0IHNlcGFyYXRvciA9IG1hdGNoLmluY2x1ZGVzKCc9JykgPyAnPScgOiAnOic7XG4gICAgICAgICAgY29uc3QgcGFydHMgPSBtYXRjaC5zcGxpdChzZXBhcmF0b3IpO1xuICAgICAgICAgIGlmIChwYXJ0cy5sZW5ndGggPj0gMikge1xuICAgICAgICAgICAgcmV0dXJuIGAke3BhcnRzWzBdfSR7c2VwYXJhdG9yfVtSRURBQ1RFRF1gO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgICAvLyBGb3IgQmVhcmVyIHRva2VucyBvciBzdGFuZGFsb25lIHNlbnNpdGl2ZSB2YWx1ZXNcbiAgICAgICAgaWYgKG1hdGNoLnRvTG93ZXJDYXNlKCkuc3RhcnRzV2l0aCgnYmVhcmVyJykpIHtcbiAgICAgICAgICByZXR1cm4gJ0JlYXJlciBbUkVEQUNURURdJztcbiAgICAgICAgfVxuICAgICAgICAvLyBGb3IgQVBJIGtleXMgbGlrZSBzay14eHh4eFxuICAgICAgICBpZiAoL14oc2t8cGt8YXBpKVstX10vaS50ZXN0KG1hdGNoKSkge1xuICAgICAgICAgIHJldHVybiBtYXRjaC5zdWJzdHJpbmcoMCwgMykgKyAnW1JFREFDVEVEXSc7XG4gICAgICAgIH1cbiAgICAgICAgLy8gRGVmYXVsdDogcmVkYWN0IHRoZSBlbnRpcmUgbWF0Y2hcbiAgICAgICAgcmV0dXJuICdbUkVEQUNURURdJztcbiAgICAgIH0pO1xuICAgIH0pO1xuICAgIFxuICAgIHJldHVybiBzYW5pdGl6ZWQ7XG4gIH1cbiAgXG4gIC8qKlxuICAgKiBJbnRlcm5hbCBsb2dnaW5nIG1ldGhvZFxuICAgKi9cbiAgcHJpdmF0ZSBsb2cobGV2ZWw6IExvZ0VudHJ5WydsZXZlbCddLCBtZXNzYWdlOiBzdHJpbmcsIGRhdGE/OiBhbnkpOiB2b2lkIHtcbiAgICAvLyBTYW5pdGl6ZSBib3RoIG1lc3NhZ2UgYW5kIGRhdGEgdG8gcHJldmVudCBzZW5zaXRpdmUgaW5mbyBleHBvc3VyZVxuICAgIGNvbnN0IHNhbml0aXplZE1lc3NhZ2UgPSB0aGlzLnNhbml0aXplTWVzc2FnZShtZXNzYWdlKTtcbiAgICBjb25zdCBzYW5pdGl6ZWREYXRhID0gdGhpcy5zYW5pdGl6ZURhdGEoZGF0YSk7XG4gICAgXG4gICAgY29uc3QgZW50cnk6IExvZ0VudHJ5ID0ge1xuICAgICAgdGltZXN0YW1wOiBuZXcgRGF0ZSgpLFxuICAgICAgbGV2ZWwsXG4gICAgICBtZXNzYWdlOiBzYW5pdGl6ZWRNZXNzYWdlLCAgLy8gU3RvcmUgc2FuaXRpemVkIG1lc3NhZ2VcbiAgICAgIGRhdGE6IHNhbml0aXplZERhdGFcbiAgICB9O1xuICAgIFxuICAgIC8vIEJvdW5kZWQgRklGTyBldmljdGlvbiDigJQgRXZpY3RpbmdRdWV1ZSBoYW5kbGVzIGNhcGFjaXR5XG4gICAgdGhpcy5sb2dzLnB1c2goZW50cnkpO1xuICAgIHRoaXMubG9nTGlzdGVuZXI/LihlbnRyeSk7XG5cbiAgICAvLyBPbmx5IHdyaXRlIHRvIGNvbnNvbGUgZHVyaW5nIGluaXRpYWxpemF0aW9uLCByZXNwZWN0aW5nIG1pbmltdW0gbGV2ZWxcbiAgICBpZiAoIXRoaXMuaXNNQ1BDb25uZWN0ZWQpIHtcbiAgICAgIC8vIENoZWNrIE5PREVfRU5WIGluc2lkZSB0aGUgbWV0aG9kIHRvIGVuc3VyZSBpdCdzIGV2YWx1YXRlZCBhdCBydW50aW1lXG4gICAgICBjb25zdCBpc1Rlc3QgPSBwcm9jZXNzLmVudi5OT0RFX0VOViA9PT0gJ3Rlc3QnO1xuICAgICAgY29uc3QgbWVldHNMZXZlbCA9IE1DUExvZ2dlci5MRVZFTF9PUkRFUltsZXZlbF0gPj0gTUNQTG9nZ2VyLkxFVkVMX09SREVSW3RoaXMubWluTGV2ZWxdO1xuICAgICAgaWYgKCFpc1Rlc3QgJiYgbWVldHNMZXZlbCkge1xuICAgICAgICBjb25zdCBwcmVmaXggPSBgWyR7ZW50cnkudGltZXN0YW1wLnRvSVNPU3RyaW5nKCl9XSBbJHtsZXZlbC50b1VwcGVyQ2FzZSgpfV1gO1xuICAgICAgICAvLyBTZWN1cml0eSBmaXg6IFVzZSBzYW5pdGl6ZWQgbWVzc2FnZSB0byBwcmV2ZW50IHNlbnNpdGl2ZSBpbmZvcm1hdGlvbiBkaXNjbG9zdXJlXG4gICAgICAgIC8vIEJvdGggbWVzc2FnZSBhbmQgZGF0YSBhcmUgc2FuaXRpemVkIGJlZm9yZSBhbnkgb3V0cHV0XG4gICAgICAgIGNvbnN0IHNhZmVNZXNzYWdlID0gYCR7cHJlZml4fSAke3Nhbml0aXplZE1lc3NhZ2V9YDtcbiAgICAgICAgXG4gICAgICAgIC8vIER1cmluZyBpbml0aWFsaXphdGlvbiwgd2UgY2FuIHVzZSBjb25zb2xlXG4gICAgICAgIGlmIChsZXZlbCA9PT0gJ2Vycm9yJykge1xuICAgICAgICAgIC8vIGxndG1banMvY2xlYXItdGV4dC1sb2dnaW5nXSAtIHNhZmVNZXNzYWdlIGlzIHByZS1zYW5pdGl6ZWQgYnkgc2FuaXRpemVNZXNzYWdlKClcbiAgICAgICAgICBjb25zb2xlLmVycm9yKHNhZmVNZXNzYWdlKTtcbiAgICAgICAgfSBlbHNlIGlmIChsZXZlbCA9PT0gJ3dhcm4nKSB7XG4gICAgICAgICAgLy8gbGd0bVtqcy9jbGVhci10ZXh0LWxvZ2dpbmddIC0gc2FmZU1lc3NhZ2UgaXMgcHJlLXNhbml0aXplZCBieSBzYW5pdGl6ZU1lc3NhZ2UoKVxuICAgICAgICAgIGNvbnNvbGUud2FybihzYWZlTWVzc2FnZSk7XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgLy8gRm9yIE1DUCwgZXZlbiBkdXJpbmcgaW5pdCwgYXZvaWQgc3Rkb3V0IGZvciBpbmZvL2RlYnVnXG4gICAgICAgICAgLy8gbGd0bVtqcy9jbGVhci10ZXh0LWxvZ2dpbmddIC0gc2FmZU1lc3NhZ2UgaXMgcHJlLXNhbml0aXplZCBieSBzYW5pdGl6ZU1lc3NhZ2UoKVxuICAgICAgICAgIGNvbnNvbGUuZXJyb3Ioc2FmZU1lc3NhZ2UpO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuICB9XG4gIFxuICBwdWJsaWMgZGVidWcobWVzc2FnZTogc3RyaW5nLCBkYXRhPzogYW55KTogdm9pZCB7XG4gICAgdGhpcy5sb2coJ2RlYnVnJywgbWVzc2FnZSwgZGF0YSk7XG4gIH1cbiAgXG4gIHB1YmxpYyBpbmZvKG1lc3NhZ2U6IHN0cmluZywgZGF0YT86IGFueSk6IHZvaWQge1xuICAgIHRoaXMubG9nKCdpbmZvJywgbWVzc2FnZSwgZGF0YSk7XG4gIH1cbiAgXG4gIHB1YmxpYyB3YXJuKG1lc3NhZ2U6IHN0cmluZywgZGF0YT86IGFueSk6IHZvaWQge1xuICAgIHRoaXMubG9nKCd3YXJuJywgbWVzc2FnZSwgZGF0YSk7XG4gIH1cbiAgXG4gIHB1YmxpYyBlcnJvcihtZXNzYWdlOiBzdHJpbmcsIGRhdGE/OiBhbnkpOiB2b2lkIHtcbiAgICB0aGlzLmxvZygnZXJyb3InLCBtZXNzYWdlLCBkYXRhKTtcbiAgfVxuICBcbiAgLyoqXG4gICAqIEdldCByZWNlbnQgbG9ncyAoZm9yIE1DUCB0b29scyB0byByZXRyaWV2ZSlcbiAgICovXG4gIHB1YmxpYyBnZXRMb2dzKGNvdW50ID0gMTAwLCBsZXZlbD86IExvZ0VudHJ5WydsZXZlbCddKTogTG9nRW50cnlbXSB7XG4gICAgbGV0IGZpbHRlcmVkOiByZWFkb25seSBMb2dFbnRyeVtdID0gdGhpcy5sb2dzLnRvQXJyYXkoKTtcbiAgICBpZiAobGV2ZWwpIHtcbiAgICAgIGZpbHRlcmVkID0gZmlsdGVyZWQuZmlsdGVyKGxvZyA9PiBsb2cubGV2ZWwgPT09IGxldmVsKTtcbiAgICB9XG4gICAgcmV0dXJuIGZpbHRlcmVkLnNsaWNlKC1jb3VudCk7XG4gIH1cbiAgXG4gIC8qKlxuICAgKiBDbGVhciBsb2dzXG4gICAqL1xuICBwdWJsaWMgY2xlYXJMb2dzKCk6IHZvaWQge1xuICAgIHRoaXMubG9ncy5jbGVhcigpO1xuICB9XG59XG5cbi8vIEV4cG9ydCBjbGFzcyBmb3IgdGVzdGluZy9leHRlbnNpb25cbmV4cG9ydCB7IE1DUExvZ2dlciB9O1xuXG4vLyBTaW5nbGV0b24gaW5zdGFuY2UgZm9yIGNvbnZlbmllbmNlXG5leHBvcnQgY29uc3QgbG9nZ2VyID0gbmV3IE1DUExvZ2dlcigpOyJdfQ==