wxt 0.8.4 → 0.8.6

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 (36) hide show
  1. package/README.md +3 -2
  2. package/dist/browser.js +3 -4
  3. package/dist/chunk-FNTE2L27.js +7 -0
  4. package/dist/chunk-VFZ5667B.js +2410 -0
  5. package/dist/chunk-YUG22S6W.js +38 -0
  6. package/dist/cli.cjs +3941 -3775
  7. package/dist/cli.d.cts +2 -0
  8. package/dist/client.d.ts +5 -171
  9. package/dist/client.js +128 -131
  10. package/dist/execa-WKZHVHC5.js +2043 -0
  11. package/dist/external-9107db91.d.ts +176 -0
  12. package/dist/external-cb0967d6.d.ts +572 -0
  13. package/dist/index.cjs +2727 -2355
  14. package/dist/index.d.cts +36 -583
  15. package/dist/index.d.ts +36 -583
  16. package/dist/index.js +330 -4484
  17. package/dist/sandbox.d.ts +2 -23
  18. package/dist/sandbox.js +1 -2
  19. package/dist/testing.cjs +161 -35
  20. package/dist/testing.d.cts +7 -273
  21. package/dist/testing.d.ts +7 -273
  22. package/dist/testing.js +12 -676
  23. package/dist/{virtual-modules → virtual}/background-entrypoint.js +6 -7
  24. package/dist/{virtual-modules → virtual}/content-script-entrypoint.js +4 -5
  25. package/dist/virtual/mock-browser.js +152 -0
  26. package/dist/{virtual-modules → virtual}/reload-html.js +2 -3
  27. package/dist/{virtual-modules → virtual}/unlisted-script-entrypoint.js +2 -3
  28. package/package.json +6 -5
  29. package/dist/index.cjs.map +0 -1
  30. package/dist/index.js.map +0 -1
  31. package/dist/virtual-modules/background-entrypoint.js.map +0 -1
  32. package/dist/virtual-modules/content-script-entrypoint.js.map +0 -1
  33. package/dist/virtual-modules/fake-browser.cjs +0 -31
  34. package/dist/virtual-modules/fake-browser.js +0 -8
  35. package/dist/virtual-modules/reload-html.js.map +0 -1
  36. package/dist/virtual-modules/unlisted-script-entrypoint.js.map +0 -1
package/dist/cli.d.cts ADDED
@@ -0,0 +1,2 @@
1
+
2
+ export { }
package/dist/client.d.ts CHANGED
@@ -1,173 +1,5 @@
1
- import { Manifest } from 'webextension-polyfill';
2
-
3
- /**
4
- * Implements [`AbortController`](https://developer.mozilla.org/en-US/docs/Web/API/AbortController).
5
- * Used to detect and stop content script code when the script is invalidated.
6
- *
7
- * It also provides several utilities like `ctx.setTimeout` and `ctx.setInterval` that should be used in
8
- * content scripts instead of `window.setTimeout` or `window.setInterval`.
9
- */
10
- declare class ContentScriptContext implements AbortController {
11
- #private;
12
- private readonly contentScriptName;
13
- readonly options?: Omit<ContentScriptDefinition, "main"> | undefined;
14
- private static SCRIPT_STARTED_MESSAGE_TYPE;
15
- constructor(contentScriptName: string, options?: Omit<ContentScriptDefinition, "main"> | undefined);
16
- get signal(): AbortSignal;
17
- abort(reason?: any): void;
18
- get isInvalid(): boolean;
19
- get isValid(): boolean;
20
- /**
21
- * Add a listener that is called when the content script's context is invalidated.
22
- *
23
- * @returns A function to remove the listener.
24
- *
25
- * @example
26
- * browser.runtime.onMessage.addListener(cb);
27
- * const removeInvalidatedListener = ctx.onInvalidated(() => {
28
- * browser.runtime.onMessage.removeListener(cb);
29
- * })
30
- * // ...
31
- * removeInvalidatedListener();
32
- */
33
- onInvalidated(cb: () => void): () => void;
34
- /**
35
- * Return a promise that never resolves. Useful if you have an async function that shouldn't run
36
- * after the context is expired.
37
- *
38
- * @example
39
- * const getValueFromStorage = async () => {
40
- * if (ctx.isInvalid) return ctx.block();
41
- *
42
- * // ...
43
- * }
44
- */
45
- block<T>(): Promise<T>;
46
- /**
47
- * Wrapper around `window.setInterval` that automatically clears the interval when invalidated.
48
- */
49
- setInterval(handler: () => void, timeout?: number): number;
50
- /**
51
- * Wrapper around `window.setTimeout` that automatically clears the interval when invalidated.
52
- */
53
- setTimeout(handler: () => void, timeout?: number): number;
54
- /**
55
- * Wrapper around `window.requestAnimationFrame` that automatically cancels the request when
56
- * invalidated.
57
- */
58
- requestAnimationFrame(callback: FrameRequestCallback): number;
59
- /**
60
- * Wrapper around `window.requestIdleCallback` that automatically cancels the request when
61
- * invalidated.
62
- */
63
- requestIdleCallback(callback: IdleRequestCallback, options?: IdleRequestOptions): number;
64
- /**
65
- * Call `target.addEventListener` and remove the event listener when the context is invalidated.
66
- *
67
- * @example
68
- * ctx.addEventListener(window, "mousemove", () => {
69
- * // ...
70
- * });
71
- * ctx.addEventListener(document, "visibilitychange", () => {
72
- * // ...
73
- * });
74
- */
75
- addEventListener(target: any, type: string, handler: (event: Event) => void, options?: AddEventListenerOptions): void;
76
- /**
77
- * @internal
78
- * Abort the abort controller and execute all `onInvalidated` listeners.
79
- */
80
- notifyInvalidated(): void;
81
- }
82
-
83
- type TargetBrowser = string;
84
- interface ContentScriptDefinition extends ExcludableEntrypoint {
85
- matches: PerBrowserOption<Manifest.ContentScript['matches']>;
86
- /**
87
- * See https://developer.chrome.com/docs/extensions/mv3/content_scripts/
88
- * @default "documentIdle"
89
- */
90
- runAt?: PerBrowserOption<Manifest.ContentScript['run_at']>;
91
- /**
92
- * See https://developer.chrome.com/docs/extensions/mv3/content_scripts/
93
- * @default false
94
- */
95
- matchAboutBlank?: PerBrowserOption<Manifest.ContentScript['match_about_blank']>;
96
- /**
97
- * See https://developer.chrome.com/docs/extensions/mv3/content_scripts/
98
- * @default []
99
- */
100
- excludeMatches?: PerBrowserOption<Manifest.ContentScript['exclude_matches']>;
101
- /**
102
- * See https://developer.chrome.com/docs/extensions/mv3/content_scripts/
103
- * @default []
104
- */
105
- includeGlobs?: PerBrowserOption<Manifest.ContentScript['include_globs']>;
106
- /**
107
- * See https://developer.chrome.com/docs/extensions/mv3/content_scripts/
108
- * @default []
109
- */
110
- excludeGlobs?: PerBrowserOption<Manifest.ContentScript['exclude_globs']>;
111
- /**
112
- * See https://developer.chrome.com/docs/extensions/mv3/content_scripts/
113
- * @default false
114
- */
115
- allFrames?: PerBrowserOption<Manifest.ContentScript['all_frames']>;
116
- /**
117
- * See https://developer.chrome.com/docs/extensions/mv3/content_scripts/
118
- * @default false
119
- */
120
- matchOriginAsFallback?: PerBrowserOption<boolean>;
121
- /**
122
- * See https://developer.chrome.com/docs/extensions/mv3/content_scripts/
123
- * @default "ISOLATED"
124
- */
125
- world?: PerBrowserOption<'ISOLATED' | 'MAIN'>;
126
- /**
127
- * Customize how imported/generated styles are injected with the content script. Regardless of the
128
- * mode selected, CSS will always be built and included in the output directory.
129
- *
130
- * - `"manifest"` - Include the CSS in the manifest, under the content script's `css` array.
131
- * - `"manual"` - Exclude the CSS from the manifest. You are responsible for manually loading it
132
- * onto the page. Use `browser.runtime.getURL("content-scripts/<name>.css")` to get the file's
133
- * URL
134
- * - `"ui"` - Exclude the CSS from the manifest. CSS will be automatically added to your UI when
135
- * calling `createContentScriptUi`
136
- *
137
- * @default "manifest"
138
- */
139
- cssInjectionMode?: PerBrowserOption<'manifest' | 'manual' | 'ui'>;
140
- /**
141
- * Main function executed when the content script is loaded.
142
- */
143
- main(ctx: ContentScriptContext): void | Promise<void>;
144
- }
145
- interface BackgroundDefinition extends ExcludableEntrypoint {
146
- type?: PerBrowserOption<'module'>;
147
- persistent?: PerBrowserOption<boolean>;
148
- main(): void;
149
- }
150
- type PerBrowserOption<T> = T | {
151
- [browser: TargetBrowser]: T;
152
- };
153
- interface ExcludableEntrypoint {
154
- /**
155
- * List of target browsers to include this entrypoint in. Defaults to being included in all
156
- * builds. Cannot be used with `exclude`. You must choose one of the two options.
157
- *
158
- * @default undefined
159
- */
160
- include?: TargetBrowser[];
161
- /**
162
- * List of target browsers to exclude this entrypoint from. Cannot be used with `include`. You
163
- * must choose one of the two options.
164
- *
165
- * @default undefined
166
- */
167
- exclude?: TargetBrowser[];
168
- }
169
-
170
- declare function defineContentScript(definition: ContentScriptDefinition): ContentScriptDefinition;
1
+ import { B as BackgroundDefinition, C as ContentScriptContext, a as ContentScriptDefinition } from './external-9107db91.js';
2
+ import 'webextension-polyfill';
171
3
 
