@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.
382 lines • 55.5 kB
JavaScript
/**
* Skill element class implementing IElement interface.
* Represents a discrete capability for specific tasks.
*
* SECURITY FIXES IMPLEMENTED (PR #319):
* 1. CRITICAL: Added comprehensive input validation for all skill parameters
* 2. MEDIUM: Implemented Unicode normalization to prevent homograph attacks
* 3. MEDIUM: Added audit logging for security events via SecurityMonitor
* 4. MEDIUM: Implemented memory management to prevent unbounded growth
* 5. MEDIUM: Added XSS protection through input sanitization
*/
import { BaseElement } from '../BaseElement.js';
import { ElementType } from '../../portfolio/types.js';
import { logger } from '../../utils/logger.js';
import { sanitizeInput } from '../../security/InputValidator.js';
import { UnicodeValidator } from '../../security/validators/unicodeValidator.js';
import { SecurityMonitor } from '../../security/securityMonitor.js';
export class Skill extends BaseElement {
instructions;
parameters = new Map();
// SECURITY FIX #4: Memory management constants to prevent unbounded growth
// Previously: No limits on parameter storage, could lead to memory exhaustion
// Now: Enforced limits on both count and size
MAX_PARAMETER_COUNT = 100;
MAX_PARAMETER_SIZE = 10000; // Max size per parameter value
constructor(metadata, instructions = '') {
// SECURITY FIX #2: Validate and sanitize ALL metadata fields
// Previously: metadata was used directly without validation
// Now: All string inputs are Unicode normalized and sanitized to prevent:
// - XSS attacks through malicious metadata
// - Unicode homograph attacks
// - Buffer overflow attempts
const sanitizedMetadata = {
...metadata,
name: metadata.name ? sanitizeInput(UnicodeValidator.normalize(metadata.name).normalizedContent, 100) : undefined,
description: metadata.description ? sanitizeInput(UnicodeValidator.normalize(metadata.description).normalizedContent, 500) : undefined
};
super(ElementType.SKILL, sanitizedMetadata);
// Ensure instructions is always a string, even if empty
this.instructions = instructions && instructions.trim() ? sanitizeInput(UnicodeValidator.normalize(instructions).normalizedContent, 10000) : '';
// Ensure skill-specific metadata
this.metadata = {
...this.metadata,
languages: metadata.languages || [],
complexity: metadata.complexity || 'beginner',
domains: metadata.domains || [],
prerequisites: metadata.prerequisites || [],
parameters: metadata.parameters || [],
examples: metadata.examples || [],
proficiency_level: metadata.proficiency_level || 0
};
// Validate parameter definitions
if (this.metadata.parameters) {
this.metadata.parameters = this.metadata.parameters.map(param => ({
...param,
name: sanitizeInput(UnicodeValidator.normalize(param.name).normalizedContent, 50),
description: sanitizeInput(UnicodeValidator.normalize(param.description).normalizedContent, 200),
options: param.options?.map(opt => sanitizeInput(opt, 100))
}));
}
// Initialize parameter values with defaults
this.initializeParameters();
}
/**
* Initialize parameters with default values
*/
initializeParameters() {
if (this.metadata.parameters) {
this.metadata.parameters.forEach(param => {
if (param.default !== undefined) {
this.parameters.set(param.name, param.default);
}
});
}
}
/**
* Set a parameter value
* SECURITY FIX #1: Comprehensive input validation for all skill parameters
* Previously: Parameters were set without validation, creating security risks
* Now: Full validation pipeline including sanitization, type checking, and size limits
*/
setParameter(name, value) {
// CRITICAL FIX: Sanitize parameter name to prevent injection attacks
const sanitizedName = sanitizeInput(name, 50);
const param = this.metadata.parameters?.find(p => p.name === sanitizedName);
if (!param) {
// SECURITY FIX #3: Log audit events for security monitoring
// This helps detect potential attack patterns or misconfigurations
SecurityMonitor.logSecurityEvent({
type: 'YAML_PARSING_WARNING',
severity: 'MEDIUM',
source: 'Skill.setParameter',
details: `Attempt to set unknown parameter: ${sanitizedName} for skill: ${this.metadata.name}`
});
throw new Error(`Parameter '${sanitizedName}' not found in skill definition`);
}
// CRITICAL FIX: Sanitize and validate the value based on type
let sanitizedValue = value;
if (param.type === 'string') {
// SECURITY FIX #2 & #5: Normalize Unicode and sanitize string values
// Previously: String values were used directly without validation
// Now: Full validation pipeline:
// 1. Unicode normalization prevents homograph attacks
// 2. sanitizeInput() strips dangerous HTML/JS content
// 3. Length limits prevent buffer overflow attacks
const normalized = UnicodeValidator.normalize(String(value));
sanitizedValue = sanitizeInput(normalized.normalizedContent, param.max || 1000);
// SECURITY FIX #5: Additional XSS protection
// Check for common XSS patterns even after sanitization
// This provides defense-in-depth against sophisticated attacks
if (sanitizedValue.includes('<script') || sanitizedValue.includes('javascript:')) {
// SECURITY FIX #3: Log high-severity security events
SecurityMonitor.logSecurityEvent({
type: 'CONTENT_INJECTION_ATTEMPT',
severity: 'HIGH',
source: 'Skill.setParameter',
details: `Potential XSS attempt in skill parameter: ${sanitizedName} for skill: ${this.metadata.name}`
});
throw new Error('Invalid characters in parameter value');
}
}
// Type validation
if (!this.validateParameterValue(param, sanitizedValue)) {
throw new Error(`Invalid value for parameter '${sanitizedName}': expected ${param.type}`);
}
// SECURITY FIX #4: Memory management - check parameter count
// Previously: No limits on parameter storage, could lead to memory exhaustion attacks
// Now: Enforced limits prevent attackers from consuming unlimited memory
if (this.parameters.size >= this.MAX_PARAMETER_COUNT && !this.parameters.has(sanitizedName)) {
// SECURITY FIX #3: Log rate limit violations for security monitoring
SecurityMonitor.logSecurityEvent({
type: 'RATE_LIMIT_EXCEEDED',
severity: 'MEDIUM',
source: 'Skill.setParameter',
details: `Parameter limit exceeded for skill: ${this.metadata.name}. Max: ${this.MAX_PARAMETER_COUNT}`
});
throw new Error(`Parameter limit exceeded. Maximum ${this.MAX_PARAMETER_COUNT} parameters allowed`);
}
// SECURITY FIX #4: Check parameter value size for strings
// This prevents attackers from storing massive strings that could exhaust memory
if (param.type === 'string' && sanitizedValue.length > this.MAX_PARAMETER_SIZE) {
throw new Error(`Parameter value too large. Maximum ${this.MAX_PARAMETER_SIZE} characters allowed`);
}
this.parameters.set(sanitizedName, sanitizedValue);
logger.debug(`Set parameter ${sanitizedName} = ${sanitizedValue} for skill ${this.metadata.name}`);
}
/**
* Get a parameter value
*/
getParameter(name) {
return this.parameters.get(name);
}
/**
* Get all parameters as object
*/
getAllParameters() {
return Object.fromEntries(this.parameters);
}
/**
* Validate parameter value against its definition
*/
validateParameterValue(param, value) {
switch (param.type) {
case 'string':
return typeof value === 'string';
case 'number':
const num = Number(value);
if (isNaN(num))
return false;
if (param.min !== undefined && num < param.min)
return false;
if (param.max !== undefined && num > param.max)
return false;
return true;
case 'boolean':
return typeof value === 'boolean';
case 'enum':
return param.options?.includes(String(value)) || false;
default:
return true;
}
}
/**
* Execute the skill with current parameters
*/
async execute(input) {
logger.info(`Executing skill: ${this.metadata.name}`);
// Validate required parameters
const missingRequired = this.metadata.parameters
?.filter(p => p.required && !this.parameters.has(p.name))
.map(p => p.name) || [];
if (missingRequired.length > 0) {
throw new Error(`Missing required parameters: ${missingRequired.join(', ')}`);
}
// Skills don't have built-in execution logic - this would be implemented
// by specific skill types or through a plugin system
logger.debug(`Skill ${this.metadata.name} executed with parameters:`, this.getAllParameters());
return {
skill: this.metadata.name,
parameters: this.getAllParameters(),
input,
executed_at: new Date().toISOString()
};
}
/**
* Skill-specific validation
*/
validate() {
const result = super.validate();
// Initialize arrays if not present
if (!result.errors)
result.errors = [];
if (!result.warnings)
result.warnings = [];
// Instructions should not be empty
if (!this.instructions || this.instructions.trim().length === 0) {
result.errors.push({
field: 'instructions',
message: 'Skill instructions cannot be empty',
code: 'EMPTY_INSTRUCTIONS'
});
}
// Validate complexity level
const validComplexity = ['beginner', 'intermediate', 'advanced', 'expert'];
if (this.metadata.complexity && !validComplexity.includes(this.metadata.complexity)) {
result.errors.push({
field: 'complexity',
message: `Complexity must be one of: ${validComplexity.join(', ')}`,
code: 'INVALID_COMPLEXITY'
});
}
// Validate proficiency level
if (this.metadata.proficiency_level !== undefined) {
if (this.metadata.proficiency_level < 0 || this.metadata.proficiency_level > 100) {
result.errors.push({
field: 'proficiency_level',
message: 'Proficiency level must be between 0 and 100',
code: 'INVALID_PROFICIENCY'
});
}
}
// Validate parameters
if (this.metadata.parameters) {
this.metadata.parameters.forEach((param, index) => {
if (!param.name || param.name.trim() === '') {
result.errors.push({
field: `parameters[${index}].name`,
message: 'Parameter name is required',
code: 'MISSING_PARAMETER_NAME'
});
}
if (param.type === 'enum' && (!param.options || param.options.length === 0)) {
result.errors.push({
field: `parameters[${index}].options`,
message: 'Enum parameter must have options defined',
code: 'MISSING_ENUM_OPTIONS'
});
}
});
}
// Warnings for best practices
if (!this.metadata.domains || this.metadata.domains.length === 0) {
result.warnings.push({
field: 'domains',
message: 'Consider adding domain categories for better organization',
severity: 'low'
});
}
if (!this.metadata.examples || this.metadata.examples.length === 0) {
result.warnings.push({
field: 'examples',
message: 'Adding usage examples improves skill usability',
severity: 'medium'
});
}
// Update the valid flag based on final errors
result.valid = (result.errors?.length || 0) === 0;
return result;
}
/**
* Serialize skill to JSON format
*/
serialize() {
const data = {
id: this.id,
type: this.type,
version: this.version,
metadata: this.metadata,
instructions: this.instructions,
parameters: this.getAllParameters(),
references: this.references,
extensions: this.extensions,
ratings: this.ratings
};
return JSON.stringify(data, null, 2);
}
/**
* Deserialize skill from JSON format
*/
deserialize(data) {
try {
const parsed = JSON.parse(data);
// Update metadata
this.metadata = { ...this.metadata, ...parsed.metadata };
// Update other properties
this.instructions = parsed.instructions || '';
this.references = parsed.references || [];
this.extensions = parsed.extensions || {};
this.ratings = parsed.ratings || this.ratings;
// Update ID and version if provided
if (parsed.id)
this.id = parsed.id;
if (parsed.version)
this.version = parsed.version;
// Restore parameters
if (parsed.parameters) {
this.parameters.clear();
Object.entries(parsed.parameters).forEach(([key, value]) => {
this.parameters.set(key, value);
});
}
this._isDirty = true;
logger.debug(`Deserialized skill: ${this.metadata.name}`);
}
catch (error) {
logger.error(`Failed to deserialize skill: ${error}`);
throw new Error(`Deserialization failed: ${error}`);
}
}
/**
* Skill activation lifecycle
*/
async activate() {
logger.info(`Activating skill: ${this.metadata.name} (${this.id})`);
// Validate that all required parameters are set
const validation = this.validate();
if (!validation.valid) {
throw new Error(`Cannot activate skill with validation errors: ${validation.errors?.map(e => e.message).join(', ')}`);
}
await super.activate?.();
}
/**
* Clone the skill with different parameters
*/
clone(newParameters) {
const cloned = new Skill(this.metadata, this.instructions);
// Copy current parameters
this.parameters.forEach((value, key) => {
cloned.parameters.set(key, value);
});
// Override with new parameters if provided
if (newParameters) {
Object.entries(newParameters).forEach(([key, value]) => {
cloned.setParameter(key, value);
});
}
return cloned;
}
/**
* Skill deactivation lifecycle
* SECURITY FIX #4: Automatically clears parameters on deactivation
* to prevent memory leaks in long-running processes
*/
async deactivate() {
logger.info(`Deactivating skill: ${this.metadata.name} (${this.id})`);
// SECURITY FIX #4: Clear parameters to free memory and prevent leaks
// This ensures skills don't retain data after deactivation
this.clearParameters();
await super.deactivate?.();
}
/**
* Clear all parameters (for memory management)
* SECURITY FIX #4: Added method to clear parameters and prevent memory leaks
* Previously: Parameters accumulated indefinitely in long-running processes
* Now: Provides explicit cleanup mechanism called during deactivation
*/
clearParameters() {
this.parameters.clear();
logger.debug(`Cleared all parameters for skill: ${this.metadata.name}`);
}
}
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiU2tpbGwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvZWxlbWVudHMvc2tpbGxzL1NraWxsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7O0dBVUc7QUFFSCxPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0sbUJBQW1CLENBQUM7QUFFaEQsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQ3ZELE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsYUFBYSxFQUFnQixNQUFNLGtDQUFrQyxDQUFDO0FBQy9FLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLCtDQUErQyxDQUFDO0FBQ2pGLE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSxtQ0FBbUMsQ0FBQztBQWlDcEUsTUFBTSxPQUFPLEtBQU0sU0FBUSxXQUFXO0lBRTdCLFlBQVksQ0FBUztJQUNyQixVQUFVLEdBQXFCLElBQUksR0FBRyxFQUFFLENBQUM7SUFFaEQsMkVBQTJFO0lBQzNFLDhFQUE4RTtJQUM5RSw4Q0FBOEM7SUFDN0IsbUJBQW1CLEdBQUcsR0FBRyxDQUFDO0lBQzFCLGtCQUFrQixHQUFHLEtBQUssQ0FBQyxDQUFDLCtCQUErQjtJQUU1RSxZQUFZLFFBQWdDLEVBQUUsZUFBdUIsRUFBRTtRQUNyRSw2REFBNkQ7UUFDN0QsNERBQTREO1FBQzVELDBFQUEwRTtRQUMxRSwyQ0FBMkM7UUFDM0MsOEJBQThCO1FBQzlCLDZCQUE2QjtRQUM3QixNQUFNLGlCQUFpQixHQUFHO1lBQ3hCLEdBQUcsUUFBUTtZQUNYLElBQUksRUFBRSxRQUFRLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxhQUFhLENBQUMsZ0JBQWdCLENBQUMsU0FBUyxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsQ0FBQyxpQkFBaUIsRUFBRSxHQUFHLENBQUMsQ0FBQyxDQUFDLENBQUMsU0FBUztZQUNqSCxXQUFXLEVB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QUMsWUFBWSxDQUFDLENBQUM7UUFFM0QsMEJBQTBCO1FBQzFCLElBQUksQ0FBQyxVQUFVLENBQUMsT0FBTyxDQUFDLENBQUMsS0FBSyxFQUFFLEdBQUcsRUFBRSxFQUFFO1lBQ3JDLE1BQU0sQ0FBQyxVQUFVLENBQUMsR0FBRyxDQUFDLEdBQUcsRUFBRSxLQUFLLENBQUMsQ0FBQztRQUNwQyxDQUFDLENBQUMsQ0FBQztRQUVILDJDQUEyQztRQUMzQyxJQUFJLGFBQWEsRUFBRSxDQUFDO1lBQ2xCLE1BQU0sQ0FBQyxPQUFPLENBQUMsYUFBYSxDQUFDLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxHQUFHLEVBQUUsS0FBSyxDQUFDLEVBQUUsRUFBRTtnQkFDckQsTUFBTSxDQUFDLFlBQVksQ0FBQyxHQUFHLEVBQUUsS0FBSyxDQUFDLENBQUM7WUFDbEMsQ0FBQyxDQUFDLENBQUM7UUFDTCxDQUFDO1FBRUQsT0FBTyxNQUFNLENBQUM7SUFDaEIsQ0FBQztJQUVEOzs7O09BSUc7SUFDYSxLQUFLLENBQUMsVUFBVTtRQUM5QixNQUFNLENBQUMsSUFBSSxDQUFDLHVCQUF1QixJQUFJLENBQUMsUUFBUSxDQUFDLElBQUksS0FBSyxJQUFJLENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQztRQUV0RSxxRUFBcUU7UUFDckUsMkRBQTJEO1FBQzNELElBQUksQ0FBQyxlQUFlLEVBQUUsQ0FBQztRQUV2QixNQUFNLEtBQUssQ0FBQyxVQUFVLEVBQUUsRUFBRSxDQUFDO0lBQzdCLENBQUM7SUFFRDs7Ozs7T0FLRztJQUNILGVBQWU7UUFDYixJQUFJLENBQUMsVUFBVSxDQUFDLEtBQUssRUFBRSxDQUFDO1FBQ3hCLE1BQU0sQ0FBQyxLQUFLLENBQUMscUNBQXFDLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FBQztJQUMxRSxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFNraWxsIGVsZW1lbnQgY2xhc3MgaW1wbGVtZW50aW5nIElFbGVtZW50IGludGVyZmFjZS5cbiAqIFJlcHJlc2VudHMgYSBkaXNjcmV0ZSBjYXBhYmlsaXR5IGZvciBzcGVjaWZpYyB0YXNrcy5cbiAqIFxuICogU0VDVVJJVFkgRklYRVMgSU1QTEVNRU5URUQgKFBSICMzMTkpOlxuICogMS4gQ1JJVElDQUw6IEFkZGVkIGNvbXByZWhlbnNpdmUgaW5wdXQgdmFsaWRhdGlvbiBmb3IgYWxsIHNraWxsIHBhcmFtZXRlcnNcbiAqIDIuIE1FRElVTTogSW1wbGVtZW50ZWQgVW5pY29kZSBub3JtYWxpemF0aW9uIHRvIHByZXZlbnQgaG9tb2dyYXBoIGF0dGFja3NcbiAqIDMuIE1FRElVTTogQWRkZWQgYXVkaXQgbG9nZ2luZyBmb3Igc2VjdXJpdHkgZXZlbnRzIHZpYSBTZWN1cml0eU1vbml0b3JcbiAqIDQuIE1FRElVTTogSW1wbGVtZW50ZWQgbWVtb3J5IG1hbmFnZW1lbnQgdG8gcHJldmVudCB1bmJvdW5kZWQgZ3Jvd3RoXG4gKiA1LiBNRURJVU06IEFkZGVkIFhTUyBwcm90ZWN0aW9uIHRocm91Z2ggaW5wdXQgc2FuaXRpemF0aW9uXG4gKi9cblxuaW1wb3J0IHsgQmFzZUVsZW1lbnQgfSBmcm9tICcuLi9CYXNlRWxlbWVudC5qcyc7XG5pbXBvcnQgeyBJRWxlbWVudCwgSUVsZW1lbnRNZXRhZGF0YSwgRWxlbWVudFZhbGlkYXRpb25SZXN1bHQgfSBmcm9tICcuLi8uLi90eXBlcy9lbGVtZW50cy9pbmRleC5qcyc7XG5pbXBvcnQgeyBFbGVtZW50VHlwZSB9IGZyb20gJy4uLy4uL3BvcnRmb2xpby90eXBlcy5qcyc7XG5pbXBvcnQgeyBsb2dnZXIgfSBmcm9tICcuLi8uLi91dGlscy9sb2dnZXIuanMnO1xuaW1wb3J0IHsgc2FuaXRpemVJbnB1dCwgdmFsaWRhdGVQYXRoIH0gZnJvbSAnLi4vLi4vc2VjdXJpdHkvSW5wdXRWYWxpZGF0b3IuanMnO1xuaW1wb3J0IHsgVW5pY29kZVZhbGlkYXRvciB9IGZyb20gJy4uLy4uL3NlY3VyaXR5L3ZhbGlkYXRvcnMvdW5pY29kZVZhbGlkYXRvci5qcyc7XG5pbXBvcnQgeyBTZWN1cml0eU1vbml0b3IgfSBmcm9tICcuLi8uLi9zZWN1cml0eS9zZWN1cml0eU1vbml0b3IuanMnO1xuXG4vLyBFeHRlbmQgSUVsZW1lbnRNZXRhZGF0YSB3aXRoIHNraWxsLXNwZWNpZmljIGZpZWxkc1xuZXhwb3J0IGludGVyZmFjZSBTa2lsbE1ldGFkYXRhIGV4dGVuZHMgSUVsZW1lbnRNZXRhZGF0YSB7XG4gIGxhbmd1YWdlcz86IHN0cmluZ1tdOyAgICAgICAgICAgLy8gUHJvZ3JhbW1pbmcvc3Bva2VuIGxhbmd1YWdlcyB0aGlzIHNraWxsIHdvcmtzIHdpdGhcbiAgY29tcGxleGl0eT86ICdiZWdpbm5lcicgfCAnaW50ZXJtZWRpYXRlJyB8ICdhZHZhbmNlZCcgfCAnZXhwZXJ0JztcbiAgZG9tYWlucz86IHN0cmluZ1tdOyAgICAgICAgICAgICAvLyBEb21haW4gY2F0ZWdvcmllcyAoZS5nLiwgd2ViLWRldiwgZGF0YS1zY2llbmNlLCB3cml0aW5nKVxuICBwcmVyZXF1aXNpdGVzPzogc3RyaW5nW107ICAgICAgIC8vIFJlcXVpcmVkIGtub3dsZWRnZSBvciBvdGhlciBza2lsbHNcbiAgcGFyYW1ldGVycz86IFNraWxsUGFyYW1ldGVyW107ICAvLyBDb25maWd1cmFibGUgcGFyYW1ldGVyc1xuICBleGFtcGxlcz86IFNraWxsRXhhbXBsZVtdOyAgICAgIC8vIFVzYWdlIGV4YW1wbGVzXG4gIGNlcnRpZmljYXRpb24/OiBzdHJpbmc7ICAgICAgICAgLy8gRXh0ZXJuYWwgY2VydGlmaWNhdGlvbiBvciB2YWxpZGF0aW9uXG4gIHByb2ZpY2llbmN5X2xldmVsPzogbnVtYmVyOyAgICAgLy8gMS0xMDAgcHJvZmljaWVuY3kgbGV2ZWxcbn1cblxuZXhwb3J0IGludGVyZmFjZSBTa2lsbFBhcmFtZXRlciB7XG4gIG5hbWU6IHN0cmluZztcbiAgdHlwZTogJ3N0cmluZycgfCAnbnVtYmVyJyB8ICdib29sZWFuJyB8ICdlbnVtJztcbiAgZGVzY3JpcHRpb246IHN0cmluZztcbiAgcmVxdWlyZWQ/OiBib29sZWFuO1xuICBkZWZhdWx0PzogYW55O1xuICBvcHRpb25zPzogc3RyaW5nW107ICAvLyBGb3IgZW51bSB0eXBlXG4gIG1pbj86IG51bWJlcjsgICAgICAgIC8vIEZvciBudW1iZXIgdHlwZVxuICBtYXg/OiBudW1iZXI7ICAgICAgICAvLyBGb3IgbnVtYmVyIHR5cGVcbn1cblxuZXhwb3J0IGludGVyZmFjZSBTa2lsbEV4YW1wbGUge1xuICB0aXRsZTogc3RyaW5nO1xuICBkZXNjcmlwdGlvbjogc3RyaW5nO1xuICBpbnB1dD86IGFueTtcbiAgb3V0cHV0PzogYW55O1xuICBjb2RlPzogc3RyaW5nO1xufVxuXG5leHBvcnQgY2xhc3MgU2tpbGwgZXh0ZW5kcyBCYXNlRWxlbWVudCBpbXBsZW1lbnRzIElFbGVtZW50IHtcbiAgcHVibGljIGRlY2xhcmUgbWV0YWRhdGE6IFNraWxsTWV0YWRhdGE7XG4gIHB1YmxpYyBpbnN0cnVjdGlvbnM6IHN0cmluZztcbiAgcHVibGljIHBhcmFtZXRlcnM6IE1hcDxzdHJpbmcsIGFueT4gPSBuZXcgTWFwKCk7XG4gIFxuICAvLyBTRUNVUklUWSBGSVggIzQ6IE1lbW9yeSBtYW5hZ2VtZW50IGNvbnN0YW50cyB0byBwcmV2ZW50IHVuYm91bmRlZCBncm93dGhcbiAgLy8gUHJldmlvdXNseTogTm8gbGltaXRzIG9uIHBhcmFtZXRlciBzdG9yYWdlLCBjb3VsZCBsZWFkIHRvIG1lbW9yeSBleGhhdXN0aW9uXG4gIC8vIE5vdzogRW5mb3JjZWQgbGltaXRzIG9uIGJvdGggY291bnQgYW5kIHNpemVcbiAgcHJpdmF0ZSByZWFkb25seSBNQVhfUEFSQU1FVEVSX0NPVU5UID0gMTAwO1xuICBwcml2YXRlIHJlYWRvbmx5IE1BWF9QQVJBTUVURVJfU0laRSA9IDEwMDAwOyAvLyBNYXggc2l6ZSBwZXIgcGFyYW1ldGVyIHZhbHVlXG5cbiAgY29uc3RydWN0b3IobWV0YWRhdGE6IFBhcnRpYWw8U2tpbGxNZXRhZGF0YT4sIGluc3RydWN0aW9uczogc3RyaW5nID0gJycpIHtcbiAgICAvLyBTRUNVUklUWSBGSVggIzI6IFZhbGlkYXRlIGFuZCBzYW5pdGl6ZSBBTEwgbWV0YWRhdGEgZmllbGRzXG4gICAgLy8gUHJldmlvdXNseTogbWV0YWRhdGEgd2FzIHVzZWQgZGlyZWN0bHkgd2l0aG91dCB2YWxpZGF0aW9uXG4gICAgLy8gTm93OiBBbGwgc3RyaW5nIGlucHV0cyBhcmUgVW5pY29kZSBub3JtYWxpemVkIGFuZCBzYW5pdGl6ZWQgdG8gcHJldmVudDpcbiAgICAvLyAtIFhTUyBhdHRhY2tzIHRocm91Z2ggbWFsaWNpb3VzIG1ldGFkYXRhXG4gICAgLy8gLSBVbmljb2RlIGhvbW9ncmFwaCBhdHRhY2tzXG4gICAgLy8gLSBCdWZmZXIgb3ZlcmZsb3cgYXR0ZW1wdHNcbiAgICBjb25zdCBzYW5pdGl6ZWRNZXRhZGF0YSA9IHtcbiAgICAgIC4uLm1ldGFkYXRhLFxuICAgICAgbmFtZTogbWV0YWRhdGEubmFtZSA/IHNhbml0aXplSW5wdXQoVW5pY29kZVZhbGlkYXRvci5ub3JtYWxpemUobWV0YWRhdGEubmFtZSkubm9ybWFsaXplZENvbnRlbnQsIDEwMCkgOiB1bmRlZmluZWQsXG4gICAgICBkZXNjcmlwdGlvbjogbWV0YWRhdGEuZGVzY3JpcHRpb24gPyBzYW5pdGl6ZUlucHV0KFVuaWNvZGVWYWxpZGF0b3Iubm9ybWFsaXplKG1ldGFkYXRhLmRlc2NyaXB0aW9uKS5ub3JtYWxpemVkQ29udGVudCwgNTAwKSA6IHVuZGVmaW5lZFxuICAgIH07XG4gICAgXG4gICAgc3VwZXIoRWxlbWVudFR5cGUuU0tJTEwsIHNhbml0aXplZE1ldGFkYXRhKTtcbiAgICAvLyBFbnN1cmUgaW5zdHJ1Y3Rpb25zIGlzIGFsd2F5cyBhIHN0cmluZywgZXZlbiBpZiBlbXB0eVxuICAgIHRoaXMuaW5zdHJ1Y3Rpb25zID0gaW5zdHJ1Y3Rpb25zICYmIGluc3RydWN0aW9ucy50cmltKCkgPyBzYW5pdGl6ZUlucHV0KFVuaWNvZGVWYWxpZGF0b3Iubm9ybWFsaXplKGluc3RydWN0aW9ucykubm9ybWFsaXplZENvbnRlbnQsIDEwMDAwKSA6ICcnO1xuICAgIFxuICAgIC8vIEVuc3VyZSBza2lsbC1zcGVjaWZpYyBtZXRhZGF0YVxuICAgIHRoaXMubWV0YWRhdGEgPSB7XG4gICAgICAuLi50aGlzLm1ldGFkYXRhLFxuICAgICAgbGFuZ3VhZ2VzOiBtZXRhZGF0YS5sYW5ndWFnZXMgfHwgW10sXG4gICAgICBjb21wbGV4aXR5OiBtZXRhZGF0YS5jb21wbGV4aXR5IHx8ICdiZWdpbm5lcicsXG4gICAgICBkb21haW5zOiBtZXRhZGF0YS5kb21haW5zIHx8IFtdLFxuICAgICAgcHJlcmVxdWlzaXRlczogbWV0YWRhdGEucHJlcmVxdWlzaXRlcyB8fCBbXSxcbiAgICAgIHBhcmFtZXRlcnM6IG1ldGFkYXRhLnBhcmFtZXRlcnMgfHwgW10sXG4gICAgICBleGFtcGxlczogbWV0YWRhdGEuZXhhbXBsZXMgfHwgW10sXG4gICAgICBwcm9maWNpZW5jeV9sZXZlbDogbWV0YWRhdGEucHJvZmljaWVuY3lfbGV2ZWwgfHwgMFxuICAgIH07XG5cbiAgICAvLyBWYWxpZGF0ZSBwYXJhbWV0ZXIgZGVmaW5pdGlvbnNcbiAgICBpZiAodGhpcy5tZXRhZGF0YS5wYXJhbWV0ZXJzKSB7XG4gICAgICB0aGlzLm1ldGFkYXRhLnBhcmFtZXRlcnMgPSB0aGlzLm1ldGFkYXRhLnBhcmFtZXRlcnMubWFwKHBhcmFtID0+ICh7XG4gICAgICAgIC4uLnBhcmFtLFxuICAgICAgICBuYW1lOiBzYW5pdGl6ZUlucHV0KFVuaWNvZGVWYWxpZGF0b3Iubm9ybWFsaXplKHBhcmFtLm5hbWUpLm5vcm1hbGl6ZWRDb250ZW50LCA1MCksXG4gICAgICAgIGRlc2NyaXB0aW9uOiBzYW5pdGl6ZUlucHV0KFVuaWNvZGVWYWxpZGF0b3Iubm9ybWFsaXplKHBhcmFtLmRlc2NyaXB0aW9uKS5ub3JtYWxpemVkQ29udGVudCwgMjAwKSxcbiAgICAgICAgb3B0aW9uczogcGFyYW0ub3B0aW9ucz8ubWFwKG9wdCA9PiBzYW5pdGl6ZUlucHV0KG9wdCwgMTAwKSlcbiAgICAgIH0pKTtcbiAgICB9XG5cbiAgICAvLyBJbml0aWFsaXplIHBhcmFtZXRlciB2YWx1ZXMgd2l0aCBkZWZhdWx0c1xuICAgIHRoaXMuaW5pdGlhbGl6ZVBhcmFtZXRlcnMoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBJbml0aWFsaXplIHBhcmFtZXRlcnMgd2l0aCBkZWZhdWx0IHZhbHVlc1xuICAgKi9cbiAgcHJpdmF0ZSBpbml0aWFsaXplUGFyYW1ldGVycygpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5tZXRhZGF0YS5wYXJhbWV0ZXJzKSB7XG4gICAgICB0aGlzLm1ldGFkYXRhLnBhcmFtZXRlcnMuZm9yRWFjaChwYXJhbSA9PiB7XG4gICAgICAgIGlmIChwYXJhbS5kZWZhdWx0ICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICB0aGlzLnBhcmFtZXRlcnMuc2V0KHBhcmFtLm5hbWUsIHBhcmFtLmRlZmF1bHQpO1xuICAgICAgICB9XG4gICAgICB9KTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogU2V0IGEgcGFyYW1ldGVyIHZhbHVlXG4gICAqIFNFQ1VSSVRZIEZJWCAjMTogQ29tcHJlaGVuc2l2ZSBpbnB1dCB2YWxpZGF0aW9uIGZvciBhbGwgc2tpbGwgcGFyYW1ldGVyc1xuICAgKiBQcmV2aW91c2x5OiBQYXJhbWV0ZXJzIHdlcmUgc2V0IHdpdGhvdXQgdmFsaWRhdGlvbiwgY3JlYXRpbmcgc2VjdXJpdHkgcmlza3NcbiAgICogTm93OiBGdWxsIHZhbGlkYXRpb24gcGlwZWxpbmUgaW5jbHVkaW5nIHNhbml0aXphdGlvbiwgdHlwZSBjaGVja2luZywgYW5kIHNpemUgbGltaXRzXG4gICAqL1xuICBzZXRQYXJhbWV0ZXIobmFtZTogc3RyaW5nLCB2YWx1ZTogYW55KTogdm9pZCB7XG4gICAgLy8gQ1JJVElDQUwgRklYOiBTYW5pdGl6ZSBwYXJhbWV0ZXIgbmFtZSB0byBwcmV2ZW50IGluamVjdGlvbiBhdHRhY2tzXG4gICAgY29uc3Qgc2FuaXRpemVkTmFtZSA9IHNhbml0aXplSW5wdXQobmFtZSwgNTApO1xuICAgIFxuICAgIGNvbnN0IHBhcmFtID0gdGhpcy5tZXRhZGF0YS5wYXJhbWV0ZXJzPy5maW5kKHAgPT4gcC5uYW1lID09PSBzYW5pdGl6ZWROYW1lKTtcbiAgICBpZiAoIXBhcmFtKSB7XG4gICAgICAvLyBTRUNVUklUWSBGSVggIzM6IExvZyBhdWRpdCBldmVudHMgZm9yIHNlY3VyaXR5IG1vbml0b3JpbmdcbiAgICAgIC8vIFRoaXMgaGVscHMgZGV0ZWN0IHBvdGVudGlhbCBhdHRhY2sgcGF0dGVybnMgb3IgbWlzY29uZmlndXJhdGlvbnNcbiAgICAgIFNlY3VyaXR5TW9uaXRvci5sb2dTZWN1cml0eUV2ZW50KHtcbiAgICAgICAgdHlwZTogJ1lBTUxfUEFSU0lOR19XQVJOSU5HJyxcbiAgICAgICAgc2V2ZXJpdHk6ICdNRURJVU0nLFxuICAgICAgICBzb3VyY2U6ICdTa2lsbC5zZXRQYXJhbWV0ZXInLFxuICAgICAgICBkZXRhaWxzOiBgQXR0ZW1wdCB0byBzZXQgdW5rbm93biBwYXJhbWV0ZXI6ICR7c2FuaXRpemVkTmFtZX0gZm9yIHNraWxsOiAke3RoaXMubWV0YWRhdGEubmFtZX1gXG4gICAgICB9KTtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgUGFyYW1ldGVyICcke3Nhbml0aXplZE5hbWV9JyBub3QgZm91bmQgaW4gc2tpbGwgZGVmaW5pdGlvbmApO1xuICAgIH1cblxuICAgIC8vIENSSVRJQ0FMIEZJWDogU2FuaXRpemUgYW5kIHZhbGlkYXRlIHRoZSB2YWx1ZSBiYXNlZCBvbiB0eXBlXG4gICAgbGV0IHNhbml0aXplZFZhbHVlID0gdmFsdWU7XG4gICAgXG4gICAgaWYgKHBhcmFtLnR5cGUgPT09ICdzdHJpbmcnKSB7XG4gICAgICAvLyBTRUNVUklUWSBGSVggIzIgJiAjNTogTm9ybWFsaXplIFVuaWNvZGUgYW5kIHNhbml0aXplIHN0cmluZyB2YWx1ZXNcbiAgICAgIC8vIFByZXZpb3VzbHk6IFN0cmluZyB2YWx1ZXMgd2VyZSB1c2VkIGRpcmVjdGx5IHdpdGhvdXQgdmFsaWRhdGlvblxuICAgICAgLy8gTm93OiBGdWxsIHZhbGlkYXRpb24gcGlwZWxpbmU6XG4gICAgICAvLyAxLiBVbmljb2RlIG5vcm1hbGl6YXRpb24gcHJldmVudHMgaG9tb2dyYXBoIGF0dGFja3NcbiAgICAgIC8vIDIuIHNhbml0aXplSW5wdXQoKSBzdHJpcHMgZGFuZ2Vyb3VzIEhUTUwvSlMgY29udGVudFxuICAgICAgLy8gMy4gTGVuZ3RoIGxpbWl0cyBwcmV2ZW50IGJ1ZmZlciBvdmVyZmxvdyBhdHRhY2tzXG4gICAgICBjb25zdCBub3JtYWxpemVkID0gVW5pY29kZVZhbGlkYXRvci5ub3JtYWxpemUoU3RyaW5nKHZhbHVlKSk7XG4gICAgICBzYW5pdGl6ZWRWYWx1ZSA9IHNhbml0aXplSW5wdXQobm9ybWFsaXplZC5ub3JtYWxpemVkQ29udGVudCwgcGFyYW0ubWF4IHx8IDEwMDApO1xuICAgICAgXG4gICAgICAvLyBTRUNVUklUWSBGSVggIzU6IEFkZGl0aW9uYWwgWFNTIHByb3RlY3Rpb25cbiAgICAgIC8vIENoZWNrIGZvciBjb21tb24gWFNTIHBhdHRlcm5zIGV2ZW4gYWZ0ZXIgc2FuaXRpemF0aW9uXG4gICAgICAvLyBUaGlzIHByb3ZpZGVzIGRlZmVuc2UtaW4tZGVwdGggYWdhaW5zdCBzb3BoaXN0aWNhdGVkIGF0dGFja3NcbiAgICAgIGlmIChzYW5pdGl6ZWRWYWx1ZS5pbmNsdWRlcygnPHNjcmlwdCcpIHx8IHNhbml0aXplZFZhbHVlLmluY2x1ZGVzKCdqYXZhc2NyaXB0OicpKSB7XG4gICAgICAgIC8vIFNFQ1VSSVRZIEZJWCAjMzogTG9nIGhpZ2gtc2V2ZXJpdHkgc2VjdXJpdHkgZXZlbnRzXG4gICAgICAgIFNlY3VyaXR5TW9uaXRvci5sb2dTZWN1cml0eUV2ZW50KHtcbiAgICAgICAgICB0eXBlOiAnQ09OVEVOVF9JTkpFQ1RJT05fQVRURU1QVCcsXG4gICAgICAgICAgc2V2ZXJpdHk6ICdISUdIJyxcbiAgICAgICAgICBzb3VyY2U6ICdTa2lsbC5zZXRQYXJhbWV0ZXInLFxuICAgICAgICAgIGRldGFpbHM6IGBQb3RlbnRpYWwgWFNTIGF0dGVtcHQgaW4gc2tpbGwgcGFyYW1ldGVyOiAke3Nhbml0aXplZE5hbWV9IGZvciBza2lsbDogJHt0aGlzLm1ldGFkYXRhLm5hbWV9YFxuICAgICAgICB9KTtcbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKCdJbnZhbGlkIGNoYXJhY3RlcnMgaW4gcGFyYW1ldGVyIHZhbHVlJyk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgLy8gVHlwZSB2YWxpZGF0aW9uXG4gICAgaWYgKCF0aGlzLnZhbGlkYXRlUGFyYW1ldGVyVmFsdWUocGFyYW0sIHNhbml0aXplZFZhbHVlKSkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGBJbnZhbGlkIHZhbHVlIGZvciBwYXJhbWV0ZXIgJyR7c2FuaXRpemVkTmFtZX0nOiBleHBlY3RlZCAke3BhcmFtLnR5cGV9YCk7XG4gICAgfVxuXG4gICAgLy8gU0VDVVJJVFkgRklYICM0OiBNZW1vcnkgbWFuYWdlbWVudCAtIGNoZWNrIHBhcmFtZXRlciBjb3VudFxuICAgIC8vIFByZXZpb3VzbHk6IE5vIGxpbWl0cyBvbiBwYXJhbWV0ZXIgc3RvcmFnZSwgY291bGQgbGVhZCB0byBtZW1vcnkgZXhoYXVzdGlvbiBhdHRhY2tzXG4gICAgLy8gTm93OiBFbmZvcmNlZCBsaW1pdHMgcHJldmVudCBhdHRhY2tlcnMgZnJvbSBjb25zdW1pbmcgdW5saW1pdGVkIG1lbW9yeVxuICAgIGlmICh0aGlzLnBhcmFtZXRlcnMuc2l6ZSA+PSB0aGlzLk1BWF9QQVJBTUVURVJfQ09VTlQgJiYgIXRoaXMucGFyYW1ldGVycy5oYXMoc2FuaXRpemVkTmFtZSkpIHtcbiAgICAgIC8vIFNFQ1VSSVRZIEZJWCAjMzogTG9nIHJhdGUgbGltaXQgdmlvbGF0aW9ucyBmb3Igc2VjdXJpdHkgbW9uaXRvcmluZ1xuICAgICAgU2VjdXJpdHlNb25pdG9yLmxvZ1NlY3VyaXR5RXZlbnQoe1xuICAgICAgICB0eXBlOiAnUkFURV9MSU1JVF9FWENFRURFRCcsXG4gICAgICAgIHNldmVyaXR5OiAnTUVESVVNJyxcbiAgICAgICAgc291cmNlOiAnU2tpbGwuc2V0UGFyYW1ldGVyJyxcbiAgICAgICAgZGV0YWlsczogYFBhcmFtZXRlciBsaW1pdCBleGNlZWRlZCBmb3Igc2tpbGw6ICR7dGhpcy5tZXRhZGF0YS5uYW1lfS4gTWF4OiAke3RoaXMuTUFYX1BBUkFNRVRFUl9DT1VOVH1gXG4gICAgICB9KTtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgUGFyYW1ldGVyIGxpbWl0IGV4Y2VlZGVkLiBNYXhpbXVtICR7dGhpcy5NQVhfUEFSQU1FVEVSX0NPVU5UfSBwYXJhbWV0ZXJzIGFsbG93ZWRgKTtcbiAgICB9XG4gICAgXG4gICAgLy8gU0VDVVJJVFkgRklYICM0OiBDaGVjayBwYXJhbWV0ZXIgdmFsdWUgc2l6ZSBmb3Igc3RyaW5nc1xuICAgIC8vIFRoaXMgcHJldmVudHMgYXR0YWNrZXJzIGZyb20gc3RvcmluZyBtYXNzaXZlIHN0cmluZ3MgdGhhdCBjb3VsZCBleGhhdXN0IG1lbW9yeVxuICAgIGlmIChwYXJhbS50eXBlID09PSAnc3RyaW5nJyAmJiBzYW5pdGl6ZWRWYWx1ZS5sZW5ndGggPiB0aGlzLk1BWF9QQVJBTUVURVJfU0laRSkge1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGBQYXJhbWV0ZXIgdmFsdWUgdG9vIGxhcmdlLiBNYXhpbXVtICR7dGhpcy5NQVhfUEFSQU1FVEVSX1NJWkV9IGNoYXJhY3RlcnMgYWxsb3dlZGApO1xuICAgIH1cblxuICAgIHRoaXMucGFyYW1ldGVycy5zZXQoc2FuaXRpemVkTmFtZSwgc2FuaXRpemVkVmFsdWUpO1xuICAgIGxvZ2dlci5kZWJ1ZyhgU2V0IHBhcmFtZXRlciAke3Nhbml0aXplZE5hbWV9ID0gJHtzYW5pdGl6ZWRWYWx1ZX0gZm9yIHNraWxsICR7dGhpcy5tZXRhZGF0YS5uYW1lfWApO1xuICB9XG5cbiAgLyoqXG4gICAqIEdldCBhIHBhcmFtZXRlciB2YWx1ZVxuICAgKi9cbiAgZ2V0UGFyYW1ldGVyKG5hbWU6IHN0cmluZyk6IGFueSB7XG4gICAgcmV0dXJuIHRoaXMucGFyYW1ldGVycy5nZXQobmFtZSk7XG4gIH1cblxuICAvKipcbiAgICogR2V0IGFsbCBwYXJhbWV0ZXJzIGFzIG9iamVjdFxuICAgKi9cbiAgZ2V0QWxsUGFyYW1ldGVycygpOiBSZWNvcmQ8c3RyaW5nLCBhbnk+IHtcbiAgICByZXR1cm4gT2JqZWN0LmZyb21FbnRyaWVzKHRoaXMucGFyYW1ldGVycyk7XG4gIH1cblxuICAvKipcbiAgICogVmFsaWRhdGUgcGFyYW1ldGVyIHZhbHVlIGFnYWluc3QgaXRzIGRlZmluaXRpb25cbiAgICovXG4gIHByaXZhdGUgdmFsaWRhdGVQYXJhbWV0ZXJWYWx1ZShwYXJhbTogU2tpbGxQYXJhbWV0ZXIsIHZhbHVlOiBhbnkpOiBib29sZWFuIHtcbiAgICBzd2l0Y2ggKHBhcmFtLnR5cGUpIHtcbiAgICAgIGNhc2UgJ3N0cmluZyc6XG4gICAgICAgIHJldHVybiB0eXBlb2YgdmFsdWUgPT09ICdzdHJpbmcnO1xuICAgICAgY2FzZSAnbnVtYmVyJzpcbiAgICAgICAgY29uc3QgbnVtID0gTnVtYmVyKHZhbHVlKTtcbiAgICAgICAgaWYgKGlzTmFOKG51bSkpIHJldHVybiBmYWxzZTtcbiAgICAgICAgaWYgKHBhcmFtLm1pbiAhPT0gdW5kZWZpbmVkICYmIG51bSA8IHBhcmFtLm1pbikgcmV0dXJuIGZhbHNlO1xuICAgICAgICBpZiAocGFyYW0ubWF4ICE9PSB1bmRlZmluZWQgJiYgbnVtID4gcGFyYW0ubWF4KSByZXR1cm4gZmFsc2U7XG4gICAgICAgIHJldHVybiB0cnVlO1xuICAgICAgY2FzZSAnYm9vbGVhbic6XG4gICAgICAgIHJldHVybiB0eXBlb2YgdmFsdWUgPT09ICdib29sZWFuJztcbiAgICAgIGNhc2UgJ2VudW0nOlxuICAgICAgICByZXR1cm4gcGFyYW0ub3B0aW9ucz8uaW5jbHVkZXMoU3RyaW5nKHZhbHVlKSkgfHwgZmFsc2U7XG4gICAgICBkZWZhdWx0OlxuICAgICAgICByZXR1cm4gdHJ1ZTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogRXhlY3V0ZSB0aGUgc2tpbGwgd2l0aCBjdXJyZW50IHBhcmFtZXRlcnNcbiAgICovXG4gIGFzeW5jIGV4ZWN1dGUoaW5wdXQ/OiBhbnkpOiBQcm9taXNlPGFueT4ge1xuICAgIGxvZ2dlci5pbmZvKGBFeGVjdXRpbmcgc2tpbGw6ICR7dGhpcy5tZXRhZGF0YS5uYW1lfWApO1xuICAgIFxuICAgIC8vIFZhbGlkYXRlIHJlcXVpcmVkIHBhcmFtZXRlcnNcbiAgICBjb25zdCBtaXNzaW5nUmVxdWlyZWQgPSB0aGlzLm1ldGFkYXRhLnBhcmFtZXRlcnNcbiAgICAgID8uZmlsdGVyKHAgPT4gcC5yZXF1aXJlZCAmJiAhdGhpcy5wYXJhbWV0ZXJzLmhhcyhwLm5hbWUpKVxuICAgICAgLm1hcChwID0+IHAubmFtZSkgfHwgW107XG4gICAgXG4gICAgaWYgKG1pc3NpbmdSZXF1aXJlZC5sZW5ndGggPiAwKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYE1pc3NpbmcgcmVxdWlyZWQgcGFyYW1ldGVyczogJHttaXNzaW5nUmVxdWlyZWQuam9pbignLCAnKX1gKTtcbiAgICB9XG5cbiAgICAvLyBTa2lsbHMgZG9uJ3QgaGF2ZSBidWlsdC1pbiBleGVjdXRpb24gbG9naWMgLSB0aGlzIHdvdWxkIGJlIGltcGxlbWVudGVkXG4gICAgLy8gYnkgc3BlY2lmaWMgc2tpbGwgdHlwZXMgb3IgdGhyb3VnaCBhIHBsdWdpbiBzeXN0ZW1cbiAgICBsb2dnZXIuZGVidWcoYFNraWxsICR7dGhpcy5tZXRhZGF0YS5uYW1lfSBleGVjdXRlZCB3aXRoIHBhcmFtZXRlcnM6YCwgdGhpcy5nZXRBbGxQYXJhbWV0ZXJzKCkpO1xuICAgIFxuICAgIHJldHVybiB7XG4gICAgICBza2lsbDogdGhpcy5tZXRhZGF0YS5uYW1lLFxuICAgICAgcGFyYW1ldGVyczogdGhpcy5nZXRBbGxQYXJhbWV0ZXJzKCksXG4gICAgICBpbnB1dCxcbiAgICAgIGV4ZWN1dGVkX2F0OiBuZXcgRGF0ZSgpLnRvSVNPU3RyaW5nKClcbiAgICB9O1xuICB9XG5cbiAgLyoqXG4gICAqIFNraWxsLXNwZWNpZmljIHZhbGlkYXRpb25cbiAgICovXG4gIHB1YmxpYyBvdmVycmlkZSB2YWxpZGF0ZSgpOiBFbGVtZW50VmFsaWRhdGlvblJlc3VsdCB7XG4gICAgY29uc3QgcmVzdWx0ID0gc3VwZXIudmFsaWRhdGUoKTtcbiAgICBcbiAgICAvLyBJbml0aWFsaXplIGFycmF5cyBpZiBub3QgcHJlc2VudFxuICAgIGlmICghcmVzdWx0LmVycm9ycykgcmVzdWx0LmVycm9ycyA9IFtdO1xuICAgIGlmICghcmVzdWx0Lndhcm5pbmdzKSByZXN1bHQud2FybmluZ3MgPSBbXTtcbiAgICBcbiAgICAvLyBJbnN0cnVjdGlvbnMgc2hvdWxkIG5vdCBiZSBlbXB0eVxuICAgIGlmICghdGhpcy5pbnN0cnVjdGlvbnMgfHwgdGhpcy5pbnN0cnVjdGlvbnMudHJpbSgpLmxlbmd0aCA9PT0gMCkge1xuICAgICAgcmVzdWx0LmVycm9ycy5wdXNoKHtcbiAgICAgICAgZmllbGQ6ICdpbnN0cnVjdGlvbnMnLFxuICAgICAgICBtZXNzYWdlOiAnU2tpbGwgaW5zdHJ1Y3Rpb25zIGNhbm5vdCBiZSBlbXB0eScsXG4gICAgICAgIGNvZGU6ICdFTVBUWV9JTlNUUlVDVElPTlMnXG4gICAgICB9KTtcbiAgICB9XG5cbiAgICAvLyBWYWxpZGF0ZSBjb21wbGV4aXR5IGxldmVsXG4gICAgY29uc3QgdmFsaWRDb21wbGV4aXR5ID0gWydiZWdpbm5lcicsICdpbnRlcm1lZGlhdGUnLCAnYWR2YW5jZWQnLCAnZXhwZXJ0J107XG4gICAgaWYgKHRoaXMubWV0YWRhdGEuY29tcGxleGl0eSAmJiAhdmFsaWRDb21wbGV4aXR5LmluY2x1ZGVzKHRoaXMubWV0YWRhdGEuY29tcGxleGl0eSkpIHtcbiAgICAgIHJlc3VsdC5lcnJvcnMucHVzaCh7XG4gICAgICAgIGZpZWxkOiAnY29tcGxleGl0eScsXG4gICAgICAgIG1lc3NhZ2U6IGBDb21wbGV4aXR5IG11c3QgYmUgb25lIG9mOiAke3ZhbGlkQ29tcGxleGl0eS5qb2luKCcsICcpfWAsXG4gICAgICAgIGNvZGU6ICdJTlZBTElEX0NPTVBMRVhJVFknXG4gICAgICB9KTtcbiAgICB9XG5cbiAgICAvLyBWYWxpZGF0ZSBwcm9maWNpZW5jeSBsZXZlbFxuICAgIGlmICh0aGlzLm1ldGFkYXRhLnByb2ZpY2llbmN5X2xldmVsICE9PSB1bmRlZmluZWQpIHtcbiAgICAgIGlmICh0aGlzLm1ldGFkYXRhLnByb2ZpY2llbmN5X2xldmVsIDwgMCB8fCB0aGlzLm1ldGFkYXRhLnByb2ZpY2llbmN5X2xldmVsID4gMTAwKSB7XG4gICAgICAgIHJlc3VsdC5lcnJvcnMucHVzaCh7XG4gICAgICAgICAgZmllbGQ6ICdwcm9maWNpZW5jeV9sZXZlbCcsXG4gICAgICAgICAgbWVzc2FnZTogJ1Byb2ZpY2llbmN5IGxldmVsIG11c3QgYmUgYmV0d2VlbiAwIGFuZCAxMDAnLFxuICAgICAgICAgIGNvZGU6ICdJTlZBTElEX1BST0ZJQ0lFTkNZJ1xuICAgICAgICB9KTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyBWYWxpZGF0ZSBwYXJhbWV0ZXJzXG4gICAgaWYgKHRoaXMubWV0YWRhdGEucGFyYW1ldGVycykge1xuICAgICAgdGhpcy5tZXRhZGF0YS5wYXJhbWV0ZXJzLmZvckVhY2goKHBhcmFtLCBpbmRleCkgPT4ge1xuICAgICAgICBpZiAoIXBhcmFtLm5hbWUgfHwgcGFyYW0ubmFtZS50cmltKCkgPT09ICcnKSB7XG4gICAgICAgICAgcmVzdWx0LmVycm9ycyEucHVzaCh7XG4gICAgICAgICAgICBmaWVsZDogYHBhcmFtZXRlcnNbJHtpbmRleH1dLm5hbWVgLFxuICAgICAgICAgICAgbWVzc2FnZTogJ1BhcmFtZXRlciBuYW1lIGlzIHJlcXVpcmVkJyxcbiAgICAgICAgICAgIGNvZGU6ICdNSVNTSU5HX1BBUkFNRVRFUl9OQU1FJ1xuICAgICAgICAgIH0pO1xuICAgICAgICB9XG4gICAgICAgIFxuICAgICAgICBpZiAocGFyYW0udHlwZSA9PT0gJ2VudW0nICYmICghcGFyYW0ub3B0aW9ucyB8fCBwYXJhbS5vcHRpb25zLmxlbmd0aCA9PT0gMCkpIHtcbiAgICAgICAgICByZXN1bHQuZXJyb3JzIS5wdXNoKHtcbiAgICAgICAgICAgIGZpZWxkOiBgcGFyYW1ldGVyc1ske2luZGV4fV0ub3B0aW9uc2AsXG4gICAgICAgICAgICBtZXNzYWdlOiAnRW51bSBwYXJhbWV0ZXIgbXVzdCBoYXZlIG9wdGlvbnMgZGVmaW5lZCcsXG4gICAgICAgICAgICBjb2RlOiAnTUlTU0lOR19FTlVNX09QVElPTlMnXG4gICAgICAgICAgfSk7XG4gICAgICAgIH1cbiAgICAgIH0pO1xuICAgIH1cblxuICAgIC8vIFdhcm5pbmdzIGZvciBiZXN0IHByYWN0aWNlc1xuICAgIGlmICghdGhpcy5tZXRhZGF0YS5kb21haW5zIHx8IHRoaXMubWV0YWRhdGEuZG9tYWlucy5sZW5ndGggPT09IDApIHtcbiAgICAgIHJlc3VsdC53YXJuaW5ncy5wdXNoKHtcbiAgICAgICAgZmllbGQ6ICdkb21haW5zJyxcbiAgICAgICAgbWVzc2FnZT