@sbb-esta/lyne-elements-dev 5.0.0-next-dev.1776240735 → 5.0.0-next-dev.1776257807

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (37) hide show
  1. package/custom-elements.json +1738 -1535
  2. package/development/autocomplete/autocomplete-base-element.d.ts +6 -0
  3. package/development/autocomplete/autocomplete-base-element.d.ts.map +1 -1
  4. package/development/autocomplete/autocomplete-base-element.js +1 -1
  5. package/development/autocomplete/autocomplete.component.js +1 -1
  6. package/development/autocomplete-base-element-qUzJFqLU.js +678 -0
  7. package/development/autocomplete.js +1 -1
  8. package/development/autocomplete.pure.js +1 -1
  9. package/development/datepicker/common/datepicker-button.d.ts +6 -0
  10. package/development/datepicker/common/datepicker-button.d.ts.map +1 -1
  11. package/development/datepicker/common/datepicker-button.js +1 -1
  12. package/development/datepicker/datepicker/datepicker.component.d.ts +3 -0
  13. package/development/datepicker/datepicker/datepicker.component.d.ts.map +1 -1
  14. package/development/datepicker/datepicker/datepicker.component.js +1 -1
  15. package/development/datepicker-button-BI5R118p.js +166 -0
  16. package/development/datepicker.component-AKlc7SAO.js +214 -0
  17. package/development/datepicker.js +2 -2
  18. package/development/datepicker.pure.js +2 -2
  19. package/development/form-field/form-field-clear/form-field-clear.component.d.ts +3 -0
  20. package/development/form-field/form-field-clear/form-field-clear.component.d.ts.map +1 -1
  21. package/development/form-field/form-field-clear/form-field-clear.component.js +4 -1
  22. package/development/timetable-form/timetable-form-swap-button/timetable-form-swap-button.component.d.ts +3 -0
  23. package/development/timetable-form/timetable-form-swap-button/timetable-form-swap-button.component.d.ts.map +1 -1
  24. package/development/timetable-form/timetable-form-swap-button/timetable-form-swap-button.component.js +1 -1
  25. package/development/{timetable-form-swap-button.component-BGziVzB8.js → timetable-form-swap-button.component-_EUhzoww.js} +4 -1
  26. package/development/timetable-form.js +1 -1
  27. package/development/timetable-form.pure.js +1 -1
  28. package/development/toggle/toggle-option/toggle-option.component.d.ts +1 -0
  29. package/development/toggle/toggle-option/toggle-option.component.d.ts.map +1 -1
  30. package/development/toggle/toggle-option/toggle-option.component.js +1 -1
  31. package/development/{toggle-option.component-D8YWH_D5.js → toggle-option.component-DscH6K3b.js} +2 -1
  32. package/development/toggle.js +1 -1
  33. package/development/toggle.pure.js +1 -1
  34. package/package.json +2 -2
  35. package/development/autocomplete-base-element-B5n-eCMO.js +0 -672
  36. package/development/datepicker-button-CIvIHG8k.js +0 -160
  37. package/development/datepicker.component-BEiMYEc-.js +0 -211
