@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.
381 lines • 48.8 kB
JavaScript
/**
* Base abstract class implementing IElement interface.
* Provides common functionality that all element types can extend.
*/
import { ElementStatus } from '../types/elements/index.js';
import { v4 as uuidv4 } from 'uuid';
import { logger } from '../utils/logger.js';
import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
import { SecurityMonitor } from '../security/securityMonitor.js';
export class BaseElement {
// Identity
id;
type;
version;
// Metadata
metadata;
// Features
references;
extensions;
ratings;
// Internal state
_status = ElementStatus.INACTIVE;
_isDirty = false;
// Constants
MAX_FEEDBACK_HISTORY = 100;
constructor(type, metadata = {}) {
this.type = type;
this.id = metadata.name ? this.generateId(metadata.name) : uuidv4();
this.version = metadata.version || '1.0.0';
// Initialize metadata with defaults
this.metadata = {
name: metadata.name || 'Unnamed Element',
description: metadata.description || '',
author: metadata.author,
version: this.version,
created: metadata.created || new Date().toISOString(),
modified: metadata.modified || new Date().toISOString(),
tags: metadata.tags || [],
dependencies: metadata.dependencies || [],
custom: metadata.custom || {}
};
// Initialize optional features
this.references = [];
this.extensions = {};
this.ratings = {
aiRating: 0,
userRating: undefined,
ratingCount: 0,
lastEvaluated: new Date(),
confidence: 0,
trend: 'stable',
feedbackHistory: []
};
}
/**
* Generate a unique ID for the element based on its name and type.
* Format: type_name-slug_timestamp
*/
generateId(name) {
const typeSlug = this.type.toLowerCase();
const nameSlug = name
.toLowerCase()
.replace(/[^a-z0-9]+/g, '-')
.replace(/^-|-$/g, '');
const timestamp = Date.now();
return `${typeSlug}_${nameSlug}_${timestamp}`;
}
/**
* Core validation that all elements share.
* Subclasses should override and call super.validate() first.
*/
validate() {
// Log security-relevant validation event
SecurityMonitor.logSecurityEvent({
type: 'YAML_PARSE_SUCCESS',
severity: 'LOW',
source: 'BaseElement.validate',
details: `Element validation performed for ${this.type}:${this.id}`,
additionalData: { elementType: this.type, elementId: this.id }
});
const errors = [];
const warnings = [];
const suggestions = [];
// Validate required fields
if (!this.id) {
errors.push({ field: 'id', message: 'Element ID is required' });
}
if (!this.metadata.name || this.metadata.name.trim() === '') {
errors.push({ field: 'metadata.name', message: 'Element name is required' });
}
if (!this.metadata.description || this.metadata.description.trim() === '') {
warnings.push({
field: 'metadata.description',
message: 'Element description is recommended',
severity: 'medium'
});
}
// Validate version format (semver)
const semverRegex = /^\d+\.\d+\.\d+(-[a-zA-Z0-9.-]+)?(\+[a-zA-Z0-9.-]+)?$/;
if (!semverRegex.test(this.version)) {
errors.push({
field: 'version',
message: 'Version must follow semantic versioning (e.g., 1.0.0)',
code: 'INVALID_VERSION_FORMAT'
});
}
// Validate references
if (this.references) {
this.references.forEach((ref, index) => {
if (!ref.uri || ref.uri.trim() === '') {
errors.push({
field: `references[${index}].uri`,
message: 'Reference URI is required'
});
}
if (!ref.title || ref.title.trim() === '') {
warnings.push({
field: `references[${index}].title`,
message: 'Reference title is recommended',
severity: 'low'
});
}
});
}
// Validate ratings if present
if (this.ratings) {
if (this.ratings.aiRating < 0 || this.ratings.aiRating > 5) {
errors.push({
field: 'ratings.aiRating',
message: 'AI rating must be between 0 and 5'
});
}
if (this.ratings.userRating !== undefined &&
(this.ratings.userRating < 0 || this.ratings.userRating > 5)) {
errors.push({
field: 'ratings.userRating',
message: 'User rating must be between 0 and 5'
});
}
}
// Add suggestions
if (!this.metadata.tags || this.metadata.tags.length === 0) {
suggestions.push('Consider adding tags to improve discoverability');
}
if (!this.metadata.author) {
suggestions.push('Consider adding author information');
}
return {
valid: errors.length === 0,
errors: errors.length > 0 ? errors : undefined,
warnings: warnings.length > 0 ? warnings : undefined,
suggestions: suggestions.length > 0 ? suggestions : undefined
};
}
/**
* Default serialization to JSON.
* Subclasses can override for custom formats.
*/
serialize() {
const data = {
id: this.id,
type: this.type,
version: this.version,
metadata: this.metadata,
references: this.references,
extensions: this.extensions,
ratings: this.ratings
};
return JSON.stringify(data, null, 2);
}
/**
* Default deserialization from JSON.
* Subclasses can override for custom formats.
*/
deserialize(data) {
try {
// Normalize Unicode input before parsing
const validationResult = UnicodeValidator.normalize(data);
const parsed = JSON.parse(validationResult.normalizedContent);
// Validate required fields
if (!parsed.id || !parsed.type || !parsed.metadata) {
throw new Error('Invalid element data: missing required fields');
}
// Update properties
this.id = parsed.id;
this.type = parsed.type;
this.version = parsed.version || '1.0.0';
this.metadata = parsed.metadata;
this.references = parsed.references || [];
this.extensions = parsed.extensions || {};
this.ratings = parsed.ratings;
this._isDirty = false;
}
catch (error) {
logger.error('Failed to deserialize element', { error, data });
throw new Error(`Failed to deserialize element: ${error instanceof Error ? error.message : 'Unknown error'}`);
}
}
/**
* Process user feedback and update ratings.
*/
receiveFeedback(feedback, context) {
// Normalize Unicode input to prevent security issues
const validationResult = UnicodeValidator.normalize(feedback);
const normalizedFeedback = validationResult.normalizedContent;
// Log security event for feedback processing
SecurityMonitor.logSecurityEvent({
type: 'CONTENT_INJECTION_ATTEMPT',
severity: 'LOW',
source: 'BaseElement.receiveFeedback',
details: `Feedback processed for element ${this.type}:${this.id}`,
additionalData: {
elementType: this.type,
elementId: this.id,
feedbackLength: feedback.length,
hasUnicodeIssues: !validationResult.isValid
}
});
if (!this.ratings) {
this.ratings = {
aiRating: 0,
userRating: undefined,
ratingCount: 0,
lastEvaluated: new Date(),
confidence: 0,
trend: 'stable',
feedbackHistory: []
};
}
// Create feedback entry with normalized content
const userFeedback = {
timestamp: new Date(),
feedback: normalizedFeedback,
sentiment: this.analyzeSentiment(normalizedFeedback),
inferredRating: this.inferRating(normalizedFeedback),
context,
elementVersion: this.version
};
// Add to history with bounds checking
if (!this.ratings.feedbackHistory) {
this.ratings.feedbackHistory = [];
}
this.ratings.feedbackHistory.push(userFeedback);
// Prevent unbounded growth
if (this.ratings.feedbackHistory.length > this.MAX_FEEDBACK_HISTORY) {
this.ratings.feedbackHistory = this.ratings.feedbackHistory.slice(-this.MAX_FEEDBACK_HISTORY);
logger.debug(`Feedback history trimmed to ${this.MAX_FEEDBACK_HISTORY} entries for element ${this.id}`);
}
// Update user rating if we inferred one
if (userFeedback.inferredRating !== undefined) {
this.updateUserRating(userFeedback.inferredRating);
}
this._isDirty = true;
}
/**
* Simple sentiment analysis.
* Subclasses can override for more sophisticated analysis.
*/
analyzeSentiment(feedback) {
const lower = feedback.toLowerCase();
const positiveWords = ['excellent', 'great', 'good', 'helpful', 'useful', 'perfect', 'amazing', 'love'];
const negativeWords = ['bad', 'poor', 'terrible', 'useless', 'broken', 'hate', 'awful', 'disappointing'];
const positiveCount = positiveWords.filter(word => lower.includes(word)).length;
const negativeCount = negativeWords.filter(word => lower.includes(word)).length;
if (positiveCount > negativeCount)
return 'positive';
if (negativeCount > positiveCount)
return 'negative';
return 'neutral';
}
/**
* Simple rating inference from feedback.
* Subclasses can override for more sophisticated inference.
*/
inferRating(feedback) {
const sentiment = this.analyzeSentiment(feedback);
const lower = feedback.toLowerCase();
// Look for explicit ratings
const ratingMatch = lower.match(/(\d+)\s*(stars?|\/5|out of 5)/);
if (ratingMatch) {
const rating = parseInt(ratingMatch[1]);
if (rating >= 1 && rating <= 5)
return rating;
}
// Infer from sentiment
if (sentiment === 'positive') {
if (lower.includes('perfect') || lower.includes('excellent'))
return 5;
if (lower.includes('great') || lower.includes('very good'))
return 4;
return 4;
}
else if (sentiment === 'negative') {
if (lower.includes('terrible') || lower.includes('awful'))
return 1;
if (lower.includes('poor') || lower.includes('bad'))
return 2;
return 2;
}
return 3; // Neutral
}
/**
* Update user rating with a new value.
*/
updateUserRating(newRating) {
if (!this.ratings)
return;
if (this.ratings.userRating === undefined) {
this.ratings.userRating = newRating;
this.ratings.ratingCount = 1;
}
else {
// Calculate running average
const totalRating = this.ratings.userRating * this.ratings.ratingCount + newRating;
this.ratings.ratingCount++;
this.ratings.userRating = totalRating / this.ratings.ratingCount;
}
// Update delta and trend
this.ratings.ratingDelta = this.ratings.userRating - this.ratings.aiRating;
// Simple trend calculation based on recent feedback
const recentFeedback = this.ratings.feedbackHistory?.slice(-5) || [];
const recentSentiments = recentFeedback.map(f => f.sentiment);
const positiveCount = recentSentiments.filter(s => s === 'positive').length;
const negativeCount = recentSentiments.filter(s => s === 'negative').length;
if (positiveCount > negativeCount + 1) {
this.ratings.trend = 'improving';
}
else if (negativeCount > positiveCount + 1) {
this.ratings.trend = 'declining';
}
else {
this.ratings.trend = 'stable';
}
}
/**
* Get current element status.
*/
getStatus() {
return this._status;
}
/**
* Default lifecycle methods - subclasses should override as needed.
*/
async beforeActivate() {
logger.debug(`Preparing to activate ${this.type} element: ${this.metadata.name}`);
this._status = ElementStatus.ACTIVATING;
}
async activate() {
logger.info(`Activating ${this.type} element: ${this.metadata.name}`);
this._status = ElementStatus.ACTIVE;
}
async afterActivate() {
logger.debug(`Completed activation of ${this.type} element: ${this.metadata.name}`);
}
async deactivate() {
logger.info(`Deactivating ${this.type} element: ${this.metadata.name}`);
this._status = ElementStatus.DEACTIVATING;
// Subclasses should implement cleanup logic
this._status = ElementStatus.INACTIVE;
}
/**
* Mark element as modified.
*/
markDirty() {
this._isDirty = true;
this.metadata.modified = new Date().toISOString();
}
/**
* Check if element has unsaved changes.
*/
isDirty() {
return this._isDirty;
}
/**
* Mark element as saved.
*/
markClean() {
this._isDirty = false;
}
}
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiQmFzZUVsZW1lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZWxlbWVudHMvQmFzZUVsZW1lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7OztHQUdHO0FBRUgsT0FBTyxFQUdMLGFBQWEsRUFRZCxNQUFNLDRCQUE0QixDQUFDO0FBRXBDLE9BQU8sRUFBRSxFQUFFLElBQUksTUFBTSxFQUFFLE1BQU0sTUFBTSxDQUFDO0FBQ3BDLE9BQU8sRUFBRSxNQUFNLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQztBQUM1QyxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSw0Q0FBNEMsQ0FBQztBQUM5RSxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sZ0NBQWdDLENBQUM7QUFFakUsTUFBTSxPQUFnQixXQUFXO0lBQy9CLFdBQVc7SUFDSixFQUFFLENBQVM7SUFDWCxJQUFJLENBQWM7SUFDbEIsT0FBTyxDQUFTO0lBRXZCLFdBQVc7SUFDSixRQUFRLENBQW1CO0lBRWxDLFdBQVc7SUFDSixVQUFVLENBQWU7SUFDekIsVUFBVSxDQUF1QjtJQUNqQyxPQUFPLENBQWtCO0lBRWhDLGlCQUFpQjtJQUNQLE9BQU8sR0FBa0IsYUFBYSxDQUFDLFFBQVEsQ0FBQztJQUNoRCxRQUFRLEdBQVksS0FBSyxDQUFDO0lBRXBDLFlBQVk7SUFDSyxvQkFBb0IsR0FBRyxHQUFHLENBQUM7SUFFNUMsWUFBWSxJQUFpQixFQUFFLFdBQXNDLEVBQUU7UUFDckUsSUFBSSxDQUFDLElBQUksR0FBRyxJQUFJLENBQUM7UUFDakIsSUFBSSxDQUFDLEVBQUUsR0FBRyxRQUFRLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxJQUFJLENBQUMsVUFBVSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUMsTUFBTSxFQUFFLENBQUM7UUFDcEUsSUFBSSxDQUFDLE9BQU8sR0FBRyxRQUFRLENBQUMsT0FBTyxJQUFJLE9BQU8sQ0FBQztRQUUzQyxvQ0FB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QUFhLENBQUMsVUFBVSxDQUFDO0lBQzFDLENBQUM7SUFFTSxLQUFLLENBQUMsUUFBUTtRQUNuQixNQUFNLENBQUMsSUFBSSxDQUFDLGNBQWMsSUFBSSxDQUFDLElBQUksYUFBYSxJQUFJLENBQUMsUUFBUSxDQUFDLElBQUksRUFBRSxDQUFDLENBQUM7UUFDdEUsSUFBSSxDQUFDLE9BQU8sR0FBRyxhQUFhLENBQUMsTUFBTSxDQUFDO0lBQ3RDLENBQUM7SUFFTSxLQUFLLENBQUMsYUFBYTtRQUN4QixNQUFNLENBQUMsS0FBSyxDQUFDLDJCQUEyQixJQUFJLENBQUMsSUFBSSxhQUFhLElBQUksQ0FBQyxRQUFRLENBQUMsSUFBSSxFQUFFLENBQUMsQ0FBQztJQUN0RixDQUFDO0lBRU0sS0FBSyxDQUFDLFVBQVU7UUFDckIsTUFBTSxDQUFDLElBQUksQ0FBQyxnQkFBZ0IsSUFBSSxDQUFDLElBQUksYUFBYSxJQUFJLENBQUMsUUFBUSxDQUFDLElBQUksRUFBRSxDQUFDLENBQUM7UUFDeEUsSUFBSSxDQUFDLE9BQU8sR0FBRyxhQUFhLENBQUMsWUFBWSxDQUFDO1FBQzFDLDRDQUE0QztRQUM1QyxJQUFJLENBQUMsT0FBTyxHQUFHLGFBQWEsQ0FBQyxRQUFRLENBQUM7SUFDeEMsQ0FBQztJQUVEOztPQUVHO0lBQ08sU0FBUztRQUNqQixJQUFJLENBQUMsUUFBUSxHQUFHLElBQUksQ0FBQztRQUNyQixJQUFJLENBQUMsUUFBUSxDQUFDLFFBQVEsR0FBRyxJQUFJLElBQUksRUFBRSxDQUFDLFdBQVcsRUFBRSxDQUFDO0lBQ3BELENBQUM7SUFFRDs7T0FFRztJQUNJLE9BQU87UUFDWixPQUFPLElBQUksQ0FBQyxRQUFRLENBQUM7SUFDdkIsQ0FBQztJQUVEOztPQUVHO0lBQ0ksU0FBUztRQUNkLElBQUksQ0FBQyxRQUFRLEdBQUcsS0FBSyxDQUFDO0lBQ3hCLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogQmFzZSBhYnN0cmFjdCBjbGFzcyBpbXBsZW1lbnRpbmcgSUVsZW1lbnQgaW50ZXJmYWNlLlxuICogUHJvdmlkZXMgY29tbW9uIGZ1bmN0aW9uYWxpdHkgdGhhdCBhbGwgZWxlbWVudCB0eXBlcyBjYW4gZXh0ZW5kLlxuICovXG5cbmltcG9ydCB7XG4gIElFbGVtZW50LFxuICBJRWxlbWVudE1ldGFkYXRhLFxuICBFbGVtZW50U3RhdHVzLFxuICBFbGVtZW50UmF0aW5ncyxcbiAgUmVmZXJlbmNlLFxuICBFbGVtZW50VmFsaWRhdGlvblJlc3VsdCxcbiAgVmFsaWRhdGlvbkVycm9yLFxuICBWYWxpZGF0aW9uV2FybmluZyxcbiAgRmVlZGJhY2tDb250ZXh0LFxuICBVc2VyRmVlZGJhY2tcbn0gZnJvbSAnLi4vdHlwZXMvZWxlbWVudHMvaW5kZXguanMnO1xuaW1wb3J0IHsgRWxlbWVudFR5cGUgfSBmcm9tICcuLi9wb3J0Zm9saW8vdHlwZXMuanMnO1xuaW1wb3J0IHsgdjQgYXMgdXVpZHY0IH0gZnJvbSAndXVpZCc7XG5pbXBvcnQgeyBsb2dnZXIgfSBmcm9tICcuLi91dGlscy9sb2dnZXIuanMnO1xuaW1wb3J0IHsgVW5pY29kZVZhbGlkYXRvciB9IGZyb20gJy4uL3NlY3VyaXR5L3ZhbGlkYXRvcnMvdW5pY29kZVZhbGlkYXRvci5qcyc7XG5pbXBvcnQgeyBTZWN1cml0eU1vbml0b3IgfSBmcm9tICcuLi9zZWN1cml0eS9zZWN1cml0eU1vbml0b3IuanMnO1xuXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgQmFzZUVsZW1lbnQgaW1wbGVtZW50cyBJRWxlbWVudCB7XG4gIC8vIElkZW50aXR5XG4gIHB1YmxpYyBpZDogc3RyaW5nO1xuICBwdWJsaWMgdHlwZTogRWxlbWVudFR5cGU7XG4gIHB1YmxpYyB2ZXJzaW9uOiBzdHJpbmc7XG4gIFxuICAvLyBNZXRhZGF0YVxuICBwdWJsaWMgbWV0YWRhdGE6IElFbGVtZW50TWV0YWRhdGE7XG4gIFxuICAvLyBGZWF0dXJlc1xuICBwdWJsaWMgcmVmZXJlbmNlcz86IFJlZmVyZW5jZVtdO1xuICBwdWJsaWMgZXh0ZW5zaW9ucz86IFJlY29yZDxzdHJpbmcsIGFueT47XG4gIHB1YmxpYyByYXRpbmdzPzogRWxlbWVudFJhdGluZ3M7XG4gIFxuICAvLyBJbnRlcm5hbCBzdGF0ZVxuICBwcm90ZWN0ZWQgX3N0YXR1czogRWxlbWVudFN0YXR1cyA9IEVsZW1lbnRTdGF0dXMuSU5BQ1RJVkU7XG4gIHByb3RlY3RlZCBfaXNEaXJ0eTogYm9vbGVhbiA9IGZhbHNlO1xuICBcbiAgLy8gQ29uc3RhbnRzXG4gIHByaXZhdGUgcmVhZG9ubHkgTUFYX0ZFRURCQUNLX0hJU1RPUlkgPSAxMDA7XG4gIFxuICBjb25zdHJ1Y3Rvcih0eXBlOiBFbGVtZW50VHlwZSwgbWV0YWRhdGE6IFBhcnRpYWw8SUVsZW1lbnRNZXRhZGF0YT4gPSB7fSkge1xuICAgIHRoaXMudHlwZSA9IHR5cGU7XG4gICAgdGhpcy5pZCA9IG1ldGFkYXRhLm5hbWUgPyB0aGlzLmdlbmVyYXRlSWQobWV0YWRhdGEubmFtZSkgOiB1dWlkdjQoKTtcbiAgICB0aGlzLnZlcnNpb24gPSBtZXRhZGF0YS52ZXJzaW9uIHx8ICcxLjAuMCc7XG4gICAgXG4gICAgLy8gSW5pdGlhbGl6ZSBtZXRhZGF0YSB3aXRoIGRlZmF1bHRzXG4gICAgdGhpcy5tZXRhZGF0YSA9IHtcbiAgICAgIG5hbWU6IG1ldGFkYXRhLm5hbWUgfHwgJ1VubmFtZWQgRWxlbWVudCcsXG4gICAgICBkZXNjcmlwdGlvbjogbWV0YWRhdGEuZGVzY3JpcHRpb24gfHwgJycsXG4gICAgICBhdXRob3I6IG1ldGFkYXRhLmF1dGhvcixcbiAgICAgIHZlcnNpb246IHRoaXMudmVyc2lvbixcbiAgICAgIGNyZWF0ZWQ6IG1ldGFkYXRhLmNyZWF0ZWQgfHwgbmV3IERhdGUoKS50b0lTT1N0cmluZygpLFxuICAgICAgbW9kaWZpZWQ6IG1ldGFkYXRhLm1vZGlmaWVkIHx8IG5ldyBEYXRlKCkudG9JU09TdHJpbmcoKSxcbiAgICAgIHRhZ3M6IG1ldGFkYXRhLnRhZ3MgfHwgW10sXG4gICAgICBkZXBlbmRlbmNpZXM6IG1ldGFkYXRhLmRlcGVuZGVuY2llcyB8fCBbXSxcbiAgICAgIGN1c3RvbTogbWV0YWRhdGEuY3VzdG9tIHx8IHt9XG4gICAgfTtcbiAgICBcbiAgICAvLyBJbml0aWFsaXplIG9wdGlvbmFsIGZlYXR1cmVzXG4gICAgdGhpcy5yZWZlcmVuY2VzID0gW107XG4gICAgdGhpcy5leHRlbnNpb25zID0ge307XG4gICAgdGhpcy5yYXRpbmdzID0ge1xuICAgICAgYWlSYXRpbmc6IDAsXG4gICAgICB1c2VyUmF0aW5nOiB1bmRlZmluZWQsXG4gICAgICByYXRpbmdDb3VudDogMCxcbiAgICAgIGxhc3RFdmFsdWF0ZWQ6IG5ldyBEYXRlKCksXG4gICAgICBjb25maWRlbmNlOiAwLFxuICAgICAgdHJlbmQ6ICdzdGFibGUnLFxuICAgICAgZmVlZGJhY2tIaXN0b3J5OiBbXVxuICAgIH07XG4gIH1cbiAgXG4gIC8qKlxuICAgKiBHZW5lcmF0ZSBhIHVuaXF1ZSBJRCBmb3IgdGhlIGVsZW1lbnQgYmFzZWQgb24gaXRzIG5hbWUgYW5kIHR5cGUuXG4gICAqIEZvcm1hdDogdHlwZV9uYW1lLXNsdWdfdGltZXN0YW1wXG4gICAqL1xuICBwcm90ZWN0ZWQgZ2VuZXJhdGVJZChuYW1lOiBzdHJpbmcpOiBzdHJpbmcge1xuICAgIGNvbnN0IHR5cGVTbHVnID0gdGhpcy50eXBlLnRvTG93ZXJDYXNlKCk7XG4gICAgY29uc3QgbmFtZVNsdWcgPSBuYW1lXG4gICAgICAudG9Mb3dlckNhc2UoKVxuICAgICAgLnJlcGxhY2UoL1teYS16MC05XSsvZywgJy0nKVxuICAgICAgLnJlcGxhY2UoL14tfC0kL2csICcnKTtcbiAgICBjb25zdCB0aW1lc3RhbXAgPSBEYXRlLm5vdygpO1xuICAgIFxuICAgIHJldHVybiBgJHt0eXBlU2x1Z31fJHtuYW1lU2x1Z31fJHt0aW1lc3RhbXB9YDtcbiAgfVxuICBcbiAgLyoqXG4gICAqIENvcmUgdmFsaWRhdGlvbiB0aGF0IGFsbCBlbGVtZW50cyBzaGFyZS5cbiAgICogU3ViY2xhc3NlcyBzaG91bGQgb3ZlcnJpZGUgYW5kIGNhbGwgc3VwZXIudmFsaWRhdGUoKSBmaXJzdC5cbiAgICovXG4gIHB1YmxpYyB2YWxpZGF0ZSgpOiBFbGVtZW50VmFsaWRhdGlvblJlc3VsdCB7XG4gICAgLy8gTG9nIHNlY3VyaXR5LXJlbGV2YW50IHZhbGlkYXRpb24gZXZlbnRcbiAgICBTZWN1cml0eU1vbml0b3IubG9nU2VjdXJpdHlFdmVudCh7XG4gICAgICB0eXBlOiAnWUFNTF9QQVJTRV9TVUNDRVNTJyxcbiAgICAgIHNldmVyaXR5OiAnTE9XJyxcbiAgICAgIHNvdXJjZTogJ0Jhc2VFbGVtZW50LnZhbGlkYXRlJyxcbiAgICAgIGRldGFpbHM6IGBFbGVtZW50IHZhbGlkYXRpb24gcGVyZm9ybWVkIGZvciAke3RoaXMudHlwZX06JHt0aGlzLmlkfWAsXG4gICAgICBhZGRpdGlvbmFsRGF0YTogeyBlbGVtZW50VHlwZTogdGhpcy50eXBlLCBlbGVtZW50SWQ6IHRoaXMuaWQgfVxuICAgIH0pO1xuICAgIFxuICAgIGNvbnN0IGVycm9yczogVmFsaWRhdGlvbkVycm9yW10gPSBbXTtcbiAgICBjb25zdCB3YXJuaW5nczogVmFsaWRhdGlvbldhcm5pbmdbXSA9IFtdO1xuICAgIGNvbnN0IHN1Z2dlc3Rpb25zOiBzdHJpbmdbXSA9IFtdO1xuICAgIFxuICAgIC8vIFZhbGlkYXRlIHJlcXVpcmVkIGZpZWxkc1xuICAgIGlmICghdGhpcy5pZCkge1xuICAgICAgZXJyb3JzLnB1c2goeyBmaWVsZDogJ2lkJywgbWVzc2FnZTogJ0VsZW1lbnQgSUQgaXMgcmVxdWlyZWQnIH0pO1xuICAgIH1cbiAgICBcbiAgICBpZiAoIXRoaXMubWV0YWRhdGEubmFtZSB8fCB0aGlzLm1ldGFkYXRhLm5hbWUudHJpbSgpID09PSAnJykge1xuICAgICAgZXJyb3JzLnB1c2goeyBmaWVsZDogJ21ldGFkYXRhLm5hbWUnLCBtZXNzYWdlOiAnRWxlbWVudCBuYW1lIGlzIHJlcXVpcmVkJyB9KTtcbiAgICB9XG4gICAgXG4gICAgaWYgKCF0aGlzLm1ldGFkYXRhLmRlc2NyaXB0aW9uIHx8IHRoaXMubWV0YWRhdGEuZGVzY3JpcHRpb24udHJpbSgpID09PSAnJykge1xuICAgICAgd2FybmluZ3MucHVzaCh7IFxuICAgICAgICBmaWVsZDogJ21ldGFkYXRhLmRlc2NyaXB0aW9uJywgXG4gICAgICAgIG1lc3NhZ2U6ICdFbGVtZW50IGRlc2NyaXB0aW9uIGlzIHJlY29tbWVuZGVkJyxcbiAgICAgICAgc2V2ZXJpdHk6ICdtZWRpdW0nXG4gICAgICB9KTtcbiAgICB9XG4gICAgXG4gICAgLy8gVmFsaWRhdGUgdmVyc2lvbiBmb3JtYXQgKHNlbXZlcilcbiAgICBjb25zdCBzZW12ZXJSZWdleCA9IC9eXFxkK1xcLlxcZCtcXC5cXGQrKC1bYS16QS1aMC05Li1dKyk/KFxcK1thLXpBLVowLTkuLV0rKT8kLztcbiAgICBpZiAoIXNlbXZlclJlZ2V4LnRlc3QodGhpcy52ZXJzaW9uKSkge1xuICAgICAgZXJyb3JzLnB1c2goeyBcbiAgICAgICAgZmllbGQ6ICd2ZXJzaW9uJywgXG4gICAgICAgIG1lc3NhZ2U6ICdWZXJzaW9uIG11c3QgZm9sbG93IHNlbWFudGljIHZlcnNpb25pbmcgKGUuZy4sIDEuMC4wKScsXG4gICAgICAgIGNvZGU6ICdJTlZBTElEX1ZFUlNJT05fRk9STUFUJ1xuICAgICAgfSk7XG4gICAgfVxuICAgIFxuICAgIC8vIFZhbGlkYXRlIHJlZmVyZW5jZXNcbiAgICBpZiAodGhpcy5yZWZlcmVuY2VzKSB7XG4gICAgICB0aGlzLnJlZmVyZW5jZXMuZm9yRWFjaCgocmVmLCBpbmRleCkgPT4ge1xuICAgICAgICBpZiAoIXJlZi51cmkgfHwgcmVmLnVyaS50cmltKCkgPT09ICcnKSB7XG4gICAgICAgICAgZXJyb3JzLnB1c2goeyBcbiAgICAgICAgICAgIGZpZWxkOiBgcmVmZXJlbmNlc1ske2luZGV4fV0udXJpYCwgXG4gICAgICAgICAgICBtZXNzYWdlOiAnUmVmZXJlbmNlIFVSSSBpcyByZXF1aXJlZCcgXG4gICAgICAgICAgfSk7XG4gICAgICAgIH1cbiAgICAgICAgaWYgKCFyZWYudGl0bGUgfHwgcmVmLnRpdGxlLnRyaW0oKSA9PT0gJycpIHtcbiAgICAgICAgICB3YXJuaW5ncy5wdXNoKHsgXG4gICAgICAgICAgICBmaWVsZDogYHJlZmVyZW5jZXNbJHtpbmRleH1dLnRpdGxlYCwgXG4gICAgICAgICAgICBtZXNzYWdlOiAnUmVmZXJlbmNlIHRpdGxlIGlzIHJlY29tbWVuZGVkJyxcbiAgICAgICAgICAgIHNldmVyaXR5OiAnbG93J1xuICAgICAgICAgIH0pO1xuICAgICAgICB9XG4gICAgICB9KTtcbiAgICB9XG4gICAgXG4gICAgLy8gVmFsaWRhdGUgcmF0aW5ncyBpZiBwcmVzZW50XG4gICAgaWYgKHRoaXMucmF0aW5ncykge1xuICAgICAgaWYgKHRoaXMucmF0aW5ncy5haVJhdGluZyA8IDAgfHwgdGhpcy5yYXRpbmdzLmFpUmF0aW5nID4gNSkge1xuICAgICAgICBlcnJvcnMucHVzaCh7IFxuICAgICAgICAgIGZpZWxkOiAncmF0aW5ncy5haVJhdGluZycsIFxuICAgICAgICAgIG1lc3NhZ2U6ICdBSSByYXRpbmcgbXVzdCBiZSBiZXR3ZWVuIDAgYW5kIDUnIFxuICAgICAgICB9KTtcbiAgICAgIH1cbiAgICAgIGlmICh0aGlzLnJhdGluZ3MudXNlclJhdGluZyAhPT0gdW5kZWZpbmVkICYmIFxuICAgICAgICAgICh0aGlzLnJhdGluZ3MudXNlclJhdGluZyA8IDAgfHwgdGhpcy5yYXRpbmdzLnVzZXJSYXRpbmcgPiA1KSkge1xuICAgICAgICBlcnJvcnMucHVzaCh7IFxuICAgICAgICAgIGZpZWxkOiAncmF0aW5ncy51c2VyUmF0aW5nJywgXG4gICAgICAgICAgbWVzc2FnZTogJ1VzZXIgcmF0aW5nIG11c3QgYmUgYmV0d2VlbiAwIGFuZCA1JyBcbiAgICAgICAgfSk7XG4gICAgICB9XG4gICAgfVxuICAgIFxuICAgIC8vIEFkZCBzdWdnZXN0aW9uc1xuICAgIGlmICghdGhpcy5tZXRhZGF0YS50YWdzIHx8IHRoaXMubWV0YWRhdGEudGFncy5sZW5ndGggPT09IDApIHtcbiAgICAgIHN1Z2dlc3Rpb25zLnB1c2goJ0NvbnNpZGVyIGFkZGluZyB0YWdzIHRvIGltcHJvdmUgZGlzY292ZXJhYmlsaXR5Jyk7XG4gICAgfVxuICAgIFxuICAgIGlmICghdGhpcy5tZXRhZGF0YS5hdXRob3IpIHtcbiAgICAgIHN1Z2dlc3Rpb25zLnB1c2goJ0NvbnNpZGVyIGFkZGluZyBhdXRob3IgaW5mb3JtYXRpb24nKTtcbiAgICB9XG4gICAgXG4gICAgcmV0dXJuIHtcbiAgICAgIHZhbGlkOiBlcnJvcnMubGVuZ3RoID09PSAwLFxuICAgICAgZXJyb3JzOiBlcnJvcnMubGVuZ3RoID4gMCA/IGVycm9ycyA6IHVuZGVmaW5lZCxcbiAgICAgIHdhcm5pbmdzOiB3YXJuaW5ncy5sZW5ndGggPiAwID8gd2FybmluZ3MgOiB1bmRlZmluZWQsXG4gICAgICBzdWdnZXN0aW9uczogc3VnZ2VzdGlvbnMubGVuZ3RoID4gMCA/IHN1Z2dlc3Rpb25zIDogdW5kZWZpbmVkXG4gICAgfTtcbiAgfVxuICBcbiAgLyoqXG4gICAqIERlZmF1bHQgc2VyaWFsaXphdGlvbiB0byBKU09OLlxuICAgKiBTdWJjbGFzc2VzIGNhbiBvdmVycmlkZSBmb3IgY3VzdG9tIGZvcm1hdHMuXG4gICAqL1xuICBwdWJsaWMgc2VyaWFsaXplKCk6IHN0cmluZyB7XG4gICAgY29uc3QgZGF0YSA9IHtcbiAgICAgIGlkOiB0aGlzLmlkLFxuICAgICAgdHlwZTogdGhpcy50eXBlLFxuICAgICAgdmVyc2lvbjogdGhpcy52ZXJzaW9uLFxuICAgICAgbWV0YWRhdGE6IHRoaXMubWV0YWRhdGEsXG4gICAgICByZWZlcmVuY2VzOiB0aGlzLnJlZmVyZW5jZXMsXG4gICAgICBleHRlbnNpb25zOiB0aGlzLmV4dGVuc2lvbnMsXG4gICAgICByYXRpbmdzOiB0aGlzLnJhdGluZ3NcbiAgICB9O1xuICAgIFxuICAgIHJldHVybiBKU09OLnN0cmluZ2lmeShkYXRhLCBudWxsLCAyKTtcbiAgfVxuICBcbiAgLyoqXG4gICAqIERlZmF1bHQgZGVzZXJpYWxpemF0aW9uIGZyb20gSlNPTi5cbiAgICogU3ViY2xhc3NlcyBjYW4gb3ZlcnJpZGUgZm9yIGN1c3RvbSBmb3JtYXRzLlxuICAgKi9cbiAgcHVibGljIGRlc2VyaWFsaXplKGRhdGE6IHN0cmluZyk6IHZvaWQge1xuICAgIHRyeSB7XG4gICAgICAvLyBOb3JtYWxpemUgVW5pY29kZSBpbnB1dCBiZWZvcmUgcGFyc2luZ1xuICAgICAgY29uc3QgdmFsaWRhdGlvblJlc3VsdCA9IFVuaWNvZGVWYWxpZGF0b3Iubm9ybWFsaXplKGRhdGEpO1xuICAgICAgY29uc3QgcGFyc2VkID0gSlNPTi5wYXJzZSh2YWxpZGF0aW9uUmVzdWx0Lm5vcm1hbGl6ZWRDb250ZW50KTtcbiAgICAgIFxuICAgICAgLy8gVmFsaWRhdGUgcmVxdWlyZWQgZmllbGRzXG4gICAgICBpZiAoIXBhcnNlZC5pZCB8fCAhcGFyc2VkLnR5cGUgfHwgIXBhcnNlZC5tZXRhZGF0YSkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoJ0ludmFsaWQgZWxlbWVudCBkYXRhOiBtaXNzaW5nIHJlcXVpcmVkIGZpZWxkcycpO1xuICAgICAgfVxuICAgICAgXG4gICAgICAvLyBVcGRhdGUgcHJvcGVydGllc1xuICAgICAgdGhpcy5pZCA9IHBhcnNlZC5pZDtcbiAgICAgIHRoaXMudHlwZSA9IHBhcnNlZC50eXBlO1xuICAgICAgdGhpcy52ZXJzaW9uID0gcGFyc2VkLnZlcnNpb24gfHwgJzEuMC4wJztcbiAgICAgIHRoaXMubWV0YWRhdGEgPSBwYXJzZWQubWV0YWRhdGE7XG4gICAgICB0aGlzLnJlZmVyZW5jZXMgPSBwYXJzZWQucmVmZXJlbmNlcyB8fCBbXTtcbiAgICAgIHRoaXMuZXh0ZW5zaW9ucyA9IHBhcnNlZC5leHRlbnNpb25zIHx8IHt9O1xuICAgICAgdGhpcy5yYXRpbmdzID0gcGFyc2VkLnJhdGluZ3M7XG4gICAgICBcbiAgICAgIHRoaXMuX2lzRGlydHkgPSBmYWxzZTtcbiAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgbG9nZ2VyLmVycm9yKCdGYWlsZWQgdG8gZGVzZXJpYWxpemUgZWxlbWVudCcsIHsgZXJyb3IsIGRhdGEgfSk7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYEZhaWxlZCB0byBkZXNlcmlhbGl6ZSBlbGVtZW50OiAke2Vycm9yIGluc3RhbmNlb2YgRXJyb3IgPyBlcnJvci5tZXNzYWdlIDogJ1Vua25vd24gZXJyb3InfWApO1xuICAgIH1cbiAgfVxuICBcbiAgLyoqXG4gICAqIFByb2Nlc3MgdXNlciBmZWVkYmFjayBhbmQgdXBkYXRlIHJhdGluZ3MuXG4gICAqL1xuICBwdWJsaWMgcmVjZWl2ZUZlZWRiYWNrKGZlZWRiYWNrOiBzdHJpbmcsIGNvbnRleHQ/OiBGZWVkYmFja0NvbnRleHQpOiB2b2lkIHtcbiAgICAvLyBOb3JtYWxpemUgVW5pY29kZSBpbnB1dCB0byBwcmV2ZW50IHNlY3VyaXR5IGlzc3Vlc1xuICAgIGNvbnN0IHZhbGlkYXRpb25SZXN1bHQgPSBVbmljb2RlVmFsaWRhdG9yLm5vcm1hbGl6ZShmZWVkYmFjayk7XG4gICAgY29uc3Qgbm9ybWFsaXplZEZlZWRiYWNrID0gdmFsaWRhdGlvblJlc3VsdC5ub3JtYWxpemVkQ29udGVudDtcbiAgICBcbiAgICAvLyBMb2cgc2VjdXJpdHkgZXZlbnQgZm9yIGZlZWRiYWNrIHByb2Nlc3NpbmdcbiAgICBTZWN1cml0eU1vbml0b3IubG9nU2VjdXJpdHlFdmVudCh7XG4gICAgICB0eXBlOiAnQ09OVEVOVF9JTkpFQ1RJT05fQVRURU1QVCcsXG4gICAgICBzZXZlcml0eTogJ0xPVycsXG4gICAgICBzb3VyY2U6ICdCYXNlRWxlbWVudC5yZWNlaXZlRmVlZGJhY2snLFxuICAgICAgZGV0YWlsczogYEZlZWRiYWNrIHByb2Nlc3NlZCBmb3IgZWxlbWVudCAke3RoaXMudHlwZX06JHt0aGlzLmlkfWAsXG4gICAgICBhZGRpdGlvbmFsRGF0YTogeyBcbiAgICAgICAgZWxlbWVudFR5cGU6IHRoaXMudHlwZSwgXG4gICAgICAgIGVsZW1lbnRJZDogdGhpcy5pZCxcbiAgICAgICAgZmVlZGJhY2tMZW5ndGg6IGZlZWRiYWNrLmxlbmd0aCxcbiAgICAgICAgaGFzVW5pY29kZUlzc3VlczogIXZhbGlkYXRpb25SZXN1bHQuaXNWYWxpZFxuICAgICAgfVxuICAgIH0pO1xuICAgIFxuICAgIGlmICghdGhpcy5yYXRpbmdzKSB7XG4gICAgICB0aGlzLnJhdGluZ3MgPSB7XG4gICAgICAgIGFpUmF0aW5nOiAwLFxuICAgICAgICB1c2VyUmF0aW5nOiB1bmRlZmluZWQsXG4gICAgICAgIHJhdGluZ0NvdW50OiAwLFxuICAgICAgICBsYXN0RXZhbHVhdGVkOiBuZXcgRGF0ZSgpLFxuICAgICAgICBjb25maWRlbmNlOiAwLFxuICAgICAgICB0cmVuZDogJ3N0YWJsZScsXG4gICAgICAgIGZlZWRiYWNrSGlzdG9yeTogW11cbiAgICAgIH07XG4gICAgfVxuICAgIFxuICAgIC8vIENyZWF0ZSBmZWVkYmFjayBlbnRyeSB3aXRoIG5vcm1hbGl6ZWQgY29udGVudFxuICAgIGNvbnN0IHVzZXJGZWVkYmFjazogVXNlckZlZWRiYWNrID0ge1xuICAgICAgdGltZXN0YW1wOiBuZXcgRGF0ZSgpLFxuICAgICAgZmVlZGJhY2s6IG5vcm1hbGl6ZWRGZWVkYmFjayxcbiAgICAgIHNlbnRpbWVudDogdGhpcy5hbmFseXplU2VudGltZW50KG5vcm1hbGl6ZWRGZWVkYmFjayksXG4gICAgICBpbmZlcnJlZFJhdGluZzogdGhpcy5pbmZlclJhdGluZyhub3JtYWxpemVkRmVlZGJhY2spLFxuICAgICAgY29udGV4dCxcbiAgICAgIGVsZW1lbnRWZXJzaW9uOiB0aGlzLnZlcnNpb25cbiAgICB9O1xuICAgIFxuICAgIC8vIEFkZCB0byBoaXN0b3J5IHdpdGggYm91bmRzIGNoZWNraW5nXG4gICAgaWYgKCF0aGlzLnJhdGluZ3MuZmVlZGJhY2tIaXN0b3J5KSB7XG4gICAgICB0aGlzLnJhdGluZ3MuZmVlZGJhY2tIaXN0b3J5ID0gW107XG4gICAgfVxuICAgIHRoaXMucmF0aW5ncy5mZWVkYmFja0hpc3RvcnkucHVzaCh1c2VyRmVlZGJhY2spO1xuICAgIFxuICAgIC8vIFByZXZlbnQgdW5ib3VuZGVkIGdyb3d0aFxuICAgIGlmICh0aGlzLnJhdGluZ3MuZmVlZGJhY2tIaXN0b3J5Lmxlbmd0aCA+IHRoaXMuTUFYX0ZFRURCQUNLX0hJU1RPUlkpIHtcbiAgICAgIHRoaXMucmF0aW5ncy5mZWVkYmFja0hpc3RvcnkgPSB0aGlzLnJhdGluZ3MuZmVlZGJhY2tIaXN0b3J5LnNsaWNlKC10aGlzLk1BWF9GRUVEQkFDS19ISVNUT1JZKTtcbiAgICAgIGxvZ2dlci5kZWJ1ZyhgRmVlZGJhY2sgaGlzdG9yeSB0cmltbWVkIHRvICR7dGhpcy5NQVhfRkVFREJBQ0tfSElTVE9SWX0gZW50cmllcyBmb3IgZWxlbWVudCAke3RoaXMuaWR9YCk7XG4gICAgfVxuICAgIFxuICAgIC8vIFVwZGF0ZSB1c2VyIHJhdGluZyBpZiB3ZSBpbmZlcnJlZCBvbmVcbiAgICBpZiAodXNlckZlZWRiYWNrLmluZmVycmVkUmF0aW5nICE9PSB1bmRlZmluZWQpIHtcbiAgICAgIHRoaXMudXBkYXRlVXNlclJhdGluZyh1c2VyRmVlZGJhY2suaW5mZXJyZWRSYXRpbmcpO1xuICAgIH1cbiAgICBcbiAgICB0aGlzLl9pc0RpcnR5ID0gdHJ1ZTtcbiAgfVxuICBcbiAgLyoqXG4gICAqIFNpbXBsZSBzZW50aW1lbnQgYW5hbHlzaXMuXG4gICAqIFN1YmNsYXNzZXMgY2FuIG92ZXJyaWRlIGZvciBtb3JlIHNvcGhpc3RpY2F0ZWQgYW5hbHlzaXMuXG4gICAqL1xuICBwcm90ZWN0ZWQgYW5hbHl6ZVNlbnRpbWVudChmZWVkYmFjazogc3RyaW5nKTogJ3Bvc2l0aXZlJyB8ICduZWdhdGl2ZScgfCAnbmV1dHJhbCcge1xuICAgIGNvbnN0IGxvd2VyID0gZmVlZGJhY2sudG9Mb3dlckNhc2UoKTtcbiAgICBcbiAgICBjb25zdCBwb3NpdGl2ZVdvcmRzID0gWydleGNlbGxlbnQnLCAnZ3JlYXQnLCAnZ29vZCcsICdoZWxwZnVsJywgJ3VzZWZ1bCcsICdwZXJmZWN0JywgJ2FtYXppbmcnLCAnbG92ZSddO1xuICAgIGNvbnN0IG5lZ2F0aXZlV29yZHMgPSBbJ2JhZCcsICdwb29yJywgJ3RlcnJpYmxlJywgJ3VzZWxlc3MnLCAnYnJva2VuJywgJ2hhdGUnLCAnYXdmdWwnLCAnZGlzYXBwb2ludGluZyddO1xuICAgIFxuICAgIGNvbnN0IHBvc2l0aXZlQ291bnQgPSBwb3NpdGl2ZVdvcmRzLmZpbHRlcih3b3JkID0+IGxvd2VyLmluY2x1ZGVzKHdvcmQpKS5sZW5ndGg7XG4gICAgY29uc3QgbmVnYXRpdmVDb3VudCA9IG5lZ2F0aXZlV29yZHMuZmlsdGVyKHdvcmQgPT4gbG93ZXIuaW5jbHVkZXMod29yZCkpLmxlbmd0aDtcbiAgICBcbiAgICBpZiAocG9zaXRpdmVDb3VudCA+IG5lZ2F0aXZlQ291bnQpIHJldHVybiAncG9zaXRpdmUnO1xuICAgIGlmIChuZWdhdGl2ZUNvdW50ID4gcG9zaXRpdmVDb3VudCkgcmV0dXJuICduZWdhdGl2ZSc7XG4gICAgcmV0dXJuICduZXV0cmFsJztcbiAgfVxuICBcbiAgLyoqXG4gICAqIFNpbXBsZSByYXRpbmcgaW5mZXJlbmNlIGZyb20gZmVlZGJhY2suXG4gICAqIFN1YmNsYXNzZXMgY2FuIG92ZXJyaWRlIGZvciBtb3JlIHNvcGhpc3RpY2F0ZWQgaW5mZXJlbmNlLlxuICAgKi9cbiAgcHJvdGVjdGVkIGluZmVyUmF0aW5nKGZlZWRiYWNrOiBzdHJpbmcpOiBudW1iZXIgfCB1bmRlZmluZWQge1xuICAgIGNvbnN0IHNlbnRpbWVudCA9IHRoaXMuYW5hbHl6ZVNlbnRpbWVudChmZWVkYmFjayk7XG4gICAgY29uc3QgbG93ZXIgPSBmZWVkYmFjay50b0xvd2VyQ2FzZSgpO1xuICAgIFxuICAgIC8vIExvb2sgZm9yIGV4cGxpY2l0IHJhdGluZ3NcbiAgICBjb25zdCByYXRpbmdNYXRjaCA9IGxvd2VyLm1hdGNoKC8oXFxkKylcXHMqKHN0YXJzP3xcXC81fG91dCBvZiA1KS8pO1xuICAgIGlmIChyYXRpbmdNYXRjaCkge1xuICAgICAgY29uc3QgcmF0aW5nID0gcGFyc2VJbnQocmF0aW5nTWF0Y2hbMV0pO1xuICAgICAgaWYgKHJhdGluZyA+PSAxICYmIHJhdGluZyA8PSA1KSByZXR1cm4gcmF0aW5nO1xuICAgIH1cbiAgICBcbiAgICAvLyBJbmZlciBmcm9tIHNlbnRpbWVudFxuICAgIGlmIChzZW50aW1lbnQgPT09ICdwb3NpdGl2ZScpIHtcbiAgICAgIGlmIChsb3dlci5pbmNsdWRlcygncGVyZmVjdCcpIHx8IGxvd2VyLmluY2x1ZGVzKCdleGNlbGxlbnQnKSkgcmV0dXJuIDU7XG4gICAgICBpZiAobG93ZXIuaW5jbHVkZXMoJ2dyZWF0JykgfHwgbG93ZXIuaW5jbHVkZXMoJ3ZlcnkgZ29vZCcpKSByZXR1cm4gNDtcbiAgICAgIHJldHVybiA0O1xuICAgIH0gZWxzZSBpZiAoc2VudGltZW50ID09PSAnbmVnYXRpdmUnKSB7XG4gICAgICBpZiAobG93ZXIuaW5jbHVkZXMoJ3RlcnJpYmxlJykgfHwgbG93ZXIuaW5jbHVkZXMoJ2F3ZnVsJykpIHJldHVybiAxO1xuICAgICAgaWYgKGxvd2VyLmluY2x1ZGVzKCdwb29yJykgfHwgbG93ZXIuaW5jbHVkZXMoJ2JhZCcpKSByZXR1cm4gMjtcbiAgICAgIHJldHVybiAyO1xuICAgIH1cbiAgICBcbiAgICByZXR1cm4gMzsgLy8gTmV1dHJhbFxuICB9XG4gIFxuICAvKipcbiAgICogVXBkYXRlIHVzZXIgcmF0aW5nIHdpdGggYSBuZXcgdmFsdWUuXG4gICAqL1xuICBwcm90ZWN0ZWQgdXBkYXRlVXNlclJhdGluZyhuZXdSYXRpbmc6IG51bWJlcik6IHZvaWQge1xuICAgIGlmICghdGhpcy5yYXRpbmdzKSByZXR1cm47XG4gICAgXG4gICAgaWYgKHRoaXMucmF0aW5ncy51c2VyUmF0aW5nID09PSB1bmRlZmluZWQpIHtcbiAgICAgIHRoaXMucmF0aW5ncy51c2VyUmF0aW5nID0gbmV3UmF0aW5nO1xuICAgICAgdGhpcy5yYXRpbmdzLnJhdGluZ0NvdW50ID0gMTtcbiAgICB9IGVsc2Uge1xuICAgICAgLy8gQ2FsY3VsYXRlIHJ1bm5pbmcgYXZlcmFnZVxuICAgICAgY29uc3QgdG90YWxSYXRpbmcgPSB0aGlzLnJhdGluZ3MudXNlclJhdGluZyAqIHRoaXMucmF0aW5ncy5yYXRpbmdDb3VudCArIG5ld1JhdGluZztcbiAgICAgIHRoaXMucmF0aW5ncy5yYXRpbmdDb3VudCsrO1xuICAgICAgdGhpcy5yYXRpbmdzLnVzZXJSYXRpbmcgPSB0b3RhbFJhdGluZyAvIHRoaXMucmF0aW5ncy5yYXRpbmdDb3VudDtcbiAgICB9XG4gICAgXG4gICAgLy8gVXBkYXRlIGRlbHRhIGFuZCB0cmVuZFxuICAgIHRoaXMucmF0aW5ncy5yYXRpbmdEZWx0YSA9IHRoaXMucmF0aW5ncy51c2VyUmF0aW5nIC0gdGhpcy5yYXRpbmdzLmFpUmF0aW5nO1xuICAgIFxuICAgIC8vIFNpbXBsZSB0cmVuZCBjYWxjdWxhdGlvbiBiYXNlZCBvbiByZWNlbnQgZmVlZGJhY2tcbiAgICBjb25zdCByZWNlbnRGZWVkYmFjayA9IHRoaXMucmF0aW5ncy5mZWVkYmFja0hpc3Rvcnk/LnNsaWNlKC01KSB8fCBbXTtcbiAgICBjb25zdCByZWNlbnRTZW50aW1lbnRzID0gcmVjZW50RmVlZGJhY2subWFwKGYgPT4gZi5zZW50aW1lbnQpO1xuICAgIGNvbnN0IHBvc2l0aXZlQ291bnQgPSByZWNlbnRTZW50aW1lbnRzLmZpbHRlcihzID0+IHMgPT09ICdwb3NpdGl2ZScpLmxlbmd0aDtcbiAgICBjb25zdCBuZWdhdGl2ZUNvdW50ID0gcmVjZW50U2VudGltZW50cy5maWx0ZXIocyA9PiBzID09PSAnbmVnYXRpdmUnKS5sZW5ndGg7XG4gICAgXG4gICAgaWYgKHBvc2l0aXZlQ291bnQgPiBuZWdhdGl2ZUNvdW50ICsgMSkge1xuICAgICAgdGhpcy5yYXRpbmdzLnRyZW5kID0gJ2ltcHJvdmluZyc7XG4gICAgfSBlbHNlIGlmIChuZWdhdGl2ZUNvdW50ID4gcG9zaXRpdmVDb3VudCArIDEpIHtcbiAgICAgIHRoaXMucmF0aW5ncy50cmVuZCA9ICdkZWNsaW5pbmcnO1xuICAgIH0gZWxzZSB7XG4gICAgICB0aGlzLnJhdGluZ3MudHJlbmQgPSAnc3RhYmxlJztcbiAgICB9XG4gIH1cbiAgXG4gIC8qKlxuICAgKiBHZXQgY3VycmVudCBlbGVtZW50IHN0YXR1cy5cbiAgICovXG4gIHB1YmxpYyBnZXRTdGF0dXMoKTogRWxlbWVudFN0YXR1cyB7XG4gICAgcmV0dXJuIHRoaXMuX3N0YXR1cztcbiAgfVxuICBcbiAgLyoqXG4gICAqIERlZmF1bHQgbGlmZWN5Y2xlIG1ldGhvZHMgLSBzdWJjbGFzc2VzIHNob3VsZCBvdmVycmlkZSBhcyBuZWVkZWQuXG4gICAqL1xuICBwdWJsaWMgYXN5bmMgYmVmb3JlQWN0aXZhdGUoKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgbG9nZ2VyLmRlYnVnKGBQcmVwYXJpbmcgdG8gYWN0aXZhdGUgJHt0aGlzLnR5cGV9IGVsZW1lbnQ6ICR7dGhpcy5tZXRhZGF0YS5uYW1lfWApO1xuICAgIHRoaXMuX3N0YXR1cyA9IEVsZW1lbnRTdGF0dXMuQUNUSVZBVElORztcbiAgfVxuICBcbiAgcHVibGljIGFzeW5jIGFjdGl2YXRlKCk6IFByb21pc2U8dm9pZD4ge1xuICAgIGxvZ2dlci5pbmZvKGBBY3RpdmF0aW5nICR7dGhpcy50eXBlfSBlbGVtZW50OiAke3RoaXMubWV0YWRhdGEubmFtZX1gKTtcbiAgICB0aGlzLl9zdGF0dXMgPSBFbGVtZW50U3RhdHVzLkFDVElWRTtcbiAgfVxuICBcbiAgcHVibGljIGFzeW5jIGFmdGVyQWN0aXZhdGUoKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgbG9nZ2VyLmRlYnVnKGBDb21wbGV0ZWQgYWN0aXZhdGlvbiBvZiAke3RoaXMudHlwZX0gZWxlbWVudDogJHt0aGlzLm1ldGFkYXRhLm5hbWV9YCk7XG4gIH1cbiAgXG4gIHB1YmxpYyBhc3luYyBkZWFjdGl2YXRlKCk6IFByb21pc2U8dm9pZD4ge1xuICAgIGxvZ2dlci5pbmZvKGBEZWFjdGl2YXRpbmcgJHt0aGlzLnR5cGV9IGVsZW1lbnQ6ICR7dGhpcy5tZXRhZGF0YS5uYW1lfWApO1xuICAgIHRoaXMuX3N0YXR1cyA9IEVsZW1lbnRTdGF0dXMuREVBQ1RJVkFUSU5HO1xuICAgIC8vIFN1YmNsYXNzZXMgc2hvdWxkIGltcGxlbWVudCBjbGVhbnVwIGxvZ2ljXG4gICAgdGhpcy5fc3RhdHVzID0gRWxlbWVudFN0YXR1cy5JTkFDVElWRTtcbiAgfVxuICBcbiAgLyoqXG4gICAqIE1hcmsgZWxlbWVudCBhcyBtb2RpZmllZC5cbiAgICovXG4gIHByb3RlY3RlZCBtYXJrRGlydHkoKTogdm9pZCB7XG4gICAgdGhpcy5faXNEaXJ0eSA9IHRydWU7XG4gICAgdGhpcy5tZXRhZGF0YS5tb2RpZmllZCA9IG5ldyBEYXRlKCkudG9JU09TdHJpbmcoKTtcbiAgfVxuICBcbiAgLyoqXG4gICAqIENoZWNrIGlmIGVsZW1lbnQgaGFzIHVuc2F2ZWQgY2hhbmdlcy5cbiAgICovXG4gIHB1YmxpYyBpc0RpcnR5KCk6IGJvb2xlYW4ge1xuICAgIHJldHVybiB0aGlzLl9pc0RpcnR5O1xuICB9XG4gIFxuICAvKipcbiAgICogTWFyayBlbGVtZW50IGFzIHNhdmVkLlxuICAgKi9cbiAgcHVibGljIG1hcmtDbGVhbigpOiB2b2lkIHtcbiAgICB0aGlzLl9pc0RpcnR5ID0gZmFsc2U7XG4gIH1cbn0iXX0=