172
4
  declare function defineBackground(main: () => void): BackgroundDefinition;
173
5
  declare function defineBackground(definition: BackgroundDefinition): BackgroundDefinition;
@@ -301,4 +133,6 @@ type ContentScriptUiOverlayAlignment = 'top-left' | 'top-right' | 'bottom-left'
301
133
  type ContentScriptAppendMode = 'last' | 'first' | 'replace' | 'before' | 'after';
302
134
  type ContentScriptUiOptions<TApp> = OverlayContentScriptUiOptions<TApp> | ModalContentScriptUiOptions<TApp> | InlineContentScriptUiOptions<TApp>;
303
135
 
304
- export { ContentScriptAppendMode, ContentScriptContext, ContentScriptUi, ContentScriptUiOptions, ContentScriptUiOverlayAlignment, InlineContentScriptUiOptions, ModalContentScriptUiOptions, OverlayContentScriptUiOptions, createContentScriptUi, defineBackground, defineContentScript };
136
+ declare function defineContentScript(definition: ContentScriptDefinition): ContentScriptDefinition;
137
+
138
+ export { BaseContentScriptUiOptions, ContentScriptAppendMode, ContentScriptContext, ContentScriptUi, ContentScriptUiOptions, ContentScriptUiOverlayAlignment, InlineContentScriptUiOptions, ModalContentScriptUiOptions, OverlayContentScriptUiOptions, createContentScriptUi, defineBackground, defineContentScript };
package/dist/client.js CHANGED
@@ -1,22 +1,14 @@
1
- // src/client/defineContentScript.ts
2
- function defineContentScript(definition) {
3
- return definition;
4
- }
1
+ import {
2
+ browser
3
+ } from "./chunk-FNTE2L27.js";
5
4
 
6
- // src/client/defineBackground.ts
5
+ // src/client/define-background.ts
7
6
  function defineBackground(arg) {
8
7
  if (typeof arg === "function")
9
8
  return { main: arg };
10
9
  return arg;
11
10
  }
12
11
 
13
- // src/client/createContentScriptUi.ts
14
- import { createIsolatedElement } from "@webext-core/isolated-element";
15
-
16
- // src/client/browser.ts
17
- import originalBrowser from "webextension-polyfill";
18
- var browser = originalBrowser;
19
-
20
12
  // src/client/utils/logger.ts
