Created
February 14, 2026 02:46
-
-
Save iamwrm/ec9bfdc0cf169bdd91be65ae16e891fc to your computer and use it in GitHub Desktop.
Pi session: Building a Tokenizer Playground with Transformers.js
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| <!DOCTYPE html> | |
| <html lang="en"> | |
| <head> | |
| <meta charset="UTF-8"> | |
| <meta name="viewport" content="width=device-width, initial-scale=1.0"> | |
| <title>Session Export</title> | |
| <style> | |
| :root { | |
| --accent: #8abeb7; | |
| --border: #5f87ff; | |
| --borderAccent: #00d7ff; | |
| --borderMuted: #505050; | |
| --success: #b5bd68; | |
| --error: #cc6666; | |
| --warning: #ffff00; | |
| --muted: #808080; | |
| --dim: #666666; | |
| --text: #e5e5e7; | |
| --thinkingText: #808080; | |
| --selectedBg: #3a3a4a; | |
| --userMessageBg: #343541; | |
| --userMessageText: #e5e5e7; | |
| --customMessageBg: #2d2838; | |
| --customMessageText: #e5e5e7; | |
| --customMessageLabel: #9575cd; | |
| --toolPendingBg: #282832; | |
| --toolSuccessBg: #283228; | |
| --toolErrorBg: #3c2828; | |
| --toolTitle: #e5e5e7; | |
| --toolOutput: #808080; | |
| --mdHeading: #f0c674; | |
| --mdLink: #81a2be; | |
| --mdLinkUrl: #666666; | |
| --mdCode: #8abeb7; | |
| --mdCodeBlock: #b5bd68; | |
| --mdCodeBlockBorder: #808080; | |
| --mdQuote: #808080; | |
| --mdQuoteBorder: #808080; | |
| --mdHr: #808080; | |
| --mdListBullet: #8abeb7; | |
| --toolDiffAdded: #b5bd68; | |
| --toolDiffRemoved: #cc6666; | |
| --toolDiffContext: #808080; | |
| --syntaxComment: #6A9955; | |
| --syntaxKeyword: #569CD6; | |
| --syntaxFunction: #DCDCAA; | |
| --syntaxVariable: #9CDCFE; | |
| --syntaxString: #CE9178; | |
| --syntaxNumber: #B5CEA8; | |
| --syntaxType: #4EC9B0; | |
| --syntaxOperator: #D4D4D4; | |
| --syntaxPunctuation: #D4D4D4; | |
| --thinkingOff: #505050; | |
| --thinkingMinimal: #6e6e6e; | |
| --thinkingLow: #5f87af; | |
| --thinkingMedium: #81a2be; | |
| --thinkingHigh: #b294bb; | |
| --thinkingXhigh: #d183e8; | |
| --bashMode: #b5bd68; | |
| --exportPageBg: #18181e; | |
| --exportCardBg: #1e1e24; | |
| --exportInfoBg: #3c3728; | |
| --body-bg: rgb(36, 37, 46); | |
| --container-bg: rgb(44, 45, 55); | |
| --info-bg: rgb(72, 68, 65); | |
| } | |
| * { margin: 0; padding: 0; box-sizing: border-box; } | |
| :root { | |
| --line-height: 18px; /* 12px font * 1.5 */ | |
| } | |
| body { | |
| font-family: ui-monospace, 'Cascadia Code', 'Source Code Pro', Menlo, Consolas, 'DejaVu Sans Mono', monospace; | |
| font-size: 12px; | |
| line-height: var(--line-height); | |
| color: var(--text); | |
| background: var(--body-bg); | |
| } | |
| #app { | |
| display: flex; | |
| min-height: 100vh; | |
| } | |
| /* Sidebar */ | |
| #sidebar { | |
| width: 400px; | |
| background: var(--container-bg); | |
| flex-shrink: 0; | |
| display: flex; | |
| flex-direction: column; | |
| position: sticky; | |
| top: 0; | |
| height: 100vh; | |
| border-right: 1px solid var(--dim); | |
| } | |
| .sidebar-header { | |
| padding: 8px 12px; | |
| flex-shrink: 0; | |
| } | |
| .sidebar-controls { | |
| padding: 8px 8px 4px 8px; | |
| } | |
| .sidebar-search { | |
| width: 100%; | |
| box-sizing: border-box; | |
| padding: 4px 8px; | |
| font-size: 11px; | |
| font-family: inherit; | |
| background: var(--body-bg); | |
| color: var(--text); | |
| border: 1px solid var(--dim); | |
| border-radius: 3px; | |
| } | |
| .sidebar-filters { | |
| display: flex; | |
| padding: 4px 8px 8px 8px; | |
| gap: 4px; | |
| align-items: center; | |
| flex-wrap: wrap; | |
| } | |
| .sidebar-search:focus { | |
| outline: none; | |
| border-color: var(--accent); | |
| } | |
| .sidebar-search::placeholder { | |
| color: var(--muted); | |
| } | |
| .filter-btn { | |
| padding: 3px 8px; | |
| font-size: 10px; | |
| font-family: inherit; | |
| background: transparent; | |
| color: var(--muted); | |
| border: 1px solid var(--dim); | |
| border-radius: 3px; | |
| cursor: pointer; | |
| } | |
| .filter-btn:hover { | |
| color: var(--text); | |
| border-color: var(--text); | |
| } | |
| .filter-btn.active { | |
| background: var(--accent); | |
| color: var(--body-bg); | |
| border-color: var(--accent); | |
| } | |
| .sidebar-close { | |
| display: none; | |
| padding: 3px 8px; | |
| font-size: 12px; | |
| font-family: inherit; | |
| background: transparent; | |
| color: var(--muted); | |
| border: 1px solid var(--dim); | |
| border-radius: 3px; | |
| cursor: pointer; | |
| margin-left: auto; | |
| } | |
| .sidebar-close:hover { | |
| color: var(--text); | |
| border-color: var(--text); | |
| } | |
| .tree-container { | |
| flex: 1; | |
| overflow: auto; | |
| padding: 4px 0; | |
| } | |
| .tree-node { | |
| padding: 0 8px; | |
| cursor: pointer; | |
| display: flex; | |
| align-items: baseline; | |
| font-size: 11px; | |
| line-height: 13px; | |
| white-space: nowrap; | |
| } | |
| .tree-node:hover { | |
| background: var(--selectedBg); | |
| } | |
| .tree-node.active { | |
| background: var(--selectedBg); | |
| } | |
| .tree-node.active .tree-content { | |
| font-weight: bold; | |
| } | |
| .tree-node.in-path { | |
| background: color-mix(in srgb, var(--accent) 10%, transparent); | |
| } | |
| .tree-node:not(.in-path) { | |
| opacity: 0.5; | |
| } | |
| .tree-node:not(.in-path):hover { | |
| opacity: 1; | |
| } | |
| .tree-prefix { | |
| color: var(--muted); | |
| flex-shrink: 0; | |
| font-family: monospace; | |
| white-space: pre; | |
| } | |
| .tree-marker { | |
| color: var(--accent); | |
| flex-shrink: 0; | |
| } | |
| .tree-content { | |
| color: var(--text); | |
| } | |
| .tree-role-user { | |
| color: var(--accent); | |
| } | |
| .tree-role-assistant { | |
| color: var(--success); | |
| } | |
| .tree-role-tool { | |
| color: var(--muted); | |
| } | |
| .tree-muted { | |
| color: var(--muted); | |
| } | |
| .tree-error { | |
| color: var(--error); | |
| } | |
| .tree-compaction { | |
| color: var(--borderAccent); | |
| } | |
| .tree-branch-summary { | |
| color: var(--warning); | |
| } | |
| .tree-custom-message { | |
| color: var(--customMessageLabel); | |
| } | |
| .tree-status { | |
| padding: 4px 12px; | |
| font-size: 10px; | |
| color: var(--muted); | |
| flex-shrink: 0; | |
| } | |
| /* Main content */ | |
| #content { | |
| flex: 1; | |
| overflow-y: auto; | |
| padding: var(--line-height) calc(var(--line-height) * 2); | |
| display: flex; | |
| flex-direction: column; | |
| align-items: center; | |
| } | |
| #content > * { | |
| width: 100%; | |
| max-width: 800px; | |
| } | |
| /* Help bar */ | |
| .help-bar { | |
| font-size: 11px; | |
| color: var(--warning); | |
| margin-bottom: var(--line-height); | |
| display: flex; | |
| align-items: center; | |
| gap: 12px; | |
| } | |
| .download-json-btn { | |
| font-size: 10px; | |
| padding: 2px 8px; | |
| background: var(--container-bg); | |
| border: 1px solid var(--border); | |
| border-radius: 3px; | |
| color: var(--text); | |
| cursor: pointer; | |
| font-family: inherit; | |
| } | |
| .download-json-btn:hover { | |
| background: var(--hover); | |
| border-color: var(--borderAccent); | |
| } | |
| /* Header */ | |
| .header { | |
| background: var(--container-bg); | |
| border-radius: 4px; | |
| padding: var(--line-height); | |
| margin-bottom: var(--line-height); | |
| } | |
| .header h1 { | |
| font-size: 12px; | |
| font-weight: bold; | |
| color: var(--borderAccent); | |
| margin-bottom: var(--line-height); | |
| } | |
| .header-info { | |
| display: flex; | |
| flex-direction: column; | |
| gap: 0; | |
| font-size: 11px; | |
| } | |
| .info-item { | |
| color: var(--dim); | |
| display: flex; | |
| align-items: baseline; | |
| } | |
| .info-label { | |
| font-weight: 600; | |
| margin-right: 8px; | |
| min-width: 100px; | |
| } | |
| .info-value { | |
| color: var(--text); | |
| flex: 1; | |
| } | |
| /* Messages */ | |
| #messages { | |
| display: flex; | |
| flex-direction: column; | |
| gap: var(--line-height); | |
| } | |
| .message-timestamp { | |
| font-size: 10px; | |
| color: var(--dim); | |
| opacity: 0.8; | |
| } | |
| .user-message { | |
| background: var(--userMessageBg); | |
| color: var(--userMessageText); | |
| padding: var(--line-height); | |
| border-radius: 4px; | |
| position: relative; | |
| } | |
| .assistant-message { | |
| padding: 0; | |
| position: relative; | |
| } | |
| /* Copy link button - appears on hover */ | |
| .copy-link-btn { | |
| position: absolute; | |
| top: 8px; | |
| right: 8px; | |
| width: 28px; | |
| height: 28px; | |
| padding: 6px; | |
| background: var(--container-bg); | |
| border: 1px solid var(--dim); | |
| border-radius: 4px; | |
| color: var(--muted); | |
| cursor: pointer; | |
| opacity: 0; | |
| transition: opacity 0.15s, background 0.15s, color 0.15s; | |
| display: flex; | |
| align-items: center; | |
| justify-content: center; | |
| z-index: 10; | |
| } | |
| .user-message:hover .copy-link-btn, | |
| .assistant-message:hover .copy-link-btn { | |
| opacity: 1; | |
| } | |
| .copy-link-btn:hover { | |
| background: var(--accent); | |
| color: var(--body-bg); | |
| border-color: var(--accent); | |
| } | |
| .copy-link-btn.copied { | |
| background: var(--success, #22c55e); | |
| color: white; | |
| border-color: var(--success, #22c55e); | |
| } | |
| /* Highlight effect for deep-linked messages */ | |
| .user-message.highlight, | |
| .assistant-message.highlight { | |
| animation: highlight-pulse 2s ease-out; | |
| } | |
| @keyframes highlight-pulse { | |
| 0% { | |
| box-shadow: 0 0 0 3px var(--accent); | |
| } | |
| 100% { | |
| box-shadow: 0 0 0 0 transparent; | |
| } | |
| } | |
| .assistant-message > .message-timestamp { | |
| padding-left: var(--line-height); | |
| } | |
| .assistant-text { | |
| padding: var(--line-height); | |
| padding-bottom: 0; | |
| } | |
| .message-timestamp + .assistant-text, | |
| .message-timestamp + .thinking-block { | |
| padding-top: 0; | |
| } | |
| .thinking-block + .assistant-text { | |
| padding-top: 0; | |
| } | |
| .thinking-text { | |
| padding: var(--line-height); | |
| color: var(--thinkingText); | |
| font-style: italic; | |
| white-space: pre-wrap; | |
| } | |
| .message-timestamp + .thinking-block .thinking-text, | |
| .message-timestamp + .thinking-block .thinking-collapsed { | |
| padding-top: 0; | |
| } | |
| .thinking-collapsed { | |
| display: none; | |
| padding: var(--line-height); | |
| color: var(--thinkingText); | |
| font-style: italic; | |
| } | |
| /* Tool execution */ | |
| .tool-execution { | |
| padding: var(--line-height); | |
| border-radius: 4px; | |
| } | |
| .tool-execution + .tool-execution { | |
| margin-top: var(--line-height); | |
| } | |
| .assistant-text + .tool-execution { | |
| margin-top: var(--line-height); | |
| } | |
| .tool-execution.pending { background: var(--toolPendingBg); } | |
| .tool-execution.success { background: var(--toolSuccessBg); } | |
| .tool-execution.error { background: var(--toolErrorBg); } | |
| .tool-header, .tool-name { | |
| font-weight: bold; | |
| } | |
| .tool-path { | |
| color: var(--accent); | |
| word-break: break-all; | |
| } | |
| .line-numbers { | |
| color: var(--warning); | |
| } | |
| .line-count { | |
| color: var(--dim); | |
| } | |
| .tool-command { | |
| font-weight: bold; | |
| white-space: pre-wrap; | |
| word-wrap: break-word; | |
| overflow-wrap: break-word; | |
| word-break: break-word; | |
| } | |
| .tool-output { | |
| margin-top: var(--line-height); | |
| color: var(--toolOutput); | |
| word-wrap: break-word; | |
| overflow-wrap: break-word; | |
| word-break: break-word; | |
| font-family: inherit; | |
| overflow-x: auto; | |
| } | |
| .tool-output > div, | |
| .output-preview, | |
| .output-full { | |
| margin: 0; | |
| padding: 0; | |
| line-height: var(--line-height); | |
| } | |
| .tool-output pre { | |
| margin: 0; | |
| padding: 0; | |
| font-family: inherit; | |
| color: inherit; | |
| white-space: pre-wrap; | |
| word-wrap: break-word; | |
| overflow-wrap: break-word; | |
| } | |
| .tool-output code { | |
| padding: 0; | |
| background: none; | |
| color: var(--text); | |
| } | |
| .tool-output.expandable { | |
| cursor: pointer; | |
| } | |
| .tool-output.expandable:hover { | |
| opacity: 0.9; | |
| } | |
| .tool-output.expandable .output-full { | |
| display: none; | |
| } | |
| .tool-output.expandable.expanded .output-preview { | |
| display: none; | |
| } | |
| .tool-output.expandable.expanded .output-full { | |
| display: block; | |
| } | |
| .ansi-line { | |
| white-space: pre-wrap; | |
| } | |
| .tool-images { | |
| } | |
| .tool-image { | |
| max-width: 100%; | |
| max-height: 500px; | |
| border-radius: 4px; | |
| margin: var(--line-height) 0; | |
| } | |
| .expand-hint { | |
| color: var(--toolOutput); | |
| } | |
| /* Diff */ | |
| .tool-diff { | |
| font-size: 11px; | |
| overflow-x: auto; | |
| white-space: pre; | |
| } | |
| .diff-added { color: var(--toolDiffAdded); } | |
| .diff-removed { color: var(--toolDiffRemoved); } | |
| .diff-context { color: var(--toolDiffContext); } | |
| /* Model change */ | |
| .model-change { | |
| padding: 0 var(--line-height); | |
| color: var(--dim); | |
| font-size: 11px; | |
| } | |
| .model-name { | |
| color: var(--borderAccent); | |
| font-weight: bold; | |
| } | |
| /* Compaction / Branch Summary - matches customMessage colors from TUI */ | |
| .compaction { | |
| background: var(--customMessageBg); | |
| border-radius: 4px; | |
| padding: var(--line-height); | |
| cursor: pointer; | |
| } | |
| .compaction-label { | |
| color: var(--customMessageLabel); | |
| font-weight: bold; | |
| } | |
| .compaction-collapsed { | |
| color: var(--customMessageText); | |
| } | |
| .compaction-content { | |
| display: none; | |
| color: var(--customMessageText); | |
| white-space: pre-wrap; | |
| margin-top: var(--line-height); | |
| } | |
| .compaction.expanded .compaction-collapsed { | |
| display: none; | |
| } | |
| .compaction.expanded .compaction-content { | |
| display: block; | |
| } | |
| /* System prompt */ | |
| .system-prompt { | |
| background: var(--customMessageBg); | |
| padding: var(--line-height); | |
| border-radius: 4px; | |
| margin-bottom: var(--line-height); | |
| } | |
| .system-prompt.expandable { | |
| cursor: pointer; | |
| } | |
| .system-prompt-header { | |
| font-weight: bold; | |
| color: var(--customMessageLabel); | |
| } | |
| .system-prompt-preview { | |
| color: var(--customMessageText); | |
| white-space: pre-wrap; | |
| word-wrap: break-word; | |
| font-size: 11px; | |
| margin-top: var(--line-height); | |
| } | |
| .system-prompt-expand-hint { | |
| color: var(--muted); | |
| font-style: italic; | |
| margin-top: 4px; | |
| } | |
| .system-prompt-full { | |
| display: none; | |
| color: var(--customMessageText); | |
| white-space: pre-wrap; | |
| word-wrap: break-word; | |
| font-size: 11px; | |
| margin-top: var(--line-height); | |
| } | |
| .system-prompt.expanded .system-prompt-preview, | |
| .system-prompt.expanded .system-prompt-expand-hint { | |
| display: none; | |
| } | |
| .system-prompt.expanded .system-prompt-full { | |
| display: block; | |
| } | |
| .system-prompt.provider-prompt { | |
| border-left: 3px solid var(--warning); | |
| } | |
| .system-prompt-note { | |
| font-size: 10px; | |
| font-style: italic; | |
| color: var(--muted); | |
| margin-top: 4px; | |
| } | |
| /* Tools list */ | |
| .tools-list { | |
| background: var(--customMessageBg); | |
| padding: var(--line-height); | |
| border-radius: 4px; | |
| margin-bottom: var(--line-height); | |
| } | |
| .tools-header { | |
| font-weight: bold; | |
| color: var(--customMessageLabel); | |
| margin-bottom: var(--line-height); | |
| } | |
| .tool-item { | |
| font-size: 11px; | |
| } | |
| .tool-item-name { | |
| font-weight: bold; | |
| color: var(--text); | |
| } | |
| .tool-item-desc { | |
| color: var(--dim); | |
| } | |
| .tool-params-hint { | |
| color: var(--muted); | |
| font-style: italic; | |
| } | |
| .tool-item:has(.tool-params-hint) { | |
| cursor: pointer; | |
| } | |
| .tool-params-hint::after { | |
| content: '[click to show parameters]'; | |
| } | |
| .tool-item.params-expanded .tool-params-hint::after { | |
| content: '[hide parameters]'; | |
| } | |
| .tool-params-content { | |
| display: none; | |
| margin-top: 4px; | |
| margin-left: 12px; | |
| padding-left: 8px; | |
| border-left: 1px solid var(--dim); | |
| } | |
| .tool-item.params-expanded .tool-params-content { | |
| display: block; | |
| } | |
| .tool-param { | |
| margin-bottom: 4px; | |
| font-size: 11px; | |
| } | |
| .tool-param-name { | |
| font-weight: bold; | |
| color: var(--text); | |
| } | |
| .tool-param-type { | |
| color: var(--dim); | |
| font-style: italic; | |
| } | |
| .tool-param-required { | |
| color: var(--warning, #e8a838); | |
| font-size: 10px; | |
| } | |
| .tool-param-optional { | |
| color: var(--dim); | |
| font-size: 10px; | |
| } | |
| .tool-param-desc { | |
| color: var(--dim); | |
| margin-left: 8px; | |
| } | |
| /* Hook/custom messages */ | |
| .hook-message { | |
| background: var(--customMessageBg); | |
| color: var(--customMessageText); | |
| padding: var(--line-height); | |
| border-radius: 4px; | |
| } | |
| .hook-type { | |
| color: var(--customMessageLabel); | |
| font-weight: bold; | |
| } | |
| /* Branch summary */ | |
| .branch-summary { | |
| background: var(--customMessageBg); | |
| padding: var(--line-height); | |
| border-radius: 4px; | |
| } | |
| .branch-summary-header { | |
| font-weight: bold; | |
| color: var(--borderAccent); | |
| } | |
| /* Error */ | |
| .error-text { | |
| color: var(--error); | |
| padding: 0 var(--line-height); | |
| } | |
| .tool-error { | |
| color: var(--error); | |
| } | |
| /* Images */ | |
| .message-images { | |
| margin-bottom: 12px; | |
| } | |
| .message-image { | |
| max-width: 100%; | |
| max-height: 400px; | |
| border-radius: 4px; | |
| margin: var(--line-height) 0; | |
| } | |
| /* Markdown content */ | |
| .markdown-content h1, | |
| .markdown-content h2, | |
| .markdown-content h3, | |
| .markdown-content h4, | |
| .markdown-content h5, | |
| .markdown-content h6 { | |
| color: var(--mdHeading); | |
| margin: var(--line-height) 0 0 0; | |
| font-weight: bold; | |
| } | |
| .markdown-content h1 { font-size: 1em; } | |
| .markdown-content h2 { font-size: 1em; } | |
| .markdown-content h3 { font-size: 1em; } | |
| .markdown-content h4 { font-size: 1em; } | |
| .markdown-content h5 { font-size: 1em; } | |
| .markdown-content h6 { font-size: 1em; } | |
| .markdown-content p { margin: 0; } | |
| .markdown-content p + p { margin-top: var(--line-height); } | |
| .markdown-content a { | |
| color: var(--mdLink); | |
| text-decoration: underline; | |
| } | |
| .markdown-content code { | |
| background: rgba(128, 128, 128, 0.2); | |
| color: var(--mdCode); | |
| padding: 0 4px; | |
| border-radius: 3px; | |
| font-family: inherit; | |
| } | |
| .markdown-content pre { | |
| background: transparent; | |
| margin: var(--line-height) 0; | |
| overflow-x: auto; | |
| } | |
| .markdown-content pre code { | |
| display: block; | |
| background: none; | |
| color: var(--text); | |
| } | |
| .markdown-content blockquote { | |
| border-left: 3px solid var(--mdQuoteBorder); | |
| padding-left: var(--line-height); | |
| margin: var(--line-height) 0; | |
| color: var(--mdQuote); | |
| font-style: italic; | |
| } | |
| .markdown-content ul, | |
| .markdown-content ol { | |
| margin: var(--line-height) 0; | |
| padding-left: calc(var(--line-height) * 2); | |
| } | |
| .markdown-content li { margin: 0; } | |
| .markdown-content li::marker { color: var(--mdListBullet); } | |
| .markdown-content hr { | |
| border: none; | |
| border-top: 1px solid var(--mdHr); | |
| margin: var(--line-height) 0; | |
| } | |
| .markdown-content table { | |
| border-collapse: collapse; | |
| margin: 0.5em 0; | |
| width: 100%; | |
| } | |
| .markdown-content th, | |
| .markdown-content td { | |
| border: 1px solid var(--mdCodeBlockBorder); | |
| padding: 6px 10px; | |
| text-align: left; | |
| } | |
| .markdown-content th { | |
| background: rgba(128, 128, 128, 0.1); | |
| font-weight: bold; | |
| } | |
| .markdown-content img { | |
| max-width: 100%; | |
| border-radius: 4px; | |
| } | |
| /* Syntax highlighting */ | |
| .hljs { background: transparent; color: var(--text); } | |
| .hljs-comment, .hljs-quote { color: var(--syntaxComment); } | |
| .hljs-keyword, .hljs-selector-tag { color: var(--syntaxKeyword); } | |
| .hljs-number, .hljs-literal { color: var(--syntaxNumber); } | |
| .hljs-string, .hljs-doctag { color: var(--syntaxString); } | |
| /* Function names: hljs v11 uses .hljs-title.function_ compound class */ | |
| .hljs-function, .hljs-title, .hljs-title.function_, .hljs-section, .hljs-name { color: var(--syntaxFunction); } | |
| /* Types: hljs v11 uses .hljs-title.class_ for class names */ | |
| .hljs-type, .hljs-class, .hljs-title.class_, .hljs-built_in { color: var(--syntaxType); } | |
| .hljs-attr, .hljs-variable, .hljs-variable.language_, .hljs-params, .hljs-property { color: var(--syntaxVariable); } | |
| .hljs-meta, .hljs-meta .hljs-keyword, .hljs-meta .hljs-string { color: var(--syntaxKeyword); } | |
| .hljs-operator { color: var(--syntaxOperator); } | |
| .hljs-punctuation { color: var(--syntaxPunctuation); } | |
| .hljs-subst { color: var(--text); } | |
| /* Footer */ | |
| .footer { | |
| margin-top: 48px; | |
| padding: 20px; | |
| text-align: center; | |
| color: var(--dim); | |
| font-size: 10px; | |
| } | |
| /* Mobile */ | |
| #hamburger { | |
| display: none; | |
| position: fixed; | |
| top: 10px; | |
| left: 10px; | |
| z-index: 100; | |
| padding: 3px 8px; | |
| font-size: 12px; | |
| font-family: inherit; | |
| background: transparent; | |
| color: var(--muted); | |
| border: 1px solid var(--dim); | |
| border-radius: 3px; | |
| cursor: pointer; | |
| } | |
| #hamburger:hover { | |
| color: var(--text); | |
| border-color: var(--text); | |
| } | |
| #sidebar-overlay { | |
| display: none; | |
| position: fixed; | |
| top: 0; | |
| left: 0; | |
| right: 0; | |
| bottom: 0; | |
| background: rgba(0, 0, 0, 0.5); | |
| z-index: 98; | |
| } | |
| @media (max-width: 900px) { | |
| #sidebar { | |
| position: fixed; | |
| left: -400px; | |
| width: 400px; | |
| top: 0; | |
| bottom: 0; | |
| height: 100vh; | |
| z-index: 99; | |
| transition: left 0.3s; | |
| } | |
| #sidebar.open { | |
| left: 0; | |
| } | |
| #sidebar-overlay.open { | |
| display: block; | |
| } | |
| #hamburger { | |
| display: block; | |
| } | |
| .sidebar-close { | |
| display: block; | |
| } | |
| #content { | |
| padding: var(--line-height) 16px; | |
| } | |
| #content > * { | |
| max-width: 100%; | |
| } | |
| } | |
| @media (max-width: 500px) { | |
| #sidebar { | |
| width: 100vw; | |
| left: -100vw; | |
| } | |
| } | |
| @media print { | |
| #sidebar, #sidebar-toggle { display: none !important; } | |
| body { background: white; color: black; } | |
| #content { max-width: none; } | |
| } | |
| </style> | |
| </head> | |
| <body> | |
| <button id="hamburger" title="Open sidebar"><svg width="14" height="14" viewBox="0 0 24 24" fill="currentColor" stroke="none"><circle cx="6" cy="6" r="2.5"/><circle cx="6" cy="18" r="2.5"/><circle cx="18" cy="12" r="2.5"/><rect x="5" y="6" width="2" height="12"/><path d="M6 12h10c1 0 2 0 2-2V8"/></svg></button> | |
| <div id="sidebar-overlay"></div> | |
| <div id="app"> | |
| <aside id="sidebar"> | |
| <div class="sidebar-header"> | |
| <div class="sidebar-controls"> | |
| <input type="text" class="sidebar-search" id="tree-search" placeholder="Search..."> | |
| </div> | |
| <div class="sidebar-filters"> | |
| <button class="filter-btn active" data-filter="default" title="Hide settings entries">Default</button> | |
| <button class="filter-btn" data-filter="no-tools" title="Default minus tool results">No-tools</button> | |
| <button class="filter-btn" data-filter="user-only" title="Only user messages">User</button> | |
| <button class="filter-btn" data-filter="labeled-only" title="Only labeled entries">Labeled</button> | |
| <button class="filter-btn" data-filter="all" title="Show everything">All</button> | |
| <button class="sidebar-close" id="sidebar-close" title="Close">✕</button> | |
| </div> | |
| </div> | |
| <div class="tree-container" id="tree-container"></div> | |
| <div class="tree-status" id="tree-status"></div> | |
| </aside> | |
| <main id="content"> | |
| <div id="header-container"></div> | |
| <div id="messages"></div> | |
| </main> | |
| <div id="image-modal" class="image-modal"> | |
| <img id="modal-image" src="" alt=""> | |
| </div> | |
| </div> | |
| <script id="session-data" type="application/json">eyJoZWFkZXIiOnsidHlwZSI6InNlc3Npb24iLCJ2ZXJzaW9uIjozLCJpZCI6IjE4YTA4MzI5LWVmZWYtNGMwYi05OGIxLTc1YjhhNDIzNmE5MyIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6MzA6MDkuNzcyWiIsImN3ZCI6Ii9ob21lL3dyL2doL3RyeV93ZWJfZ3B1In0sImVudHJpZXMiOlt7InR5cGUiOiJtb2RlbF9jaGFuZ2UiLCJpZCI6IjQwZWI1YjI4IiwicGFyZW50SWQiOm51bGwsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6MzA6MDkuNzczWiIsInByb3ZpZGVyIjoiYW50aHJvcGljIiwibW9kZWxJZCI6ImNsYXVkZS1vcHVzLTQtNiJ9LHsidHlwZSI6InRoaW5raW5nX2xldmVsX2NoYW5nZSIsImlkIjoiNTcwM2VlYTciLCJwYXJlbnRJZCI6IjQwZWI1YjI4IiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjozMDowOS43NzNaIiwidGhpbmtpbmdMZXZlbCI6ImhpZ2gifSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiJkNmU3NDVjZCIsInBhcmVudElkIjoiNTcwM2VlYTciLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjMwOjE1LjY3OFoiLCJtZXNzYWdlIjp7InJvbGUiOiJ1c2VyIiwiY29udGVudCI6W3sidHlwZSI6InRleHQiLCJ0ZXh0IjoiaHR0cHM6Ly9odWdnaW5nZmFjZS5jby9ibG9nL3RyYW5zZm9ybWVyc2pzLXY0In1dLCJ0aW1lc3RhbXAiOjE3NzEwMzYyMTU2NjR9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiJlNzRmZmQ5ZiIsInBhcmVudElkIjoiZDZlNzQ1Y2QiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjMwOjE2LjU2MloiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbXSwiYXBpIjoiYW50aHJvcGljLW1lc3NhZ2VzIiwicHJvdmlkZXIiOiJhbnRocm9waWMiLCJtb2RlbCI6ImNsYXVkZS1vcHVzLTQtNiIsInVzYWdlIjp7ImlucHV0IjowLCJvdXRwdXQiOjAsImNhY2hlUmVhZCI6MCwiY2FjaGVXcml0ZSI6MCwidG90YWxUb2tlbnMiOjAsImNvc3QiOnsiaW5wdXQiOjAsIm91dHB1dCI6MCwiY2FjaGVSZWFkIjowLCJjYWNoZVdyaXRlIjowLCJ0b3RhbCI6MH19LCJzdG9wUmVhc29uIjoiYWJvcnRlZCIsInRpbWVzdGFtcCI6MTc3MTAzNjIxNTY3MSwiZXJyb3JNZXNzYWdlIjoiT3BlcmF0aW9uIGFib3J0ZWQifX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiNmVhOGNkYTAiLCJwYXJlbnRJZCI6ImU3NGZmZDlmIiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjozMToyOS41MTlaIiwibWVzc2FnZSI6eyJyb2xlIjoidXNlciIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6Imh0dHBzOi8vaHVnZ2luZ2ZhY2UuY28vYmxvZy90cmFuc2Zvcm1lcnNqcy12NFxuXG5jYW4gd2UgYnVpbGQgYSBzdGF0aWMgd2Vic2l0ZSB0byBpbXBsbWVuZXQgYSB0b2tlbml6ZXIgd2VicGFnZSB1c2luZyB0aGlzPyBsZXQncyB1c2UgZ3B0NG8ncyB0b2tlbml6ZXIsIGNyZWF0ZSB0aGUgc3RhdGljIHdlYnBhZ2UsIGNvbnRhaW5pbmcgYSBpbnB1dCBib3gsIGEgYnV0dG9uLCBhbmQgb3V0cHV0IGJveCwgdGhlbiBzdGFydCB0aGUgc2VydmVyIn1dLCJ0aW1lc3RhbXAiOjE3NzEwMzYyODk1MDZ9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiJlOTU3NDllOSIsInBhcmVudElkIjoiNmVhOGNkYTAiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjMxOjQyLjQzMVoiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJcblxuIn0seyJ0eXBlIjoidGhpbmtpbmciLCJ0aGlua2luZyI6IlRoZSB1c2VyIHdhbnRzIG1lIHRvIGJ1aWxkIGEgc3RhdGljIHdlYnNpdGUgdGhhdCBpbXBsZW1lbnRzIGEgdG9rZW5pemVyIHVzaW5nIFRyYW5zZm9ybWVycy5qcyB2NCwgc3BlY2lmaWNhbGx5IHVzaW5nIEdQVC00bydzIHRva2VuaXplci4gTGV0IG1lIGZpcnN0IGNoZWNrIHdoYXQncyBpbiB0aGUgY3VycmVudCBkaXJlY3RvcnksIHRoZW4gY3JlYXRlIHRoZSBzdGF0aWMgd2VicGFnZS5cblxuTGV0IG1lIHRoaW5rIGFib3V0IHdoYXQgd2UgbmVlZDpcbjEuIEEgc3RhdGljIEhUTUwgcGFnZSB3aXRoIGFuIGlucHV0IGJveCwgYnV0dG9uLCBhbmQgb3V0cHV0IGJveFxuMi4gVXNlIFRyYW5zZm9ybWVycy5qcyB2NCB0byBsb2FkIEdQVC00bydzIHRva2VuaXplclxuMy4gU3RhcnQgYSBzZXJ2ZXJcblxuR1BULTRvIHVzZXMgdGhlIGBvMjAwa19iYXNlYCB0b2tlbml6ZXIuIEluIFRyYW5zZm9ybWVycy5qcywgd2UgY2FuIGxvYWQgaXQgdmlhIGBBdXRvVG9rZW5pemVyLmZyb21fcHJldHJhaW5lZGAuIFRoZSBtb2RlbCBuYW1lIGZvciBHUFQtNG8ncyB0b2tlbml6ZXIgd291bGQgYmUgc29tZXRoaW5nIGxpa2UgYFhlbm92YS9ncHQtNG9gIG9yIHdlIG1pZ2h0IG5lZWQgdG8gY2hlY2suXG5cblxuSSdsbCB1c2UgdGhlIFRyYW5zZm9ybWVycy5qcyBsaWJyYXJ5IGZyb20gYSBDRE4gdG8gbG9hZCB0aGUgdG9rZW5pemVyLCB0aG91Z2ggSSBuZWVkIHRvIHZlcmlmeSB3aGljaCB2ZXJzaW9uIGlzIGFjdHVhbGx5IGF2YWlsYWJsZSBzaW5jZSB2NCBtaWdodCBzdGlsbCBiZSBpbiBkZXZlbG9wbWVudC4gSSdsbCBjcmVhdGUgYW4gSFRNTCBwYWdlIHRoYXQgaW1wb3J0cyB0aGUgdG9rZW5pemVyIGxpYnJhcnkgYW5kIHNldHMgdXAgdGhlIGludGVyZmFjZSBmb3IgdG9rZW5pemluZyB0ZXh0LiIsInRoaW5raW5nU2lnbmF0dXJlIjoiRXVNTENrWUlDeGdDS2tCR0h3VVFacWIrOERYdWc4ckFZeHVsbmI3YmFtbkdzK0tYeVM2bzRVOXI1bzd2WkFpekhwRGJhRDBmOXpjWjhRRHRscHIvK2RBa05hQ2ZtcDVkRVl2RUVneVFxWTk1Y2kxMGtwcmRSNTRhRER3d2ZDdFFRMi92cWNSUExTSXdNMzNSaDhyUlI2c1VEcDRWcFFhc3hPaUhvQVpHMXhxcnJ3YjNJZnRSMWFiNDNvZElCVUc4ZXZpV1AwMkI0Y2VZS3NvS3EyTStvNy8wQVVTQURPWGo4QjlrS3Z4WWJyZm1FN2dmTjRYa1JCWENVcldQM1lTblBua0FEQUZhNWM5LzhTNi82eWZVM2RrczY5YUd1TWdtaXA1Z0NhajZrWFBqcnU1OWZubjd5VVBZanVHbnZ0TzFaOGhTSUxkZFhkUUR3ZitLYXhzK2xFUUhDRkdhQWxSb29WZDU0bHJOY3dLS0FWUWZGa2tpQWxHdUlWM2hOdlRNZ1o5WjA2SmkxVXN4WHR2TmVZSVI0eUtmbjVEckNlM0VPdFlFNTlENzhEMGQ4T3pnRDdCd0ZEQ2VPdTEwcCtYTWtEL3J5eU8rQzNoMjVGMHBzOFJWd0oxbWpQWTM4Z09DRE9jaE8wOUJkSzZVcC80RXc0WVgxODh5aUVlZG45ODU4c0xITXJsL3ZOOEpkN0tjamdhUmNkcEhPMEU1VXYxaG9zajFIVlJzR1B4dVNwWHJudTJHK2crNVF4ZldJN25GSm1BUjdhcU1YMVZ4NlZxZ1lrSUdDNG5hdTRVMnhyV1dJMGVsa2Y1Ykt2Nk5TdlR3ZVdVME1PM3ZqY1R6WE9QZjFJOVRkaEdVTzhHRXcvc3VGelVjRjJobDZHZVFYYUlCUnI3OHcvQnZBWERkK1FqRC93cnQyM2I5U3NDRVhFVWZXK0xBRFVpNjVhZlBYRG5kK1NJM0kybDNROFhXdjByY0NwRCtpNk1PS2NidTlxZUZ5dGYwWFNJRVdUdHNBSk9URHdKU0RGK2JQQnI2RmhuKzVBWHdmVGhsd2QrczMrekRyd0JUbXJBMXpmWDJ0UEYrYkg0T3V1T09VZDRpV0dmdy8rcnJhYTZvNWhhQU9BendBbE1FdUkyLzhwVncwVkhtYWhCSEk4YWQ2SUFsVGx6ejdXbFJIMjBlaS9abC8wSlNJa0RpSlFSdDhRZFM2MS9ndDNqUE9Dcm92UittR05ta2ZGbVZUSjZDOGVjTlhCQUs2bWdiZzZ0QlZjQlVGemd6V2NvakJ2dXkrM3VlamgwQjR1VzdPMUtuTWxYVmtaWDhnTU5SZWFtb0tVeUsvbHhqS0YxOGttZE15M3FTdkdoNUlDRVJsdjZJczdOVTRoTjNvSy96aGF6TW1PZ2lJcjRqb0REOVV5bVhyaC82S1Nta2RSNUdtNDZjZUhrN3RUODNoZkNHSmV4V2J1Ri9NaXRnTkZFUjhIZ2lpTDdlSDRpZjRIZGxRbXNZSkRiRWxTbFpxUXhFTEx5Zjh2Z3F5QkM4UnNiUzFoVXVHREZldjl0cEFhQmhYTnVzM2h4aHVmN1hsaWt6QXJXU3dNbFg4NGlVbzB5YjRndGJQZ2trdlZhazJ2dm1QR1BPbzZ6S2tacksyVmttVm53MEpuenZNTk1hNHpNYjIvNzgzZHc4azg2dkhHNjZwSlZZUjBjRjlXQ1NLeGNPdGVnQyt6SWZKeWpNZ0Y0QTcza2RtSHFoSnk2cFJ5YWF0WEFTWGxYVXNYTWNNRDZjWGZkRzM3SzJLWEptQlh1dEc3TDg3QUVlV09WYlpaZFBNdW1rckozcWZQdXdxbFBjM0czSlJhZDNWTUE2R2JLWjRIRnpzSzF6azI5a0FDV2tGNE5KZ0hVaFZ1ZjlDNmpTV2dtaWtQTmlBVjBZYktkV3lORVJGampla2FYQmFaVFEyRUQxZkd0SXNzNWFzZ1pDMTlkSFlyUEVnenRtZG9PR3MvYnZHNWMvczFQQ0hxL0xFdk5sSEZjWFdUU281cDhmNW9LNkRhZld5YVJXWG53NkcwM0ZlcG5KOHA2QjJZQjUrTFdKUHo1bTNXVWN2cDlvaWNYVjdVZ3VjbHEwTnU3WVJHeVgyWkRrdVFsZ2J6eFZPNDd1L1pLbkhySmVnek5XZHVqSTJhYUxhZWdac0JNUFVzRGxoYktjR3lMbXc4MGpqekNjbTlaNVA0TTRnRGlhN2RVTktyUHpQWmRSZXVQZ3V4RlY5Y3VUUTQ1YW5JOTd0bXNaRnFHb3RmVlhQYmkzaURMWGF3czIrNEpSSVZRMDNCMVlOdSsrS1VraXYvcUxMcjNKR2VzV2xCdjYxTFVSNVYySkxUTUVERTlrTVdjK0t5c1Z0RU9NL0dMN3hVT2VzaXozZjFtRGQ5bUs4SXNjcjkvbEh2UlVEb1Bsb3JzdXY4aG82anpmcHc5bXUwbFlTQVRtUWsyNlpEU1BLNGM1TVo1cWdjOHIzWVVDZVVCV2Z6WS9tL29kS3Focm45bTZPREFJMTVCQkpoNndTS01wemNtZ3AwdzVHODVkUlZ2RmJOdk9LcjhxdDZMWTJ3aGJyTVpFMjgzejQyeWJCTEtYcmp6T3ZWL0NiYVY3Q29qN2NuVmZhNy9LWjZXVTF1cGNVYy9kYmMrTE5OU0ZzR0VJLy9YU1ZhZU4xQmdCIn0seyJ0eXBlIjoidG9vbENhbGwiLCJpZCI6InRvb2x1XzAxSE4yNzFCUWo2cUFCZDlocjFTSEo5YyIsIm5hbWUiOiJiYXNoIiwiYXJndW1lbnRzIjp7ImNvbW1hbmQiOiJscyAvaG9tZS93ci9naC90cnlfd2ViX2dwdS8ifX1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjE3ODgsIm91dHB1dCI6NDkxLCJjYWNoZVJlYWQiOjAsImNhY2hlV3JpdGUiOjAsInRvdGFsVG9rZW5zIjoyMjc5LCJjb3N0Ijp7ImlucHV0IjowLjAwODk0LCJvdXRwdXQiOjAuMDEyMjc1MDAwMDAwMDAwMDAxLCJjYWNoZVJlYWQiOjAsImNhY2hlV3JpdGUiOjAsInRvdGFsIjowLjAyMTIxNX19LCJzdG9wUmVhc29uIjoidG9vbFVzZSIsInRpbWVzdGFtcCI6MTc3MTAzNjI4OTUxOH19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6ImI4ZGQ5NGI5IiwicGFyZW50SWQiOiJlOTU3NDllOSIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6MzE6NDIuNDM0WiIsIm1lc3NhZ2UiOnsicm9sZSI6InRvb2xSZXN1bHQiLCJ0b29sQ2FsbElkIjoidG9vbHVfMDFITjI3MUJRajZxQUJkOWhyMVNISjljIiwidG9vbE5hbWUiOiJiYXNoIiwiY29udGVudCI6W3sidHlwZSI6InRleHQiLCJ0ZXh0IjoiKG5vIG91dHB1dCkifV0sImlzRXJyb3IiOmZhbHNlLCJ0aW1lc3RhbXAiOjE3NzEwMzYzMDI0MzR9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiI5MmE2ZmUwMiIsInBhcmVudElkIjoiYjhkZDk0YjkiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjMyOjE0LjI5OFoiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidG9vbENhbGwiLCJpZCI6InRvb2x1XzAxTWs4VVVIYWpKS0xldXg3VGJURHFtZSIsIm5hbWUiOiJ3cml0ZSIsImFyZ3VtZW50cyI6eyJwYXRoIjoiL2hvbWUvd3IvZ2gvdHJ5X3dlYl9ncHUvaW5kZXguaHRtbCIsImNvbnRlbnQiOiI8IURPQ1RZUEUgaHRtbD5cbjxodG1sIGxhbmc9XCJlblwiPlxuPGhlYWQ+XG4gIDxtZXRhIGNoYXJzZXQ9XCJVVEYtOFwiPlxuICA8bWV0YSBuYW1lPVwidmlld3BvcnRcIiBjb250ZW50PVwid2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEuMFwiPlxuICA8dGl0bGU+R1BULTRvIFRva2VuaXplcjwvdGl0bGU+XG4gIDxzdHlsZT5cbiAgICAqIHsgYm94LXNpemluZzogYm9yZGVyLWJveDsgbWFyZ2luOiAwOyBwYWRkaW5nOiAwOyB9XG4gICAgYm9keSB7XG4gICAgICBmb250LWZhbWlseTogLWFwcGxlLXN5c3RlbSwgQmxpbmtNYWNTeXN0ZW1Gb250LCAnU2Vnb2UgVUknLCBSb2JvdG8sIHNhbnMtc2VyaWY7XG4gICAgICBiYWNrZ3JvdW5kOiAjMGYwZjBmO1xuICAgICAgY29sb3I6ICNlMGUwZTA7XG4gICAgICBtaW4taGVpZ2h0OiAxMDB2aDtcbiAgICAgIGRpc3BsYXk6IGZsZXg7XG4gICAgICBqdXN0aWZ5LWNvbnRlbnQ6IGNlbnRlcjtcbiAgICAgIHBhZGRpbmc6IDQwcHggMjBweDtcbiAgICB9XG4gICAgLmNvbnRhaW5lciB7XG4gICAgICBtYXgtd2lkdGg6IDcyMHB4O1xuICAgICAgd2lkdGg6IDEwMCU7XG4gICAgfVxuICAgIGgxIHtcbiAgICAgIGZvbnQtc2l6ZTogMS44cmVtO1xuICAgICAgbWFyZ2luLWJvdHRvbTogOHB4O1xuICAgICAgY29sb3I6ICNmZmY7XG4gICAgfVxuICAgIC5zdWJ0aXRsZSB7XG4gICAgICBjb2xvcjogIzg4ODtcbiAgICAgIG1hcmdpbi1ib3R0b206IDI0cHg7XG4gICAgICBmb250LXNpemU6IDAuOTVyZW07XG4gICAgfVxuICAgIGxhYmVsIHtcbiAgICAgIGRpc3BsYXk6IGJsb2NrO1xuICAgICAgZm9udC13ZWlnaHQ6IDYwMDtcbiAgICAgIG1hcmdpbi1ib3R0b206IDZweDtcbiAgICAgIGNvbG9yOiAjYmJiO1xuICAgIH1cbiAgICB0ZXh0YXJlYSB7XG4gICAgICB3aWR0aDogMTAwJTtcbiAgICAgIG1pbi1oZWlnaHQ6IDEyMHB4O1xuICAgICAgcGFkZGluZzogMTJweDtcbiAgICAgIGJvcmRlcjogMXB4IHNvbGlkICMzMzM7XG4gICAgICBib3JkZXItcmFkaXVzOiA4cHg7XG4gICAgICBiYWNrZ3JvdW5kOiAjMWExYTFhO1xuICAgICAgY29sb3I6ICNlMGUwZTA7XG4gICAgICBmb250LXNpemU6IDFyZW07XG4gICAgICBmb250LWZhbWlseTogaW5oZXJpdDtcbiAgICAgIHJlc2l6ZTogdmVydGljYWw7XG4gICAgfVxuICAgIHRleHRhcmVhOmZvY3VzIHsgb3V0bGluZTogbm9uZTsgYm9yZGVyLWNvbG9yOiAjNTU1OyB9XG4gICAgYnV0dG9uIHtcbiAgICAgIG1hcmdpbi10b3A6IDE2cHg7XG4gICAgICBwYWRkaW5nOiAxMHB4IDI4cHg7XG4gICAgICBmb250LXNpemU6IDFyZW07XG4gICAgICBmb250LXdlaWdodDogNjAwO1xuICAgICAgYm9yZGVyOiBub25lO1xuICAgICAgYm9yZGVyLXJhZGl1czogOHB4O1xuICAgICAgY3Vyc29yOiBwb2ludGVyO1xuICAgICAgYmFja2dyb3VuZDogIzI1NjNlYjtcbiAgICAgIGNvbG9yOiAjZmZmO1xuICAgICAgdHJhbnNpdGlvbjogYmFja2dyb3VuZCAwLjJzO1xuICAgIH1cbiAgICBidXR0b246aG92ZXIgeyBiYWNrZ3JvdW5kOiAjMWQ0ZWQ4OyB9XG4gICAgYnV0dG9uOmRpc2FibGVkIHsgYmFja2dyb3VuZDogIzQ0NDsgY3Vyc29yOiBub3QtYWxsb3dlZDsgfVxuICAgIC5zdGF0cyB7XG4gICAgICBtYXJnaW4tdG9wOiAyMHB4O1xuICAgICAgcGFkZGluZzogMTJweCAxNnB4O1xuICAgICAgYmFja2dyb3VuZDogIzFhMWExYTtcbiAgICAgIGJvcmRlcjogMXB4IHNvbGlkICMzMzM7XG4gICAgICBib3JkZXItcmFkaXVzOiA4cHg7XG4gICAgICBmb250LXNpemU6IDAuOTVyZW07XG4gICAgICBkaXNwbGF5OiBub25lO1xuICAgIH1cbiAgICAuc3RhdHMgc3BhbiB7IGNvbG9yOiAjNjBhNWZhOyBmb250LXdlaWdodDogNzAwOyB9XG4gICAgLm91dHB1dC1zZWN0aW9uIHsgbWFyZ2luLXRvcDogMjBweDsgZGlzcGxheTogbm9uZTsgfVxuICAgIC50b2tlbi1saXN0IHtcbiAgICAgIGRpc3BsYXk6IGZsZXg7XG4gICAgICBmbGV4LXdyYXA6IHdyYXA7XG4gICAgICBnYXA6IDRweDtcbiAgICAgIHBhZGRpbmc6IDEycHg7XG4gICAgICBiYWNrZ3JvdW5kOiAjMWExYTFhO1xuICAgICAgYm9yZGVyOiAxcHggc29saWQgIzMzMztcbiAgICAgIGJvcmRlci1yYWRpdXM6IDhweDtcbiAgICAgIG1pbi1oZWlnaHQ6IDQ4cHg7XG4gICAgICBsaW5lLWhlaWdodDogMS44O1xuICAgIH1cbiAgICAudG9rZW4ge1xuICAgICAgZGlzcGxheTogaW5saW5lLWJsb2NrO1xuICAgICAgcGFkZGluZzogMnB4IDZweDtcbiAgICAgIGJvcmRlci1yYWRpdXM6IDRweDtcbiAgICAgIGZvbnQtZmFtaWx5OiAnU0YgTW9ubycsICdGaXJhIENvZGUnLCAnQ29uc29sYXMnLCBtb25vc3BhY2U7XG4gICAgICBmb250LXNpemU6IDAuOXJlbTtcbiAgICAgIGN1cnNvcjogZGVmYXVsdDtcbiAgICAgIHBvc2l0aW9uOiByZWxhdGl2ZTtcbiAgICB9XG4gICAgLnRva2VuOmhvdmVyOjphZnRlciB7XG4gICAgICBjb250ZW50OiBhdHRyKGRhdGEtaWQpO1xuICAgICAgcG9zaXRpb246IGFic29sdXRlO1xuICAgICAgYm90dG9tOiAxMDAlO1xuICAgICAgbGVmdDogNTAlO1xuICAgICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVYKC01MCUpO1xuICAgICAgYmFja2dyb3VuZDogIzMzMztcbiAgICAgIGNvbG9yOiAjZmZmO1xuICAgICAgcGFkZGluZzogMnB4IDhweDtcbiAgICAgIGJvcmRlci1yYWRpdXM6IDRweDtcbiAgICAgIGZvbnQtc2l6ZTogMC43NXJlbTtcbiAgICAgIHdoaXRlLXNwYWNlOiBub3dyYXA7XG4gICAgICB6LWluZGV4OiAxMDtcbiAgICB9XG4gICAgLmlkcy1ib3gge1xuICAgICAgbWFyZ2luLXRvcDogMTJweDtcbiAgICAgIHBhZGRpbmc6IDEycHg7XG4gICAgICBiYWNrZ3JvdW5kOiAjMWExYTFhO1xuICAgICAgYm9yZGVyOiAxcHggc29saWQgIzMzMztcbiAgICAgIGJvcmRlci1yYWRpdXM6IDhweDtcbiAgICAgIGZvbnQtZmFtaWx5OiAnU0YgTW9ubycsICdGaXJhIENvZGUnLCAnQ29uc29sYXMnLCBtb25vc3BhY2U7XG4gICAgICBmb250LXNpemU6IDAuODVyZW07XG4gICAgICB3b3JkLWJyZWFrOiBicmVhay1hbGw7XG4gICAgICBjb2xvcjogI2FhYTtcbiAgICAgIG1heC1oZWlnaHQ6IDIwMHB4O1xuICAgICAgb3ZlcmZsb3cteTogYXV0bztcbiAgICB9XG4gICAgI3N0YXR1cyB7XG4gICAgICBtYXJnaW4tdG9wOiAxMnB4O1xuICAgICAgZm9udC1zaXplOiAwLjlyZW07XG4gICAgICBjb2xvcjogIzg4ODtcbiAgICB9XG4gICAgLnNwaW5uZXIge1xuICAgICAgZGlzcGxheTogaW5saW5lLWJsb2NrO1xuICAgICAgd2lkdGg6IDE0cHg7IGhlaWdodDogMTRweDtcbiAgICAgIGJvcmRlcjogMnB4IHNvbGlkICM1NTU7XG4gICAgICBib3JkZXItdG9wLWNvbG9yOiAjNjBhNWZhO1xuICAgICAgYm9yZGVyLXJhZGl1czogNTAlO1xuICAgICAgYW5pbWF0aW9uOiBzcGluIDAuNnMgbGluZWFyIGluZmluaXRlO1xuICAgICAgdmVydGljYWwtYWxpZ246IG1pZGRsZTtcbiAgICAgIG1hcmdpbi1yaWdodDogNnB4O1xuICAgIH1cbiAgICBAa2V5ZnJhbWVzIHNwaW4geyB0byB7IHRyYW5zZm9ybTogcm90YXRlKDM2MGRlZyk7IH0gfVxuICA8L3N0eWxlPlxuPC9oZWFkPlxuPGJvZHk+XG4gIDxkaXYgY2xhc3M9XCJjb250YWluZXJcIj5cbiAgICA8aDE+8J+UpCBHUFQtNG8gVG9rZW5pemVyPC9oMT5cbiAgICA8cCBjbGFzcz1cInN1YnRpdGxlXCI+UG93ZXJlZCBieSA8YSBocmVmPVwiaHR0cHM6Ly9odWdnaW5nZmFjZS5jby9kb2NzL3RyYW5zZm9ybWVycy5qc1wiIHN0eWxlPVwiY29sb3I6IzYwYTVmYVwiPlRyYW5zZm9ybWVycy5qczwvYT4g4oCUIHJ1bnMgZW50aXJlbHkgaW4geW91ciBicm93c2VyPC9wPlxuXG4gICAgPGxhYmVsIGZvcj1cImlucHV0XCI+SW5wdXQgVGV4dDwvbGFiZWw+XG4gICAgPHRleHRhcmVhIGlkPVwiaW5wdXRcIiBwbGFjZWhvbGRlcj1cIlR5cGUgb3IgcGFzdGUgdGV4dCBoZXJl4oCmXCI+SGVsbG8sIHdvcmxkISBUaGlzIGlzIGEgdGVzdCBvZiB0aGUgR1BULTRvIHRva2VuaXplciBydW5uaW5nIGluIHRoZSBicm93c2VyLjwvdGV4dGFyZWE+XG5cbiAgICA8YnV0dG9uIGlkPVwidG9rZW5pemUtYnRuXCIgZGlzYWJsZWQ+TG9hZGluZyB0b2tlbml6ZXLigKY8L2J1dHRvbj5cblxuICAgIDxkaXYgaWQ9XCJzdGF0dXNcIj48L2Rpdj5cblxuICAgIDxkaXYgY2xhc3M9XCJzdGF0c1wiIGlkPVwic3RhdHNcIj5cbiAgICAgIFRva2VuczogPHNwYW4gaWQ9XCJ0b2tlbi1jb3VudFwiPjA8L3NwYW4+ICZuYnNwO3wmbmJzcDsgQ2hhcmFjdGVyczogPHNwYW4gaWQ9XCJjaGFyLWNvdW50XCI+MDwvc3Bhbj5cbiAgICA8L2Rpdj5cblxuICAgIDxkaXYgY2xhc3M9XCJvdXRwdXQtc2VjdGlvblwiIGlkPVwib3V0cHV0LXNlY3Rpb25cIj5cbiAgICAgIDxsYWJlbD5Ub2tlbnMgPHNtYWxsIHN0eWxlPVwiZm9udC13ZWlnaHQ6NDAwO2NvbG9yOiM2NjZcIj4oaG92ZXIgZm9yIHRva2VuIElEKTwvc21hbGw+PC9sYWJlbD5cbiAgICAgIDxkaXYgY2xhc3M9XCJ0b2tlbi1saXN0XCIgaWQ9XCJ0b2tlbi1saXN0XCI+PC9kaXY+XG5cbiAgICAgIDxsYWJlbCBzdHlsZT1cIm1hcmdpbi10b3A6MTZweFwiPlRva2VuIElEczwvbGFiZWw+XG4gICAgICA8ZGl2IGNsYXNzPVwiaWRzLWJveFwiIGlkPVwiaWRzLWJveFwiPjwvZGl2PlxuICAgIDwvZGl2PlxuICA8L2Rpdj5cblxuICA8c2NyaXB0IHR5cGU9XCJtb2R1bGVcIj5cbiAgICBpbXBvcnQgeyBBdXRvVG9rZW5pemVyIH0gZnJvbSAnaHR0cHM6Ly9jZG4uanNkZWxpdnIubmV0L25wbS9AaHVnZ2luZ2ZhY2UvdHJhbnNmb3JtZXJzQDMnO1xuXG4gICAgY29uc3QgQ09MT1JTID0gW1xuICAgICAgJyMzYjMwMjQnLCcjMWUzYTJmJywnIzFlMmQzZCcsJyMzMzI0M2InLCcjM2IyNDI0JyxcbiAgICAgICcjMmUzYjFlJywnIzFlM2IzYicsJyMzYjFlMzMnLCcjM2IzNTFlJywnIzFlMmUzYicsXG4gICAgXTtcblxuICAgIGNvbnN0IHN0YXR1c0VsID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ3N0YXR1cycpO1xuICAgIGNvbnN0IGJ0biA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCd0b2tlbml6ZS1idG4nKTtcbiAgICBjb25zdCBpbnB1dEVsID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ2lucHV0Jyk7XG4gICAgY29uc3Qgc3RhdHNFbCA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCdzdGF0cycpO1xuICAgIGNvbnN0IHRva2VuQ291bnRFbCA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCd0b2tlbi1jb3VudCcpO1xuICAgIGNvbnN0IGNoYXJDb3VudEVsID0gZG9jdW1lbnQuZ2V0RWxlbWVudEJ5SWQoJ2NoYXItY291bnQnKTtcbiAgICBjb25zdCB0b2tlbkxpc3RFbCA9IGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCd0b2tlbi1saXN0Jyk7XG4gICAgY29uc3QgaWRzQm94RWwgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgnaWRzLWJveCcpO1xuICAgIGNvbnN0IG91dHB1dFNlY3Rpb24gPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgnb3V0cHV0LXNlY3Rpb24nKTtcblxuICAgIHN0YXR1c0VsLmlubmVySFRNTCA9ICc8c3BhbiBjbGFzcz1cInNwaW5uZXJcIj48L3NwYW4+IERvd25sb2FkaW5nICYgbG9hZGluZyBHUFQtNG8gdG9rZW5pemVy4oCmJztcblxuICAgIGxldCB0b2tlbml6ZXI7XG4gICAgdHJ5IHtcbiAgICAgIHRva2VuaXplciA9IGF3YWl0IEF1dG9Ub2tlbml6ZXIuZnJvbV9wcmV0cmFpbmVkKCdYZW5vdmEvZ3B0LTRvJyk7XG4gICAgICBzdGF0dXNFbC50ZXh0Q29udGVudCA9ICfinIUgVG9rZW5pemVyIHJlYWR5ISc7XG4gICAgICBidG4uZGlzYWJsZWQgPSBmYWxzZTtcbiAgICAgIGJ0bi50ZXh0Q29udGVudCA9ICdUb2tlbml6ZSc7XG4gICAgICBzZXRUaW1lb3V0KCgpID0+IHN0YXR1c0VsLnRleHRDb250ZW50ID0gJycsIDIwMDApO1xuICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgIHN0YXR1c0VsLnRleHRDb250ZW50ID0gJ+KdjCBGYWlsZWQgdG8gbG9hZCB0b2tlbml6ZXI6ICcgKyBlLm1lc3NhZ2U7XG4gICAgICBjb25zb2xlLmVycm9yKGUpO1xuICAgIH1cblxuICAgIGJ0bi5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsICgpID0+IHtcbiAgICAgIGNvbnN0IHRleHQgPSBpbnB1dEVsLnZhbHVlO1xuICAgICAgaWYgKCF0ZXh0IHx8ICF0b2tlbml6ZXIpIHJldHVybjtcblxuICAgICAgY29uc3QgZW5jb2RlZCA9IHRva2VuaXplci5lbmNvZGUodGV4dCk7XG4gICAgICAvLyBlbmNvZGVkIGlzIGFuIGFycmF5IG9mIHRva2VuIElEcyAoaW5jbHVkZXMgc3BlY2lhbCB0b2tlbnMpOyBza2lwIEJPUyBpZiBwcmVzZW50XG4gICAgICBjb25zdCBpZHMgPSBBcnJheS5mcm9tKGVuY29kZWQpO1xuXG4gICAgICAvLyBEZWNvZGUgZWFjaCB0b2tlbiBpbmRpdmlkdWFsbHkgdG8gZ2V0IGl0cyBzdHJpbmdcbiAgICAgIHRva2VuTGlzdEVsLmlubmVySFRNTCA9ICcnO1xuICAgICAgY29uc3QgZnJhZ21lbnRzID0gW107XG4gICAgICBmb3IgKGxldCBpID0gMDsgaSA8IGlkcy5sZW5ndGg7IGkrKykge1xuICAgICAgICBjb25zdCBkZWNvZGVkID0gdG9rZW5pemVyLmRlY29kZShbaWRzW2ldXSwgeyBza2lwX3NwZWNpYWxfdG9rZW5zOiBmYWxzZSB9KTtcbiAgICAgICAgZnJhZ21lbnRzLnB1c2goZGVjb2RlZCk7XG4gICAgICAgIGNvbnN0IHNwYW4gPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCdzcGFuJyk7XG4gICAgICAgIHNwYW4uY2xhc3NOYW1lID0gJ3Rva2VuJztcbiAgICAgICAgc3Bhbi50ZXh0Q29udGVudCA9IGRlY29kZWQucmVwbGFjZSgvIC9nLCAnwrcnKS5yZXBsYWNlKC9cXG4vZywgJ+KGtScpO1xuICAgICAgICBzcGFuLnN0eWxlLmJhY2tncm91bmQgPSBDT0xPUlNbaSAlIENPTE9SUy5sZW5ndGhdO1xuICAgICAgICBzcGFuLnNldEF0dHJpYnV0ZSgnZGF0YS1pZCcsIGlkc1tpXSk7XG4gICAgICAgIHRva2VuTGlzdEVsLmFwcGVuZENoaWxkKHNwYW4pO1xuICAgICAgfVxuXG4gICAgICBpZHNCb3hFbC50ZXh0Q29udGVudCA9ICdbJyArIGlkcy5qb2luKCcsICcpICsgJ10nO1xuICAgICAgdG9rZW5Db3VudEVsLnRleHRDb250ZW50ID0gaWRzLmxlbmd0aDtcbiAgICAgIGNoYXJDb3VudEVsLnRleHRDb250ZW50ID0gdGV4dC5sZW5ndGg7XG4gICAgICBzdGF0c0VsLnN0eWxlLmRpc3BsYXkgPSAnYmxvY2snO1xuICAgICAgb3V0cHV0U2VjdGlvbi5zdHlsZS5kaXNwbGF5ID0gJ2Jsb2NrJztcbiAgICB9KTtcblxuICAgIC8vIEFsbG93IEN0cmwrRW50ZXIgdG8gdG9rZW5pemVcbiAgICBpbnB1dEVsLmFkZEV2ZW50TGlzdGVuZXIoJ2tleWRvd24nLCAoZSkgPT4ge1xuICAgICAgaWYgKGUua2V5ID09PSAnRW50ZXInICYmIChlLmN0cmxLZXkgfHwgZS5tZXRhS2V5KSkge1xuICAgICAgICBidG4uY2xpY2soKTtcbiAgICAgIH1cbiAgICB9KTtcbiAgPC9zY3JpcHQ+XG48L2JvZHk+XG48L2h0bWw+XG4ifX1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjIyODcsIm91dHB1dCI6MjU0MSwiY2FjaGVSZWFkIjowLCJjYWNoZVdyaXRlIjowLCJ0b3RhbFRva2VucyI6NDgyOCwiY29zdCI6eyJpbnB1dCI6MC4wMTE0MzUsIm91dHB1dCI6MC4wNjM1MjUsImNhY2hlUmVhZCI6MCwiY2FjaGVXcml0ZSI6MCwidG90YWwiOjAuMDc0OTZ9fSwic3RvcFJlYXNvbiI6InRvb2xVc2UiLCJ0aW1lc3RhbXAiOjE3NzEwMzYzMDI0MzR9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiJkMDFhYjRiNSIsInBhcmVudElkIjoiOTJhNmZlMDIiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjMyOjE0LjMwN1oiLCJtZXNzYWdlIjp7InJvbGUiOiJ0b29sUmVzdWx0IiwidG9vbENhbGxJZCI6InRvb2x1XzAxTWs4VVVIYWpKS0xldXg3VGJURHFtZSIsInRvb2xOYW1lIjoid3JpdGUiLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJTdWNjZXNzZnVsbHkgd3JvdGUgNzA2NCBieXRlcyB0byAvaG9tZS93ci9naC90cnlfd2ViX2dwdS9pbmRleC5odG1sIn1dLCJpc0Vycm9yIjpmYWxzZSwidGltZXN0YW1wIjoxNzcxMDM2MzM0MzAwfX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiMDVjYjU4YzMiLCJwYXJlbnRJZCI6ImQwMWFiNGI1IiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjozMjoxNy44ODlaIiwibWVzc2FnZSI6eyJyb2xlIjoiYXNzaXN0YW50IiwiY29udGVudCI6W3sidHlwZSI6InRvb2xDYWxsIiwiaWQiOiJ0b29sdV8wMUZ1NlVuaVFQZThzYzVDaUxmanR2Qk4iLCJuYW1lIjoiYmFzaCIsImFyZ3VtZW50cyI6eyJjb21tYW5kIjoiY2QgL2hvbWUvd3IvZ2gvdHJ5X3dlYl9ncHUgJiYgcHl0aG9uMyAtbSBodHRwLnNlcnZlciA4MDgwICZcbmVjaG8gXCJTZXJ2ZXIgUElEOiAkIVwiXG5zbGVlcCAxXG5lY2hvIFwiU2VydmVyIHJ1bm5pbmcgYXQgaHR0cDovL2xvY2FsaG9zdDo4MDgwXCIifX1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjEsIm91dHB1dCI6MTA1LCJjYWNoZVJlYWQiOjAsImNhY2hlV3JpdGUiOjQ4NjQsInRvdGFsVG9rZW5zIjo0OTcwLCJjb3N0Ijp7ImlucHV0IjowLjAwMDAwNSwib3V0cHV0IjowLjAwMjYyNSwiY2FjaGVSZWFkIjowLCJjYWNoZVdyaXRlIjowLjAzMDQsInRvdGFsIjowLjAzMzAzfX0sInN0b3BSZWFzb24iOiJ0b29sVXNlIiwidGltZXN0YW1wIjoxNzcxMDM2MzM0MzA3fX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiNTNiYTg3YzIiLCJwYXJlbnRJZCI6IjA1Y2I1OGMzIiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjozMzo0Ni4xOTVaIiwibWVzc2FnZSI6eyJyb2xlIjoidG9vbFJlc3VsdCIsInRvb2xDYWxsSWQiOiJ0b29sdV8wMUZ1NlVuaVFQZThzYzVDaUxmanR2Qk4iLCJ0b29sTmFtZSI6ImJhc2giLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJTZXJ2ZXIgUElEOiAyODM4MDdcblNlcnZlciBydW5uaW5nIGF0IGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MFxuMTI3LjAuMC4xIC0gLSBbMTQvRmViLzIwMjYgMTA6MzM6MzFdIFwiR0VUIC8gSFRUUC8xLjFcIiAyMDAgLVxuMTI3LjAuMC4xIC0gLSBbMTQvRmViLzIwMjYgMTA6MzM6MzJdIGNvZGUgNDA0LCBtZXNzYWdlIEZpbGUgbm90IGZvdW5kXG4xMjcuMC4wLjEgLSAtIFsxNC9GZWIvMjAyNiAxMDozMzozMl0gXCJHRVQgL2Zhdmljb24uaWNvIEhUVFAvMS4xXCIgNDA0IC1cblxuXG5Db21tYW5kIGFib3J0ZWQifV0sImRldGFpbHMiOnt9LCJpc0Vycm9yIjp0cnVlLCJ0aW1lc3RhbXAiOjE3NzEwMzY0MjYxOTN9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiJjNzZlYmMyMSIsInBhcmVudElkIjoiNTNiYTg3YzIiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjMzOjQ2LjIxM1oiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbXSwiYXBpIjoiYW50aHJvcGljLW1lc3NhZ2VzIiwicHJvdmlkZXIiOiJhbnRocm9waWMiLCJtb2RlbCI6ImNsYXVkZS1vcHVzLTQtNiIsInVzYWdlIjp7ImlucHV0IjowLCJvdXRwdXQiOjAsImNhY2hlUmVhZCI6MCwiY2FjaGVXcml0ZSI6MCwidG90YWxUb2tlbnMiOjAsImNvc3QiOnsiaW5wdXQiOjAsIm91dHB1dCI6MCwiY2FjaGVSZWFkIjowLCJjYWNoZVdyaXRlIjowLCJ0b3RhbCI6MH19LCJzdG9wUmVhc29uIjoiYWJvcnRlZCIsInRpbWVzdGFtcCI6MTc3MTAzNjQyNjE5NywiZXJyb3JNZXNzYWdlIjoiT3BlcmF0aW9uIGFib3J0ZWQifX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiMTk5ZmE3NWIiLCJwYXJlbnRJZCI6ImM3NmViYzIxIiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjozNDoxMi4wNDlaIiwibWVzc2FnZSI6eyJyb2xlIjoidXNlciIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6Iml0J3Mgd29ya2luZywgbm93IHVwbG9hZCB0aGUgaW5kZXguaHRtbCB0byBnaXN0IHVzaW5nIGdoIGNsaSwgdGhlbiBnaXZlIG1lIHRoZSBsaW5rIn1dLCJ0aW1lc3RhbXAiOjE3NzEwMzY0NTIwNDZ9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiJiNWVlN2Y2MiIsInBhcmVudElkIjoiMTk5ZmE3NWIiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjM0OjE2LjQxMloiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidG9vbENhbGwiLCJpZCI6InRvb2x1XzAxRndLbm0ySHFUQ0NoQ3hlVHdZWjVRYyIsIm5hbWUiOiJiYXNoIiwiYXJndW1lbnRzIjp7ImNvbW1hbmQiOiJjZCAvaG9tZS93ci9naC90cnlfd2ViX2dwdSAmJiBnaCBnaXN0IGNyZWF0ZSBpbmRleC5odG1sIC1kIFwiR1BULTRvIFRva2VuaXplciAtIFN0YXRpYyB3ZWJwYWdlIHVzaW5nIFRyYW5zZm9ybWVycy5qc1wiIC1wIn19XSwiYXBpIjoiYW50aHJvcGljLW1lc3NhZ2VzIiwicHJvdmlkZXIiOiJhbnRocm9waWMiLCJtb2RlbCI6ImNsYXVkZS1vcHVzLTQtNiIsInVzYWdlIjp7ImlucHV0IjozLCJvdXRwdXQiOjk3LCJjYWNoZVJlYWQiOjQ4NjQsImNhY2hlV3JpdGUiOjI4MywidG90YWxUb2tlbnMiOjUyNDcsImNvc3QiOnsiaW5wdXQiOjAuMDAwMDE1MDAwMDAwMDAwMDAwMDAyLCJvdXRwdXQiOjAuMDAyNDI1LCJjYWNoZVJlYWQiOjAuMDAyNDMxOTk5OTk5OTk5OTk5NywiY2FjaGVXcml0ZSI6MC4wMDE3Njg3NTAwMDAwMDAwMDAxLCJ0b3RhbCI6MC4wMDY2NDA3NX19LCJzdG9wUmVhc29uIjoidG9vbFVzZSIsInRpbWVzdGFtcCI6MTc3MTAzNjQ1MjA0N319LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6IjEwZmIzZDg0IiwicGFyZW50SWQiOiJiNWVlN2Y2MiIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6MzQ6MTguNTkwWiIsIm1lc3NhZ2UiOnsicm9sZSI6InRvb2xSZXN1bHQiLCJ0b29sQ2FsbElkIjoidG9vbHVfMDFGd0tubTJIcVRDQ2hDeGVUd1laNVFjIiwidG9vbE5hbWUiOiJiYXNoIiwiY29udGVudCI6W3sidHlwZSI6InRleHQiLCJ0ZXh0IjoiLSBDcmVhdGluZyBnaXN0IGluZGV4Lmh0bWxcbuKckyBDcmVhdGVkIHB1YmxpYyBnaXN0IGluZGV4Lmh0bWxcbmh0dHBzOi8vZ2lzdC5naXRodWIuY29tL2lhbXdybS80MjQxMzJhYzE5Yzk5MzRjNDZkZTM5ZjA4OTYzMmZhZVxuIn1dLCJpc0Vycm9yIjpmYWxzZSwidGltZXN0YW1wIjoxNzcxMDM2NDU4NTkwfX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiMzI2OTExYzgiLCJwYXJlbnRJZCI6IjEwZmIzZDg0IiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjozNDoyMy4yNzJaIiwibWVzc2FnZSI6eyJyb2xlIjoiYXNzaXN0YW50IiwiY29udGVudCI6W3sidHlwZSI6InRleHQiLCJ0ZXh0IjoiSGVyZSdzIHlvdXIgZ2lzdDpcblxuKipHaXN0OioqIGh0dHBzOi8vZ2lzdC5naXRodWIuY29tL2lhbXdybS80MjQxMzJhYzE5Yzk5MzRjNDZkZTM5ZjA4OTYzMmZhZVxuXG5Zb3UgY2FuIGFsc28gdmlldyBpdCBhcyBhIGxpdmUgcGFnZSB2aWEgR2l0SHViJ3MgcmF3IHByZXZpZXcgb3IgdXNlIGEgc2VydmljZSBsaWtlIFtibC5vY2tzLm9yZ10oaHR0cHM6Ly9ibC5vY2tzLm9yZykgLyBbcmF3LmdpdGhhY2suY29tXShodHRwczovL3Jhdy5naXRoYWNrLmNvbSkuIEZvciBhIGRpcmVjdCByYXcgSFRNTCBsaW5rOlxuXG5odHRwczovL2dpc3QuZ2l0aGFjay5jb20vaWFtd3JtLzQyNDEzMmFjMTljOTkzNGM0NmRlMzlmMDg5NjMyZmFlL3Jhdy9pbmRleC5odG1sIn1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjEsIm91dHB1dCI6MTQ3LCJjYWNoZVJlYWQiOjUxNDcsImNhY2hlV3JpdGUiOjE2MiwidG90YWxUb2tlbnMiOjU0NTcsImNvc3QiOnsiaW5wdXQiOjAuMDAwMDA1LCJvdXRwdXQiOjAuMDAzNjc1MDAwMDAwMDAwMDAwMywiY2FjaGVSZWFkIjowLjAwMjU3MzUsImNhY2hlV3JpdGUiOjAuMDAxMDEyNSwidG90YWwiOjAuMDA3MjY2fX0sInN0b3BSZWFzb24iOiJzdG9wIiwidGltZXN0YW1wIjoxNzcxMDM2NDU4NTkwfX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiOTI1ODVjMWEiLCJwYXJlbnRJZCI6IjMyNjkxMWM4IiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjozNjoxOC42MDNaIiwibWVzc2FnZSI6eyJyb2xlIjoidXNlciIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6IndoYXQgYXJlIG90aGVyIHdlbGwga25vdyB0b2tlbml6ZXJzPyBJIHdhbnQgYWRkIGEgZHJvcGJhciBpbiB0aGUgd2VicGFnZSB0byBzZWxlY3QsIGFsc28gSSB3YW50IHRoZSB3ZWJwYWdlIHRvIGhhdmUgbWluaW1hbCBhc3RoZXRpY3MgbGlrZSBhc3RybyJ9XSwidGltZXN0YW1wIjoxNzcxMDM2NTc4NTk1fX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiZDE1NTMyMDYiLCJwYXJlbnRJZCI6IjkyNTg1YzFhIiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjozNjozMS4zNTNaIiwibWVzc2FnZSI6eyJyb2xlIjoiYXNzaXN0YW50IiwiY29udGVudCI6W3sidHlwZSI6InRleHQiLCJ0ZXh0IjoiXG5cbiJ9LHsidHlwZSI6InRoaW5raW5nIiwidGhpbmtpbmciOiJUaGUgdXNlciB3YW50czpcbjEuIEEgZHJvcGRvd24gdG8gc2VsZWN0IGZyb20gd2VsbC1rbm93biB0b2tlbml6ZXJzXG4yLiBNaW5pbWFsIGFlc3RoZXRpY3MgbGlrZSBBc3RybyAodGhlIHdlYiBmcmFtZXdvcmsncyBjbGVhbiwgbWluaW1hbCBkZXNpZ24pXG5cbkxldCBtZSB0aGluayBhYm91dCB3ZWxsLWtub3duIHRva2VuaXplcnMgYXZhaWxhYmxlIHZpYSBUcmFuc2Zvcm1lcnMuanM6XG5cbjEuICoqR1BULTRvKiogLSBgWGVub3ZhL2dwdC00b2AgKG8yMDBrX2Jhc2UpXG4yLiAqKkdQVC00IC8gR1BULTMuNSoqIC0gYFhlbm92YS9ncHQtNGAgKGNsMTAwa19iYXNlKVxuMy4gKipHUFQtMioqIC0gYFhlbm92YS9ncHQyYCAoZ3B0MiBCUEUpXG40LiAqKkxMYU1BIDMqKiAtIGBYZW5vdmEvbGxhbWEtMy10b2tlbml6ZXJgIG9yIGBtZXRhLWxsYW1hL01ldGEtTGxhbWEtMy04QmBcblxuXG41LiAqKk1pc3RyYWwqKiAtIGBYZW5vdmEvbWlzdHJhbC10b2tlbml6ZXItdjNgXG42LiAqKkdlbW1hKiogLSBgWGVub3ZhL2dlbW1hLXRva2VuaXplcmAgb3IgYGdvb2dsZS9nZW1tYS0yYmBcbjcuICoqVDUqKiAtIGBYZW5vdmEvdDUtc21hbGxgXG44LiAqKkJFUlQqKiAtIGBYZW5vdmEvYmVydC1iYXNlLXVuY2FzZWRgXG5cbkNsYXVkZSdzIHRva2VuaXplciBpc24ndCBhdmFpbGFibGUgb24gSHVnZ2luZ0ZhY2UsIGFuZCBEZWVwU2VlayBtaWdodCBiZSBhdmFpbGFibGUgYnV0IEkgc2hvdWxkIGZvY3VzIG9uIHRoZSBtb3N0IHBvcHVsYXIgYW5kIHJlbGlhYmx5IGFjY2Vzc2libGUgb3B0aW9ucyB0aHJvdWdoIFRyYW5zZm9ybWVycy5qcy5cblxuTm93IEknbSB0aGlua2luZyBhYm91dCB0aGUgZGVzaWduIGRpcmVjdGlvbuKAlEkgd2FudCB0byBjcmVhdGUgc29tZXRoaW5nIHdpdGggQXN0cm8ncyBhZXN0aGV0aWMgaW4iLCJ0aGlua2luZ1NpZ25hdHVyZSI6IiJ9XSwiYXBpIjoiYW50aHJvcGljLW1lc3NhZ2VzIiwicHJvdmlkZXIiOiJhbnRocm9waWMiLCJtb2RlbCI6ImNsYXVkZS1vcHVzLTQtNiIsInVzYWdlIjp7ImlucHV0IjozLCJvdXRwdXQiOjEyLCJjYWNoZVJlYWQiOjUzMDksImNhY2hlV3JpdGUiOjE4MiwidG90YWxUb2tlbnMiOjU1MDYsImNvc3QiOnsiaW5wdXQiOjAuMDAwMDE1MDAwMDAwMDAwMDAwMDAyLCJvdXRwdXQiOjAuMDAwMzAwMDAwMDAwMDAwMDAwMDMsImNhY2hlUmVhZCI6MC4wMDI2NTQ0OTk5OTk5OTk5OTk3LCJjYWNoZVdyaXRlIjowLjAwMTEzNzUsInRvdGFsIjowLjAwNDEwNjk5OTk5OTk5OTk5OTV9fSwic3RvcFJlYXNvbiI6ImFib3J0ZWQiLCJ0aW1lc3RhbXAiOjE3NzEwMzY1Nzg1OTgsImVycm9yTWVzc2FnZSI6Ik9wZXJhdGlvbiBhYm9ydGVkIn19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6IjdmMDA4N2UxIiwicGFyZW50SWQiOiJkMTU1MzIwNiIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6MzY6NTQuNDg0WiIsIm1lc3NhZ2UiOnsicm9sZSI6InVzZXIiLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJ5b3VyIGtub3dsZWRnZSBpcyBvdXRkYXRlZCwgc2VhcmNoIHRoZSB3ZWIgYW5kIGZpbmQgbW9yZSwgb3IgY2hlY2sgaHVnZ2luZ2ZhY2VzIHBhY2thZ2VzIn1dLCJ0aW1lc3RhbXAiOjE3NzEwMzY2MTQ0ODN9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiI5OGFmNDk0YiIsInBhcmVudElkIjoiN2YwMDg3ZTEiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjM3OjAxLjQ3M1oiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJcblxuIn0seyJ0eXBlIjoidGhpbmtpbmciLCJ0aGlua2luZyI6IlRoZSB1c2VyIHdhbnRzIG1lIHRvIGZpbmQgd2VsbC1rbm93biB0b2tlbml6ZXJzIGF2YWlsYWJsZSBpbiBUcmFuc2Zvcm1lcnMuanMgYW5kIHVwZGF0ZSB0aGUgd2VicGFnZSB3aXRoIGEgZHJvcGRvd24gc2VsZWN0b3IuIExldCBtZSBzZWFyY2ggZm9yIGF2YWlsYWJsZSB0b2tlbml6ZXJzIG9uIEh1Z2dpbmdGYWNlIHRoYXQgd29yayB3aXRoIFRyYW5zZm9ybWVycy5qcy4iLCJ0aGlua2luZ1NpZ25hdHVyZSI6IkV2d0NDa1lJQ3hnQ0trRGM3dUtGQUh1R211dWpWZzFHNkRUaDA2Yzk2c0owb0JnbmRCQ1hSU3pwZjd5U2YvcjArN3FPcVpqdm9Kc0tNVzFEaVN1TVNKMDRoQmhjdzZHYm5DcElFZ3hGalFjNHZOd241Q01VT0gwYURJcS9oWXl6c0VKalo4ZE04aUl3ejFrUDhNQVJsYUVxT1I3UmRPM1NRS3ZVNldYZ1ZxRzB6U2x3TDR5c2lmRWVtNDlTaityT0E4Q2EyOG9obW5QNUt1TUJPMmtvY3dZYzByR0lJamdTQ1pHZXVZVjNITGx0cDB1T0pETXlmNDhsMFpCZEt0QXNET0lLQVRSM1FraUpudXZDalRrb09acTdXelFDRmh5OVpncVlVUjZjT2k0ZkM2WllMcVQ0K0VXS0d6T0k4a2YyUktKVG53WFo3M0pXY1RsWFV1SzIvbW9ZcFpTdmtjV3FyM3RjVUh2NFJnbDUrTzArT28yU1JuVzEzNUpKUXZIM2syT3o2ZmZ2K0lBSTZnbzE4WkhOUmN4YzBEL3hnSmpkL3hqWTR3T1lUWmI4MnkyUUprNnFIKzlRNkxmUFQ2aFk0OEp6REJKc3dRVTRjNUdWK2EycWo0SkhhaG1oRHdjSjMwYmljMUhlSDBNSmxsVDhucXNOVi80WXJUZGNSYzRZQVE9PSJ9LHsidHlwZSI6InRvb2xDYWxsIiwiaWQiOiJ0b29sdV8wMTFnYTJ3eGl5d2NLTTY1b0J2bzlZenIiLCJuYW1lIjoiYmFzaCIsImFyZ3VtZW50cyI6eyJjb21tYW5kIjoiY3VybCAtcyBcImh0dHBzOi8vaHVnZ2luZ2ZhY2UuY28vYXBpL21vZGVscz9zZWFyY2g9dG9rZW5pemVyJmxpYnJhcnk9dHJhbnNmb3JtZXJzLmpzJnNvcnQ9ZG93bmxvYWRzJmxpbWl0PTMwXCIgfCBweXRob24zIC1jIFwiXG5pbXBvcnQganNvbiwgc3lzXG5kYXRhID0ganNvbi5sb2FkKHN5cy5zdGRpbilcbmZvciBtIGluIGRhdGE6XG4gICAgcHJpbnQoZlxcXCJ7bVsnaWQnXTo1MHN9ICBkb3dubG9hZHM9e20uZ2V0KCdkb3dubG9hZHMnLCc/Jyl9XFxcIilcblwiIDI+L2Rldi9udWxsIHwgaGVhZCAtNDAifX1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjMsIm91dHB1dCI6MjIyLCJjYWNoZVJlYWQiOjU0OTEsImNhY2hlV3JpdGUiOjIxLCJ0b3RhbFRva2VucyI6NTczNywiY29zdCI6eyJpbnB1dCI6MC4wMDAwMTUwMDAwMDAwMDAwMDAwMDIsIm91dHB1dCI6MC4wMDU1NSwiY2FjaGVSZWFkIjowLjAwMjc0NTQ5OTk5OTk5OTk5OTcsImNhY2hlV3JpdGUiOjAuMDAwMTMxMjUwMDAwMDAwMDAwMDIsInRvdGFsIjowLjAwODQ0MTc1fX0sInN0b3BSZWFzb24iOiJ0b29sVXNlIiwidGltZXN0YW1wIjoxNzcxMDM2NjE0NDgzfX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiNDc2YmUwZjIiLCJwYXJlbnRJZCI6Ijk4YWY0OTRiIiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjozNzowMi41ODZaIiwibWVzc2FnZSI6eyJyb2xlIjoidG9vbFJlc3VsdCIsInRvb2xDYWxsSWQiOiJ0b29sdV8wMTFnYTJ3eGl5d2NLTTY1b0J2bzlZenIiLCJ0b29sTmFtZSI6ImJhc2giLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJOZW9RdWFzYXIvS3Jvbm9zLVRva2VuaXplci1iYXNlICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTI0NTUxNlxuemFpLW9yZy9nbG0tNC12b2ljZS10b2tlbml6ZXIgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0xMjY3OTZcblF3ZW4vUXdlbjMtVFRTLVRva2VuaXplci0xMkh6ICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9NzE5NDVcbmZ4bWFydHkvdGlueS1sbGFtYS1mYXN0LXRva2VuaXplciAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MTM2NThcbk5lb1F1YXNhci9Lcm9ub3MtVG9rZW5pemVyLTJrICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9NTcyMFxuRVBGTC1WSUxBQi80TV90b2tlbml6ZXJzX3JnYl8xNmtfMjI0LTQ0OCAgICAgICAgICAgIGRvd25sb2Fkcz01MzA5XG5qdWxpZW4tYy9kdW1teS1kaWZmLXRva2VuaXplciAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTQ2NDNcbnJvYm90LXRlc3QvZHVtbXktdG9rZW5pemVyLWZhc3Qtd2l0aC1tb2RlbC1jb25maWcgICBkb3dubG9hZHM9NDMxMFxuYW1kLXF1YXJrL3RpbnktbGxhbWEtZmFzdC10b2tlbml6ZXIgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0yNzM4XG5ib3NvbmFpL2hpZ2dzLWF1ZGlvLXYyLXRva2VuaXplciAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTI0NTdcbk9wZW5NT1NTLVRlYW0vTU9TUy1BdWRpby1Ub2tlbml6ZXIgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MjM3MFxuRVBGTC1WSUxBQi80TV90b2tlbml6ZXJzX2h1bWFuLXBvc2VzXzFrXzggICAgICAgICAgIGRvd25sb2Fkcz0xNTY3XG5FUEZMLVZJTEFCLzRNX3Rva2VuaXplcnNfZGVwdGhfOGtfMjI0LTQ0OCAgICAgICAgICAgZG93bmxvYWRzPTE1NDhcbnl1Y29ybmV0dG8vdG9rZW5pemVyX3RpdG9rX3MxMjhfaW1hZ2VuZXQgICAgICAgICAgICBkb3dubG9hZHM9MTQyN1xuZXVzdGxiL2hpZ2dzLWF1ZGlvLXYyLXRva2VuaXplciAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0xMTgwXG5pYm0tZXNhLWdlb3NwYXRpYWwvVGVycmFNaW5kLTEuMC1Ub2tlbml6ZXItTFVMQyAgICAgZG93bmxvYWRzPTExMjhcbm1haXRyaXgtb3JnL1ZvaWxhLVRva2VuaXplciAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MTEwNFxuRVBGTC1WSUxBQi80TV90b2tlbml6ZXJzX25vcm1hbF84a18yMjQtNDQ4ICAgICAgICAgIGRvd25sb2Fkcz0xMDk0XG5LaWphaS9sbGF2YS1sbGFtYS0zLThiLXRleHQtZW5jb2Rlci10b2tlbml6ZXIgICAgICAgZG93bmxvYWRzPTg0NVxuQ2hhcmFuU2FpVmFkZGkvQmlnX0VuZGlhbl9Ub2tlbml6ZXJzICAgICAgICAgICAgICAgIGRvd25sb2Fkcz03OTRcbnl1Y29ybmV0dG8vdG9rZW5pemVyX3RpdG9rX2wzMl9pbWFnZW5ldCAgICAgICAgICAgICBkb3dubG9hZHM9NzIyXG5PcGVuTU9TUy1UZWFtL1hZX1Rva2VuaXplcl9UVFNEX1YwX2hmICAgICAgICAgICAgICAgZG93bmxvYWRzPTcxM1xuRVBGTC1WSUxBQi80TV90b2tlbml6ZXJzX3NlbXNlZ180a18yMjQtNDQ4ICAgICAgICAgIGRvd25sb2Fkcz03MDRcbkVQRkwtVklMQUIvNE1fdG9rZW5pemVyc19DTElQLUIxNl84a18yMjQtNDQ4ICAgICAgICBkb3dubG9hZHM9NjU4XG55dWNvcm5ldHRvL3Rva2VuaXplcl90aXRva19ibDY0X3ZxOGtfaW1hZ2VuZXQgICAgICAgZG93bmxvYWRzPTY0OVxuWGlhb21pTWlNby9NaU1vLUF1ZGlvLVRva2VuaXplciAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz02MTFcbmFsbGVncm8vaGVyYmVydC1rbGVqLWNhc2VkLXRva2VuaXplci12MSAgICAgICAgICAgICBkb3dubG9hZHM9NTgyXG55dWNvcm5ldHRvL3Rva2VuaXplcl90aXRva19iNjRfaW1hZ2VuZXQgICAgICAgICAgICAgZG93bmxvYWRzPTQ5OFxuYWktbW9kZWxzL2dpbGJlcnRvLXVuY2FzZWQtZnJvbS1jYW1lbWJlcnQtZmFzdC10b2tlbml6ZXIgIGRvd25sb2Fkcz00MDhcblF1YW5nRHV5L2h1aXQtdG9rZW5pemVyICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MzgyXG4ifV0sImlzRXJyb3IiOmZhbHNlLCJ0aW1lc3RhbXAiOjE3NzEwMzY2MjI1ODZ9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiJkNzczNmY2ZSIsInBhcmVudElkIjoiNDc2YmUwZjIiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjM3OjA2LjE2NVoiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidG9vbENhbGwiLCJpZCI6InRvb2x1XzAxR3dmcExZZmFFMVlwaW9aa3JhVTltbiIsIm5hbWUiOiJiYXNoIiwiYXJndW1lbnRzIjp7ImNvbW1hbmQiOiIjIFNlYXJjaCBmb3Igd2VsbC1rbm93biBMTE0gbW9kZWxzIG9uIFhlbm92YSdzIG5hbWVzcGFjZSAoVHJhbnNmb3JtZXJzLmpzIGNvbXBhdGlibGUpXG5jdXJsIC1zIFwiaHR0cHM6Ly9odWdnaW5nZmFjZS5jby9hcGkvbW9kZWxzP2F1dGhvcj1YZW5vdmEmc29ydD1kb3dubG9hZHMmbGltaXQ9NTBcIiB8IHB5dGhvbjMgLWMgXCJcbmltcG9ydCBqc29uLCBzeXNcbmRhdGEgPSBqc29uLmxvYWQoc3lzLnN0ZGluKVxuZm9yIG0gaW4gZGF0YTpcbiAgICBwcmludChmXFxcInttWydpZCddOjU1c30gIGRvd25sb2Fkcz17bS5nZXQoJ2Rvd25sb2FkcycsJz8nKX1cXFwiKVxuXCIgMj4vZGV2L251bGwgfCBoZWFkIC01MCJ9fV0sImFwaSI6ImFudGhyb3BpYy1tZXNzYWdlcyIsInByb3ZpZGVyIjoiYW50aHJvcGljIiwibW9kZWwiOiJjbGF1ZGUtb3B1cy00LTYiLCJ1c2FnZSI6eyJpbnB1dCI6MSwib3V0cHV0IjoxNzUsImNhY2hlUmVhZCI6NTUxMiwiY2FjaGVXcml0ZSI6OTU0LCJ0b3RhbFRva2VucyI6NjY0MiwiY29zdCI6eyJpbnB1dCI6MC4wMDAwMDUsIm91dHB1dCI6MC4wMDQzNzUsImNhY2hlUmVhZCI6MC4wMDI3NTU5OTk5OTk5OTk5OTk4LCJjYWNoZVdyaXRlIjowLjAwNTk2MjUsInRvdGFsIjowLjAxMzA5ODQ5OTk5OTk5OTk5OX19LCJzdG9wUmVhc29uIjoidG9vbFVzZSIsInRpbWVzdGFtcCI6MTc3MTAzNjYyMjU4Nn19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6IjY4ZTBmODU3IiwicGFyZW50SWQiOiJkNzczNmY2ZSIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6Mzc6MDcuMDM3WiIsIm1lc3NhZ2UiOnsicm9sZSI6InRvb2xSZXN1bHQiLCJ0b29sQ2FsbElkIjoidG9vbHVfMDFHd2ZwTFlmYUUxWXBpb1prcmFVOW1uIiwidG9vbE5hbWUiOiJiYXNoIiwiY29udGVudCI6W3sidHlwZSI6InRleHQiLCJ0ZXh0IjoiWGVub3ZhL2JnZS1iYXNlLWVuLXYxLjUgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTEyNzU1NjRcblhlbm92YS9hbGwtTWluaUxNLUw2LXYyICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0xMDcwMjkyXG5YZW5vdmEvc2VnZm9ybWVyLWIwLWZpbmV0dW5lZC1hZGUtNTEyLTUxMiAgICAgICAgICAgICAgICBkb3dubG9hZHM9ODg0OTU3XG5YZW5vdmEvcGFyYXBocmFzZS1tdWx0aWxpbmd1YWwtTWluaUxNLUwxMi12MiAgICAgICAgICAgICBkb3dubG9hZHM9MTA2OTAwXG5YZW5vdmEvY2xpcC12aXQtYmFzZS1wYXRjaDMyICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9NzU3Mzdcblhlbm92YS9lNS1zbWFsbC12MiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz02NTM4MFxuWGVub3ZhL2JnZS1zbWFsbC1lbi12MS41ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTU4ODExXG5YZW5vdmEvd2hpc3Blci10aW55LmVuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9NDk4MDRcblhlbm92YS9tcy1tYXJjby1NaW5pTE0tTC02LXYyICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0zODk5MFxuWGVub3ZhL3doaXNwZXItdGlueSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTM2NzAxXG5YZW5vdmEvZGlzdGlsYmVydC1iYXNlLXVuY2FzZWQtZmluZXR1bmVkLXNzdC0yLWVuZ2xpc2ggICBkb3dubG9hZHM9MzU3NDRcblhlbm92YS93aGlzcGVyLWJhc2UgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0zMjkwN1xuWGVub3ZhL2JlcnQtYmFzZS1ORVIgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTI0Nzc0XG5YZW5vdmEvYmdlLW0zICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MjQyODdcblhlbm92YS9hbGwtbXBuZXQtYmFzZS12MiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0yMTA2NlxuWGVub3ZhL2ppbmEtZW1iZWRkaW5ncy12Mi1zbWFsbC1lbiAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTE5NDA4XG5YZW5vdmEvZmxhbi10NS1iYXNlICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MTc4OTVcblhlbm92YS9qaW5hLWVtYmVkZGluZ3MtdjItYmFzZS1lbiAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0xNjg2M1xuWGVub3ZhL3doaXNwZXItYmFzZS5lbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTE2MDkyXG5YZW5vdmEvZ3RlLXNtYWxsICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MTU0ODhcblhlbm92YS9jbGlwLXZpdC1iYXNlLXBhdGNoMTYgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0xNDkwOFxuWGVub3ZhL2Rpbm92Mi1zbWFsbCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTEzOTc2XG5YZW5vdmEvZGlzdGlsYmVydC1iYXNlLW11bHRpbGluZ3VhbC1jYXNlZC1zZW50aW1lbnRzLXN0dWRlbnQgIGRvd25sb2Fkcz0xMzc5MFxuWGVub3ZhL21vZG5ldCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTExNTU0XG5YZW5vdmEvZGlzdGlsdXNlLWJhc2UtbXVsdGlsaW5ndWFsLWNhc2VkLXYyICAgICAgICAgICAgICBkb3dubG9hZHM9MTA4ODNcblhlbm92YS9vcHVzLW10LWphLWVuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0xMDY5M1xuWGVub3ZhL3NsaW1zYW0tNzctdW5pZm9ybSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTk2MzNcblhlbm92YS9tdWx0aWxpbmd1YWwtZTUtc21hbGwgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz05NTM2XG5YZW5vdmEvYmdlLWxhcmdlLWVuLXYxLjUgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9ODgyNFxuWGVub3ZhL3RveGljLWJlcnQgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTc1OTJcblhlbm92YS9kaXN0aWxncHQyICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz02NjA2XG5YZW5vdmEvdGlueS1yYW5kb20tUGhpM0ZvckNhdXNhbExNICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9NjU0M1xuWGVub3ZhL2RldHItcmVzbmV0LTUwICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTU4Njlcblhlbm92YS9ub21pYy1lbWJlZC10ZXh0LXYxICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz01MTA3XG5YZW5vdmEvdml0LWdwdDItaW1hZ2UtY2FwdGlvbmluZyAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9NTA3OFxuWGVub3ZhL21zLW1hcmNvLU1pbmlMTS1MLTEyLXYyICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTQ2OTRcblhlbm92YS93aGlzcGVyLXNtYWxsICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz00NDIzXG5YZW5vdmEvZGVwdGgtYW55dGhpbmctc21hbGwtaGYgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9NDIxNlxuWGVub3ZhL2dwdDIgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTQwNzdcblhlbm92YS9kaXN0aWxiYXJ0LWNubi02LTYgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz00MDI1XG5YZW5vdmEvbXVsdGlsaW5ndWFsLWU1LWJhc2UgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MzkyN1xuWGVub3ZhL2Rpbm8tdml0czE2ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTM3OTFcblhlbm92YS9tdXNpY2dlbi1zbWFsbCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0zNzA5XG5YZW5vdmEvc3BlZWNodDVfdHRzICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MzU0NlxuWGVub3ZhL25sbGItMjAwLWRpc3RpbGxlZC02MDBNICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTM0MTNcblhlbm92YS9hbGwtTWluaUxNLUwxMi12MiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0zMzg2XG5YZW5vdmEvdml0LWJhc2UtcGF0Y2gxNi0yMjQgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MzI0M1xuWGVub3ZhL3BhcmFwaHJhc2UtbXVsdGlsaW5ndWFsLW1wbmV0LWJhc2UtdjIgICAgICAgICAgICAgZG93bmxvYWRzPTMwODlcblhlbm92YS9tc21hcmNvLWRpc3RpbGJlcnQtYmFzZS12NCAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0zMDM2XG5YZW5vdmEvZGlzdGlsYmVydC1iYXNlLWNhc2VkLWRpc3RpbGxlZC1zcXVhZCAgICAgICAgICAgICBkb3dubG9hZHM9Mjg0NFxuIn1dLCJpc0Vycm9yIjpmYWxzZSwidGltZXN0YW1wIjoxNzcxMDM2NjI3MDM2fX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiMmM0MTQ4MmUiLCJwYXJlbnRJZCI6IjY4ZTBmODU3IiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjozNzoxMS40MjhaIiwibWVzc2FnZSI6eyJyb2xlIjoiYXNzaXN0YW50IiwiY29udGVudCI6W3sidHlwZSI6InRvb2xDYWxsIiwiaWQiOiJ0b29sdV8wMVc4bkZDdkFLN21XeDZxNzlvdGptOEoiLCJuYW1lIjoiYmFzaCIsImFyZ3VtZW50cyI6eyJjb21tYW5kIjoiIyBTZWFyY2ggZm9yIHNwZWNpZmljIHdlbGwta25vd24gTExNIHRva2VuaXplcnMgb24gWGVub3ZhJ3MgbmFtZXNwYWNlXG5mb3IgbW9kZWwgaW4gXCJncHQtNG9cIiBcImdwdC00XCIgXCJncHQtMy41XCIgXCJsbGFtYVwiIFwibWlzdHJhbFwiIFwiZ2VtbWFcIiBcInBoaVwiIFwiY2xhdWRlXCIgXCJxd2VuXCIgXCJkZWVwc2Vla1wiIFwibzIwMGtcIiBcImNsMTAwa1wiOyBkb1xuICBlY2hvIFwiPT09ICRtb2RlbCA9PT1cIlxuICBjdXJsIC1zIFwiaHR0cHM6Ly9odWdnaW5nZmFjZS5jby9hcGkvbW9kZWxzP2F1dGhvcj1YZW5vdmEmc2VhcmNoPSRtb2RlbCZzb3J0PWRvd25sb2FkcyZsaW1pdD01XCIgfCBweXRob24zIC1jIFwiXG5pbXBvcnQganNvbiwgc3lzXG5kYXRhID0ganNvbi5sb2FkKHN5cy5zdGRpbilcbmZvciBtIGluIGRhdGE6XG4gICAgcHJpbnQoZlxcXCIgIHttWydpZCddOjU1c30gIGRvd25sb2Fkcz17bS5nZXQoJ2Rvd25sb2FkcycsJz8nKX1cXFwiKVxuXCIgMj4vZGV2L251bGxcbmRvbmUifX1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjEsIm91dHB1dCI6MjQ2LCJjYWNoZVJlYWQiOjY0NjYsImNhY2hlV3JpdGUiOjEyMTQsInRvdGFsVG9rZW5zIjo3OTI3LCJjb3N0Ijp7ImlucHV0IjowLjAwMDAwNSwib3V0cHV0IjowLjAwNjE1LCJjYWNoZVJlYWQiOjAuMDAzMjMyOTk5OTk5OTk5OTk5OCwiY2FjaGVXcml0ZSI6MC4wMDc1ODc1MDAwMDAwMDAwMDA1LCJ0b3RhbCI6MC4wMTY5NzU1fX0sInN0b3BSZWFzb24iOiJ0b29sVXNlIiwidGltZXN0YW1wIjoxNzcxMDM2NjI3MDM3fX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiYTc1MjI1ZmUiLCJwYXJlbnRJZCI6IjJjNDE0ODJlIiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjozNzoyNC4zNTFaIiwibWVzc2FnZSI6eyJyb2xlIjoidG9vbFJlc3VsdCIsInRvb2xDYWxsSWQiOiJ0b29sdV8wMVc4bkZDdkFLN21XeDZxNzlvdGptOEoiLCJ0b29sTmFtZSI6ImJhc2giLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiI9PT0gZ3B0LTRvID09PVxuICBYZW5vdmEvZ3B0LTRvICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MFxuPT09IGdwdC00ID09PVxuICBYZW5vdmEvZ3B0LTQgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MFxuICBYZW5vdmEvZ3B0LTRvICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MFxuPT09IGdwdC0zLjUgPT09XG4gIFhlbm92YS9ncHQtMy41LXR1cmJvICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0wXG4gIFhlbm92YS9ncHQtMy41LXR1cmJvLTE2ayAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0wXG49PT0gbGxhbWEgPT09XG4gIFhlbm92YS9sbGFtYTIuYy1zdG9yaWVzMTVNICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0yMjc4XG4gIFhlbm92YS9sbGFtYTIuYy1zdG9yaWVzMTEwTSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0zNFxuICBYZW5vdmEvdGlueS1yYW5kb20tTGxhbWFGb3JDYXVzYWxMTSAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MjdcbiAgWGVub3ZhL2xsYW1hLTE2MG0gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTIwXG4gIFhlbm92YS9sbGFtYTIuYy1zdG9yaWVzNDJNICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0xNlxuPT09IG1pc3RyYWwgPT09XG4gIFhlbm92YS90aW55LXJhbmRvbS1taXN0cmFsICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz04XG4gIFhlbm92YS90aW55LXJhbmRvbS1NaXN0cmFsRm9yQ2F1c2FsTE1fZXh0ZXJuYWwtZGF0YSAgICAgIGRvd25sb2Fkcz00XG4gIFhlbm92YS9taXN0cmFsLXRva2VuaXplciAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0wXG4gIFhlbm92YS9taXN0cmFsLXRva2VuaXplci12MyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0wXG4gIFhlbm92YS9taXN0cmFsLXRva2VuaXplci12MSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0wXG49PT0gZ2VtbWEgPT09XG4gIFhlbm92YS90aW55LXJhbmRvbS1HZW1tYUZvckNhdXNhbExNICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz00MDZcbiAgWGVub3ZhL2dlbW1hMi10b2tlbml6ZXIgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTZcbiAgWGVub3ZhL2dlbW1hLXRva2VuaXplciAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTBcbiAgWGVub3ZhL2dlbW1hLTItdG9rZW5pemVyICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTBcbj09PSBwaGkgPT09XG4gIFhlbm92YS90aW55LXJhbmRvbS1QaGkzRm9yQ2F1c2FsTE0gICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz02NTQzXG4gIFhlbm92YS9QaGktMy1taW5pLTRrLWluc3RydWN0ICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz01MTlcbiAgWGVub3ZhL1BoaS0zLW1pbmktNGstaW5zdHJ1Y3RfZnAxNiAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTE3NVxuICBYZW5vdmEvcGhpLTFfNV9kZXYgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MzlcbiAgWGVub3ZhL3RpbnktcmFuZG9tLVBoaUZvckNhdXNhbExNICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTExXG49PT0gY2xhdWRlID09PVxuICBYZW5vdmEvY2xhdWRlLXRva2VuaXplciAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MFxuPT09IHF3ZW4gPT09XG4gIFhlbm92YS9Rd2VuMS41LTAuNUItQ2hhdCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0xMTk1XG4gIFhlbm92YS9Rd2VuMS41LTEuOEItQ2hhdCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0xMDFcbiAgWGVub3ZhL1F3ZW4xLjUtMC41QiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTM2XG4gIFhlbm92YS9Rd2VuMS41LTEuOEIgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz04XG49PT0gZGVlcHNlZWsgPT09XG4gIFhlbm92YS9kZWVwc2Vlay1jb2Rlci0xLjNiLWluc3RydWN0ICAgICAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0xOVxuICBYZW5vdmEvZGVlcHNlZWstY29kZXItMS4zYi1iYXNlICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9N1xuPT09IG8yMDBrID09PVxuPT09IGNsMTAwayA9PT1cbiJ9XSwiaXNFcnJvciI6ZmFsc2UsInRpbWVzdGFtcCI6MTc3MTAzNjY0NDM1MH19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6ImUzZDI3MjUyIiwicGFyZW50SWQiOiJhNzUyMjVmZSIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6Mzc6MjkuODE3WiIsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0b29sQ2FsbCIsImlkIjoidG9vbHVfMDFWazVZWHRadnVBWDJYcEROZXlKc0VQIiwibmFtZSI6ImJhc2giLCJhcmd1bWVudHMiOnsiY29tbWFuZCI6IiMgQ2hlY2sgZm9yIFF3ZW4yLjUgYW5kIFF3ZW4zIHRva2VuaXplcnNcbmZvciBtb2RlbCBpbiBcIlF3ZW4yXCIgXCJRd2VuM1wiIFwiZGVlcHNlZWstdlwiIFwiTWV0YS1MbGFtYS0zXCIgXCJsbGFtYS0zXCI7IGRvXG4gIGVjaG8gXCI9PT0gJG1vZGVsID09PVwiXG4gIGN1cmwgLXMgXCJodHRwczovL2h1Z2dpbmdmYWNlLmNvL2FwaS9tb2RlbHM/c2VhcmNoPSRtb2RlbCt0b2tlbml6ZXImc29ydD1kb3dubG9hZHMmbGltaXQ9NVwiIHwgcHl0aG9uMyAtYyBcIlxuaW1wb3J0IGpzb24sIHN5c1xuZGF0YSA9IGpzb24ubG9hZChzeXMuc3RkaW4pXG5mb3IgbSBpbiBkYXRhWzo1XTpcbiAgICBwcmludChmXFxcIiAge21bJ2lkJ106NjBzfSAgZG93bmxvYWRzPXttLmdldCgnZG93bmxvYWRzJywnPycpfVxcXCIpXG5cIiAyPi9kZXYvbnVsbFxuZG9uZSJ9fV0sImFwaSI6ImFudGhyb3BpYy1tZXNzYWdlcyIsInByb3ZpZGVyIjoiYW50aHJvcGljIiwibW9kZWwiOiJjbGF1ZGUtb3B1cy00LTYiLCJ1c2FnZSI6eyJpbnB1dCI6MSwib3V0cHV0IjoyMjEsImNhY2hlUmVhZCI6NzY4MCwiY2FjaGVXcml0ZSI6MTAwNSwidG90YWxUb2tlbnMiOjg5MDcsImNvc3QiOnsiaW5wdXQiOjAuMDAwMDA1LCJvdXRwdXQiOjAuMDA1NTI1LCJjYWNoZVJlYWQiOjAuMDAzODM5OTk5OTk5OTk5OTk5NywiY2FjaGVXcml0ZSI6MC4wMDYyODEyNSwidG90YWwiOjAuMDE1NjUxMjV9fSwic3RvcFJlYXNvbiI6InRvb2xVc2UiLCJ0aW1lc3RhbXAiOjE3NzEwMzY2NDQzNTF9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiI1M2MzMWY2ZSIsInBhcmVudElkIjoiZTNkMjcyNTIiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjM3OjM1LjAyM1oiLCJtZXNzYWdlIjp7InJvbGUiOiJ0b29sUmVzdWx0IiwidG9vbENhbGxJZCI6InRvb2x1XzAxVms1WVh0WnZ1QVgyWHBETmV5SnNFUCIsInRvb2xOYW1lIjoiYmFzaCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6Ij09PSBRd2VuMiA9PT1cbiAgUmljaGFyZEVya2hvdi9haG1lZGhlYWtsXy1fYXNtMmFzbS1xd2VuMi41Y29kZXItMC41Yi0zMDBrLTJlcC10b2tlbml6ZXItZ2d1ZiAgZG93bmxvYWRzPTE5XG4gIFJpY2hhcmRFcmtob3YvYWhtZWRoZWFrbF8tX2FzbTJhc20tcXdlbjIuNWNvZGVyLTAuNWItNTAway0yZXAtdG9rZW5pemVyLWdndWYgIGRvd25sb2Fkcz0xMlxuICBSaWNoYXJkRXJraG92L2FobWVkaGVha2xfLV9hc20yYXNtLXF3ZW4yLjVjb2Rlci0wLjViLTQwMGstMmVwLXRva2VuaXplci1nZ3VmICBkb3dubG9hZHM9OVxuICBSaWNoYXJkRXJraG92L2FobWVkaGVha2xfLV9hc20yYXNtLXF3ZW4yLjVjb2Rlci0wLjViLTIwMGstMmVwLXRva2VuaXplci1nZ3VmICBkb3dubG9hZHM9OFxuICBSaWNoYXJkRXJraG92L2FobWVkaGVha2xfLV9hc20yYXNtLXF3ZW4yLjVjb2Rlci0wLjViLTMwMGstMmVwLXRva2VuaXplci1hd3EgIGRvd25sb2Fkcz01XG49PT0gUXdlbjMgPT09XG4gIFF3ZW4vUXdlbjMtVFRTLVRva2VuaXplci0xMkh6ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZG93bmxvYWRzPTcxOTQ1XG4gIG1yYWRlcm1hY2hlci9Jcml4LTEyQi1Rd2VuMy1Ub2tlbml6ZXItSG9tdW5jdWx1cy1TbGVycC1HR1VGICAgZG93bmxvYWRzPTI3XG4gIHByYW5qYWxjaGl0YWxlL3F3ZW4zLWVtYmVkZGluZy0wLjZCLXJhbmRvbS1lbWItZ2VtbWEtdG9rZW5pemVyICBkb3dubG9hZHM9MTZcbiAgQWNjb3JkaWMvcXdlbjMtdHRzLXRva2VuaXplci0xMmh6LW1vZGVsICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MTVcbiAgc2hhZG93bGlsYWMvcXdlbjMtMC42Qi1iYXNlLXdpdGgtaW5zdHJ1Y3QtdG9rZW5pemVyICAgICAgICAgICBkb3dubG9hZHM9OFxuPT09IGRlZXBzZWVrLXYgPT09XG4gIGNhbmVydHVncnVsL0RlZXBTZWVrLVIxLURpc3RpbGwtUXdlbi0xNEItVG9vbC1Vc2UtVG9rZW5pemVyX3YyICBkb3dubG9hZHM9MFxuICBjYW5lcnR1Z3J1bC9EZWVwU2Vlay1SMS1EaXN0aWxsLVF3ZW4tMTRCLVRvb2wtVXNlLVRva2VuaXplcl92MyAgZG93bmxvYWRzPTBcbiAgQWNlc0RlbnkvZGVlcHNlZWtfdjNfdG9rZW5pemVyICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9MFxuICBOb3VzUmVzZWFyY2gvRGVlcFNlZWstVjMuMS1BbHRlcm5hdGUtVG9rZW5pemVyICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0wXG49PT0gTWV0YS1MbGFtYS0zID09PVxuICBOaWtvbGF5S296bG9mZi9NZXRhLUxsYW1hLTMtOEItSW5zdHJ1Y3QtYmYxNi1jb3JyZWN0LXByZS10b2tlbml6ZXItYW5kLUVPUy10b2tlbi1ROF8wLVE2X2stUTRfS19NLUdHVUYgIGRvd25sb2Fkcz0xNjlcbiAgTm91c1Jlc2VhcmNoL01ldGEtTGxhbWEtMy04Qi1BbHRlcm5hdGUtVG9rZW5pemVyICAgICAgICAgICAgICBkb3dubG9hZHM9NTVcbiAgZmVhdGhlcmxlc3MtYWktcXVhbnRzL05vdXNSZXNlYXJjaC1NZXRhLUxsYW1hLTMtOEItQWx0ZXJuYXRlLVRva2VuaXplci1HR1VGICBkb3dubG9hZHM9MzhcbiAgUmljaGFyZEVya2hvdi9Ob3VzUmVzZWFyY2hfLV9NZXRhLUxsYW1hLTMtOEItQWx0ZXJuYXRlLVRva2VuaXplci1nZ3VmICBkb3dubG9hZHM9M1xuICBtYXJ0aW50b21vdi9NZXRhLUxsYW1hLTMtOEItQWx0ZXJuYXRlLVRva2VuaXplci1RNF9LX1MtR0dVRiAgIGRvd25sb2Fkcz0xXG49PT0gbGxhbWEtMyA9PT1cbiAgS2lqYWkvbGxhdmEtbGxhbWEtMy04Yi10ZXh0LWVuY29kZXItdG9rZW5pemVyICAgICAgICAgICAgICAgICBkb3dubG9hZHM9ODQ1XG4gIE5pa29sYXlLb3psb2ZmL01ldGEtTGxhbWEtMy04Qi1JbnN0cnVjdC1iZjE2LWNvcnJlY3QtcHJlLXRva2VuaXplci1hbmQtRU9TLXRva2VuLVE4XzAtUTZfay1RNF9LX00tR0dVRiAgZG93bmxvYWRzPTE2OVxuICBqYWNxdWVsaW5laGUvdGlueWNvbW1hLTEuOGItbGxhbWEzLXRva2VuaXplciAgICAgICAgICAgICAgICAgIGRvd25sb2Fkcz0xNDJcbiAgcGN1ZW5xL0xsYW1hLTMuMi0xQi1JbnN0cnVjdC10b2tlbml6ZXIgICAgICAgICAgICAgICAgICAgICAgICBkb3dubG9hZHM9OTdcbiAgTm91c1Jlc2VhcmNoL01ldGEtTGxhbWEtMy04Qi1BbHRlcm5hdGUtVG9rZW5pemVyICAgICAgICAgICAgICBkb3dubG9hZHM9NTVcbiJ9XSwiaXNFcnJvciI6ZmFsc2UsInRpbWVzdGFtcCI6MTc3MTAzNjY1NTAyM319LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6IjMzYmMzZWI5IiwicGFyZW50SWQiOiI1M2MzMWY2ZSIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6Mzc6NDMuMzk5WiIsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0b29sQ2FsbCIsImlkIjoidG9vbHVfMDFSWG5EU2ZKR3VUaEUxdjdLMTJnaXp6IiwibmFtZSI6ImJhc2giLCJhcmd1bWVudHMiOnsiY29tbWFuZCI6IiMgQ2hlY2sgd2hpY2ggWGVub3ZhIG1vZGVscyBoYXZlIHRva2VuaXplciBjb25maWdzIChpLmUuIGNhbiBiZSB1c2VkIHB1cmVseSBhcyB0b2tlbml6ZXJzKVxuZm9yIG1vZGVsIGluIFwiWGVub3ZhL2dwdC00b1wiIFwiWGVub3ZhL2dwdC00XCIgXCJYZW5vdmEvZ3B0LTMuNS10dXJib1wiIFwiWGVub3ZhL2NsYXVkZS10b2tlbml6ZXJcIiBcIlhlbm92YS9taXN0cmFsLXRva2VuaXplci12M1wiIFwiWGVub3ZhL21pc3RyYWwtdG9rZW5pemVyLXYxXCIgXCJYZW5vdmEvZ2VtbWEtMi10b2tlbml6ZXJcIiBcIlhlbm92YS9nZW1tYTItdG9rZW5pemVyXCIgXCJYZW5vdmEvbGxhbWEtMTYwbVwiIFwiWGVub3ZhL1F3ZW4xLjUtMC41QlwiIFwiWGVub3ZhL1BoaS0zLW1pbmktNGstaW5zdHJ1Y3RcIiBcIlhlbm92YS9ncHQyXCIgXCJYZW5vdmEvZGlzdGlsZ3B0MlwiIFwiWGVub3ZhL2RlZXBzZWVrLWNvZGVyLTEuM2ItYmFzZVwiOyBkb1xuICBlY2hvIC1uIFwiJG1vZGVsOiBcIlxuICBjdXJsIC1zIFwiaHR0cHM6Ly9odWdnaW5nZmFjZS5jby9hcGkvbW9kZWxzLyRtb2RlbFwiIHwgcHl0aG9uMyAtYyBcIlxuaW1wb3J0IGpzb24sIHN5c1xuZGF0YSA9IGpzb24ubG9hZChzeXMuc3RkaW4pXG5zaWJsaW5ncyA9IFtzWydyZmlsZW5hbWUnXSBmb3IgcyBpbiBkYXRhLmdldCgnc2libGluZ3MnLCBbXSldXG5oYXNfdG9rID0gJ3Rva2VuaXplci5qc29uJyBpbiBzaWJsaW5ncyBvciAndG9rZW5pemVyX2NvbmZpZy5qc29uJyBpbiBzaWJsaW5nc1xucHJpbnQoJ+KchScgaWYgaGFzX3RvayBlbHNlICfinYwnLCBbZiBmb3IgZiBpbiBzaWJsaW5ncyBpZiAndG9rZW4nIGluIGYubG93ZXIoKV0pXG5cIiAyPi9kZXYvbnVsbFxuZG9uZSJ9fV0sImFwaSI6ImFudGhyb3BpYy1tZXNzYWdlcyIsInByb3ZpZGVyIjoiYW50aHJvcGljIiwibW9kZWwiOiJjbGF1ZGUtb3B1cy00LTYiLCJ1c2FnZSI6eyJpbnB1dCI6MSwib3V0cHV0Ijo0MDYsImNhY2hlUmVhZCI6ODY4NSwiY2FjaGVXcml0ZSI6MTIyNywidG90YWxUb2tlbnMiOjEwMzE5LCJjb3N0Ijp7ImlucHV0IjowLjAwMDAwNSwib3V0cHV0IjowLjAxMDE1MDAwMDAwMDAwMDAwMSwiY2FjaGVSZWFkIjowLjAwNDM0MjUsImNhY2hlV3JpdGUiOjAuMDA3NjY4NzUsInRvdGFsIjowLjAyMjE2NjI1fX0sInN0b3BSZWFzb24iOiJ0b29sVXNlIiwidGltZXN0YW1wIjoxNzcxMDM2NjU1MDIzfX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiZmYxYTMxOWYiLCJwYXJlbnRJZCI6IjMzYmMzZWI5IiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjozNzo1NS42MDdaIiwibWVzc2FnZSI6eyJyb2xlIjoidG9vbFJlc3VsdCIsInRvb2xDYWxsSWQiOiJ0b29sdV8wMVJYbkRTZkpHdVRoRTF2N0sxMmdpenoiLCJ0b29sTmFtZSI6ImJhc2giLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJYZW5vdmEvZ3B0LTRvOiDinIUgWydzcGVjaWFsX3Rva2Vuc19tYXAuanNvbicsICd0b2tlbml6ZXIuanNvbicsICd0b2tlbml6ZXJfY29uZmlnLmpzb24nXVxuWGVub3ZhL2dwdC00OiDinIUgWydzcGVjaWFsX3Rva2Vuc19tYXAuanNvbicsICd0b2tlbml6ZXIuanNvbicsICd0b2tlbml6ZXJfY29uZmlnLmpzb24nXVxuWGVub3ZhL2dwdC0zLjUtdHVyYm86IOKchSBbJ3NwZWNpYWxfdG9rZW5zX21hcC5qc29uJywgJ3Rva2VuaXplci5qc29uJywgJ3Rva2VuaXplcl9jb25maWcuanNvbiddXG5YZW5vdmEvY2xhdWRlLXRva2VuaXplcjog4pyFIFsndG9rZW5pemVyLmpzb24nLCAndG9rZW5pemVyX2NvbmZpZy5qc29uJ11cblhlbm92YS9taXN0cmFsLXRva2VuaXplci12Mzog4pyFIFsnc3BlY2lhbF90b2tlbnNfbWFwLmpzb24nLCAndG9rZW5pemVyLmpzb24nLCAndG9rZW5pemVyLm1vZGVsJywgJ3Rva2VuaXplci5tb2RlbC52MycsICd0b2tlbml6ZXJfY29uZmlnLmpzb24nXVxuWGVub3ZhL21pc3RyYWwtdG9rZW5pemVyLXYxOiDinIUgWydzcGVjaWFsX3Rva2Vuc19tYXAuanNvbicsICd0b2tlbml6ZXIuanNvbicsICd0b2tlbml6ZXIubW9kZWwnLCAndG9rZW5pemVyX2NvbmZpZy5qc29uJ11cblhlbm92YS9nZW1tYS0yLXRva2VuaXplcjog4pyFIFsnc3BlY2lhbF90b2tlbnNfbWFwLmpzb24nLCAndG9rZW5pemVyLmpzb24nLCAndG9rZW5pemVyLm1vZGVsJywgJ3Rva2VuaXplcl9jb25maWcuanNvbiddXG5YZW5vdmEvZ2VtbWEyLXRva2VuaXplcjog4pyFIFsnc3BlY2lhbF90b2tlbnNfbWFwLmpzb24nLCAndG9rZW5pemVyLmpzb24nLCAndG9rZW5pemVyLm1vZGVsJywgJ3Rva2VuaXplcl9jb25maWcuanNvbiddXG5YZW5vdmEvbGxhbWEtMTYwbTog4pyFIFsnc3BlY2lhbF90b2tlbnNfbWFwLmpzb24nLCAndG9rZW5pemVyLmpzb24nLCAndG9rZW5pemVyLm1vZGVsJywgJ3Rva2VuaXplcl9jb25maWcuanNvbiddXG5YZW5vdmEvUXdlbjEuNS0wLjVCOiDinIUgWydhZGRlZF90b2tlbnMuanNvbicsICdzcGVjaWFsX3Rva2Vuc19tYXAuanNvbicsICd0b2tlbml6ZXIuanNvbicsICd0b2tlbml6ZXJfY29uZmlnLmpzb24nXVxuWGVub3ZhL1BoaS0zLW1pbmktNGstaW5zdHJ1Y3Q6IOKchSBbJ2FkZGVkX3Rva2Vucy5qc29uJywgJ3NwZWNpYWxfdG9rZW5zX21hcC5qc29uJywgJ3Rva2VuaXplci5qc29uJywgJ3Rva2VuaXplci5tb2RlbCcsICd0b2tlbml6ZXJfY29uZmlnLmpzb24nXVxuWGVub3ZhL2dwdDI6IOKchSBbJ3NwZWNpYWxfdG9rZW5zX21hcC5qc29uJywgJ3Rva2VuaXplci5qc29uJywgJ3Rva2VuaXplcl9jb25maWcuanNvbiddXG5YZW5vdmEvZGlzdGlsZ3B0Mjog4pyFIFsnc3BlY2lhbF90b2tlbnNfbWFwLmpzb24nLCAndG9rZW5pemVyLmpzb24nLCAndG9rZW5pemVyX2NvbmZpZy5qc29uJ11cblhlbm92YS9kZWVwc2Vlay1jb2Rlci0xLjNiLWJhc2U6IOKchSBbJ3NwZWNpYWxfdG9rZW5zX21hcC5qc29uJywgJ3Rva2VuaXplci5qc29uJywgJ3Rva2VuaXplcl9jb25maWcuanNvbiddXG4ifV0sImlzRXJyb3IiOmZhbHNlLCJ0aW1lc3RhbXAiOjE3NzEwMzY2NzU2MDZ9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiJlMWVlODdjNiIsInBhcmVudElkIjoiZmYxYTMxOWYiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjM4OjAzLjg0NFoiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidG9vbENhbGwiLCJpZCI6InRvb2x1XzAxVWZHZXhvUGRLNWZzZUVmRkZpWlFYcCIsIm5hbWUiOiJiYXNoIiwiYXJndW1lbnRzIjp7ImNvbW1hbmQiOiIjIENoZWNrIGZvciBMTGFNQSAzIGFuZCBEZWVwU2VlayBWMyB0b2tlbml6ZXJzIG9uIFhlbm92YVxuZm9yIG1vZGVsIGluIFwiWGVub3ZhL2xsYW1hLTMtdG9rZW5pemVyXCIgXCJYZW5vdmEvTWV0YS1MbGFtYS0zXCIgXCJYZW5vdmEvbGxhbWEzLXRva2VuaXplclwiIFwiWGVub3ZhL2RlZXBzZWVrLXYzXCIgXCJYZW5vdmEvZGVlcHNlZWstdjJcIjsgZG9cbiAgZWNobyAtbiBcIiRtb2RlbDogXCJcbiAgY3VybCAtcyBcImh0dHBzOi8vaHVnZ2luZ2ZhY2UuY28vYXBpL21vZGVscy8kbW9kZWxcIiB8IHB5dGhvbjMgLWMgXCJcbmltcG9ydCBqc29uLCBzeXNcbmRhdGEgPSBqc29uLmxvYWQoc3lzLnN0ZGluKVxuaWYgJ2Vycm9yJyBpbiBkYXRhOlxuICAgIHByaW50KCdOT1QgRk9VTkQnKVxuZWxzZTpcbiAgICBzaWJsaW5ncyA9IFtzWydyZmlsZW5hbWUnXSBmb3IgcyBpbiBkYXRhLmdldCgnc2libGluZ3MnLCBbXSldXG4gICAgaGFzX3RvayA9ICd0b2tlbml6ZXIuanNvbicgaW4gc2libGluZ3NcbiAgICBwcmludCgn4pyFJyBpZiBoYXNfdG9rIGVsc2UgJ+KdjCcsIFtmIGZvciBmIGluIHNpYmxpbmdzIGlmICd0b2tlbicgaW4gZi5sb3dlcigpXSlcblwiIDI+L2Rldi9udWxsXG5kb25lXG5cbiMgQWxzbyBjaGVjayB0aGUgb3JpZ2luYWwgcmVwb3MgZm9yIExMYU1BIDNcbmZvciBtb2RlbCBpbiBcIm1ldGEtbGxhbWEvTGxhbWEtMy4xLThCXCIgXCJtZXRhLWxsYW1hL0xsYW1hLTMuMi0xQlwiIFwiZGVlcHNlZWstYWkvRGVlcFNlZWstVjNcIjsgZG9cbiAgZWNobyAtbiBcIiRtb2RlbDogXCJcbiAgY3VybCAtcyBcImh0dHBzOi8vaHVnZ2luZ2ZhY2UuY28vYXBpL21vZGVscy8kbW9kZWxcIiB8IHB5dGhvbjMgLWMgXCJcbmltcG9ydCBqc29uLCBzeXNcbmRhdGEgPSBqc29uLmxvYWQoc3lzLnN0ZGluKVxuaWYgJ2Vycm9yJyBpbiBkYXRhOlxuICAgIHByaW50KCdOT1QgRk9VTkQgLyBHQVRFRCcpXG5lbHNlOlxuICAgIHNpYmxpbmdzID0gW3NbJ3JmaWxlbmFtZSddIGZvciBzIGluIGRhdGEuZ2V0KCdzaWJsaW5ncycsIFtdKV1cbiAgICBoYXNfdG9rID0gJ3Rva2VuaXplci5qc29uJyBpbiBzaWJsaW5nc1xuICAgIHByaW50KCfinIUnIGlmIGhhc190b2sgZWxzZSAn4p2MJywgW2YgZm9yIGYgaW4gc2libGluZ3MgaWYgJ3Rva2VuJyBpbiBmLmxvd2VyKCldKVxuXCIgMj4vZGV2L251bGxcbmRvbmUifX1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjEsIm91dHB1dCI6NTIwLCJjYWNoZVJlYWQiOjk5MTIsImNhY2hlV3JpdGUiOjEwNjYsInRvdGFsVG9rZW5zIjoxMTQ5OSwiY29zdCI6eyJpbnB1dCI6MC4wMDAwMDUsIm91dHB1dCI6MC4wMTMwMDAwMDAwMDAwMDAwMDEsImNhY2hlUmVhZCI6MC4wMDQ5NTU5OTk5OTk5OTk5OTk1LCJjYWNoZVdyaXRlIjowLjAwNjY2MjUsInRvdGFsIjowLjAyNDYyMzV9fSwic3RvcFJlYXNvbiI6InRvb2xVc2UiLCJ0aW1lc3RhbXAiOjE3NzEwMzY2NzU2MDh9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiIzOGQwNWJlZiIsInBhcmVudElkIjoiZTFlZTg3YzYiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjM4OjEwLjc1NFoiLCJtZXNzYWdlIjp7InJvbGUiOiJ0b29sUmVzdWx0IiwidG9vbENhbGxJZCI6InRvb2x1XzAxVWZHZXhvUGRLNWZzZUVmRkZpWlFYcCIsInRvb2xOYW1lIjoiYmFzaCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6Ilhlbm92YS9sbGFtYS0zLXRva2VuaXplcjogWGVub3ZhL01ldGEtTGxhbWEtMzogTk9UIEZPVU5EXG5YZW5vdmEvbGxhbWEzLXRva2VuaXplcjog4pyFIFsnc3BlY2lhbF90b2tlbnNfbWFwLmpzb24nLCAndG9rZW5pemVyLmpzb24nLCAndG9rZW5pemVyX2NvbmZpZy5qc29uJ11cblhlbm92YS9kZWVwc2Vlay12MzogTk9UIEZPVU5EXG5YZW5vdmEvZGVlcHNlZWstdjI6IE5PVCBGT1VORFxubWV0YS1sbGFtYS9MbGFtYS0zLjEtOEI6IOKchSBbJ29yaWdpbmFsL3Rva2VuaXplci5tb2RlbCcsICdzcGVjaWFsX3Rva2Vuc19tYXAuanNvbicsICd0b2tlbml6ZXIuanNvbicsICd0b2tlbml6ZXJfY29uZmlnLmpzb24nXVxubWV0YS1sbGFtYS9MbGFtYS0zLjItMUI6IOKchSBbJ29yaWdpbmFsL3Rva2VuaXplci5tb2RlbCcsICdzcGVjaWFsX3Rva2Vuc19tYXAuanNvbicsICd0b2tlbml6ZXIuanNvbicsICd0b2tlbml6ZXJfY29uZmlnLmpzb24nXVxuZGVlcHNlZWstYWkvRGVlcFNlZWstVjM6IOKchSBbJ3Rva2VuaXplci5qc29uJywgJ3Rva2VuaXplcl9jb25maWcuanNvbiddXG4ifV0sImlzRXJyb3IiOmZhbHNlLCJ0aW1lc3RhbXAiOjE3NzEwMzY2OTA3NTN9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiIzNjJlOTBlMyIsInBhcmVudElkIjoiMzhkMDViZWYiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjM4OjE2LjI2MloiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidG9vbENhbGwiLCJpZCI6InRvb2x1XzAxUTNWZzF2UDZod3RyaWc1RUpBMkd4NCIsIm5hbWUiOiJiYXNoIiwiYXJndW1lbnRzIjp7ImNvbW1hbmQiOiIjIENoZWNrIGZvciBtb3JlIFhlbm92YSB0b2tlbml6ZXItb25seSByZXBvc1xuZm9yIG1vZGVsIGluIFwiWGVub3ZhL2xsYW1hLTMuMS10b2tlbml6ZXJcIiBcIlhlbm92YS9sbGFtYS0zLjItdG9rZW5pemVyXCIgXCJYZW5vdmEvYzRhaS1jb21tYW5kLXJcIiBcIlhlbm92YS9kZWVwc2Vlay10b2tlbml6ZXJcIjsgZG9cbiAgZWNobyAtbiBcIiRtb2RlbDogXCJcbiAgY3VybCAtcyBcImh0dHBzOi8vaHVnZ2luZ2ZhY2UuY28vYXBpL21vZGVscy8kbW9kZWxcIiB8IHB5dGhvbjMgLWMgXCJcbmltcG9ydCBqc29uLCBzeXNcbmRhdGEgPSBqc29uLmxvYWQoc3lzLnN0ZGluKVxuaWYgJ2Vycm9yJyBpbiBkYXRhOlxuICAgIHByaW50KCdOT1QgRk9VTkQnKVxuZWxzZTpcbiAgICBzaWJsaW5ncyA9IFtzWydyZmlsZW5hbWUnXSBmb3IgcyBpbiBkYXRhLmdldCgnc2libGluZ3MnLCBbXSldXG4gICAgaGFzX3RvayA9ICd0b2tlbml6ZXIuanNvbicgaW4gc2libGluZ3NcbiAgICBwcmludCgn4pyFJyBpZiBoYXNfdG9rIGVsc2UgJ+KdjCcsIFtmIGZvciBmIGluIHNpYmxpbmdzIGlmICd0b2tlbicgaW4gZi5sb3dlcigpXSlcblwiIDI+L2Rldi9udWxsXG5kb25lIn19XSwiYXBpIjoiYW50aHJvcGljLW1lc3NhZ2VzIiwicHJvdmlkZXIiOiJhbnRocm9waWMiLCJtb2RlbCI6ImNsYXVkZS1vcHVzLTQtNiIsInVzYWdlIjp7ImlucHV0IjoxLCJvdXRwdXQiOjI3NiwiY2FjaGVSZWFkIjoxMDk3OCwiY2FjaGVXcml0ZSI6Nzc1LCJ0b3RhbFRva2VucyI6MTIwMzAsImNvc3QiOnsiaW5wdXQiOjAuMDAwMDA1LCJvdXRwdXQiOjAuMDA2OTAwMDAwMDAwMDAwMDAxLCJjYWNoZVJlYWQiOjAuMDA1NDg5LCJjYWNoZVdyaXRlIjowLjAwNDg0Mzc1LCJ0b3RhbCI6MC4wMTcyMzc3NX19LCJzdG9wUmVhc29uIjoidG9vbFVzZSIsInRpbWVzdGFtcCI6MTc3MTAzNjY5MDc1NH19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6ImUzNDk1ZTE5IiwicGFyZW50SWQiOiIzNjJlOTBlMyIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6Mzg6MTkuNTcxWiIsIm1lc3NhZ2UiOnsicm9sZSI6InRvb2xSZXN1bHQiLCJ0b29sQ2FsbElkIjoidG9vbHVfMDFRM1ZnMXZQNmh3dHJpZzVFSkEyR3g0IiwidG9vbE5hbWUiOiJiYXNoIiwiY29udGVudCI6W3sidHlwZSI6InRleHQiLCJ0ZXh0IjoiWGVub3ZhL2xsYW1hLTMuMS10b2tlbml6ZXI6IE5PVCBGT1VORFxuWGVub3ZhL2xsYW1hLTMuMi10b2tlbml6ZXI6IFhlbm92YS9jNGFpLWNvbW1hbmQtcjogTk9UIEZPVU5EXG5YZW5vdmEvZGVlcHNlZWstdG9rZW5pemVyOiBOT1QgRk9VTkRcbiJ9XSwiaXNFcnJvciI6ZmFsc2UsInRpbWVzdGFtcCI6MTc3MTAzNjY5OTU3MH19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6IjhkMmNjNWYyIiwicGFyZW50SWQiOiJlMzQ5NWUxOSIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6Mzk6MjMuMzMwWiIsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6Ikdvb2QsIEkgaGF2ZSBhIHNvbGlkIGxpc3Qgbm93LiBMZXQgbWUgYnVpbGQgdGhlIHJlZGVzaWduZWQgcGFnZSB3aXRoIGFuIEFzdHJvLWluc3BpcmVkIG1pbmltYWwgYWVzdGhldGljIGFuZCBhIGRyb3Bkb3duIGZvciBtdWx0aXBsZSB0b2tlbml6ZXJzLiJ9LHsidHlwZSI6InRvb2xDYWxsIiwiaWQiOiJ0b29sdV8wMTFTN3FFaHgzM1JVblp2MnNuUHJjakUiLCJuYW1lIjoid3JpdGUiLCJhcmd1bWVudHMiOnsicGF0aCI6Ii9ob21lL3dyL2doL3RyeV93ZWJfZ3B1L2luZGV4Lmh0bWwiLCJjb250ZW50IjoiPCFET0NUWVBFIGh0bWw+XG48aHRtbCBsYW5nPVwiZW5cIj5cbjxoZWFkPlxuICA8bWV0YSBjaGFyc2V0PVwiVVRGLThcIj5cbiAgPG1ldGEgbmFtZT1cInZpZXdwb3J0XCIgY29udGVudD1cIndpZHRoPWRldmljZS13aWR0aCwgaW5pdGlhbC1zY2FsZT0xLjBcIj5cbiAgPHRpdGxlPlRva2VuaXplciBQbGF5Z3JvdW5kPC90aXRsZT5cbiAgPHN0eWxlPlxuICAgIEBpbXBvcnQgdXJsKCdodHRwczovL2ZvbnRzLmdvb2dsZWFwaXMuY29tL2NzczI/ZmFtaWx5PUludGVyOndnaHRANDAwOzUwMDs2MDA7NzAwJmZhbWlseT1JQk0rUGxleCtNb25vOndnaHRANDAwOzUwMCZkaXNwbGF5PXN3YXAnKTtcblxuICAgIDpyb290IHtcbiAgICAgIC0tYmc6ICMwZDBmMTQ7XG4gICAgICAtLXN1cmZhY2U6ICMxMzE1MWM7XG4gICAgICAtLXN1cmZhY2UtMjogIzE5MWMyNjtcbiAgICAgIC0tYm9yZGVyOiAjMjMyNzNhO1xuICAgICAgLS1ib3JkZXItaG92ZXI6ICMzNDNhNTQ7XG4gICAgICAtLXRleHQ6ICNlNmVkZjM7XG4gICAgICAtLXRleHQtbXV0ZWQ6ICM3ZDg1OTA7XG4gICAgICAtLXRleHQtZGltOiAjNDg0ZjU4O1xuICAgICAgLS1hY2NlbnQ6ICNiYzhjZmY7XG4gICAgICAtLWFjY2VudC1kaW06IHJnYmEoMTg4LCAxNDAsIDI1NSwgMC4xMik7XG4gICAgICAtLWFjY2VudC0yOiAjZDJhOGZmO1xuICAgICAgLS1tb25vOiAnSUJNIFBsZXggTW9ubycsICdTRiBNb25vJywgJ0ZpcmEgQ29kZScsIG1vbm9zcGFjZTtcbiAgICAgIC0tc2FuczogJ0ludGVyJywgLWFwcGxlLXN5c3RlbSwgQmxpbmtNYWNTeXN0ZW1Gb250LCBzYW5zLXNlcmlmO1xuICAgICAgLS1yYWRpdXM6IDhweDtcbiAgICB9XG5cbiAgICAqIHsgYm94LXNpemluZzogYm9yZGVyLWJveDsgbWFyZ2luOiAwOyBwYWRkaW5nOiAwOyB9XG5cbiAgICBib2R5IHtcbiAgICAgIGZvbnQtZmFtaWx5OiB2YXIoLS1zYW5zKTtcbiAgICAgIGJhY2tncm91bmQ6IHZhcigtLWJnKTtcbiAgICAgIGNvbG9yOiB2YXIoLS10ZXh0KTtcbiAgICAgIG1pbi1oZWlnaHQ6IDEwMHZoO1xuICAgICAgLXdlYmtpdC1mb250LXNtb290aGluZzogYW50aWFsaWFzZWQ7XG4gICAgfVxuXG4gICAgLyog4oCUIE5vaXNlIHRleHR1cmUgb3ZlcmxheSDigJQgKi9cbiAgICBib2R5OjpiZWZvcmUge1xuICAgICAgY29udGVudDogJyc7XG4gICAgICBwb3NpdGlvbjogZml4ZWQ7XG4gICAgICBpbnNldDogMDtcbiAgICAgIGJhY2tncm91bmQ6IHVybChcImRhdGE6aW1hZ2Uvc3ZnK3htbCwlM0Nzdmcgdmlld0JveD0nMCAwIDI1NiAyNTYnIHhtbG5zPSdodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyclM0UlM0NmaWx0ZXIgaWQ9J24nJTNFJTNDZmVUdXJidWxlbmNlIHR5cGU9J2ZyYWN0YWxOb2lzZScgYmFzZUZyZXF1ZW5jeT0nMC45JyBudW1PY3RhdmVzPSc0JyBzdGl0Y2hUaWxlcz0nc3RpdGNoJy8lM0UlM0MvZmlsdGVyJTNFJTNDcmVjdCB3aWR0aD0nMTAwJTI1JyBoZWlnaHQ9JzEwMCUyNScgZmlsdGVyPSd1cmwoJTIzbiknIG9wYWNpdHk9JzAuMDMnLyUzRSUzQy9zdmclM0VcIik7XG4gICAgICBwb2ludGVyLWV2ZW50czogbm9uZTtcbiAgICAgIHotaW5kZXg6IDA7XG4gICAgfVxuXG4gICAgLnBhZ2Uge1xuICAgICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgICAgei1pbmRleDogMTtcbiAgICAgIG1heC13aWR0aDogNzgwcHg7XG4gICAgICBtYXJnaW46IDAgYXV0bztcbiAgICAgIHBhZGRpbmc6IDYwcHggMjRweCA4MHB4O1xuICAgIH1cblxuICAgIC8qIOKAlCBIZWFkZXIg4oCUICovXG4gICAgaGVhZGVyIHsgbWFyZ2luLWJvdHRvbTogNDhweDsgfVxuXG4gICAgLmJhZGdlIHtcbiAgICAgIGRpc3BsYXk6IGlubGluZS1mbGV4O1xuICAgICAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgICAgIGdhcDogNnB4O1xuICAgICAgcGFkZGluZzogNHB4IDEycHg7XG4gICAgICBib3JkZXI6IDFweCBzb2xpZCB2YXIoLS1ib3JkZXIpO1xuICAgICAgYm9yZGVyLXJhZGl1czogOTk5cHg7XG4gICAgICBmb250LXNpemU6IDAuNzVyZW07XG4gICAgICBjb2xvcjogdmFyKC0tdGV4dC1tdXRlZCk7XG4gICAgICBtYXJnaW4tYm90dG9tOiAyMHB4O1xuICAgICAgYmFja2dyb3VuZDogdmFyKC0tc3VyZmFjZSk7XG4gICAgfVxuICAgIC5iYWRnZSAuZG90IHtcbiAgICAgIHdpZHRoOiA2cHg7IGhlaWdodDogNnB4O1xuICAgICAgYm9yZGVyLXJhZGl1czogNTAlO1xuICAgICAgYmFja2dyb3VuZDogIzNmYjk1MDtcbiAgICAgIGJveC1zaGFkb3c6IDAgMCA2cHggIzNmYjk1MDY2O1xuICAgIH1cblxuICAgIGgxIHtcbiAgICAgIGZvbnQtc2l6ZTogMi40cmVtO1xuICAgICAgZm9udC13ZWlnaHQ6IDcwMDtcbiAgICAgIGxldHRlci1zcGFjaW5nOiAtMC4wM2VtO1xuICAgICAgbGluZS1oZWlnaHQ6IDEuMTU7XG4gICAgICBiYWNrZ3JvdW5kOiBsaW5lYXItZ3JhZGllbnQoMTM1ZGVnLCAjZmZmIDAlLCB2YXIoLS1hY2NlbnQtMikgMTAwJSk7XG4gICAgICAtd2Via2l0LWJhY2tncm91bmQtY2xpcDogdGV4dDtcbiAgICAgIC13ZWJraXQtdGV4dC1maWxsLWNvbG9yOiB0cmFuc3BhcmVudDtcbiAgICAgIGJhY2tncm91bmQtY2xpcDogdGV4dDtcbiAgICB9XG5cbiAgICAuc3VidGl0bGUge1xuICAgICAgbWFyZ2luLXRvcDogMTBweDtcbiAgICAgIGZvbnQtc2l6ZTogMS4wNXJlbTtcbiAgICAgIGNvbG9yOiB2YXIoLS10ZXh0LW11dGVkKTtcbiAgICAgIGxpbmUtaGVpZ2h0OiAxLjY7XG4gICAgfVxuICAgIC5zdWJ0aXRsZSBhIHtcbiAgICAgIGNvbG9yOiB2YXIoLS1hY2NlbnQpO1xuICAgICAgdGV4dC1kZWNvcmF0aW9uOiBub25lO1xuICAgICAgYm9yZGVyLWJvdHRvbTogMXB4IHNvbGlkIHRyYW5zcGFyZW50O1xuICAgICAgdHJhbnNpdGlvbjogYm9yZGVyLWNvbG9yIDAuMnM7XG4gICAgfVxuICAgIC5zdWJ0aXRsZSBhOmhvdmVyIHsgYm9yZGVyLWJvdHRvbS1jb2xvcjogdmFyKC0tYWNjZW50KTsgfVxuXG4gICAgLyog4oCUIEZvcm0gY29udHJvbHMg4oCUICovXG4gICAgLmZpZWxkIHsgbWFyZ2luLWJvdHRvbTogMjBweDsgfVxuXG4gICAgbGFiZWwge1xuICAgICAgZGlzcGxheTogYmxvY2s7XG4gICAgICBmb250LXNpemU6IDAuOHJlbTtcbiAgICAgIGZvbnQtd2VpZ2h0OiA2MDA7XG4gICAgICB0ZXh0LXRyYW5zZm9ybTogdXBwZXJjYXNlO1xuICAgICAgbGV0dGVyLXNwYWNpbmc6IDAuMDZlbTtcbiAgICAgIGNvbG9yOiB2YXIoLS10ZXh0LW11dGVkKTtcbiAgICAgIG1hcmdpbi1ib3R0b206IDhweDtcbiAgICB9XG5cbiAgICBzZWxlY3QsIHRleHRhcmVhIHtcbiAgICAgIHdpZHRoOiAxMDAlO1xuICAgICAgcGFkZGluZzogMTJweCAxNHB4O1xuICAgICAgZm9udC1zaXplOiAwLjk1cmVtO1xuICAgICAgZm9udC1mYW1pbHk6IHZhcigtLXNhbnMpO1xuICAgICAgY29sb3I6IHZhcigtLXRleHQpO1xuICAgICAgYmFja2dyb3VuZDogdmFyKC0tc3VyZmFjZSk7XG4gICAgICBib3JkZXI6IDFweCBzb2xpZCB2YXIoLS1ib3JkZXIpO1xuICAgICAgYm9yZGVyLXJhZGl1czogdmFyKC0tcmFkaXVzKTtcbiAgICAgIHRyYW5zaXRpb246IGJvcmRlci1jb2xvciAwLjJzLCBib3gtc2hhZG93IDAuMnM7XG4gICAgICBhcHBlYXJhbmNlOiBub25lO1xuICAgIH1cbiAgICBzZWxlY3Q6Zm9jdXMsIHRleHRhcmVhOmZvY3VzIHtcbiAgICAgIG91dGxpbmU6IG5vbmU7XG4gICAgICBib3JkZXItY29sb3I6IHZhcigtLWFjY2VudCk7XG4gICAgICBib3gtc2hhZG93OiAwIDAgMCAzcHggdmFyKC0tYWNjZW50LWRpbSk7XG4gICAgfVxuICAgIHNlbGVjdCB7XG4gICAgICBiYWNrZ3JvdW5kLWltYWdlOiB1cmwoXCJkYXRhOmltYWdlL3N2Zyt4bWwsJTNDc3ZnIHhtbG5zPSdodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2Zycgd2lkdGg9JzE2JyBoZWlnaHQ9JzE2JyBmaWxsPSclMjM3ZDg1OTAnIHZpZXdCb3g9JzAgMCAxNiAxNiclM0UlM0NwYXRoIGQ9J000LjQyNyA2LjQyN2wzLjM5NiAzLjM5NmEuMjUuMjUgMCAwMC4zNTQgMGwzLjM5Ni0zLjM5NkEuMjUuMjUgMCAwMDExLjM5NiA2SDQuNjA0YS4yNS4yNSAwIDAwLS4xNzcuNDI3eicvJTNFJTNDL3N2ZyUzRVwiKTtcbiAgICAgIGJhY2tncm91bmQtcmVwZWF0OiBuby1yZXBlYXQ7XG4gICAgICBiYWNrZ3JvdW5kLXBvc2l0aW9uOiByaWdodCAxMnB4IGNlbnRlcjtcbiAgICAgIGN1cnNvcjogcG9pbnRlcjtcbiAgICAgIHBhZGRpbmctcmlnaHQ6IDM2cHg7XG4gICAgfVxuICAgIHRleHRhcmVhIHtcbiAgICAgIGZvbnQtZmFtaWx5OiB2YXIoLS1tb25vKTtcbiAgICAgIGZvbnQtc2l6ZTogMC45cmVtO1xuICAgICAgbWluLWhlaWdodDogMTMwcHg7XG4gICAgICByZXNpemU6IHZlcnRpY2FsO1xuICAgICAgbGluZS1oZWlnaHQ6IDEuNjU7XG4gICAgfVxuXG4gICAgLyog4oCUIEJ1dHRvbiDigJQgKi9cbiAgICAuYWN0aW9ucyB7IGRpc3BsYXk6IGZsZXg7IGdhcDogMTJweDsgYWxpZ24taXRlbXM6IGNlbnRlcjsgfVxuXG4gICAgYnV0dG9uIHtcbiAgICAgIHBhZGRpbmc6IDEwcHggMjRweDtcbiAgICAgIGZvbnQtc2l6ZTogMC45cmVtO1xuICAgICAgZm9udC13ZWlnaHQ6IDYwMDtcbiAgICAgIGZvbnQtZmFtaWx5OiB2YXIoLS1zYW5zKTtcbiAgICAgIGJvcmRlcjogbm9uZTtcbiAgICAgIGJvcmRlci1yYWRpdXM6IHZhcigtLXJhZGl1cyk7XG4gICAgICBjdXJzb3I6IHBvaW50ZXI7XG4gICAgICB0cmFuc2l0aW9uOiBhbGwgMC4ycztcbiAgICAgIGJhY2tncm91bmQ6IHZhcigtLWFjY2VudCk7XG4gICAgICBjb2xvcjogIzBkMGYxNDtcbiAgICB9XG4gICAgYnV0dG9uOmhvdmVyIHsgZmlsdGVyOiBicmlnaHRuZXNzKDEuMTUpOyB0cmFuc2Zvcm06IHRyYW5zbGF0ZVkoLTFweCk7IH1cbiAgICBidXR0b246YWN0aXZlIHsgdHJhbnNmb3JtOiB0cmFuc2xhdGVZKDApOyB9XG4gICAgYnV0dG9uOmRpc2FibGVkIHtcbiAgICAgIGJhY2tncm91bmQ6IHZhcigtLXN1cmZhY2UtMik7XG4gICAgICBjb2xvcjogdmFyKC0tdGV4dC1kaW0pO1xuICAgICAgY3Vyc29yOiBub3QtYWxsb3dlZDtcbiAgICAgIGZpbHRlcjogbm9uZTtcbiAgICAgIHRyYW5zZm9ybTogbm9uZTtcbiAgICB9XG5cbiAgICAjc3RhdHVzIHtcbiAgICAgIGZvbnQtc2l6ZTogMC44NXJlbTtcbiAgICAgIGNvbG9yOiB2YXIoLS10ZXh0LW11dGVkKTtcbiAgICB9XG4gICAgLnNwaW5uZXIge1xuICAgICAgZGlzcGxheTogaW5saW5lLWJsb2NrO1xuICAgICAgd2lkdGg6IDEzcHg7IGhlaWdodDogMTNweDtcbiAgICAgIGJvcmRlcjogMnB4IHNvbGlkIHZhcigtLWJvcmRlcik7XG4gICAgICBib3JkZXItdG9wLWNvbG9yOiB2YXIoLS1hY2NlbnQpO1xuICAgICAgYm9yZGVyLXJhZGl1czogNTAlO1xuICAgICAgYW5pbWF0aW9uOiBzcGluIDAuNnMgbGluZWFyIGluZmluaXRlO1xuICAgICAgdmVydGljYWwtYWxpZ246IG1pZGRsZTtcbiAgICAgIG1hcmdpbi1yaWdodDogNXB4O1xuICAgIH1cbiAgICBAa2V5ZnJhbWVzIHNwaW4geyB0byB7IHRyYW5zZm9ybTogcm90YXRlKDM2MGRlZyk7IH0gfVxuXG4gICAgLyog4oCUIFN0YXRzIGJhciDigJQgKi9cbiAgICAuc3RhdHMtYmFyIHtcbiAgICAgIGRpc3BsYXk6IG5vbmU7XG4gICAgICBnYXA6IDI0cHg7XG4gICAgICBtYXJnaW4tdG9wOiAyOHB4O1xuICAgICAgcGFkZGluZzogMTRweCAxOHB4O1xuICAgICAgYmFja2dyb3VuZDogdmFyKC0tc3VyZmFjZSk7XG4gICAgICBib3JkZXI6IDFweCBzb2xpZCB2YXIoLS1ib3JkZXIpO1xuICAgICAgYm9yZGVyLXJhZGl1czogdmFyKC0tcmFkaXVzKTtcbiAgICAgIGZvbnQtc2l6ZTogMC44NXJlbTtcbiAgICAgIGNvbG9yOiB2YXIoLS10ZXh0LW11dGVkKTtcbiAgICB9XG4gICAgLnN0YXRzLWJhci52aXNpYmxlIHsgZGlzcGxheTogZmxleDsgfVxuICAgIC5zdGF0LXZhbHVlIHtcbiAgICAgIGZvbnQtZmFtaWx5OiB2YXIoLS1tb25vKTtcbiAgICAgIGZvbnQtd2VpZ2h0OiA2MDA7XG4gICAgICBjb2xvcjogdmFyKC0tYWNjZW50LTIpO1xuICAgICAgZm9udC1zaXplOiAxLjFyZW07XG4gICAgfVxuICAgIC5zdGF0LWxhYmVsIHsgZm9udC1zaXplOiAwLjc1cmVtOyBtYXJnaW4tdG9wOiAycHg7IH1cblxuICAgIC8qIOKAlCBPdXRwdXQg4oCUICovXG4gICAgLm91dHB1dCB7IGRpc3BsYXk6IG5vbmU7IG1hcmdpbi10b3A6IDI4cHg7IH1cbiAgICAub3V0cHV0LnZpc2libGUgeyBkaXNwbGF5OiBibG9jazsgfVxuXG4gICAgLnNlY3Rpb24tdGl0bGUge1xuICAgICAgZm9udC1zaXplOiAwLjhyZW07XG4gICAgICBmb250LXdlaWdodDogNjAwO1xuICAgICAgdGV4dC10cmFuc2Zvcm06IHVwcGVyY2FzZTtcbiAgICAgIGxldHRlci1zcGFjaW5nOiAwLjA2ZW07XG4gICAgICBjb2xvcjogdmFyKC0tdGV4dC1tdXRlZCk7XG4gICAgICBtYXJnaW4tYm90dG9tOiAxMHB4O1xuICAgIH1cbiAgICAuc2VjdGlvbi10aXRsZSBzbWFsbCB7XG4gICAgICBmb250LXdlaWdodDogNDAwO1xuICAgICAgdGV4dC10cmFuc2Zvcm06IG5vbmU7XG4gICAgICBsZXR0ZXItc3BhY2luZzogMDtcbiAgICAgIGNvbG9yOiB2YXIoLS10ZXh0LWRpbSk7XG4gICAgfVxuXG4gICAgLnRva2VuLWdyaWQge1xuICAgICAgZGlzcGxheTogZmxleDtcbiAgICAgIGZsZXgtd3JhcDogd3JhcDtcbiAgICAgIGdhcDogNHB4O1xuICAgICAgcGFkZGluZzogMTRweDtcbiAgICAgIGJhY2tncm91bmQ6IHZhcigtLXN1cmZhY2UpO1xuICAgICAgYm9yZGVyOiAxcHggc29saWQgdmFyKC0tYm9yZGVyKTtcbiAgICAgIGJvcmRlci1yYWRpdXM6IHZhcigtLXJhZGl1cyk7XG4gICAgICBsaW5lLWhlaWdodDogMS45O1xuICAgICAgbWluLWhlaWdodDogNTJweDtcbiAgICB9XG4gICAgLnRvayB7XG4gICAgICBkaXNwbGF5OiBpbmxpbmUtYmxvY2s7XG4gICAgICBwYWRkaW5nOiAycHggN3B4O1xuICAgICAgYm9yZGVyLXJhZGl1czogNHB4O1xuICAgICAgZm9udC1mYW1pbHk6IHZhcigtLW1vbm8pO1xuICAgICAgZm9udC1zaXplOiAwLjg1cmVtO1xuICAgICAgY3Vyc29yOiBkZWZhdWx0O1xuICAgICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgICAgdHJhbnNpdGlvbjogZmlsdGVyIDAuMTVzO1xuICAgICAgd2hpdGUtc3BhY2U6IHByZTtcbiAgICB9XG4gICAgLnRvazpob3ZlciB7IGZpbHRlcjogYnJpZ2h0bmVzcygxLjQpOyB9XG4gICAgLnRvazo6YWZ0ZXIge1xuICAgICAgY29udGVudDogYXR0cihkYXRhLWlkKTtcbiAgICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICAgIGJvdHRvbTogY2FsYygxMDAlICsgNHB4KTtcbiAgICAgIGxlZnQ6IDUwJTtcbiAgICAgIHRyYW5zZm9ybTogdHJhbnNsYXRlWCgtNTAlKSBzY2FsZSgwLjkpO1xuICAgICAgYmFja2dyb3VuZDogIzJkMzMzYjtcbiAgICAgIGNvbG9yOiAjZTZlZGYzO1xuICAgICAgcGFkZGluZzogM3B4IDhweDtcbiAgICAgIGJvcmRlci1yYWRpdXM6IDVweDtcbiAgICAgIGZvbnQtc2l6ZTogMC43cmVtO1xuICAgICAgd2hpdGUtc3BhY2U6IG5vd3JhcDtcbiAgICAgIG9wYWNpdHk6IDA7XG4gICAgICBwb2ludGVyLWV2ZW50czogbm9uZTtcbiAgICAgIHRyYW5zaXRpb246IG9wYWNpdHkgMC4xNXMsIHRyYW5zZm9ybSAwLjE1cztcbiAgICAgIHotaW5kZXg6IDEwO1xuICAgICAgYm94LXNoYWRvdzogMCAycHggOHB4IHJnYmEoMCwwLDAsMC40KTtcbiAgICB9XG4gICAgLnRvazpob3Zlcjo6YWZ0ZXIgeyBvcGFjaXR5OiAxOyB0cmFuc2Zvcm06IHRyYW5zbGF0ZVgoLTUwJSkgc2NhbGUoMSk7IH1cblxuICAgIC5pZHMtYm94IHtcbiAgICAgIG1hcmdpbi10b3A6IDE2cHg7XG4gICAgICBwYWRkaW5nOiAxNHB4O1xuICAgICAgYmFja2dyb3VuZDogdmFyKC0tc3VyZmFjZSk7XG4gICAgICBib3JkZXI6IDFweCBzb2xpZCB2YXIoLS1ib3JkZXIpO1xuICAgICAgYm9yZGVyLXJhZGl1czogdmFyKC0tcmFkaXVzKTtcbiAgICAgIGZvbnQtZmFtaWx5OiB2YXIoLS1tb25vKTtcbiAgICAgIGZvbnQtc2l6ZTogMC44cmVtO1xuICAgICAgY29sb3I6IHZhcigtLXRleHQtbXV0ZWQpO1xuICAgICAgd29yZC1icmVhazogYnJlYWstYWxsO1xuICAgICAgbWF4LWhlaWdodDogMTgwcHg7XG4gICAgICBvdmVyZmxvdy15OiBhdXRvO1xuICAgICAgbGluZS1oZWlnaHQ6IDEuNztcbiAgICB9XG5cbiAgICAvKiDigJQgRGl2aWRlciDigJQgKi9cbiAgICBociB7XG4gICAgICBib3JkZXI6IG5vbmU7XG4gICAgICBib3JkZXItdG9wOiAxcHggc29saWQgdmFyKC0tYm9yZGVyKTtcbiAgICAgIG1hcmdpbjogMjhweCAwO1xuICAgIH1cblxuICAgIC8qIHNjcm9sbGJhciAqL1xuICAgIDo6LXdlYmtpdC1zY3JvbGxiYXIgeyB3aWR0aDogNnB4OyB9XG4gICAgOjotd2Via2l0LXNjcm9sbGJhci10cmFjayB7IGJhY2tncm91bmQ6IHRyYW5zcGFyZW50OyB9XG4gICAgOjotd2Via2l0LXNjcm9sbGJhci10aHVtYiB7IGJhY2tncm91bmQ6IHZhcigtLWJvcmRlcik7IGJvcmRlci1yYWRpdXM6IDNweDsgfVxuICA8L3N0eWxlPlxuPC9oZWFkPlxuPGJvZHk+XG48ZGl2IGNsYXNzPVwicGFnZVwiPlxuICA8aGVhZGVyPlxuICAgIDxkaXYgY2xhc3M9XCJiYWRnZVwiPjxzcGFuIGNsYXNzPVwiZG90XCI+PC9zcGFuPiBSdW5zIGVudGlyZWx5IGluIHlvdXIgYnJvd3NlcjwvZGl2PlxuICAgIDxoMT5Ub2tlbml6ZXIgUGxheWdyb3VuZDwvaDE+XG4gICAgPHAgY2xhc3M9XCJzdWJ0aXRsZVwiPlxuICAgICAgVmlzdWFsaXplIGhvdyBMTE1zIGJyZWFrIHRleHQgaW50byB0b2tlbnMuIFBvd2VyZWQgYnlcbiAgICAgIDxhIGhyZWY9XCJodHRwczovL2h1Z2dpbmdmYWNlLmNvL2RvY3MvdHJhbnNmb3JtZXJzLmpzXCIgdGFyZ2V0PVwiX2JsYW5rXCI+VHJhbnNmb3JtZXJzLmpzPC9hPlxuICAgIDwvcD5cbiAgPC9oZWFkZXI+XG5cbiAgPGRpdiBjbGFzcz1cImZpZWxkXCI+XG4gICAgPGxhYmVsIGZvcj1cIm1vZGVsLXNlbGVjdFwiPlRva2VuaXplcjwvbGFiZWw+XG4gICAgPHNlbGVjdCBpZD1cIm1vZGVsLXNlbGVjdFwiPlxuICAgICAgPG9wdGdyb3VwIGxhYmVsPVwiT3BlbkFJXCI+XG4gICAgICAgIDxvcHRpb24gdmFsdWU9XCJYZW5vdmEvZ3B0LTRvXCIgc2VsZWN0ZWQ+R1BULTRvICZuYnNwOyhvMjAwa19iYXNlKTwvb3B0aW9uPlxuICAgICAgICA8b3B0aW9uIHZhbHVlPVwiWGVub3ZhL2dwdC00XCI+R1BULTQgLyAzLjUgVHVyYm8gJm5ic3A7KGNsMTAwa19iYXNlKTwvb3B0aW9uPlxuICAgICAgICA8b3B0aW9uIHZhbHVlPVwiWGVub3ZhL2dwdDJcIj5HUFQtMiAmbmJzcDsoYnBlIDUwayk8L29wdGlvbj5cbiAgICAgIDwvb3B0Z3JvdXA+XG4gICAgICA8b3B0Z3JvdXAgbGFiZWw9XCJNZXRhXCI+XG4gICAgICAgIDxvcHRpb24gdmFsdWU9XCJYZW5vdmEvbGxhbWEzLXRva2VuaXplclwiPkxMYU1BIDMgJm5ic3A7KDEyOGsgdm9jYWIpPC9vcHRpb24+XG4gICAgICAgIDxvcHRpb24gdmFsdWU9XCJYZW5vdmEvbGxhbWEtMTYwbVwiPkxMYU1BIDIgJm5ic3A7KDMyayB2b2NhYik8L29wdGlvbj5cbiAgICAgIDwvb3B0Z3JvdXA+XG4gICAgICA8b3B0Z3JvdXAgbGFiZWw9XCJNaXN0cmFsXCI+XG4gICAgICAgIDxvcHRpb24gdmFsdWU9XCJYZW5vdmEvbWlzdHJhbC10b2tlbml6ZXItdjNcIj5NaXN0cmFsIHYzICZuYnNwOyhUZWtrZW4pPC9vcHRpb24+XG4gICAgICAgIDxvcHRpb24gdmFsdWU9XCJYZW5vdmEvbWlzdHJhbC10b2tlbml6ZXItdjFcIj5NaXN0cmFsIHYxICZuYnNwOyhTZW50ZW5jZVBpZWNlKTwvb3B0aW9uPlxuICAgICAgPC9vcHRncm91cD5cbiAgICAgIDxvcHRncm91cCBsYWJlbD1cIkdvb2dsZVwiPlxuICAgICAgICA8b3B0aW9uIHZhbHVlPVwiWGVub3ZhL2dlbW1hLTItdG9rZW5pemVyXCI+R2VtbWEgMiAmbmJzcDsoMjU2ayB2b2NhYik8L29wdGlvbj5cbiAgICAgIDwvb3B0Z3JvdXA+XG4gICAgICA8b3B0Z3JvdXAgbGFiZWw9XCJNaWNyb3NvZnRcIj5cbiAgICAgICAgPG9wdGlvbiB2YWx1ZT1cIlhlbm92YS9QaGktMy1taW5pLTRrLWluc3RydWN0XCI+UGhpLTMgJm5ic3A7KDMyayB2b2NhYik8L29wdGlvbj5cbiAgICAgIDwvb3B0Z3JvdXA+XG4gICAgICA8b3B0Z3JvdXAgbGFiZWw9XCJBbGliYWJhXCI+XG4gICAgICAgIDxvcHRpb24gdmFsdWU9XCJYZW5vdmEvUXdlbjEuNS0wLjVCXCI+UXdlbiAxLjUgJm5ic3A7KDE1Mmsgdm9jYWIpPC9vcHRpb24+XG4gICAgICA8L29wdGdyb3VwPlxuICAgICAgPG9wdGdyb3VwIGxhYmVsPVwiRGVlcFNlZWtcIj5cbiAgICAgICAgPG9wdGlvbiB2YWx1ZT1cIlhlbm92YS9kZWVwc2Vlay1jb2Rlci0xLjNiLWJhc2VcIj5EZWVwU2VlayBDb2RlciAmbmJzcDsoMzJrIHZvY2FiKTwvb3B0aW9uPlxuICAgICAgPC9vcHRncm91cD5cbiAgICAgIDxvcHRncm91cCBsYWJlbD1cIkFudGhyb3BpY1wiPlxuICAgICAgICA8b3B0aW9uIHZhbHVlPVwiWGVub3ZhL2NsYXVkZS10b2tlbml6ZXJcIj5DbGF1ZGUgJm5ic3A7KGJwZSk8L29wdGlvbj5cbiAgICAgIDwvb3B0Z3JvdXA+XG4gICAgPC9zZWxlY3Q+XG4gIDwvZGl2PlxuXG4gIDxkaXYgY2xhc3M9XCJmaWVsZFwiPlxuICAgIDxsYWJlbCBmb3I9XCJpbnB1dFwiPklucHV0IFRleHQ8L2xhYmVsPlxuICAgIDx0ZXh0YXJlYSBpZD1cImlucHV0XCIgcGxhY2Vob2xkZXI9XCJUeXBlIG9yIHBhc3RlIHRleHQgaGVyZeKAplwiPkhlbGxvLCB3b3JsZCEgVGhpcyBpcyBhIHRlc3Qgb2YgdGhlIHRva2VuaXplciBydW5uaW5nIGVudGlyZWx5IGluIHlvdXIgYnJvd3Nlci4g8J+agDwvdGV4dGFyZWE+XG4gIDwvZGl2PlxuXG4gIDxkaXYgY2xhc3M9XCJhY3Rpb25zXCI+XG4gICAgPGJ1dHRvbiBpZD1cImJ0blwiIGRpc2FibGVkPkxvYWRpbmfigKY8L2J1dHRvbj5cbiAgICA8c3BhbiBpZD1cInN0YXR1c1wiPjwvc3Bhbj5cbiAgPC9kaXY+XG5cbiAgPGRpdiBjbGFzcz1cInN0YXRzLWJhclwiIGlkPVwic3RhdHNcIj5cbiAgICA8ZGl2PjxkaXYgY2xhc3M9XCJzdGF0LXZhbHVlXCIgaWQ9XCJ0b2tlbi1jb3VudFwiPjA8L2Rpdj48ZGl2IGNsYXNzPVwic3RhdC1sYWJlbFwiPlRva2VuczwvZGl2PjwvZGl2PlxuICAgIDxkaXY+PGRpdiBjbGFzcz1cInN0YXQtdmFsdWVcIiBpZD1cImNoYXItY291bnRcIj4wPC9kaXY+PGRpdiBjbGFzcz1cInN0YXQtbGFiZWxcIj5DaGFyYWN0ZXJzPC9kaXY+PC9kaXY+XG4gICAgPGRpdj48ZGl2IGNsYXNzPVwic3RhdC12YWx1ZVwiIGlkPVwicmF0aW9cIj4wPC9kaXY+PGRpdiBjbGFzcz1cInN0YXQtbGFiZWxcIj5DaGFycyAvIFRva2VuPC9kaXY+PC9kaXY+XG4gIDwvZGl2PlxuXG4gIDxkaXYgY2xhc3M9XCJvdXRwdXRcIiBpZD1cIm91dHB1dFwiPlxuICAgIDxocj5cbiAgICA8ZGl2IGNsYXNzPVwic2VjdGlvbi10aXRsZVwiPlRva2VucyA8c21hbGw+4oCUIGhvdmVyIGZvciBJRDwvc21hbGw+PC9kaXY+XG4gICAgPGRpdiBjbGFzcz1cInRva2VuLWdyaWRcIiBpZD1cInRva2VuLWdyaWRcIj48L2Rpdj5cblxuICAgIDxkaXYgY2xhc3M9XCJzZWN0aW9uLXRpdGxlXCIgc3R5bGU9XCJtYXJnaW4tdG9wOjIwcHhcIj5Ub2tlbiBJRHM8L2Rpdj5cbiAgICA8ZGl2IGNsYXNzPVwiaWRzLWJveFwiIGlkPVwiaWRzLWJveFwiPjwvZGl2PlxuICA8L2Rpdj5cbjwvZGl2PlxuXG48c2NyaXB0IHR5cGU9XCJtb2R1bGVcIj5cbiAgaW1wb3J0IHsgQXV0b1Rva2VuaXplciB9IGZyb20gJ2h0dHBzOi8vY2RuLmpzZGVsaXZyLm5ldC9ucG0vQGh1Z2dpbmdmYWNlL3RyYW5zZm9ybWVyc0AzJztcblxuICBjb25zdCBQQUxFVFRFID0gW1xuICAgIFsnIzJhMWYzZCcsJyNjNGI1ZmQnXSwgWycjMWEyZTFhJywnIzg2ZWZhYyddLCBbJyMxZTI5M2InLCcjN2RkM2ZjJ10sXG4gICAgWycjMzEyZTJhJywnI2ZjZDM0ZCddLCBbJyMyZDFmMmYnLCcjZjlhOGQ0J10sIFsnIzFhMmEyYScsJyM1ZWVhZDQnXSxcbiAgICBbJyMyZTI0MTgnLCcjZmRiYTc0J10sIFsnIzFlMWUzMCcsJyNhNWI0ZmMnXSwgWycjMmExYTFhJywnI2ZjYTVhNSddLFxuICAgIFsnIzFhMmUyNScsJyM2ZWU3YjcnXSxcbiAgXTtcblxuICBjb25zdCAkID0gKHNlbCkgPT4gZG9jdW1lbnQucXVlcnlTZWxlY3RvcihzZWwpO1xuICBjb25zdCBzZWxlY3RFbCAgPSAkKCcjbW9kZWwtc2VsZWN0Jyk7XG4gIGNvbnN0IGlucHV0RWwgICA9ICQoJyNpbnB1dCcpO1xuICBjb25zdCBidG4gICAgICAgPSAkKCcjYnRuJyk7XG4gIGNvbnN0IHN0YXR1c0VsICA9ICQoJyNzdGF0dXMnKTtcbiAgY29uc3Qgc3RhdHNFbCAgID0gJCgnI3N0YXRzJyk7XG4gIGNvbnN0IHRva2VuR3JpZCA9ICQoJyN0b2tlbi1ncmlkJyk7XG4gIGNvbnN0IGlkc0JveCAgICA9ICQoJyNpZHMtYm94Jyk7XG4gIGNvbnN0IG91dHB1dCAgICA9ICQoJyNvdXRwdXQnKTtcblxuICBsZXQgdG9rZW5pemVyID0gbnVsbDtcbiAgbGV0IGN1cnJlbnRNb2RlbCA9ICcnO1xuXG4gIGFzeW5jIGZ1bmN0aW9uIGxvYWRUb2tlbml6ZXIobW9kZWxJZCkge1xuICAgIGlmIChtb2RlbElkID09PSBjdXJyZW50TW9kZWwgJiYgdG9rZW5pemVyKSByZXR1cm47XG4gICAgYnRuLmRpc2FibGVkID0gdHJ1ZTtcbiAgICBidG4udGV4dENvbnRlbnQgPSAnTG9hZGluZ+KApic7XG4gICAgc3RhdHVzRWwuaW5uZXJIVE1MID0gYDxzcGFuIGNsYXNzPVwic3Bpbm5lclwiPjwvc3Bhbj4gRG93bmxvYWRpbmcgJHttb2RlbElkfeKApmA7XG4gICAgdHJ5IHtcbiAgICAgIHRva2VuaXplciA9IGF3YWl0IEF1dG9Ub2tlbml6ZXIuZnJvbV9wcmV0cmFpbmVkKG1vZGVsSWQpO1xuICAgICAgY3VycmVudE1vZGVsID0gbW9kZWxJZDtcbiAgICAgIHN0YXR1c0VsLnRleHRDb250ZW50ID0gJyc7XG4gICAgICBidG4uZGlzYWJsZWQgPSBmYWxzZTtcbiAgICAgIGJ0bi50ZXh0Q29udGVudCA9ICdUb2tlbml6ZSc7XG4gICAgfSBjYXRjaCAoZSkge1xuICAgICAgc3RhdHVzRWwudGV4dENvbnRlbnQgPSAn4p2MICcgKyBlLm1lc3NhZ2U7XG4gICAgICBjb25zb2xlLmVycm9yKGUpO1xuICAgIH1cbiAgfVxuXG4gIGZ1bmN0aW9uIHRva2VuaXplKCkge1xuICAgIGNvbnN0IHRleHQgPSBpbnB1dEVsLnZhbHVlO1xuICAgIGlmICghdGV4dCB8fCAhdG9rZW5pemVyKSByZXR1cm47XG5cbiAgICBjb25zdCBpZHMgPSBBcnJheS5mcm9tKHRva2VuaXplci5lbmNvZGUodGV4dCkpO1xuXG4gICAgdG9rZW5HcmlkLmlubmVySFRNTCA9ICcnO1xuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgaWRzLmxlbmd0aDsgaSsrKSB7XG4gICAgICBjb25zdCBkZWNvZGVkID0gdG9rZW5pemVyLmRlY29kZShbaWRzW2ldXSwgeyBza2lwX3NwZWNpYWxfdG9rZW5zOiBmYWxzZSB9KTtcbiAgICAgIGNvbnN0IFtiZywgZmddID0gUEFMRVRURVtpICUgUEFMRVRURS5sZW5ndGhdO1xuICAgICAgY29uc3Qgc3BhbiA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoJ3NwYW4nKTtcbiAgICAgIHNwYW4uY2xhc3NOYW1lID0gJ3Rvayc7XG4gICAgICBzcGFuLnN0eWxlLmJhY2tncm91bmQgPSBiZztcbiAgICAgIHNwYW4uc3R5bGUuY29sb3IgPSBmZztcbiAgICAgIHNwYW4udGV4dENvbnRlbnQgPSBkZWNvZGVkLnJlcGxhY2UoLyAvZywgJ8K3JykucmVwbGFjZSgvXFxuL2csICfihrUnKS5yZXBsYWNlKC9cXHQvZywgJ+KHpScpO1xuICAgICAgc3Bhbi5zZXRBdHRyaWJ1dGUoJ2RhdGEtaWQnLCBpZHNbaV0pO1xuICAgICAgdG9rZW5HcmlkLmFwcGVuZENoaWxkKHNwYW4pO1xuICAgIH1cblxuICAgIGlkc0JveC50ZXh0Q29udGVudCA9ICdbJyArIGlkcy5qb2luKCcsICcpICsgJ10nO1xuICAgICQoJyN0b2tlbi1jb3VudCcpLnRleHRDb250ZW50ID0gaWRzLmxlbmd0aDtcbiAgICAkKCcjY2hhci1jb3VudCcpLnRleHRDb250ZW50ID0gdGV4dC5sZW5ndGg7XG4gICAgJCgnI3JhdGlvJykudGV4dENvbnRlbnQgPSBpZHMubGVuZ3RoID8gKHRleHQubGVuZ3RoIC8gaWRzLmxlbmd0aCkudG9GaXhlZCgyKSA6ICfigJQnO1xuICAgIHN0YXRzRWwuY2xhc3NMaXN0LmFkZCgndmlzaWJsZScpO1xuICAgIG91dHB1dC5jbGFzc0xpc3QuYWRkKCd2aXNpYmxlJyk7XG4gIH1cblxuICAvLyBFdmVudHNcbiAgc2VsZWN0RWwuYWRkRXZlbnRMaXN0ZW5lcignY2hhbmdlJywgKCkgPT4gbG9hZFRva2VuaXplcihzZWxlY3RFbC52YWx1ZSkpO1xuICBidG4uYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCB0b2tlbml6ZSk7XG4gIGlucHV0RWwuYWRkRXZlbnRMaXN0ZW5lcigna2V5ZG93bicsIChlKSA9PiB7XG4gICAgaWYgKGUua2V5ID09PSAnRW50ZXInICYmIChlLmN0cmxLZXkgfHwgZS5tZXRhS2V5KSkgdG9rZW5pemUoKTtcbiAgfSk7XG5cbiAgLy8gSW5pdGlhbCBsb2FkXG4gIGF3YWl0IGxvYWRUb2tlbml6ZXIoc2VsZWN0RWwudmFsdWUpO1xuPC9zY3JpcHQ+XG48L2JvZHk+XG48L2h0bWw+XG4ifX1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjEsIm91dHB1dCI6NTE1MCwiY2FjaGVSZWFkIjoxMTc1MywiY2FjaGVXcml0ZSI6MzUxLCJ0b3RhbFRva2VucyI6MTcyNTUsImNvc3QiOnsiaW5wdXQiOjAuMDAwMDA1LCJvdXRwdXQiOjAuMTI4NzUsImNhY2hlUmVhZCI6MC4wMDU4NzY1LCJjYWNoZVdyaXRlIjowLjAwMjE5Mzc1LCJ0b3RhbCI6MC4xMzY4MjUyNX19LCJzdG9wUmVhc29uIjoidG9vbFVzZSIsInRpbWVzdGFtcCI6MTc3MTAzNjY5OTU3MX19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6IjljNDgwMGNmIiwicGFyZW50SWQiOiI4ZDJjYzVmMiIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6Mzk6MjMuMzQ0WiIsIm1lc3NhZ2UiOnsicm9sZSI6InRvb2xSZXN1bHQiLCJ0b29sQ2FsbElkIjoidG9vbHVfMDExUzdxRWh4MzNSVW5adjJzblByY2pFIiwidG9vbE5hbWUiOiJ3cml0ZSIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6IlN1Y2Nlc3NmdWxseSB3cm90ZSAxMzYyMiBieXRlcyB0byAvaG9tZS93ci9naC90cnlfd2ViX2dwdS9pbmRleC5odG1sIn1dLCJpc0Vycm9yIjpmYWxzZSwidGltZXN0YW1wIjoxNzcxMDM2NzYzMzM3fX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiY2MxYmUwM2UiLCJwYXJlbnRJZCI6IjljNDgwMGNmIiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjozOToyNy4xNjhaIiwibWVzc2FnZSI6eyJyb2xlIjoiYXNzaXN0YW50IiwiY29udGVudCI6W3sidHlwZSI6InRvb2xDYWxsIiwiaWQiOiJ0b29sdV8wMUY0QXBNUEZtOXBBU3B1ZmpWd2MxbkYiLCJuYW1lIjoiYmFzaCIsImFyZ3VtZW50cyI6eyJjb21tYW5kIjoiIyBLaWxsIG9sZCBzZXJ2ZXIgaWYgc3RpbGwgcnVubmluZ1xucGtpbGwgLWYgXCJweXRob24zIC1tIGh0dHAuc2VydmVyIDgwODBcIiAyPi9kZXYvbnVsbFxuc2xlZXAgMC41XG5jZCAvaG9tZS93ci9naC90cnlfd2ViX2dwdSAmJiBweXRob24zIC1tIGh0dHAuc2VydmVyIDgwODAgJlxuZWNobyBcIlBJRDogJCFcIlxuc2xlZXAgMVxuZWNobyBcIlNlcnZlciBydW5uaW5nIGF0IGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MFwiIn19XSwiYXBpIjoiYW50aHJvcGljLW1lc3NhZ2VzIiwicHJvdmlkZXIiOiJhbnRocm9waWMiLCJtb2RlbCI6ImNsYXVkZS1vcHVzLTQtNiIsInVzYWdlIjp7ImlucHV0IjoxLCJvdXRwdXQiOjE0MiwiY2FjaGVSZWFkIjoxMjEwNCwiY2FjaGVXcml0ZSI6NTE4NywidG90YWxUb2tlbnMiOjE3NDM0LCJjb3N0Ijp7ImlucHV0IjowLjAwMDAwNSwib3V0cHV0IjowLjAwMzU1LCJjYWNoZVJlYWQiOjAuMDA2MDUyLCJjYWNoZVdyaXRlIjowLjAzMjQxODc1LCJ0b3RhbCI6MC4wNDIwMjU3NX19LCJzdG9wUmVhc29uIjoidG9vbFVzZSIsInRpbWVzdGFtcCI6MTc3MTAzNjc2MzM0NH19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6ImExN2ZmMjc3IiwicGFyZW50SWQiOiJjYzFiZTAzZSIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6Mzk6MjcuMjA0WiIsIm1lc3NhZ2UiOnsicm9sZSI6InRvb2xSZXN1bHQiLCJ0b29sQ2FsbElkIjoidG9vbHVfMDFGNEFwTVBGbTlwQVNwdWZqVndjMW5GIiwidG9vbE5hbWUiOiJiYXNoIiwiY29udGVudCI6W3sidHlwZSI6InRleHQiLCJ0ZXh0IjoiKG5vIG91dHB1dCkifV0sImlzRXJyb3IiOmZhbHNlLCJ0aW1lc3RhbXAiOjE3NzEwMzY3NjcyMDR9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiI4Y2JlZjBhZCIsInBhcmVudElkIjoiYTE3ZmYyNzciLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjM5OjI5Ljk1NloiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidG9vbENhbGwiLCJpZCI6InRvb2x1XzAxRjlRWGtZUzJCU1pWWFM0QVpWNjkxZSIsIm5hbWUiOiJiYXNoIiwiYXJndW1lbnRzIjp7ImNvbW1hbmQiOiJjZCAvaG9tZS93ci9naC90cnlfd2ViX2dwdSAmJiBweXRob24zIC1tIGh0dHAuc2VydmVyIDgwODAgPiAvZGV2L251bGwgMj4mMSAmXG5lY2hvIFwiU2VydmVyIFBJRDogJCFcIlxuc2xlZXAgMC41XG5jdXJsIC1zIC1vIC9kZXYvbnVsbCAtdyBcIiV7aHR0cF9jb2RlfVwiIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC8ifX1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjEsIm91dHB1dCI6MTI5LCJjYWNoZVJlYWQiOjE3MjkxLCJjYWNoZVdyaXRlIjoxNTYsInRvdGFsVG9rZW5zIjoxNzU3NywiY29zdCI6eyJpbnB1dCI6MC4wMDAwMDUsIm91dHB1dCI6MC4wMDMyMjUsImNhY2hlUmVhZCI6MC4wMDg2NDU1LCJjYWNoZVdyaXRlIjowLjAwMDk3NTAwMDAwMDAwMDAwMDEsInRvdGFsIjowLjAxMjg1MDV9fSwic3RvcFJlYXNvbiI6InRvb2xVc2UiLCJ0aW1lc3RhbXAiOjE3NzEwMzY3NjcyMDV9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiIxYzZkMzM1YSIsInBhcmVudElkIjoiOGNiZWYwYWQiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjQwOjM0LjQ0NFoiLCJtZXNzYWdlIjp7InJvbGUiOiJ0b29sUmVzdWx0IiwidG9vbENhbGxJZCI6InRvb2x1XzAxRjlRWGtZUzJCU1pWWFM0QVpWNjkxZSIsInRvb2xOYW1lIjoiYmFzaCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6IlNlcnZlciBQSUQ6IDI4NDAxNlxuMjAwXG5cbkNvbW1hbmQgYWJvcnRlZCJ9XSwiZGV0YWlscyI6e30sImlzRXJyb3IiOnRydWUsInRpbWVzdGFtcCI6MTc3MTAzNjgzNDQ0M319LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6Ijc5NTFmYzE0IiwicGFyZW50SWQiOiIxYzZkMzM1YSIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDA6MzQuNDU2WiIsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOltdLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjAsIm91dHB1dCI6MCwiY2FjaGVSZWFkIjowLCJjYWNoZVdyaXRlIjowLCJ0b3RhbFRva2VucyI6MCwiY29zdCI6eyJpbnB1dCI6MCwib3V0cHV0IjowLCJjYWNoZVJlYWQiOjAsImNhY2hlV3JpdGUiOjAsInRvdGFsIjowfX0sInN0b3BSZWFzb24iOiJhYm9ydGVkIiwidGltZXN0YW1wIjoxNzcxMDM2ODM0NDQ1LCJlcnJvck1lc3NhZ2UiOiJPcGVyYXRpb24gYWJvcnRlZCJ9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiJiMGU1YmQ2YSIsInBhcmVudElkIjoiNzk1MWZjMTQiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjQwOjQzLjIzMVoiLCJtZXNzYWdlIjp7InJvbGUiOiJ1c2VyIiwiY29udGVudCI6W3sidHlwZSI6InRleHQiLCJ0ZXh0IjoiYWRkIGEgbGlnaHQvZGFyayB0aGVtZSB0b2dnbGUifV0sInRpbWVzdGFtcCI6MTc3MTAzNjg0MzIyOH19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6ImJlNmQ5NmIzIiwicGFyZW50SWQiOiJiMGU1YmQ2YSIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDA6NDYuODY5WiIsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0b29sQ2FsbCIsImlkIjoidG9vbHVfMDFHSk1MMkJZU0FFaHM4RExSTU1IbUxlIiwibmFtZSI6InJlYWQiLCJhcmd1bWVudHMiOnsicGF0aCI6Ii9ob21lL3dyL2doL3RyeV93ZWJfZ3B1L2luZGV4Lmh0bWwifX1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjMsIm91dHB1dCI6NjgsImNhY2hlUmVhZCI6MTc0NDcsImNhY2hlV3JpdGUiOjE3MiwidG90YWxUb2tlbnMiOjE3NjkwLCJjb3N0Ijp7ImlucHV0IjowLjAwMDAxNTAwMDAwMDAwMDAwMDAwMiwib3V0cHV0IjowLjAwMTcwMDAwMDAwMDAwMDAwMDEsImNhY2hlUmVhZCI6MC4wMDg3MjM1LCJjYWNoZVdyaXRlIjowLjAwMTA3NSwidG90YWwiOjAuMDExNTEzNX19LCJzdG9wUmVhc29uIjoidG9vbFVzZSIsInRpbWVzdGFtcCI6MTc3MTAzNjg0MzIyOX19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6ImU0MTI3ZGQ1IiwicGFyZW50SWQiOiJiZTZkOTZiMyIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDA6NDYuODkxWiIsIm1lc3NhZ2UiOnsicm9sZSI6InRvb2xSZXN1bHQiLCJ0b29sQ2FsbElkIjoidG9vbHVfMDFHSk1MMkJZU0FFaHM4RExSTU1IbUxlIiwidG9vbE5hbWUiOiJyZWFkIiwiY29udGVudCI6W3sidHlwZSI6InRleHQiLCJ0ZXh0IjoiPCFET0NUWVBFIGh0bWw+XG48aHRtbCBsYW5nPVwiZW5cIj5cbjxoZWFkPlxuICA8bWV0YSBjaGFyc2V0PVwiVVRGLThcIj5cbiAgPG1ldGEgbmFtZT1cInZpZXdwb3J0XCIgY29udGVudD1cIndpZHRoPWRldmljZS13aWR0aCwgaW5pdGlhbC1zY2FsZT0xLjBcIj5cbiAgPHRpdGxlPlRva2VuaXplciBQbGF5Z3JvdW5kPC90aXRsZT5cbiAgPHN0eWxlPlxuICAgIEBpbXBvcnQgdXJsKCdodHRwczovL2ZvbnRzLmdvb2dsZWFwaXMuY29tL2NzczI/ZmFtaWx5PUludGVyOndnaHRANDAwOzUwMDs2MDA7NzAwJmZhbWlseT1JQk0rUGxleCtNb25vOndnaHRANDAwOzUwMCZkaXNwbGF5PXN3YXAnKTtcblxuICAgIDpyb290IHtcbiAgICAgIC0tYmc6ICMwZDBmMTQ7XG4gICAgICAtLXN1cmZhY2U6ICMxMzE1MWM7XG4gICAgICAtLXN1cmZhY2UtMjogIzE5MWMyNjtcbiAgICAgIC0tYm9yZGVyOiAjMjMyNzNhO1xuICAgICAgLS1ib3JkZXItaG92ZXI6ICMzNDNhNTQ7XG4gICAgICAtLXRleHQ6ICNlNmVkZjM7XG4gICAgICAtLXRleHQtbXV0ZWQ6ICM3ZDg1OTA7XG4gICAgICAtLXRleHQtZGltOiAjNDg0ZjU4O1xuICAgICAgLS1hY2NlbnQ6ICNiYzhjZmY7XG4gICAgICAtLWFjY2VudC1kaW06IHJnYmEoMTg4LCAxNDAsIDI1NSwgMC4xMik7XG4gICAgICAtLWFjY2VudC0yOiAjZDJhOGZmO1xuICAgICAgLS1tb25vOiAnSUJNIFBsZXggTW9ubycsICdTRiBNb25vJywgJ0ZpcmEgQ29kZScsIG1vbm9zcGFjZTtcbiAgICAgIC0tc2FuczogJ0ludGVyJywgLWFwcGxlLXN5c3RlbSwgQmxpbmtNYWNTeXN0ZW1Gb250LCBzYW5zLXNlcmlmO1xuICAgICAgLS1yYWRpdXM6IDhweDtcbiAgICB9XG5cbiAgICAqIHsgYm94LXNpemluZzogYm9yZGVyLWJveDsgbWFyZ2luOiAwOyBwYWRkaW5nOiAwOyB9XG5cbiAgICBib2R5IHtcbiAgICAgIGZvbnQtZmFtaWx5OiB2YXIoLS1zYW5zKTtcbiAgICAgIGJhY2tncm91bmQ6IHZhcigtLWJnKTtcbiAgICAgIGNvbG9yOiB2YXIoLS10ZXh0KTtcbiAgICAgIG1pbi1oZWlnaHQ6IDEwMHZoO1xuICAgICAgLXdlYmtpdC1mb250LXNtb290aGluZzogYW50aWFsaWFzZWQ7XG4gICAgfVxuXG4gICAgLyog4oCUIE5vaXNlIHRleHR1cmUgb3ZlcmxheSDigJQgKi9cbiAgICBib2R5OjpiZWZvcmUge1xuICAgICAgY29udGVudDogJyc7XG4gICAgICBwb3NpdGlvbjogZml4ZWQ7XG4gICAgICBpbnNldDogMDtcbiAgICAgIGJhY2tncm91bmQ6IHVybChcImRhdGE6aW1hZ2Uvc3ZnK3htbCwlM0Nzdmcgdmlld0JveD0nMCAwIDI1NiAyNTYnIHhtbG5zPSdodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyclM0UlM0NmaWx0ZXIgaWQ9J24nJTNFJTNDZmVUdXJidWxlbmNlIHR5cGU9J2ZyYWN0YWxOb2lzZScgYmFzZUZyZXF1ZW5jeT0nMC45JyBudW1PY3RhdmVzPSc0JyBzdGl0Y2hUaWxlcz0nc3RpdGNoJy8lM0UlM0MvZmlsdGVyJTNFJTNDcmVjdCB3aWR0aD0nMTAwJTI1JyBoZWlnaHQ9JzEwMCUyNScgZmlsdGVyPSd1cmwoJTIzbiknIG9wYWNpdHk9JzAuMDMnLyUzRSUzQy9zdmclM0VcIik7XG4gICAgICBwb2ludGVyLWV2ZW50czogbm9uZTtcbiAgICAgIHotaW5kZXg6IDA7XG4gICAgfVxuXG4gICAgLnBhZ2Uge1xuICAgICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgICAgei1pbmRleDogMTtcbiAgICAgIG1heC13aWR0aDogNzgwcHg7XG4gICAgICBtYXJnaW46IDAgYXV0bztcbiAgICAgIHBhZGRpbmc6IDYwcHggMjRweCA4MHB4O1xuICAgIH1cblxuICAgIC8qIOKAlCBIZWFkZXIg4oCUICovXG4gICAgaGVhZGVyIHsgbWFyZ2luLWJvdHRvbTogNDhweDsgfVxuXG4gICAgLmJhZGdlIHtcbiAgICAgIGRpc3BsYXk6IGlubGluZS1mbGV4O1xuICAgICAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiAgICAgIGdhcDogNnB4O1xuICAgICAgcGFkZGluZzogNHB4IDEycHg7XG4gICAgICBib3JkZXI6IDFweCBzb2xpZCB2YXIoLS1ib3JkZXIpO1xuICAgICAgYm9yZGVyLXJhZGl1czogOTk5cHg7XG4gICAgICBmb250LXNpemU6IDAuNzVyZW07XG4gICAgICBjb2xvcjogdmFyKC0tdGV4dC1tdXRlZCk7XG4gICAgICBtYXJnaW4tYm90dG9tOiAyMHB4O1xuICAgICAgYmFja2dyb3VuZDogdmFyKC0tc3VyZmFjZSk7XG4gICAgfVxuICAgIC5iYWRnZSAuZG90IHtcbiAgICAgIHdpZHRoOiA2cHg7IGhlaWdodDogNnB4O1xuICAgICAgYm9yZGVyLXJhZGl1czogNTAlO1xuICAgICAgYmFja2dyb3VuZDogIzNmYjk1MDtcbiAgICAgIGJveC1zaGFkb3c6IDAgMCA2cHggIzNmYjk1MDY2O1xuICAgIH1cblxuICAgIGgxIHtcbiAgICAgIGZvbnQtc2l6ZTogMi40cmVtO1xuICAgICAgZm9udC13ZWlnaHQ6IDcwMDtcbiAgICAgIGxldHRlci1zcGFjaW5nOiAtMC4wM2VtO1xuICAgICAgbGluZS1oZWlnaHQ6IDEuMTU7XG4gICAgICBiYWNrZ3JvdW5kOiBsaW5lYXItZ3JhZGllbnQoMTM1ZGVnLCAjZmZmIDAlLCB2YXIoLS1hY2NlbnQtMikgMTAwJSk7XG4gICAgICAtd2Via2l0LWJhY2tncm91bmQtY2xpcDogdGV4dDtcbiAgICAgIC13ZWJraXQtdGV4dC1maWxsLWNvbG9yOiB0cmFuc3BhcmVudDtcbiAgICAgIGJhY2tncm91bmQtY2xpcDogdGV4dDtcbiAgICB9XG5cbiAgICAuc3VidGl0bGUge1xuICAgICAgbWFyZ2luLXRvcDogMTBweDtcbiAgICAgIGZvbnQtc2l6ZTogMS4wNXJlbTtcbiAgICAgIGNvbG9yOiB2YXIoLS10ZXh0LW11dGVkKTtcbiAgICAgIGxpbmUtaGVpZ2h0OiAxLjY7XG4gICAgfVxuICAgIC5zdWJ0aXRsZSBhIHtcbiAgICAgIGNvbG9yOiB2YXIoLS1hY2NlbnQpO1xuICAgICAgdGV4dC1kZWNvcmF0aW9uOiBub25lO1xuICAgICAgYm9yZGVyLWJvdHRvbTogMXB4IHNvbGlkIHRyYW5zcGFyZW50O1xuICAgICAgdHJhbnNpdGlvbjogYm9yZGVyLWNvbG9yIDAuMnM7XG4gICAgfVxuICAgIC5zdWJ0aXRsZSBhOmhvdmVyIHsgYm9yZGVyLWJvdHRvbS1jb2xvcjogdmFyKC0tYWNjZW50KTsgfVxuXG4gICAgLyog4oCUIEZvcm0gY29udHJvbHMg4oCUICovXG4gICAgLmZpZWxkIHsgbWFyZ2luLWJvdHRvbTogMjBweDsgfVxuXG4gICAgbGFiZWwge1xuICAgICAgZGlzcGxheTogYmxvY2s7XG4gICAgICBmb250LXNpemU6IDAuOHJlbTtcbiAgICAgIGZvbnQtd2VpZ2h0OiA2MDA7XG4gICAgICB0ZXh0LXRyYW5zZm9ybTogdXBwZXJjYXNlO1xuICAgICAgbGV0dGVyLXNwYWNpbmc6IDAuMDZlbTtcbiAgICAgIGNvbG9yOiB2YXIoLS10ZXh0LW11dGVkKTtcbiAgICAgIG1hcmdpbi1ib3R0b206IDhweDtcbiAgICB9XG5cbiAgICBzZWxlY3QsIHRleHRhcmVhIHtcbiAgICAgIHdpZHRoOiAxMDAlO1xuICAgICAgcGFkZGluZzogMTJweCAxNHB4O1xuICAgICAgZm9udC1zaXplOiAwLjk1cmVtO1xuICAgICAgZm9udC1mYW1pbHk6IHZhcigtLXNhbnMpO1xuICAgICAgY29sb3I6IHZhcigtLXRleHQpO1xuICAgICAgYmFja2dyb3VuZDogdmFyKC0tc3VyZmFjZSk7XG4gICAgICBib3JkZXI6IDFweCBzb2xpZCB2YXIoLS1ib3JkZXIpO1xuICAgICAgYm9yZGVyLXJhZGl1czogdmFyKC0tcmFkaXVzKTtcbiAgICAgIHRyYW5zaXRpb246IGJvcmRlci1jb2xvciAwLjJzLCBib3gtc2hhZG93IDAuMnM7XG4gICAgICBhcHBlYXJhbmNlOiBub25lO1xuICAgIH1cbiAgICBzZWxlY3Q6Zm9jdXMsIHRleHRhcmVhOmZvY3VzIHtcbiAgICAgIG91dGxpbmU6IG5vbmU7XG4gICAgICBib3JkZXItY29sb3I6IHZhcigtLWFjY2VudCk7XG4gICAgICBib3gtc2hhZG93OiAwIDAgMCAzcHggdmFyKC0tYWNjZW50LWRpbSk7XG4gICAgfVxuICAgIHNlbGVjdCB7XG4gICAgICBiYWNrZ3JvdW5kLWltYWdlOiB1cmwoXCJkYXRhOmltYWdlL3N2Zyt4bWwsJTNDc3ZnIHhtbG5zPSdodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2Zycgd2lkdGg9JzE2JyBoZWlnaHQ9JzE2JyBmaWxsPSclMjM3ZDg1OTAnIHZpZXdCb3g9JzAgMCAxNiAxNiclM0UlM0NwYXRoIGQ9J000LjQyNyA2LjQyN2wzLjM5NiAzLjM5NmEuMjUuMjUgMCAwMC4zNTQgMGwzLjM5Ni0zLjM5NkEuMjUuMjUgMCAwMDExLjM5NiA2SDQuNjA0YS4yNS4yNSAwIDAwLS4xNzcuNDI3eicvJTNFJTNDL3N2ZyUzRVwiKTtcbiAgICAgIGJhY2tncm91bmQtcmVwZWF0OiBuby1yZXBlYXQ7XG4gICAgICBiYWNrZ3JvdW5kLXBvc2l0aW9uOiByaWdodCAxMnB4IGNlbnRlcjtcbiAgICAgIGN1cnNvcjogcG9pbnRlcjtcbiAgICAgIHBhZGRpbmctcmlnaHQ6IDM2cHg7XG4gICAgfVxuICAgIHRleHRhcmVhIHtcbiAgICAgIGZvbnQtZmFtaWx5OiB2YXIoLS1tb25vKTtcbiAgICAgIGZvbnQtc2l6ZTogMC45cmVtO1xuICAgICAgbWluLWhlaWdodDogMTMwcHg7XG4gICAgICByZXNpemU6IHZlcnRpY2FsO1xuICAgICAgbGluZS1oZWlnaHQ6IDEuNjU7XG4gICAgfVxuXG4gICAgLyog4oCUIEJ1dHRvbiDigJQgKi9cbiAgICAuYWN0aW9ucyB7IGRpc3BsYXk6IGZsZXg7IGdhcDogMTJweDsgYWxpZ24taXRlbXM6IGNlbnRlcjsgfVxuXG4gICAgYnV0dG9uIHtcbiAgICAgIHBhZGRpbmc6IDEwcHggMjRweDtcbiAgICAgIGZvbnQtc2l6ZTogMC45cmVtO1xuICAgICAgZm9udC13ZWlnaHQ6IDYwMDtcbiAgICAgIGZvbnQtZmFtaWx5OiB2YXIoLS1zYW5zKTtcbiAgICAgIGJvcmRlcjogbm9uZTtcbiAgICAgIGJvcmRlci1yYWRpdXM6IHZhcigtLXJhZGl1cyk7XG4gICAgICBjdXJzb3I6IHBvaW50ZXI7XG4gICAgICB0cmFuc2l0aW9uOiBhbGwgMC4ycztcbiAgICAgIGJhY2tncm91bmQ6IHZhcigtLWFjY2VudCk7XG4gICAgICBjb2xvcjogIzBkMGYxNDtcbiAgICB9XG4gICAgYnV0dG9uOmhvdmVyIHsgZmlsdGVyOiBicmlnaHRuZXNzKDEuMTUpOyB0cmFuc2Zvcm06IHRyYW5zbGF0ZVkoLTFweCk7IH1cbiAgICBidXR0b246YWN0aXZlIHsgdHJhbnNmb3JtOiB0cmFuc2xhdGVZKDApOyB9XG4gICAgYnV0dG9uOmRpc2FibGVkIHtcbiAgICAgIGJhY2tncm91bmQ6IHZhcigtLXN1cmZhY2UtMik7XG4gICAgICBjb2xvcjogdmFyKC0tdGV4dC1kaW0pO1xuICAgICAgY3Vyc29yOiBub3QtYWxsb3dlZDtcbiAgICAgIGZpbHRlcjogbm9uZTtcbiAgICAgIHRyYW5zZm9ybTogbm9uZTtcbiAgICB9XG5cbiAgICAjc3RhdHVzIHtcbiAgICAgIGZvbnQtc2l6ZTogMC44NXJlbTtcbiAgICAgIGNvbG9yOiB2YXIoLS10ZXh0LW11dGVkKTtcbiAgICB9XG4gICAgLnNwaW5uZXIge1xuICAgICAgZGlzcGxheTogaW5saW5lLWJsb2NrO1xuICAgICAgd2lkdGg6IDEzcHg7IGhlaWdodDogMTNweDtcbiAgICAgIGJvcmRlcjogMnB4IHNvbGlkIHZhcigtLWJvcmRlcik7XG4gICAgICBib3JkZXItdG9wLWNvbG9yOiB2YXIoLS1hY2NlbnQpO1xuICAgICAgYm9yZGVyLXJhZGl1czogNTAlO1xuICAgICAgYW5pbWF0aW9uOiBzcGluIDAuNnMgbGluZWFyIGluZmluaXRlO1xuICAgICAgdmVydGljYWwtYWxpZ246IG1pZGRsZTtcbiAgICAgIG1hcmdpbi1yaWdodDogNXB4O1xuICAgIH1cbiAgICBAa2V5ZnJhbWVzIHNwaW4geyB0byB7IHRyYW5zZm9ybTogcm90YXRlKDM2MGRlZyk7IH0gfVxuXG4gICAgLyog4oCUIFN0YXRzIGJhciDigJQgKi9cbiAgICAuc3RhdHMtYmFyIHtcbiAgICAgIGRpc3BsYXk6IG5vbmU7XG4gICAgICBnYXA6IDI0cHg7XG4gICAgICBtYXJnaW4tdG9wOiAyOHB4O1xuICAgICAgcGFkZGluZzogMTRweCAxOHB4O1xuICAgICAgYmFja2dyb3VuZDogdmFyKC0tc3VyZmFjZSk7XG4gICAgICBib3JkZXI6IDFweCBzb2xpZCB2YXIoLS1ib3JkZXIpO1xuICAgICAgYm9yZGVyLXJhZGl1czogdmFyKC0tcmFkaXVzKTtcbiAgICAgIGZvbnQtc2l6ZTogMC44NXJlbTtcbiAgICAgIGNvbG9yOiB2YXIoLS10ZXh0LW11dGVkKTtcbiAgICB9XG4gICAgLnN0YXRzLWJhci52aXNpYmxlIHsgZGlzcGxheTogZmxleDsgfVxuICAgIC5zdGF0LXZhbHVlIHtcbiAgICAgIGZvbnQtZmFtaWx5OiB2YXIoLS1tb25vKTtcbiAgICAgIGZvbnQtd2VpZ2h0OiA2MDA7XG4gICAgICBjb2xvcjogdmFyKC0tYWNjZW50LTIpO1xuICAgICAgZm9udC1zaXplOiAxLjFyZW07XG4gICAgfVxuICAgIC5zdGF0LWxhYmVsIHsgZm9udC1zaXplOiAwLjc1cmVtOyBtYXJnaW4tdG9wOiAycHg7IH1cblxuICAgIC8qIOKAlCBPdXRwdXQg4oCUICovXG4gICAgLm91dHB1dCB7IGRpc3BsYXk6IG5vbmU7IG1hcmdpbi10b3A6IDI4cHg7IH1cbiAgICAub3V0cHV0LnZpc2libGUgeyBkaXNwbGF5OiBibG9jazsgfVxuXG4gICAgLnNlY3Rpb24tdGl0bGUge1xuICAgICAgZm9udC1zaXplOiAwLjhyZW07XG4gICAgICBmb250LXdlaWdodDogNjAwO1xuICAgICAgdGV4dC10cmFuc2Zvcm06IHVwcGVyY2FzZTtcbiAgICAgIGxldHRlci1zcGFjaW5nOiAwLjA2ZW07XG4gICAgICBjb2xvcjogdmFyKC0tdGV4dC1tdXRlZCk7XG4gICAgICBtYXJnaW4tYm90dG9tOiAxMHB4O1xuICAgIH1cbiAgICAuc2VjdGlvbi10aXRsZSBzbWFsbCB7XG4gICAgICBmb250LXdlaWdodDogNDAwO1xuICAgICAgdGV4dC10cmFuc2Zvcm06IG5vbmU7XG4gICAgICBsZXR0ZXItc3BhY2luZzogMDtcbiAgICAgIGNvbG9yOiB2YXIoLS10ZXh0LWRpbSk7XG4gICAgfVxuXG4gICAgLnRva2VuLWdyaWQge1xuICAgICAgZGlzcGxheTogZmxleDtcbiAgICAgIGZsZXgtd3JhcDogd3JhcDtcbiAgICAgIGdhcDogNHB4O1xuICAgICAgcGFkZGluZzogMTRweDtcbiAgICAgIGJhY2tncm91bmQ6IHZhcigtLXN1cmZhY2UpO1xuICAgICAgYm9yZGVyOiAxcHggc29saWQgdmFyKC0tYm9yZGVyKTtcbiAgICAgIGJvcmRlci1yYWRpdXM6IHZhcigtLXJhZGl1cyk7XG4gICAgICBsaW5lLWhlaWdodDogMS45O1xuICAgICAgbWluLWhlaWdodDogNTJweDtcbiAgICB9XG4gICAgLnRvayB7XG4gICAgICBkaXNwbGF5OiBpbmxpbmUtYmxvY2s7XG4gICAgICBwYWRkaW5nOiAycHggN3B4O1xuICAgICAgYm9yZGVyLXJhZGl1czogNHB4O1xuICAgICAgZm9udC1mYW1pbHk6IHZhcigtLW1vbm8pO1xuICAgICAgZm9udC1zaXplOiAwLjg1cmVtO1xuICAgICAgY3Vyc29yOiBkZWZhdWx0O1xuICAgICAgcG9zaXRpb246IHJlbGF0aXZlO1xuICAgICAgdHJhbnNpdGlvbjogZmlsdGVyIDAuMTVzO1xuICAgICAgd2hpdGUtc3BhY2U6IHByZTtcbiAgICB9XG4gICAgLnRvazpob3ZlciB7IGZpbHRlcjogYnJpZ2h0bmVzcygxLjQpOyB9XG4gICAgLnRvazo6YWZ0ZXIge1xuICAgICAgY29udGVudDogYXR0cihkYXRhLWlkKTtcbiAgICAgIHBvc2l0aW9uOiBhYnNvbHV0ZTtcbiAgICAgIGJvdHRvbTogY2FsYygxMDAlICsgNHB4KTtcbiAgICAgIGxlZnQ6IDUwJTtcbiAgICAgIHRyYW5zZm9ybTogdHJhbnNsYXRlWCgtNTAlKSBzY2FsZSgwLjkpO1xuICAgICAgYmFja2dyb3VuZDogIzJkMzMzYjtcbiAgICAgIGNvbG9yOiAjZTZlZGYzO1xuICAgICAgcGFkZGluZzogM3B4IDhweDtcbiAgICAgIGJvcmRlci1yYWRpdXM6IDVweDtcbiAgICAgIGZvbnQtc2l6ZTogMC43cmVtO1xuICAgICAgd2hpdGUtc3BhY2U6IG5vd3JhcDtcbiAgICAgIG9wYWNpdHk6IDA7XG4gICAgICBwb2ludGVyLWV2ZW50czogbm9uZTtcbiAgICAgIHRyYW5zaXRpb246IG9wYWNpdHkgMC4xNXMsIHRyYW5zZm9ybSAwLjE1cztcbiAgICAgIHotaW5kZXg6IDEwO1xuICAgICAgYm94LXNoYWRvdzogMCAycHggOHB4IHJnYmEoMCwwLDAsMC40KTtcbiAgICB9XG4gICAgLnRvazpob3Zlcjo6YWZ0ZXIgeyBvcGFjaXR5OiAxOyB0cmFuc2Zvcm06IHRyYW5zbGF0ZVgoLTUwJSkgc2NhbGUoMSk7IH1cblxuICAgIC5pZHMtYm94IHtcbiAgICAgIG1hcmdpbi10b3A6IDE2cHg7XG4gICAgICBwYWRkaW5nOiAxNHB4O1xuICAgICAgYmFja2dyb3VuZDogdmFyKC0tc3VyZmFjZSk7XG4gICAgICBib3JkZXI6IDFweCBzb2xpZCB2YXIoLS1ib3JkZXIpO1xuICAgICAgYm9yZGVyLXJhZGl1czogdmFyKC0tcmFkaXVzKTtcbiAgICAgIGZvbnQtZmFtaWx5OiB2YXIoLS1tb25vKTtcbiAgICAgIGZvbnQtc2l6ZTogMC44cmVtO1xuICAgICAgY29sb3I6IHZhcigtLXRleHQtbXV0ZWQpO1xuICAgICAgd29yZC1icmVhazogYnJlYWstYWxsO1xuICAgICAgbWF4LWhlaWdodDogMTgwcHg7XG4gICAgICBvdmVyZmxvdy15OiBhdXRvO1xuICAgICAgbGluZS1oZWlnaHQ6IDEuNztcbiAgICB9XG5cbiAgICAvKiDigJQgRGl2aWRlciDigJQgKi9cbiAgICBociB7XG4gICAgICBib3JkZXI6IG5vbmU7XG4gICAgICBib3JkZXItdG9wOiAxcHggc29saWQgdmFyKC0tYm9yZGVyKTtcbiAgICAgIG1hcmdpbjogMjhweCAwO1xuICAgIH1cblxuICAgIC8qIHNjcm9sbGJhciAqL1xuICAgIDo6LXdlYmtpdC1zY3JvbGxiYXIgeyB3aWR0aDogNnB4OyB9XG4gICAgOjotd2Via2l0LXNjcm9sbGJhci10cmFjayB7IGJhY2tncm91bmQ6IHRyYW5zcGFyZW50OyB9XG4gICAgOjotd2Via2l0LXNjcm9sbGJhci10aHVtYiB7IGJhY2tncm91bmQ6IHZhcigtLWJvcmRlcik7IGJvcmRlci1yYWRpdXM6IDNweDsgfVxuICA8L3N0eWxlPlxuPC9oZWFkPlxuPGJvZHk+XG48ZGl2IGNsYXNzPVwicGFnZVwiPlxuICA8aGVhZGVyPlxuICAgIDxkaXYgY2xhc3M9XCJiYWRnZVwiPjxzcGFuIGNsYXNzPVwiZG90XCI+PC9zcGFuPiBSdW5zIGVudGlyZWx5IGluIHlvdXIgYnJvd3NlcjwvZGl2PlxuICAgIDxoMT5Ub2tlbml6ZXIgUGxheWdyb3VuZDwvaDE+XG4gICAgPHAgY2xhc3M9XCJzdWJ0aXRsZVwiPlxuICAgICAgVmlzdWFsaXplIGhvdyBMTE1zIGJyZWFrIHRleHQgaW50byB0b2tlbnMuIFBvd2VyZWQgYnlcbiAgICAgIDxhIGhyZWY9XCJodHRwczovL2h1Z2dpbmdmYWNlLmNvL2RvY3MvdHJhbnNmb3JtZXJzLmpzXCIgdGFyZ2V0PVwiX2JsYW5rXCI+VHJhbnNmb3JtZXJzLmpzPC9hPlxuICAgIDwvcD5cbiAgPC9oZWFkZXI+XG5cbiAgPGRpdiBjbGFzcz1cImZpZWxkXCI+XG4gICAgPGxhYmVsIGZvcj1cIm1vZGVsLXNlbGVjdFwiPlRva2VuaXplcjwvbGFiZWw+XG4gICAgPHNlbGVjdCBpZD1cIm1vZGVsLXNlbGVjdFwiPlxuICAgICAgPG9wdGdyb3VwIGxhYmVsPVwiT3BlbkFJXCI+XG4gICAgICAgIDxvcHRpb24gdmFsdWU9XCJYZW5vdmEvZ3B0LTRvXCIgc2VsZWN0ZWQ+R1BULTRvICZuYnNwOyhvMjAwa19iYXNlKTwvb3B0aW9uPlxuICAgICAgICA8b3B0aW9uIHZhbHVlPVwiWGVub3ZhL2dwdC00XCI+R1BULTQgLyAzLjUgVHVyYm8gJm5ic3A7KGNsMTAwa19iYXNlKTwvb3B0aW9uPlxuICAgICAgICA8b3B0aW9uIHZhbHVlPVwiWGVub3ZhL2dwdDJcIj5HUFQtMiAmbmJzcDsoYnBlIDUwayk8L29wdGlvbj5cbiAgICAgIDwvb3B0Z3JvdXA+XG4gICAgICA8b3B0Z3JvdXAgbGFiZWw9XCJNZXRhXCI+XG4gICAgICAgIDxvcHRpb24gdmFsdWU9XCJYZW5vdmEvbGxhbWEzLXRva2VuaXplclwiPkxMYU1BIDMgJm5ic3A7KDEyOGsgdm9jYWIpPC9vcHRpb24+XG4gICAgICAgIDxvcHRpb24gdmFsdWU9XCJYZW5vdmEvbGxhbWEtMTYwbVwiPkxMYU1BIDIgJm5ic3A7KDMyayB2b2NhYik8L29wdGlvbj5cbiAgICAgIDwvb3B0Z3JvdXA+XG4gICAgICA8b3B0Z3JvdXAgbGFiZWw9XCJNaXN0cmFsXCI+XG4gICAgICAgIDxvcHRpb24gdmFsdWU9XCJYZW5vdmEvbWlzdHJhbC10b2tlbml6ZXItdjNcIj5NaXN0cmFsIHYzICZuYnNwOyhUZWtrZW4pPC9vcHRpb24+XG4gICAgICAgIDxvcHRpb24gdmFsdWU9XCJYZW5vdmEvbWlzdHJhbC10b2tlbml6ZXItdjFcIj5NaXN0cmFsIHYxICZuYnNwOyhTZW50ZW5jZVBpZWNlKTwvb3B0aW9uPlxuICAgICAgPC9vcHRncm91cD5cbiAgICAgIDxvcHRncm91cCBsYWJlbD1cIkdvb2dsZVwiPlxuICAgICAgICA8b3B0aW9uIHZhbHVlPVwiWGVub3ZhL2dlbW1hLTItdG9rZW5pemVyXCI+R2VtbWEgMiAmbmJzcDsoMjU2ayB2b2NhYik8L29wdGlvbj5cbiAgICAgIDwvb3B0Z3JvdXA+XG4gICAgICA8b3B0Z3JvdXAgbGFiZWw9XCJNaWNyb3NvZnRcIj5cbiAgICAgICAgPG9wdGlvbiB2YWx1ZT1cIlhlbm92YS9QaGktMy1taW5pLTRrLWluc3RydWN0XCI+UGhpLTMgJm5ic3A7KDMyayB2b2NhYik8L29wdGlvbj5cbiAgICAgIDwvb3B0Z3JvdXA+XG4gICAgICA8b3B0Z3JvdXAgbGFiZWw9XCJBbGliYWJhXCI+XG4gICAgICAgIDxvcHRpb24gdmFsdWU9XCJYZW5vdmEvUXdlbjEuNS0wLjVCXCI+UXdlbiAxLjUgJm5ic3A7KDE1Mmsgdm9jYWIpPC9vcHRpb24+XG4gICAgICA8L29wdGdyb3VwPlxuICAgICAgPG9wdGdyb3VwIGxhYmVsPVwiRGVlcFNlZWtcIj5cbiAgICAgICAgPG9wdGlvbiB2YWx1ZT1cIlhlbm92YS9kZWVwc2Vlay1jb2Rlci0xLjNiLWJhc2VcIj5EZWVwU2VlayBDb2RlciAmbmJzcDsoMzJrIHZvY2FiKTwvb3B0aW9uPlxuICAgICAgPC9vcHRncm91cD5cbiAgICAgIDxvcHRncm91cCBsYWJlbD1cIkFudGhyb3BpY1wiPlxuICAgICAgICA8b3B0aW9uIHZhbHVlPVwiWGVub3ZhL2NsYXVkZS10b2tlbml6ZXJcIj5DbGF1ZGUgJm5ic3A7KGJwZSk8L29wdGlvbj5cbiAgICAgIDwvb3B0Z3JvdXA+XG4gICAgPC9zZWxlY3Q+XG4gIDwvZGl2PlxuXG4gIDxkaXYgY2xhc3M9XCJmaWVsZFwiPlxuICAgIDxsYWJlbCBmb3I9XCJpbnB1dFwiPklucHV0IFRleHQ8L2xhYmVsPlxuICAgIDx0ZXh0YXJlYSBpZD1cImlucHV0XCIgcGxhY2Vob2xkZXI9XCJUeXBlIG9yIHBhc3RlIHRleHQgaGVyZeKAplwiPkhlbGxvLCB3b3JsZCEgVGhpcyBpcyBhIHRlc3Qgb2YgdGhlIHRva2VuaXplciBydW5uaW5nIGVudGlyZWx5IGluIHlvdXIgYnJvd3Nlci4g8J+agDwvdGV4dGFyZWE+XG4gIDwvZGl2PlxuXG4gIDxkaXYgY2xhc3M9XCJhY3Rpb25zXCI+XG4gICAgPGJ1dHRvbiBpZD1cImJ0blwiIGRpc2FibGVkPkxvYWRpbmfigKY8L2J1dHRvbj5cbiAgICA8c3BhbiBpZD1cInN0YXR1c1wiPjwvc3Bhbj5cbiAgPC9kaXY+XG5cbiAgPGRpdiBjbGFzcz1cInN0YXRzLWJhclwiIGlkPVwic3RhdHNcIj5cbiAgICA8ZGl2PjxkaXYgY2xhc3M9XCJzdGF0LXZhbHVlXCIgaWQ9XCJ0b2tlbi1jb3VudFwiPjA8L2Rpdj48ZGl2IGNsYXNzPVwic3RhdC1sYWJlbFwiPlRva2VuczwvZGl2PjwvZGl2PlxuICAgIDxkaXY+PGRpdiBjbGFzcz1cInN0YXQtdmFsdWVcIiBpZD1cImNoYXItY291bnRcIj4wPC9kaXY+PGRpdiBjbGFzcz1cInN0YXQtbGFiZWxcIj5DaGFyYWN0ZXJzPC9kaXY+PC9kaXY+XG4gICAgPGRpdj48ZGl2IGNsYXNzPVwic3RhdC12YWx1ZVwiIGlkPVwicmF0aW9cIj4wPC9kaXY+PGRpdiBjbGFzcz1cInN0YXQtbGFiZWxcIj5DaGFycyAvIFRva2VuPC9kaXY+PC9kaXY+XG4gIDwvZGl2PlxuXG4gIDxkaXYgY2xhc3M9XCJvdXRwdXRcIiBpZD1cIm91dHB1dFwiPlxuICAgIDxocj5cbiAgICA8ZGl2IGNsYXNzPVwic2VjdGlvbi10aXRsZVwiPlRva2VucyA8c21hbGw+4oCUIGhvdmVyIGZvciBJRDwvc21hbGw+PC9kaXY+XG4gICAgPGRpdiBjbGFzcz1cInRva2VuLWdyaWRcIiBpZD1cInRva2VuLWdyaWRcIj48L2Rpdj5cblxuICAgIDxkaXYgY2xhc3M9XCJzZWN0aW9uLXRpdGxlXCIgc3R5bGU9XCJtYXJnaW4tdG9wOjIwcHhcIj5Ub2tlbiBJRHM8L2Rpdj5cbiAgICA8ZGl2IGNsYXNzPVwiaWRzLWJveFwiIGlkPVwiaWRzLWJveFwiPjwvZGl2PlxuICA8L2Rpdj5cbjwvZGl2PlxuXG48c2NyaXB0IHR5cGU9XCJtb2R1bGVcIj5cbiAgaW1wb3J0IHsgQXV0b1Rva2VuaXplciB9IGZyb20gJ2h0dHBzOi8vY2RuLmpzZGVsaXZyLm5ldC9ucG0vQGh1Z2dpbmdmYWNlL3RyYW5zZm9ybWVyc0AzJztcblxuICBjb25zdCBQQUxFVFRFID0gW1xuICAgIFsnIzJhMWYzZCcsJyNjNGI1ZmQnXSwgWycjMWEyZTFhJywnIzg2ZWZhYyddLCBbJyMxZTI5M2InLCcjN2RkM2ZjJ10sXG4gICAgWycjMzEyZTJhJywnI2ZjZDM0ZCddLCBbJyMyZDFmMmYnLCcjZjlhOGQ0J10sIFsnIzFhMmEyYScsJyM1ZWVhZDQnXSxcbiAgICBbJyMyZTI0MTgnLCcjZmRiYTc0J10sIFsnIzFlMWUzMCcsJyNhNWI0ZmMnXSwgWycjMmExYTFhJywnI2ZjYTVhNSddLFxuICAgIFsnIzFhMmUyNScsJyM2ZWU3YjcnXSxcbiAgXTtcblxuICBjb25zdCAkID0gKHNlbCkgPT4gZG9jdW1lbnQucXVlcnlTZWxlY3RvcihzZWwpO1xuICBjb25zdCBzZWxlY3RFbCAgPSAkKCcjbW9kZWwtc2VsZWN0Jyk7XG4gIGNvbnN0IGlucHV0RWwgICA9ICQoJyNpbnB1dCcpO1xuICBjb25zdCBidG4gICAgICAgPSAkKCcjYnRuJyk7XG4gIGNvbnN0IHN0YXR1c0VsICA9ICQoJyNzdGF0dXMnKTtcbiAgY29uc3Qgc3RhdHNFbCAgID0gJCgnI3N0YXRzJyk7XG4gIGNvbnN0IHRva2VuR3JpZCA9ICQoJyN0b2tlbi1ncmlkJyk7XG4gIGNvbnN0IGlkc0JveCAgICA9ICQoJyNpZHMtYm94Jyk7XG4gIGNvbnN0IG91dHB1dCAgICA9ICQoJyNvdXRwdXQnKTtcblxuICBsZXQgdG9rZW5pemVyID0gbnVsbDtcbiAgbGV0IGN1cnJlbnRNb2RlbCA9ICcnO1xuXG4gIGFzeW5jIGZ1bmN0aW9uIGxvYWRUb2tlbml6ZXIobW9kZWxJZCkge1xuICAgIGlmIChtb2RlbElkID09PSBjdXJyZW50TW9kZWwgJiYgdG9rZW5pemVyKSByZXR1cm47XG4gICAgYnRuLmRpc2FibGVkID0gdHJ1ZTtcbiAgICBidG4udGV4dENvbnRlbnQgPSAnTG9hZGluZ+KApic7XG4gICAgc3RhdHVzRWwuaW5uZXJIVE1MID0gYDxzcGFuIGNsYXNzPVwic3Bpbm5lclwiPjwvc3Bhbj4gRG93bmxvYWRpbmcgJHttb2RlbElkfeKApmA7XG4gICAgdHJ5IHtcbiAgICAgIHRva2VuaXplciA9IGF3YWl0IEF1dG9Ub2tlbml6ZXIuZnJvbV9wcmV0cmFpbmVkKG1vZGVsSWQpO1xuICAgICAgY3VycmVudE1vZGVsID0gbW9kZWxJZDtcbiAgICAgIHN0YXR1c0VsLnRleHRDb250ZW50ID0gJyc7XG4gICAgICBidG4uZGlzYWJsZWQgPSBmYWxzZTtcbiAgICAgIGJ0bi50ZXh0Q29udGVudCA9ICdUb2tlbml6ZSc7XG4gICAgfSBjYXRjaCAoZSkge1xuICAgICAgc3RhdHVzRWwudGV4dENvbnRlbnQgPSAn4p2MICcgKyBlLm1lc3NhZ2U7XG4gICAgICBjb25zb2xlLmVycm9yKGUpO1xuICAgIH1cbiAgfVxuXG4gIGZ1bmN0aW9uIHRva2VuaXplKCkge1xuICAgIGNvbnN0IHRleHQgPSBpbnB1dEVsLnZhbHVlO1xuICAgIGlmICghdGV4dCB8fCAhdG9rZW5pemVyKSByZXR1cm47XG5cbiAgICBjb25zdCBpZHMgPSBBcnJheS5mcm9tKHRva2VuaXplci5lbmNvZGUodGV4dCkpO1xuXG4gICAgdG9rZW5HcmlkLmlubmVySFRNTCA9ICcnO1xuICAgIGZvciAobGV0IGkgPSAwOyBpIDwgaWRzLmxlbmd0aDsgaSsrKSB7XG4gICAgICBjb25zdCBkZWNvZGVkID0gdG9rZW5pemVyLmRlY29kZShbaWRzW2ldXSwgeyBza2lwX3NwZWNpYWxfdG9rZW5zOiBmYWxzZSB9KTtcbiAgICAgIGNvbnN0IFtiZywgZmddID0gUEFMRVRURVtpICUgUEFMRVRURS5sZW5ndGhdO1xuICAgICAgY29uc3Qgc3BhbiA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoJ3NwYW4nKTtcbiAgICAgIHNwYW4uY2xhc3NOYW1lID0gJ3Rvayc7XG4gICAgICBzcGFuLnN0eWxlLmJhY2tncm91bmQgPSBiZztcbiAgICAgIHNwYW4uc3R5bGUuY29sb3IgPSBmZztcbiAgICAgIHNwYW4udGV4dENvbnRlbnQgPSBkZWNvZGVkLnJlcGxhY2UoLyAvZywgJ8K3JykucmVwbGFjZSgvXFxuL2csICfihrUnKS5yZXBsYWNlKC9cXHQvZywgJ+KHpScpO1xuICAgICAgc3Bhbi5zZXRBdHRyaWJ1dGUoJ2RhdGEtaWQnLCBpZHNbaV0pO1xuICAgICAgdG9rZW5HcmlkLmFwcGVuZENoaWxkKHNwYW4pO1xuICAgIH1cblxuICAgIGlkc0JveC50ZXh0Q29udGVudCA9ICdbJyArIGlkcy5qb2luKCcsICcpICsgJ10nO1xuICAgICQoJyN0b2tlbi1jb3VudCcpLnRleHRDb250ZW50ID0gaWRzLmxlbmd0aDtcbiAgICAkKCcjY2hhci1jb3VudCcpLnRleHRDb250ZW50ID0gdGV4dC5sZW5ndGg7XG4gICAgJCgnI3JhdGlvJykudGV4dENvbnRlbnQgPSBpZHMubGVuZ3RoID8gKHRleHQubGVuZ3RoIC8gaWRzLmxlbmd0aCkudG9GaXhlZCgyKSA6ICfigJQnO1xuICAgIHN0YXRzRWwuY2xhc3NMaXN0LmFkZCgndmlzaWJsZScpO1xuICAgIG91dHB1dC5jbGFzc0xpc3QuYWRkKCd2aXNpYmxlJyk7XG4gIH1cblxuICAvLyBFdmVudHNcbiAgc2VsZWN0RWwuYWRkRXZlbnRMaXN0ZW5lcignY2hhbmdlJywgKCkgPT4gbG9hZFRva2VuaXplcihzZWxlY3RFbC52YWx1ZSkpO1xuICBidG4uYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCB0b2tlbml6ZSk7XG4gIGlucHV0RWwuYWRkRXZlbnRMaXN0ZW5lcigna2V5ZG93bicsIChlKSA9PiB7XG4gICAgaWYgKGUua2V5ID09PSAnRW50ZXInICYmIChlLmN0cmxLZXkgfHwgZS5tZXRhS2V5KSkgdG9rZW5pemUoKTtcbiAgfSk7XG5cbiAgLy8gSW5pdGlhbCBsb2FkXG4gIGF3YWl0IGxvYWRUb2tlbml6ZXIoc2VsZWN0RWwudmFsdWUpO1xuPC9zY3JpcHQ+XG48L2JvZHk+XG48L2h0bWw+XG4ifV0sImlzRXJyb3IiOmZhbHNlLCJ0aW1lc3RhbXAiOjE3NzEwMzY4NDY4Nzd9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiIzNDhmNThhMCIsInBhcmVudElkIjoiZTQxMjdkZDUiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjQwOjU5LjM3NVoiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidG9vbENhbGwiLCJpZCI6InRvb2x1XzAxQnlnWFh0WnpBMmMxVE1QZ05LTGJjaCIsIm5hbWUiOiJlZGl0IiwiYXJndW1lbnRzIjp7InBhdGgiOiIvaG9tZS93ci9naC90cnlfd2ViX2dwdS9pbmRleC5odG1sIiwib2xkVGV4dCI6IiAgICA6cm9vdCB7XG4gICAgICAtLWJnOiAjMGQwZjE0O1xuICAgICAgLS1zdXJmYWNlOiAjMTMxNTFjO1xuICAgICAgLS1zdXJmYWNlLTI6ICMxOTFjMjY7XG4gICAgICAtLWJvcmRlcjogIzIzMjczYTtcbiAgICAgIC0tYm9yZGVyLWhvdmVyOiAjMzQzYTU0O1xuICAgICAgLS10ZXh0OiAjZTZlZGYzO1xuICAgICAgLS10ZXh0LW11dGVkOiAjN2Q4NTkwO1xuICAgICAgLS10ZXh0LWRpbTogIzQ4NGY1ODtcbiAgICAgIC0tYWNjZW50OiAjYmM4Y2ZmO1xuICAgICAgLS1hY2NlbnQtZGltOiByZ2JhKDE4OCwgMTQwLCAyNTUsIDAuMTIpO1xuICAgICAgLS1hY2NlbnQtMjogI2QyYThmZjtcbiAgICAgIC0tbW9ubzogJ0lCTSBQbGV4IE1vbm8nLCAnU0YgTW9ubycsICdGaXJhIENvZGUnLCBtb25vc3BhY2U7XG4gICAgICAtLXNhbnM6ICdJbnRlcicsIC1hcHBsZS1zeXN0ZW0sIEJsaW5rTWFjU3lzdGVtRm9udCwgc2Fucy1zZXJpZjtcbiAgICAgIC0tcmFkaXVzOiA4cHg7XG4gICAgfSIsIm5ld1RleHQiOiIgICAgOnJvb3QsXG4gICAgW2RhdGEtdGhlbWU9XCJkYXJrXCJdIHtcbiAgICAgIC0tYmc6ICMwZDBmMTQ7XG4gICAgICAtLXN1cmZhY2U6ICMxMzE1MWM7XG4gICAgICAtLXN1cmZhY2UtMjogIzE5MWMyNjtcbiAgICAgIC0tYm9yZGVyOiAjMjMyNzNhO1xuICAgICAgLS1ib3JkZXItaG92ZXI6ICMzNDNhNTQ7XG4gICAgICAtLXRleHQ6ICNlNmVkZjM7XG4gICAgICAtLXRleHQtbXV0ZWQ6ICM3ZDg1OTA7XG4gICAgICAtLXRleHQtZGltOiAjNDg0ZjU4O1xuICAgICAgLS1hY2NlbnQ6ICNiYzhjZmY7XG4gICAgICAtLWFjY2VudC1kaW06IHJnYmEoMTg4LCAxNDAsIDI1NSwgMC4xMik7XG4gICAgICAtLWFjY2VudC0yOiAjZDJhOGZmO1xuICAgICAgLS1idG4tdGV4dDogIzBkMGYxNDtcbiAgICAgIC0tdG9vbHRpcC1iZzogIzJkMzMzYjtcbiAgICAgIC0tdG9vbHRpcC10ZXh0OiAjZTZlZGYzO1xuICAgICAgLS1oMS1mcm9tOiAjZmZmO1xuICAgICAgLS1ub2lzZS1vcGFjaXR5OiAwLjAzO1xuICAgICAgLS1tb25vOiAnSUJNIFBsZXggTW9ubycsICdTRiBNb25vJywgJ0ZpcmEgQ29kZScsIG1vbm9zcGFjZTtcbiAgICAgIC0tc2FuczogJ0ludGVyJywgLWFwcGxlLXN5c3RlbSwgQmxpbmtNYWNTeXN0ZW1Gb250LCBzYW5zLXNlcmlmO1xuICAgICAgLS1yYWRpdXM6IDhweDtcbiAgICB9XG5cbiAgICBbZGF0YS10aGVtZT1cImxpZ2h0XCJdIHtcbiAgICAgIC0tYmc6ICNmOGY5ZmI7XG4gICAgICAtLXN1cmZhY2U6ICNmZmZmZmY7XG4gICAgICAtLXN1cmZhY2UtMjogI2VlZjBmNDtcbiAgICAgIC0tYm9yZGVyOiAjZDFkNWRlO1xuICAgICAgLS1ib3JkZXItaG92ZXI6ICNiMGI4Yzk7XG4gICAgICAtLXRleHQ6ICMxYTFlMmM7XG4gICAgICAtLXRleHQtbXV0ZWQ6ICM1YzYzNzA7XG4gICAgICAtLXRleHQtZGltOiAjOWNhM2FmO1xuICAgICAgLS1hY2NlbnQ6ICM3YzNhZWQ7XG4gICAgICAtLWFjY2VudC1kaW06IHJnYmEoMTI0LCA1OCwgMjM3LCAwLjEwKTtcbiAgICAgIC0tYWNjZW50LTI6ICM2ZDI4ZDk7XG4gICAgICAtLWJ0bi10ZXh0OiAjZmZmZmZmO1xuICAgICAgLS10b29sdGlwLWJnOiAjMWUyMDMwO1xuICAgICAgLS10b29sdGlwLXRleHQ6ICNlNmVkZjM7XG4gICAgICAtLWgxLWZyb206ICMxYTFlMmM7XG4gICAgICAtLW5vaXNlLW9wYWNpdHk6IDAuMDE1O1xuICAgIH0ifX1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjEsIm91dHB1dCI6ODI2LCJjYWNoZVJlYWQiOjE3NjE5LCJjYWNoZVdyaXRlIjo1MTE1LCJ0b3RhbFRva2VucyI6MjM1NjEsImNvc3QiOnsiaW5wdXQiOjAuMDAwMDA1LCJvdXRwdXQiOjAuMDIwNjUsImNhY2hlUmVhZCI6MC4wMDg4MDk1LCJjYWNoZVdyaXRlIjowLjAzMTk2ODc1MDAwMDAwMDAwNCwidG90YWwiOjAuMDYxNDMzMjUwMDAwMDAwMDF9fSwic3RvcFJlYXNvbiI6InRvb2xVc2UiLCJ0aW1lc3RhbXAiOjE3NzEwMzY4NDY4OTJ9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiJkNDRiMGRmYyIsInBhcmVudElkIjoiMzQ4ZjU4YTAiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjQwOjU5LjM5OFoiLCJtZXNzYWdlIjp7InJvbGUiOiJ0b29sUmVzdWx0IiwidG9vbENhbGxJZCI6InRvb2x1XzAxQnlnWFh0WnpBMmMxVE1QZ05LTGJjaCIsInRvb2xOYW1lIjoiZWRpdCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6IlN1Y2Nlc3NmdWxseSByZXBsYWNlZCB0ZXh0IGluIC9ob21lL3dyL2doL3RyeV93ZWJfZ3B1L2luZGV4Lmh0bWwuIn1dLCJkZXRhaWxzIjp7ImRpZmYiOiIgICAgIC4uLlxuICAgNiAgIDx0aXRsZT5Ub2tlbml6ZXIgUGxheWdyb3VuZDwvdGl0bGU+XG4gICA3ICAgPHN0eWxlPlxuICAgOCAgICAgQGltcG9ydCB1cmwoJ2h0dHBzOi8vZm9udHMuZ29vZ2xlYXBpcy5jb20vY3NzMj9mYW1pbHk9SW50ZXI6d2dodEA0MDA7NTAwOzYwMDs3MDAmZmFtaWx5PUlCTStQbGV4K01vbm86d2dodEA0MDA7NTAwJmRpc3BsYXk9c3dhcCcpO1xuICAgOSBcbi0gMTAgICAgIDpyb290IHtcbisgMTAgICAgIDpyb290LFxuKyAxMSAgICAgW2RhdGEtdGhlbWU9XCJkYXJrXCJdIHtcbiAgMTEgICAgICAgLS1iZzogIzBkMGYxNDtcbiAgMTIgICAgICAgLS1zdXJmYWNlOiAjMTMxNTFjO1xuICAxMyAgICAgICAtLXN1cmZhY2UtMjogIzE5MWMyNjtcbiAgMTQgICAgICAgLS1ib3JkZXI6ICMyMzI3M2E7XG4gIDE1ICAgICAgIC0tYm9yZGVyLWhvdmVyOiAjMzQzYTU0O1xuICAxNiAgICAgICAtLXRleHQ6ICNlNmVkZjM7XG4gIDE3ICAgICAgIC0tdGV4dC1tdXRlZDogIzdkODU5MDtcbiAgMTggICAgICAgLS10ZXh0LWRpbTogIzQ4NGY1ODtcbiAgMTkgICAgICAgLS1hY2NlbnQ6ICNiYzhjZmY7XG4gIDIwICAgICAgIC0tYWNjZW50LWRpbTogcmdiYSgxODgsIDE0MCwgMjU1LCAwLjEyKTtcbiAgMjEgICAgICAgLS1hY2NlbnQtMjogI2QyYThmZjtcbisgMjMgICAgICAgLS1idG4tdGV4dDogIzBkMGYxNDtcbisgMjQgICAgICAgLS10b29sdGlwLWJnOiAjMmQzMzNiO1xuKyAyNSAgICAgICAtLXRvb2x0aXAtdGV4dDogI2U2ZWRmMztcbisgMjYgICAgICAgLS1oMS1mcm9tOiAjZmZmO1xuKyAyNyAgICAgICAtLW5vaXNlLW9wYWNpdHk6IDAuMDM7XG4gIDIyICAgICAgIC0tbW9ubzogJ0lCTSBQbGV4IE1vbm8nLCAnU0YgTW9ubycsICdGaXJhIENvZGUnLCBtb25vc3BhY2U7XG4gIDIzICAgICAgIC0tc2FuczogJ0ludGVyJywgLWFwcGxlLXN5c3RlbSwgQmxpbmtNYWNTeXN0ZW1Gb250LCBzYW5zLXNlcmlmO1xuICAyNCAgICAgICAtLXJhZGl1czogOHB4O1xuICAyNSAgICAgfVxuICAyNiBcbisgMzMgICAgIFtkYXRhLXRoZW1lPVwibGlnaHRcIl0ge1xuKyAzNCAgICAgICAtLWJnOiAjZjhmOWZiO1xuKyAzNSAgICAgICAtLXN1cmZhY2U6ICNmZmZmZmY7XG4rIDM2ICAgICAgIC0tc3VyZmFjZS0yOiAjZWVmMGY0O1xuKyAzNyAgICAgICAtLWJvcmRlcjogI2QxZDVkZTtcbisgMzggICAgICAgLS1ib3JkZXItaG92ZXI6ICNiMGI4Yzk7XG4rIDM5ICAgICAgIC0tdGV4dDogIzFhMWUyYztcbisgNDAgICAgICAgLS10ZXh0LW11dGVkOiAjNWM2MzcwO1xuKyA0MSAgICAgICAtLXRleHQtZGltOiAjOWNhM2FmO1xuKyA0MiAgICAgICAtLWFjY2VudDogIzdjM2FlZDtcbisgNDMgICAgICAgLS1hY2NlbnQtZGltOiByZ2JhKDEyNCwgNTgsIDIzNywgMC4xMCk7XG4rIDQ0ICAgICAgIC0tYWNjZW50LTI6ICM2ZDI4ZDk7XG4rIDQ1ICAgICAgIC0tYnRuLXRleHQ6ICNmZmZmZmY7XG4rIDQ2ICAgICAgIC0tdG9vbHRpcC1iZzogIzFlMjAzMDtcbisgNDcgICAgICAgLS10b29sdGlwLXRleHQ6ICNlNmVkZjM7XG4rIDQ4ICAgICAgIC0taDEtZnJvbTogIzFhMWUyYztcbisgNDkgICAgICAgLS1ub2lzZS1vcGFjaXR5OiAwLjAxNTtcbisgNTAgICAgIH1cbisgNTEgXG4gIDI3ICAgICAqIHsgYm94LXNpemluZzogYm9yZGVyLWJveDsgbWFyZ2luOiAwOyBwYWRkaW5nOiAwOyB9XG4gIDI4IFxuICAyOSAgICAgYm9keSB7XG4gIDMwICAgICAgIGZvbnQtZmFtaWx5OiB2YXIoLS1zYW5zKTtcbiAgICAgLi4uIiwiZmlyc3RDaGFuZ2VkTGluZSI6MTB9LCJpc0Vycm9yIjpmYWxzZSwidGltZXN0YW1wIjoxNzcxMDM2ODU5Mzk4fX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiYmRhNjFiNTUiLCJwYXJlbnRJZCI6ImQ0NGIwZGZjIiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjo0MTowNS4xODNaIiwibWVzc2FnZSI6eyJyb2xlIjoiYXNzaXN0YW50IiwiY29udGVudCI6W3sidHlwZSI6InRvb2xDYWxsIiwiaWQiOiJ0b29sdV8wMVJ6NTl4TEFkQXV3UFJjYUdYcDFDMXEiLCJuYW1lIjoiZWRpdCIsImFyZ3VtZW50cyI6eyJwYXRoIjoiL2hvbWUvd3IvZ2gvdHJ5X3dlYl9ncHUvaW5kZXguaHRtbCIsIm9sZFRleHQiOiIgICAgICBiYWNrZ3JvdW5kOiB1cmwoXCJkYXRhOmltYWdlL3N2Zyt4bWwsJTNDc3ZnIHZpZXdCb3g9JzAgMCAyNTYgMjU2JyB4bWxucz0naHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmcnJTNFJTNDZmlsdGVyIGlkPSduJyUzRSUzQ2ZlVHVyYnVsZW5jZSB0eXBlPSdmcmFjdGFsTm9pc2UnIGJhc2VGcmVxdWVuY3k9JzAuOScgbnVtT2N0YXZlcz0nNCcgc3RpdGNoVGlsZXM9J3N0aXRjaCcvJTNFJTNDL2ZpbHRlciUzRSUzQ3JlY3Qgd2lkdGg9JzEwMCUyNScgaGVpZ2h0PScxMDAlMjUnIGZpbHRlcj0ndXJsKCUyM24pJyBvcGFjaXR5PScwLjAzJy8lM0UlM0Mvc3ZnJTNFXCIpOyIsIm5ld1RleHQiOiIgICAgICBiYWNrZ3JvdW5kOiB1cmwoXCJkYXRhOmltYWdlL3N2Zyt4bWwsJTNDc3ZnIHZpZXdCb3g9JzAgMCAyNTYgMjU2JyB4bWxucz0naHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmcnJTNFJTNDZmlsdGVyIGlkPSduJyUzRSUzQ2ZlVHVyYnVsZW5jZSB0eXBlPSdmcmFjdGFsTm9pc2UnIGJhc2VGcmVxdWVuY3k9JzAuOScgbnVtT2N0YXZlcz0nNCcgc3RpdGNoVGlsZXM9J3N0aXRjaCcvJTNFJTNDL2ZpbHRlciUzRSUzQ3JlY3Qgd2lkdGg9JzEwMCUyNScgaGVpZ2h0PScxMDAlMjUnIGZpbHRlcj0ndXJsKCUyM24pJyBvcGFjaXR5PScxJy8lM0UlM0Mvc3ZnJTNFXCIpO1xuICAgICAgb3BhY2l0eTogdmFyKC0tbm9pc2Utb3BhY2l0eSk7In19XSwiYXBpIjoiYW50aHJvcGljLW1lc3NhZ2VzIiwicHJvdmlkZXIiOiJhbnRocm9waWMiLCJtb2RlbCI6ImNsYXVkZS1vcHVzLTQtNiIsInVzYWdlIjp7ImlucHV0IjoxLCJvdXRwdXQiOjQxOCwiY2FjaGVSZWFkIjoyMjczNCwiY2FjaGVXcml0ZSI6ODU5LCJ0b3RhbFRva2VucyI6MjQwMTIsImNvc3QiOnsiaW5wdXQiOjAuMDAwMDA1LCJvdXRwdXQiOjAuMDEwNDUwMDAwMDAwMDAwMDAxLCJjYWNoZVJlYWQiOjAuMDExMzY2OTk5OTk5OTk5OTk5LCJjYWNoZVdyaXRlIjowLjAwNTM2ODc1LCJ0b3RhbCI6MC4wMjcxOTA3NX19LCJzdG9wUmVhc29uIjoidG9vbFVzZSIsInRpbWVzdGFtcCI6MTc3MTAzNjg1OTM5OH19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6ImU4NTY3MDc1IiwicGFyZW50SWQiOiJiZGE2MWI1NSIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDE6MDUuMjAwWiIsIm1lc3NhZ2UiOnsicm9sZSI6InRvb2xSZXN1bHQiLCJ0b29sQ2FsbElkIjoidG9vbHVfMDFSejU5eExBZEF1d1BSY2FHWHAxQzFxIiwidG9vbE5hbWUiOiJlZGl0IiwiY29udGVudCI6W3sidHlwZSI6InRleHQiLCJ0ZXh0IjoiU3VjY2Vzc2Z1bGx5IHJlcGxhY2VkIHRleHQgaW4gL2hvbWUvd3IvZ2gvdHJ5X3dlYl9ncHUvaW5kZXguaHRtbC4ifV0sImRldGFpbHMiOnsiZGlmZiI6IiAgICAgLi4uXG4gIDYzICAgICBib2R5OjpiZWZvcmUge1xuICA2NCAgICAgICBjb250ZW50OiAnJztcbiAgNjUgICAgICAgcG9zaXRpb246IGZpeGVkO1xuICA2NiAgICAgICBpbnNldDogMDtcbi0gNjcgICAgICAgYmFja2dyb3VuZDogdXJsKFwiZGF0YTppbWFnZS9zdmcreG1sLCUzQ3N2ZyB2aWV3Qm94PScwIDAgMjU2IDI1NicgeG1sbnM9J2h0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnJyUzRSUzQ2ZpbHRlciBpZD0nbiclM0UlM0NmZVR1cmJ1bGVuY2UgdHlwZT0nZnJhY3RhbE5vaXNlJyBiYXNlRnJlcXVlbmN5PScwLjknIG51bU9jdGF2ZXM9JzQnIHN0aXRjaFRpbGVzPSdzdGl0Y2gnLyUzRSUzQy9maWx0ZXIlM0UlM0NyZWN0IHdpZHRoPScxMDAlMjUnIGhlaWdodD0nMTAwJTI1JyBmaWx0ZXI9J3VybCglMjNuKScgb3BhY2l0eT0nMC4wMycvJTNFJTNDL3N2ZyUzRVwiKTtcbisgNjcgICAgICAgYmFja2dyb3VuZDogdXJsKFwiZGF0YTppbWFnZS9zdmcreG1sLCUzQ3N2ZyB2aWV3Qm94PScwIDAgMjU2IDI1NicgeG1sbnM9J2h0dHA6Ly93d3cudzMub3JnLzIwMDAvc3ZnJyUzRSUzQ2ZpbHRlciBpZD0nbiclM0UlM0NmZVR1cmJ1bGVuY2UgdHlwZT0nZnJhY3RhbE5vaXNlJyBiYXNlRnJlcXVlbmN5PScwLjknIG51bU9jdGF2ZXM9JzQnIHN0aXRjaFRpbGVzPSdzdGl0Y2gnLyUzRSUzQy9maWx0ZXIlM0UlM0NyZWN0IHdpZHRoPScxMDAlMjUnIGhlaWdodD0nMTAwJTI1JyBmaWx0ZXI9J3VybCglMjNuKScgb3BhY2l0eT0nMScvJTNFJTNDL3N2ZyUzRVwiKTtcbisgNjggICAgICAgb3BhY2l0eTogdmFyKC0tbm9pc2Utb3BhY2l0eSk7XG4gIDY4ICAgICAgIHBvaW50ZXItZXZlbnRzOiBub25lO1xuICA2OSAgICAgICB6LWluZGV4OiAwO1xuICA3MCAgICAgfVxuICA3MSBcbiAgICAgLi4uIiwiZmlyc3RDaGFuZ2VkTGluZSI6Njd9LCJpc0Vycm9yIjpmYWxzZSwidGltZXN0YW1wIjoxNzcxMDM2ODY1MTk5fX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiZjllMmEyMGUiLCJwYXJlbnRJZCI6ImU4NTY3MDc1IiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjo0MTowOC4xNzZaIiwibWVzc2FnZSI6eyJyb2xlIjoiYXNzaXN0YW50IiwiY29udGVudCI6W3sidHlwZSI6InRvb2xDYWxsIiwiaWQiOiJ0b29sdV8wMUNqSmlWdEFzYWZKMlBzUDNNbkQ0U3AiLCJuYW1lIjoiZWRpdCIsImFyZ3VtZW50cyI6eyJwYXRoIjoiL2hvbWUvd3IvZ2gvdHJ5X3dlYl9ncHUvaW5kZXguaHRtbCIsIm9sZFRleHQiOiIgICAgICBiYWNrZ3JvdW5kOiBsaW5lYXItZ3JhZGllbnQoMTM1ZGVnLCAjZmZmIDAlLCB2YXIoLS1hY2NlbnQtMikgMTAwJSk7IiwibmV3VGV4dCI6IiAgICAgIGJhY2tncm91bmQ6IGxpbmVhci1ncmFkaWVudCgxMzVkZWcsIHZhcigtLWgxLWZyb20pIDAlLCB2YXIoLS1hY2NlbnQtMikgMTAwJSk7In19XSwiYXBpIjoiYW50aHJvcGljLW1lc3NhZ2VzIiwicHJvdmlkZXIiOiJhbnRocm9waWMiLCJtb2RlbCI6ImNsYXVkZS1vcHVzLTQtNiIsInVzYWdlIjp7ImlucHV0IjoxLCJvdXRwdXQiOjE1NywiY2FjaGVSZWFkIjoyMzU5MywiY2FjaGVXcml0ZSI6NDUxLCJ0b3RhbFRva2VucyI6MjQyMDIsImNvc3QiOnsiaW5wdXQiOjAuMDAwMDA1LCJvdXRwdXQiOjAuMDAzOTI1MDAwMDAwMDAwMDAwNSwiY2FjaGVSZWFkIjowLjAxMTc5NjUsImNhY2hlV3JpdGUiOjAuMDAyODE4NzUsInRvdGFsIjowLjAxODU0NTI1fX0sInN0b3BSZWFzb24iOiJ0b29sVXNlIiwidGltZXN0YW1wIjoxNzcxMDM2ODY1MjAwfX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiMjc1MzIwODgiLCJwYXJlbnRJZCI6ImY5ZTJhMjBlIiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjo0MTowOC4xOTlaIiwibWVzc2FnZSI6eyJyb2xlIjoidG9vbFJlc3VsdCIsInRvb2xDYWxsSWQiOiJ0b29sdV8wMUNqSmlWdEFzYWZKMlBzUDNNbkQ0U3AiLCJ0b29sTmFtZSI6ImVkaXQiLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJTdWNjZXNzZnVsbHkgcmVwbGFjZWQgdGV4dCBpbiAvaG9tZS93ci9naC90cnlfd2ViX2dwdS9pbmRleC5odG1sLiJ9XSwiZGV0YWlscyI6eyJkaWZmIjoiICAgICAuLi5cbiAxMDQgICAgICAgZm9udC1zaXplOiAyLjRyZW07XG4gMTA1ICAgICAgIGZvbnQtd2VpZ2h0OiA3MDA7XG4gMTA2ICAgICAgIGxldHRlci1zcGFjaW5nOiAtMC4wM2VtO1xuIDEwNyAgICAgICBsaW5lLWhlaWdodDogMS4xNTtcbi0xMDggICAgICAgYmFja2dyb3VuZDogbGluZWFyLWdyYWRpZW50KDEzNWRlZywgI2ZmZiAwJSwgdmFyKC0tYWNjZW50LTIpIDEwMCUpO1xuKzEwOCAgICAgICBiYWNrZ3JvdW5kOiBsaW5lYXItZ3JhZGllbnQoMTM1ZGVnLCB2YXIoLS1oMS1mcm9tKSAwJSwgdmFyKC0tYWNjZW50LTIpIDEwMCUpO1xuIDEwOSAgICAgICAtd2Via2l0LWJhY2tncm91bmQtY2xpcDogdGV4dDtcbiAxMTAgICAgICAgLXdlYmtpdC10ZXh0LWZpbGwtY29sb3I6IHRyYW5zcGFyZW50O1xuIDExMSAgICAgICBiYWNrZ3JvdW5kLWNsaXA6IHRleHQ7XG4gMTEyICAgICB9XG4gICAgIC4uLiIsImZpcnN0Q2hhbmdlZExpbmUiOjEwOH0sImlzRXJyb3IiOmZhbHNlLCJ0aW1lc3RhbXAiOjE3NzEwMzY4NjgxOTh9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiJiMjU3MGE4MyIsInBhcmVudElkIjoiMjc1MzIwODgiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjQxOjExLjcyMloiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidG9vbENhbGwiLCJpZCI6InRvb2x1XzAxTGZXS3BDQ0tIbzc4NTRISzJicWlBdCIsIm5hbWUiOiJlZGl0IiwiYXJndW1lbnRzIjp7InBhdGgiOiIvaG9tZS93ci9naC90cnlfd2ViX2dwdS9pbmRleC5odG1sIiwib2xkVGV4dCI6IiAgICAgIGJhY2tncm91bmQ6IHZhcigtLWFjY2VudCk7XG4gICAgICBjb2xvcjogIzBkMGYxNDsiLCJuZXdUZXh0IjoiICAgICAgYmFja2dyb3VuZDogdmFyKC0tYWNjZW50KTtcbiAgICAgIGNvbG9yOiB2YXIoLS1idG4tdGV4dCk7In19XSwiYXBpIjoiYW50aHJvcGljLW1lc3NhZ2VzIiwicHJvdmlkZXIiOiJhbnRocm9waWMiLCJtb2RlbCI6ImNsYXVkZS1vcHVzLTQtNiIsInVzYWdlIjp7ImlucHV0IjoxLCJvdXRwdXQiOjEzOCwiY2FjaGVSZWFkIjoyNDA0NCwiY2FjaGVXcml0ZSI6MTkwLCJ0b3RhbFRva2VucyI6MjQzNzMsImNvc3QiOnsiaW5wdXQiOjAuMDAwMDA1LCJvdXRwdXQiOjAuMDAzNDUwMDAwMDAwMDAwMDAwNCwiY2FjaGVSZWFkIjowLjAxMjAyMiwiY2FjaGVXcml0ZSI6MC4wMDExODc1LCJ0b3RhbCI6MC4wMTY2NjQ1fX0sInN0b3BSZWFzb24iOiJ0b29sVXNlIiwidGltZXN0YW1wIjoxNzcxMDM2ODY4MTk5fX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiNDAwYWU1MTIiLCJwYXJlbnRJZCI6ImIyNTcwYTgzIiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjo0MToxMS43NTRaIiwibWVzc2FnZSI6eyJyb2xlIjoidG9vbFJlc3VsdCIsInRvb2xDYWxsSWQiOiJ0b29sdV8wMUxmV0twQ0NLSG83ODU0SEsyYnFpQXQiLCJ0b29sTmFtZSI6ImVkaXQiLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJTdWNjZXNzZnVsbHkgcmVwbGFjZWQgdGV4dCBpbiAvaG9tZS93ci9naC90cnlfd2ViX2dwdS9pbmRleC5odG1sLiJ9XSwiZGV0YWlscyI6eyJkaWZmIjoiICAgICAuLi5cbiAxODIgICAgICAgYm9yZGVyLXJhZGl1czogdmFyKC0tcmFkaXVzKTtcbiAxODMgICAgICAgY3Vyc29yOiBwb2ludGVyO1xuIDE4NCAgICAgICB0cmFuc2l0aW9uOiBhbGwgMC4ycztcbiAxODUgICAgICAgYmFja2dyb3VuZDogdmFyKC0tYWNjZW50KTtcbi0xODYgICAgICAgY29sb3I6ICMwZDBmMTQ7XG4rMTg2ICAgICAgIGNvbG9yOiB2YXIoLS1idG4tdGV4dCk7XG4gMTg3ICAgICB9XG4gMTg4ICAgICBidXR0b246aG92ZXIgeyBmaWx0ZXI6IGJyaWdodG5lc3MoMS4xNSk7IHRyYW5zZm9ybTogdHJhbnNsYXRlWSgtMXB4KTsgfVxuIDE4OSAgICAgYnV0dG9uOmFjdGl2ZSB7IHRyYW5zZm9ybTogdHJhbnNsYXRlWSgwKTsgfVxuIDE5MCAgICAgYnV0dG9uOmRpc2FibGVkIHtcbiAgICAgLi4uIiwiZmlyc3RDaGFuZ2VkTGluZSI6MTg2fSwiaXNFcnJvciI6ZmFsc2UsInRpbWVzdGFtcCI6MTc3MTAzNjg3MTc1Mn19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6IjBiOGQ3NDRiIiwicGFyZW50SWQiOiI0MDBhZTUxMiIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDE6MTUuNTU4WiIsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0b29sQ2FsbCIsImlkIjoidG9vbHVfMDFQQlp1ZVg4SEpldnRlNXEyQzdtM3FoIiwibmFtZSI6ImVkaXQiLCJhcmd1bWVudHMiOnsicGF0aCI6Ii9ob21lL3dyL2doL3RyeV93ZWJfZ3B1L2luZGV4Lmh0bWwiLCJvbGRUZXh0IjoiICAgICAgYmFja2dyb3VuZDogIzJkMzMzYjtcbiAgICAgIGNvbG9yOiAjZTZlZGYzOyIsIm5ld1RleHQiOiIgICAgICBiYWNrZ3JvdW5kOiB2YXIoLS10b29sdGlwLWJnKTtcbiAgICAgIGNvbG9yOiB2YXIoLS10b29sdGlwLXRleHQpOyJ9fV0sImFwaSI6ImFudGhyb3BpYy1tZXNzYWdlcyIsInByb3ZpZGVyIjoiYW50aHJvcGljIiwibW9kZWwiOiJjbGF1ZGUtb3B1cy00LTYiLCJ1c2FnZSI6eyJpbnB1dCI6MSwib3V0cHV0IjoxNDEsImNhY2hlUmVhZCI6MjQyMzQsImNhY2hlV3JpdGUiOjE3MSwidG90YWxUb2tlbnMiOjI0NTQ3LCJjb3N0Ijp7ImlucHV0IjowLjAwMDAwNSwib3V0cHV0IjowLjAwMzUyNTAwMDAwMDAwMDAwMDQsImNhY2hlUmVhZCI6MC4wMTIxMTcsImNhY2hlV3JpdGUiOjAuMDAxMDY4NzUsInRvdGFsIjowLjAxNjcxNTc1fX0sInN0b3BSZWFzb24iOiJ0b29sVXNlIiwidGltZXN0YW1wIjoxNzcxMDM2ODcxNzU1fX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiNTliNjVkMGMiLCJwYXJlbnRJZCI6IjBiOGQ3NDRiIiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjo0MToxNS41NzBaIiwibWVzc2FnZSI6eyJyb2xlIjoidG9vbFJlc3VsdCIsInRvb2xDYWxsSWQiOiJ0b29sdV8wMVBCWnVlWDhISmV2dGU1cTJDN20zcWgiLCJ0b29sTmFtZSI6ImVkaXQiLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJTdWNjZXNzZnVsbHkgcmVwbGFjZWQgdGV4dCBpbiAvaG9tZS93ci9naC90cnlfd2ViX2dwdS9pbmRleC5odG1sLiJ9XSwiZGV0YWlscyI6eyJkaWZmIjoiICAgICAuLi5cbiAyNzkgICAgICAgcG9zaXRpb246IGFic29sdXRlO1xuIDI4MCAgICAgICBib3R0b206IGNhbGMoMTAwJSArIDRweCk7XG4gMjgxICAgICAgIGxlZnQ6IDUwJTtcbiAyODIgICAgICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVYKC01MCUpIHNjYWxlKDAuOSk7XG4tMjgzICAgICAgIGJhY2tncm91bmQ6ICMyZDMzM2I7XG4tMjg0ICAgICAgIGNvbG9yOiAjZTZlZGYzO1xuKzI4MyAgICAgICBiYWNrZ3JvdW5kOiB2YXIoLS10b29sdGlwLWJnKTtcbisyODQgICAgICAgY29sb3I6IHZhcigtLXRvb2x0aXAtdGV4dCk7XG4gMjg1ICAgICAgIHBhZGRpbmc6IDNweCA4cHg7XG4gMjg2ICAgICAgIGJvcmRlci1yYWRpdXM6IDVweDtcbiAyODcgICAgICAgZm9udC1zaXplOiAwLjdyZW07XG4gMjg4ICAgICAgIHdoaXRlLXNwYWNlOiBub3dyYXA7XG4gICAgIC4uLiIsImZpcnN0Q2hhbmdlZExpbmUiOjI4M30sImlzRXJyb3IiOmZhbHNlLCJ0aW1lc3RhbXAiOjE3NzEwMzY4NzU1NzB9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiIyOTM3NWNlNiIsInBhcmVudElkIjoiNTliNjVkMGMiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjQxOjI2LjMyMloiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJOb3cgYWRkIHRoZSB0aGVtZSB0b2dnbGUgYnV0dG9uIGFuZCB0aGUgbGlnaHQtbW9kZSB0b2tlbiBwYWxldHRlIGxvZ2ljLCBwbHVzIHRoZSB0b2dnbGUgQ1NTOiJ9LHsidHlwZSI6InRvb2xDYWxsIiwiaWQiOiJ0b29sdV8wMVRxYzQzYzRxdG1Td1BEdGdqWU5yYWIiLCJuYW1lIjoiZWRpdCIsImFyZ3VtZW50cyI6eyJwYXRoIjoiL2hvbWUvd3IvZ2gvdHJ5X3dlYl9ncHUvaW5kZXguaHRtbCIsIm9sZFRleHQiOiIgICAgLyogc2Nyb2xsYmFyICovXG4gICAgOjotd2Via2l0LXNjcm9sbGJhciB7IHdpZHRoOiA2cHg7IH1cbiAgICA6Oi13ZWJraXQtc2Nyb2xsYmFyLXRyYWNrIHsgYmFja2dyb3VuZDogdHJhbnNwYXJlbnQ7IH1cbiAgICA6Oi13ZWJraXQtc2Nyb2xsYmFyLXRodW1iIHsgYmFja2dyb3VuZDogdmFyKC0tYm9yZGVyKTsgYm9yZGVyLXJhZGl1czogM3B4OyB9XG4gIDwvc3R5bGU+IiwibmV3VGV4dCI6IiAgICAvKiDigJQgVGhlbWUgdG9nZ2xlIOKAlCAqL1xuICAgIC50aGVtZS10b2dnbGUge1xuICAgICAgcG9zaXRpb246IGZpeGVkO1xuICAgICAgdG9wOiAyMHB4O1xuICAgICAgcmlnaHQ6IDIwcHg7XG4gICAgICB6LWluZGV4OiAxMDA7XG4gICAgICB3aWR0aDogNDBweDsgaGVpZ2h0OiA0MHB4O1xuICAgICAgYm9yZGVyOiAxcHggc29saWQgdmFyKC0tYm9yZGVyKTtcbiAgICAgIGJvcmRlci1yYWRpdXM6IDUwJTtcbiAgICAgIGJhY2tncm91bmQ6IHZhcigtLXN1cmZhY2UpO1xuICAgICAgY29sb3I6IHZhcigtLXRleHQtbXV0ZWQpO1xuICAgICAgY3Vyc29yOiBwb2ludGVyO1xuICAgICAgZGlzcGxheTogZmxleDtcbiAgICAgIGFsaWduLWl0ZW1zOiBjZW50ZXI7XG4gICAgICBqdXN0aWZ5LWNvbnRlbnQ6IGNlbnRlcjtcbiAgICAgIHRyYW5zaXRpb246IGJvcmRlci1jb2xvciAwLjJzLCBiYWNrZ3JvdW5kIDAuMnMsIHRyYW5zZm9ybSAwLjJzO1xuICAgICAgcGFkZGluZzogMDtcbiAgICAgIGZvbnQtc2l6ZTogMDtcbiAgICB9XG4gICAgLnRoZW1lLXRvZ2dsZTpob3ZlciB7XG4gICAgICBib3JkZXItY29sb3I6IHZhcigtLWJvcmRlci1ob3Zlcik7XG4gICAgICB0cmFuc2Zvcm06IHNjYWxlKDEuMDgpO1xuICAgIH1cbiAgICAudGhlbWUtdG9nZ2xlIHN2ZyB7XG4gICAgICB3aWR0aDogMThweDsgaGVpZ2h0OiAxOHB4O1xuICAgICAgZmlsbDogbm9uZTtcbiAgICAgIHN0cm9rZTogY3VycmVudENvbG9yO1xuICAgICAgc3Ryb2tlLXdpZHRoOiAyO1xuICAgICAgc3Ryb2tlLWxpbmVjYXA6IHJvdW5kO1xuICAgICAgc3Ryb2tlLWxpbmVqb2luOiByb3VuZDtcbiAgICB9XG4gICAgLnRoZW1lLXRvZ2dsZSAuaWNvbi1zdW4gIHsgZGlzcGxheTogbm9uZTsgfVxuICAgIC50aGVtZS10b2dnbGUgLmljb24tbW9vbiB7IGRpc3BsYXk6IGJsb2NrOyB9XG4gICAgW2RhdGEtdGhlbWU9XCJsaWdodFwiXSAudGhlbWUtdG9nZ2xlIC5pY29uLXN1biAgeyBkaXNwbGF5OiBibG9jazsgfVxuICAgIFtkYXRhLXRoZW1lPVwibGlnaHRcIl0gLnRoZW1lLXRvZ2dsZSAuaWNvbi1tb29uIHsgZGlzcGxheTogbm9uZTsgfVxuXG4gICAgLyogc21vb3RoIHRyYW5zaXRpb24gb24gdGhlbWUgY2hhbmdlICovXG4gICAgYm9keSwgLnBhZ2UsIHNlbGVjdCwgdGV4dGFyZWEsIC5iYWRnZSwgLnN0YXRzLWJhciwgLnRva2VuLWdyaWQsIC5pZHMtYm94LFxuICAgIC50aGVtZS10b2dnbGUsIGJ1dHRvbiwgLnRvazo6YWZ0ZXIge1xuICAgICAgdHJhbnNpdGlvbjogYmFja2dyb3VuZCAwLjNzLCBjb2xvciAwLjNzLCBib3JkZXItY29sb3IgMC4zcztcbiAgICB9XG5cbiAgICAvKiBzY3JvbGxiYXIgKi9cbiAgICA6Oi13ZWJraXQtc2Nyb2xsYmFyIHsgd2lkdGg6IDZweDsgfVxuICAgIDo6LXdlYmtpdC1zY3JvbGxiYXItdHJhY2sgeyBiYWNrZ3JvdW5kOiB0cmFuc3BhcmVudDsgfVxuICAgIDo6LXdlYmtpdC1zY3JvbGxiYXItdGh1bWIgeyBiYWNrZ3JvdW5kOiB2YXIoLS1ib3JkZXIpOyBib3JkZXItcmFkaXVzOiAzcHg7IH1cbiAgPC9zdHlsZT4ifX1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjEsIm91dHB1dCI6Njk5LCJjYWNoZVJlYWQiOjI0NDA1LCJjYWNoZVdyaXRlIjoxNzQsInRvdGFsVG9rZW5zIjoyNTI3OSwiY29zdCI6eyJpbnB1dCI6MC4wMDAwMDUsIm91dHB1dCI6MC4wMTc0NzUsImNhY2hlUmVhZCI6MC4wMTIyMDI1LCJjYWNoZVdyaXRlIjowLjAwMTA4NzUwMDAwMDAwMDAwMDEsInRvdGFsIjowLjAzMDc3MDAwMDAwMDAwMDAwMn19LCJzdG9wUmVhc29uIjoidG9vbFVzZSIsInRpbWVzdGFtcCI6MTc3MTAzNjg3NTU3MH19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6ImI2N2M4N2ZhIiwicGFyZW50SWQiOiIyOTM3NWNlNiIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDE6MjYuMzM4WiIsIm1lc3NhZ2UiOnsicm9sZSI6InRvb2xSZXN1bHQiLCJ0b29sQ2FsbElkIjoidG9vbHVfMDFUcWM0M2M0cXRtU3dQRHRnallOcmFiIiwidG9vbE5hbWUiOiJlZGl0IiwiY29udGVudCI6W3sidHlwZSI6InRleHQiLCJ0ZXh0IjoiU3VjY2Vzc2Z1bGx5IHJlcGxhY2VkIHRleHQgaW4gL2hvbWUvd3IvZ2gvdHJ5X3dlYl9ncHUvaW5kZXguaHRtbC4ifV0sImRldGFpbHMiOnsiZGlmZiI6IiAgICAgLi4uXG4gMzE1ICAgICAgIGJvcmRlci10b3A6IDFweCBzb2xpZCB2YXIoLS1ib3JkZXIpO1xuIDMxNiAgICAgICBtYXJnaW46IDI4cHggMDtcbiAzMTcgICAgIH1cbiAzMTggXG4rMzE5ICAgICAvKiDigJQgVGhlbWUgdG9nZ2xlIOKAlCAqL1xuKzMyMCAgICAgLnRoZW1lLXRvZ2dsZSB7XG4rMzIxICAgICAgIHBvc2l0aW9uOiBmaXhlZDtcbiszMjIgICAgICAgdG9wOiAyMHB4O1xuKzMyMyAgICAgICByaWdodDogMjBweDtcbiszMjQgICAgICAgei1pbmRleDogMTAwO1xuKzMyNSAgICAgICB3aWR0aDogNDBweDsgaGVpZ2h0OiA0MHB4O1xuKzMyNiAgICAgICBib3JkZXI6IDFweCBzb2xpZCB2YXIoLS1ib3JkZXIpO1xuKzMyNyAgICAgICBib3JkZXItcmFkaXVzOiA1MCU7XG4rMzI4ICAgICAgIGJhY2tncm91bmQ6IHZhcigtLXN1cmZhY2UpO1xuKzMyOSAgICAgICBjb2xvcjogdmFyKC0tdGV4dC1tdXRlZCk7XG4rMzMwICAgICAgIGN1cnNvcjogcG9pbnRlcjtcbiszMzEgICAgICAgZGlzcGxheTogZmxleDtcbiszMzIgICAgICAgYWxpZ24taXRlbXM6IGNlbnRlcjtcbiszMzMgICAgICAganVzdGlmeS1jb250ZW50OiBjZW50ZXI7XG4rMzM0ICAgICAgIHRyYW5zaXRpb246IGJvcmRlci1jb2xvciAwLjJzLCBiYWNrZ3JvdW5kIDAuMnMsIHRyYW5zZm9ybSAwLjJzO1xuKzMzNSAgICAgICBwYWRkaW5nOiAwO1xuKzMzNiAgICAgICBmb250LXNpemU6IDA7XG4rMzM3ICAgICB9XG4rMzM4ICAgICAudGhlbWUtdG9nZ2xlOmhvdmVyIHtcbiszMzkgICAgICAgYm9yZGVyLWNvbG9yOiB2YXIoLS1ib3JkZXItaG92ZXIpO1xuKzM0MCAgICAgICB0cmFuc2Zvcm06IHNjYWxlKDEuMDgpO1xuKzM0MSAgICAgfVxuKzM0MiAgICAgLnRoZW1lLXRvZ2dsZSBzdmcge1xuKzM0MyAgICAgICB3aWR0aDogMThweDsgaGVpZ2h0OiAxOHB4O1xuKzM0NCAgICAgICBmaWxsOiBub25lO1xuKzM0NSAgICAgICBzdHJva2U6IGN1cnJlbnRDb2xvcjtcbiszNDYgICAgICAgc3Ryb2tlLXdpZHRoOiAyO1xuKzM0NyAgICAgICBzdHJva2UtbGluZWNhcDogcm91bmQ7XG4rMzQ4ICAgICAgIHN0cm9rZS1saW5lam9pbjogcm91bmQ7XG4rMzQ5ICAgICB9XG4rMzUwICAgICAudGhlbWUtdG9nZ2xlIC5pY29uLXN1biAgeyBkaXNwbGF5OiBub25lOyB9XG4rMzUxICAgICAudGhlbWUtdG9nZ2xlIC5pY29uLW1vb24geyBkaXNwbGF5OiBibG9jazsgfVxuKzM1MiAgICAgW2RhdGEtdGhlbWU9XCJsaWdodFwiXSAudGhlbWUtdG9nZ2xlIC5pY29uLXN1biAgeyBkaXNwbGF5OiBibG9jazsgfVxuKzM1MyAgICAgW2RhdGEtdGhlbWU9XCJsaWdodFwiXSAudGhlbWUtdG9nZ2xlIC5pY29uLW1vb24geyBkaXNwbGF5OiBub25lOyB9XG4rMzU0IFxuKzM1NSAgICAgLyogc21vb3RoIHRyYW5zaXRpb24gb24gdGhlbWUgY2hhbmdlICovXG4rMzU2ICAgICBib2R5LCAucGFnZSwgc2VsZWN0LCB0ZXh0YXJlYSwgLmJhZGdlLCAuc3RhdHMtYmFyLCAudG9rZW4tZ3JpZCwgLmlkcy1ib3gsXG4rMzU3ICAgICAudGhlbWUtdG9nZ2xlLCBidXR0b24sIC50b2s6OmFmdGVyIHtcbiszNTggICAgICAgdHJhbnNpdGlvbjogYmFja2dyb3VuZCAwLjNzLCBjb2xvciAwLjNzLCBib3JkZXItY29sb3IgMC4zcztcbiszNTkgICAgIH1cbiszNjAgXG4gMzE5ICAgICAvKiBzY3JvbGxiYXIgKi9cbiAzMjAgICAgIDo6LXdlYmtpdC1zY3JvbGxiYXIgeyB3aWR0aDogNnB4OyB9XG4gMzIxICAgICA6Oi13ZWJraXQtc2Nyb2xsYmFyLXRyYWNrIHsgYmFja2dyb3VuZDogdHJhbnNwYXJlbnQ7IH1cbiAzMjIgICAgIDo6LXdlYmtpdC1zY3JvbGxiYXItdGh1bWIgeyBiYWNrZ3JvdW5kOiB2YXIoLS1ib3JkZXIpOyBib3JkZXItcmFkaXVzOiAzcHg7IH1cbiAgICAgLi4uIiwiZmlyc3RDaGFuZ2VkTGluZSI6MzE5fSwiaXNFcnJvciI6ZmFsc2UsInRpbWVzdGFtcCI6MTc3MTAzNjg4NjMzN319LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6IjdlYzJhNTY0IiwicGFyZW50SWQiOiJiNjdjODdmYSIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDE6MzQuNzA5WiIsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6Ik5vdyBhZGQgdGhlIHRvZ2dsZSBidXR0b24gdG8gdGhlIEhUTUw6In0seyJ0eXBlIjoidG9vbENhbGwiLCJpZCI6InRvb2x1XzAxMUJMMzNNTTdiU3BkSDRGSldYSER4MiIsIm5hbWUiOiJlZGl0IiwiYXJndW1lbnRzIjp7InBhdGgiOiIvaG9tZS93ci9naC90cnlfd2ViX2dwdS9pbmRleC5odG1sIiwib2xkVGV4dCI6IjxkaXYgY2xhc3M9XCJwYWdlXCI+XG4gIDxoZWFkZXI+IiwibmV3VGV4dCI6IjxidXR0b24gY2xhc3M9XCJ0aGVtZS10b2dnbGVcIiBpZD1cInRoZW1lLXRvZ2dsZVwiIGFyaWEtbGFiZWw9XCJUb2dnbGUgdGhlbWVcIj5cbiAgPCEtLSBNb29uIGljb24gKHNob3duIGluIGRhcmsgbW9kZSDihpIgY2xpY2sgdG8gZ28gbGlnaHQpIC0tPlxuICA8c3ZnIGNsYXNzPVwiaWNvbi1tb29uXCIgdmlld0JveD1cIjAgMCAyNCAyNFwiPjxwYXRoIGQ9XCJNMjEgMTIuNzlBOSA5IDAgMSAxIDExLjIxIDNhNyA3IDAgMCAwIDkuNzkgOS43OXpcIi8+PC9zdmc+XG4gIDwhLS0gU3VuIGljb24gKHNob3duIGluIGxpZ2h0IG1vZGUg4oaSIGNsaWNrIHRvIGdvIGRhcmspIC0tPlxuICA8c3ZnIGNsYXNzPVwiaWNvbi1zdW5cIiB2aWV3Qm94PVwiMCAwIDI0IDI0XCI+PGNpcmNsZSBjeD1cIjEyXCIgY3k9XCIxMlwiIHI9XCI1XCIvPjxsaW5lIHgxPVwiMTJcIiB5MT1cIjFcIiB4Mj1cIjEyXCIgeTI9XCIzXCIvPjxsaW5lIHgxPVwiMTJcIiB5MT1cIjIxXCIgeDI9XCIxMlwiIHkyPVwiMjNcIi8+PGxpbmUgeDE9XCI0LjIyXCIgeTE9XCI0LjIyXCIgeDI9XCI1LjY0XCIgeTI9XCI1LjY0XCIvPjxsaW5lIHgxPVwiMTguMzZcIiB5MT1cIjE4LjM2XCIgeDI9XCIxOS43OFwiIHkyPVwiMTkuNzhcIi8+PGxpbmUgeDE9XCIxXCIgeTE9XCIxMlwiIHgyPVwiM1wiIHkyPVwiMTJcIi8+PGxpbmUgeDE9XCIyMVwiIHkxPVwiMTJcIiB4Mj1cIjIzXCIgeTI9XCIxMlwiLz48bGluZSB4MT1cIjQuMjJcIiB5MT1cIjE5Ljc4XCIgeDI9XCI1LjY0XCIgeTI9XCIxOC4zNlwiLz48bGluZSB4MT1cIjE4LjM2XCIgeTE9XCI1LjY0XCIgeDI9XCIxOS43OFwiIHkyPVwiNC4yMlwiLz48L3N2Zz5cbjwvYnV0dG9uPlxuXG48ZGl2IGNsYXNzPVwicGFnZVwiPlxuICA8aGVhZGVyPiJ9fV0sImFwaSI6ImFudGhyb3BpYy1tZXNzYWdlcyIsInByb3ZpZGVyIjoiYW50aHJvcGljIiwibW9kZWwiOiJjbGF1ZGUtb3B1cy00LTYiLCJ1c2FnZSI6eyJpbnB1dCI6MSwib3V0cHV0Ijo1MDMsImNhY2hlUmVhZCI6MjQ1NzksImNhY2hlV3JpdGUiOjczMiwidG90YWxUb2tlbnMiOjI1ODE1LCJjb3N0Ijp7ImlucHV0IjowLjAwMDAwNSwib3V0cHV0IjowLjAxMjU3NTAwMDAwMDAwMDAwMSwiY2FjaGVSZWFkIjowLjAxMjI4OTUsImNhY2hlV3JpdGUiOjAuMDA0NTc1LCJ0b3RhbCI6MC4wMjk0NDQ1MDAwMDAwMDAwMDJ9fSwic3RvcFJlYXNvbiI6InRvb2xVc2UiLCJ0aW1lc3RhbXAiOjE3NzEwMzY4ODYzMzh9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiI3ZDU3NDAyMyIsInBhcmVudElkIjoiN2VjMmE1NjQiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjQxOjM0LjcyMFoiLCJtZXNzYWdlIjp7InJvbGUiOiJ0b29sUmVzdWx0IiwidG9vbENhbGxJZCI6InRvb2x1XzAxMUJMMzNNTTdiU3BkSDRGSldYSER4MiIsInRvb2xOYW1lIjoiZWRpdCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6IlN1Y2Nlc3NmdWxseSByZXBsYWNlZCB0ZXh0IGluIC9ob21lL3dyL2doL3RyeV93ZWJfZ3B1L2luZGV4Lmh0bWwuIn1dLCJkZXRhaWxzIjp7ImRpZmYiOiIgICAgIC4uLlxuIDM2NCAgICAgOjotd2Via2l0LXNjcm9sbGJhci10aHVtYiB7IGJhY2tncm91bmQ6IHZhcigtLWJvcmRlcik7IGJvcmRlci1yYWRpdXM6IDNweDsgfVxuIDM2NSAgIDwvc3R5bGU+XG4gMzY2IDwvaGVhZD5cbiAzNjcgPGJvZHk+XG4rMzY4IDxidXR0b24gY2xhc3M9XCJ0aGVtZS10b2dnbGVcIiBpZD1cInRoZW1lLXRvZ2dsZVwiIGFyaWEtbGFiZWw9XCJUb2dnbGUgdGhlbWVcIj5cbiszNjkgICA8IS0tIE1vb24gaWNvbiAoc2hvd24gaW4gZGFyayBtb2RlIOKGkiBjbGljayB0byBnbyBsaWdodCkgLS0+XG4rMzcwICAgPHN2ZyBjbGFzcz1cImljb24tbW9vblwiIHZpZXdCb3g9XCIwIDAgMjQgMjRcIj48cGF0aCBkPVwiTTIxIDEyLjc5QTkgOSAwIDEgMSAxMS4yMSAzYTcgNyAwIDAgMCA5Ljc5IDkuNzl6XCIvPjwvc3ZnPlxuKzM3MSAgIDwhLS0gU3VuIGljb24gKHNob3duIGluIGxpZ2h0IG1vZGUg4oaSIGNsaWNrIHRvIGdvIGRhcmspIC0tPlxuKzM3MiAgIDxzdmcgY2xhc3M9XCJpY29uLXN1blwiIHZpZXdCb3g9XCIwIDAgMjQgMjRcIj48Y2lyY2xlIGN4PVwiMTJcIiBjeT1cIjEyXCIgcj1cIjVcIi8+PGxpbmUgeDE9XCIxMlwiIHkxPVwiMVwiIHgyPVwiMTJcIiB5Mj1cIjNcIi8+PGxpbmUgeDE9XCIxMlwiIHkxPVwiMjFcIiB4Mj1cIjEyXCIgeTI9XCIyM1wiLz48bGluZSB4MT1cIjQuMjJcIiB5MT1cIjQuMjJcIiB4Mj1cIjUuNjRcIiB5Mj1cIjUuNjRcIi8+PGxpbmUgeDE9XCIxOC4zNlwiIHkxPVwiMTguMzZcIiB4Mj1cIjE5Ljc4XCIgeTI9XCIxOS43OFwiLz48bGluZSB4MT1cIjFcIiB5MT1cIjEyXCIgeDI9XCIzXCIgeTI9XCIxMlwiLz48bGluZSB4MT1cIjIxXCIgeTE9XCIxMlwiIHgyPVwiMjNcIiB5Mj1cIjEyXCIvPjxsaW5lIHgxPVwiNC4yMlwiIHkxPVwiMTkuNzhcIiB4Mj1cIjUuNjRcIiB5Mj1cIjE4LjM2XCIvPjxsaW5lIHgxPVwiMTguMzZcIiB5MT1cIjUuNjRcIiB4Mj1cIjE5Ljc4XCIgeTI9XCI0LjIyXCIvPjwvc3ZnPlxuKzM3MyA8L2J1dHRvbj5cbiszNzQgXG4gMzY4IDxkaXYgY2xhc3M9XCJwYWdlXCI+XG4gMzY5ICAgPGhlYWRlcj5cbiAzNzAgICAgIDxkaXYgY2xhc3M9XCJiYWRnZVwiPjxzcGFuIGNsYXNzPVwiZG90XCI+PC9zcGFuPiBSdW5zIGVudGlyZWx5IGluIHlvdXIgYnJvd3NlcjwvZGl2PlxuIDM3MSAgICAgPGgxPlRva2VuaXplciBQbGF5Z3JvdW5kPC9oMT5cbiAgICAgLi4uIiwiZmlyc3RDaGFuZ2VkTGluZSI6MzY4fSwiaXNFcnJvciI6ZmFsc2UsInRpbWVzdGFtcCI6MTc3MTAzNjg5NDcyMH19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6ImNlZGQxODI1IiwicGFyZW50SWQiOiI3ZDU3NDAyMyIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDE6NDYuNTI0WiIsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6Ik5vdyB1cGRhdGUgdGhlIEpTIHRvIGFkZCB0aGVtZSB0b2dnbGUgbG9naWMgYW5kIGEgbGlnaHQtbW9kZSBwYWxldHRlOiJ9LHsidHlwZSI6InRvb2xDYWxsIiwiaWQiOiJ0b29sdV8wMVdwNXBXZW9LbXk3RXFRRHhNMlRNa1AiLCJuYW1lIjoiZWRpdCIsImFyZ3VtZW50cyI6eyJwYXRoIjoiL2hvbWUvd3IvZ2gvdHJ5X3dlYl9ncHUvaW5kZXguaHRtbCIsIm9sZFRleHQiOiI8c2NyaXB0IHR5cGU9XCJtb2R1bGVcIj5cbiAgaW1wb3J0IHsgQXV0b1Rva2VuaXplciB9IGZyb20gJ2h0dHBzOi8vY2RuLmpzZGVsaXZyLm5ldC9ucG0vQGh1Z2dpbmdmYWNlL3RyYW5zZm9ybWVyc0AzJztcblxuICBjb25zdCBQQUxFVFRFID0gW1xuICAgIFsnIzJhMWYzZCcsJyNjNGI1ZmQnXSwgWycjMWEyZTFhJywnIzg2ZWZhYyddLCBbJyMxZTI5M2InLCcjN2RkM2ZjJ10sXG4gICAgWycjMzEyZTJhJywnI2ZjZDM0ZCddLCBbJyMyZDFmMmYnLCcjZjlhOGQ0J10sIFsnIzFhMmEyYScsJyM1ZWVhZDQnXSxcbiAgICBbJyMyZTI0MTgnLCcjZmRiYTc0J10sIFsnIzFlMWUzMCcsJyNhNWI0ZmMnXSwgWycjMmExYTFhJywnI2ZjYTVhNSddLFxuICAgIFsnIzFhMmUyNScsJyM2ZWU3YjcnXSxcbiAgXTtcblxuICBjb25zdCAkID0gKHNlbCkgPT4gZG9jdW1lbnQucXVlcnlTZWxlY3RvcihzZWwpOyIsIm5ld1RleHQiOiI8c2NyaXB0IHR5cGU9XCJtb2R1bGVcIj5cbiAgaW1wb3J0IHsgQXV0b1Rva2VuaXplciB9IGZyb20gJ2h0dHBzOi8vY2RuLmpzZGVsaXZyLm5ldC9ucG0vQGh1Z2dpbmdmYWNlL3RyYW5zZm9ybWVyc0AzJztcblxuICAvLyDigJQgVGhlbWUgdG9nZ2xlIOKAlFxuICBjb25zdCByb290ID0gZG9jdW1lbnQuZG9jdW1lbnRFbGVtZW50O1xuICBjb25zdCBzdG9yZWQgPSBsb2NhbFN0b3JhZ2UuZ2V0SXRlbSgndGhlbWUnKTtcbiAgY29uc3QgcHJlZmVyc0RhcmsgPSB3aW5kb3cubWF0Y2hNZWRpYSgnKHByZWZlcnMtY29sb3Itc2NoZW1lOiBkYXJrKScpLm1hdGNoZXM7XG4gIHJvb3Quc2V0QXR0cmlidXRlKCdkYXRhLXRoZW1lJywgc3RvcmVkIHx8IChwcmVmZXJzRGFyayA/ICdkYXJrJyA6ICdsaWdodCcpKTtcblxuICBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCgndGhlbWUtdG9nZ2xlJykuYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCAoKSA9PiB7XG4gICAgY29uc3QgbmV4dCA9IHJvb3QuZ2V0QXR0cmlidXRlKCdkYXRhLXRoZW1lJykgPT09ICdkYXJrJyA/ICdsaWdodCcgOiAnZGFyayc7XG4gICAgcm9vdC5zZXRBdHRyaWJ1dGUoJ2RhdGEtdGhlbWUnLCBuZXh0KTtcbiAgICBsb2NhbFN0b3JhZ2Uuc2V0SXRlbSgndGhlbWUnLCBuZXh0KTtcbiAgICAvLyByZS1yZW5kZXIgdG9rZW5zIHdpdGggY29ycmVjdCBwYWxldHRlIGlmIG91dHB1dCBpcyBzaG93aW5nXG4gICAgaWYgKGxhc3RJZHMubGVuZ3RoKSByZW5kZXJUb2tlbnMoKTtcbiAgfSk7XG5cbiAgZnVuY3Rpb24gaXNEYXJrKCkgeyByZXR1cm4gcm9vdC5nZXRBdHRyaWJ1dGUoJ2RhdGEtdGhlbWUnKSA9PT0gJ2RhcmsnOyB9XG5cbiAgY29uc3QgUEFMRVRURV9EQVJLID0gW1xuICAgIFsnIzJhMWYzZCcsJyNjNGI1ZmQnXSwgWycjMWEyZTFhJywnIzg2ZWZhYyddLCBbJyMxZTI5M2InLCcjN2RkM2ZjJ10sXG4gICAgWycjMzEyZTJhJywnI2ZjZDM0ZCddLCBbJyMyZDFmMmYnLCcjZjlhOGQ0J10sIFsnIzFhMmEyYScsJyM1ZWVhZDQnXSxcbiAgICBbJyMyZTI0MTgnLCcjZmRiYTc0J10sIFsnIzFlMWUzMCcsJyNhNWI0ZmMnXSwgWycjMmExYTFhJywnI2ZjYTVhNSddLFxuICAgIFsnIzFhMmUyNScsJyM2ZWU3YjcnXSxcbiAgXTtcbiAgY29uc3QgUEFMRVRURV9MSUdIVCA9IFtcbiAgICBbJyNlZGU5ZmUnLCcjNWIyMWI2J10sIFsnI2RjZmNlNycsJyMxNjY1MzQnXSwgWycjZGJlYWZlJywnIzFlNDBhZiddLFxuICAgIFsnI2ZlZjljMycsJyM4NTRkMGUnXSwgWycjZmNlN2YzJywnIzlkMTc0ZCddLCBbJyNjY2ZiZjEnLCcjMTE1ZTU5J10sXG4gICAgWycjZmZlZGQ1JywnIzlhMzQxMiddLCBbJyNlMGU3ZmYnLCcjMzczMGEzJ10sIFsnI2ZlZTJlMicsJyM5OTFiMWInXSxcbiAgICBbJyNkMWZhZTUnLCcjMDY1ZjQ2J10sXG4gIF07XG5cbiAgZnVuY3Rpb24gcGFsZXR0ZSgpIHsgcmV0dXJuIGlzRGFyaygpID8gUEFMRVRURV9EQVJLIDogUEFMRVRURV9MSUdIVDsgfVxuXG4gIGNvbnN0ICQgPSAoc2VsKSA9PiBkb2N1bWVudC5xdWVyeVNlbGVjdG9yKHNlbCk7In19XSwiYXBpIjoiYW50aHJvcGljLW1lc3NhZ2VzIiwicHJvdmlkZXIiOiJhbnRocm9waWMiLCJtb2RlbCI6ImNsYXVkZS1vcHVzLTQtNiIsInVzYWdlIjp7ImlucHV0IjoxLCJvdXRwdXQiOjkyOCwiY2FjaGVSZWFkIjoyNTMxMSwiY2FjaGVXcml0ZSI6NTM2LCJ0b3RhbFRva2VucyI6MjY3NzYsImNvc3QiOnsiaW5wdXQiOjAuMDAwMDA1LCJvdXRwdXQiOjAuMDIzMjAwMDAwMDAwMDAwMDAyLCJjYWNoZVJlYWQiOjAuMDEyNjU1NSwiY2FjaGVXcml0ZSI6MC4wMDMzNSwidG90YWwiOjAuMDM5MjEwNX19LCJzdG9wUmVhc29uIjoidG9vbFVzZSIsInRpbWVzdGFtcCI6MTc3MTAzNjg5NDcyMX19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6IjlmY2JkNzFkIiwicGFyZW50SWQiOiJjZWRkMTgyNSIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDE6NDYuNTQzWiIsIm1lc3NhZ2UiOnsicm9sZSI6InRvb2xSZXN1bHQiLCJ0b29sQ2FsbElkIjoidG9vbHVfMDFXcDVwV2VvS215N0VxUUR4TTJUTWtQIiwidG9vbE5hbWUiOiJlZGl0IiwiY29udGVudCI6W3sidHlwZSI6InRleHQiLCJ0ZXh0IjoiU3VjY2Vzc2Z1bGx5IHJlcGxhY2VkIHRleHQgaW4gL2hvbWUvd3IvZ2gvdHJ5X3dlYl9ncHUvaW5kZXguaHRtbC4ifV0sImRldGFpbHMiOnsiZGlmZiI6IiAgICAgLi4uXG4gNDQ0IFxuIDQ0NSA8c2NyaXB0IHR5cGU9XCJtb2R1bGVcIj5cbiA0NDYgICBpbXBvcnQgeyBBdXRvVG9rZW5pemVyIH0gZnJvbSAnaHR0cHM6Ly9jZG4uanNkZWxpdnIubmV0L25wbS9AaHVnZ2luZ2ZhY2UvdHJhbnNmb3JtZXJzQDMnO1xuIDQ0NyBcbi00NDggICBjb25zdCBQQUxFVFRFID0gW1xuKzQ0OCAgIC8vIOKAlCBUaGVtZSB0b2dnbGUg4oCUXG4rNDQ5ICAgY29uc3Qgcm9vdCA9IGRvY3VtZW50LmRvY3VtZW50RWxlbWVudDtcbis0NTAgICBjb25zdCBzdG9yZWQgPSBsb2NhbFN0b3JhZ2UuZ2V0SXRlbSgndGhlbWUnKTtcbis0NTEgICBjb25zdCBwcmVmZXJzRGFyayA9IHdpbmRvdy5tYXRjaE1lZGlhKCcocHJlZmVycy1jb2xvci1zY2hlbWU6IGRhcmspJykubWF0Y2hlcztcbis0NTIgICByb290LnNldEF0dHJpYnV0ZSgnZGF0YS10aGVtZScsIHN0b3JlZCB8fCAocHJlZmVyc0RhcmsgPyAnZGFyaycgOiAnbGlnaHQnKSk7XG4rNDUzIFxuKzQ1NCAgIGRvY3VtZW50LmdldEVsZW1lbnRCeUlkKCd0aGVtZS10b2dnbGUnKS5hZGRFdmVudExpc3RlbmVyKCdjbGljaycsICgpID0+IHtcbis0NTUgICAgIGNvbnN0IG5leHQgPSByb290LmdldEF0dHJpYnV0ZSgnZGF0YS10aGVtZScpID09PSAnZGFyaycgPyAnbGlnaHQnIDogJ2RhcmsnO1xuKzQ1NiAgICAgcm9vdC5zZXRBdHRyaWJ1dGUoJ2RhdGEtdGhlbWUnLCBuZXh0KTtcbis0NTcgICAgIGxvY2FsU3RvcmFnZS5zZXRJdGVtKCd0aGVtZScsIG5leHQpO1xuKzQ1OCAgICAgLy8gcmUtcmVuZGVyIHRva2VucyB3aXRoIGNvcnJlY3QgcGFsZXR0ZSBpZiBvdXRwdXQgaXMgc2hvd2luZ1xuKzQ1OSAgICAgaWYgKGxhc3RJZHMubGVuZ3RoKSByZW5kZXJUb2tlbnMoKTtcbis0NjAgICB9KTtcbis0NjEgXG4rNDYyICAgZnVuY3Rpb24gaXNEYXJrKCkgeyByZXR1cm4gcm9vdC5nZXRBdHRyaWJ1dGUoJ2RhdGEtdGhlbWUnKSA9PT0gJ2RhcmsnOyB9XG4rNDYzIFxuKzQ2NCAgIGNvbnN0IFBBTEVUVEVfREFSSyA9IFtcbiA0NDkgICAgIFsnIzJhMWYzZCcsJyNjNGI1ZmQnXSwgWycjMWEyZTFhJywnIzg2ZWZhYyddLCBbJyMxZTI5M2InLCcjN2RkM2ZjJ10sXG4gNDUwICAgICBbJyMzMTJlMmEnLCcjZmNkMzRkJ10sIFsnIzJkMWYyZicsJyNmOWE4ZDQnXSwgWycjMWEyYTJhJywnIzVlZWFkNCddLFxuIDQ1MSAgICAgWycjMmUyNDE4JywnI2ZkYmE3NCddLCBbJyMxZTFlMzAnLCcjYTViNGZjJ10sIFsnIzJhMWExYScsJyNmY2E1YTUnXSxcbiA0NTIgICAgIFsnIzFhMmUyNScsJyM2ZWU3YjcnXSxcbiA0NTMgICBdO1xuKzQ3MCAgIGNvbnN0IFBBTEVUVEVfTElHSFQgPSBbXG4rNDcxICAgICBbJyNlZGU5ZmUnLCcjNWIyMWI2J10sIFsnI2RjZmNlNycsJyMxNjY1MzQnXSwgWycjZGJlYWZlJywnIzFlNDBhZiddLFxuKzQ3MiAgICAgWycjZmVmOWMzJywnIzg1NGQwZSddLCBbJyNmY2U3ZjMnLCcjOWQxNzRkJ10sIFsnI2NjZmJmMScsJyMxMTVlNTknXSxcbis0NzMgICAgIFsnI2ZmZWRkNScsJyM5YTM0MTInXSwgWycjZTBlN2ZmJywnIzM3MzBhMyddLCBbJyNmZWUyZTInLCcjOTkxYjFiJ10sXG4rNDc0ICAgICBbJyNkMWZhZTUnLCcjMDY1ZjQ2J10sXG4rNDc1ICAgXTtcbiA0NTQgXG4rNDc3ICAgZnVuY3Rpb24gcGFsZXR0ZSgpIHsgcmV0dXJuIGlzRGFyaygpID8gUEFMRVRURV9EQVJLIDogUEFMRVRURV9MSUdIVDsgfVxuKzQ3OCBcbiA0NTUgICBjb25zdCAkID0gKHNlbCkgPT4gZG9jdW1lbnQucXVlcnlTZWxlY3RvcihzZWwpO1xuIDQ1NiAgIGNvbnN0IHNlbGVjdEVsICA9ICQoJyNtb2RlbC1zZWxlY3QnKTtcbiA0NTcgICBjb25zdCBpbnB1dEVsICAgPSAkKCcjaW5wdXQnKTtcbiA0NTggICBjb25zdCBidG4gICAgICAgPSAkKCcjYnRuJyk7XG4gICAgIC4uLiIsImZpcnN0Q2hhbmdlZExpbmUiOjQ0OH0sImlzRXJyb3IiOmZhbHNlLCJ0aW1lc3RhbXAiOjE3NzEwMzY5MDY1NDN9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiJmOGZiOWM1MSIsInBhcmVudElkIjoiOWZjYmQ3MWQiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjQyOjAxLjQ5MloiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJOb3cgcmVmYWN0b3IgdGhlIHRva2VuaXplIGZ1bmN0aW9uIHRvIHNlcGFyYXRlIHJlbmRlcmluZyBzbyB0aGUgdGhlbWUgdG9nZ2xlIGNhbiByZS1yZW5kZXI6In0seyJ0eXBlIjoidG9vbENhbGwiLCJpZCI6InRvb2x1XzAxRG1leUJKWTRYV1QxS3pkQndMWWhCTSIsIm5hbWUiOiJlZGl0IiwiYXJndW1lbnRzIjp7InBhdGgiOiIvaG9tZS93ci9naC90cnlfd2ViX2dwdS9pbmRleC5odG1sIiwib2xkVGV4dCI6IiAgbGV0IHRva2VuaXplciA9IG51bGw7XG4gIGxldCBjdXJyZW50TW9kZWwgPSAnJztcblxuICBhc3luYyBmdW5jdGlvbiBsb2FkVG9rZW5pemVyKG1vZGVsSWQpIHtcbiAgICBpZiAobW9kZWxJZCA9PT0gY3VycmVudE1vZGVsICYmIHRva2VuaXplcikgcmV0dXJuO1xuICAgIGJ0bi5kaXNhYmxlZCA9IHRydWU7XG4gICAgYnRuLnRleHRDb250ZW50ID0gJ0xvYWRpbmfigKYnO1xuICAgIHN0YXR1c0VsLmlubmVySFRNTCA9IGA8c3BhbiBjbGFzcz1cInNwaW5uZXJcIj48L3NwYW4+IERvd25sb2FkaW5nICR7bW9kZWxJZH3igKZgO1xuICAgIHRyeSB7XG4gICAgICB0b2tlbml6ZXIgPSBhd2FpdCBBdXRvVG9rZW5pemVyLmZyb21fcHJldHJhaW5lZChtb2RlbElkKTtcbiAgICAgIGN1cnJlbnRNb2RlbCA9IG1vZGVsSWQ7XG4gICAgICBzdGF0dXNFbC50ZXh0Q29udGVudCA9ICcnO1xuICAgICAgYnRuLmRpc2FibGVkID0gZmFsc2U7XG4gICAgICBidG4udGV4dENvbnRlbnQgPSAnVG9rZW5pemUnO1xuICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgIHN0YXR1c0VsLnRleHRDb250ZW50ID0gJ+KdjCAnICsgZS5tZXNzYWdlO1xuICAgICAgY29uc29sZS5lcnJvcihlKTtcbiAgICB9XG4gIH1cblxuICBmdW5jdGlvbiB0b2tlbml6ZSgpIHtcbiAgICBjb25zdCB0ZXh0ID0gaW5wdXRFbC52YWx1ZTtcbiAgICBpZiAoIXRleHQgfHwgIXRva2VuaXplcikgcmV0dXJuO1xuXG4gICAgY29uc3QgaWRzID0gQXJyYXkuZnJvbSh0b2tlbml6ZXIuZW5jb2RlKHRleHQpKTtcblxuICAgIHRva2VuR3JpZC5pbm5lckhUTUwgPSAnJztcbiAgICBmb3IgKGxldCBpID0gMDsgaSA8IGlkcy5sZW5ndGg7IGkrKykge1xuICAgICAgY29uc3QgZGVjb2RlZCA9IHRva2VuaXplci5kZWNvZGUoW2lkc1tpXV0sIHsgc2tpcF9zcGVjaWFsX3Rva2VuczogZmFsc2UgfSk7XG4gICAgICBjb25zdCBbYmcsIGZnXSA9IFBBTEVUVEVbaSAlIFBBTEVUVEUubGVuZ3RoXTtcbiAgICAgIGNvbnN0IHNwYW4gPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCdzcGFuJyk7XG4gICAgICBzcGFuLmNsYXNzTmFtZSA9ICd0b2snO1xuICAgICAgc3Bhbi5zdHlsZS5iYWNrZ3JvdW5kID0gYmc7XG4gICAgICBzcGFuLnN0eWxlLmNvbG9yID0gZmc7XG4gICAgICBzcGFuLnRleHRDb250ZW50ID0gZGVjb2RlZC5yZXBsYWNlKC8gL2csICfCtycpLnJlcGxhY2UoL1xcbi9nLCAn4oa1JykucmVwbGFjZSgvXFx0L2csICfih6UnKTtcbiAgICAgIHNwYW4uc2V0QXR0cmlidXRlKCdkYXRhLWlkJywgaWRzW2ldKTtcbiAgICAgIHRva2VuR3JpZC5hcHBlbmRDaGlsZChzcGFuKTtcbiAgICB9XG5cbiAgICBpZHNCb3gudGV4dENvbnRlbnQgPSAnWycgKyBpZHMuam9pbignLCAnKSArICddJztcbiAgICAkKCcjdG9rZW4tY291bnQnKS50ZXh0Q29udGVudCA9IGlkcy5sZW5ndGg7XG4gICAgJCgnI2NoYXItY291bnQnKS50ZXh0Q29udGVudCA9IHRleHQubGVuZ3RoO1xuICAgICQoJyNyYXRpbycpLnRleHRDb250ZW50ID0gaWRzLmxlbmd0aCA/ICh0ZXh0Lmxlbmd0aCAvIGlkcy5sZW5ndGgpLnRvRml4ZWQoMikgOiAn4oCUJztcbiAgICBzdGF0c0VsLmNsYXNzTGlzdC5hZGQoJ3Zpc2libGUnKTtcbiAgICBvdXRwdXQuY2xhc3NMaXN0LmFkZCgndmlzaWJsZScpO1xuICB9IiwibmV3VGV4dCI6IiAgbGV0IHRva2VuaXplciA9IG51bGw7XG4gIGxldCBjdXJyZW50TW9kZWwgPSAnJztcbiAgbGV0IGxhc3RJZHMgPSBbXTtcbiAgbGV0IGxhc3RUZXh0ID0gJyc7XG5cbiAgYXN5bmMgZnVuY3Rpb24gbG9hZFRva2VuaXplcihtb2RlbElkKSB7XG4gICAgaWYgKG1vZGVsSWQgPT09IGN1cnJlbnRNb2RlbCAmJiB0b2tlbml6ZXIpIHJldHVybjtcbiAgICBidG4uZGlzYWJsZWQgPSB0cnVlO1xuICAgIGJ0bi50ZXh0Q29udGVudCA9ICdMb2FkaW5n4oCmJztcbiAgICBzdGF0dXNFbC5pbm5lckhUTUwgPSBgPHNwYW4gY2xhc3M9XCJzcGlubmVyXCI+PC9zcGFuPiBEb3dubG9hZGluZyAke21vZGVsSWR94oCmYDtcbiAgICB0cnkge1xuICAgICAgdG9rZW5pemVyID0gYXdhaXQgQXV0b1Rva2VuaXplci5mcm9tX3ByZXRyYWluZWQobW9kZWxJZCk7XG4gICAgICBjdXJyZW50TW9kZWwgPSBtb2RlbElkO1xuICAgICAgc3RhdHVzRWwudGV4dENvbnRlbnQgPSAnJztcbiAgICAgIGJ0bi5kaXNhYmxlZCA9IGZhbHNlO1xuICAgICAgYnRuLnRleHRDb250ZW50ID0gJ1Rva2VuaXplJztcbiAgICB9IGNhdGNoIChlKSB7XG4gICAgICBzdGF0dXNFbC50ZXh0Q29udGVudCA9ICfinYwgJyArIGUubWVzc2FnZTtcbiAgICAgIGNvbnNvbGUuZXJyb3IoZSk7XG4gICAgfVxuICB9XG5cbiAgZnVuY3Rpb24gcmVuZGVyVG9rZW5zKCkge1xuICAgIGlmICghbGFzdElkcy5sZW5ndGggfHwgIXRva2VuaXplcikgcmV0dXJuO1xuICAgIGNvbnN0IHBhbCA9IHBhbGV0dGUoKTtcbiAgICB0b2tlbkdyaWQuaW5uZXJIVE1MID0gJyc7XG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCBsYXN0SWRzLmxlbmd0aDsgaSsrKSB7XG4gICAgICBjb25zdCBkZWNvZGVkID0gdG9rZW5pemVyLmRlY29kZShbbGFzdElkc1tpXV0sIHsgc2tpcF9zcGVjaWFsX3Rva2VuczogZmFsc2UgfSk7XG4gICAgICBjb25zdCBbYmcsIGZnXSA9IHBhbFtpICUgcGFsLmxlbmd0aF07XG4gICAgICBjb25zdCBzcGFuID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgnc3BhbicpO1xuICAgICAgc3Bhbi5jbGFzc05hbWUgPSAndG9rJztcbiAgICAgIHNwYW4uc3R5bGUuYmFja2dyb3VuZCA9IGJnO1xuICAgICAgc3Bhbi5zdHlsZS5jb2xvciA9IGZnO1xuICAgICAgc3Bhbi50ZXh0Q29udGVudCA9IGRlY29kZWQucmVwbGFjZSgvIC9nLCAnwrcnKS5yZXBsYWNlKC9cXG4vZywgJ+KGtScpLnJlcGxhY2UoL1xcdC9nLCAn4oelJyk7XG4gICAgICBzcGFuLnNldEF0dHJpYnV0ZSgnZGF0YS1pZCcsIGxhc3RJZHNbaV0pO1xuICAgICAgdG9rZW5HcmlkLmFwcGVuZENoaWxkKHNwYW4pO1xuICAgIH1cbiAgfVxuXG4gIGZ1bmN0aW9uIHRva2VuaXplKCkge1xuICAgIGxhc3RUZXh0ID0gaW5wdXRFbC52YWx1ZTtcbiAgICBpZiAoIWxhc3RUZXh0IHx8ICF0b2tlbml6ZXIpIHJldHVybjtcblxuICAgIGxhc3RJZHMgPSBBcnJheS5mcm9tKHRva2VuaXplci5lbmNvZGUobGFzdFRleHQpKTtcbiAgICByZW5kZXJUb2tlbnMoKTtcblxuICAgIGlkc0JveC50ZXh0Q29udGVudCA9ICdbJyArIGxhc3RJZHMuam9pbignLCAnKSArICddJztcbiAgICAkKCcjdG9rZW4tY291bnQnKS50ZXh0Q29udGVudCA9IGxhc3RJZHMubGVuZ3RoO1xuICAgICQoJyNjaGFyLWNvdW50JykudGV4dENvbnRlbnQgPSBsYXN0VGV4dC5sZW5ndGg7XG4gICAgJCgnI3JhdGlvJykudGV4dENvbnRlbnQgPSBsYXN0SWRzLmxlbmd0aCA/IChsYXN0VGV4dC5sZW5ndGggLyBsYXN0SWRzLmxlbmd0aCkudG9GaXhlZCgyKSA6ICfigJQnO1xuICAgIHN0YXRzRWwuY2xhc3NMaXN0LmFkZCgndmlzaWJsZScpO1xuICAgIG91dHB1dC5jbGFzc0xpc3QuYWRkKCd2aXNpYmxlJyk7XG4gIH0ifX1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjEsIm91dHB1dCI6MTIxMCwiY2FjaGVSZWFkIjoyNTg0NywiY2FjaGVXcml0ZSI6OTYxLCJ0b3RhbFRva2VucyI6MjgwMTksImNvc3QiOnsiaW5wdXQiOjAuMDAwMDA1LCJvdXRwdXQiOjAuMDMwMjUwMDAwMDAwMDAwMDAzLCJjYWNoZVJlYWQiOjAuMDEyOTIzNSwiY2FjaGVXcml0ZSI6MC4wMDYwMDYyNSwidG90YWwiOjAuMDQ5MTg0NzV9fSwic3RvcFJlYXNvbiI6InRvb2xVc2UiLCJ0aW1lc3RhbXAiOjE3NzEwMzY5MDY1NDR9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiI1ZmUwZTUzZCIsInBhcmVudElkIjoiZjhmYjljNTEiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjQyOjAxLjUxOVoiLCJtZXNzYWdlIjp7InJvbGUiOiJ0b29sUmVzdWx0IiwidG9vbENhbGxJZCI6InRvb2x1XzAxRG1leUJKWTRYV1QxS3pkQndMWWhCTSIsInRvb2xOYW1lIjoiZWRpdCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6IlN1Y2Nlc3NmdWxseSByZXBsYWNlZCB0ZXh0IGluIC9ob21lL3dyL2doL3RyeV93ZWJfZ3B1L2luZGV4Lmh0bWwuIn1dLCJkZXRhaWxzIjp7ImRpZmYiOiIgICAgIC4uLlxuIDQ4NyAgIGNvbnN0IG91dHB1dCAgICA9ICQoJyNvdXRwdXQnKTtcbiA0ODggXG4gNDg5ICAgbGV0IHRva2VuaXplciA9IG51bGw7XG4gNDkwICAgbGV0IGN1cnJlbnRNb2RlbCA9ICcnO1xuKzQ5MSAgIGxldCBsYXN0SWRzID0gW107XG4rNDkyICAgbGV0IGxhc3RUZXh0ID0gJyc7XG4gNDkxIFxuIDQ5MiAgIGFzeW5jIGZ1bmN0aW9uIGxvYWRUb2tlbml6ZXIobW9kZWxJZCkge1xuIDQ5MyAgICAgaWYgKG1vZGVsSWQgPT09IGN1cnJlbnRNb2RlbCAmJiB0b2tlbml6ZXIpIHJldHVybjtcbiA0OTQgICAgIGJ0bi5kaXNhYmxlZCA9IHRydWU7XG4gNDk1ICAgICBidG4udGV4dENvbnRlbnQgPSAnTG9hZGluZ+KApic7XG4gNDk2ICAgICBzdGF0dXNFbC5pbm5lckhUTUwgPSBgPHNwYW4gY2xhc3M9XCJzcGlubmVyXCI+PC9zcGFuPiBEb3dubG9hZGluZyAke21vZGVsSWR94oCmYDtcbiA0OTcgICAgIHRyeSB7XG4gNDk4ICAgICAgIHRva2VuaXplciA9IGF3YWl0IEF1dG9Ub2tlbml6ZXIuZnJvbV9wcmV0cmFpbmVkKG1vZGVsSWQpO1xuIDQ5OSAgICAgICBjdXJyZW50TW9kZWwgPSBtb2RlbElkO1xuIDUwMCAgICAgICBzdGF0dXNFbC50ZXh0Q29udGVudCA9ICcnO1xuIDUwMSAgICAgICBidG4uZGlzYWJsZWQgPSBmYWxzZTtcbiA1MDIgICAgICAgYnRuLnRleHRDb250ZW50ID0gJ1Rva2VuaXplJztcbiA1MDMgICAgIH0gY2F0Y2ggKGUpIHtcbiA1MDQgICAgICAgc3RhdHVzRWwudGV4dENvbnRlbnQgPSAn4p2MICcgKyBlLm1lc3NhZ2U7XG4gNTA1ICAgICAgIGNvbnNvbGUuZXJyb3IoZSk7XG4gNTA2ICAgICB9XG4gNTA3ICAgfVxuIDUwOCBcbi01MDkgICBmdW5jdGlvbiB0b2tlbml6ZSgpIHtcbi01MTAgICAgIGNvbnN0IHRleHQgPSBpbnB1dEVsLnZhbHVlO1xuLTUxMSAgICAgaWYgKCF0ZXh0IHx8ICF0b2tlbml6ZXIpIHJldHVybjtcbi01MTIgXG4tNTEzICAgICBjb25zdCBpZHMgPSBBcnJheS5mcm9tKHRva2VuaXplci5lbmNvZGUodGV4dCkpO1xuLTUxNCBcbis1MTEgICBmdW5jdGlvbiByZW5kZXJUb2tlbnMoKSB7XG4rNTEyICAgICBpZiAoIWxhc3RJZHMubGVuZ3RoIHx8ICF0b2tlbml6ZXIpIHJldHVybjtcbis1MTMgICAgIGNvbnN0IHBhbCA9IHBhbGV0dGUoKTtcbiA1MTUgICAgIHRva2VuR3JpZC5pbm5lckhUTUwgPSAnJztcbi01MTYgICAgIGZvciAobGV0IGkgPSAwOyBpIDwgaWRzLmxlbmd0aDsgaSsrKSB7XG4tNTE3ICAgICAgIGNvbnN0IGRlY29kZWQgPSB0b2tlbml6ZXIuZGVjb2RlKFtpZHNbaV1dLCB7IHNraXBfc3BlY2lhbF90b2tlbnM6IGZhbHNlIH0pO1xuLTUxOCAgICAgICBjb25zdCBbYmcsIGZnXSA9IFBBTEVUVEVbaSAlIFBBTEVUVEUubGVuZ3RoXTtcbis1MTUgICAgIGZvciAobGV0IGkgPSAwOyBpIDwgbGFzdElkcy5sZW5ndGg7IGkrKykge1xuKzUxNiAgICAgICBjb25zdCBkZWNvZGVkID0gdG9rZW5pemVyLmRlY29kZShbbGFzdElkc1tpXV0sIHsgc2tpcF9zcGVjaWFsX3Rva2VuczogZmFsc2UgfSk7XG4rNTE3ICAgICAgIGNvbnN0IFtiZywgZmddID0gcGFsW2kgJSBwYWwubGVuZ3RoXTtcbiA1MTkgICAgICAgY29uc3Qgc3BhbiA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoJ3NwYW4nKTtcbiA1MjAgICAgICAgc3Bhbi5jbGFzc05hbWUgPSAndG9rJztcbiA1MjEgICAgICAgc3Bhbi5zdHlsZS5iYWNrZ3JvdW5kID0gYmc7XG4gNTIyICAgICAgIHNwYW4uc3R5bGUuY29sb3IgPSBmZztcbiA1MjMgICAgICAgc3Bhbi50ZXh0Q29udGVudCA9IGRlY29kZWQucmVwbGFjZSgvIC9nLCAnwrcnKS5yZXBsYWNlKC9cXG4vZywgJ+KGtScpLnJlcGxhY2UoL1xcdC9nLCAn4oelJyk7XG4tNTI0ICAgICAgIHNwYW4uc2V0QXR0cmlidXRlKCdkYXRhLWlkJywgaWRzW2ldKTtcbis1MjMgICAgICAgc3Bhbi5zZXRBdHRyaWJ1dGUoJ2RhdGEtaWQnLCBsYXN0SWRzW2ldKTtcbiA1MjUgICAgICAgdG9rZW5HcmlkLmFwcGVuZENoaWxkKHNwYW4pO1xuIDUyNiAgICAgfVxuKzUyNiAgIH1cbiA1MjcgXG4tNTI4ICAgICBpZHNCb3gudGV4dENvbnRlbnQgPSAnWycgKyBpZHMuam9pbignLCAnKSArICddJztcbi01MjkgICAgICQoJyN0b2tlbi1jb3VudCcpLnRleHRDb250ZW50ID0gaWRzLmxlbmd0aDtcbi01MzAgICAgICQoJyNjaGFyLWNvdW50JykudGV4dENvbnRlbnQgPSB0ZXh0Lmxlbmd0aDtcbi01MzEgICAgICQoJyNyYXRpbycpLnRleHRDb250ZW50ID0gaWRzLmxlbmd0aCA/ICh0ZXh0Lmxlbmd0aCAvIGlkcy5sZW5ndGgpLnRvRml4ZWQoMikgOiAn4oCUJztcbis1MjggICBmdW5jdGlvbiB0b2tlbml6ZSgpIHtcbis1MjkgICAgIGxhc3RUZXh0ID0gaW5wdXRFbC52YWx1ZTtcbis1MzAgICAgIGlmICghbGFzdFRleHQgfHwgIXRva2VuaXplcikgcmV0dXJuO1xuKzUzMSBcbis1MzIgICAgIGxhc3RJZHMgPSBBcnJheS5mcm9tKHRva2VuaXplci5lbmNvZGUobGFzdFRleHQpKTtcbis1MzMgICAgIHJlbmRlclRva2VucygpO1xuKzUzNCBcbis1MzUgICAgIGlkc0JveC50ZXh0Q29udGVudCA9ICdbJyArIGxhc3RJZHMuam9pbignLCAnKSArICddJztcbis1MzYgICAgICQoJyN0b2tlbi1jb3VudCcpLnRleHRDb250ZW50ID0gbGFzdElkcy5sZW5ndGg7XG4rNTM3ICAgICAkKCcjY2hhci1jb3VudCcpLnRleHRDb250ZW50ID0gbGFzdFRleHQubGVuZ3RoO1xuKzUzOCAgICAgJCgnI3JhdGlvJykudGV4dENvbnRlbnQgPSBsYXN0SWRzLmxlbmd0aCA/IChsYXN0VGV4dC5sZW5ndGggLyBsYXN0SWRzLmxlbmd0aCkudG9GaXhlZCgyKSA6ICfigJQnO1xuIDUzMiAgICAgc3RhdHNFbC5jbGFzc0xpc3QuYWRkKCd2aXNpYmxlJyk7XG4gNTMzICAgICBvdXRwdXQuY2xhc3NMaXN0LmFkZCgndmlzaWJsZScpO1xuIDUzNCAgIH1cbiA1MzUgXG4gICAgIC4uLiIsImZpcnN0Q2hhbmdlZExpbmUiOjQ5MX0sImlzRXJyb3IiOmZhbHNlLCJ0aW1lc3RhbXAiOjE3NzEwMzY5MjE1MTl9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiI0OGY5OTFkNyIsInBhcmVudElkIjoiNWZlMGU1M2QiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjQyOjA1LjM1MloiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJOb3cgdXBkYXRlIHRoZSBnaXN0IGFuZCB2ZXJpZnkgdGhlIHNlcnZlciBpcyBzdGlsbCBydW5uaW5nOiJ9LHsidHlwZSI6InRvb2xDYWxsIiwiaWQiOiJ0b29sdV8wMTVNVVBSVUpiQnBYaGJIWXZtajhLUFYiLCJuYW1lIjoiYmFzaCIsImFyZ3VtZW50cyI6eyJjb21tYW5kIjoiY2QgL2hvbWUvd3IvZ2gvdHJ5X3dlYl9ncHUgJiYgY3VybCAtcyAtbyAvZGV2L251bGwgLXcgXCIle2h0dHBfY29kZX1cIiBodHRwOi8vbG9jYWxob3N0OjgwODAvIn19XSwiYXBpIjoiYW50aHJvcGljLW1lc3NhZ2VzIiwicHJvdmlkZXIiOiJhbnRocm9waWMiLCJtb2RlbCI6ImNsYXVkZS1vcHVzLTQtNiIsInVzYWdlIjp7ImlucHV0IjoxLCJvdXRwdXQiOjEwNCwiY2FjaGVSZWFkIjoyNjgwOCwiY2FjaGVXcml0ZSI6MTI0MywidG90YWxUb2tlbnMiOjI4MTU2LCJjb3N0Ijp7ImlucHV0IjowLjAwMDAwNSwib3V0cHV0IjowLjAwMjYwMDAwMDAwMDAwMDAwMDMsImNhY2hlUmVhZCI6MC4wMTM0MDQsImNhY2hlV3JpdGUiOjAuMDA3NzY4NzUwMDAwMDAwMDAwNSwidG90YWwiOjAuMDIzNzc3NzV9fSwic3RvcFJlYXNvbiI6InRvb2xVc2UiLCJ0aW1lc3RhbXAiOjE3NzEwMzY5MjE1MjB9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiJkM2ViNzZhNiIsInBhcmVudElkIjoiNDhmOTkxZDciLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjQyOjA1LjQwOFoiLCJtZXNzYWdlIjp7InJvbGUiOiJ0b29sUmVzdWx0IiwidG9vbENhbGxJZCI6InRvb2x1XzAxNU1VUFJVSmJCcFhoYkhZdm1qOEtQViIsInRvb2xOYW1lIjoiYmFzaCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6IjAwMFxuXG5Db21tYW5kIGV4aXRlZCB3aXRoIGNvZGUgNyJ9XSwiZGV0YWlscyI6e30sImlzRXJyb3IiOnRydWUsInRpbWVzdGFtcCI6MTc3MTAzNjkyNTQwOH19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6IjAzMjBkN2UyIiwicGFyZW50SWQiOiJkM2ViNzZhNiIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDI6MDkuMTMxWiIsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0b29sQ2FsbCIsImlkIjoidG9vbHVfMDFTaGNnU2o1UlY5TTFFNkMyanZrQUxRIiwibmFtZSI6ImJhc2giLCJhcmd1bWVudHMiOnsiY29tbWFuZCI6ImNkIC9ob21lL3dyL2doL3RyeV93ZWJfZ3B1ICYmIHB5dGhvbjMgLW0gaHR0cC5zZXJ2ZXIgODA4MCA+IC9kZXYvbnVsbCAyPiYxICZcbmVjaG8gXCJQSUQ6ICQhXCJcbnNsZWVwIDAuNVxuY3VybCAtcyAtbyAvZGV2L251bGwgLXcgXCIle2h0dHBfY29kZX1cIiBodHRwOi8vbG9jYWxob3N0OjgwODAvIn19XSwiYXBpIjoiYW50aHJvcGljLW1lc3NhZ2VzIiwicHJvdmlkZXIiOiJhbnRocm9waWMiLCJtb2RlbCI6ImNsYXVkZS1vcHVzLTQtNiIsInVzYWdlIjp7ImlucHV0IjoxLCJvdXRwdXQiOjEyOCwiY2FjaGVSZWFkIjoyODA1MSwiY2FjaGVXcml0ZSI6MTMzLCJ0b3RhbFRva2VucyI6MjgzMTMsImNvc3QiOnsiaW5wdXQiOjAuMDAwMDA1LCJvdXRwdXQiOjAuMDAzMiwiY2FjaGVSZWFkIjowLjAxNDAyNTUsImNhY2hlV3JpdGUiOjAuMDAwODMxMjUwMDAwMDAwMDAwMSwidG90YWwiOjAuMDE4MDYxNzQ5OTk5OTk5OTk4fX0sInN0b3BSZWFzb24iOiJ0b29sVXNlIiwidGltZXN0YW1wIjoxNzcxMDM2OTI1NDA4fX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiYjE2YWVjNzAiLCJwYXJlbnRJZCI6IjAzMjBkN2UyIiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjo0Mjo1Mi4yMTBaIiwibWVzc2FnZSI6eyJyb2xlIjoidG9vbFJlc3VsdCIsInRvb2xDYWxsSWQiOiJ0b29sdV8wMVNoY2dTajVSVjlNMUU2QzJqdmtBTFEiLCJ0b29sTmFtZSI6ImJhc2giLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJQSUQ6IDI4NDAyNFxuMjAwXG5cbkNvbW1hbmQgYWJvcnRlZCJ9XSwiZGV0YWlscyI6e30sImlzRXJyb3IiOnRydWUsInRpbWVzdGFtcCI6MTc3MTAzNjk3MjIwOH19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6ImQwNGVlODQ2IiwicGFyZW50SWQiOiJiMTZhZWM3MCIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDI6NTIuMjE5WiIsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOltdLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjAsIm91dHB1dCI6MCwiY2FjaGVSZWFkIjowLCJjYWNoZVdyaXRlIjowLCJ0b3RhbFRva2VucyI6MCwiY29zdCI6eyJpbnB1dCI6MCwib3V0cHV0IjowLCJjYWNoZVJlYWQiOjAsImNhY2hlV3JpdGUiOjAsInRvdGFsIjowfX0sInN0b3BSZWFzb24iOiJhYm9ydGVkIiwidGltZXN0YW1wIjoxNzcxMDM2OTcyMjEzLCJlcnJvck1lc3NhZ2UiOiJPcGVyYXRpb24gYWJvcnRlZCJ9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiJiOGRkNDc4YiIsInBhcmVudElkIjoiZDA0ZWU4NDYiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjQyOjU4LjM0NloiLCJtZXNzYWdlIjp7InJvbGUiOiJ1c2VyIiwiY29udGVudCI6W3sidHlwZSI6InRleHQiLCJ0ZXh0IjoiZ3JlYXQuIE5vdyB1cGxvYWQgaXQgdG8gZ2lzdCJ9XSwidGltZXN0YW1wIjoxNzcxMDM2OTc4MzQzfX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiZTZlOGI0NmQiLCJwYXJlbnRJZCI6ImI4ZGQ0NzhiIiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjo0MzowMy44NjJaIiwibWVzc2FnZSI6eyJyb2xlIjoiYXNzaXN0YW50IiwiY29udGVudCI6W3sidHlwZSI6InRvb2xDYWxsIiwiaWQiOiJ0b29sdV8wMVg1ZzJiUFVxdWN3dU42TExDdjZkU3MiLCJuYW1lIjoiYmFzaCIsImFyZ3VtZW50cyI6eyJjb21tYW5kIjoiY2QgL2hvbWUvd3IvZ2gvdHJ5X3dlYl9ncHUgJiYgZ2ggZ2lzdCBlZGl0IDQyNDEzMmFjMTljOTkzNGM0NmRlMzlmMDg5NjMyZmFlIGluZGV4Lmh0bWwifX1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjMsIm91dHB1dCI6OTEsImNhY2hlUmVhZCI6MjgxODQsImNhY2hlV3JpdGUiOjE3MSwidG90YWxUb2tlbnMiOjI4NDQ5LCJjb3N0Ijp7ImlucHV0IjowLjAwMDAxNTAwMDAwMDAwMDAwMDAwMiwib3V0cHV0IjowLjAwMjI3NSwiY2FjaGVSZWFkIjowLjAxNDA5MTk5OTk5OTk5OTk5OSwiY2FjaGVXcml0ZSI6MC4wMDEwNjg3NSwidG90YWwiOjAuMDE3NDUwNzQ5OTk5OTk5OTk4fX0sInN0b3BSZWFzb24iOiJ0b29sVXNlIiwidGltZXN0YW1wIjoxNzcxMDM2OTc4MzQ0fX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiOGQ4YzFjMjQiLCJwYXJlbnRJZCI6ImU2ZThiNDZkIiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjo0MzowNy4wNzNaIiwibWVzc2FnZSI6eyJyb2xlIjoidG9vbFJlc3VsdCIsInRvb2xDYWxsSWQiOiJ0b29sdV8wMVg1ZzJiUFVxdWN3dU42TExDdjZkU3MiLCJ0b29sTmFtZSI6ImJhc2giLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiIobm8gb3V0cHV0KSJ9XSwiaXNFcnJvciI6ZmFsc2UsInRpbWVzdGFtcCI6MTc3MTAzNjk4NzA0N319LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6IjNhNjFhNTc5IiwicGFyZW50SWQiOiI4ZDhjMWMyNCIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDM6MTAuNjQxWiIsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6IlVwZGF0ZWQgdGhlIGV4aXN0aW5nIGdpc3Q6XG5cbioqR2lzdDoqKiBodHRwczovL2dpc3QuZ2l0aHViLmNvbS9pYW13cm0vNDI0MTMyYWMxOWM5OTM0YzQ2ZGUzOWYwODk2MzJmYWVcblxuKipMaXZlIHByZXZpZXc6KiogaHR0cHM6Ly9naXN0LmdpdGhhY2suY29tL2lhbXdybS80MjQxMzJhYzE5Yzk5MzRjNDZkZTM5ZjA4OTYzMmZhZS9yYXcvaW5kZXguaHRtbCJ9XSwiYXBpIjoiYW50aHJvcGljLW1lc3NhZ2VzIiwicHJvdmlkZXIiOiJhbnRocm9waWMiLCJtb2RlbCI6ImNsYXVkZS1vcHVzLTQtNiIsInVzYWdlIjp7ImlucHV0IjoxLCJvdXRwdXQiOjg5LCJjYWNoZVJlYWQiOjI4MzU1LCJjYWNoZVdyaXRlIjoxMDcsInRvdGFsVG9rZW5zIjoyODU1MiwiY29zdCI6eyJpbnB1dCI6MC4wMDAwMDUsIm91dHB1dCI6MC4wMDIyMjUsImNhY2hlUmVhZCI6MC4wMTQxNzc0OTk5OTk5OTk5OTksImNhY2hlV3JpdGUiOjAuMDAwNjY4NzUwMDAwMDAwMDAwMSwidG90YWwiOjAuMDE3MDc2MjQ5OTk5OTk5OTk4fX0sInN0b3BSZWFzb24iOiJzdG9wIiwidGltZXN0YW1wIjoxNzcxMDM2OTg3MDczfX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiZTM4NmI0OWMiLCJwYXJlbnRJZCI6IjNhNjFhNTc5IiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjo0NDowNi42NDBaIiwibWVzc2FnZSI6eyJyb2xlIjoidXNlciIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6ImFyZSB5b3Ugc3VyZSwgaSdtIHN0aWxsIHNlZWluZyB0aGUgb2xkZXIgcGFnZSJ9XSwidGltZXN0YW1wIjoxNzcxMDM3MDQ2NjMyfX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiMjRkYWY3OTYiLCJwYXJlbnRJZCI6ImUzODZiNDljIiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjo0NDoxMC41NzlaIiwibWVzc2FnZSI6eyJyb2xlIjoiYXNzaXN0YW50IiwiY29udGVudCI6W3sidHlwZSI6InRvb2xDYWxsIiwiaWQiOiJ0b29sdV8wMVRXbVM5OTljODNXcUwzWTh2cEMxMkUiLCJuYW1lIjoiYmFzaCIsImFyZ3VtZW50cyI6eyJjb21tYW5kIjoiZ2ggZ2lzdCB2aWV3IDQyNDEzMmFjMTljOTkzNGM0NmRlMzlmMDg5NjMyZmFlIC0tZmlsZXMifX1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjMsIm91dHB1dCI6NzUsImNhY2hlUmVhZCI6Mjg0NjIsImNhY2hlV3JpdGUiOjEwMSwidG90YWxUb2tlbnMiOjI4NjQxLCJjb3N0Ijp7ImlucHV0IjowLjAwMDAxNTAwMDAwMDAwMDAwMDAwMiwib3V0cHV0IjowLjAwMTg3NTAwMDAwMDAwMDAwMDEsImNhY2hlUmVhZCI6MC4wMTQyMzA5OTk5OTk5OTk5OTksImNhY2hlV3JpdGUiOjAuMDAwNjMxMjUwMDAwMDAwMDAwMSwidG90YWwiOjAuMDE2NzUyMjV9fSwic3RvcFJlYXNvbiI6InRvb2xVc2UiLCJ0aW1lc3RhbXAiOjE3NzEwMzcwNDY2MzV9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiJmMTEzNjE3NiIsInBhcmVudElkIjoiMjRkYWY3OTYiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjQ0OjExLjg0MVoiLCJtZXNzYWdlIjp7InJvbGUiOiJ0b29sUmVzdWx0IiwidG9vbENhbGxJZCI6InRvb2x1XzAxVFdtUzk5OWM4M1dxTDNZOHZwQzEyRSIsInRvb2xOYW1lIjoiYmFzaCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6ImluZGV4Lmh0bWxcbiJ9XSwiaXNFcnJvciI6ZmFsc2UsInRpbWVzdGFtcCI6MTc3MTAzNzA1MTg0MX19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6IjNiZTlkODY3IiwicGFyZW50SWQiOiJmMTEzNjE3NiIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDQ6MTQuOTMxWiIsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0b29sQ2FsbCIsImlkIjoidG9vbHVfMDFDekdnREFTRG1rakFYazdEaUNLRllrIiwibmFtZSI6ImJhc2giLCJhcmd1bWVudHMiOnsiY29tbWFuZCI6ImdoIGdpc3QgdmlldyA0MjQxMzJhYzE5Yzk5MzRjNDZkZTM5ZjA4OTYzMmZhZSAtZiBpbmRleC5odG1sIHwgaGVhZCAtMzAifX1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjEsIm91dHB1dCI6ODEsImNhY2hlUmVhZCI6Mjg1NjMsImNhY2hlV3JpdGUiOjkzLCJ0b3RhbFRva2VucyI6Mjg3MzgsImNvc3QiOnsiaW5wdXQiOjAuMDAwMDA1LCJvdXRwdXQiOjAuMDAyMDI1LCJjYWNoZVJlYWQiOjAuMDE0MjgxNDk5OTk5OTk5OTk5LCJjYWNoZVdyaXRlIjowLjAwMDU4MTI1MDAwMDAwMDAwMDEsInRvdGFsIjowLjAxNjg5Mjc0OTk5OTk5OTk5OH19LCJzdG9wUmVhc29uIjoidG9vbFVzZSIsInRpbWVzdGFtcCI6MTc3MTAzNzA1MTg0MX19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6ImUwZjcwNjgxIiwicGFyZW50SWQiOiIzYmU5ZDg2NyIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDQ6MTcuMDk5WiIsIm1lc3NhZ2UiOnsicm9sZSI6InRvb2xSZXN1bHQiLCJ0b29sQ2FsbElkIjoidG9vbHVfMDFDekdnREFTRG1rakFYazdEaUNLRllrIiwidG9vbE5hbWUiOiJiYXNoIiwiY29udGVudCI6W3sidHlwZSI6InRleHQiLCJ0ZXh0IjoiPCFET0NUWVBFIGh0bWw+XG48aHRtbCBsYW5nPVwiZW5cIj5cbjxoZWFkPlxuICA8bWV0YSBjaGFyc2V0PVwiVVRGLThcIj5cbiAgPG1ldGEgbmFtZT1cInZpZXdwb3J0XCIgY29udGVudD1cIndpZHRoPWRldmljZS13aWR0aCwgaW5pdGlhbC1zY2FsZT0xLjBcIj5cbiAgPHRpdGxlPlRva2VuaXplciBQbGF5Z3JvdW5kPC90aXRsZT5cbiAgPHN0eWxlPlxuICAgIEBpbXBvcnQgdXJsKCdodHRwczovL2ZvbnRzLmdvb2dsZWFwaXMuY29tL2NzczI/ZmFtaWx5PUludGVyOndnaHRANDAwOzUwMDs2MDA7NzAwJmZhbWlseT1JQk0rUGxleCtNb25vOndnaHRANDAwOzUwMCZkaXNwbGF5PXN3YXAnKTtcblxuICAgIDpyb290LFxuICAgIFtkYXRhLXRoZW1lPVwiZGFya1wiXSB7XG4gICAgICAtLWJnOiAjMGQwZjE0O1xuICAgICAgLS1zdXJmYWNlOiAjMTMxNTFjO1xuICAgICAgLS1zdXJmYWNlLTI6ICMxOTFjMjY7XG4gICAgICAtLWJvcmRlcjogIzIzMjczYTtcbiAgICAgIC0tYm9yZGVyLWhvdmVyOiAjMzQzYTU0O1xuICAgICAgLS10ZXh0OiAjZTZlZGYzO1xuICAgICAgLS10ZXh0LW11dGVkOiAjN2Q4NTkwO1xuICAgICAgLS10ZXh0LWRpbTogIzQ4NGY1ODtcbiAgICAgIC0tYWNjZW50OiAjYmM4Y2ZmO1xuICAgICAgLS1hY2NlbnQtZGltOiByZ2JhKDE4OCwgMTQwLCAyNTUsIDAuMTIpO1xuICAgICAgLS1hY2NlbnQtMjogI2QyYThmZjtcbiAgICAgIC0tYnRuLXRleHQ6ICMwZDBmMTQ7XG4gICAgICAtLXRvb2x0aXAtYmc6ICMyZDMzM2I7XG4gICAgICAtLXRvb2x0aXAtdGV4dDogI2U2ZWRmMztcbiAgICAgIC0taDEtZnJvbTogI2ZmZjtcbiAgICAgIC0tbm9pc2Utb3BhY2l0eTogMC4wMztcbiAgICAgIC0tbW9ubzogJ0lCTSBQbGV4IE1vbm8nLCAnU0YgTW9ubycsICdGaXJhIENvZGUnLCBtb25vc3BhY2U7XG4gICAgICAtLXNhbnM6ICdJbnRlcicsIC1hcHBsZS1zeXN0ZW0sIEJsaW5rTWFjU3lzdGVtRm9udCwgc2Fucy1zZXJpZjtcbiAgICAgIC0tcmFkaXVzOiA4cHg7XG4ifV0sImlzRXJyb3IiOmZhbHNlLCJ0aW1lc3RhbXAiOjE3NzEwMzcwNTcwOTh9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiJjZDEyYzAyYyIsInBhcmVudElkIjoiZTBmNzA2ODEiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjQ0OjE5LjcyNloiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidG9vbENhbGwiLCJpZCI6InRvb2x1XzAxNll3R1hka0I4bTZLQkc0alA0Y2hTRiIsIm5hbWUiOiJiYXNoIiwiYXJndW1lbnRzIjp7ImNvbW1hbmQiOiJnaCBnaXN0IHZpZXcgNDI0MTMyYWMxOWM5OTM0YzQ2ZGUzOWYwODk2MzJmYWUgLWYgaW5kZXguaHRtbCB8IGdyZXAgLWMgXCJ0aGVtZS10b2dnbGVcIiJ9fV0sImFwaSI6ImFudGhyb3BpYy1tZXNzYWdlcyIsInByb3ZpZGVyIjoiYW50aHJvcGljIiwibW9kZWwiOiJjbGF1ZGUtb3B1cy00LTYiLCJ1c2FnZSI6eyJpbnB1dCI6MSwib3V0cHV0Ijo4NiwiY2FjaGVSZWFkIjoyODY1NiwiY2FjaGVXcml0ZSI6NDk5LCJ0b3RhbFRva2VucyI6MjkyNDIsImNvc3QiOnsiaW5wdXQiOjAuMDAwMDA1LCJvdXRwdXQiOjAuMDAyMTUsImNhY2hlUmVhZCI6MC4wMTQzMjc5OTk5OTk5OTk5OTksImNhY2hlV3JpdGUiOjAuMDAzMTE4NzUsInRvdGFsIjowLjAxOTYwMTc0OTk5OTk5OTk5OH19LCJzdG9wUmVhc29uIjoidG9vbFVzZSIsInRpbWVzdGFtcCI6MTc3MTAzNzA1NzEwMH19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6IjdjNWE3YzIwIiwicGFyZW50SWQiOiJjZDEyYzAyYyIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDQ6MjAuODk0WiIsIm1lc3NhZ2UiOnsicm9sZSI6InRvb2xSZXN1bHQiLCJ0b29sQ2FsbElkIjoidG9vbHVfMDE2WXdHWGRrQjhtNktCRzRqUDRjaFNGIiwidG9vbE5hbWUiOiJiYXNoIiwiY29udGVudCI6W3sidHlwZSI6InRleHQiLCJ0ZXh0IjoiMTBcbiJ9XSwiaXNFcnJvciI6ZmFsc2UsInRpbWVzdGFtcCI6MTc3MTAzNzA2MDg5M319LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6ImY4YzU2ZGY1IiwicGFyZW50SWQiOiI3YzVhN2MyMCIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDQ6MjUuNzkwWiIsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6IlRoZSBnaXN0ICoqaXMqKiB1cGRhdGVkIOKAlCBpdCBoYXMgdGhlIHRoZW1lIHRvZ2dsZSBjb2RlLiBUaGUgaXNzdWUgaXMgbGlrZWx5IENETiBjYWNoaW5nIG9uIGBnaXN0LmdpdGhhY2suY29tYC4gVHJ5IHRoaXMgZnJlc2ggVVJMIGluc3RlYWQgKHVzZXMgYHJhd2NkbmAgZm9yIGNhY2hlLWJ1c3RpbmcpOiJ9LHsidHlwZSI6InRvb2xDYWxsIiwiaWQiOiJ0b29sdV8wMUFWWFJremVQMmNkeUZiMUhieFZ1U2EiLCJuYW1lIjoiYmFzaCIsImFyZ3VtZW50cyI6eyJjb21tYW5kIjoiIyBHZXQgdGhlIGxhdGVzdCBjb21taXQgU0hBIGZvciB0aGUgZ2lzdFxuZ2ggYXBpIFwiZ2lzdHMvNDI0MTMyYWMxOWM5OTM0YzQ2ZGUzOWYwODk2MzJmYWVcIiAtLWpxICcuaGlzdG9yeVswXS52ZXJzaW9uJyJ9fV0sImFwaSI6ImFudGhyb3BpYy1tZXNzYWdlcyIsInByb3ZpZGVyIjoiYW50aHJvcGljIiwibW9kZWwiOiJjbGF1ZGUtb3B1cy00LTYiLCJ1c2FnZSI6eyJpbnB1dCI6MSwib3V0cHV0IjoxNDYsImNhY2hlUmVhZCI6MjkxNTUsImNhY2hlV3JpdGUiOjEwMCwidG90YWxUb2tlbnMiOjI5NDAyLCJjb3N0Ijp7ImlucHV0IjowLjAwMDAwNSwib3V0cHV0IjowLjAwMzY1LCJjYWNoZVJlYWQiOjAuMDE0NTc3NSwiY2FjaGVXcml0ZSI6MC4wMDA2MjUsInRvdGFsIjowLjAxODg1NzV9fSwic3RvcFJlYXNvbiI6InRvb2xVc2UiLCJ0aW1lc3RhbXAiOjE3NzEwMzcwNjA4OTR9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiI2OGUxZTk4MyIsInBhcmVudElkIjoiZjhjNTZkZjUiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjQ0OjI3LjA4NloiLCJtZXNzYWdlIjp7InJvbGUiOiJ0b29sUmVzdWx0IiwidG9vbENhbGxJZCI6InRvb2x1XzAxQVZYUmt6ZVAyY2R5RmIxSGJ4VnVTYSIsInRvb2xOYW1lIjoiYmFzaCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6Ijg0OTBlYzIxOTA1N2QxZmJjNWE5ZGY0MWQ5MWQxZDA5MjY4YjU4YTFcbiJ9XSwiaXNFcnJvciI6ZmFsc2UsInRpbWVzdGFtcCI6MTc3MTAzNzA2NzA4Nn19LHsidHlwZSI6Im1lc3NhZ2UiLCJpZCI6IjY2ZjYyMzU1IiwicGFyZW50SWQiOiI2OGUxZTk4MyIsInRpbWVzdGFtcCI6IjIwMjYtMDItMTRUMDI6NDQ6MzEuNzg5WiIsIm1lc3NhZ2UiOnsicm9sZSI6ImFzc2lzdGFudCIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6IkhlcmUncyBhIGNhY2hlLXByb29mIGRpcmVjdCBsaW5rIHVzaW5nIHRoZSBjb21taXQgU0hBOlxuXG5odHRwczovL2dpc3QuZ2l0aGFjay5jb20vaWFtd3JtLzQyNDEzMmFjMTljOTkzNGM0NmRlMzlmMDg5NjMyZmFlL3Jhdy84NDkwZWMyMTkwNTdkMWZiYzVhOWRmNDFkOTFkMWQwOTI2OGI1OGExL2luZGV4Lmh0bWxcblxuVGhlIG9sZCBVUkwgd2FzIHNlcnZpbmcgYSBjYWNoZWQgdmVyc2lvbi4gWW91IGNhbiBhbHNvIHRyeSBoYXJkLXJlZnJlc2hpbmcgKGBDdHJsK1NoaWZ0K1JgKSBvbiB0aGUgb3JpZ2luYWwgbGluaywgb3IgdXNlIHRoaXMgcGlubmVkLXRvLWNvbW1pdCBVUkwgYWJvdmUuIn1dLCJhcGkiOiJhbnRocm9waWMtbWVzc2FnZXMiLCJwcm92aWRlciI6ImFudGhyb3BpYyIsIm1vZGVsIjoiY2xhdWRlLW9wdXMtNC02IiwidXNhZ2UiOnsiaW5wdXQiOjEsIm91dHB1dCI6MTI1LCJjYWNoZVJlYWQiOjI5MjU1LCJjYWNoZVdyaXRlIjoxODQsInRvdGFsVG9rZW5zIjoyOTU2NSwiY29zdCI6eyJpbnB1dCI6MC4wMDAwMDUsIm91dHB1dCI6MC4wMDMxMjUsImNhY2hlUmVhZCI6MC4wMTQ2Mjc1LCJjYWNoZVdyaXRlIjowLjAwMTE1LCJ0b3RhbCI6MC4wMTg5MDc1fX0sInN0b3BSZWFzb24iOiJzdG9wIiwidGltZXN0YW1wIjoxNzcxMDM3MDY3MDg3fX0seyJ0eXBlIjoibWVzc2FnZSIsImlkIjoiNmMyNzJhNzIiLCJwYXJlbnRJZCI6IjY2ZjYyMzU1IiwidGltZXN0YW1wIjoiMjAyNi0wMi0xNFQwMjo0NDo1Mi43NDZaIiwibWVzc2FnZSI6eyJyb2xlIjoidXNlciIsImNvbnRlbnQiOlt7InR5cGUiOiJ0ZXh0IiwidGV4dCI6Imdvb2QhIn1dLCJ0aW1lc3RhbXAiOjE3NzEwMzcwOTI3NDB9fSx7InR5cGUiOiJtZXNzYWdlIiwiaWQiOiI2ZGVlNWU1NiIsInBhcmVudElkIjoiNmMyNzJhNzIiLCJ0aW1lc3RhbXAiOiIyMDI2LTAyLTE0VDAyOjQ0OjU5LjY2MVoiLCJtZXNzYWdlIjp7InJvbGUiOiJhc3Npc3RhbnQiLCJjb250ZW50IjpbeyJ0eXBlIjoidGV4dCIsInRleHQiOiJcblxuR2xhZCBpdCB3b3JrcyEgVGhlIHRva2VuaXplciBwbGF5Z3JvdW5kIG5vdyBoYXM6XG5cbi0gKioxMiB0b2tlbml6ZXJzKiogYWNyb3NzIE9wZW5BSSwgTWV0YSwgTWlzdHJhbCwgR29vZ2xlLCBNaWNyb3NvZnQsIEFsaWJhYmEsIERlZXBTZWVrLCBhbmQgQW50aHJvcGljXG4tICoqTGlnaHQvZGFyayB0aGVtZSB0b2dnbGUqKiAodG9wLXJpZ2h0IG1vb24vc3VuIGJ1dHRvbikgd2l0aCBgbG9jYWxTdG9yYWdlYCBwZXJzaXN0ZW5jZSBhbmQgT1MgcHJlZmVyZW5jZSBkZXRlY3Rpb25cbi0gKipBc3Ryby1pbnNwaXJlZCBtaW5pbWFsIGFlc3RoZXRpYyoqIOKAlCBncmFkaWVudCBoZWFkaW5nLCBub2lzZSB0ZXh0dXJlLCBwaWxsIGJhZGdlLCBzbW9vdGggdHJhbnNpdGlvbnMsIHByb3BlciBsaWdodC9kYXJrIHBhbGV0dGVzIGZvciB0b2tlbiBjaGlwc1xuXG5FbmpveSEg8J+agCJ9XSwiYXBpIjoiYW50aHJvcGljLW1lc3NhZ2VzIiwicHJvdmlkZXIiOiJhbnRocm9waWMiLCJtb2RlbCI6ImNsYXVkZS1vcHVzLTQtNiIsInVzYWdlIjp7ImlucHV0IjozLCJvdXRwdXQiOjEyMCwiY2FjaGVSZWFkIjoyOTQzOSwiY2FjaGVXcml0ZSI6MTI4LCJ0b3RhbFRva2VucyI6Mjk2OTAsImNvc3QiOnsiaW5wdXQiOjAuMDAwMDE1MDAwMDAwMDAwMDAwMDAyLCJvdXRwdXQiOjAuMDAzLCJjYWNoZVJlYWQiOjAuMDE0NzE5NSwiY2FjaGVXcml0ZSI6MC4wMDA4LCJ0b3RhbCI6MC4wMTg1MzQ1fX0sInN0b3BSZWFzb24iOiJzdG9wIiwidGltZXN0YW1wIjoxNzcxMDM3MDkyNzQyfX1dLCJsZWFmSWQiOiI2ZGVlNWU1NiIsInN5c3RlbVByb21wdCI6IllvdSBhcmUgYW4gZXhwZXJ0IGNvZGluZyBhc3Npc3RhbnQgb3BlcmF0aW5nIGluc2lkZSBwaSwgYSBjb2RpbmcgYWdlbnQgaGFybmVzcy4gWW91IGhlbHAgdXNlcnMgYnkgcmVhZGluZyBmaWxlcywgZXhlY3V0aW5nIGNvbW1hbmRzLCBlZGl0aW5nIGNvZGUsIGFuZCB3cml0aW5nIG5ldyBmaWxlcy5cblxuQXZhaWxhYmxlIHRvb2xzOlxuLSByZWFkOiBSZWFkIGZpbGUgY29udGVudHNcbi0gYmFzaDogRXhlY3V0ZSBiYXNoIGNvbW1hbmRzIChscywgZ3JlcCwgZmluZCwgZXRjLilcbi0gZWRpdDogTWFrZSBzdXJnaWNhbCBlZGl0cyB0byBmaWxlcyAoZmluZCBleGFjdCB0ZXh0IGFuZCByZXBsYWNlKVxuLSB3cml0ZTogQ3JlYXRlIG9yIG92ZXJ3cml0ZSBmaWxlc1xuXG5JbiBhZGRpdGlvbiB0byB0aGUgdG9vbHMgYWJvdmUsIHlvdSBtYXkgaGF2ZSBhY2Nlc3MgdG8gb3RoZXIgY3VzdG9tIHRvb2xzIGRlcGVuZGluZyBvbiB0aGUgcHJvamVjdC5cblxuR3VpZGVsaW5lczpcbi0gVXNlIGJhc2ggZm9yIGZpbGUgb3BlcmF0aW9ucyBsaWtlIGxzLCByZywgZmluZFxuLSBVc2UgcmVhZCB0byBleGFtaW5lIGZpbGVzIGJlZm9yZSBlZGl0aW5nLiBZb3UgbXVzdCB1c2UgdGhpcyB0b29sIGluc3RlYWQgb2YgY2F0IG9yIHNlZC5cbi0gVXNlIGVkaXQgZm9yIHByZWNpc2UgY2hhbmdlcyAob2xkIHRleHQgbXVzdCBtYXRjaCBleGFjdGx5KVxuLSBVc2Ugd3JpdGUgb25seSBmb3IgbmV3IGZpbGVzIG9yIGNvbXBsZXRlIHJld3JpdGVzXG4tIFdoZW4gc3VtbWFyaXppbmcgeW91ciBhY3Rpb25zLCBvdXRwdXQgcGxhaW4gdGV4dCBkaXJlY3RseSAtIGRvIE5PVCB1c2UgY2F0IG9yIGJhc2ggdG8gZGlzcGxheSB3aGF0IHlvdSBkaWRcbi0gQmUgY29uY2lzZSBpbiB5b3VyIHJlc3BvbnNlc1xuLSBTaG93IGZpbGUgcGF0aHMgY2xlYXJseSB3aGVuIHdvcmtpbmcgd2l0aCBmaWxlc1xuXG5QaSBkb2N1bWVudGF0aW9uIChyZWFkIG9ubHkgd2hlbiB0aGUgdXNlciBhc2tzIGFib3V0IHBpIGl0c2VsZiwgaXRzIFNESywgZXh0ZW5zaW9ucywgdGhlbWVzLCBza2lsbHMsIG9yIFRVSSk6XG4tIE1haW4gZG9jdW1lbnRhdGlvbjogL2hvbWUvd3IvLm5wbS9fbnB4L2M5MDM3MGEzNDlkNWExMTgvbm9kZV9tb2R1bGVzL0BtYXJpb3plY2huZXIvcGktY29kaW5nLWFnZW50L1JFQURNRS5tZFxuLSBBZGRpdGlvbmFsIGRvY3M6IC9ob21lL3dyLy5ucG0vX25weC9jOTAzNzBhMzQ5ZDVhMTE4L25vZGVfbW9kdWxlcy9AbWFyaW96ZWNobmVyL3BpLWNvZGluZy1hZ2VudC9kb2NzXG4tIEV4YW1wbGVzOiAvaG9tZS93ci8ubnBtL19ucHgvYzkwMzcwYTM0OWQ1YTExOC9ub2RlX21vZHVsZXMvQG1hcmlvemVjaG5lci9waS1jb2RpbmctYWdlbnQvZXhhbXBsZXMgKGV4dGVuc2lvbnMsIGN1c3RvbSB0b29scywgU0RLKVxuLSBXaGVuIGFza2VkIGFib3V0OiBleHRlbnNpb25zIChkb2NzL2V4dGVuc2lvbnMubWQsIGV4YW1wbGVzL2V4dGVuc2lvbnMvKSwgdGhlbWVzIChkb2NzL3RoZW1lcy5tZCksIHNraWxscyAoZG9jcy9za2lsbHMubWQpLCBwcm9tcHQgdGVtcGxhdGVzIChkb2NzL3Byb21wdC10ZW1wbGF0ZXMubWQpLCBUVUkgY29tcG9uZW50cyAoZG9jcy90dWkubWQpLCBrZXliaW5kaW5ncyAoZG9jcy9rZXliaW5kaW5ncy5tZCksIFNESyBpbnRlZ3JhdGlvbnMgKGRvY3Mvc2RrLm1kKSwgY3VzdG9tIHByb3ZpZGVycyAoZG9jcy9jdXN0b20tcHJvdmlkZXIubWQpLCBhZGRpbmcgbW9kZWxzIChkb2NzL21vZGVscy5tZCksIHBpIHBhY2thZ2VzIChkb2NzL3BhY2thZ2VzLm1kKVxuLSBXaGVuIHdvcmtpbmcgb24gcGkgdG9waWNzLCByZWFkIHRoZSBkb2NzIGFuZCBleGFtcGxlcywgYW5kIGZvbGxvdyAubWQgY3Jvc3MtcmVmZXJlbmNlcyBiZWZvcmUgaW1wbGVtZW50aW5nXG4tIEFsd2F5cyByZWFkIHBpIC5tZCBmaWxlcyBjb21wbGV0ZWx5IGFuZCBmb2xsb3cgbGlua3MgdG8gcmVsYXRlZCBkb2NzIChlLmcuLCB0dWkubWQgZm9yIFRVSSBBUEkgZGV0YWlscylcbkN1cnJlbnQgZGF0ZSBhbmQgdGltZTogU2F0dXJkYXksIEZlYnJ1YXJ5IDE0LCAyMDI2IGF0IDEwOjMwOjA5IEFNIEdNVCs4XG5DdXJyZW50IHdvcmtpbmcgZGlyZWN0b3J5OiAvaG9tZS93ci9naC90cnlfd2ViX2dwdSIsInRvb2xzIjpbeyJuYW1lIjoicmVhZCIsImRlc2NyaXB0aW9uIjoiUmVhZCB0aGUgY29udGVudHMgb2YgYSBmaWxlLiBTdXBwb3J0cyB0ZXh0IGZpbGVzIGFuZCBpbWFnZXMgKGpwZywgcG5nLCBnaWYsIHdlYnApLiBJbWFnZXMgYXJlIHNlbnQgYXMgYXR0YWNobWVudHMuIEZvciB0ZXh0IGZpbGVzLCBvdXRwdXQgaXMgdHJ1bmNhdGVkIHRvIDIwMDAgbGluZXMgb3IgNTBLQiAod2hpY2hldmVyIGlzIGhpdCBmaXJzdCkuIFVzZSBvZmZzZXQvbGltaXQgZm9yIGxhcmdlIGZpbGVzLiBXaGVuIHlvdSBuZWVkIHRoZSBmdWxsIGZpbGUsIGNvbnRpbnVlIHdpdGggb2Zmc2V0IHVudGlsIGNvbXBsZXRlLiIsInBhcmFtZXRlcnMiOnsidHlwZSI6Im9iamVjdCIsInJlcXVpcmVkIjpbInBhdGgiXSwicHJvcGVydGllcyI6eyJwYXRoIjp7ImRlc2NyaXB0aW9uIjoiUGF0aCB0byB0aGUgZmlsZSB0byByZWFkIChyZWxhdGl2ZSBvciBhYnNvbHV0ZSkiLCJ0eXBlIjoic3RyaW5nIn0sIm9mZnNldCI6eyJkZXNjcmlwdGlvbiI6IkxpbmUgbnVtYmVyIHRvIHN0YXJ0IHJlYWRpbmcgZnJvbSAoMS1pbmRleGVkKSIsInR5cGUiOiJudW1iZXIifSwibGltaXQiOnsiZGVzY3JpcHRpb24iOiJNYXhpbXVtIG51bWJlciBvZiBsaW5lcyB0byByZWFkIiwidHlwZSI6Im51bWJlciJ9fX19LHsibmFtZSI6ImJhc2giLCJkZXNjcmlwdGlvbiI6IkV4ZWN1dGUgYSBiYXNoIGNvbW1hbmQgaW4gdGhlIGN1cnJlbnQgd29ya2luZyBkaXJlY3RvcnkuIFJldHVybnMgc3Rkb3V0IGFuZCBzdGRlcnIuIE91dHB1dCBpcyB0cnVuY2F0ZWQgdG8gbGFzdCAyMDAwIGxpbmVzIG9yIDUwS0IgKHdoaWNoZXZlciBpcyBoaXQgZmlyc3QpLiBJZiB0cnVuY2F0ZWQsIGZ1bGwgb3V0cHV0IGlzIHNhdmVkIHRvIGEgdGVtcCBmaWxlLiBPcHRpb25hbGx5IHByb3ZpZGUgYSB0aW1lb3V0IGluIHNlY29uZHMuIiwicGFyYW1ldGVycyI6eyJ0eXBlIjoib2JqZWN0IiwicmVxdWlyZWQiOlsiY29tbWFuZCJdLCJwcm9wZXJ0aWVzIjp7ImNvbW1hbmQiOnsiZGVzY3JpcHRpb24iOiJCYXNoIGNvbW1hbmQgdG8gZXhlY3V0ZSIsInR5cGUiOiJzdHJpbmcifSwidGltZW91dCI6eyJkZXNjcmlwdGlvbiI6IlRpbWVvdXQgaW4gc2Vjb25kcyAob3B0aW9uYWwsIG5vIGRlZmF1bHQgdGltZW91dCkiLCJ0eXBlIjoibnVtYmVyIn19fX0seyJuYW1lIjoiZWRpdCIsImRlc2NyaXB0aW9uIjoiRWRpdCBhIGZpbGUgYnkgcmVwbGFjaW5nIGV4YWN0IHRleHQuIFRoZSBvbGRUZXh0IG11c3QgbWF0Y2ggZXhhY3RseSAoaW5jbHVkaW5nIHdoaXRlc3BhY2UpLiBVc2UgdGhpcyBmb3IgcHJlY2lzZSwgc3VyZ2ljYWwgZWRpdHMuIiwicGFyYW1ldGVycyI6eyJ0eXBlIjoib2JqZWN0IiwicmVxdWlyZWQiOlsicGF0aCIsIm9sZFRleHQiLCJuZXdUZXh0Il0sInByb3BlcnRpZXMiOnsicGF0aCI6eyJkZXNjcmlwdGlvbiI6IlBhdGggdG8gdGhlIGZpbGUgdG8gZWRpdCAocmVsYXRpdmUgb3IgYWJzb2x1dGUpIiwidHlwZSI6InN0cmluZyJ9LCJvbGRUZXh0Ijp7ImRlc2NyaXB0aW9uIjoiRXhhY3QgdGV4dCB0byBmaW5kIGFuZCByZXBsYWNlIChtdXN0IG1hdGNoIGV4YWN0bHkpIiwidHlwZSI6InN0cmluZyJ9LCJuZXdUZXh0Ijp7ImRlc2NyaXB0aW9uIjoiTmV3IHRleHQgdG8gcmVwbGFjZSB0aGUgb2xkIHRleHQgd2l0aCIsInR5cGUiOiJzdHJpbmcifX19fSx7Im5hbWUiOiJ3cml0ZSIsImRlc2NyaXB0aW9uIjoiV3JpdGUgY29udGVudCB0byBhIGZpbGUuIENyZWF0ZXMgdGhlIGZpbGUgaWYgaXQgZG9lc24ndCBleGlzdCwgb3ZlcndyaXRlcyBpZiBpdCBkb2VzLiBBdXRvbWF0aWNhbGx5IGNyZWF0ZXMgcGFyZW50IGRpcmVjdG9yaWVzLiIsInBhcmFtZXRlcnMiOnsidHlwZSI6Im9iamVjdCIsInJlcXVpcmVkIjpbInBhdGgiLCJjb250ZW50Il0sInByb3BlcnRpZXMiOnsicGF0aCI6eyJkZXNjcmlwdGlvbiI6IlBhdGggdG8gdGhlIGZpbGUgdG8gd3JpdGUgKHJlbGF0aXZlIG9yIGFic29sdXRlKSIsInR5cGUiOiJzdHJpbmcifSwiY29udGVudCI6eyJkZXNjcmlwdGlvbiI6IkNvbnRlbnQgdG8gd3JpdGUgdG8gdGhlIGZpbGUiLCJ0eXBlIjoic3RyaW5nIn19fX1dfQ==</script> | |
| <!-- Vendored libraries --> | |
| <script>/** | |
| * marked v15.0.4 - a markdown parser | |
| * Copyright (c) 2011-2024, Christopher Jeffrey. (MIT Licensed) | |
| * https://github.com/markedjs/marked | |
| */ | |
| !function(e,t){"object"==typeof exports&&"undefined"!=typeof module?t(exports):"function"==typeof define&&define.amd?define(["exports"],t):t((e="undefined"!=typeof globalThis?globalThis:e||self).marked={})}(this,(function(e){"use strict";function t(){return{async:!1,breaks:!1,extensions:null,gfm:!0,hooks:null,pedantic:!1,renderer:null,silent:!1,tokenizer:null,walkTokens:null}}function n(t){e.defaults=t}e.defaults={async:!1,breaks:!1,extensions:null,gfm:!0,hooks:null,pedantic:!1,renderer:null,silent:!1,tokenizer:null,walkTokens:null};const s={exec:()=>null};function r(e,t=""){let n="string"==typeof e?e:e.source;const s={replace:(e,t)=>{let r="string"==typeof t?t:t.source;return r=r.replace(i.caret,"$1"),n=n.replace(e,r),s},getRegex:()=>new RegExp(n,t)};return s}const i={codeRemoveIndent:/^(?: {1,4}| {0,3}\t)/gm,outputLinkReplace:/\\([\[\]])/g,indentCodeCompensation:/^(\s+)(?:```)/,beginningSpace:/^\s+/,endingHash:/#$/,startingSpaceChar:/^ /,endingSpaceChar:/ $/,nonSpaceChar:/[^ ]/,newLineCharGlobal:/\n/g,tabCharGlobal:/\t/g,multipleSpaceGlobal:/\s+/g,blankLine:/^[ \t]*$/,doubleBlankLine:/\n[ \t]*\n[ \t]*$/,blockquoteStart:/^ {0,3}>/,blockquoteSetextReplace:/\n {0,3}((?:=+|-+) *)(?=\n|$)/g,blockquoteSetextReplace2:/^ {0,3}>[ \t]?/gm,listReplaceTabs:/^\t+/,listReplaceNesting:/^ {1,4}(?=( {4})*[^ ])/g,listIsTask:/^\[[ xX]\] /,listReplaceTask:/^\[[ xX]\] +/,anyLine:/\n.*\n/,hrefBrackets:/^<(.*)>$/,tableDelimiter:/[:|]/,tableAlignChars:/^\||\| *$/g,tableRowBlankLine:/\n[ \t]*$/,tableAlignRight:/^ *-+: *$/,tableAlignCenter:/^ *:-+: *$/,tableAlignLeft:/^ *:-+ *$/,startATag:/^<a /i,endATag:/^<\/a>/i,startPreScriptTag:/^<(pre|code|kbd|script)(\s|>)/i,endPreScriptTag:/^<\/(pre|code|kbd|script)(\s|>)/i,startAngleBracket:/^</,endAngleBracket:/>$/,pedanticHrefTitle:/^([^'"]*[^\s])\s+(['"])(.*)\2/,unicodeAlphaNumeric:/[\p{L}\p{N}]/u,escapeTest:/[&<>"']/,escapeReplace:/[&<>"']/g,escapeTestNoEncode:/[<>"']|&(?!(#\d{1,7}|#[Xx][a-fA-F0-9]{1,6}|\w+);)/,escapeReplaceNoEncode:/[<>"']|&(?!(#\d{1,7}|#[Xx][a-fA-F0-9]{1,6}|\w+);)/g,unescapeTest:/&(#(?:\d+)|(?:#x[0-9A-Fa-f]+)|(?:\w+));?/gi,caret:/(^|[^\[])\^/g,percentDecode:/%25/g,findPipe:/\|/g,splitPipe:/ \|/,slashPipe:/\\\|/g,carriageReturn:/\r\n|\r/g,spaceLine:/^ +$/gm,notSpaceStart:/^\S*/,endingNewline:/\n$/,listItemRegex:e=>new RegExp(`^( {0,3}${e})((?:[\t ][^\\n]*)?(?:\\n|$))`),nextBulletRegex:e=>new RegExp(`^ {0,${Math.min(3,e-1)}}(?:[*+-]|\\d{1,9}[.)])((?:[ \t][^\\n]*)?(?:\\n|$))`),hrRegex:e=>new RegExp(`^ {0,${Math.min(3,e-1)}}((?:- *){3,}|(?:_ *){3,}|(?:\\* *){3,})(?:\\n+|$)`),fencesBeginRegex:e=>new RegExp(`^ {0,${Math.min(3,e-1)}}(?:\`\`\`|~~~)`),headingBeginRegex:e=>new RegExp(`^ {0,${Math.min(3,e-1)}}#`),htmlBeginRegex:e=>new RegExp(`^ {0,${Math.min(3,e-1)}}<(?:[a-z].*>|!--)`,"i")},l=/^ {0,3}((?:-[\t ]*){3,}|(?:_[ \t]*){3,}|(?:\*[ \t]*){3,})(?:\n+|$)/,o=/(?:[*+-]|\d{1,9}[.)])/,a=r(/^(?!bull |blockCode|fences|blockquote|heading|html)((?:.|\n(?!\s*?\n|bull |blockCode|fences|blockquote|heading|html))+?)\n {0,3}(=+|-+) *(?:\n+|$)/).replace(/bull/g,o).replace(/blockCode/g,/(?: {4}| {0,3}\t)/).replace(/fences/g,/ {0,3}(?:`{3,}|~{3,})/).replace(/blockquote/g,/ {0,3}>/).replace(/heading/g,/ {0,3}#{1,6}/).replace(/html/g,/ {0,3}<[^\n>]+>\n/).getRegex(),c=/^([^\n]+(?:\n(?!hr|heading|lheading|blockquote|fences|list|html|table| +\n)[^\n]+)*)/,h=/(?!\s*\])(?:\\.|[^\[\]\\])+/,p=r(/^ {0,3}\[(label)\]: *(?:\n[ \t]*)?([^<\s][^\s]*|<.*?>)(?:(?: +(?:\n[ \t]*)?| *\n[ \t]*)(title))? *(?:\n+|$)/).replace("label",h).replace("title",/(?:"(?:\\"?|[^"\\])*"|'[^'\n]*(?:\n[^'\n]+)*\n?'|\([^()]*\))/).getRegex(),u=r(/^( {0,3}bull)([ \t][^\n]+?)?(?:\n|$)/).replace(/bull/g,o).getRegex(),g="address|article|aside|base|basefont|blockquote|body|caption|center|col|colgroup|dd|details|dialog|dir|div|dl|dt|fieldset|figcaption|figure|footer|form|frame|frameset|h[1-6]|head|header|hr|html|iframe|legend|li|link|main|menu|menuitem|meta|nav|noframes|ol|optgroup|option|p|param|search|section|summary|table|tbody|td|tfoot|th|thead|title|tr|track|ul",k=/<!--(?:-?>|[\s\S]*?(?:-->|$))/,f=r("^ {0,3}(?:<(script|pre|style|textarea)[\\s>][\\s\\S]*?(?:</\\1>[^\\n]*\\n+|$)|comment[^\\n]*(\\n+|$)|<\\?[\\s\\S]*?(?:\\?>\\n*|$)|<![A-Z][\\s\\S]*?(?:>\\n*|$)|<!\\[CDATA\\[[\\s\\S]*?(?:\\]\\]>\\n*|$)|</?(tag)(?: +|\\n|/?>)[\\s\\S]*?(?:(?:\\n[ \t]*)+\\n|$)|<(?!script|pre|style|textarea)([a-z][\\w-]*)(?:attribute)*? */?>(?=[ \\t]*(?:\\n|$))[\\s\\S]*?(?:(?:\\n[ \t]*)+\\n|$)|</(?!script|pre|style|textarea)[a-z][\\w-]*\\s*>(?=[ \\t]*(?:\\n|$))[\\s\\S]*?(?:(?:\\n[ \t]*)+\\n|$))","i").replace("comment",k).replace("tag",g).replace("attribute",/ +[a-zA-Z:_][\w.:-]*(?: *= *"[^"\n]*"| *= *'[^'\n]*'| *= *[^\s"'=<>`]+)?/).getRegex(),d=r(c).replace("hr",l).replace("heading"," {0,3}#{1,6}(?:\\s|$)").replace("|lheading","").replace("|table","").replace("blockquote"," {0,3}>").replace("fences"," {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n").replace("list"," {0,3}(?:[*+-]|1[.)]) ").replace("html","</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|textarea|!--)").replace("tag",g).getRegex(),x={blockquote:r(/^( {0,3}> ?(paragraph|[^\n]*)(?:\n|$))+/).replace("paragraph",d).getRegex(),code:/^((?: {4}| {0,3}\t)[^\n]+(?:\n(?:[ \t]*(?:\n|$))*)?)+/,def:p,fences:/^ {0,3}(`{3,}(?=[^`\n]*(?:\n|$))|~{3,})([^\n]*)(?:\n|$)(?:|([\s\S]*?)(?:\n|$))(?: {0,3}\1[~`]* *(?=\n|$)|$)/,heading:/^ {0,3}(#{1,6})(?=\s|$)(.*)(?:\n+|$)/,hr:l,html:f,lheading:a,list:u,newline:/^(?:[ \t]*(?:\n|$))+/,paragraph:d,table:s,text:/^[^\n]+/},b=r("^ *([^\\n ].*)\\n {0,3}((?:\\| *)?:?-+:? *(?:\\| *:?-+:? *)*(?:\\| *)?)(?:\\n((?:(?! *\\n|hr|heading|blockquote|code|fences|list|html).*(?:\\n|$))*)\\n*|$)").replace("hr",l).replace("heading"," {0,3}#{1,6}(?:\\s|$)").replace("blockquote"," {0,3}>").replace("code","(?: {4}| {0,3}\t)[^\\n]").replace("fences"," {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n").replace("list"," {0,3}(?:[*+-]|1[.)]) ").replace("html","</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|textarea|!--)").replace("tag",g).getRegex(),w={...x,table:b,paragraph:r(c).replace("hr",l).replace("heading"," {0,3}#{1,6}(?:\\s|$)").replace("|lheading","").replace("table",b).replace("blockquote"," {0,3}>").replace("fences"," {0,3}(?:`{3,}(?=[^`\\n]*\\n)|~{3,})[^\\n]*\\n").replace("list"," {0,3}(?:[*+-]|1[.)]) ").replace("html","</?(?:tag)(?: +|\\n|/?>)|<(?:script|pre|style|textarea|!--)").replace("tag",g).getRegex()},m={...x,html:r("^ *(?:comment *(?:\\n|\\s*$)|<(tag)[\\s\\S]+?</\\1> *(?:\\n{2,}|\\s*$)|<tag(?:\"[^\"]*\"|'[^']*'|\\s[^'\"/>\\s]*)*?/?> *(?:\\n{2,}|\\s*$))").replace("comment",k).replace(/tag/g,"(?!(?:a|em|strong|small|s|cite|q|dfn|abbr|data|time|code|var|samp|kbd|sub|sup|i|b|u|mark|ruby|rt|rp|bdi|bdo|span|br|wbr|ins|del|img)\\b)\\w+(?!:|[^\\w\\s@]*@)\\b").getRegex(),def:/^ *\[([^\]]+)\]: *<?([^\s>]+)>?(?: +(["(][^\n]+[")]))? *(?:\n+|$)/,heading:/^(#{1,6})(.*)(?:\n+|$)/,fences:s,lheading:/^(.+?)\n {0,3}(=+|-+) *(?:\n+|$)/,paragraph:r(c).replace("hr",l).replace("heading"," *#{1,6} *[^\n]").replace("lheading",a).replace("|table","").replace("blockquote"," {0,3}>").replace("|fences","").replace("|list","").replace("|html","").replace("|tag","").getRegex()},y=/^\\([!"#$%&'()*+,\-./:;<=>?@\[\]\\^_`{|}~])/,$=/^( {2,}|\\)\n(?!\s*$)/,R=/[\p{P}\p{S}]/u,S=/[\s\p{P}\p{S}]/u,T=/[^\s\p{P}\p{S}]/u,z=r(/^((?![*_])punctSpace)/,"u").replace(/punctSpace/g,S).getRegex(),A=r(/^(?:\*+(?:((?!\*)punct)|[^\s*]))|^_+(?:((?!_)punct)|([^\s_]))/,"u").replace(/punct/g,R).getRegex(),_=r("^[^_*]*?__[^_*]*?\\*[^_*]*?(?=__)|[^*]+(?=[^*])|(?!\\*)punct(\\*+)(?=[\\s]|$)|notPunctSpace(\\*+)(?!\\*)(?=punctSpace|$)|(?!\\*)punctSpace(\\*+)(?=notPunctSpace)|[\\s](\\*+)(?!\\*)(?=punct)|(?!\\*)punct(\\*+)(?!\\*)(?=punct)|notPunctSpace(\\*+)(?=notPunctSpace)","gu").replace(/notPunctSpace/g,T).replace(/punctSpace/g,S).replace(/punct/g,R).getRegex(),P=r("^[^_*]*?\\*\\*[^_*]*?_[^_*]*?(?=\\*\\*)|[^_]+(?=[^_])|(?!_)punct(_+)(?=[\\s]|$)|notPunctSpace(_+)(?!_)(?=punctSpace|$)|(?!_)punctSpace(_+)(?=notPunctSpace)|[\\s](_+)(?!_)(?=punct)|(?!_)punct(_+)(?!_)(?=punct)","gu").replace(/notPunctSpace/g,T).replace(/punctSpace/g,S).replace(/punct/g,R).getRegex(),I=r(/\\(punct)/,"gu").replace(/punct/g,R).getRegex(),L=r(/^<(scheme:[^\s\x00-\x1f<>]*|email)>/).replace("scheme",/[a-zA-Z][a-zA-Z0-9+.-]{1,31}/).replace("email",/[a-zA-Z0-9.!#$%&'*+/=?^_`{|}~-]+(@)[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?(?:\.[a-zA-Z0-9](?:[a-zA-Z0-9-]{0,61}[a-zA-Z0-9])?)+(?![-_])/).getRegex(),B=r(k).replace("(?:--\x3e|$)","--\x3e").getRegex(),C=r("^comment|^</[a-zA-Z][\\w:-]*\\s*>|^<[a-zA-Z][\\w-]*(?:attribute)*?\\s*/?>|^<\\?[\\s\\S]*?\\?>|^<![a-zA-Z]+\\s[\\s\\S]*?>|^<!\\[CDATA\\[[\\s\\S]*?\\]\\]>").replace("comment",B).replace("attribute",/\s+[a-zA-Z:_][\w.:-]*(?:\s*=\s*"[^"]*"|\s*=\s*'[^']*'|\s*=\s*[^\s"'=<>`]+)?/).getRegex(),E=/(?:\[(?:\\.|[^\[\]\\])*\]|\\.|`[^`]*`|[^\[\]\\`])*?/,q=r(/^!?\[(label)\]\(\s*(href)(?:\s+(title))?\s*\)/).replace("label",E).replace("href",/<(?:\\.|[^\n<>\\])+>|[^\s\x00-\x1f]*/).replace("title",/"(?:\\"?|[^"\\])*"|'(?:\\'?|[^'\\])*'|\((?:\\\)?|[^)\\])*\)/).getRegex(),Z=r(/^!?\[(label)\]\[(ref)\]/).replace("label",E).replace("ref",h).getRegex(),v=r(/^!?\[(ref)\](?:\[\])?/).replace("ref",h).getRegex(),D={_backpedal:s,anyPunctuation:I,autolink:L,blockSkip:/\[[^[\]]*?\]\((?:\\.|[^\\\(\)]|\((?:\\.|[^\\\(\)])*\))*\)|`[^`]*?`|<[^<>]*?>/g,br:$,code:/^(`+)([^`]|[^`][\s\S]*?[^`])\1(?!`)/,del:s,emStrongLDelim:A,emStrongRDelimAst:_,emStrongRDelimUnd:P,escape:y,link:q,nolink:v,punctuation:z,reflink:Z,reflinkSearch:r("reflink|nolink(?!\\()","g").replace("reflink",Z).replace("nolink",v).getRegex(),tag:C,text:/^(`+|[^`])(?:(?= {2,}\n)|[\s\S]*?(?:(?=[\\<!\[`*_]|\b_|$)|[^ ](?= {2,}\n)))/,url:s},M={...D,link:r(/^!?\[(label)\]\((.*?)\)/).replace("label",E).getRegex(),reflink:r(/^!?\[(label)\]\s*\[([^\]]*)\]/).replace("label",E).getRegex()},O={...D,escape:r(y).replace("])","~|])").getRegex(),url:r(/^((?:ftp|https?):\/\/|www\.)(?:[a-zA-Z0-9\-]+\.?)+[^\s<]*|^email/,"i").replace("email",/[A-Za-z0-9._+-]+(@)[a-zA-Z0-9-_]+(?:\.[a-zA-Z0-9-_]*[a-zA-Z0-9])+(?![-_])/).getRegex(),_backpedal:/(?:[^?!.,:;*_'"~()&]+|\([^)]*\)|&(?![a-zA-Z0-9]+;$)|[?!.,:;*_'"~)]+(?!$))+/,del:/^(~~?)(?=[^\s~])((?:\\.|[^\\])*?(?:\\.|[^\s~\\]))\1(?=[^~]|$)/,text:/^([`~]+|[^`~])(?:(?= {2,}\n)|(?=[a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-]+@)|[\s\S]*?(?:(?=[\\<!\[`*~_]|\b_|https?:\/\/|ftp:\/\/|www\.|$)|[^ ](?= {2,}\n)|[^a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-](?=[a-zA-Z0-9.!#$%&'*+\/=?_`{\|}~-]+@)))/},Q={...O,br:r($).replace("{2,}","*").getRegex(),text:r(O.text).replace("\\b_","\\b_| {2,}\\n").replace(/\{2,\}/g,"*").getRegex()},j={normal:x,gfm:w,pedantic:m},N={normal:D,gfm:O,breaks:Q,pedantic:M},G={"&":"&","<":"<",">":">",'"':""","'":"'"},H=e=>G[e];function X(e,t){if(t){if(i.escapeTest.test(e))return e.replace(i.escapeReplace,H)}else if(i.escapeTestNoEncode.test(e))return e.replace(i.escapeReplaceNoEncode,H);return e}function F(e){try{e=encodeURI(e).replace(i.percentDecode,"%")}catch{return null}return e}function U(e,t){const n=e.replace(i.findPipe,((e,t,n)=>{let s=!1,r=t;for(;--r>=0&&"\\"===n[r];)s=!s;return s?"|":" |"})).split(i.splitPipe);let s=0;if(n[0].trim()||n.shift(),n.length>0&&!n.at(-1)?.trim()&&n.pop(),t)if(n.length>t)n.splice(t);else for(;n.length<t;)n.push("");for(;s<n.length;s++)n[s]=n[s].trim().replace(i.slashPipe,"|");return n}function J(e,t,n){const s=e.length;if(0===s)return"";let r=0;for(;r<s;){const i=e.charAt(s-r-1);if(i!==t||n){if(i===t||!n)break;r++}else r++}return e.slice(0,s-r)}function K(e,t,n,s,r){const i=t.href,l=t.title||null,o=e[1].replace(r.other.outputLinkReplace,"$1");if("!"!==e[0].charAt(0)){s.state.inLink=!0;const e={type:"link",raw:n,href:i,title:l,text:o,tokens:s.inlineTokens(o)};return s.state.inLink=!1,e}return{type:"image",raw:n,href:i,title:l,text:o}}class V{options;rules;lexer;constructor(t){this.options=t||e.defaults}space(e){const t=this.rules.block.newline.exec(e);if(t&&t[0].length>0)return{type:"space",raw:t[0]}}code(e){const t=this.rules.block.code.exec(e);if(t){const e=t[0].replace(this.rules.other.codeRemoveIndent,"");return{type:"code",raw:t[0],codeBlockStyle:"indented",text:this.options.pedantic?e:J(e,"\n")}}}fences(e){const t=this.rules.block.fences.exec(e);if(t){const e=t[0],n=function(e,t,n){const s=e.match(n.other.indentCodeCompensation);if(null===s)return t;const r=s[1];return t.split("\n").map((e=>{const t=e.match(n.other.beginningSpace);if(null===t)return e;const[s]=t;return s.length>=r.length?e.slice(r.length):e})).join("\n")}(e,t[3]||"",this.rules);return{type:"code",raw:e,lang:t[2]?t[2].trim().replace(this.rules.inline.anyPunctuation,"$1"):t[2],text:n}}}heading(e){const t=this.rules.block.heading.exec(e);if(t){let e=t[2].trim();if(this.rules.other.endingHash.test(e)){const t=J(e,"#");this.options.pedantic?e=t.trim():t&&!this.rules.other.endingSpaceChar.test(t)||(e=t.trim())}return{type:"heading",raw:t[0],depth:t[1].length,text:e,tokens:this.lexer.inline(e)}}}hr(e){const t=this.rules.block.hr.exec(e);if(t)return{type:"hr",raw:J(t[0],"\n")}}blockquote(e){const t=this.rules.block.blockquote.exec(e);if(t){let e=J(t[0],"\n").split("\n"),n="",s="";const r=[];for(;e.length>0;){let t=!1;const i=[];let l;for(l=0;l<e.length;l++)if(this.rules.other.blockquoteStart.test(e[l]))i.push(e[l]),t=!0;else{if(t)break;i.push(e[l])}e=e.slice(l);const o=i.join("\n"),a=o.replace(this.rules.other.blockquoteSetextReplace,"\n $1").replace(this.rules.other.blockquoteSetextReplace2,"");n=n?`${n}\n${o}`:o,s=s?`${s}\n${a}`:a;const c=this.lexer.state.top;if(this.lexer.state.top=!0,this.lexer.blockTokens(a,r,!0),this.lexer.state.top=c,0===e.length)break;const h=r.at(-1);if("code"===h?.type)break;if("blockquote"===h?.type){const t=h,i=t.raw+"\n"+e.join("\n"),l=this.blockquote(i);r[r.length-1]=l,n=n.substring(0,n.length-t.raw.length)+l.raw,s=s.substring(0,s.length-t.text.length)+l.text;break}if("list"!==h?.type);else{const t=h,i=t.raw+"\n"+e.join("\n"),l=this.list(i);r[r.length-1]=l,n=n.substring(0,n.length-h.raw.length)+l.raw,s=s.substring(0,s.length-t.raw.length)+l.raw,e=i.substring(r.at(-1).raw.length).split("\n")}}return{type:"blockquote",raw:n,tokens:r,text:s}}}list(e){let t=this.rules.block.list.exec(e);if(t){let n=t[1].trim();const s=n.length>1,r={type:"list",raw:"",ordered:s,start:s?+n.slice(0,-1):"",loose:!1,items:[]};n=s?`\\d{1,9}\\${n.slice(-1)}`:`\\${n}`,this.options.pedantic&&(n=s?n:"[*+-]");const i=this.rules.other.listItemRegex(n);let l=!1;for(;e;){let n=!1,s="",o="";if(!(t=i.exec(e)))break;if(this.rules.block.hr.test(e))break;s=t[0],e=e.substring(s.length);let a=t[2].split("\n",1)[0].replace(this.rules.other.listReplaceTabs,(e=>" ".repeat(3*e.length))),c=e.split("\n",1)[0],h=!a.trim(),p=0;if(this.options.pedantic?(p=2,o=a.trimStart()):h?p=t[1].length+1:(p=t[2].search(this.rules.other.nonSpaceChar),p=p>4?1:p,o=a.slice(p),p+=t[1].length),h&&this.rules.other.blankLine.test(c)&&(s+=c+"\n",e=e.substring(c.length+1),n=!0),!n){const t=this.rules.other.nextBulletRegex(p),n=this.rules.other.hrRegex(p),r=this.rules.other.fencesBeginRegex(p),i=this.rules.other.headingBeginRegex(p),l=this.rules.other.htmlBeginRegex(p);for(;e;){const u=e.split("\n",1)[0];let g;if(c=u,this.options.pedantic?(c=c.replace(this.rules.other.listReplaceNesting," "),g=c):g=c.replace(this.rules.other.tabCharGlobal," "),r.test(c))break;if(i.test(c))break;if(l.test(c))break;if(t.test(c))break;if(n.test(c))break;if(g.search(this.rules.other.nonSpaceChar)>=p||!c.trim())o+="\n"+g.slice(p);else{if(h)break;if(a.replace(this.rules.other.tabCharGlobal," ").search(this.rules.other.nonSpaceChar)>=4)break;if(r.test(a))break;if(i.test(a))break;if(n.test(a))break;o+="\n"+c}h||c.trim()||(h=!0),s+=u+"\n",e=e.substring(u.length+1),a=g.slice(p)}}r.loose||(l?r.loose=!0:this.rules.other.doubleBlankLine.test(s)&&(l=!0));let u,g=null;this.options.gfm&&(g=this.rules.other.listIsTask.exec(o),g&&(u="[ ] "!==g[0],o=o.replace(this.rules.other.listReplaceTask,""))),r.items.push({type:"list_item",raw:s,task:!!g,checked:u,loose:!1,text:o,tokens:[]}),r.raw+=s}const o=r.items.at(-1);if(!o)return;o.raw=o.raw.trimEnd(),o.text=o.text.trimEnd(),r.raw=r.raw.trimEnd();for(let e=0;e<r.items.length;e++)if(this.lexer.state.top=!1,r.items[e].tokens=this.lexer.blockTokens(r.items[e].text,[]),!r.loose){const t=r.items[e].tokens.filter((e=>"space"===e.type)),n=t.length>0&&t.some((e=>this.rules.other.anyLine.test(e.raw)));r.loose=n}if(r.loose)for(let e=0;e<r.items.length;e++)r.items[e].loose=!0;return r}}html(e){const t=this.rules.block.html.exec(e);if(t){return{type:"html",block:!0,raw:t[0],pre:"pre"===t[1]||"script"===t[1]||"style"===t[1],text:t[0]}}}def(e){const t=this.rules.block.def.exec(e);if(t){const e=t[1].toLowerCase().replace(this.rules.other.multipleSpaceGlobal," "),n=t[2]?t[2].replace(this.rules.other.hrefBrackets,"$1").replace(this.rules.inline.anyPunctuation,"$1"):"",s=t[3]?t[3].substring(1,t[3].length-1).replace(this.rules.inline.anyPunctuation,"$1"):t[3];return{type:"def",tag:e,raw:t[0],href:n,title:s}}}table(e){const t=this.rules.block.table.exec(e);if(!t)return;if(!this.rules.other.tableDelimiter.test(t[2]))return;const n=U(t[1]),s=t[2].replace(this.rules.other.tableAlignChars,"").split("|"),r=t[3]?.trim()?t[3].replace(this.rules.other.tableRowBlankLine,"").split("\n"):[],i={type:"table",raw:t[0],header:[],align:[],rows:[]};if(n.length===s.length){for(const e of s)this.rules.other.tableAlignRight.test(e)?i.align.push("right"):this.rules.other.tableAlignCenter.test(e)?i.align.push("center"):this.rules.other.tableAlignLeft.test(e)?i.align.push("left"):i.align.push(null);for(let e=0;e<n.length;e++)i.header.push({text:n[e],tokens:this.lexer.inline(n[e]),header:!0,align:i.align[e]});for(const e of r)i.rows.push(U(e,i.header.length).map(((e,t)=>({text:e,tokens:this.lexer.inline(e),header:!1,align:i.align[t]}))));return i}}lheading(e){const t=this.rules.block.lheading.exec(e);if(t)return{type:"heading",raw:t[0],depth:"="===t[2].charAt(0)?1:2,text:t[1],tokens:this.lexer.inline(t[1])}}paragraph(e){const t=this.rules.block.paragraph.exec(e);if(t){const e="\n"===t[1].charAt(t[1].length-1)?t[1].slice(0,-1):t[1];return{type:"paragraph",raw:t[0],text:e,tokens:this.lexer.inline(e)}}}text(e){const t=this.rules.block.text.exec(e);if(t)return{type:"text",raw:t[0],text:t[0],tokens:this.lexer.inline(t[0])}}escape(e){const t=this.rules.inline.escape.exec(e);if(t)return{type:"escape",raw:t[0],text:t[1]}}tag(e){const t=this.rules.inline.tag.exec(e);if(t)return!this.lexer.state.inLink&&this.rules.other.startATag.test(t[0])?this.lexer.state.inLink=!0:this.lexer.state.inLink&&this.rules.other.endATag.test(t[0])&&(this.lexer.state.inLink=!1),!this.lexer.state.inRawBlock&&this.rules.other.startPreScriptTag.test(t[0])?this.lexer.state.inRawBlock=!0:this.lexer.state.inRawBlock&&this.rules.other.endPreScriptTag.test(t[0])&&(this.lexer.state.inRawBlock=!1),{type:"html",raw:t[0],inLink:this.lexer.state.inLink,inRawBlock:this.lexer.state.inRawBlock,block:!1,text:t[0]}}link(e){const t=this.rules.inline.link.exec(e);if(t){const e=t[2].trim();if(!this.options.pedantic&&this.rules.other.startAngleBracket.test(e)){if(!this.rules.other.endAngleBracket.test(e))return;const t=J(e.slice(0,-1),"\\");if((e.length-t.length)%2==0)return}else{const e=function(e,t){if(-1===e.indexOf(t[1]))return-1;let n=0;for(let s=0;s<e.length;s++)if("\\"===e[s])s++;else if(e[s]===t[0])n++;else if(e[s]===t[1]&&(n--,n<0))return s;return-1}(t[2],"()");if(e>-1){const n=(0===t[0].indexOf("!")?5:4)+t[1].length+e;t[2]=t[2].substring(0,e),t[0]=t[0].substring(0,n).trim(),t[3]=""}}let n=t[2],s="";if(this.options.pedantic){const e=this.rules.other.pedanticHrefTitle.exec(n);e&&(n=e[1],s=e[3])}else s=t[3]?t[3].slice(1,-1):"";return n=n.trim(),this.rules.other.startAngleBracket.test(n)&&(n=this.options.pedantic&&!this.rules.other.endAngleBracket.test(e)?n.slice(1):n.slice(1,-1)),K(t,{href:n?n.replace(this.rules.inline.anyPunctuation,"$1"):n,title:s?s.replace(this.rules.inline.anyPunctuation,"$1"):s},t[0],this.lexer,this.rules)}}reflink(e,t){let n;if((n=this.rules.inline.reflink.exec(e))||(n=this.rules.inline.nolink.exec(e))){const e=t[(n[2]||n[1]).replace(this.rules.other.multipleSpaceGlobal," ").toLowerCase()];if(!e){const e=n[0].charAt(0);return{type:"text",raw:e,text:e}}return K(n,e,n[0],this.lexer,this.rules)}}emStrong(e,t,n=""){let s=this.rules.inline.emStrongLDelim.exec(e);if(!s)return;if(s[3]&&n.match(this.rules.other.unicodeAlphaNumeric))return;if(!(s[1]||s[2]||"")||!n||this.rules.inline.punctuation.exec(n)){const n=[...s[0]].length-1;let r,i,l=n,o=0;const a="*"===s[0][0]?this.rules.inline.emStrongRDelimAst:this.rules.inline.emStrongRDelimUnd;for(a.lastIndex=0,t=t.slice(-1*e.length+n);null!=(s=a.exec(t));){if(r=s[1]||s[2]||s[3]||s[4]||s[5]||s[6],!r)continue;if(i=[...r].length,s[3]||s[4]){l+=i;continue}if((s[5]||s[6])&&n%3&&!((n+i)%3)){o+=i;continue}if(l-=i,l>0)continue;i=Math.min(i,i+l+o);const t=[...s[0]][0].length,a=e.slice(0,n+s.index+t+i);if(Math.min(n,i)%2){const e=a.slice(1,-1);return{type:"em",raw:a,text:e,tokens:this.lexer.inlineTokens(e)}}const c=a.slice(2,-2);return{type:"strong",raw:a,text:c,tokens:this.lexer.inlineTokens(c)}}}}codespan(e){const t=this.rules.inline.code.exec(e);if(t){let e=t[2].replace(this.rules.other.newLineCharGlobal," ");const n=this.rules.other.nonSpaceChar.test(e),s=this.rules.other.startingSpaceChar.test(e)&&this.rules.other.endingSpaceChar.test(e);return n&&s&&(e=e.substring(1,e.length-1)),{type:"codespan",raw:t[0],text:e}}}br(e){const t=this.rules.inline.br.exec(e);if(t)return{type:"br",raw:t[0]}}del(e){const t=this.rules.inline.del.exec(e);if(t)return{type:"del",raw:t[0],text:t[2],tokens:this.lexer.inlineTokens(t[2])}}autolink(e){const t=this.rules.inline.autolink.exec(e);if(t){let e,n;return"@"===t[2]?(e=t[1],n="mailto:"+e):(e=t[1],n=e),{type:"link",raw:t[0],text:e,href:n,tokens:[{type:"text",raw:e,text:e}]}}}url(e){let t;if(t=this.rules.inline.url.exec(e)){let e,n;if("@"===t[2])e=t[0],n="mailto:"+e;else{let s;do{s=t[0],t[0]=this.rules.inline._backpedal.exec(t[0])?.[0]??""}while(s!==t[0]);e=t[0],n="www."===t[1]?"http://"+t[0]:t[0]}return{type:"link",raw:t[0],text:e,href:n,tokens:[{type:"text",raw:e,text:e}]}}}inlineText(e){const t=this.rules.inline.text.exec(e);if(t){const e=this.lexer.state.inRawBlock;return{type:"text",raw:t[0],text:t[0],escaped:e}}}}class W{tokens;options;state;tokenizer;inlineQueue;constructor(t){this.tokens=[],this.tokens.links=Object.create(null),this.options=t||e.defaults,this.options.tokenizer=this.options.tokenizer||new V,this.tokenizer=this.options.tokenizer,this.tokenizer.options=this.options,this.tokenizer.lexer=this,this.inlineQueue=[],this.state={inLink:!1,inRawBlock:!1,top:!0};const n={other:i,block:j.normal,inline:N.normal};this.options.pedantic?(n.block=j.pedantic,n.inline=N.pedantic):this.options.gfm&&(n.block=j.gfm,this.options.breaks?n.inline=N.breaks:n.inline=N.gfm),this.tokenizer.rules=n}static get rules(){return{block:j,inline:N}}static lex(e,t){return new W(t).lex(e)}static lexInline(e,t){return new W(t).inlineTokens(e)}lex(e){e=e.replace(i.carriageReturn,"\n"),this.blockTokens(e,this.tokens);for(let e=0;e<this.inlineQueue.length;e++){const t=this.inlineQueue[e];this.inlineTokens(t.src,t.tokens)}return this.inlineQueue=[],this.tokens}blockTokens(e,t=[],n=!1){for(this.options.pedantic&&(e=e.replace(i.tabCharGlobal," ").replace(i.spaceLine,""));e;){let s;if(this.options.extensions?.block?.some((n=>!!(s=n.call({lexer:this},e,t))&&(e=e.substring(s.raw.length),t.push(s),!0))))continue;if(s=this.tokenizer.space(e)){e=e.substring(s.raw.length);const n=t.at(-1);1===s.raw.length&&void 0!==n?n.raw+="\n":t.push(s);continue}if(s=this.tokenizer.code(e)){e=e.substring(s.raw.length);const n=t.at(-1);"paragraph"===n?.type||"text"===n?.type?(n.raw+="\n"+s.raw,n.text+="\n"+s.text,this.inlineQueue.at(-1).src=n.text):t.push(s);continue}if(s=this.tokenizer.fences(e)){e=e.substring(s.raw.length),t.push(s);continue}if(s=this.tokenizer.heading(e)){e=e.substring(s.raw.length),t.push(s);continue}if(s=this.tokenizer.hr(e)){e=e.substring(s.raw.length),t.push(s);continue}if(s=this.tokenizer.blockquote(e)){e=e.substring(s.raw.length),t.push(s);continue}if(s=this.tokenizer.list(e)){e=e.substring(s.raw.length),t.push(s);continue}if(s=this.tokenizer.html(e)){e=e.substring(s.raw.length),t.push(s);continue}if(s=this.tokenizer.def(e)){e=e.substring(s.raw.length);const n=t.at(-1);"paragraph"===n?.type||"text"===n?.type?(n.raw+="\n"+s.raw,n.text+="\n"+s.raw,this.inlineQueue.at(-1).src=n.text):this.tokens.links[s.tag]||(this.tokens.links[s.tag]={href:s.href,title:s.title});continue}if(s=this.tokenizer.table(e)){e=e.substring(s.raw.length),t.push(s);continue}if(s=this.tokenizer.lheading(e)){e=e.substring(s.raw.length),t.push(s);continue}let r=e;if(this.options.extensions?.startBlock){let t=1/0;const n=e.slice(1);let s;this.options.extensions.startBlock.forEach((e=>{s=e.call({lexer:this},n),"number"==typeof s&&s>=0&&(t=Math.min(t,s))})),t<1/0&&t>=0&&(r=e.substring(0,t+1))}if(this.state.top&&(s=this.tokenizer.paragraph(r))){const i=t.at(-1);n&&"paragraph"===i?.type?(i.raw+="\n"+s.raw,i.text+="\n"+s.text,this.inlineQueue.pop(),this.inlineQueue.at(-1).src=i.text):t.push(s),n=r.length!==e.length,e=e.substring(s.raw.length)}else if(s=this.tokenizer.text(e)){e=e.substring(s.raw.length);const n=t.at(-1);"text"===n?.type?(n.raw+="\n"+s.raw,n.text+="\n"+s.text,this.inlineQueue.pop(),this.inlineQueue.at(-1).src=n.text):t.push(s)}else if(e){const t="Infinite loop on byte: "+e.charCodeAt(0);if(this.options.silent){console.error(t);break}throw new Error(t)}}return this.state.top=!0,t}inline(e,t=[]){return this.inlineQueue.push({src:e,tokens:t}),t}inlineTokens(e,t=[]){let n=e,s=null;if(this.tokens.links){const e=Object.keys(this.tokens.links);if(e.length>0)for(;null!=(s=this.tokenizer.rules.inline.reflinkSearch.exec(n));)e.includes(s[0].slice(s[0].lastIndexOf("[")+1,-1))&&(n=n.slice(0,s.index)+"["+"a".repeat(s[0].length-2)+"]"+n.slice(this.tokenizer.rules.inline.reflinkSearch.lastIndex))}for(;null!=(s=this.tokenizer.rules.inline.blockSkip.exec(n));)n=n.slice(0,s.index)+"["+"a".repeat(s[0].length-2)+"]"+n.slice(this.tokenizer.rules.inline.blockSkip.lastIndex);for(;null!=(s=this.tokenizer.rules.inline.anyPunctuation.exec(n));)n=n.slice(0,s.index)+"++"+n.slice(this.tokenizer.rules.inline.anyPunctuation.lastIndex);let r=!1,i="";for(;e;){let s;if(r||(i=""),r=!1,this.options.extensions?.inline?.some((n=>!!(s=n.call({lexer:this},e,t))&&(e=e.substring(s.raw.length),t.push(s),!0))))continue;if(s=this.tokenizer.escape(e)){e=e.substring(s.raw.length),t.push(s);continue}if(s=this.tokenizer.tag(e)){e=e.substring(s.raw.length),t.push(s);continue}if(s=this.tokenizer.link(e)){e=e.substring(s.raw.length),t.push(s);continue}if(s=this.tokenizer.reflink(e,this.tokens.links)){e=e.substring(s.raw.length);const n=t.at(-1);"text"===s.type&&"text"===n?.type?(n.raw+=s.raw,n.text+=s.text):t.push(s);continue}if(s=this.tokenizer.emStrong(e,n,i)){e=e.substring(s.raw.length),t.push(s);continue}if(s=this.tokenizer.codespan(e)){e=e.substring(s.raw.length),t.push(s);continue}if(s=this.tokenizer.br(e)){e=e.substring(s.raw.length),t.push(s);continue}if(s=this.tokenizer.del(e)){e=e.substring(s.raw.length),t.push(s);continue}if(s=this.tokenizer.autolink(e)){e=e.substring(s.raw.length),t.push(s);continue}if(!this.state.inLink&&(s=this.tokenizer.url(e))){e=e.substring(s.raw.length),t.push(s);continue}let l=e;if(this.options.extensions?.startInline){let t=1/0;const n=e.slice(1);let s;this.options.extensions.startInline.forEach((e=>{s=e.call({lexer:this},n),"number"==typeof s&&s>=0&&(t=Math.min(t,s))})),t<1/0&&t>=0&&(l=e.substring(0,t+1))}if(s=this.tokenizer.inlineText(l)){e=e.substring(s.raw.length),"_"!==s.raw.slice(-1)&&(i=s.raw.slice(-1)),r=!0;const n=t.at(-1);"text"===n?.type?(n.raw+=s.raw,n.text+=s.text):t.push(s)}else if(e){const t="Infinite loop on byte: "+e.charCodeAt(0);if(this.options.silent){console.error(t);break}throw new Error(t)}}return t}}class Y{options;parser;constructor(t){this.options=t||e.defaults}space(e){return""}code({text:e,lang:t,escaped:n}){const s=(t||"").match(i.notSpaceStart)?.[0],r=e.replace(i.endingNewline,"")+"\n";return s?'<pre><code class="language-'+X(s)+'">'+(n?r:X(r,!0))+"</code></pre>\n":"<pre><code>"+(n?r:X(r,!0))+"</code></pre>\n"}blockquote({tokens:e}){return`<blockquote>\n${this.parser.parse(e)}</blockquote>\n`}html({text:e}){return e}heading({tokens:e,depth:t}){return`<h${t}>${this.parser.parseInline(e)}</h${t}>\n`}hr(e){return"<hr>\n"}list(e){const t=e.ordered,n=e.start;let s="";for(let t=0;t<e.items.length;t++){const n=e.items[t];s+=this.listitem(n)}const r=t?"ol":"ul";return"<"+r+(t&&1!==n?' start="'+n+'"':"")+">\n"+s+"</"+r+">\n"}listitem(e){let t="";if(e.task){const n=this.checkbox({checked:!!e.checked});e.loose?"paragraph"===e.tokens[0]?.type?(e.tokens[0].text=n+" "+e.tokens[0].text,e.tokens[0].tokens&&e.tokens[0].tokens.length>0&&"text"===e.tokens[0].tokens[0].type&&(e.tokens[0].tokens[0].text=n+" "+X(e.tokens[0].tokens[0].text),e.tokens[0].tokens[0].escaped=!0)):e.tokens.unshift({type:"text",raw:n+" ",text:n+" ",escaped:!0}):t+=n+" "}return t+=this.parser.parse(e.tokens,!!e.loose),`<li>${t}</li>\n`}checkbox({checked:e}){return"<input "+(e?'checked="" ':"")+'disabled="" type="checkbox">'}paragraph({tokens:e}){return`<p>${this.parser.parseInline(e)}</p>\n`}table(e){let t="",n="";for(let t=0;t<e.header.length;t++)n+=this.tablecell(e.header[t]);t+=this.tablerow({text:n});let s="";for(let t=0;t<e.rows.length;t++){const r=e.rows[t];n="";for(let e=0;e<r.length;e++)n+=this.tablecell(r[e]);s+=this.tablerow({text:n})}return s&&(s=`<tbody>${s}</tbody>`),"<table>\n<thead>\n"+t+"</thead>\n"+s+"</table>\n"}tablerow({text:e}){return`<tr>\n${e}</tr>\n`}tablecell(e){const t=this.parser.parseInline(e.tokens),n=e.header?"th":"td";return(e.align?`<${n} align="${e.align}">`:`<${n}>`)+t+`</${n}>\n`}strong({tokens:e}){return`<strong>${this.parser.parseInline(e)}</strong>`}em({tokens:e}){return`<em>${this.parser.parseInline(e)}</em>`}codespan({text:e}){return`<code>${X(e,!0)}</code>`}br(e){return"<br>"}del({tokens:e}){return`<del>${this.parser.parseInline(e)}</del>`}link({href:e,title:t,tokens:n}){const s=this.parser.parseInline(n),r=F(e);if(null===r)return s;let i='<a href="'+(e=r)+'"';return t&&(i+=' title="'+X(t)+'"'),i+=">"+s+"</a>",i}image({href:e,title:t,text:n}){const s=F(e);if(null===s)return X(n);let r=`<img src="${e=s}" alt="${n}"`;return t&&(r+=` title="${X(t)}"`),r+=">",r}text(e){return"tokens"in e&&e.tokens?this.parser.parseInline(e.tokens):"escaped"in e&&e.escaped?e.text:X(e.text)}}class ee{strong({text:e}){return e}em({text:e}){return e}codespan({text:e}){return e}del({text:e}){return e}html({text:e}){return e}text({text:e}){return e}link({text:e}){return""+e}image({text:e}){return""+e}br(){return""}}class te{options;renderer;textRenderer;constructor(t){this.options=t||e.defaults,this.options.renderer=this.options.renderer||new Y,this.renderer=this.options.renderer,this.renderer.options=this.options,this.renderer.parser=this,this.textRenderer=new ee}static parse(e,t){return new te(t).parse(e)}static parseInline(e,t){return new te(t).parseInline(e)}parse(e,t=!0){let n="";for(let s=0;s<e.length;s++){const r=e[s];if(this.options.extensions?.renderers?.[r.type]){const e=r,t=this.options.extensions.renderers[e.type].call({parser:this},e);if(!1!==t||!["space","hr","heading","code","table","blockquote","list","html","paragraph","text"].includes(e.type)){n+=t||"";continue}}const i=r;switch(i.type){case"space":n+=this.renderer.space(i);continue;case"hr":n+=this.renderer.hr(i);continue;case"heading":n+=this.renderer.heading(i);continue;case"code":n+=this.renderer.code(i);continue;case"table":n+=this.renderer.table(i);continue;case"blockquote":n+=this.renderer.blockquote(i);continue;case"list":n+=this.renderer.list(i);continue;case"html":n+=this.renderer.html(i);continue;case"paragraph":n+=this.renderer.paragraph(i);continue;case"text":{let r=i,l=this.renderer.text(r);for(;s+1<e.length&&"text"===e[s+1].type;)r=e[++s],l+="\n"+this.renderer.text(r);n+=t?this.renderer.paragraph({type:"paragraph",raw:l,text:l,tokens:[{type:"text",raw:l,text:l,escaped:!0}]}):l;continue}default:{const e='Token with "'+i.type+'" type was not found.';if(this.options.silent)return console.error(e),"";throw new Error(e)}}}return n}parseInline(e,t=this.renderer){let n="";for(let s=0;s<e.length;s++){const r=e[s];if(this.options.extensions?.renderers?.[r.type]){const e=this.options.extensions.renderers[r.type].call({parser:this},r);if(!1!==e||!["escape","html","link","image","strong","em","codespan","br","del","text"].includes(r.type)){n+=e||"";continue}}const i=r;switch(i.type){case"escape":case"text":n+=t.text(i);break;case"html":n+=t.html(i);break;case"link":n+=t.link(i);break;case"image":n+=t.image(i);break;case"strong":n+=t.strong(i);break;case"em":n+=t.em(i);break;case"codespan":n+=t.codespan(i);break;case"br":n+=t.br(i);break;case"del":n+=t.del(i);break;default:{const e='Token with "'+i.type+'" type was not found.';if(this.options.silent)return console.error(e),"";throw new Error(e)}}}return n}}class ne{options;block;constructor(t){this.options=t||e.defaults}static passThroughHooks=new Set(["preprocess","postprocess","processAllTokens"]);preprocess(e){return e}postprocess(e){return e}processAllTokens(e){return e}provideLexer(){return this.block?W.lex:W.lexInline}provideParser(){return this.block?te.parse:te.parseInline}}class se{defaults={async:!1,breaks:!1,extensions:null,gfm:!0,hooks:null,pedantic:!1,renderer:null,silent:!1,tokenizer:null,walkTokens:null};options=this.setOptions;parse=this.parseMarkdown(!0);parseInline=this.parseMarkdown(!1);Parser=te;Renderer=Y;TextRenderer=ee;Lexer=W;Tokenizer=V;Hooks=ne;constructor(...e){this.use(...e)}walkTokens(e,t){let n=[];for(const s of e)switch(n=n.concat(t.call(this,s)),s.type){case"table":{const e=s;for(const s of e.header)n=n.concat(this.walkTokens(s.tokens,t));for(const s of e.rows)for(const e of s)n=n.concat(this.walkTokens(e.tokens,t));break}case"list":{const e=s;n=n.concat(this.walkTokens(e.items,t));break}default:{const e=s;this.defaults.extensions?.childTokens?.[e.type]?this.defaults.extensions.childTokens[e.type].forEach((s=>{const r=e[s].flat(1/0);n=n.concat(this.walkTokens(r,t))})):e.tokens&&(n=n.concat(this.walkTokens(e.tokens,t)))}}return n}use(...e){const t=this.defaults.extensions||{renderers:{},childTokens:{}};return e.forEach((e=>{const n={...e};if(n.async=this.defaults.async||n.async||!1,e.extensions&&(e.extensions.forEach((e=>{if(!e.name)throw new Error("extension name required");if("renderer"in e){const n=t.renderers[e.name];t.renderers[e.name]=n?function(...t){let s=e.renderer.apply(this,t);return!1===s&&(s=n.apply(this,t)),s}:e.renderer}if("tokenizer"in e){if(!e.level||"block"!==e.level&&"inline"!==e.level)throw new Error("extension level must be 'block' or 'inline'");const n=t[e.level];n?n.unshift(e.tokenizer):t[e.level]=[e.tokenizer],e.start&&("block"===e.level?t.startBlock?t.startBlock.push(e.start):t.startBlock=[e.start]:"inline"===e.level&&(t.startInline?t.startInline.push(e.start):t.startInline=[e.start]))}"childTokens"in e&&e.childTokens&&(t.childTokens[e.name]=e.childTokens)})),n.extensions=t),e.renderer){const t=this.defaults.renderer||new Y(this.defaults);for(const n in e.renderer){if(!(n in t))throw new Error(`renderer '${n}' does not exist`);if(["options","parser"].includes(n))continue;const s=n,r=e.renderer[s],i=t[s];t[s]=(...e)=>{let n=r.apply(t,e);return!1===n&&(n=i.apply(t,e)),n||""}}n.renderer=t}if(e.tokenizer){const t=this.defaults.tokenizer||new V(this.defaults);for(const n in e.tokenizer){if(!(n in t))throw new Error(`tokenizer '${n}' does not exist`);if(["options","rules","lexer"].includes(n))continue;const s=n,r=e.tokenizer[s],i=t[s];t[s]=(...e)=>{let n=r.apply(t,e);return!1===n&&(n=i.apply(t,e)),n}}n.tokenizer=t}if(e.hooks){const t=this.defaults.hooks||new ne;for(const n in e.hooks){if(!(n in t))throw new Error(`hook '${n}' does not exist`);if(["options","block"].includes(n))continue;const s=n,r=e.hooks[s],i=t[s];ne.passThroughHooks.has(n)?t[s]=e=>{if(this.defaults.async)return Promise.resolve(r.call(t,e)).then((e=>i.call(t,e)));const n=r.call(t,e);return i.call(t,n)}:t[s]=(...e)=>{let n=r.apply(t,e);return!1===n&&(n=i.apply(t,e)),n}}n.hooks=t}if(e.walkTokens){const t=this.defaults.walkTokens,s=e.walkTokens;n.walkTokens=function(e){let n=[];return n.push(s.call(this,e)),t&&(n=n.concat(t.call(this,e))),n}}this.defaults={...this.defaults,...n}})),this}setOptions(e){return this.defaults={...this.defaults,...e},this}lexer(e,t){return W.lex(e,t??this.defaults)}parser(e,t){return te.parse(e,t??this.defaults)}parseMarkdown(e){return(t,n)=>{const s={...n},r={...this.defaults,...s},i=this.onError(!!r.silent,!!r.async);if(!0===this.defaults.async&&!1===s.async)return i(new Error("marked(): The async option was set to true by an extension. Remove async: false from the parse options object to return a Promise."));if(null==t)return i(new Error("marked(): input parameter is undefined or null"));if("string"!=typeof t)return i(new Error("marked(): input parameter is of type "+Object.prototype.toString.call(t)+", string expected"));r.hooks&&(r.hooks.options=r,r.hooks.block=e);const l=r.hooks?r.hooks.provideLexer():e?W.lex:W.lexInline,o=r.hooks?r.hooks.provideParser():e?te.parse:te.parseInline;if(r.async)return Promise.resolve(r.hooks?r.hooks.preprocess(t):t).then((e=>l(e,r))).then((e=>r.hooks?r.hooks.processAllTokens(e):e)).then((e=>r.walkTokens?Promise.all(this.walkTokens(e,r.walkTokens)).then((()=>e)):e)).then((e=>o(e,r))).then((e=>r.hooks?r.hooks.postprocess(e):e)).catch(i);try{r.hooks&&(t=r.hooks.preprocess(t));let e=l(t,r);r.hooks&&(e=r.hooks.processAllTokens(e)),r.walkTokens&&this.walkTokens(e,r.walkTokens);let n=o(e,r);return r.hooks&&(n=r.hooks.postprocess(n)),n}catch(e){return i(e)}}}onError(e,t){return n=>{if(n.message+="\nPlease report this to https://github.com/markedjs/marked.",e){const e="<p>An error occurred:</p><pre>"+X(n.message+"",!0)+"</pre>";return t?Promise.resolve(e):e}if(t)return Promise.reject(n);throw n}}}const re=new se;function ie(e,t){return re.parse(e,t)}ie.options=ie.setOptions=function(e){return re.setOptions(e),ie.defaults=re.defaults,n(ie.defaults),ie},ie.getDefaults=t,ie.defaults=e.defaults,ie.use=function(...e){return re.use(...e),ie.defaults=re.defaults,n(ie.defaults),ie},ie.walkTokens=function(e,t){return re.walkTokens(e,t)},ie.parseInline=re.parseInline,ie.Parser=te,ie.parser=te.parse,ie.Renderer=Y,ie.TextRenderer=ee,ie.Lexer=W,ie.lexer=W.lex,ie.Tokenizer=V,ie.Hooks=ne,ie.parse=ie;const le=ie.options,oe=ie.setOptions,ae=ie.use,ce=ie.walkTokens,he=ie.parseInline,pe=ie,ue=te.parse,ge=W.lex;e.Hooks=ne,e.Lexer=W,e.Marked=se,e.Parser=te,e.Renderer=Y,e.TextRenderer=ee,e.Tokenizer=V,e.getDefaults=t,e.lexer=ge,e.marked=ie,e.options=le,e.parse=pe,e.parseInline=he,e.parser=ue,e.setOptions=oe,e.use=ae,e.walkTokens=ce})); | |
| </script> | |
| <!-- highlight.js --> | |
| <script>/*! | |
| Highlight.js v11.9.0 (git: f47103d4f1) | |
| (c) 2006-2023 undefined and other contributors | |
| License: BSD-3-Clause | |
| */ | |
| var hljs=function(){"use strict";function e(n){ | |
| return n instanceof Map?n.clear=n.delete=n.set=()=>{ | |
| throw Error("map is read-only")}:n instanceof Set&&(n.add=n.clear=n.delete=()=>{ | |
| throw Error("set is read-only") | |
| }),Object.freeze(n),Object.getOwnPropertyNames(n).forEach((t=>{ | |
| const a=n[t],i=typeof a;"object"!==i&&"function"!==i||Object.isFrozen(a)||e(a) | |
| })),n}class n{constructor(e){ | |
| void 0===e.data&&(e.data={}),this.data=e.data,this.isMatchIgnored=!1} | |
| ignoreMatch(){this.isMatchIgnored=!0}}function t(e){ | |
| return e.replace(/&/g,"&").replace(/</g,"<").replace(/>/g,">").replace(/"/g,""").replace(/'/g,"'") | |
| }function a(e,...n){const t=Object.create(null);for(const n in e)t[n]=e[n] | |
| ;return n.forEach((e=>{for(const n in e)t[n]=e[n]})),t}const i=e=>!!e.scope | |
| ;class r{constructor(e,n){ | |
| this.buffer="",this.classPrefix=n.classPrefix,e.walk(this)}addText(e){ | |
| this.buffer+=t(e)}openNode(e){if(!i(e))return;const n=((e,{prefix:n})=>{ | |
| if(e.startsWith("language:"))return e.replace("language:","language-") | |
| ;if(e.includes(".")){const t=e.split(".") | |
| ;return[`${n}${t.shift()}`,...t.map(((e,n)=>`${e}${"_".repeat(n+1)}`))].join(" ") | |
| }return`${n}${e}`})(e.scope,{prefix:this.classPrefix});this.span(n)} | |
| closeNode(e){i(e)&&(this.buffer+="</span>")}value(){return this.buffer}span(e){ | |
| this.buffer+=`<span class="${e}">`}}const s=(e={})=>{const n={children:[]} | |
| ;return Object.assign(n,e),n};class o{constructor(){ | |
| this.rootNode=s(),this.stack=[this.rootNode]}get top(){ | |
| return this.stack[this.stack.length-1]}get root(){return this.rootNode}add(e){ | |
| this.top.children.push(e)}openNode(e){const n=s({scope:e}) | |
| ;this.add(n),this.stack.push(n)}closeNode(){ | |
| if(this.stack.length>1)return this.stack.pop()}closeAllNodes(){ | |
| for(;this.closeNode(););}toJSON(){return JSON.stringify(this.rootNode,null,4)} | |
| walk(e){return this.constructor._walk(e,this.rootNode)}static _walk(e,n){ | |
| return"string"==typeof n?e.addText(n):n.children&&(e.openNode(n), | |
| n.children.forEach((n=>this._walk(e,n))),e.closeNode(n)),e}static _collapse(e){ | |
| "string"!=typeof e&&e.children&&(e.children.every((e=>"string"==typeof e))?e.children=[e.children.join("")]:e.children.forEach((e=>{ | |
| o._collapse(e)})))}}class l extends o{constructor(e){super(),this.options=e} | |
| addText(e){""!==e&&this.add(e)}startScope(e){this.openNode(e)}endScope(){ | |
| this.closeNode()}__addSublanguage(e,n){const t=e.root | |
| ;n&&(t.scope="language:"+n),this.add(t)}toHTML(){ | |
| return new r(this,this.options).value()}finalize(){ | |
| return this.closeAllNodes(),!0}}function c(e){ | |
| return e?"string"==typeof e?e:e.source:null}function d(e){return b("(?=",e,")")} | |
| function g(e){return b("(?:",e,")*")}function u(e){return b("(?:",e,")?")} | |
| function b(...e){return e.map((e=>c(e))).join("")}function m(...e){const n=(e=>{ | |
| const n=e[e.length-1] | |
| ;return"object"==typeof n&&n.constructor===Object?(e.splice(e.length-1,1),n):{} | |
| })(e);return"("+(n.capture?"":"?:")+e.map((e=>c(e))).join("|")+")"} | |
| function p(e){return RegExp(e.toString()+"|").exec("").length-1} | |
| const _=/\[(?:[^\\\]]|\\.)*\]|\(\??|\\([1-9][0-9]*)|\\./ | |
| ;function h(e,{joinWith:n}){let t=0;return e.map((e=>{t+=1;const n=t | |
| ;let a=c(e),i="";for(;a.length>0;){const e=_.exec(a);if(!e){i+=a;break} | |
| i+=a.substring(0,e.index), | |
| a=a.substring(e.index+e[0].length),"\\"===e[0][0]&&e[1]?i+="\\"+(Number(e[1])+n):(i+=e[0], | |
| "("===e[0]&&t++)}return i})).map((e=>`(${e})`)).join(n)} | |
| const f="[a-zA-Z]\\w*",E="[a-zA-Z_]\\w*",y="\\b\\d+(\\.\\d+)?",N="(-?)(\\b0[xX][a-fA-F0-9]+|(\\b\\d+(\\.\\d*)?|\\.\\d+)([eE][-+]?\\d+)?)",w="\\b(0b[01]+)",v={ | |
| begin:"\\\\[\\s\\S]",relevance:0},O={scope:"string",begin:"'",end:"'", | |
| illegal:"\\n",contains:[v]},k={scope:"string",begin:'"',end:'"',illegal:"\\n", | |
| contains:[v]},x=(e,n,t={})=>{const i=a({scope:"comment",begin:e,end:n, | |
| contains:[]},t);i.contains.push({scope:"doctag", | |
| begin:"[ ]*(?=(TODO|FIXME|NOTE|BUG|OPTIMIZE|HACK|XXX):)", | |
| end:/(TODO|FIXME|NOTE|BUG|OPTIMIZE|HACK|XXX):/,excludeBegin:!0,relevance:0}) | |
| ;const r=m("I","a","is","so","us","to","at","if","in","it","on",/[A-Za-z]+['](d|ve|re|ll|t|s|n)/,/[A-Za-z]+[-][a-z]+/,/[A-Za-z][a-z]{2,}/) | |
| ;return i.contains.push({begin:b(/[ ]+/,"(",r,/[.]?[:]?([.][ ]|[ ])/,"){3}")}),i | |
| },M=x("//","$"),S=x("/\\*","\\*/"),A=x("#","$");var C=Object.freeze({ | |
| __proto__:null,APOS_STRING_MODE:O,BACKSLASH_ESCAPE:v,BINARY_NUMBER_MODE:{ | |
| scope:"number",begin:w,relevance:0},BINARY_NUMBER_RE:w,COMMENT:x, | |
| C_BLOCK_COMMENT_MODE:S,C_LINE_COMMENT_MODE:M,C_NUMBER_MODE:{scope:"number", | |
| begin:N,relevance:0},C_NUMBER_RE:N,END_SAME_AS_BEGIN:e=>Object.assign(e,{ | |
| "on:begin":(e,n)=>{n.data._beginMatch=e[1]},"on:end":(e,n)=>{ | |
| n.data._beginMatch!==e[1]&&n.ignoreMatch()}}),HASH_COMMENT_MODE:A,IDENT_RE:f, | |
| MATCH_NOTHING_RE:/\b\B/,METHOD_GUARD:{begin:"\\.\\s*"+E,relevance:0}, | |
| NUMBER_MODE:{scope:"number",begin:y,relevance:0},NUMBER_RE:y, | |
| PHRASAL_WORDS_MODE:{ | |
| begin:/\b(a|an|the|are|I'm|isn't|don't|doesn't|won't|but|just|should|pretty|simply|enough|gonna|going|wtf|so|such|will|you|your|they|like|more)\b/ | |
| },QUOTE_STRING_MODE:k,REGEXP_MODE:{scope:"regexp",begin:/\/(?=[^/\n]*\/)/, | |
| end:/\/[gimuy]*/,contains:[v,{begin:/\[/,end:/\]/,relevance:0,contains:[v]}]}, | |
| RE_STARTERS_RE:"!|!=|!==|%|%=|&|&&|&=|\\*|\\*=|\\+|\\+=|,|-|-=|/=|/|:|;|<<|<<=|<=|<|===|==|=|>>>=|>>=|>=|>>>|>>|>|\\?|\\[|\\{|\\(|\\^|\\^=|\\||\\|=|\\|\\||~", | |
| SHEBANG:(e={})=>{const n=/^#![ ]*\// | |
| ;return e.binary&&(e.begin=b(n,/.*\b/,e.binary,/\b.*/)),a({scope:"meta",begin:n, | |
| end:/$/,relevance:0,"on:begin":(e,n)=>{0!==e.index&&n.ignoreMatch()}},e)}, | |
| TITLE_MODE:{scope:"title",begin:f,relevance:0},UNDERSCORE_IDENT_RE:E, | |
| UNDERSCORE_TITLE_MODE:{scope:"title",begin:E,relevance:0}});function T(e,n){ | |
| "."===e.input[e.index-1]&&n.ignoreMatch()}function R(e,n){ | |
| void 0!==e.className&&(e.scope=e.className,delete e.className)}function D(e,n){ | |
| n&&e.beginKeywords&&(e.begin="\\b("+e.beginKeywords.split(" ").join("|")+")(?!\\.)(?=\\b|\\s)", | |
| e.__beforeBegin=T,e.keywords=e.keywords||e.beginKeywords,delete e.beginKeywords, | |
| void 0===e.relevance&&(e.relevance=0))}function I(e,n){ | |
| Array.isArray(e.illegal)&&(e.illegal=m(...e.illegal))}function L(e,n){ | |
| if(e.match){ | |
| if(e.begin||e.end)throw Error("begin & end are not supported with match") | |
| ;e.begin=e.match,delete e.match}}function B(e,n){ | |
| void 0===e.relevance&&(e.relevance=1)}const $=(e,n)=>{if(!e.beforeMatch)return | |
| ;if(e.starts)throw Error("beforeMatch cannot be used with starts") | |
| ;const t=Object.assign({},e);Object.keys(e).forEach((n=>{delete e[n] | |
| })),e.keywords=t.keywords,e.begin=b(t.beforeMatch,d(t.begin)),e.starts={ | |
| relevance:0,contains:[Object.assign(t,{endsParent:!0})] | |
| },e.relevance=0,delete t.beforeMatch | |
| },z=["of","and","for","in","not","or","if","then","parent","list","value"],F="keyword" | |
| ;function U(e,n,t=F){const a=Object.create(null) | |
| ;return"string"==typeof e?i(t,e.split(" ")):Array.isArray(e)?i(t,e):Object.keys(e).forEach((t=>{ | |
| Object.assign(a,U(e[t],n,t))})),a;function i(e,t){ | |
| n&&(t=t.map((e=>e.toLowerCase()))),t.forEach((n=>{const t=n.split("|") | |
| ;a[t[0]]=[e,j(t[0],t[1])]}))}}function j(e,n){ | |
| return n?Number(n):(e=>z.includes(e.toLowerCase()))(e)?0:1}const P={},K=e=>{ | |
| console.error(e)},H=(e,...n)=>{console.log("WARN: "+e,...n)},q=(e,n)=>{ | |
| P[`${e}/${n}`]||(console.log(`Deprecated as of ${e}. ${n}`),P[`${e}/${n}`]=!0) | |
| },G=Error();function Z(e,n,{key:t}){let a=0;const i=e[t],r={},s={} | |
| ;for(let e=1;e<=n.length;e++)s[e+a]=i[e],r[e+a]=!0,a+=p(n[e-1]) | |
| ;e[t]=s,e[t]._emit=r,e[t]._multi=!0}function W(e){(e=>{ | |
| e.scope&&"object"==typeof e.scope&&null!==e.scope&&(e.beginScope=e.scope, | |
| delete e.scope)})(e),"string"==typeof e.beginScope&&(e.beginScope={ | |
| _wrap:e.beginScope}),"string"==typeof e.endScope&&(e.endScope={_wrap:e.endScope | |
| }),(e=>{if(Array.isArray(e.begin)){ | |
| if(e.skip||e.excludeBegin||e.returnBegin)throw K("skip, excludeBegin, returnBegin not compatible with beginScope: {}"), | |
| G | |
| ;if("object"!=typeof e.beginScope||null===e.beginScope)throw K("beginScope must be object"), | |
| G;Z(e,e.begin,{key:"beginScope"}),e.begin=h(e.begin,{joinWith:""})}})(e),(e=>{ | |
| if(Array.isArray(e.end)){ | |
| if(e.skip||e.excludeEnd||e.returnEnd)throw K("skip, excludeEnd, returnEnd not compatible with endScope: {}"), | |
| G | |
| ;if("object"!=typeof e.endScope||null===e.endScope)throw K("endScope must be object"), | |
| G;Z(e,e.end,{key:"endScope"}),e.end=h(e.end,{joinWith:""})}})(e)}function Q(e){ | |
| function n(n,t){ | |
| return RegExp(c(n),"m"+(e.case_insensitive?"i":"")+(e.unicodeRegex?"u":"")+(t?"g":"")) | |
| }class t{constructor(){ | |
| this.matchIndexes={},this.regexes=[],this.matchAt=1,this.position=0} | |
| addRule(e,n){ | |
| n.position=this.position++,this.matchIndexes[this.matchAt]=n,this.regexes.push([n,e]), | |
| this.matchAt+=p(e)+1}compile(){0===this.regexes.length&&(this.exec=()=>null) | |
| ;const e=this.regexes.map((e=>e[1]));this.matcherRe=n(h(e,{joinWith:"|" | |
| }),!0),this.lastIndex=0}exec(e){this.matcherRe.lastIndex=this.lastIndex | |
| ;const n=this.matcherRe.exec(e);if(!n)return null | |
| ;const t=n.findIndex(((e,n)=>n>0&&void 0!==e)),a=this.matchIndexes[t] | |
| ;return n.splice(0,t),Object.assign(n,a)}}class i{constructor(){ | |
| this.rules=[],this.multiRegexes=[], | |
| this.count=0,this.lastIndex=0,this.regexIndex=0}getMatcher(e){ | |
| if(this.multiRegexes[e])return this.multiRegexes[e];const n=new t | |
| ;return this.rules.slice(e).forEach((([e,t])=>n.addRule(e,t))), | |
| n.compile(),this.multiRegexes[e]=n,n}resumingScanAtSamePosition(){ | |
| return 0!==this.regexIndex}considerAll(){this.regexIndex=0}addRule(e,n){ | |
| this.rules.push([e,n]),"begin"===n.type&&this.count++}exec(e){ | |
| const n=this.getMatcher(this.regexIndex);n.lastIndex=this.lastIndex | |
| ;let t=n.exec(e) | |
| ;if(this.resumingScanAtSamePosition())if(t&&t.index===this.lastIndex);else{ | |
| const n=this.getMatcher(0);n.lastIndex=this.lastIndex+1,t=n.exec(e)} | |
| return t&&(this.regexIndex+=t.position+1, | |
| this.regexIndex===this.count&&this.considerAll()),t}} | |
| if(e.compilerExtensions||(e.compilerExtensions=[]), | |
| e.contains&&e.contains.includes("self"))throw Error("ERR: contains `self` is not supported at the top-level of a language. See documentation.") | |
| ;return e.classNameAliases=a(e.classNameAliases||{}),function t(r,s){const o=r | |
| ;if(r.isCompiled)return o | |
| ;[R,L,W,$].forEach((e=>e(r,s))),e.compilerExtensions.forEach((e=>e(r,s))), | |
| r.__beforeBegin=null,[D,I,B].forEach((e=>e(r,s))),r.isCompiled=!0;let l=null | |
| ;return"object"==typeof r.keywords&&r.keywords.$pattern&&(r.keywords=Object.assign({},r.keywords), | |
| l=r.keywords.$pattern, | |
| delete r.keywords.$pattern),l=l||/\w+/,r.keywords&&(r.keywords=U(r.keywords,e.case_insensitive)), | |
| o.keywordPatternRe=n(l,!0), | |
| s&&(r.begin||(r.begin=/\B|\b/),o.beginRe=n(o.begin),r.end||r.endsWithParent||(r.end=/\B|\b/), | |
| r.end&&(o.endRe=n(o.end)), | |
| o.terminatorEnd=c(o.end)||"",r.endsWithParent&&s.terminatorEnd&&(o.terminatorEnd+=(r.end?"|":"")+s.terminatorEnd)), | |
| r.illegal&&(o.illegalRe=n(r.illegal)), | |
| r.contains||(r.contains=[]),r.contains=[].concat(...r.contains.map((e=>(e=>(e.variants&&!e.cachedVariants&&(e.cachedVariants=e.variants.map((n=>a(e,{ | |
| variants:null},n)))),e.cachedVariants?e.cachedVariants:X(e)?a(e,{ | |
| starts:e.starts?a(e.starts):null | |
| }):Object.isFrozen(e)?a(e):e))("self"===e?r:e)))),r.contains.forEach((e=>{t(e,o) | |
| })),r.starts&&t(r.starts,s),o.matcher=(e=>{const n=new i | |
| ;return e.contains.forEach((e=>n.addRule(e.begin,{rule:e,type:"begin" | |
| }))),e.terminatorEnd&&n.addRule(e.terminatorEnd,{type:"end" | |
| }),e.illegal&&n.addRule(e.illegal,{type:"illegal"}),n})(o),o}(e)}function X(e){ | |
| return!!e&&(e.endsWithParent||X(e.starts))}class V extends Error{ | |
| constructor(e,n){super(e),this.name="HTMLInjectionError",this.html=n}} | |
| const J=t,Y=a,ee=Symbol("nomatch"),ne=t=>{ | |
| const a=Object.create(null),i=Object.create(null),r=[];let s=!0 | |
| ;const o="Could not find the language '{}', did you forget to load/include a language module?",c={ | |
| disableAutodetect:!0,name:"Plain text",contains:[]};let p={ | |
| ignoreUnescapedHTML:!1,throwUnescapedHTML:!1,noHighlightRe:/^(no-?highlight)$/i, | |
| languageDetectRe:/\blang(?:uage)?-([\w-]+)\b/i,classPrefix:"hljs-", | |
| cssSelector:"pre code",languages:null,__emitter:l};function _(e){ | |
| return p.noHighlightRe.test(e)}function h(e,n,t){let a="",i="" | |
| ;"object"==typeof n?(a=e, | |
| t=n.ignoreIllegals,i=n.language):(q("10.7.0","highlight(lang, code, ...args) has been deprecated."), | |
| q("10.7.0","Please use highlight(code, options) instead.\nhttps://github.com/highlightjs/highlight.js/issues/2277"), | |
| i=e,a=n),void 0===t&&(t=!0);const r={code:a,language:i};x("before:highlight",r) | |
| ;const s=r.result?r.result:f(r.language,r.code,t) | |
| ;return s.code=r.code,x("after:highlight",s),s}function f(e,t,i,r){ | |
| const l=Object.create(null);function c(){if(!x.keywords)return void S.addText(A) | |
| ;let e=0;x.keywordPatternRe.lastIndex=0;let n=x.keywordPatternRe.exec(A),t="" | |
| ;for(;n;){t+=A.substring(e,n.index) | |
| ;const i=w.case_insensitive?n[0].toLowerCase():n[0],r=(a=i,x.keywords[a]);if(r){ | |
| const[e,a]=r | |
| ;if(S.addText(t),t="",l[i]=(l[i]||0)+1,l[i]<=7&&(C+=a),e.startsWith("_"))t+=n[0];else{ | |
| const t=w.classNameAliases[e]||e;g(n[0],t)}}else t+=n[0] | |
| ;e=x.keywordPatternRe.lastIndex,n=x.keywordPatternRe.exec(A)}var a | |
| ;t+=A.substring(e),S.addText(t)}function d(){null!=x.subLanguage?(()=>{ | |
| if(""===A)return;let e=null;if("string"==typeof x.subLanguage){ | |
| if(!a[x.subLanguage])return void S.addText(A) | |
| ;e=f(x.subLanguage,A,!0,M[x.subLanguage]),M[x.subLanguage]=e._top | |
| }else e=E(A,x.subLanguage.length?x.subLanguage:null) | |
| ;x.relevance>0&&(C+=e.relevance),S.__addSublanguage(e._emitter,e.language) | |
| })():c(),A=""}function g(e,n){ | |
| ""!==e&&(S.startScope(n),S.addText(e),S.endScope())}function u(e,n){let t=1 | |
| ;const a=n.length-1;for(;t<=a;){if(!e._emit[t]){t++;continue} | |
| const a=w.classNameAliases[e[t]]||e[t],i=n[t];a?g(i,a):(A=i,c(),A=""),t++}} | |
| function b(e,n){ | |
| return e.scope&&"string"==typeof e.scope&&S.openNode(w.classNameAliases[e.scope]||e.scope), | |
| e.beginScope&&(e.beginScope._wrap?(g(A,w.classNameAliases[e.beginScope._wrap]||e.beginScope._wrap), | |
| A=""):e.beginScope._multi&&(u(e.beginScope,n),A="")),x=Object.create(e,{parent:{ | |
| value:x}}),x}function m(e,t,a){let i=((e,n)=>{const t=e&&e.exec(n) | |
| ;return t&&0===t.index})(e.endRe,a);if(i){if(e["on:end"]){const a=new n(e) | |
| ;e["on:end"](t,a),a.isMatchIgnored&&(i=!1)}if(i){ | |
| for(;e.endsParent&&e.parent;)e=e.parent;return e}} | |
| if(e.endsWithParent)return m(e.parent,t,a)}function _(e){ | |
| return 0===x.matcher.regexIndex?(A+=e[0],1):(D=!0,0)}function h(e){ | |
| const n=e[0],a=t.substring(e.index),i=m(x,e,a);if(!i)return ee;const r=x | |
| ;x.endScope&&x.endScope._wrap?(d(), | |
| g(n,x.endScope._wrap)):x.endScope&&x.endScope._multi?(d(), | |
| u(x.endScope,e)):r.skip?A+=n:(r.returnEnd||r.excludeEnd||(A+=n), | |
| d(),r.excludeEnd&&(A=n));do{ | |
| x.scope&&S.closeNode(),x.skip||x.subLanguage||(C+=x.relevance),x=x.parent | |
| }while(x!==i.parent);return i.starts&&b(i.starts,e),r.returnEnd?0:n.length} | |
| let y={};function N(a,r){const o=r&&r[0];if(A+=a,null==o)return d(),0 | |
| ;if("begin"===y.type&&"end"===r.type&&y.index===r.index&&""===o){ | |
| if(A+=t.slice(r.index,r.index+1),!s){const n=Error(`0 width match regex (${e})`) | |
| ;throw n.languageName=e,n.badRule=y.rule,n}return 1} | |
| if(y=r,"begin"===r.type)return(e=>{ | |
| const t=e[0],a=e.rule,i=new n(a),r=[a.__beforeBegin,a["on:begin"]] | |
| ;for(const n of r)if(n&&(n(e,i),i.isMatchIgnored))return _(t) | |
| ;return a.skip?A+=t:(a.excludeBegin&&(A+=t), | |
| d(),a.returnBegin||a.excludeBegin||(A=t)),b(a,e),a.returnBegin?0:t.length})(r) | |
| ;if("illegal"===r.type&&!i){ | |
| const e=Error('Illegal lexeme "'+o+'" for mode "'+(x.scope||"<unnamed>")+'"') | |
| ;throw e.mode=x,e}if("end"===r.type){const e=h(r);if(e!==ee)return e} | |
| if("illegal"===r.type&&""===o)return 1 | |
| ;if(R>1e5&&R>3*r.index)throw Error("potential infinite loop, way more iterations than matches") | |
| ;return A+=o,o.length}const w=v(e) | |
| ;if(!w)throw K(o.replace("{}",e)),Error('Unknown language: "'+e+'"') | |
| ;const O=Q(w);let k="",x=r||O;const M={},S=new p.__emitter(p);(()=>{const e=[] | |
| ;for(let n=x;n!==w;n=n.parent)n.scope&&e.unshift(n.scope) | |
| ;e.forEach((e=>S.openNode(e)))})();let A="",C=0,T=0,R=0,D=!1;try{ | |
| if(w.__emitTokens)w.__emitTokens(t,S);else{for(x.matcher.considerAll();;){ | |
| R++,D?D=!1:x.matcher.considerAll(),x.matcher.lastIndex=T | |
| ;const e=x.matcher.exec(t);if(!e)break;const n=N(t.substring(T,e.index),e) | |
| ;T=e.index+n}N(t.substring(T))}return S.finalize(),k=S.toHTML(),{language:e, | |
| value:k,relevance:C,illegal:!1,_emitter:S,_top:x}}catch(n){ | |
| if(n.message&&n.message.includes("Illegal"))return{language:e,value:J(t), | |
| illegal:!0,relevance:0,_illegalBy:{message:n.message,index:T, | |
| context:t.slice(T-100,T+100),mode:n.mode,resultSoFar:k},_emitter:S};if(s)return{ | |
| language:e,value:J(t),illegal:!1,relevance:0,errorRaised:n,_emitter:S,_top:x} | |
| ;throw n}}function E(e,n){n=n||p.languages||Object.keys(a);const t=(e=>{ | |
| const n={value:J(e),illegal:!1,relevance:0,_top:c,_emitter:new p.__emitter(p)} | |
| ;return n._emitter.addText(e),n})(e),i=n.filter(v).filter(k).map((n=>f(n,e,!1))) | |
| ;i.unshift(t);const r=i.sort(((e,n)=>{ | |
| if(e.relevance!==n.relevance)return n.relevance-e.relevance | |
| ;if(e.language&&n.language){if(v(e.language).supersetOf===n.language)return 1 | |
| ;if(v(n.language).supersetOf===e.language)return-1}return 0})),[s,o]=r,l=s | |
| ;return l.secondBest=o,l}function y(e){let n=null;const t=(e=>{ | |
| let n=e.className+" ";n+=e.parentNode?e.parentNode.className:"" | |
| ;const t=p.languageDetectRe.exec(n);if(t){const n=v(t[1]) | |
| ;return n||(H(o.replace("{}",t[1])), | |
| H("Falling back to no-highlight mode for this block.",e)),n?t[1]:"no-highlight"} | |
| return n.split(/\s+/).find((e=>_(e)||v(e)))})(e);if(_(t))return | |
| ;if(x("before:highlightElement",{el:e,language:t | |
| }),e.dataset.highlighted)return void console.log("Element previously highlighted. To highlight again, first unset `dataset.highlighted`.",e) | |
| ;if(e.children.length>0&&(p.ignoreUnescapedHTML||(console.warn("One of your code blocks includes unescaped HTML. This is a potentially serious security risk."), | |
| console.warn("https://github.com/highlightjs/highlight.js/wiki/security"), | |
| console.warn("The element with unescaped HTML:"), | |
| console.warn(e)),p.throwUnescapedHTML))throw new V("One of your code blocks includes unescaped HTML.",e.innerHTML) | |
| ;n=e;const a=n.textContent,r=t?h(a,{language:t,ignoreIllegals:!0}):E(a) | |
| ;e.innerHTML=r.value,e.dataset.highlighted="yes",((e,n,t)=>{const a=n&&i[n]||t | |
| ;e.classList.add("hljs"),e.classList.add("language-"+a) | |
| })(e,t,r.language),e.result={language:r.language,re:r.relevance, | |
| relevance:r.relevance},r.secondBest&&(e.secondBest={ | |
| language:r.secondBest.language,relevance:r.secondBest.relevance | |
| }),x("after:highlightElement",{el:e,result:r,text:a})}let N=!1;function w(){ | |
| "loading"!==document.readyState?document.querySelectorAll(p.cssSelector).forEach(y):N=!0 | |
| }function v(e){return e=(e||"").toLowerCase(),a[e]||a[i[e]]} | |
| function O(e,{languageName:n}){"string"==typeof e&&(e=[e]),e.forEach((e=>{ | |
| i[e.toLowerCase()]=n}))}function k(e){const n=v(e) | |
| ;return n&&!n.disableAutodetect}function x(e,n){const t=e;r.forEach((e=>{ | |
| e[t]&&e[t](n)}))} | |
| "undefined"!=typeof window&&window.addEventListener&&window.addEventListener("DOMContentLoaded",(()=>{ | |
| N&&w()}),!1),Object.assign(t,{highlight:h,highlightAuto:E,highlightAll:w, | |
| highlightElement:y, | |
| highlightBlock:e=>(q("10.7.0","highlightBlock will be removed entirely in v12.0"), | |
| q("10.7.0","Please use highlightElement now."),y(e)),configure:e=>{p=Y(p,e)}, | |
| initHighlighting:()=>{ | |
| w(),q("10.6.0","initHighlighting() deprecated. Use highlightAll() now.")}, | |
| initHighlightingOnLoad:()=>{ | |
| w(),q("10.6.0","initHighlightingOnLoad() deprecated. Use highlightAll() now.") | |
| },registerLanguage:(e,n)=>{let i=null;try{i=n(t)}catch(n){ | |
| if(K("Language definition for '{}' could not be registered.".replace("{}",e)), | |
| !s)throw n;K(n),i=c} | |
| i.name||(i.name=e),a[e]=i,i.rawDefinition=n.bind(null,t),i.aliases&&O(i.aliases,{ | |
| languageName:e})},unregisterLanguage:e=>{delete a[e] | |
| ;for(const n of Object.keys(i))i[n]===e&&delete i[n]}, | |
| listLanguages:()=>Object.keys(a),getLanguage:v,registerAliases:O, | |
| autoDetection:k,inherit:Y,addPlugin:e=>{(e=>{ | |
| e["before:highlightBlock"]&&!e["before:highlightElement"]&&(e["before:highlightElement"]=n=>{ | |
| e["before:highlightBlock"](Object.assign({block:n.el},n)) | |
| }),e["after:highlightBlock"]&&!e["after:highlightElement"]&&(e["after:highlightElement"]=n=>{ | |
| e["after:highlightBlock"](Object.assign({block:n.el},n))})})(e),r.push(e)}, | |
| removePlugin:e=>{const n=r.indexOf(e);-1!==n&&r.splice(n,1)}}),t.debugMode=()=>{ | |
| s=!1},t.safeMode=()=>{s=!0},t.versionString="11.9.0",t.regex={concat:b, | |
| lookahead:d,either:m,optional:u,anyNumberOfTimes:g} | |
| ;for(const n in C)"object"==typeof C[n]&&e(C[n]);return Object.assign(t,C),t | |
| },te=ne({});te.newInstance=()=>ne({});var ae=te;const ie=e=>({IMPORTANT:{ | |
| scope:"meta",begin:"!important"},BLOCK_COMMENT:e.C_BLOCK_COMMENT_MODE,HEXCOLOR:{ | |
| scope:"number",begin:/#(([0-9a-fA-F]{3,4})|(([0-9a-fA-F]{2}){3,4}))\b/}, | |
| FUNCTION_DISPATCH:{className:"built_in",begin:/[\w-]+(?=\()/}, | |
| ATTRIBUTE_SELECTOR_MODE:{scope:"selector-attr",begin:/\[/,end:/\]/,illegal:"$", | |
| contains:[e.APOS_STRING_MODE,e.QUOTE_STRING_MODE]},CSS_NUMBER_MODE:{ | |
| scope:"number", | |
| begin:e.NUMBER_RE+"(%|em|ex|ch|rem|vw|vh|vmin|vmax|cm|mm|in|pt|pc|px|deg|grad|rad|turn|s|ms|Hz|kHz|dpi|dpcm|dppx)?", | |
| relevance:0},CSS_VARIABLE:{className:"attr",begin:/--[A-Za-z_][A-Za-z0-9_-]*/} | |
| }),re=["a","abbr","address","article","aside","audio","b","blockquote","body","button","canvas","caption","cite","code","dd","del","details","dfn","div","dl","dt","em","fieldset","figcaption","figure","footer","form","h1","h2","h3","h4","h5","h6","header","hgroup","html","i","iframe","img","input","ins","kbd","label","legend","li","main","mark","menu","nav","object","ol","p","q","quote","samp","section","span","strong","summary","sup","table","tbody","td","textarea","tfoot","th","thead","time","tr","ul","var","video"],se=["any-hover","any-pointer","aspect-ratio","color","color-gamut","color-index","device-aspect-ratio","device-height","device-width","display-mode","forced-colors","grid","height","hover","inverted-colors","monochrome","orientation","overflow-block","overflow-inline","pointer","prefers-color-scheme","prefers-contrast","prefers-reduced-motion","prefers-reduced-transparency","resolution","scan","scripting","update","width","min-width","max-width","min-height","max-height"],oe=["active","any-link","blank","checked","current","default","defined","dir","disabled","drop","empty","enabled","first","first-child","first-of-type","fullscreen","future","focus","focus-visible","focus-within","has","host","host-context","hover","indeterminate","in-range","invalid","is","lang","last-child","last-of-type","left","link","local-link","not","nth-child","nth-col","nth-last-child","nth-last-col","nth-last-of-type","nth-of-type","only-child","only-of-type","optional","out-of-range","past","placeholder-shown","read-only","read-write","required","right","root","scope","target","target-within","user-invalid","valid","visited","where"],le=["after","backdrop","before","cue","cue-region","first-letter","first-line","grammar-error","marker","part","placeholder","selection","slotted","spelling-error"],ce=["align-content","align-items","align-self","all","animation","animation-delay","animation-direction","animation-duration","animation-fill-mode","animation-iteration-count","animation-name","animation-play-state","animation-timing-function","backface-visibility","background","background-attachment","background-blend-mode","background-clip","background-color","background-image","background-origin","background-position","background-repeat","background-size","block-size","border","border-block","border-block-color","border-block-end","border-block-end-color","border-block-end-style","border-block-end-width","border-block-start","border-block-start-color","border-block-start-style","border-block-start-width","border-block-style","border-block-width","border-bottom","border-bottom-color","border-bottom-left-radius","border-bottom-right-radius","border-bottom-style","border-bottom-width","border-collapse","border-color","border-image","border-image-outset","border-image-repeat","border-image-slice","border-image-source","border-image-width","border-inline","border-inline-color","border-inline-end","border-inline-end-color","border-inline-end-style","border-inline-end-width","border-inline-start","border-inline-start-color","border-inline-start-style","border-inline-start-width","border-inline-style","border-inline-width","border-left","border-left-color","border-left-style","border-left-width","border-radius","border-right","border-right-color","border-right-style","border-right-width","border-spacing","border-style","border-top","border-top-color","border-top-left-radius","border-top-right-radius","border-top-style","border-top-width","border-width","bottom","box-decoration-break","box-shadow","box-sizing","break-after","break-before","break-inside","caption-side","caret-color","clear","clip","clip-path","clip-rule","color","column-count","column-fill","column-gap","column-rule","column-rule-color","column-rule-style","column-rule-width","column-span","column-width","columns","contain","content","content-visibility","counter-increment","counter-reset","cue","cue-after","cue-before","cursor","direction","display","empty-cells","filter","flex","flex-basis","flex-direction","flex-flow","flex-grow","flex-shrink","flex-wrap","float","flow","font","font-display","font-family","font-feature-settings","font-kerning","font-language-override","font-size","font-size-adjust","font-smoothing","font-stretch","font-style","font-synthesis","font-variant","font-variant-caps","font-variant-east-asian","font-variant-ligatures","font-variant-numeric","font-variant-position","font-variation-settings","font-weight","gap","glyph-orientation-vertical","grid","grid-area","grid-auto-columns","grid-auto-flow","grid-auto-rows","grid-column","grid-column-end","grid-column-start","grid-gap","grid-row","grid-row-end","grid-row-start","grid-template","grid-template-areas","grid-template-columns","grid-template-rows","hanging-punctuation","height","hyphens","icon","image-orientation","image-rendering","image-resolution","ime-mode","inline-size","isolation","justify-content","left","letter-spacing","line-break","line-height","list-style","list-style-image","list-style-position","list-style-type","margin","margin-block","margin-block-end","margin-block-start","margin-bottom","margin-inline","margin-inline-end","margin-inline-start","margin-left","margin-right","margin-top","marks","mask","mask-border","mask-border-mode","mask-border-outset","mask-border-repeat","mask-border-slice","mask-border-source","mask-border-width","mask-clip","mask-composite","mask-image","mask-mode","mask-origin","mask-position","mask-repeat","mask-size","mask-type","max-block-size","max-height","max-inline-size","max-width","min-block-size","min-height","min-inline-size","min-width","mix-blend-mode","nav-down","nav-index","nav-left","nav-right","nav-up","none","normal","object-fit","object-position","opacity","order","orphans","outline","outline-color","outline-offset","outline-style","outline-width","overflow","overflow-wrap","overflow-x","overflow-y","padding","padding-block","padding-block-end","padding-block-start","padding-bottom","padding-inline","padding-inline-end","padding-inline-start","padding-left","padding-right","padding-top","page-break-after","page-break-before","page-break-inside","pause","pause-after","pause-before","perspective","perspective-origin","pointer-events","position","quotes","resize","rest","rest-after","rest-before","right","row-gap","scroll-margin","scroll-margin-block","scroll-margin-block-end","scroll-margin-block-start","scroll-margin-bottom","scroll-margin-inline","scroll-margin-inline-end","scroll-margin-inline-start","scroll-margin-left","scroll-margin-right","scroll-margin-top","scroll-padding","scroll-padding-block","scroll-padding-block-end","scroll-padding-block-start","scroll-padding-bottom","scroll-padding-inline","scroll-padding-inline-end","scroll-padding-inline-start","scroll-padding-left","scroll-padding-right","scroll-padding-top","scroll-snap-align","scroll-snap-stop","scroll-snap-type","scrollbar-color","scrollbar-gutter","scrollbar-width","shape-image-threshold","shape-margin","shape-outside","speak","speak-as","src","tab-size","table-layout","text-align","text-align-all","text-align-last","text-combine-upright","text-decoration","text-decoration-color","text-decoration-line","text-decoration-style","text-emphasis","text-emphasis-color","text-emphasis-position","text-emphasis-style","text-indent","text-justify","text-orientation","text-overflow","text-rendering","text-shadow","text-transform","text-underline-position","top","transform","transform-box","transform-origin","transform-style","transition","transition-delay","transition-duration","transition-property","transition-timing-function","unicode-bidi","vertical-align","visibility","voice-balance","voice-duration","voice-family","voice-pitch","voice-range","voice-rate","voice-stress","voice-volume","white-space","widows","width","will-change","word-break","word-spacing","word-wrap","writing-mode","z-index"].reverse(),de=oe.concat(le) | |
| ;var ge="[0-9](_*[0-9])*",ue=`\\.(${ge})`,be="[0-9a-fA-F](_*[0-9a-fA-F])*",me={ | |
| className:"number",variants:[{ | |
| begin:`(\\b(${ge})((${ue})|\\.)?|(${ue}))[eE][+-]?(${ge})[fFdD]?\\b`},{ | |
| begin:`\\b(${ge})((${ue})[fFdD]?\\b|\\.([fFdD]\\b)?)`},{ | |
| begin:`(${ue})[fFdD]?\\b`},{begin:`\\b(${ge})[fFdD]\\b`},{ | |
| begin:`\\b0[xX]((${be})\\.?|(${be})?\\.(${be}))[pP][+-]?(${ge})[fFdD]?\\b`},{ | |
| begin:"\\b(0|[1-9](_*[0-9])*)[lL]?\\b"},{begin:`\\b0[xX](${be})[lL]?\\b`},{ | |
| begin:"\\b0(_*[0-7])*[lL]?\\b"},{begin:"\\b0[bB][01](_*[01])*[lL]?\\b"}], | |
| relevance:0};function pe(e,n,t){return-1===t?"":e.replace(n,(a=>pe(e,n,t-1)))} | |
| const _e="[A-Za-z$_][0-9A-Za-z$_]*",he=["as","in","of","if","for","while","finally","var","new","function","do","return","void","else","break","catch","instanceof","with","throw","case","default","try","switch","continue","typeof","delete","let","yield","const","class","debugger","async","await","static","import","from","export","extends"],fe=["true","false","null","undefined","NaN","Infinity"],Ee=["Object","Function","Boolean","Symbol","Math","Date","Number","BigInt","String","RegExp","Array","Float32Array","Float64Array","Int8Array","Uint8Array","Uint8ClampedArray","Int16Array","Int32Array","Uint16Array","Uint32Array","BigInt64Array","BigUint64Array","Set","Map","WeakSet","WeakMap","ArrayBuffer","SharedArrayBuffer","Atomics","DataView","JSON","Promise","Generator","GeneratorFunction","AsyncFunction","Reflect","Proxy","Intl","WebAssembly"],ye=["Error","EvalError","InternalError","RangeError","ReferenceError","SyntaxError","TypeError","URIError"],Ne=["setInterval","setTimeout","clearInterval","clearTimeout","require","exports","eval","isFinite","isNaN","parseFloat","parseInt","decodeURI","decodeURIComponent","encodeURI","encodeURIComponent","escape","unescape"],we=["arguments","this","super","console","window","document","localStorage","sessionStorage","module","global"],ve=[].concat(Ne,Ee,ye) | |
| ;function Oe(e){const n=e.regex,t=_e,a={begin:/<[A-Za-z0-9\\._:-]+/, | |
| end:/\/[A-Za-z0-9\\._:-]+>|\/>/,isTrulyOpeningTag:(e,n)=>{ | |
| const t=e[0].length+e.index,a=e.input[t] | |
| ;if("<"===a||","===a)return void n.ignoreMatch();let i | |
| ;">"===a&&(((e,{after:n})=>{const t="</"+e[0].slice(1) | |
| ;return-1!==e.input.indexOf(t,n)})(e,{after:t})||n.ignoreMatch()) | |
| ;const r=e.input.substring(t) | |
| ;((i=r.match(/^\s*=/))||(i=r.match(/^\s+extends\s+/))&&0===i.index)&&n.ignoreMatch() | |
| }},i={$pattern:_e,keyword:he,literal:fe,built_in:ve,"variable.language":we | |
| },r="[0-9](_?[0-9])*",s=`\\.(${r})`,o="0|[1-9](_?[0-9])*|0[0-7]*[89][0-9]*",l={ | |
| className:"number",variants:[{ | |
| begin:`(\\b(${o})((${s})|\\.)?|(${s}))[eE][+-]?(${r})\\b`},{ | |
| begin:`\\b(${o})\\b((${s})\\b|\\.)?|(${s})\\b`},{ | |
| begin:"\\b(0|[1-9](_?[0-9])*)n\\b"},{ | |
| begin:"\\b0[xX][0-9a-fA-F](_?[0-9a-fA-F])*n?\\b"},{ | |
| begin:"\\b0[bB][0-1](_?[0-1])*n?\\b"},{begin:"\\b0[oO][0-7](_?[0-7])*n?\\b"},{ | |
| begin:"\\b0[0-7]+n?\\b"}],relevance:0},c={className:"subst",begin:"\\$\\{", | |
| end:"\\}",keywords:i,contains:[]},d={begin:"html`",end:"",starts:{end:"`", | |
| returnEnd:!1,contains:[e.BACKSLASH_ESCAPE,c],subLanguage:"xml"}},g={ | |
| begin:"css`",end:"",starts:{end:"`",returnEnd:!1, | |
| contains:[e.BACKSLASH_ESCAPE,c],subLanguage:"css"}},u={begin:"gql`",end:"", | |
| starts:{end:"`",returnEnd:!1,contains:[e.BACKSLASH_ESCAPE,c], | |
| subLanguage:"graphql"}},b={className:"string",begin:"`",end:"`", | |
| contains:[e.BACKSLASH_ESCAPE,c]},m={className:"comment", | |
| variants:[e.COMMENT(/\/\*\*(?!\/)/,"\\*/",{relevance:0,contains:[{ | |
| begin:"(?=@[A-Za-z]+)",relevance:0,contains:[{className:"doctag", | |
| begin:"@[A-Za-z]+"},{className:"type",begin:"\\{",end:"\\}",excludeEnd:!0, | |
| excludeBegin:!0,relevance:0},{className:"variable",begin:t+"(?=\\s*(-)|$)", | |
| endsParent:!0,relevance:0},{begin:/(?=[^\n])\s/,relevance:0}]}] | |
| }),e.C_BLOCK_COMMENT_MODE,e.C_LINE_COMMENT_MODE] | |
| },p=[e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,d,g,u,b,{match:/\$\d+/},l] | |
| ;c.contains=p.concat({begin:/\{/,end:/\}/,keywords:i,contains:["self"].concat(p) | |
| });const _=[].concat(m,c.contains),h=_.concat([{begin:/\(/,end:/\)/,keywords:i, | |
| contains:["self"].concat(_)}]),f={className:"params",begin:/\(/,end:/\)/, | |
| excludeBegin:!0,excludeEnd:!0,keywords:i,contains:h},E={variants:[{ | |
| match:[/class/,/\s+/,t,/\s+/,/extends/,/\s+/,n.concat(t,"(",n.concat(/\./,t),")*")], | |
| scope:{1:"keyword",3:"title.class",5:"keyword",7:"title.class.inherited"}},{ | |
| match:[/class/,/\s+/,t],scope:{1:"keyword",3:"title.class"}}]},y={relevance:0, | |
| match:n.either(/\bJSON/,/\b[A-Z][a-z]+([A-Z][a-z]*|\d)*/,/\b[A-Z]{2,}([A-Z][a-z]+|\d)+([A-Z][a-z]*)*/,/\b[A-Z]{2,}[a-z]+([A-Z][a-z]+|\d)*([A-Z][a-z]*)*/), | |
| className:"title.class",keywords:{_:[...Ee,...ye]}},N={variants:[{ | |
| match:[/function/,/\s+/,t,/(?=\s*\()/]},{match:[/function/,/\s*(?=\()/]}], | |
| className:{1:"keyword",3:"title.function"},label:"func.def",contains:[f], | |
| illegal:/%/},w={ | |
| match:n.concat(/\b/,(v=[...Ne,"super","import"],n.concat("(?!",v.join("|"),")")),t,n.lookahead(/\(/)), | |
| className:"title.function",relevance:0};var v;const O={ | |
| begin:n.concat(/\./,n.lookahead(n.concat(t,/(?![0-9A-Za-z$_(])/))),end:t, | |
| excludeBegin:!0,keywords:"prototype",className:"property",relevance:0},k={ | |
| match:[/get|set/,/\s+/,t,/(?=\()/],className:{1:"keyword",3:"title.function"}, | |
| contains:[{begin:/\(\)/},f] | |
| },x="(\\([^()]*(\\([^()]*(\\([^()]*\\)[^()]*)*\\)[^()]*)*\\)|"+e.UNDERSCORE_IDENT_RE+")\\s*=>",M={ | |
| match:[/const|var|let/,/\s+/,t,/\s*/,/=\s*/,/(async\s*)?/,n.lookahead(x)], | |
| keywords:"async",className:{1:"keyword",3:"title.function"},contains:[f]} | |
| ;return{name:"JavaScript",aliases:["js","jsx","mjs","cjs"],keywords:i,exports:{ | |
| PARAMS_CONTAINS:h,CLASS_REFERENCE:y},illegal:/#(?![$_A-z])/, | |
| contains:[e.SHEBANG({label:"shebang",binary:"node",relevance:5}),{ | |
| label:"use_strict",className:"meta",relevance:10, | |
| begin:/^\s*['"]use (strict|asm)['"]/ | |
| },e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,d,g,u,b,m,{match:/\$\d+/},l,y,{ | |
| className:"attr",begin:t+n.lookahead(":"),relevance:0},M,{ | |
| begin:"("+e.RE_STARTERS_RE+"|\\b(case|return|throw)\\b)\\s*", | |
| keywords:"return throw case",relevance:0,contains:[m,e.REGEXP_MODE,{ | |
| className:"function",begin:x,returnBegin:!0,end:"\\s*=>",contains:[{ | |
| className:"params",variants:[{begin:e.UNDERSCORE_IDENT_RE,relevance:0},{ | |
| className:null,begin:/\(\s*\)/,skip:!0},{begin:/\(/,end:/\)/,excludeBegin:!0, | |
| excludeEnd:!0,keywords:i,contains:h}]}]},{begin:/,/,relevance:0},{match:/\s+/, | |
| relevance:0},{variants:[{begin:"<>",end:"</>"},{ | |
| match:/<[A-Za-z0-9\\._:-]+\s*\/>/},{begin:a.begin, | |
| "on:begin":a.isTrulyOpeningTag,end:a.end}],subLanguage:"xml",contains:[{ | |
| begin:a.begin,end:a.end,skip:!0,contains:["self"]}]}]},N,{ | |
| beginKeywords:"while if switch catch for"},{ | |
| begin:"\\b(?!function)"+e.UNDERSCORE_IDENT_RE+"\\([^()]*(\\([^()]*(\\([^()]*\\)[^()]*)*\\)[^()]*)*\\)\\s*\\{", | |
| returnBegin:!0,label:"func.def",contains:[f,e.inherit(e.TITLE_MODE,{begin:t, | |
| className:"title.function"})]},{match:/\.\.\./,relevance:0},O,{match:"\\$"+t, | |
| relevance:0},{match:[/\bconstructor(?=\s*\()/],className:{1:"title.function"}, | |
| contains:[f]},w,{relevance:0,match:/\b[A-Z][A-Z_0-9]+\b/, | |
| className:"variable.constant"},E,k,{match:/\$[(.]/}]}} | |
| const ke=e=>b(/\b/,e,/\w$/.test(e)?/\b/:/\B/),xe=["Protocol","Type"].map(ke),Me=["init","self"].map(ke),Se=["Any","Self"],Ae=["actor","any","associatedtype","async","await",/as\?/,/as!/,"as","borrowing","break","case","catch","class","consume","consuming","continue","convenience","copy","default","defer","deinit","didSet","distributed","do","dynamic","each","else","enum","extension","fallthrough",/fileprivate\(set\)/,"fileprivate","final","for","func","get","guard","if","import","indirect","infix",/init\?/,/init!/,"inout",/internal\(set\)/,"internal","in","is","isolated","nonisolated","lazy","let","macro","mutating","nonmutating",/open\(set\)/,"open","operator","optional","override","postfix","precedencegroup","prefix",/private\(set\)/,"private","protocol",/public\(set\)/,"public","repeat","required","rethrows","return","set","some","static","struct","subscript","super","switch","throws","throw",/try\?/,/try!/,"try","typealias",/unowned\(safe\)/,/unowned\(unsafe\)/,"unowned","var","weak","where","while","willSet"],Ce=["false","nil","true"],Te=["assignment","associativity","higherThan","left","lowerThan","none","right"],Re=["#colorLiteral","#column","#dsohandle","#else","#elseif","#endif","#error","#file","#fileID","#fileLiteral","#filePath","#function","#if","#imageLiteral","#keyPath","#line","#selector","#sourceLocation","#warning"],De=["abs","all","any","assert","assertionFailure","debugPrint","dump","fatalError","getVaList","isKnownUniquelyReferenced","max","min","numericCast","pointwiseMax","pointwiseMin","precondition","preconditionFailure","print","readLine","repeatElement","sequence","stride","swap","swift_unboxFromSwiftValueWithType","transcode","type","unsafeBitCast","unsafeDowncast","withExtendedLifetime","withUnsafeMutablePointer","withUnsafePointer","withVaList","withoutActuallyEscaping","zip"],Ie=m(/[/=\-+!*%<>&|^~?]/,/[\u00A1-\u00A7]/,/[\u00A9\u00AB]/,/[\u00AC\u00AE]/,/[\u00B0\u00B1]/,/[\u00B6\u00BB\u00BF\u00D7\u00F7]/,/[\u2016-\u2017]/,/[\u2020-\u2027]/,/[\u2030-\u203E]/,/[\u2041-\u2053]/,/[\u2055-\u205E]/,/[\u2190-\u23FF]/,/[\u2500-\u2775]/,/[\u2794-\u2BFF]/,/[\u2E00-\u2E7F]/,/[\u3001-\u3003]/,/[\u3008-\u3020]/,/[\u3030]/),Le=m(Ie,/[\u0300-\u036F]/,/[\u1DC0-\u1DFF]/,/[\u20D0-\u20FF]/,/[\uFE00-\uFE0F]/,/[\uFE20-\uFE2F]/),Be=b(Ie,Le,"*"),$e=m(/[a-zA-Z_]/,/[\u00A8\u00AA\u00AD\u00AF\u00B2-\u00B5\u00B7-\u00BA]/,/[\u00BC-\u00BE\u00C0-\u00D6\u00D8-\u00F6\u00F8-\u00FF]/,/[\u0100-\u02FF\u0370-\u167F\u1681-\u180D\u180F-\u1DBF]/,/[\u1E00-\u1FFF]/,/[\u200B-\u200D\u202A-\u202E\u203F-\u2040\u2054\u2060-\u206F]/,/[\u2070-\u20CF\u2100-\u218F\u2460-\u24FF\u2776-\u2793]/,/[\u2C00-\u2DFF\u2E80-\u2FFF]/,/[\u3004-\u3007\u3021-\u302F\u3031-\u303F\u3040-\uD7FF]/,/[\uF900-\uFD3D\uFD40-\uFDCF\uFDF0-\uFE1F\uFE30-\uFE44]/,/[\uFE47-\uFEFE\uFF00-\uFFFD]/),ze=m($e,/\d/,/[\u0300-\u036F\u1DC0-\u1DFF\u20D0-\u20FF\uFE20-\uFE2F]/),Fe=b($e,ze,"*"),Ue=b(/[A-Z]/,ze,"*"),je=["attached","autoclosure",b(/convention\(/,m("swift","block","c"),/\)/),"discardableResult","dynamicCallable","dynamicMemberLookup","escaping","freestanding","frozen","GKInspectable","IBAction","IBDesignable","IBInspectable","IBOutlet","IBSegueAction","inlinable","main","nonobjc","NSApplicationMain","NSCopying","NSManaged",b(/objc\(/,Fe,/\)/),"objc","objcMembers","propertyWrapper","requires_stored_property_inits","resultBuilder","Sendable","testable","UIApplicationMain","unchecked","unknown","usableFromInline","warn_unqualified_access"],Pe=["iOS","iOSApplicationExtension","macOS","macOSApplicationExtension","macCatalyst","macCatalystApplicationExtension","watchOS","watchOSApplicationExtension","tvOS","tvOSApplicationExtension","swift"] | |
| ;var Ke=Object.freeze({__proto__:null,grmr_bash:e=>{const n=e.regex,t={},a={ | |
| begin:/\$\{/,end:/\}/,contains:["self",{begin:/:-/,contains:[t]}]} | |
| ;Object.assign(t,{className:"variable",variants:[{ | |
| begin:n.concat(/\$[\w\d#@][\w\d_]*/,"(?![\\w\\d])(?![$])")},a]});const i={ | |
| className:"subst",begin:/\$\(/,end:/\)/,contains:[e.BACKSLASH_ESCAPE]},r={ | |
| begin:/<<-?\s*(?=\w+)/,starts:{contains:[e.END_SAME_AS_BEGIN({begin:/(\w+)/, | |
| end:/(\w+)/,className:"string"})]}},s={className:"string",begin:/"/,end:/"/, | |
| contains:[e.BACKSLASH_ESCAPE,t,i]};i.contains.push(s);const o={begin:/\$?\(\(/, | |
| end:/\)\)/,contains:[{begin:/\d+#[0-9a-f]+/,className:"number"},e.NUMBER_MODE,t] | |
| },l=e.SHEBANG({binary:"(fish|bash|zsh|sh|csh|ksh|tcsh|dash|scsh)",relevance:10 | |
| }),c={className:"function",begin:/\w[\w\d_]*\s*\(\s*\)\s*\{/,returnBegin:!0, | |
| contains:[e.inherit(e.TITLE_MODE,{begin:/\w[\w\d_]*/})],relevance:0};return{ | |
| name:"Bash",aliases:["sh"],keywords:{$pattern:/\b[a-z][a-z0-9._-]+\b/, | |
| keyword:["if","then","else","elif","fi","for","while","until","in","do","done","case","esac","function","select"], | |
| literal:["true","false"], | |
| built_in:["break","cd","continue","eval","exec","exit","export","getopts","hash","pwd","readonly","return","shift","test","times","trap","umask","unset","alias","bind","builtin","caller","command","declare","echo","enable","help","let","local","logout","mapfile","printf","read","readarray","source","type","typeset","ulimit","unalias","set","shopt","autoload","bg","bindkey","bye","cap","chdir","clone","comparguments","compcall","compctl","compdescribe","compfiles","compgroups","compquote","comptags","comptry","compvalues","dirs","disable","disown","echotc","echoti","emulate","fc","fg","float","functions","getcap","getln","history","integer","jobs","kill","limit","log","noglob","popd","print","pushd","pushln","rehash","sched","setcap","setopt","stat","suspend","ttyctl","unfunction","unhash","unlimit","unsetopt","vared","wait","whence","where","which","zcompile","zformat","zftp","zle","zmodload","zparseopts","zprof","zpty","zregexparse","zsocket","zstyle","ztcp","chcon","chgrp","chown","chmod","cp","dd","df","dir","dircolors","ln","ls","mkdir","mkfifo","mknod","mktemp","mv","realpath","rm","rmdir","shred","sync","touch","truncate","vdir","b2sum","base32","base64","cat","cksum","comm","csplit","cut","expand","fmt","fold","head","join","md5sum","nl","numfmt","od","paste","ptx","pr","sha1sum","sha224sum","sha256sum","sha384sum","sha512sum","shuf","sort","split","sum","tac","tail","tr","tsort","unexpand","uniq","wc","arch","basename","chroot","date","dirname","du","echo","env","expr","factor","groups","hostid","id","link","logname","nice","nohup","nproc","pathchk","pinky","printenv","printf","pwd","readlink","runcon","seq","sleep","stat","stdbuf","stty","tee","test","timeout","tty","uname","unlink","uptime","users","who","whoami","yes"] | |
| },contains:[l,e.SHEBANG(),c,o,e.HASH_COMMENT_MODE,r,{match:/(\/[a-z._-]+)+/},s,{ | |
| match:/\\"/},{className:"string",begin:/'/,end:/'/},{match:/\\'/},t]}}, | |
| grmr_c:e=>{const n=e.regex,t=e.COMMENT("//","$",{contains:[{begin:/\\\n/}] | |
| }),a="decltype\\(auto\\)",i="[a-zA-Z_]\\w*::",r="("+a+"|"+n.optional(i)+"[a-zA-Z_]\\w*"+n.optional("<[^<>]+>")+")",s={ | |
| className:"type",variants:[{begin:"\\b[a-z\\d_]*_t\\b"},{ | |
| match:/\batomic_[a-z]{3,6}\b/}]},o={className:"string",variants:[{ | |
| begin:'(u8?|U|L)?"',end:'"',illegal:"\\n",contains:[e.BACKSLASH_ESCAPE]},{ | |
| begin:"(u8?|U|L)?'(\\\\(x[0-9A-Fa-f]{2}|u[0-9A-Fa-f]{4,8}|[0-7]{3}|\\S)|.)", | |
| end:"'",illegal:"."},e.END_SAME_AS_BEGIN({ | |
| begin:/(?:u8?|U|L)?R"([^()\\ ]{0,16})\(/,end:/\)([^()\\ ]{0,16})"/})]},l={ | |
| className:"number",variants:[{begin:"\\b(0b[01']+)"},{ | |
| begin:"(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)((ll|LL|l|L)(u|U)?|(u|U)(ll|LL|l|L)?|f|F|b|B)" | |
| },{ | |
| begin:"(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)" | |
| }],relevance:0},c={className:"meta",begin:/#\s*[a-z]+\b/,end:/$/,keywords:{ | |
| keyword:"if else elif endif define undef warning error line pragma _Pragma ifdef ifndef include" | |
| },contains:[{begin:/\\\n/,relevance:0},e.inherit(o,{className:"string"}),{ | |
| className:"string",begin:/<.*?>/},t,e.C_BLOCK_COMMENT_MODE]},d={ | |
| className:"title",begin:n.optional(i)+e.IDENT_RE,relevance:0 | |
| },g=n.optional(i)+e.IDENT_RE+"\\s*\\(",u={ | |
| keyword:["asm","auto","break","case","continue","default","do","else","enum","extern","for","fortran","goto","if","inline","register","restrict","return","sizeof","struct","switch","typedef","union","volatile","while","_Alignas","_Alignof","_Atomic","_Generic","_Noreturn","_Static_assert","_Thread_local","alignas","alignof","noreturn","static_assert","thread_local","_Pragma"], | |
| type:["float","double","signed","unsigned","int","short","long","char","void","_Bool","_Complex","_Imaginary","_Decimal32","_Decimal64","_Decimal128","const","static","complex","bool","imaginary"], | |
| literal:"true false NULL", | |
| built_in:"std string wstring cin cout cerr clog stdin stdout stderr stringstream istringstream ostringstream auto_ptr deque list queue stack vector map set pair bitset multiset multimap unordered_set unordered_map unordered_multiset unordered_multimap priority_queue make_pair array shared_ptr abort terminate abs acos asin atan2 atan calloc ceil cosh cos exit exp fabs floor fmod fprintf fputs free frexp fscanf future isalnum isalpha iscntrl isdigit isgraph islower isprint ispunct isspace isupper isxdigit tolower toupper labs ldexp log10 log malloc realloc memchr memcmp memcpy memset modf pow printf putchar puts scanf sinh sin snprintf sprintf sqrt sscanf strcat strchr strcmp strcpy strcspn strlen strncat strncmp strncpy strpbrk strrchr strspn strstr tanh tan vfprintf vprintf vsprintf endl initializer_list unique_ptr" | |
| },b=[c,s,t,e.C_BLOCK_COMMENT_MODE,l,o],m={variants:[{begin:/=/,end:/;/},{ | |
| begin:/\(/,end:/\)/},{beginKeywords:"new throw return else",end:/;/}], | |
| keywords:u,contains:b.concat([{begin:/\(/,end:/\)/,keywords:u, | |
| contains:b.concat(["self"]),relevance:0}]),relevance:0},p={ | |
| begin:"("+r+"[\\*&\\s]+)+"+g,returnBegin:!0,end:/[{;=]/,excludeEnd:!0, | |
| keywords:u,illegal:/[^\w\s\*&:<>.]/,contains:[{begin:a,keywords:u,relevance:0},{ | |
| begin:g,returnBegin:!0,contains:[e.inherit(d,{className:"title.function"})], | |
| relevance:0},{relevance:0,match:/,/},{className:"params",begin:/\(/,end:/\)/, | |
| keywords:u,relevance:0,contains:[t,e.C_BLOCK_COMMENT_MODE,o,l,s,{begin:/\(/, | |
| end:/\)/,keywords:u,relevance:0,contains:["self",t,e.C_BLOCK_COMMENT_MODE,o,l,s] | |
| }]},s,t,e.C_BLOCK_COMMENT_MODE,c]};return{name:"C",aliases:["h"],keywords:u, | |
| disableAutodetect:!0,illegal:"</",contains:[].concat(m,p,b,[c,{ | |
| begin:e.IDENT_RE+"::",keywords:u},{className:"class", | |
| beginKeywords:"enum class struct union",end:/[{;:<>=]/,contains:[{ | |
| beginKeywords:"final class struct"},e.TITLE_MODE]}]),exports:{preprocessor:c, | |
| strings:o,keywords:u}}},grmr_cpp:e=>{const n=e.regex,t=e.COMMENT("//","$",{ | |
| contains:[{begin:/\\\n/}] | |
| }),a="decltype\\(auto\\)",i="[a-zA-Z_]\\w*::",r="(?!struct)("+a+"|"+n.optional(i)+"[a-zA-Z_]\\w*"+n.optional("<[^<>]+>")+")",s={ | |
| className:"type",begin:"\\b[a-z\\d_]*_t\\b"},o={className:"string",variants:[{ | |
| begin:'(u8?|U|L)?"',end:'"',illegal:"\\n",contains:[e.BACKSLASH_ESCAPE]},{ | |
| begin:"(u8?|U|L)?'(\\\\(x[0-9A-Fa-f]{2}|u[0-9A-Fa-f]{4,8}|[0-7]{3}|\\S)|.)", | |
| end:"'",illegal:"."},e.END_SAME_AS_BEGIN({ | |
| begin:/(?:u8?|U|L)?R"([^()\\ ]{0,16})\(/,end:/\)([^()\\ ]{0,16})"/})]},l={ | |
| className:"number",variants:[{begin:"\\b(0b[01']+)"},{ | |
| begin:"(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)((ll|LL|l|L)(u|U)?|(u|U)(ll|LL|l|L)?|f|F|b|B)" | |
| },{ | |
| begin:"(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)" | |
| }],relevance:0},c={className:"meta",begin:/#\s*[a-z]+\b/,end:/$/,keywords:{ | |
| keyword:"if else elif endif define undef warning error line pragma _Pragma ifdef ifndef include" | |
| },contains:[{begin:/\\\n/,relevance:0},e.inherit(o,{className:"string"}),{ | |
| className:"string",begin:/<.*?>/},t,e.C_BLOCK_COMMENT_MODE]},d={ | |
| className:"title",begin:n.optional(i)+e.IDENT_RE,relevance:0 | |
| },g=n.optional(i)+e.IDENT_RE+"\\s*\\(",u={ | |
| type:["bool","char","char16_t","char32_t","char8_t","double","float","int","long","short","void","wchar_t","unsigned","signed","const","static"], | |
| keyword:["alignas","alignof","and","and_eq","asm","atomic_cancel","atomic_commit","atomic_noexcept","auto","bitand","bitor","break","case","catch","class","co_await","co_return","co_yield","compl","concept","const_cast|10","consteval","constexpr","constinit","continue","decltype","default","delete","do","dynamic_cast|10","else","enum","explicit","export","extern","false","final","for","friend","goto","if","import","inline","module","mutable","namespace","new","noexcept","not","not_eq","nullptr","operator","or","or_eq","override","private","protected","public","reflexpr","register","reinterpret_cast|10","requires","return","sizeof","static_assert","static_cast|10","struct","switch","synchronized","template","this","thread_local","throw","transaction_safe","transaction_safe_dynamic","true","try","typedef","typeid","typename","union","using","virtual","volatile","while","xor","xor_eq"], | |
| literal:["NULL","false","nullopt","nullptr","true"],built_in:["_Pragma"], | |
| _type_hints:["any","auto_ptr","barrier","binary_semaphore","bitset","complex","condition_variable","condition_variable_any","counting_semaphore","deque","false_type","future","imaginary","initializer_list","istringstream","jthread","latch","lock_guard","multimap","multiset","mutex","optional","ostringstream","packaged_task","pair","promise","priority_queue","queue","recursive_mutex","recursive_timed_mutex","scoped_lock","set","shared_future","shared_lock","shared_mutex","shared_timed_mutex","shared_ptr","stack","string_view","stringstream","timed_mutex","thread","true_type","tuple","unique_lock","unique_ptr","unordered_map","unordered_multimap","unordered_multiset","unordered_set","variant","vector","weak_ptr","wstring","wstring_view"] | |
| },b={className:"function.dispatch",relevance:0,keywords:{ | |
| _hint:["abort","abs","acos","apply","as_const","asin","atan","atan2","calloc","ceil","cerr","cin","clog","cos","cosh","cout","declval","endl","exchange","exit","exp","fabs","floor","fmod","forward","fprintf","fputs","free","frexp","fscanf","future","invoke","isalnum","isalpha","iscntrl","isdigit","isgraph","islower","isprint","ispunct","isspace","isupper","isxdigit","labs","launder","ldexp","log","log10","make_pair","make_shared","make_shared_for_overwrite","make_tuple","make_unique","malloc","memchr","memcmp","memcpy","memset","modf","move","pow","printf","putchar","puts","realloc","scanf","sin","sinh","snprintf","sprintf","sqrt","sscanf","std","stderr","stdin","stdout","strcat","strchr","strcmp","strcpy","strcspn","strlen","strncat","strncmp","strncpy","strpbrk","strrchr","strspn","strstr","swap","tan","tanh","terminate","to_underlying","tolower","toupper","vfprintf","visit","vprintf","vsprintf"] | |
| }, | |
| begin:n.concat(/\b/,/(?!decltype)/,/(?!if)/,/(?!for)/,/(?!switch)/,/(?!while)/,e.IDENT_RE,n.lookahead(/(<[^<>]+>|)\s*\(/)) | |
| },m=[b,c,s,t,e.C_BLOCK_COMMENT_MODE,l,o],p={variants:[{begin:/=/,end:/;/},{ | |
| begin:/\(/,end:/\)/},{beginKeywords:"new throw return else",end:/;/}], | |
| keywords:u,contains:m.concat([{begin:/\(/,end:/\)/,keywords:u, | |
| contains:m.concat(["self"]),relevance:0}]),relevance:0},_={className:"function", | |
| begin:"("+r+"[\\*&\\s]+)+"+g,returnBegin:!0,end:/[{;=]/,excludeEnd:!0, | |
| keywords:u,illegal:/[^\w\s\*&:<>.]/,contains:[{begin:a,keywords:u,relevance:0},{ | |
| begin:g,returnBegin:!0,contains:[d],relevance:0},{begin:/::/,relevance:0},{ | |
| begin:/:/,endsWithParent:!0,contains:[o,l]},{relevance:0,match:/,/},{ | |
| className:"params",begin:/\(/,end:/\)/,keywords:u,relevance:0, | |
| contains:[t,e.C_BLOCK_COMMENT_MODE,o,l,s,{begin:/\(/,end:/\)/,keywords:u, | |
| relevance:0,contains:["self",t,e.C_BLOCK_COMMENT_MODE,o,l,s]}] | |
| },s,t,e.C_BLOCK_COMMENT_MODE,c]};return{name:"C++", | |
| aliases:["cc","c++","h++","hpp","hh","hxx","cxx"],keywords:u,illegal:"</", | |
| classNameAliases:{"function.dispatch":"built_in"}, | |
| contains:[].concat(p,_,b,m,[c,{ | |
| begin:"\\b(deque|list|queue|priority_queue|pair|stack|vector|map|set|bitset|multiset|multimap|unordered_map|unordered_set|unordered_multiset|unordered_multimap|array|tuple|optional|variant|function)\\s*<(?!<)", | |
| end:">",keywords:u,contains:["self",s]},{begin:e.IDENT_RE+"::",keywords:u},{ | |
| match:[/\b(?:enum(?:\s+(?:class|struct))?|class|struct|union)/,/\s+/,/\w+/], | |
| className:{1:"keyword",3:"title.class"}}])}},grmr_csharp:e=>{const n={ | |
| keyword:["abstract","as","base","break","case","catch","class","const","continue","do","else","event","explicit","extern","finally","fixed","for","foreach","goto","if","implicit","in","interface","internal","is","lock","namespace","new","operator","out","override","params","private","protected","public","readonly","record","ref","return","scoped","sealed","sizeof","stackalloc","static","struct","switch","this","throw","try","typeof","unchecked","unsafe","using","virtual","void","volatile","while"].concat(["add","alias","and","ascending","async","await","by","descending","equals","from","get","global","group","init","into","join","let","nameof","not","notnull","on","or","orderby","partial","remove","select","set","unmanaged","value|0","var","when","where","with","yield"]), | |
| built_in:["bool","byte","char","decimal","delegate","double","dynamic","enum","float","int","long","nint","nuint","object","sbyte","short","string","ulong","uint","ushort"], | |
| literal:["default","false","null","true"]},t=e.inherit(e.TITLE_MODE,{ | |
| begin:"[a-zA-Z](\\.?\\w)*"}),a={className:"number",variants:[{ | |
| begin:"\\b(0b[01']+)"},{ | |
| begin:"(-?)\\b([\\d']+(\\.[\\d']*)?|\\.[\\d']+)(u|U|l|L|ul|UL|f|F|b|B)"},{ | |
| begin:"(-?)(\\b0[xX][a-fA-F0-9']+|(\\b[\\d']+(\\.[\\d']*)?|\\.[\\d']+)([eE][-+]?[\\d']+)?)" | |
| }],relevance:0},i={className:"string",begin:'@"',end:'"',contains:[{begin:'""'}] | |
| },r=e.inherit(i,{illegal:/\n/}),s={className:"subst",begin:/\{/,end:/\}/, | |
| keywords:n},o=e.inherit(s,{illegal:/\n/}),l={className:"string",begin:/\$"/, | |
| end:'"',illegal:/\n/,contains:[{begin:/\{\{/},{begin:/\}\}/ | |
| },e.BACKSLASH_ESCAPE,o]},c={className:"string",begin:/\$@"/,end:'"',contains:[{ | |
| begin:/\{\{/},{begin:/\}\}/},{begin:'""'},s]},d=e.inherit(c,{illegal:/\n/, | |
| contains:[{begin:/\{\{/},{begin:/\}\}/},{begin:'""'},o]}) | |
| ;s.contains=[c,l,i,e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,a,e.C_BLOCK_COMMENT_MODE], | |
| o.contains=[d,l,r,e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,a,e.inherit(e.C_BLOCK_COMMENT_MODE,{ | |
| illegal:/\n/})];const g={variants:[c,l,i,e.APOS_STRING_MODE,e.QUOTE_STRING_MODE] | |
| },u={begin:"<",end:">",contains:[{beginKeywords:"in out"},t] | |
| },b=e.IDENT_RE+"(<"+e.IDENT_RE+"(\\s*,\\s*"+e.IDENT_RE+")*>)?(\\[\\])?",m={ | |
| begin:"@"+e.IDENT_RE,relevance:0};return{name:"C#",aliases:["cs","c#"], | |
| keywords:n,illegal:/::/,contains:[e.COMMENT("///","$",{returnBegin:!0, | |
| contains:[{className:"doctag",variants:[{begin:"///",relevance:0},{ | |
| begin:"\x3c!--|--\x3e"},{begin:"</?",end:">"}]}] | |
| }),e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE,{className:"meta",begin:"#", | |
| end:"$",keywords:{ | |
| keyword:"if else elif endif define undef warning error line region endregion pragma checksum" | |
| }},g,a,{beginKeywords:"class interface",relevance:0,end:/[{;=]/, | |
| illegal:/[^\s:,]/,contains:[{beginKeywords:"where class" | |
| },t,u,e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},{beginKeywords:"namespace", | |
| relevance:0,end:/[{;=]/,illegal:/[^\s:]/, | |
| contains:[t,e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},{ | |
| beginKeywords:"record",relevance:0,end:/[{;=]/,illegal:/[^\s:]/, | |
| contains:[t,u,e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},{className:"meta", | |
| begin:"^\\s*\\[(?=[\\w])",excludeBegin:!0,end:"\\]",excludeEnd:!0,contains:[{ | |
| className:"string",begin:/"/,end:/"/}]},{ | |
| beginKeywords:"new return throw await else",relevance:0},{className:"function", | |
| begin:"("+b+"\\s+)+"+e.IDENT_RE+"\\s*(<[^=]+>\\s*)?\\(",returnBegin:!0, | |
| end:/\s*[{;=]/,excludeEnd:!0,keywords:n,contains:[{ | |
| beginKeywords:"public private protected static internal protected abstract async extern override unsafe virtual new sealed partial", | |
| relevance:0},{begin:e.IDENT_RE+"\\s*(<[^=]+>\\s*)?\\(",returnBegin:!0, | |
| contains:[e.TITLE_MODE,u],relevance:0},{match:/\(\)/},{className:"params", | |
| begin:/\(/,end:/\)/,excludeBegin:!0,excludeEnd:!0,keywords:n,relevance:0, | |
| contains:[g,a,e.C_BLOCK_COMMENT_MODE] | |
| },e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},m]}},grmr_css:e=>{ | |
| const n=e.regex,t=ie(e),a=[e.APOS_STRING_MODE,e.QUOTE_STRING_MODE];return{ | |
| name:"CSS",case_insensitive:!0,illegal:/[=|'\$]/,keywords:{ | |
| keyframePosition:"from to"},classNameAliases:{keyframePosition:"selector-tag"}, | |
| contains:[t.BLOCK_COMMENT,{begin:/-(webkit|moz|ms|o)-(?=[a-z])/ | |
| },t.CSS_NUMBER_MODE,{className:"selector-id",begin:/#[A-Za-z0-9_-]+/,relevance:0 | |
| },{className:"selector-class",begin:"\\.[a-zA-Z-][a-zA-Z0-9_-]*",relevance:0 | |
| },t.ATTRIBUTE_SELECTOR_MODE,{className:"selector-pseudo",variants:[{ | |
| begin:":("+oe.join("|")+")"},{begin:":(:)?("+le.join("|")+")"}] | |
| },t.CSS_VARIABLE,{className:"attribute",begin:"\\b("+ce.join("|")+")\\b"},{ | |
| begin:/:/,end:/[;}{]/, | |
| contains:[t.BLOCK_COMMENT,t.HEXCOLOR,t.IMPORTANT,t.CSS_NUMBER_MODE,...a,{ | |
| begin:/(url|data-uri)\(/,end:/\)/,relevance:0,keywords:{built_in:"url data-uri" | |
| },contains:[...a,{className:"string",begin:/[^)]/,endsWithParent:!0, | |
| excludeEnd:!0}]},t.FUNCTION_DISPATCH]},{begin:n.lookahead(/@/),end:"[{;]", | |
| relevance:0,illegal:/:/,contains:[{className:"keyword",begin:/@-?\w[\w]*(-\w+)*/ | |
| },{begin:/\s/,endsWithParent:!0,excludeEnd:!0,relevance:0,keywords:{ | |
| $pattern:/[a-z-]+/,keyword:"and or not only",attribute:se.join(" ")},contains:[{ | |
| begin:/[a-z-]+(?=:)/,className:"attribute"},...a,t.CSS_NUMBER_MODE]}]},{ | |
| className:"selector-tag",begin:"\\b("+re.join("|")+")\\b"}]}},grmr_diff:e=>{ | |
| const n=e.regex;return{name:"Diff",aliases:["patch"],contains:[{ | |
| className:"meta",relevance:10, | |
| match:n.either(/^@@ +-\d+,\d+ +\+\d+,\d+ +@@/,/^\*\*\* +\d+,\d+ +\*\*\*\*$/,/^--- +\d+,\d+ +----$/) | |
| },{className:"comment",variants:[{ | |
| begin:n.either(/Index: /,/^index/,/={3,}/,/^-{3}/,/^\*{3} /,/^\+{3}/,/^diff --git/), | |
| end:/$/},{match:/^\*{15}$/}]},{className:"addition",begin:/^\+/,end:/$/},{ | |
| className:"deletion",begin:/^-/,end:/$/},{className:"addition",begin:/^!/, | |
| end:/$/}]}},grmr_go:e=>{const n={ | |
| keyword:["break","case","chan","const","continue","default","defer","else","fallthrough","for","func","go","goto","if","import","interface","map","package","range","return","select","struct","switch","type","var"], | |
| type:["bool","byte","complex64","complex128","error","float32","float64","int8","int16","int32","int64","string","uint8","uint16","uint32","uint64","int","uint","uintptr","rune"], | |
| literal:["true","false","iota","nil"], | |
| built_in:["append","cap","close","complex","copy","imag","len","make","new","panic","print","println","real","recover","delete"] | |
| };return{name:"Go",aliases:["golang"],keywords:n,illegal:"</", | |
| contains:[e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE,{className:"string", | |
| variants:[e.QUOTE_STRING_MODE,e.APOS_STRING_MODE,{begin:"`",end:"`"}]},{ | |
| className:"number",variants:[{begin:e.C_NUMBER_RE+"[i]",relevance:1 | |
| },e.C_NUMBER_MODE]},{begin:/:=/},{className:"function",beginKeywords:"func", | |
| end:"\\s*(\\{|$)",excludeEnd:!0,contains:[e.TITLE_MODE,{className:"params", | |
| begin:/\(/,end:/\)/,endsParent:!0,keywords:n,illegal:/["']/}]}]}}, | |
| grmr_graphql:e=>{const n=e.regex;return{name:"GraphQL",aliases:["gql"], | |
| case_insensitive:!0,disableAutodetect:!1,keywords:{ | |
| keyword:["query","mutation","subscription","type","input","schema","directive","interface","union","scalar","fragment","enum","on"], | |
| literal:["true","false","null"]}, | |
| contains:[e.HASH_COMMENT_MODE,e.QUOTE_STRING_MODE,e.NUMBER_MODE,{ | |
| scope:"punctuation",match:/[.]{3}/,relevance:0},{scope:"punctuation", | |
| begin:/[\!\(\)\:\=\[\]\{\|\}]{1}/,relevance:0},{scope:"variable",begin:/\$/, | |
| end:/\W/,excludeEnd:!0,relevance:0},{scope:"meta",match:/@\w+/,excludeEnd:!0},{ | |
| scope:"symbol",begin:n.concat(/[_A-Za-z][_0-9A-Za-z]*/,n.lookahead(/\s*:/)), | |
| relevance:0}],illegal:[/[;<']/,/BEGIN/]}},grmr_ini:e=>{const n=e.regex,t={ | |
| className:"number",relevance:0,variants:[{begin:/([+-]+)?[\d]+_[\d_]+/},{ | |
| begin:e.NUMBER_RE}]},a=e.COMMENT();a.variants=[{begin:/;/,end:/$/},{begin:/#/, | |
| end:/$/}];const i={className:"variable",variants:[{begin:/\$[\w\d"][\w\d_]*/},{ | |
| begin:/\$\{(.*?)\}/}]},r={className:"literal", | |
| begin:/\bon|off|true|false|yes|no\b/},s={className:"string", | |
| contains:[e.BACKSLASH_ESCAPE],variants:[{begin:"'''",end:"'''",relevance:10},{ | |
| begin:'"""',end:'"""',relevance:10},{begin:'"',end:'"'},{begin:"'",end:"'"}] | |
| },o={begin:/\[/,end:/\]/,contains:[a,r,i,s,t,"self"],relevance:0 | |
| },l=n.either(/[A-Za-z0-9_-]+/,/"(\\"|[^"])*"/,/'[^']*'/);return{ | |
| name:"TOML, also INI",aliases:["toml"],case_insensitive:!0,illegal:/\S/, | |
| contains:[a,{className:"section",begin:/\[+/,end:/\]+/},{ | |
| begin:n.concat(l,"(\\s*\\.\\s*",l,")*",n.lookahead(/\s*=\s*[^#\s]/)), | |
| className:"attr",starts:{end:/$/,contains:[a,o,r,i,s,t]}}]}},grmr_java:e=>{ | |
| const n=e.regex,t="[\xc0-\u02b8a-zA-Z_$][\xc0-\u02b8a-zA-Z_$0-9]*",a=t+pe("(?:<"+t+"~~~(?:\\s*,\\s*"+t+"~~~)*>)?",/~~~/g,2),i={ | |
| keyword:["synchronized","abstract","private","var","static","if","const ","for","while","strictfp","finally","protected","import","native","final","void","enum","else","break","transient","catch","instanceof","volatile","case","assert","package","default","public","try","switch","continue","throws","protected","public","private","module","requires","exports","do","sealed","yield","permits"], | |
| literal:["false","true","null"], | |
| type:["char","boolean","long","float","int","byte","short","double"], | |
| built_in:["super","this"]},r={className:"meta",begin:"@"+t,contains:[{ | |
| begin:/\(/,end:/\)/,contains:["self"]}]},s={className:"params",begin:/\(/, | |
| end:/\)/,keywords:i,relevance:0,contains:[e.C_BLOCK_COMMENT_MODE],endsParent:!0} | |
| ;return{name:"Java",aliases:["jsp"],keywords:i,illegal:/<\/|#/, | |
| contains:[e.COMMENT("/\\*\\*","\\*/",{relevance:0,contains:[{begin:/\w+@/, | |
| relevance:0},{className:"doctag",begin:"@[A-Za-z]+"}]}),{ | |
| begin:/import java\.[a-z]+\./,keywords:"import",relevance:2 | |
| },e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE,{begin:/"""/,end:/"""/, | |
| className:"string",contains:[e.BACKSLASH_ESCAPE] | |
| },e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,{ | |
| match:[/\b(?:class|interface|enum|extends|implements|new)/,/\s+/,t],className:{ | |
| 1:"keyword",3:"title.class"}},{match:/non-sealed/,scope:"keyword"},{ | |
| begin:[n.concat(/(?!else)/,t),/\s+/,t,/\s+/,/=(?!=)/],className:{1:"type", | |
| 3:"variable",5:"operator"}},{begin:[/record/,/\s+/,t],className:{1:"keyword", | |
| 3:"title.class"},contains:[s,e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},{ | |
| beginKeywords:"new throw return else",relevance:0},{ | |
| begin:["(?:"+a+"\\s+)",e.UNDERSCORE_IDENT_RE,/\s*(?=\()/],className:{ | |
| 2:"title.function"},keywords:i,contains:[{className:"params",begin:/\(/, | |
| end:/\)/,keywords:i,relevance:0, | |
| contains:[r,e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,me,e.C_BLOCK_COMMENT_MODE] | |
| },e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},me,r]}},grmr_javascript:Oe, | |
| grmr_json:e=>{const n=["true","false","null"],t={scope:"literal", | |
| beginKeywords:n.join(" ")};return{name:"JSON",keywords:{literal:n},contains:[{ | |
| className:"attr",begin:/"(\\.|[^\\"\r\n])*"(?=\s*:)/,relevance:1.01},{ | |
| match:/[{}[\],:]/,className:"punctuation",relevance:0 | |
| },e.QUOTE_STRING_MODE,t,e.C_NUMBER_MODE,e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE], | |
| illegal:"\\S"}},grmr_kotlin:e=>{const n={ | |
| keyword:"abstract as val var vararg get set class object open private protected public noinline crossinline dynamic final enum if else do while for when throw try catch finally import package is in fun override companion reified inline lateinit init interface annotation data sealed internal infix operator out by constructor super tailrec where const inner suspend typealias external expect actual", | |
| built_in:"Byte Short Char Int Long Boolean Float Double Void Unit Nothing", | |
| literal:"true false null"},t={className:"symbol",begin:e.UNDERSCORE_IDENT_RE+"@" | |
| },a={className:"subst",begin:/\$\{/,end:/\}/,contains:[e.C_NUMBER_MODE]},i={ | |
| className:"variable",begin:"\\$"+e.UNDERSCORE_IDENT_RE},r={className:"string", | |
| variants:[{begin:'"""',end:'"""(?=[^"])',contains:[i,a]},{begin:"'",end:"'", | |
| illegal:/\n/,contains:[e.BACKSLASH_ESCAPE]},{begin:'"',end:'"',illegal:/\n/, | |
| contains:[e.BACKSLASH_ESCAPE,i,a]}]};a.contains.push(r);const s={ | |
| className:"meta", | |
| begin:"@(?:file|property|field|get|set|receiver|param|setparam|delegate)\\s*:(?:\\s*"+e.UNDERSCORE_IDENT_RE+")?" | |
| },o={className:"meta",begin:"@"+e.UNDERSCORE_IDENT_RE,contains:[{begin:/\(/, | |
| end:/\)/,contains:[e.inherit(r,{className:"string"}),"self"]}] | |
| },l=me,c=e.COMMENT("/\\*","\\*/",{contains:[e.C_BLOCK_COMMENT_MODE]}),d={ | |
| variants:[{className:"type",begin:e.UNDERSCORE_IDENT_RE},{begin:/\(/,end:/\)/, | |
| contains:[]}]},g=d;return g.variants[1].contains=[d],d.variants[1].contains=[g], | |
| {name:"Kotlin",aliases:["kt","kts"],keywords:n, | |
| contains:[e.COMMENT("/\\*\\*","\\*/",{relevance:0,contains:[{className:"doctag", | |
| begin:"@[A-Za-z]+"}]}),e.C_LINE_COMMENT_MODE,c,{className:"keyword", | |
| begin:/\b(break|continue|return|this)\b/,starts:{contains:[{className:"symbol", | |
| begin:/@\w+/}]}},t,s,o,{className:"function",beginKeywords:"fun",end:"[(]|$", | |
| returnBegin:!0,excludeEnd:!0,keywords:n,relevance:5,contains:[{ | |
| begin:e.UNDERSCORE_IDENT_RE+"\\s*\\(",returnBegin:!0,relevance:0, | |
| contains:[e.UNDERSCORE_TITLE_MODE]},{className:"type",begin:/</,end:/>/, | |
| keywords:"reified",relevance:0},{className:"params",begin:/\(/,end:/\)/, | |
| endsParent:!0,keywords:n,relevance:0,contains:[{begin:/:/,end:/[=,\/]/, | |
| endsWithParent:!0,contains:[d,e.C_LINE_COMMENT_MODE,c],relevance:0 | |
| },e.C_LINE_COMMENT_MODE,c,s,o,r,e.C_NUMBER_MODE]},c]},{ | |
| begin:[/class|interface|trait/,/\s+/,e.UNDERSCORE_IDENT_RE],beginScope:{ | |
| 3:"title.class"},keywords:"class interface trait",end:/[:\{(]|$/,excludeEnd:!0, | |
| illegal:"extends implements",contains:[{ | |
| beginKeywords:"public protected internal private constructor" | |
| },e.UNDERSCORE_TITLE_MODE,{className:"type",begin:/</,end:/>/,excludeBegin:!0, | |
| excludeEnd:!0,relevance:0},{className:"type",begin:/[,:]\s*/,end:/[<\(,){\s]|$/, | |
| excludeBegin:!0,returnEnd:!0},s,o]},r,{className:"meta",begin:"^#!/usr/bin/env", | |
| end:"$",illegal:"\n"},l]}},grmr_less:e=>{ | |
| const n=ie(e),t=de,a="[\\w-]+",i="("+a+"|@\\{"+a+"\\})",r=[],s=[],o=e=>({ | |
| className:"string",begin:"~?"+e+".*?"+e}),l=(e,n,t)=>({className:e,begin:n, | |
| relevance:t}),c={$pattern:/[a-z-]+/,keyword:"and or not only", | |
| attribute:se.join(" ")},d={begin:"\\(",end:"\\)",contains:s,keywords:c, | |
| relevance:0} | |
| ;s.push(e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE,o("'"),o('"'),n.CSS_NUMBER_MODE,{ | |
| begin:"(url|data-uri)\\(",starts:{className:"string",end:"[\\)\\n]", | |
| excludeEnd:!0} | |
| },n.HEXCOLOR,d,l("variable","@@?"+a,10),l("variable","@\\{"+a+"\\}"),l("built_in","~?`[^`]*?`"),{ | |
| className:"attribute",begin:a+"\\s*:",end:":",returnBegin:!0,excludeEnd:!0 | |
| },n.IMPORTANT,{beginKeywords:"and not"},n.FUNCTION_DISPATCH);const g=s.concat({ | |
| begin:/\{/,end:/\}/,contains:r}),u={beginKeywords:"when",endsWithParent:!0, | |
| contains:[{beginKeywords:"and not"}].concat(s)},b={begin:i+"\\s*:", | |
| returnBegin:!0,end:/[;}]/,relevance:0,contains:[{begin:/-(webkit|moz|ms|o)-/ | |
| },n.CSS_VARIABLE,{className:"attribute",begin:"\\b("+ce.join("|")+")\\b", | |
| end:/(?=:)/,starts:{endsWithParent:!0,illegal:"[<=$]",relevance:0,contains:s}}] | |
| },m={className:"keyword", | |
| begin:"@(import|media|charset|font-face|(-[a-z]+-)?keyframes|supports|document|namespace|page|viewport|host)\\b", | |
| starts:{end:"[;{}]",keywords:c,returnEnd:!0,contains:s,relevance:0}},p={ | |
| className:"variable",variants:[{begin:"@"+a+"\\s*:",relevance:15},{begin:"@"+a | |
| }],starts:{end:"[;}]",returnEnd:!0,contains:g}},_={variants:[{ | |
| begin:"[\\.#:&\\[>]",end:"[;{}]"},{begin:i,end:/\{/}],returnBegin:!0, | |
| returnEnd:!0,illegal:"[<='$\"]",relevance:0, | |
| contains:[e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE,u,l("keyword","all\\b"),l("variable","@\\{"+a+"\\}"),{ | |
| begin:"\\b("+re.join("|")+")\\b",className:"selector-tag" | |
| },n.CSS_NUMBER_MODE,l("selector-tag",i,0),l("selector-id","#"+i),l("selector-class","\\."+i,0),l("selector-tag","&",0),n.ATTRIBUTE_SELECTOR_MODE,{ | |
| className:"selector-pseudo",begin:":("+oe.join("|")+")"},{ | |
| className:"selector-pseudo",begin:":(:)?("+le.join("|")+")"},{begin:/\(/, | |
| end:/\)/,relevance:0,contains:g},{begin:"!important"},n.FUNCTION_DISPATCH]},h={ | |
| begin:a+":(:)?"+`(${t.join("|")})`,returnBegin:!0,contains:[_]} | |
| ;return r.push(e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE,m,p,h,b,_,u,n.FUNCTION_DISPATCH), | |
| {name:"Less",case_insensitive:!0,illegal:"[=>'/<($\"]",contains:r}}, | |
| grmr_lua:e=>{const n="\\[=*\\[",t="\\]=*\\]",a={begin:n,end:t,contains:["self"] | |
| },i=[e.COMMENT("--(?!"+n+")","$"),e.COMMENT("--"+n,t,{contains:[a],relevance:10 | |
| })];return{name:"Lua",keywords:{$pattern:e.UNDERSCORE_IDENT_RE, | |
| literal:"true false nil", | |
| keyword:"and break do else elseif end for goto if in local not or repeat return then until while", | |
| built_in:"_G _ENV _VERSION __index __newindex __mode __call __metatable __tostring __len __gc __add __sub __mul __div __mod __pow __concat __unm __eq __lt __le assert collectgarbage dofile error getfenv getmetatable ipairs load loadfile loadstring module next pairs pcall print rawequal rawget rawset require select setfenv setmetatable tonumber tostring type unpack xpcall arg self coroutine resume yield status wrap create running debug getupvalue debug sethook getmetatable gethook setmetatable setlocal traceback setfenv getinfo setupvalue getlocal getregistry getfenv io lines write close flush open output type read stderr stdin input stdout popen tmpfile math log max acos huge ldexp pi cos tanh pow deg tan cosh sinh random randomseed frexp ceil floor rad abs sqrt modf asin min mod fmod log10 atan2 exp sin atan os exit setlocale date getenv difftime remove time clock tmpname rename execute package preload loadlib loaded loaders cpath config path seeall string sub upper len gfind rep find match char dump gmatch reverse byte format gsub lower table setn insert getn foreachi maxn foreach concat sort remove" | |
| },contains:i.concat([{className:"function",beginKeywords:"function",end:"\\)", | |
| contains:[e.inherit(e.TITLE_MODE,{ | |
| begin:"([_a-zA-Z]\\w*\\.)*([_a-zA-Z]\\w*:)?[_a-zA-Z]\\w*"}),{className:"params", | |
| begin:"\\(",endsWithParent:!0,contains:i}].concat(i) | |
| },e.C_NUMBER_MODE,e.APOS_STRING_MODE,e.QUOTE_STRING_MODE,{className:"string", | |
| begin:n,end:t,contains:[a],relevance:5}])}},grmr_makefile:e=>{const n={ | |
| className:"variable",variants:[{begin:"\\$\\("+e.UNDERSCORE_IDENT_RE+"\\)", | |
| contains:[e.BACKSLASH_ESCAPE]},{begin:/\$[@%<?\^\+\*]/}]},t={className:"string", | |
| begin:/"/,end:/"/,contains:[e.BACKSLASH_ESCAPE,n]},a={className:"variable", | |
| begin:/\$\([\w-]+\s/,end:/\)/,keywords:{ | |
| built_in:"subst patsubst strip findstring filter filter-out sort word wordlist firstword lastword dir notdir suffix basename addsuffix addprefix join wildcard realpath abspath error warning shell origin flavor foreach if or and call eval file value" | |
| },contains:[n]},i={begin:"^"+e.UNDERSCORE_IDENT_RE+"\\s*(?=[:+?]?=)"},r={ | |
| className:"section",begin:/^[^\s]+:/,end:/$/,contains:[n]};return{ | |
| name:"Makefile",aliases:["mk","mak","make"],keywords:{$pattern:/[\w-]+/, | |
| keyword:"define endef undefine ifdef ifndef ifeq ifneq else endif include -include sinclude override export unexport private vpath" | |
| },contains:[e.HASH_COMMENT_MODE,n,t,a,i,{className:"meta",begin:/^\.PHONY:/, | |
| end:/$/,keywords:{$pattern:/[\.\w]+/,keyword:".PHONY"}},r]}},grmr_markdown:e=>{ | |
| const n={begin:/<\/?[A-Za-z_]/,end:">",subLanguage:"xml",relevance:0},t={ | |
| variants:[{begin:/\[.+?\]\[.*?\]/,relevance:0},{ | |
| begin:/\[.+?\]\(((data|javascript|mailto):|(?:http|ftp)s?:\/\/).*?\)/, | |
| relevance:2},{ | |
| begin:e.regex.concat(/\[.+?\]\(/,/[A-Za-z][A-Za-z0-9+.-]*/,/:\/\/.*?\)/), | |
| relevance:2},{begin:/\[.+?\]\([./?&#].*?\)/,relevance:1},{ | |
| begin:/\[.*?\]\(.*?\)/,relevance:0}],returnBegin:!0,contains:[{match:/\[(?=\])/ | |
| },{className:"string",relevance:0,begin:"\\[",end:"\\]",excludeBegin:!0, | |
| returnEnd:!0},{className:"link",relevance:0,begin:"\\]\\(",end:"\\)", | |
| excludeBegin:!0,excludeEnd:!0},{className:"symbol",relevance:0,begin:"\\]\\[", | |
| end:"\\]",excludeBegin:!0,excludeEnd:!0}]},a={className:"strong",contains:[], | |
| variants:[{begin:/_{2}(?!\s)/,end:/_{2}/},{begin:/\*{2}(?!\s)/,end:/\*{2}/}] | |
| },i={className:"emphasis",contains:[],variants:[{begin:/\*(?![*\s])/,end:/\*/},{ | |
| begin:/_(?![_\s])/,end:/_/,relevance:0}]},r=e.inherit(a,{contains:[] | |
| }),s=e.inherit(i,{contains:[]});a.contains.push(s),i.contains.push(r) | |
| ;let o=[n,t];return[a,i,r,s].forEach((e=>{e.contains=e.contains.concat(o) | |
| })),o=o.concat(a,i),{name:"Markdown",aliases:["md","mkdown","mkd"],contains:[{ | |
| className:"section",variants:[{begin:"^#{1,6}",end:"$",contains:o},{ | |
| begin:"(?=^.+?\\n[=-]{2,}$)",contains:[{begin:"^[=-]*$"},{begin:"^",end:"\\n", | |
| contains:o}]}]},n,{className:"bullet",begin:"^[ \t]*([*+-]|(\\d+\\.))(?=\\s+)", | |
| end:"\\s+",excludeEnd:!0},a,i,{className:"quote",begin:"^>\\s+",contains:o, | |
| end:"$"},{className:"code",variants:[{begin:"(`{3,})[^`](.|\\n)*?\\1`*[ ]*"},{ | |
| begin:"(~{3,})[^~](.|\\n)*?\\1~*[ ]*"},{begin:"```",end:"```+[ ]*$"},{ | |
| begin:"~~~",end:"~~~+[ ]*$"},{begin:"`.+?`"},{begin:"(?=^( {4}|\\t))", | |
| contains:[{begin:"^( {4}|\\t)",end:"(\\n)$"}],relevance:0}]},{ | |
| begin:"^[-\\*]{3,}",end:"$"},t,{begin:/^\[[^\n]+\]:/,returnBegin:!0,contains:[{ | |
| className:"symbol",begin:/\[/,end:/\]/,excludeBegin:!0,excludeEnd:!0},{ | |
| className:"link",begin:/:\s*/,end:/$/,excludeBegin:!0}]}]}},grmr_objectivec:e=>{ | |
| const n=/[a-zA-Z@][a-zA-Z0-9_]*/,t={$pattern:n, | |
| keyword:["@interface","@class","@protocol","@implementation"]};return{ | |
| name:"Objective-C",aliases:["mm","objc","obj-c","obj-c++","objective-c++"], | |
| keywords:{"variable.language":["this","super"],$pattern:n, | |
| keyword:["while","export","sizeof","typedef","const","struct","for","union","volatile","static","mutable","if","do","return","goto","enum","else","break","extern","asm","case","default","register","explicit","typename","switch","continue","inline","readonly","assign","readwrite","self","@synchronized","id","typeof","nonatomic","IBOutlet","IBAction","strong","weak","copy","in","out","inout","bycopy","byref","oneway","__strong","__weak","__block","__autoreleasing","@private","@protected","@public","@try","@property","@end","@throw","@catch","@finally","@autoreleasepool","@synthesize","@dynamic","@selector","@optional","@required","@encode","@package","@import","@defs","@compatibility_alias","__bridge","__bridge_transfer","__bridge_retained","__bridge_retain","__covariant","__contravariant","__kindof","_Nonnull","_Nullable","_Null_unspecified","__FUNCTION__","__PRETTY_FUNCTION__","__attribute__","getter","setter","retain","unsafe_unretained","nonnull","nullable","null_unspecified","null_resettable","class","instancetype","NS_DESIGNATED_INITIALIZER","NS_UNAVAILABLE","NS_REQUIRES_SUPER","NS_RETURNS_INNER_POINTER","NS_INLINE","NS_AVAILABLE","NS_DEPRECATED","NS_ENUM","NS_OPTIONS","NS_SWIFT_UNAVAILABLE","NS_ASSUME_NONNULL_BEGIN","NS_ASSUME_NONNULL_END","NS_REFINED_FOR_SWIFT","NS_SWIFT_NAME","NS_SWIFT_NOTHROW","NS_DURING","NS_HANDLER","NS_ENDHANDLER","NS_VALUERETURN","NS_VOIDRETURN"], | |
| literal:["false","true","FALSE","TRUE","nil","YES","NO","NULL"], | |
| built_in:["dispatch_once_t","dispatch_queue_t","dispatch_sync","dispatch_async","dispatch_once"], | |
| type:["int","float","char","unsigned","signed","short","long","double","wchar_t","unichar","void","bool","BOOL","id|0","_Bool"] | |
| },illegal:"</",contains:[{className:"built_in", | |
| begin:"\\b(AV|CA|CF|CG|CI|CL|CM|CN|CT|MK|MP|MTK|MTL|NS|SCN|SK|UI|WK|XC)\\w+" | |
| },e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE,e.C_NUMBER_MODE,e.QUOTE_STRING_MODE,e.APOS_STRING_MODE,{ | |
| className:"string",variants:[{begin:'@"',end:'"',illegal:"\\n", | |
| contains:[e.BACKSLASH_ESCAPE]}]},{className:"meta",begin:/#\s*[a-z]+\b/,end:/$/, | |
| keywords:{ | |
| keyword:"if else elif endif define undef warning error line pragma ifdef ifndef include" | |
| },contains:[{begin:/\\\n/,relevance:0},e.inherit(e.QUOTE_STRING_MODE,{ | |
| className:"string"}),{className:"string",begin:/<.*?>/,end:/$/,illegal:"\\n" | |
| },e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE]},{className:"class", | |
| begin:"("+t.keyword.join("|")+")\\b",end:/(\{|$)/,excludeEnd:!0,keywords:t, | |
| contains:[e.UNDERSCORE_TITLE_MODE]},{begin:"\\."+e.UNDERSCORE_IDENT_RE, | |
| relevance:0}]}},grmr_perl:e=>{const n=e.regex,t=/[dualxmsipngr]{0,12}/,a={ | |
| $pattern:/[\w.]+/, | |
| keyword:"abs accept alarm and atan2 bind binmode bless break caller chdir chmod chomp chop chown chr chroot close closedir connect continue cos crypt dbmclose dbmopen defined delete die do dump each else elsif endgrent endhostent endnetent endprotoent endpwent endservent eof eval exec exists exit exp fcntl fileno flock for foreach fork format formline getc getgrent getgrgid getgrnam gethostbyaddr gethostbyname gethostent getlogin getnetbyaddr getnetbyname getnetent getpeername getpgrp getpriority getprotobyname getprotobynumber getprotoent getpwent getpwnam getpwuid getservbyname getservbyport getservent getsockname getsockopt given glob gmtime goto grep gt hex if index int ioctl join keys kill last lc lcfirst length link listen local localtime log lstat lt ma map mkdir msgctl msgget msgrcv msgsnd my ne next no not oct open opendir or ord our pack package pipe pop pos print printf prototype push q|0 qq quotemeta qw qx rand read readdir readline readlink readpipe recv redo ref rename require reset return reverse rewinddir rindex rmdir say scalar seek seekdir select semctl semget semop send setgrent sethostent setnetent setpgrp setpriority setprotoent setpwent setservent setsockopt shift shmctl shmget shmread shmwrite shutdown sin sleep socket socketpair sort splice split sprintf sqrt srand stat state study sub substr symlink syscall sysopen sysread sysseek system syswrite tell telldir tie tied time times tr truncate uc ucfirst umask undef unless unlink unpack unshift untie until use utime values vec wait waitpid wantarray warn when while write x|0 xor y|0" | |
| },i={className:"subst",begin:"[$@]\\{",end:"\\}",keywords:a},r={begin:/->\{/, | |
| end:/\}/},s={variants:[{begin:/\$\d/},{ | |
| begin:n.concat(/[$%@](\^\w\b|#\w+(::\w+)*|\{\w+\}|\w+(::\w*)*)/,"(?![A-Za-z])(?![@$%])") | |
| },{begin:/[$%@][^\s\w{]/,relevance:0}] | |
| },o=[e.BACKSLASH_ESCAPE,i,s],l=[/!/,/\//,/\|/,/\?/,/'/,/"/,/#/],c=(e,a,i="\\1")=>{ | |
| const r="\\1"===i?i:n.concat(i,a) | |
| ;return n.concat(n.concat("(?:",e,")"),a,/(?:\\.|[^\\\/])*?/,r,/(?:\\.|[^\\\/])*?/,i,t) | |
| },d=(e,a,i)=>n.concat(n.concat("(?:",e,")"),a,/(?:\\.|[^\\\/])*?/,i,t),g=[s,e.HASH_COMMENT_MODE,e.COMMENT(/^=\w/,/=cut/,{ | |
| endsWithParent:!0}),r,{className:"string",contains:o,variants:[{ | |
| begin:"q[qwxr]?\\s*\\(",end:"\\)",relevance:5},{begin:"q[qwxr]?\\s*\\[", | |
| end:"\\]",relevance:5},{begin:"q[qwxr]?\\s*\\{",end:"\\}",relevance:5},{ | |
| begin:"q[qwxr]?\\s*\\|",end:"\\|",relevance:5},{begin:"q[qwxr]?\\s*<",end:">", | |
| relevance:5},{begin:"qw\\s+q",end:"q",relevance:5},{begin:"'",end:"'", | |
| contains:[e.BACKSLASH_ESCAPE]},{begin:'"',end:'"'},{begin:"`",end:"`", | |
| contains:[e.BACKSLASH_ESCAPE]},{begin:/\{\w+\}/,relevance:0},{ | |
| begin:"-?\\w+\\s*=>",relevance:0}]},{className:"number", | |
| begin:"(\\b0[0-7_]+)|(\\b0x[0-9a-fA-F_]+)|(\\b[1-9][0-9_]*(\\.[0-9_]+)?)|[0_]\\b", | |
| relevance:0},{ | |
| begin:"(\\/\\/|"+e.RE_STARTERS_RE+"|\\b(split|return|print|reverse|grep)\\b)\\s*", | |
| keywords:"split return print reverse grep",relevance:0, | |
| contains:[e.HASH_COMMENT_MODE,{className:"regexp",variants:[{ | |
| begin:c("s|tr|y",n.either(...l,{capture:!0}))},{begin:c("s|tr|y","\\(","\\)")},{ | |
| begin:c("s|tr|y","\\[","\\]")},{begin:c("s|tr|y","\\{","\\}")}],relevance:2},{ | |
| className:"regexp",variants:[{begin:/(m|qr)\/\//,relevance:0},{ | |
| begin:d("(?:m|qr)?",/\//,/\//)},{begin:d("m|qr",n.either(...l,{capture:!0 | |
| }),/\1/)},{begin:d("m|qr",/\(/,/\)/)},{begin:d("m|qr",/\[/,/\]/)},{ | |
| begin:d("m|qr",/\{/,/\}/)}]}]},{className:"function",beginKeywords:"sub", | |
| end:"(\\s*\\(.*?\\))?[;{]",excludeEnd:!0,relevance:5,contains:[e.TITLE_MODE]},{ | |
| begin:"-\\w\\b",relevance:0},{begin:"^__DATA__$",end:"^__END__$", | |
| subLanguage:"mojolicious",contains:[{begin:"^@@.*",end:"$",className:"comment"}] | |
| }];return i.contains=g,r.contains=g,{name:"Perl",aliases:["pl","pm"],keywords:a, | |
| contains:g}},grmr_php:e=>{ | |
| const n=e.regex,t=/(?![A-Za-z0-9])(?![$])/,a=n.concat(/[a-zA-Z_\x7f-\xff][a-zA-Z0-9_\x7f-\xff]*/,t),i=n.concat(/(\\?[A-Z][a-z0-9_\x7f-\xff]+|\\?[A-Z]+(?=[A-Z][a-z0-9_\x7f-\xff])){1,}/,t),r={ | |
| scope:"variable",match:"\\$+"+a},s={scope:"subst",variants:[{begin:/\$\w+/},{ | |
| begin:/\{\$/,end:/\}/}]},o=e.inherit(e.APOS_STRING_MODE,{illegal:null | |
| }),l="[ \t\n]",c={scope:"string",variants:[e.inherit(e.QUOTE_STRING_MODE,{ | |
| illegal:null,contains:e.QUOTE_STRING_MODE.contains.concat(s)}),o,{ | |
| begin:/<<<[ \t]*(?:(\w+)|"(\w+)")\n/,end:/[ \t]*(\w+)\b/, | |
| contains:e.QUOTE_STRING_MODE.contains.concat(s),"on:begin":(e,n)=>{ | |
| n.data._beginMatch=e[1]||e[2]},"on:end":(e,n)=>{ | |
| n.data._beginMatch!==e[1]&&n.ignoreMatch()}},e.END_SAME_AS_BEGIN({ | |
| begin:/<<<[ \t]*'(\w+)'\n/,end:/[ \t]*(\w+)\b/})]},d={scope:"number",variants:[{ | |
| begin:"\\b0[bB][01]+(?:_[01]+)*\\b"},{begin:"\\b0[oO][0-7]+(?:_[0-7]+)*\\b"},{ | |
| begin:"\\b0[xX][\\da-fA-F]+(?:_[\\da-fA-F]+)*\\b"},{ | |
| begin:"(?:\\b\\d+(?:_\\d+)*(\\.(?:\\d+(?:_\\d+)*))?|\\B\\.\\d+)(?:[eE][+-]?\\d+)?" | |
| }],relevance:0 | |
| },g=["false","null","true"],u=["__CLASS__","__DIR__","__FILE__","__FUNCTION__","__COMPILER_HALT_OFFSET__","__LINE__","__METHOD__","__NAMESPACE__","__TRAIT__","die","echo","exit","include","include_once","print","require","require_once","array","abstract","and","as","binary","bool","boolean","break","callable","case","catch","class","clone","const","continue","declare","default","do","double","else","elseif","empty","enddeclare","endfor","endforeach","endif","endswitch","endwhile","enum","eval","extends","final","finally","float","for","foreach","from","global","goto","if","implements","instanceof","insteadof","int","integer","interface","isset","iterable","list","match|0","mixed","new","never","object","or","private","protected","public","readonly","real","return","string","switch","throw","trait","try","unset","use","var","void","while","xor","yield"],b=["Error|0","AppendIterator","ArgumentCountError","ArithmeticError","ArrayIterator","ArrayObject","AssertionError","BadFunctionCallException","BadMethodCallException","CachingIterator","CallbackFilterIterator","CompileError","Countable","DirectoryIterator","DivisionByZeroError","DomainException","EmptyIterator","ErrorException","Exception","FilesystemIterator","FilterIterator","GlobIterator","InfiniteIterator","InvalidArgumentException","IteratorIterator","LengthException","LimitIterator","LogicException","MultipleIterator","NoRewindIterator","OutOfBoundsException","OutOfRangeException","OuterIterator","OverflowException","ParentIterator","ParseError","RangeException","RecursiveArrayIterator","RecursiveCachingIterator","RecursiveCallbackFilterIterator","RecursiveDirectoryIterator","RecursiveFilterIterator","RecursiveIterator","RecursiveIteratorIterator","RecursiveRegexIterator","RecursiveTreeIterator","RegexIterator","RuntimeException","SeekableIterator","SplDoublyLinkedList","SplFileInfo","SplFileObject","SplFixedArray","SplHeap","SplMaxHeap","SplMinHeap","SplObjectStorage","SplObserver","SplPriorityQueue","SplQueue","SplStack","SplSubject","SplTempFileObject","TypeError","UnderflowException","UnexpectedValueException","UnhandledMatchError","ArrayAccess","BackedEnum","Closure","Fiber","Generator","Iterator","IteratorAggregate","Serializable","Stringable","Throwable","Traversable","UnitEnum","WeakReference","WeakMap","Directory","__PHP_Incomplete_Class","parent","php_user_filter","self","static","stdClass"],m={ | |
| keyword:u,literal:(e=>{const n=[];return e.forEach((e=>{ | |
| n.push(e),e.toLowerCase()===e?n.push(e.toUpperCase()):n.push(e.toLowerCase()) | |
| })),n})(g),built_in:b},p=e=>e.map((e=>e.replace(/\|\d+$/,""))),_={variants:[{ | |
| match:[/new/,n.concat(l,"+"),n.concat("(?!",p(b).join("\\b|"),"\\b)"),i],scope:{ | |
| 1:"keyword",4:"title.class"}}]},h=n.concat(a,"\\b(?!\\()"),f={variants:[{ | |
| match:[n.concat(/::/,n.lookahead(/(?!class\b)/)),h],scope:{2:"variable.constant" | |
| }},{match:[/::/,/class/],scope:{2:"variable.language"}},{ | |
| match:[i,n.concat(/::/,n.lookahead(/(?!class\b)/)),h],scope:{1:"title.class", | |
| 3:"variable.constant"}},{match:[i,n.concat("::",n.lookahead(/(?!class\b)/))], | |
| scope:{1:"title.class"}},{match:[i,/::/,/class/],scope:{1:"title.class", | |
| 3:"variable.language"}}]},E={scope:"attr", | |
| match:n.concat(a,n.lookahead(":"),n.lookahead(/(?!::)/))},y={relevance:0, | |
| begin:/\(/,end:/\)/,keywords:m,contains:[E,r,f,e.C_BLOCK_COMMENT_MODE,c,d,_] | |
| },N={relevance:0, | |
| match:[/\b/,n.concat("(?!fn\\b|function\\b|",p(u).join("\\b|"),"|",p(b).join("\\b|"),"\\b)"),a,n.concat(l,"*"),n.lookahead(/(?=\()/)], | |
| scope:{3:"title.function.invoke"},contains:[y]};y.contains.push(N) | |
| ;const w=[E,f,e.C_BLOCK_COMMENT_MODE,c,d,_];return{case_insensitive:!1, | |
| keywords:m,contains:[{begin:n.concat(/#\[\s*/,i),beginScope:"meta",end:/]/, | |
| endScope:"meta",keywords:{literal:g,keyword:["new","array"]},contains:[{ | |
| begin:/\[/,end:/]/,keywords:{literal:g,keyword:["new","array"]}, | |
| contains:["self",...w]},...w,{scope:"meta",match:i}] | |
| },e.HASH_COMMENT_MODE,e.COMMENT("//","$"),e.COMMENT("/\\*","\\*/",{contains:[{ | |
| scope:"doctag",match:"@[A-Za-z]+"}]}),{match:/__halt_compiler\(\);/, | |
| keywords:"__halt_compiler",starts:{scope:"comment",end:e.MATCH_NOTHING_RE, | |
| contains:[{match:/\?>/,scope:"meta",endsParent:!0}]}},{scope:"meta",variants:[{ | |
| begin:/<\?php/,relevance:10},{begin:/<\?=/},{begin:/<\?/,relevance:.1},{ | |
| begin:/\?>/}]},{scope:"variable.language",match:/\$this\b/},r,N,f,{ | |
| match:[/const/,/\s/,a],scope:{1:"keyword",3:"variable.constant"}},_,{ | |
| scope:"function",relevance:0,beginKeywords:"fn function",end:/[;{]/, | |
| excludeEnd:!0,illegal:"[$%\\[]",contains:[{beginKeywords:"use" | |
| },e.UNDERSCORE_TITLE_MODE,{begin:"=>",endsParent:!0},{scope:"params", | |
| begin:"\\(",end:"\\)",excludeBegin:!0,excludeEnd:!0,keywords:m, | |
| contains:["self",r,f,e.C_BLOCK_COMMENT_MODE,c,d]}]},{scope:"class",variants:[{ | |
| beginKeywords:"enum",illegal:/[($"]/},{beginKeywords:"class interface trait", | |
| illegal:/[:($"]/}],relevance:0,end:/\{/,excludeEnd:!0,contains:[{ | |
| beginKeywords:"extends implements"},e.UNDERSCORE_TITLE_MODE]},{ | |
| beginKeywords:"namespace",relevance:0,end:";",illegal:/[.']/, | |
| contains:[e.inherit(e.UNDERSCORE_TITLE_MODE,{scope:"title.class"})]},{ | |
| beginKeywords:"use",relevance:0,end:";",contains:[{ | |
| match:/\b(as|const|function)\b/,scope:"keyword"},e.UNDERSCORE_TITLE_MODE]},c,d]} | |
| },grmr_php_template:e=>({name:"PHP template",subLanguage:"xml",contains:[{ | |
| begin:/<\?(php|=)?/,end:/\?>/,subLanguage:"php",contains:[{begin:"/\\*", | |
| end:"\\*/",skip:!0},{begin:'b"',end:'"',skip:!0},{begin:"b'",end:"'",skip:!0 | |
| },e.inherit(e.APOS_STRING_MODE,{illegal:null,className:null,contains:null, | |
| skip:!0}),e.inherit(e.QUOTE_STRING_MODE,{illegal:null,className:null, | |
| contains:null,skip:!0})]}]}),grmr_plaintext:e=>({name:"Plain text", | |
| aliases:["text","txt"],disableAutodetect:!0}),grmr_python:e=>{ | |
| const n=e.regex,t=/[\p{XID_Start}_]\p{XID_Continue}*/u,a=["and","as","assert","async","await","break","case","class","continue","def","del","elif","else","except","finally","for","from","global","if","import","in","is","lambda","match","nonlocal|10","not","or","pass","raise","return","try","while","with","yield"],i={ | |
| $pattern:/[A-Za-z]\w+|__\w+__/,keyword:a, | |
| built_in:["__import__","abs","all","any","ascii","bin","bool","breakpoint","bytearray","bytes","callable","chr","classmethod","compile","complex","delattr","dict","dir","divmod","enumerate","eval","exec","filter","float","format","frozenset","getattr","globals","hasattr","hash","help","hex","id","input","int","isinstance","issubclass","iter","len","list","locals","map","max","memoryview","min","next","object","oct","open","ord","pow","print","property","range","repr","reversed","round","set","setattr","slice","sorted","staticmethod","str","sum","super","tuple","type","vars","zip"], | |
| literal:["__debug__","Ellipsis","False","None","NotImplemented","True"], | |
| type:["Any","Callable","Coroutine","Dict","List","Literal","Generic","Optional","Sequence","Set","Tuple","Type","Union"] | |
| },r={className:"meta",begin:/^(>>>|\.\.\.) /},s={className:"subst",begin:/\{/, | |
| end:/\}/,keywords:i,illegal:/#/},o={begin:/\{\{/,relevance:0},l={ | |
| className:"string",contains:[e.BACKSLASH_ESCAPE],variants:[{ | |
| begin:/([uU]|[bB]|[rR]|[bB][rR]|[rR][bB])?'''/,end:/'''/, | |
| contains:[e.BACKSLASH_ESCAPE,r],relevance:10},{ | |
| begin:/([uU]|[bB]|[rR]|[bB][rR]|[rR][bB])?"""/,end:/"""/, | |
| contains:[e.BACKSLASH_ESCAPE,r],relevance:10},{ | |
| begin:/([fF][rR]|[rR][fF]|[fF])'''/,end:/'''/, | |
| contains:[e.BACKSLASH_ESCAPE,r,o,s]},{begin:/([fF][rR]|[rR][fF]|[fF])"""/, | |
| end:/"""/,contains:[e.BACKSLASH_ESCAPE,r,o,s]},{begin:/([uU]|[rR])'/,end:/'/, | |
| relevance:10},{begin:/([uU]|[rR])"/,end:/"/,relevance:10},{ | |
| begin:/([bB]|[bB][rR]|[rR][bB])'/,end:/'/},{begin:/([bB]|[bB][rR]|[rR][bB])"/, | |
| end:/"/},{begin:/([fF][rR]|[rR][fF]|[fF])'/,end:/'/, | |
| contains:[e.BACKSLASH_ESCAPE,o,s]},{begin:/([fF][rR]|[rR][fF]|[fF])"/,end:/"/, | |
| contains:[e.BACKSLASH_ESCAPE,o,s]},e.APOS_STRING_MODE,e.QUOTE_STRING_MODE] | |
| },c="[0-9](_?[0-9])*",d=`(\\b(${c}))?\\.(${c})|\\b(${c})\\.`,g="\\b|"+a.join("|"),u={ | |
| className:"number",relevance:0,variants:[{ | |
| begin:`(\\b(${c})|(${d}))[eE][+-]?(${c})[jJ]?(?=${g})`},{begin:`(${d})[jJ]?`},{ | |
| begin:`\\b([1-9](_?[0-9])*|0+(_?0)*)[lLjJ]?(?=${g})`},{ | |
| begin:`\\b0[bB](_?[01])+[lL]?(?=${g})`},{begin:`\\b0[oO](_?[0-7])+[lL]?(?=${g})` | |
| },{begin:`\\b0[xX](_?[0-9a-fA-F])+[lL]?(?=${g})`},{begin:`\\b(${c})[jJ](?=${g})` | |
| }]},b={className:"comment",begin:n.lookahead(/# type:/),end:/$/,keywords:i, | |
| contains:[{begin:/# type:/},{begin:/#/,end:/\b\B/,endsWithParent:!0}]},m={ | |
| className:"params",variants:[{className:"",begin:/\(\s*\)/,skip:!0},{begin:/\(/, | |
| end:/\)/,excludeBegin:!0,excludeEnd:!0,keywords:i, | |
| contains:["self",r,u,l,e.HASH_COMMENT_MODE]}]};return s.contains=[l,u,r],{ | |
| name:"Python",aliases:["py","gyp","ipython"],unicodeRegex:!0,keywords:i, | |
| illegal:/(<\/|\?)|=>/,contains:[r,u,{begin:/\bself\b/},{beginKeywords:"if", | |
| relevance:0},l,b,e.HASH_COMMENT_MODE,{match:[/\bdef/,/\s+/,t],scope:{ | |
| 1:"keyword",3:"title.function"},contains:[m]},{variants:[{ | |
| match:[/\bclass/,/\s+/,t,/\s*/,/\(\s*/,t,/\s*\)/]},{match:[/\bclass/,/\s+/,t]}], | |
| scope:{1:"keyword",3:"title.class",6:"title.class.inherited"}},{ | |
| className:"meta",begin:/^[\t ]*@/,end:/(?=#)|$/,contains:[u,m,l]}]}}, | |
| grmr_python_repl:e=>({aliases:["pycon"],contains:[{className:"meta.prompt", | |
| starts:{end:/ |$/,starts:{end:"$",subLanguage:"python"}},variants:[{ | |
| begin:/^>>>(?=[ ]|$)/},{begin:/^\.\.\.(?=[ ]|$)/}]}]}),grmr_r:e=>{ | |
| const n=e.regex,t=/(?:(?:[a-zA-Z]|\.[._a-zA-Z])[._a-zA-Z0-9]*)|\.(?!\d)/,a=n.either(/0[xX][0-9a-fA-F]+\.[0-9a-fA-F]*[pP][+-]?\d+i?/,/0[xX][0-9a-fA-F]+(?:[pP][+-]?\d+)?[Li]?/,/(?:\d+(?:\.\d*)?|\.\d+)(?:[eE][+-]?\d+)?[Li]?/),i=/[=!<>:]=|\|\||&&|:::?|<-|<<-|->>|->|\|>|[-+*\/?!{{HIGHLIGHT_JS}}|:<=>@^~]|\*\*/,r=n.either(/[()]/,/[{}]/,/\[\[/,/[[\]]/,/\\/,/,/) | |
| ;return{name:"R",keywords:{$pattern:t, | |
| keyword:"function if in break next repeat else for while", | |
| literal:"NULL NA TRUE FALSE Inf NaN NA_integer_|10 NA_real_|10 NA_character_|10 NA_complex_|10", | |
| built_in:"LETTERS letters month.abb month.name pi T F abs acos acosh all any anyNA Arg as.call as.character as.complex as.double as.environment as.integer as.logical as.null.default as.numeric as.raw asin asinh atan atanh attr attributes baseenv browser c call ceiling class Conj cos cosh cospi cummax cummin cumprod cumsum digamma dim dimnames emptyenv exp expression floor forceAndCall gamma gc.time globalenv Im interactive invisible is.array is.atomic is.call is.character is.complex is.double is.environment is.expression is.finite is.function is.infinite is.integer is.language is.list is.logical is.matrix is.na is.name is.nan is.null is.numeric is.object is.pairlist is.raw is.recursive is.single is.symbol lazyLoadDBfetch length lgamma list log max min missing Mod names nargs nzchar oldClass on.exit pos.to.env proc.time prod quote range Re rep retracemem return round seq_along seq_len seq.int sign signif sin sinh sinpi sqrt standardGeneric substitute sum switch tan tanh tanpi tracemem trigamma trunc unclass untracemem UseMethod xtfrm" | |
| },contains:[e.COMMENT(/#'/,/$/,{contains:[{scope:"doctag",match:/@examples/, | |
| starts:{end:n.lookahead(n.either(/\n^#'\s*(?=@[a-zA-Z]+)/,/\n^(?!#')/)), | |
| endsParent:!0}},{scope:"doctag",begin:"@param",end:/$/,contains:[{ | |
| scope:"variable",variants:[{match:t},{match:/`(?:\\.|[^`\\])+`/}],endsParent:!0 | |
| }]},{scope:"doctag",match:/@[a-zA-Z]+/},{scope:"keyword",match:/\\[a-zA-Z]+/}] | |
| }),e.HASH_COMMENT_MODE,{scope:"string",contains:[e.BACKSLASH_ESCAPE], | |
| variants:[e.END_SAME_AS_BEGIN({begin:/[rR]"(-*)\(/,end:/\)(-*)"/ | |
| }),e.END_SAME_AS_BEGIN({begin:/[rR]"(-*)\{/,end:/\}(-*)"/ | |
| }),e.END_SAME_AS_BEGIN({begin:/[rR]"(-*)\[/,end:/\](-*)"/ | |
| }),e.END_SAME_AS_BEGIN({begin:/[rR]'(-*)\(/,end:/\)(-*)'/ | |
| }),e.END_SAME_AS_BEGIN({begin:/[rR]'(-*)\{/,end:/\}(-*)'/ | |
| }),e.END_SAME_AS_BEGIN({begin:/[rR]'(-*)\[/,end:/\](-*)'/}),{begin:'"',end:'"', | |
| relevance:0},{begin:"'",end:"'",relevance:0}]},{relevance:0,variants:[{scope:{ | |
| 1:"operator",2:"number"},match:[i,a]},{scope:{1:"operator",2:"number"}, | |
| match:[/%[^%]*%/,a]},{scope:{1:"punctuation",2:"number"},match:[r,a]},{scope:{ | |
| 2:"number"},match:[/[^a-zA-Z0-9._]|^/,a]}]},{scope:{3:"operator"}, | |
| match:[t,/\s+/,/<-/,/\s+/]},{scope:"operator",relevance:0,variants:[{match:i},{ | |
| match:/%[^%]*%/}]},{scope:"punctuation",relevance:0,match:r},{begin:"`",end:"`", | |
| contains:[{begin:/\\./}]}]}},grmr_ruby:e=>{ | |
| const n=e.regex,t="([a-zA-Z_]\\w*[!?=]?|[-+~]@|<<|>>|=~|===?|<=>|[<>]=?|\\*\\*|[-/+%^&*~`|]|\\[\\]=?)",a=n.either(/\b([A-Z]+[a-z0-9]+)+/,/\b([A-Z]+[a-z0-9]+)+[A-Z]+/),i=n.concat(a,/(::\w+)*/),r={ | |
| "variable.constant":["__FILE__","__LINE__","__ENCODING__"], | |
| "variable.language":["self","super"], | |
| keyword:["alias","and","begin","BEGIN","break","case","class","defined","do","else","elsif","end","END","ensure","for","if","in","module","next","not","or","redo","require","rescue","retry","return","then","undef","unless","until","when","while","yield","include","extend","prepend","public","private","protected","raise","throw"], | |
| built_in:["proc","lambda","attr_accessor","attr_reader","attr_writer","define_method","private_constant","module_function"], | |
| literal:["true","false","nil"]},s={className:"doctag",begin:"@[A-Za-z]+"},o={ | |
| begin:"#<",end:">"},l=[e.COMMENT("#","$",{contains:[s] | |
| }),e.COMMENT("^=begin","^=end",{contains:[s],relevance:10 | |
| }),e.COMMENT("^__END__",e.MATCH_NOTHING_RE)],c={className:"subst",begin:/#\{/, | |
| end:/\}/,keywords:r},d={className:"string",contains:[e.BACKSLASH_ESCAPE,c], | |
| variants:[{begin:/'/,end:/'/},{begin:/"/,end:/"/},{begin:/`/,end:/`/},{ | |
| begin:/%[qQwWx]?\(/,end:/\)/},{begin:/%[qQwWx]?\[/,end:/\]/},{ | |
| begin:/%[qQwWx]?\{/,end:/\}/},{begin:/%[qQwWx]?</,end:/>/},{begin:/%[qQwWx]?\//, | |
| end:/\//},{begin:/%[qQwWx]?%/,end:/%/},{begin:/%[qQwWx]?-/,end:/-/},{ | |
| begin:/%[qQwWx]?\|/,end:/\|/},{begin:/\B\?(\\\d{1,3})/},{ | |
| begin:/\B\?(\\x[A-Fa-f0-9]{1,2})/},{begin:/\B\?(\\u\{?[A-Fa-f0-9]{1,6}\}?)/},{ | |
| begin:/\B\?(\\M-\\C-|\\M-\\c|\\c\\M-|\\M-|\\C-\\M-)[\x20-\x7e]/},{ | |
| begin:/\B\?\\(c|C-)[\x20-\x7e]/},{begin:/\B\?\\?\S/},{ | |
| begin:n.concat(/<<[-~]?'?/,n.lookahead(/(\w+)(?=\W)[^\n]*\n(?:[^\n]*\n)*?\s*\1\b/)), | |
| contains:[e.END_SAME_AS_BEGIN({begin:/(\w+)/,end:/(\w+)/, | |
| contains:[e.BACKSLASH_ESCAPE,c]})]}]},g="[0-9](_?[0-9])*",u={className:"number", | |
| relevance:0,variants:[{ | |
| begin:`\\b([1-9](_?[0-9])*|0)(\\.(${g}))?([eE][+-]?(${g})|r)?i?\\b`},{ | |
| begin:"\\b0[dD][0-9](_?[0-9])*r?i?\\b"},{begin:"\\b0[bB][0-1](_?[0-1])*r?i?\\b" | |
| },{begin:"\\b0[oO][0-7](_?[0-7])*r?i?\\b"},{ | |
| begin:"\\b0[xX][0-9a-fA-F](_?[0-9a-fA-F])*r?i?\\b"},{ | |
| begin:"\\b0(_?[0-7])+r?i?\\b"}]},b={variants:[{match:/\(\)/},{ | |
| className:"params",begin:/\(/,end:/(?=\))/,excludeBegin:!0,endsParent:!0, | |
| keywords:r}]},m=[d,{variants:[{match:[/class\s+/,i,/\s+<\s+/,i]},{ | |
| match:[/\b(class|module)\s+/,i]}],scope:{2:"title.class", | |
| 4:"title.class.inherited"},keywords:r},{match:[/(include|extend)\s+/,i],scope:{ | |
| 2:"title.class"},keywords:r},{relevance:0,match:[i,/\.new[. (]/],scope:{ | |
| 1:"title.class"}},{relevance:0,match:/\b[A-Z][A-Z_0-9]+\b/, | |
| className:"variable.constant"},{relevance:0,match:a,scope:"title.class"},{ | |
| match:[/def/,/\s+/,t],scope:{1:"keyword",3:"title.function"},contains:[b]},{ | |
| begin:e.IDENT_RE+"::"},{className:"symbol", | |
| begin:e.UNDERSCORE_IDENT_RE+"(!|\\?)?:",relevance:0},{className:"symbol", | |
| begin:":(?!\\s)",contains:[d,{begin:t}],relevance:0},u,{className:"variable", | |
| begin:"(\\$\\W)|((\\$|@@?)(\\w+))(?=[^@$?])(?![A-Za-z])(?![@$?'])"},{ | |
| className:"params",begin:/\|/,end:/\|/,excludeBegin:!0,excludeEnd:!0, | |
| relevance:0,keywords:r},{begin:"("+e.RE_STARTERS_RE+"|unless)\\s*", | |
| keywords:"unless",contains:[{className:"regexp",contains:[e.BACKSLASH_ESCAPE,c], | |
| illegal:/\n/,variants:[{begin:"/",end:"/[a-z]*"},{begin:/%r\{/,end:/\}[a-z]*/},{ | |
| begin:"%r\\(",end:"\\)[a-z]*"},{begin:"%r!",end:"![a-z]*"},{begin:"%r\\[", | |
| end:"\\][a-z]*"}]}].concat(o,l),relevance:0}].concat(o,l) | |
| ;c.contains=m,b.contains=m;const p=[{begin:/^\s*=>/,starts:{end:"$",contains:m} | |
| },{className:"meta.prompt", | |
| begin:"^([>?]>|[\\w#]+\\(\\w+\\):\\d+:\\d+[>*]|(\\w+-)?\\d+\\.\\d+\\.\\d+(p\\d+)?[^\\d][^>]+>)(?=[ ])", | |
| starts:{end:"$",keywords:r,contains:m}}];return l.unshift(o),{name:"Ruby", | |
| aliases:["rb","gemspec","podspec","thor","irb"],keywords:r,illegal:/\/\*/, | |
| contains:[e.SHEBANG({binary:"ruby"})].concat(p).concat(l).concat(m)}}, | |
| grmr_rust:e=>{const n=e.regex,t={className:"title.function.invoke",relevance:0, | |
| begin:n.concat(/\b/,/(?!let|for|while|if|else|match\b)/,e.IDENT_RE,n.lookahead(/\s*\(/)) | |
| },a="([ui](8|16|32|64|128|size)|f(32|64))?",i=["drop ","Copy","Send","Sized","Sync","Drop","Fn","FnMut","FnOnce","ToOwned","Clone","Debug","PartialEq","PartialOrd","Eq","Ord","AsRef","AsMut","Into","From","Default","Iterator","Extend","IntoIterator","DoubleEndedIterator","ExactSizeIterator","SliceConcatExt","ToString","assert!","assert_eq!","bitflags!","bytes!","cfg!","col!","concat!","concat_idents!","debug_assert!","debug_assert_eq!","env!","eprintln!","panic!","file!","format!","format_args!","include_bytes!","include_str!","line!","local_data_key!","module_path!","option_env!","print!","println!","select!","stringify!","try!","unimplemented!","unreachable!","vec!","write!","writeln!","macro_rules!","assert_ne!","debug_assert_ne!"],r=["i8","i16","i32","i64","i128","isize","u8","u16","u32","u64","u128","usize","f32","f64","str","char","bool","Box","Option","Result","String","Vec"] | |
| ;return{name:"Rust",aliases:["rs"],keywords:{$pattern:e.IDENT_RE+"!?",type:r, | |
| keyword:["abstract","as","async","await","become","box","break","const","continue","crate","do","dyn","else","enum","extern","false","final","fn","for","if","impl","in","let","loop","macro","match","mod","move","mut","override","priv","pub","ref","return","self","Self","static","struct","super","trait","true","try","type","typeof","unsafe","unsized","use","virtual","where","while","yield"], | |
| literal:["true","false","Some","None","Ok","Err"],built_in:i},illegal:"</", | |
| contains:[e.C_LINE_COMMENT_MODE,e.COMMENT("/\\*","\\*/",{contains:["self"] | |
| }),e.inherit(e.QUOTE_STRING_MODE,{begin:/b?"/,illegal:null}),{ | |
| className:"string",variants:[{begin:/b?r(#*)"(.|\n)*?"\1(?!#)/},{ | |
| begin:/b?'\\?(x\w{2}|u\w{4}|U\w{8}|.)'/}]},{className:"symbol", | |
| begin:/'[a-zA-Z_][a-zA-Z0-9_]*/},{className:"number",variants:[{ | |
| begin:"\\b0b([01_]+)"+a},{begin:"\\b0o([0-7_]+)"+a},{ | |
| begin:"\\b0x([A-Fa-f0-9_]+)"+a},{ | |
| begin:"\\b(\\d[\\d_]*(\\.[0-9_]+)?([eE][+-]?[0-9_]+)?)"+a}],relevance:0},{ | |
| begin:[/fn/,/\s+/,e.UNDERSCORE_IDENT_RE],className:{1:"keyword", | |
| 3:"title.function"}},{className:"meta",begin:"#!?\\[",end:"\\]",contains:[{ | |
| className:"string",begin:/"/,end:/"/}]},{ | |
| begin:[/let/,/\s+/,/(?:mut\s+)?/,e.UNDERSCORE_IDENT_RE],className:{1:"keyword", | |
| 3:"keyword",4:"variable"}},{ | |
| begin:[/for/,/\s+/,e.UNDERSCORE_IDENT_RE,/\s+/,/in/],className:{1:"keyword", | |
| 3:"variable",5:"keyword"}},{begin:[/type/,/\s+/,e.UNDERSCORE_IDENT_RE], | |
| className:{1:"keyword",3:"title.class"}},{ | |
| begin:[/(?:trait|enum|struct|union|impl|for)/,/\s+/,e.UNDERSCORE_IDENT_RE], | |
| className:{1:"keyword",3:"title.class"}},{begin:e.IDENT_RE+"::",keywords:{ | |
| keyword:"Self",built_in:i,type:r}},{className:"punctuation",begin:"->"},t]}}, | |
| grmr_scss:e=>{const n=ie(e),t=le,a=oe,i="@[a-z-]+",r={className:"variable", | |
| begin:"(\\$[a-zA-Z-][a-zA-Z0-9_-]*)\\b",relevance:0};return{name:"SCSS", | |
| case_insensitive:!0,illegal:"[=/|']", | |
| contains:[e.C_LINE_COMMENT_MODE,e.C_BLOCK_COMMENT_MODE,n.CSS_NUMBER_MODE,{ | |
| className:"selector-id",begin:"#[A-Za-z0-9_-]+",relevance:0},{ | |
| className:"selector-class",begin:"\\.[A-Za-z0-9_-]+",relevance:0 | |
| },n.ATTRIBUTE_SELECTOR_MODE,{className:"selector-tag", | |
| begin:"\\b("+re.join("|")+")\\b",relevance:0},{className:"selector-pseudo", | |
| begin:":("+a.join("|")+")"},{className:"selector-pseudo", | |
| begin:":(:)?("+t.join("|")+")"},r,{begin:/\(/,end:/\)/, | |
| contains:[n.CSS_NUMBER_MODE]},n.CSS_VARIABLE,{className:"attribute", | |
| begin:"\\b("+ce.join("|")+")\\b"},{ | |
| begin:"\\b(whitespace|wait|w-resize|visible|vertical-text|vertical-ideographic|uppercase|upper-roman|upper-alpha|underline|transparent|top|thin|thick|text|text-top|text-bottom|tb-rl|table-header-group|table-footer-group|sw-resize|super|strict|static|square|solid|small-caps|separate|se-resize|scroll|s-resize|rtl|row-resize|ridge|right|repeat|repeat-y|repeat-x|relative|progress|pointer|overline|outside|outset|oblique|nowrap|not-allowed|normal|none|nw-resize|no-repeat|no-drop|newspaper|ne-resize|n-resize|move|middle|medium|ltr|lr-tb|lowercase|lower-roman|lower-alpha|loose|list-item|line|line-through|line-edge|lighter|left|keep-all|justify|italic|inter-word|inter-ideograph|inside|inset|inline|inline-block|inherit|inactive|ideograph-space|ideograph-parenthesis|ideograph-numeric|ideograph-alpha|horizontal|hidden|help|hand|groove|fixed|ellipsis|e-resize|double|dotted|distribute|distribute-space|distribute-letter|distribute-all-lines|disc|disabled|default|decimal|dashed|crosshair|collapse|col-resize|circle|char|center|capitalize|break-word|break-all|bottom|both|bolder|bold|block|bidi-override|below|baseline|auto|always|all-scroll|absolute|table|table-cell)\\b" | |
| },{begin:/:/,end:/[;}{]/,relevance:0, | |
| contains:[n.BLOCK_COMMENT,r,n.HEXCOLOR,n.CSS_NUMBER_MODE,e.QUOTE_STRING_MODE,e.APOS_STRING_MODE,n.IMPORTANT,n.FUNCTION_DISPATCH] | |
| },{begin:"@(page|font-face)",keywords:{$pattern:i,keyword:"@page @font-face"}},{ | |
| begin:"@",end:"[{;]",returnBegin:!0,keywords:{$pattern:/[a-z-]+/, | |
| keyword:"and or not only",attribute:se.join(" ")},contains:[{begin:i, | |
| className:"keyword"},{begin:/[a-z-]+(?=:)/,className:"attribute" | |
| },r,e.QUOTE_STRING_MODE,e.APOS_STRING_MODE,n.HEXCOLOR,n.CSS_NUMBER_MODE] | |
| },n.FUNCTION_DISPATCH]}},grmr_shell:e=>({name:"Shell Session", | |
| aliases:["console","shellsession"],contains:[{className:"meta.prompt", | |
| begin:/^\s{0,3}[/~\w\d[\]()@-]*[>%$#][ ]?/,starts:{end:/[^\\](?=\s*$)/, | |
| subLanguage:"bash"}}]}),grmr_sql:e=>{ | |
| const n=e.regex,t=e.COMMENT("--","$"),a=["true","false","unknown"],i=["bigint","binary","blob","boolean","char","character","clob","date","dec","decfloat","decimal","float","int","integer","interval","nchar","nclob","national","numeric","real","row","smallint","time","timestamp","varchar","varying","varbinary"],r=["abs","acos","array_agg","asin","atan","avg","cast","ceil","ceiling","coalesce","corr","cos","cosh","count","covar_pop","covar_samp","cume_dist","dense_rank","deref","element","exp","extract","first_value","floor","json_array","json_arrayagg","json_exists","json_object","json_objectagg","json_query","json_table","json_table_primitive","json_value","lag","last_value","lead","listagg","ln","log","log10","lower","max","min","mod","nth_value","ntile","nullif","percent_rank","percentile_cont","percentile_disc","position","position_regex","power","rank","regr_avgx","regr_avgy","regr_count","regr_intercept","regr_r2","regr_slope","regr_sxx","regr_sxy","regr_syy","row_number","sin","sinh","sqrt","stddev_pop","stddev_samp","substring","substring_regex","sum","tan","tanh","translate","translate_regex","treat","trim","trim_array","unnest","upper","value_of","var_pop","var_samp","width_bucket"],s=["create table","insert into","primary key","foreign key","not null","alter table","add constraint","grouping sets","on overflow","character set","respect nulls","ignore nulls","nulls first","nulls last","depth first","breadth first"],o=r,l=["abs","acos","all","allocate","alter","and","any","are","array","array_agg","array_max_cardinality","as","asensitive","asin","asymmetric","at","atan","atomic","authorization","avg","begin","begin_frame","begin_partition","between","bigint","binary","blob","boolean","both","by","call","called","cardinality","cascaded","case","cast","ceil","ceiling","char","char_length","character","character_length","check","classifier","clob","close","coalesce","collate","collect","column","commit","condition","connect","constraint","contains","convert","copy","corr","corresponding","cos","cosh","count","covar_pop","covar_samp","create","cross","cube","cume_dist","current","current_catalog","current_date","current_default_transform_group","current_path","current_role","current_row","current_schema","current_time","current_timestamp","current_path","current_role","current_transform_group_for_type","current_user","cursor","cycle","date","day","deallocate","dec","decimal","decfloat","declare","default","define","delete","dense_rank","deref","describe","deterministic","disconnect","distinct","double","drop","dynamic","each","element","else","empty","end","end_frame","end_partition","end-exec","equals","escape","every","except","exec","execute","exists","exp","external","extract","false","fetch","filter","first_value","float","floor","for","foreign","frame_row","free","from","full","function","fusion","get","global","grant","group","grouping","groups","having","hold","hour","identity","in","indicator","initial","inner","inout","insensitive","insert","int","integer","intersect","intersection","interval","into","is","join","json_array","json_arrayagg","json_exists","json_object","json_objectagg","json_query","json_table","json_table_primitive","json_value","lag","language","large","last_value","lateral","lead","leading","left","like","like_regex","listagg","ln","local","localtime","localtimestamp","log","log10","lower","match","match_number","match_recognize","matches","max","member","merge","method","min","minute","mod","modifies","module","month","multiset","national","natural","nchar","nclob","new","no","none","normalize","not","nth_value","ntile","null","nullif","numeric","octet_length","occurrences_regex","of","offset","old","omit","on","one","only","open","or","order","out","outer","over","overlaps","overlay","parameter","partition","pattern","per","percent","percent_rank","percentile_cont","percentile_disc","period","portion","position","position_regex","power","precedes","precision","prepare","primary","procedure","ptf","range","rank","reads","real","recursive","ref","references","referencing","regr_avgx","regr_avgy","regr_count","regr_intercept","regr_r2","regr_slope","regr_sxx","regr_sxy","regr_syy","release","result","return","returns","revoke","right","rollback","rollup","row","row_number","rows","running","savepoint","scope","scroll","search","second","seek","select","sensitive","session_user","set","show","similar","sin","sinh","skip","smallint","some","specific","specifictype","sql","sqlexception","sqlstate","sqlwarning","sqrt","start","static","stddev_pop","stddev_samp","submultiset","subset","substring","substring_regex","succeeds","sum","symmetric","system","system_time","system_user","table","tablesample","tan","tanh","then","time","timestamp","timezone_hour","timezone_minute","to","trailing","translate","translate_regex","translation","treat","trigger","trim","trim_array","true","truncate","uescape","union","unique","unknown","unnest","update","upper","user","using","value","values","value_of","var_pop","var_samp","varbinary","varchar","varying","versioning","when","whenever","where","width_bucket","window","with","within","without","year","add","asc","collation","desc","final","first","last","view"].filter((e=>!r.includes(e))),c={ | |
| begin:n.concat(/\b/,n.either(...o),/\s*\(/),relevance:0,keywords:{built_in:o}} | |
| ;return{name:"SQL",case_insensitive:!0,illegal:/[{}]|<\//,keywords:{ | |
| $pattern:/\b[\w\.]+/,keyword:((e,{exceptions:n,when:t}={})=>{const a=t | |
| ;return n=n||[],e.map((e=>e.match(/\|\d+$/)||n.includes(e)?e:a(e)?e+"|0":e)) | |
| })(l,{when:e=>e.length<3}),literal:a,type:i, | |
| built_in:["current_catalog","current_date","current_default_transform_group","current_path","current_role","current_schema","current_transform_group_for_type","current_user","session_user","system_time","system_user","current_time","localtime","current_timestamp","localtimestamp"] | |
| },contains:[{begin:n.either(...s),relevance:0,keywords:{$pattern:/[\w\.]+/, | |
| keyword:l.concat(s),literal:a,type:i}},{className:"type", | |
| begin:n.either("double precision","large object","with timezone","without timezone") | |
| },c,{className:"variable",begin:/@[a-z0-9][a-z0-9_]*/},{className:"string", | |
| variants:[{begin:/'/,end:/'/,contains:[{begin:/''/}]}]},{begin:/"/,end:/"/, | |
| contains:[{begin:/""/}]},e.C_NUMBER_MODE,e.C_BLOCK_COMMENT_MODE,t,{ | |
| className:"operator",begin:/[-+*/=%^~]|&&?|\|\|?|!=?|<(?:=>?|<|>)?|>[>=]?/, | |
| relevance:0}]}},grmr_swift:e=>{const n={match:/\s+/,relevance:0 | |
| },t=e.COMMENT("/\\*","\\*/",{contains:["self"]}),a=[e.C_LINE_COMMENT_MODE,t],i={ | |
| match:[/\./,m(...xe,...Me)],className:{2:"keyword"}},r={match:b(/\./,m(...Ae)), | |
| relevance:0},s=Ae.filter((e=>"string"==typeof e)).concat(["_|0"]),o={variants:[{ | |
| className:"keyword", | |
| match:m(...Ae.filter((e=>"string"!=typeof e)).concat(Se).map(ke),...Me)}]},l={ | |
| $pattern:m(/\b\w+/,/#\w+/),keyword:s.concat(Re),literal:Ce},c=[i,r,o],g=[{ | |
| match:b(/\./,m(...De)),relevance:0},{className:"built_in", | |
| match:b(/\b/,m(...De),/(?=\()/)}],u={match:/->/,relevance:0},p=[u,{ | |
| className:"operator",relevance:0,variants:[{match:Be},{match:`\\.(\\.|${Le})+`}] | |
| }],_="([0-9]_*)+",h="([0-9a-fA-F]_*)+",f={className:"number",relevance:0, | |
| variants:[{match:`\\b(${_})(\\.(${_}))?([eE][+-]?(${_}))?\\b`},{ | |
| match:`\\b0x(${h})(\\.(${h}))?([pP][+-]?(${_}))?\\b`},{match:/\b0o([0-7]_*)+\b/ | |
| },{match:/\b0b([01]_*)+\b/}]},E=(e="")=>({className:"subst",variants:[{ | |
| match:b(/\\/,e,/[0\\tnr"']/)},{match:b(/\\/,e,/u\{[0-9a-fA-F]{1,8}\}/)}] | |
| }),y=(e="")=>({className:"subst",match:b(/\\/,e,/[\t ]*(?:[\r\n]|\r\n)/) | |
| }),N=(e="")=>({className:"subst",label:"interpol",begin:b(/\\/,e,/\(/),end:/\)/ | |
| }),w=(e="")=>({begin:b(e,/"""/),end:b(/"""/,e),contains:[E(e),y(e),N(e)] | |
| }),v=(e="")=>({begin:b(e,/"/),end:b(/"/,e),contains:[E(e),N(e)]}),O={ | |
| className:"string", | |
| variants:[w(),w("#"),w("##"),w("###"),v(),v("#"),v("##"),v("###")] | |
| },k=[e.BACKSLASH_ESCAPE,{begin:/\[/,end:/\]/,relevance:0, | |
| contains:[e.BACKSLASH_ESCAPE]}],x={begin:/\/[^\s](?=[^/\n]*\/)/,end:/\//, | |
| contains:k},M=e=>{const n=b(e,/\//),t=b(/\//,e);return{begin:n,end:t, | |
| contains:[...k,{scope:"comment",begin:`#(?!.*${t})`,end:/$/}]}},S={ | |
| scope:"regexp",variants:[M("###"),M("##"),M("#"),x]},A={match:b(/`/,Fe,/`/) | |
| },C=[A,{className:"variable",match:/\$\d+/},{className:"variable", | |
| match:`\\${ze}+`}],T=[{match:/(@|#(un)?)available/,scope:"keyword",starts:{ | |
| contains:[{begin:/\(/,end:/\)/,keywords:Pe,contains:[...p,f,O]}]}},{ | |
| scope:"keyword",match:b(/@/,m(...je))},{scope:"meta",match:b(/@/,Fe)}],R={ | |
| match:d(/\b[A-Z]/),relevance:0,contains:[{className:"type", | |
| match:b(/(AV|CA|CF|CG|CI|CL|CM|CN|CT|MK|MP|MTK|MTL|NS|SCN|SK|UI|WK|XC)/,ze,"+") | |
| },{className:"type",match:Ue,relevance:0},{match:/[?!]+/,relevance:0},{ | |
| match:/\.\.\./,relevance:0},{match:b(/\s+&\s+/,d(Ue)),relevance:0}]},D={ | |
| begin:/</,end:/>/,keywords:l,contains:[...a,...c,...T,u,R]};R.contains.push(D) | |
| ;const I={begin:/\(/,end:/\)/,relevance:0,keywords:l,contains:["self",{ | |
| match:b(Fe,/\s*:/),keywords:"_|0",relevance:0 | |
| },...a,S,...c,...g,...p,f,O,...C,...T,R]},L={begin:/</,end:/>/, | |
| keywords:"repeat each",contains:[...a,R]},B={begin:/\(/,end:/\)/,keywords:l, | |
| contains:[{begin:m(d(b(Fe,/\s*:/)),d(b(Fe,/\s+/,Fe,/\s*:/))),end:/:/, | |
| relevance:0,contains:[{className:"keyword",match:/\b_\b/},{className:"params", | |
| match:Fe}]},...a,...c,...p,f,O,...T,R,I],endsParent:!0,illegal:/["']/},$={ | |
| match:[/(func|macro)/,/\s+/,m(A.match,Fe,Be)],className:{1:"keyword", | |
| 3:"title.function"},contains:[L,B,n],illegal:[/\[/,/%/]},z={ | |
| match:[/\b(?:subscript|init[?!]?)/,/\s*(?=[<(])/],className:{1:"keyword"}, | |
| contains:[L,B,n],illegal:/\[|%/},F={match:[/operator/,/\s+/,Be],className:{ | |
| 1:"keyword",3:"title"}},U={begin:[/precedencegroup/,/\s+/,Ue],className:{ | |
| 1:"keyword",3:"title"},contains:[R],keywords:[...Te,...Ce],end:/}/} | |
| ;for(const e of O.variants){const n=e.contains.find((e=>"interpol"===e.label)) | |
| ;n.keywords=l;const t=[...c,...g,...p,f,O,...C];n.contains=[...t,{begin:/\(/, | |
| end:/\)/,contains:["self",...t]}]}return{name:"Swift",keywords:l, | |
| contains:[...a,$,z,{beginKeywords:"struct protocol class extension enum actor", | |
| end:"\\{",excludeEnd:!0,keywords:l,contains:[e.inherit(e.TITLE_MODE,{ | |
| className:"title.class",begin:/[A-Za-z$_][\u00C0-\u02B80-9A-Za-z$_]*/}),...c] | |
| },F,U,{beginKeywords:"import",end:/$/,contains:[...a],relevance:0 | |
| },S,...c,...g,...p,f,O,...C,...T,R,I]}},grmr_typescript:e=>{ | |
| const n=Oe(e),t=_e,a=["any","void","number","boolean","string","object","never","symbol","bigint","unknown"],i={ | |
| beginKeywords:"namespace",end:/\{/,excludeEnd:!0, | |
| contains:[n.exports.CLASS_REFERENCE]},r={beginKeywords:"interface",end:/\{/, | |
| excludeEnd:!0,keywords:{keyword:"interface extends",built_in:a}, | |
| contains:[n.exports.CLASS_REFERENCE]},s={$pattern:_e, | |
| keyword:he.concat(["type","namespace","interface","public","private","protected","implements","declare","abstract","readonly","enum","override"]), | |
| literal:fe,built_in:ve.concat(a),"variable.language":we},o={className:"meta", | |
| begin:"@"+t},l=(e,n,t)=>{const a=e.contains.findIndex((e=>e.label===n)) | |
| ;if(-1===a)throw Error("can not find mode to replace");e.contains.splice(a,1,t)} | |
| ;return Object.assign(n.keywords,s), | |
| n.exports.PARAMS_CONTAINS.push(o),n.contains=n.contains.concat([o,i,r]), | |
| l(n,"shebang",e.SHEBANG()),l(n,"use_strict",{className:"meta",relevance:10, | |
| begin:/^\s*['"]use strict['"]/ | |
| }),n.contains.find((e=>"func.def"===e.label)).relevance=0,Object.assign(n,{ | |
| name:"TypeScript",aliases:["ts","tsx","mts","cts"]}),n},grmr_vbnet:e=>{ | |
| const n=e.regex,t=/\d{1,2}\/\d{1,2}\/\d{4}/,a=/\d{4}-\d{1,2}-\d{1,2}/,i=/(\d|1[012])(:\d+){0,2} *(AM|PM)/,r=/\d{1,2}(:\d{1,2}){1,2}/,s={ | |
| className:"literal",variants:[{begin:n.concat(/# */,n.either(a,t),/ *#/)},{ | |
| begin:n.concat(/# */,r,/ *#/)},{begin:n.concat(/# */,i,/ *#/)},{ | |
| begin:n.concat(/# */,n.either(a,t),/ +/,n.either(i,r),/ *#/)}] | |
| },o=e.COMMENT(/'''/,/$/,{contains:[{className:"doctag",begin:/<\/?/,end:/>/}] | |
| }),l=e.COMMENT(null,/$/,{variants:[{begin:/'/},{begin:/([\t ]|^)REM(?=\s)/}]}) | |
| ;return{name:"Visual Basic .NET",aliases:["vb"],case_insensitive:!0, | |
| classNameAliases:{label:"symbol"},keywords:{ | |
| keyword:"addhandler alias aggregate ansi as async assembly auto binary by byref byval call case catch class compare const continue custom declare default delegate dim distinct do each equals else elseif end enum erase error event exit explicit finally for friend from function get global goto group handles if implements imports in inherits interface into iterator join key let lib loop me mid module mustinherit mustoverride mybase myclass namespace narrowing new next notinheritable notoverridable of off on operator option optional order overloads overridable overrides paramarray partial preserve private property protected public raiseevent readonly redim removehandler resume return select set shadows shared skip static step stop structure strict sub synclock take text then throw to try unicode until using when where while widening with withevents writeonly yield", | |
| built_in:"addressof and andalso await directcast gettype getxmlnamespace is isfalse isnot istrue like mod nameof new not or orelse trycast typeof xor cbool cbyte cchar cdate cdbl cdec cint clng cobj csbyte cshort csng cstr cuint culng cushort", | |
| type:"boolean byte char date decimal double integer long object sbyte short single string uinteger ulong ushort", | |
| literal:"true false nothing"}, | |
| illegal:"//|\\{|\\}|endif|gosub|variant|wend|^\\$ ",contains:[{ | |
| className:"string",begin:/"(""|[^/n])"C\b/},{className:"string",begin:/"/, | |
| end:/"/,illegal:/\n/,contains:[{begin:/""/}]},s,{className:"number",relevance:0, | |
| variants:[{begin:/\b\d[\d_]*((\.[\d_]+(E[+-]?[\d_]+)?)|(E[+-]?[\d_]+))[RFD@!#]?/ | |
| },{begin:/\b\d[\d_]*((U?[SIL])|[%&])?/},{begin:/&H[\dA-F_]+((U?[SIL])|[%&])?/},{ | |
| begin:/&O[0-7_]+((U?[SIL])|[%&])?/},{begin:/&B[01_]+((U?[SIL])|[%&])?/}]},{ | |
| className:"label",begin:/^\w+:/},o,l,{className:"meta", | |
| begin:/[\t ]*#(const|disable|else|elseif|enable|end|externalsource|if|region)\b/, | |
| end:/$/,keywords:{ | |
| keyword:"const disable else elseif enable end externalsource if region then"}, | |
| contains:[l]}]}},grmr_wasm:e=>{e.regex;const n=e.COMMENT(/\(;/,/;\)/) | |
| ;return n.contains.push("self"),{name:"WebAssembly",keywords:{$pattern:/[\w.]+/, | |
| keyword:["anyfunc","block","br","br_if","br_table","call","call_indirect","data","drop","elem","else","end","export","func","global.get","global.set","local.get","local.set","local.tee","get_global","get_local","global","if","import","local","loop","memory","memory.grow","memory.size","module","mut","nop","offset","param","result","return","select","set_global","set_local","start","table","tee_local","then","type","unreachable"] | |
| },contains:[e.COMMENT(/;;/,/$/),n,{match:[/(?:offset|align)/,/\s*/,/=/], | |
| className:{1:"keyword",3:"operator"}},{className:"variable",begin:/\$[\w_]+/},{ | |
| match:/(\((?!;)|\))+/,className:"punctuation",relevance:0},{ | |
| begin:[/(?:func|call|call_indirect)/,/\s+/,/\$[^\s)]+/],className:{1:"keyword", | |
| 3:"title.function"}},e.QUOTE_STRING_MODE,{match:/(i32|i64|f32|f64)(?!\.)/, | |
| className:"type"},{className:"keyword", | |
| match:/\b(f32|f64|i32|i64)(?:\.(?:abs|add|and|ceil|clz|const|convert_[su]\/i(?:32|64)|copysign|ctz|demote\/f64|div(?:_[su])?|eqz?|extend_[su]\/i32|floor|ge(?:_[su])?|gt(?:_[su])?|le(?:_[su])?|load(?:(?:8|16|32)_[su])?|lt(?:_[su])?|max|min|mul|nearest|neg?|or|popcnt|promote\/f32|reinterpret\/[fi](?:32|64)|rem_[su]|rot[lr]|shl|shr_[su]|store(?:8|16|32)?|sqrt|sub|trunc(?:_[su]\/f(?:32|64))?|wrap\/i64|xor))\b/ | |
| },{className:"number",relevance:0, | |
| match:/[+-]?\b(?:\d(?:_?\d)*(?:\.\d(?:_?\d)*)?(?:[eE][+-]?\d(?:_?\d)*)?|0x[\da-fA-F](?:_?[\da-fA-F])*(?:\.[\da-fA-F](?:_?[\da-fA-D])*)?(?:[pP][+-]?\d(?:_?\d)*)?)\b|\binf\b|\bnan(?::0x[\da-fA-F](?:_?[\da-fA-D])*)?\b/ | |
| }]}},grmr_xml:e=>{ | |
| const n=e.regex,t=n.concat(/[\p{L}_]/u,n.optional(/[\p{L}0-9_.-]*:/u),/[\p{L}0-9_.-]*/u),a={ | |
| className:"symbol",begin:/&[a-z]+;|&#[0-9]+;|&#x[a-f0-9]+;/},i={begin:/\s/, | |
| contains:[{className:"keyword",begin:/#?[a-z_][a-z1-9_-]+/,illegal:/\n/}] | |
| },r=e.inherit(i,{begin:/\(/,end:/\)/}),s=e.inherit(e.APOS_STRING_MODE,{ | |
| className:"string"}),o=e.inherit(e.QUOTE_STRING_MODE,{className:"string"}),l={ | |
| endsWithParent:!0,illegal:/</,relevance:0,contains:[{className:"attr", | |
| begin:/[\p{L}0-9._:-]+/u,relevance:0},{begin:/=\s*/,relevance:0,contains:[{ | |
| className:"string",endsParent:!0,variants:[{begin:/"/,end:/"/,contains:[a]},{ | |
| begin:/'/,end:/'/,contains:[a]},{begin:/[^\s"'=<>`]+/}]}]}]};return{ | |
| name:"HTML, XML", | |
| aliases:["html","xhtml","rss","atom","xjb","xsd","xsl","plist","wsf","svg"], | |
| case_insensitive:!0,unicodeRegex:!0,contains:[{className:"meta",begin:/<![a-z]/, | |
| end:/>/,relevance:10,contains:[i,o,s,r,{begin:/\[/,end:/\]/,contains:[{ | |
| className:"meta",begin:/<![a-z]/,end:/>/,contains:[i,r,o,s]}]}] | |
| },e.COMMENT(/<!--/,/-->/,{relevance:10}),{begin:/<!\[CDATA\[/,end:/\]\]>/, | |
| relevance:10},a,{className:"meta",end:/\?>/,variants:[{begin:/<\?xml/, | |
| relevance:10,contains:[o]},{begin:/<\?[a-z][a-z0-9]+/}]},{className:"tag", | |
| begin:/<style(?=\s|>)/,end:/>/,keywords:{name:"style"},contains:[l],starts:{ | |
| end:/<\/style>/,returnEnd:!0,subLanguage:["css","xml"]}},{className:"tag", | |
| begin:/<script(?=\s|>)/,end:/>/,keywords:{name:"script"},contains:[l],starts:{ | |
| end:/<\/script>/,returnEnd:!0,subLanguage:["javascript","handlebars","xml"]}},{ | |
| className:"tag",begin:/<>|<\/>/},{className:"tag", | |
| begin:n.concat(/</,n.lookahead(n.concat(t,n.either(/\/>/,/>/,/\s/)))), | |
| end:/\/?>/,contains:[{className:"name",begin:t,relevance:0,starts:l}]},{ | |
| className:"tag",begin:n.concat(/<\//,n.lookahead(n.concat(t,/>/))),contains:[{ | |
| className:"name",begin:t,relevance:0},{begin:/>/,relevance:0,endsParent:!0}]}]} | |
| },grmr_yaml:e=>{ | |
| const n="true false yes no null",t="[\\w#;/?:@&=+$,.~*'()[\\]]+",a={ | |
| className:"string",relevance:0,variants:[{begin:/'/,end:/'/},{begin:/"/,end:/"/ | |
| },{begin:/\S+/}],contains:[e.BACKSLASH_ESCAPE,{className:"template-variable", | |
| variants:[{begin:/\{\{/,end:/\}\}/},{begin:/%\{/,end:/\}/}]}]},i=e.inherit(a,{ | |
| variants:[{begin:/'/,end:/'/},{begin:/"/,end:/"/},{begin:/[^\s,{}[\]]+/}]}),r={ | |
| end:",",endsWithParent:!0,excludeEnd:!0,keywords:n,relevance:0},s={begin:/\{/, | |
| end:/\}/,contains:[r],illegal:"\\n",relevance:0},o={begin:"\\[",end:"\\]", | |
| contains:[r],illegal:"\\n",relevance:0},l=[{className:"attr",variants:[{ | |
| begin:"\\w[\\w :\\/.-]*:(?=[ \t]|$)"},{begin:'"\\w[\\w :\\/.-]*":(?=[ \t]|$)'},{ | |
| begin:"'\\w[\\w :\\/.-]*':(?=[ \t]|$)"}]},{className:"meta",begin:"^---\\s*$", | |
| relevance:10},{className:"string", | |
| begin:"[\\|>]([1-9]?[+-])?[ ]*\\n( +)[^ ][^\\n]*\\n(\\2[^\\n]+\\n?)*"},{ | |
| begin:"<%[%=-]?",end:"[%-]?%>",subLanguage:"ruby",excludeBegin:!0,excludeEnd:!0, | |
| relevance:0},{className:"type",begin:"!\\w+!"+t},{className:"type", | |
| begin:"!<"+t+">"},{className:"type",begin:"!"+t},{className:"type",begin:"!!"+t | |
| },{className:"meta",begin:"&"+e.UNDERSCORE_IDENT_RE+"$"},{className:"meta", | |
| begin:"\\*"+e.UNDERSCORE_IDENT_RE+"$"},{className:"bullet",begin:"-(?=[ ]|$)", | |
| relevance:0},e.HASH_COMMENT_MODE,{beginKeywords:n,keywords:{literal:n}},{ | |
| className:"number", | |
| begin:"\\b[0-9]{4}(-[0-9][0-9]){0,2}([Tt \\t][0-9][0-9]?(:[0-9][0-9]){2})?(\\.[0-9]*)?([ \\t])*(Z|[-+][0-9][0-9]?(:[0-9][0-9])?)?\\b" | |
| },{className:"number",begin:e.C_NUMBER_RE+"\\b",relevance:0},s,o,a],c=[...l] | |
| ;return c.pop(),c.push(i),r.contains=c,{name:"YAML",case_insensitive:!0, | |
| aliases:["yml"],contains:l}}});const He=ae;for(const e of Object.keys(Ke)){ | |
| const n=e.replace("grmr_","").replace("_","-");He.registerLanguage(n,Ke[e])} | |
| return He}() | |
| ;"object"==typeof exports&&"undefined"!=typeof module&&(module.exports=hljs);</script> | |
| <!-- Main application code --> | |
| <script> | |
| (function() { | |
| 'use strict'; | |
| // ============================================================ | |
| // DATA LOADING | |
| // ============================================================ | |
| const base64 = document.getElementById('session-data').textContent; | |
| const binary = atob(base64); | |
| const bytes = new Uint8Array(binary.length); | |
| for (let i = 0; i < binary.length; i++) { | |
| bytes[i] = binary.charCodeAt(i); | |
| } | |
| const data = JSON.parse(new TextDecoder('utf-8').decode(bytes)); | |
| const { header, entries, leafId: defaultLeafId, systemPrompt, tools, renderedTools } = data; | |
| // ============================================================ | |
| // URL PARAMETER HANDLING | |
| // ============================================================ | |
| // Parse URL parameters for deep linking: leafId and targetId | |
| // Check for injected params (when loaded in iframe via srcdoc) or use window.location | |
| const injectedParams = document.querySelector('meta[name="pi-url-params"]'); | |
| const searchString = injectedParams ? injectedParams.content : window.location.search.substring(1); | |
| const urlParams = new URLSearchParams(searchString); | |
| const urlLeafId = urlParams.get('leafId'); | |
| const urlTargetId = urlParams.get('targetId'); | |
| // Use URL leafId if provided, otherwise fall back to session default | |
| const leafId = urlLeafId || defaultLeafId; | |
| // ============================================================ | |
| // DATA STRUCTURES | |
| // ============================================================ | |
| // Entry lookup by ID | |
| const byId = new Map(); | |
| for (const entry of entries) { | |
| byId.set(entry.id, entry); | |
| } | |
| // Tool call lookup (toolCallId -> {name, arguments}) | |
| const toolCallMap = new Map(); | |
| for (const entry of entries) { | |
| if (entry.type === 'message' && entry.message.role === 'assistant') { | |
| const content = entry.message.content; | |
| if (Array.isArray(content)) { | |
| for (const block of content) { | |
| if (block.type === 'toolCall') { | |
| toolCallMap.set(block.id, { name: block.name, arguments: block.arguments }); | |
| } | |
| } | |
| } | |
| } | |
| } | |
| // Label lookup (entryId -> label string) | |
| // Labels are stored in 'label' entries that reference their target via targetId | |
| const labelMap = new Map(); | |
| for (const entry of entries) { | |
| if (entry.type === 'label' && entry.targetId && entry.label) { | |
| labelMap.set(entry.targetId, entry.label); | |
| } | |
| } | |
| // ============================================================ | |
| // TREE DATA PREPARATION (no DOM, pure data) | |
| // ============================================================ | |
| /** | |
| * Build tree structure from flat entries. | |
| * Returns array of root nodes, each with { entry, children, label }. | |
| */ | |
| function buildTree() { | |
| const nodeMap = new Map(); | |
| const roots = []; | |
| // Create nodes | |
| for (const entry of entries) { | |
| nodeMap.set(entry.id, { | |
| entry, | |
| children: [], | |
| label: labelMap.get(entry.id) | |
| }); | |
| } | |
| // Build parent-child relationships | |
| for (const entry of entries) { | |
| const node = nodeMap.get(entry.id); | |
| if (entry.parentId === null || entry.parentId === undefined || entry.parentId === entry.id) { | |
| roots.push(node); | |
| } else { | |
| const parent = nodeMap.get(entry.parentId); | |
| if (parent) { | |
| parent.children.push(node); | |
| } else { | |
| roots.push(node); | |
| } | |
| } | |
| } | |
| // Sort children by timestamp | |
| function sortChildren(node) { | |
| node.children.sort((a, b) => | |
| new Date(a.entry.timestamp).getTime() - new Date(b.entry.timestamp).getTime() | |
| ); | |
| node.children.forEach(sortChildren); | |
| } | |
| roots.forEach(sortChildren); | |
| return roots; | |
| } | |
| /** | |
| * Build set of entry IDs on path from root to target. | |
| */ | |
| function buildActivePathIds(targetId) { | |
| const ids = new Set(); | |
| let current = byId.get(targetId); | |
| while (current) { | |
| ids.add(current.id); | |
| // Stop if no parent or self-referencing (root) | |
| if (!current.parentId || current.parentId === current.id) { | |
| break; | |
| } | |
| current = byId.get(current.parentId); | |
| } | |
| return ids; | |
| } | |
| /** | |
| * Get array of entries from root to target (the conversation path). | |
| */ | |
| function getPath(targetId) { | |
| const path = []; | |
| let current = byId.get(targetId); | |
| while (current) { | |
| path.unshift(current); | |
| // Stop if no parent or self-referencing (root) | |
| if (!current.parentId || current.parentId === current.id) { | |
| break; | |
| } | |
| current = byId.get(current.parentId); | |
| } | |
| return path; | |
| } | |
| // Tree node lookup for finding leaves | |
| let treeNodeMap = null; | |
| /** | |
| * Find the newest leaf node reachable from a given node. | |
| * This allows clicking any node in a branch to show the full branch. | |
| * Children are sorted by timestamp, so the newest is always last. | |
| */ | |
| function findNewestLeaf(nodeId) { | |
| // Build tree node map lazily | |
| if (!treeNodeMap) { | |
| treeNodeMap = new Map(); | |
| const tree = buildTree(); | |
| function mapNodes(node) { | |
| treeNodeMap.set(node.entry.id, node); | |
| node.children.forEach(mapNodes); | |
| } | |
| tree.forEach(mapNodes); | |
| } | |
| const node = treeNodeMap.get(nodeId); | |
| if (!node) return nodeId; | |
| // Follow the newest (last) child at each level | |
| let current = node; | |
| while (current.children.length > 0) { | |
| current = current.children[current.children.length - 1]; | |
| } | |
| return current.entry.id; | |
| } | |
| /** | |
| * Flatten tree into list with indentation and connector info. | |
| * Returns array of { node, indent, showConnector, isLast, gutters, isVirtualRootChild, multipleRoots }. | |
| * Matches tree-selector.ts logic exactly. | |
| */ | |
| function flattenTree(roots, activePathIds) { | |
| const result = []; | |
| const multipleRoots = roots.length > 1; | |
| // Mark which subtrees contain the active leaf | |
| const containsActive = new Map(); | |
| function markActive(node) { | |
| let has = activePathIds.has(node.entry.id); | |
| for (const child of node.children) { | |
| if (markActive(child)) has = true; | |
| } | |
| containsActive.set(node, has); | |
| return has; | |
| } | |
| roots.forEach(markActive); | |
| // Stack: [node, indent, justBranched, showConnector, isLast, gutters, isVirtualRootChild] | |
| const stack = []; | |
| // Add roots (prioritize branch containing active leaf) | |
| const orderedRoots = [...roots].sort((a, b) => | |
| Number(containsActive.get(b)) - Number(containsActive.get(a)) | |
| ); | |
| for (let i = orderedRoots.length - 1; i >= 0; i--) { | |
| const isLast = i === orderedRoots.length - 1; | |
| stack.push([orderedRoots[i], multipleRoots ? 1 : 0, multipleRoots, multipleRoots, isLast, [], multipleRoots]); | |
| } | |
| while (stack.length > 0) { | |
| const [node, indent, justBranched, showConnector, isLast, gutters, isVirtualRootChild] = stack.pop(); | |
| result.push({ node, indent, showConnector, isLast, gutters, isVirtualRootChild, multipleRoots }); | |
| const children = node.children; | |
| const multipleChildren = children.length > 1; | |
| // Order children (active branch first) | |
| const orderedChildren = [...children].sort((a, b) => | |
| Number(containsActive.get(b)) - Number(containsActive.get(a)) | |
| ); | |
| // Calculate child indent (matches tree-selector.ts) | |
| let childIndent; | |
| if (multipleChildren) { | |
| // Parent branches: children get +1 | |
| childIndent = indent + 1; | |
| } else if (justBranched && indent > 0) { | |
| // First generation after a branch: +1 for visual grouping | |
| childIndent = indent + 1; | |
| } else { | |
| // Single-child chain: stay flat | |
| childIndent = indent; | |
| } | |
| // Build gutters for children | |
| const connectorDisplayed = showConnector && !isVirtualRootChild; | |
| const currentDisplayIndent = multipleRoots ? Math.max(0, indent - 1) : indent; | |
| const connectorPosition = Math.max(0, currentDisplayIndent - 1); | |
| const childGutters = connectorDisplayed | |
| ? [...gutters, { position: connectorPosition, show: !isLast }] | |
| : gutters; | |
| // Add children in reverse order for stack | |
| for (let i = orderedChildren.length - 1; i >= 0; i--) { | |
| const childIsLast = i === orderedChildren.length - 1; | |
| stack.push([orderedChildren[i], childIndent, multipleChildren, multipleChildren, childIsLast, childGutters, false]); | |
| } | |
| } | |
| return result; | |
| } | |
| /** | |
| * Build ASCII prefix string for tree node. | |
| */ | |
| function buildTreePrefix(flatNode) { | |
| const { indent, showConnector, isLast, gutters, isVirtualRootChild, multipleRoots } = flatNode; | |
| const displayIndent = multipleRoots ? Math.max(0, indent - 1) : indent; | |
| const connector = showConnector && !isVirtualRootChild ? (isLast ? '└─ ' : '├─ ') : ''; | |
| const connectorPosition = connector ? displayIndent - 1 : -1; | |
| const totalChars = displayIndent * 3; | |
| const prefixChars = []; | |
| for (let i = 0; i < totalChars; i++) { | |
| const level = Math.floor(i / 3); | |
| const posInLevel = i % 3; | |
| const gutter = gutters.find(g => g.position === level); | |
| if (gutter) { | |
| prefixChars.push(posInLevel === 0 ? (gutter.show ? '│' : ' ') : ' '); | |
| } else if (connector && level === connectorPosition) { | |
| if (posInLevel === 0) { | |
| prefixChars.push(isLast ? '└' : '├'); | |
| } else if (posInLevel === 1) { | |
| prefixChars.push('─'); | |
| } else { | |
| prefixChars.push(' '); | |
| } | |
| } else { | |
| prefixChars.push(' '); | |
| } | |
| } | |
| return prefixChars.join(''); | |
| } | |
| // ============================================================ | |
| // FILTERING (pure data) | |
| // ============================================================ | |
| let filterMode = 'default'; | |
| let searchQuery = ''; | |
| function hasTextContent(content) { | |
| if (typeof content === 'string') return content.trim().length > 0; | |
| if (Array.isArray(content)) { | |
| for (const c of content) { | |
| if (c.type === 'text' && c.text && c.text.trim().length > 0) return true; | |
| } | |
| } | |
| return false; | |
| } | |
| function extractContent(content) { | |
| if (typeof content === 'string') return content; | |
| if (Array.isArray(content)) { | |
| return content | |
| .filter(c => c.type === 'text' && c.text) | |
| .map(c => c.text) | |
| .join(''); | |
| } | |
| return ''; | |
| } | |
| function getSearchableText(entry, label) { | |
| const parts = []; | |
| if (label) parts.push(label); | |
| switch (entry.type) { | |
| case 'message': { | |
| const msg = entry.message; | |
| parts.push(msg.role); | |
| if (msg.content) parts.push(extractContent(msg.content)); | |
| if (msg.role === 'bashExecution' && msg.command) parts.push(msg.command); | |
| break; | |
| } | |
| case 'custom_message': | |
| parts.push(entry.customType); | |
| parts.push(typeof entry.content === 'string' ? entry.content : extractContent(entry.content)); | |
| break; | |
| case 'compaction': | |
| parts.push('compaction'); | |
| break; | |
| case 'branch_summary': | |
| parts.push('branch summary', entry.summary); | |
| break; | |
| case 'model_change': | |
| parts.push('model', entry.modelId); | |
| break; | |
| case 'thinking_level_change': | |
| parts.push('thinking', entry.thinkingLevel); | |
| break; | |
| } | |
| return parts.join(' ').toLowerCase(); | |
| } | |
| /** | |
| * Filter flat nodes based on current filterMode and searchQuery. | |
| */ | |
| function filterNodes(flatNodes, currentLeafId) { | |
| const searchTokens = searchQuery.toLowerCase().split(/\s+/).filter(Boolean); | |
| const filtered = flatNodes.filter(flatNode => { | |
| const entry = flatNode.node.entry; | |
| const label = flatNode.node.label; | |
| const isCurrentLeaf = entry.id === currentLeafId; | |
| // Always show current leaf | |
| if (isCurrentLeaf) return true; | |
| // Hide assistant messages with only tool calls (no text) unless error/aborted | |
| if (entry.type === 'message' && entry.message.role === 'assistant') { | |
| const msg = entry.message; | |
| const hasText = hasTextContent(msg.content); | |
| const isErrorOrAborted = msg.stopReason && msg.stopReason !== 'stop' && msg.stopReason !== 'toolUse'; | |
| if (!hasText && !isErrorOrAborted) return false; | |
| } | |
| // Apply filter mode | |
| const isSettingsEntry = ['label', 'custom', 'model_change', 'thinking_level_change'].includes(entry.type); | |
| let passesFilter = true; | |
| switch (filterMode) { | |
| case 'user-only': | |
| passesFilter = entry.type === 'message' && entry.message.role === 'user'; | |
| break; | |
| case 'no-tools': | |
| passesFilter = !isSettingsEntry && !(entry.type === 'message' && entry.message.role === 'toolResult'); | |
| break; | |
| case 'labeled-only': | |
| passesFilter = label !== undefined; | |
| break; | |
| case 'all': | |
| passesFilter = true; | |
| break; | |
| default: // 'default' | |
| passesFilter = !isSettingsEntry; | |
| break; | |
| } | |
| if (!passesFilter) return false; | |
| // Apply search filter | |
| if (searchTokens.length > 0) { | |
| const nodeText = getSearchableText(entry, label); | |
| if (!searchTokens.every(t => nodeText.includes(t))) return false; | |
| } | |
| return true; | |
| }); | |
| // Recalculate visual structure based on visible tree | |
| recalculateVisualStructure(filtered, flatNodes); | |
| return filtered; | |
| } | |
| /** | |
| * Recompute indentation/connectors for the filtered view | |
| * | |
| * Filtering can hide intermediate entries; descendants attach to the nearest visible ancestor. | |
| * Keep indentation semantics aligned with flattenTree() so single-child chains don't drift right. | |
| */ | |
| function recalculateVisualStructure(filteredNodes, allFlatNodes) { | |
| if (filteredNodes.length === 0) return; | |
| const visibleIds = new Set(filteredNodes.map(n => n.node.entry.id)); | |
| // Build entry map for parent lookup (using full tree) | |
| const entryMap = new Map(); | |
| for (const flatNode of allFlatNodes) { | |
| entryMap.set(flatNode.node.entry.id, flatNode); | |
| } | |
| // Find nearest visible ancestor for a node | |
| function findVisibleAncestor(nodeId) { | |
| let currentId = entryMap.get(nodeId)?.node.entry.parentId; | |
| while (currentId != null) { | |
| if (visibleIds.has(currentId)) { | |
| return currentId; | |
| } | |
| currentId = entryMap.get(currentId)?.node.entry.parentId; | |
| } | |
| return null; | |
| } | |
| // Build visible tree structure | |
| const visibleParent = new Map(); | |
| const visibleChildren = new Map(); | |
| visibleChildren.set(null, []); // root-level nodes | |
| for (const flatNode of filteredNodes) { | |
| const nodeId = flatNode.node.entry.id; | |
| const ancestorId = findVisibleAncestor(nodeId); | |
| visibleParent.set(nodeId, ancestorId); | |
| if (!visibleChildren.has(ancestorId)) { | |
| visibleChildren.set(ancestorId, []); | |
| } | |
| visibleChildren.get(ancestorId).push(nodeId); | |
| } | |
| // Update multipleRoots based on visible roots | |
| const visibleRootIds = visibleChildren.get(null); | |
| const multipleRoots = visibleRootIds.length > 1; | |
| // Build a map for quick lookup: nodeId → FlatNode | |
| const filteredNodeMap = new Map(); | |
| for (const flatNode of filteredNodes) { | |
| filteredNodeMap.set(flatNode.node.entry.id, flatNode); | |
| } | |
| // DFS traversal of visible tree, applying same indentation rules as flattenTree() | |
| // Stack items: [nodeId, indent, justBranched, showConnector, isLast, gutters, isVirtualRootChild] | |
| const stack = []; | |
| // Add visible roots in reverse order (to process in forward order via stack) | |
| for (let i = visibleRootIds.length - 1; i >= 0; i--) { | |
| const isLast = i === visibleRootIds.length - 1; | |
| stack.push([ | |
| visibleRootIds[i], | |
| multipleRoots ? 1 : 0, | |
| multipleRoots, | |
| multipleRoots, | |
| isLast, | |
| [], | |
| multipleRoots | |
| ]); | |
| } | |
| while (stack.length > 0) { | |
| const [nodeId, indent, justBranched, showConnector, isLast, gutters, isVirtualRootChild] = stack.pop(); | |
| const flatNode = filteredNodeMap.get(nodeId); | |
| if (!flatNode) continue; | |
| // Update this node's visual properties | |
| flatNode.indent = indent; | |
| flatNode.showConnector = showConnector; | |
| flatNode.isLast = isLast; | |
| flatNode.gutters = gutters; | |
| flatNode.isVirtualRootChild = isVirtualRootChild; | |
| flatNode.multipleRoots = multipleRoots; | |
| // Get visible children of this node | |
| const children = visibleChildren.get(nodeId) || []; | |
| const multipleChildren = children.length > 1; | |
| // Calculate child indent using same rules as flattenTree(): | |
| // - Parent branches (multiple children): children get +1 | |
| // - Just branched and indent > 0: children get +1 for visual grouping | |
| // - Single-child chain: stay flat | |
| let childIndent; | |
| if (multipleChildren) { | |
| childIndent = indent + 1; | |
| } else if (justBranched && indent > 0) { | |
| childIndent = indent + 1; | |
| } else { | |
| childIndent = indent; | |
| } | |
| // Build gutters for children (same logic as flattenTree) | |
| const connectorDisplayed = showConnector && !isVirtualRootChild; | |
| const currentDisplayIndent = multipleRoots ? Math.max(0, indent - 1) : indent; | |
| const connectorPosition = Math.max(0, currentDisplayIndent - 1); | |
| const childGutters = connectorDisplayed | |
| ? [...gutters, { position: connectorPosition, show: !isLast }] | |
| : gutters; | |
| // Add children in reverse order (to process in forward order via stack) | |
| for (let i = children.length - 1; i >= 0; i--) { | |
| const childIsLast = i === children.length - 1; | |
| stack.push([ | |
| children[i], | |
| childIndent, | |
| multipleChildren, | |
| multipleChildren, | |
| childIsLast, | |
| childGutters, | |
| false | |
| ]); | |
| } | |
| } | |
| } | |
| // ============================================================ | |
| // TREE DISPLAY TEXT (pure data -> string) | |
| // ============================================================ | |
| function shortenPath(p) { | |
| if (typeof p !== 'string') return ''; | |
| if (p.startsWith('/Users/')) { | |
| const parts = p.split('/'); | |
| if (parts.length > 2) return '~' + p.slice(('/Users/' + parts[2]).length); | |
| } | |
| if (p.startsWith('/home/')) { | |
| const parts = p.split('/'); | |
| if (parts.length > 2) return '~' + p.slice(('/home/' + parts[2]).length); | |
| } | |
| return p; | |
| } | |
| function formatToolCall(name, args) { | |
| switch (name) { | |
| case 'read': { | |
| const path = shortenPath(String(args.path || args.file_path || '')); | |
| const offset = args.offset; | |
| const limit = args.limit; | |
| let display = path; | |
| if (offset !== undefined || limit !== undefined) { | |
| const start = offset ?? 1; | |
| const end = limit !== undefined ? start + limit - 1 : ''; | |
| display += `:${start}${end ? `-${end}` : ''}`; | |
| } | |
| return `[read: ${display}]`; | |
| } | |
| case 'write': | |
| return `[write: ${shortenPath(String(args.path || args.file_path || ''))}]`; | |
| case 'edit': | |
| return `[edit: ${shortenPath(String(args.path || args.file_path || ''))}]`; | |
| case 'bash': { | |
| const rawCmd = String(args.command || ''); | |
| const cmd = rawCmd.replace(/[\n\t]/g, ' ').trim().slice(0, 50); | |
| return `[bash: ${cmd}${rawCmd.length > 50 ? '...' : ''}]`; | |
| } | |
| case 'grep': | |
| return `[grep: /${args.pattern || ''}/ in ${shortenPath(String(args.path || '.'))}]`; | |
| case 'find': | |
| return `[find: ${args.pattern || ''} in ${shortenPath(String(args.path || '.'))}]`; | |
| case 'ls': | |
| return `[ls: ${shortenPath(String(args.path || '.'))}]`; | |
| default: { | |
| const argsStr = JSON.stringify(args).slice(0, 40); | |
| return `[${name}: ${argsStr}${JSON.stringify(args).length > 40 ? '...' : ''}]`; | |
| } | |
| } | |
| } | |
| function escapeHtml(text) { | |
| const div = document.createElement('div'); | |
| div.textContent = text; | |
| return div.innerHTML; | |
| } | |
| /** | |
| * Truncate string to maxLen chars, append "..." if truncated. | |
| */ | |
| function truncate(s, maxLen = 100) { | |
| if (s.length <= maxLen) return s; | |
| return s.slice(0, maxLen) + '...'; | |
| } | |
| /** | |
| * Get display text for tree node (returns HTML string). | |
| */ | |
| function getTreeNodeDisplayHtml(entry, label) { | |
| const normalize = s => s.replace(/[\n\t]/g, ' ').trim(); | |
| const labelHtml = label ? `<span class="tree-label">[${escapeHtml(label)}]</span> ` : ''; | |
| switch (entry.type) { | |
| case 'message': { | |
| const msg = entry.message; | |
| if (msg.role === 'user') { | |
| const content = truncate(normalize(extractContent(msg.content))); | |
| return labelHtml + `<span class="tree-role-user">user:</span> ${escapeHtml(content)}`; | |
| } | |
| if (msg.role === 'assistant') { | |
| const textContent = truncate(normalize(extractContent(msg.content))); | |
| if (textContent) { | |
| return labelHtml + `<span class="tree-role-assistant">assistant:</span> ${escapeHtml(textContent)}`; | |
| } | |
| if (msg.stopReason === 'aborted') { | |
| return labelHtml + `<span class="tree-role-assistant">assistant:</span> <span class="tree-muted">(aborted)</span>`; | |
| } | |
| if (msg.errorMessage) { | |
| return labelHtml + `<span class="tree-role-assistant">assistant:</span> <span class="tree-error">${escapeHtml(truncate(msg.errorMessage))}</span>`; | |
| } | |
| return labelHtml + `<span class="tree-role-assistant">assistant:</span> <span class="tree-muted">(no text)</span>`; | |
| } | |
| if (msg.role === 'toolResult') { | |
| const toolCall = msg.toolCallId ? toolCallMap.get(msg.toolCallId) : null; | |
| if (toolCall) { | |
| return labelHtml + `<span class="tree-role-tool">${escapeHtml(formatToolCall(toolCall.name, toolCall.arguments))}</span>`; | |
| } | |
| return labelHtml + `<span class="tree-role-tool">[${msg.toolName || 'tool'}]</span>`; | |
| } | |
| if (msg.role === 'bashExecution') { | |
| const cmd = truncate(normalize(msg.command || '')); | |
| return labelHtml + `<span class="tree-role-tool">[bash]:</span> ${escapeHtml(cmd)}`; | |
| } | |
| return labelHtml + `<span class="tree-muted">[${msg.role}]</span>`; | |
| } | |
| case 'compaction': | |
| return labelHtml + `<span class="tree-compaction">[compaction: ${Math.round(entry.tokensBefore/1000)}k tokens]</span>`; | |
| case 'branch_summary': { | |
| const summary = truncate(normalize(entry.summary || '')); | |
| return labelHtml + `<span class="tree-branch-summary">[branch summary]:</span> ${escapeHtml(summary)}`; | |
| } | |
| case 'custom_message': { | |
| const content = typeof entry.content === 'string' ? entry.content : extractContent(entry.content); | |
| return labelHtml + `<span class="tree-custom">[${escapeHtml(entry.customType)}]:</span> ${escapeHtml(truncate(normalize(content)))}`; | |
| } | |
| case 'model_change': | |
| return labelHtml + `<span class="tree-muted">[model: ${entry.modelId}]</span>`; | |
| case 'thinking_level_change': | |
| return labelHtml + `<span class="tree-muted">[thinking: ${entry.thinkingLevel}]</span>`; | |
| default: | |
| return labelHtml + `<span class="tree-muted">[${entry.type}]</span>`; | |
| } | |
| } | |
| // ============================================================ | |
| // TREE RENDERING (DOM manipulation) | |
| // ============================================================ | |
| let currentLeafId = leafId; | |
| let currentTargetId = urlTargetId || leafId; | |
| let treeRendered = false; | |
| function renderTree() { | |
| const tree = buildTree(); | |
| const activePathIds = buildActivePathIds(currentLeafId); | |
| const flatNodes = flattenTree(tree, activePathIds); | |
| const filtered = filterNodes(flatNodes, currentLeafId); | |
| const container = document.getElementById('tree-container'); | |
| // Full render only on first call or when filter/search changes | |
| if (!treeRendered) { | |
| container.innerHTML = ''; | |
| for (const flatNode of filtered) { | |
| const entry = flatNode.node.entry; | |
| const isOnPath = activePathIds.has(entry.id); | |
| const isTarget = entry.id === currentTargetId; | |
| const div = document.createElement('div'); | |
| div.className = 'tree-node'; | |
| if (isOnPath) div.classList.add('in-path'); | |
| if (isTarget) div.classList.add('active'); | |
| div.dataset.id = entry.id; | |
| const prefix = buildTreePrefix(flatNode); | |
| const prefixSpan = document.createElement('span'); | |
| prefixSpan.className = 'tree-prefix'; | |
| prefixSpan.textContent = prefix; | |
| const marker = document.createElement('span'); | |
| marker.className = 'tree-marker'; | |
| marker.textContent = isOnPath ? '•' : ' '; | |
| const content = document.createElement('span'); | |
| content.className = 'tree-content'; | |
| content.innerHTML = getTreeNodeDisplayHtml(entry, flatNode.node.label); | |
| div.appendChild(prefixSpan); | |
| div.appendChild(marker); | |
| div.appendChild(content); | |
| // Navigate to the newest leaf through this node, but scroll to the clicked node | |
| div.addEventListener('click', () => { | |
| const leafId = findNewestLeaf(entry.id); | |
| navigateTo(leafId, 'target', entry.id); | |
| }); | |
| container.appendChild(div); | |
| } | |
| treeRendered = true; | |
| } else { | |
| // Just update markers and classes | |
| const nodes = container.querySelectorAll('.tree-node'); | |
| for (const node of nodes) { | |
| const id = node.dataset.id; | |
| const isOnPath = activePathIds.has(id); | |
| const isTarget = id === currentTargetId; | |
| node.classList.toggle('in-path', isOnPath); | |
| node.classList.toggle('active', isTarget); | |
| const marker = node.querySelector('.tree-marker'); | |
| if (marker) { | |
| marker.textContent = isOnPath ? '•' : ' '; | |
| } | |
| } | |
| } | |
| document.getElementById('tree-status').textContent = `${filtered.length} / ${flatNodes.length} entries`; | |
| // Scroll active node into view after layout | |
| setTimeout(() => { | |
| const activeNode = container.querySelector('.tree-node.active'); | |
| if (activeNode) { | |
| activeNode.scrollIntoView({ block: 'nearest' }); | |
| } | |
| }, 0); | |
| } | |
| function forceTreeRerender() { | |
| treeRendered = false; | |
| renderTree(); | |
| } | |
| // ============================================================ | |
| // MESSAGE RENDERING | |
| // ============================================================ | |
| function formatTokens(count) { | |
| if (count < 1000) return count.toString(); | |
| if (count < 10000) return (count / 1000).toFixed(1) + 'k'; | |
| if (count < 1000000) return Math.round(count / 1000) + 'k'; | |
| return (count / 1000000).toFixed(1) + 'M'; | |
| } | |
| function formatTimestamp(ts) { | |
| if (!ts) return ''; | |
| const date = new Date(ts); | |
| return date.toLocaleTimeString(undefined, { hour: '2-digit', minute: '2-digit', second: '2-digit' }); | |
| } | |
| function replaceTabs(text) { | |
| return text.replace(/\t/g, ' '); | |
| } | |
| /** Safely coerce value to string for display. Returns null if invalid type. */ | |
| function str(value) { | |
| if (typeof value === 'string') return value; | |
| if (value == null) return ''; | |
| return null; | |
| } | |
| function getLanguageFromPath(filePath) { | |
| const ext = filePath.split('.').pop()?.toLowerCase(); | |
| const extToLang = { | |
| ts: 'typescript', tsx: 'typescript', js: 'javascript', jsx: 'javascript', | |
| py: 'python', rb: 'ruby', rs: 'rust', go: 'go', java: 'java', | |
| c: 'c', cpp: 'cpp', h: 'c', hpp: 'cpp', cs: 'csharp', | |
| php: 'php', sh: 'bash', bash: 'bash', zsh: 'bash', | |
| sql: 'sql', html: 'html', css: 'css', scss: 'scss', | |
| json: 'json', yaml: 'yaml', yml: 'yaml', xml: 'xml', | |
| md: 'markdown', dockerfile: 'dockerfile' | |
| }; | |
| return extToLang[ext]; | |
| } | |
| function findToolResult(toolCallId) { | |
| for (const entry of entries) { | |
| if (entry.type === 'message' && entry.message.role === 'toolResult') { | |
| if (entry.message.toolCallId === toolCallId) { | |
| return entry.message; | |
| } | |
| } | |
| } | |
| return null; | |
| } | |
| function formatExpandableOutput(text, maxLines, lang) { | |
| text = replaceTabs(text); | |
| const lines = text.split('\n'); | |
| const displayLines = lines.slice(0, maxLines); | |
| const remaining = lines.length - maxLines; | |
| if (lang) { | |
| let highlighted; | |
| try { | |
| highlighted = hljs.highlight(text, { language: lang }).value; | |
| } catch { | |
| highlighted = escapeHtml(text); | |
| } | |
| if (remaining > 0) { | |
| const previewCode = displayLines.join('\n'); | |
| let previewHighlighted; | |
| try { | |
| previewHighlighted = hljs.highlight(previewCode, { language: lang }).value; | |
| } catch { | |
| previewHighlighted = escapeHtml(previewCode); | |
| } | |
| return `<div class="tool-output expandable" onclick="this.classList.toggle('expanded')"> | |
| <div class="output-preview"><pre><code class="hljs">${previewHighlighted}</code></pre> | |
| <div class="expand-hint">... (${remaining} more lines)</div></div> | |
| <div class="output-full"><pre><code class="hljs">${highlighted}</code></pre></div></div>`; | |
| } | |
| return `<div class="tool-output"><pre><code class="hljs">${highlighted}</code></pre></div>`; | |
| } | |
| // Plain text output | |
| if (remaining > 0) { | |
| let out = '<div class="tool-output expandable" onclick="this.classList.toggle(\'expanded\')">'; | |
| out += '<div class="output-preview">'; | |
| for (const line of displayLines) { | |
| out += `<div>${escapeHtml(replaceTabs(line))}</div>`; | |
| } | |
| out += `<div class="expand-hint">... (${remaining} more lines)</div></div>`; | |
| out += '<div class="output-full">'; | |
| for (const line of lines) { | |
| out += `<div>${escapeHtml(replaceTabs(line))}</div>`; | |
| } | |
| out += '</div></div>'; | |
| return out; | |
| } | |
| let out = '<div class="tool-output">'; | |
| for (const line of displayLines) { | |
| out += `<div>${escapeHtml(replaceTabs(line))}</div>`; | |
| } | |
| out += '</div>'; | |
| return out; | |
| } | |
| function renderToolCall(call) { | |
| const result = findToolResult(call.id); | |
| const isError = result?.isError || false; | |
| const statusClass = result ? (isError ? 'error' : 'success') : 'pending'; | |
| const getResultText = () => { | |
| if (!result) return ''; | |
| const textBlocks = result.content.filter(c => c.type === 'text'); | |
| return textBlocks.map(c => c.text).join('\n'); | |
| }; | |
| const getResultImages = () => { | |
| if (!result) return []; | |
| return result.content.filter(c => c.type === 'image'); | |
| }; | |
| const renderResultImages = () => { | |
| const images = getResultImages(); | |
| if (images.length === 0) return ''; | |
| return '<div class="tool-images">' + | |
| images.map(img => `<img src="data:${img.mimeType};base64,${img.data}" class="tool-image" />`).join('') + | |
| '</div>'; | |
| }; | |
| let html = `<div class="tool-execution ${statusClass}">`; | |
| const args = call.arguments || {}; | |
| const name = call.name; | |
| const invalidArg = '<span class="tool-error">[invalid arg]</span>'; | |
| switch (name) { | |
| case 'bash': { | |
| const command = str(args.command); | |
| const cmdDisplay = command === null ? invalidArg : escapeHtml(command || '...'); | |
| html += `<div class="tool-command">$ ${cmdDisplay}</div>`; | |
| if (result) { | |
| const output = getResultText().trim(); | |
| if (output) html += formatExpandableOutput(output, 5); | |
| } | |
| break; | |
| } | |
| case 'read': { | |
| const filePath = str(args.file_path ?? args.path); | |
| const offset = args.offset; | |
| const limit = args.limit; | |
| let pathHtml = filePath === null ? invalidArg : escapeHtml(shortenPath(filePath || '')); | |
| if (filePath !== null && (offset !== undefined || limit !== undefined)) { | |
| const startLine = offset ?? 1; | |
| const endLine = limit !== undefined ? startLine + limit - 1 : ''; | |
| pathHtml += `<span class="line-numbers">:${startLine}${endLine ? '-' + endLine : ''}</span>`; | |
| } | |
| html += `<div class="tool-header"><span class="tool-name">read</span> <span class="tool-path">${pathHtml}</span></div>`; | |
| if (result) { | |
| html += renderResultImages(); | |
| const output = getResultText(); | |
| const lang = filePath ? getLanguageFromPath(filePath) : null; | |
| if (output) html += formatExpandableOutput(output, 10, lang); | |
| } | |
| break; | |
| } | |
| case 'write': { | |
| const filePath = str(args.file_path ?? args.path); | |
| const content = str(args.content); | |
| html += `<div class="tool-header"><span class="tool-name">write</span> <span class="tool-path">${filePath === null ? invalidArg : escapeHtml(shortenPath(filePath || ''))}</span>`; | |
| if (content !== null && content) { | |
| const lines = content.split('\n'); | |
| if (lines.length > 10) html += ` <span class="line-count">(${lines.length} lines)</span>`; | |
| } | |
| html += '</div>'; | |
| if (content === null) { | |
| html += `<div class="tool-error">[invalid content arg - expected string]</div>`; | |
| } else if (content) { | |
| const lang = filePath ? getLanguageFromPath(filePath) : null; | |
| html += formatExpandableOutput(content, 10, lang); | |
| } | |
| if (result) { | |
| const output = getResultText().trim(); | |
| if (output) html += `<div class="tool-output"><div>${escapeHtml(output)}</div></div>`; | |
| } | |
| break; | |
| } | |
| case 'edit': { | |
| const filePath = str(args.file_path ?? args.path); | |
| html += `<div class="tool-header"><span class="tool-name">edit</span> <span class="tool-path">${filePath === null ? invalidArg : escapeHtml(shortenPath(filePath || ''))}</span></div>`; | |
| if (result?.details?.diff) { | |
| const diffLines = result.details.diff.split('\n'); | |
| html += '<div class="tool-diff">'; | |
| for (const line of diffLines) { | |
| const cls = line.match(/^\+/) ? 'diff-added' : line.match(/^-/) ? 'diff-removed' : 'diff-context'; | |
| html += `<div class="${cls}">${escapeHtml(replaceTabs(line))}</div>`; | |
| } | |
| html += '</div>'; | |
| } else if (result) { | |
| const output = getResultText().trim(); | |
| if (output) html += `<div class="tool-output"><pre>${escapeHtml(output)}</pre></div>`; | |
| } | |
| break; | |
| } | |
| default: { | |
| // Check for pre-rendered custom tool HTML | |
| const rendered = renderedTools?.[call.id]; | |
| if (rendered?.callHtml || rendered?.resultHtml) { | |
| // Custom tool with pre-rendered HTML from TUI renderer | |
| if (rendered.callHtml) { | |
| html += `<div class="tool-header ansi-rendered">${rendered.callHtml}</div>`; | |
| } else { | |
| html += `<div class="tool-header"><span class="tool-name">${escapeHtml(name)}</span></div>`; | |
| } | |
| if (rendered.resultHtml) { | |
| // Apply same truncation as built-in tools (10 lines) | |
| const lines = rendered.resultHtml.split('\n'); | |
| if (lines.length > 10) { | |
| const preview = lines.slice(0, 10).join('\n'); | |
| html += `<div class="tool-output expandable ansi-rendered" onclick="this.classList.toggle('expanded')"> | |
| <div class="output-preview">${preview}<div class="expand-hint">... (${lines.length - 10} more lines)</div></div> | |
| <div class="output-full">${rendered.resultHtml}</div> | |
| </div>`; | |
| } else { | |
| html += `<div class="tool-output ansi-rendered">${rendered.resultHtml}</div>`; | |
| } | |
| } else if (result) { | |
| // Fallback to JSON for result if no pre-rendered HTML | |
| const output = getResultText(); | |
| if (output) html += formatExpandableOutput(output, 10); | |
| } | |
| } else { | |
| // Fallback to JSON display (existing behavior) | |
| html += `<div class="tool-header"><span class="tool-name">${escapeHtml(name)}</span></div>`; | |
| html += `<div class="tool-output"><pre>${escapeHtml(JSON.stringify(args, null, 2))}</pre></div>`; | |
| if (result) { | |
| const output = getResultText(); | |
| if (output) html += formatExpandableOutput(output, 10); | |
| } | |
| } | |
| } | |
| } | |
| html += '</div>'; | |
| return html; | |
| } | |
| /** | |
| * Download the session data as a JSONL file. | |
| * Reconstructs the original format: header line + entry lines. | |
| */ | |
| window.downloadSessionJson = function() { | |
| // Build JSONL content: header first, then all entries | |
| const lines = []; | |
| if (header) { | |
| lines.push(JSON.stringify({ type: 'header', ...header })); | |
| } | |
| for (const entry of entries) { | |
| lines.push(JSON.stringify(entry)); | |
| } | |
| const jsonlContent = lines.join('\n'); | |
| // Create download | |
| const blob = new Blob([jsonlContent], { type: 'application/x-ndjson' }); | |
| const url = URL.createObjectURL(blob); | |
| const a = document.createElement('a'); | |
| a.href = url; | |
| a.download = `${header?.id || 'session'}.jsonl`; | |
| document.body.appendChild(a); | |
| a.click(); | |
| document.body.removeChild(a); | |
| URL.revokeObjectURL(url); | |
| } | |
| /** | |
| * Build a shareable URL for a specific message. | |
| * URL format: base?gistId&leafId=<leafId>&targetId=<entryId> | |
| */ | |
| function buildShareUrl(entryId) { | |
| // Check for injected base URL (used when loaded in iframe via srcdoc) | |
| const baseUrlMeta = document.querySelector('meta[name="pi-share-base-url"]'); | |
| const baseUrl = baseUrlMeta ? baseUrlMeta.content : window.location.href.split('?')[0]; | |
| const url = new URL(window.location.href); | |
| // Find the gist ID (first query param without value, e.g., ?abc123) | |
| const gistId = Array.from(url.searchParams.keys()).find(k => !url.searchParams.get(k)); | |
| // Build the share URL | |
| const params = new URLSearchParams(); | |
| params.set('leafId', currentLeafId); | |
| params.set('targetId', entryId); | |
| // If we have an injected base URL (iframe context), use it directly | |
| if (baseUrlMeta) { | |
| return `${baseUrl}&${params.toString()}`; | |
| } | |
| // Otherwise build from current location (direct file access) | |
| url.search = gistId ? `?${gistId}&${params.toString()}` : `?${params.toString()}`; | |
| return url.toString(); | |
| } | |
| /** | |
| * Copy text to clipboard with visual feedback. | |
| * Uses navigator.clipboard with fallback to execCommand for HTTP contexts. | |
| */ | |
| async function copyToClipboard(text, button) { | |
| let success = false; | |
| try { | |
| if (navigator.clipboard && navigator.clipboard.writeText) { | |
| await navigator.clipboard.writeText(text); | |
| success = true; | |
| } | |
| } catch (err) { | |
| // Clipboard API failed, try fallback | |
| } | |
| // Fallback for HTTP or when Clipboard API is unavailable | |
| if (!success) { | |
| try { | |
| const textarea = document.createElement('textarea'); | |
| textarea.value = text; | |
| textarea.style.position = 'fixed'; | |
| textarea.style.opacity = '0'; | |
| document.body.appendChild(textarea); | |
| textarea.select(); | |
| success = document.execCommand('copy'); | |
| document.body.removeChild(textarea); | |
| } catch (err) { | |
| console.error('Failed to copy:', err); | |
| } | |
| } | |
| if (success && button) { | |
| const originalHtml = button.innerHTML; | |
| button.innerHTML = '✓'; | |
| button.classList.add('copied'); | |
| setTimeout(() => { | |
| button.innerHTML = originalHtml; | |
| button.classList.remove('copied'); | |
| }, 1500); | |
| } | |
| } | |
| /** | |
| * Render the copy-link button HTML for a message. | |
| */ | |
| function renderCopyLinkButton(entryId) { | |
| return `<button class="copy-link-btn" data-entry-id="${entryId}" title="Copy link to this message"> | |
| <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"> | |
| <path d="M10 13a5 5 0 0 0 7.54.54l3-3a5 5 0 0 0-7.07-7.07l-1.72 1.71"/> | |
| <path d="M14 11a5 5 0 0 0-7.54-.54l-3 3a5 5 0 0 0 7.07 7.07l1.71-1.71"/> | |
| </svg> | |
| </button>`; | |
| } | |
| function renderEntry(entry) { | |
| const ts = formatTimestamp(entry.timestamp); | |
| const tsHtml = ts ? `<div class="message-timestamp">${ts}</div>` : ''; | |
| const entryId = `entry-${entry.id}`; | |
| const copyBtnHtml = renderCopyLinkButton(entry.id); | |
| if (entry.type === 'message') { | |
| const msg = entry.message; | |
| if (msg.role === 'user') { | |
| let html = `<div class="user-message" id="${entryId}">${copyBtnHtml}${tsHtml}`; | |
| const content = msg.content; | |
| if (Array.isArray(content)) { | |
| const images = content.filter(c => c.type === 'image'); | |
| if (images.length > 0) { | |
| html += '<div class="message-images">'; | |
| for (const img of images) { | |
| html += `<img src="data:${img.mimeType};base64,${img.data}" class="message-image" />`; | |
| } | |
| html += '</div>'; | |
| } | |
| } | |
| const text = typeof content === 'string' ? content : | |
| content.filter(c => c.type === 'text').map(c => c.text).join('\n'); | |
| if (text.trim()) { | |
| html += `<div class="markdown-content">${safeMarkedParse(text)}</div>`; | |
| } | |
| html += '</div>'; | |
| return html; | |
| } | |
| if (msg.role === 'assistant') { | |
| let html = `<div class="assistant-message" id="${entryId}">${copyBtnHtml}${tsHtml}`; | |
| for (const block of msg.content) { | |
| if (block.type === 'text' && block.text.trim()) { | |
| html += `<div class="assistant-text markdown-content">${safeMarkedParse(block.text)}</div>`; | |
| } else if (block.type === 'thinking' && block.thinking.trim()) { | |
| html += `<div class="thinking-block"> | |
| <div class="thinking-text">${escapeHtml(block.thinking)}</div> | |
| <div class="thinking-collapsed">Thinking ...</div> | |
| </div>`; | |
| } | |
| } | |
| for (const block of msg.content) { | |
| if (block.type === 'toolCall') { | |
| html += renderToolCall(block); | |
| } | |
| } | |
| if (msg.stopReason === 'aborted') { | |
| html += '<div class="error-text">Aborted</div>'; | |
| } else if (msg.stopReason === 'error') { | |
| html += `<div class="error-text">Error: ${escapeHtml(msg.errorMessage || 'Unknown error')}</div>`; | |
| } | |
| html += '</div>'; | |
| return html; | |
| } | |
| if (msg.role === 'bashExecution') { | |
| const isError = msg.cancelled || (msg.exitCode !== 0 && msg.exitCode !== null); | |
| let html = `<div class="tool-execution ${isError ? 'error' : 'success'}" id="${entryId}">${tsHtml}`; | |
| html += `<div class="tool-command">$ ${escapeHtml(msg.command)}</div>`; | |
| if (msg.output) html += formatExpandableOutput(msg.output, 10); | |
| if (msg.cancelled) { | |
| html += '<div style="color: var(--warning)">(cancelled)</div>'; | |
| } else if (msg.exitCode !== 0 && msg.exitCode !== null) { | |
| html += `<div style="color: var(--error)">(exit ${msg.exitCode})</div>`; | |
| } | |
| html += '</div>'; | |
| return html; | |
| } | |
| if (msg.role === 'toolResult') return ''; | |
| } | |
| if (entry.type === 'model_change') { | |
| return `<div class="model-change" id="${entryId}">${tsHtml}Switched to model: <span class="model-name">${escapeHtml(entry.provider)}/${escapeHtml(entry.modelId)}</span></div>`; | |
| } | |
| if (entry.type === 'compaction') { | |
| return `<div class="compaction" id="${entryId}" onclick="this.classList.toggle('expanded')"> | |
| <div class="compaction-label">[compaction]</div> | |
| <div class="compaction-collapsed">Compacted from ${entry.tokensBefore.toLocaleString()} tokens</div> | |
| <div class="compaction-content"><strong>Compacted from ${entry.tokensBefore.toLocaleString()} tokens</strong>\n\n${escapeHtml(entry.summary)}</div> | |
| </div>`; | |
| } | |
| if (entry.type === 'branch_summary') { | |
| return `<div class="branch-summary" id="${entryId}">${tsHtml} | |
| <div class="branch-summary-header">Branch Summary</div> | |
| <div class="markdown-content">${safeMarkedParse(entry.summary)}</div> | |
| </div>`; | |
| } | |
| if (entry.type === 'custom_message' && entry.display) { | |
| return `<div class="hook-message" id="${entryId}">${tsHtml} | |
| <div class="hook-type">[${escapeHtml(entry.customType)}]</div> | |
| <div class="markdown-content">${safeMarkedParse(typeof entry.content === 'string' ? entry.content : JSON.stringify(entry.content))}</div> | |
| </div>`; | |
| } | |
| return ''; | |
| } | |
| // ============================================================ | |
| // HEADER / STATS | |
| // ============================================================ | |
| function computeStats(entryList) { | |
| let userMessages = 0, assistantMessages = 0, toolResults = 0; | |
| let customMessages = 0, compactions = 0, branchSummaries = 0, toolCalls = 0; | |
| const tokens = { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 }; | |
| const cost = { input: 0, output: 0, cacheRead: 0, cacheWrite: 0 }; | |
| const models = new Set(); | |
| for (const entry of entryList) { | |
| if (entry.type === 'message') { | |
| const msg = entry.message; | |
| if (msg.role === 'user') userMessages++; | |
| if (msg.role === 'assistant') { | |
| assistantMessages++; | |
| if (msg.model) models.add(msg.provider ? `${msg.provider}/${msg.model}` : msg.model); | |
| if (msg.usage) { | |
| tokens.input += msg.usage.input || 0; | |
| tokens.output += msg.usage.output || 0; | |
| tokens.cacheRead += msg.usage.cacheRead || 0; | |
| tokens.cacheWrite += msg.usage.cacheWrite || 0; | |
| if (msg.usage.cost) { | |
| cost.input += msg.usage.cost.input || 0; | |
| cost.output += msg.usage.cost.output || 0; | |
| cost.cacheRead += msg.usage.cost.cacheRead || 0; | |
| cost.cacheWrite += msg.usage.cost.cacheWrite || 0; | |
| } | |
| } | |
| toolCalls += msg.content.filter(c => c.type === 'toolCall').length; | |
| } | |
| if (msg.role === 'toolResult') toolResults++; | |
| } else if (entry.type === 'compaction') { | |
| compactions++; | |
| } else if (entry.type === 'branch_summary') { | |
| branchSummaries++; | |
| } else if (entry.type === 'custom_message') { | |
| customMessages++; | |
| } | |
| } | |
| return { userMessages, assistantMessages, toolResults, customMessages, compactions, branchSummaries, toolCalls, tokens, cost, models: Array.from(models) }; | |
| } | |
| const globalStats = computeStats(entries); | |
| function renderHeader() { | |
| const totalCost = globalStats.cost.input + globalStats.cost.output + globalStats.cost.cacheRead + globalStats.cost.cacheWrite; | |
| const tokenParts = []; | |
| if (globalStats.tokens.input) tokenParts.push(`↑${formatTokens(globalStats.tokens.input)}`); | |
| if (globalStats.tokens.output) tokenParts.push(`↓${formatTokens(globalStats.tokens.output)}`); | |
| if (globalStats.tokens.cacheRead) tokenParts.push(`R${formatTokens(globalStats.tokens.cacheRead)}`); | |
| if (globalStats.tokens.cacheWrite) tokenParts.push(`W${formatTokens(globalStats.tokens.cacheWrite)}`); | |
| const msgParts = []; | |
| if (globalStats.userMessages) msgParts.push(`${globalStats.userMessages} user`); | |
| if (globalStats.assistantMessages) msgParts.push(`${globalStats.assistantMessages} assistant`); | |
| if (globalStats.toolResults) msgParts.push(`${globalStats.toolResults} tool results`); | |
| if (globalStats.customMessages) msgParts.push(`${globalStats.customMessages} custom`); | |
| if (globalStats.compactions) msgParts.push(`${globalStats.compactions} compactions`); | |
| if (globalStats.branchSummaries) msgParts.push(`${globalStats.branchSummaries} branch summaries`); | |
| let html = ` | |
| <div class="header"> | |
| <h1>Session: ${escapeHtml(header?.id || 'unknown')}</h1> | |
| <div class="help-bar"> | |
| <span>Ctrl+T toggle thinking · Ctrl+O toggle tools</span> | |
| <button class="download-json-btn" onclick="downloadSessionJson()" title="Download session as JSONL">↓ JSONL</button> | |
| </div> | |
| <div class="header-info"> | |
| <div class="info-item"><span class="info-label">Date:</span><span class="info-value">${header?.timestamp ? new Date(header.timestamp).toLocaleString() : 'unknown'}</span></div> | |
| <div class="info-item"><span class="info-label">Models:</span><span class="info-value">${globalStats.models.join(', ') || 'unknown'}</span></div> | |
| <div class="info-item"><span class="info-label">Messages:</span><span class="info-value">${msgParts.join(', ') || '0'}</span></div> | |
| <div class="info-item"><span class="info-label">Tool Calls:</span><span class="info-value">${globalStats.toolCalls}</span></div> | |
| <div class="info-item"><span class="info-label">Tokens:</span><span class="info-value">${tokenParts.join(' ') || '0'}</span></div> | |
| <div class="info-item"><span class="info-label">Cost:</span><span class="info-value">${totalCost.toFixed(3)}</span></div> | |
| </div> | |
| </div>`; | |
| // Render system prompt (user's base prompt, applies to all providers) | |
| if (systemPrompt) { | |
| const lines = systemPrompt.split('\n'); | |
| const previewLines = 10; | |
| if (lines.length > previewLines) { | |
| const preview = lines.slice(0, previewLines).join('\n'); | |
| const remaining = lines.length - previewLines; | |
| html += `<div class="system-prompt expandable" onclick="this.classList.toggle('expanded')"> | |
| <div class="system-prompt-header">System Prompt</div> | |
| <div class="system-prompt-preview">${escapeHtml(preview)}</div> | |
| <div class="system-prompt-expand-hint">... (${remaining} more lines, click to expand)</div> | |
| <div class="system-prompt-full">${escapeHtml(systemPrompt)}</div> | |
| </div>`; | |
| } else { | |
| html += `<div class="system-prompt"> | |
| <div class="system-prompt-header">System Prompt</div> | |
| <div class="system-prompt-full" style="display: block">${escapeHtml(systemPrompt)}</div> | |
| </div>`; | |
| } | |
| } | |
| if (tools && tools.length > 0) { | |
| html += `<div class="tools-list"> | |
| <div class="tools-header">Available Tools</div> | |
| <div class="tools-content"> | |
| ${tools.map(t => { | |
| const hasParams = t.parameters && typeof t.parameters === 'object' && t.parameters.properties && Object.keys(t.parameters.properties).length > 0; | |
| if (!hasParams) { | |
| return `<div class="tool-item"><span class="tool-item-name">${escapeHtml(t.name)}</span> - <span class="tool-item-desc">${escapeHtml(t.description)}</span></div>`; | |
| } | |
| const params = t.parameters; | |
| const properties = params.properties; | |
| const required = params.required || []; | |
| let paramsHtml = ''; | |
| for (const [name, prop] of Object.entries(properties)) { | |
| const isRequired = required.includes(name); | |
| const typeStr = prop.type || 'any'; | |
| const reqLabel = isRequired ? '<span class="tool-param-required">required</span>' : '<span class="tool-param-optional">optional</span>'; | |
| paramsHtml += `<div class="tool-param"><span class="tool-param-name">${escapeHtml(name)}</span> <span class="tool-param-type">${escapeHtml(typeStr)}</span> ${reqLabel}`; | |
| if (prop.description) { | |
| paramsHtml += `<div class="tool-param-desc">${escapeHtml(prop.description)}</div>`; | |
| } | |
| paramsHtml += `</div>`; | |
| } | |
| return `<div class="tool-item" onclick="this.classList.toggle('params-expanded')"><span class="tool-item-name">${escapeHtml(t.name)}</span> - <span class="tool-item-desc">${escapeHtml(t.description)}</span> <span class="tool-params-hint"></span><div class="tool-params-content">${paramsHtml}</div></div>`; | |
| }).join('')} | |
| </div> | |
| </div>`; | |
| } | |
| return html; | |
| } | |
| // ============================================================ | |
| // NAVIGATION | |
| // ============================================================ | |
| // Cache for rendered entry DOM nodes | |
| const entryCache = new Map(); | |
| function renderEntryToNode(entry) { | |
| // Check cache first | |
| if (entryCache.has(entry.id)) { | |
| return entryCache.get(entry.id).cloneNode(true); | |
| } | |
| // Render to HTML string, then parse to node | |
| const html = renderEntry(entry); | |
| if (!html) return null; | |
| const template = document.createElement('template'); | |
| template.innerHTML = html; | |
| const node = template.content.firstElementChild; | |
| // Cache the node | |
| if (node) { | |
| entryCache.set(entry.id, node.cloneNode(true)); | |
| } | |
| return node; | |
| } | |
| function navigateTo(targetId, scrollMode = 'target', scrollToEntryId = null) { | |
| currentLeafId = targetId; | |
| currentTargetId = scrollToEntryId || targetId; | |
| const path = getPath(targetId); | |
| renderTree(); | |
| document.getElementById('header-container').innerHTML = renderHeader(); | |
| // Build messages using cached DOM nodes | |
| const messagesEl = document.getElementById('messages'); | |
| const fragment = document.createDocumentFragment(); | |
| for (const entry of path) { | |
| const node = renderEntryToNode(entry); | |
| if (node) { | |
| fragment.appendChild(node); | |
| } | |
| } | |
| messagesEl.innerHTML = ''; | |
| messagesEl.appendChild(fragment); | |
| // Attach click handlers for copy-link buttons | |
| messagesEl.querySelectorAll('.copy-link-btn').forEach(btn => { | |
| btn.addEventListener('click', (e) => { | |
| e.stopPropagation(); | |
| const entryId = btn.dataset.entryId; | |
| const shareUrl = buildShareUrl(entryId); | |
| copyToClipboard(shareUrl, btn); | |
| }); | |
| }); | |
| // Use setTimeout(0) to ensure DOM is fully laid out before scrolling | |
| setTimeout(() => { | |
| const content = document.getElementById('content'); | |
| if (scrollMode === 'bottom') { | |
| content.scrollTop = content.scrollHeight; | |
| } else if (scrollMode === 'target') { | |
| // If scrollToEntryId is provided, scroll to that specific entry | |
| const scrollTargetId = scrollToEntryId || targetId; | |
| const targetEl = document.getElementById(`entry-${scrollTargetId}`); | |
| if (targetEl) { | |
| targetEl.scrollIntoView({ block: 'center' }); | |
| // Briefly highlight the target message | |
| if (scrollToEntryId) { | |
| targetEl.classList.add('highlight'); | |
| setTimeout(() => targetEl.classList.remove('highlight'), 2000); | |
| } | |
| } | |
| } | |
| }, 0); | |
| } | |
| // ============================================================ | |
| // INITIALIZATION | |
| // ============================================================ | |
| // Escape HTML tags in text (but not code blocks) | |
| function escapeHtmlTags(text) { | |
| return text.replace(/<(?=[a-zA-Z\/])/g, '<'); | |
| } | |
| // Configure marked with syntax highlighting and HTML escaping for text | |
| marked.use({ | |
| breaks: true, | |
| gfm: true, | |
| renderer: { | |
| // Code blocks: syntax highlight, no HTML escaping | |
| code(token) { | |
| const code = token.text; | |
| const lang = token.lang; | |
| let highlighted; | |
| if (lang && hljs.getLanguage(lang)) { | |
| try { | |
| highlighted = hljs.highlight(code, { language: lang }).value; | |
| } catch { | |
| highlighted = escapeHtml(code); | |
| } | |
| } else { | |
| // Auto-detect language if not specified | |
| try { | |
| highlighted = hljs.highlightAuto(code).value; | |
| } catch { | |
| highlighted = escapeHtml(code); | |
| } | |
| } | |
| return `<pre><code class="hljs">${highlighted}</code></pre>`; | |
| }, | |
| // Text content: escape HTML tags | |
| text(token) { | |
| return escapeHtmlTags(escapeHtml(token.text)); | |
| }, | |
| // Inline code: escape HTML | |
| codespan(token) { | |
| return `<code>${escapeHtml(token.text)}</code>`; | |
| } | |
| } | |
| }); | |
| // Simple marked parse (escaping handled in renderers) | |
| function safeMarkedParse(text) { | |
| return marked.parse(text); | |
| } | |
| // Search input | |
| const searchInput = document.getElementById('tree-search'); | |
| searchInput.addEventListener('input', (e) => { | |
| searchQuery = e.target.value; | |
| forceTreeRerender(); | |
| }); | |
| // Filter buttons | |
| document.querySelectorAll('.filter-btn').forEach(btn => { | |
| btn.addEventListener('click', () => { | |
| document.querySelectorAll('.filter-btn').forEach(b => b.classList.remove('active')); | |
| btn.classList.add('active'); | |
| filterMode = btn.dataset.filter; | |
| forceTreeRerender(); | |
| }); | |
| }); | |
| // Sidebar toggle | |
| const sidebar = document.getElementById('sidebar'); | |
| const overlay = document.getElementById('sidebar-overlay'); | |
| const hamburger = document.getElementById('hamburger'); | |
| hamburger.addEventListener('click', () => { | |
| sidebar.classList.add('open'); | |
| overlay.classList.add('open'); | |
| hamburger.style.display = 'none'; | |
| }); | |
| const closeSidebar = () => { | |
| sidebar.classList.remove('open'); | |
| overlay.classList.remove('open'); | |
| hamburger.style.display = ''; | |
| }; | |
| overlay.addEventListener('click', closeSidebar); | |
| document.getElementById('sidebar-close').addEventListener('click', closeSidebar); | |
| // Toggle states | |
| let thinkingExpanded = true; | |
| let toolOutputsExpanded = false; | |
| const toggleThinking = () => { | |
| thinkingExpanded = !thinkingExpanded; | |
| document.querySelectorAll('.thinking-text').forEach(el => { | |
| el.style.display = thinkingExpanded ? '' : 'none'; | |
| }); | |
| document.querySelectorAll('.thinking-collapsed').forEach(el => { | |
| el.style.display = thinkingExpanded ? 'none' : 'block'; | |
| }); | |
| }; | |
| const toggleToolOutputs = () => { | |
| toolOutputsExpanded = !toolOutputsExpanded; | |
| document.querySelectorAll('.tool-output.expandable').forEach(el => { | |
| el.classList.toggle('expanded', toolOutputsExpanded); | |
| }); | |
| document.querySelectorAll('.compaction').forEach(el => { | |
| el.classList.toggle('expanded', toolOutputsExpanded); | |
| }); | |
| }; | |
| // Keyboard shortcuts | |
| document.addEventListener('keydown', (e) => { | |
| if (e.key === 'Escape') { | |
| searchInput.value = ''; | |
| searchQuery = ''; | |
| navigateTo(leafId, 'bottom'); | |
| } | |
| if (e.ctrlKey && e.key === 't') { | |
| e.preventDefault(); | |
| toggleThinking(); | |
| } | |
| if (e.ctrlKey && e.key === 'o') { | |
| e.preventDefault(); | |
| toggleToolOutputs(); | |
| } | |
| }); | |
| // Initial render | |
| // If URL has targetId, scroll to that specific message; otherwise stay at top | |
| if (leafId) { | |
| if (urlTargetId && byId.has(urlTargetId)) { | |
| // Deep link: navigate to leaf and scroll to target message | |
| navigateTo(leafId, 'target', urlTargetId); | |
| } else { | |
| navigateTo(leafId, 'none'); | |
| } | |
| } else if (entries.length > 0) { | |
| // Fallback: use last entry if no leafId | |
| navigateTo(entries[entries.length - 1].id, 'none'); | |
| } | |
| })(); | |
| </script> | |
| </body> | |
| </html> |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment