@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.
326 lines • 59 kB
JavaScript
/**
* Unicode Validator for DollhouseMCP
*
* Prevents Unicode-based bypass attacks including:
* - Homograph attacks (visually similar characters)
* - Direction override attacks (RLO/LRO)
* - Mixed script attacks
* - Zero-width character injection
* - Unicode normalization bypasses
*
* Security: SEC-001 - Unicode attack prevention
*/
import { SecurityMonitor } from '../securityMonitor.js';
import { escalateSeverity } from '../constants.js';
export class UnicodeValidator {
/**
* Unicode attack patterns and confusable characters
*/
/**
* Direction override characters that can hide or reverse text display
* @see https://unicode.org/reports/tr9/#Directional_Formatting_Characters
* U+202A-U+202E: Left/Right embedding and override marks (LRE, RLE, PDF, LRO, RLO)
* U+2066-U+2069: Isolate formatting characters (LRI, RLI, FSI, PDI)
*/
static DIRECTION_OVERRIDE_CHARS = /[\u202A-\u202E\u2066-\u2069]/g;
/**
* Zero-width and invisible formatting characters often used to hide payloads
* U+200B-U+200F: Zero-width spaces and directional marks
* U+2028-U+202F: Line/paragraph separators and formatting characters
* U+FEFF: Zero-width no-break space (Byte Order Mark)
*/
static ZERO_WIDTH_CHARS = /[\u200B-\u200F\u2028-\u202F\uFEFF]/g;
/**
* Non-printable control characters that should not appear in normal text
* U+0000-U+0008, U+000B-U+000C, U+000E-U+001F: C0 control codes (except TAB, LF, CR)
* U+007F-U+009F: Delete and C1 control codes
* U+FFFE-U+FFFF: Non-characters that should never appear in valid text
*/
// eslint-disable-next-line no-control-regex -- Intentionally matching control chars for security sanitization
static NON_PRINTABLE_CHARS = /[\u0000-\u0008\u000B\u000C\u000E-\u001F\u007F-\u009F\uFFFE\uFFFF]/g; // NOSONAR - Intentionally matching control characters for security sanitization
/**
* Common homograph/confusable character mappings
* Maps visually similar Unicode characters to their ASCII equivalents.
* Performance: ~110 entries; replaceConfusables() iterates the full map
* per call but uses Map.has() lookups (O(1) per char). Negligible at
* current size — profile before optimizing if the map grows beyond ~500.
*/
static CONFUSABLE_MAPPINGS = new Map([
// Cyrillic to Latin
['а', 'a'], ['е', 'e'], ['о', 'o'], ['р', 'p'], ['с', 'c'], ['х', 'x'], ['у', 'y'],
['А', 'A'], ['В', 'B'], ['Е', 'E'], ['К', 'K'], ['М', 'M'], ['Н', 'H'], ['О', 'O'],
['Р', 'P'], ['С', 'C'], ['Т', 'T'], ['У', 'Y'], ['Х', 'X'],
// Greek lowercase to Latin
['α', 'a'], ['β', 'b'], ['γ', 'g'], ['δ', 'd'], ['ε', 'e'], ['ζ', 'z'], ['η', 'h'],
['θ', 'th'], ['ι', 'i'], ['κ', 'k'], ['λ', 'l'], ['μ', 'm'], ['ν', 'n'], ['ξ', 'x'],
['ο', 'o'], ['π', 'p'], ['ρ', 'r'], ['σ', 's'], ['τ', 't'], ['υ', 'u'], ['φ', 'f'],
['χ', 'ch'], ['ψ', 'ps'], ['ω', 'w'],
// Greek uppercase to Latin — visually identical to Latin capitals (#1782)
['\u0391', 'A'], ['\u0392', 'B'], ['\u0395', 'E'], ['\u0397', 'H'], ['\u0399', 'I'],
['\u039A', 'K'], ['\u039C', 'M'], ['\u039D', 'N'], ['\u039F', 'O'], ['\u03A1', 'P'],
['\u03A4', 'T'], ['\u03A5', 'Y'], ['\u03A7', 'X'],
// Mathematical symbols to ASCII (various styles)
['𝒂', 'a'], ['𝒃', 'b'], ['𝒄', 'c'], ['𝒅', 'd'], ['𝒆', 'e'], ['𝒇', 'f'], ['𝒈', 'g'], ['𝒉', 'h'], ['𝒊', 'i'], ['𝒋', 'j'], ['𝒌', 'k'], ['𝒍', 'l'], ['𝒎', 'm'], ['𝒏', 'n'], ['𝒐', 'o'], ['𝒑', 'p'], ['𝒒', 'q'], ['𝒓', 'r'], ['𝒔', 's'], ['𝒕', 't'], ['𝒖', 'u'], ['𝒗', 'v'], ['𝒘', 'w'], ['𝒙', 'x'], ['𝒚', 'y'], ['𝒛', 'z'],
['𝐚', 'a'], ['𝐛', 'b'], ['𝐜', 'c'], ['𝐝', 'd'], ['𝐞', 'e'], ['𝐟', 'f'], ['𝐠', 'g'], ['𝐡', 'h'], ['𝐢', 'i'], ['𝐣', 'j'], ['𝐤', 'k'], ['𝐥', 'l'], ['𝐦', 'm'], ['𝐧', 'n'], ['𝐨', 'o'], ['𝐩', 'p'], ['𝐪', 'q'], ['𝐫', 'r'], ['𝐬', 's'], ['𝐭', 't'], ['𝐮', 'u'], ['𝐯', 'v'], ['𝐰', 'w'], ['𝐱', 'x'], ['𝐲', 'y'], ['𝐳', 'z'],
// Special i variants (Turkish, etc.)
['ı', 'i'], ['İ', 'I'], ['і', 'i'], ['Ӏ', 'I'],
// Other common confusables
['ǝ', 'e'], ['ɐ', 'a'], ['ɔ', 'o'], ['ʇ', 't'], ['ʌ', 'v'], ['ʍ', 'w'],
['℃', 'C'], ['℉', 'F'], ['№', 'No'], ['™', 'TM'], ['®', 'R'],
// Fullwidth characters
['A', 'A'], ['B', 'B'], ['C', 'C'], ['D', 'D'], ['E', 'E'], ['F', 'F'], ['G', 'G'], ['H', 'H'], ['I', 'I'], ['J', 'J'], ['K', 'K'], ['L', 'L'], ['M', 'M'], ['N', 'N'], ['O', 'O'], ['P', 'P'], ['Q', 'Q'], ['R', 'R'], ['S', 'S'], ['T', 'T'], ['U', 'U'], ['V', 'V'], ['W', 'W'], ['X', 'X'], ['Y', 'Y'], ['Z', 'Z'],
['a', 'a'], ['b', 'b'], ['c', 'c'], ['d', 'd'], ['e', 'e'], ['f', 'f'], ['g', 'g'], ['h', 'h'], ['i', 'i'], ['j', 'j'], ['k', 'k'], ['l', 'l'], ['m', 'm'], ['n', 'n'], ['o', 'o'], ['p', 'p'], ['q', 'q'], ['r', 'r'], ['s', 's'], ['t', 't'], ['u', 'u'], ['v', 'v'], ['w', 'w'], ['x', 'x'], ['y', 'y'], ['z', 'z'],
['0', '0'], ['1', '1'], ['2', '2'], ['3', '3'], ['4', '4'], ['5', '5'], ['6', '6'], ['7', '7'], ['8', '8'], ['9', '9'],
]);
/**
* Script mixing detection patterns
* Detects suspicious mixing of different Unicode scripts
*/
static SCRIPT_PATTERNS = {
// eslint-disable-next-line no-control-regex -- Intentionally includes control chars for comprehensive Latin script detection
LATIN: /[\u0000-\u007F\u00A0-\u00FF\u0100-\u017F\u0180-\u024F]/, // NOSONAR - Intentionally includes control characters for comprehensive Latin script detection
// Use alternation to avoid SonarCloud thinking \u052F\u2DE0 is a combined character
CYRILLIC: /(?:[\u0400-\u04FF]|[\u0500-\u052F]|[\u2DE0-\u2DFF]|[\uA640-\uA69F])/,
GREEK: /[\u0370-\u03FF\u1F00-\u1FFF]/,
ARABIC: /[\u0600-\u06FF\u0750-\u077F\u08A0-\u08FF\uFB50-\uFDFF\uFE70-\uFEFF]/,
HEBREW: /[\u0590-\u05FF\uFB1D-\uFB4F]/,
CJK: /[\u2E80-\u2EFF\u2F00-\u2FDF\u3000-\u303F\u3040-\u309F\u30A0-\u30FF\u3100-\u312F\u3130-\u318F\u3190-\u319F\u31A0-\u31BF\u31C0-\u31EF\u31F0-\u31FF\u3200-\u32FF\u3300-\u33FF\u3400-\u4DBF\u4DC0-\u4DFF\u4E00-\u9FFF]/,
};
/**
* Normalize Unicode content to prevent bypass attacks
*/
static normalize(content) {
const issues = [];
let normalized = content;
let severity = 'low';
try {
// 1. Detect and log suspicious Unicode patterns before normalization
const suspiciousPatterns = this.detectSuspiciousPatterns(content);
issues.push(...suspiciousPatterns.issues);
if (suspiciousPatterns.severity) {
severity = escalateSeverity(severity, suspiciousPatterns.severity);
}
// 2. Remove direction override characters (prevents RLO/LRO attacks)
if (this.DIRECTION_OVERRIDE_CHARS.test(normalized)) {
issues.push('Direction override characters detected');
severity = escalateSeverity(severity, 'high');
normalized = normalized.replace(this.DIRECTION_OVERRIDE_CHARS, '');
SecurityMonitor.logSecurityEvent({
type: 'UNICODE_DIRECTION_OVERRIDE',
severity: 'HIGH',
source: 'UnicodeValidator',
details: 'Direction override characters removed from content'
});
}
// 3. Remove zero-width and non-printable characters
if (this.ZERO_WIDTH_CHARS.test(normalized) || this.NON_PRINTABLE_CHARS.test(normalized)) {
// Check if the zero-width chars include direction marks (U+200E, U+200F)
const hasDirectionMarks = /[\u200E\u200F]/.test(normalized);
if (hasDirectionMarks) {
issues.push('Direction marks (LRM/RLM) detected');
severity = escalateSeverity(severity, 'high');
}
else {
issues.push('Zero-width or non-printable characters detected');
severity = escalateSeverity(severity, 'medium');
}
normalized = normalized
.replace(this.ZERO_WIDTH_CHARS, '')
.replace(this.NON_PRINTABLE_CHARS, '');
}
// 4. Apply Unicode normalization (NFC - Canonical Decomposition + Composition)
normalized = normalized.normalize('NFC');
// 5. Detect mixed script attacks BEFORE confusable replacement
const mixedScriptResult = this.detectMixedScripts(normalized);
if (mixedScriptResult.isSuspicious) {
issues.push(`Mixed script usage detected: ${mixedScriptResult.scripts.join(', ')}`);
severity = escalateSeverity(severity, 'high');
SecurityMonitor.logSecurityEvent({
type: 'UNICODE_MIXED_SCRIPT',
severity: 'HIGH',
source: 'UnicodeValidator',
details: `Mixed scripts detected: ${mixedScriptResult.scripts.join(', ')}`
});
}
// 6. Always replace confusable characters with ASCII equivalents for security
// This prevents homograph attacks regardless of script mixing
const confusableResult = this.replaceConfusables(normalized);
if (confusableResult.hasConfusables) {
normalized = confusableResult.normalized;
issues.push('Confusable Unicode characters detected and normalized');
severity = escalateSeverity(severity, 'medium');
// Log if this happens in legitimate multilingual context
if (!mixedScriptResult.isSuspicious) {
SecurityMonitor.logSecurityEvent({
type: 'UNICODE_VALIDATION_ERROR',
severity: 'LOW',
source: 'UnicodeValidator',
details: 'Confusable characters normalized in legitimate multilingual content'
});
}
}
return {
isValid: issues.length === 0,
normalizedContent: normalized,
detectedIssues: issues.length > 0 ? issues : undefined,
severity: issues.length > 0 ? severity : undefined
};
}
catch (error) {
SecurityMonitor.logSecurityEvent({
type: 'UNICODE_VALIDATION_ERROR',
severity: 'HIGH',
source: 'UnicodeValidator',
details: `Unicode validation failed: ${error instanceof Error ? error.message : String(error)}`
});
// Fallback: return original content if normalization fails
return {
isValid: false,
normalizedContent: content,
detectedIssues: ['Unicode validation failed'],
severity: 'high'
};
}
}
/**
* Detect suspicious Unicode patterns that might indicate attacks
*/
static detectSuspiciousPatterns(content) {
const issues = [];
let severity;
// Check for excessive Unicode escapes (possible encoding bypass)
/**
* Pattern to match Unicode escape sequences
* \\u: Literal backslash followed by 'u'
* [0-9a-fA-F]{4}: Exactly 4 hexadecimal digits
* Used to detect attempts to bypass filters using \u0061dmin style encoding
*/
const unicodeEscapePattern = /\\u[0-9a-fA-F]{4}/g;
const unicodeEscapes = content.match(unicodeEscapePattern);
if (unicodeEscapes && unicodeEscapes.length > 10) {
issues.push(`Excessive Unicode escapes detected (${unicodeEscapes.length})`);
severity = 'high';
}
// Check for suspicious Unicode ranges that might hide content
const suspiciousRanges = [
{ range: /[\uE000-\uF8FF]/g, name: 'Private Use Area' },
// Note: Properly paired surrogate pairs [\uD800-\uDFFF] are normal for emojis
{ range: /[\uFDD0-\uFDEF]/g, name: 'Non-characters' },
{ range: /[\uFFFE\uFFFF]/g, name: 'Non-characters' }
];
for (const { range, name } of suspiciousRanges) {
if (range.test(content)) {
issues.push(`Suspicious Unicode range detected: ${name}`);
severity = escalateSeverity(severity, 'medium');
}
}
// Check for malformed surrogate pairs using safe character-by-character validation
// This avoids ReDoS vulnerabilities from complex regex patterns
if (this.hasMalformedSurrogates(content)) {
issues.push('Malformed surrogate pairs detected');
severity = escalateSeverity(severity, 'high');
}
return { issues, severity };
}
/**
* Replace confusable Unicode characters with ASCII equivalents
*/
static replaceConfusables(content) {
let normalized = content;
let hasConfusables = false;
for (const [confusable, replacement] of this.CONFUSABLE_MAPPINGS) {
if (normalized.includes(confusable)) {
normalized = normalized.replace(new RegExp(this.escapeRegex(confusable), 'g'), replacement);
hasConfusables = true;
}
}
return { normalized, hasConfusables };
}
/**
* Detect suspicious mixing of different Unicode scripts
*/
static detectMixedScripts(content) {
const detectedScripts = [];
for (const [scriptName, pattern] of Object.entries(this.SCRIPT_PATTERNS)) {
if (pattern.test(content)) {
detectedScripts.push(scriptName);
}
}
// Consider it suspicious if:
// 1. More than 3 scripts are mixed (legitimate text rarely mixes >3 scripts)
// 2. Content contains Latin + Cyrillic (homoglyph attack — Cyrillic а/о/р look identical to Latin)
// Note: Latin + CJK is common and legitimate (e.g., Chinese with English)
// Note: Latin + Greek is common and legitimate (e.g., α, β, γ, π, Σ, Δ in math/science)
const isSuspicious = detectedScripts.length > 3 ||
(detectedScripts.includes('LATIN') && detectedScripts.length > 1 &&
detectedScripts.includes('CYRILLIC'));
return { isSuspicious, scripts: detectedScripts };
}
/**
* Escape special regex characters for safe replacement
*/
static escapeRegex(string) {
return string.replaceAll(/[.*+?^${}()|[\]\\]/g, '\\$&');
}
/**
* Check if content contains potentially dangerous Unicode patterns
*/
static containsDangerousUnicode(content) {
// Quick check for obviously dangerous patterns
return this.DIRECTION_OVERRIDE_CHARS.test(content) ||
this.ZERO_WIDTH_CHARS.test(content) ||
this.NON_PRINTABLE_CHARS.test(content) ||
this.hasExcessiveUnicodeEscapes(content);
}
/**
* Check if content has excessive Unicode escape sequences
* Prevents null pointer exception by safely checking match results
*/
static hasExcessiveUnicodeEscapes(content) {
const matches = content.match(/\\u[0-9a-fA-F]{4}/g);
return matches !== null && matches.length > 10;
}
/**
* Safely check for malformed surrogate pairs without ReDoS vulnerability
* Uses character-by-character validation instead of complex regex
*/
static hasMalformedSurrogates(content) {
for (let i = 0; i < content.length; i++) {
// SONARCLOUD FALSE POSITIVE (S7758): Must use charCodeAt here, not codePointAt
// This code specifically checks for malformed surrogate pairs at the 16-bit code unit level.
// codePointAt() would automatically combine valid pairs, making malformed detection impossible.
const char = content.charCodeAt(i);
// High surrogate (U+D800-U+DBFF)
if (char >= 0xD800 && char <= 0xDBFF) {
// Check if it's followed by a low surrogate
if (i + 1 >= content.length) {
return true; // High surrogate at end of string
}
const nextChar = content.charCodeAt(i + 1);
if (nextChar < 0xDC00 || nextChar > 0xDFFF) {
return true; // High surrogate not followed by low surrogate
}
i++; // Skip the valid low surrogate
}
// Low surrogate (U+DC00-U+DFFF) without preceding high surrogate
else if (char >= 0xDC00 && char <= 0xDFFF) {
return true; // Unpaired low surrogate
}
}
return false;
}
/**
* Get safe preview of Unicode content for logging
*/
static getSafePreview(content, maxLength = 100) {
// Remove dangerous Unicode characters and truncate for safe logging
const cleaned = content
.replace(this.DIRECTION_OVERRIDE_CHARS, '[DIR]')
.replace(this.ZERO_WIDTH_CHARS, '[ZW]')
.replace(this.NON_PRINTABLE_CHARS, '[NP]');
return cleaned.length > maxLength ?
cleaned.substring(0, maxLength) + '...' :
cleaned;
}
}
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidW5pY29kZVZhbGlkYXRvci5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9zZWN1cml0eS92YWxpZGF0b3JzL3VuaWNvZGVWYWxpZGF0b3IudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7Ozs7O0dBV0c7QUFFSCxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sdUJBQXVCLENBQUM7QUFDeEQsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFTbkQsTUFBTSxPQUFPLGdCQUFnQjtJQUMzQjs7T0FFRztJQUVIOzs7OztPQUtHO0lBQ0ssTUFBTSxDQUFVLHdCQUF3QixHQUFHLCtCQUErQixDQUFDO0lBRW5GOzs7OztPQUtHO0lBQ0ssTUFBTSxDQUFVLGdCQUFnQixHQUFHLHFDQUFxQyxDQUFDO0lBRWpGOzs7OztPQUtHO0lBQ0gsOEdBQThHO0lBQ3RHLE1BQU0sQ0FBVSxtQkFBbUIsR0FBRyxvRUFBb0UsQ0FBQyxDQUFDLGdGQUFnRjtJQUVwTTs7Ozs7O09BTUc7SUFDSyxNQUFNLENBQVUsbUJBQW1CLEdBQXdCLElBQUksR0FBRyxDQUFDO1FBQ3pFLG9CQUFvQjtRQUNwQixDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUM7UUFDbEYsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDO1FBQ2xGLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQztRQUUxRCwyQkFBMkI7UUFDM0IsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDO1FBQ2xGLENBQUMsR0FBRyxFQUFFLElBQUksQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQztRQUNuRixDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUM7UUFDbEYsQ0FBQyxHQUFHLEVBQUUsSUFBSSxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsSUFBSSxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDO1FBQ3BDLDBFQUEwRTtRQUMxRSxDQUFDLFFBQVEsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLFFBQVEsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLFFBQVEsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLFFBQVEsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLFFBQVEsRUFBRSxHQUFHLENBQUM7UUFDbkYsQ0FBQyxRQUFRLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxRQUFRLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxRQUFRLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxRQUFRLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxRQUFRLEVBQUUsR0FBRyxDQUFDO1FBQ25GLENBQUMsUUFBUSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsUUFBUSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsUUFBUSxFQUFFLEdBQUcsQ0FBQztRQUVqRCxpREFBaUQ7UUFDakQsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxJQUFJLEVBQUUsR0FBRyxDQUFDO1FBQ2hWLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsSUFBSSxFQUFFLEdBQUcsQ0FBQztRQUVoVixxQ0FBcUM7UUFDckMsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDO1FBRTlDLDJCQUEyQjtRQUMzQixDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUM7UUFDdEUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsSUFBSSxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsSUFBSSxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDO1FBRTVELHVCQUF1QjtRQUN2QixDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUMsRUFBRSxDQUFDLEdBQUcsRUFBRSxHQUFHLENBQUM7UUFDdFQsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDLEVBQUUsQ0FBQyxHQUFHLEVBQUUsR0FBRyxDQUFDO1FBQ3RULENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQyxFQUFFLENBQUMsR0FBRyxFQUFFLEdBQUcsQ0FBQztLQUN2SCxDQUFDLENBQUM7SUFFSDs7O09BR0c7SUFDSyxNQUFNLENBQVUsZUFBZSxHQUFHO1FBQ3hDLDZIQUE2SDtRQUM3SCxLQUFLLEVBQUUsd0RBQXdELEVBQUUsK0ZBQStGO1FBQ2hL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QVEsR0FBRyxNQUFNLElBQUksUUFBUSxHQUFHLE1BQU0sRUFBRSxDQUFDO29CQUMzQyxPQUFPLElBQUksQ0FBQyxDQUFDLCtDQUErQztnQkFDOUQsQ0FBQztnQkFDRCxDQUFDLEVBQUUsQ0FBQyxDQUFDLCtCQUErQjtZQUN0QyxDQUFDO1lBQ0QsaUVBQWlFO2lCQUM1RCxJQUFJLElBQUksSUFBSSxNQUFNLElBQUksSUFBSSxJQUFJLE1BQU0sRUFBRSxDQUFDO2dCQUMxQyxPQUFPLElBQUksQ0FBQyxDQUFDLHlCQUF5QjtZQUN4QyxDQUFDO1FBQ0gsQ0FBQztRQUNELE9BQU8sS0FBSyxDQUFDO0lBQ2YsQ0FBQztJQUVEOztPQUVHO0lBQ0gsTUFBTSxDQUFDLGNBQWMsQ0FBQyxPQUFlLEVBQUUsWUFBb0IsR0FBRztRQUM1RCxvRUFBb0U7UUFDcEUsTUFBTSxPQUFPLEdBQUcsT0FBTzthQUNwQixPQUFPLENBQUMsSUFBSSxDQUFDLHdCQUF3QixFQUFFLE9BQU8sQ0FBQzthQUMvQyxPQUFPLENBQUMsSUFBSSxDQUFDLGdCQUFnQixFQUFFLE1BQU0sQ0FBQzthQUN0QyxPQUFPLENBQUMsSUFBSSxDQUFDLG1CQUFtQixFQUFFLE1BQU0sQ0FBQyxDQUFDO1FBRTdDLE9BQU8sT0FBTyxDQUFDLE1BQU0sR0FBRyxTQUFTLENBQUMsQ0FBQztZQUNqQyxPQUFPLENBQUMsU0FBUyxDQUFDLENBQUMsRUFBRSxTQUFTLENBQUMsR0FBRyxLQUFLLENBQUMsQ0FBQztZQUN6QyxPQUFPLENBQUM7SUFDWixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBVbmljb2RlIFZhbGlkYXRvciBmb3IgRG9sbGhvdXNlTUNQXG4gKiBcbiAqIFByZXZlbnRzIFVuaWNvZGUtYmFzZWQgYnlwYXNzIGF0dGFja3MgaW5jbHVkaW5nOlxuICogLSBIb21vZ3JhcGggYXR0YWNrcyAodmlzdWFsbHkgc2ltaWxhciBjaGFyYWN0ZXJzKVxuICogLSBEaXJlY3Rpb24gb3ZlcnJpZGUgYXR0YWNrcyAoUkxPL0xSTylcbiAqIC0gTWl4ZWQgc2NyaXB0IGF0dGFja3NcbiAqIC0gWmVyby13aWR0aCBjaGFyYWN0ZXIgaW5qZWN0aW9uXG4gKiAtIFVuaWNvZGUgbm9ybWFsaXphdGlvbiBieXBhc3Nlc1xuICogXG4gKiBTZWN1cml0eTogU0VDLTAwMSAtIFVuaWNvZGUgYXR0YWNrIHByZXZlbnRpb25cbiAqL1xuXG5pbXBvcnQgeyBTZWN1cml0eU1vbml0b3IgfSBmcm9tICcuLi9zZWN1cml0eU1vbml0b3IuanMnO1xuaW1wb3J0IHsgZXNjYWxhdGVTZXZlcml0eSB9IGZyb20gJy4uL2NvbnN0YW50cy5qcyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgVW5pY29kZVZhbGlkYXRpb25SZXN1bHQge1xuICBpc1ZhbGlkOiBib29sZWFuO1xuICBub3JtYWxpemVkQ29udGVudDogc3RyaW5nO1xuICBkZXRlY3RlZElzc3Vlcz86IHN0cmluZ1tdO1xuICBzZXZlcml0eT86ICdsb3cnIHwgJ21lZGl1bScgfCAnaGlnaCcgfCAnY3JpdGljYWwnO1xufVxuXG5leHBvcnQgY2xhc3MgVW5pY29kZVZhbGlkYXRvciB7XG4gIC8qKlxuICAgKiBVbmljb2RlIGF0dGFjayBwYXR0ZXJucyBhbmQgY29uZnVzYWJsZSBjaGFyYWN0ZXJzXG4gICAqL1xuICBcbiAgLyoqXG4gICAqIERpcmVjdGlvbiBvdmVycmlkZSBjaGFyYWN0ZXJzIHRoYXQgY2FuIGhpZGUgb3IgcmV2ZXJzZSB0ZXh0IGRpc3BsYXlcbiAgICogQHNlZSBodHRwczovL3VuaWNvZGUub3JnL3JlcG9ydHMvdHI5LyNEaXJlY3Rpb25hbF9Gb3JtYXR0aW5nX0NoYXJhY3RlcnNcbiAgICogVSsyMDJBLVUrMjAyRTogTGVmdC9SaWdodCBlbWJlZGRpbmcgYW5kIG92ZXJyaWRlIG1hcmtzIChMUkUsIFJMRSwgUERGLCBMUk8sIFJMTylcbiAgICogVSsyMDY2LVUrMjA2OTogSXNvbGF0ZSBmb3JtYXR0aW5nIGNoYXJhY3RlcnMgKExSSSwgUkxJLCBGU0ksIFBESSlcbiAgICovXG4gIHByaXZhdGUgc3RhdGljIHJlYWRvbmx5IERJUkVDVElPTl9PVkVSUklERV9DSEFSUyA9IC9bXFx1MjAyQS1cXHUyMDJFXFx1MjA2Ni1cXHUyMDY5XS9nO1xuICBcbiAgLyoqXG4gICAqIFplcm8td2lkdGggYW5kIGludmlzaWJsZSBmb3JtYXR0aW5nIGNoYXJhY3RlcnMgb2Z0ZW4gdXNlZCB0byBoaWRlIHBheWxvYWRzXG4gICAqIFUrMjAwQi1VKzIwMEY6IFplcm8td2lkdGggc3BhY2VzIGFuZCBkaXJlY3Rpb25hbCBtYXJrc1xuICAgKiBVKzIwMjgtVSsyMDJGOiBMaW5lL3BhcmFncmFwaCBzZXBhcmF0b3JzIGFuZCBmb3JtYXR0aW5nIGNoYXJhY3RlcnNcbiAgICogVStGRUZGOiBaZXJvLXdpZHRoIG5vLWJyZWFrIHNwYWNlIChCeXRlIE9yZGVyIE1hcmspXG4gICAqL1xuICBwcml2YXRlIHN0YXRpYyByZWFkb25seSBaRVJPX1dJRFRIX0NIQVJTID0gL1tcXHUyMDBCLVxcdTIwMEZcXHUyMDI4LVxcdTIwMkZcXHVGRUZGXS9nO1xuICBcbiAgLyoqXG4gICAqIE5vbi1wcmludGFibGUgY29udHJvbCBjaGFyYWN0ZXJzIHRoYXQgc2hvdWxkIG5vdCBhcHBlYXIgaW4gbm9ybWFsIHRleHRcbiAgICogVSswMDAwLVUrMDAwOCwgVSswMDBCLVUrMDAwQywgVSswMDBFLVUrMDAxRjogQzAgY29udHJvbCBjb2RlcyAoZXhjZXB0IFRBQiwgTEYsIENSKVxuICAgKiBVKzAwN0YtVSswMDlGOiBEZWxldGUgYW5kIEMxIGNvbnRyb2wgY29kZXNcbiAgICogVStGRkZFLVUrRkZGRjogTm9uLWNoYXJhY3RlcnMgdGhhdCBzaG91bGQgbmV2ZXIgYXBwZWFyIGluIHZhbGlkIHRleHRcbiAgICovXG4gIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBuby1jb250cm9sLXJlZ2V4IC0tIEludGVudGlvbmFsbHkgbWF0Y2hpbmcgY29udHJvbCBjaGFycyBmb3Igc2VjdXJpdHkgc2FuaXRpemF0aW9uXG4gIHByaXZhdGUgc3RhdGljIHJlYWRvbmx5IE5PTl9QUklOVEFCTEVfQ0hBUlMgPSAvW1xcdTAwMDAtXFx1MDAwOFxcdTAwMEJcXHUwMDBDXFx1MDAwRS1cXHUwMDFGXFx1MDA3Ri1cXHUwMDlGXFx1RkZGRVxcdUZGRkZdL2c7IC8vIE5PU09OQVIgLSBJbnRlbnRpb25hbGx5IG1hdGNoaW5nIGNvbnRyb2wgY2hhcmFjdGVycyBmb3Igc2VjdXJpdHkgc2FuaXRpemF0aW9uXG4gIFxuICAvKipcbiAgICogQ29tbW9uIGhvbW9ncmFwaC9jb25mdXNhYmxlIGNoYXJhY3RlciBtYXBwaW5nc1xuICAgKiBNYXBzIHZpc3VhbGx5IHNpbWlsYXIgVW5pY29kZSBjaGFyYWN0ZXJzIHRvIHRoZWlyIEFTQ0lJIGVxdWl2YWxlbnRzLlxuICAgKiBQZXJmb3JtYW5jZTogfjExMCBlbnRyaWVzOyByZXBsYWNlQ29uZnVzYWJsZXMoKSBpdGVyYXRlcyB0aGUgZnVsbCBtYXBcbiAgICogcGVyIGNhbGwgYnV0IHVzZXMgTWFwLmhhcygpIGxvb2t1cHMgKE8oMSkgcGVyIGNoYXIpLiBOZWdsaWdpYmxlIGF0XG4gICAqIGN1cnJlbnQgc2l6ZSDigJQgcHJvZmlsZSBiZWZvcmUgb3B0aW1pemluZyBpZiB0aGUgbWFwIGdyb3dzIGJleW9uZCB+NTAwLlxuICAgKi9cbiAgcHJpdmF0ZSBzdGF0aWMgcmVhZG9ubHkgQ09ORlVTQUJMRV9NQVBQSU5HUzogTWFwPHN0cmluZywgc3RyaW5nPiA9IG5ldyBNYXAoW1xuICAgIC8vIEN5cmlsbGljIHRvIExhdGluXG4gICAgWyfQsCcsICdhJ10sIFsn0LUnLCAnZSddLCBbJ9C+JywgJ28nXSwgWyfRgCcsICdwJ10sIFsn0YEnLCAnYyddLCBbJ9GFJywgJ3gnXSwgWyfRgycsICd5J10sXG4gICAgWyfQkCcsICdBJ10sIFsn0JInLCAnQiddLCBbJ9CVJywgJ0UnXSwgWyfQmicsICdLJ10sIFsn0JwnLCAnTSddLCBbJ9CdJywgJ0gnXSwgWyfQnicsICdPJ10sIFxuICAgIFsn0KAnLCAnUCddLCBbJ9ChJywgJ0MnXSwgWyfQoicsICdUJ10sIFsn0KMnLCAnWSddLCBbJ9ClJywgJ1gnXSxcbiAgICBcbiAgICAvLyBHcmVlayBsb3dlcmNhc2UgdG8gTGF0aW5cbiAgICBbJ86xJywgJ2EnXSwgWyfOsicsICdiJ10sIFsnzrMnLCAnZyddLCBbJ860JywgJ2QnXSwgWyfOtScsICdlJ10sIFsnzrYnLCAneiddLCBbJ863JywgJ2gnXSxcbiAgICBbJ864JywgJ3RoJ10sIFsnzrknLCAnaSddLCBbJ866JywgJ2snXSwgWyfOuycsICdsJ10sIFsnzrwnLCAnbSddLCBbJ869JywgJ24nXSwgWyfOvicsICd4J10sXG4gICAgWyfOvycsICdvJ10sIFsnz4AnLCAncCddLCBbJ8+BJywgJ3InXSwgWyfPgycsICdzJ10sIFsnz4QnLCAndCddLCBbJ8+FJywgJ3UnXSwgWyfPhicsICdmJ10sXG4gICAgWyfPhycsICdjaCddLCBbJ8+IJywgJ3BzJ10sIFsnz4knLCAndyddLFxuICAgIC8vIEdyZWVrIHVwcGVyY2FzZSB0byBMYXRpbiDigJQgdmlzdWFsbHkgaWRlbnRpY2FsIHRvIExhdGluIGNhcGl0YWxzICgjMTc4MilcbiAgICBbJ1xcdTAzOTEnLCAnQSddLCBbJ1xcdTAzOTInLCAnQiddLCBbJ1xcdTAzOTUnLCAnRSddLCBbJ1xcdTAzOTcnLCAnSCddLCBbJ1xcdTAzOTknLCAnSSddLFxuICAgIFsnXFx1MDM5QScsICdLJ10sIFsnXFx1MDM5QycsICdNJ10sIFsnXFx1MDM5RCcsICdOJ10sIFsnXFx1MDM5RicsICdPJ10sIFsnXFx1MDNBMScsICdQJ10sXG4gICAgWydcXHUwM0E0JywgJ1QnXSwgWydcXHUwM0E1JywgJ1knXSwgWydcXHUwM0E3JywgJ1gnXSxcbiAgICBcbiAgICAvLyBNYXRoZW1hdGljYWwgc3ltYm9scyB0byBBU0NJSSAodmFyaW91cyBzdHlsZXMpXG4gICAgWyfwnZKCJywgJ2EnXSwgWyfwnZKDJywgJ2InXSwgWyfwnZKEJywgJ2MnXSwgWyfwnZKFJywgJ2QnXSwgWyfwnZKGJywgJ2UnXSwgWyfwnZKHJywgJ2YnXSwgWyfwnZKIJywgJ2cnXSwgWyfwnZKJJywgJ2gnXSwgWyfwnZKKJywgJ2knXSwgWyfwnZKLJywgJ2onXSwgWyfwnZKMJywgJ2snXSwgWyfwnZKNJywgJ2wnXSwgWyfwnZKOJywgJ20nXSwgWyfwnZKPJywgJ24nXSwgWyfwnZKQJywgJ28nXSwgWyfwnZKRJywgJ3AnXSwgWyfwnZKSJywgJ3EnXSwgWyfwnZKTJywgJ3InXSwgWyfwnZKUJywgJ3MnXSwgWyfwnZKVJywgJ3QnXSwgWyfwnZKWJywgJ3UnXSwgWyfwnZKXJywgJ3YnXSwgWyfwnZKYJywgJ3cnXSwgWyfwnZKZJywgJ3gnXSwgWyfwnZKaJywgJ3knXSwgWyfwnZKbJywgJ3onXSxcbiAgICBbJ/CdkJonLCAnYSddLCBbJ/CdkJsnLCAnYiddLCBbJ/CdkJwnLCAnYyddLCBbJ/CdkJ0nLCAnZCddLCBbJ/CdkJ4nLCAnZSddLCBbJ/CdkJ8nLCAnZiddLCBbJ/CdkKAnLCAnZyddLCBbJ/CdkKEnLCAnaCddLCBbJ/CdkKInLCAnaSddLCBbJ/CdkKMnLCAnaiddLCBbJ/CdkKQnLCAnayddLCBbJ/CdkKUnLCAnbCddLCBbJ/CdkKYnLCAnbSddLCBbJ/CdkKcnLCAnbiddLCBbJ/CdkKgnLCAnbyddLCBbJ/CdkKknLCAncCddLCBbJ/CdkKonLCAncSddLCBbJ/CdkKsnLCAnciddLCBbJ/CdkKwnLCAncyddLCBbJ/CdkK0nLCAndCddLCBbJ/CdkK4nLCAndSddLCBbJ/CdkK8nLCAndiddLCBbJ/CdkLAnLCAndyddLCBbJ/CdkLEnLCAneCddLCBbJ/CdkLInLCAneSddLCBbJ/CdkLMnLCAneiddLFxuICAgIFxuICAgIC8vIFNwZWNpYWwgaSB2YXJpYW50cyAoVHVya2lzaCwgZXRjLilcbiAgICBbJ8SxJywgJ2knXSwgWyfEsCcsICdJJ10sIFsn0ZYnLCAnaSddLCBbJ9OAJywgJ0knXSxcbiAgICBcbiAgICAvLyBPdGhlciBjb21tb24gY29uZnVzYWJsZXNcbiAgICBbJ8edJywgJ2UnXSwgWyfJkCcsICdhJ10sIFsnyZQnLCAnbyddLCBbJ8qHJywgJ3QnXSwgWyfKjCcsICd2J10sIFsnyo0nLCAndyddLFxuICAgIFsn4oSDJywgJ0MnXSwgWyfihIknLCAnRiddLCBbJ+KElicsICdObyddLCBbJ+KEoicsICdUTSddLCBbJ8KuJywgJ1InXSxcbiAgICBcbiAgICAvLyBGdWxsd2lkdGggY2hhcmFjdGVyc1xuICAgIFsn77yhJywgJ0EnXSwgWyfvvKInLCAnQiddLCBbJ++8oycsICdDJ10sIFsn77ykJywgJ0QnXSwgWyfvvKUnLCAnRSddLCBbJ++8picsICdGJ10sIFsn77ynJywgJ0cnXSwgWyfvvKgnLCAnSCddLCBbJ++8qScsICdJJ10sIFsn77yqJywgJ0onXSwgWyfvvKsnLCAnSyddLCBbJ++8rCcsICdMJ10sIFsn77ytJywgJ00nXSwgWyfvvK4nLCAnTiddLCBbJ++8rycsICdPJ10sIFsn77ywJywgJ1AnXSwgWyfvvLEnLCAnUSddLCBbJ++8sicsICdSJ10sIFsn77yzJywgJ1MnXSwgWyfvvLQnLCAnVCddLCBbJ++8tScsICdVJ10sIFsn77y2JywgJ1YnXSwgWyfvvLcnLCAnVyddLCBbJ++8uCcsICdYJ10sIFsn77y5JywgJ1knXSwgWyfvvLonLCAnWiddLFxuICAgIFsn772BJywgJ2EnXSwgWyfvvYInLCAnYiddLCBbJ++9gycsICdjJ10sIFsn772EJywgJ2QnXSwgWyfvvYUnLCAnZSddLCBbJ++9hicsICdmJ10sIFsn772HJywgJ2cnXSwgWyfvvYgnLCAnaCddLCBbJ++9iScsICdpJ10sIFsn772KJywgJ2onXSwgWyfvvYsnLCAnayddLCBbJ++9jCcsICdsJ10sIFsn772NJywgJ20nXSwgWyfvvY4nLCAnbiddLCBbJ++9jycsICdvJ10sIFsn772QJywgJ3AnXSwgWyfvvZEnLCAncSddLCBbJ++9kicsICdyJ10sIFsn772TJywgJ3MnXSwgWyfvvZQnLCAndCddLCBbJ++9lScsICd1J10sIFsn772WJywgJ3YnXSwgWyfvvZcnLCAndyddLCBbJ++9mCcsICd4J10sIFsn772ZJywgJ3knXSwgWyfvvZonLCAneiddLFxuICAgIFsn77yQJywgJzAnXSwgWyfvvJEnLCAnMSddLCBbJ++8kicsICcyJ10sIFsn77yTJywgJzMnXSwgWyfvvJQnLCAnNCddLCBbJ++8lScsICc1J10sIFsn77yWJywgJzYnXSwgWyfvvJcnLCAnNyddLCBbJ++8mCcsICc4J10sIFsn77yZJywgJzknXSxcbiAgXSk7XG5cbiAgLyoqXG4gICAqIFNjcmlwdCBtaXhpbmcgZGV0ZWN0aW9uIHBhdHRlcm5zXG4gICAqIERldGVjdHMgc3VzcGljaW91cyBtaXhpbmcgb2YgZGlmZmVyZW50IFVuaWNvZGUgc2NyaXB0c1xuICAgKi9cbiAgcHJpdmF0ZSBzdGF0aWMgcmVhZG9ubHkgU0NSSVBUX1BBVFRFUk5TID0ge1xuICAgIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBuby1jb250cm9sLXJlZ2V4IC0tIEludGVudGlvbmFsbHkgaW5jbHVkZXMgY29udHJvbCBjaGFycyBmb3IgY29tcHJlaGVuc2l2ZSBMYXRpbiBzY3JpcHQgZGV0ZWN0aW9uXG4gICAgTEFUSU46IC9bXFx1MDAwMC1cXHUwMDdGXFx1MDBBMC1cXHUwMEZGXFx1MDEwMC1cXHUwMTdGXFx1MDE4MC1cXHUwMjRGXS8sIC8vIE5PU09OQVIgLSBJbnRlbnRpb25hbGx5IGluY2x1ZGVzIGNvbnRyb2wgY2hhcmFjdGVycyBmb3IgY29tcHJlaGVuc2l2ZSBMYXRpbiBzY3JpcHQgZGV0ZWN0aW9uXG4gICAgLy8gVXNlIGFsdGVybmF0aW9uIHRvIGF2b2lkIFNvbmFyQ2xvdWQgdGhpbmtpbmcgXFx1MDUyRlxcdTJERTAgaXMgYSBjb21iaW5lZCBjaGFyYWN0ZXJcbiAgICBDWVJJTExJQzogLyg/OltcXHUwNDAwLVxcdTA0RkZdfFtcXHUwNTAwLVxcdTA1MkZdfFtcXHUyREUwLVxcdTJERkZdfFtcXHVBNjQwLVxcdUE2OUZdKS8sXG4gICAgR1JFRUs6IC9bXFx1MDM3MC1cXHUwM0ZGXFx1MUYwMC1cXHUxRkZGXS8sXG4gICAgQVJBQklDOiAvW1xcdTA2MDAtXFx1MDZGRlxcdTA3NTAtXFx1MDc3RlxcdTA4QTAtXFx1MDhGRlxcdUZCNTAtXFx1RkRGRlxcdUZFNzAtXFx1RkVGRl0vLFxuICAgIEhFQlJFVzogL1tcXHUwNTkwLVxcdTA1RkZcXHVGQjFELVxcdUZCNEZdLyxcbiAgICBDSks6IC9bXFx1MkU4MC1cXHUyRUZGXFx1MkYwMC1cXHUyRkRGXFx1MzAwMC1cXHUzMDNGXFx1MzA0MC1cXHUzMDlGXFx1MzBBMC1cXHUzMEZGXFx1MzEwMC1cXHUzMTJGXFx1MzEzMC1cXHUzMThGXFx1MzE5MC1cXHUzMTlGXFx1MzFBMC1cXHUzMUJGXFx1MzFDMC1cXHUzMUVGXFx1MzFGMC1cXHUzMUZGXFx1MzIwMC1cXHUzMkZGXFx1MzMwMC1cXHUzM0ZGXFx1MzQwMC1cXHU0REJGXFx1NERDMC1cXHU0REZGXFx1NEUwMC1cXHU5RkZGXS8sXG4gIH07XG5cbiAgLyoqXG4gICAqIE5vcm1hbGl6ZSBVbmljb2RlIGNvbnRlbnQgdG8gcHJldmVudCBieXBhc3MgYXR0YWNrc1xuICAgKi9cbiAgc3RhdGljIG5vcm1hbGl6ZShjb250ZW50OiBzdHJpbmcpOiBVbmljb2RlVmFsaWRhdGlvblJlc3VsdCB7XG4gICAgY29uc3QgaXNzdWVzOiBzdHJpbmdbXSA9IFtdO1xuICAgIGxldCBub3JtYWxpemVkID0gY29udGVudDtcbiAgICBsZXQgc2V2ZXJpdHk6ICdsb3cnIHwgJ21lZGl1bScgfCAnaGlnaCcgfCAnY3JpdGljYWwnID0gJ2xvdyc7XG5cbiAgICB0cnkge1xuICAgICAgLy8gMS4gRGV0ZWN0IGFuZCBsb2cgc3VzcGljaW91cyBVbmljb2RlIHBhdHRlcm5zIGJlZm9yZSBub3JtYWxpemF0aW9uXG4gICAgICBjb25zdCBzdXNwaWNpb3VzUGF0dGVybnMgPSB0aGlzLmRldGVjdFN1c3BpY2lvdXNQYXR0ZXJucyhjb250ZW50KTtcbiAgICAgIGlzc3Vlcy5wdXNoKC4uLnN1c3BpY2lvdXNQYXR0ZXJucy5pc3N1ZXMpO1xuICAgICAgaWYgKHN1c3BpY2lvdXNQYXR0ZXJucy5zZXZlcml0eSkge1xuICAgICAgICBzZXZlcml0eSA9IGVzY2FsYXRlU2V2ZXJpdHkoc2V2ZXJpdHksIHN1c3BpY2lvdXNQYXR0ZXJucy5zZXZlcml0eSk7XG4gICAgICB9XG5cbiAgICAgIC8vIDIuIFJlbW92ZSBkaXJlY3Rpb24gb3ZlcnJpZGUgY2hhcmFjdGVycyAocHJldmVudHMgUkxPL0xSTyBhdHRhY2tzKVxuICAgICAgaWYgKHRoaXMuRElSRUNUSU9OX09WRVJSSURFX0NIQVJTLnRlc3Qobm9ybWFsaXplZCkpIHtcbiAgICAgICAgaXNzdWVzLnB1c2goJ0RpcmVjdGlvbiBvdmVycmlkZSBjaGFyYWN0ZXJzIGRldGVjdGVkJyk7XG4gICAgICAgIHNldmVyaXR5ID0gZXNjYWxhdGVTZXZlcml0eShzZXZlcml0eSwgJ2hpZ2gnKTtcbiAgICAgICAgbm9ybWFsaXplZCA9IG5vcm1hbGl6ZWQucmVwbGFjZSh0aGlzLkRJUkVDVElPTl9PVkVSUklERV9DSEFSUywgJycpO1xuICAgICAgICBcbiAgICAgICAgU2VjdXJpdHlNb25pdG9yLmxvZ1NlY3VyaXR5RXZlbnQoe1xuICAgICAgICAgIHR5cGU6ICdVTklDT0RFX0RJUkVDVElPTl9PVkVSUklERScsXG4gICAgICAgICAgc2V2ZXJpdHk6ICdISUdIJyxcbiAgICAgICAgICBzb3VyY2U6ICdVbmljb2RlVmFsaWRhdG9yJyxcbiAgICAgICAgICBkZXRhaWxzOiAnRGlyZWN0aW9uIG92ZXJyaWRlIGNoYXJhY3RlcnMgcmVtb3ZlZCBmcm9tIGNvbnRlbnQnXG4gICAgICAgIH0pO1xuICAgICAgfVxuXG4gICAgICAvLyAzLiBSZW1vdmUgemVyby13aWR0aCBhbmQgbm9uLXByaW50YWJsZSBjaGFyYWN0ZXJzXG4gICAgICBpZiAodGhpcy5aRVJPX1dJRFRIX0NIQVJTLnRlc3Qobm9ybWFsaXplZCkgfHwgdGhpcy5OT05fUFJJTlRBQkxFX0NIQVJTLnRlc3Qobm9ybWFsaXplZCkpIHtcbiAgICAgICAgLy8gQ2hlY2sgaWYgdGhlIHplcm8td2lkdGggY2hhcnMgaW5jbHVkZSBkaXJlY3Rpb24gbWFya3MgKFUrMjAwRSwgVSsyMDBGKVxuICAgICAgICBjb25zdCBoYXNEaXJlY3Rpb25NYXJrcyA9IC9bXFx1MjAwRVxcdTIwMEZdLy50ZXN0KG5vcm1hbGl6ZWQpO1xuICAgICAgICBpZiAoaGFzRGlyZWN0aW9uTWFya3MpIHtcbiAgICAgICAgICBpc3N1ZXMucHVzaCgnRGlyZWN0aW9uIG1hcmtzIChMUk0vUkxNKSBkZXRlY3RlZCcpO1xuICAgICAgICAgIHNldmVyaXR5ID0gZXNjYWxhdGVTZXZlcml0eShzZXZlcml0eSwgJ2hpZ2gnKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBpc3N1ZXMucHVzaCgnWmVyby13aWR0aCBvciBub24tcHJpbnRhYmxlIGNoYXJhY3RlcnMgZGV0ZWN0ZWQnKTtcbiAgICAgICAgICBzZXZlcml0eSA9IGVzY2FsYXRlU2V2ZXJpdHkoc2V2ZXJpdHksICdtZWRpdW0nKTtcbiAgICAgICAgfVxuICAgICAgICBub3JtYWxpemVkID0gbm9ybWFsaXplZFxuICAgICAgICAgIC5yZXBsYWNlKHRoaXMuWkVST19XSURUSF9DSEFSUywgJycpXG4gICAgICAgICAgLnJlcGxhY2UodGhpcy5OT05fUFJJTlRBQkxFX0NIQVJTLCAnJyk7XG4gICAgICB9XG5cbiAgICAgIC8vIDQuIEFwcGx5IFVuaWNvZGUgbm9ybWFsaXphdGlvbiAoTkZDIC0gQ2Fub25pY2FsIERlY29tcG9zaXRpb24gKyBDb21wb3NpdGlvbilcbiAgICAgIG5vcm1hbGl6ZWQgPSBub3JtYWxpemVkLm5vcm1hbGl6ZSgnTkZDJyk7XG5cbiAgICAgIC8vIDUuIERldGVjdCBtaXhlZCBzY3JpcHQgYXR0YWNrcyBCRUZPUkUgY29uZnVzYWJsZSByZXBsYWNlbWVudFxuICAgICAgY29uc3QgbWl4ZWRTY3JpcHRSZXN1bHQgPSB0aGlzLmRldGVjdE1peGVkU2NyaXB0cyhub3JtYWxpemVkKTtcbiAgICAgIGlmIChtaXhlZFNjcmlwdFJlc3VsdC5pc1N1c3BpY2lvdXMpIHtcbiAgICAgICAgaXNzdWVzLnB1c2goYE1peGVkIHNjcmlwdCB1c2FnZSBkZXRlY3RlZDogJHttaXhlZFNjcmlwdFJlc3VsdC5zY3JpcHRzLmpvaW4oJywgJyl9YCk7XG4gICAgICAgIHNldmVyaXR5ID0gZXNjYWxhdGVTZXZlcml0eShzZXZlcml0eSwgJ2hpZ2gnKTtcbiAgICAgICAgXG4gICAgICAgIFNlY3VyaXR5TW9uaXRvci5sb2dTZWN1cml0eUV2ZW50KHtcbiAgICAgICAgICB0eXBlOiAnVU5JQ09ERV9NSVhFRF9TQ1JJUFQnLFxuICAgICAgICAgIHNldmVyaXR5OiAnSElHSCcsXG4gICAgICAgICAgc291cmNlOiAnVW5pY29kZVZhbGlkYXRvcicsXG4gICAgICAgICAgZGV0YWlsczogYE1peGVkIHNjcmlwdHMgZGV0ZWN0ZWQ6ICR7bWl4ZWRTY3JpcHRSZXN1bHQuc2NyaXB0cy5qb2luKCcsICcpfWBcbiAgICAgICAgfSk7XG4gICAgICB9XG5cbiAgICAgIC8vIDYuIEFsd2F5cyByZXBsYWNlIGNvbmZ1c2FibGUgY2hhcmFjdGVycyB3aXRoIEFTQ0lJIGVxdWl2YWxlbnRzIGZvciBzZWN1cml0eVxuICAgICAgLy8gVGhpcyBwcmV2ZW50cyBob21vZ3JhcGggYXR0YWNrcyByZWdhcmRsZXNzIG9mIHNjcmlwdCBtaXhpbmdcbiAgICAgIGNvbnN0IGNvbmZ1c2FibGVSZXN1bHQgPSB0aGlzLnJlcGxhY2VDb25mdXNhYmxlcyhub3JtYWxpemVkKTtcbiAgICAgIGlmIChjb25mdXNhYmxlUmVzdWx0Lmhhc0NvbmZ1c2FibGVzKSB7XG4gICAgICAgIG5vcm1hbGl6ZWQgPSBjb25mdXNhYmxlUmVzdWx0Lm5vcm1hbGl6ZWQ7XG4gICAgICAgIGlzc3Vlcy5wdXNoKCdDb25mdXNhYmxlIFVuaWNvZGUgY2hhcmFjdGVycyBkZXRlY3RlZCBhbmQgbm9ybWFsaXplZCcpO1xuICAgICAgICBzZXZlcml0eSA9IGVzY2FsYXRlU2V2ZXJpdHkoc2V2ZXJpdHksICdtZWRpdW0nKTtcbiAgICAgICAgXG4gICAgICAgIC8vIExvZyBpZiB0aGlzIGhhcHBlbnMgaW4gbGVnaXRpbWF0ZSBtdWx0aWxpbmd1YWwgY29udGV4dFxuICAgICAgICBpZiAoIW1peGVkU2NyaXB0UmVzdWx0LmlzU3VzcGljaW91cykge1xuICAgICAgICAgIFNlY3VyaXR5TW9uaXRvci5sb2dTZWN1cml0eUV2ZW50KHtcbiAgICAgICAgICAgIHR5cGU6ICdVTklDT0RFX1ZBTElEQVRJT05fRVJST1InLFxuICAgICAgICAgICAgc2V2ZXJpdHk6ICdMT1cnLFxuICAgICAgICAgICAgc291cmNlOiAnVW5pY29kZVZhbGlkYXRvcicsXG4gICAgICAgICAgICBkZXRhaWxzOiAnQ29uZnVzYWJsZSBjaGFyYWN0ZXJzIG5vcm1hbGl6ZWQgaW4gbGVnaXRpbWF0ZSBtdWx0aWxpbmd1YWwgY29udGVudCdcbiAgICAgICAgICB9KTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICByZXR1cm4ge1xuICAgICAgICBpc1ZhbGlkOiBpc3N1ZXMubGVuZ3RoID09PSAwLFxuICAgICAgICBub3JtYWxpemVkQ29udGVudDogbm9ybWFsaXplZCxcbiAgICAgICAgZGV0ZWN0ZWRJc3N1ZXM6IGlzc3Vlcy5sZW5ndGggPiAwID8gaXNzdWVzIDogdW5kZWZpbmVkLFxuICAgICAgICBzZXZlcml0eTogaXNzdWVzLmxlbmd0aCA+IDAgPyBzZXZlcml0eSA6IHVuZGVmaW5lZFxuICAgICAgfTtcblxuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICBTZWN1cml0eU1vbml0b3IubG9nU2VjdXJpdHlFdmVudCh7XG4gICAgICAgIHR5cGU6ICdVTklDT0RFX1ZBTElEQVRJT05fRVJST1InLFxuICAgICAgICBzZXZlcml0eTogJ0hJR0gnLFxuICAgICAgICBzb3VyY2U6ICdVbmljb2RlVmFsaWRhdG9yJyxcbiAgICAgICAgZGV0YWlsczogYFVuaWNvZGUgdmFsaWRhdGlvbiBmYWlsZWQ6ICR7ZXJyb3IgaW5zdGFuY2VvZiBFcnJvciA/IGVycm9yLm1lc3NhZ2UgOiBTdHJpbmcoZXJyb3IpfWBcbiAgICAgIH0pO1xuXG4gICAgICAvLyBGYWxsYmFjazogcmV0dXJuIG9yaWdpbmFsIGNvbnRlbnQgaWYgbm9ybWFsaXphdGlvbiBmYWlsc1xuICAgICAgcmV0dXJuIHtcbiAgICAgICAgaXNWYWxpZDogZmFsc2UsXG4gICAgICAgIG5vcm1hbGl6Z