21
13
  function print(method, ...args) {
22
14
  if (import.meta.env.MODE === "production")
@@ -35,124 +27,7 @@ var logger = {
35
27
  error: (...args) => print(console.error, ...args)
36
28
  };
37
29
 
38
- // src/client/createContentScriptUi.ts
39
- async function createContentScriptUi(ctx, options) {
40
- const css = [options.css ?? ""];
41
- if (ctx.options?.cssInjectionMode === "ui") {
42
- css.push(await loadCss());
43
- }
44
- const {
45
- isolatedElement: uiContainer,
46
- parentElement: shadowHost,
47
- shadow
48
- } = await createIsolatedElement({
49
- name: options.name,
50
- css: {
51
- textContent: css.join("\n").trim()
52
- },
53
- mode: "open"
54
- });
55
- const getAnchor = () => {
56
- if (options.anchor == null)
57
- return document.body;
58
- let resolved = typeof options.anchor === "function" ? options.anchor() : options.anchor;
59
- if (typeof resolved === "string")
60
- return document.querySelector(resolved) ?? void 0;
61
- return resolved ?? void 0;
62
- };
63
- let mounted;
64
- const mount = () => {
65
- const anchor = getAnchor();
66
- if (anchor == null)
67
- throw Error(
68
- "Failed to mount content script ui: could not find anchor element"
69
- );
70
- mounted = options.mount(uiContainer);
71
- switch (options.append) {
72
- case void 0:
73
- case "last":
74
- anchor.append(shadowHost);
75
- break;
76
- case "first":
77
- if (anchor.firstChild) {
78
- anchor.insertBefore(shadowHost, anchor.firstChild);
79
- } else {
80
- anchor.append(shadowHost);
81
- }
82
- break;
83
- case "replace":
84
- anchor.replaceWith(shadowHost);
85
- break;
86
- case "after":
87
- anchor.replaceWith(anchor, shadowHost);
88
- break;
89
- case "before":
90
- anchor.replaceWith(shadowHost, anchor);
91
- break;
92
- default:
93
- options.append(anchor, shadowHost);
94
- break;
95
- }
96
- if (options.type !== "inline") {
97
- if (options.zIndex != null)
98
- shadowHost.style.zIndex = String(options.zIndex);
99
- shadowHost.style.overflow = "visible";
100
- shadowHost.style.position = "relative";
101
- shadowHost.style.width = "0";
102
- shadowHost.style.height = "0";
103
- shadowHost.style.display = "block";
104
- const html = shadow.querySelector("html");
105
- if (options.type === "overlay") {
106
- html.style.position = "absolute";
107
- if (options.alignment?.startsWith("bottom-"))
108
- html.style.bottom = "0";
109
- else
110
- html.style.top = "0";
111
- if (options.alignment?.endsWith("-right"))
112
- html.style.right = "0";
113
- else
114
- html.style.left = "0";
115
- } else {
116
- html.style.position = "fixed";
117
- html.style.top = "0";
118
- html.style.bottom = "0";
119
- html.style.left = "0";
120
- html.style.right = "0";
121
- }
122
- }
123
- };
124
- const remove = () => {
125
- shadowHost.remove();
126
- options.onRemove?.(mounted);
127
- while (uiContainer.lastChild)
128
- uiContainer.removeChild(uiContainer.lastChild);
129
- };
130
- ctx.onInvalidated(remove);
131
- return {
132
- shadow,
133
- shadowHost,
134
- uiContainer,
135
- mount,
136
- remove,
137
- mounted
138
- };
139
- }
140
- async function loadCss() {
141
- const url = browser.runtime.getURL(`/content-scripts/${__ENTRYPOINT__}.css`);
142
- try {
143
- const res = await fetch(url);
144
- const css = await res.text();
145
- return css.replaceAll(":root", ":host");
146
- } catch (err) {
147
- logger.warn(
148
- `Failed to load styles @ ${url}. Did you forget to import the stylesheet in your entrypoint?`,
149
- err
150
- );
151
- return "";
152
- }
153
- }
154
-
155
- // src/client/utils/ContentScriptContext.ts
30
+ // src/client/content-scripts/content-script-context.ts
156
31
  var ContentScriptContext = class _ContentScriptContext {
157
32
  constructor(contentScriptName, options) {
158
33
  this.contentScriptName = contentScriptName;
@@ -307,10 +182,132 @@ var ContentScriptContext = class _ContentScriptContext {
307
182
  this.onInvalidated(() => removeEventListener("message", cb));
308
183
  }
309
184
  };
185
+
186
+ // src/client/content-scripts/content-script-ui.ts
187
+ import { createIsolatedElement } from "@webext-core/isolated-element";
188
+ async function createContentScriptUi(ctx, options) {
189
+ const css = [options.css ?? ""];
190
+ if (ctx.options?.cssInjectionMode === "ui") {
191
+ css.push(await loadCss());
192
+ }
193
+ const {
194
+ isolatedElement: uiContainer,
195
+ parentElement: shadowHost,
196
+ shadow
197
+ } = await createIsolatedElement({
198
+ name: options.name,
199
+ css: {
200
+ textContent: css.join("\n").trim()
201
+ },
202
+ mode: "open"
203
+ });
204
+ const getAnchor = () => {
205
+ if (options.anchor == null)
206
+ return document.body;
207
+ let resolved = typeof options.anchor === "function" ? options.anchor() : options.anchor;
208
+ if (typeof resolved === "string")
209
+ return document.querySelector(resolved) ?? void 0;
210
+ return resolved ?? void 0;
211
+ };
212
+ let mounted;
213
+ const mount = () => {
214
+ const anchor = getAnchor();
215
+ if (anchor == null)
216
+ throw Error(
217
+ "Failed to mount content script ui: could not find anchor element"
218
+ );
219
+ mounted = options.mount(uiContainer);
220
+ switch (options.append) {
221
+ case void 0:
222
+ case "last":
223
+ anchor.append(shadowHost);
224
+ break;
225
+ case "first":
226
+ if (anchor.firstChild) {
227
+ anchor.insertBefore(shadowHost, anchor.firstChild);
228
+ } else {
229
+ anchor.append(shadowHost);
230
+ }
231
+ break;
232
+ case "replace":
233
+ anchor.replaceWith(shadowHost);
234
+ break;
235
+ case "after":
236
+ anchor.replaceWith(anchor, shadowHost);
237
+ break;
238
+ case "before":
239
+ anchor.replaceWith(shadowHost, anchor);
240
+ break;
241
+ default:
242
+ options.append(anchor, shadowHost);
243
+ break;
244
+ }
245
+ if (options.type !== "inline") {
246
+ if (options.zIndex != null)
247
+ shadowHost.style.zIndex = String(options.zIndex);
248
+ shadowHost.style.overflow = "visible";
249
+ shadowHost.style.position = "relative";
250
+ shadowHost.style.width = "0";
251
+ shadowHost.style.height = "0";
252
+ shadowHost.style.display = "block";
253
+ const html = shadow.querySelector("html");
254
+ if (options.type === "overlay") {
255
+ html.style.position = "absolute";
256
+ if (options.alignment?.startsWith("bottom-"))
257
+ html.style.bottom = "0";
258
+ else
259
+ html.style.top = "0";
260
+ if (options.alignment?.endsWith("-right"))
261
+ html.style.right = "0";
262
+ else
263
+ html.style.left = "0";
264
+ } else {
265
+ html.style.position = "fixed";
266
+ html.style.top = "0";
267
+ html.style.bottom = "0";
268
+ html.style.left = "0";
269
+ html.style.right = "0";
270
+ }
271
+ }
272
+ };
273
+ const remove = () => {
274
+ shadowHost.remove();
275
+ options.onRemove?.(mounted);
276
+ while (uiContainer.lastChild)
277
+ uiContainer.removeChild(uiContainer.lastChild);
278
+ };
279
+ ctx.onInvalidated(remove);
280
+ return {
281
+ shadow,
282
+ shadowHost,
283
+ uiContainer,
284
+ mount,
285
+ remove,
286
+ mounted
287
+ };
288
+ }
289
+ async function loadCss() {
290
+ const url = browser.runtime.getURL(`/content-scripts/${__ENTRYPOINT__}.css`);
291
+ try {
292
+ const res = await fetch(url);
293
+ const css = await res.text();
294
+ return css.replaceAll(":root", ":host");
295
+ } catch (err) {
296
+ logger.warn(
297
+ `Failed to load styles @ ${url}. Did you forget to import the stylesheet in your entrypoint?`,
298
+ err
299
+ );
300
+ return "";
301
+ }
302
+ }
303
+
304
+ // src/client/content-scripts/define-content-script.ts
305
+ function defineContentScript(definition) {
306
+ return definition;
307
+ }
310
308
  export {
311
309
  ContentScriptContext,
312
310
  createContentScriptUi,
313
311
  defineBackground,
314
312
  defineContentScript
315
313
  };
316
- //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vc3JjL2NsaWVudC9kZWZpbmVDb250ZW50U2NyaXB0LnRzIiwgIi4uL3NyYy9jbGllbnQvZGVmaW5lQmFja2dyb3VuZC50cyIsICIuLi9zcmMvY2xpZW50L2NyZWF0ZUNvbnRlbnRTY3JpcHRVaS50cyIsICIuLi9zcmMvY2xpZW50L2Jyb3dzZXIudHMiLCAiLi4vc3JjL2NsaWVudC91dGlscy9sb2dnZXIudHMiLCAiLi4vc3JjL2NsaWVudC91dGlscy9Db250ZW50U2NyaXB0Q29udGV4dC50cyJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiaW1wb3J0IHsgQ29udGVudFNjcmlwdERlZmluaXRpb24gfSBmcm9tICcuLi9jb3JlL3R5cGVzJztcblxuZXhwb3J0IGZ1bmN0aW9uIGRlZmluZUNvbnRlbnRTY3JpcHQoXG4gIGRlZmluaXRpb246IENvbnRlbnRTY3JpcHREZWZpbml0aW9uLFxuKTogQ29udGVudFNjcmlwdERlZmluaXRpb24ge1xuICByZXR1cm4gZGVmaW5pdGlvbjtcbn1cbiIsICJpbXBvcnQgeyBCYWNrZ3JvdW5kRGVmaW5pdGlvbiB9IGZyb20gJy4uJztcblxuZXhwb3J0IGZ1bmN0aW9uIGRlZmluZUJhY2tncm91bmQobWFpbjogKCkgPT4gdm9pZCk6IEJhY2tncm91bmREZWZpbml0aW9uO1xuZXhwb3J0IGZ1bmN0aW9uIGRlZmluZUJhY2tncm91bmQoXG4gIGRlZmluaXRpb246IEJhY2tncm91bmREZWZpbml0aW9uLFxuKTogQmFja2dyb3VuZERlZmluaXRpb247XG5leHBvcnQgZnVuY3Rpb24gZGVmaW5lQmFja2dyb3VuZChcbiAgYXJnOiAoKCkgPT4gdm9pZCkgfCBCYWNrZ3JvdW5kRGVmaW5pdGlvbixcbik6IEJhY2tncm91bmREZWZpbml0aW9uIHtcbiAgaWYgKHR5cGVvZiBhcmcgPT09ICdmdW5jdGlvbicpIHJldHVybiB7IG1haW46IGFyZyB9O1xuICByZXR1cm4gYXJnO1xufVxuIiwgImltcG9ydCB7IGNyZWF0ZUlzb2xhdGVkRWxlbWVudCB9IGZyb20gJ0B3ZWJleHQtY29yZS9pc29sYXRlZC1lbGVtZW50JztcbmltcG9ydCB7IGJyb3dzZXIgfSBmcm9tICcuL2Jyb3dzZXInO1xuaW1wb3J0IHsgbG9nZ2VyIH0gZnJvbSAnLi91dGlscy9sb2dnZXInO1xuaW1wb3J0IHsgQ29udGVudFNjcmlwdENvbnRleHQgfSBmcm9tICcuJztcblxuLyoqXG4gKiBVdGlsaXR5IGZvciBtb3VudGluZyBjb250ZW50IHNjcmlwdCBVSSdzIHdpdGggaXNvbGF0ZWQgc3R5bGVzLiBBdXRvbWF0aWNhbGx5IHJlbW92ZWQgZnJvbSB0aGUgRE9NXG4gKiB3aGVuIHRoZSBjb250ZW50IHNjcmlwdCdzIGNvbnRleHQgaXMgaW52YWxpZGF0ZWQuXG4gKlxuICogU2VlIGh0dHBzOi8vd3h0LmRldi9lbnRyeXBvaW50cy9jb250ZW50LXNjcmlwdHMuaHRtbCN1aSBmb3IgZnVsbCBkb2N1bWVudGF0aW9uLlxuICpcbiAqIEBleGFtcGxlXG4gKiAvLyBlbnRyeXBvaW50cy9leGFtcGxlLXVpLmNvbnRlbnQvaW5kZXgudHNcbiAqIGltcG9ydCBcIi4vc3R5bGUuY3NzXCJcbiAqXG4gKiBleHBvcnQgZGVmYXVsdCBkZWZpbmVDb250ZW50U2NyaXB0KHtcbiAqICAgbWF0Y2hlczogW1wiKjovLyouZ29vZ2xlLmNvbS8qXCJdLFxuICogICBjc3NJbmplY3Rpb25Nb2RlOiBcInVpXCIsXG4gKlxuICogICBhc3luYyBtYWluKGN0eCkge1xuICogICAgIGNvbnN0IHVpID0gYXdhaXQgY3JlYXRlQ29udGVudFNjcmlwdFVpKGN0eCwge1xuICogICAgICAgbmFtZTogXCJleGFtcGxlLW92ZXJsYXlcIixcbiAqICAgICAgIHR5cGU6IFwibW9kYWxcIixcbiAqICAgICAgIG1vdW50KGNvbnRhaW5lcikge1xuICogICAgICAgICBjb25zdCBhcHAgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KFwiZGl2XCIpO1xuICogICAgICAgICBhcHAudGV4dENvbnRlbnQgPSBcIkNvbnRlbnQgU2NyaXB0IFVJXCI7XG4gKiAgICAgICAgIGNvbnRhaW5lci5hcHBlbmQoYXBwKTtcbiAqICAgICAgIH1cbiAqICAgICB9KVxuICogICAgIHVpLm1vdW50KCk7XG4gKiAgIH1cbiAqIH0pXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBjcmVhdGVDb250ZW50U2NyaXB0VWk8VEFwcD4oXG4gIGN0eDogQ29udGVudFNjcmlwdENvbnRleHQsXG4gIG9wdGlvbnM6IENvbnRlbnRTY3JpcHRVaU9wdGlvbnM8VEFwcD4sXG4pOiBQcm9taXNlPENvbnRlbnRTY3JpcHRVaTxUQXBwPj4ge1xuICBjb25zdCBjc3MgPSBbb3B0aW9ucy5jc3MgPz8gJyddO1xuICBpZiAoY3R4Lm9wdGlvbnM/LmNzc0luamVjdGlvbk1vZGUgPT09ICd1aScpIHtcbiAgICBjc3MucHVzaChhd2FpdCBsb2FkQ3NzKCkpO1xuICB9XG5cbiAgY29uc3Qge1xuICAgIGlzb2xhdGVkRWxlbWVudDogdWlDb250YWluZXIsXG4gICAgcGFyZW50RWxlbWVudDogc2hhZG93SG9zdCxcbiAgICBzaGFkb3csXG4gIH0gPSBhd2FpdCBjcmVhdGVJc29sYXRlZEVsZW1lbnQoe1xuICAgIG5hbWU6IG9wdGlvbnMubmFtZSxcbiAgICBjc3M6IHtcbiAgICAgIHRleHRDb250ZW50OiBjc3Muam9pbignXFxuJykudHJpbSgpLFxuICAgIH0sXG4gICAgbW9kZTogJ29wZW4nLFxuICB9KTtcblxuICBjb25zdCBnZXRBbmNob3IgPSAoKTogRWxlbWVudCB8IHVuZGVmaW5lZCA9PiB7XG4gICAgaWYgKG9wdGlvbnMuYW5jaG9yID09IG51bGwpIHJldHVybiBkb2N1bWVudC5ib2R5O1xuXG4gICAgbGV0IHJlc29sdmVkID1cbiAgICAgIHR5cGVvZiBvcHRpb25zLmFuY2hvciA9PT0gJ2Z1bmN0aW9uJyA/IG9wdGlvbnMuYW5jaG9yKCkgOiBvcHRpb25zLmFuY2hvcjtcbiAgICBpZiAodHlwZW9mIHJlc29sdmVkID09PSAnc3RyaW5nJylcbiAgICAgIHJldHVybiBkb2N1bWVudC5xdWVyeVNlbGVjdG9yPEVsZW1lbnQ+KHJlc29sdmVkKSA/PyB1bmRlZmluZWQ7XG4gICAgcmV0dXJuIHJlc29sdmVkID8/IHVuZGVmaW5lZDtcbiAgfTtcblxuICBsZXQgbW91bnRlZDogVEFwcDtcblxuICBjb25zdCBtb3VudCA9ICgpID0+IHtcbiAgICBjb25zdCBhbmNob3IgPSBnZXRBbmNob3IoKTtcbiAgICBpZiAoYW5jaG9yID09IG51bGwpXG4gICAgICB0aHJvdyBFcnJvcihcbiAgICAgICAgJ0ZhaWxlZCB0byBtb3VudCBjb250ZW50IHNjcmlwdCB1aTogY291bGQgbm90IGZpbmQgYW5jaG9yIGVsZW1lbnQnLFxuICAgICAgKTtcblxuICAgIC8vIE1vdW50IFVJIGluc2lkZSBzaGFkb3cgcm9vdFxuICAgIG1vdW50ZWQgPSBvcHRpb25zLm1vdW50KHVpQ29udGFpbmVyKTtcblxuICAgIC8vIEFkZCBzaGFkb3cgcm9vdCBlbGVtZW50IHRvIERPTVxuICAgIHN3aXRjaCAob3B0aW9ucy5hcHBlbmQpIHtcbiAgICAgIGNhc2UgdW5kZWZpbmVkOlxuICAgICAgY2FzZSAnbGFzdCc6XG4gICAgICAgIGFuY2hvci5hcHBlbmQoc2hhZG93SG9zdCk7XG4gICAgICAgIGJyZWFrO1xuICAgICAgY2FzZSAnZmlyc3QnOlxuICAgICAgICBpZiAoYW5jaG9yLmZpcnN0Q2hpbGQpIHtcbiAgICAgICAgICBhbmNob3IuaW5zZXJ0QmVmb3JlKHNoYWRvd0hvc3QsIGFuY2hvci5maXJzdENoaWxkKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBhbmNob3IuYXBwZW5kKHNoYWRvd0hvc3QpO1xuICAgICAgICB9XG4gICAgICAgIGJyZWFrO1xuICAgICAgY2FzZSAncmVwbGFjZSc6XG4gICAgICAgIGFuY2hvci5yZXBsYWNlV2l0aChzaGFkb3dIb3N0KTtcbiAgICAgICAgYnJlYWs7XG4gICAgICBjYXNlICdhZnRlcic6XG4gICAgICAgIGFuY2hvci5yZXBsYWNlV2l0aChhbmNob3IsIHNoYWRvd0hvc3QpO1xuICAgICAgICBicmVhaztcbiAgICAgIGNhc2UgJ2JlZm9yZSc6XG4gICAgICAgIGFuY2hvci5yZXBsYWNlV2l0aChzaGFkb3dIb3N0LCBhbmNob3IpO1xuICAgICAgICBicmVhaztcbiAgICAgIGRlZmF1bHQ6XG4gICAgICAgIG9wdGlvbnMuYXBwZW5kKGFuY2hvciwgc2hhZG93SG9zdCk7XG4gICAgICAgIGJyZWFrO1xuICAgIH1cblxuICAgIC8vIEFwcGx5IHR5cGVzXG4gICAgaWYgKG9wdGlvbnMudHlwZSAhPT0gJ2lubGluZScpIHtcbiAgICAgIGlmIChvcHRpb25zLnpJbmRleCAhPSBudWxsKVxuICAgICAgICBzaGFkb3dIb3N0LnN0eWxlLnpJbmRleCA9IFN0cmluZyhvcHRpb25zLnpJbmRleCk7XG5cbiAgICAgIHNoYWRvd0hvc3Quc3R5bGUub3ZlcmZsb3cgPSAndmlzaWJsZSc7XG4gICAgICBzaGFkb3dIb3N0LnN0eWxlLnBvc2l0aW9uID0gJ3JlbGF0aXZlJztcbiAgICAgIHNoYWRvd0hvc3Quc3R5bGUud2lkdGggPSAnMCc7XG4gICAgICBzaGFkb3dIb3N0LnN0eWxlLmhlaWdodCA9ICcwJztcbiAgICAgIHNoYWRvd0hvc3Quc3R5bGUuZGlzcGxheSA9ICdibG9jayc7XG5cbiAgICAgIGNvbnN0IGh0bWwgPSBzaGFkb3cucXVlcnlTZWxlY3RvcignaHRtbCcpITtcbiAgICAgIC8vIEhUTUwgZG9lc24ndCBleGlzdCBpbiB0ZXN0c1xuICAgICAgaWYgKG9wdGlvbnMudHlwZSA9PT0gJ292ZXJsYXknKSB7XG4gICAgICAgIGh0bWwuc3R5bGUucG9zaXRpb24gPSAnYWJzb2x1dGUnO1xuICAgICAgICBpZiAob3B0aW9ucy5hbGlnbm1lbnQ/LnN0YXJ0c1dpdGgoJ2JvdHRvbS0nKSkgaHRtbC5zdHlsZS5ib3R0b20gPSAnMCc7XG4gICAgICAgIGVsc2UgaHRtbC5zdHlsZS50b3AgPSAnMCc7XG5cbiAgICAgICAgaWYgKG9wdGlvbnMuYWxpZ25tZW50Py5lbmRzV2l0aCgnLXJpZ2h0JykpIGh0bWwuc3R5bGUucmlnaHQgPSAnMCc7XG4gICAgICAgIGVsc2UgaHRtbC5zdHlsZS5sZWZ0ID0gJzAnO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgaHRtbC5zdHlsZS5wb3NpdGlvbiA9ICdmaXhlZCc7XG4gICAgICAgIGh0bWwuc3R5bGUudG9wID0gJzAnO1xuICAgICAgICBodG1sLnN0eWxlLmJvdHRvbSA9ICcwJztcbiAgICAgICAgaHRtbC5zdHlsZS5sZWZ0ID0gJzAnO1xuICAgICAgICBodG1sLnN0eWxlLnJpZ2h0ID0gJzAnO1xuICAgICAgfVxuICAgIH1cbiAgfTtcblxuICBjb25zdCByZW1vdmUgPSAoKSA9PiB7XG4gICAgLy8gRGV0YXRjaCBzaGFkb3cgcm9vdCBmcm9tIERPTVxuICAgIHNoYWRvd0hvc3QucmVtb3ZlKCk7XG4gICAgLy8gQ2xlYW51cCBtb3VudGVkIHN0YXRlXG4gICAgb3B0aW9ucy5vblJlbW92ZT8uKG1vdW50ZWQpO1xuICAgIC8vIFJlbW92ZSBjaGlsZHJlbiBmcm9tIHVpQ29udGFpbmVyXG4gICAgd2hpbGUgKHVpQ29udGFpbmVyLmxhc3RDaGlsZClcbiAgICAgIHVpQ29udGFpbmVyLnJlbW92ZUNoaWxkKHVpQ29udGFpbmVyLmxhc3RDaGlsZCk7XG4gIH07XG5cbiAgY3R4Lm9uSW52YWxpZGF0ZWQocmVtb3ZlKTtcblxuICByZXR1cm4ge1xuICAgIHNoYWRvdyxcbiAgICBzaGFkb3dIb3N0LFxuICAgIHVpQ29udGFpbmVyLFxuICAgIG1vdW50LFxuICAgIHJlbW92ZSxcbiAgICBtb3VudGVkOiBtb3VudGVkISxcbiAgfTtcbn1cblxuLyoqXG4gKiBMb2FkIHRoZSBDU1MgZm9yIHRoZSBjdXJyZW50IGVudHJ5cG9pbnQuXG4gKi9cbmFzeW5jIGZ1bmN0aW9uIGxvYWRDc3MoKTogUHJvbWlzZTxzdHJpbmc+IHtcbiAgY29uc3QgdXJsID0gYnJvd3Nlci5ydW50aW1lLmdldFVSTChgL2NvbnRlbnQtc2NyaXB0cy8ke19fRU5UUllQT0lOVF9ffS5jc3NgKTtcbiAgdHJ5IHtcbiAgICBjb25zdCByZXMgPSBhd2FpdCBmZXRjaCh1cmwpO1xuICAgIGNvbnN0IGNzcyA9IGF3YWl0IHJlcy50ZXh0KCk7XG5cbiAgICAvLyBSZXBsYWNlIDpyb290IHNlbGVjdG9ycyB3aXRoIDpob3N0IHNpbmNlIHdlJ3JlIGluIGEgc2hhZG93IHJvb3RcbiAgICByZXR1cm4gY3NzLnJlcGxhY2VBbGwoJzpyb290JywgJzpob3N0Jyk7XG4gIH0gY2F0Y2ggKGVycikge1xuICAgIGxvZ2dlci53YXJuKFxuICAgICAgYEZhaWxlZCB0byBsb2FkIHN0eWxlcyBAICR7dXJsfS4gRGlkIHlvdSBmb3JnZXQgdG8gaW1wb3J0IHRoZSBzdHlsZXNoZWV0IGluIHlvdXIgZW50cnlwb2ludD9gLFxuICAgICAgZXJyLFxuICAgICk7XG4gICAgcmV0dXJuICcnO1xuICB9XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgQ29udGVudFNjcmlwdFVpPFRBcHA+IHtcbiAgLyoqXG4gICAqIFRoZSBgSFRNTEVsZW1lbnRgIGhvc3RpbmcgdGhlIHNoYWRvdyByb290IHVzZWQgdG8gaXNvbGF0ZSB0aGUgVUkncyBzdHlsZXMuIFRoaXMgaXMgdGhlIGVsZW1lbnRcbiAgICogdGhhdCBnZXQncyBhZGRlZCB0byB0aGUgRE9NLiBUaGlzIGVsZW1lbnQncyBzdHlsZSBpcyBub3QgaXNvbGF0ZWQgZnJvbSB0aGUgd2VicGFnZS5cbiAgICovXG4gIHNoYWRvd0hvc3Q6IEhUTUxFbGVtZW50O1xuICAvKipcbiAgICogVGhlIGNvbnRhaW5lciBlbGVtZW50IGluc2lkZSB0aGUgYFNoYWRvd1Jvb3RgIHdob3NlIHN0eWxlcyBhcmUgaXNvbGF0ZWQuIFRoZSBVSSBpcyBtb3VudGVkXG4gICAqIGluc2lkZSB0aGlzIGBIVE1MRWxlbWVudGAuXG4gICAqL1xuICB1aUNvbnRhaW5lcjogSFRNTEVsZW1lbnQ7XG4gIC8qKlxuICAgKiBUaGUgc2hhZG93IHJvb3QgcGVyZm9ybWluZyB0aGUgaXNvbGF0aW9uLlxuICAgKi9cbiAgc2hhZG93OiBTaGFkb3dSb290O1xuICAvKipcbiAgICogQ3VzdG9tIGRhdGEgcmV0dXJuZWQgZnJvbSB0aGUgYG9wdGlvbnMubW91bnRgIGZ1bmN0aW9uLlxuICAgKi9cbiAgbW91bnRlZDogVEFwcDtcbiAgLyoqXG4gICAqIEZ1bmN0aW9uIHRoYXQgbW91bnRzIG9yIHJlbW91bnRzIHRoZSBVSSBvbiB0aGUgcGFnZS5cbiAgICovXG4gIG1vdW50OiAoKSA9PiB2b2lkO1xuICAvKipcbiAgICogRnVuY3Rpb24gdGhhdCByZW1vdmVzIHRoZSBVSSBmcm9tIHRoZSB3ZWJwYWdlLlxuICAgKi9cbiAgcmVtb3ZlOiAoKSA9PiB2b2lkO1xufVxuXG5pbnRlcmZhY2UgQmFzZUNvbnRlbnRTY3JpcHRVaU9wdGlvbnM8VEFwcD4ge1xuICAvKipcbiAgICogVGhlIG5hbWUgb2YgdGhlIGN1c3RvbSBjb21wb25lbnQgdXNlZCB0byBob3N0IHRoZSBTaGFkb3dSb290LiBNdXN0IGJlIGtlYmFiLWNhc2UuXG4gICAqL1xuICBuYW1lOiBzdHJpbmc7XG4gIC8qKlxuICAgKiBJbiBjb21iaW5hdGlvbiB3aXRoIGBhbmNob3JgLCBkZWNpZGUgaG93IHRvIGFkZCB0aGUgVUkgdG8gdGhlIERPTS5cbiAgICpcbiAgICogLSBgXCJsYXN0XCJgIChkZWZhdWx0KSAtIEFkZCB0aGUgVUkgYXMgdGhlIGxhc3QgY2hpbGQgb2YgdGhlIGBhbmNob3JgIGVsZW1lbnRcbiAgICogLSBgXCJmaXJzdFwiYCAtIEFkZCB0aGUgVUkgYXMgdGhlIGxhc3QgY2hpbGQgb2YgdGhlIGBhbmNob3JgIGVsZW1lbnRcbiAgICogLSBgXCJyZXBsYWNlXCJgIC0gUmVwbGFjZSB0aGUgYGFuY2hvcmAgZWxlbWVudCB3aXRoIHRoZSBVSS5cbiAgICogLSBgXCJiZWZvcmVcImAgLSBBZGQgdGhlIFVJIGFzIHRoZSBzaWJsaW5nIGJlZm9yZSB0aGUgYGFuY2hvcmAgZWxlbWVudFxuICAgKiAtIGBcImFmdGVyXCJgIC0gQWRkIHRoZSBVSSBhcyB0aGUgc2libGluZyBhZnRlciB0aGUgYGFuY2hvcmAgZWxlbWVudFxuICAgKiAtIGAoYW5jaG9yLCB1aSkgPT4gdm9pZGAgLSBDdXN0b21pemFibGUgZnVuY3Rpb24gdGhhdCBsZXQncyB5b3UgYWRkIHRoZSBVSSB0byB0aGUgRE9NXG4gICAqL1xuICBhcHBlbmQ/OiBDb250ZW50U2NyaXB0QXBwZW5kTW9kZSB8ICgoYW5jaG9yOiBFbGVtZW50LCB1aTogRWxlbWVudCkgPT4gdm9pZCk7XG4gIC8qKlxuICAgKiBBIENTUyBzZWxlY3RvciwgZWxlbWVudCwgb3IgZnVuY3Rpb24gdGhhdCByZXR1cm5zIG9uZSBvZiB0aGUgdHdvLiBBbG9uZyB3aXRoIGBhcHBlbmRgLCB0aGVcbiAgICogYGFuY2hvcmAgZGljdGF0ZXMgd2hlcmUgaW4gdGhlIHBhZ2UgdGhlIFVJIHdpbGwgYmUgYWRkZWQuXG4gICAqL1xuICBhbmNob3I/OlxuICAgIHwgc3RyaW5nXG4gICAgfCBFbGVtZW50XG4gICAgfCBudWxsXG4gICAgfCB1bmRlZmluZWRcbiAgICB8ICgoKSA9PiBzdHJpbmcgfCBFbGVtZW50IHwgbnVsbCB8IHVuZGVmaW5lZCk7XG4gIC8qKlxuICAgKiBDYWxsYmFjayBleGVjdXRlZCB3aGVuIG1vdW50aW5nIHRoZSBVSS4gVGhpcyBmdW5jdGlvbiBzaG91bGQgY3JlYXRlIGFuZCBhcHBlbmQgdGhlIFVJIHRvIHRoZVxuICAgKiBgY29udGFpbmVyYCBlbGVtZW50LiBJdCBpcyBjYWxsZWQgZXZlcnkgdGltZSBgdWkubW91bnQoKWAgaXMgY2FsbGVkXG4gICAqXG4gICAqIE9wdGlvbmFsbHkgcmV0dXJuIGEgdmFsdWUgdGhhdCBjYW4gYmUgYWNjZXNzZWQgYXQgYHVpLm1vdW50ZWRgIG9yIGluIHRoZSBgb25SZW1vdmVgIGNhbGxiYWNrLlxuICAgKi9cbiAgbW91bnQ6IChjb250YWluZXI6IEVsZW1lbnQpID0+IFRBcHA7XG4gIC8qKlxuICAgKiBDYWxsYmFjayBjYWxsZWQgd2hlbiB0aGUgVUkgaXMgcmVtb3ZlZCBmcm9tIHRoZSB3ZWJwYWdlLiBVc2UgdG8gY2xlYW51cCB5b3VyIFVJLCBsaWtlXG4gICAqIHVubW91bnRpbmcgeW91ciB2dWUgb3IgcmVhY3QgYXBwcy5cbiAgICovXG4gIG9uUmVtb3ZlPzogKG1vdW50ZWQ6IFRBcHApID0+IHZvaWQ7XG4gIC8qKlxuICAgKiBDdXN0b20gQ1NTIHRleHQgdG8gYXBwbHkgdG8gdGhlIFVJLiBJZiB5b3VyIGNvbnRlbnQgc2NyaXB0IGltcG9ydHMvZ2VuZXJhdGVzIENTUyBhbmQgeW91J3ZlXG4gICAqIHNldCBgY3NzSW5qZWN0aW9uTW9kZTogXCJ1aVwiYCwgdGhlIGltcG9ydGVkIENTUyB3aWxsIGJlIGluY2x1ZGVkIGF1dG9tYXRpY2FsbHkuIFlvdSBkbyBub3QgbmVlZFxuICAgKiB0byBwYXNzIHRob3NlIHN0eWxlcyBpbiBoZXJlLiBUaGlzIGlzIGZvciBhbnkgYWRkaXRpb25hbCBzdHlsZXMgbm90IGluIHRoZSBpbXBvcnRlZCBDU1MuXG4gICAqXG4gICAqIFNlZSBodHRwczovL3d4dC5kZXYvZW50cnlwb2ludHMvY29udGVudC1zY3JpcHRzLmh0bWwjdWkgZm9yIG1vcmUgaW5mby5cbiAgICovXG4gIGNzcz86IHN0cmluZztcbn1cblxuZXhwb3J0IHR5cGUgT3ZlcmxheUNvbnRlbnRTY3JpcHRVaU9wdGlvbnM8VEFwcD4gPVxuICBCYXNlQ29udGVudFNjcmlwdFVpT3B0aW9uczxUQXBwPiAmIHtcbiAgICB0eXBlOiAnb3ZlcmxheSc7XG4gICAgLyoqXG4gICAgICogV2hlbiB1c2luZyBgdHlwZTogXCJvdmVybGF5XCJgLCB0aGUgbW91bnRlZCBlbGVtZW50IGlzIDBweCBieSAwcHggaW4gc2l6ZS4gQWxpZ25tZW50IHNwZWNpZmllc1xuICAgICAqIHdoaWNoIGNvcm5lciBpcyBhbGlnbmVkIHdpdGggdGhhdCAweDAgcGl4ZWwgc3BhY2UuXG4gICAgICpcbiAgICAgKiBAZGVmYXVsdCBcInRvcC1sZWZ0XCJcbiAgICAgKi9cbiAgICBhbGlnbm1lbnQ/OiBDb250ZW50U2NyaXB0VWlPdmVybGF5QWxpZ25tZW50O1xuICAgIC8qKlxuICAgICAqIFRoZSBgei1pbmRleGAgdXNlZCBvbiB0aGUgYHNoYWRvd0hvc3RgLiBTZXQgdG8gYSBwb3NpdGl2ZSBudW1iZXIgdG8gc2hvdyB5b3VyIFVJIG92ZXIgd2Vic2l0ZVxuICAgICAqIGNvbnRlbnQuXG4gICAgICovXG4gICAgekluZGV4PzogbnVtYmVyO1xuICB9O1xuXG5leHBvcnQgdHlwZSBNb2RhbENvbnRlbnRTY3JpcHRVaU9wdGlvbnM8VEFwcD4gPVxuICBCYXNlQ29udGVudFNjcmlwdFVpT3B0aW9uczxUQXBwPiAmIHtcbiAgICB0eXBlOiAnbW9kYWwnO1xuICAgIC8qKlxuICAgICAqIFRoZSBgei1pbmRleGAgdXNlZCBvbiB0aGUgYHNoYWRvd0hvc3RgLiBTZXQgdG8gYSBwb3NpdGl2ZSBudW1iZXIgdG8gc2hvdyB5b3VyIFVJIG92ZXIgd2Vic2l0ZVxuICAgICAqIGNvbnRlbnQuXG4gICAgICovXG4gICAgekluZGV4PzogbnVtYmVyO1xuICB9O1xuXG5leHBvcnQgdHlwZSBJbmxpbmVDb250ZW50U2NyaXB0VWlPcHRpb25zPFRBcHA+ID1cbiAgQmFzZUNvbnRlbnRTY3JpcHRVaU9wdGlvbnM8VEFwcD4gJiB7XG4gICAgdHlwZTogJ2lubGluZSc7XG4gIH07XG5cbmV4cG9ydCB0eXBlIENvbnRlbnRTY3JpcHRVaU92ZXJsYXlBbGlnbm1lbnQgPVxuICB8ICd0b3AtbGVmdCdcbiAgfCAndG9wLXJpZ2h0J1xuICB8ICdib3R0b20tbGVmdCdcbiAgfCAnYm90dG9tLXJpZ2h0JztcblxuZXhwb3J0IHR5cGUgQ29udGVudFNjcmlwdEFwcGVuZE1vZGUgPVxuICB8ICdsYXN0J1xuICB8ICdmaXJzdCdcbiAgfCAncmVwbGFjZSdcbiAgfCAnYmVmb3JlJ1xuICB8ICdhZnRlcic7XG5cbmV4cG9ydCB0eXBlIENvbnRlbnRTY3JpcHRVaU9wdGlvbnM8VEFwcD4gPVxuICB8IE92ZXJsYXlDb250ZW50U2NyaXB0VWlPcHRpb25zPFRBcHA+XG4gIHwgTW9kYWxDb250ZW50U2NyaXB0VWlPcHRpb25zPFRBcHA+XG4gIHwgSW5saW5lQ29udGVudFNjcmlwdFVpT3B0aW9uczxUQXBwPjtcbiIsICIvKipcbiAqIEBtb2R1bGUgd3h0L2Jyb3dzZXJcbiAqL1xuaW1wb3J0IG9yaWdpbmFsQnJvd3NlciwgeyBCcm93c2VyLCBSdW50aW1lLCBJMThuIH0gZnJvbSAnd2ViZXh0ZW5zaW9uLXBvbHlmaWxsJztcblxuZXhwb3J0IGludGVyZmFjZSBBdWdtZW50ZWRCcm93c2VyIGV4dGVuZHMgQnJvd3NlciB7XG4gIHJ1bnRpbWU6IFd4dFJ1bnRpbWU7XG4gIGkxOG46IFd4dEkxOG47XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgV3h0UnVudGltZSBleHRlbmRzIFJ1bnRpbWUuU3RhdGljIHtcbiAgLy8gT3ZlcnJpZGVuIHBlci1wcm9qZWN0XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgV3h0STE4biBleHRlbmRzIEkxOG4uU3RhdGljIHtcbiAgLy8gT3ZlcnJpZGVuIHBlci1wcm9qZWN0XG59XG5cbmV4cG9ydCBjb25zdCBicm93c2VyOiBBdWdtZW50ZWRCcm93c2VyID0gb3JpZ2luYWxCcm93c2VyO1xuIiwgImZ1bmN0aW9uIHByaW50KG1ldGhvZDogKC4uLmFyZ3M6IGFueVtdKSA9PiB2b2lkLCAuLi5hcmdzOiBhbnlbXSkge1xuICBpZiAoaW1wb3J0Lm1ldGEuZW52Lk1PREUgPT09ICdwcm9kdWN0aW9uJykgcmV0dXJuO1xuXG4gIGlmICh0eXBlb2YgYXJnc1swXSA9PT0gJ3N0cmluZycpIHtcbiAgICBjb25zdCBtZXNzYWdlID0gYXJncy5zaGlmdCgpO1xuICAgIG1ldGhvZChgW3d4dF0gJHttZXNzYWdlfWAsIC4uLmFyZ3MpO1xuICB9IGVsc2Uge1xuICAgIG1ldGhvZCgnW3d4dF0nLCAuLi5hcmdzKTtcbiAgfVxufVxuXG4vKipcbiAqIFdyYXBwZXIgYXJvdW5kIGBjb25zb2xlYCB3aXRoIGEgXCJbd3h0XVwiIHByZWZpeFxuICovXG5leHBvcnQgY29uc3QgbG9nZ2VyID0ge1xuICBkZWJ1ZzogKC4uLmFyZ3M6IGFueVtdKSA9PiBwcmludChjb25zb2xlLmRlYnVnLCAuLi5hcmdzKSxcbiAgbG9nOiAoLi4uYXJnczogYW55W10pID0+IHByaW50KGNvbnNvbGUubG9nLCAuLi5hcmdzKSxcbiAgd2FybjogKC4uLmFyZ3M6IGFueVtdKSA9PiBwcmludChjb25zb2xlLndhcm4sIC4uLmFyZ3MpLFxuICBlcnJvcjogKC4uLmFyZ3M6IGFueVtdKSA9PiBwcmludChjb25zb2xlLmVycm9yLCAuLi5hcmdzKSxcbn07XG4iLCAiaW1wb3J0IHsgQ29udGVudFNjcmlwdERlZmluaXRpb24gfSBmcm9tICcuLi8uLi9jb3JlL3R5cGVzJztcbmltcG9ydCB7IGJyb3dzZXIgfSBmcm9tICcuLi9icm93c2VyJztcbmltcG9ydCB7IGxvZ2dlciB9IGZyb20gJy4vbG9nZ2VyJztcblxuLyoqXG4gKiBJbXBsZW1lbnRzIFtgQWJvcnRDb250cm9sbGVyYF0oaHR0cHM6Ly9kZXZlbG9wZXIubW96aWxsYS5vcmcvZW4tVVMvZG9jcy9XZWIvQVBJL0Fib3J0Q29udHJvbGxlcikuXG4gKiBVc2VkIHRvIGRldGVjdCBhbmQgc3RvcCBjb250ZW50IHNjcmlwdCBjb2RlIHdoZW4gdGhlIHNjcmlwdCBpcyBpbnZhbGlkYXRlZC5cbiAqXG4gKiBJdCBhbHNvIHByb3ZpZGVzIHNldmVyYWwgdXRpbGl0aWVzIGxpa2UgYGN0eC5zZXRUaW1lb3V0YCBhbmQgYGN0eC5zZXRJbnRlcnZhbGAgdGhhdCBzaG91bGQgYmUgdXNlZCBpblxuICogY29udGVudCBzY3JpcHRzIGluc3RlYWQgb2YgYHdpbmRvdy5zZXRUaW1lb3V0YCBvciBgd2luZG93LnNldEludGVydmFsYC5cbiAqL1xuZXhwb3J0IGNsYXNzIENvbnRlbnRTY3JpcHRDb250ZXh0IGltcGxlbWVudHMgQWJvcnRDb250cm9sbGVyIHtcbiAgcHJpdmF0ZSBzdGF0aWMgU0NSSVBUX1NUQVJURURfTUVTU0FHRV9UWVBFID0gJ3d4dDpjb250ZW50LXNjcmlwdC1zdGFydGVkJztcblxuICAjaXNUb3BGcmFtZSA9IHdpbmRvdy5zZWxmID09PSB3aW5kb3cudG9wO1xuICAjYWJvcnRDb250cm9sbGVyOiBBYm9ydENvbnRyb2xsZXI7XG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHJpdmF0ZSByZWFkb25seSBjb250ZW50U2NyaXB0TmFtZTogc3RyaW5nLFxuICAgIHB1YmxpYyByZWFkb25seSBvcHRpb25zPzogT21pdDxDb250ZW50U2NyaXB0RGVmaW5pdGlvbiwgJ21haW4nPixcbiAgKSB7XG4gICAgdGhpcy4jYWJvcnRDb250cm9sbGVyID0gbmV3IEFib3J0Q29udHJvbGxlcigpO1xuICAgIGlmICh0aGlzLiNpc1RvcEZyYW1lKSB7XG4gICAgICB0aGlzLiNzdG9wT2xkU2NyaXB0cygpO1xuICAgIH1cbiAgICB0aGlzLnNldFRpbWVvdXQoKCkgPT4ge1xuICAgICAgLy8gUnVuIG9uIG5leHQgdGljayBzbyB0aGUgbGlzdGVuZXIgaXQgYWRkcyBpc24ndCB0cmlnZ2VyZWQgYnkgc3RvcE9sZFNjcmlwdFxuICAgICAgdGhpcy4jbGlzdGVuRm9yTmV3ZXJTY3JpcHRzKCk7XG4gICAgfSk7XG4gIH1cblxuICBnZXQgc2lnbmFsKCkge1xuICAgIHJldHVybiB0aGlzLiNhYm9ydENvbnRyb2xsZXIuc2lnbmFsO1xuICB9XG5cbiAgYWJvcnQocmVhc29uPzogYW55KTogdm9pZCB7XG4gICAgcmV0dXJuIHRoaXMuI2Fib3J0Q29udHJvbGxlci5hYm9ydChyZWFzb24pO1xuICB9XG5cbiAgZ2V0IGlzSW52YWxpZCgpOiBib29sZWFuIHtcbiAgICBpZiAoYnJvd3Nlci5ydW50aW1lLmlkID09IG51bGwpIHtcbiAgICAgIHRoaXMubm90aWZ5SW52YWxpZGF0ZWQoKTsgLy8gU2V0cyBgc2lnbmFsLmFib3J0ZWRgIHRvIHRydWVcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMuc2lnbmFsLmFib3J0ZWQ7XG4gIH1cblxuICBnZXQgaXNWYWxpZCgpOiBib29sZWFuIHtcbiAgICByZXR1cm4gIXRoaXMuaXNJbnZhbGlkO1xuICB9XG5cbiAgLyoqXG4gICAqIEFkZCBhIGxpc3RlbmVyIHRoYXQgaXMgY2FsbGVkIHdoZW4gdGhlIGNvbnRlbnQgc2NyaXB0J3MgY29udGV4dCBpcyBpbnZhbGlkYXRlZC5cbiAgICpcbiAgICogQHJldHVybnMgQSBmdW5jdGlvbiB0byByZW1vdmUgdGhlIGxpc3RlbmVyLlxuICAgKlxuICAgKiBAZXhhbXBsZVxuICAgKiBicm93c2VyLnJ1bnRpbWUub25NZXNzYWdlLmFkZExpc3RlbmVyKGNiKTtcbiAgICogY29uc3QgcmVtb3ZlSW52YWxpZGF0ZWRMaXN0ZW5lciA9IGN0eC5vbkludmFsaWRhdGVkKCgpID0+IHtcbiAgICogICBicm93c2VyLnJ1bnRpbWUub25NZXNzYWdlLnJlbW92ZUxpc3RlbmVyKGNiKTtcbiAgICogfSlcbiAgICogLy8gLi4uXG4gICAqIHJlbW92ZUludmFsaWRhdGVkTGlzdGVuZXIoKTtcbiAgICovXG4gIG9uSW52YWxpZGF0ZWQoY2I6ICgpID0+IHZvaWQpOiAoKSA9PiB2b2lkIHtcbiAgICB0aGlzLnNpZ25hbC5hZGRFdmVudExpc3RlbmVyKCdhYm9ydCcsIGNiKTtcbiAgICByZXR1cm4gKCkgPT4gdGhpcy5zaWduYWwucmVtb3ZlRXZlbnRMaXN0ZW5lcignYWJvcnQnLCBjYik7XG4gIH1cblxuICAvKipcbiAgICogUmV0dXJuIGEgcHJvbWlzZSB0aGF0IG5ldmVyIHJlc29sdmVzLiBVc2VmdWwgaWYgeW91IGhhdmUgYW4gYXN5bmMgZnVuY3Rpb24gdGhhdCBzaG91bGRuJ3QgcnVuXG4gICAqIGFmdGVyIHRoZSBjb250ZXh0IGlzIGV4cGlyZWQuXG4gICAqXG4gICAqIEBleGFtcGxlXG4gICAqIGNvbnN0IGdldFZhbHVlRnJvbVN0b3JhZ2UgPSBhc3luYyAoKSA9PiB7XG4gICAqICAgaWYgKGN0eC5pc0ludmFsaWQpIHJldHVybiBjdHguYmxvY2soKTtcbiAgICpcbiAgICogICAvLyAuLi5cbiAgICogfVxuICAgKi9cbiAgYmxvY2s8VD4oKTogUHJvbWlzZTxUPiB7XG4gICAgcmV0dXJuIG5ldyBQcm9taXNlKCgpID0+IHtcbiAgICAgIC8vIG5vb3BcbiAgICB9KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBXcmFwcGVyIGFyb3VuZCBgd2luZG93LnNldEludGVydmFsYCB0aGF0IGF1dG9tYXRpY2FsbHkgY2xlYXJzIHRoZSBpbnRlcnZhbCB3aGVuIGludmFsaWRhdGVkLlxuICAgKi9cbiAgc2V0SW50ZXJ2YWwoaGFuZGxlcjogKCkgPT4gdm9pZCwgdGltZW91dD86IG51bWJlcik6IG51bWJlciB7XG4gICAgY29uc3QgaWQgPSBzZXRJbnRlcnZhbCgoKSA9PiB7XG4gICAgICBpZiAodGhpcy5pc1ZhbGlkKSBoYW5kbGVyKCk7XG4gICAgfSwgdGltZW91dCkgYXMgdW5rbm93biBhcyBudW1iZXI7XG4gICAgdGhpcy5vbkludmFsaWRhdGVkKCgpID0+IGNsZWFySW50ZXJ2YWwoaWQpKTtcbiAgICByZXR1cm4gaWQ7XG4gIH1cblxuICAvKipcbiAgICogV3JhcHBlciBhcm91bmQgYHdpbmRvdy5zZXRUaW1lb3V0YCB0aGF0IGF1dG9tYXRpY2FsbHkgY2xlYXJzIHRoZSBpbnRlcnZhbCB3aGVuIGludmFsaWRhdGVkLlxuICAgKi9cbiAgc2V0VGltZW91dChoYW5kbGVyOiAoKSA9PiB2b2lkLCB0aW1lb3V0PzogbnVtYmVyKTogbnVtYmVyIHtcbiAgICBjb25zdCBpZCA9IHNldFRpbWVvdXQoKCkgPT4ge1xuICAgICAgaWYgKHRoaXMuaXNWYWxpZCkgaGFuZGxlcigpO1xuICAgIH0sIHRpbWVvdXQpIGFzIHVua25vd24gYXMgbnVtYmVyO1xuICAgIHRoaXMub25JbnZhbGlkYXRlZCgoKSA9PiBjbGVhclRpbWVvdXQoaWQpKTtcbiAgICByZXR1cm4gaWQ7XG4gIH1cblxuICAvKipcbiAgICogV3JhcHBlciBhcm91bmQgYHdpbmRvdy5yZXF1ZXN0QW5pbWF0aW9uRnJhbWVgIHRoYXQgYXV0b21hdGljYWxseSBjYW5jZWxzIHRoZSByZXF1ZXN0IHdoZW5cbiAgICogaW52YWxpZGF0ZWQuXG4gICAqL1xuICByZXF1ZXN0QW5pbWF0aW9uRnJhbWUoY2FsbGJhY2s6IEZyYW1lUmVxdWVzdENhbGxiYWNrKTogbnVtYmVyIHtcbiAgICBjb25zdCBpZCA9IHJlcXVlc3RBbmltYXRpb25GcmFtZSgoLi4uYXJncykgPT4ge1xuICAgICAgaWYgKHRoaXMuaXNWYWxpZCkgY2FsbGJhY2soLi4uYXJncyk7XG4gICAgfSk7XG5cbiAgICB0aGlzLm9uSW52YWxpZGF0ZWQoKCkgPT4gY2FuY2VsQW5pbWF0aW9uRnJhbWUoaWQpKTtcbiAgICByZXR1cm4gaWQ7XG4gIH1cblxuICAvKipcbiAgICogV3JhcHBlciBhcm91bmQgYHdpbmRvdy5yZXF1ZXN0SWRsZUNhbGxiYWNrYCB0aGF0IGF1dG9tYXRpY2FsbHkgY2FuY2VscyB0aGUgcmVxdWVzdCB3aGVuXG4gICAqIGludmFsaWRhdGVkLlxuICAgKi9cbiAgcmVxdWVzdElkbGVDYWxsYmFjayhcbiAgICBjYWxsYmFjazogSWRsZVJlcXVlc3RDYWxsYmFjayxcbiAgICBvcHRpb25zPzogSWRsZVJlcXVlc3RPcHRpb25zLFxuICApOiBudW1iZXIge1xuICAgIGNvbnN0IGlkID0gcmVxdWVzdElkbGVDYWxsYmFjaygoLi4uYXJncykgPT4ge1xuICAgICAgaWYgKCF0aGlzLnNpZ25hbC5hYm9ydGVkKSBjYWxsYmFjayguLi5hcmdzKTtcbiAgICB9LCBvcHRpb25zKTtcblxuICAgIHRoaXMub25JbnZhbGlkYXRlZCgoKSA9PiBjYW5jZWxJZGxlQ2FsbGJhY2soaWQpKTtcbiAgICByZXR1cm4gaWQ7XG4gIH1cblxuICAvKipcbiAgICogQ2FsbCBgdGFyZ2V0LmFkZEV2ZW50TGlzdGVuZXJgIGFuZCByZW1vdmUgdGhlIGV2ZW50IGxpc3RlbmVyIHdoZW4gdGhlIGNvbnRleHQgaXMgaW52YWxpZGF0ZWQuXG4gICAqXG4gICAqIEBleGFtcGxlXG4gICAqIGN0eC5hZGRFdmVudExpc3RlbmVyKHdpbmRvdywgXCJtb3VzZW1vdmVcIiwgKCkgPT4ge1xuICAgKiAgIC8vIC4uLlxuICAgKiB9KTtcbiAgICogY3R4LmFkZEV2ZW50TGlzdGVuZXIoZG9jdW1lbnQsIFwidmlzaWJpbGl0eWNoYW5nZVwiLCAoKSA9PiB7XG4gICAqICAgLy8gLi4uXG4gICAqIH0pO1xuICAgKi9cbiAgYWRkRXZlbnRMaXN0ZW5lcihcbiAgICB0YXJnZXQ6IGFueSxcbiAgICB0eXBlOiBzdHJpbmcsXG4gICAgaGFuZGxlcjogKGV2ZW50OiBFdmVudCkgPT4gdm9pZCxcbiAgICBvcHRpb25zPzogQWRkRXZlbnRMaXN0ZW5lck9wdGlvbnMsXG4gICkge1xuICAgIHRhcmdldC5hZGRFdmVudExpc3RlbmVyPy4odHlwZSwgaGFuZGxlciwgb3B0aW9ucyk7XG4gICAgdGhpcy5vbkludmFsaWRhdGVkKFxuICAgICAgKCkgPT4gdGFyZ2V0LnJlbW92ZUV2ZW50TGlzdGVuZXI/Lih0eXBlLCBoYW5kbGVyLCBvcHRpb25zKSxcbiAgICApO1xuICB9XG5cbiAgLyoqXG4gICAqIEBpbnRlcm5hbFxuICAgKiBBYm9ydCB0aGUgYWJvcnQgY29udHJvbGxlciBhbmQgZXhlY3V0ZSBhbGwgYG9uSW52YWxpZGF0ZWRgIGxpc3RlbmVycy5cbiAgICovXG4gIG5vdGlmeUludmFsaWRhdGVkKCkge1xuICAgIHRoaXMuYWJvcnQoJ0NvbnRlbnQgc2NyaXB0IGNvbnRleHQgaW52YWxpZGF0ZWQnKTtcbiAgICBsb2dnZXIuZGVidWcoXG4gICAgICBgQ29udGVudCBzY3JpcHQgXCIke3RoaXMuY29udGVudFNjcmlwdE5hbWV9XCIgY29udGV4dCBpbnZhbGlkYXRlZGAsXG4gICAgKTtcbiAgfVxuXG4gICNzdG9wT2xkU2NyaXB0cygpIHtcbiAgICAvLyBVc2UgcG9zdE1lc3NhZ2Ugc28gaXQgZ2V0J3Mgc2VudCB0byBhbGwgdGhlIGZyYW1lcyBvZiB0aGUgcGFnZS5cbiAgICB3aW5kb3cucG9zdE1lc3NhZ2UoXG4gICAgICB7XG4gICAgICAgIGV2ZW50OiBDb250ZW50U2NyaXB0Q29udGV4dC5TQ1JJUFRfU1RBUlRFRF9NRVNTQUdFX1RZUEUsXG4gICAgICAgIGNvbnRlbnRTY3JpcHROYW1lOiB0aGlzLmNvbnRlbnRTY3JpcHROYW1lLFxuICAgICAgfSxcbiAgICAgICcqJyxcbiAgICApO1xuICB9XG5cbiAgI2xpc3RlbkZvck5ld2VyU2NyaXB0cygpIHtcbiAgICBjb25zdCBjYiA9IChldmVudDogTWVzc2FnZUV2ZW50KSA9PiB7XG4gICAgICBpZiAoXG4gICAgICAgIGV2ZW50LmRhdGE/LnR5cGUgPT09IENvbnRlbnRTY3JpcHRDb250ZXh0LlNDUklQVF9TVEFSVEVEX01FU1NBR0VfVFlQRSAmJlxuICAgICAgICBldmVudC5kYXRhPy5jb250ZW50U2NyaXB0TmFtZSA9PT0gdGhpcy5jb250ZW50U2NyaXB0TmFtZVxuICAgICAgKSB7XG4gICAgICAgIHRoaXMubm90aWZ5SW52YWxpZGF0ZWQoKTtcbiAgICAgIH1cbiAgICB9O1xuXG4gICAgYWRkRXZlbnRMaXN0ZW5lcignbWVzc2FnZScsIGNiKTtcbiAgICB0aGlzLm9uSW52YWxpZGF0ZWQoKCkgPT4gcmVtb3ZlRXZlbnRMaXN0ZW5lcignbWVzc2FnZScsIGNiKSk7XG4gIH1cbn1cbiJdLAogICJtYXBwaW5ncyI6ICI7QUFFTyxTQUFTLG9CQUNkLFlBQ3lCO0FBQ3pCLFNBQU87QUFDVDs7O0FDQU8sU0FBUyxpQkFDZCxLQUNzQjtBQUN0QixNQUFJLE9BQU8sUUFBUTtBQUFZLFdBQU8sRUFBRSxNQUFNLElBQUk7QUFDbEQsU0FBTztBQUNUOzs7QUNYQSxTQUFTLDZCQUE2Qjs7O0FDR3RDLE9BQU8scUJBQWlEO0FBZWpELElBQU0sVUFBNEI7OztBQ2xCekMsU0FBUyxNQUFNLFdBQXFDLE1BQWE7QUFDL0QsTUFBSSxZQUFZLElBQUksU0FBUztBQUFjO0FBRTNDLE1BQUksT0FBTyxLQUFLLENBQUMsTUFBTSxVQUFVO0FBQy9CLFVBQU0sVUFBVSxLQUFLLE1BQU07QUFDM0IsV0FBTyxTQUFTLE9BQU8sSUFBSSxH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QUFBO0FBQUE7QUFBQTtBQUFBLEVBTUEsc0JBQXNCLFVBQXdDO0FBQzVELFVBQU0sS0FBSyxzQkFBc0IsSUFBSSxTQUFTO0FBQzVDLFVBQUksS0FBSztBQUFTLGlCQUFTLEdBQUcsSUFBSTtBQUFBLElBQ3BDLENBQUM7QUFFRCxTQUFLLGNBQWMsTUFBTSxxQkFBcUIsRUFBRSxDQUFDO0FBQ2pELFdBQU87QUFBQSxFQUNUO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQSxFQU1BLG9CQUNFLFVBQ0EsU0FDUTtBQUNSLFVBQU0sS0FBSyxvQkFBb0IsSUFBSSxTQUFTO0FBQzFDLFVBQUksQ0FBQyxLQUFLLE9BQU87QUFBUyxpQkFBUyxHQUFHLElBQUk7QUFBQSxJQUM1QyxHQUFHLE9BQU87QUFFVixTQUFLLGNBQWMsTUFBTSxtQkFBbUIsRUFBRSxDQUFDO0FBQy9DLFdBQU87QUFBQSxFQUNUO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBLEVBYUEsaUJBQ0UsUUFDQSxNQUNBLFNBQ0EsU0FDQTtBQUNBLFdBQU8sbUJBQW1CLE1BQU0sU0FBUyxPQUFPO0FBQ2hELFNBQUs7QUFBQSxNQUNILE1BQU0sT0FBTyxzQkFBc0IsTUFBTSxTQUFTLE9BQU87QUFBQSxJQUMzRDtBQUFBLEVBQ0Y7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBLEVBTUEsb0JBQW9CO0FBQ2xCLFNBQUssTUFBTSxvQ0FBb0M7QUFDL0MsV0FBTztBQUFBLE1BQ0wsbUJBQW1CLEtBQUssaUJBQWlCO0FBQUEsSUFDM0M7QUFBQSxFQUNGO0FBQUEsRUFFQSxrQkFBa0I7QUFFaEIsV0FBTztBQUFBLE1BQ0w7QUFBQSxRQUNFLE9BQU8sc0JBQXFCO0FBQUEsUUFDNUIsbUJBQW1CLEtBQUs7QUFBQSxNQUMxQjtBQUFBLE1BQ0E7QUFBQSxJQUNGO0FBQUEsRUFDRjtBQUFBLEVBRUEseUJBQXlCO0FBQ3ZCLFVBQU0sS0FBSyxDQUFDLFVBQXdCO0FBQ2xDLFVBQ0UsTUFBTSxNQUFNLFNBQVMsc0JBQXFCLCtCQUMxQyxNQUFNLE1BQU0sc0JBQXNCLEtBQUssbUJBQ3ZDO0FBQ0EsYUFBSyxrQkFBa0I7QUFBQSxNQUN6QjtBQUFBLElBQ0Y7QUFFQSxxQkFBaUIsV0FBVyxFQUFFO0FBQzlCLFNBQUssY0FBYyxNQUFNLG9CQUFvQixXQUFXLEVBQUUsQ0FBQztBQUFBLEVBQzdEO0FBQ0Y7IiwKICAibmFtZXMiOiBbXQp9Cg==