@@ -1,672 +0,0 @@
1
- import { __esDecorate, __runInitializers } from "tslib";
2
- import { html, isServer, nothing, unsafeCSS } from "lit";
3
- import { property } from "lit/decorators.js";
4
- import { SbbEscapableOverlayController, SbbNegativeMixin, SbbOpenCloseBaseElement, SbbPropertyWatcherController, boxSizingStyles, forceType, idReference, isEventOnElement, isLean, isSafari, isZeroAnimationDuration, overlayGapFixCorners, removeAriaComboBoxAttributes, setOverlayPosition } from "./core.js";
5
- import { ResizeController } from "@lit-labs/observers/resize-controller.js";
6
- import { ref } from "lit/directives/ref.js";
7
- //#region src/elements/autocomplete/autocomplete-base-element.scss?inline
8
- var autocomplete_base_element_default = ".sbb-gap-fix-wrapper {\n position: relative;\n overflow: hidden;\n width: var(--sbb-options-panel-border-radius);\n height: var(--sbb-options-panel-border-radius);\n background-color: transparent;\n}\n\n.sbb-gap-fix-corner {\n position: absolute;\n border-radius: 50%;\n border: calc(var(--sbb-options-panel-border-radius) / 2) solid var(--sbb-options-panel-background-color);\n width: calc(var(--sbb-options-panel-border-radius) * 3);\n height: calc(var(--sbb-options-panel-border-radius) * 3);\n bottom: calc(var(--sbb-options-panel-border-radius) / 2 * -1);\n}\n\n.sbb-gap-fix-corner#left {\n left: calc(var(--sbb-options-panel-border-radius) / 2 * -1);\n}\n:host(:dir(rtl)) .sbb-gap-fix-corner#left {\n right: calc(var(--sbb-options-panel-border-radius) / 2 * -1);\n left: unset;\n}\n\n.sbb-gap-fix-corner#right {\n right: calc(var(--sbb-options-panel-border-radius) / 2 * -1);\n}\n:host(:dir(rtl)) .sbb-gap-fix-corner#right {\n left: calc(var(--sbb-options-panel-border-radius) / 2 * -1);\n right: unset;\n}\n\n:host {\n --sbb-scrollbar-thumb-width: 0.125rem;\n --sbb-scrollbar-thumb-width-hover: 0.25rem;\n --sbb-scrollbar-width-firefox: thin;\n --sbb-scrollbar-color: color-mix(in srgb, var(--sbb-color-black) 30%, transparent);\n --sbb-scrollbar-color: light-dark(\n color-mix(in srgb, var(--sbb-color-black) 30%, transparent),\n color-mix(in srgb, var(--sbb-color-white) 30%, transparent)\n );\n --sbb-scrollbar-color-hover: color-mix(in srgb, var(--sbb-color-black) 60%, transparent);\n --sbb-scrollbar-color-hover: light-dark(\n color-mix(in srgb, var(--sbb-color-black) 60%, transparent),\n color-mix(in srgb, var(--sbb-color-white) 60%, transparent)\n );\n --sbb-scrollbar-track-color: transparent;\n --sbb-options-panel-position-x: 0;\n --sbb-options-panel-position-y: 0;\n --_sbb-options-panel-max-height: min(\n var(--sbb-options-panel-max-height-calculated),\n var(--sbb-options-panel-max-height, 100000em)\n );\n --sbb-options-panel-min-height: var(--sbb-options-panel-origin-height);\n --sbb-options-panel-options-border-radius: 0 0 var(--sbb-options-panel-border-radius)\n var(--sbb-options-panel-border-radius);\n --sbb-options-panel-animation-duration: var(\n --sbb-disable-animation-duration,\n var(--sbb-animation-duration-4x)\n );\n --sbb-options-panel-gap-fix-top: calc(\n var(--sbb-options-panel-position-y) - var(--sbb-options-panel-border-radius)\n );\n --sbb-focus-outline-color: var(--sbb-focus-outline-color-default);\n --sbb-focus-outline-color: light-dark(\n var(--sbb-focus-outline-color-default),\n var(--sbb-focus-outline-color-dark)\n );\n --sbb-options-pointer-events: all;\n --sbb-options-panel-internal-z-index: var(\n --sbb-autocomplete-z-index,\n var(--sbb-overlay-default-z-index)\n );\n display: none;\n font-weight: normal;\n}\n\n:host([size=s]) {\n --sbb-options-panel-padding-block: var(--sbb-spacing-fixed-1x);\n}\n\n:host([size=m]) {\n --sbb-options-panel-padding-block: var(--sbb-spacing-fixed-2x);\n}\n\n:host(:is(:is(:state(state-opening),[state--state-opening]), :is(:state(state-opened),[state--state-opened]), :is(:state(state-closing),[state--state-closing]))) {\n display: block;\n}\n\n:host([negative]) {\n --sbb-scrollbar-color: color-mix(in srgb, var(--sbb-color-white) 30%, transparent);\n --sbb-scrollbar-color-hover: color-mix(in srgb, var(--sbb-color-white) 60%, transparent);\n --sbb-options-panel-background-color: var(--sbb-background-color-1-negative);\n --sbb-focus-outline-color: var(--sbb-focus-outline-color-dark);\n}\n\n:host(:not(:is(:state(state-opening),[state--state-opening]), :is(:state(state-opened),[state--state-opened]), :is(:state(state-closing),[state--state-closing]))),\n:host(:is(:state(state-closed),[state--state-closed])) {\n --sbb-options-panel-visibility: hidden;\n --sbb-options-pointer-events: none;\n}\n\n:host(:is(:state(state-opening),[state--state-opening])) {\n --sbb-options-panel-animation-name: open;\n}\n\n:host(:is(:state(state-closing),[state--state-closing])) {\n --sbb-options-panel-animation-name: close;\n}\n\n:host(:is(:state(state-opened),[state--state-opened])),\n:host(:is(:state(state-opening),[state--state-opening])) {\n --sbb-options-panel-gap-fix-opacity: 1;\n}\n\n:host(:is(:state(options-panel-position-below),[state--options-panel-position-below])) {\n --sbb-options-panel-animation-transform: translateY(\n calc((var(--sbb-options-panel-origin-height) / 2) * -1)\n );\n}\n\n:host(:is(:state(options-panel-position-above),[state--options-panel-position-above])) {\n --sbb-options-panel-options-border-radius: var(--sbb-options-panel-border-radius)\n var(--sbb-options-panel-border-radius) 0 0;\n --sbb-options-panel-gap-fix-top: var(--_sbb-options-panel-max-height);\n --sbb-options-panel-gap-fix-transform: rotate(180deg);\n --sbb-options-panel-animation-transform: translateY(\n calc(var(--sbb-options-panel-origin-height) / 2)\n );\n}\n\n:host([preserve-icon-space]) {\n --sbb-option-icon-container-display: block;\n}\n\n::slotted(sbb-divider) {\n margin-block: var(--sbb-options-panel-divider-margin-block);\n}\n\n.sbb-autocomplete__container {\n position: fixed;\n pointer-events: none;\n inset: 0;\n z-index: var(--sbb-options-panel-internal-z-index);\n}\n\n.sbb-autocomplete__gap-fix {\n display: flex;\n justify-content: space-between;\n position: fixed;\n visibility: var(--sbb-options-panel-visibility, visible);\n opacity: var(--sbb-options-panel-gap-fix-opacity, 0);\n background-color: transparent;\n width: var(--sbb-options-panel-width, fit-content);\n height: var(--sbb-options-panel-border-radius);\n top: var(--sbb-options-panel-gap-fix-top);\n left: var(--sbb-options-panel-position-x);\n transform: var(--sbb-options-panel-gap-fix-transform, none);\n transition: opacity var(--sbb-options-panel-animation-duration) var(--sbb-options-panel-animation-timing-function);\n}\n\n.sbb-autocomplete__panel {\n display: block;\n position: absolute;\n visibility: var(--sbb-options-panel-visibility, \"visible\");\n overflow: hidden;\n background-color: transparent;\n border: none;\n border-radius: var(--sbb-options-panel-border-radius);\n top: var(--sbb-options-panel-position-y);\n left: var(--sbb-options-panel-position-x);\n width: var(--sbb-options-panel-width, fit-content);\n transition: box-shadow var(--sbb-options-panel-animation-duration) var(--sbb-options-panel-animation-timing-function);\n}\n.sbb-autocomplete__panel::before, .sbb-autocomplete__panel::after {\n content: \"\";\n display: none;\n position: relative;\n width: 100%;\n height: var(--sbb-options-panel-origin-height);\n background-color: transparent;\n border-radius: var(--sbb-options-panel-border-radius);\n pointer-events: none;\n transition: box-shadow var(--sbb-options-panel-animation-duration) var(--sbb-options-panel-animation-timing-function);\n z-index: 1;\n}\n:host(:is(:state(options-panel-position-below),[state--options-panel-position-below])) .sbb-autocomplete__panel {\n inset-block-start: calc(var(--sbb-options-panel-position-y) - var(--sbb-options-panel-origin-height));\n}\n:host(:is(:is(:state(state-opened),[state--state-opened]), :is(:state(state-opening),[state--state-opening]))) .sbb-autocomplete__panel {\n box-shadow: var(--sbb-box-shadow-level-5-hard);\n}\n:host(:is(:is(:state(state-opened),[state--state-opened]), :is(:state(state-opening),[state--state-opening]))[negative]) .sbb-autocomplete__panel {\n box-shadow: var(--sbb-box-shadow-level-5-hard-negative);\n}\n:host(:is(:state(options-panel-position-below),[state--options-panel-position-below])) .sbb-autocomplete__panel::before {\n display: block;\n}\n:host(:is(:state(options-panel-position-above),[state--options-panel-position-above])) .sbb-autocomplete__panel::after {\n display: block;\n}\n:host(:is(:is(:state(state-opened),[state--state-opened]), :is(:state(state-opening),[state--state-opening])):is(:state(option-panel-origin-borderless),[state--option-panel-origin-borderless])) .sbb-autocomplete__panel::before, :host(:is(:is(:state(state-opened),[state--state-opened]), :is(:state(state-opening),[state--state-opening])):is(:state(option-panel-origin-borderless),[state--option-panel-origin-borderless])) .sbb-autocomplete__panel::after {\n box-shadow: var(--sbb-box-shadow-level-5-hard);\n}\n:host(:is(:is(:state(state-opened),[state--state-opened]), :is(:state(state-opening),[state--state-opening])):is(:state(option-panel-origin-borderless),[state--option-panel-origin-borderless])[negative]) .sbb-autocomplete__panel::before, :host(:is(:is(:state(state-opened),[state--state-opened]), :is(:state(state-opening),[state--state-opening])):is(:state(option-panel-origin-borderless),[state--option-panel-origin-borderless])[negative]) .sbb-autocomplete__panel::after {\n box-shadow: var(--sbb-box-shadow-level-5-hard-negative);\n}\n\n.sbb-autocomplete__wrapper {\n overflow: hidden;\n}\n\n.sbb-autocomplete__options {\n background-color: var(--sbb-options-panel-background-color);\n padding-block: var(--sbb-options-panel-padding-block);\n padding-inline: 0;\n border-radius: var(--sbb-options-panel-options-border-radius);\n max-height: var(--_sbb-options-panel-max-height);\n min-height: var(--sbb-options-panel-min-height);\n pointer-events: all;\n overflow-y: auto;\n animation-name: var(--sbb-options-panel-animation-name);\n animation-duration: var(--sbb-options-panel-animation-duration);\n animation-timing-function: var(--sbb-options-panel-animation-timing-function);\n --sbb-scrollbar-width: var(--sbb-spacing-fixed-3x);\n}\n.sbb-autocomplete__options::-webkit-scrollbar {\n width: var(--sbb-scrollbar-width);\n height: var(--sbb-scrollbar-width);\n background-color: var(--sbb-scrollbar-track-color, transparent);\n}\n.sbb-autocomplete__options::-webkit-scrollbar-corner {\n background-color: var(--sbb-scrollbar-track-color, transparent);\n}\n.sbb-autocomplete__options::-webkit-scrollbar-thumb {\n background-color: var(--sbb-scrollbar-color, currentcolor);\n border: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width))) solid transparent;\n border-radius: var(--sbb-border-radius-4x);\n background-clip: padding-box;\n}\n.sbb-autocomplete__options::-webkit-scrollbar-thumb:hover {\n background-color: var(--sbb-scrollbar-color-hover, currentcolor);\n border-width: calc(0.5 * (var(--sbb-scrollbar-width) - var(--sbb-scrollbar-thumb-width-hover)));\n}\n.sbb-autocomplete__options::-webkit-scrollbar-button, .sbb-autocomplete__options::-webkit-scrollbar-corner {\n display: none;\n}\n@supports not selector(::-webkit-scrollbar) {\n .sbb-autocomplete__options {\n scrollbar-width: var(--sbb-scrollbar-width-firefox);\n scrollbar-color: var(--sbb-scrollbar-color, currentcolor) var(--sbb-scrollbar-track-color, transparent);\n }\n}\n.sbb-autocomplete__options {\n pointer-events: var(--sbb-options-pointer-events);\n}\n@media (forced-colors: active) {\n .sbb-autocomplete__options {\n border: var(--sbb-border-width-1x) solid CanvasText;\n border-top: none;\n }\n}\n\n@keyframes open {\n from {\n transform: var(--sbb-options-panel-animation-transform);\n opacity: 0;\n }\n to {\n transform: translateY(0);\n opacity: 1;\n }\n}\n@keyframes close {\n from {\n transform: translateY(0);\n opacity: 1;\n }\n to {\n transform: var(--sbb-options-panel-animation-transform);\n opacity: 0;\n }\n}";
9
- //#endregion
10
- //#region src/elements/autocomplete/autocomplete-base-element.ts
11
- /**
12
- * On Safari, the aria role 'listbox' must be on the host element, or else VoiceOver won't work at all.
13
- * On the other hand, JAWS and NVDA need the role to be "closer" to the options, or else optgroups won't work.
14
- */
15
- var ariaRoleOnHost = isSafari;
16
- var SbbAutocompleteBaseElement = (() => {
17
- let _classSuper = SbbNegativeMixin(SbbOpenCloseBaseElement);
18
- let _origin_decorators;
19
- let _origin_initializers = [];
20
- let _origin_extraInitializers = [];
21
- let _trigger_decorators;
22
- let _trigger_initializers = [];
23
- let _trigger_extraInitializers = [];
24
- let _preserveIconSpace_decorators;
25
- let _preserveIconSpace_initializers = [];
26
- let _preserveIconSpace_extraInitializers = [];
27
- let _autoActiveFirstOption_decorators;
28
- let _autoActiveFirstOption_initializers = [];
29
- let _autoActiveFirstOption_extraInitializers = [];
30
- let _displayWith_decorators;
31
- let _displayWith_initializers = [];
32
- let _displayWith_extraInitializers = [];
33
- let _size_decorators;
34
- let _size_initializers = [];
35
- let _size_extraInitializers = [];
36
- let _autoSelectActiveOption_decorators;
37
- let _autoSelectActiveOption_initializers = [];
38
- let _autoSelectActiveOption_extraInitializers = [];
39
- let _autoSelectActiveOptionOnBlur_decorators;
40
- let _autoSelectActiveOptionOnBlur_initializers = [];
41
- let _autoSelectActiveOptionOnBlur_extraInitializers = [];
42
- let _requireSelection_decorators;
43
- let _requireSelection_initializers = [];
44
- let _requireSelection_extraInitializers = [];
45
- let _position_decorators;
46
- let _position_initializers = [];
47
- let _position_extraInitializers = [];
48
- return class SbbAutocompleteBaseElement extends _classSuper {
49
- static {
50
- const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(_classSuper[Symbol.metadata] ?? null) : void 0;
51
- _origin_decorators = [idReference(), property()];
52
- _trigger_decorators = [idReference(), property()];
53
- _preserveIconSpace_decorators = [forceType(), property({
54
- attribute: "preserve-icon-space",
55
- reflect: true,
56
- type: Boolean
57
- })];
58
- _autoActiveFirstOption_decorators = [forceType(), property({
59
- attribute: "auto-active-first-option",
60
- type: Boolean
61
- })];
62
- _displayWith_decorators = [property({ attribute: false })];
63
- _size_decorators = [property({ reflect: true })];
64
- _autoSelectActiveOption_decorators = [forceType(), property({
65
- attribute: "auto-select-active-option",
66
- type: Boolean
67
- })];
68
- _autoSelectActiveOptionOnBlur_decorators = [forceType(), property({
69
- attribute: "auto-select-active-option-on-blur",
70
- type: Boolean
71
- })];
72
- _requireSelection_decorators = [forceType(), property({
73
- attribute: "require-selection",
74
- type: Boolean
75
- })];
76
- _position_decorators = [property()];
77
- __esDecorate(this, null, _origin_decorators, {
78
- kind: "accessor",
79
- name: "origin",
80
- static: false,
81
- private: false,
82
- access: {
83
- has: (obj) => "origin" in obj,
84
- get: (obj) => obj.origin,
85
- set: (obj, value) => {
86
- obj.origin = value;
87
- }
88
- },
89
- metadata: _metadata
90
- }, _origin_initializers, _origin_extraInitializers);
91
- __esDecorate(this, null, _trigger_decorators, {
92
- kind: "accessor",
93
- name: "trigger",
94
- static: false,
95
- private: false,
96
- access: {
97
- has: (obj) => "trigger" in obj,
98
- get: (obj) => obj.trigger,
99
- set: (obj, value) => {
100
- obj.trigger = value;
101
- }
102
- },
103
- metadata: _metadata
104
- }, _trigger_initializers, _trigger_extraInitializers);
105
- __esDecorate(this, null, _preserveIconSpace_decorators, {
106
- kind: "accessor",
107
- name: "preserveIconSpace",
108
- static: false,
109
- private: false,
110
- access: {
111
- has: (obj) => "preserveIconSpace" in obj,
112
- get: (obj) => obj.preserveIconSpace,
113
- set: (obj, value) => {
114
- obj.preserveIconSpace = value;
115
- }
116
- },
117
- metadata: _metadata
118
- }, _preserveIconSpace_initializers, _preserveIconSpace_extraInitializers);
119
- __esDecorate(this, null, _autoActiveFirstOption_decorators, {
120
- kind: "accessor",
121
- name: "autoActiveFirstOption",
122
- static: false,
123
- private: false,
124
- access: {
125
- has: (obj) => "autoActiveFirstOption" in obj,
126
- get: (obj) => obj.autoActiveFirstOption,
127
- set: (obj, value) => {
128
- obj.autoActiveFirstOption = value;
129
- }
130
- },
131
- metadata: _metadata
132
- }, _autoActiveFirstOption_initializers, _autoActiveFirstOption_extraInitializers);
133
- __esDecorate(this, null, _displayWith_decorators, {
134
- kind: "accessor",
135
- name: "displayWith",
136
- static: false,
137
- private: false,
138
- access: {
139
- has: (obj) => "displayWith" in obj,
140
- get: (obj) => obj.displayWith,
141
- set: (obj, value) => {
142
- obj.displayWith = value;
143
- }
144
- },
145
- metadata: _metadata
146
- }, _displayWith_initializers, _displayWith_extraInitializers);
147
- __esDecorate(this, null, _size_decorators, {
148
- kind: "accessor",
149
- name: "size",
150
- static: false,
151
- private: false,
152
- access: {
153
- has: (obj) => "size" in obj,
154
- get: (obj) => obj.size,
155
- set: (obj, value) => {
156
- obj.size = value;
157
- }
158
- },
159
- metadata: _metadata
160
- }, _size_initializers, _size_extraInitializers);
161
- __esDecorate(this, null, _autoSelectActiveOption_decorators, {
162
- kind: "accessor",
163
- name: "autoSelectActiveOption",
164
- static: false,
165
- private: false,
166
- access: {
167
- has: (obj) => "autoSelectActiveOption" in obj,
168
- get: (obj) => obj.autoSelectActiveOption,
169
- set: (obj, value) => {
170
- obj.autoSelectActiveOption = value;
171
- }
172
- },
173
- metadata: _metadata
174
- }, _autoSelectActiveOption_initializers, _autoSelectActiveOption_extraInitializers);
175
- __esDecorate(this, null, _autoSelectActiveOptionOnBlur_decorators, {
176
- kind: "accessor",
177
- name: "autoSelectActiveOptionOnBlur",
178
- static: false,
179
- private: false,
180
- access: {
181
- has: (obj) => "autoSelectActiveOptionOnBlur" in obj,
182
- get: (obj) => obj.autoSelectActiveOptionOnBlur,
183
- set: (obj, value) => {
184
- obj.autoSelectActiveOptionOnBlur = value;
185
- }
186
- },
187
- metadata: _metadata
188
- }, _autoSelectActiveOptionOnBlur_initializers, _autoSelectActiveOptionOnBlur_extraInitializers);
189
- __esDecorate(this, null, _requireSelection_decorators, {
190
- kind: "accessor",
191
- name: "requireSelection",
192
- static: false,
193
- private: false,
194
- access: {
195
- has: (obj) => "requireSelection" in obj,
196
- get: (obj) => obj.requireSelection,
197
- set: (obj, value) => {
198
- obj.requireSelection = value;
199
- }
200
- },
201
- metadata: _metadata
202
- }, _requireSelection_initializers, _requireSelection_extraInitializers);
203
- __esDecorate(this, null, _position_decorators, {
204
- kind: "accessor",
205
- name: "position",
206
- static: false,
207
- private: false,
208
- access: {
209
- has: (obj) => "position" in obj,
210
- get: (obj) => obj.position,
211
- set: (obj, value) => {
212
- obj.position = value;
213
- }
214
- },
215
- metadata: _metadata
216
- }, _position_initializers, _position_extraInitializers);
217
- if (_metadata) Object.defineProperty(this, Symbol.metadata, {
218
- enumerable: true,
219
- configurable: true,
220
- writable: true,
221
- value: _metadata
222
- });
223
- }
224
- static {
225
- this.styles = [boxSizingStyles, unsafeCSS(autocomplete_base_element_default)];
226
- }
227
- #origin_accessor_storage;
228
- /**
229
- * The element where the autocomplete will attach.
230
- * If not set, as fallback there are two elements which can act as origin with following priority order:
231
- * 1. `sbb-form-field` if it is an ancestor.
232
- * 2. trigger element if set.
233
- *
234
- * For attribute usage, provide an id reference.
235
- */
236
- get origin() {
237
- return this.#origin_accessor_storage;
238
- }
239
- set origin(value) {
240
- this.#origin_accessor_storage = value;
241
- }
242
- #trigger_accessor_storage;
243
- /**
244
- * The input element that will trigger the autocomplete opening.
245
- * By default, the autocomplete will open on focus, click, input or `ArrowDown` keypress of the 'trigger' element.
246
- * If not set, will search for the first 'input' child of a 'sbb-form-field' ancestor.
247
- *
248
- * For attribute usage, provide an id reference.
249
- */
250
- get trigger() {
251
- return this.#trigger_accessor_storage;
252
- }
253
- set trigger(value) {
254
- this.#trigger_accessor_storage = value;
255
- }
256
- #preserveIconSpace_accessor_storage;
257
- /** Whether the icon space is preserved when no icon is set. */
258
- get preserveIconSpace() {
259
- return this.#preserveIconSpace_accessor_storage;
260
- }
261
- set preserveIconSpace(value) {
262
- this.#preserveIconSpace_accessor_storage = value;
263
- }
264
- #autoActiveFirstOption_accessor_storage;
265
- /** Whether the first option is automatically activated when the autocomplete is opened. */
266
- get autoActiveFirstOption() {
267
- return this.#autoActiveFirstOption_accessor_storage;
268
- }
269
- set autoActiveFirstOption(value) {
270
- this.#autoActiveFirstOption_accessor_storage = value;
271
- }
272
- #displayWith_accessor_storage;
273
- /** Function that maps an option's control value to its display value in the trigger. */
274
- get displayWith() {
275
- return this.#displayWith_accessor_storage;
276
- }
277
- set displayWith(value) {
278
- this.#displayWith_accessor_storage = value;
279
- }
280
- #size_accessor_storage;
281
- /**
282
- * Size variant, either m or s.
283
- * @default 'm' / 's' (lean)
284
- */
285
- get size() {
286
- return this.#size_accessor_storage;
287
- }
288
- set size(value) {
289
- this.#size_accessor_storage = value;
290
- }
291
- #autoSelectActiveOption_accessor_storage;
292
- /** Whether the active option should be selected as the user is navigating. */
293
- get autoSelectActiveOption() {
294
- return this.#autoSelectActiveOption_accessor_storage;
295
- }
296
- set autoSelectActiveOption(value) {
297
- this.#autoSelectActiveOption_accessor_storage = value;
298
- }
299
- #autoSelectActiveOptionOnBlur_accessor_storage;
300
- /**
301
- * When enabled, the active option is automatically selected on blur.
302
- * This is an experimental feature. It might be subject to changes.
303
- */
304
- get autoSelectActiveOptionOnBlur() {
305
- return this.#autoSelectActiveOptionOnBlur_accessor_storage;
306
- }
307
- set autoSelectActiveOptionOnBlur(value) {
308
- this.#autoSelectActiveOptionOnBlur_accessor_storage = value;
309
- }
310
- #requireSelection_accessor_storage;
311
- /**
312
- * Whether the user is required to make a selection when they're interacting with the
313
- * autocomplete. If the user moves away from the autocomplete without selecting an option from
314
- * the list, the value will be reset. If the user opens the panel and closes it without
315
- * interacting or selecting a value, the initial value will be kept.
316
- */
317
- get requireSelection() {
318
- return this.#requireSelection_accessor_storage;
319
- }
320
- set requireSelection(value) {
321
- this.#requireSelection_accessor_storage = value;
322
- }
323
- #position_accessor_storage;
324
- /**
325
- * The position of the autocomplete panel relative to the trigger.
326
- * @default 'auto'
327
- */
328
- get position() {
329
- return this.#position_accessor_storage;
330
- }
331
- set position(value) {
332
- this.#position_accessor_storage = value;
333
- }
334
- /** Returns the element where the autocomplete overlay is attached to. */
335
- get originElement() {
336
- return this.origin ?? this.closest?.("sbb-form-field")?.shadowRoot?.querySelector?.("#overlay-anchor") ?? this.trigger ?? null;
337
- }
338
- /** Returns the trigger element. */
339
- get triggerElement() {
340
- return this._triggerElement ?? null;
341
- }
342
- constructor() {
343
- super();
344
- this.#origin_accessor_storage = __runInitializers(this, _origin_initializers, null);
345
- this.#trigger_accessor_storage = (__runInitializers(this, _origin_extraInitializers), __runInitializers(this, _trigger_initializers, null));
346
- this.#preserveIconSpace_accessor_storage = (__runInitializers(this, _trigger_extraInitializers), __runInitializers(this, _preserveIconSpace_initializers, false));
347
- this.#autoActiveFirstOption_accessor_storage = (__runInitializers(this, _preserveIconSpace_extraInitializers), __runInitializers(this, _autoActiveFirstOption_initializers, false));
348
- this.#displayWith_accessor_storage = (__runInitializers(this, _autoActiveFirstOption_extraInitializers), __runInitializers(this, _displayWith_initializers, null));
349
- this.#size_accessor_storage = (__runInitializers(this, _displayWith_extraInitializers), __runInitializers(this, _size_initializers, isLean() ? "s" : "m"));
350
- this.#autoSelectActiveOption_accessor_storage = (__runInitializers(this, _size_extraInitializers), __runInitializers(this, _autoSelectActiveOption_initializers, false));
351
- this.#autoSelectActiveOptionOnBlur_accessor_storage = (__runInitializers(this, _autoSelectActiveOption_extraInitializers), __runInitializers(this, _autoSelectActiveOptionOnBlur_initializers, false));
352
- this.#requireSelection_accessor_storage = (__runInitializers(this, _autoSelectActiveOptionOnBlur_extraInitializers), __runInitializers(this, _requireSelection_initializers, false));
353
- this.#position_accessor_storage = (__runInitializers(this, _requireSelection_extraInitializers), __runInitializers(this, _position_initializers, "auto"));
354
- this._triggerElement = __runInitializers(this, _position_extraInitializers);
355
- this.activeOption = null;
356
- this.pendingAutoSelectedOption = null;
357
- this._originResizeObserver = new ResizeController(this, {
358
- target: null,
359
- skipInitial: true,
360
- callback: () => {
361
- if (this.state === "opened") this._setOverlayPosition();
362
- }
363
- });
364
- /** Listens to the changes on the `disabled` or `readonly` attribute of the trigger. */
365
- this._triggerAttributeObserver = !isServer ? new MutationObserver((mutations) => {
366
- const input = mutations[0].target;
367
- if (input.hasAttribute("disabled") || input.hasAttribute("readonly")) this.close();
368
- }) : null;
369
- this._isPointerDownEventOnMenu = false;
370
- this._escapableOverlayController = new SbbEscapableOverlayController(this);
371
- this._optionsCount = 0;
372
- /** Tracks input from keyboard. */
373
- this._lastUserInput = null;
374
- /** If true, the 'change' event source is an option selection or a 'requireSelection' field cleanup */
375
- this._isCustomChangeEvent = false;
376
- this._pointerDownListener = (event) => {
377
- this._isPointerDownEventOnMenu = isEventOnElement(this._overlay, event);
378
- };
379
- this._closeOnBackdropClick = (event) => {
380
- if (!this._isPointerDownEventOnMenu && !isEventOnElement(this._overlay, event) && !isEventOnElement(this.originElement, event)) this.close();
381
- };
382
- this.addEventListener?.("optionselected", (e) => this.onOptionSelected(e));
383
- this.addEventListener?.("ɵoptgroupslotchange", () => this._handleSlotchange(), { capture: true });
384
- this.addController(new SbbPropertyWatcherController(this, () => this.closest("sbb-form-field"), {
385
- negative: (e) => {
386
- this.negative = e.negative;
387
- this.syncNegative();
388
- },
389
- borderless: (e) => this.toggleState("option-panel-origin-borderless", e.borderless)
390
- }));
391
- }
392
- /** Opens the autocomplete. */
393
- open() {
394
- if (this.state === "opening" || this.state === "opened" || !this._overlay || this.options.length === 0 || this._readonly() || !this.dispatchBeforeOpenEvent()) return;
395
- this.showPopover?.();
396
- this.state = "opening";
397
- this.triggerElement?.toggleAttribute("data-expanded", true);
398
- const originElement = this.originElement;
399
- if (!originElement) throw new Error("Cannot find the origin element. Please specify a valid element or check the usage of the \"origin\" property from the documentation");
400
- this._setOverlayPosition(originElement);
401
- this._setNextActiveOptionIfAutoActiveFirstOption();
402
- this._attachOpenPanelEvents();
403
- this._escapableOverlayController.connect();
404
- if (this._isZeroAnimationDuration()) this._handleOpening();
405
- }
406
- /** Closes the autocomplete. */
407
- close() {
408
- if (this.state === "closing" || this.state === "closed" || !this.dispatchBeforeCloseEvent()) return;
409
- if (this.pendingAutoSelectedOption) {
410
- this.pendingAutoSelectedOption.selected = true;
411
- this._setValueAndDispatchEvents(this.pendingAutoSelectedOption, true);
412
- }
413
- this.state = "closing";
414
- this.triggerElement?.removeAttribute("data-expanded");
415
- this._openPanelEventsController.abort();
416
- if (this.originElement) this._originResizeObserver.unobserve(this.originElement);
417
- if (this._isZeroAnimationDuration()) this._handleClosing();
418
- }
419
- _isZeroAnimationDuration() {
420
- return isZeroAnimationDuration(this, "--sbb-options-panel-animation-duration");
421
- }
422
- connectedCallback() {
423
- this.popover = "manual";
424
- super.connectedCallback();
425
- if (ariaRoleOnHost) this.id ||= this.overlayId;
426
- if (this.hasUpdated) this._componentSetup();
427
- this.syncNegative();
428
- }
429
- willUpdate(changedProperties) {
430
- super.willUpdate(changedProperties);
431
- if (changedProperties.has("negative")) this.syncNegative();
432
- if (changedProperties.has("autoActiveFirstOption") && this.isOpen) this._setNextActiveOptionIfAutoActiveFirstOption();
433
- }
434
- firstUpdated(changedProperties) {
435
- super.firstUpdated(changedProperties);
436
- this._componentSetup();
437
- }
438
- disconnectedCallback() {
439
- super.disconnectedCallback();
440
- this._triggerElement = null;
441
- this._triggerAbortController?.abort();
442
- this._openPanelEventsController?.abort();
443
- }
444
- requestUpdate(name, oldValue, options) {
445
- super.requestUpdate(name, oldValue, options);
446
- if (isServer || !this.hasUpdated) return;
447
- if (!name || name === "trigger") this._configureTrigger();
448
- else if ((!name || name === "origin") && this.isOpen) this._setOverlayPosition();
449
- }
450
- /** When an option is selected, update the input value and close the autocomplete. */
451
- onOptionSelected(event) {
452
- const target = event.target;
453
- this._setValueAndDispatchEvents(target);
454
- this.close();
455
- }
456
- /** @deprecated */
457
- onOptionArrowsSelected(_activeOption) {}
458
- /**
459
- * A 'pending selection' sets the option value in the input element without emitting events.
460
- * A 'pending selection' is confirmed when the panel closes. Any other user interaction
461
- * will reset the pending value.
462
- */
463
- setPendingSelection(activeOption) {
464
- this.pendingAutoSelectedOption = activeOption;
465
- this._setInputValue(activeOption);
466
- }
467
- _setValueAndDispatchEvents(selectedOption, preventFocus = false) {
468
- this.options.filter((option) => option.id !== selectedOption.id && option.selected).forEach((option) => option.selected = false);
469
- this.pendingAutoSelectedOption = null;
470
- if (this.triggerElement) {
471
- this._setInputValue(selectedOption);
472
- this._isCustomChangeEvent = true;
473
- this.triggerElement.dispatchEvent(new Event("change", { bubbles: true }));
474
- this.triggerElement.dispatchEvent(new InputEvent("input", {
475
- bubbles: true,
476
- composed: true
477
- }));
478
- this._lastUserInput = null;
479
- this.triggerElement.dispatchEvent(new CustomEvent("inputAutocomplete", { detail: { option: selectedOption } }));
480
- if (!preventFocus) this.triggerElement.focus();
481
- }
482
- }
483
- /** Set the option value within the input element */
484
- _setInputValue(option) {
485
- if (!this.triggerElement) return;
486
- const toDisplay = this.displayWith?.(option.value) ?? option.value;
487
- Object.getOwnPropertyDescriptor(HTMLInputElement.prototype, "value").set.call(this.triggerElement, toDisplay);
488
- }
489
- _handleSlotchange() {
490
- this._highlightOptions(this.triggerElement?.value);
491
- /**
492
- * It's possible to filter out options with an opened panel on input change.
493
- * In this case, the panel's position must be recalculated considering the new option's list.
494
- */
495
- if (this.isOpen) {
496
- this._setOverlayPosition();
497
- this._setNextActiveOptionIfAutoActiveFirstOption();
498
- if (this._optionsCount > 0 && this.options.length === 0) this.close();
499
- } else if (document?.activeElement === this.triggerElement && this._optionsCount === 0 && this.options.length > 0) this.open();
500
- this._optionsCount = this.options.length;
501
- }
502
- _setNextActiveOptionIfAutoActiveFirstOption() {
503
- if (this.autoActiveFirstOption) {
504
- this.resetActiveElement();
505
- this.setNextActiveOption();
506
- }
507
- }
508
- /** The autocomplete should inherit 'readonly' state from the trigger. */
509
- _readonly() {
510
- return this.triggerElement?.hasAttribute("readonly") ?? false;
511
- }
512
- _componentSetup() {
513
- if (isServer) return;
514
- this._configureTrigger();
515
- }
516
- _configureTrigger() {
517
- const triggerElement = this.trigger ?? this.closest?.("sbb-form-field")?.querySelector("input");
518
- if (triggerElement === this.triggerElement) return;
519
- this._triggerAbortController?.abort();
520
- removeAriaComboBoxAttributes(this.triggerElement);
521
- this.triggerElement?.removeAttribute("data-expanded");
522
- this._triggerElement = triggerElement;
523
- if (!this.triggerElement) return;
524
- const originElement = this.originElement;
525
- if (this.triggerElement === originElement && this.isOpen) this._setOverlayPosition(originElement);
526
- this._triggerAttributeObserver?.observe(this.triggerElement, {
527
- attributes: true,
528
- attributeFilter: ["disabled", "readonly"]
529
- });
530
- this.setTriggerAttributes(this.triggerElement);
531
- this._triggerAbortController = new AbortController();
532
- this.triggerElement.addEventListener("focus", () => this.open(), { signal: this._triggerAbortController.signal });
533
- this.triggerElement.addEventListener("click", () => this.open(), { signal: this._triggerAbortController.signal });
534
- this.triggerElement.addEventListener("input", (event) => {
535
- const value = event.target.value;
536
- if (value) this.open();
537
- this._highlightOptions(value);
538
- this._lastUserInput = value;
539
- this.pendingAutoSelectedOption = null;
540
- }, { signal: this._triggerAbortController.signal });
541
- this.triggerElement.addEventListener("change", (event) => {
542
- /**
543
- * In 'requireSelection' mode, we block the native change events and
544
- * let only pass the ones that come with a valid value (when an option is selected)
545
- */
546
- if (this.requireSelection && !this._isCustomChangeEvent) event.stopImmediatePropagation();
547
- this._isCustomChangeEvent = false;
548
- }, {
549
- signal: this._triggerAbortController.signal,
550
- capture: true
551
- });
552
- this.triggerElement.addEventListener("keydown", (event) => this._closedPanelKeyboardInteraction(event), {
553
- signal: this._triggerAbortController.signal,
554
- capture: true
555
- });
556
- this.triggerElement.addEventListener("blur", (e) => {
557
- if (this.contains(e.relatedTarget)) return;
558
- if (this.autoSelectActiveOptionOnBlur && this.activeOption && this._lastUserInput && this.triggerElement?.value) {
559
- this.activeOption.selected = true;
560
- this._setValueAndDispatchEvents(this.activeOption, true);
561
- }
562
- if (this.requireSelection && this.triggerElement && this._lastUserInput != null) {
563
- Object.getOwnPropertyDescriptor(HTMLInputElement.prototype, "value").set.call(this.triggerElement, "");
564
- this._highlightOptions("");
565
- this._isCustomChangeEvent = true;
566
- this.triggerElement.dispatchEvent(new Event("change", { bubbles: true }));
567
- this.triggerElement.dispatchEvent(new InputEvent("input", {
568
- bubbles: true,
569
- composed: true
570
- }));
571
- }
572
- this.close();
573
- }, {
574
- signal: this._triggerAbortController.signal,
575
- capture: true
576
- });
577
- }
578
- _setOverlayPosition(originElement = this.originElement) {
579
- if (!originElement) return;
580
- setOverlayPosition(this._overlay, originElement, this._optionContainer, this.shadowRoot.querySelector(".sbb-autocomplete__container"), this, this.position);
581
- }
582
- /**
583
- * On open/close animation end.
584
- * In rare cases it can be that the animationEnd event is triggered twice.
585
- * To avoid entering a corrupt state, exit when state is not expected.
586
- */
587
- _onAnimationEnd(event) {
588
- if (event.animationName === "open" && this.state === "opening") this._handleOpening();
589
- else if (event.animationName === "close" && this.state === "closing") this._handleClosing();
590
- }
591
- _handleOpening() {
592
- this.state = "opened";
593
- if (this.originElement) this._originResizeObserver.observe(this.originElement);
594
- this.triggerElement?.setAttribute("aria-expanded", "true");
595
- this.dispatchOpenEvent();
596
- }
597
- _handleClosing() {
598
- this.state = "closed";
599
- this.hidePopover?.();
600
- this.triggerElement?.setAttribute("aria-expanded", "false");
601
- this.resetActiveElement();
602
- this._optionContainer.scrollTop = 0;
603
- this._escapableOverlayController.disconnect();
604
- this.dispatchCloseEvent();
605
- }
606
- _attachOpenPanelEvents() {
607
- this._openPanelEventsController = new AbortController();
608
- document.addEventListener("scroll", () => this._setOverlayPosition(), {
609
- passive: true,
610
- signal: this._openPanelEventsController.signal,
611
- capture: true
612
- });
613
- window.addEventListener("resize", () => this._setOverlayPosition(), {
614
- passive: true,
615
- signal: this._openPanelEventsController.signal
616
- });
617
- window.addEventListener("pointerdown", (ev) => this._pointerDownListener(ev), { signal: this._openPanelEventsController.signal });
618
- window.addEventListener("pointerup", (ev) => this._closeOnBackdropClick(ev), { signal: this._openPanelEventsController.signal });
619
- this.addEventListener("ɵdisabledchange", () => {
620
- if (this.activeOption?.disabled) this.resetActiveElement();
621
- this._setNextActiveOptionIfAutoActiveFirstOption();
622
- }, { signal: this._openPanelEventsController.signal });
623
- this.triggerElement?.addEventListener("keydown", (event) => this.openedPanelKeyboardInteraction(event), {
624
- signal: this._openPanelEventsController.signal,
625
- capture: true
626
- });
627
- }
628
- _closedPanelKeyboardInteraction(event) {
629
- if (this.state === "opening" || this.state === "opened") return;
630
- switch (event.key) {
631
- case "ArrowDown":
632
- case "ArrowUp":
633
- this.open();
634
- break;
635
- }
636
- }
637
- /** Highlight the searched text on the options. */
638
- _highlightOptions(searchTerm) {
639
- if (searchTerm === null || searchTerm === void 0) return;
640
- this.options.forEach((option) => option.highlight(searchTerm));
641
- }
642
- render() {
643
- return html`
644
- <div class="sbb-autocomplete__gap-fix"></div>
645
- <div class="sbb-autocomplete__container">
646
- <div class="sbb-autocomplete__gap-fix">${overlayGapFixCorners()}</div>
647
- <div
648
- @animationend=${this._onAnimationEnd}
649
- class="sbb-autocomplete__panel"
650
- ${ref((overlayRef) => this._overlay = overlayRef)}
651
- >
652
- <div class="sbb-autocomplete__wrapper">
653
- <div
654
- class="sbb-autocomplete__options"
655
- role=${!ariaRoleOnHost ? this.panelRole : nothing}
656
- id=${!ariaRoleOnHost ? this.overlayId : nothing}
657
- tabindex="-1"
658
- ${ref((containerRef) => this._optionContainer = containerRef)}
659
- >
660
- <slot @slotchange=${this._handleSlotchange}></slot>
661
- </div>
662
- </div>
663
- </div>
664
- </div>
665
- `;
666
- }
667
- };
668
- })();
669
- //#endregion
670
- export { SbbAutocompleteBaseElement as t };
671
-
672
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0b2NvbXBsZXRlLWJhc2UtZWxlbWVudC1CNW4tZUNNTy5qcyIsIm5hbWVzIjpbXSwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvZWxlbWVudHMvYXV0b2NvbXBsZXRlL2F1dG9jb21wbGV0ZS1iYXNlLWVsZW1lbnQuc2Nzcz9pbmxpbmUiLCIuLi8uLi8uLi9zcmMvZWxlbWVudHMvYXV0b2NvbXBsZXRlL2F1dG9jb21wbGV0ZS1iYXNlLWVsZW1lbnQudHMiXSwic291cmNlc0NvbnRlbnQiOlsiQHVzZSAnLi4vY29yZS9zdHlsZXMnIGFzIHNiYjtcblxuLy8gRml4ZXMgdGhlIGdhcCBiZXR3ZWVuIHRoZSBvcmlnaW4gYW5kIHRoZSBvdmVybGF5IGJ5IGNyZWF0aW5nIGNvbmp1bmN0aW9uXG4vLyBjb3JuZXJzIGJhc2VkIG9uIHRoZSBvcmlnaW4gZWxlbWVudCBib3JkZXIgcmFkaXVzXG5AaW5jbHVkZSBzYmIub3ZlcmxheS1nYXAtZml4O1xuXG46aG9zdCB7XG4gIEBpbmNsdWRlIHNiYi5vcHRpb25zLXBhbmVsLW92ZXJsYXktdmFyaWFibGVzO1xuXG4gIC0tc2JiLW9wdGlvbnMtcG9pbnRlci1ldmVudHM6IGFsbDtcbiAgLS1zYmItb3B0aW9ucy1wYW5lbC1pbnRlcm5hbC16LWluZGV4OiB2YXIoXG4gICAgLS1zYmItYXV0b2NvbXBsZXRlLXotaW5kZXgsXG4gICAgdmFyKC0tc2JiLW92ZXJsYXktZGVmYXVsdC16LWluZGV4KVxuICApO1xuXG4gIGRpc3BsYXk6IG5vbmU7XG4gIGZvbnQtd2VpZ2h0OiBub3JtYWw7XG59XG5cbjpob3N0KFtzaXplPSdzJ10pIHtcbiAgQGluY2x1ZGUgc2JiLm9wdGlvbnMtcGFuZWwtb3ZlcmxheS1zaXplLXMtdmFyaWFibGVzO1xufVxuXG46aG9zdChbc2l6ZT0nbSddKSB7XG4gIEBpbmNsdWRlIHNiYi5vcHRpb25zLXBhbmVsLW92ZXJsYXktc2l6ZS1tLXZhcmlhYmxlcztcbn1cblxuOmhvc3QoOmlzKDpzdGF0ZShzdGF0ZS1vcGVuaW5nKSwgOnN0YXRlKHN0YXRlLW9wZW5lZCksIDpzdGF0ZShzdGF0ZS1jbG9zaW5nKSkpIHtcbiAgZGlzcGxheTogYmxvY2s7XG59XG5cbjpob3N0KFtuZWdhdGl2ZV0pIHtcbiAgQGluY2x1ZGUgc2JiLm9wdGlvbnMtcGFuZWwtb3ZlcmxheS1uZWdhdGl2ZS12YXJpYWJsZXM7XG59XG5cbjpob3N0KDpub3QoOnN0YXRlKHN0YXRlLW9wZW5pbmcpLCA6c3RhdGUoc3RhdGUtb3BlbmVkKSwgOnN0YXRlKHN0YXRlLWNsb3NpbmcpKSksXG46aG9zdCg6c3RhdGUoc3RhdGUtY2xvc2VkKSkge1xuICAtLXNiYi1vcHRpb25zLXBhbmVsLXZpc2liaWxpdHk6IGhpZGRlbjtcbiAgLS1zYmItb3B0aW9ucy1wb2ludGVyLWV2ZW50czogbm9uZTtcbn1cblxuOmhvc3QoOnN0YXRlKHN0YXRlLW9wZW5pbmcpKSB7XG4gIC0tc2JiLW9wdGlvbnMtcGFuZWwtYW5pbWF0aW9uLW5hbWU6IG9wZW47XG59XG5cbjpob3N0KDpzdGF0ZShzdGF0ZS1jbG9zaW5nKSkge1xuICAtLXNiYi1vcHRpb25zLXBhbmVsLWFuaW1hdGlvbi1uYW1lOiBjbG9zZTtcbn1cblxuOmhvc3QoOnN0YXRlKHN0YXRlLW9wZW5lZCkpLFxuOmhvc3QoOnN0YXRlKHN0YXRlLW9wZW5pbmcpKSB7XG4gIC0tc2JiLW9wdGlvbnMtcGFuZWwtZ2FwLWZpeC1vcGFjaXR5OiAxO1xufVxuXG46aG9zdCg6c3RhdGUob3B0aW9ucy1wYW5lbC1wb3NpdGlvbi1iZWxvdykpIHtcbiAgLS1zYmItb3B0aW9ucy1wYW5lbC1hbmltYXRpb24tdHJhbnNmb3JtOiB0cmFuc2xhdGVZKFxuICAgIGNhbGMoKHZhcigtLXNiYi1vcHRpb25zLXBhbmVsLW9yaWdpbi1oZWlnaHQpIC8gMikgKiAtMSlcbiAgKTtcbn1cblxuOmhvc3QoOnN0YXRlKG9wdGlvbnMtcGFuZWwtcG9zaXRpb24tYWJvdmUpKSB7XG4gIC0tc2JiLW9wdGlvbnMtcGFuZWwtb3B0aW9ucy1ib3JkZXItcmFkaXVzOiB2YXIoLS1zYmItb3B0aW9ucy1wYW5lbC1ib3JkZXItcmFkaXVzKVxuICAgIHZhcigtLXNiYi1vcHRpb25zLXBhbmVsLWJvcmRlci1yYWRpdXMpIDAgMDtcbiAgLS1zYmItb3B0aW9ucy1wYW5lbC1nYXAtZml4LXRvcDogdmFyKC0tX3NiYi1vcHRpb25zLXBhbmVsLW1heC1oZWlnaHQpO1xuICAtLXNiYi1vcHRpb25zLXBhbmVsLWdhcC1maXgtdHJhbnNmb3JtOiByb3RhdGUoMTgwZGVnKTtcbiAgLS1zYmItb3B0aW9ucy1wYW5lbC1hbmltYXRpb24tdHJhbnNmb3JtOiB0cmFuc2xhdGVZKFxuICAgIGNhbGModmFyKC0tc2JiLW9wdGlvbnMtcGFuZWwtb3JpZ2luLWhlaWdodCkgLyAyKVxuICApO1xufVxuXG46aG9zdChbcHJlc2VydmUtaWNvbi1zcGFjZV0pIHtcbiAgLS1zYmItb3B0aW9uLWljb24tY29udGFpbmVyLWRpc3BsYXk6IGJsb2NrO1xufVxuXG46OnNsb3R0ZWQoc2JiLWRpdmlkZXIpIHtcbiAgbWFyZ2luLWJsb2NrOiB2YXIoLS1zYmItb3B0aW9ucy1wYW5lbC1kaXZpZGVyLW1hcmdpbi1ibG9jayk7XG59XG5cbi5zYmItYXV0b2NvbXBsZXRlX19jb250YWluZXIge1xuICBAaW5jbHVkZSBzYmIub3B0aW9ucy1wYW5lbC1vdmVybGF5LWNvbnRhaW5lcjtcbn1cblxuLnNiYi1hdXRvY29tcGxldGVfX2dhcC1maXgge1xuICBAaW5jbHVkZSBzYmIub3B0aW9ucy1wYW5lbC1vdmVybGF5LWdhcDtcbn1cblxuLnNiYi1hdXRvY29tcGxldGVfX3BhbmVsIHtcbiAgQGluY2x1ZGUgc2JiLm9wdGlvbnMtcGFuZWwtb3ZlcmxheTtcblxuICA6aG9zdCg6c3RhdGUob3B0aW9ucy1wYW5lbC1wb3NpdGlvbi1iZWxvdykpICYge1xuICAgIGluc2V0LWJsb2NrLXN0YXJ0OiBjYWxjKFxuICAgICAgdmFyKC0tc2JiLW9wdGlvbnMtcGFuZWwtcG9zaXRpb24teSkgLSB2YXIoLS1zYmItb3B0aW9ucy1wYW5lbC1vcmlnaW4taGVpZ2h0KVxuICAgICk7XG4gIH1cblxuICA6aG9zdCg6aXMoOnN0YXRlKHN0YXRlLW9wZW5lZCksIDpzdGF0ZShzdGF0ZS1vcGVuaW5nKSkpICYge1xuICAgIGJveC1zaGFkb3c6IHZhcigtLXNiYi1ib3gtc2hhZG93LWxldmVsLTUtaGFyZCk7XG4gIH1cblxuICA6aG9zdCg6aXMoOnN0YXRlKHN0YXRlLW9wZW5lZCksIDpzdGF0ZShzdGF0ZS1vcGVuaW5nKSlbbmVnYXRpdmVdKSAmIHtcbiAgICBib3gtc2hhZG93OiB2YXIoLS1zYmItYm94LXNoYWRvdy1sZXZlbC01LWhhcmQtbmVnYXRpdmUpO1xuICB9XG5cbiAgJjo6YmVmb3JlIHtcbiAgICA6aG9zdCg6c3RhdGUob3B0aW9ucy1wYW5lbC1wb3NpdGlvbi1iZWxvdykpICYge1xuICAgICAgZGlzcGxheTogYmxvY2s7XG4gICAgfVxuICB9XG5cbiAgJjo6YWZ0ZXIge1xuICAgIDpob3N0KDpzdGF0ZShvcHRpb25zLXBhbmVsLXBvc2l0aW9uLWFib3ZlKSkgJiB7XG4gICAgICBkaXNwbGF5OiBibG9jaztcbiAgICB9XG4gIH1cblxuICAvLyBzdHlsZWxpbnQtZGlzYWJsZS1uZXh0LWxpbmUgbm8tZGVzY2VuZGluZy1zcGVjaWZpY2l0eVxuICAmOjpiZWZvcmUsXG4gICY6OmFmdGVyIHtcbiAgICA6aG9zdCg6aXMoOnN0YXRlKHN0YXRlLW9wZW5lZCksIDpzdGF0ZShzdGF0ZS1vcGVuaW5nKSk6c3RhdGUob3B0aW9uLXBhbmVsLW9yaWdpbi1ib3JkZXJsZXNzKSlcbiAgICAgICYge1xuICAgICAgYm94LXNoYWRvdzogdmFyKC0tc2JiLWJveC1zaGFkb3ctbGV2ZWwtNS1oYXJkKTtcbiAgICB9XG5cbiAgICA6aG9zdChcbiAgICAgICAgOmlzKDpzdGF0ZShzdGF0ZS1vcGVuZWQpLCA6c3RhdGUoc3RhdGUtb3BlbmluZykpOnN0YXRlKFxuICAgICAgICAgICAgb3B0aW9uLXBhbmVsLW9yaWdpbi1ib3JkZXJsZXNzXG4gICAgICAgICAgKVtuZWdhdGl2ZV1cbiAgICAgIClcbiAgICAgICYge1xuICAgICAgYm94LXNoYWRvdzogdmFyKC0tc2JiLWJveC1zaGFkb3ctbGV2ZWwtNS1oYXJkLW5lZ2F0aXZlKTtcbiAgICB9XG4gIH1cbn1cblxuLnNiYi1hdXRvY29tcGxldGVfX3dyYXBwZXIge1xuICBvdmVyZmxvdzogaGlkZGVuO1xufVxuXG4uc2JiLWF1dG9jb21wbGV0ZV9fb3B0aW9ucyB7XG4gIEBpbmNsdWRlIHNiYi5vcHRpb25zT3ZlcmxheTtcbiAgQGluY2x1ZGUgc2JiLnNjcm9sbGJhci1ydWxlcztcblxuICBwb2ludGVyLWV2ZW50czogdmFyKC0tc2JiLW9wdGlvbnMtcG9pbnRlci1ldmVudHMpO1xuXG4gIEBpbmNsdWRlIHNiYi5pZi1mb3JjZWQtY29sb3JzIHtcbiAgICBib3JkZXI6IHZhcigtLXNiYi1ib3JkZXItd2lkdGgtMXgpIHNvbGlkIENhbnZhc1RleHQ7XG4gICAgYm9yZGVyLXRvcDogbm9uZTtcbiAgfVxufVxuXG5Aa2V5ZnJhbWVzIG9wZW4ge1xuICBmcm9tIHtcbiAgICB0cmFuc2Zvcm06IHZhcigtLXNiYi1vcHRpb25zLXBhbmVsLWFuaW1hdGlvbi10cmFuc2Zvcm0pO1xuICAgIG9wYWNpdHk6IDA7XG4gIH1cblxuICB0byB7XG4gICAgdHJhbnNmb3JtOiB0cmFuc2xhdGVZKDApO1xuICAgIG9wYWNpdHk6IDE7XG4gIH1cbn1cblxuQGtleWZyYW1lcyBjbG9zZSB7XG4gIGZyb20ge1xuICAgIHRyYW5zZm9ybTogdHJhbnNsYXRlWSgwKTtcbiAgICBvcGFjaXR5OiAxO1xuICB9XG5cbiAgdG8ge1xuICAgIHRyYW5zZm9ybTogdmFyKC0tc2JiLW9wdGlvbnMtcGFuZWwtYW5pbWF0aW9uLXRyYW5zZm9ybSk7XG4gICAgb3BhY2l0eTogMDtcbiAgfVxufVxuIiwiaW1wb3J0IHsgUmVzaXplQ29udHJvbGxlciB9IGZyb20gJ0BsaXQtbGFicy9vYnNlcnZlcnMvcmVzaXplLWNvbnRyb2xsZXIuanMnO1xuaW1wb3J0IHtcbiAgdHlwZSBDU1NSZXN1bHRHcm91cCxcbiAgaHRtbCxcbiAgaXNTZXJ2ZXIsXG4gIG5vdGhpbmcsXG4gIHR5cGUgUHJvcGVydHlEZWNsYXJhdGlvbixcbiAgdHlwZSBQcm9wZXJ0eVZhbHVlcyxcbiAgdHlwZSBUZW1wbGF0ZVJlc3VsdCxcbiAgdW5zYWZlQ1NTLFxufSBmcm9tICdsaXQnO1xuaW1wb3J0IHsgcHJvcGVydHkgfSBmcm9tICdsaXQvZGVjb3JhdG9ycy5qcyc7XG5pbXBvcnQgeyByZWYgfSBmcm9tICdsaXQvZGlyZWN0aXZlcy9yZWYuanMnO1xuXG5pbXBvcnQge1xuICBTYmJPcGVuQ2xvc2VCYXNlRWxlbWVudCxcbiAgU2JiUHJvcGVydHlXYXRjaGVyQ29udHJvbGxlcixcbiAgU2JiRXNjYXBhYmxlT3ZlcmxheUNvbnRyb2xsZXIsXG4gIGZvcmNlVHlwZSxcbiAgaWRSZWZlcmVuY2UsXG4gIGlzTGVhbixcbiAgaXNTYWZhcmksXG4gIGlzWmVyb0FuaW1hdGlvbkR1cmF0aW9uLFxuICBTYmJOZWdhdGl2ZU1peGluLFxuICBpc0V2ZW50T25FbGVtZW50LFxuICBvdmVybGF5R2FwRml4Q29ybmVycyxcbiAgcmVtb3ZlQXJpYUNvbWJvQm94QXR0cmlidXRlcyxcbiAgc2V0T3ZlcmxheVBvc2l0aW9uLFxuICBib3hTaXppbmdTdHlsZXMsXG59IGZyb20gJy4uL2NvcmUudHMnO1xuaW1wb3J0IHR5cGUgeyBTYmJGb3JtRmllbGRFbGVtZW50IH0gZnJvbSAnLi4vZm9ybS1maWVsZC9mb3JtLWZpZWxkL2Zvcm0tZmllbGQuY29tcG9uZW50LnRzJztcbmltcG9ydCB0eXBlIHsgU2JiT3B0aW9uQmFzZUVsZW1lbnQgfSBmcm9tICcuLi9vcHRpb24ucHVyZS50cyc7XG5cbmltcG9ydCBzdHlsZSBmcm9tICcuL2F1dG9jb21wbGV0ZS1iYXNlLWVsZW1lbnQuc2Nzcz9pbmxpbmUnO1xuXG4vKipcbiAqIE9uIFNhZmFyaSwgdGhlIGFyaWEgcm9sZSAnbGlzdGJveCcgbXVzdCBiZSBvbiB0aGUgaG9zdCBlbGVtZW50LCBvciBlbHNlIFZvaWNlT3ZlciB3b24ndCB3b3JrIGF0IGFsbC5cbiAqIE9uIHRoZSBvdGhlciBoYW5kLCBKQVdTIGFuZCBOVkRBIG5lZWQgdGhlIHJvbGUgdG8gYmUgXCJjbG9zZXJcIiB0byB0aGUgb3B0aW9ucywgb3IgZWxzZSBvcHRncm91cHMgd29uJ3Qgd29yay5cbiAqL1xuY29uc3QgYXJpYVJvbGVPbkhvc3QgPSBpc1NhZmFyaTtcblxuZXhwb3J0IGFic3RyYWN0IGNsYXNzIFNiYkF1dG9jb21wbGV0ZUJhc2VFbGVtZW50PFQgPSBzdHJpbmc+IGV4dGVuZHMgU2JiTmVnYXRpdmVNaXhpbihcbiAgU2JiT3BlbkNsb3NlQmFzZUVsZW1lbnQsXG4pIHtcbiAgcHVibGljIHN0YXRpYyBvdmVycmlkZSBzdHlsZXM6IENTU1Jlc3VsdEdyb3VwID0gW2JveFNpemluZ1N0eWxlcywgdW5zYWZlQ1NTKHN0eWxlKV07XG5cbiAgLyoqXG4gICAqIFRoZSBlbGVtZW50IHdoZXJlIHRoZSBhdXRvY29tcGxldGUgd2lsbCBhdHRhY2guXG4gICAqIElmIG5vdCBzZXQsIGFzIGZhbGxiYWNrIHRoZXJlIGFyZSB0d28gZWxlbWVudHMgd2hpY2ggY2FuIGFjdCBhcyBvcmlnaW4gd2l0aCBmb2xsb3dpbmcgcHJpb3JpdHkgb3JkZXI6XG4gICAqIDEuIGBzYmItZm9ybS1maWVsZGAgaWYgaXQgaXMgYW4gYW5jZXN0b3IuXG4gICAqIDIuIHRyaWdnZXIgZWxlbWVudCBpZiBzZXQuXG4gICAqXG4gICAqIEZvciBhdHRyaWJ1dGUgdXNhZ2UsIHByb3ZpZGUgYW4gaWQgcmVmZXJlbmNlLlxuICAgKi9cbiAgQGlkUmVmZXJlbmNlKClcbiAgQHByb3BlcnR5KClcbiAgcHVibGljIGFjY2Vzc29yIG9yaWdpbjogSFRNTEVsZW1lbnQgfCBudWxsID0gbnVsbDtcblxuICAvKipcbiAgICogVGhlIGlucHV0IGVsZW1lbnQgdGhhdCB3aWxsIHRyaWdnZXIgdGhlIGF1dG9jb21wbGV0ZSBvcGVuaW5nLlxuICAgKiBCeSBkZWZhdWx0LCB0aGUgYXV0b2NvbXBsZXRlIHdpbGwgb3BlbiBvbiBmb2N1cywgY2xpY2ssIGlucHV0IG9yIGBBcnJvd0Rvd25gIGtleXByZXNzIG9mIHRoZSAndHJpZ2dlcicgZWxlbWVudC5cbiAgICogSWYgbm90IHNldCwgd2lsbCBzZWFyY2ggZm9yIHRoZSBmaXJzdCAnaW5wdXQnIGNoaWxkIG9mIGEgJ3NiYi1mb3JtLWZpZWxkJyBhbmNlc3Rvci5cbiAgICpcbiAgICogRm9yIGF0dHJpYnV0ZSB1c2FnZSwgcHJvdmlkZSBhbiBpZCByZWZlcmVuY2UuXG4gICAqL1xuICBAaWRSZWZlcmVuY2UoKVxuICBAcHJvcGVydHkoKVxuICBwdWJsaWMgYWNjZXNzb3IgdHJpZ2dlcjogSFRNTElucHV0RWxlbWVudCB8IG51bGwgPSBudWxsO1xuXG4gIC8qKiBXaGV0aGVyIHRoZSBpY29uIHNwYWNlIGlzIHByZXNlcnZlZCB3aGVuIG5vIGljb24gaXMgc2V0LiAqL1xuICBAZm9yY2VUeXBlKClcbiAgQHByb3BlcnR5KHsgYXR0cmlidXRlOiAncHJlc2VydmUtaWNvbi1zcGFjZScsIHJlZmxlY3Q6IHRydWUsIHR5cGU6IEJvb2xlYW4gfSlcbiAgcHVibGljIGFjY2Vzc29yIHByZXNlcnZlSWNvblNwYWNlOiBib29sZWFuID0gZmFsc2U7XG5cbiAgLyoqIFdoZXRoZXIgdGhlIGZpcnN0IG9wdGlvbiBpcyBhdXRvbWF0aWNhbGx5IGFjdGl2YXRlZCB3aGVuIHRoZSBhdXRvY29tcGxldGUgaXMgb3BlbmVkLiAqL1xuICBAZm9yY2VUeXBlKClcbiAgQHByb3BlcnR5KHsgYXR0cmlidXRlOiAnYXV0by1hY3RpdmUtZmlyc3Qtb3B0aW9uJywgdHlwZTogQm9vbGVhbiB9KVxuICBwdWJsaWMgYWNjZXNzb3IgYXV0b0FjdGl2ZUZpcnN0T3B0aW9uOiBib29sZWFuID0gZmFsc2U7XG5cbiAgLyoqIEZ1bmN0aW9uIHRoYXQgbWFwcyBhbiBvcHRpb24ncyBjb250cm9sIHZhbHVlIHRvIGl0cyBkaXNwbGF5IHZhbHVlIGluIHRoZSB0cmlnZ2VyLiAqL1xuICBAcHJvcGVydHkoeyBhdHRyaWJ1dGU6IGZhbHNlIH0pXG4gIHB1YmxpYyBhY2Nlc3NvciBkaXNwbGF5V2l0aDogKCh2YWx1ZTogVCkgPT4gc3RyaW5nKSB8IG51bGwgPSBudWxsO1xuXG4gIC8qKlxuICAgKiBTaXplIHZhcmlhbnQsIGVpdGhlciBtIG9yIHMuXG4gICAqIEBkZWZhdWx0ICdtJyAvICdzJyAobGVhbilcbiAgICovXG4gIEBwcm9wZXJ0eSh7IHJlZmxlY3Q6IHRydWUgfSkgcHVibGljIGFjY2Vzc29yIHNpemU6ICdtJyB8ICdzJyA9IGlzTGVhbigpID8gJ3MnIDogJ20nO1xuXG4gIC8qKiBXaGV0aGVyIHRoZSBhY3RpdmUgb3B0aW9uIHNob3VsZCBiZSBzZWxlY3RlZCBhcyB0aGUgdXNlciBpcyBuYXZpZ2F0aW5nLiAqL1xuICBAZm9yY2VUeXBlKClcbiAgQHByb3BlcnR5KHsgYXR0cmlidXRlOiAnYXV0by1zZWxlY3QtYWN0aXZlLW9wdGlvbicsIHR5cGU6IEJvb2xlYW4gfSlcbiAgcHVibGljIGFjY2Vzc29yIGF1dG9TZWxlY3RBY3RpdmVPcHRpb246IGJvb2xlYW4gPSBmYWxzZTtcblxuICAvKipcbiAgICogV2hlbiBlbmFibGVkLCB0aGUgYWN0aXZlIG9wdGlvbiBpcyBhdXRvbWF0aWNhbGx5IHNlbGVjdGVkIG9uIGJsdXIuXG4gICAqIFRoaXMgaXMgYW4gZXhwZXJpbWVudGFsIGZlYXR1cmUuIEl0IG1pZ2h0IGJlIHN1YmplY3QgdG8gY2hhbmdlcy5cbiAgICovXG4gIEBmb3JjZVR5cGUoKVxuICBAcHJvcGVydHkoeyBhdHRyaWJ1dGU6ICdhdXRvLXNlbGVjdC1hY3RpdmUtb3B0aW9uLW9uLWJsdXInLCB0eXBlOiBCb29sZWFuIH0pXG4gIHB1YmxpYyBhY2Nlc3NvciBhdXRvU2VsZWN0QWN0aXZlT3B0aW9uT25CbHVyOiBib29sZWFuID0gZmFsc2U7XG5cbiAgLyoqXG4gICAqIFdoZXRoZXIgdGhlIHVzZXIgaXMgcmVxdWlyZWQgdG8gbWFrZSBhIHNlbGVjdGlvbiB3aGVuIHRoZXkncmUgaW50ZXJhY3Rpbmcgd2l0aCB0aGVcbiAgICogYXV0b2NvbXBsZXRlLiBJZiB0aGUgdXNlciBtb3ZlcyBhd2F5IGZyb20gdGhlIGF1dG9jb21wbGV0ZSB3aXRob3V0IHNlbGVjdGluZyBhbiBvcHRpb24gZnJvbVxuICAgKiB0aGUgbGlzdCwgdGhlIHZhbHVlIHdpbGwgYmUgcmVzZXQuIElmIHRoZSB1c2VyIG9wZW5zIHRoZSBwYW5lbCBhbmQgY2xvc2VzIGl0IHdpdGhvdXRcbiAgICogaW50ZXJhY3Rpbmcgb3Igc2VsZWN0aW5nIGEgdmFsdWUsIHRoZSBpbml0aWFsIHZhbHVlIHdpbGwgYmUga2VwdC5cbiAgICovXG4gIEBmb3JjZVR5cGUoKVxuICBAcHJvcGVydHkoeyBhdHRyaWJ1dGU6ICdyZXF1aXJlLXNlbGVjdGlvbicsIHR5cGU6IEJvb2xlYW4gfSlcbiAgcHVibGljIGFjY2Vzc29yIHJlcXVpcmVTZWxlY3Rpb246IGJvb2xlYW4gPSBmYWxzZTtcblxuICAvKipcbiAgICogVGhlIHBvc2l0aW9uIG9mIHRoZSBhdXRvY29tcGxldGUgcGFuZWwgcmVsYXRpdmUgdG8gdGhlIHRyaWdnZXIuXG4gICAqIEBkZWZhdWx0ICdhdXRvJ1xuICAgKi9cbiAgQHByb3BlcnR5KClcbiAgcHVibGljIGFjY2Vzc29yIHBvc2l0aW9uOiAnYXV0bycgfCAnYWJvdmUnIHwgJ2JlbG93JyA9ICdhdXRvJztcblxuICAvKiogUmV0dXJucyB0aGUgZWxlbWVudCB3aGVyZSB0aGUgYXV0b2NvbXBsZXRlIG92ZXJsYXkgaXMgYXR0YWNoZWQgdG8uICovXG4gIHB1YmxpYyBnZXQgb3JpZ2luRWxlbWVudCgpOiBIVE1MRWxlbWVudCB8IG51bGwge1xuICAgIHJldHVybiAoXG4gICAgICB0aGlzLm9yaWdpbiA/P1xuICAgICAgdGhpcy5jbG9zZXN0Py4oJ3NiYi1mb3JtLWZpZWxkJyk/LnNoYWRvd1Jvb3Q/LnF1ZXJ5U2VsZWN0b3I/LignI292ZXJsYXktYW5jaG9yJykgPz9cbiAgICAgIHRoaXMudHJpZ2dlciA/P1xuICAgICAgbnVsbFxuICAgICk7XG4gIH1cblxuICAvKiogUmV0dXJucyB0aGUgdHJpZ2dlciBlbGVtZW50LiAqL1xuICBwdWJsaWMgZ2V0IHRyaWdnZXJFbGVtZW50KCk6IEhUTUxJbnB1dEVsZW1lbnQgfCBudWxsIHtcbiAgICByZXR1cm4gdGhpcy5fdHJpZ2dlckVsZW1lbnQgPz8gbnVsbDtcbiAgfVxuICBwcml2YXRlIF90cmlnZ2VyRWxlbWVudD86IEhUTUxJbnB1dEVsZW1lbnQgfCBudWxsO1xuXG4gIHByb3RlY3RlZCBhYnN0cmFjdCBvdmVybGF5SWQ6IHN0cmluZztcbiAgcHJvdGVjdGVkIGFic3RyYWN0IHBhbmVsUm9sZTogc3RyaW5nO1xuICBwcm90ZWN0ZWQgYWN0aXZlT3B0aW9uOiBTYmJPcHRpb25CYXNlRWxlbWVudDxUPiB8IG51bGwgPSBudWxsO1xuICBwcm90ZWN0ZWQgcGVuZGluZ0F1dG9TZWxlY3RlZE9wdGlvbjogU2JiT3B0aW9uQmFzZUVsZW1lbnQ8VD4gfCBudWxsID0gbnVsbDtcbiAgcHJpdmF0ZSBfb3JpZ2luUmVzaXplT2JzZXJ2ZXIgPSBuZXcgUmVzaXplQ29udHJvbGxlcih0aGlzLCB7XG4gICAgdGFyZ2V0OiBudWxsLFxuICAgIHNraXBJbml0aWFsOiB0cnVlLFxuICAgIGNhbGxiYWNrOiAoKSA9PiB7XG4gICAgICBpZiAodGhpcy5zdGF0ZSA9PT0gJ29wZW5lZCcpIHtcbiAgICAgICAgdGhpcy5fc2V0T3ZlcmxheVBvc2l0aW9uKCk7XG4gICAgICB9XG4gICAgfSxcbiAgfSk7XG4gIC8qKiBMaXN0ZW5zIHRvIHRoZSBjaGFuZ2VzIG9uIHRoZSBgZGlzYWJsZWRgIG9yIGByZWFkb25seWAgYXR0cmlidXRlIG9mIHRoZSB0cmlnZ2VyLiAqL1xuICBwcml2YXRlIF90cmlnZ2VyQXR0cmlidXRlT2JzZXJ2ZXIgPSAhaXNTZXJ2ZXJcbiAgICA/IG5ldyBNdXRhdGlvbk9ic2VydmVyKChtdXRhdGlvbnM6IE11dGF0aW9uUmVjb3JkW10pOiB2b2lkID0+IHtcbiAgICAgICAgY29uc3QgaW5wdXQgPSBtdXRhdGlvbnNbMF0udGFyZ2V0IGFzIEhUTUxJbnB1dEVsZW1lbnQ7XG4gICAgICAgIGlmIChpbnB1dC5oYXNBdHRyaWJ1dGUoJ2Rpc2FibGVkJykgfHwgaW5wdXQuaGFzQXR0cmlidXRlKCdyZWFkb25seScpKSB7XG4gICAgICAgICAgdGhpcy5jbG9zZSgpO1xuICAgICAgICB9XG4gICAgICB9KVxuICAgIDogbnVsbDtcbiAgcHJpdmF0ZSBfb3ZlcmxheSE6IEhUTUxFbGVtZW50O1xuICBwcml2YXRlIF9vcHRpb25Db250YWluZXIhOiBIVE1MRWxlbWVudDtcbiAgcHJpdmF0ZSBfdHJpZ2dlckFib3J0Q29udHJvbGxlciE6IEFib3J0Q29udHJvbGxlcjtcbiAgcHJpdmF0ZSBfb3BlblBhbmVsRXZlbnRzQ29udHJvbGxlciE6IEFib3J0Q29udHJvbGxlcjtcbiAgcHJpdmF0ZSBfaXNQb2ludGVyRG93bkV2ZW50T25NZW51OiBib29sZWFuID0gZmFsc2U7XG4gIHByaXZhdGUgX2VzY2FwYWJsZU92ZXJsYXlDb250cm9sbGVyID0gbmV3IFNiYkVzY2FwYWJsZU92ZXJsYXlDb250cm9sbGVyKHRoaXMpO1xuICBwcml2YXRlIF9vcHRpb25zQ291bnQgPSAwO1xuXG4gIC8qKiBUcmFja3MgaW5wdXQgZnJvbSBrZXlib2FyZC4gKi9cbiAgcHJpdmF0ZSBfbGFzdFVzZXJJbnB1dDogc3RyaW5nIHwgbnVsbCA9IG51bGw7XG5cbiAgLyoqIElmIHRydWUsIHRoZSAnY2hhbmdlJyBldmVudCBzb3VyY2UgaXMgYW4gb3B0aW9uIHNlbGVjdGlvbiBvciBhICdyZXF1aXJlU2VsZWN0aW9uJyBmaWVsZCBjbGVhbnVwICovXG4gIHByaXZhdGUgX2lzQ3VzdG9tQ2hhbmdlRXZlbnQgPSBmYWxzZTtcblxuICBwcm90ZWN0ZWQgYWJzdHJhY3QgZ2V0IG9wdGlvbnMoKTogU2JiT3B0aW9uQmFzZUVsZW1lbnQ8VD5bXTtcblxuICBwdWJsaWMgY29uc3RydWN0b3IoKSB7XG4gICAgc3VwZXIoKTtcbiAgICB0aGlzLmFkZEV2ZW50TGlzdGVuZXI/Lignb3B0aW9uc2VsZWN0ZWQnLCAoZTogRXZlbnQpID0+IHRoaXMub25PcHRpb25TZWxlY3RlZChlKSk7XG4gICAgdGhpcy5hZGRFdmVudExpc3RlbmVyPy4oJ8m1b3B0Z3JvdXBzbG90Y2hhbmdlJywgKCkgPT4gdGhpcy5faGFuZGxlU2xvdGNoYW5nZSgpLCB7XG4gICAgICBjYXB0dXJlOiB0cnVlLFxuICAgIH0pO1xuICAgIHRoaXMuYWRkQ29udHJvbGxlcihcbiAgICAgIG5ldyBTYmJQcm9wZXJ0eVdhdGNoZXJDb250cm9sbGVyKFxuICAgICAgICB0aGlzLFxuICAgICAgICAoKSA9PiB0aGlzLmNsb3Nlc3Q8U2JiRm9ybUZpZWxkRWxlbWVudD4oJ3NiYi1mb3JtLWZpZWxkJyksXG4gICAgICAgIHtcbiAgICAgICAgICBuZWdhdGl2ZTogKGUpID0+IHtcbiAgICAgICAgICAgIHRoaXMubmVnYXRpdmUgPSBlLm5lZ2F0aXZlO1xuICAgICAgICAgICAgdGhpcy5zeW5jTmVnYXRpdmUoKTtcbiAgICAgICAgICB9LFxuICAgICAgICAgIGJvcmRlcmxlc3M6IChlKSA9PiB0aGlzLnRvZ2dsZVN0YXRlKCdvcHRpb24tcGFuZWwtb3JpZ2luLWJvcmRlcmxlc3MnLCBlLmJvcmRlcmxlc3MpLFxuICAgICAgICB9LFxuICAgICAgKSxcbiAgICApO1xuICB9XG5cbiAgcHJvdGVjdGVkIGFic3RyYWN0IHN5bmNOZWdhdGl2ZSgpOiB2b2lkO1xuICBwcm90ZWN0ZWQgYWJzdHJhY3Qgc2V0VHJpZ2dlckF0dHJpYnV0ZXMoZWxlbWVudDogSFRNTElucHV0RWxlbWVudCk6IHZvaWQ7XG4gIHByb3RlY3RlZCBhYnN0cmFjdCBvcGVuZWRQYW5lbEtleWJvYXJkSW50ZXJhY3Rpb24oZXZlbnQ6IEtleWJvYXJkRXZlbnQpOiB2b2lkO1xuICBwcm90ZWN0ZWQgYWJzdHJhY3Qgc2VsZWN0QnlLZXlib2FyZChldmVudDogS2V5Ym9hcmRFdmVudCk6IHZvaWQ7XG4gIHByb3RlY3RlZCBhYnN0cmFjdCBzZXROZXh0QWN0aXZlT3B0aW9uKGV2ZW50PzogS2V5Ym9hcmRFdmVudCk6IHZvaWQ7XG4gIHByb3RlY3RlZCBhYnN0cmFjdCByZXNldEFjdGl2ZUVsZW1lbnQoKTogdm9pZDtcblxuICAvKiogT3BlbnMgdGhlIGF1dG9jb21wbGV0ZS4gKi9cbiAgcHVibGljIG9wZW4oKTogdm9pZCB7XG4gICAgaWYgKFxuICAgICAgdGhpcy5zdGF0ZSA9PT0gJ29wZW5pbmcnIHx8XG4gICAgICB0aGlzLnN0YXRlID09PSAnb3BlbmVkJyB8fFxuICAgICAgIXRoaXMuX292ZXJsYXkgfHxcbiAgICAgIHRoaXMub3B0aW9ucy5sZW5ndGggPT09IDAgfHxcbiAgICAgIHRoaXMuX3JlYWRvbmx5KCkgfHxcbiAgICAgICF0aGlzLmRpc3BhdGNoQmVmb3JlT3BlbkV2ZW50KClcbiAgICApIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICB0aGlzLnNob3dQb3BvdmVyPy4oKTtcbiAgICB0aGlzLnN0YXRlID0gJ29wZW5pbmcnO1xuICAgIHRoaXMudHJpZ2dlckVsZW1lbnQ/LnRvZ2dsZUF0dHJpYnV0ZSgnZGF0YS1leHBhbmRlZCcsIHRydWUpO1xuICAgIGNvbnN0IG9yaWdpbkVsZW1lbnQgPSB0aGlzLm9yaWdpbkVsZW1lbnQ7XG4gICAgaWYgKCFvcmlnaW5FbGVtZW50KSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgICdDYW5ub3QgZmluZCB0aGUgb3JpZ2luIGVsZW1lbnQuIFBsZWFzZSBzcGVjaWZ5IGEgdmFsaWQgZWxlbWVudCBvciBjaGVjayB0aGUgdXNhZ2Ugb2YgdGhlIFwib3JpZ2luXCIgcHJvcGVydHkgZnJvbSB0aGUgZG9jdW1lbnRhdGlvbicsXG4gICAgICApO1xuICAgIH1cbiAgICB0aGlzLl9zZXRPdmVybGF5UG9zaXRpb24ob3JpZ2luRWxlbWVudCk7XG4gICAgdGhpcy5fc2V0TmV4dEFjdGl2ZU9wdGlvbklmQXV0b0FjdGl2ZUZpcnN0T3B0aW9uKCk7XG4gICAgdGhpcy5fYXR0YWNoT3BlblBhbmVsRXZlbnRzKCk7XG4gICAgdGhpcy5fZXNjYXBhYmxlT3ZlcmxheUNvbnRyb2xsZXIuY29ubmVjdCgpO1xuXG4gICAgLy8gSWYgdGhlIGFuaW1hdGlvbiBkdXJhdGlvbiBpcyB6ZXJvLCB0aGUgYW5pbWF0aW9uZW5kIGV2ZW50IGlzIG5vdCBhbHdheXMgZmlyZWQgcmVsaWFibHkuXG4gICAgLy8gSW4gdGhpcyBjYXNlIHdlIGRpcmVjdGx5IHNldCB0aGUgYG9wZW5lZGAgc3RhdGUuXG4gICAgaWYgKHRoaXMuX2lzWmVyb0FuaW1hdGlvbkR1cmF0aW9uKCkpIHtcbiAgICAgIHRoaXMuX2hhbmRsZU9wZW5pbmcoKTtcbiAgICB9XG4gIH1cblxuICAvKiogQ2xvc2VzIHRoZSBhdXRvY29tcGxldGUuICovXG4gIHB1YmxpYyBjbG9zZSgpOiB2b2lkIHtcbiAgICBpZiAodGhpcy5zdGF0ZSA9PT0gJ2Nsb3NpbmcnIHx8IHRoaXMuc3RhdGUgPT09ICdjbG9zZWQnIHx8ICF0aGlzLmRpc3BhdGNoQmVmb3JlQ2xvc2VFdmVudCgpKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgLy8gQSAncGVuZGluZyBzZWxlY3Rpb24nIGlzIGNvbmZpcm1lZCBvbiBwYW5lbCBjbG9zZVxuICAgIGlmICh0aGlzLnBlbmRpbmdBdXRvU2VsZWN0ZWRPcHRpb24pIHtcbiAgICAgIHRoaXMucGVuZGluZ0F1dG9TZWxlY3RlZE9wdGlvbi5zZWxlY3RlZCA9IHRydWU7XG4gICAgICB0aGlzLl9zZXRWYWx1ZUFuZERpc3BhdGNoRXZlbnRzKHRoaXMucGVuZGluZ0F1dG9TZWxlY3RlZE9wdGlvbiwgdHJ1ZSk7XG4gICAgfVxuXG4gICAgdGhpcy5zdGF0ZSA9ICdjbG9zaW5nJztcbiAgICB0aGlzLnRyaWdnZXJFbGVtZW50Py5yZW1vdmVBdHRyaWJ1dGUoJ2RhdGEtZXhwYW5kZWQnKTtcbiAgICB0aGlzLl9vcGVuUGFuZWxFdmVudHNDb250cm9sbGVyLmFib3J0KCk7XG4gICAgaWYgKHRoaXMub3JpZ2luRWxlbWVudCkge1xuICAgICAgdGhpcy5fb3JpZ2luUmVzaXplT2JzZXJ2ZXIudW5vYnNlcnZlKHRoaXMub3JpZ2luRWxlbWVudCk7XG4gICAgfVxuXG4gICAgLy8gSWYgdGhlIGFuaW1hdGlvbiBkdXJhdGlvbiBpcyB6ZXJvLCB0aGUgYW5pbWF0aW9uZW5kIGV2ZW50IGlzIG5vdCBhbHdheXMgZmlyZWQgcmVsaWFibHkuXG4gICAgLy8gSW4gdGhpcyBjYXNlIHdlIGRpcmVjdGx5IHNldCB0aGUgYGNsb3NlZGAgc3RhdGUuXG4gICAgaWYgKHRoaXMuX2lzWmVyb0FuaW1hdGlvbkR1cmF0aW9uKCkpIHtcbiAgICAgIHRoaXMuX2hhbmRsZUNsb3NpbmcoKTtcbiAgICB9XG4gIH1cblxuICBwcml2YXRlIF9pc1plcm9BbmltYXRpb25EdXJhdGlvbigpOiBib29sZWFuIHtcbiAgICByZXR1cm4gaXNaZXJvQW5pbWF0aW9uRHVyYXRpb24odGhpcywgJy0tc2JiLW9wdGlvbnMtcGFuZWwtYW5pbWF0aW9uLWR1cmF0aW9uJyk7XG4gIH1cblxuICBwdWJsaWMgb3ZlcnJpZGUgY29ubmVjdGVkQ2FsbGJhY2soKTogdm9pZCB7XG4gICAgdGhpcy5wb3BvdmVyID0gJ21hbnVhbCc7XG4gICAgc3VwZXIuY29ubmVjdGVkQ2FsbGJhY2soKTtcbiAgICBpZiAoYXJpYVJvbGVPbkhvc3QpIHtcbiAgICAgIHRoaXMuaWQgfHw9IHRoaXMub3ZlcmxheUlkO1xuICAgIH1cblxuICAgIGlmICh0aGlzLmhhc1VwZGF0ZWQpIHtcbiAgICAgIHRoaXMuX2NvbXBvbmVudFNldHVwKCk7XG4gICAgfVxuICAgIHRoaXMuc3luY05lZ2F0aXZlKCk7XG4gIH1cblxuICBwcm90ZWN0ZWQgb3ZlcnJpZGUgd2lsbFVwZGF0ZShjaGFuZ2VkUHJvcGVydGllczogUHJvcGVydHlWYWx1ZXM8dGhpcz4pOiB2b2lkIHtcbiAgICBzdXBlci53aWxsVXBkYXRlKGNoYW5nZWRQcm9wZXJ0aWVzKTtcblxuICAgIGlmIChjaGFuZ2VkUHJvcGVydGllcy5oYXMoJ25lZ2F0aXZlJykpIHtcbiAgICAgIHRoaXMuc3luY05lZ2F0aXZlKCk7XG4gICAgfVxuXG4gICAgaWYgKGNoYW5nZWRQcm9wZXJ0aWVzLmhhcygnYXV0b0FjdGl2ZUZpcnN0T3B0aW9uJykgJiYgdGhpcy5pc09wZW4pIHtcbiAgICAgIHRoaXMuX3NldE5leHRBY3RpdmVPcHRpb25JZkF1dG9BY3RpdmVGaXJzdE9wdGlvbigpO1xuICAgIH1cbiAgfVxuXG4gIHByb3RlY3RlZCBvdmVycmlkZSBmaXJzdFVwZGF0ZWQoY2hhbmdlZFByb3BlcnRpZXM6IFByb3BlcnR5VmFsdWVzPHRoaXM+KTogdm9pZCB7XG4gICAgc3VwZXIuZmlyc3RVcGRhdGVkKGNoYW5nZWRQcm9wZXJ0aWVzKTtcbiAgICB0aGlzLl9jb21wb25lbnRTZXR1cCgpO1xuICB9XG5cbiAgcHVibGljIG92ZXJyaWRlIGRpc2Nvbm5lY3RlZENhbGxiYWNrKCk6IHZvaWQge1xuICAgIHN1cGVyLmRpc2Nvbm5lY3RlZENhbGxiYWNrKCk7XG4gICAgdGhpcy5fdHJpZ2dlckVsZW1lbnQgPSBudWxsO1xuICAgIHRoaXMuX3RyaWdnZXJBYm9ydENvbnRyb2xsZXI/LmFib3J0KCk7XG4gICAgdGhpcy5fb3BlblBhbmVsRXZlbnRzQ29udHJvbGxlcj8uYWJvcnQoKTtcbiAgfVxuXG4gIHB1YmxpYyBvdmVycmlkZSByZXF1ZXN0VXBkYXRlKFxuICAgIG5hbWU/OiBQcm9wZXJ0eUtleSxcbiAgICBvbGRWYWx1ZT86IHVua25vd24sXG4gICAgb3B0aW9ucz86IFByb3BlcnR5RGVjbGFyYXRpb24sXG4gICk6IHZvaWQge1xuICAgIHN1cGVyLnJlcXVlc3RVcGRhdGUobmFtZSwgb2xkVmFsdWUsIG9wdGlvbnMpO1xuXG4gICAgaWYgKGlzU2VydmVyIHx8ICF0aGlzLmhhc1VwZGF0ZWQpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgaWYgKCFuYW1lIHx8IG5hbWUgPT09ICd0cmlnZ2VyJykge1xuICAgICAgdGhpcy5fY29uZmlndXJlVHJpZ2dlcigpO1xuICAgIH0gZWxzZSBpZiAoKCFuYW1lIHx8IG5hbWUgPT09ICdvcmlnaW4nKSAmJiB0aGlzLmlzT3Blbikge1xuICAgICAgdGhpcy5fc2V0T3ZlcmxheVBvc2l0aW9uKCk7XG4gICAgfVxuICB9XG5cbiAgLyoqIFdoZW4gYW4gb3B0aW9uIGlzIHNlbGVjdGVkLCB1cGRhdGUgdGhlIGlucHV0IHZhbHVlIGFuZCBjbG9zZSB0aGUgYXV0b2NvbXBsZXRlLiAqL1xuICBwcm90ZWN0ZWQgb25PcHRpb25TZWxlY3RlZChldmVudDogRXZlbnQpOiB2b2lkIHtcbiAgICBjb25zdCB0YXJnZXQgPSBldmVudC50YXJnZXQgYXMgU2JiT3B0aW9uQmFzZUVsZW1lbnQ8VD47XG4gICAgdGhpcy5fc2V0VmFsdWVBbmREaXNwYXRjaEV2ZW50cyh0YXJnZXQpO1xuICAgIHRoaXMuY2xvc2UoKTtcbiAgfVxuXG4gIC8qKiBAZGVwcmVjYXRlZCAqL1xuICBwcm90ZWN0ZWQgb25PcHRpb25BcnJvd3NTZWxlY3RlZChfYWN0aXZlT3B0aW9uOiBTYmJPcHRpb25CYXNlRWxlbWVudDxUPik6IHZvaWQge31cblxuICAvKipcbiAgICogQSAncGVuZGluZyBzZWxlY3Rpb24nIHNldHMgdGhlIG9wdGlvbiB2YWx1ZSBpbiB0aGUgaW5wdXQgZWxlbWVudCB3aXRob3V0IGVtaXR0aW5nIGV2ZW50cy5cbiAgICogQSAncGVuZGluZyBzZWxlY3Rpb24nIGlzIGNvbmZpcm1lZCB3aGVuIHRoZSBwYW5lbCBjbG9zZXMuIEFueSBvdGhlciB1c2VyIGludGVyYWN0aW9uXG4gICAqIHdpbGwgcmVzZXQgdGhlIHBlbmRpbmcgdmFsdWUuXG4gICAqL1xuICBwcm90ZWN0ZWQgc2V0UGVuZGluZ1NlbGVjdGlvbihhY3RpdmVPcHRpb246IFNiYk9wdGlvbkJhc2VFbGVtZW50PFQ+KTogdm9pZCB7XG4gICAgdGhpcy5wZW5kaW5nQXV0b1NlbGVjdGVkT3B0aW9uID0gYWN0aXZlT3B0aW9uO1xuICAgIHRoaXMuX3NldElucHV0VmFsdWUoYWN0aXZlT3B0aW9uKTtcbiAgfVxuXG4gIHByaXZhdGUgX3NldFZhbHVlQW5kRGlzcGF0Y2hFdmVudHMoXG4gICAgc2VsZWN0ZWRPcHRpb246IFNiYk9wdGlvbkJhc2VFbGVtZW50PFQ+LFxuICAgIHByZXZlbnRGb2N1cyA9IGZhbHNlLFxuICApOiB2b2lkIHtcbiAgICAvLyBEZXNlbGVjdCB0aGUgcHJldmlvdXMgb3B0aW9uc1xuICAgIHRoaXMub3B0aW9uc1xuICAgICAgLmZpbHRlcigob3B0aW9uKSA9PiBvcHRpb24uaWQgIT09IHNlbGVjdGVkT3B0aW9uLmlkICYmIG9wdGlvbi5zZWxlY3RlZClcbiAgICAgIC5mb3JFYWNoKChvcHRpb24pID0+IChvcHRpb24uc2VsZWN0ZWQgPSBmYWxzZSkpO1xuICAgIHRoaXMucGVuZGluZ0F1dG9TZWxlY3RlZE9wdGlvbiA9IG51bGw7XG5cbiAgICBpZiAodGhpcy50cmlnZ2VyRWxlbWVudCkge1xuICAgICAgdGhpcy5fc2V0SW5wdXRWYWx1ZShzZWxlY3RlZE9wdGlvbik7XG5cbiAgICAgIC8vIE1hbnVhbGx5IHRyaWdnZXIgdGhlIGNoYW5nZSBldmVudHNcbiAgICAgIHRoaXMuX2lzQ3VzdG9tQ2hhbmdlRXZlbnQgPSB0cnVlO1xuICAgICAgdGhpcy50cmlnZ2VyRWxlbWVudC5kaXNwYXRjaEV2ZW50KG5ldyBFdmVudCgnY2hhbmdlJywgeyBidWJibGVzOiB0cnVlIH0pKTtcbiAgICAgIHRoaXMudHJpZ2dlckVsZW1lbnQuZGlzcGF0Y2hFdmVudChuZXcgSW5wdXRFdmVudCgnaW5wdXQnLCB7IGJ1YmJsZXM6IHRydWUsIGNvbXBvc2VkOiB0cnVlIH0pKTtcblxuICAgICAgLy8gRGlzcGF0Y2hpbmcgdGhlIGlucHV0IGV2ZW50IGNoYW5nZXMgX2xhc3RJbnB1dCwgd2hpY2ggc2hvdWxkIHN0YXkgZW1wdHkgc2luY2UgdGhlcmUncyBubyBrZXlib2FyZCBpbnRlcmFjdGlvbi5cbiAgICAgIHRoaXMuX2xhc3RVc2VySW5wdXQgPSBudWxsO1xuXG4gICAgICAvLyBDdXN0b20gaW5wdXQgZXZlbnQgZW1pdHRlZCB3aGVuIGlucHV0IHZhbHVlIGNoYW5nZXMgYWZ0ZXIgYW4gb3B0aW9uIGlzIHNlbGVjdGVkXG4gICAgICB0aGlzLnRyaWdnZXJFbGVtZW50LmRpc3BhdGNoRXZlbnQoXG4gICAgICAgIG5ldyBDdXN0b21FdmVudDx7IG9wdGlvbjogU2JiT3B0aW9uQmFzZUVsZW1lbnQ8VD4gfT4oJ2lucHV0QXV0b2NvbXBsZXRlJywge1xuICAgICAgICAgIGRldGFpbDogeyBvcHRpb246IHNlbGVjdGVkT3B0aW9uIH0sXG4gICAgICAgIH0pLFxuICAgICAgKTtcbiAgICAgIGlmICghcHJldmVudEZvY3VzKSB7XG4gICAgICAgIHRoaXMudHJpZ2dlckVsZW1lbnQuZm9jdXMoKTtcbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICAvKiogU2V0IHRoZSBvcHRpb24gdmFsdWUgd2l0aGluIHRoZSBpbnB1dCBlbGVtZW50ICovXG4gIHByaXZhdGUgX3NldElucHV0VmFsdWUob3B0aW9uOiBTYmJPcHRpb25CYXNlRWxlbWVudDxUPik6IHZvaWQge1xuICAgIGlmICghdGhpcy50cmlnZ2VyRWxlbWVudCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIC8vIEdpdmVuIGEgdmFsdWUsIHJldHVybnMgdGhlIHN0cmluZyB0aGF0IHNob3VsZCBiZSBzaG93biB3aXRoaW4gdGhlIGlucHV0LlxuICAgIGNvbnN0IHRvRGlzcGxheSA9IHRoaXMuZGlzcGxheVdpdGg/LihvcHRpb24udmFsdWUgYXMgVCkgPz8gb3B0aW9uLnZhbHVlO1xuXG4gICAgLy8gU2V0IHRoZSBvcHRpb24gdmFsdWVcbiAgICAvLyBJbiBvcmRlciB0byBzdXBwb3J0IFJlYWN0IG9uQ2hhbmdlIGV2ZW50LCB3ZSBoYXZlIHRvIGdldCB0aGUgc2V0dGVyIGFuZCBjYWxsIGl0LlxuICAgIC8vIGh0dHBzOi8vZ2l0aHViLmNvbS9mYWNlYm9vay9yZWFjdC9pc3N1ZXMvMTE2MDAjaXNzdWVjb21tZW50LTM0NTgxMzEzMFxuICAgIGNvbnN0IHNldFZhbHVlID0gT2JqZWN0LmdldE93blByb3BlcnR5RGVzY3JpcHRvcihIVE1MSW5wdXRFbGVtZW50LnByb3RvdHlwZSwgJ3ZhbHVlJykhLnNldCE7XG4gICAgc2V0VmFsdWUuY2FsbCh0aGlzLnRyaWdnZXJFbGVtZW50LCB0b0Rpc3BsYXkpO1xuICB9XG5cbiAgcHJpdmF0ZSBfaGFuZGxlU2xvdGNoYW5nZSgpOiB2b2lkIHtcbiAgICB0aGlzLl9oaWdobGlnaHRPcHRpb25zKHRoaXMudHJpZ2dlckVsZW1lbnQ/LnZhbHVlKTtcblxuICAgIC8qKlxuICAgICAqIEl0J3MgcG9zc2libGUgdG8gZmlsdGVyIG91dCBvcHRpb25zIHdpdGggYW4gb3BlbmVkIHBhbmVsIG9uIGlucHV0IGNoYW5nZS5cbiAgICAgKiBJbiB0aGlzIGNhc2UsIHRoZSBwYW5lbCdzIHBvc2l0aW9uIG11c3QgYmUgcmVjYWxjdWxhdGVkIGNvbnNpZGVyaW5nIHRoZSBuZXcgb3B0aW9uJ3MgbGlzdC5cbiAgICAgKi9cbiAgICBpZiAodGhpcy5pc09wZW4pIHtcbiAgICAgIHRoaXMuX3NldE92ZXJsYXlQb3NpdGlvbigpO1xuICAgICAgdGhpcy5fc2V0TmV4dEFjdGl2ZU9wdGlvbklmQXV0b0FjdGl2ZUZpcnN0T3B0aW9uKCk7XG5cbiAgICAgIC8vIElmIHRoZSBhdXRvY29tcGxldGUgaXMgb3BlbiBhbmQgdGhlIG9wdGlvbiBjb3VudCBnZXRzIHRvIHplcm8sIHdlIGNsb3NlIHRoZSBhdXRvY29tcGxldGUuXG4gICAgICBpZiAodGhpcy5fb3B0aW9uc0NvdW50ID4gMCAmJiB0aGlzLm9wdGlvbnMubGVuZ3RoID09PSAwKSB7XG4gICAgICAgIHRoaXMuY2xvc2UoKTtcbiAgICAgIH1cbiAgICB9IGVsc2UgaWYgKFxuICAgICAgLy8gSWYgdGhlICdpbnB1dCcgaXMgZm9jdXNlZCBhbmQgdGhlIGNvdW50IG9mIG9wdGlvbnMgY2hhbmdlcyBmcm9tIDAgdG8gPiAwLFxuICAgICAgLy8gdGhlIGF1dG9jb21wbGV0ZSBzaG91bGQgb3BlbiBhdXRvbWF0aWNhbGx5LlxuICAgICAgZG9jdW1lbnQ/LmFjdGl2ZUVsZW1lbnQgPT09IHRoaXMudHJpZ2dlckVsZW1lbnQgJiZcbiAgICAgIHRoaXMuX29wdGlvbnNDb3VudCA9PT0gMCAmJlxuICAgICAgdGhpcy5vcHRpb25zLmxlbmd0aCA+IDBcbiAgICApIHtcbiAgICAgIHRoaXMub3BlbigpO1xuICAgIH1cblxuICAgIHRoaXMuX29wdGlvbnNDb3VudCA9IHRoaXMub3B0aW9ucy5sZW5ndGg7XG4gIH1cblxuICBwcml2YXRlIF9zZXROZXh0QWN0aXZlT3B0aW9uSWZBdXRvQWN0aXZlRmlyc3RPcHRpb24oKTogdm9pZCB7XG4gICAgaWYgKHRoaXMuYXV0b0FjdGl2ZUZpcnN0T3B0aW9uKSB7XG4gICAgICB0aGlzLnJlc2V0QWN0aXZlRWxlbWVudCgpO1xuICAgICAgdGhpcy5zZXROZXh0QWN0aXZlT3B0aW9uKCk7XG4gICAgfVxuICB9XG5cbiAgLyoqIFRoZSBhdXRvY29tcGxldGUgc2hvdWxkIGluaGVyaXQgJ3JlYWRvbmx5JyBzdGF0ZSBmcm9tIHRoZSB0cmlnZ2VyLiAqL1xuICBwcml2YXRlIF9yZWFkb25seSgpOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy50cmlnZ2VyRWxlbWVudD8uaGFzQXR0cmlidXRlKCdyZWFkb25seScpID8/IGZhbHNlO1xuICB9XG5cbiAgcHJpdmF0ZSBfY29tcG9uZW50U2V0dXAoKTogdm9pZCB7XG4gICAgaWYgKGlzU2VydmVyKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgdGhpcy5fY29uZmlndXJlVHJpZ2dlcigpO1xuICB9XG5cbiAgcHJpdmF0ZSBfY29uZmlndXJlVHJpZ2dlcigpOiB2b2lkIHtcbiAgICBjb25zdCB0cmlnZ2VyRWxlbWVudCA9ICh0aGlzLnRyaWdnZXIgPz9cbiAgICAgIHRoaXMuY2xvc2VzdD8uKCdzYmItZm9ybS1maWVsZCcpPy5xdWVyeVNlbGVjdG9yKCdpbnB1dCcpKSBhcyBIVE1MSW5wdXRFbGVtZW50IHwgbnVsbDtcbiAgICBpZiAodHJpZ2dlckVsZW1lbnQgPT09IHRoaXMudHJpZ2dlckVsZW1lbnQpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICB0aGlzLl90cmlnZ2VyQWJvcnRDb250cm9sbGVyPy5hYm9ydCgpO1xuICAgIHJlbW92ZUFyaWFDb21ib0JveEF0dHJpYnV0ZXModGhpcy50cmlnZ2VyRWxlbWVudCk7XG4gICAgdGhpcy50cmlnZ2VyRWxlbWVudD8ucmVtb3ZlQXR0cmlidXRlKCdkYXRhLWV4cGFuZGVkJyk7XG4gICAgdGhpcy5fdHJpZ2dlckVsZW1lbnQgPSB0cmlnZ2VyRWxlbWVudDtcblxuICAgIGlmICghdGhpcy50cmlnZ2VyRWxlbWVudCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIC8vIEFzIHRoZSB0cmlnZ2VyIGNhbiBiZSB0aGUgZmFsbGJhY2sgb2YgdGhlIG9yaWdpbiwgd2UgZXZlbnR1YWxseSBoYXZlIHRvIHVwZGF0ZSB0aGUgcG9zaXRpb24uXG4gICAgY29uc3Qgb3JpZ2luRWxlbWVudCA9IHRoaXMub3JpZ2luRWxlbWVudDtcbiAgICBpZiAodGhpcy50cmlnZ2VyRWxlbWVudCA9PT0gb3JpZ2luRWxlbWVudCAmJiB0aGlzLmlzT3Blbikge1xuICAgICAgdGhpcy5fc2V0T3ZlcmxheVBvc2l0aW9uKG9yaWdpbkVsZW1lbnQpO1xuICAgIH1cblxuICAgIHRoaXMuX3RyaWdnZXJBdHRyaWJ1dGVPYnNlcnZlcj8ub2JzZXJ2ZSh0aGlzLnRyaWdnZXJFbGVtZW50ISwge1xuICAgICAgYXR0cmlidXRlczogdHJ1ZSxcbiAgICAgIGF0dHJpYnV0ZUZpbHRlcjogWydkaXNhYmxlZCcsICdyZWFkb25seSddLFxuICAgIH0pO1xuXG4gICAgdGhpcy5zZXRUcmlnZ2VyQXR0cmlidXRlcyh0aGlzLnRyaWdnZXJFbGVtZW50KTtcbiAgICB0aGlzLl90cmlnZ2VyQWJvcnRDb250cm9sbGVyID0gbmV3IEFib3J0Q29udHJvbGxlcigpO1xuXG4gICAgLy8gT3BlbiB0aGUgb3ZlcmxheSBvbiBmb2N1cywgY2xpY2ssIGlucHV0IGFuZCBgQXJyb3dEb3duYCBldmVudFxuICAgIHRoaXMudHJpZ2dlckVsZW1lbnQuYWRkRXZlbnRMaXN0ZW5lcignZm9jdXMnLCAoKSA9PiB0aGlzLm9wZW4oKSwge1xuICAgICAgc2lnbmFsOiB0aGlzLl90cmlnZ2VyQWJvcnRDb250cm9sbGVyLnNpZ25hbCxcbiAgICB9KTtcbiAgICB0aGlzLnRyaWdnZXJFbGVtZW50LmFkZEV2ZW50TGlzdGVuZXIoJ2NsaWNrJywgKCkgPT4gdGhpcy5vcGVuKCksIHtcbiAgICAgIHNpZ25hbDogdGhpcy5fdHJpZ2dlckFib3J0Q29udHJvbGxlci5zaWduYWwsXG4gICAgfSk7XG4gICAgdGhpcy50cmlnZ2VyRWxlbWVudC5hZGRFdmVudExpc3RlbmVyKFxuICAgICAgJ2lucHV0JyxcbiAgICAgIChldmVudCkgPT4ge1xuICAgICAgICBjb25zdCB2YWx1ZTogc3RyaW5nID0gKGV2ZW50LnRhcmdldCBhcyBIVE1MSW5wdXRFbGVtZW50KS52YWx1ZTtcblxuICAgICAgICBpZiAodmFsdWUpIHtcbiAgICAgICAgICB0aGlzLm9wZW4oKTtcbiAgICAgICAgfVxuICAgICAgICB0aGlzLl9oaWdobGlnaHRPcHRpb25zKHZhbHVlKTtcbiAgICAgICAgdGhpcy5fbGFzdFVzZXJJbnB1dCA9IHZhbHVlO1xuICAgICAgICB0aGlzLnBlbmRpbmdBdXRvU2VsZWN0ZWRPcHRpb24gPSBudWxsO1xuICAgICAgfSxcbiAgICAgIHsgc2lnbmFsOiB0aGlzLl90cmlnZ2VyQWJvcnRDb250cm9sbGVyLnNpZ25hbCB9LFxuICAgICk7XG4gICAgdGhpcy50cmlnZ2VyRWxlbWVudC5hZGRFdmVudExpc3RlbmVyKFxuICAgICAgJ2NoYW5nZScsXG4gICAgICAoZXZlbnQpID0+IHtcbiAgICAgICAgLyoqXG4gICAgICAgICAqIEluICdyZXF1aXJlU2VsZWN0aW9uJyBtb2RlLCB3ZSBibG9jayB0aGUgbmF0aXZlIGNoYW5nZSBldmVudHMgYW5kXG4gICAgICAgICAqIGxldCBvbmx5IHBhc3MgdGhlIG9uZXMgdGhhdCBjb21lIHdpdGggYSB2YWxpZCB2YWx1ZSAod2hlbiBhbiBvcHRpb24gaXMgc2VsZWN0ZWQpXG4gICAgICAgICAqL1xuICAgICAgICBpZiAodGhpcy5yZXF1aXJlU2VsZWN0aW9uICYmICF0aGlzLl9pc0N1c3RvbUNoYW5nZUV2ZW50KSB7XG4gICAgICAgICAgZXZlbnQuc3RvcEltbWVkaWF0ZVByb3BhZ2F0aW9uKCk7XG4gICAgICAgIH1cbiAgICAgICAgdGhpcy5faXNDdXN0b21DaGFuZ2VFdmVudCA9IGZhbHNlO1xuICAgICAgfSxcbiAgICAgIHsgc2lnbmFsOiB0aGlzLl90cmlnZ2VyQWJvcnRDb250cm9sbGVyLnNpZ25hbCwgY2FwdHVyZTogdHJ1ZSB9LFxuICAgICk7XG4gICAgdGhpcy50cmlnZ2VyRWxlbWVudC5hZGRFdmVudExpc3RlbmVyKFxuICAgICAgJ2tleWRvd24nLFxuICAgICAgKGV2ZW50OiBLZXlib2FyZEV2ZW50KSA9PiB0aGlzLl9jbG9zZWRQYW5lbEtleWJvYXJkSW50ZXJhY3Rpb24oZXZlbnQpLFxuICAgICAge1xuICAgICAgICBzaWduYWw6IHRoaXMuX3RyaWdnZXJBYm9ydENvbnRyb2xsZXIuc2lnbmFsLFxuICAgICAgICAvLyBXZSBuZWVkIGtleSBldmVudCB0byBydW4gYmVmb3JlIGFueSBvdGhlciBzdWJzY3JpcHRpb24gdG8gZ3VhcmFudGVlIGEgY29ycmVjdFxuICAgICAgICAvLyBpbnRlcmFjdGlvbiB3aXRoIG90aGVyIGNvbXBvbmVudHMgKG5lY2Vzc2FyeSBmb3IgdGhlICdzYmItY2hpcC1ncm91cCcgdXNlIGNhc2UpLlxuICAgICAgICBjYXB0dXJlOiB0cnVlLFxuICAgICAgfSxcbiAgICApO1xuXG4gICAgdGhpcy50cmlnZ2VyRWxlbWVudC5hZGRFdmVudExpc3RlbmVyKFxuICAgICAgJ2JsdXInLFxuICAgICAgKGUpID0+IHtcbiAgICAgICAgLy8gSWYgdGhlIG5ldyBmb2N1cyBpcyB0aGUgYXV0b2NvbXBsZXRlIG9yIGluc2lkZSBvZiBpdCB0aGVuIGFuIG9wdGlvblxuICAgICAgICAvLyB3YXMgc2VsZWN0ZWQuIFRoZXJlZm9yZSwgdGhlIGZvY3VzIGlzIHN0aWxsIG9uIHRoZSBjb21wb25lbnQuXG4gICAgICAgIGlmICh0aGlzLmNvbnRhaW5zKGUucmVsYXRlZFRhcmdldCBhcyBOb2RlKSkge1xuICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIElmICdhdXRvU2VsZWN0QWN0aXZlT3B0aW9uT25CbHVyJyBpcyBlbmFibGVkLCBzZWxlY3QgdGhlIGFjdGl2ZSBvcHRpb24gb24gYmx1clxuICAgICAgICBpZiAoXG4gICAgICAgICAgdGhpcy5hdXRvU2VsZWN0QWN0aXZlT3B0aW9uT25CbHVyICYmXG4gICAgICAgICAgdGhpcy5hY3RpdmVPcHRpb24gJiZcbiAgICAgICAgICB0aGlzLl9sYXN0VXNlcklucHV0ICYmXG4gICAgICAgICAgdGhpcy50cmlnZ2VyRWxlbWVudD8udmFsdWVcbiAgICAgICAgKSB7XG4gICAgICAgICAgdGhpcy5hY3RpdmVPcHRpb24uc2VsZWN0ZWQgPSB0cnVlO1xuICAgICAgICAgIHRoaXMuX3NldFZhbHVlQW5kRGlzcGF0Y2hFdmVudHModGhpcy5hY3RpdmVPcHRpb24sIHRydWUpO1xuICAgICAgICB9XG5cbiAgICAgICAgLy8gQ2xlYXJzIHRoZSBpbnB1dCBpZiB0aGVyZSdzIHVzZXIgaW50ZXJhY3Rpb24gd2l0aG91dCBzZWxlY3Rpb24gKHNlbGVjdGlvbiBjbGVhcnMgYF9sYXN0VXNlcklucHV0YCkuXG4gICAgICAgIGlmICh0aGlzLnJlcXVpcmVTZWxlY3Rpb24gJiYgdGhpcy50cmlnZ2VyRWxlbWVudCAmJiB0aGlzLl9sYXN0VXNlcklucHV0ICE9IG51bGwpIHtcbiAgICAgICAgICBjb25zdCBzZXRWYWx1ZSA9IE9iamVjdC5nZXRPd25Qcm9wZXJ0eURlc2NyaXB0b3IoXG4gICAgICAgICAgICBIVE1MSW5wdXRFbGVtZW50LnByb3RvdHlwZSxcbiAgICAgICAgICAgICd2YWx1ZScsXG4gICAgICAgICAgKSEuc2V0ITtcbiAgICAgICAgICBzZXRWYWx1ZS5jYWxsKHRoaXMudHJpZ2dlckVsZW1lbnQsICcnKTtcbiAgICAgICAgICB0aGlzLl9oaWdobGlnaHRPcHRpb25zKCcnKTtcbiAgICAgICAgICB0aGlzLl9pc0N1c3RvbUNoYW5nZUV2ZW50ID0gdHJ1ZTtcbiAgICAgICAgICB0aGlzLnRyaWdnZXJFbGVtZW50LmRpc3BhdGNoRXZlbnQobmV3IEV2ZW50KCdjaGFuZ2UnLCB7IGJ1YmJsZXM6IHRydWUgfSkpO1xuICAgICAgICAgIHRoaXMudHJpZ2dlckVsZW1lbnQuZGlzcGF0Y2hFdmVudChcbiAgICAgICAgICAgIG5ldyBJbnB1dEV2ZW50KCdpbnB1dCcsIHsgYnViYmxlczogdHJ1ZSwgY29tcG9zZWQ6IHRydWUgfSksXG4gICAgICAgICAgKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHRoaXMuY2xvc2UoKTtcbiAgICAgIH0sXG4gICAgICB7IHNpZ25hbDogdGhpcy5fdHJpZ2dlckFib3J0Q29udHJvbGxlci5zaWduYWwsIGNhcHR1cmU6IHRydWUgfSxcbiAgICApO1xuICB9XG5cbiAgLy8gU2V0IG92ZXJsYXkgcG9zaXRpb24sIHdpZHRoIGFuZCBtYXggaGVpZ2h0XG4gIHByaXZhdGUgX3NldE92ZXJsYXlQb3NpdGlvbihvcmlnaW5FbGVtZW50ID0gdGhpcy5vcmlnaW5FbGVtZW50KTogdm9pZCB7XG4gICAgLy8gQW4gdW5kZWZpbmVkIG9yaWdpbkVsZW1lbnQgc2hvdWxkIG9ubHkgb2NjdXIgaW4gdGhlIHVubGlrZWx5IGV2ZW50XG4gICAgLy8gdGhhdCB0aGUgYXV0b2NvbXBsZXRlIGxvc2VzIGl0cyBvcmlnaW5FbGVtZW50IGFuZCB0cmlnZ2VyRWxlbWVudCBkdXJpbmcgYW4gb3BlbiBzdGF0ZS5cbiAgICBpZiAoIW9yaWdpbkVsZW1lbnQpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgc2V0T3ZlcmxheVBvc2l0aW9uKFxuICAgICAgdGhpcy5fb3ZlcmxheSxcbiAgICAgIG9yaWdpbkVsZW1lbnQsXG4gICAgICB0aGlzLl9vcHRpb25Db250YWluZXIsXG4gICAgICB0aGlzLnNoYWRvd1Jvb3QhLnF1ZXJ5U2VsZWN0b3IoJy5zYmItYXV0b2NvbXBsZXRlX19jb250YWluZXInKSEsXG4gICAgICB0aGlzLFxuICAgICAgdGhpcy5wb3NpdGlvbixcbiAgICApO1xuICB9XG5cbiAgLyoqXG4gICAqIE9uIG9wZW4vY2xvc2UgYW5pbWF0aW9uIGVuZC5cbiAgICogSW4gcmFyZSBjYXNlcyBpdCBjYW4gYmUgdGhhdCB0aGUgYW5pbWF0aW9uRW5kIGV2ZW50IGlzIHRyaWdnZXJlZCB0d2ljZS5cbiAgICogVG8gYXZvaWQgZW50ZXJpbmcgYSBjb3JydXB0IHN0YXRlLCBleGl0IHdoZW4gc3RhdGUgaXMgbm90IGV4cGVjdGVkLlxuICAgKi9cbiAgcHJpdmF0ZSBfb25BbmltYXRpb25FbmQoZXZlbnQ6IEFuaW1hdGlvbkV2ZW50KTogdm9pZCB7XG4gICAgaWYgKGV2ZW50LmFuaW1hdGlvbk5hbWUgPT09ICdvcGVuJyAmJiB0aGlzLnN0YXRlID09PSAnb3BlbmluZycpIHtcbiAgICAgIHRoaXMuX2hhbmRsZU9wZW5pbmcoKTtcbiAgICB9IGVsc2UgaWYgKGV2ZW50LmFuaW1hdGlvbk5hbWUgPT09ICdjbG9zZScgJiYgdGhpcy5zdGF0ZSA9PT0gJ2Nsb3NpbmcnKSB7XG4gICAgICB0aGlzLl9oYW5kbGVDbG9zaW5nKCk7XG4gICAgfVxuICB9XG5cbiAgcHJpdmF0ZSBfaGFuZGxlT3BlbmluZygpOiB2b2lkIHtcbiAgICB0aGlzLnN0YXRlID0gJ29wZW5lZCc7XG4gICAgaWYgKHRoaXMub3JpZ2luRWxlbWVudCkge1xuICAgICAgdGhpcy5fb3JpZ2luUmVzaXplT2JzZXJ2ZXIub2JzZXJ2ZSh0aGlzLm9yaWdpbkVsZW1lbnQpO1xuICAgIH1cbiAgICB0aGlzLnRyaWdnZXJFbGVtZW50Py5zZXRBdHRyaWJ1dGUoJ2FyaWEtZXhwYW5kZWQnLCAndHJ1ZScpO1xuICAgIHRoaXMuZGlzcGF0Y2hPcGVuRXZlbnQoKTtcbiAgfVxuXG4gIHByaXZhdGUgX2hhbmRsZUNsb3NpbmcoKTogdm9pZCB7XG4gICAgdGhpcy5zdGF0ZSA9ICdjbG9zZWQnO1xuICAgIHRoaXMuaGlkZVBvcG92ZXI/LigpO1xuICAgIHRoaXMudHJpZ2dlckVsZW1lbnQ/LnNldEF0dHJpYnV0ZSgnYXJpYS1leHBhbmRlZCcsICdmYWxzZScpO1xuXG4gICAgdGhpcy5yZXNldEFjdGl2ZUVsZW1lbnQoKTtcbiAgICB0aGlzLl9vcHRpb25Db250YWluZXIuc2Nyb2xsVG9wID0gMDtcbiAgICB0aGlzLl9lc2NhcGFibGVPdmVybGF5Q29udHJvbGxlci5kaXNjb25uZWN0KCk7XG4gICAgdGhpcy5kaXNwYXRjaENsb3NlRXZlbnQoKTtcbiAgfVxuXG4gIHByaXZhdGUgX2F0dGFjaE9wZW5QYW5lbEV2ZW50cygpOiB2b2lkIHtcbiAgICB0aGlzLl9vcGVuUGFuZWxFdmVudHNDb250cm9sbGVyID0gbmV3IEFib3J0Q29udHJvbGxlcigpO1xuXG4gICAgLy8gUmVjYWxjdWxhdGUgdGhlIG92ZXJsYXkgcG9zaXRpb24gb24gc2Nyb2xsIGFuZCB3aW5kb3cgcmVzaXplXG4gICAgZG9jdW1lbnQuYWRkRXZlbnRMaXN0ZW5lcignc2Nyb2xsJywgKCkgPT4gdGhpcy5fc2V0T3ZlcmxheVBvc2l0aW9uKCksIHtcbiAgICAgIHBhc3NpdmU6IHRydWUsXG4gICAgICBzaWduYWw6IHRoaXMuX29wZW5QYW5lbEV2ZW50c0NvbnRyb2xsZXIuc2lnbmFsLFxuICAgICAgLy8gV2l0aG91dCBjYXB0dXJlLCBvdGhlciBzY3JvbGwgY29udGV4dHMgd291bGQgbm90IGJ1YmJsZSB0byB0aGlzIGV2ZW50IGxpc3RlbmVyLlxuICAgICAgLy8gQ2FwdHVyZSBhbGxvd3MgdXMgdG8gcmVhY3QgdG8gYWxsIHNjcm9sbCBjb250ZXh0cyBpbiB0aGlzIERPTS5cbiAgICAgIGNhcHR1cmU6IHRydWUsXG4gICAgfSk7XG4gICAgd2luZG93LmFkZEV2ZW50TGlzdGVuZXIoJ3Jlc2l6ZScsICgpID0+IHRoaXMuX3NldE92ZXJsYXlQb3NpdGlvbigpLCB7XG4gICAgICBwYXNzaXZlOiB0cnVlLFxuICAgICAgc2lnbmFsOiB0aGlzLl9vcGVuUGFuZWxFdmVudHNDb250cm9sbGVyLnNpZ25hbCxcbiAgICB9KTtcblxuICAgIC8vIENsb3NlIGF1dG9jb21wbGV0ZSBvbiBiYWNrZHJvcCBjbGlja1xuICAgIHdpbmRvdy5hZGRFdmVudExpc3RlbmVyKCdwb2ludGVyZG93bicsIChldikgPT4gdGhpcy5fcG9pbnRlckRvd25MaXN0ZW5lcihldiksIHtcbiAgICAgIHNpZ25hbDogdGhpcy5fb3BlblBhbmVsRXZlbnRzQ29udHJvbGxlci5zaWduYWwsXG4gICAgfSk7XG4gICAgd2luZG93LmFkZEV2ZW50TGlzdGVuZXIoJ3BvaW50ZXJ1cCcsIChldikgPT4gdGhpcy5fY2xvc2VPbkJhY2tkcm9wQ2xpY2soZXYpLCB7XG4gICAgICBzaWduYWw6IHRoaXMuX29wZW5QYW5lbEV2ZW50c0NvbnRyb2xsZXIuc2lnbmFsLFxuICAgIH0pO1xuXG4gICAgdGhpcy5hZGRFdmVudExpc3RlbmVyKFxuICAgICAgJ8m1ZGlzYWJsZWRjaGFuZ2UnLFxuICAgICAgKCkgPT4ge1xuICAgICAgICBpZiAodGhpcy5hY3RpdmVPcHRpb24/LmRpc2FibGVkKSB7XG4gICAgICAgICAgdGhpcy5yZXNldEFjdGl2ZUVsZW1lbnQoKTtcbiAgICAgICAgfVxuICAgICAgICB0aGlzLl9zZXROZXh0QWN0aXZlT3B0aW9uSWZBdXRvQWN0aXZlRmlyc3RPcHRpb24oKTtcbiAgICAgIH0sXG4gICAgICB7XG4gICAgICAgIHNpZ25hbDogdGhpcy5fb3BlblBhbmVsRXZlbnRzQ29udHJvbGxlci5zaWduYWwsXG4gICAgICB9LFxuICAgICk7XG5cbiAgICAvLyBLZXlib2FyZCBpbnRlcmFjdGlvbnNcbiAgICB0aGlzLnRyaWdnZXJFbGVtZW50Py5hZGRFdmVudExpc3RlbmVyKFxuICAgICAgJ2tleWRvd24nLFxuICAgICAgKGV2ZW50OiBLZXlib2FyZEV2ZW50KSA9PiB0aGlzLm9wZW5lZFBhbmVsS2V5Ym9hcmRJbnRlcmFjdGlvbihldmVudCksXG4gICAgICB7XG4gICAgICAgIHNpZ25hbDogdGhpcy5fb3BlblBhbmVsRXZlbnRzQ29udHJvbGxlci5zaWduYWwsXG4gICAgICAgIC8vIFdlIG5lZWQga2V5IGV2ZW50IHRvIHJ1biBiZWZvcmUgYW55IG90aGVyIHN1YnNjcmlwdGlvbiB0byBndWFyYW50ZWUgYSBjb3JyZWN0XG4gICAgICAgIC8vIGludGVyYWN0aW9uIHdpdGggb3RoZXIgY29tcG9uZW50cyAobmVjZXNzYXJ5IGZvciB0aGUgJ3NiYi1jaGlwLWdyb3VwJyB1c2UgY2FzZSkuXG4gICAgICAgIGNhcHR1cmU6IHRydWUsXG4gICAgICB9LFxuICAgICk7XG4gIH1cblxuICAvLyBDaGVjayBpZiB0aGUgcG9pbnRlcmRvd24gZXZlbnQgdGFyZ2V0IGlzIHRyaWdnZXJlZCBvbiB0aGUgbWVudS5cbiAgcHJpdmF0ZSBfcG9pbnRlckRvd25MaXN0ZW5lciA9IChldmVudDogUG9pbnRlckV2ZW50KTogdm9pZCA9PiB7XG4gICAgdGhpcy5faXNQb2ludGVyRG93bkV2ZW50T25NZW51ID0gaXNFdmVudE9uRWxlbWVudCh0aGlzLl9vdmVybGF5LCBldmVudCk7XG4gIH07XG5cbiAgLy8gSWYgdGhlIGNsaWNrIGlzIG91dHNpZGUgdGhlIGF1dG9jb21wbGV0ZSwgY2xvc2VzIHRoZSBwYW5lbC5cbiAgcHJpdmF0ZSBfY2xvc2VPbkJhY2tkcm9wQ2xpY2sgPSAoZXZlbnQ6IFBvaW50ZXJFdmVudCk6IHZvaWQgPT4ge1xuICAgIGlmIChcbiAgICAgICF0aGlzLl9pc1BvaW50ZXJEb3duRXZlbnRPbk1lbnUgJiZcbiAgICAgICFpc0V2ZW50T25FbGVtZW50KHRoaXMuX292ZXJsYXksIGV2ZW50KSAmJlxuICAgICAgIWlzRXZlbnRPbkVsZW1lbnQodGhpcy5vcmlnaW5FbGVtZW50LCBldmVudClcbiAgICApIHtcbiAgICAgIHRoaXMuY2xvc2UoKTtcbiAgICB9XG4gIH07XG5cbiAgcHJpdmF0ZSBfY2xvc2VkUGFuZWxLZXlib2FyZEludGVyYWN0aW9uKGV2ZW50OiBLZXlib2FyZEV2ZW50KTogdm9pZCB7XG4gICAgaWYgKHRoaXMuc3RhdGUgPT09ICdvcGVuaW5nJyB8fCB0aGlzLnN0YXRlID09PSAnb3BlbmVkJykge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIHN3aXRjaCAoZXZlbnQua2V5KSB7XG4gICAgICBjYXNlICdBcnJvd0Rvd24nOlxuICAgICAgY2FzZSAnQXJyb3dVcCc6XG4gICAgICAgIHRoaXMub3BlbigpO1xuICAgICAgICBicmVhaztcbiAgICB9XG4gIH1cblxuICAvKiogSGlnaGxpZ2h0IHRoZSBzZWFyY2hlZCB0ZXh0IG9uIHRoZSBvcHRpb25zLiAqL1xuICBwcml2YXRlIF9oaWdobGlnaHRPcHRpb25zKHNlYXJjaFRlcm0/OiBzdHJpbmcpOiB2b2lkIHtcbiAgICBpZiAoc2VhcmNoVGVybSA9PT0gbnVsbCB8fCBzZWFyY2hUZXJtID09PSB1bmRlZmluZWQpIHtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gICAgdGhpcy5vcHRpb25zLmZvckVhY2goKG9wdGlvbikgPT4gb3B0aW9uLmhpZ2hsaWdodChzZWFyY2hUZXJtKSk7XG4gIH1cblxuICBwcm90ZWN0ZWQgb3ZlcnJpZGUgcmVuZGVyKCk6IFRlbXBsYXRlUmVzdWx0IHtcbiAgICAvLyBTY3JvbGwgYXJlYXMgd2l0aG91dCBjb250YWluaW5nIGFuIGludGVyYWN0aXZlIGVsZW1lbnQgd2lsbCByZWNlaXZlIGZvY3VzIHdoZW4gdGFiYmluZyB0aHJvdWdoIHRoZSBkb2N1bWVudC5cbiAgICAvLyBJZiB0aGVyZSBhcmUgYSBsb3Qgb2Ygb3B0aW9ucyBhbmQgd2hlbiBwcmVzc2luZyB0YWIga2V5LCB0aGUgc2Nyb2xsIGFyZWEgb24gc2JiLWF1dG9jb21wbGV0ZV9fb3B0aW9ucyBnZXRzIGZvY3VzLlxuICAgIC8vIEFzIGVsZW1lbnRzIGluc2lkZSB0aGUgcGFuZWwgc2hvdWxkIG5ldmVyIGdldCBmb2N1cywgd2UgaGF2ZSB0byBhdm9pZCB0aGF0IGJ5IHNldHRpbmcgdGFiaW5kZXg9LTEuXG4gICAgcmV0dXJuIGh0bWxgXG4gICAgICA8ZGl2IGNsYXNzPVwic2JiLWF1dG9jb21wbGV0ZV9fZ2FwLWZpeFwiPjwvZGl2PlxuICAgICAgPGRpdiBjbGFzcz1cInNiYi1hdXRvY29tcGxldGVfX2NvbnRhaW5lclwiPlxuICAgICAgICA8ZGl2IGNsYXNzPVwic2JiLWF1dG9jb21wbGV0ZV9fZ2FwLWZpeFwiPiR7b3ZlcmxheUdhcEZpeENvcm5lcnMoKX08L2Rpdj5cbiAgICAgICAgPGRpdlxuICAgICAgICAgIEBhbmltYXRpb25lbmQ9JHt0aGlzLl9vbkFuaW1hdGlvbkVuZH1cbiAgICAgICAgICBjbGFzcz1cInNiYi1hdXRvY29tcGxldGVfX3BhbmVsXCJcbiAgICAgICAgICAke3JlZigob3ZlcmxheVJlZj86IEVsZW1lbnQpID0+ICh0aGlzLl9vdmVybGF5ID0gb3ZlcmxheVJlZiBhcyBIVE1MRWxlbWVudCkpfVxuICAgICAgICA+XG4gICAgICAgICAgPGRpdiBjbGFzcz1cInNiYi1hdXRvY29tcGxldGVfX3dyYXBwZXJcIj5cbiAgICAgICAgICAgIDxkaXZcbiAgICAgICAgICAgICAgY2xhc3M9XCJzYmItYXV0b2NvbXBsZXRlX19vcHRpb25zXCJcbiAgICAgICAgICAgICAgcm9sZT0keyFhcmlhUm9sZU9uSG9zdCA/IHRoaXMucGFuZWxSb2xlIDogbm90aGluZ31cbiAgICAgICAgICAgICAgaWQ9JHshYXJpYVJvbGVPbkhvc3QgPyB0aGlzLm92ZXJsYXlJZCA6IG5vdGhpbmd9XG4gICAgICAgICAgICAgIHRhYmluZGV4PVwiLTFcIlxuICAgICAgICAgICAgICAke3JlZigoY29udGFpbmVyUmVmKSA9PiAodGhpcy5fb3B0aW9uQ29udGFpbmVyID0gY29udGFpbmVyUmVmIGFzIEhUTUxFbGVtZW50KSl9XG4gICAgICAgICAgICA+XG4gICAgICAgICAgICAgIDxzbG90IEBzbG90Y2hhbmdlPSR7dGhpcy5faGFuZGxlU2xvdGNoYW5nZX0+PC9zbG90PlxuICAgICAgICAgICAgPC9kaXY+XG4gICAgICAgICAgPC9kaXY+XG4gICAgICAgIDwvZGl2PlxuICAgICAgPC9kaXY+XG4gICAgYDtcbiAgfVxufVxuXG5kZWNsYXJlIGdsb2JhbCB7XG4gIGludGVyZmFjZSBIVE1MRWxlbWVudEV2ZW50TWFwIHtcbiAgICBpbnB1dEF1dG9jb21wbGV0ZTogQ3VzdG9tRXZlbnQ8eyBvcHRpb246IFNiYk9wdGlvbkJhc2VFbGVtZW50PGFueT4gfT47XG4gIH1cbn1cbiJdLCJtYXBwaW5ncyI6Ijs7Ozs7Ozs7Ozs7Ozs7QUN1Q0EsSUFBTSxpQkFBaUI7SUFFRCxvQ0FBMEI7bUJBQXFCLGlCQUNuRSx3QkFDRDs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztjQUZxQixtQ0FBK0MsWUFFcEU7Ozt5QkFXRSxhQUFhLEVBQ2IsVUFBVSxDQUFBOzBCQVVWLGFBQWEsRUFDYixVQUFVLENBQUE7b0NBSVYsV0FBVyxFQUNYLFNBQVM7SUFBRSxXQUFXO0lBQXVCLFNBQVM7SUFBTSxNQUFNO0lBQVMsQ0FBQyxDQUFBO3dDQUk1RSxXQUFXLEVBQ1gsU0FBUztJQUFFLFdBQVc7SUFBNEIsTUFBTTtJQUFTLENBQUMsQ0FBQTs4QkFJbEUsU0FBUyxFQUFFLFdBQVcsT0FBTyxDQUFDLENBQUE7dUJBTzlCLFNBQVMsRUFBRSxTQUFTLE1BQU0sQ0FBQyxDQUFBO3lDQUczQixXQUFXLEVBQ1gsU0FBUztJQUFFLFdBQVc7SUFBNkIsTUFBTTtJQUFTLENBQUMsQ0FBQTsrQ0FPbkUsV0FBVyxFQUNYLFNBQVM7SUFBRSxXQUFXO0lBQXFDLE1BQU07SUFBUyxDQUFDLENBQUE7bUNBUzNFLFdBQVcsRUFDWCxTQUFTO0lBQUUsV0FBVztJQUFxQixNQUFNO0lBQVMsQ0FBQyxDQUFBOzJCQU8zRCxV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