experimental-agent 0.7.1 → 0.8.0-alpha.2

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.
@@ -0,0 +1,3779 @@
1
+ import {
2
+ DEFAULT_WAIT_UNTIL
3
+ } from "./chunk-C4VSUEY2.mjs";
4
+ import {
5
+ kvStorageFallback,
6
+ localStorage,
7
+ shouldFallbackToKV2,
8
+ toStorage
9
+ } from "./chunk-CS2SEUAA.mjs";
10
+ import {
11
+ createLogger
12
+ } from "./chunk-MSWINCCM.mjs";
13
+
14
+ // src/errors.ts
15
+ import * as errore from "errore";
16
+ var SessionNotFoundError = class extends errore.createTaggedError({
17
+ name: "SessionNotFoundError",
18
+ message: "Session $id not found"
19
+ }) {
20
+ };
21
+ var SessionError = class extends errore.createTaggedError({
22
+ name: "SessionError",
23
+ message: "Session $id failed: $reason"
24
+ }) {
25
+ };
26
+ var SandboxNotFoundError = class extends errore.createTaggedError({
27
+ name: "SandboxNotFoundError",
28
+ message: "Sandbox $id not found"
29
+ }) {
30
+ };
31
+ var StorageError = class extends errore.createTaggedError({
32
+ name: "StorageError",
33
+ message: "$reason"
34
+ }) {
35
+ };
36
+ var StorageConflictError = class extends errore.createTaggedError({
37
+ name: "StorageConflictError",
38
+ message: "$reason"
39
+ }) {
40
+ };
41
+ var SandboxError = class extends errore.createTaggedError({
42
+ name: "SandboxError",
43
+ message: "$reason"
44
+ }) {
45
+ };
46
+ var MessageNotFoundError = class extends errore.createTaggedError({
47
+ name: "MessageNotFoundError",
48
+ message: "Message $id not found"
49
+ }) {
50
+ };
51
+
52
+ // src/tools/tool.ts
53
+ import {
54
+ tool as aiTool
55
+ } from "ai";
56
+ function tool(def) {
57
+ const { execute, contextSchema, stateSchema, ...rest } = def;
58
+ const wrappedExecute = execute ? (
59
+ // biome-ignore lint/suspicious/noExplicitAny: input type erased at runtime
60
+ (input, options) => {
61
+ const ctx = options.experimental_context;
62
+ const context = contextSchema ? contextSchema.parse(ctx.context) : ctx.context;
63
+ if (stateSchema) {
64
+ stateSchema.parse(ctx.state);
65
+ }
66
+ const state = ctx.state;
67
+ return execute(input, {
68
+ toolCallId: options.toolCallId,
69
+ abortSignal: options.abortSignal,
70
+ context,
71
+ state,
72
+ sessionId: ctx.sessionId,
73
+ messages: ctx.messages,
74
+ sandbox: ctx.sandbox
75
+ });
76
+ }
77
+ ) : void 0;
78
+ return wrappedExecute ? aiTool({ ...rest, execute: wrappedExecute }) : (
79
+ // biome-ignore lint/suspicious/noExplicitAny: no-execute path needs cast
80
+ aiTool(rest)
81
+ );
82
+ }
83
+
84
+ // src/tools/index.ts
85
+ import {
86
+ tool as tool3
87
+ } from "ai";
88
+ import { ulid as ulid2 } from "ulid";
89
+ import { z as z2 } from "zod";
90
+
91
+ // src/session/send.ts
92
+ import { all } from "better-all";
93
+ import * as errore2 from "errore";
94
+ import { ulid } from "ulid";
95
+
96
+ // src/session/ai-loop/stream-text.ts
97
+ import {
98
+ convertToModelMessages,
99
+ createUIMessageStream,
100
+ stepCountIs,
101
+ streamText as streamTextAi
102
+ } from "ai";
103
+
104
+ // src/skills/parser.ts
105
+ function parseSkillFrontmatter(content) {
106
+ const trimmed = content.trim();
107
+ if (!trimmed.startsWith("---")) {
108
+ return null;
109
+ }
110
+ const endMarkerIndex = trimmed.indexOf("---", 3);
111
+ if (endMarkerIndex === -1) {
112
+ return null;
113
+ }
114
+ const frontmatterBlock = trimmed.slice(3, endMarkerIndex).trim();
115
+ const parsed = parseSimpleYaml(frontmatterBlock);
116
+ if (!(parsed.name && parsed.description)) {
117
+ return null;
118
+ }
119
+ return {
120
+ name: String(parsed.name),
121
+ description: String(parsed.description)
122
+ };
123
+ }
124
+ function parseSimpleYaml(yaml) {
125
+ const result = {};
126
+ for (const line of yaml.split("\n")) {
127
+ const trimmedLine = line.trim();
128
+ if (!trimmedLine || trimmedLine.startsWith("#")) {
129
+ continue;
130
+ }
131
+ const colonIndex = trimmedLine.indexOf(":");
132
+ if (colonIndex === -1) {
133
+ continue;
134
+ }
135
+ const key = trimmedLine.slice(0, colonIndex).trim();
136
+ let value = trimmedLine.slice(colonIndex + 1).trim();
137
+ if (value.startsWith('"') && value.endsWith('"') || value.startsWith("'") && value.endsWith("'")) {
138
+ value = value.slice(1, -1);
139
+ }
140
+ if (key) {
141
+ result[key] = value;
142
+ }
143
+ }
144
+ return result;
145
+ }
146
+ function normalizeSkills(skills) {
147
+ if (!skills || skills.length === 0) {
148
+ return [];
149
+ }
150
+ return skills.map((skill) => {
151
+ if (typeof skill !== "object" || skill === null) {
152
+ throw new Error(
153
+ "Invalid skills entry: expected an object with a `type` field."
154
+ );
155
+ }
156
+ if ("type" in skill && skill.type) {
157
+ return skill;
158
+ }
159
+ throw new Error("Invalid skills entry: expected `type`.");
160
+ });
161
+ }
162
+
163
+ // src/skills/discover.ts
164
+ async function discoverSkillsInSandbox(opts) {
165
+ const { agent: agent2, sandbox, skillsDirs, sessionId } = opts;
166
+ const baseLog = createLogger({
167
+ config: { ...agent2.options.logging, name: agent2.name },
168
+ subsystem: "skills"
169
+ });
170
+ const log2 = sessionId ? baseLog.withContext({ sessionId }) : baseLog;
171
+ const done = log2.time("discoverSkillsInSandbox");
172
+ const summaries = [];
173
+ const seenNames = /* @__PURE__ */ new Set();
174
+ for (const skillsDir of skillsDirs) {
175
+ const dirDone = log2.time("scan directory", { dir: skillsDir });
176
+ const dirSummaries = await discoverSkillsInDirectory({
177
+ sandbox,
178
+ skillsDir,
179
+ log: log2
180
+ });
181
+ dirDone({ count: dirSummaries.length });
182
+ for (const summary of dirSummaries) {
183
+ if (!seenNames.has(summary.name)) {
184
+ seenNames.add(summary.name);
185
+ summaries.push(summary);
186
+ }
187
+ }
188
+ }
189
+ done({ totalSkills: summaries.length });
190
+ return summaries;
191
+ }
192
+ async function discoverSkillsInDirectory(opts) {
193
+ const { sandbox, skillsDir, log: log2 } = opts;
194
+ const skillPaths = await findSkillFiles({ sandbox, skillsDir, log: log2 });
195
+ if (skillPaths.length === 0) {
196
+ return [];
197
+ }
198
+ const summaries = [];
199
+ for (const skillMdPath of skillPaths) {
200
+ const summary = await parseSkillFile({ sandbox, skillMdPath, log: log2 });
201
+ if (summary) {
202
+ summaries.push(summary);
203
+ }
204
+ }
205
+ return summaries;
206
+ }
207
+ async function findSkillFiles(opts) {
208
+ const { sandbox, skillsDir, log: log2 } = opts;
209
+ const execResult = await sandbox.exec({
210
+ command: "find",
211
+ args: [skillsDir, "-name", "SKILL.md", "-type", "f"]
212
+ });
213
+ if (execResult instanceof Error) {
214
+ log2.warn("failed to scan skills directory", {
215
+ dir: skillsDir,
216
+ error: execResult.message
217
+ });
218
+ return [];
219
+ }
220
+ const { stdout, exitCode } = await execResult.result;
221
+ if (exitCode !== 0) {
222
+ log2.warn("skills directory not found or inaccessible", { dir: skillsDir });
223
+ return [];
224
+ }
225
+ return stdout.trim().split("\n").filter((p) => p.length > 0);
226
+ }
227
+ async function parseSkillFile(opts) {
228
+ const { sandbox, skillMdPath, log: log2 } = opts;
229
+ const execResult = await sandbox.exec({
230
+ command: "cat",
231
+ args: [skillMdPath]
232
+ });
233
+ if (execResult instanceof Error) {
234
+ log2.warn("failed to read skill file", {
235
+ path: skillMdPath,
236
+ error: execResult.message
237
+ });
238
+ return null;
239
+ }
240
+ const { stdout, exitCode } = await execResult.result;
241
+ if (exitCode !== 0) {
242
+ log2.warn("could not read skill file", { path: skillMdPath });
243
+ return null;
244
+ }
245
+ const parsed = parseSkillFrontmatter(stdout);
246
+ if (!parsed) {
247
+ log2.warn("invalid or missing frontmatter", { path: skillMdPath });
248
+ return null;
249
+ }
250
+ return {
251
+ name: parsed.name,
252
+ description: parsed.description,
253
+ skillMdPath
254
+ };
255
+ }
256
+
257
+ // src/skills/resolve-host.ts
258
+ async function collectFiles(opts) {
259
+ const { readdir, readFile: readFile2, stat } = await import("fs/promises");
260
+ const { resolve: resolve2, posix: pathPosix } = await import("path");
261
+ const fileStat = await stat(opts.absolutePath);
262
+ if (fileStat.isFile()) {
263
+ const content = await readFile2(opts.absolutePath);
264
+ return [{ path: opts.relativePrefix ?? "SKILL.md", content }];
265
+ }
266
+ if (!fileStat.isDirectory()) {
267
+ return [];
268
+ }
269
+ const entries = await readdir(opts.absolutePath, { withFileTypes: true });
270
+ const files = [];
271
+ for (const entry of entries) {
272
+ const nextAbs = resolve2(opts.absolutePath, entry.name);
273
+ const nextRel = opts.relativePrefix ? pathPosix.join(opts.relativePrefix, entry.name) : entry.name;
274
+ if (entry.isDirectory()) {
275
+ files.push(
276
+ ...await collectFiles({
277
+ absolutePath: nextAbs,
278
+ relativePrefix: nextRel
279
+ })
280
+ );
281
+ continue;
282
+ }
283
+ if (entry.isFile()) {
284
+ files.push({
285
+ path: nextRel,
286
+ content: await readFile2(nextAbs)
287
+ });
288
+ }
289
+ }
290
+ return files;
291
+ }
292
+ async function stableDirName(inputPath) {
293
+ const { createHash } = await import("crypto");
294
+ const hash = createHash("sha1").update(inputPath).digest("hex").slice(0, 10);
295
+ return `host-${hash}`;
296
+ }
297
+ async function resolveHostSkill(opts) {
298
+ const { resolve: resolve2, posix: pathPosix } = await import("path");
299
+ const hostPath = resolve2(process.cwd(), opts.skill.path);
300
+ const files = await collectFiles({ absolutePath: hostPath });
301
+ const dirName = await stableDirName(hostPath);
302
+ const destPath = pathPosix.join(".agent", "skills", ".materialized", dirName);
303
+ if (files.length === 0) {
304
+ return destPath;
305
+ }
306
+ await opts.sandbox.writeFiles({
307
+ files,
308
+ destPath
309
+ });
310
+ return destPath;
311
+ }
312
+
313
+ // src/skills/resolve-inline.ts
314
+ function slugify(value) {
315
+ return value.toLowerCase().replace(/[^a-z0-9]+/g, "-").replace(/(^-|-$)/g, "");
316
+ }
317
+ async function stableInlineDirName(name) {
318
+ const { createHash } = await import("crypto");
319
+ const slug = slugify(name) || "inline-skill";
320
+ const hash = createHash("sha1").update(name).digest("hex").slice(0, 8);
321
+ return `${slug}-${hash}`;
322
+ }
323
+ function buildSkillMarkdown(skill) {
324
+ return `---
325
+ name: ${skill.name}
326
+ description: ${skill.description}
327
+ ---
328
+
329
+ ${skill.instructions.trim()}
330
+ `;
331
+ }
332
+ async function resolveInlineSkill(opts) {
333
+ const { posix: pathPosix } = await import("path");
334
+ const dirName = await stableInlineDirName(opts.skill.name);
335
+ const destPath = pathPosix.join(".agent", "skills", ".materialized", dirName);
336
+ await opts.sandbox.writeFiles({
337
+ destPath,
338
+ files: [
339
+ {
340
+ path: "SKILL.md",
341
+ content: buildSkillMarkdown(opts.skill)
342
+ }
343
+ ]
344
+ });
345
+ return destPath;
346
+ }
347
+
348
+ // src/skills/resolve-remote.ts
349
+ function shellQuote(value) {
350
+ return `'${value.replace(/'/g, `'\\''`)}'`;
351
+ }
352
+ async function stableRepoDirName(skill) {
353
+ const { createHash } = await import("crypto");
354
+ const hashInput = `${skill.repo}::${skill.ref ?? "HEAD"}`;
355
+ const hash = createHash("sha1").update(hashInput).digest("hex").slice(0, 12);
356
+ return `git-${hash}`;
357
+ }
358
+ async function resolveRemoteSkill(opts) {
359
+ const { posix: pathPosix } = await import("path");
360
+ const dirName = await stableRepoDirName(opts.skill);
361
+ const cloneDir = pathPosix.join(".agent", "skills", ".materialized", dirName);
362
+ const quotedDir = shellQuote(cloneDir);
363
+ const quotedRepo = shellQuote(opts.skill.repo);
364
+ const quotedRef = shellQuote(opts.skill.ref ?? "HEAD");
365
+ const syncCommand = [
366
+ "set -euo pipefail",
367
+ `if [ ! -d ${quotedDir}/.git ]; then`,
368
+ ` rm -rf ${quotedDir}`,
369
+ ` mkdir -p ${quotedDir}`,
370
+ ` git clone --depth 1 ${quotedRepo} ${quotedDir}`,
371
+ "fi",
372
+ `if [ ${quotedRef} != 'HEAD' ]; then`,
373
+ ` git -C ${quotedDir} fetch --depth 1 origin ${quotedRef}`,
374
+ ` git -C ${quotedDir} checkout FETCH_HEAD`,
375
+ "fi"
376
+ ].join("\n");
377
+ const execResult = await opts.sandbox.exec({
378
+ command: "bash",
379
+ args: ["-lc", syncCommand]
380
+ });
381
+ const { exitCode, stderr } = await execResult.result;
382
+ if (exitCode !== 0) {
383
+ throw new Error(
384
+ `Failed to materialize git skill from ${opts.skill.repo}: ${stderr || "unknown error"}`
385
+ );
386
+ }
387
+ let resolvedDir = cloneDir;
388
+ if (opts.skill.path) {
389
+ resolvedDir = pathPosix.join(resolvedDir, opts.skill.path);
390
+ }
391
+ if (opts.skill.name) {
392
+ resolvedDir = pathPosix.join(resolvedDir, opts.skill.name);
393
+ }
394
+ return resolvedDir;
395
+ }
396
+
397
+ // src/skills/resolve.ts
398
+ async function resolveSkillInputsToSandboxDirs(opts) {
399
+ const baseLog = createLogger({
400
+ config: { ...opts.agent.options.logging, name: opts.agent.name },
401
+ subsystem: "skills"
402
+ });
403
+ const log2 = opts.sessionId ? baseLog.withContext({ sessionId: opts.sessionId }) : baseLog;
404
+ const entries = normalizeSkills(opts.skills);
405
+ const resolved = [];
406
+ for (const entry of entries) {
407
+ if (entry.type === "sandbox") {
408
+ resolved.push(entry.path);
409
+ continue;
410
+ }
411
+ if (entry.type === "host") {
412
+ resolved.push(
413
+ await resolveHostSkill({
414
+ sandbox: opts.sandbox,
415
+ skill: entry
416
+ })
417
+ );
418
+ continue;
419
+ }
420
+ if (entry.type === "git") {
421
+ resolved.push(
422
+ await resolveRemoteSkill({
423
+ sandbox: opts.sandbox,
424
+ skill: entry
425
+ })
426
+ );
427
+ continue;
428
+ }
429
+ if (entry.type === "inline") {
430
+ resolved.push(
431
+ await resolveInlineSkill({
432
+ sandbox: opts.sandbox,
433
+ skill: entry
434
+ })
435
+ );
436
+ continue;
437
+ }
438
+ log2.warn("unknown skill source entry encountered", { entry });
439
+ }
440
+ return resolved;
441
+ }
442
+
443
+ // src/utils/prompt-cache.ts
444
+ var ANTHROPIC_MAX_CACHE_BREAKPOINTS_PER_REQUEST = 4;
445
+ var CLAUDE_LIKE_MODEL_MATCHERS = ["claude", "anthropic"];
446
+ var CLAUDE_PROMPT_CACHING_PROVIDER_OPTIONS = {
447
+ anthropic: {
448
+ cacheControl: { type: "ephemeral" }
449
+ },
450
+ openrouter: {
451
+ cacheControl: { type: "ephemeral" }
452
+ },
453
+ bedrock: {
454
+ cachePoint: { type: "default" }
455
+ },
456
+ openaiCompatible: {
457
+ cache_control: { type: "ephemeral" }
458
+ },
459
+ copilot: {
460
+ copilot_cache_control: { type: "ephemeral" }
461
+ }
462
+ };
463
+ function getGatewayProvider(model) {
464
+ const slashIndex = model.indexOf("/");
465
+ if (slashIndex === -1) {
466
+ return null;
467
+ }
468
+ return model.slice(0, slashIndex);
469
+ }
470
+ function countAnthropicCacheBreakpoints(messages) {
471
+ let count = 0;
472
+ for (const message of messages) {
473
+ const providerOptions = message.providerOptions;
474
+ if (providerOptions?.anthropic?.cacheControl) {
475
+ count += 1;
476
+ }
477
+ if (Array.isArray(message.content)) {
478
+ for (const part of message.content) {
479
+ const partProviderOptions = part.providerOptions;
480
+ if (partProviderOptions?.anthropic?.cacheControl) {
481
+ count += 1;
482
+ }
483
+ }
484
+ }
485
+ }
486
+ return count;
487
+ }
488
+ function isClaudeLikeModel(model) {
489
+ const lower = model.toLowerCase();
490
+ return CLAUDE_LIKE_MODEL_MATCHERS.some((m) => lower.includes(m));
491
+ }
492
+ function mergeProviderOptions(opts) {
493
+ const next = { ...opts.current ?? {} };
494
+ for (const [key, value] of Object.entries(opts.patch)) {
495
+ next[key] = { ...next[key] ?? {}, ...value };
496
+ }
497
+ return next;
498
+ }
499
+ function isCacheableClaudePart(part) {
500
+ if (!part || typeof part !== "object") {
501
+ return false;
502
+ }
503
+ if ("type" in part && part.type === "text") {
504
+ const text = part.text;
505
+ if (typeof text === "string") {
506
+ return text.trim().length > 0;
507
+ }
508
+ }
509
+ if ("type" in part) {
510
+ const type = part.type;
511
+ if (type === "thinking" || type === "reasoning") {
512
+ return false;
513
+ }
514
+ }
515
+ return true;
516
+ }
517
+ function hasCacheableContent(message) {
518
+ if (typeof message.content === "string") {
519
+ return message.content.trim().length > 0;
520
+ }
521
+ if (Array.isArray(message.content)) {
522
+ return message.content.some((part) => isCacheableClaudePart(part));
523
+ }
524
+ return false;
525
+ }
526
+ function findLastCacheablePartIndex(content) {
527
+ for (let i = content.length - 1; i >= 0; i -= 1) {
528
+ if (isCacheableClaudePart(content[i])) {
529
+ return i;
530
+ }
531
+ }
532
+ return null;
533
+ }
534
+ function selectClaudeCachingTargets(messages) {
535
+ const systemIndices = [];
536
+ const finalIndices = [];
537
+ for (let i = 0; i < messages.length && systemIndices.length < 2; i += 1) {
538
+ const message = messages[i];
539
+ if (message.role === "system" && hasCacheableContent(message)) {
540
+ systemIndices.push(i);
541
+ }
542
+ }
543
+ for (let i = messages.length - 1; i >= 0 && finalIndices.length < 2; i -= 1) {
544
+ const message = messages[i];
545
+ if (message.role !== "system" && hasCacheableContent(message)) {
546
+ finalIndices.push(i);
547
+ }
548
+ }
549
+ finalIndices.reverse();
550
+ return [...systemIndices, ...finalIndices];
551
+ }
552
+ function applyClaudePromptCaching(opts) {
553
+ const existingBreakpoints = countAnthropicCacheBreakpoints(opts.messages);
554
+ const remainingBudget = Math.max(
555
+ 0,
556
+ opts.maxBreakpointsPerRequest - existingBreakpoints
557
+ );
558
+ if (remainingBudget === 0) {
559
+ return opts.messages;
560
+ }
561
+ const targetIndices = selectClaudeCachingTargets(opts.messages).slice(
562
+ 0,
563
+ remainingBudget
564
+ );
565
+ if (targetIndices.length === 0) {
566
+ return opts.messages;
567
+ }
568
+ const useMessageLevelOptions = opts.providerId === "anthropic" || Boolean(opts.providerId?.includes("bedrock"));
569
+ const nextMessages = opts.messages.slice();
570
+ for (const messageIndex of targetIndices) {
571
+ const message = nextMessages[messageIndex];
572
+ const shouldUseContentOptions = !useMessageLevelOptions && Array.isArray(message.content) && message.content.length > 0;
573
+ if (shouldUseContentOptions && Array.isArray(message.content)) {
574
+ const partIndex = findLastCacheablePartIndex(message.content);
575
+ if (partIndex !== null) {
576
+ const part = message.content[partIndex];
577
+ if (part && typeof part === "object") {
578
+ const partProviderOptions = part.providerOptions;
579
+ const nextContent = message.content.slice();
580
+ nextContent[partIndex] = {
581
+ ...part,
582
+ providerOptions: mergeProviderOptions({
583
+ current: partProviderOptions,
584
+ patch: CLAUDE_PROMPT_CACHING_PROVIDER_OPTIONS
585
+ })
586
+ };
587
+ nextMessages[messageIndex] = {
588
+ ...message,
589
+ content: nextContent
590
+ };
591
+ continue;
592
+ }
593
+ }
594
+ }
595
+ const messageProviderOptions = message.providerOptions;
596
+ nextMessages[messageIndex] = {
597
+ ...message,
598
+ providerOptions: mergeProviderOptions({
599
+ current: messageProviderOptions,
600
+ patch: CLAUDE_PROMPT_CACHING_PROVIDER_OPTIONS
601
+ })
602
+ };
603
+ }
604
+ return nextMessages;
605
+ }
606
+ function applyPromptCachingToModelRequest(opts) {
607
+ const provider = getGatewayProvider(opts.model);
608
+ const providerOptions = {};
609
+ if ((provider === "openai" || provider === "azure") && opts.openai?.setPromptCacheKey !== false) {
610
+ providerOptions.openai = { promptCacheKey: opts.sessionId };
611
+ }
612
+ const maxBreakpointsPerRequest = opts.anthropic?.maxBreakpointsPerRequest ?? ANTHROPIC_MAX_CACHE_BREAKPOINTS_PER_REQUEST;
613
+ const cachedMessages = isClaudeLikeModel(opts.model) ? applyClaudePromptCaching({
614
+ messages: opts.messages,
615
+ providerId: provider,
616
+ maxBreakpointsPerRequest
617
+ }) : opts.messages;
618
+ return {
619
+ messages: cachedMessages,
620
+ providerOptions
621
+ };
622
+ }
623
+
624
+ // src/utils/ui.ts
625
+ function isSubToolPart(part) {
626
+ const p = part.part;
627
+ return typeof p?.toolCallId === "string" && p.toolCallId.startsWith("js_tc_");
628
+ }
629
+ function isStreamingContent(part) {
630
+ return typeof part === "object" && part !== null && "type" in part && (part.type === "text" || part.type === "reasoning");
631
+ }
632
+ function applyInterruptCutoff({
633
+ parts,
634
+ lastPart
635
+ }) {
636
+ return parts.filter((p) => p.index <= lastPart.index).map((p) => {
637
+ if (p.index !== lastPart.index) {
638
+ return p;
639
+ }
640
+ if (isStreamingContent(lastPart.part)) {
641
+ return { ...p, part: lastPart.part };
642
+ }
643
+ return p;
644
+ });
645
+ }
646
+ function assembleUIMessages(opts) {
647
+ let filtered = opts.messages;
648
+ if (opts.until !== void 0) {
649
+ const until = opts.until;
650
+ filtered = filtered.filter((m) => m.createdAt <= until);
651
+ }
652
+ if (!opts.includeQueued) {
653
+ filtered = filtered.filter((m) => m.startedAt !== null);
654
+ }
655
+ filtered = [...filtered].sort((a, b) => a.createdAt - b.createdAt);
656
+ const parts = opts.excludeSubToolParts ? opts.parts.filter((p) => !isSubToolPart(p)) : opts.parts;
657
+ const partsByMessage = /* @__PURE__ */ new Map();
658
+ for (const part of parts) {
659
+ const messageId = String(part.messageId);
660
+ const existing = partsByMessage.get(messageId) ?? [];
661
+ existing.push(part);
662
+ partsByMessage.set(messageId, existing);
663
+ }
664
+ return filtered.map((m) => {
665
+ const messageId = String(m.id);
666
+ let messageParts = partsByMessage.get(messageId) ?? [];
667
+ messageParts.sort((a, b) => a.index - b.index);
668
+ if (m.interruptedLastPart != null) {
669
+ messageParts = applyInterruptCutoff({
670
+ parts: messageParts,
671
+ lastPart: m.interruptedLastPart
672
+ });
673
+ }
674
+ return {
675
+ id: messageId,
676
+ role: m.role,
677
+ parts: messageParts.map((p) => p.part),
678
+ ...m.metadata != null ? { metadata: m.metadata } : {}
679
+ };
680
+ }).filter((m) => m.parts.length > 0);
681
+ }
682
+
683
+ // src/session/ai-loop/stream-text.ts
684
+ var INTERRUPT_POLL_INTERVAL_MS = 250;
685
+ function pollForInterrupt({
686
+ agent: agent2,
687
+ messageId,
688
+ signal,
689
+ abortController
690
+ }) {
691
+ let resolveFirstCheck;
692
+ const firstCheck = new Promise((resolve2) => {
693
+ resolveFirstCheck = resolve2;
694
+ });
695
+ const interruptedLastPartRef = {
696
+ current: null
697
+ };
698
+ const log2 = createLogger({
699
+ config: { ...agent2.options.logging, name: agent2.name },
700
+ subsystem: "ai-loop"
701
+ });
702
+ const poll = async () => {
703
+ let isFirst = true;
704
+ while (!signal.aborted) {
705
+ const message = await agent2.storage.message.get(messageId);
706
+ if (!message) {
707
+ if (isFirst) {
708
+ resolveFirstCheck();
709
+ return;
710
+ }
711
+ throw new Error(`Message ${messageId} not found`);
712
+ }
713
+ if (message.interruptedAt !== null) {
714
+ log2.info("interrupt detected", { messageId });
715
+ interruptedLastPartRef.current = message.interruptedLastPart;
716
+ abortController.abort();
717
+ if (isFirst) {
718
+ resolveFirstCheck();
719
+ }
720
+ return;
721
+ }
722
+ if (isFirst) {
723
+ isFirst = false;
724
+ resolveFirstCheck();
725
+ }
726
+ await new Promise(
727
+ (resolve2) => setTimeout(resolve2, INTERRUPT_POLL_INTERVAL_MS)
728
+ );
729
+ }
730
+ };
731
+ poll();
732
+ return { firstCheck, interruptedLastPartRef };
733
+ }
734
+ async function streamText({
735
+ stepThroughput,
736
+ assistantMessageId,
737
+ agent: agent2,
738
+ sessionId,
739
+ context,
740
+ state,
741
+ createdAt,
742
+ writable,
743
+ lastPartIndex,
744
+ stepIndexOffset,
745
+ discoveredSkills
746
+ }) {
747
+ const log2 = createLogger({
748
+ config: { ...agent2.options.logging, name: agent2.name },
749
+ subsystem: "ai-loop"
750
+ });
751
+ const stepLog = log2.withContext({
752
+ sessionId,
753
+ messageId: assistantMessageId
754
+ });
755
+ const doneStep = stepLog.time("streamText");
756
+ const throttledEmitStatus = createThrottledEmitStatus(agent2);
757
+ const abortController = new AbortController();
758
+ const pollController = new AbortController();
759
+ const { firstCheck, interruptedLastPartRef } = pollForInterrupt({
760
+ agent: agent2,
761
+ messageId: assistantMessageId,
762
+ signal: pollController.signal,
763
+ abortController
764
+ });
765
+ const doneLoad = stepLog.time("load session + messages + parts + sandbox");
766
+ const [, messagesResult, partsResult, { sandbox, sandboxRecord }] = await Promise.all([
767
+ firstCheck,
768
+ agent2.storage.message.listBySession(sessionId),
769
+ agent2.storage.part.listBySession(sessionId),
770
+ agent2.storage.session.get(sessionId).then(async (session) => {
771
+ if (!session) {
772
+ throw new SessionNotFoundError({ id: sessionId });
773
+ }
774
+ const sandboxId = session.sandboxId ?? session.id;
775
+ const sandbox2 = agent2.sandbox(sandboxId);
776
+ const sandboxRecord2 = await agent2.storage.sandbox.get(sandboxId);
777
+ return { sandbox: sandbox2, sandboxRecord: sandboxRecord2 };
778
+ })
779
+ ]);
780
+ doneLoad();
781
+ if (abortController.signal.aborted) {
782
+ pollController.abort();
783
+ return {
784
+ finishReason: "stop",
785
+ lastPartIndex,
786
+ usageSteps: [],
787
+ pendingApprovals: [],
788
+ maxSteps: void 0,
789
+ discoveredSkills,
790
+ context
791
+ };
792
+ }
793
+ const setStartedPromise = lastPartIndex === 0 ? (async () => {
794
+ const now = Date.now();
795
+ const pendingMessages = messagesResult.filter(
796
+ (m) => m.createdAt <= createdAt && m.startedAt === null
797
+ );
798
+ if (pendingMessages.length > 0) {
799
+ return await Promise.all(
800
+ pendingMessages.map(
801
+ (m) => agent2.storage.message.update(m.id, {
802
+ startedAt: now,
803
+ completedAt: m.role === "assistant" ? null : now
804
+ })
805
+ )
806
+ );
807
+ }
808
+ })() : null;
809
+ const allParts = [...partsResult];
810
+ const nextPartIndexRef = {
811
+ current: Math.max(
812
+ lastPartIndex,
813
+ ...allParts.filter((p) => p.messageId === assistantMessageId).map((p) => p.index + 1),
814
+ 0
815
+ )
816
+ };
817
+ const skillsRef = {
818
+ current: discoveredSkills ?? []
819
+ };
820
+ let uiMessages = assembleUIMessages({
821
+ messages: messagesResult,
822
+ parts: allParts,
823
+ until: createdAt,
824
+ includeQueued: true,
825
+ excludeSubToolParts: true
826
+ });
827
+ const resolvableArgs = {
828
+ context,
829
+ state,
830
+ sessionId,
831
+ sandbox,
832
+ get messages() {
833
+ return uiMessages;
834
+ }
835
+ };
836
+ if (lastPartIndex === 0 && !discoveredSkills) {
837
+ const resolvedSkills = await resolveOption(
838
+ agent2.options.skills,
839
+ resolvableArgs
840
+ );
841
+ const skillsDirs = resolvedSkills?.length ? await resolveSkillInputsToSandboxDirs({
842
+ agent: agent2,
843
+ sandbox,
844
+ skills: resolvedSkills,
845
+ sessionId
846
+ }) : [];
847
+ if (skillsDirs.length > 0) {
848
+ const doneSkills = stepLog.time("discover skills", { skillsDirs });
849
+ skillsRef.current = await discoverSkillsInSandbox({
850
+ agent: agent2,
851
+ sandbox,
852
+ skillsDirs,
853
+ sessionId
854
+ });
855
+ doneSkills({ count: skillsRef.current.length });
856
+ }
857
+ }
858
+ const usageSteps = [];
859
+ let internalStepIndex = 0;
860
+ let loopFinishReason = "unknown";
861
+ let wasAborted = false;
862
+ const allPendingApprovals = [];
863
+ let currentMaxSteps;
864
+ try {
865
+ for (let iteration = 0; iteration < stepThroughput; iteration++) {
866
+ const resolvedModel = await resolveOption(
867
+ agent2.options.model,
868
+ resolvableArgs
869
+ );
870
+ const resolvedGeneration = await resolveOption(
871
+ agent2.options.generation,
872
+ resolvableArgs
873
+ );
874
+ const resolvedActiveTools = await resolveOption(
875
+ agent2.options.activeTools,
876
+ resolvableArgs
877
+ );
878
+ currentMaxSteps = resolvedGeneration?.maxSteps;
879
+ if (!resolvedModel) {
880
+ throw new Error("Agent model is not set");
881
+ }
882
+ const stepParts = [];
883
+ let streamTextFinishReason;
884
+ try {
885
+ const stream2 = createUIMessageStream({
886
+ execute: async ({ writer }) => {
887
+ const { rawTools, toolsWithNeedsApproval, toolContext } = getTools({
888
+ agent: agent2,
889
+ sandbox,
890
+ skillsRef,
891
+ writer,
892
+ abortController,
893
+ nextPartIndexRef,
894
+ assistantMessageId,
895
+ sessionId,
896
+ stepLog,
897
+ context,
898
+ state,
899
+ messages: uiMessages,
900
+ activeTools: resolvedActiveTools
901
+ });
902
+ if (iteration === 0) {
903
+ await processApprovals({
904
+ allParts,
905
+ messagesResult,
906
+ createdAt,
907
+ rawTools,
908
+ abortController,
909
+ toolContext,
910
+ agent: agent2,
911
+ stepLog,
912
+ throttledEmitStatus,
913
+ writer
914
+ });
915
+ }
916
+ throttledEmitStatus({
917
+ writer,
918
+ status: { type: "thinking" }
919
+ });
920
+ const iterResult = streamTextAi({
921
+ messages: [{ role: "user", content: "." }],
922
+ tools: toolsWithNeedsApproval,
923
+ model: resolvedModel,
924
+ abortSignal: abortController.signal,
925
+ stopWhen: stepCountIs(1),
926
+ temperature: resolvedGeneration?.temperature,
927
+ topK: resolvedGeneration?.topK,
928
+ topP: resolvedGeneration?.topP,
929
+ frequencyPenalty: resolvedGeneration?.frequencyPenalty,
930
+ presencePenalty: resolvedGeneration?.presencePenalty,
931
+ maxOutputTokens: resolvedGeneration?.maxOutputTokens,
932
+ headers: resolvedGeneration?.headers,
933
+ experimental_context: {
934
+ context,
935
+ state,
936
+ sessionId,
937
+ sandbox,
938
+ messages: uiMessages
939
+ },
940
+ prepareStep: async ({ model }) => {
941
+ const skillsContext = buildSkillsContext(skillsRef.current);
942
+ const cwdPrompt = sandboxRecord?.setup?.cwd ? `Your working directory is ${sandboxRecord.setup.cwd}. All file paths should be absolute.` : "All file paths must be absolute.";
943
+ const resolvedSystem = await resolveSystem(
944
+ agent2.options.system,
945
+ resolvableArgs
946
+ );
947
+ const systemHeader = joinPromptSections(
948
+ BASE_SYSTEM_PROMPT,
949
+ cwdPrompt,
950
+ resolvedSystem
951
+ );
952
+ const systemContext = joinPromptSections(
953
+ skillsContext,
954
+ cwdPrompt
955
+ );
956
+ const convertedMessages = await convertToModelMessages(
957
+ uiMessages,
958
+ { ignoreIncompleteToolCalls: true }
959
+ );
960
+ const modelMessages = [
961
+ ...systemHeader.trim() ? [{ role: "system", content: systemHeader }] : [],
962
+ ...systemContext.trim() ? [{ role: "system", content: systemContext }] : [],
963
+ ...convertedMessages
964
+ ];
965
+ const promptCaching = applyPromptCachingToModelRequest({
966
+ model: typeof model === "string" ? model : model.modelId,
967
+ sessionId,
968
+ messages: modelMessages
969
+ });
970
+ let activeTools = resolvedActiveTools ?? void 0;
971
+ if (skillsRef.current.length === 0 && activeTools) {
972
+ activeTools = activeTools.filter(
973
+ (t) => t !== "Skill"
974
+ );
975
+ }
976
+ return {
977
+ messages: promptCaching.messages,
978
+ providerOptions: mergeProviderOptions({
979
+ current: resolvedGeneration?.providerOptions,
980
+ patch: promptCaching.providerOptions
981
+ }),
982
+ activeTools
983
+ };
984
+ },
985
+ onStepFinish: ({ usage: usage2 }) => {
986
+ if (usage2) {
987
+ usageSteps.push({
988
+ stepIndex: stepIndexOffset + internalStepIndex,
989
+ model: resolvedModel ?? "unknown",
990
+ inputTokens: usage2.inputTokens ?? 0,
991
+ outputTokens: usage2.outputTokens ?? 0,
992
+ totalTokens: usage2.totalTokens ?? 0,
993
+ cacheReadTokens: usage2.inputTokenDetails?.cacheReadTokens ?? 0,
994
+ cacheWriteTokens: usage2.inputTokenDetails?.cacheWriteTokens ?? 0,
995
+ reasoningTokens: usage2.outputTokenDetails?.reasoningTokens ?? 0
996
+ });
997
+ }
998
+ internalStepIndex++;
999
+ }
1000
+ });
1001
+ writer.merge(
1002
+ iterResult.toUIMessageStream({
1003
+ generateMessageId: () => assistantMessageId,
1004
+ onFinish: ({ messages }) => {
1005
+ for (const m of messages) {
1006
+ if (m.role === "assistant") {
1007
+ stepParts.push(...m.parts);
1008
+ }
1009
+ }
1010
+ }
1011
+ })
1012
+ );
1013
+ streamTextFinishReason = await iterResult.finishReason;
1014
+ }
1015
+ });
1016
+ await stream2.pipeTo(writable, {
1017
+ preventClose: true,
1018
+ preventAbort: true
1019
+ });
1020
+ } catch (err) {
1021
+ if (abortController.signal.aborted) {
1022
+ wasAborted = true;
1023
+ } else {
1024
+ throw err;
1025
+ }
1026
+ }
1027
+ if (abortController.signal.aborted) {
1028
+ wasAborted = true;
1029
+ }
1030
+ if (wasAborted) {
1031
+ const terminalStates = /* @__PURE__ */ new Set([
1032
+ "output-available",
1033
+ "output-error",
1034
+ "output-denied",
1035
+ "done"
1036
+ ]);
1037
+ for (const part of stepParts) {
1038
+ if ("type" in part && typeof part.type === "string" && part.type.startsWith("tool-") && "state" in part && !terminalStates.has(part.state)) {
1039
+ part.state = "output-error";
1040
+ part.errorText = "interrupted";
1041
+ }
1042
+ }
1043
+ }
1044
+ const lastPart = interruptedLastPartRef.current;
1045
+ const newPartRecords = stepParts.map((uiPart, i) => {
1046
+ const index = nextPartIndexRef.current + i;
1047
+ if (lastPart != null && index > lastPart.index) {
1048
+ return null;
1049
+ }
1050
+ const useClientPart = lastPart != null && index === lastPart.index;
1051
+ const content = useClientPart ? lastPart.part : uiPart;
1052
+ const partId = "approval" in content && content.approval?.id ? approvalPartId(content.approval.id) : `part_${assistantMessageId}_${index}`;
1053
+ return {
1054
+ id: partId,
1055
+ index,
1056
+ messageId: assistantMessageId,
1057
+ sessionId,
1058
+ part: content
1059
+ };
1060
+ }).filter((r) => r !== null);
1061
+ const persistPromise = Promise.all([
1062
+ ...newPartRecords.map(
1063
+ (record) => agent2.storage.part.set(record.id, record)
1064
+ ),
1065
+ agent2.storage.session.update(sessionId, { state })
1066
+ ]);
1067
+ nextPartIndexRef.current = lastPart != null ? Math.min(
1068
+ nextPartIndexRef.current + stepParts.length,
1069
+ lastPart.index + 1
1070
+ ) : nextPartIndexRef.current + stepParts.length;
1071
+ const iterPendingApprovals = stepParts.filter(
1072
+ (p) => "state" in p && p.state === "approval-requested" && "approval" in p && !!p.approval?.id
1073
+ ).map((p) => ({
1074
+ approvalId: p.approval.id,
1075
+ toolName: "type" in p ? String(p.type).replace("tool-", "") : "unknown"
1076
+ }));
1077
+ loopFinishReason = wasAborted ? "stop" : streamTextFinishReason ?? "unknown";
1078
+ const shouldBreak = iterPendingApprovals.length > 0 || loopFinishReason === "stop" || wasAborted || lastPart != null || currentMaxSteps != null && stepIndexOffset + usageSteps.length >= currentMaxSteps || iteration >= stepThroughput - 1;
1079
+ if (shouldBreak) {
1080
+ await persistPromise;
1081
+ allParts.push(...newPartRecords);
1082
+ allPendingApprovals.push(...iterPendingApprovals);
1083
+ break;
1084
+ }
1085
+ await persistPromise;
1086
+ allParts.push(...newPartRecords);
1087
+ uiMessages = assembleUIMessages({
1088
+ messages: messagesResult,
1089
+ parts: allParts,
1090
+ until: createdAt,
1091
+ includeQueued: true,
1092
+ excludeSubToolParts: true
1093
+ });
1094
+ }
1095
+ } finally {
1096
+ pollController.abort();
1097
+ throttledEmitStatus.dispose();
1098
+ }
1099
+ if (setStartedPromise) {
1100
+ await setStartedPromise;
1101
+ }
1102
+ doneStep({
1103
+ finishReason: loopFinishReason,
1104
+ wasAborted,
1105
+ iterations: internalStepIndex,
1106
+ pendingApprovals: allPendingApprovals.length
1107
+ });
1108
+ return {
1109
+ finishReason: loopFinishReason,
1110
+ lastPartIndex: nextPartIndexRef.current,
1111
+ usageSteps,
1112
+ pendingApprovals: allPendingApprovals,
1113
+ maxSteps: currentMaxSteps,
1114
+ discoveredSkills: skillsRef.current,
1115
+ context
1116
+ };
1117
+ }
1118
+ async function streamTextStep(...args) {
1119
+ "use step";
1120
+ return await streamText(...args);
1121
+ }
1122
+ async function processApprovals({
1123
+ allParts,
1124
+ messagesResult,
1125
+ createdAt,
1126
+ rawTools,
1127
+ abortController,
1128
+ toolContext,
1129
+ agent: agent2,
1130
+ stepLog,
1131
+ throttledEmitStatus,
1132
+ writer
1133
+ }) {
1134
+ const approvedParts = allParts.filter(
1135
+ (p) => p.part && "state" in p.part && p.part.state === "approval-responded" && "approval" in p.part && p.part.approval?.approved === true
1136
+ );
1137
+ if (approvedParts.length > 0) {
1138
+ stepLog.info("executing approved tools", {
1139
+ count: approvedParts.length
1140
+ });
1141
+ throttledEmitStatus({
1142
+ writer,
1143
+ status: { type: "processing-approvals" }
1144
+ });
1145
+ const preExecMessages = await convertToModelMessages(
1146
+ assembleUIMessages({
1147
+ messages: messagesResult,
1148
+ parts: allParts,
1149
+ until: createdAt,
1150
+ includeQueued: true,
1151
+ excludeSubToolParts: true
1152
+ }),
1153
+ { ignoreIncompleteToolCalls: true }
1154
+ );
1155
+ await Promise.all(
1156
+ approvedParts.map(async (ap) => {
1157
+ if (!ap.part.type.startsWith("tool-")) {
1158
+ return;
1159
+ }
1160
+ const part = ap.part;
1161
+ const toolName = part.type.replace("tool-", "");
1162
+ const toolDef = rawTools[toolName];
1163
+ if (toolDef?.execute && part.input !== void 0) {
1164
+ try {
1165
+ const toolOutput = await toolDef.execute(part.input, {
1166
+ toolCallId: part.toolCallId,
1167
+ messages: preExecMessages,
1168
+ abortSignal: abortController.signal,
1169
+ experimental_context: toolContext
1170
+ });
1171
+ part.state = "output-available";
1172
+ part.output = toolOutput;
1173
+ writer.write({
1174
+ type: "tool-output-available",
1175
+ toolCallId: part.toolCallId,
1176
+ output: toolOutput
1177
+ });
1178
+ } catch (err) {
1179
+ part.state = "output-error";
1180
+ part.errorText = err instanceof Error ? err.message : String(err);
1181
+ writer.write({
1182
+ type: "tool-output-error",
1183
+ toolCallId: part.toolCallId,
1184
+ errorText: part.errorText
1185
+ });
1186
+ }
1187
+ await agent2.storage.part.set(ap.id, { ...ap, part });
1188
+ }
1189
+ })
1190
+ );
1191
+ }
1192
+ const pendingApprovalParts = allParts.filter(
1193
+ (p) => p.part.state === "approval-requested"
1194
+ );
1195
+ if (pendingApprovalParts.length > 0) {
1196
+ await Promise.all(
1197
+ pendingApprovalParts.map(async (p) => {
1198
+ const denied = {
1199
+ ...p,
1200
+ part: {
1201
+ ...p.part,
1202
+ state: "output-denied",
1203
+ approval: {
1204
+ ...p.part.approval ?? {},
1205
+ approved: false,
1206
+ reason: "auto-denied: new message sent"
1207
+ }
1208
+ }
1209
+ };
1210
+ Object.assign(p, denied);
1211
+ await agent2.storage.part.set(p.id, denied);
1212
+ })
1213
+ );
1214
+ }
1215
+ }
1216
+ function emitStatus({
1217
+ writer,
1218
+ status,
1219
+ agent: agent2
1220
+ }) {
1221
+ writer.write({ type: "data-status", data: status, transient: true });
1222
+ agent2.hooks?.status?.(status);
1223
+ }
1224
+ var STATUS_THROTTLE_MS = 500;
1225
+ function createThrottledEmitStatus(agent2) {
1226
+ let timer = null;
1227
+ let pending = null;
1228
+ function throttled({
1229
+ writer,
1230
+ status
1231
+ }) {
1232
+ pending = { writer, status };
1233
+ if (timer !== null) {
1234
+ return;
1235
+ }
1236
+ flush();
1237
+ timer = setTimeout(() => {
1238
+ timer = null;
1239
+ if (pending) {
1240
+ flush();
1241
+ }
1242
+ }, STATUS_THROTTLE_MS);
1243
+ }
1244
+ function flush() {
1245
+ if (!pending) {
1246
+ return;
1247
+ }
1248
+ const { writer, status } = pending;
1249
+ pending = null;
1250
+ emitStatus({ writer, status, agent: agent2 });
1251
+ }
1252
+ throttled.dispose = () => {
1253
+ if (timer !== null) {
1254
+ clearTimeout(timer);
1255
+ timer = null;
1256
+ }
1257
+ flush();
1258
+ };
1259
+ return throttled;
1260
+ }
1261
+ async function resolveOption(value, args) {
1262
+ if (value === void 0) {
1263
+ return void 0;
1264
+ }
1265
+ if (typeof value === "function") {
1266
+ return await value(args);
1267
+ }
1268
+ return value;
1269
+ }
1270
+ async function resolveSystem(input, args) {
1271
+ const result = await resolveOption(input, args);
1272
+ if (!result) {
1273
+ return null;
1274
+ }
1275
+ return Array.isArray(result) ? result.filter(Boolean).join("\n") : result;
1276
+ }
1277
+ var BASE_SYSTEM_PROMPT = "You are an AI assistant with basic tools to interact with your environment. Explore and work freely.";
1278
+ function joinPromptSections(...sections) {
1279
+ return sections.filter((s) => s?.trim()).join("\n\n");
1280
+ }
1281
+ var backgroundProcessPrompt = `## Background Processes
1282
+ These background process instructions are for you to manipulate the processes, do not be to verbose to the user about the response details like "how to debug the process output" the user will have an UI.
1283
+
1284
+ Use \`waitUntil: 0\` for dev servers that should run indefinitely.
1285
+ It is a good practice to check the output log after running processes like dev servers to make sure they started correctly.
1286
+
1287
+ To run a background process:
1288
+ Bash({ command: "npm run dev", waitUntil: 0 })
1289
+ Returns immediately with \`commandId\`. The process keeps running in the sandbox.
1290
+
1291
+ To kill a process:
1292
+ Bash({ command: "ps aux | grep node" }) // Find the PID
1293
+ Bash({ command: "kill {pid}" }) // Graceful
1294
+ Bash({ command: "kill -9 {pid}" }) // Force
1295
+ `;
1296
+ function buildSkillsContext(skills) {
1297
+ if (skills.length === 0) {
1298
+ return "";
1299
+ }
1300
+ const skillLines = skills.map((s) => `- ${s.name}: ${s.description}`).join("\n");
1301
+ return `## Available Skills
1302
+ ${skillLines}
1303
+
1304
+ ${backgroundProcessPrompt}
1305
+
1306
+ Use the Skill tool to load a skill's full instructions before following it.`;
1307
+ }
1308
+
1309
+ // src/session/interrupt.ts
1310
+ async function interrupt(sessionContext, opts) {
1311
+ "use step";
1312
+ const messages = await sessionContext.agent.storage.message.listBySession(
1313
+ sessionContext.sessionId
1314
+ );
1315
+ const now = Date.now();
1316
+ let pendingAssistantMessages = messages.filter(
1317
+ (m) => m.role === "assistant" && m.completedAt === null && m.interruptedAt === null
1318
+ );
1319
+ if (pendingAssistantMessages.length === 0) {
1320
+ const MAX_RETROACTIVE_MS = 5e3;
1321
+ const latest = messages.filter(
1322
+ (m) => m.role === "assistant" && m.interruptedAt === null && m.completedAt !== null && now - m.completedAt < MAX_RETROACTIVE_MS
1323
+ ).sort((a, b) => b.createdAt - a.createdAt).at(0);
1324
+ if (latest) {
1325
+ pendingAssistantMessages = [latest];
1326
+ }
1327
+ }
1328
+ await Promise.all(
1329
+ pendingAssistantMessages.map(
1330
+ (m) => sessionContext.agent.storage.message.set(m.id, {
1331
+ ...m,
1332
+ interruptedAt: now,
1333
+ interruptedLastPart: opts?.lastPart ?? null
1334
+ })
1335
+ )
1336
+ );
1337
+ const messageIds = new Set(pendingAssistantMessages.map((m) => m.id));
1338
+ if (messageIds.size === 0) {
1339
+ return;
1340
+ }
1341
+ const allParts = await sessionContext.agent.storage.part.listBySession(
1342
+ sessionContext.sessionId
1343
+ );
1344
+ for (const p of allParts) {
1345
+ if (!messageIds.has(p.messageId)) {
1346
+ continue;
1347
+ }
1348
+ if (p.part && "state" in p.part && p.part.state === "approval-requested" && "approval" in p.part) {
1349
+ const approval = p.part.approval;
1350
+ if (approval?.id) {
1351
+ await sessionContext.agent.storage.part.set(p.id, {
1352
+ ...p,
1353
+ part: {
1354
+ ...p.part,
1355
+ state: "output-denied",
1356
+ approval: {
1357
+ ...approval,
1358
+ approved: false,
1359
+ reason: "interrupted"
1360
+ }
1361
+ }
1362
+ });
1363
+ }
1364
+ }
1365
+ }
1366
+ }
1367
+
1368
+ // src/session/send.ts
1369
+ var DEFAULT_STEP_THROUGHPUT = 100;
1370
+ var ACTIVE_RUNS_KEY = /* @__PURE__ */ Symbol.for("experimental-agent:activeRuns");
1371
+ function getActiveRuns() {
1372
+ const g = globalThis;
1373
+ if (!g[ACTIVE_RUNS_KEY]) {
1374
+ g[ACTIVE_RUNS_KEY] = /* @__PURE__ */ new Map();
1375
+ }
1376
+ return g[ACTIVE_RUNS_KEY];
1377
+ }
1378
+ async function send(sessionContext, input, opts) {
1379
+ if (opts?.interruptIfStreaming) {
1380
+ const lastPart = typeof opts.interruptIfStreaming === "object" ? opts.interruptIfStreaming.lastPart : void 0;
1381
+ await interrupt(sessionContext, { lastPart });
1382
+ }
1383
+ const {
1384
+ workflowRunId,
1385
+ workflowWritable,
1386
+ existingPartCount,
1387
+ assistantMessageId,
1388
+ assistantMessageCreatedAt,
1389
+ state
1390
+ } = await errore2.tryAsync({
1391
+ try: () => initSessionStep({
1392
+ sessionContext,
1393
+ sessionId: sessionContext.sessionId,
1394
+ sandboxId: opts?.sandboxId,
1395
+ initialState: opts?.initialState,
1396
+ input
1397
+ }),
1398
+ catch: (e) => {
1399
+ if (e instanceof Error) {
1400
+ throw e;
1401
+ }
1402
+ throw new SendError({
1403
+ id: sessionContext.sessionId,
1404
+ reason: String(e),
1405
+ cause: e
1406
+ });
1407
+ }
1408
+ });
1409
+ const loopOpts = {
1410
+ sessionContext,
1411
+ assistantMessageId,
1412
+ createdAt: assistantMessageCreatedAt,
1413
+ context: opts?.context ?? {},
1414
+ state,
1415
+ lastPartIndex: existingPartCount,
1416
+ workflowRunId
1417
+ };
1418
+ if (workflowWritable) {
1419
+ await runLoop({ ...loopOpts, writable: workflowWritable });
1420
+ return { assistantMessageId, done: true };
1421
+ }
1422
+ const chunks = [];
1423
+ let notifyChunk;
1424
+ let onChunk = new Promise((r) => {
1425
+ notifyChunk = r;
1426
+ });
1427
+ const writable = new WritableStream({
1428
+ write(chunk) {
1429
+ chunks.push(chunk);
1430
+ const prev = notifyChunk;
1431
+ onChunk = new Promise((r) => {
1432
+ notifyChunk = r;
1433
+ });
1434
+ prev();
1435
+ }
1436
+ });
1437
+ const done = runLoop({ ...loopOpts, writable });
1438
+ const activeRun = {
1439
+ chunks,
1440
+ writable,
1441
+ done,
1442
+ get onChunk() {
1443
+ return onChunk;
1444
+ }
1445
+ };
1446
+ const activeRuns = getActiveRuns();
1447
+ activeRuns.set(assistantMessageId, activeRun);
1448
+ done.finally(() => activeRuns.delete(assistantMessageId));
1449
+ return { assistantMessageId, done: done.then(() => true) };
1450
+ }
1451
+ async function runLoop({
1452
+ workflowRunId,
1453
+ sessionContext,
1454
+ assistantMessageId,
1455
+ createdAt,
1456
+ context: initialContext,
1457
+ state,
1458
+ writable,
1459
+ lastPartIndex: initialPartIndex
1460
+ }) {
1461
+ const agent2 = sessionContext.agent;
1462
+ const log2 = createLogger({
1463
+ config: { ...agent2.options.logging, name: agent2.name },
1464
+ subsystem: "ai-loop"
1465
+ });
1466
+ const msgLog = log2.withContext({
1467
+ sessionId: sessionContext.sessionId,
1468
+ messageId: assistantMessageId
1469
+ });
1470
+ const doneMessage = msgLog.time("onMessage");
1471
+ let finishReason;
1472
+ let lastPartIndex = initialPartIndex;
1473
+ const usageSteps = [];
1474
+ let discoveredSkills = null;
1475
+ let context = initialContext;
1476
+ try {
1477
+ while (finishReason !== "stop") {
1478
+ const streamTextFn = workflowRunId ? streamTextStep : streamText;
1479
+ const throughput = workflowRunId ? 5 : DEFAULT_STEP_THROUGHPUT;
1480
+ const result = await streamTextFn({
1481
+ stepThroughput: throughput,
1482
+ assistantMessageId,
1483
+ agent: agent2,
1484
+ sessionId: sessionContext.sessionId,
1485
+ context,
1486
+ state,
1487
+ createdAt,
1488
+ writable,
1489
+ lastPartIndex,
1490
+ stepIndexOffset: usageSteps.length,
1491
+ discoveredSkills
1492
+ });
1493
+ finishReason = result.finishReason;
1494
+ lastPartIndex = result.lastPartIndex;
1495
+ usageSteps.push(...result.usageSteps);
1496
+ discoveredSkills = result.discoveredSkills;
1497
+ context = result.context;
1498
+ if (result.maxSteps != null && usageSteps.length >= result.maxSteps) {
1499
+ msgLog.info("reached maxSteps limit", { maxSteps: result.maxSteps });
1500
+ break;
1501
+ }
1502
+ if (result.pendingApprovals.length > 0) {
1503
+ msgLog.info("pending approvals, stopping loop", {
1504
+ count: result.pendingApprovals.length,
1505
+ tools: result.pendingApprovals.map((p) => p.toolName)
1506
+ });
1507
+ break;
1508
+ }
1509
+ }
1510
+ doneMessage({ totalSteps: usageSteps.length });
1511
+ } catch (err) {
1512
+ msgLog.error("AI loop failed", {
1513
+ error: err instanceof Error ? err.message : String(err)
1514
+ });
1515
+ throw err;
1516
+ } finally {
1517
+ await completeMessageStep({
1518
+ assistantMessageId,
1519
+ sessionContext,
1520
+ writable,
1521
+ usageSteps
1522
+ });
1523
+ }
1524
+ }
1525
+ async function initSessionStep({
1526
+ sessionContext,
1527
+ sessionId,
1528
+ sandboxId,
1529
+ initialState,
1530
+ input
1531
+ }) {
1532
+ "use step";
1533
+ let workflowRunId;
1534
+ let workflowWritable;
1535
+ try {
1536
+ const { getWorkflowMetadata, getWritable } = await import("workflow");
1537
+ workflowRunId = getWorkflowMetadata().workflowRunId;
1538
+ workflowWritable = getWritable();
1539
+ } catch {
1540
+ }
1541
+ const newMessages = [];
1542
+ const newParts = [];
1543
+ const newApprovals = [];
1544
+ if (Array.isArray(input)) {
1545
+ for (const i of input) {
1546
+ const { message, parts, approvals } = toMessageAndParts({
1547
+ sessionId: sessionContext.sessionId,
1548
+ input: i,
1549
+ defaultRole: "user"
1550
+ });
1551
+ if (message) {
1552
+ newMessages.push(message);
1553
+ }
1554
+ newParts.push(...parts);
1555
+ newApprovals.push(...approvals);
1556
+ }
1557
+ } else {
1558
+ const { message, parts, approvals } = toMessageAndParts({
1559
+ input,
1560
+ sessionId: sessionContext.sessionId,
1561
+ defaultRole: "user"
1562
+ });
1563
+ if (message) {
1564
+ newMessages.push(message);
1565
+ }
1566
+ newParts.push(...parts);
1567
+ newApprovals.push(...approvals);
1568
+ }
1569
+ const lastNewMessage = newMessages.at(-1);
1570
+ const reuseAssistant = lastNewMessage?.role === "assistant";
1571
+ const assistantMessageId = reuseAssistant ? lastNewMessage.id : `message_${ulid()}`;
1572
+ const baseTime = Date.now();
1573
+ const assistantMessageCreatedAt = baseTime + newMessages.length;
1574
+ if (!reuseAssistant) {
1575
+ newMessages.push({
1576
+ id: assistantMessageId,
1577
+ sessionId: sessionContext.sessionId,
1578
+ role: "assistant",
1579
+ createdAt: assistantMessageCreatedAt,
1580
+ startedAt: null,
1581
+ completedAt: null,
1582
+ interruptedAt: null,
1583
+ interruptedLastPart: null,
1584
+ usage: null,
1585
+ workflowRunId: null,
1586
+ metadata: null
1587
+ });
1588
+ }
1589
+ const existingPartCount = reuseAssistant ? newParts.filter((p) => p.messageId === assistantMessageId).length : 0;
1590
+ let session = await sessionContext.agent.storage.session.get(sessionId);
1591
+ const now = Date.now();
1592
+ if (!session) {
1593
+ session = {
1594
+ id: sessionId,
1595
+ sandboxId: sandboxId ?? null,
1596
+ lastMessageId: null,
1597
+ createdAt: now,
1598
+ updatedAt: now
1599
+ };
1600
+ await sessionContext.agent.storage.session.set(session.id, session);
1601
+ } else if (sandboxId && sandboxId !== session.sandboxId) {
1602
+ session = { ...session, sandboxId, updatedAt: now };
1603
+ await sessionContext.agent.storage.session.update(session.id, {
1604
+ sandboxId
1605
+ });
1606
+ }
1607
+ await all({
1608
+ async saveMessages() {
1609
+ await Promise.all(
1610
+ newMessages.map(async (m, i) => {
1611
+ const existing = await sessionContext.agent.storage.message.get(m.id);
1612
+ if (existing) {
1613
+ return;
1614
+ }
1615
+ await sessionContext.agent.storage.message.set(m.id, {
1616
+ ...m,
1617
+ createdAt: baseTime + i,
1618
+ workflowRunId: m.id === assistantMessageId ? workflowRunId ?? null : m.workflowRunId
1619
+ });
1620
+ })
1621
+ );
1622
+ },
1623
+ async saveParts() {
1624
+ await Promise.all(
1625
+ newParts.map(
1626
+ async (p) => sessionContext.agent.storage.part.set(p.id, p)
1627
+ )
1628
+ );
1629
+ },
1630
+ async resolveApprovals() {
1631
+ await Promise.all(
1632
+ newApprovals.map(async (a) => {
1633
+ const partId = approvalPartId(a.approvalId);
1634
+ const part = await sessionContext.agent.storage.part.get(partId);
1635
+ if (!part) {
1636
+ throw new ApprovalNotFoundError({
1637
+ id: sessionContext.sessionId,
1638
+ approvalId: a.approvalId
1639
+ });
1640
+ }
1641
+ await sessionContext.agent.storage.part.set(partId, {
1642
+ ...part,
1643
+ part: {
1644
+ ...part.part,
1645
+ state: a.approved ? "approval-responded" : "output-denied",
1646
+ approval: {
1647
+ id: a.approvalId,
1648
+ approved: a.approved,
1649
+ reason: a.reason
1650
+ }
1651
+ }
1652
+ });
1653
+ })
1654
+ );
1655
+ },
1656
+ async updateSession() {
1657
+ await sessionContext.agent.storage.session.set(session.id, {
1658
+ ...session,
1659
+ lastMessageId: assistantMessageId,
1660
+ updatedAt: Date.now()
1661
+ });
1662
+ }
1663
+ });
1664
+ const state = session.state ? { ...session.state } : { ...initialState ?? {} };
1665
+ return {
1666
+ workflowRunId,
1667
+ workflowWritable,
1668
+ existingPartCount,
1669
+ assistantMessageId,
1670
+ assistantMessageCreatedAt,
1671
+ state
1672
+ };
1673
+ }
1674
+ async function completeMessageStep({
1675
+ assistantMessageId,
1676
+ sessionContext,
1677
+ writable,
1678
+ usageSteps
1679
+ }) {
1680
+ "use step";
1681
+ const log2 = createLogger({
1682
+ config: {
1683
+ ...sessionContext.agent.options.logging,
1684
+ name: sessionContext.agent.name
1685
+ },
1686
+ subsystem: "ai-loop"
1687
+ });
1688
+ const msgLog = log2.withContext({
1689
+ sessionId: sessionContext.sessionId,
1690
+ messageId: assistantMessageId
1691
+ });
1692
+ const usage2 = usageSteps.length > 0 ? {
1693
+ steps: usageSteps,
1694
+ summary: {
1695
+ model: usageSteps[0]?.model ?? "unknown",
1696
+ inputTokens: usageSteps.reduce((a, s) => a + s.inputTokens, 0),
1697
+ outputTokens: usageSteps.reduce((a, s) => a + s.outputTokens, 0),
1698
+ totalTokens: usageSteps.reduce((a, s) => a + s.totalTokens, 0),
1699
+ cacheReadTokens: usageSteps.reduce(
1700
+ (a, s) => a + s.cacheReadTokens,
1701
+ 0
1702
+ ),
1703
+ cacheWriteTokens: usageSteps.reduce(
1704
+ (a, s) => a + s.cacheWriteTokens,
1705
+ 0
1706
+ ),
1707
+ reasoningTokens: usageSteps.reduce(
1708
+ (a, s) => a + s.reasoningTokens,
1709
+ 0
1710
+ ),
1711
+ stepCount: usageSteps.length
1712
+ }
1713
+ } : null;
1714
+ await Promise.all([
1715
+ sessionContext.agent.storage.message.update(assistantMessageId, {
1716
+ completedAt: Date.now(),
1717
+ usage: usage2
1718
+ }),
1719
+ writable.close().catch(() => void 0)
1720
+ ]);
1721
+ msgLog.info("message completed", {
1722
+ steps: usageSteps.length,
1723
+ totalTokens: usage2?.summary.totalTokens
1724
+ });
1725
+ }
1726
+ function approvalPartId(approvalId) {
1727
+ return `part_${approvalId}`;
1728
+ }
1729
+ function replayActiveRun(active, startIndex = 0) {
1730
+ let cursor = startIndex;
1731
+ let cancelled = false;
1732
+ return new ReadableStream({
1733
+ async pull(controller) {
1734
+ while (cursor < active.chunks.length) {
1735
+ controller.enqueue(active.chunks[cursor++]);
1736
+ }
1737
+ while (!cancelled && cursor >= active.chunks.length) {
1738
+ const result = await Promise.race([
1739
+ active.onChunk.then(() => "chunk"),
1740
+ active.done.then(() => "done")
1741
+ ]);
1742
+ while (cursor < active.chunks.length) {
1743
+ controller.enqueue(active.chunks[cursor++]);
1744
+ }
1745
+ if (result === "done") {
1746
+ controller.close();
1747
+ return;
1748
+ }
1749
+ }
1750
+ },
1751
+ cancel() {
1752
+ cancelled = true;
1753
+ }
1754
+ });
1755
+ }
1756
+ function toMessageAndParts({
1757
+ sessionId,
1758
+ id,
1759
+ input,
1760
+ defaultRole
1761
+ }) {
1762
+ let messageId = id ?? `message_${ulid()}`;
1763
+ if (typeof input === "string") {
1764
+ const now2 = Date.now();
1765
+ return {
1766
+ approvals: [],
1767
+ message: {
1768
+ id: messageId,
1769
+ sessionId,
1770
+ role: defaultRole,
1771
+ createdAt: now2,
1772
+ startedAt: null,
1773
+ completedAt: null,
1774
+ interruptedAt: null,
1775
+ interruptedLastPart: null,
1776
+ usage: null,
1777
+ workflowRunId: null,
1778
+ metadata: null
1779
+ },
1780
+ parts: [
1781
+ {
1782
+ sessionId,
1783
+ messageId,
1784
+ id: `part_${messageId}_0`,
1785
+ index: 0,
1786
+ part: { type: "text", text: input }
1787
+ }
1788
+ ]
1789
+ };
1790
+ }
1791
+ if ("type" in input && input.type === "approval") {
1792
+ return {
1793
+ message: void 0,
1794
+ parts: [],
1795
+ approvals: [input.approval]
1796
+ };
1797
+ }
1798
+ const msg = "type" in input && input.type === "message" ? input.message : input;
1799
+ if ("id" in msg && msg.id) {
1800
+ messageId = msg.id;
1801
+ }
1802
+ const now = Date.now();
1803
+ const metadata = "metadata" in msg && msg.metadata != null ? msg.metadata : null;
1804
+ return {
1805
+ approvals: [],
1806
+ message: {
1807
+ sessionId,
1808
+ id: messageId,
1809
+ createdAt: now,
1810
+ startedAt: null,
1811
+ completedAt: null,
1812
+ interruptedAt: null,
1813
+ interruptedLastPart: null,
1814
+ usage: null,
1815
+ workflowRunId: null,
1816
+ role: msg.role ?? defaultRole,
1817
+ metadata
1818
+ },
1819
+ parts: msg.parts.map((part, index) => {
1820
+ const ap = part;
1821
+ const id2 = ap.approval?.id ? approvalPartId(ap.approval.id) : `part_${messageId}_${index}`;
1822
+ return { sessionId, messageId, id: id2, index, part };
1823
+ })
1824
+ };
1825
+ }
1826
+ var SendError = class extends errore2.createTaggedError({
1827
+ name: "SendError",
1828
+ message: "Send failed for session $id: $reason"
1829
+ }) {
1830
+ };
1831
+ var ApprovalNotFoundError = class extends errore2.createTaggedError({
1832
+ name: "ApprovalNotFoundError",
1833
+ message: "Approval $approvalId not found in session $id"
1834
+ }) {
1835
+ };
1836
+
1837
+ // src/tools/javascript.ts
1838
+ import { tool as tool2 } from "ai";
1839
+ import { z } from "zod";
1840
+ var REQUEST_MARKER_START = "__TOOL_REQUEST__";
1841
+ var REQUEST_MARKER_END = "__TOOL_REQUEST_END__";
1842
+ var EXECUTION_TIMEOUT_MS = 5 * 60 * 1e3;
1843
+ var RUNNER_SCRIPT = `
1844
+ import { readFileSync, existsSync, unlinkSync, mkdirSync } from 'node:fs';
1845
+ import { join, dirname } from 'node:path';
1846
+ import { fileURLToPath } from 'node:url';
1847
+
1848
+ const SCRIPT_DIR = dirname(fileURLToPath(import.meta.url));
1849
+ const TOOL_CALL_TIMEOUT_MS = 300_000;
1850
+
1851
+ let reqCounter = 0;
1852
+
1853
+ async function callTool(name, input) {
1854
+ const id = 'req_' + (++reqCounter) + '_' + Date.now();
1855
+ const responseFile = join(SCRIPT_DIR, id + '.response.json');
1856
+
1857
+ const request = JSON.stringify({ id, tool: name, input });
1858
+ process.stderr.write('${REQUEST_MARKER_START}' + request + '${REQUEST_MARKER_END}' + '\\n');
1859
+
1860
+ const start = Date.now();
1861
+ while (!existsSync(responseFile)) {
1862
+ if (Date.now() - start > TOOL_CALL_TIMEOUT_MS) {
1863
+ throw new Error('Tool call ' + name + ' timed out waiting for response');
1864
+ }
1865
+ await new Promise(r => setTimeout(r, 30));
1866
+ }
1867
+
1868
+ let raw;
1869
+ try {
1870
+ raw = readFileSync(responseFile, 'utf8');
1871
+ } catch {
1872
+ await new Promise(r => setTimeout(r, 50));
1873
+ raw = readFileSync(responseFile, 'utf8');
1874
+ }
1875
+
1876
+ const response = JSON.parse(raw);
1877
+ try { unlinkSync(responseFile); } catch {}
1878
+
1879
+ if (response.error) {
1880
+ throw new Error(response.error);
1881
+ }
1882
+ return response.result;
1883
+ }
1884
+
1885
+ const code = readFileSync(join(SCRIPT_DIR, 'code.js'), 'utf8');
1886
+ const toolNames = JSON.parse(readFileSync(join(SCRIPT_DIR, 'tools.json'), 'utf8'));
1887
+
1888
+ const ctx = { tools: {} };
1889
+ for (const name of toolNames) {
1890
+ ctx.tools[name] = (input) => callTool(name, input);
1891
+ }
1892
+
1893
+ const logs = [];
1894
+ const customConsole = {
1895
+ log: (...args) => logs.push(args.map(a => typeof a === 'string' ? a : JSON.stringify(a)).join(' ')),
1896
+ error: (...args) => logs.push('[error] ' + args.map(a => typeof a === 'string' ? a : JSON.stringify(a)).join(' ')),
1897
+ warn: (...args) => logs.push('[warn] ' + args.map(a => typeof a === 'string' ? a : JSON.stringify(a)).join(' ')),
1898
+ };
1899
+
1900
+ try {
1901
+ const AsyncFunction = Object.getPrototypeOf(async function() {}).constructor;
1902
+ const fn = new AsyncFunction('ctx', 'console', code);
1903
+ const result = await fn(ctx, customConsole);
1904
+
1905
+ const output = { success: true, result: result !== undefined ? result : null };
1906
+ if (logs.length > 0) output.logs = logs;
1907
+ process.stdout.write(JSON.stringify(output));
1908
+ } catch (err) {
1909
+ const output = {
1910
+ success: false,
1911
+ error: err instanceof Error ? err.message : String(err),
1912
+ };
1913
+ if (logs.length > 0) output.logs = logs;
1914
+ process.stdout.write(JSON.stringify(output));
1915
+ }
1916
+ `;
1917
+ function extractJsonSchema(schema) {
1918
+ if (!schema || typeof schema !== "object") {
1919
+ return null;
1920
+ }
1921
+ if ("_def" in schema && typeof schema._def === "object") {
1922
+ try {
1923
+ return z.toJSONSchema(schema);
1924
+ } catch {
1925
+ return null;
1926
+ }
1927
+ }
1928
+ const s = schema;
1929
+ if (/* @__PURE__ */ Symbol.for("vercel.ai.schema") in s && "jsonSchema" in s) {
1930
+ return s.jsonSchema;
1931
+ }
1932
+ if ("type" in s || "properties" in s) {
1933
+ return s;
1934
+ }
1935
+ return null;
1936
+ }
1937
+ function jsonSchemaToTs(schema, indent = 0) {
1938
+ if (!schema) {
1939
+ return "unknown";
1940
+ }
1941
+ if (schema.enum && Array.isArray(schema.enum)) {
1942
+ return schema.enum.map((v) => JSON.stringify(v)).join(" | ");
1943
+ }
1944
+ if (schema.anyOf && Array.isArray(schema.anyOf)) {
1945
+ return schema.anyOf.map((s) => jsonSchemaToTs(s, indent)).join(" | ");
1946
+ }
1947
+ if (schema.oneOf && Array.isArray(schema.oneOf)) {
1948
+ return schema.oneOf.map((s) => jsonSchemaToTs(s, indent)).join(" | ");
1949
+ }
1950
+ const type = schema.type;
1951
+ switch (type) {
1952
+ case "string":
1953
+ return "string";
1954
+ case "number":
1955
+ case "integer":
1956
+ return "number";
1957
+ case "boolean":
1958
+ return "boolean";
1959
+ case "null":
1960
+ return "null";
1961
+ case "array": {
1962
+ const items = schema.items ? jsonSchemaToTs(schema.items, indent) : "unknown";
1963
+ return `Array<${items}>`;
1964
+ }
1965
+ case "object": {
1966
+ const properties = schema.properties;
1967
+ if (!properties) {
1968
+ return "Record<string, unknown>";
1969
+ }
1970
+ const required = new Set(schema.required || []);
1971
+ const pad = " ".repeat(indent + 1);
1972
+ const closePad = " ".repeat(indent);
1973
+ const props = Object.entries(properties).map(([key, value]) => {
1974
+ const opt = required.has(key) ? "" : "?";
1975
+ return `${pad}${key}${opt}: ${jsonSchemaToTs(value, indent + 1)}`;
1976
+ });
1977
+ return `{
1978
+ ${props.join(";\n")};
1979
+ ${closePad}}`;
1980
+ }
1981
+ default:
1982
+ return "unknown";
1983
+ }
1984
+ }
1985
+ function generateContextTypeString(tools) {
1986
+ const sigs = [];
1987
+ for (const [name, t] of Object.entries(tools)) {
1988
+ const raw = t;
1989
+ const inputSchema = extractJsonSchema(raw.inputSchema ?? raw.parameters);
1990
+ const outputSchema = extractJsonSchema(raw.outputSchema);
1991
+ const inputType = inputSchema ? jsonSchemaToTs(inputSchema, 2) : "{}";
1992
+ const outputType = outputSchema ? jsonSchemaToTs(outputSchema, 2) : "unknown";
1993
+ sigs.push(` ${name}(input: ${inputType}): Promise<${outputType}>`);
1994
+ }
1995
+ return `type Context = {
1996
+ tools: {
1997
+ ${sigs.join(";\n")};
1998
+ }
1999
+ }`;
2000
+ }
2001
+ async function executeInSandbox({
2002
+ code,
2003
+ rpcDir,
2004
+ abortController,
2005
+ sandbox,
2006
+ availableTools,
2007
+ onSubToolCall,
2008
+ toolContext
2009
+ }) {
2010
+ const mkdirResult = await sandbox.exec({
2011
+ command: "mkdir",
2012
+ args: ["-p", rpcDir]
2013
+ });
2014
+ if (mkdirResult instanceof Error) {
2015
+ return { success: false, error: mkdirResult.message };
2016
+ }
2017
+ await mkdirResult.result;
2018
+ await sandbox.writeFiles({
2019
+ files: [
2020
+ { path: "runner.mjs", content: RUNNER_SCRIPT },
2021
+ { path: "code.js", content: code },
2022
+ {
2023
+ path: "tools.json",
2024
+ content: JSON.stringify(Object.keys(availableTools))
2025
+ }
2026
+ ],
2027
+ destPath: rpcDir
2028
+ });
2029
+ const nodeMajor = Number.parseInt(process.versions.node.split(".")[0], 10);
2030
+ const permissionFlag = nodeMajor >= 22 ? "--permission" : "--experimental-permission";
2031
+ const execResult = await sandbox.exec({
2032
+ command: "node",
2033
+ args: [
2034
+ permissionFlag,
2035
+ `--allow-fs-read=${rpcDir}`,
2036
+ `--allow-fs-write=${rpcDir}`,
2037
+ `${rpcDir}/runner.mjs`
2038
+ ],
2039
+ signal: abortController.signal
2040
+ });
2041
+ if (execResult instanceof Error) {
2042
+ return { success: false, error: execResult.message };
2043
+ }
2044
+ const fatal = { error: null };
2045
+ const abort = (error) => {
2046
+ if (!fatal.error) {
2047
+ fatal.error = error;
2048
+ abortController.abort();
2049
+ }
2050
+ };
2051
+ const handleToolRequest = async (requestJson) => {
2052
+ let parsed;
2053
+ try {
2054
+ parsed = JSON.parse(requestJson);
2055
+ } catch {
2056
+ return;
2057
+ }
2058
+ const { id, tool: toolName, input } = parsed;
2059
+ let response;
2060
+ try {
2061
+ const t = availableTools[toolName];
2062
+ if (!t?.execute) {
2063
+ throw new Error(
2064
+ `Tool ${toolName} not found or has no execute function`
2065
+ );
2066
+ }
2067
+ const exec2 = t.execute.bind(t);
2068
+ const execOptions = {
2069
+ toolCallId: `js_${toolName}_${Date.now()}`,
2070
+ messages: [],
2071
+ experimental_context: toolContext
2072
+ };
2073
+ if (onSubToolCall) {
2074
+ response = await onSubToolCall(
2075
+ toolName,
2076
+ input,
2077
+ () => exec2(input, execOptions)
2078
+ );
2079
+ } else {
2080
+ const result = await exec2(input, execOptions);
2081
+ response = { result };
2082
+ }
2083
+ } catch (err) {
2084
+ const msg = err instanceof Error ? err.message : String(err);
2085
+ response = { error: msg };
2086
+ }
2087
+ try {
2088
+ await sandbox.writeFiles({
2089
+ files: [
2090
+ {
2091
+ path: `${id}.response.json`,
2092
+ content: JSON.stringify(response)
2093
+ }
2094
+ ],
2095
+ destPath: rpcDir
2096
+ });
2097
+ } catch (err) {
2098
+ const msg = err instanceof Error ? err.message : String(err);
2099
+ abort(new Error(`Failed to write RPC response for ${toolName}: ${msg}`));
2100
+ }
2101
+ };
2102
+ const toolCallPromises = [];
2103
+ let stderrBuffer = "";
2104
+ let nonMarkerStderr = "";
2105
+ const timeout = setTimeout(() => {
2106
+ abort(
2107
+ new Error(
2108
+ `JavaScript execution timed out after ${EXECUTION_TIMEOUT_MS / 1e3}s`
2109
+ )
2110
+ );
2111
+ }, EXECUTION_TIMEOUT_MS);
2112
+ try {
2113
+ for await (const entry of execResult.logs()) {
2114
+ if (fatal.error) {
2115
+ break;
2116
+ }
2117
+ if (entry.stream === "stderr") {
2118
+ stderrBuffer += entry.data;
2119
+ while (true) {
2120
+ const startIdx = stderrBuffer.indexOf(REQUEST_MARKER_START);
2121
+ if (startIdx === -1) {
2122
+ break;
2123
+ }
2124
+ const beforeMarker = stderrBuffer.slice(0, startIdx);
2125
+ if (beforeMarker.trim()) {
2126
+ nonMarkerStderr += beforeMarker;
2127
+ }
2128
+ const contentStart = startIdx + REQUEST_MARKER_START.length;
2129
+ const endIdx = stderrBuffer.indexOf(REQUEST_MARKER_END, contentStart);
2130
+ if (endIdx === -1) {
2131
+ break;
2132
+ }
2133
+ const requestJson = stderrBuffer.slice(contentStart, endIdx);
2134
+ stderrBuffer = stderrBuffer.slice(endIdx + REQUEST_MARKER_END.length);
2135
+ toolCallPromises.push(handleToolRequest(requestJson));
2136
+ }
2137
+ }
2138
+ }
2139
+ } finally {
2140
+ clearTimeout(timeout);
2141
+ }
2142
+ if (stderrBuffer.trim()) {
2143
+ nonMarkerStderr += stderrBuffer;
2144
+ }
2145
+ await Promise.allSettled(toolCallPromises);
2146
+ if (fatal.error) {
2147
+ execResult.result.catch(() => void 0);
2148
+ return {
2149
+ success: false,
2150
+ error: fatal.error.message,
2151
+ ...nonMarkerStderr ? { stderr: nonMarkerStderr.slice(0, 2e3) } : {}
2152
+ };
2153
+ }
2154
+ const { stdout, stderr, exitCode } = await execResult.result;
2155
+ try {
2156
+ return JSON.parse(stdout);
2157
+ } catch {
2158
+ return {
2159
+ success: false,
2160
+ error: `Runner failed (exit ${exitCode}).`,
2161
+ stderr: (nonMarkerStderr + stderr).slice(0, 2e3),
2162
+ stdout: stdout.slice(0, 1e3)
2163
+ };
2164
+ }
2165
+ }
2166
+ function createJavaScriptTool(opts) {
2167
+ const { sandbox, onSubToolCall } = opts;
2168
+ const activeSet = opts.activeTools ? new Set(opts.activeTools) : null;
2169
+ const availableTools = {};
2170
+ for (const [name, t] of Object.entries(opts.tools)) {
2171
+ if (!activeSet || activeSet.has(name)) {
2172
+ availableTools[name] = t;
2173
+ }
2174
+ }
2175
+ const contextType = generateContextTypeString(availableTools);
2176
+ return tool2({
2177
+ ...builtInTools.JavaScript,
2178
+ description: `Execute JavaScript to orchestrate multiple tool calls in a single step. Use this when you need to run several tools in sequence, transform intermediate results, or parallelize independent operations with Promise.all.
2179
+
2180
+ The code runs as an async function body with \`ctx\` in scope:
2181
+
2182
+ \`\`\`typescript
2183
+ ${contextType}
2184
+ \`\`\`
2185
+
2186
+ Examples:
2187
+ - Sequential: \`const file = await ctx.tools.Read({ path: "package.json" }); return JSON.parse(file.content);\`
2188
+ - Parallel: \`const [a, b] = await Promise.all([ctx.tools.Read({ path: "a.ts" }), ctx.tools.Read({ path: "b.ts" })]); return { a: a.content, b: b.content };\`
2189
+ - Transform: \`const grep = await ctx.tools.Grep({ pattern: "TODO" }); return grep.matches.split("\\n").length;\`
2190
+ - Bash: \`const r = await ctx.tools.Bash({ command: "node -e \\"console.log(JSON.stringify({v:1}))\\"" }); return JSON.parse(r.stdout);\`
2191
+
2192
+ Return a value to pass results back. Use console.log() for debug output.
2193
+ Important: Each tool returns a structured object matching its output schema \u2014 not a raw string. Access the specific field you need (e.g. Bash's result.stdout, Read's result.content) rather than trying to parse the entire result.`,
2194
+ execute: ({ code }) => {
2195
+ const runId = `run_${Date.now()}_${Math.random().toString(36).slice(2, 8)}`;
2196
+ const rpcDir = `.agent/js-rpc/${runId}`;
2197
+ const abortController = new AbortController();
2198
+ return executeInSandbox({
2199
+ code,
2200
+ rpcDir,
2201
+ abortController,
2202
+ sandbox,
2203
+ availableTools,
2204
+ onSubToolCall,
2205
+ toolContext: opts.toolContext
2206
+ });
2207
+ }
2208
+ });
2209
+ }
2210
+
2211
+ // src/tools/needs-approval.ts
2212
+ var builtinNeedsApproval = {};
2213
+ async function checkNeedsApproval({
2214
+ agent: agent2,
2215
+ toolName,
2216
+ input,
2217
+ toolCallId,
2218
+ messages
2219
+ }) {
2220
+ const toolDef = agent2.options.tools?.[toolName];
2221
+ if (toolDef?.needsApproval != null) {
2222
+ if (typeof toolDef.needsApproval === "boolean") {
2223
+ return toolDef.needsApproval;
2224
+ }
2225
+ return await toolDef.needsApproval(input, {
2226
+ toolCallId,
2227
+ messages,
2228
+ experimental_context: void 0
2229
+ });
2230
+ }
2231
+ const agentEntry = agent2.options.needsApproval?.[toolName];
2232
+ if (agentEntry != null) {
2233
+ if (typeof agentEntry === "boolean") {
2234
+ return agentEntry;
2235
+ }
2236
+ return await agentEntry(input, {
2237
+ toolCallId,
2238
+ messages,
2239
+ experimental_context: void 0
2240
+ });
2241
+ }
2242
+ return builtinNeedsApproval[toolName] ?? false;
2243
+ }
2244
+
2245
+ // src/tools/index.ts
2246
+ var log = createLogger({ config: { name: "tools" }, subsystem: "tools" });
2247
+ function formatFileSize(bytes) {
2248
+ if (bytes < 1024) {
2249
+ return `${bytes}`;
2250
+ }
2251
+ if (bytes < 1024 * 1024) {
2252
+ return `${(bytes / 1024).toFixed(1)}K`;
2253
+ }
2254
+ if (bytes < 1024 * 1024 * 1024) {
2255
+ return `${(bytes / (1024 * 1024)).toFixed(1)}M`;
2256
+ }
2257
+ return `${(bytes / (1024 * 1024 * 1024)).toFixed(1)}G`;
2258
+ }
2259
+ function isRgNotFoundError(err) {
2260
+ const parts = [err.message];
2261
+ const anyErr = err;
2262
+ if (typeof anyErr.reason === "string") {
2263
+ parts.push(anyErr.reason);
2264
+ }
2265
+ if (anyErr.cause instanceof Error) {
2266
+ parts.push(anyErr.cause.message);
2267
+ const anyCause = anyErr.cause;
2268
+ if (typeof anyCause.text === "string") {
2269
+ parts.push(anyCause.text);
2270
+ }
2271
+ }
2272
+ const msg = parts.join(" ").toLowerCase();
2273
+ return msg.includes("executable file not found") || msg.includes("executable_not_found") || msg.includes("enoent") && msg.includes("rg");
2274
+ }
2275
+ var builtInTools = {
2276
+ Read: tool3({
2277
+ description: "Reads a file and returns its contents with metadata. For files over 200 lines, automatically shows first 100 lines unless a specific line range is provided. Use startLine and endLine parameters to read specific portions of large files.",
2278
+ inputSchema: z2.object({
2279
+ label: z2.string().describe("A label that describes the action being performed"),
2280
+ path: z2.string().describe("Absolute path to the file"),
2281
+ startLine: z2.number().optional().describe(
2282
+ "Starting line number (1-indexed). If provided with endLine, reads exact range regardless of file size."
2283
+ ),
2284
+ endLine: z2.number().optional().describe(
2285
+ "Ending line number (1-indexed, inclusive). If provided with startLine, reads exact range regardless of file size."
2286
+ )
2287
+ }),
2288
+ outputSchema: z2.object({
2289
+ content: z2.string().describe("File content"),
2290
+ metadata: z2.object({
2291
+ totalLines: z2.number().describe("Total number of lines in the file"),
2292
+ linesShown: z2.number().describe("Number of lines included in this response"),
2293
+ startLine: z2.number().describe("First line number shown (1-indexed)"),
2294
+ endLine: z2.number().describe("Last line number shown (1-indexed)"),
2295
+ isPaginated: z2.boolean().describe("Whether this is a partial view of the file"),
2296
+ fileSize: z2.string().describe("Human-readable file size (e.g., '2.5K', '1.2M')"),
2297
+ path: z2.string().describe("Absolute path to the file")
2298
+ })
2299
+ })
2300
+ }),
2301
+ Grep: tool3({
2302
+ description: "Search for patterns in files using ripgrep. Use this to find code patterns, function definitions, imports, etc.",
2303
+ inputSchema: z2.object({
2304
+ label: z2.string().describe("A label that describes the action being performed"),
2305
+ pattern: z2.string().describe("Regex pattern to search for (ripgrep syntax)"),
2306
+ path: z2.string().optional().describe(
2307
+ "Absolute path to search in (defaults to working directory). Can be a file or directory."
2308
+ ),
2309
+ fileType: z2.string().optional().describe(
2310
+ "File type to filter by (e.g., 'ts', 'js', 'py', 'md'). Uses ripgrep's built-in type filters."
2311
+ ),
2312
+ glob: z2.string().optional().describe(
2313
+ "Glob pattern to filter files (e.g., '*.tsx', 'src/**/*.ts')"
2314
+ ),
2315
+ caseSensitive: z2.boolean().optional().default(true).describe("Whether search is case-sensitive (default: true)"),
2316
+ contextLines: z2.number().optional().describe(
2317
+ "Number of context lines to show before and after each match"
2318
+ ),
2319
+ maxCount: z2.number().optional().describe(
2320
+ "Maximum number of matches per file (useful for limiting output)"
2321
+ ),
2322
+ filesWithMatches: z2.boolean().optional().default(false).describe(
2323
+ "Only show file paths that contain matches, not the matching lines themselves"
2324
+ )
2325
+ }),
2326
+ outputSchema: z2.object({
2327
+ matches: z2.string().describe(
2328
+ "Search results with file paths, line numbers, and matching content"
2329
+ ),
2330
+ summary: z2.object({
2331
+ matchCount: z2.number().describe("Number of matches found"),
2332
+ fileCount: z2.number().describe("Number of files containing matches"),
2333
+ searchPath: z2.string().describe("Path that was searched"),
2334
+ pattern: z2.string().describe("Pattern that was searched for")
2335
+ })
2336
+ })
2337
+ }),
2338
+ List: tool3({
2339
+ description: "Recursively list directory contents. Use this to understand the codebase structure, find files, or explore directories. Control depth to balance detail vs. overview. Depth 1 shows immediate children, depth 2 includes subdirectories, etc.",
2340
+ inputSchema: z2.object({
2341
+ label: z2.string().describe("A label that describes the action being performed"),
2342
+ path: z2.string().optional().describe("Absolute path to list (defaults to working directory)"),
2343
+ depth: z2.number().optional().describe(
2344
+ "Maximum depth to traverse. Choose based on context: 1-2 for quick overview, 3-4 for detailed exploration, 5+ for comprehensive mapping"
2345
+ ),
2346
+ includeHidden: z2.boolean().optional().default(false).describe(
2347
+ "Include hidden files and directories (those starting with '.')"
2348
+ ),
2349
+ filesOnly: z2.boolean().optional().default(false).describe("Only show files, not directories"),
2350
+ pattern: z2.string().optional().describe("Glob pattern to filter results (e.g., '*.ts', '*test*')")
2351
+ }),
2352
+ outputSchema: z2.object({
2353
+ listing: z2.string().describe(
2354
+ "Directory tree listing showing paths relative to search root"
2355
+ ),
2356
+ summary: z2.object({
2357
+ totalItems: z2.number().describe("Total number of items found"),
2358
+ totalFiles: z2.number().describe("Total number of files found"),
2359
+ totalDirs: z2.number().describe("Total number of directories found"),
2360
+ searchPath: z2.string().describe("Path that was listed"),
2361
+ depth: z2.number().optional().describe("Maximum depth used (if specified)")
2362
+ })
2363
+ })
2364
+ }),
2365
+ Write: tool3({
2366
+ description: "Write content to a file. Creates parent directories automatically. Overwrites existing files.",
2367
+ inputSchema: z2.object({
2368
+ label: z2.string().describe("A label that describes the action being performed"),
2369
+ path: z2.string().describe("Absolute path to the file"),
2370
+ content: z2.string().describe("Content to write to the file")
2371
+ }),
2372
+ outputSchema: z2.object({
2373
+ success: z2.boolean().describe("Whether the write succeeded"),
2374
+ path: z2.string().describe("Path to the written file"),
2375
+ bytesWritten: z2.number().describe("Number of bytes written"),
2376
+ error: z2.string().optional().describe("Error message if write failed")
2377
+ })
2378
+ }),
2379
+ Edit: tool3({
2380
+ description: "Edit a file by replacing an exact string. Fails if old_string is not found or appears multiple times (not unique). For multiple replacements, call this tool multiple times with unique context.",
2381
+ inputSchema: z2.object({
2382
+ label: z2.string().describe("A label that describes the action being performed"),
2383
+ path: z2.string().describe("Absolute path to the file"),
2384
+ old_string: z2.string().describe("Exact string to find and replace (must be unique in file)"),
2385
+ new_string: z2.string().describe("String to replace old_string with")
2386
+ }),
2387
+ outputSchema: z2.object({
2388
+ success: z2.boolean().describe("Whether the edit succeeded"),
2389
+ path: z2.string().describe("Path to the edited file"),
2390
+ error: z2.string().optional().describe("Error message if edit failed")
2391
+ })
2392
+ }),
2393
+ Bash: tool3({
2394
+ description: "Executes a bash command. Returns stdout and stderr separately. Use waitUntil to control how long to wait (0 = return immediately, process keeps running). Large outputs are tail-truncated; full output in outputDir.",
2395
+ inputSchema: z2.object({
2396
+ label: z2.string().describe("A label that describes the action being performed"),
2397
+ command: z2.string().describe("The shell command to execute"),
2398
+ waitUntil: z2.number().optional().describe(
2399
+ `Max ms to wait for completion (default: ${DEFAULT_WAIT_UNTIL}). Use 0 to return immediately while the process keeps running.`
2400
+ )
2401
+ }),
2402
+ outputSchema: z2.object({
2403
+ commandId: z2.string().describe("Command ID. Use to reference or kill running processes."),
2404
+ stdout: z2.string().describe(
2405
+ "Command stdout. Tail-truncated if large; full content in outputDir/stdout.txt."
2406
+ ),
2407
+ stderr: z2.string().describe(
2408
+ "Command stderr. Tail-truncated if large; full content in outputDir/stderr.txt."
2409
+ ),
2410
+ exitCode: z2.number().describe("Exit code (-1 if still running)"),
2411
+ status: z2.enum(["running", "completed", "failed"]).describe("Process status"),
2412
+ outputDir: z2.string().describe(
2413
+ "Path to output directory containing stdout.txt, stderr.txt, and metadata.json. Empty if still running."
2414
+ )
2415
+ })
2416
+ }),
2417
+ Skill: tool3({
2418
+ description: "Load a skill's full instructions by name. Call this before following a skill.",
2419
+ inputSchema: z2.object({
2420
+ label: z2.string().describe("A label that describes the action being performed"),
2421
+ name: z2.string().describe("Skill name from the Available Skills list")
2422
+ }),
2423
+ outputSchema: z2.object({
2424
+ name: z2.string(),
2425
+ description: z2.string(),
2426
+ content: z2.string().describe("Full SKILL.md content"),
2427
+ path: z2.string().describe("Path to the skill directory in the sandbox")
2428
+ })
2429
+ }),
2430
+ JavaScript: tool3({
2431
+ inputSchema: z2.object({
2432
+ label: z2.string().describe("A label that describes the action being performed"),
2433
+ code: z2.string().describe(
2434
+ "JavaScript async function body. `ctx` is in scope. Must use `return` to produce output."
2435
+ )
2436
+ })
2437
+ })
2438
+ };
2439
+ var builtinToolNames = Object.fromEntries(
2440
+ Object.entries(builtInTools).map(([name]) => [name, name])
2441
+ );
2442
+ var SKILL_MD_SUFFIX = /\/?SKILL\.md$/;
2443
+ function getTools({
2444
+ agent: agent2,
2445
+ sandbox,
2446
+ skillsRef,
2447
+ writer,
2448
+ abortController,
2449
+ nextPartIndexRef,
2450
+ assistantMessageId,
2451
+ sessionId,
2452
+ stepLog,
2453
+ context,
2454
+ state,
2455
+ messages,
2456
+ activeTools
2457
+ }) {
2458
+ const rawTools = {
2459
+ [builtinToolNames.Read]: tool3({
2460
+ ...builtInTools.Read,
2461
+ execute: async ({ path, startLine, endLine }) => {
2462
+ const filePath = path;
2463
+ const result = await sandbox.readFile({ path: filePath });
2464
+ if (result instanceof Error) {
2465
+ log.error("Read failed", { error: result.message });
2466
+ throw result;
2467
+ }
2468
+ if (result === null) {
2469
+ return {
2470
+ content: `Error: File not found - ${filePath}`,
2471
+ metadata: {
2472
+ totalLines: 0,
2473
+ linesShown: 0,
2474
+ startLine: 0,
2475
+ endLine: 0,
2476
+ isPaginated: false,
2477
+ fileSize: "0",
2478
+ path: filePath
2479
+ }
2480
+ };
2481
+ }
2482
+ const fullContent = result.toString("utf-8");
2483
+ const lines = fullContent.split("\n");
2484
+ if (lines.length > 0 && lines.at(-1) === "") {
2485
+ lines.pop();
2486
+ }
2487
+ const totalLines = lines.length;
2488
+ const fileBytes = Buffer.byteLength(fullContent);
2489
+ const fileSize = formatFileSize(fileBytes);
2490
+ const PAGE_SIZE = 100;
2491
+ let actualStart;
2492
+ let actualEnd;
2493
+ if (startLine !== void 0 && endLine !== void 0) {
2494
+ actualStart = startLine;
2495
+ actualEnd = endLine;
2496
+ } else if (startLine !== void 0) {
2497
+ actualStart = startLine;
2498
+ actualEnd = Math.min(startLine + PAGE_SIZE - 1, totalLines);
2499
+ } else if (endLine !== void 0) {
2500
+ actualStart = 1;
2501
+ actualEnd = endLine;
2502
+ } else if (totalLines > 200) {
2503
+ actualStart = 1;
2504
+ actualEnd = PAGE_SIZE;
2505
+ } else {
2506
+ actualStart = 1;
2507
+ actualEnd = totalLines;
2508
+ }
2509
+ const slicedLines = lines.slice(actualStart - 1, actualEnd);
2510
+ const content = slicedLines.join("\n");
2511
+ return {
2512
+ metadata: {
2513
+ totalLines,
2514
+ linesShown: Math.max(0, actualEnd - actualStart + 1),
2515
+ startLine: actualStart,
2516
+ endLine: actualEnd,
2517
+ isPaginated: actualEnd < totalLines,
2518
+ fileSize,
2519
+ path: filePath
2520
+ },
2521
+ content
2522
+ };
2523
+ }
2524
+ }),
2525
+ [builtinToolNames.Grep]: tool3({
2526
+ ...builtInTools.Grep,
2527
+ execute: async ({
2528
+ pattern,
2529
+ path,
2530
+ fileType,
2531
+ glob,
2532
+ caseSensitive,
2533
+ contextLines,
2534
+ maxCount,
2535
+ filesWithMatches
2536
+ }) => {
2537
+ const searchPath = path ?? ".";
2538
+ const args = [];
2539
+ args.push("--line-number");
2540
+ args.push("--heading");
2541
+ args.push("--color", "never");
2542
+ if (!caseSensitive) {
2543
+ args.push("-i");
2544
+ }
2545
+ if (fileType) {
2546
+ args.push("--type", fileType);
2547
+ }
2548
+ if (glob) {
2549
+ args.push("--glob", glob);
2550
+ }
2551
+ if (contextLines !== void 0) {
2552
+ args.push("-C", String(contextLines));
2553
+ }
2554
+ if (maxCount !== void 0) {
2555
+ args.push("--max-count", String(maxCount));
2556
+ }
2557
+ if (filesWithMatches) {
2558
+ args.push("--files-with-matches");
2559
+ }
2560
+ args.push("--", pattern, searchPath);
2561
+ let result = await sandbox.exec({ command: "rg", args });
2562
+ if (result instanceof Error && isRgNotFoundError(result)) {
2563
+ log.warn("rg not found, installing ripgrep");
2564
+ const installResult = await sandbox.exec({
2565
+ command: "bash",
2566
+ args: [
2567
+ "-c",
2568
+ [
2569
+ "command -v rg >/dev/null 2>&1 && exit 0",
2570
+ "dnf install -y ripgrep 2>/dev/null && exit 0",
2571
+ "(apt-get update -qq && apt-get install -y -qq ripgrep) 2>/dev/null && exit 0",
2572
+ "apk add --no-cache ripgrep 2>/dev/null && exit 0",
2573
+ "curl -sL https://github.com/BurntSushi/ripgrep/releases/download/14.1.1/ripgrep-14.1.1-x86_64-unknown-linux-musl.tar.gz | tar xz -C /tmp && install /tmp/ripgrep-14.1.1-x86_64-unknown-linux-musl/rg /usr/local/bin/rg && rm -rf /tmp/ripgrep-14.1.1-x86_64-unknown-linux-musl"
2574
+ ].join("\n")
2575
+ ],
2576
+ sudo: true
2577
+ });
2578
+ if (!(installResult instanceof Error)) {
2579
+ const installOutput = await installResult.result;
2580
+ if (installOutput.exitCode !== 0) {
2581
+ log.warn("ripgrep install failed", {
2582
+ stderr: installOutput.stderr
2583
+ });
2584
+ }
2585
+ }
2586
+ result = await sandbox.exec({ command: "rg", args });
2587
+ }
2588
+ if (result instanceof Error && isRgNotFoundError(result)) {
2589
+ log.warn("rg unavailable, falling back to grep");
2590
+ const grepArgs = ["-rn", "--color=never"];
2591
+ if (!caseSensitive) {
2592
+ grepArgs.push("-i");
2593
+ }
2594
+ if (contextLines !== void 0) {
2595
+ grepArgs.push("-C", String(contextLines));
2596
+ }
2597
+ if (maxCount !== void 0) {
2598
+ grepArgs.push("-m", String(maxCount));
2599
+ }
2600
+ if (filesWithMatches) {
2601
+ grepArgs.push("-l");
2602
+ }
2603
+ if (glob) {
2604
+ grepArgs.push(`--include=${glob}`);
2605
+ }
2606
+ if (fileType) {
2607
+ const includeMap = {
2608
+ ts: ["*.ts", "*.tsx", "*.mts", "*.cts"],
2609
+ js: ["*.js", "*.jsx", "*.mjs", "*.cjs"],
2610
+ py: ["*.py"],
2611
+ rust: ["*.rs"],
2612
+ go: ["*.go"],
2613
+ java: ["*.java"],
2614
+ md: ["*.md"],
2615
+ json: ["*.json"],
2616
+ css: ["*.css"],
2617
+ html: ["*.html"],
2618
+ yaml: ["*.yml", "*.yaml"]
2619
+ };
2620
+ for (const ext of includeMap[fileType] ?? [`*.${fileType}`]) {
2621
+ grepArgs.push(`--include=${ext}`);
2622
+ }
2623
+ }
2624
+ grepArgs.push("--", pattern, searchPath);
2625
+ result = await sandbox.exec({
2626
+ command: "grep",
2627
+ args: grepArgs
2628
+ });
2629
+ }
2630
+ if (result instanceof Error) {
2631
+ log.error("Grep failed", { error: result.message });
2632
+ throw result;
2633
+ }
2634
+ const { stdout, stderr } = await result.result;
2635
+ if (stderr && !stderr.toLowerCase().includes("no matches")) {
2636
+ log.warn("Grep stderr", { stderr });
2637
+ }
2638
+ const MAX_GREP_OUTPUT_CHARS = 5e4;
2639
+ let finalOutput = stdout;
2640
+ let wasTruncated = false;
2641
+ if (finalOutput.length > MAX_GREP_OUTPUT_CHARS) {
2642
+ finalOutput = finalOutput.slice(0, MAX_GREP_OUTPUT_CHARS) + "\n\n[Output truncated - use more specific pattern or path]";
2643
+ wasTruncated = true;
2644
+ }
2645
+ const lines = finalOutput.trim().split("\n").filter((l) => l.length > 0);
2646
+ const fileCount = filesWithMatches ? lines.length : new Set(
2647
+ lines.filter((l) => !l.startsWith(" ") && l.includes(":")).map((l) => l.split(":")[0])
2648
+ ).size;
2649
+ return {
2650
+ summary: {
2651
+ matchCount: filesWithMatches ? 0 : lines.filter((l) => l.includes(":")).length,
2652
+ fileCount,
2653
+ searchPath,
2654
+ pattern,
2655
+ wasTruncated
2656
+ },
2657
+ matches: finalOutput || "(no matches found)"
2658
+ };
2659
+ }
2660
+ }),
2661
+ [builtinToolNames.List]: tool3({
2662
+ ...builtInTools.List,
2663
+ execute: async ({ path, depth, includeHidden, filesOnly, pattern }) => {
2664
+ const searchPath = path ?? ".";
2665
+ const result = await sandbox.exec({
2666
+ command: "bash",
2667
+ args: [
2668
+ "-c",
2669
+ `
2670
+ set -e
2671
+ SEARCH_PATH="$1"
2672
+ DEPTH="$2"
2673
+ INCLUDE_HIDDEN="$3"
2674
+ FILES_ONLY="$4"
2675
+ PATTERN="$5"
2676
+
2677
+ # Build find command arguments
2678
+ FIND_ARGS=""
2679
+ [ -n "$DEPTH" ] && FIND_ARGS="$FIND_ARGS -maxdepth $DEPTH"
2680
+ [ "$INCLUDE_HIDDEN" != "true" ] && FIND_ARGS="$FIND_ARGS ! -path '*/.*'"
2681
+ [ "$FILES_ONLY" = "true" ] && FIND_ARGS="$FIND_ARGS -type f"
2682
+ [ -n "$PATTERN" ] && FIND_ARGS="$FIND_ARGS -name '$PATTERN'"
2683
+
2684
+ # Get listing
2685
+ LISTING=$(eval "find '$SEARCH_PATH' $FIND_ARGS" 2>/dev/null | sort)
2686
+
2687
+ # Get counts
2688
+ COUNT_ARGS=""
2689
+ [ -n "$DEPTH" ] && COUNT_ARGS="$COUNT_ARGS -maxdepth $DEPTH"
2690
+ [ "$INCLUDE_HIDDEN" != "true" ] && COUNT_ARGS="$COUNT_ARGS ! -path '*/.*'"
2691
+
2692
+ FILE_COUNT=$(eval "find '$SEARCH_PATH' $COUNT_ARGS -type f" 2>/dev/null | wc -l)
2693
+ DIR_COUNT=$(eval "find '$SEARCH_PATH' $COUNT_ARGS -type d" 2>/dev/null | wc -l)
2694
+
2695
+ # Output: counts first, then listing
2696
+ echo "$FILE_COUNT|$DIR_COUNT"
2697
+ echo "|||LISTING|||"
2698
+ echo "$LISTING" | sed "s|^$SEARCH_PATH|.|"
2699
+ `,
2700
+ "--",
2701
+ searchPath,
2702
+ depth?.toString() || "",
2703
+ includeHidden ? "true" : "false",
2704
+ filesOnly ? "true" : "false",
2705
+ pattern || ""
2706
+ ]
2707
+ });
2708
+ if (result instanceof Error) {
2709
+ log.error("List failed", { error: result.message });
2710
+ throw result;
2711
+ }
2712
+ const { stdout, stderr } = await result.result;
2713
+ if (stderr) {
2714
+ log.warn("List stderr", { stderr });
2715
+ }
2716
+ const [countsLine, ...rest] = stdout.split("|||LISTING|||");
2717
+ const listing = rest.join("|||LISTING|||").trim();
2718
+ const [fileCountStr, dirCountStr] = countsLine.trim().split("|");
2719
+ const totalFiles = Number.parseInt(fileCountStr, 10) || 0;
2720
+ const totalDirs = Number.parseInt(dirCountStr, 10) || 0;
2721
+ const lines = listing.split("\n").filter((l) => l.length > 0);
2722
+ return {
2723
+ summary: {
2724
+ totalItems: lines.length,
2725
+ totalFiles,
2726
+ totalDirs,
2727
+ searchPath,
2728
+ depth
2729
+ },
2730
+ listing
2731
+ };
2732
+ }
2733
+ }),
2734
+ [builtinToolNames.Write]: tool3({
2735
+ ...builtInTools.Write,
2736
+ execute: async ({ path, content }) => {
2737
+ const filePath = path;
2738
+ try {
2739
+ await sandbox.writeFiles({
2740
+ files: [{ path: filePath, content }],
2741
+ destPath: "."
2742
+ });
2743
+ return {
2744
+ success: true,
2745
+ path: filePath,
2746
+ bytesWritten: Buffer.byteLength(content, "utf8")
2747
+ };
2748
+ } catch (err) {
2749
+ const errorMsg = err instanceof Error ? err.message : String(err);
2750
+ return {
2751
+ success: false,
2752
+ path: filePath,
2753
+ bytesWritten: 0,
2754
+ error: errorMsg
2755
+ };
2756
+ }
2757
+ }
2758
+ }),
2759
+ [builtinToolNames.Edit]: tool3({
2760
+ ...builtInTools.Edit,
2761
+ execute: async ({ path, old_string, new_string }) => {
2762
+ const filePath = path;
2763
+ const result = await sandbox.readFile({ path: filePath });
2764
+ if (result instanceof Error) {
2765
+ return { success: false, path: filePath, error: result.message };
2766
+ }
2767
+ if (result === null) {
2768
+ return {
2769
+ success: false,
2770
+ path: filePath,
2771
+ error: `File not found: ${filePath}`
2772
+ };
2773
+ }
2774
+ const content = result.toString("utf-8");
2775
+ const occurrences = content.split(old_string).length - 1;
2776
+ if (occurrences === 0) {
2777
+ return {
2778
+ success: false,
2779
+ path: filePath,
2780
+ error: "old_string not found in file"
2781
+ };
2782
+ }
2783
+ if (occurrences > 1) {
2784
+ return {
2785
+ success: false,
2786
+ path: filePath,
2787
+ error: `old_string appears ${occurrences} times in file (must be unique). Include more surrounding context to make the match unique.`
2788
+ };
2789
+ }
2790
+ const newContent = content.replace(old_string, new_string);
2791
+ try {
2792
+ await sandbox.writeFiles({
2793
+ files: [{ path: filePath, content: newContent }],
2794
+ destPath: "."
2795
+ });
2796
+ return { success: true, path: filePath };
2797
+ } catch (err) {
2798
+ const errorMsg = err instanceof Error ? err.message : String(err);
2799
+ return { success: false, path: filePath, error: errorMsg };
2800
+ }
2801
+ }
2802
+ }),
2803
+ [builtinToolNames.Bash]: tool3({
2804
+ ...builtInTools.Bash,
2805
+ execute: async ({ command, waitUntil }) => {
2806
+ const { createProcessManager } = await import("./process-manager-WQHAIVRB.mjs");
2807
+ const processManager = createProcessManager({ sandbox });
2808
+ const result = await processManager.run({ command, waitUntil });
2809
+ const MAX_STDOUT = 5e4;
2810
+ const MAX_STDERR = 1e4;
2811
+ let { stdout, stderr } = result;
2812
+ if (stdout.length > MAX_STDOUT) {
2813
+ stdout = `[truncated \u2014 showing last ${MAX_STDOUT} chars. Full: ${result.outputDir}/stdout.txt]
2814
+
2815
+ ` + stdout.slice(-MAX_STDOUT);
2816
+ }
2817
+ if (stderr.length > MAX_STDERR) {
2818
+ stderr = `[truncated \u2014 showing last ${MAX_STDERR} chars. Full: ${result.outputDir}/stderr.txt]
2819
+
2820
+ ` + stderr.slice(-MAX_STDERR);
2821
+ }
2822
+ return { ...result, stdout, stderr };
2823
+ }
2824
+ })
2825
+ };
2826
+ for (const [name, t] of Object.entries(agent2.options.tools ?? {})) {
2827
+ rawTools[name] = t;
2828
+ }
2829
+ if (skillsRef.current.length > 0) {
2830
+ rawTools[builtinToolNames.Skill] = tool3({
2831
+ ...builtInTools.Skill,
2832
+ execute: async ({ name }) => {
2833
+ const skills = skillsRef.current;
2834
+ const skill = skills.find(
2835
+ (s) => s.name.toLowerCase() === name.toLowerCase()
2836
+ );
2837
+ if (!skill) {
2838
+ throw new Error(
2839
+ `Skill not found: "${name}". Available: ${skills.map((s) => s.name).join(", ")}`
2840
+ );
2841
+ }
2842
+ const result = await sandbox.readFile({
2843
+ path: skill.skillMdPath
2844
+ });
2845
+ if (result instanceof Error) {
2846
+ throw result;
2847
+ }
2848
+ if (result === null) {
2849
+ throw new Error(`Skill file not found: ${skill.skillMdPath}`);
2850
+ }
2851
+ const raw = typeof result === "string" ? result : result.toString("utf-8");
2852
+ const endMarker = raw.indexOf("---", 3);
2853
+ const content = endMarker === -1 ? raw : raw.slice(endMarker + 3).trim();
2854
+ const skillDir = skill.skillMdPath.replace(SKILL_MD_SUFFIX, "");
2855
+ return {
2856
+ name: skill.name,
2857
+ description: skill.description,
2858
+ content,
2859
+ path: skillDir
2860
+ };
2861
+ }
2862
+ });
2863
+ }
2864
+ const onSubToolCall = async (toolName, toolInput, execute) => {
2865
+ const needsApproval = await checkNeedsApproval({
2866
+ agent: agent2,
2867
+ toolName,
2868
+ input: toolInput,
2869
+ toolCallId: `js_${toolName}_${Date.now()}`,
2870
+ messages: []
2871
+ });
2872
+ if (!needsApproval) {
2873
+ try {
2874
+ const result = await execute();
2875
+ return { result };
2876
+ } catch (err) {
2877
+ return {
2878
+ error: err instanceof Error ? err.message : String(err)
2879
+ };
2880
+ }
2881
+ }
2882
+ const approvalId = `js_approval_${ulid2()}`;
2883
+ const toolCallId = `js_tc_${ulid2()}`;
2884
+ const partId = approvalPartId(approvalId);
2885
+ await agent2.storage.part.set(partId, {
2886
+ id: partId,
2887
+ index: nextPartIndexRef.current++,
2888
+ messageId: assistantMessageId,
2889
+ sessionId,
2890
+ part: {
2891
+ type: `tool-${toolName}`,
2892
+ toolCallId,
2893
+ state: "approval-requested",
2894
+ input: toolInput,
2895
+ approval: { id: approvalId }
2896
+ }
2897
+ });
2898
+ writer.write({
2899
+ type: "tool-input-start",
2900
+ toolCallId,
2901
+ toolName
2902
+ });
2903
+ writer.write({
2904
+ type: "tool-input-available",
2905
+ toolCallId,
2906
+ toolName,
2907
+ input: toolInput
2908
+ });
2909
+ writer.write({
2910
+ type: "tool-approval-request",
2911
+ approvalId,
2912
+ toolCallId
2913
+ });
2914
+ const POLL_MS = 500;
2915
+ const TIMEOUT_MS = 5 * 60 * 1e3;
2916
+ const start2 = Date.now();
2917
+ while (Date.now() - start2 < TIMEOUT_MS && !abortController.signal.aborted) {
2918
+ const updated = await agent2.storage.part.get(partId);
2919
+ if (updated && "state" in updated.part && (updated.part.state === "approval-responded" || updated.part.state === "output-denied")) {
2920
+ const state2 = updated.part.state;
2921
+ if (state2 === "output-denied") {
2922
+ const reason = updated.part.approval?.reason;
2923
+ return {
2924
+ error: `Tool "${toolName}" denied: ${reason || "user denied"}`
2925
+ };
2926
+ }
2927
+ const approval = updated.part.approval;
2928
+ if (approval?.approved) {
2929
+ try {
2930
+ const result = await execute();
2931
+ writer.write({
2932
+ type: "tool-output-available",
2933
+ toolCallId,
2934
+ output: result
2935
+ });
2936
+ return { result };
2937
+ } catch (err) {
2938
+ const error = err instanceof Error ? err.message : String(err);
2939
+ writer.write({
2940
+ type: "tool-output-error",
2941
+ toolCallId,
2942
+ errorText: error
2943
+ });
2944
+ return { error };
2945
+ }
2946
+ }
2947
+ writer.write({
2948
+ type: "tool-output-denied",
2949
+ toolCallId
2950
+ });
2951
+ return {
2952
+ error: `Tool "${toolName}" denied: ${approval?.reason || "user denied"}`
2953
+ };
2954
+ }
2955
+ await new Promise((r) => setTimeout(r, POLL_MS));
2956
+ }
2957
+ return {
2958
+ error: abortController.signal.aborted ? "interrupted" : "Approval timed out"
2959
+ };
2960
+ };
2961
+ const toolContext = {
2962
+ context,
2963
+ state,
2964
+ sessionId,
2965
+ sandbox,
2966
+ messages
2967
+ };
2968
+ rawTools[builtinToolNames.JavaScript] = createJavaScriptTool({
2969
+ tools: rawTools,
2970
+ sandbox,
2971
+ activeTools,
2972
+ onSubToolCall,
2973
+ toolContext
2974
+ });
2975
+ const toolsWithNeedsApproval = Object.fromEntries(
2976
+ Object.entries(rawTools).map(([name, t]) => {
2977
+ const originalExecute = t.execute;
2978
+ const wrappedExecute = originalExecute ? async (...args) => {
2979
+ const done = stepLog.time(`tool:${name}`);
2980
+ try {
2981
+ let toolInput = args[0];
2982
+ if (agent2.hooks["tool.before"]) {
2983
+ const beforeResult = await agent2.hooks["tool.before"]({
2984
+ name,
2985
+ input: toolInput,
2986
+ context: toolContext
2987
+ });
2988
+ if (beforeResult?.input !== void 0) {
2989
+ toolInput = beforeResult.input;
2990
+ args[0] = toolInput;
2991
+ }
2992
+ }
2993
+ let result = await originalExecute(...args);
2994
+ if (agent2.hooks["tool.after"]) {
2995
+ const afterResult = await agent2.hooks["tool.after"]({
2996
+ name,
2997
+ input: toolInput,
2998
+ result,
2999
+ context: toolContext
3000
+ });
3001
+ if (afterResult?.result !== void 0) {
3002
+ result = afterResult.result;
3003
+ }
3004
+ }
3005
+ done();
3006
+ return result;
3007
+ } catch (e) {
3008
+ done({
3009
+ error: e instanceof Error ? e.message : String(e)
3010
+ });
3011
+ throw e;
3012
+ }
3013
+ } : void 0;
3014
+ return [
3015
+ name,
3016
+ {
3017
+ ...t,
3018
+ ...wrappedExecute ? { execute: wrappedExecute } : {},
3019
+ needsApproval: async (toolInput, opts) => {
3020
+ return await checkNeedsApproval({
3021
+ agent: agent2,
3022
+ toolName: name,
3023
+ input: toolInput,
3024
+ toolCallId: opts.toolCallId,
3025
+ messages: opts.messages
3026
+ });
3027
+ }
3028
+ }
3029
+ ];
3030
+ })
3031
+ );
3032
+ return { rawTools, toolsWithNeedsApproval, toolContext };
3033
+ }
3034
+
3035
+ // src/agent/registry.ts
3036
+ import * as errore3 from "errore";
3037
+ import { z as z3 } from "zod";
3038
+ var AgentStatus = z3.discriminatedUnion("type", [
3039
+ z3.object({ type: z3.literal("sandbox-setup") }),
3040
+ z3.object({ type: z3.literal("sandbox-setup-cold") }),
3041
+ z3.object({ type: z3.literal("loading-skills") }),
3042
+ z3.object({ type: z3.literal("processing-approvals") }),
3043
+ z3.object({ type: z3.literal("needs-approval") }),
3044
+ z3.object({ type: z3.literal("thinking") }),
3045
+ z3.object({ type: z3.literal("custom"), status: z3.string() })
3046
+ ]);
3047
+ var AgentNotFoundError = class extends errore3.createTaggedError({
3048
+ name: "AgentNotFoundError",
3049
+ message: `Agent "$agentName" not found in registry.
3050
+
3051
+ hint: agent() must be called before workflow steps execute.
3052
+ Valid locations: module scope, or inside a "use workflow" body.
3053
+
3054
+ // Module scope (static agents)
3055
+ export const myAgent = agent("$agentName", { ... });
3056
+
3057
+ // Workflow body (dynamic agents)
3058
+ async function myWorkflow({ input }) {
3059
+ "use workflow";
3060
+ const myAgent = agent("$agentName", { ... });
3061
+ ...
3062
+ }`
3063
+ }) {
3064
+ };
3065
+ var AGENT_REGISTRY_KEY = /* @__PURE__ */ Symbol.for("experimental-agent-registry");
3066
+ function getAgentRegistry() {
3067
+ const g = globalThis;
3068
+ let registry = g[AGENT_REGISTRY_KEY];
3069
+ if (!registry) {
3070
+ registry = /* @__PURE__ */ new Map();
3071
+ g[AGENT_REGISTRY_KEY] = registry;
3072
+ }
3073
+ return registry;
3074
+ }
3075
+ function registerAgent(name, agent2) {
3076
+ getAgentRegistry().set(name, agent2);
3077
+ }
3078
+ function getAgent(name, required) {
3079
+ const agent2 = getAgentRegistry().get(name);
3080
+ if (required && !agent2) {
3081
+ throw new AgentNotFoundError({ agentName: name });
3082
+ }
3083
+ return agent2;
3084
+ }
3085
+
3086
+ // src/agent/client.ts
3087
+ import { WORKFLOW_DESERIALIZE, WORKFLOW_SERIALIZE } from "@workflow/serde";
3088
+
3089
+ // src/sandbox/handle.ts
3090
+ import { ulid as ulid3 } from "ulid";
3091
+
3092
+ // src/agent/is-vercel.ts
3093
+ var isVercel = async () => {
3094
+ try {
3095
+ const { getVercelOidcToken } = await import("@vercel/oidc");
3096
+ return Boolean(await getVercelOidcToken());
3097
+ } catch {
3098
+ return false;
3099
+ }
3100
+ };
3101
+
3102
+ // src/sandbox/setup.ts
3103
+ var POLL_INTERVAL_MS = 50;
3104
+ var POLL_TIMEOUT_MS = 5 * 60 * 1e3;
3105
+ function isSingleBinding(sandbox) {
3106
+ return !Array.isArray(sandbox);
3107
+ }
3108
+ async function getDefaultBinding() {
3109
+ if (await isVercel()) {
3110
+ const { vercelSandbox } = await import("./vercel-7TYF67KD.mjs");
3111
+ return vercelSandbox();
3112
+ }
3113
+ if (await isDockerAvailable()) {
3114
+ const { dockerSandbox } = await import("./docker-OBUQX5ZI.mjs");
3115
+ return dockerSandbox();
3116
+ }
3117
+ const { localSandbox } = await import("./local-4C3V4EFS.mjs");
3118
+ return localSandbox();
3119
+ }
3120
+ async function isDockerAvailable() {
3121
+ try {
3122
+ const { execSync } = await import("child_process");
3123
+ execSync("docker sandbox --help", { stdio: "ignore", timeout: 5e3 });
3124
+ return true;
3125
+ } catch {
3126
+ return false;
3127
+ }
3128
+ }
3129
+ async function resolveBinding(sandbox, type) {
3130
+ if (!sandbox) {
3131
+ const binding = await getDefaultBinding();
3132
+ return { binding, bindingName: binding.type };
3133
+ }
3134
+ if (isSingleBinding(sandbox)) {
3135
+ return { binding: sandbox, bindingName: sandbox.type };
3136
+ }
3137
+ if (type) {
3138
+ const match = sandbox.find((b) => b.type === type);
3139
+ if (!match) {
3140
+ throw new SandboxError({
3141
+ reason: `Unknown binding type: "${type}". Available: ${sandbox.map((b) => b.type).join(", ")}`
3142
+ });
3143
+ }
3144
+ return { binding: match, bindingName: match.type };
3145
+ }
3146
+ if (sandbox.length === 0) {
3147
+ throw new SandboxError({ reason: "No sandbox bindings configured" });
3148
+ }
3149
+ return { binding: sandbox[0], bindingName: sandbox[0].type };
3150
+ }
3151
+ async function resolveSandboxId(ctx) {
3152
+ if (ctx.sandboxId) {
3153
+ return ctx.sandboxId;
3154
+ }
3155
+ if (!ctx._sessionId) {
3156
+ throw new Error(
3157
+ "SandboxContext has no sandboxId and no sessionId to resolve from"
3158
+ );
3159
+ }
3160
+ const record = await ctx.agent.storage.session.get(ctx._sessionId);
3161
+ ctx.sandboxId = record?.sandboxId ?? ctx._sessionId;
3162
+ return ctx.sandboxId;
3163
+ }
3164
+ async function setup(ctx, opts) {
3165
+ "use step";
3166
+ const sandboxId = await resolveSandboxId(ctx);
3167
+ if (ctx._pendingSetup) {
3168
+ return { done: ctx._pendingSetup };
3169
+ }
3170
+ const agent2 = ctx.agent;
3171
+ const log2 = createLogger({
3172
+ config: { ...agent2.options.logging, name: agent2.name },
3173
+ subsystem: "sandbox:setup",
3174
+ context: { sandboxId }
3175
+ });
3176
+ const doneResolveBinding = log2.time("resolve binding", {
3177
+ requestedType: opts?.type
3178
+ });
3179
+ const { binding, bindingName } = await resolveBinding(
3180
+ agent2.options.sandbox,
3181
+ opts?.type
3182
+ );
3183
+ doneResolveBinding();
3184
+ if (opts?.cwd) {
3185
+ ctx._cwd = opts.cwd;
3186
+ }
3187
+ const setupLog = log2.withContext({
3188
+ binding: bindingName,
3189
+ version: opts?.version ?? binding.defaults?.version
3190
+ });
3191
+ let resolveBlockingPromise = (value) => value;
3192
+ let rejectBlockingPromise = (_reason) => {
3193
+ };
3194
+ const blockingPromise = new Promise((resolve2, reject) => {
3195
+ resolveBlockingPromise = resolve2;
3196
+ rejectBlockingPromise = reject;
3197
+ });
3198
+ ctx._pendingSetup = (async () => {
3199
+ let sleepOldInstance;
3200
+ const donePendingSetup = setupLog.time("pending setup");
3201
+ try {
3202
+ const doneStorageGet = setupLog.time("storage.sandbox.get");
3203
+ const record = await agent2.storage.sandbox.get(sandboxId);
3204
+ doneStorageGet();
3205
+ const bindingDefaults = binding.defaults ?? {};
3206
+ const effectiveVersion = opts?.version ?? bindingDefaults.version ?? null;
3207
+ const effectiveRun = opts?.run ?? bindingDefaults.run;
3208
+ const effectiveNetworkPolicy = opts?.networkPolicy ?? bindingDefaults.networkPolicy ?? null;
3209
+ const effectiveCwd = ctx._cwd || opts?.cwd || bindingDefaults.cwd || null;
3210
+ if (effectiveCwd) {
3211
+ ctx._cwd = effectiveCwd;
3212
+ }
3213
+ const effectiveConfig = {
3214
+ ...bindingDefaults.config,
3215
+ // biome-ignore lint/suspicious/noExplicitAny: opts is a union; config exists on binding-specific branches
3216
+ ...opts?.config
3217
+ };
3218
+ if (record) {
3219
+ const versionMatches = !effectiveVersion || record.setup?.version === effectiveVersion;
3220
+ if (versionMatches && record.setup?.completedAt) {
3221
+ resolveBlockingPromise();
3222
+ const doneConnectExisting = setupLog.time(
3223
+ "connect existing sandbox",
3224
+ {
3225
+ setupState: "completed"
3226
+ }
3227
+ );
3228
+ const instance = await binding.connect({
3229
+ agent: agent2,
3230
+ metadata: record.setup.metadata,
3231
+ cwd: record.setup.cwd,
3232
+ signal: opts?.signal,
3233
+ log: setupLog
3234
+ });
3235
+ doneConnectExisting();
3236
+ ctx._cwd = instance.cwd;
3237
+ return instance;
3238
+ }
3239
+ if (versionMatches && !record.setup?.completedAt) {
3240
+ resolveBlockingPromise();
3241
+ const donePollUntilReady = setupLog.time("poll until ready");
3242
+ await pollUntilReady(ctx, opts?.signal);
3243
+ donePollUntilReady();
3244
+ const doneStorageGetPostPoll = setupLog.time(
3245
+ "storage.sandbox.get (post-poll)"
3246
+ );
3247
+ const ready = await agent2.storage.sandbox.get(sandboxId);
3248
+ doneStorageGetPostPoll();
3249
+ if (!ready) {
3250
+ throw new SandboxNotFoundError({ id: sandboxId });
3251
+ }
3252
+ const doneConnectExisting = setupLog.time(
3253
+ "connect existing sandbox",
3254
+ {
3255
+ setupState: "post-poll"
3256
+ }
3257
+ );
3258
+ const instance = await binding.connect({
3259
+ agent: agent2,
3260
+ metadata: ready.setup.metadata,
3261
+ cwd: ready.setup.cwd,
3262
+ signal: opts?.signal,
3263
+ log: setupLog
3264
+ });
3265
+ doneConnectExisting();
3266
+ ctx._cwd = instance.cwd;
3267
+ return instance;
3268
+ }
3269
+ if (record.setup.metadata) {
3270
+ const oldInstanceMetadata = record.setup.metadata;
3271
+ sleepOldInstance = (async () => {
3272
+ const doneStopOldInstance = setupLog.time(
3273
+ "stop old sandbox instance"
3274
+ );
3275
+ try {
3276
+ const old = await binding.connect({
3277
+ agent: agent2,
3278
+ metadata: oldInstanceMetadata,
3279
+ log: setupLog
3280
+ });
3281
+ await old.stop();
3282
+ } catch {
3283
+ doneStopOldInstance();
3284
+ return;
3285
+ }
3286
+ doneStopOldInstance();
3287
+ })();
3288
+ }
3289
+ }
3290
+ const now = Date.now();
3291
+ const setup_ = {
3292
+ binding: bindingName,
3293
+ version: effectiveVersion ?? record?.setup?.version ?? null,
3294
+ networkPolicy: effectiveNetworkPolicy ?? record?.setup?.networkPolicy ?? null,
3295
+ completedAt: null,
3296
+ metadata: null,
3297
+ cwd: null
3298
+ };
3299
+ const newRecord = {
3300
+ id: sandboxId,
3301
+ setup: setup_,
3302
+ createdAt: record?.createdAt ?? now,
3303
+ lastActiveAt: now
3304
+ };
3305
+ const doneStorageSetPending = setupLog.time(
3306
+ "storage.sandbox.set (setup pending)"
3307
+ );
3308
+ await agent2.storage.sandbox.set(sandboxId, newRecord);
3309
+ doneStorageSetPending();
3310
+ resolveBlockingPromise();
3311
+ const doneBindingCreate = setupLog.time("binding.create");
3312
+ const result = await binding.create({
3313
+ cwd: ctx._cwd ?? ".",
3314
+ agent: agent2,
3315
+ setup: {
3316
+ ...setup_,
3317
+ run: effectiveRun,
3318
+ config: effectiveConfig
3319
+ },
3320
+ record: newRecord,
3321
+ signal: opts?.signal,
3322
+ log: setupLog
3323
+ });
3324
+ doneBindingCreate();
3325
+ setup_.completedAt = Date.now();
3326
+ setup_.metadata = result.metadata;
3327
+ setup_.cwd = result.instance.cwd;
3328
+ const doneStorageUpdateComplete = setupLog.time(
3329
+ "storage.sandbox.update (setup complete)"
3330
+ );
3331
+ await agent2.storage.sandbox.update(sandboxId, {
3332
+ setup: setup_,
3333
+ lastActiveAt: Date.now()
3334
+ });
3335
+ doneStorageUpdateComplete();
3336
+ ctx._cwd = result.instance.cwd;
3337
+ return result.instance;
3338
+ } catch (err) {
3339
+ ctx._pendingSetup = void 0;
3340
+ rejectBlockingPromise(err);
3341
+ throw err;
3342
+ } finally {
3343
+ const doneFinalize = setupLog.time("finalize setup");
3344
+ await sleepOldInstance;
3345
+ doneFinalize();
3346
+ donePendingSetup();
3347
+ }
3348
+ })();
3349
+ const doneWaitForUnblock = setupLog.time("wait for setup unblock");
3350
+ await blockingPromise;
3351
+ doneWaitForUnblock();
3352
+ return { done: ctx._pendingSetup };
3353
+ }
3354
+ async function resolve(ctx, signal) {
3355
+ const sandboxId = await resolveSandboxId(ctx);
3356
+ const agent2 = ctx.agent;
3357
+ const log2 = createLogger({
3358
+ config: { ...agent2.options.logging, name: agent2.name },
3359
+ subsystem: "sandbox:resolve",
3360
+ context: { sandboxId }
3361
+ });
3362
+ if (ctx._pendingSetup) {
3363
+ return await ctx._pendingSetup;
3364
+ }
3365
+ let record = await agent2.storage.sandbox.get(sandboxId);
3366
+ if (!record) {
3367
+ const { done } = await setup(ctx, { signal });
3368
+ return await done;
3369
+ }
3370
+ if (!record.setup?.completedAt) {
3371
+ await pollUntilReady(ctx, signal);
3372
+ record = await agent2.storage.sandbox.get(sandboxId);
3373
+ }
3374
+ if (!record) {
3375
+ throw new SandboxNotFoundError({ id: sandboxId });
3376
+ }
3377
+ const { binding } = await resolveBinding(
3378
+ ctx.agent.options.sandbox,
3379
+ record.setup.binding
3380
+ );
3381
+ agent2.storage.sandbox.update(sandboxId, { lastActiveAt: Date.now() }).catch((error) => {
3382
+ log2.error("Failed to update sandbox lastActiveAt", { error });
3383
+ });
3384
+ const instance = await binding.connect({
3385
+ agent: agent2,
3386
+ metadata: record.setup.metadata,
3387
+ cwd: record.setup.cwd,
3388
+ signal,
3389
+ log: log2
3390
+ });
3391
+ ctx._cwd = instance.cwd;
3392
+ return instance;
3393
+ }
3394
+ async function pollUntilReady(ctx, signal) {
3395
+ const id = ctx.sandboxId;
3396
+ const deadline = Date.now() + POLL_TIMEOUT_MS;
3397
+ while (Date.now() < deadline) {
3398
+ signal?.throwIfAborted();
3399
+ const record = await ctx.agent.storage.sandbox.get(id);
3400
+ if (record?.setup?.completedAt) {
3401
+ return;
3402
+ }
3403
+ await new Promise((r) => setTimeout(r, POLL_INTERVAL_MS));
3404
+ }
3405
+ throw new SandboxError({
3406
+ reason: `Timed out waiting for sandbox "${id}" setup to complete`
3407
+ });
3408
+ }
3409
+
3410
+ // src/sandbox/exec.ts
3411
+ async function exec(sandboxContext, ...opts) {
3412
+ const sbx = await resolve(sandboxContext);
3413
+ return await sbx.exec(...opts);
3414
+ }
3415
+
3416
+ // src/sandbox/get-domain.ts
3417
+ async function getDomain(sandboxContext, ...opts) {
3418
+ "use step";
3419
+ const sbx = await resolve(sandboxContext);
3420
+ return await sbx.getDomain(...opts);
3421
+ }
3422
+
3423
+ // src/sandbox/get-status.ts
3424
+ async function getStatus(sandboxContext, ...opts) {
3425
+ "use step";
3426
+ const sbx = await resolve(sandboxContext);
3427
+ return await sbx.getStatus(...opts);
3428
+ }
3429
+
3430
+ // src/sandbox/kill.ts
3431
+ async function kill(sandboxContext, ...opts) {
3432
+ "use step";
3433
+ const sbx = await resolve(sandboxContext);
3434
+ return await sbx.kill(...opts);
3435
+ }
3436
+
3437
+ // src/sandbox/read-file.ts
3438
+ async function readFile(sandboxContext, ...opts) {
3439
+ "use step";
3440
+ const sbx = await resolve(sandboxContext);
3441
+ return await sbx.readFile(...opts);
3442
+ }
3443
+
3444
+ // src/sandbox/snapshot.ts
3445
+ async function snapshot(sandboxContext, ...opts) {
3446
+ "use step";
3447
+ const sbx = await resolve(sandboxContext);
3448
+ return await sbx.snapshot(...opts);
3449
+ }
3450
+
3451
+ // src/sandbox/start.ts
3452
+ async function start(sandboxContext, ...opts) {
3453
+ "use step";
3454
+ const sbx = await resolve(sandboxContext);
3455
+ return await sbx.start(...opts);
3456
+ }
3457
+
3458
+ // src/sandbox/stop.ts
3459
+ async function stop(sandboxContext, ...opts) {
3460
+ "use step";
3461
+ const sbx = await resolve(sandboxContext);
3462
+ return await sbx.stop(...opts);
3463
+ }
3464
+
3465
+ // src/sandbox/update-network-policy.ts
3466
+ async function updateNetworkPolicy(sandboxContext, ...opts) {
3467
+ "use step";
3468
+ const sbx = await resolve(sandboxContext);
3469
+ return await sbx.updateNetworkPolicy(...opts);
3470
+ }
3471
+
3472
+ // src/sandbox/write-files.ts
3473
+ async function writeFiles(sandboxContext, ...opts) {
3474
+ "use step";
3475
+ const sbx = await resolve(sandboxContext);
3476
+ return await sbx.writeFiles(...opts);
3477
+ }
3478
+
3479
+ // src/sandbox/handle.ts
3480
+ function sandboxHandle(agent2, id) {
3481
+ const sandboxId = typeof id === "string" ? id : id?.id ?? `sandbox_${ulid3()}`;
3482
+ const contexts = getSandboxContexts(agent2);
3483
+ let sandboxContext = contexts.get(sandboxId);
3484
+ if (!sandboxContext) {
3485
+ sandboxContext = { agent: agent2, sandboxId };
3486
+ contexts.set(sandboxId, sandboxContext);
3487
+ }
3488
+ return {
3489
+ id: sandboxId,
3490
+ get cwd() {
3491
+ return sandboxContext._cwd ?? ".";
3492
+ },
3493
+ setup: setup.bind(null, sandboxContext),
3494
+ exec: exec.bind(null, sandboxContext),
3495
+ readFile: readFile.bind(null, sandboxContext),
3496
+ writeFiles: writeFiles.bind(null, sandboxContext),
3497
+ getDomain: getDomain.bind(null, sandboxContext),
3498
+ updateNetworkPolicy: updateNetworkPolicy.bind(null, sandboxContext),
3499
+ start: start.bind(null, sandboxContext),
3500
+ stop: stop.bind(null, sandboxContext),
3501
+ kill: kill.bind(null, sandboxContext),
3502
+ snapshot: snapshot.bind(null, sandboxContext),
3503
+ getStatus: getStatus.bind(null, sandboxContext)
3504
+ };
3505
+ }
3506
+ function lazySandboxHandle(agent2, sessionId) {
3507
+ const sandboxContext = {
3508
+ agent: agent2,
3509
+ sandboxId: null,
3510
+ _sessionId: sessionId
3511
+ };
3512
+ return {
3513
+ setup: setup.bind(null, sandboxContext),
3514
+ exec: exec.bind(null, sandboxContext),
3515
+ readFile: readFile.bind(null, sandboxContext),
3516
+ writeFiles: writeFiles.bind(null, sandboxContext),
3517
+ getDomain: getDomain.bind(null, sandboxContext),
3518
+ updateNetworkPolicy: updateNetworkPolicy.bind(null, sandboxContext),
3519
+ start: start.bind(null, sandboxContext),
3520
+ stop: stop.bind(null, sandboxContext),
3521
+ kill: kill.bind(null, sandboxContext),
3522
+ snapshot: snapshot.bind(null, sandboxContext),
3523
+ getStatus: getStatus.bind(null, sandboxContext)
3524
+ };
3525
+ }
3526
+
3527
+ // src/session/handle.ts
3528
+ import { ulid as ulid4 } from "ulid";
3529
+
3530
+ // src/utils/usage.ts
3531
+ function computeSessionUsage(messages) {
3532
+ const byMessageId = {};
3533
+ for (const m of messages) {
3534
+ byMessageId[m.id] = m.usage?.summary ?? null;
3535
+ }
3536
+ const summaries = messages.map((m) => m.usage?.summary).filter((s) => s !== void 0);
3537
+ const total = {
3538
+ model: summaries[0]?.model ?? "unknown",
3539
+ inputTokens: summaries.reduce((acc, s) => acc + s.inputTokens, 0),
3540
+ outputTokens: summaries.reduce((acc, s) => acc + s.outputTokens, 0),
3541
+ totalTokens: summaries.reduce((acc, s) => acc + s.totalTokens, 0),
3542
+ cacheReadTokens: summaries.reduce((acc, s) => acc + s.cacheReadTokens, 0),
3543
+ cacheWriteTokens: summaries.reduce((acc, s) => acc + s.cacheWriteTokens, 0),
3544
+ reasoningTokens: summaries.reduce((acc, s) => acc + s.reasoningTokens, 0),
3545
+ stepCount: summaries.reduce((acc, s) => acc + s.stepCount, 0),
3546
+ messageCount: summaries.length
3547
+ };
3548
+ return { total, byMessageId };
3549
+ }
3550
+
3551
+ // src/session/history.ts
3552
+ async function history(sessionContext) {
3553
+ "use step";
3554
+ const [session, messages, parts] = await Promise.all([
3555
+ sessionContext.agent.storage.session.get(sessionContext.sessionId),
3556
+ sessionContext.agent.storage.message.listBySession(
3557
+ sessionContext.sessionId
3558
+ ),
3559
+ sessionContext.agent.storage.part.listBySession(sessionContext.sessionId)
3560
+ ]);
3561
+ const lastAssistant = messages.find(
3562
+ (m) => m.id === session?.lastMessageId && m.role === "assistant"
3563
+ ) ?? messages.filter((m) => m.role === "assistant").at(-1);
3564
+ const isStreaming = lastAssistant?.completedAt === null && lastAssistant?.interruptedAt === null;
3565
+ const streamingMessageId = isStreaming ? lastAssistant.id : null;
3566
+ const visibleUntil = lastAssistant?.createdAt;
3567
+ const assembled = assembleUIMessages({
3568
+ messages,
3569
+ parts,
3570
+ includeQueued: visibleUntil !== void 0,
3571
+ until: visibleUntil
3572
+ });
3573
+ const usage2 = computeSessionUsage(messages);
3574
+ return { messages: assembled, streamingMessageId, usage: usage2 };
3575
+ }
3576
+
3577
+ // src/session/stream.ts
3578
+ import * as errore4 from "errore";
3579
+ function isWorkflowRun(arg) {
3580
+ return typeof arg === "object" && arg !== null && "getReadable" in arg && typeof arg.getReadable === "function";
3581
+ }
3582
+ async function stream(sessionContext, runOrOpts) {
3583
+ if (isWorkflowRun(runOrOpts)) {
3584
+ const reader = runOrOpts.getReadable().getReader();
3585
+ await reader.read().catch(() => {
3586
+ throw new StreamError({
3587
+ id: sessionContext.sessionId,
3588
+ reason: "stream() failed to read first chunk from workflow run"
3589
+ });
3590
+ });
3591
+ reader.cancel();
3592
+ return runOrOpts.getReadable();
3593
+ }
3594
+ const opts = runOrOpts;
3595
+ const session = await sessionContext.agent.storage.session.get(
3596
+ sessionContext.sessionId
3597
+ );
3598
+ if (!session) {
3599
+ throw new SessionNotFoundError({ id: sessionContext.sessionId });
3600
+ }
3601
+ const messageId = opts?.messageId ?? session.lastMessageId;
3602
+ if (!messageId) {
3603
+ throw new StreamError({
3604
+ id: sessionContext.sessionId,
3605
+ reason: "No message to stream"
3606
+ });
3607
+ }
3608
+ const activeRuns = getActiveRuns();
3609
+ const active = activeRuns.get(messageId);
3610
+ if (!active) {
3611
+ const message = await sessionContext.agent.storage.message.get(messageId);
3612
+ if (message?.workflowRunId) {
3613
+ const readable = await import("workflow/api").then(({ getRun }) => getRun(message.workflowRunId).getReadable()).catch(() => void 0);
3614
+ if (readable) {
3615
+ return readable;
3616
+ }
3617
+ }
3618
+ throw new StreamError({
3619
+ id: sessionContext.sessionId,
3620
+ reason: `No active run for message ${messageId}`
3621
+ });
3622
+ }
3623
+ return replayActiveRun(active, opts?.startIndex);
3624
+ }
3625
+ var StreamError = class extends errore4.createTaggedError({
3626
+ name: "StreamError",
3627
+ message: "Stream failed for session $id: $reason"
3628
+ }) {
3629
+ };
3630
+
3631
+ // src/session/update.ts
3632
+ async function update(sessionContext, opts) {
3633
+ "use step";
3634
+ const updates = Object.fromEntries(
3635
+ Object.entries(opts).filter(([, value]) => value !== void 0)
3636
+ );
3637
+ if (Object.keys(updates).length === 0) {
3638
+ return;
3639
+ }
3640
+ await sessionContext.agent.storage.session.update(sessionContext.sessionId, {
3641
+ ...updates,
3642
+ updatedAt: Date.now()
3643
+ });
3644
+ }
3645
+
3646
+ // src/session/usage.ts
3647
+ async function usage(sessionContext) {
3648
+ "use step";
3649
+ const messages = await sessionContext.agent.storage.message.listBySession(
3650
+ sessionContext.sessionId
3651
+ );
3652
+ return computeSessionUsage(messages);
3653
+ }
3654
+
3655
+ // src/session/handle.ts
3656
+ function sessionHandle(agent2, id) {
3657
+ const sessionId = typeof id === "string" ? id : id?.id ?? `session_${ulid4()}`;
3658
+ const contexts = getSessionContexts(agent2);
3659
+ let sessionContext = contexts.get(sessionId);
3660
+ if (!sessionContext) {
3661
+ sessionContext = { agent: agent2, sessionId };
3662
+ contexts.set(sessionId, sessionContext);
3663
+ }
3664
+ return {
3665
+ id: sessionId,
3666
+ send: send.bind(
3667
+ null,
3668
+ sessionContext
3669
+ ),
3670
+ stream: stream.bind(null, sessionContext),
3671
+ history: history.bind(null, sessionContext),
3672
+ interrupt: interrupt.bind(null, sessionContext),
3673
+ usage: usage.bind(null, sessionContext),
3674
+ update: update.bind(null, sessionContext),
3675
+ sandbox: lazySandboxHandle(agent2, sessionId)
3676
+ };
3677
+ }
3678
+
3679
+ // src/agent/client.ts
3680
+ var sessionContextsMap = /* @__PURE__ */ new WeakMap();
3681
+ var sandboxContextsMap = /* @__PURE__ */ new WeakMap();
3682
+ function getSessionContexts(agent2) {
3683
+ let map = sessionContextsMap.get(agent2);
3684
+ if (!map) {
3685
+ map = /* @__PURE__ */ new Map();
3686
+ sessionContextsMap.set(agent2, map);
3687
+ }
3688
+ return map;
3689
+ }
3690
+ function getSandboxContexts(agent2) {
3691
+ let map = sandboxContextsMap.get(agent2);
3692
+ if (!map) {
3693
+ map = /* @__PURE__ */ new Map();
3694
+ sandboxContextsMap.set(agent2, map);
3695
+ }
3696
+ return map;
3697
+ }
3698
+ var Agent = class _Agent {
3699
+ name;
3700
+ options;
3701
+ constructor(name, options) {
3702
+ this.name = name;
3703
+ this.options = options ?? {};
3704
+ this.session = sessionHandle.bind(null, this);
3705
+ this.sandbox = sandboxHandle.bind(
3706
+ null,
3707
+ this
3708
+ );
3709
+ }
3710
+ static [WORKFLOW_SERIALIZE](instance) {
3711
+ return { name: instance.name };
3712
+ }
3713
+ static [WORKFLOW_DESERIALIZE](data) {
3714
+ return getAgent(data.name, true);
3715
+ }
3716
+ _resolvedStorage;
3717
+ get storage() {
3718
+ if (this._resolvedStorage) {
3719
+ return this._resolvedStorage;
3720
+ }
3721
+ if (this.options.storage) {
3722
+ this._resolvedStorage = toStorage(this.options.storage);
3723
+ return this._resolvedStorage;
3724
+ }
3725
+ const log2 = createLogger({
3726
+ config: { ...this.options.logging, name: this.name },
3727
+ subsystem: "storage"
3728
+ });
3729
+ if (shouldFallbackToKV2()) {
3730
+ log2.info(
3731
+ `No storage adapter configured \u2014 using KV2 (namespace: "${this.name}").`
3732
+ );
3733
+ this._resolvedStorage = toStorage(
3734
+ kvStorageFallback({ prefix: this.name })
3735
+ );
3736
+ return this._resolvedStorage;
3737
+ }
3738
+ log2.warn(
3739
+ "No storage adapter configured \u2014 using localStorage(). Data will not persist across restarts."
3740
+ );
3741
+ this._resolvedStorage = toStorage(localStorage());
3742
+ return this._resolvedStorage;
3743
+ }
3744
+ get tools() {
3745
+ return { ...builtInTools, ...this.options.tools };
3746
+ }
3747
+ get hooks() {
3748
+ return this.options.hooks ?? {};
3749
+ }
3750
+ session;
3751
+ sandbox;
3752
+ /**
3753
+ * Create a derived agent with merged options. The returned agent is NOT
3754
+ * registered in the workflow registry – it's a one-off instance used by
3755
+ * the framework to layer fs-scanned config onto an agent.ts definition.
3756
+ */
3757
+ extend(fn) {
3758
+ return new _Agent(this.name, fn(this.options));
3759
+ }
3760
+ };
3761
+ function agent(name, options) {
3762
+ const instance = new Agent(name, options);
3763
+ registerAgent(name, instance);
3764
+ return instance;
3765
+ }
3766
+
3767
+ export {
3768
+ SessionNotFoundError,
3769
+ SandboxNotFoundError,
3770
+ StorageError,
3771
+ StorageConflictError,
3772
+ SandboxError,
3773
+ MessageNotFoundError,
3774
+ tool,
3775
+ builtinToolNames,
3776
+ AgentStatus,
3777
+ agent
3778
+ };
3779
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vc3JjL2Vycm9ycy50cyIsICIuLi9zcmMvdG9vbHMvdG9vbC50cyIsICIuLi9zcmMvdG9vbHMvaW5kZXgudHMiLCAiLi4vc3JjL3Nlc3Npb24vc2VuZC50cyIsICIuLi9zcmMvc2Vzc2lvbi9haS1sb29wL3N0cmVhbS10ZXh0LnRzIiwgIi4uL3NyYy9za2lsbHMvcGFyc2VyLnRzIiwgIi4uL3NyYy9za2lsbHMvZGlzY292ZXIudHMiLCAiLi4vc3JjL3NraWxscy9yZXNvbHZlLWhvc3QudHMiLCAiLi4vc3JjL3NraWxscy9yZXNvbHZlLWlubGluZS50cyIsICIuLi9zcmMvc2tpbGxzL3Jlc29sdmUtcmVtb3RlLnRzIiwgIi4uL3NyYy9za2lsbHMvcmVzb2x2ZS50cyIsICIuLi9zcmMvdXRpbHMvcHJvbXB0LWNhY2hlLnRzIiwgIi4uL3NyYy91dGlscy91aS50cyIsICIuLi9zcmMvc2Vzc2lvbi9pbnRlcnJ1cHQudHMiLCAiLi4vc3JjL3Rvb2xzL2phdmFzY3JpcHQudHMiLCAiLi4vc3JjL3Rvb2xzL25lZWRzLWFwcHJvdmFsLnRzIiwgIi4uL3NyYy9hZ2VudC9yZWdpc3RyeS50cyIsICIuLi9zcmMvYWdlbnQvY2xpZW50LnRzIiwgIi4uL3NyYy9zYW5kYm94L2hhbmRsZS50cyIsICIuLi9zcmMvYWdlbnQvaXMtdmVyY2VsLnRzIiwgIi4uL3NyYy9zYW5kYm94L3NldHVwLnRzIiwgIi4uL3NyYy9zYW5kYm94L2V4ZWMudHMiLCAiLi4vc3JjL3NhbmRib3gvZ2V0LWRvbWFpbi50cyIsICIuLi9zcmMvc2FuZGJveC9nZXQtc3RhdHVzLnRzIiwgIi4uL3NyYy9zYW5kYm94L2tpbGwudHMiLCAiLi4vc3JjL3NhbmRib3gvcmVhZC1maWxlLnRzIiwgIi4uL3NyYy9zYW5kYm94L3NuYXBzaG90LnRzIiwgIi4uL3NyYy9zYW5kYm94L3N0YXJ0LnRzIiwgIi4uL3NyYy9zYW5kYm94L3N0b3AudHMiLCAiLi4vc3JjL3NhbmRib3gvdXBkYXRlLW5ldHdvcmstcG9saWN5LnRzIiwgIi4uL3NyYy9zYW5kYm94L3dyaXRlLWZpbGVzLnRzIiwgIi4uL3NyYy9zZXNzaW9uL2hhbmRsZS50cyIsICIuLi9zcmMvdXRpbHMvdXNhZ2UudHMiLCAiLi4vc3JjL3Nlc3Npb24vaGlzdG9yeS50cyIsICIuLi9zcmMvc2Vzc2lvbi9zdHJlYW0udHMiLCAiLi4vc3JjL3Nlc3Npb24vdXBkYXRlLnRzIiwgIi4uL3NyYy9zZXNzaW9uL3VzYWdlLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyJpbXBvcnQgKiBhcyBlcnJvcmUgZnJvbSBcImVycm9yZVwiO1xuXG5leHBvcnQgY2xhc3MgU2Vzc2lvbk5vdEZvdW5kRXJyb3IgZXh0ZW5kcyBlcnJvcmUuY3JlYXRlVGFnZ2VkRXJyb3Ioe1xuICBuYW1lOiBcIlNlc3Npb25Ob3RGb3VuZEVycm9yXCIsXG4gIG1lc3NhZ2U6IFwiU2Vzc2lvbiAkaWQgbm90IGZvdW5kXCIsXG59KSB7fVxuXG5leHBvcnQgY2xhc3MgU2Vzc2lvbkVycm9yIGV4dGVuZHMgZXJyb3JlLmNyZWF0ZVRhZ2dlZEVycm9yKHtcbiAgbmFtZTogXCJTZXNzaW9uRXJyb3JcIixcbiAgbWVzc2FnZTogXCJTZXNzaW9uICRpZCBmYWlsZWQ6ICRyZWFzb25cIixcbn0pIHt9XG5cbmV4cG9ydCBjbGFzcyBTYW5kYm94Tm90Rm91bmRFcnJvciBleHRlbmRzIGVycm9yZS5jcmVhdGVUYWdnZWRFcnJvcih7XG4gIG5hbWU6IFwiU2FuZGJveE5vdEZvdW5kRXJyb3JcIixcbiAgbWVzc2FnZTogXCJTYW5kYm94ICRpZCBub3QgZm91bmRcIixcbn0pIHt9XG5cbmV4cG9ydCBjbGFzcyBTdG9yYWdlRXJyb3IgZXh0ZW5kcyBlcnJvcmUuY3JlYXRlVGFnZ2VkRXJyb3Ioe1xuICBuYW1lOiBcIlN0b3JhZ2VFcnJvclwiLFxuICBtZXNzYWdlOiBcIiRyZWFzb25cIixcbn0pIHt9XG5cbmV4cG9ydCBjbGFzcyBTdG9yYWdlQ29uZmxpY3RFcnJvciBleHRlbmRzIGVycm9yZS5jcmVhdGVUYWdnZWRFcnJvcih7XG4gIG5hbWU6IFwiU3RvcmFnZUNvbmZsaWN0RXJyb3JcIixcbiAgbWVzc2FnZTogXCIkcmVhc29uXCIsXG59KSB7fVxuXG5leHBvcnQgY2xhc3MgU2FuZGJveEVycm9yIGV4dGVuZHMgZXJyb3JlLmNyZWF0ZVRhZ2dlZEVycm9yKHtcbiAgbmFtZTogXCJTYW5kYm94RXJyb3JcIixcbiAgbWVzc2FnZTogXCIkcmVhc29uXCIsXG59KSB7fVxuXG5leHBvcnQgY2xhc3MgTWVzc2FnZU5vdEZvdW5kRXJyb3IgZXh0ZW5kcyBlcnJvcmUuY3JlYXRlVGFnZ2VkRXJyb3Ioe1xuICBuYW1lOiBcIk1lc3NhZ2VOb3RGb3VuZEVycm9yXCIsXG4gIG1lc3NhZ2U6IFwiTWVzc2FnZSAkaWQgbm90IGZvdW5kXCIsXG59KSB7fVxuIiwgImltcG9ydCB7XG4gIHRvb2wgYXMgYWlUb29sLFxuICB0eXBlIFRvb2wsXG4gIHR5cGUgVG9vbEV4ZWN1dGlvbk9wdGlvbnMsXG4gIHR5cGUgVUlNZXNzYWdlLFxufSBmcm9tIFwiYWlcIjtcbmltcG9ydCB0eXBlIHsgeiB9IGZyb20gXCJ6b2RcIjtcbmltcG9ydCB0eXBlIHsgU2FuZGJveEluc3RhbmNlIH0gZnJvbSBcIi4uL3NhbmRib3gvYWRhcHRlclwiO1xuaW1wb3J0IHR5cGUgeyBUb29sQ29udGV4dCB9IGZyb20gXCIuL2luZGV4XCI7XG5cbi8qKlxuICogRGVmaW5lIGFuIGFnZW50IHRvb2wgd2l0aCB0eXBlZCBjb250ZXh0IGFuZCBzdGF0ZS5cbiAqXG4gKiBQYXNzIHRoZSBzYW1lIGBjb250ZXh0U2NoZW1hYCBhbmQvb3IgYHN0YXRlU2NoZW1hYCAoWm9kIG9iamVjdHMpIHVzZWQgaW5cbiAqIHlvdXIgYWdlbnQgZGVmaW5pdGlvbiB0byBnZXQgdHlwZWQgYGNvbnRleHRgIGFuZCBgc3RhdGVgIGluc2lkZSBgZXhlY3V0ZWAuXG4gKlxuICogYGBgdHNcbiAqIGltcG9ydCB7IHRvb2wgfSBmcm9tIFwiZXhwZXJpbWVudGFsLWFnZW50XCI7XG4gKiBpbXBvcnQgeyBjb250ZXh0U2NoZW1hLCBzdGF0ZVNjaGVtYSB9IGZyb20gXCIuL2NvbnRleHRcIjtcbiAqXG4gKiBleHBvcnQgY29uc3QgTXlUb29sID0gdG9vbCh7XG4gKiAgIGNvbnRleHRTY2hlbWEsXG4gKiAgIHN0YXRlU2NoZW1hLFxuICogICBpbnB1dFNjaGVtYTogei5vYmplY3QoeyBxdWVyeTogei5zdHJpbmcoKSB9KSxcbiAqICAgZXhlY3V0ZTogYXN5bmMgKHsgcXVlcnkgfSwgeyBjb250ZXh0LCBzdGF0ZSwgc2FuZGJveCB9KSA9PiB7XG4gKiAgICAgY29udGV4dC5jaGFubmVsSWQgLy8gXHUyNzEzIHR5cGVkIGZyb20gY29udGV4dFNjaGVtYVxuICogICAgIHN0YXRlLmNvdW50ZXIrKyAgIC8vIFx1MjcxMyB0eXBlZCBmcm9tIHN0YXRlU2NoZW1hLCBtdXRhYmxlXG4gKiAgIH0sXG4gKiB9KTtcbiAqIGBgYFxuICovXG5leHBvcnQgZnVuY3Rpb24gdG9vbDxcbiAgVElucHV0U2NoZW1hIGV4dGVuZHMgei5ab2RUeXBlLFxuICBUQ29udGV4dFNjaGVtYSBleHRlbmRzIHouWm9kVHlwZSA9IHouWm9kVHlwZTxSZWNvcmQ8c3RyaW5nLCB1bmtub3duPj4sXG4gIFRTdGF0ZVNjaGVtYSBleHRlbmRzIHouWm9kVHlwZSA9IHouWm9kVHlwZTxSZWNvcmQ8c3RyaW5nLCB1bmtub3duPj4sXG4gIC8vIGJpb21lLWlnbm9yZSBsaW50L3N1c3BpY2lvdXMvbm9FeHBsaWNpdEFueTogZGVmYXVsdCBlcmFzZWRcbiAgVE91dHB1dFNjaGVtYSBleHRlbmRzIHouWm9kVHlwZSA9IHouWm9kVHlwZTxhbnk+LFxuPihkZWY6IHtcbiAgZGVzY3JpcHRpb24/OiBzdHJpbmc7XG4gIGlucHV0U2NoZW1hOiBUSW5wdXRTY2hlbWE7XG4gIG91dHB1dFNjaGVtYT86IFRPdXRwdXRTY2hlbWE7XG4gIGNvbnRleHRTY2hlbWE/OiBUQ29udGV4dFNjaGVtYTtcbiAgc3RhdGVTY2hlbWE/OiBUU3RhdGVTY2hlbWE7XG4gIGV4ZWN1dGU/OiAoXG4gICAgaW5wdXQ6IHouaW5mZXI8VElucHV0U2NoZW1hPixcbiAgICBvcHRpb25zOiBUb29sQ29udGV4dDx6LmluZmVyPFRDb250ZXh0U2NoZW1hPiwgei5pbmZlcjxUU3RhdGVTY2hlbWE+PiAmIHtcbiAgICAgIHRvb2xDYWxsSWQ6IHN0cmluZztcbiAgICAgIGFib3J0U2lnbmFsOiBBYm9ydFNpZ25hbDtcbiAgICB9XG4gICAgLy8gYmlvbWUtaWdub3JlIGxpbnQvc3VzcGljaW91cy9ub0V4cGxpY2l0QW55OiBleGVjdXRlIGNhbiByZXR1cm4gYW55dGhpbmdcbiAgKSA9PiBhbnk7XG4gIC8vIGJpb21lLWlnbm9yZSBsaW50L3N1c3BpY2lvdXMvbm9FeHBsaWNpdEFueTogZXJhc2VkIHRvb2wgZ2VuZXJpY3Ncbn0pOiBUb29sPGFueSwgYW55PiB7XG4gIGNvbnN0IHsgZXhlY3V0ZSwgY29udGV4dFNjaGVtYSwgc3RhdGVTY2hlbWEsIC4uLnJlc3QgfSA9IGRlZjtcblxuICBjb25zdCB3cmFwcGVkRXhlY3V0ZSA9IGV4ZWN1dGVcbiAgICA/IC8vIGJpb21lLWlnbm9yZSBsaW50L3N1c3BpY2lvdXMvbm9FeHBsaWNpdEFueTogaW5wdXQgdHlwZSBlcmFzZWQgYXQgcnVudGltZVxuICAgICAgKGlucHV0OiBhbnksIG9wdGlvbnM6IFRvb2xFeGVjdXRpb25PcHRpb25zKSA9PiB7XG4gICAgICAgIGNvbnN0IGN0eCA9IG9wdGlvbnMuZXhwZXJpbWVudGFsX2NvbnRleHQgYXMge1xuICAgICAgICAgIGNvbnRleHQ6IHVua25vd247XG4gICAgICAgICAgc3RhdGU6IHVua25vd247XG4gICAgICAgICAgc2Vzc2lvbklkOiBzdHJpbmc7XG4gICAgICAgICAgbWVzc2FnZXM6IFVJTWVzc2FnZVtdO1xuICAgICAgICAgIHNhbmRib3g6IFNhbmRib3hJbnN0YW5jZTtcbiAgICAgICAgfTtcblxuICAgICAgICBjb25zdCBjb250ZXh0ID0gKFxuICAgICAgICAgIGNvbnRleHRTY2hlbWEgPyBjb250ZXh0U2NoZW1hLnBhcnNlKGN0eC5jb250ZXh0KSA6IGN0eC5jb250ZXh0XG4gICAgICAgICkgYXMgei5pbmZlcjxUQ29udGV4dFNjaGVtYT47XG5cbiAgICAgICAgaWYgKHN0YXRlU2NoZW1hKSB7XG4gICAgICAgICAgc3RhdGVTY2hlbWEucGFyc2UoY3R4LnN0YXRlKTtcbiAgICAgICAgfVxuICAgICAgICBjb25zdCBzdGF0ZSA9IGN0eC5zdGF0ZSBhcyB6LmluZmVyPFRTdGF0ZVNjaGVtYT47XG5cbiAgICAgICAgcmV0dXJuIGV4ZWN1dGUoaW5wdXQsIHtcbiAgICAgICAgICB0b29sQ2FsbElkOiBvcHRpb25zLnRvb2xDYWxsSWQsXG4gICAgICAgICAgYWJvcnRTaWduYWw6IG9wdGlvbnMuYWJvcnRTaWduYWwhLFxuICAgICAgICAgIGNvbnRleHQsXG4gICAgICAgICAgc3RhdGUsXG4gICAgICAgICAgc2Vzc2lvbklkOiBjdHguc2Vzc2lvbklkLFxuICAgICAgICAgIG1lc3NhZ2VzOiBjdHgubWVzc2FnZXMsXG4gICAgICAgICAgc2FuZGJveDogY3R4LnNhbmRib3gsXG4gICAgICAgIH0pO1xuICAgICAgfVxuICAgIDogdW5kZWZpbmVkO1xuXG4gIHJldHVybiB3cmFwcGVkRXhlY3V0ZVxuICAgID8gYWlUb29sKHsgLi4ucmVzdCwgZXhlY3V0ZTogd3JhcHBlZEV4ZWN1dGUgfSlcbiAgICA6IC8vIGJpb21lLWlnbm9yZSBsaW50L3N1c3BpY2lvdXMvbm9FeHBsaWNpdEFueTogbm8tZXhlY3V0ZSBwYXRoIG5lZWRzIGNhc3RcbiAgICAgIGFpVG9vbChyZXN0IGFzIGFueSk7XG59XG4iLCAiaW1wb3J0IHtcbiAgdHlwZSBUb29sU2V0LFxuICB0b29sLFxuICB0eXBlIFVJTWVzc2FnZSxcbiAgdHlwZSBVSU1lc3NhZ2VTdHJlYW1Xcml0ZXIsXG59IGZyb20gXCJhaVwiO1xuaW1wb3J0IHsgdWxpZCB9IGZyb20gXCJ1bGlkXCI7XG5pbXBvcnQgeyB6IH0gZnJvbSBcInpvZFwiO1xuaW1wb3J0IHR5cGUgeyBBbnlBZ2VudCB9IGZyb20gXCIuLi9hZ2VudC9jbGllbnRcIjtcbmltcG9ydCB0eXBlIHsgU2FuZGJveEluc3RhbmNlIH0gZnJvbSBcIi4uL3NhbmRib3gvYWRhcHRlclwiO1xuaW1wb3J0IHsgREVGQVVMVF9XQUlUX1VOVElMIH0gZnJvbSBcIi4uL3NhbmRib3gvcHJvY2Vzcy1tYW5hZ2VyXCI7XG5pbXBvcnQgeyBhcHByb3ZhbFBhcnRJZCB9IGZyb20gXCIuLi9zZXNzaW9uL3NlbmRcIjtcbmltcG9ydCB0eXBlIHsgU2tpbGxTdW1tYXJ5IH0gZnJvbSBcIi4uL3NraWxscy90eXBlc1wiO1xuaW1wb3J0IHsgY3JlYXRlTG9nZ2VyLCB0eXBlIExvZ2dlciB9IGZyb20gXCIuLi91dGlscy9sb2dnZXJcIjtcbmltcG9ydCB7IGNyZWF0ZUphdmFTY3JpcHRUb29sLCB0eXBlIE9uU3ViVG9vbENhbGwgfSBmcm9tIFwiLi9qYXZhc2NyaXB0XCI7XG5pbXBvcnQgeyBjaGVja05lZWRzQXBwcm92YWwgfSBmcm9tIFwiLi9uZWVkcy1hcHByb3ZhbFwiO1xuXG5leHBvcnQgeyB0b29sIGFzIGFnZW50VG9vbCB9IGZyb20gXCIuL3Rvb2xcIjtcblxuY29uc3QgbG9nID0gY3JlYXRlTG9nZ2VyKHsgY29uZmlnOiB7IG5hbWU6IFwidG9vbHNcIiB9LCBzdWJzeXN0ZW06IFwidG9vbHNcIiB9KTtcblxuZnVuY3Rpb24gZm9ybWF0RmlsZVNpemUoYnl0ZXM6IG51bWJlcik6IHN0cmluZyB7XG4gIGlmIChieXRlcyA8IDEwMjQpIHtcbiAgICByZXR1cm4gYCR7Ynl0ZXN9YDtcbiAgfVxuICBpZiAoYnl0ZXMgPCAxMDI0ICogMTAyNCkge1xuICAgIHJldHVybiBgJHsoYnl0ZXMgLyAxMDI0KS50b0ZpeGVkKDEpfUtgO1xuICB9XG4gIGlmIChieXRlcyA8IDEwMjQgKiAxMDI0ICogMTAyNCkge1xuICAgIHJldHVybiBgJHsoYnl0ZXMgLyAoMTAyNCAqIDEwMjQpKS50b0ZpeGVkKDEpfU1gO1xuICB9XG4gIHJldHVybiBgJHsoYnl0ZXMgLyAoMTAyNCAqIDEwMjQgKiAxMDI0KSkudG9GaXhlZCgxKX1HYDtcbn1cblxudHlwZSBTdHJlYW1Xcml0ZXIgPSBVSU1lc3NhZ2VTdHJlYW1Xcml0ZXI7XG5cbmV4cG9ydCB0eXBlIFRvb2xDb250ZXh0PFxuICBUQ29udGV4dCA9IFJlY29yZDxzdHJpbmcsIHVua25vd24+LFxuICBUU3RhdGUgPSBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPixcbj4gPSB7XG4gIGNvbnRleHQ6IFRDb250ZXh0O1xuICBzdGF0ZTogVFN0YXRlO1xuICBzZXNzaW9uSWQ6IHN0cmluZztcbiAgc2FuZGJveDogU2FuZGJveEluc3RhbmNlO1xuICBtZXNzYWdlczogVUlNZXNzYWdlW107XG59O1xuXG5mdW5jdGlvbiBpc1JnTm90Rm91bmRFcnJvcihlcnI6IEVycm9yKTogYm9vbGVhbiB7XG4gIGNvbnN0IHBhcnRzID0gW2Vyci5tZXNzYWdlXTtcblxuICBjb25zdCBhbnlFcnIgPSBlcnIgYXMgdW5rbm93biBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgaWYgKHR5cGVvZiBhbnlFcnIucmVhc29uID09PSBcInN0cmluZ1wiKSB7XG4gICAgcGFydHMucHVzaChhbnlFcnIucmVhc29uKTtcbiAgfVxuICBpZiAoYW55RXJyLmNhdXNlIGluc3RhbmNlb2YgRXJyb3IpIHtcbiAgICBwYXJ0cy5wdXNoKGFueUVyci5jYXVzZS5tZXNzYWdlKTtcbiAgICBjb25zdCBhbnlDYXVzZSA9IGFueUVyci5jYXVzZSBhcyB1bmtub3duIGFzIFJlY29yZDxzdHJpbmcsIHVua25vd24+O1xuICAgIGlmICh0eXBlb2YgYW55Q2F1c2UudGV4dCA9PT0gXCJzdHJpbmdcIikge1xuICAgICAgcGFydHMucHVzaChhbnlDYXVzZS50ZXh0KTtcbiAgICB9XG4gIH1cblxuICBjb25zdCBtc2cgPSBwYXJ0cy5qb2luKFwiIFwiKS50b0xvd2VyQ2FzZSgpO1xuICByZXR1cm4gKFxuICAgIG1zZy5pbmNsdWRlcyhcImV4ZWN1dGFibGUgZmlsZSBub3QgZm91bmRcIikgfHxcbiAgICBtc2cuaW5jbHVkZXMoXCJleGVjdXRhYmxlX25vdF9mb3VuZFwiKSB8fFxuICAgIChtc2cuaW5jbHVkZXMoXCJlbm9lbnRcIikgJiYgbXNnLmluY2x1ZGVzKFwicmdcIikpXG4gICk7XG59XG5cbmV4cG9ydCBjb25zdCBidWlsdEluVG9vbHMgPSB7XG4gIFJlYWQ6IHRvb2woe1xuICAgIGRlc2NyaXB0aW9uOlxuICAgICAgXCJSZWFkcyBhIGZpbGUgYW5kIHJldHVybnMgaXRzIGNvbnRlbnRzIHdpdGggbWV0YWRhdGEuIEZvciBmaWxlcyBvdmVyIDIwMCBsaW5lcywgYXV0b21hdGljYWxseSBzaG93cyBmaXJzdCAxMDAgbGluZXMgdW5sZXNzIGEgc3BlY2lmaWMgbGluZSByYW5nZSBpcyBwcm92aWRlZC4gVXNlIHN0YXJ0TGluZSBhbmQgZW5kTGluZSBwYXJhbWV0ZXJzIHRvIHJlYWQgc3BlY2lmaWMgcG9ydGlvbnMgb2YgbGFyZ2UgZmlsZXMuXCIsXG4gICAgaW5wdXRTY2hlbWE6IHoub2JqZWN0KHtcbiAgICAgIGxhYmVsOiB6XG4gICAgICAgIC5zdHJpbmcoKVxuICAgICAgICAuZGVzY3JpYmUoXCJBIGxhYmVsIHRoYXQgZGVzY3JpYmVzIHRoZSBhY3Rpb24gYmVpbmcgcGVyZm9ybWVkXCIpLFxuICAgICAgcGF0aDogei5zdHJpbmcoKS5kZXNjcmliZShcIkFic29sdXRlIHBhdGggdG8gdGhlIGZpbGVcIiksXG4gICAgICBzdGFydExpbmU6IHpcbiAgICAgICAgLm51bWJlcigpXG4gICAgICAgIC5vcHRpb25hbCgpXG4gICAgICAgIC5kZXNjcmliZShcbiAgICAgICAgICBcIlN0YXJ0aW5nIGxpbmUgbnVtYmVyICgxLWluZGV4ZWQpLiBJZiBwcm92aWRlZCB3aXRoIGVuZExpbmUsIHJlYWRzIGV4YWN0IHJhbmdlIHJlZ2FyZGxlc3Mgb2YgZmlsZSBzaXplLlwiXG4gICAgICAgICksXG4gICAgICBlbmRMaW5lOiB6XG4gICAgICAgIC5udW1iZXIoKVxuICAgICAgICAub3B0aW9uYWwoKVxuICAgICAgICAuZGVzY3JpYmUoXG4gICAgICAgICAgXCJFbmRpbmcgbGluZSBudW1iZXIgKDEtaW5kZXhlZCwgaW5jbHVzaXZlKS4gSWYgcHJvdmlkZWQgd2l0aCBzdGFydExpbmUsIHJlYWRzIGV4YWN0IHJhbmdlIHJlZ2FyZGxlc3Mgb2YgZmlsZSBzaXplLlwiXG4gICAgICAgICksXG4gICAgfSksXG4gICAgb3V0cHV0U2NoZW1hOiB6Lm9iamVjdCh7XG4gICAgICBjb250ZW50OiB6LnN0cmluZygpLmRlc2NyaWJlKFwiRmlsZSBjb250ZW50XCIpLFxuICAgICAgbWV0YWRhdGE6IHoub2JqZWN0KHtcbiAgICAgICAgdG90YWxMaW5lczogei5udW1iZXIoKS5kZXNjcmliZShcIlRvdGFsIG51bWJlciBvZiBsaW5lcyBpbiB0aGUgZmlsZVwiKSxcbiAgICAgICAgbGluZXNTaG93bjogelxuICAgICAgICAgIC5udW1iZXIoKVxuICAgICAgICAgIC5kZXNjcmliZShcIk51bWJlciBvZiBsaW5lcyBpbmNsdWRlZCBpbiB0aGlzIHJlc3BvbnNlXCIpLFxuICAgICAgICBzdGFydExpbmU6IHoubnVtYmVyKCkuZGVzY3JpYmUoXCJGaXJzdCBsaW5lIG51bWJlciBzaG93biAoMS1pbmRleGVkKVwiKSxcbiAgICAgICAgZW5kTGluZTogei5udW1iZXIoKS5kZXNjcmliZShcIkxhc3QgbGluZSBudW1iZXIgc2hvd24gKDEtaW5kZXhlZClcIiksXG4gICAgICAgIGlzUGFnaW5hdGVkOiB6XG4gICAgICAgICAgLmJvb2xlYW4oKVxuICAgICAgICAgIC5kZXNjcmliZShcIldoZXRoZXIgdGhpcyBpcyBhIHBhcnRpYWwgdmlldyBvZiB0aGUgZmlsZVwiKSxcbiAgICAgICAgZmlsZVNpemU6IHpcbiAgICAgICAgICAuc3RyaW5nKClcbiAgICAgICAgICAuZGVzY3JpYmUoXCJIdW1hbi1yZWFkYWJsZSBmaWxlIHNpemUgKGUuZy4sICcyLjVLJywgJzEuMk0nKVwiKSxcbiAgICAgICAgcGF0aDogei5zdHJpbmcoKS5kZXNjcmliZShcIkFic29sdXRlIHBhdGggdG8gdGhlIGZpbGVcIiksXG4gICAgICB9KSxcbiAgICB9KSxcbiAgfSksXG4gIEdyZXA6IHRvb2woe1xuICAgIGRlc2NyaXB0aW9uOlxuICAgICAgXCJTZWFyY2ggZm9yIHBhdHRlcm5zIGluIGZpbGVzIHVzaW5nIHJpcGdyZXAuIFVzZSB0aGlzIHRvIGZpbmQgY29kZSBwYXR0ZXJucywgZnVuY3Rpb24gZGVmaW5pdGlvbnMsIGltcG9ydHMsIGV0Yy5cIixcbiAgICBpbnB1dFNjaGVtYTogei5vYmplY3Qoe1xuICAgICAgbGFiZWw6IHpcbiAgICAgICAgLnN0cmluZygpXG4gICAgICAgIC5kZXNjcmliZShcIkEgbGFiZWwgdGhhdCBkZXNjcmliZXMgdGhlIGFjdGlvbiBiZWluZyBwZXJmb3JtZWRcIiksXG4gICAgICBwYXR0ZXJuOiB6XG4gICAgICAgIC5zdHJpbmcoKVxuICAgICAgICAuZGVzY3JpYmUoXCJSZWdleCBwYXR0ZXJuIHRvIHNlYXJjaCBmb3IgKHJpcGdyZXAgc3ludGF4KVwiKSxcbiAgICAgIHBhdGg6IHpcbiAgICAgICAgLnN0cmluZygpXG4gICAgICAgIC5vcHRpb25hbCgpXG4gICAgICAgIC5kZXNjcmliZShcbiAgICAgICAgICBcIkFic29sdXRlIHBhdGggdG8gc2VhcmNoIGluIChkZWZhdWx0cyB0byB3b3JraW5nIGRpcmVjdG9yeSkuIENhbiBiZSBhIGZpbGUgb3IgZGlyZWN0b3J5LlwiXG4gICAgICAgICksXG4gICAgICBmaWxlVHlwZTogelxuICAgICAgICAuc3RyaW5nKClcbiAgICAgICAgLm9wdGlvbmFsKClcbiAgICAgICAgLmRlc2NyaWJlKFxuICAgICAgICAgIFwiRmlsZSB0eXBlIHRvIGZpbHRlciBieSAoZS5nLiwgJ3RzJywgJ2pzJywgJ3B5JywgJ21kJykuIFVzZXMgcmlwZ3JlcCdzIGJ1aWx0LWluIHR5cGUgZmlsdGVycy5cIlxuICAgICAgICApLFxuICAgICAgZ2xvYjogelxuICAgICAgICAuc3RyaW5nKClcbiAgICAgICAgLm9wdGlvbmFsKClcbiAgICAgICAgLmRlc2NyaWJlKFxuICAgICAgICAgIFwiR2xvYiBwYXR0ZXJuIHRvIGZpbHRlciBmaWxlcyAoZS5nLiwgJyoudHN4JywgJ3NyYy8qKi8qLnRzJylcIlxuICAgICAgICApLFxuICAgICAgY2FzZVNlbnNpdGl2ZTogelxuICAgICAgICAuYm9vbGVhbigpXG4gICAgICAgIC5vcHRpb25hbCgpXG4gICAgICAgIC5kZWZhdWx0KHRydWUpXG4gICAgICAgIC5kZXNjcmliZShcIldoZXRoZXIgc2VhcmNoIGlzIGNhc2Utc2Vuc2l0aXZlIChkZWZhdWx0OiB0cnVlKVwiKSxcbiAgICAgIGNvbnRleHRMaW5lczogelxuICAgICAgICAubnVtYmVyKClcbiAgICAgICAgLm9wdGlvbmFsKClcbiAgICAgICAgLmRlc2NyaWJlKFxuICAgICAgICAgIFwiTnVtYmVyIG9mIGNvbnRleHQgbGluZXMgdG8gc2hvdyBiZWZvcmUgYW5kIGFmdGVyIGVhY2ggbWF0Y2hcIlxuICAgICAgICApLFxuICAgICAgbWF4Q291bnQ6IHpcbiAgICAgICAgLm51bWJlcigpXG4gICAgICAgIC5vcHRpb25hbCgpXG4gICAgICAgIC5kZXNjcmliZShcbiAgICAgICAgICBcIk1heGltdW0gbnVtYmVyIG9mIG1hdGNoZXMgcGVyIGZpbGUgKHVzZWZ1bCBmb3IgbGltaXRpbmcgb3V0cHV0KVwiXG4gICAgICAgICksXG4gICAgICBmaWxlc1dpdGhNYXRjaGVzOiB6XG4gICAgICAgIC5ib29sZWFuKClcbiAgICAgICAgLm9wdGlvbmFsKClcbiAgICAgICAgLmRlZmF1bHQoZmFsc2UpXG4gICAgICAgIC5kZXNjcmliZShcbiAgICAgICAgICBcIk9ubHkgc2hvdyBmaWxlIHBhdGhzIHRoYXQgY29udGFpbiBtYXRjaGVzLCBub3QgdGhlIG1hdGNoaW5nIGxpbmVzIHRoZW1zZWx2ZXNcIlxuICAgICAgICApLFxuICAgIH0pLFxuICAgIG91dHB1dFNjaGVtYTogei5vYmplY3Qoe1xuICAgICAgbWF0Y2hlczogelxuICAgICAgICAuc3RyaW5nKClcbiAgICAgICAgLmRlc2NyaWJlKFxuICAgICAgICAgIFwiU2VhcmNoIHJlc3VsdHMgd2l0aCBmaWxlIHBhdGhzLCBsaW5lIG51bWJlcnMsIGFuZCBtYXRjaGluZyBjb250ZW50XCJcbiAgICAgICAgKSxcbiAgICAgIHN1bW1hcnk6IHoub2JqZWN0KHtcbiAgICAgICAgbWF0Y2hDb3VudDogei5udW1iZXIoKS5kZXNjcmliZShcIk51bWJlciBvZiBtYXRjaGVzIGZvdW5kXCIpLFxuICAgICAgICBmaWxlQ291bnQ6IHoubnVtYmVyKCkuZGVzY3JpYmUoXCJOdW1iZXIgb2YgZmlsZXMgY29udGFpbmluZyBtYXRjaGVzXCIpLFxuICAgICAgICBzZWFyY2hQYXRoOiB6LnN0cmluZygpLmRlc2NyaWJlKFwiUGF0aCB0aGF0IHdhcyBzZWFyY2hlZFwiKSxcbiAgICAgICAgcGF0dGVybjogei5zdHJpbmcoKS5kZXNjcmliZShcIlBhdHRlcm4gdGhhdCB3YXMgc2VhcmNoZWQgZm9yXCIpLFxuICAgICAgfSksXG4gICAgfSksXG4gIH0pLFxuICBMaXN0OiB0b29sKHtcbiAgICBkZXNjcmlwdGlvbjpcbiAgICAgIFwiUmVjdXJzaXZlbHkgbGlzdCBkaXJlY3RvcnkgY29udGVudHMuIFVzZSB0aGlzIHRvIHVuZGVyc3RhbmQgdGhlIGNvZGViYXNlIHN0cnVjdHVyZSwgZmluZCBmaWxlcywgb3IgZXhwbG9yZSBkaXJlY3Rvcmllcy4gQ29udHJvbCBkZXB0aCB0byBiYWxhbmNlIGRldGFpbCB2cy4gb3ZlcnZpZXcuIERlcHRoIDEgc2hvd3MgaW1tZWRpYXRlIGNoaWxkcmVuLCBkZXB0aCAyIGluY2x1ZGVzIHN1YmRpcmVjdG9yaWVzLCBldGMuXCIsXG4gICAgaW5wdXRTY2hlbWE6IHoub2JqZWN0KHtcbiAgICAgIGxhYmVsOiB6XG4gICAgICAgIC5zdHJpbmcoKVxuICAgICAgICAuZGVzY3JpYmUoXCJBIGxhYmVsIHRoYXQgZGVzY3JpYmVzIHRoZSBhY3Rpb24gYmVpbmcgcGVyZm9ybWVkXCIpLFxuICAgICAgcGF0aDogelxuICAgICAgICAuc3RyaW5nKClcbiAgICAgICAgLm9wdGlvbmFsKClcbiAgICAgICAgLmRlc2NyaWJlKFwiQWJzb2x1dGUgcGF0aCB0byBsaXN0IChkZWZhdWx0cyB0byB3b3JraW5nIGRpcmVjdG9yeSlcIiksXG4gICAgICBkZXB0aDogelxuICAgICAgICAubnVtYmVyKClcbiAgICAgICAgLm9wdGlvbmFsKClcbiAgICAgICAgLmRlc2NyaWJlKFxuICAgICAgICAgIFwiTWF4aW11bSBkZXB0aCB0byB0cmF2ZXJzZS4gQ2hvb3NlIGJhc2VkIG9uIGNvbnRleHQ6IDEtMiBmb3IgcXVpY2sgb3ZlcnZpZXcsIDMtNCBmb3IgZGV0YWlsZWQgZXhwbG9yYXRpb24sIDUrIGZvciBjb21wcmVoZW5zaXZlIG1hcHBpbmdcIlxuICAgICAgICApLFxuICAgICAgaW5jbHVkZUhpZGRlbjogelxuICAgICAgICAuYm9vbGVhbigpXG4gICAgICAgIC5vcHRpb25hbCgpXG4gICAgICAgIC5kZWZhdWx0KGZhbHNlKVxuICAgICAgICAuZGVzY3JpYmUoXG4gICAgICAgICAgXCJJbmNsdWRlIGhpZGRlbiBmaWxlcyBhbmQgZGlyZWN0b3JpZXMgKHRob3NlIHN0YXJ0aW5nIHdpdGggJy4nKVwiXG4gICAgICAgICksXG4gICAgICBmaWxlc09ubHk6IHpcbiAgICAgICAgLmJvb2xlYW4oKVxuICAgICAgICAub3B0aW9uYWwoKVxuICAgICAgICAuZGVmYXVsdChmYWxzZSlcbiAgICAgICAgLmRlc2NyaWJlKFwiT25seSBzaG93IGZpbGVzLCBub3QgZGlyZWN0b3JpZXNcIiksXG4gICAgICBwYXR0ZXJuOiB6XG4gICAgICAgIC5zdHJpbmcoKVxuICAgICAgICAub3B0aW9uYWwoKVxuICAgICAgICAuZGVzY3JpYmUoXCJHbG9iIHBhdHRlcm4gdG8gZmlsdGVyIHJlc3VsdHMgKGUuZy4sICcqLnRzJywgJyp0ZXN0KicpXCIpLFxuICAgIH0pLFxuICAgIG91dHB1dFNjaGVtYTogei5vYmplY3Qoe1xuICAgICAgbGlzdGluZzogelxuICAgICAgICAuc3RyaW5nKClcbiAgICAgICAgLmRlc2NyaWJlKFxuICAgICAgICAgIFwiRGlyZWN0b3J5IHRyZWUgbGlzdGluZyBzaG93aW5nIHBhdGhzIHJlbGF0aXZlIHRvIHNlYXJjaCByb290XCJcbiAgICAgICAgKSxcbiAgICAgIHN1bW1hcnk6IHoub2JqZWN0KHtcbiAgICAgICAgdG90YWxJdGVtczogei5udW1iZXIoKS5kZXNjcmliZShcIlRvdGFsIG51bWJlciBvZiBpdGVtcyBmb3VuZFwiKSxcbiAgICAgICAgdG90YWxGaWxlczogei5udW1iZXIoKS5kZXNjcmliZShcIlRvdGFsIG51bWJlciBvZiBmaWxlcyBmb3VuZFwiKSxcbiAgICAgICAgdG90YWxEaXJzOiB6Lm51bWJlcigpLmRlc2NyaWJlKFwiVG90YWwgbnVtYmVyIG9mIGRpcmVjdG9yaWVzIGZvdW5kXCIpLFxuICAgICAgICBzZWFyY2hQYXRoOiB6LnN0cmluZygpLmRlc2NyaWJlKFwiUGF0aCB0aGF0IHdhcyBsaXN0ZWRcIiksXG4gICAgICAgIGRlcHRoOiB6XG4gICAgICAgICAgLm51bWJlcigpXG4gICAgICAgICAgLm9wdGlvbmFsKClcbiAgICAgICAgICAuZGVzY3JpYmUoXCJNYXhpbXVtIGRlcHRoIHVzZWQgKGlmIHNwZWNpZmllZClcIiksXG4gICAgICB9KSxcbiAgICB9KSxcbiAgfSksXG4gIFdyaXRlOiB0b29sKHtcbiAgICBkZXNjcmlwdGlvbjpcbiAgICAgIFwiV3JpdGUgY29udGVudCB0byBhIGZpbGUuIENyZWF0ZXMgcGFyZW50IGRpcmVjdG9yaWVzIGF1dG9tYXRpY2FsbHkuIE92ZXJ3cml0ZXMgZXhpc3RpbmcgZmlsZXMuXCIsXG4gICAgaW5wdXRTY2hlbWE6IHoub2JqZWN0KHtcbiAgICAgIGxhYmVsOiB6XG4gICAgICAgIC5zdHJpbmcoKVxuICAgICAgICAuZGVzY3JpYmUoXCJBIGxhYmVsIHRoYXQgZGVzY3JpYmVzIHRoZSBhY3Rpb24gYmVpbmcgcGVyZm9ybWVkXCIpLFxuICAgICAgcGF0aDogei5zdHJpbmcoKS5kZXNjcmliZShcIkFic29sdXRlIHBhdGggdG8gdGhlIGZpbGVcIiksXG4gICAgICBjb250ZW50OiB6LnN0cmluZygpLmRlc2NyaWJlKFwiQ29udGVudCB0byB3cml0ZSB0byB0aGUgZmlsZVwiKSxcbiAgICB9KSxcbiAgICBvdXRwdXRTY2hlbWE6IHoub2JqZWN0KHtcbiAgICAgIHN1Y2Nlc3M6IHouYm9vbGVhbigpLmRlc2NyaWJlKFwiV2hldGhlciB0aGUgd3JpdGUgc3VjY2VlZGVkXCIpLFxuICAgICAgcGF0aDogei5zdHJpbmcoKS5kZXNjcmliZShcIlBhdGggdG8gdGhlIHdyaXR0ZW4gZmlsZVwiKSxcbiAgICAgIGJ5dGVzV3JpdHRlbjogei5udW1iZXIoKS5kZXNjcmliZShcIk51bWJlciBvZiBieXRlcyB3cml0dGVuXCIpLFxuICAgICAgZXJyb3I6IHouc3RyaW5nKCkub3B0aW9uYWwoKS5kZXNjcmliZShcIkVycm9yIG1lc3NhZ2UgaWYgd3JpdGUgZmFpbGVkXCIpLFxuICAgIH0pLFxuICB9KSxcbiAgRWRpdDogdG9vbCh7XG4gICAgZGVzY3JpcHRpb246XG4gICAgICBcIkVkaXQgYSBmaWxlIGJ5IHJlcGxhY2luZyBhbiBleGFjdCBzdHJpbmcuIEZhaWxzIGlmIG9sZF9zdHJpbmcgaXMgbm90IGZvdW5kIG9yIGFwcGVhcnMgbXVsdGlwbGUgdGltZXMgKG5vdCB1bmlxdWUpLiBGb3IgbXVsdGlwbGUgcmVwbGFjZW1lbnRzLCBjYWxsIHRoaXMgdG9vbCBtdWx0aXBsZSB0aW1lcyB3aXRoIHVuaXF1ZSBjb250ZXh0LlwiLFxuICAgIGlucHV0U2NoZW1hOiB6Lm9iamVjdCh7XG4gICAgICBsYWJlbDogelxuICAgICAgICAuc3RyaW5nKClcbiAgICAgICAgLmRlc2NyaWJlKFwiQSBsYWJlbCB0aGF0IGRlc2NyaWJlcyB0aGUgYWN0aW9uIGJlaW5nIHBlcmZvcm1lZFwiKSxcbiAgICAgIHBhdGg6IHouc3RyaW5nKCkuZGVzY3JpYmUoXCJBYnNvbHV0ZSBwYXRoIHRvIHRoZSBmaWxlXCIpLFxuICAgICAgb2xkX3N0cmluZzogelxuICAgICAgICAuc3RyaW5nKClcbiAgICAgICAgLmRlc2NyaWJlKFwiRXhhY3Qgc3RyaW5nIHRvIGZpbmQgYW5kIHJlcGxhY2UgKG11c3QgYmUgdW5pcXVlIGluIGZpbGUpXCIpLFxuICAgICAgbmV3X3N0cmluZzogei5zdHJpbmcoKS5kZXNjcmliZShcIlN0cmluZyB0byByZXBsYWNlIG9sZF9zdHJpbmcgd2l0aFwiKSxcbiAgICB9KSxcbiAgICBvdXRwdXRTY2hlbWE6IHoub2JqZWN0KHtcbiAgICAgIHN1Y2Nlc3M6IHouYm9vbGVhbigpLmRlc2NyaWJlKFwiV2hldGhlciB0aGUgZWRpdCBzdWNjZWVkZWRcIiksXG4gICAgICBwYXRoOiB6LnN0cmluZygpLmRlc2NyaWJlKFwiUGF0aCB0byB0aGUgZWRpdGVkIGZpbGVcIiksXG4gICAgICBlcnJvcjogei5zdHJpbmcoKS5vcHRpb25hbCgpLmRlc2NyaWJlKFwiRXJyb3IgbWVzc2FnZSBpZiBlZGl0IGZhaWxlZFwiKSxcbiAgICB9KSxcbiAgfSksXG4gIEJhc2g6IHRvb2woe1xuICAgIGRlc2NyaXB0aW9uOlxuICAgICAgXCJFeGVjdXRlcyBhIGJhc2ggY29tbWFuZC4gUmV0dXJucyBzdGRvdXQgYW5kIHN0ZGVyciBzZXBhcmF0ZWx5LiBVc2Ugd2FpdFVudGlsIHRvIGNvbnRyb2wgaG93IGxvbmcgdG8gd2FpdCAoMCA9IHJldHVybiBpbW1lZGlhdGVseSwgcHJvY2VzcyBrZWVwcyBydW5uaW5nKS4gTGFyZ2Ugb3V0cHV0cyBhcmUgdGFpbC10cnVuY2F0ZWQ7IGZ1bGwgb3V0cHV0IGluIG91dHB1dERpci5cIixcbiAgICBpbnB1dFNjaGVtYTogei5vYmplY3Qoe1xuICAgICAgbGFiZWw6IHpcbiAgICAgICAgLnN0cmluZygpXG4gICAgICAgIC5kZXNjcmliZShcIkEgbGFiZWwgdGhhdCBkZXNjcmliZXMgdGhlIGFjdGlvbiBiZWluZyBwZXJmb3JtZWRcIiksXG4gICAgICBjb21tYW5kOiB6LnN0cmluZygpLmRlc2NyaWJlKFwiVGhlIHNoZWxsIGNvbW1hbmQgdG8gZXhlY3V0ZVwiKSxcbiAgICAgIHdhaXRVbnRpbDogelxuICAgICAgICAubnVtYmVyKClcbiAgICAgICAgLm9wdGlvbmFsKClcbiAgICAgICAgLmRlc2NyaWJlKFxuICAgICAgICAgIGBNYXggbXMgdG8gd2FpdCBmb3IgY29tcGxldGlvbiAoZGVmYXVsdDogJHtERUZBVUxUX1dBSVRfVU5USUx9KS4gVXNlIDAgdG8gcmV0dXJuIGltbWVkaWF0ZWx5IHdoaWxlIHRoZSBwcm9jZXNzIGtlZXBzIHJ1bm5pbmcuYFxuICAgICAgICApLFxuICAgIH0pLFxuICAgIG91dHB1dFNjaGVtYTogei5vYmplY3Qoe1xuICAgICAgY29tbWFuZElkOiB6XG4gICAgICAgIC5zdHJpbmcoKVxuICAgICAgICAuZGVzY3JpYmUoXCJDb21tYW5kIElELiBVc2UgdG8gcmVmZXJlbmNlIG9yIGtpbGwgcnVubmluZyBwcm9jZXNzZXMuXCIpLFxuICAgICAgc3Rkb3V0OiB6XG4gICAgICAgIC5zdHJpbmcoKVxuICAgICAgICAuZGVzY3JpYmUoXG4gICAgICAgICAgXCJDb21tYW5kIHN0ZG91dC4gVGFpbC10cnVuY2F0ZWQgaWYgbGFyZ2U7IGZ1bGwgY29udGVudCBpbiBvdXRwdXREaXIvc3Rkb3V0LnR4dC5cIlxuICAgICAgICApLFxuICAgICAgc3RkZXJyOiB6XG4gICAgICAgIC5zdHJpbmcoKVxuICAgICAgICAuZGVzY3JpYmUoXG4gICAgICAgICAgXCJDb21tYW5kIHN0ZGVyci4gVGFpbC10cnVuY2F0ZWQgaWYgbGFyZ2U7IGZ1bGwgY29udGVudCBpbiBvdXRwdXREaXIvc3RkZXJyLnR4dC5cIlxuICAgICAgICApLFxuICAgICAgZXhpdENvZGU6IHoubnVtYmVyKCkuZGVzY3JpYmUoXCJFeGl0IGNvZGUgKC0xIGlmIHN0aWxsIHJ1bm5pbmcpXCIpLFxuICAgICAgc3RhdHVzOiB6XG4gICAgICAgIC5lbnVtKFtcInJ1bm5pbmdcIiwgXCJjb21wbGV0ZWRcIiwgXCJmYWlsZWRcIl0pXG4gICAgICAgIC5kZXNjcmliZShcIlByb2Nlc3Mgc3RhdHVzXCIpLFxuICAgICAgb3V0cHV0RGlyOiB6XG4gICAgICAgIC5zdHJpbmcoKVxuICAgICAgICAuZGVzY3JpYmUoXG4gICAgICAgICAgXCJQYXRoIHRvIG91dHB1dCBkaXJlY3RvcnkgY29udGFpbmluZyBzdGRvdXQudHh0LCBzdGRlcnIudHh0LCBhbmQgbWV0YWRhdGEuanNvbi4gRW1wdHkgaWYgc3RpbGwgcnVubmluZy5cIlxuICAgICAgICApLFxuICAgIH0pLFxuICB9KSxcbiAgU2tpbGw6IHRvb2woe1xuICAgIGRlc2NyaXB0aW9uOlxuICAgICAgXCJMb2FkIGEgc2tpbGwncyBmdWxsIGluc3RydWN0aW9ucyBieSBuYW1lLiBDYWxsIHRoaXMgYmVmb3JlIGZvbGxvd2luZyBhIHNraWxsLlwiLFxuICAgIGlucHV0U2NoZW1hOiB6Lm9iamVjdCh7XG4gICAgICBsYWJlbDogelxuICAgICAgICAuc3RyaW5nKClcbiAgICAgICAgLmRlc2NyaWJlKFwiQSBsYWJlbCB0aGF0IGRlc2NyaWJlcyB0aGUgYWN0aW9uIGJlaW5nIHBlcmZvcm1lZFwiKSxcbiAgICAgIG5hbWU6IHouc3RyaW5nKCkuZGVzY3JpYmUoXCJTa2lsbCBuYW1lIGZyb20gdGhlIEF2YWlsYWJsZSBTa2lsbHMgbGlzdFwiKSxcbiAgICB9KSxcbiAgICBvdXRwdXRTY2hlbWE6IHoub2JqZWN0KHtcbiAgICAgIG5hbWU6IHouc3RyaW5nKCksXG4gICAgICBkZXNjcmlwdGlvbjogei5zdHJpbmcoKSxcbiAgICAgIGNvbnRlbnQ6IHouc3RyaW5nKCkuZGVzY3JpYmUoXCJGdWxsIFNLSUxMLm1kIGNvbnRlbnRcIiksXG4gICAgICBwYXRoOiB6LnN0cmluZygpLmRlc2NyaWJlKFwiUGF0aCB0byB0aGUgc2tpbGwgZGlyZWN0b3J5IGluIHRoZSBzYW5kYm94XCIpLFxuICAgIH0pLFxuICB9KSxcbiAgSmF2YVNjcmlwdDogdG9vbCh7XG4gICAgaW5wdXRTY2hlbWE6IHoub2JqZWN0KHtcbiAgICAgIGxhYmVsOiB6XG4gICAgICAgIC5zdHJpbmcoKVxuICAgICAgICAuZGVzY3JpYmUoXCJBIGxhYmVsIHRoYXQgZGVzY3JpYmVzIHRoZSBhY3Rpb24gYmVpbmcgcGVyZm9ybWVkXCIpLFxuICAgICAgY29kZTogelxuICAgICAgICAuc3RyaW5nKClcbiAgICAgICAgLmRlc2NyaWJlKFxuICAgICAgICAgIFwiSmF2YVNjcmlwdCBhc3luYyBmdW5jdGlvbiBib2R5LiBgY3R4YCBpcyBpbiBzY29wZS4gTXVzdCB1c2UgYHJldHVybmAgdG8gcHJvZHVjZSBvdXRwdXQuXCJcbiAgICAgICAgKSxcbiAgICB9KSxcbiAgfSksXG59IHNhdGlzZmllcyBUb29sU2V0O1xuXG5leHBvcnQgdHlwZSBCdWlsdEluVG9vbE5hbWUgPSBrZXlvZiB0eXBlb2YgYnVpbHRJblRvb2xzO1xuZXhwb3J0IGNvbnN0IGJ1aWx0aW5Ub29sTmFtZXMgPSBPYmplY3QuZnJvbUVudHJpZXMoXG4gIE9iamVjdC5lbnRyaWVzKGJ1aWx0SW5Ub29scykubWFwKChbbmFtZV0pID0+IFtuYW1lLCBuYW1lXSlcbikgYXMgeyBbSyBpbiBCdWlsdEluVG9vbE5hbWVdOiBLIH07XG5cbmNvbnN0IFNLSUxMX01EX1NVRkZJWCA9IC9cXC8/U0tJTExcXC5tZCQvO1xuXG5leHBvcnQgZnVuY3Rpb24gZ2V0VG9vbHMoe1xuICBhZ2VudCxcbiAgc2FuZGJveCxcbiAgc2tpbGxzUmVmLFxuICB3cml0ZXIsXG4gIGFib3J0Q29udHJvbGxlcixcbiAgbmV4dFBhcnRJbmRleFJlZixcbiAgYXNzaXN0YW50TWVzc2FnZUlkLFxuICBzZXNzaW9uSWQsXG4gIHN0ZXBMb2csXG4gIGNvbnRleHQsXG4gIHN0YXRlLFxuICBtZXNzYWdlcyxcbiAgYWN0aXZlVG9vbHMsXG59OiB7XG4gIGFnZW50OiBBbnlBZ2VudDtcbiAgc2FuZGJveDogU2FuZGJveEluc3RhbmNlO1xuICBza2lsbHNSZWY6IHsgY3VycmVudDogU2tpbGxTdW1tYXJ5W10gfTtcbiAgd3JpdGVyOiBTdHJlYW1Xcml0ZXI7XG4gIGFib3J0Q29udHJvbGxlcjogQWJvcnRDb250cm9sbGVyO1xuICBuZXh0UGFydEluZGV4UmVmOiB7IGN1cnJlbnQ6IG51bWJlciB9O1xuICBhc3Npc3RhbnRNZXNzYWdlSWQ6IHN0cmluZztcbiAgc2Vzc2lvbklkOiBzdHJpbmc7XG4gIHN0ZXBMb2c6IExvZ2dlcjtcbiAgY29udGV4dDogUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG4gIHN0YXRlOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgbWVzc2FnZXM6IFVJTWVzc2FnZVtdO1xuICBhY3RpdmVUb29scz86IHN0cmluZ1tdO1xufSk6IHtcbiAgcmF3VG9vbHM6IFRvb2xTZXQ7XG4gIHRvb2xzV2l0aE5lZWRzQXBwcm92YWw6IFRvb2xTZXQ7XG4gIHRvb2xDb250ZXh0OiBUb29sQ29udGV4dDtcbn0ge1xuICBjb25zdCByYXdUb29sczogVG9vbFNldCA9IHtcbiAgICBbYnVpbHRpblRvb2xOYW1lcy5SZWFkXTogdG9vbCh7XG4gICAgICAuLi5idWlsdEluVG9vbHMuUmVhZCxcbiAgICAgIGV4ZWN1dGU6IGFzeW5jICh7IHBhdGgsIHN0YXJ0TGluZSwgZW5kTGluZSB9KSA9PiB7XG4gICAgICAgIGNvbnN0IGZpbGVQYXRoID0gcGF0aDtcblxuICAgICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBzYW5kYm94LnJlYWRGaWxlKHsgcGF0aDogZmlsZVBhdGggfSk7XG5cbiAgICAgICAgaWYgKHJlc3VsdCBpbnN0YW5jZW9mIEVycm9yKSB7XG4gICAgICAgICAgbG9nLmVycm9yKFwiUmVhZCBmYWlsZWRcIiwgeyBlcnJvcjogcmVzdWx0Lm1lc3NhZ2UgfSk7XG4gICAgICAgICAgdGhyb3cgcmVzdWx0O1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKHJlc3VsdCA9PT0gbnVsbCkge1xuICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICBjb250ZW50OiBgRXJyb3I6IEZpbGUgbm90IGZvdW5kIC0gJHtmaWxlUGF0aH1gLFxuICAgICAgICAgICAgbWV0YWRhdGE6IHtcbiAgICAgICAgICAgICAgdG90YWxMaW5lczogMCxcbiAgICAgICAgICAgICAgbGluZXNTaG93bjogMCxcbiAgICAgICAgICAgICAgc3RhcnRMaW5lOiAwLFxuICAgICAgICAgICAgICBlbmRMaW5lOiAwLFxuICAgICAgICAgICAgICBpc1BhZ2luYXRlZDogZmFsc2UsXG4gICAgICAgICAgICAgIGZpbGVTaXplOiBcIjBcIixcbiAgICAgICAgICAgICAgcGF0aDogZmlsZVBhdGgsXG4gICAgICAgICAgICB9LFxuICAgICAgICAgIH07XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBmdWxsQ29udGVudCA9IHJlc3VsdC50b1N0cmluZyhcInV0Zi04XCIpO1xuICAgICAgICBjb25zdCBsaW5lcyA9IGZ1bGxDb250ZW50LnNwbGl0KFwiXFxuXCIpO1xuICAgICAgICAvLyBBIHRyYWlsaW5nIG5ld2xpbmUgcHJvZHVjZXMgYW4gZW1wdHkgbGFzdCBlbGVtZW50IFx1MjAxNCByZW1vdmUgaXRcbiAgICAgICAgLy8gc28gbGluZSBjb3VudCBtYXRjaGVzIGB3YyAtbGAgLyBgYXdrICdFTkR7cHJpbnQgTlJ9J2AgYmVoYXZpb3IuXG4gICAgICAgIGlmIChsaW5lcy5sZW5ndGggPiAwICYmIGxpbmVzLmF0KC0xKSA9PT0gXCJcIikge1xuICAgICAgICAgIGxpbmVzLnBvcCgpO1xuICAgICAgICB9XG4gICAgICAgIGNvbnN0IHRvdGFsTGluZXMgPSBsaW5lcy5sZW5ndGg7XG4gICAgICAgIGNvbnN0IGZpbGVCeXRlcyA9IEJ1ZmZlci5ieXRlTGVuZ3RoKGZ1bGxDb250ZW50KTtcbiAgICAgICAgY29uc3QgZmlsZVNpemUgPSBmb3JtYXRGaWxlU2l6ZShmaWxlQnl0ZXMpO1xuXG4gICAgICAgIC8vIERldGVybWluZSByYW5nZVxuICAgICAgICBjb25zdCBQQUdFX1NJWkUgPSAxMDA7XG4gICAgICAgIGxldCBhY3R1YWxTdGFydDogbnVtYmVyO1xuICAgICAgICBsZXQgYWN0dWFsRW5kOiBudW1iZXI7XG5cbiAgICAgICAgaWYgKHN0YXJ0TGluZSAhPT0gdW5kZWZpbmVkICYmIGVuZExpbmUgIT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgIGFjdHVhbFN0YXJ0ID0gc3RhcnRMaW5lO1xuICAgICAgICAgIGFjdHVhbEVuZCA9IGVuZExpbmU7XG4gICAgICAgIH0gZWxzZSBpZiAoc3RhcnRMaW5lICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICBhY3R1YWxTdGFydCA9IHN0YXJ0TGluZTtcbiAgICAgICAgICBhY3R1YWxFbmQgPSBNYXRoLm1pbihzdGFydExpbmUgKyBQQUdFX1NJWkUgLSAxLCB0b3RhbExpbmVzKTtcbiAgICAgICAgfSBlbHNlIGlmIChlbmRMaW5lICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICBhY3R1YWxTdGFydCA9IDE7XG4gICAgICAgICAgYWN0dWFsRW5kID0gZW5kTGluZTtcbiAgICAgICAgfSBlbHNlIGlmICh0b3RhbExpbmVzID4gMjAwKSB7XG4gICAgICAgICAgYWN0dWFsU3RhcnQgPSAxO1xuICAgICAgICAgIGFjdHVhbEVuZCA9IFBBR0VfU0laRTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBhY3R1YWxTdGFydCA9IDE7XG4gICAgICAgICAgYWN0dWFsRW5kID0gdG90YWxMaW5lcztcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIEV4dHJhY3QgdGhlIGxpbmUgcmFuZ2UgKDEtaW5kZXhlZClcbiAgICAgICAgY29uc3Qgc2xpY2VkTGluZXMgPSBsaW5lcy5zbGljZShhY3R1YWxTdGFydCAtIDEsIGFjdHVhbEVuZCk7XG4gICAgICAgIGNvbnN0IGNvbnRlbnQgPSBzbGljZWRMaW5lcy5qb2luKFwiXFxuXCIpO1xuXG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgbWV0YWRhdGE6IHtcbiAgICAgICAgICAgIHRvdGFsTGluZXMsXG4gICAgICAgICAgICBsaW5lc1Nob3duOiBNYXRoLm1heCgwLCBhY3R1YWxFbmQgLSBhY3R1YWxTdGFydCArIDEpLFxuICAgICAgICAgICAgc3RhcnRMaW5lOiBhY3R1YWxTdGFydCxcbiAgICAgICAgICAgIGVuZExpbmU6IGFjdHVhbEVuZCxcbiAgICAgICAgICAgIGlzUGFnaW5hdGVkOiBhY3R1YWxFbmQgPCB0b3RhbExpbmVzLFxuICAgICAgICAgICAgZmlsZVNpemUsXG4gICAgICAgICAgICBwYXRoOiBmaWxlUGF0aCxcbiAgICAgICAgICB9LFxuICAgICAgICAgIGNvbnRlbnQsXG4gICAgICAgIH07XG4gICAgICB9LFxuICAgIH0pLFxuICAgIFtidWlsdGluVG9vbE5hbWVzLkdyZXBdOiB0b29sKHtcbiAgICAgIC4uLmJ1aWx0SW5Ub29scy5HcmVwLFxuICAgICAgZXhlY3V0ZTogYXN5bmMgKHtcbiAgICAgICAgcGF0dGVybixcbiAgICAgICAgcGF0aCxcbiAgICAgICAgZmlsZVR5cGUsXG4gICAgICAgIGdsb2IsXG4gICAgICAgIGNhc2VTZW5zaXRpdmUsXG4gICAgICAgIGNvbnRleHRMaW5lcyxcbiAgICAgICAgbWF4Q291bnQsXG4gICAgICAgIGZpbGVzV2l0aE1hdGNoZXMsXG4gICAgICB9KSA9PiB7XG4gICAgICAgIGNvbnN0IHNlYXJjaFBhdGggPSBwYXRoID8/IFwiLlwiO1xuXG4gICAgICAgIGNvbnN0IGFyZ3M6IHN0cmluZ1tdID0gW107XG5cbiAgICAgICAgYXJncy5wdXNoKFwiLS1saW5lLW51bWJlclwiKTtcbiAgICAgICAgYXJncy5wdXNoKFwiLS1oZWFkaW5nXCIpO1xuICAgICAgICBhcmdzLnB1c2goXCItLWNvbG9yXCIsIFwibmV2ZXJcIik7XG5cbiAgICAgICAgaWYgKCFjYXNlU2Vuc2l0aXZlKSB7XG4gICAgICAgICAgYXJncy5wdXNoKFwiLWlcIik7XG4gICAgICAgIH1cblxuICAgICAgICBpZiAoZmlsZVR5cGUpIHtcbiAgICAgICAgICBhcmdzLnB1c2goXCItLXR5cGVcIiwgZmlsZVR5cGUpO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKGdsb2IpIHtcbiAgICAgICAgICBhcmdzLnB1c2goXCItLWdsb2JcIiwgZ2xvYik7XG4gICAgICAgIH1cblxuICAgICAgICBpZiAoY29udGV4dExpbmVzICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICBhcmdzLnB1c2goXCItQ1wiLCBTdHJpbmcoY29udGV4dExpbmVzKSk7XG4gICAgICAgIH1cblxuICAgICAgICBpZiAobWF4Q291bnQgIT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgIGFyZ3MucHVzaChcIi0tbWF4LWNvdW50XCIsIFN0cmluZyhtYXhDb3VudCkpO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKGZpbGVzV2l0aE1hdGNoZXMpIHtcbiAgICAgICAgICBhcmdzLnB1c2goXCItLWZpbGVzLXdpdGgtbWF0Y2hlc1wiKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGFyZ3MucHVzaChcIi0tXCIsIHBhdHRlcm4sIHNlYXJjaFBhdGgpO1xuXG4gICAgICAgIGxldCByZXN1bHQgPSBhd2FpdCBzYW5kYm94LmV4ZWMoeyBjb21tYW5kOiBcInJnXCIsIGFyZ3MgfSk7XG5cbiAgICAgICAgaWYgKHJlc3VsdCBpbnN0YW5jZW9mIEVycm9yICYmIGlzUmdOb3RGb3VuZEVycm9yKHJlc3VsdCkpIHtcbiAgICAgICAgICBsb2cud2FybihcInJnIG5vdCBmb3VuZCwgaW5zdGFsbGluZyByaXBncmVwXCIpO1xuICAgICAgICAgIGNvbnN0IGluc3RhbGxSZXN1bHQgPSBhd2FpdCBzYW5kYm94LmV4ZWMoe1xuICAgICAgICAgICAgY29tbWFuZDogXCJiYXNoXCIsXG4gICAgICAgICAgICBhcmdzOiBbXG4gICAgICAgICAgICAgIFwiLWNcIixcbiAgICAgICAgICAgICAgW1xuICAgICAgICAgICAgICAgIFwiY29tbWFuZCAtdiByZyA+L2Rldi9udWxsIDI+JjEgJiYgZXhpdCAwXCIsXG4gICAgICAgICAgICAgICAgXCJkbmYgaW5zdGFsbCAteSByaXBncmVwIDI+L2Rldi9udWxsICYmIGV4aXQgMFwiLFxuICAgICAgICAgICAgICAgIFwiKGFwdC1nZXQgdXBkYXRlIC1xcSAmJiBhcHQtZ2V0IGluc3RhbGwgLXkgLXFxIHJpcGdyZXApIDI+L2Rldi9udWxsICYmIGV4aXQgMFwiLFxuICAgICAgICAgICAgICAgIFwiYXBrIGFkZCAtLW5vLWNhY2hlIHJpcGdyZXAgMj4vZGV2L251bGwgJiYgZXhpdCAwXCIsXG4gICAgICAgICAgICAgICAgXCJjdXJsIC1zTCBodHRwczovL2dpdGh1Yi5jb20vQnVybnRTdXNoaS9yaXBncmVwL3JlbGVhc2VzL2Rvd25sb2FkLzE0LjEuMS9yaXBncmVwLTE0LjEuMS14ODZfNjQtdW5rbm93bi1saW51eC1tdXNsLnRhci5neiB8IHRhciB4eiAtQyAvdG1wICYmIGluc3RhbGwgL3RtcC9yaXBncmVwLTE0LjEuMS14ODZfNjQtdW5rbm93bi1saW51eC1tdXNsL3JnIC91c3IvbG9jYWwvYmluL3JnICYmIHJtIC1yZiAvdG1wL3JpcGdyZXAtMTQuMS4xLXg4Nl82NC11bmtub3duLWxpbnV4LW11c2xcIixcbiAgICAgICAgICAgICAgXS5qb2luKFwiXFxuXCIpLFxuICAgICAgICAgICAgXSxcbiAgICAgICAgICAgIHN1ZG86IHRydWUsXG4gICAgICAgICAgfSk7XG4gICAgICAgICAgaWYgKCEoaW5zdGFsbFJlc3VsdCBpbnN0YW5jZW9mIEVycm9yKSkge1xuICAgICAgICAgICAgY29uc3QgaW5zdGFsbE91dHB1dCA9IGF3YWl0IGluc3RhbGxSZXN1bHQucmVzdWx0O1xuICAgICAgICAgICAgaWYgKGluc3RhbGxPdXRwdXQuZXhpdENvZGUgIT09IDApIHtcbiAgICAgICAgICAgICAgbG9nLndhcm4oXCJyaXBncmVwIGluc3RhbGwgZmFpbGVkXCIsIHtcbiAgICAgICAgICAgICAgICBzdGRlcnI6IGluc3RhbGxPdXRwdXQuc3RkZXJyLFxuICAgICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgICAgcmVzdWx0ID0gYXdhaXQgc2FuZGJveC5leGVjKHsgY29tbWFuZDogXCJyZ1wiLCBhcmdzIH0pO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKHJlc3VsdCBpbnN0YW5jZW9mIEVycm9yICYmIGlzUmdOb3RGb3VuZEVycm9yKHJlc3VsdCkpIHtcbiAgICAgICAgICBsb2cud2FybihcInJnIHVuYXZhaWxhYmxlLCBmYWxsaW5nIGJhY2sgdG8gZ3JlcFwiKTtcbiAgICAgICAgICBjb25zdCBncmVwQXJncyA9IFtcIi1yblwiLCBcIi0tY29sb3I9bmV2ZXJcIl07XG4gICAgICAgICAgaWYgKCFjYXNlU2Vuc2l0aXZlKSB7XG4gICAgICAgICAgICBncmVwQXJncy5wdXNoKFwiLWlcIik7XG4gICAgICAgICAgfVxuICAgICAgICAgIGlmIChjb250ZXh0TGluZXMgIT09IHVuZGVmaW5lZCkge1xuICAgICAgICAgICAgZ3JlcEFyZ3MucHVzaChcIi1DXCIsIFN0cmluZyhjb250ZXh0TGluZXMpKTtcbiAgICAgICAgICB9XG4gICAgICAgICAgaWYgKG1heENvdW50ICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgIGdyZXBBcmdzLnB1c2goXCItbVwiLCBTdHJpbmcobWF4Q291bnQpKTtcbiAgICAgICAgICB9XG4gICAgICAgICAgaWYgKGZpbGVzV2l0aE1hdGNoZXMpIHtcbiAgICAgICAgICAgIGdyZXBBcmdzLnB1c2goXCItbFwiKTtcbiAgICAgICAgICB9XG4gICAgICAgICAgaWYgKGdsb2IpIHtcbiAgICAgICAgICAgIGdyZXBBcmdzLnB1c2goYC0taW5jbHVkZT0ke2dsb2J9YCk7XG4gICAgICAgICAgfVxuICAgICAgICAgIGlmIChmaWxlVHlwZSkge1xuICAgICAgICAgICAgY29uc3QgaW5jbHVkZU1hcDogUmVjb3JkPHN0cmluZywgc3RyaW5nW10+ID0ge1xuICAgICAgICAgICAgICB0czogW1wiKi50c1wiLCBcIioudHN4XCIsIFwiKi5tdHNcIiwgXCIqLmN0c1wiXSxcbiAgICAgICAgICAgICAganM6IFtcIiouanNcIiwgXCIqLmpzeFwiLCBcIioubWpzXCIsIFwiKi5janNcIl0sXG4gICAgICAgICAgICAgIHB5OiBbXCIqLnB5XCJdLFxuICAgICAgICAgICAgICBydXN0OiBbXCIqLnJzXCJdLFxuICAgICAgICAgICAgICBnbzogW1wiKi5nb1wiXSxcbiAgICAgICAgICAgICAgamF2YTogW1wiKi5qYXZhXCJdLFxuICAgICAgICAgICAgICBtZDogW1wiKi5tZFwiXSxcbiAgICAgICAgICAgICAganNvbjogW1wiKi5qc29uXCJdLFxuICAgICAgICAgICAgICBjc3M6IFtcIiouY3NzXCJdLFxuICAgICAgICAgICAgICBodG1sOiBbXCIqLmh0bWxcIl0sXG4gICAgICAgICAgICAgIHlhbWw6IFtcIioueW1sXCIsIFwiKi55YW1sXCJdLFxuICAgICAgICAgICAgfTtcbiAgICAgICAgICAgIGZvciAoY29uc3QgZXh0IG9mIGluY2x1ZGVNYXBbZmlsZVR5cGVdID8/IFtgKi4ke2ZpbGVUeXBlfWBdKSB7XG4gICAgICAgICAgICAgIGdyZXBBcmdzLnB1c2goYC0taW5jbHVkZT0ke2V4dH1gKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgICAgZ3JlcEFyZ3MucHVzaChcIi0tXCIsIHBhdHRlcm4sIHNlYXJjaFBhdGgpO1xuICAgICAgICAgIHJlc3VsdCA9IGF3YWl0IHNhbmRib3guZXhlYyh7XG4gICAgICAgICAgICBjb21tYW5kOiBcImdyZXBcIixcbiAgICAgICAgICAgIGFyZ3M6IGdyZXBBcmdzLFxuICAgICAgICAgIH0pO1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKHJlc3VsdCBpbnN0YW5jZW9mIEVycm9yKSB7XG4gICAgICAgICAgbG9nLmVycm9yKFwiR3JlcCBmYWlsZWRcIiwgeyBlcnJvcjogcmVzdWx0Lm1lc3NhZ2UgfSk7XG4gICAgICAgICAgdGhyb3cgcmVzdWx0O1xuICAgICAgICB9XG5cbiAgICAgICAgY29uc3QgeyBzdGRvdXQsIHN0ZGVyciB9ID0gYXdhaXQgcmVzdWx0LnJlc3VsdDtcblxuICAgICAgICBpZiAoc3RkZXJyICYmICFzdGRlcnIudG9Mb3dlckNhc2UoKS5pbmNsdWRlcyhcIm5vIG1hdGNoZXNcIikpIHtcbiAgICAgICAgICBsb2cud2FybihcIkdyZXAgc3RkZXJyXCIsIHsgc3RkZXJyIH0pO1xuICAgICAgICB9XG5cbiAgICAgICAgLy8gVHJ1bmNhdGUgb3V0cHV0IHRvIHByZXZlbnQgXCJpbnB1dCB0b28gbG9uZ1wiIGVycm9ycyAoNTBrIGNoYXJzIFx1MjI0OCAxMi41ayB0b2tlbnMpXG4gICAgICAgIGNvbnN0IE1BWF9HUkVQX09VVFBVVF9DSEFSUyA9IDUwXzAwMDtcbiAgICAgICAgbGV0IGZpbmFsT3V0cHV0ID0gc3Rkb3V0O1xuICAgICAgICBsZXQgd2FzVHJ1bmNhdGVkID0gZmFsc2U7XG4gICAgICAgIGlmIChmaW5hbE91dHB1dC5sZW5ndGggPiBNQVhfR1JFUF9PVVRQVVRfQ0hBUlMpIHtcbiAgICAgICAgICBmaW5hbE91dHB1dCA9XG4gICAgICAgICAgICBmaW5hbE91dHB1dC5zbGljZSgwLCBNQVhfR1JFUF9PVVRQVVRfQ0hBUlMpICtcbiAgICAgICAgICAgIFwiXFxuXFxuW091dHB1dCB0cnVuY2F0ZWQgLSB1c2UgbW9yZSBzcGVjaWZpYyBwYXR0ZXJuIG9yIHBhdGhdXCI7XG4gICAgICAgICAgd2FzVHJ1bmNhdGVkID0gdHJ1ZTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IGxpbmVzID0gZmluYWxPdXRwdXRcbiAgICAgICAgICAudHJpbSgpXG4gICAgICAgICAgLnNwbGl0KFwiXFxuXCIpXG4gICAgICAgICAgLmZpbHRlcigobCkgPT4gbC5sZW5ndGggPiAwKTtcbiAgICAgICAgY29uc3QgZmlsZUNvdW50ID0gZmlsZXNXaXRoTWF0Y2hlc1xuICAgICAgICAgID8gbGluZXMubGVuZ3RoXG4gICAgICAgICAgOiBuZXcgU2V0KFxuICAgICAgICAgICAgICBsaW5lc1xuICAgICAgICAgICAgICAgIC5maWx0ZXIoKGwpID0+ICFsLnN0YXJ0c1dpdGgoXCIgXCIpICYmIGwuaW5jbHVkZXMoXCI6XCIpKVxuICAgICAgICAgICAgICAgIC5tYXAoKGwpID0+IGwuc3BsaXQoXCI6XCIpWzBdKVxuICAgICAgICAgICAgKS5zaXplO1xuXG4gICAgICAgIHJldHVybiB7XG4gICAgICAgICAgc3VtbWFyeToge1xuICAgICAgICAgICAgbWF0Y2hDb3VudDogZmlsZXNXaXRoTWF0Y2hlc1xuICAgICAgICAgICAgICA/IDBcbiAgICAgICAgICAgICAgOiBsaW5lcy5maWx0ZXIoKGwpID0+IGwuaW5jbHVkZXMoXCI6XCIpKS5sZW5ndGgsXG4gICAgICAgICAgICBmaWxlQ291bnQsXG4gICAgICAgICAgICBzZWFyY2hQYXRoLFxuICAgICAgICAgICAgcGF0dGVybixcbiAgICAgICAgICAgIHdhc1RydW5jYXRlZCxcbiAgICAgICAgICB9LFxuICAgICAgICAgIG1hdGNoZXM6IGZpbmFsT3V0cHV0IHx8IFwiKG5vIG1hdGNoZXMgZm91bmQpXCIsXG4gICAgICAgIH07XG4gICAgICB9LFxuICAgIH0pLFxuICAgIFtidWlsdGluVG9vbE5hbWVzLkxpc3RdOiB0b29sKHtcbiAgICAgIC4uLmJ1aWx0SW5Ub29scy5MaXN0LFxuICAgICAgZXhlY3V0ZTogYXN5bmMgKHsgcGF0aCwgZGVwdGgsIGluY2x1ZGVIaWRkZW4sIGZpbGVzT25seSwgcGF0dGVybiB9KSA9PiB7XG4gICAgICAgIGNvbnN0IHNlYXJjaFBhdGggPSBwYXRoID8/IFwiLlwiO1xuXG4gICAgICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IHNhbmRib3guZXhlYyh7XG4gICAgICAgICAgY29tbWFuZDogXCJiYXNoXCIsXG4gICAgICAgICAgYXJnczogW1xuICAgICAgICAgICAgXCItY1wiLFxuICAgICAgICAgICAgYFxuICAgICAgICAgICAgc2V0IC1lXG4gICAgICAgICAgICBTRUFSQ0hfUEFUSD1cIiQxXCJcbiAgICAgICAgICAgIERFUFRIPVwiJDJcIlxuICAgICAgICAgICAgSU5DTFVERV9ISURERU49XCIkM1wiXG4gICAgICAgICAgICBGSUxFU19PTkxZPVwiJDRcIlxuICAgICAgICAgICAgUEFUVEVSTj1cIiQ1XCJcblxuICAgICAgICAgICAgIyBCdWlsZCBmaW5kIGNvbW1hbmQgYXJndW1lbnRzXG4gICAgICAgICAgICBGSU5EX0FSR1M9XCJcIlxuICAgICAgICAgICAgWyAtbiBcIiRERVBUSFwiIF0gJiYgRklORF9BUkdTPVwiJEZJTkRfQVJHUyAtbWF4ZGVwdGggJERFUFRIXCJcbiAgICAgICAgICAgIFsgXCIkSU5DTFVERV9ISURERU5cIiAhPSBcInRydWVcIiBdICYmIEZJTkRfQVJHUz1cIiRGSU5EX0FSR1MgISAtcGF0aCAnKi8uKidcIlxuICAgICAgICAgICAgWyBcIiRGSUxFU19PTkxZXCIgPSBcInRydWVcIiBdICYmIEZJTkRfQVJHUz1cIiRGSU5EX0FSR1MgLXR5cGUgZlwiXG4gICAgICAgICAgICBbIC1uIFwiJFBBVFRFUk5cIiBdICYmIEZJTkRfQVJHUz1cIiRGSU5EX0FSR1MgLW5hbWUgJyRQQVRURVJOJ1wiXG5cbiAgICAgICAgICAgICMgR2V0IGxpc3RpbmdcbiAgICAgICAgICAgIExJU1RJTkc9JChldmFsIFwiZmluZCAnJFNFQVJDSF9QQVRIJyAkRklORF9BUkdTXCIgMj4vZGV2L251bGwgfCBzb3J0KVxuXG4gICAgICAgICAgICAjIEdldCBjb3VudHNcbiAgICAgICAgICAgIENPVU5UX0FSR1M9XCJcIlxuICAgICAgICAgICAgWyAtbiBcIiRERVBUSFwiIF0gJiYgQ09VTlRfQVJHUz1cIiRDT1VOVF9BUkdTIC1tYXhkZXB0aCAkREVQVEhcIlxuICAgICAgICAgICAgWyBcIiRJTkNMVURFX0hJRERFTlwiICE9IFwidHJ1ZVwiIF0gJiYgQ09VTlRfQVJHUz1cIiRDT1VOVF9BUkdTICEgLXBhdGggJyovLionXCJcblxuICAgICAgICAgICAgRklMRV9DT1VOVD0kKGV2YWwgXCJmaW5kICckU0VBUkNIX1BBVEgnICRDT1VOVF9BUkdTIC10eXBlIGZcIiAyPi9kZXYvbnVsbCB8IHdjIC1sKVxuICAgICAgICAgICAgRElSX0NPVU5UPSQoZXZhbCBcImZpbmQgJyRTRUFSQ0hfUEFUSCcgJENPVU5UX0FSR1MgLXR5cGUgZFwiIDI+L2Rldi9udWxsIHwgd2MgLWwpXG5cbiAgICAgICAgICAgICMgT3V0cHV0OiBjb3VudHMgZmlyc3QsIHRoZW4gbGlzdGluZ1xuICAgICAgICAgICAgZWNobyBcIiRGSUxFX0NPVU5UfCRESVJfQ09VTlRcIlxuICAgICAgICAgICAgZWNobyBcInx8fExJU1RJTkd8fHxcIlxuICAgICAgICAgICAgZWNobyBcIiRMSVNUSU5HXCIgfCBzZWQgXCJzfF4kU0VBUkNIX1BBVEh8LnxcIlxuICAgICAgICAgIGAsXG4gICAgICAgICAgICBcIi0tXCIsXG4gICAgICAgICAgICBzZWFyY2hQYXRoLFxuICAgICAgICAgICAgZGVwdGg/LnRvU3RyaW5nKCkgfHwgXCJcIixcbiAgICAgICAgICAgIGluY2x1ZGVIaWRkZW4gPyBcInRydWVcIiA6IFwiZmFsc2VcIixcbiAgICAgICAgICAgIGZpbGVzT25seSA/IFwidHJ1ZVwiIDogXCJmYWxzZVwiLFxuICAgICAgICAgICAgcGF0dGVybiB8fCBcIlwiLFxuICAgICAgICAgIF0sXG4gICAgICAgIH0pO1xuXG4gICAgICAgIGlmIChyZXN1bHQgaW5zdGFuY2VvZiBFcnJvcikge1xuICAgICAgICAgIGxvZy5lcnJvcihcIkxpc3QgZmFpbGVkXCIsIHsgZXJyb3I6IHJlc3VsdC5tZXNzYWdlIH0pO1xuICAgICAgICAgIHRocm93IHJlc3VsdDtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IHsgc3Rkb3V0LCBzdGRlcnIgfSA9IGF3YWl0IHJlc3VsdC5yZXN1bHQ7XG5cbiAgICAgICAgaWYgKHN0ZGVycikge1xuICAgICAgICAgIGxvZy53YXJuKFwiTGlzdCBzdGRlcnJcIiwgeyBzdGRlcnIgfSk7XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBbY291bnRzTGluZSwgLi4ucmVzdF0gPSBzdGRvdXQuc3BsaXQoXCJ8fHxMSVNUSU5HfHx8XCIpO1xuICAgICAgICBjb25zdCBsaXN0aW5nID0gcmVzdC5qb2luKFwifHx8TElTVElOR3x8fFwiKS50cmltKCk7XG4gICAgICAgIGNvbnN0IFtmaWxlQ291bnRTdHIsIGRpckNvdW50U3RyXSA9IGNvdW50c0xpbmUudHJpbSgpLnNwbGl0KFwifFwiKTtcblxuICAgICAgICBjb25zdCB0b3RhbEZpbGVzID0gTnVtYmVyLnBhcnNlSW50KGZpbGVDb3VudFN0ciwgMTApIHx8IDA7XG4gICAgICAgIGNvbnN0IHRvdGFsRGlycyA9IE51bWJlci5wYXJzZUludChkaXJDb3VudFN0ciwgMTApIHx8IDA7XG4gICAgICAgIGNvbnN0IGxpbmVzID0gbGlzdGluZy5zcGxpdChcIlxcblwiKS5maWx0ZXIoKGwpID0+IGwubGVuZ3RoID4gMCk7XG5cbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICBzdW1tYXJ5OiB7XG4gICAgICAgICAgICB0b3RhbEl0ZW1zOiBsaW5lcy5sZW5ndGgsXG4gICAgICAgICAgICB0b3RhbEZpbGVzLFxuICAgICAgICAgICAgdG90YWxEaXJzLFxuICAgICAgICAgICAgc2VhcmNoUGF0aCxcbiAgICAgICAgICAgIGRlcHRoLFxuICAgICAgICAgIH0sXG4gICAgICAgICAgbGlzdGluZyxcbiAgICAgICAgfTtcbiAgICAgIH0sXG4gICAgfSksXG4gICAgW2J1aWx0aW5Ub29sTmFtZXMuV3JpdGVdOiB0b29sKHtcbiAgICAgIC4uLmJ1aWx0SW5Ub29scy5Xcml0ZSxcbiAgICAgIGV4ZWN1dGU6IGFzeW5jICh7IHBhdGgsIGNvbnRlbnQgfSkgPT4ge1xuICAgICAgICBjb25zdCBmaWxlUGF0aCA9IHBhdGg7XG5cbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICBhd2FpdCBzYW5kYm94LndyaXRlRmlsZXMoe1xuICAgICAgICAgICAgZmlsZXM6IFt7IHBhdGg6IGZpbGVQYXRoLCBjb250ZW50IH1dLFxuICAgICAgICAgICAgZGVzdFBhdGg6IFwiLlwiLFxuICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgIHN1Y2Nlc3M6IHRydWUsXG4gICAgICAgICAgICBwYXRoOiBmaWxlUGF0aCxcbiAgICAgICAgICAgIGJ5dGVzV3JpdHRlbjogQnVmZmVyLmJ5dGVMZW5ndGgoY29udGVudCwgXCJ1dGY4XCIpLFxuICAgICAgICAgIH07XG4gICAgICAgIH0gY2F0Y2ggKGVycikge1xuICAgICAgICAgIGNvbnN0IGVycm9yTXNnID0gZXJyIGluc3RhbmNlb2YgRXJyb3IgPyBlcnIubWVzc2FnZSA6IFN0cmluZyhlcnIpO1xuICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICBzdWNjZXNzOiBmYWxzZSxcbiAgICAgICAgICAgIHBhdGg6IGZpbGVQYXRoLFxuICAgICAgICAgICAgYnl0ZXNXcml0dGVuOiAwLFxuICAgICAgICAgICAgZXJyb3I6IGVycm9yTXNnLFxuICAgICAgICAgIH07XG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgfSksXG4gICAgW2J1aWx0aW5Ub29sTmFtZXMuRWRpdF06IHRvb2woe1xuICAgICAgLi4uYnVpbHRJblRvb2xzLkVkaXQsXG4gICAgICBleGVjdXRlOiBhc3luYyAoeyBwYXRoLCBvbGRfc3RyaW5nLCBuZXdfc3RyaW5nIH0pID0+IHtcbiAgICAgICAgY29uc3QgZmlsZVBhdGggPSBwYXRoO1xuXG4gICAgICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IHNhbmRib3gucmVhZEZpbGUoeyBwYXRoOiBmaWxlUGF0aCB9KTtcblxuICAgICAgICBpZiAocmVzdWx0IGluc3RhbmNlb2YgRXJyb3IpIHtcbiAgICAgICAgICByZXR1cm4geyBzdWNjZXNzOiBmYWxzZSwgcGF0aDogZmlsZVBhdGgsIGVycm9yOiByZXN1bHQubWVzc2FnZSB9O1xuICAgICAgICB9XG5cbiAgICAgICAgaWYgKHJlc3VsdCA9PT0gbnVsbCkge1xuICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICBzdWNjZXNzOiBmYWxzZSxcbiAgICAgICAgICAgIHBhdGg6IGZpbGVQYXRoLFxuICAgICAgICAgICAgZXJyb3I6IGBGaWxlIG5vdCBmb3VuZDogJHtmaWxlUGF0aH1gLFxuICAgICAgICAgIH07XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBjb250ZW50ID0gcmVzdWx0LnRvU3RyaW5nKFwidXRmLThcIik7XG4gICAgICAgIGNvbnN0IG9jY3VycmVuY2VzID0gY29udGVudC5zcGxpdChvbGRfc3RyaW5nKS5sZW5ndGggLSAxO1xuXG4gICAgICAgIGlmIChvY2N1cnJlbmNlcyA9PT0gMCkge1xuICAgICAgICAgIHJldHVybiB7XG4gICAgICAgICAgICBzdWNjZXNzOiBmYWxzZSxcbiAgICAgICAgICAgIHBhdGg6IGZpbGVQYXRoLFxuICAgICAgICAgICAgZXJyb3I6IFwib2xkX3N0cmluZyBub3QgZm91bmQgaW4gZmlsZVwiLFxuICAgICAgICAgIH07XG4gICAgICAgIH1cblxuICAgICAgICBpZiAob2NjdXJyZW5jZXMgPiAxKSB7XG4gICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgIHN1Y2Nlc3M6IGZhbHNlLFxuICAgICAgICAgICAgcGF0aDogZmlsZVBhdGgsXG4gICAgICAgICAgICBlcnJvcjogYG9sZF9zdHJpbmcgYXBwZWFycyAke29jY3VycmVuY2VzfSB0aW1lcyBpbiBmaWxlIChtdXN0IGJlIHVuaXF1ZSkuIEluY2x1ZGUgbW9yZSBzdXJyb3VuZGluZyBjb250ZXh0IHRvIG1ha2UgdGhlIG1hdGNoIHVuaXF1ZS5gLFxuICAgICAgICAgIH07XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBuZXdDb250ZW50ID0gY29udGVudC5yZXBsYWNlKG9sZF9zdHJpbmcsIG5ld19zdHJpbmcpO1xuXG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgYXdhaXQgc2FuZGJveC53cml0ZUZpbGVzKHtcbiAgICAgICAgICAgIGZpbGVzOiBbeyBwYXRoOiBmaWxlUGF0aCwgY29udGVudDogbmV3Q29udGVudCB9XSxcbiAgICAgICAgICAgIGRlc3RQYXRoOiBcIi5cIixcbiAgICAgICAgICB9KTtcblxuICAgICAgICAgIHJldHVybiB7IHN1Y2Nlc3M6IHRydWUsIHBhdGg6IGZpbGVQYXRoIH07XG4gICAgICAgIH0gY2F0Y2ggKGVycikge1xuICAgICAgICAgIGNvbnN0IGVycm9yTXNnID0gZXJyIGluc3RhbmNlb2YgRXJyb3IgPyBlcnIubWVzc2FnZSA6IFN0cmluZyhlcnIpO1xuICAgICAgICAgIHJldHVybiB7IHN1Y2Nlc3M6IGZhbHNlLCBwYXRoOiBmaWxlUGF0aCwgZXJyb3I6IGVycm9yTXNnIH07XG4gICAgICAgIH1cbiAgICAgIH0sXG4gICAgfSksXG4gICAgW2J1aWx0aW5Ub29sTmFtZXMuQmFzaF06IHRvb2woe1xuICAgICAgLi4uYnVpbHRJblRvb2xzLkJhc2gsXG4gICAgICBleGVjdXRlOiBhc3luYyAoeyBjb21tYW5kLCB3YWl0VW50aWwgfSkgPT4ge1xuICAgICAgICBjb25zdCB7IGNyZWF0ZVByb2Nlc3NNYW5hZ2VyIH0gPSBhd2FpdCBpbXBvcnQoXG4gICAgICAgICAgXCIuLi9zYW5kYm94L3Byb2Nlc3MtbWFuYWdlclwiXG4gICAgICAgICk7XG5cbiAgICAgICAgY29uc3QgcHJvY2Vzc01hbmFnZXIgPSBjcmVhdGVQcm9jZXNzTWFuYWdlcih7IHNhbmRib3ggfSk7XG5cbiAgICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgcHJvY2Vzc01hbmFnZXIucnVuKHsgY29tbWFuZCwgd2FpdFVudGlsIH0pO1xuXG4gICAgICAgIGNvbnN0IE1BWF9TVERPVVQgPSA1MF8wMDA7XG4gICAgICAgIGNvbnN0IE1BWF9TVERFUlIgPSAxMF8wMDA7XG5cbiAgICAgICAgbGV0IHsgc3Rkb3V0LCBzdGRlcnIgfSA9IHJlc3VsdDtcbiAgICAgICAgaWYgKHN0ZG91dC5sZW5ndGggPiBNQVhfU1RET1VUKSB7XG4gICAgICAgICAgc3Rkb3V0ID1cbiAgICAgICAgICAgIGBbdHJ1bmNhdGVkIFx1MjAxNCBzaG93aW5nIGxhc3QgJHtNQVhfU1RET1VUfSBjaGFycy4gRnVsbDogJHtyZXN1bHQub3V0cHV0RGlyfS9zdGRvdXQudHh0XVxcblxcbmAgK1xuICAgICAgICAgICAgc3Rkb3V0LnNsaWNlKC1NQVhfU1RET1VUKTtcbiAgICAgICAgfVxuICAgICAgICBpZiAoc3RkZXJyLmxlbmd0aCA+IE1BWF9TVERFUlIpIHtcbiAgICAgICAgICBzdGRlcnIgPVxuICAgICAgICAgICAgYFt0cnVuY2F0ZWQgXHUyMDE0IHNob3dpbmcgbGFzdCAke01BWF9TVERFUlJ9IGNoYXJzLiBGdWxsOiAke3Jlc3VsdC5vdXRwdXREaXJ9L3N0ZGVyci50eHRdXFxuXFxuYCArXG4gICAgICAgICAgICBzdGRlcnIuc2xpY2UoLU1BWF9TVERFUlIpO1xuICAgICAgICB9XG5cbiAgICAgICAgcmV0dXJuIHsgLi4ucmVzdWx0LCBzdGRvdXQsIHN0ZGVyciB9O1xuICAgICAgfSxcbiAgICB9KSxcbiAgfTtcblxuICAvLyBBZGQgdXNlci1kZWZpbmVkIHRvb2xzXG4gIGZvciAoY29uc3QgW25hbWUsIHRdIG9mIE9iamVjdC5lbnRyaWVzKGFnZW50Lm9wdGlvbnMudG9vbHMgPz8ge30pKSB7XG4gICAgcmF3VG9vbHNbbmFtZV0gPSB0IGFzIFRvb2xTZXRbc3RyaW5nXTtcbiAgfVxuXG4gIGlmIChza2lsbHNSZWYuY3VycmVudC5sZW5ndGggPiAwKSB7XG4gICAgcmF3VG9vbHNbYnVpbHRpblRvb2xOYW1lcy5Ta2lsbF0gPSB0b29sKHtcbiAgICAgIC4uLmJ1aWx0SW5Ub29scy5Ta2lsbCxcbiAgICAgIGV4ZWN1dGU6IGFzeW5jICh7IG5hbWUgfSkgPT4ge1xuICAgICAgICBjb25zdCBza2lsbHMgPSBza2lsbHNSZWYuY3VycmVudDtcbiAgICAgICAgY29uc3Qgc2tpbGwgPSBza2lsbHMuZmluZChcbiAgICAgICAgICAocykgPT4gcy5uYW1lLnRvTG93ZXJDYXNlKCkgPT09IG5hbWUudG9Mb3dlckNhc2UoKVxuICAgICAgICApO1xuICAgICAgICBpZiAoIXNraWxsKSB7XG4gICAgICAgICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgICAgICAgYFNraWxsIG5vdCBmb3VuZDogXCIke25hbWV9XCIuIEF2YWlsYWJsZTogJHtza2lsbHMubWFwKChzKSA9PiBzLm5hbWUpLmpvaW4oXCIsIFwiKX1gXG4gICAgICAgICAgKTtcbiAgICAgICAgfVxuICAgICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBzYW5kYm94LnJlYWRGaWxlKHtcbiAgICAgICAgICBwYXRoOiBza2lsbC5za2lsbE1kUGF0aCxcbiAgICAgICAgfSk7XG4gICAgICAgIGlmIChyZXN1bHQgaW5zdGFuY2VvZiBFcnJvcikge1xuICAgICAgICAgIHRocm93IHJlc3VsdDtcbiAgICAgICAgfVxuICAgICAgICBpZiAocmVzdWx0ID09PSBudWxsKSB7XG4gICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBTa2lsbCBmaWxlIG5vdCBmb3VuZDogJHtza2lsbC5za2lsbE1kUGF0aH1gKTtcbiAgICAgICAgfVxuICAgICAgICBjb25zdCByYXcgPVxuICAgICAgICAgIHR5cGVvZiByZXN1bHQgPT09IFwic3RyaW5nXCIgPyByZXN1bHQgOiByZXN1bHQudG9TdHJpbmcoXCJ1dGYtOFwiKTtcbiAgICAgICAgY29uc3QgZW5kTWFya2VyID0gcmF3LmluZGV4T2YoXCItLS1cIiwgMyk7XG4gICAgICAgIGNvbnN0IGNvbnRlbnQgPVxuICAgICAgICAgIGVuZE1hcmtlciA9PT0gLTEgPyByYXcgOiByYXcuc2xpY2UoZW5kTWFya2VyICsgMykudHJpbSgpO1xuICAgICAgICBjb25zdCBza2lsbERpciA9IHNraWxsLnNraWxsTWRQYXRoLnJlcGxhY2UoU0tJTExfTURfU1VGRklYLCBcIlwiKTtcbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICBuYW1lOiBza2lsbC5uYW1lLFxuICAgICAgICAgIGRlc2NyaXB0aW9uOiBza2lsbC5kZXNjcmlwdGlvbixcbiAgICAgICAgICBjb250ZW50LFxuICAgICAgICAgIHBhdGg6IHNraWxsRGlyLFxuICAgICAgICB9O1xuICAgICAgfSxcbiAgICB9KTtcbiAgfVxuXG4gIGNvbnN0IG9uU3ViVG9vbENhbGw6IE9uU3ViVG9vbENhbGwgPSBhc3luYyAodG9vbE5hbWUsIHRvb2xJbnB1dCwgZXhlY3V0ZSkgPT4ge1xuICAgIGNvbnN0IG5lZWRzQXBwcm92YWwgPSBhd2FpdCBjaGVja05lZWRzQXBwcm92YWwoe1xuICAgICAgYWdlbnQsXG4gICAgICB0b29sTmFtZSxcbiAgICAgIGlucHV0OiB0b29sSW5wdXQsXG4gICAgICB0b29sQ2FsbElkOiBganNfJHt0b29sTmFtZX1fJHtEYXRlLm5vdygpfWAsXG4gICAgICBtZXNzYWdlczogW10sXG4gICAgfSk7XG5cbiAgICBpZiAoIW5lZWRzQXBwcm92YWwpIHtcbiAgICAgIHRyeSB7XG4gICAgICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IGV4ZWN1dGUoKTtcbiAgICAgICAgcmV0dXJuIHsgcmVzdWx0IH07XG4gICAgICB9IGNhdGNoIChlcnIpIHtcbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICBlcnJvcjogZXJyIGluc3RhbmNlb2YgRXJyb3IgPyBlcnIubWVzc2FnZSA6IFN0cmluZyhlcnIpLFxuICAgICAgICB9O1xuICAgICAgfVxuICAgIH1cblxuICAgIGNvbnN0IGFwcHJvdmFsSWQgPSBganNfYXBwcm92YWxfJHt1bGlkKCl9YDtcbiAgICBjb25zdCB0b29sQ2FsbElkID0gYGpzX3RjXyR7dWxpZCgpfWA7XG4gICAgY29uc3QgcGFydElkID0gYXBwcm92YWxQYXJ0SWQoYXBwcm92YWxJZCk7XG5cbiAgICBhd2FpdCBhZ2VudC5zdG9yYWdlLnBhcnQuc2V0KHBhcnRJZCwge1xuICAgICAgaWQ6IHBhcnRJZCxcbiAgICAgIGluZGV4OiBuZXh0UGFydEluZGV4UmVmLmN1cnJlbnQrKyxcbiAgICAgIG1lc3NhZ2VJZDogYXNzaXN0YW50TWVzc2FnZUlkLFxuICAgICAgc2Vzc2lvbklkLFxuICAgICAgcGFydDoge1xuICAgICAgICB0eXBlOiBgdG9vbC0ke3Rvb2xOYW1lfWAsXG4gICAgICAgIHRvb2xDYWxsSWQsXG4gICAgICAgIHN0YXRlOiBcImFwcHJvdmFsLXJlcXVlc3RlZFwiLFxuICAgICAgICBpbnB1dDogdG9vbElucHV0LFxuICAgICAgICBhcHByb3ZhbDogeyBpZDogYXBwcm92YWxJZCB9LFxuICAgICAgfSBhcyB1bmtub3duIGFzIFVJTWVzc2FnZVtcInBhcnRzXCJdW251bWJlcl0sXG4gICAgfSk7XG5cbiAgICB3cml0ZXIud3JpdGUoe1xuICAgICAgdHlwZTogXCJ0b29sLWlucHV0LXN0YXJ0XCIsXG4gICAgICB0b29sQ2FsbElkLFxuICAgICAgdG9vbE5hbWUsXG4gICAgfSk7XG4gICAgd3JpdGVyLndyaXRlKHtcbiAgICAgIHR5cGU6IFwidG9vbC1pbnB1dC1hdmFpbGFibGVcIixcbiAgICAgIHRvb2xDYWxsSWQsXG4gICAgICB0b29sTmFtZSxcbiAgICAgIGlucHV0OiB0b29sSW5wdXQsXG4gICAgfSk7XG4gICAgd3JpdGVyLndyaXRlKHtcbiAgICAgIHR5cGU6IFwidG9vbC1hcHByb3ZhbC1yZXF1ZXN0XCIsXG4gICAgICBhcHByb3ZhbElkLFxuICAgICAgdG9vbENhbGxJZCxcbiAgICB9KTtcblxuICAgIGNvbnN0IFBPTExfTVMgPSA1MDA7XG4gICAgY29uc3QgVElNRU9VVF9NUyA9IDUgKiA2MCAqIDEwMDA7XG4gICAgY29uc3Qgc3RhcnQgPSBEYXRlLm5vdygpO1xuXG4gICAgd2hpbGUgKERhdGUubm93KCkgLSBzdGFydCA8IFRJTUVPVVRfTVMgJiYgIWFib3J0Q29udHJvbGxlci5zaWduYWwuYWJvcnRlZCkge1xuICAgICAgY29uc3QgdXBkYXRlZCA9IGF3YWl0IGFnZW50LnN0b3JhZ2UucGFydC5nZXQocGFydElkKTtcbiAgICAgIGlmIChcbiAgICAgICAgdXBkYXRlZCAmJlxuICAgICAgICBcInN0YXRlXCIgaW4gdXBkYXRlZC5wYXJ0ICYmXG4gICAgICAgICgodXBkYXRlZC5wYXJ0IGFzIHsgc3RhdGU6IHN0cmluZyB9KS5zdGF0ZSA9PT0gXCJhcHByb3ZhbC1yZXNwb25kZWRcIiB8fFxuICAgICAgICAgICh1cGRhdGVkLnBhcnQgYXMgeyBzdGF0ZTogc3RyaW5nIH0pLnN0YXRlID09PSBcIm91dHB1dC1kZW5pZWRcIilcbiAgICAgICkge1xuICAgICAgICBjb25zdCBzdGF0ZSA9ICh1cGRhdGVkLnBhcnQgYXMgeyBzdGF0ZTogc3RyaW5nIH0pLnN0YXRlO1xuXG4gICAgICAgIGlmIChzdGF0ZSA9PT0gXCJvdXRwdXQtZGVuaWVkXCIpIHtcbiAgICAgICAgICBjb25zdCByZWFzb24gPSAodXBkYXRlZC5wYXJ0IGFzIHsgYXBwcm92YWw/OiB7IHJlYXNvbj86IHN0cmluZyB9IH0pXG4gICAgICAgICAgICAuYXBwcm92YWw/LnJlYXNvbjtcbiAgICAgICAgICByZXR1cm4ge1xuICAgICAgICAgICAgZXJyb3I6IGBUb29sIFwiJHt0b29sTmFtZX1cIiBkZW5pZWQ6ICR7cmVhc29uIHx8IFwidXNlciBkZW5pZWRcIn1gLFxuICAgICAgICAgIH07XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBhcHByb3ZhbCA9IChcbiAgICAgICAgICB1cGRhdGVkLnBhcnQgYXMge1xuICAgICAgICAgICAgYXBwcm92YWw/OiB7IGFwcHJvdmVkOiBib29sZWFuOyByZWFzb24/OiBzdHJpbmcgfTtcbiAgICAgICAgICB9XG4gICAgICAgICkuYXBwcm92YWw7XG4gICAgICAgIGlmIChhcHByb3ZhbD8uYXBwcm92ZWQpIHtcbiAgICAgICAgICB0cnkge1xuICAgICAgICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgZXhlY3V0ZSgpO1xuICAgICAgICAgICAgd3JpdGVyLndyaXRlKHtcbiAgICAgICAgICAgICAgdHlwZTogXCJ0b29sLW91dHB1dC1hdmFpbGFibGVcIixcbiAgICAgICAgICAgICAgdG9vbENhbGxJZCxcbiAgICAgICAgICAgICAgb3V0cHV0OiByZXN1bHQsXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIHJldHVybiB7IHJlc3VsdCB9O1xuICAgICAgICAgIH0gY2F0Y2ggKGVycikge1xuICAgICAgICAgICAgY29uc3QgZXJyb3IgPSBlcnIgaW5zdGFuY2VvZiBFcnJvciA/IGVyci5tZXNzYWdlIDogU3RyaW5nKGVycik7XG4gICAgICAgICAgICB3cml0ZXIud3JpdGUoe1xuICAgICAgICAgICAgICB0eXBlOiBcInRvb2wtb3V0cHV0LWVycm9yXCIsXG4gICAgICAgICAgICAgIHRvb2xDYWxsSWQsXG4gICAgICAgICAgICAgIGVycm9yVGV4dDogZXJyb3IsXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIHJldHVybiB7IGVycm9yIH07XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIHdyaXRlci53cml0ZSh7XG4gICAgICAgICAgdHlwZTogXCJ0b29sLW91dHB1dC1kZW5pZWRcIixcbiAgICAgICAgICB0b29sQ2FsbElkLFxuICAgICAgICB9KTtcbiAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICBlcnJvcjogYFRvb2wgXCIke3Rvb2xOYW1lfVwiIGRlbmllZDogJHthcHByb3ZhbD8ucmVhc29uIHx8IFwidXNlciBkZW5pZWRcIn1gLFxuICAgICAgICB9O1xuICAgICAgfVxuICAgICAgYXdhaXQgbmV3IFByb21pc2UoKHIpID0+IHNldFRpbWVvdXQociwgUE9MTF9NUykpO1xuICAgIH1cblxuICAgIHJldHVybiB7XG4gICAgICBlcnJvcjogYWJvcnRDb250cm9sbGVyLnNpZ25hbC5hYm9ydGVkXG4gICAgICAgID8gXCJpbnRlcnJ1cHRlZFwiXG4gICAgICAgIDogXCJBcHByb3ZhbCB0aW1lZCBvdXRcIixcbiAgICB9O1xuICB9O1xuXG4gIGNvbnN0IHRvb2xDb250ZXh0OiBUb29sQ29udGV4dCA9IHtcbiAgICBjb250ZXh0LFxuICAgIHN0YXRlLFxuICAgIHNlc3Npb25JZCxcbiAgICBzYW5kYm94LFxuICAgIG1lc3NhZ2VzOiBtZXNzYWdlcyBhcyBUb29sQ29udGV4dFtcIm1lc3NhZ2VzXCJdLFxuICB9O1xuXG4gIHJhd1Rvb2xzW2J1aWx0aW5Ub29sTmFtZXMuSmF2YVNjcmlwdF0gPSBjcmVhdGVKYXZhU2NyaXB0VG9vbCh7XG4gICAgdG9vbHM6IHJhd1Rvb2xzLFxuICAgIHNhbmRib3gsXG4gICAgYWN0aXZlVG9vbHMsXG4gICAgb25TdWJUb29sQ2FsbCxcbiAgICB0b29sQ29udGV4dCxcbiAgfSk7XG5cbiAgY29uc3QgdG9vbHNXaXRoTmVlZHNBcHByb3ZhbCA9IE9iamVjdC5mcm9tRW50cmllcyhcbiAgICBPYmplY3QuZW50cmllcyhyYXdUb29scykubWFwKChbbmFtZSwgdF0pID0+IHtcbiAgICAgIGNvbnN0IG9yaWdpbmFsRXhlY3V0ZSA9IHQuZXhlY3V0ZTtcbiAgICAgIGNvbnN0IHdyYXBwZWRFeGVjdXRlID0gb3JpZ2luYWxFeGVjdXRlXG4gICAgICAgID8gYXN5bmMgKC4uLmFyZ3M6IFBhcmFtZXRlcnM8dHlwZW9mIG9yaWdpbmFsRXhlY3V0ZT4pID0+IHtcbiAgICAgICAgICAgIGNvbnN0IGRvbmUgPSBzdGVwTG9nLnRpbWUoYHRvb2w6JHtuYW1lfWApO1xuICAgICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgICAgbGV0IHRvb2xJbnB1dCA9IGFyZ3NbMF07XG4gICAgICAgICAgICAgIGlmIChhZ2VudC5ob29rc1tcInRvb2wuYmVmb3JlXCJdKSB7XG4gICAgICAgICAgICAgICAgY29uc3QgYmVmb3JlUmVzdWx0ID0gYXdhaXQgYWdlbnQuaG9va3NbXCJ0b29sLmJlZm9yZVwiXSh7XG4gICAgICAgICAgICAgICAgICBuYW1lLFxuICAgICAgICAgICAgICAgICAgaW5wdXQ6IHRvb2xJbnB1dCxcbiAgICAgICAgICAgICAgICAgIGNvbnRleHQ6IHRvb2xDb250ZXh0LFxuICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgICAgIGlmIChiZWZvcmVSZXN1bHQ/LmlucHV0ICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICAgIHRvb2xJbnB1dCA9IGJlZm9yZVJlc3VsdC5pbnB1dDtcbiAgICAgICAgICAgICAgICAgIGFyZ3NbMF0gPSB0b29sSW5wdXQgYXMgKHR5cGVvZiBhcmdzKVswXTtcbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH1cblxuICAgICAgICAgICAgICBsZXQgcmVzdWx0ID0gYXdhaXQgb3JpZ2luYWxFeGVjdXRlKC4uLmFyZ3MpO1xuXG4gICAgICAgICAgICAgIGlmIChhZ2VudC5ob29rc1tcInRvb2wuYWZ0ZXJcIl0pIHtcbiAgICAgICAgICAgICAgICBjb25zdCBhZnRlclJlc3VsdCA9IGF3YWl0IGFnZW50Lmhvb2tzW1widG9vbC5hZnRlclwiXSh7XG4gICAgICAgICAgICAgICAgICBuYW1lLFxuICAgICAgICAgICAgICAgICAgaW5wdXQ6IHRvb2xJbnB1dCxcbiAgICAgICAgICAgICAgICAgIHJlc3VsdCxcbiAgICAgICAgICAgICAgICAgIGNvbnRleHQ6IHRvb2xDb250ZXh0LFxuICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgICAgIGlmIChhZnRlclJlc3VsdD8ucmVzdWx0ICE9PSB1bmRlZmluZWQpIHtcbiAgICAgICAgICAgICAgICAgIHJlc3VsdCA9IGFmdGVyUmVzdWx0LnJlc3VsdCBhcyB0eXBlb2YgcmVzdWx0O1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAgIGRvbmUoKTtcbiAgICAgICAgICAgICAgcmV0dXJuIHJlc3VsdDtcbiAgICAgICAgICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgICAgICAgZG9uZSh7XG4gICAgICAgICAgICAgICAgZXJyb3I6IGUgaW5zdGFuY2VvZiBFcnJvciA/IGUubWVzc2FnZSA6IFN0cmluZyhlKSxcbiAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICAgIHRocm93IGU7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICA6IHVuZGVmaW5lZDtcbiAgICAgIHJldHVybiBbXG4gICAgICAgIG5hbWUsXG4gICAgICAgIHtcbiAgICAgICAgICAuLi50LFxuICAgICAgICAgIC4uLih3cmFwcGVkRXhlY3V0ZSA/IHsgZXhlY3V0ZTogd3JhcHBlZEV4ZWN1dGUgfSA6IHt9KSxcbiAgICAgICAgICBuZWVkc0FwcHJvdmFsOiBhc3luYyAoXG4gICAgICAgICAgICB0b29sSW5wdXQ6IHVua25vd24sXG4gICAgICAgICAgICBvcHRzOiB7IHRvb2xDYWxsSWQ6IHN0cmluZzsgbWVzc2FnZXM6IHVua25vd25bXSB9XG4gICAgICAgICAgKSA9PiB7XG4gICAgICAgICAgICByZXR1cm4gYXdhaXQgY2hlY2tOZWVkc0FwcHJvdmFsKHtcbiAgICAgICAgICAgICAgYWdlbnQsXG4gICAgICAgICAgICAgIHRvb2xOYW1lOiBuYW1lLFxuICAgICAgICAgICAgICBpbnB1dDogdG9vbElucHV0LFxuICAgICAgICAgICAgICB0b29sQ2FsbElkOiBvcHRzLnRvb2xDYWxsSWQsXG4gICAgICAgICAgICAgIG1lc3NhZ2VzOiBvcHRzLm1lc3NhZ2VzLFxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgfSxcbiAgICAgICAgfSxcbiAgICAgIF07XG4gICAgfSlcbiAgKTtcblxuICByZXR1cm4geyByYXdUb29scywgdG9vbHNXaXRoTmVlZHNBcHByb3ZhbCwgdG9vbENvbnRleHQgfTtcbn1cbiIsICJpbXBvcnQgdHlwZSB7IEZpbmlzaFJlYXNvbiwgVG9vbFNldCwgVUlNZXNzYWdlLCBVSU1lc3NhZ2VDaHVuayB9IGZyb20gXCJhaVwiO1xuaW1wb3J0IHsgYWxsIH0gZnJvbSBcImJldHRlci1hbGxcIjtcbmltcG9ydCAqIGFzIGVycm9yZSBmcm9tIFwiZXJyb3JlXCI7XG5pbXBvcnQgeyB1bGlkIH0gZnJvbSBcInVsaWRcIjtcbmltcG9ydCB0eXBlIHsgVHlwZWRVSU1lc3NhZ2UgfSBmcm9tIFwiLi4vYWdlbnQvcmVnaXN0cnlcIjtcbmltcG9ydCB0eXBlIHsgU2tpbGxTdW1tYXJ5IH0gZnJvbSBcIi4uL3NraWxscy90eXBlc1wiO1xuaW1wb3J0IHR5cGUgeyBNZXNzYWdlLCBQYXJ0IH0gZnJvbSBcIi4uL3N0b3JhZ2UvdHlwZXNcIjtcbmltcG9ydCB7IGNyZWF0ZUxvZ2dlciB9IGZyb20gXCIuLi91dGlscy9sb2dnZXJcIjtcbmltcG9ydCB0eXBlIHsgU3RlcFVzYWdlIH0gZnJvbSBcIi4uL3V0aWxzL3VzYWdlXCI7XG5pbXBvcnQge1xuICB0eXBlIFN0cmVhbVRleHRSZXN1bHQsXG4gIHN0cmVhbVRleHQsXG4gIHN0cmVhbVRleHRTdGVwLFxufSBmcm9tIFwiLi9haS1sb29wL3N0cmVhbS10ZXh0XCI7XG5pbXBvcnQgdHlwZSB7IFNlc3Npb25Db250ZXh0IH0gZnJvbSBcIi4vaGFuZGxlXCI7XG5pbXBvcnQgeyBpbnRlcnJ1cHQgfSBmcm9tIFwiLi9pbnRlcnJ1cHRcIjtcblxuY29uc3QgREVGQVVMVF9TVEVQX1RIUk9VR0hQVVQgPSAxMDA7XG5cbmV4cG9ydCB0eXBlIEFwcHJvdmFsRGF0YSA9IHtcbiAgYXBwcm92YWxJZDogc3RyaW5nO1xuICBhcHByb3ZlZDogYm9vbGVhbjtcbiAgcmVhc29uPzogc3RyaW5nO1xufTtcblxudHlwZSBNZXNzYWdlRGF0YTxUb29scyBleHRlbmRzIFRvb2xTZXQsIFRNZXNzYWdlTWV0YWRhdGEgPSB1bmtub3duPiA9XG4gIHwgVUlNZXNzYWdlPFRNZXNzYWdlTWV0YWRhdGE+XG4gIHwge1xuICAgICAgcm9sZT86IFR5cGVkVUlNZXNzYWdlPHVua25vd24sIFRvb2xzPltcInJvbGVcIl07XG4gICAgICBwYXJ0czogVHlwZWRVSU1lc3NhZ2U8dW5rbm93biwgVG9vbHM+W1wicGFydHNcIl07XG4gICAgICBtZXRhZGF0YT86IFRNZXNzYWdlTWV0YWRhdGEgfCBudWxsO1xuICAgICAgaWQ/OiBzdHJpbmc7XG4gICAgfTtcblxuZXhwb3J0IHR5cGUgU2VuZElucHV0PFRvb2xzIGV4dGVuZHMgVG9vbFNldCwgVE1lc3NhZ2VNZXRhZGF0YSA9IHVua25vd24+ID1cbiAgfCBzdHJpbmdcbiAgfCBNZXNzYWdlRGF0YTxUb29scywgVE1lc3NhZ2VNZXRhZGF0YT5cbiAgfCB7IHR5cGU6IFwibWVzc2FnZVwiOyBtZXNzYWdlOiBNZXNzYWdlRGF0YTxUb29scywgVE1lc3NhZ2VNZXRhZGF0YT4gfVxuICB8IHsgdHlwZTogXCJhcHByb3ZhbFwiOyBhcHByb3ZhbDogQXBwcm92YWxEYXRhIH07XG5cbmV4cG9ydCB0eXBlIFNlbmRPcHRpb25zPFRDb250ZXh0LCBUU3RhdGUgPSB1bmtub3duPiA9IHtcbiAgaW50ZXJydXB0SWZTdHJlYW1pbmc/OlxuICAgIHwgYm9vbGVhblxuICAgIHwgeyBsYXN0UGFydDogeyBpbmRleDogbnVtYmVyOyBwYXJ0OiB1bmtub3duIH0gfTtcbiAgY29udGV4dD86IFRDb250ZXh0O1xuICBpbml0aWFsU3RhdGU/OiBUU3RhdGU7XG4gIGFib3J0U2lnbmFsPzogQWJvcnRTaWduYWw7XG4gIHNhbmRib3hJZD86IHN0cmluZztcbn07XG5cbmV4cG9ydCB0eXBlIFNlbmRSZXN1bHQgPSB7XG4gIGFzc2lzdGFudE1lc3NhZ2VJZDogc3RyaW5nO1xuICBkb25lOiB0cnVlIHwgUHJvbWlzZTx0cnVlPjtcbn07XG5cbmV4cG9ydCB0eXBlIEFjdGl2ZVJ1biA9IHtcbiAgY2h1bmtzOiBVSU1lc3NhZ2VDaHVua1tdO1xuICB3cml0YWJsZTogV3JpdGFibGVTdHJlYW08VUlNZXNzYWdlQ2h1bms+O1xuICBkb25lOiBQcm9taXNlPHZvaWQ+O1xuICAvKiogUmVzb2x2ZXMgd2hlbiBhIG5ldyBjaHVuayBpcyBhdmFpbGFibGUuIFJlcGxhY2VkIGFmdGVyIGVhY2ggbm90aWZpY2F0aW9uLiAqL1xuICBvbkNodW5rOiBQcm9taXNlPHZvaWQ+O1xufTtcblxuY29uc3QgQUNUSVZFX1JVTlNfS0VZID0gU3ltYm9sLmZvcihcImV4cGVyaW1lbnRhbC1hZ2VudDphY3RpdmVSdW5zXCIpO1xuXG5leHBvcnQgZnVuY3Rpb24gZ2V0QWN0aXZlUnVucygpOiBNYXA8c3RyaW5nLCBBY3RpdmVSdW4+IHtcbiAgY29uc3QgZyA9IGdsb2JhbFRoaXMgYXMgUmVjb3JkPHN5bWJvbCwgTWFwPHN0cmluZywgQWN0aXZlUnVuPiB8IHVuZGVmaW5lZD47XG4gIGlmICghZ1tBQ1RJVkVfUlVOU19LRVldKSB7XG4gICAgZ1tBQ1RJVkVfUlVOU19LRVldID0gbmV3IE1hcCgpO1xuICB9XG4gIHJldHVybiBnW0FDVElWRV9SVU5TX0tFWV07XG59XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBzZW5kPFxuICBUb29scyBleHRlbmRzIFRvb2xTZXQsXG4gIFRDb250ZXh0LFxuICBUTWVzc2FnZU1ldGFkYXRhID0gdW5rbm93bixcbiAgVFN0YXRlID0gdW5rbm93bixcbj4oXG4gIHNlc3Npb25Db250ZXh0OiBTZXNzaW9uQ29udGV4dCxcbiAgaW5wdXQ6XG4gICAgfCBTZW5kSW5wdXQ8VG9vbHMsIFRNZXNzYWdlTWV0YWRhdGE+XG4gICAgfCBTZW5kSW5wdXQ8VG9vbHMsIFRNZXNzYWdlTWV0YWRhdGE+W10sXG4gIG9wdHM/OiBTZW5kT3B0aW9uczxUQ29udGV4dCwgVFN0YXRlPlxuKTogUHJvbWlzZTxTZW5kUmVzdWx0PiB7XG4gIGlmIChvcHRzPy5pbnRlcnJ1cHRJZlN0cmVhbWluZykge1xuICAgIGNvbnN0IGxhc3RQYXJ0ID1cbiAgICAgIHR5cGVvZiBvcHRzLmludGVycnVwdElmU3RyZWFtaW5nID09PSBcIm9iamVjdFwiXG4gICAgICAgID8gb3B0cy5pbnRlcnJ1cHRJZlN0cmVhbWluZy5sYXN0UGFydFxuICAgICAgICA6IHVuZGVmaW5lZDtcbiAgICBhd2FpdCBpbnRlcnJ1cHQoc2Vzc2lvbkNvbnRleHQsIHsgbGFzdFBhcnQgfSk7XG4gIH1cblxuICBjb25zdCB7XG4gICAgd29ya2Zsb3dSdW5JZCxcbiAgICB3b3JrZmxvd1dyaXRhYmxlLFxuICAgIGV4aXN0aW5nUGFydENvdW50LFxuICAgIGFzc2lzdGFudE1lc3NhZ2VJZCxcbiAgICBhc3Npc3RhbnRNZXNzYWdlQ3JlYXRlZEF0LFxuICAgIHN0YXRlLFxuICB9ID0gYXdhaXQgZXJyb3JlLnRyeUFzeW5jKHtcbiAgICB0cnk6ICgpID0+XG4gICAgICBpbml0U2Vzc2lvblN0ZXA8VG9vbHM+KHtcbiAgICAgICAgc2Vzc2lvbkNvbnRleHQsXG4gICAgICAgIHNlc3Npb25JZDogc2Vzc2lvbkNvbnRleHQuc2Vzc2lvbklkLFxuICAgICAgICBzYW5kYm94SWQ6IG9wdHM/LnNhbmRib3hJZCxcbiAgICAgICAgaW5pdGlhbFN0YXRlOiBvcHRzPy5pbml0aWFsU3RhdGUgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4gfCB1bmRlZmluZWQsXG4gICAgICAgIGlucHV0LFxuICAgICAgfSksXG4gICAgY2F0Y2g6IChlKSA9PiB7XG4gICAgICBpZiAoZSBpbnN0YW5jZW9mIEVycm9yKSB7XG4gICAgICAgIHRocm93IGU7XG4gICAgICB9XG4gICAgICB0aHJvdyBuZXcgU2VuZEVycm9yKHtcbiAgICAgICAgaWQ6IHNlc3Npb25Db250ZXh0LnNlc3Npb25JZCxcbiAgICAgICAgcmVhc29uOiBTdHJpbmcoZSksXG4gICAgICAgIGNhdXNlOiBlLFxuICAgICAgfSk7XG4gICAgfSxcbiAgfSk7XG5cbiAgY29uc3QgbG9vcE9wdHMgPSB7XG4gICAgc2Vzc2lvbkNvbnRleHQsXG4gICAgYXNzaXN0YW50TWVzc2FnZUlkLFxuICAgIGNyZWF0ZWRBdDogYXNzaXN0YW50TWVzc2FnZUNyZWF0ZWRBdCxcbiAgICBjb250ZXh0OiAob3B0cz8uY29udGV4dCA/PyB7fSkgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4sXG4gICAgc3RhdGUsXG4gICAgbGFzdFBhcnRJbmRleDogZXhpc3RpbmdQYXJ0Q291bnQsXG4gICAgd29ya2Zsb3dSdW5JZCxcbiAgfTtcblxuICBpZiAod29ya2Zsb3dXcml0YWJsZSkge1xuICAgIGF3YWl0IHJ1bkxvb3A8VG9vbHM+KHsgLi4ubG9vcE9wdHMsIHdyaXRhYmxlOiB3b3JrZmxvd1dyaXRhYmxlIH0pO1xuICAgIHJldHVybiB7IGFzc2lzdGFudE1lc3NhZ2VJZCwgZG9uZTogdHJ1ZSB9O1xuICB9XG5cbiAgY29uc3QgY2h1bmtzOiBVSU1lc3NhZ2VDaHVua1tdID0gW107XG4gIGxldCBub3RpZnlDaHVuazogKCkgPT4gdm9pZDtcbiAgbGV0IG9uQ2h1bmsgPSBuZXcgUHJvbWlzZTx2b2lkPigocikgPT4ge1xuICAgIG5vdGlmeUNodW5rID0gcjtcbiAgfSk7XG5cbiAgY29uc3Qgd3JpdGFibGUgPSBuZXcgV3JpdGFibGVTdHJlYW08VUlNZXNzYWdlQ2h1bms+KHtcbiAgICB3cml0ZShjaHVuaykge1xuICAgICAgY2h1bmtzLnB1c2goY2h1bmspO1xuICAgICAgY29uc3QgcHJldiA9IG5vdGlmeUNodW5rO1xuICAgICAgb25DaHVuayA9IG5ldyBQcm9taXNlPHZvaWQ+KChyKSA9PiB7XG4gICAgICAgIG5vdGlmeUNodW5rID0gcjtcbiAgICAgIH0pO1xuICAgICAgcHJldigpO1xuICAgIH0sXG4gIH0pO1xuXG4gIGNvbnN0IGRvbmUgPSBydW5Mb29wPFRvb2xzPih7IC4uLmxvb3BPcHRzLCB3cml0YWJsZSB9KTtcblxuICBjb25zdCBhY3RpdmVSdW46IEFjdGl2ZVJ1biA9IHtcbiAgICBjaHVua3MsXG4gICAgd3JpdGFibGUsXG4gICAgZG9uZSxcbiAgICBnZXQgb25DaHVuaygpIHtcbiAgICAgIHJldHVybiBvbkNodW5rO1xuICAgIH0sXG4gIH07XG4gIGNvbnN0IGFjdGl2ZVJ1bnMgPSBnZXRBY3RpdmVSdW5zKCk7XG4gIGFjdGl2ZVJ1bnMuc2V0KGFzc2lzdGFudE1lc3NhZ2VJZCwgYWN0aXZlUnVuKTtcbiAgZG9uZS5maW5hbGx5KCgpID0+IGFjdGl2ZVJ1bnMuZGVsZXRlKGFzc2lzdGFudE1lc3NhZ2VJZCkpO1xuXG4gIHJldHVybiB7IGFzc2lzdGFudE1lc3NhZ2VJZCwgZG9uZTogZG9uZS50aGVuKCgpID0+IHRydWUpIH07XG59XG5cbi8qIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cbiAqIEFJIExvb3BcbiAqIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi9cblxuYXN5bmMgZnVuY3Rpb24gcnVuTG9vcDxUb29scyBleHRlbmRzIFRvb2xTZXQ+KHtcbiAgd29ya2Zsb3dSdW5JZCxcbiAgc2Vzc2lvbkNvbnRleHQsXG4gIGFzc2lzdGFudE1lc3NhZ2VJZCxcbiAgY3JlYXRlZEF0LFxuICBjb250ZXh0OiBpbml0aWFsQ29udGV4dCxcbiAgc3RhdGUsXG4gIHdyaXRhYmxlLFxuICBsYXN0UGFydEluZGV4OiBpbml0aWFsUGFydEluZGV4LFxufToge1xuICB3b3JrZmxvd1J1bklkOiBzdHJpbmcgfCB1bmRlZmluZWQ7XG4gIHNlc3Npb25Db250ZXh0OiBTZXNzaW9uQ29udGV4dDtcbiAgYXNzaXN0YW50TWVzc2FnZUlkOiBzdHJpbmc7XG4gIGNyZWF0ZWRBdDogbnVtYmVyO1xuICBjb250ZXh0OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgc3RhdGU6IFJlY29yZDxzdHJpbmcsIHVua25vd24+O1xuICB3cml0YWJsZTogV3JpdGFibGVTdHJlYW07XG4gIGxhc3RQYXJ0SW5kZXg6IG51bWJlcjtcbn0pOiBQcm9taXNlPHZvaWQ+IHtcbiAgY29uc3QgYWdlbnQgPSBzZXNzaW9uQ29udGV4dC5hZ2VudDtcblxuICBjb25zdCBsb2cgPSBjcmVhdGVMb2dnZXIoe1xuICAgIGNvbmZpZzogeyAuLi5hZ2VudC5vcHRpb25zLmxvZ2dpbmcsIG5hbWU6IGFnZW50Lm5hbWUgfSxcbiAgICBzdWJzeXN0ZW06IFwiYWktbG9vcFwiLFxuICB9KTtcbiAgY29uc3QgbXNnTG9nID0gbG9nLndpdGhDb250ZXh0KHtcbiAgICBzZXNzaW9uSWQ6IHNlc3Npb25Db250ZXh0LnNlc3Npb25JZCxcbiAgICBtZXNzYWdlSWQ6IGFzc2lzdGFudE1lc3NhZ2VJZCxcbiAgfSk7XG4gIGNvbnN0IGRvbmVNZXNzYWdlID0gbXNnTG9nLnRpbWUoXCJvbk1lc3NhZ2VcIik7XG5cbiAgbGV0IGZpbmlzaFJlYXNvbjogRmluaXNoUmVhc29uIHwgdW5kZWZpbmVkO1xuICBsZXQgbGFzdFBhcnRJbmRleCA9IGluaXRpYWxQYXJ0SW5kZXg7XG4gIGNvbnN0IHVzYWdlU3RlcHM6IFN0ZXBVc2FnZVtdID0gW107XG4gIGxldCBkaXNjb3ZlcmVkU2tpbGxzOiBTa2lsbFN1bW1hcnlbXSB8IG51bGwgPSBudWxsO1xuICBsZXQgY29udGV4dCA9IGluaXRpYWxDb250ZXh0O1xuXG4gIHRyeSB7XG4gICAgd2hpbGUgKGZpbmlzaFJlYXNvbiAhPT0gXCJzdG9wXCIpIHtcbiAgICAgIGNvbnN0IHN0cmVhbVRleHRGbiA9IHdvcmtmbG93UnVuSWRcbiAgICAgICAgPyBzdHJlYW1UZXh0U3RlcDxUb29scz5cbiAgICAgICAgOiBzdHJlYW1UZXh0PFRvb2xzPjtcbiAgICAgIGNvbnN0IHRocm91Z2hwdXQgPSB3b3JrZmxvd1J1bklkID8gNSA6IERFRkFVTFRfU1RFUF9USFJPVUdIUFVUO1xuXG4gICAgICBjb25zdCByZXN1bHQ6IFN0cmVhbVRleHRSZXN1bHQgPSBhd2FpdCBzdHJlYW1UZXh0Rm4oe1xuICAgICAgICBzdGVwVGhyb3VnaHB1dDogdGhyb3VnaHB1dCxcbiAgICAgICAgYXNzaXN0YW50TWVzc2FnZUlkLFxuICAgICAgICBhZ2VudCxcbiAgICAgICAgc2Vzc2lvbklkOiBzZXNzaW9uQ29udGV4dC5zZXNzaW9uSWQsXG4gICAgICAgIGNvbnRleHQsXG4gICAgICAgIHN0YXRlLFxuICAgICAgICBjcmVhdGVkQXQsXG4gICAgICAgIHdyaXRhYmxlLFxuICAgICAgICBsYXN0UGFydEluZGV4LFxuICAgICAgICBzdGVwSW5kZXhPZmZzZXQ6IHVzYWdlU3RlcHMubGVuZ3RoLFxuICAgICAgICBkaXNjb3ZlcmVkU2tpbGxzLFxuICAgICAgfSk7XG5cbiAgICAgIGZpbmlzaFJlYXNvbiA9IHJlc3VsdC5maW5pc2hSZWFzb247XG4gICAgICBsYXN0UGFydEluZGV4ID0gcmVzdWx0Lmxhc3RQYXJ0SW5kZXg7XG4gICAgICB1c2FnZVN0ZXBzLnB1c2goLi4ucmVzdWx0LnVzYWdlU3RlcHMpO1xuICAgICAgZGlzY292ZXJlZFNraWxscyA9IHJlc3VsdC5kaXNjb3ZlcmVkU2tpbGxzO1xuICAgICAgY29udGV4dCA9IHJlc3VsdC5jb250ZXh0O1xuXG4gICAgICBpZiAocmVzdWx0Lm1heFN0ZXBzICE9IG51bGwgJiYgdXNhZ2VTdGVwcy5sZW5ndGggPj0gcmVzdWx0Lm1heFN0ZXBzKSB7XG4gICAgICAgIG1zZ0xvZy5pbmZvKFwicmVhY2hlZCBtYXhTdGVwcyBsaW1pdFwiLCB7IG1heFN0ZXBzOiByZXN1bHQubWF4U3RlcHMgfSk7XG4gICAgICAgIGJyZWFrO1xuICAgICAgfVxuXG4gICAgICBpZiAocmVzdWx0LnBlbmRpbmdBcHByb3ZhbHMubGVuZ3RoID4gMCkge1xuICAgICAgICBtc2dMb2cuaW5mbyhcInBlbmRpbmcgYXBwcm92YWxzLCBzdG9wcGluZyBsb29wXCIsIHtcbiAgICAgICAgICBjb3VudDogcmVzdWx0LnBlbmRpbmdBcHByb3ZhbHMubGVuZ3RoLFxuICAgICAgICAgIHRvb2xzOiByZXN1bHQucGVuZGluZ0FwcHJvdmFscy5tYXAoKHApID0+IHAudG9vbE5hbWUpLFxuICAgICAgICB9KTtcbiAgICAgICAgYnJlYWs7XG4gICAgICB9XG4gICAgfVxuXG4gICAgZG9uZU1lc3NhZ2UoeyB0b3RhbFN0ZXBzOiB1c2FnZVN0ZXBzLmxlbmd0aCB9KTtcbiAgfSBjYXRjaCAoZXJyKSB7XG4gICAgbXNnTG9nLmVycm9yKFwiQUkgbG9vcCBmYWlsZWRcIiwge1xuICAgICAgZXJyb3I6IGVyciBpbnN0YW5jZW9mIEVycm9yID8gZXJyLm1lc3NhZ2UgOiBTdHJpbmcoZXJyKSxcbiAgICB9KTtcbiAgICB0aHJvdyBlcnI7XG4gIH0gZmluYWxseSB7XG4gICAgYXdhaXQgY29tcGxldGVNZXNzYWdlU3RlcCh7XG4gICAgICBhc3Npc3RhbnRNZXNzYWdlSWQsXG4gICAgICBzZXNzaW9uQ29udGV4dCxcbiAgICAgIHdyaXRhYmxlLFxuICAgICAgdXNhZ2VTdGVwcyxcbiAgICB9KTtcbiAgfVxufVxuXG4vKiAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG4gKiBTdGVwc1xuICogLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qL1xuXG5hc3luYyBmdW5jdGlvbiBpbml0U2Vzc2lvblN0ZXA8VG9vbHMgZXh0ZW5kcyBUb29sU2V0Pih7XG4gIHNlc3Npb25Db250ZXh0LFxuICBzZXNzaW9uSWQsXG4gIHNhbmRib3hJZCxcbiAgaW5pdGlhbFN0YXRlLFxuICBpbnB1dCxcbn06IHtcbiAgc2Vzc2lvbkNvbnRleHQ6IFNlc3Npb25Db250ZXh0O1xuICBzZXNzaW9uSWQ6IHN0cmluZztcbiAgc2FuZGJveElkPzogc3RyaW5nO1xuICBpbml0aWFsU3RhdGU/OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgaW5wdXQ6IFNlbmRJbnB1dDxUb29scz4gfCBTZW5kSW5wdXQ8VG9vbHM+W107XG59KSB7XG4gIFwidXNlIHN0ZXBcIjtcblxuICBsZXQgd29ya2Zsb3dSdW5JZDogc3RyaW5nIHwgdW5kZWZpbmVkO1xuICBsZXQgd29ya2Zsb3dXcml0YWJsZTogV3JpdGFibGVTdHJlYW0gfCB1bmRlZmluZWQ7XG4gIHRyeSB7XG4gICAgY29uc3QgeyBnZXRXb3JrZmxvd01ldGFkYXRhLCBnZXRXcml0YWJsZSB9ID0gYXdhaXQgaW1wb3J0KFwid29ya2Zsb3dcIik7XG4gICAgd29ya2Zsb3dSdW5JZCA9IGdldFdvcmtmbG93TWV0YWRhdGEoKS53b3JrZmxvd1J1bklkO1xuICAgIHdvcmtmbG93V3JpdGFibGUgPSBnZXRXcml0YWJsZSgpO1xuICB9IGNhdGNoIHtcbiAgICAvLyB3b3JrZmxvdyBub3QgaW5zdGFsbGVkXG4gIH1cblxuICBjb25zdCBuZXdNZXNzYWdlczogTWVzc2FnZVtdID0gW107XG4gIGNvbnN0IG5ld1BhcnRzOiBQYXJ0W10gPSBbXTtcbiAgY29uc3QgbmV3QXBwcm92YWxzOiBBcHByb3ZhbERhdGFbXSA9IFtdO1xuICBpZiAoQXJyYXkuaXNBcnJheShpbnB1dCkpIHtcbiAgICBmb3IgKGNvbnN0IGkgb2YgaW5wdXQpIHtcbiAgICAgIGNvbnN0IHsgbWVzc2FnZSwgcGFydHMsIGFwcHJvdmFscyB9ID0gdG9NZXNzYWdlQW5kUGFydHMoe1xuICAgICAgICBzZXNzaW9uSWQ6IHNlc3Npb25Db250ZXh0LnNlc3Npb25JZCxcbiAgICAgICAgaW5wdXQ6IGksXG4gICAgICAgIGRlZmF1bHRSb2xlOiBcInVzZXJcIixcbiAgICAgIH0pO1xuICAgICAgaWYgKG1lc3NhZ2UpIHtcbiAgICAgICAgbmV3TWVzc2FnZXMucHVzaChtZXNzYWdlKTtcbiAgICAgIH1cbiAgICAgIG5ld1BhcnRzLnB1c2goLi4ucGFydHMpO1xuICAgICAgbmV3QXBwcm92YWxzLnB1c2goLi4uYXBwcm92YWxzKTtcbiAgICB9XG4gIH0gZWxzZSB7XG4gICAgY29uc3QgeyBtZXNzYWdlLCBwYXJ0cywgYXBwcm92YWxzIH0gPSB0b01lc3NhZ2VBbmRQYXJ0cyh7XG4gICAgICBpbnB1dCxcbiAgICAgIHNlc3Npb25JZDogc2Vzc2lvbkNvbnRleHQuc2Vzc2lvbklkLFxuICAgICAgZGVmYXVsdFJvbGU6IFwidXNlclwiLFxuICAgIH0pO1xuICAgIGlmIChtZXNzYWdlKSB7XG4gICAgICBuZXdNZXNzYWdlcy5wdXNoKG1lc3NhZ2UpO1xuICAgIH1cbiAgICBuZXdQYXJ0cy5wdXNoKC4uLnBhcnRzKTtcbiAgICBuZXdBcHByb3ZhbHMucHVzaCguLi5hcHByb3ZhbHMpO1xuICB9XG5cbiAgY29uc3QgbGFzdE5ld01lc3NhZ2UgPSBuZXdNZXNzYWdlcy5hdCgtMSk7XG4gIGNvbnN0IHJldXNlQXNzaXN0YW50ID0gbGFzdE5ld01lc3NhZ2U/LnJvbGUgPT09IFwiYXNzaXN0YW50XCI7XG5cbiAgY29uc3QgYXNzaXN0YW50TWVzc2FnZUlkID0gcmV1c2VBc3Npc3RhbnRcbiAgICA/IGxhc3ROZXdNZXNzYWdlLmlkXG4gICAgOiBgbWVzc2FnZV8ke3VsaWQoKX1gO1xuICBjb25zdCBiYXNlVGltZSA9IERhdGUubm93KCk7XG4gIGNvbnN0IGFzc2lzdGFudE1lc3NhZ2VDcmVhdGVkQXQgPSBiYXNlVGltZSArIG5ld01lc3NhZ2VzLmxlbmd0aDtcblxuICBpZiAoIXJldXNlQXNzaXN0YW50KSB7XG4gICAgbmV3TWVzc2FnZXMucHVzaCh7XG4gICAgICBpZDogYXNzaXN0YW50TWVzc2FnZUlkLFxuICAgICAgc2Vzc2lvbklkOiBzZXNzaW9uQ29udGV4dC5zZXNzaW9uSWQsXG4gICAgICByb2xlOiBcImFzc2lzdGFudFwiLFxuICAgICAgY3JlYXRlZEF0OiBhc3Npc3RhbnRNZXNzYWdlQ3JlYXRlZEF0LFxuICAgICAgc3RhcnRlZEF0OiBudWxsLFxuICAgICAgY29tcGxldGVkQXQ6IG51bGwsXG4gICAgICBpbnRlcnJ1cHRlZEF0OiBudWxsLFxuICAgICAgaW50ZXJydXB0ZWRMYXN0UGFydDogbnVsbCxcbiAgICAgIHVzYWdlOiBudWxsLFxuICAgICAgd29ya2Zsb3dSdW5JZDogbnVsbCxcbiAgICAgIG1ldGFkYXRhOiBudWxsLFxuICAgIH0pO1xuICB9XG5cbiAgY29uc3QgZXhpc3RpbmdQYXJ0Q291bnQgPSByZXVzZUFzc2lzdGFudFxuICAgID8gbmV3UGFydHMuZmlsdGVyKChwKSA9PiBwLm1lc3NhZ2VJZCA9PT0gYXNzaXN0YW50TWVzc2FnZUlkKS5sZW5ndGhcbiAgICA6IDA7XG5cbiAgbGV0IHNlc3Npb24gPSBhd2FpdCBzZXNzaW9uQ29udGV4dC5hZ2VudC5zdG9yYWdlLnNlc3Npb24uZ2V0KHNlc3Npb25JZCk7XG4gIGNvbnN0IG5vdyA9IERhdGUubm93KCk7XG4gIGlmICghc2Vzc2lvbikge1xuICAgIHNlc3Npb24gPSB7XG4gICAgICBpZDogc2Vzc2lvbklkLFxuICAgICAgc2FuZGJveElkOiBzYW5kYm94SWQgPz8gbnVsbCxcbiAgICAgIGxhc3RNZXNzYWdlSWQ6IG51bGwsXG4gICAgICBjcmVhdGVkQXQ6IG5vdyxcbiAgICAgIHVwZGF0ZWRBdDogbm93LFxuICAgIH07XG4gICAgYXdhaXQgc2Vzc2lvbkNvbnRleHQuYWdlbnQuc3RvcmFnZS5zZXNzaW9uLnNldChzZXNzaW9uLmlkLCBzZXNzaW9uKTtcbiAgfSBlbHNlIGlmIChzYW5kYm94SWQgJiYgc2FuZGJveElkICE9PSBzZXNzaW9uLnNhbmRib3hJZCkge1xuICAgIHNlc3Npb24gPSB7IC4uLnNlc3Npb24sIHNhbmRib3hJZCwgdXBkYXRlZEF0OiBub3cgfTtcbiAgICBhd2FpdCBzZXNzaW9uQ29udGV4dC5hZ2VudC5zdG9yYWdlLnNlc3Npb24udXBkYXRlKHNlc3Npb24uaWQsIHtcbiAgICAgIHNhbmRib3hJZCxcbiAgICB9KTtcbiAgfVxuXG4gIGF3YWl0IGFsbCh7XG4gICAgYXN5bmMgc2F2ZU1lc3NhZ2VzKCkge1xuICAgICAgYXdhaXQgUHJvbWlzZS5hbGwoXG4gICAgICAgIG5ld01lc3NhZ2VzLm1hcChhc3luYyAobSwgaSkgPT4ge1xuICAgICAgICAgIGNvbnN0IGV4aXN0aW5nID0gYXdhaXQgc2Vzc2lvbkNvbnRleHQuYWdlbnQuc3RvcmFnZS5tZXNzYWdlLmdldChtLmlkKTtcbiAgICAgICAgICBpZiAoZXhpc3RpbmcpIHtcbiAgICAgICAgICAgIHJldHVybjtcbiAgICAgICAgICB9XG4gICAgICAgICAgYXdhaXQgc2Vzc2lvbkNvbnRleHQuYWdlbnQuc3RvcmFnZS5tZXNzYWdlLnNldChtLmlkLCB7XG4gICAgICAgICAgICAuLi5tLFxuICAgICAgICAgICAgY3JlYXRlZEF0OiBiYXNlVGltZSArIGksXG4gICAgICAgICAgICB3b3JrZmxvd1J1bklkOlxuICAgICAgICAgICAgICBtLmlkID09PSBhc3Npc3RhbnRNZXNzYWdlSWRcbiAgICAgICAgICAgICAgICA/ICh3b3JrZmxvd1J1bklkID8/IG51bGwpXG4gICAgICAgICAgICAgICAgOiBtLndvcmtmbG93UnVuSWQsXG4gICAgICAgICAgfSk7XG4gICAgICAgIH0pXG4gICAgICApO1xuICAgIH0sXG4gICAgYXN5bmMgc2F2ZVBhcnRzKCkge1xuICAgICAgYXdhaXQgUHJvbWlzZS5hbGwoXG4gICAgICAgIG5ld1BhcnRzLm1hcChhc3luYyAocCkgPT5cbiAgICAgICAgICBzZXNzaW9uQ29udGV4dC5hZ2VudC5zdG9yYWdlLnBhcnQuc2V0KHAuaWQsIHApXG4gICAgICAgIClcbiAgICAgICk7XG4gICAgfSxcbiAgICBhc3luYyByZXNvbHZlQXBwcm92YWxzKCkge1xuICAgICAgYXdhaXQgUHJvbWlzZS5hbGwoXG4gICAgICAgIG5ld0FwcHJvdmFscy5tYXAoYXN5bmMgKGEpID0+IHtcbiAgICAgICAgICBjb25zdCBwYXJ0SWQgPSBhcHByb3ZhbFBhcnRJZChhLmFwcHJvdmFsSWQpO1xuICAgICAgICAgIGNvbnN0IHBhcnQgPSBhd2FpdCBzZXNzaW9uQ29udGV4dC5hZ2VudC5zdG9yYWdlLnBhcnQuZ2V0KHBhcnRJZCk7XG4gICAgICAgICAgaWYgKCFwYXJ0KSB7XG4gICAgICAgICAgICB0aHJvdyBuZXcgQXBwcm92YWxOb3RGb3VuZEVycm9yKHtcbiAgICAgICAgICAgICAgaWQ6IHNlc3Npb25Db250ZXh0LnNlc3Npb25JZCxcbiAgICAgICAgICAgICAgYXBwcm92YWxJZDogYS5hcHByb3ZhbElkLFxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgfVxuICAgICAgICAgIGF3YWl0IHNlc3Npb25Db250ZXh0LmFnZW50LnN0b3JhZ2UucGFydC5zZXQocGFydElkLCB7XG4gICAgICAgICAgICAuLi5wYXJ0LFxuICAgICAgICAgICAgcGFydDoge1xuICAgICAgICAgICAgICAuLi5wYXJ0LnBhcnQsXG4gICAgICAgICAgICAgIHN0YXRlOiBhLmFwcHJvdmVkID8gXCJhcHByb3ZhbC1yZXNwb25kZWRcIiA6IFwib3V0cHV0LWRlbmllZFwiLFxuICAgICAgICAgICAgICBhcHByb3ZhbDoge1xuICAgICAgICAgICAgICAgIGlkOiBhLmFwcHJvdmFsSWQsXG4gICAgICAgICAgICAgICAgYXBwcm92ZWQ6IGEuYXBwcm92ZWQsXG4gICAgICAgICAgICAgICAgcmVhc29uOiBhLnJlYXNvbixcbiAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgIH0gYXMgdHlwZW9mIHBhcnQucGFydCxcbiAgICAgICAgICB9KTtcbiAgICAgICAgfSlcbiAgICAgICk7XG4gICAgfSxcbiAgICBhc3luYyB1cGRhdGVTZXNzaW9uKCkge1xuICAgICAgYXdhaXQgc2Vzc2lvbkNvbnRleHQuYWdlbnQuc3RvcmFnZS5zZXNzaW9uLnNldChzZXNzaW9uLmlkLCB7XG4gICAgICAgIC4uLnNlc3Npb24sXG4gICAgICAgIGxhc3RNZXNzYWdlSWQ6IGFzc2lzdGFudE1lc3NhZ2VJZCxcbiAgICAgICAgdXBkYXRlZEF0OiBEYXRlLm5vdygpLFxuICAgICAgfSk7XG4gICAgfSxcbiAgfSk7XG5cbiAgY29uc3Qgc3RhdGU6IFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0gc2Vzc2lvbi5zdGF0ZVxuICAgID8geyAuLi5zZXNzaW9uLnN0YXRlIH1cbiAgICA6IHsgLi4uKGluaXRpYWxTdGF0ZSA/PyB7fSkgfTtcblxuICByZXR1cm4ge1xuICAgIHdvcmtmbG93UnVuSWQsXG4gICAgd29ya2Zsb3dXcml0YWJsZSxcbiAgICBleGlzdGluZ1BhcnRDb3VudCxcbiAgICBhc3Npc3RhbnRNZXNzYWdlSWQsXG4gICAgYXNzaXN0YW50TWVzc2FnZUNyZWF0ZWRBdCxcbiAgICBzdGF0ZSxcbiAgfTtcbn1cblxuYXN5bmMgZnVuY3Rpb24gY29tcGxldGVNZXNzYWdlU3RlcCh7XG4gIGFzc2lzdGFudE1lc3NhZ2VJZCxcbiAgc2Vzc2lvbkNvbnRleHQsXG4gIHdyaXRhYmxlLFxuICB1c2FnZVN0ZXBzLFxufToge1xuICBhc3Npc3RhbnRNZXNzYWdlSWQ6IHN0cmluZztcbiAgc2Vzc2lvbkNvbnRleHQ6IFNlc3Npb25Db250ZXh0O1xuICB3cml0YWJsZTogV3JpdGFibGVTdHJlYW07XG4gIHVzYWdlU3RlcHM6IFN0ZXBVc2FnZVtdO1xufSkge1xuICBcInVzZSBzdGVwXCI7XG5cbiAgY29uc3QgbG9nID0gY3JlYXRlTG9nZ2VyKHtcbiAgICBjb25maWc6IHtcbiAgICAgIC4uLnNlc3Npb25Db250ZXh0LmFnZW50Lm9wdGlvbnMubG9nZ2luZyxcbiAgICAgIG5hbWU6IHNlc3Npb25Db250ZXh0LmFnZW50Lm5hbWUsXG4gICAgfSxcbiAgICBzdWJzeXN0ZW06IFwiYWktbG9vcFwiLFxuICB9KTtcbiAgY29uc3QgbXNnTG9nID0gbG9nLndpdGhDb250ZXh0KHtcbiAgICBzZXNzaW9uSWQ6IHNlc3Npb25Db250ZXh0LnNlc3Npb25JZCxcbiAgICBtZXNzYWdlSWQ6IGFzc2lzdGFudE1lc3NhZ2VJZCxcbiAgfSk7XG5cbiAgY29uc3QgdXNhZ2UgPVxuICAgIHVzYWdlU3RlcHMubGVuZ3RoID4gMFxuICAgICAgPyB7XG4gICAgICAgICAgc3RlcHM6IHVzYWdlU3RlcHMsXG4gICAgICAgICAgc3VtbWFyeToge1xuICAgICAgICAgICAgbW9kZWw6IHVzYWdlU3RlcHNbMF0/Lm1vZGVsID8/IFwidW5rbm93blwiLFxuICAgICAgICAgICAgaW5wdXRUb2tlbnM6IHVzYWdlU3RlcHMucmVkdWNlKChhLCBzKSA9PiBhICsgcy5pbnB1dFRva2VucywgMCksXG4gICAgICAgICAgICBvdXRwdXRUb2tlbnM6IHVzYWdlU3RlcHMucmVkdWNlKChhLCBzKSA9PiBhICsgcy5vdXRwdXRUb2tlbnMsIDApLFxuICAgICAgICAgICAgdG90YWxUb2tlbnM6IHVzYWdlU3RlcHMucmVkdWNlKChhLCBzKSA9PiBhICsgcy50b3RhbFRva2VucywgMCksXG4gICAgICAgICAgICBjYWNoZVJlYWRUb2tlbnM6IHVzYWdlU3RlcHMucmVkdWNlKFxuICAgICAgICAgICAgICAoYSwgcykgPT4gYSArIHMuY2FjaGVSZWFkVG9rZW5zLFxuICAgICAgICAgICAgICAwXG4gICAgICAgICAgICApLFxuICAgICAgICAgICAgY2FjaGVXcml0ZVRva2VuczogdXNhZ2VTdGVwcy5yZWR1Y2UoXG4gICAgICAgICAgICAgIChhLCBzKSA9PiBhICsgcy5jYWNoZVdyaXRlVG9rZW5zLFxuICAgICAgICAgICAgICAwXG4gICAgICAgICAgICApLFxuICAgICAgICAgICAgcmVhc29uaW5nVG9rZW5zOiB1c2FnZVN0ZXBzLnJlZHVjZShcbiAgICAgICAgICAgICAgKGEsIHMpID0+IGEgKyBzLnJlYXNvbmluZ1Rva2VucyxcbiAgICAgICAgICAgICAgMFxuICAgICAgICAgICAgKSxcbiAgICAgICAgICAgIHN0ZXBDb3VudDogdXNhZ2VTdGVwcy5sZW5ndGgsXG4gICAgICAgICAgfSxcbiAgICAgICAgfVxuICAgICAgOiBudWxsO1xuXG4gIGF3YWl0IFByb21pc2UuYWxsKFtcbiAgICBzZXNzaW9uQ29udGV4dC5hZ2VudC5zdG9yYWdlLm1lc3NhZ2UudXBkYXRlKGFzc2lzdGFudE1lc3NhZ2VJZCwge1xuICAgICAgY29tcGxldGVkQXQ6IERhdGUubm93KCksXG4gICAgICB1c2FnZSxcbiAgICB9KSxcbiAgICB3cml0YWJsZS5jbG9zZSgpLmNhdGNoKCgpID0+IHVuZGVmaW5lZCksXG4gIF0pO1xuXG4gIG1zZ0xvZy5pbmZvKFwibWVzc2FnZSBjb21wbGV0ZWRcIiwge1xuICAgIHN0ZXBzOiB1c2FnZVN0ZXBzLmxlbmd0aCxcbiAgICB0b3RhbFRva2VuczogdXNhZ2U/LnN1bW1hcnkudG90YWxUb2tlbnMsXG4gIH0pO1xufVxuXG4vKiAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG4gKiBVdGlsc1xuICogLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0qL1xuXG5leHBvcnQgZnVuY3Rpb24gYXBwcm92YWxQYXJ0SWQoYXBwcm92YWxJZDogc3RyaW5nKTogc3RyaW5nIHtcbiAgcmV0dXJuIGBwYXJ0XyR7YXBwcm92YWxJZH1gO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gcmVwbGF5QWN0aXZlUnVuKFxuICBhY3RpdmU6IEFjdGl2ZVJ1bixcbiAgc3RhcnRJbmRleCA9IDBcbik6IFJlYWRhYmxlU3RyZWFtPFVJTWVzc2FnZUNodW5rPiB7XG4gIGxldCBjdXJzb3IgPSBzdGFydEluZGV4O1xuICBsZXQgY2FuY2VsbGVkID0gZmFsc2U7XG5cbiAgcmV0dXJuIG5ldyBSZWFkYWJsZVN0cmVhbTxVSU1lc3NhZ2VDaHVuaz4oe1xuICAgIGFzeW5jIHB1bGwoY29udHJvbGxlcikge1xuICAgICAgLy8gRHJhaW4gYW55IGJ1ZmZlcmVkIGNodW5rc1xuICAgICAgd2hpbGUgKGN1cnNvciA8IGFjdGl2ZS5jaHVua3MubGVuZ3RoKSB7XG4gICAgICAgIGNvbnRyb2xsZXIuZW5xdWV1ZShhY3RpdmUuY2h1bmtzW2N1cnNvcisrXSk7XG4gICAgICB9XG5cbiAgICAgIC8vIFdhaXQgZm9yIGVpdGhlciBhIG5ldyBjaHVuayBvciBjb21wbGV0aW9uXG4gICAgICB3aGlsZSAoIWNhbmNlbGxlZCAmJiBjdXJzb3IgPj0gYWN0aXZlLmNodW5rcy5sZW5ndGgpIHtcbiAgICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgUHJvbWlzZS5yYWNlKFtcbiAgICAgICAgICBhY3RpdmUub25DaHVuay50aGVuKCgpID0+IFwiY2h1bmtcIiBhcyBjb25zdCksXG4gICAgICAgICAgYWN0aXZlLmRvbmUudGhlbigoKSA9PiBcImRvbmVcIiBhcyBjb25zdCksXG4gICAgICAgIF0pO1xuXG4gICAgICAgIC8vIERyYWluIG5ldyBjaHVua3NcbiAgICAgICAgd2hpbGUgKGN1cnNvciA8IGFjdGl2ZS5jaHVua3MubGVuZ3RoKSB7XG4gICAgICAgICAgY29udHJvbGxlci5lbnF1ZXVlKGFjdGl2ZS5jaHVua3NbY3Vyc29yKytdKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmIChyZXN1bHQgPT09IFwiZG9uZVwiKSB7XG4gICAgICAgICAgY29udHJvbGxlci5jbG9zZSgpO1xuICAgICAgICAgIHJldHVybjtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH0sXG4gICAgY2FuY2VsKCkge1xuICAgICAgY2FuY2VsbGVkID0gdHJ1ZTtcbiAgICB9LFxuICB9KTtcbn1cblxuZnVuY3Rpb24gdG9NZXNzYWdlQW5kUGFydHM8VG9vbHMgZXh0ZW5kcyBUb29sU2V0Pih7XG4gIHNlc3Npb25JZCxcbiAgaWQsXG4gIGlucHV0LFxuICBkZWZhdWx0Um9sZSxcbn06IHtcbiAgc2Vzc2lvbklkOiBzdHJpbmc7XG4gIGlkPzogc3RyaW5nO1xuICBpbnB1dDogU2VuZElucHV0PFRvb2xzPjtcbiAgZGVmYXVsdFJvbGU6IFVJTWVzc2FnZVtcInJvbGVcIl07XG59KToge1xuICBtZXNzYWdlOiBNZXNzYWdlIHwgdW5kZWZpbmVkO1xuICBwYXJ0czogUGFydFtdO1xuICBhcHByb3ZhbHM6IEFwcHJvdmFsRGF0YVtdO1xufSB7XG4gIGxldCBtZXNzYWdlSWQgPSBpZCA/PyBgbWVzc2FnZV8ke3VsaWQoKX1gO1xuICBpZiAodHlwZW9mIGlucHV0ID09PSBcInN0cmluZ1wiKSB7XG4gICAgY29uc3Qgbm93ID0gRGF0ZS5ub3coKTtcbiAgICByZXR1cm4ge1xuICAgICAgYXBwcm92YWxzOiBbXSxcbiAgICAgIG1lc3NhZ2U6IHtcbiAgICAgICAgaWQ6IG1lc3NhZ2VJZCxcbiAgICAgICAgc2Vzc2lvbklkLFxuICAgICAgICByb2xlOiBkZWZhdWx0Um9sZSxcbiAgICAgICAgY3JlYXRlZEF0OiBub3csXG4gICAgICAgIHN0YXJ0ZWRBdDogbnVsbCxcbiAgICAgICAgY29tcGxldGVkQXQ6IG51bGwsXG4gICAgICAgIGludGVycnVwdGVkQXQ6IG51bGwsXG4gICAgICAgIGludGVycnVwdGVkTGFzdFBhcnQ6IG51bGwsXG4gICAgICAgIHVzYWdlOiBudWxsLFxuICAgICAgICB3b3JrZmxvd1J1bklkOiBudWxsLFxuICAgICAgICBtZXRhZGF0YTogbnVsbCxcbiAgICAgIH0sXG4gICAgICBwYXJ0czogW1xuICAgICAgICB7XG4gICAgICAgICAgc2Vzc2lvbklkLFxuICAgICAgICAgIG1lc3NhZ2VJZCxcbiAgICAgICAgICBpZDogYHBhcnRfJHttZXNzYWdlSWR9XzBgLFxuICAgICAgICAgIGluZGV4OiAwLFxuICAgICAgICAgIHBhcnQ6IHsgdHlwZTogXCJ0ZXh0XCIsIHRleHQ6IGlucHV0IH0sXG4gICAgICAgIH0sXG4gICAgICBdLFxuICAgIH07XG4gIH1cbiAgaWYgKFwidHlwZVwiIGluIGlucHV0ICYmIGlucHV0LnR5cGUgPT09IFwiYXBwcm92YWxcIikge1xuICAgIHJldHVybiB7XG4gICAgICBtZXNzYWdlOiB1bmRlZmluZWQsXG4gICAgICBwYXJ0czogW10sXG4gICAgICBhcHByb3ZhbHM6IFtpbnB1dC5hcHByb3ZhbF0sXG4gICAgfTtcbiAgfVxuXG4gIGNvbnN0IG1zZyA9XG4gICAgXCJ0eXBlXCIgaW4gaW5wdXQgJiYgaW5wdXQudHlwZSA9PT0gXCJtZXNzYWdlXCIgPyBpbnB1dC5tZXNzYWdlIDogaW5wdXQ7XG4gIGlmIChcImlkXCIgaW4gbXNnICYmIG1zZy5pZCkge1xuICAgIG1lc3NhZ2VJZCA9IG1zZy5pZDtcbiAgfVxuXG4gIGNvbnN0IG5vdyA9IERhdGUubm93KCk7XG4gIGNvbnN0IG1ldGFkYXRhID1cbiAgICBcIm1ldGFkYXRhXCIgaW4gbXNnICYmIG1zZy5tZXRhZGF0YSAhPSBudWxsXG4gICAgICA/IChtc2cubWV0YWRhdGEgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4pXG4gICAgICA6IG51bGw7XG4gIHJldHVybiB7XG4gICAgYXBwcm92YWxzOiBbXSxcbiAgICBtZXNzYWdlOiB7XG4gICAgICBzZXNzaW9uSWQsXG4gICAgICBpZDogbWVzc2FnZUlkLFxuICAgICAgY3JlYXRlZEF0OiBub3csXG4gICAgICBzdGFydGVkQXQ6IG51bGwsXG4gICAgICBjb21wbGV0ZWRBdDogbnVsbCxcbiAgICAgIGludGVycnVwdGVkQXQ6IG51bGwsXG4gICAgICBpbnRlcnJ1cHRlZExhc3RQYXJ0OiBudWxsLFxuICAgICAgdXNhZ2U6IG51bGwsXG4gICAgICB3b3JrZmxvd1J1bklkOiBudWxsLFxuICAgICAgcm9sZTogbXNnLnJvbGUgPz8gZGVmYXVsdFJvbGUsXG4gICAgICBtZXRhZGF0YSxcbiAgICB9LFxuICAgIHBhcnRzOiBtc2cucGFydHMubWFwKChwYXJ0LCBpbmRleCkgPT4ge1xuICAgICAgY29uc3QgYXAgPSBwYXJ0IGFzIHsgYXBwcm92YWw/OiB7IGlkOiBzdHJpbmcgfSB9O1xuICAgICAgY29uc3QgaWQgPSBhcC5hcHByb3ZhbD8uaWRcbiAgICAgICAgPyBhcHByb3ZhbFBhcnRJZChhcC5hcHByb3ZhbC5pZClcbiAgICAgICAgOiBgcGFydF8ke21lc3NhZ2VJZH1fJHtpbmRleH1gO1xuICAgICAgcmV0dXJuIHsgc2Vzc2lvbklkLCBtZXNzYWdlSWQsIGlkLCBpbmRleCwgcGFydCB9O1xuICAgIH0pLFxuICB9O1xufVxuXG4vKiAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG4gKiBFcnJvcnNcbiAqIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi9cblxuZXhwb3J0IGNsYXNzIFNlbmRFcnJvciBleHRlbmRzIGVycm9yZS5jcmVhdGVUYWdnZWRFcnJvcih7XG4gIG5hbWU6IFwiU2VuZEVycm9yXCIsXG4gIG1lc3NhZ2U6IFwiU2VuZCBmYWlsZWQgZm9yIHNlc3Npb24gJGlkOiAkcmVhc29uXCIsXG59KSB7fVxuXG5leHBvcnQgY2xhc3MgQXBwcm92YWxOb3RGb3VuZEVycm9yIGV4dGVuZHMgZXJyb3JlLmNyZWF0ZVRhZ2dlZEVycm9yKHtcbiAgbmFtZTogXCJBcHByb3ZhbE5vdEZvdW5kRXJyb3JcIixcbiAgbWVzc2FnZTogXCJBcHByb3ZhbCAkYXBwcm92YWxJZCBub3QgZm91bmQgaW4gc2Vzc2lvbiAkaWRcIixcbn0pIHt9XG4iLCAiaW1wb3J0IHR5cGUge1xuICBGaW5pc2hSZWFzb24sXG4gIFRvb2xTZXQsXG4gIFVJTWVzc2FnZSxcbiAgVUlNZXNzYWdlU3RyZWFtV3JpdGVyLFxufSBmcm9tIFwiYWlcIjtcbmltcG9ydCB7XG4gIGNvbnZlcnRUb01vZGVsTWVzc2FnZXMsXG4gIGNyZWF0ZVVJTWVzc2FnZVN0cmVhbSxcbiAgc3RlcENvdW50SXMsXG4gIHN0cmVhbVRleHQgYXMgc3RyZWFtVGV4dEFpLFxufSBmcm9tIFwiYWlcIjtcbmltcG9ydCB0eXBlIHtcbiAgQWdlbnRTdGF0dXMsXG4gIEFueUFnZW50LFxuICBSZXNvbHZhYmxlLFxuICBSZXNvbHZhYmxlQXJncyxcbiAgVHlwZWRVSU1lc3NhZ2UsXG59IGZyb20gXCIuLi8uLi9hZ2VudC9jbGllbnRcIjtcbmltcG9ydCB7IFNlc3Npb25Ob3RGb3VuZEVycm9yIH0gZnJvbSBcIi4uLy4uL2Vycm9yc1wiO1xuaW1wb3J0IHsgZGlzY292ZXJTa2lsbHNJblNhbmRib3ggfSBmcm9tIFwiLi4vLi4vc2tpbGxzL2Rpc2NvdmVyXCI7XG5pbXBvcnQgeyByZXNvbHZlU2tpbGxJbnB1dHNUb1NhbmRib3hEaXJzIH0gZnJvbSBcIi4uLy4uL3NraWxscy9yZXNvbHZlXCI7XG5pbXBvcnQgdHlwZSB7IFNraWxsU3VtbWFyeSB9IGZyb20gXCIuLi8uLi9za2lsbHMvdHlwZXNcIjtcbmltcG9ydCB0eXBlIHsgUGFydCB9IGZyb20gXCIuLi8uLi9zdG9yYWdlL3R5cGVzXCI7XG5pbXBvcnQgeyB0eXBlIEJ1aWx0SW5Ub29sTmFtZSwgZ2V0VG9vbHMsIHR5cGUgVG9vbENvbnRleHQgfSBmcm9tIFwiLi4vLi4vdG9vbHNcIjtcbmltcG9ydCB7IGNyZWF0ZUxvZ2dlciwgdHlwZSBMb2dnZXIgfSBmcm9tIFwiLi4vLi4vdXRpbHMvbG9nZ2VyXCI7XG5pbXBvcnQge1xuICBhcHBseVByb21wdENhY2hpbmdUb01vZGVsUmVxdWVzdCxcbiAgbWVyZ2VQcm92aWRlck9wdGlvbnMsXG59IGZyb20gXCIuLi8uLi91dGlscy9wcm9tcHQtY2FjaGVcIjtcbmltcG9ydCB7IGFzc2VtYmxlVUlNZXNzYWdlcyB9IGZyb20gXCIuLi8uLi91dGlscy91aVwiO1xuaW1wb3J0IHR5cGUgeyBTdGVwVXNhZ2UgfSBmcm9tIFwiLi4vLi4vdXRpbHMvdXNhZ2VcIjtcbmltcG9ydCB7IGFwcHJvdmFsUGFydElkIH0gZnJvbSBcIi4uL3NlbmRcIjtcblxuY29uc3QgSU5URVJSVVBUX1BPTExfSU5URVJWQUxfTVMgPSAyNTA7XG5cbmZ1bmN0aW9uIHBvbGxGb3JJbnRlcnJ1cHQoe1xuICBhZ2VudCxcbiAgbWVzc2FnZUlkLFxuICBzaWduYWwsXG4gIGFib3J0Q29udHJvbGxlcixcbn06IHtcbiAgYWdlbnQ6IEFueUFnZW50O1xuICBtZXNzYWdlSWQ6IHN0cmluZztcbiAgc2lnbmFsOiBBYm9ydFNpZ25hbDtcbiAgYWJvcnRDb250cm9sbGVyOiBBYm9ydENvbnRyb2xsZXI7XG59KToge1xuICBmaXJzdENoZWNrOiBQcm9taXNlPHZvaWQ+O1xuICBpbnRlcnJ1cHRlZExhc3RQYXJ0UmVmOiB7IGN1cnJlbnQ6IHsgaW5kZXg6IG51bWJlcjsgcGFydDogdW5rbm93biB9IHwgbnVsbCB9O1xufSB7XG4gIGxldCByZXNvbHZlRmlyc3RDaGVjazogKCkgPT4gdm9pZDtcbiAgY29uc3QgZmlyc3RDaGVjayA9IG5ldyBQcm9taXNlPHZvaWQ+KChyZXNvbHZlKSA9PiB7XG4gICAgcmVzb2x2ZUZpcnN0Q2hlY2sgPSByZXNvbHZlO1xuICB9KTtcbiAgY29uc3QgaW50ZXJydXB0ZWRMYXN0UGFydFJlZiA9IHtcbiAgICBjdXJyZW50OiBudWxsIGFzIHsgaW5kZXg6IG51bWJlcjsgcGFydDogdW5rbm93biB9IHwgbnVsbCxcbiAgfTtcblxuICBjb25zdCBsb2cgPSBjcmVhdGVMb2dnZXIoe1xuICAgIGNvbmZpZzogeyAuLi5hZ2VudC5vcHRpb25zLmxvZ2dpbmcsIG5hbWU6IGFnZW50Lm5hbWUgfSxcbiAgICBzdWJzeXN0ZW06IFwiYWktbG9vcFwiLFxuICB9KTtcblxuICBjb25zdCBwb2xsID0gYXN5bmMgKCkgPT4ge1xuICAgIGxldCBpc0ZpcnN0ID0gdHJ1ZTtcbiAgICB3aGlsZSAoIXNpZ25hbC5hYm9ydGVkKSB7XG4gICAgICBjb25zdCBtZXNzYWdlID0gYXdhaXQgYWdlbnQuc3RvcmFnZS5tZXNzYWdlLmdldChtZXNzYWdlSWQpO1xuICAgICAgaWYgKCFtZXNzYWdlKSB7XG4gICAgICAgIGlmIChpc0ZpcnN0KSB7XG4gICAgICAgICAgcmVzb2x2ZUZpcnN0Q2hlY2soKTtcbiAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cbiAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBNZXNzYWdlICR7bWVzc2FnZUlkfSBub3QgZm91bmRgKTtcbiAgICAgIH1cbiAgICAgIGlmIChtZXNzYWdlLmludGVycnVwdGVkQXQgIT09IG51bGwpIHtcbiAgICAgICAgbG9nLmluZm8oXCJpbnRlcnJ1cHQgZGV0ZWN0ZWRcIiwgeyBtZXNzYWdlSWQgfSk7XG4gICAgICAgIGludGVycnVwdGVkTGFzdFBhcnRSZWYuY3VycmVudCA9IG1lc3NhZ2UuaW50ZXJydXB0ZWRMYXN0UGFydDtcbiAgICAgICAgYWJvcnRDb250cm9sbGVyLmFib3J0KCk7XG4gICAgICAgIGlmIChpc0ZpcnN0KSB7XG4gICAgICAgICAgcmVzb2x2ZUZpcnN0Q2hlY2soKTtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm47XG4gICAgICB9XG4gICAgICBpZiAoaXNGaXJzdCkge1xuICAgICAgICBpc0ZpcnN0ID0gZmFsc2U7XG4gICAgICAgIHJlc29sdmVGaXJzdENoZWNrKCk7XG4gICAgICB9XG4gICAgICBhd2FpdCBuZXcgUHJvbWlzZSgocmVzb2x2ZSkgPT5cbiAgICAgICAgc2V0VGltZW91dChyZXNvbHZlLCBJTlRFUlJVUFRfUE9MTF9JTlRFUlZBTF9NUylcbiAgICAgICk7XG4gICAgfVxuICB9O1xuXG4gIHBvbGwoKTtcbiAgcmV0dXJuIHsgZmlyc3RDaGVjaywgaW50ZXJydXB0ZWRMYXN0UGFydFJlZiB9O1xufVxuXG50eXBlIFN0cmVhbVRleHRPcHRpb25zID0ge1xuICBzdGVwVGhyb3VnaHB1dDogbnVtYmVyO1xuICBhc3Npc3RhbnRNZXNzYWdlSWQ6IHN0cmluZztcbiAgc2Vzc2lvbklkOiBzdHJpbmc7XG4gIGFnZW50OiBBbnlBZ2VudDtcbiAgY3JlYXRlZEF0OiBudW1iZXI7XG4gIGNvbnRleHQ6IFJlY29yZDxzdHJpbmcsIHVua25vd24+O1xuICBzdGF0ZTogUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG4gIHdyaXRhYmxlOiBXcml0YWJsZVN0cmVhbTtcbiAgbGFzdFBhcnRJbmRleDogbnVtYmVyO1xuICBzdGVwSW5kZXhPZmZzZXQ6IG51bWJlcjtcbiAgZGlzY292ZXJlZFNraWxsczogU2tpbGxTdW1tYXJ5W10gfCBudWxsO1xufTtcblxuZXhwb3J0IHR5cGUgU3RyZWFtVGV4dFJlc3VsdCA9IHtcbiAgZmluaXNoUmVhc29uOiBGaW5pc2hSZWFzb247XG4gIGxhc3RQYXJ0SW5kZXg6IG51bWJlcjtcbiAgdXNhZ2VTdGVwczogU3RlcFVzYWdlW107XG4gIHBlbmRpbmdBcHByb3ZhbHM6IHsgYXBwcm92YWxJZDogc3RyaW5nOyB0b29sTmFtZTogc3RyaW5nIH1bXTtcbiAgbWF4U3RlcHM6IG51bWJlciB8IHVuZGVmaW5lZDtcbiAgZGlzY292ZXJlZFNraWxsczogU2tpbGxTdW1tYXJ5W10gfCBudWxsO1xuICBjb250ZXh0OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbn07XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBzdHJlYW1UZXh0PFRvb2xzIGV4dGVuZHMgVG9vbFNldD4oe1xuICBzdGVwVGhyb3VnaHB1dCxcbiAgYXNzaXN0YW50TWVzc2FnZUlkLFxuICBhZ2VudCxcbiAgc2Vzc2lvbklkLFxuICBjb250ZXh0LFxuICBzdGF0ZSxcbiAgY3JlYXRlZEF0LFxuICB3cml0YWJsZSxcbiAgbGFzdFBhcnRJbmRleCxcbiAgc3RlcEluZGV4T2Zmc2V0LFxuICBkaXNjb3ZlcmVkU2tpbGxzLFxufTogU3RyZWFtVGV4dE9wdGlvbnMpOiBQcm9taXNlPFN0cmVhbVRleHRSZXN1bHQ+IHtcbiAgY29uc3QgbG9nID0gY3JlYXRlTG9nZ2VyKHtcbiAgICBjb25maWc6IHsgLi4uYWdlbnQub3B0aW9ucy5sb2dnaW5nLCBuYW1lOiBhZ2VudC5uYW1lIH0sXG4gICAgc3Vic3lzdGVtOiBcImFpLWxvb3BcIixcbiAgfSk7XG5cbiAgY29uc3Qgc3RlcExvZyA9IGxvZy53aXRoQ29udGV4dCh7XG4gICAgc2Vzc2lvbklkLFxuICAgIG1lc3NhZ2VJZDogYXNzaXN0YW50TWVzc2FnZUlkLFxuICB9KTtcbiAgY29uc3QgZG9uZVN0ZXAgPSBzdGVwTG9nLnRpbWUoXCJzdHJlYW1UZXh0XCIpO1xuXG4gIGNvbnN0IHRocm90dGxlZEVtaXRTdGF0dXMgPSBjcmVhdGVUaHJvdHRsZWRFbWl0U3RhdHVzKGFnZW50KTtcblxuICBjb25zdCBhYm9ydENvbnRyb2xsZXIgPSBuZXcgQWJvcnRDb250cm9sbGVyKCk7XG4gIGNvbnN0IHBvbGxDb250cm9sbGVyID0gbmV3IEFib3J0Q29udHJvbGxlcigpO1xuXG4gIGNvbnN0IHsgZmlyc3RDaGVjaywgaW50ZXJydXB0ZWRMYXN0UGFydFJlZiB9ID0gcG9sbEZvckludGVycnVwdCh7XG4gICAgYWdlbnQsXG4gICAgbWVzc2FnZUlkOiBhc3Npc3RhbnRNZXNzYWdlSWQsXG4gICAgc2lnbmFsOiBwb2xsQ29udHJvbGxlci5zaWduYWwsXG4gICAgYWJvcnRDb250cm9sbGVyLFxuICB9KTtcblxuICBjb25zdCBkb25lTG9hZCA9IHN0ZXBMb2cudGltZShcImxvYWQgc2Vzc2lvbiArIG1lc3NhZ2VzICsgcGFydHMgKyBzYW5kYm94XCIpO1xuICBjb25zdCBbLCBtZXNzYWdlc1Jlc3VsdCwgcGFydHNSZXN1bHQsIHsgc2FuZGJveCwgc2FuZGJveFJlY29yZCB9XSA9XG4gICAgYXdhaXQgUHJvbWlzZS5hbGwoW1xuICAgICAgZmlyc3RDaGVjayxcbiAgICAgIGFnZW50LnN0b3JhZ2UubWVzc2FnZS5saXN0QnlTZXNzaW9uKHNlc3Npb25JZCksXG4gICAgICBhZ2VudC5zdG9yYWdlLnBhcnQubGlzdEJ5U2Vzc2lvbihzZXNzaW9uSWQpLFxuICAgICAgYWdlbnQuc3RvcmFnZS5zZXNzaW9uLmdldChzZXNzaW9uSWQpLnRoZW4oYXN5bmMgKHNlc3Npb24pID0+IHtcbiAgICAgICAgaWYgKCFzZXNzaW9uKSB7XG4gICAgICAgICAgdGhyb3cgbmV3IFNlc3Npb25Ob3RGb3VuZEVycm9yKHsgaWQ6IHNlc3Npb25JZCB9KTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IHNhbmRib3hJZCA9IHNlc3Npb24uc2FuZGJveElkID8/IHNlc3Npb24uaWQ7XG4gICAgICAgIGNvbnN0IHNhbmRib3ggPSBhZ2VudC5zYW5kYm94KHNhbmRib3hJZCk7XG4gICAgICAgIGNvbnN0IHNhbmRib3hSZWNvcmQgPSBhd2FpdCBhZ2VudC5zdG9yYWdlLnNhbmRib3guZ2V0KHNhbmRib3hJZCk7XG5cbiAgICAgICAgcmV0dXJuIHsgc2FuZGJveCwgc2FuZGJveFJlY29yZCB9O1xuICAgICAgfSksXG4gICAgXSk7XG5cbiAgZG9uZUxvYWQoKTtcblxuICBpZiAoYWJvcnRDb250cm9sbGVyLnNpZ25hbC5hYm9ydGVkKSB7XG4gICAgcG9sbENvbnRyb2xsZXIuYWJvcnQoKTtcbiAgICByZXR1cm4ge1xuICAgICAgZmluaXNoUmVhc29uOiBcInN0b3BcIiBhcyBGaW5pc2hSZWFzb24sXG4gICAgICBsYXN0UGFydEluZGV4LFxuICAgICAgdXNhZ2VTdGVwczogW10sXG4gICAgICBwZW5kaW5nQXBwcm92YWxzOiBbXSxcbiAgICAgIG1heFN0ZXBzOiB1bmRlZmluZWQsXG4gICAgICBkaXNjb3ZlcmVkU2tpbGxzLFxuICAgICAgY29udGV4dCxcbiAgICB9O1xuICB9XG5cbiAgY29uc3Qgc2V0U3RhcnRlZFByb21pc2UgPVxuICAgIGxhc3RQYXJ0SW5kZXggPT09IDBcbiAgICAgID8gKGFzeW5jICgpID0+IHtcbiAgICAgICAgICBjb25zdCBub3cgPSBEYXRlLm5vdygpO1xuICAgICAgICAgIGNvbnN0IHBlbmRpbmdNZXNzYWdlcyA9IG1lc3NhZ2VzUmVzdWx0LmZpbHRlcihcbiAgICAgICAgICAgIChtKSA9PiBtLmNyZWF0ZWRBdCA8PSBjcmVhdGVkQXQgJiYgbS5zdGFydGVkQXQgPT09IG51bGxcbiAgICAgICAgICApO1xuICAgICAgICAgIGlmIChwZW5kaW5nTWVzc2FnZXMubGVuZ3RoID4gMCkge1xuICAgICAgICAgICAgcmV0dXJuIGF3YWl0IFByb21pc2UuYWxsKFxuICAgICAgICAgICAgICBwZW5kaW5nTWVzc2FnZXMubWFwKChtKSA9PlxuICAgICAgICAgICAgICAgIGFnZW50LnN0b3JhZ2UubWVzc2FnZS51cGRhdGUobS5pZCwge1xuICAgICAgICAgICAgICAgICAgc3RhcnRlZEF0OiBub3csXG4gICAgICAgICAgICAgICAgICBjb21wbGV0ZWRBdDogbS5yb2xlID09PSBcImFzc2lzdGFudFwiID8gbnVsbCA6IG5vdyxcbiAgICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgICApXG4gICAgICAgICAgICApO1xuICAgICAgICAgIH1cbiAgICAgICAgfSkoKVxuICAgICAgOiBudWxsO1xuXG4gIGNvbnN0IGFsbFBhcnRzID0gWy4uLnBhcnRzUmVzdWx0XTtcbiAgY29uc3QgbmV4dFBhcnRJbmRleFJlZiA9IHtcbiAgICBjdXJyZW50OiBNYXRoLm1heChcbiAgICAgIGxhc3RQYXJ0SW5kZXgsXG4gICAgICAuLi5hbGxQYXJ0c1xuICAgICAgICAuZmlsdGVyKChwKSA9PiBwLm1lc3NhZ2VJZCA9PT0gYXNzaXN0YW50TWVzc2FnZUlkKVxuICAgICAgICAubWFwKChwKSA9PiBwLmluZGV4ICsgMSksXG4gICAgICAwXG4gICAgKSxcbiAgfTtcblxuICBjb25zdCBza2lsbHNSZWY6IHsgY3VycmVudDogU2tpbGxTdW1tYXJ5W10gfSA9IHtcbiAgICBjdXJyZW50OiBkaXNjb3ZlcmVkU2tpbGxzID8/IFtdLFxuICB9O1xuXG4gIC8vIEJ1aWxkIHVpTWVzc2FnZXMgZnJvbSBjdXJyZW50IHBhcnRzIChyZWJ1aWx0IGJldHdlZW4gaXRlcmF0aW9ucylcbiAgbGV0IHVpTWVzc2FnZXMgPSBhc3NlbWJsZVVJTWVzc2FnZXMoe1xuICAgIG1lc3NhZ2VzOiBtZXNzYWdlc1Jlc3VsdCxcbiAgICBwYXJ0czogYWxsUGFydHMsXG4gICAgdW50aWw6IGNyZWF0ZWRBdCxcbiAgICBpbmNsdWRlUXVldWVkOiB0cnVlLFxuICAgIGV4Y2x1ZGVTdWJUb29sUGFydHM6IHRydWUsXG4gIH0pO1xuXG4gIC8vIEJ1aWxkIHJlc29sdmFibGUgYXJncyBmb3IgcmVzb2x2aW5nIGFnZW50IG9wdGlvbnNcbiAgY29uc3QgcmVzb2x2YWJsZUFyZ3M6IFJlc29sdmFibGVBcmdzID0ge1xuICAgIGNvbnRleHQsXG4gICAgc3RhdGUsXG4gICAgc2Vzc2lvbklkLFxuICAgIHNhbmRib3gsXG4gICAgZ2V0IG1lc3NhZ2VzKCkge1xuICAgICAgcmV0dXJuIHVpTWVzc2FnZXM7XG4gICAgfSxcbiAgfTtcblxuICAvLyBza2lsbHMgZGlzY292ZXJ5IFx1MjAxNCBmaXJzdCBpdGVyYXRpb24gb25seVxuICBpZiAobGFzdFBhcnRJbmRleCA9PT0gMCAmJiAhZGlzY292ZXJlZFNraWxscykge1xuICAgIGNvbnN0IHJlc29sdmVkU2tpbGxzID0gYXdhaXQgcmVzb2x2ZU9wdGlvbihcbiAgICAgIGFnZW50Lm9wdGlvbnMuc2tpbGxzLFxuICAgICAgcmVzb2x2YWJsZUFyZ3NcbiAgICApO1xuICAgIGNvbnN0IHNraWxsc0RpcnMgPSByZXNvbHZlZFNraWxscz8ubGVuZ3RoXG4gICAgICA/IGF3YWl0IHJlc29sdmVTa2lsbElucHV0c1RvU2FuZGJveERpcnMoe1xuICAgICAgICAgIGFnZW50LFxuICAgICAgICAgIHNhbmRib3gsXG4gICAgICAgICAgc2tpbGxzOiByZXNvbHZlZFNraWxscyxcbiAgICAgICAgICBzZXNzaW9uSWQsXG4gICAgICAgIH0pXG4gICAgICA6IFtdO1xuICAgIGlmIChza2lsbHNEaXJzLmxlbmd0aCA+IDApIHtcbiAgICAgIGNvbnN0IGRvbmVTa2lsbHMgPSBzdGVwTG9nLnRpbWUoXCJkaXNjb3ZlciBza2lsbHNcIiwgeyBza2lsbHNEaXJzIH0pO1xuICAgICAgc2tpbGxzUmVmLmN1cnJlbnQgPSBhd2FpdCBkaXNjb3ZlclNraWxsc0luU2FuZGJveCh7XG4gICAgICAgIGFnZW50LFxuICAgICAgICBzYW5kYm94LFxuICAgICAgICBza2lsbHNEaXJzLFxuICAgICAgICBzZXNzaW9uSWQsXG4gICAgICB9KTtcbiAgICAgIGRvbmVTa2lsbHMoeyBjb3VudDogc2tpbGxzUmVmLmN1cnJlbnQubGVuZ3RoIH0pO1xuICAgIH1cbiAgfVxuXG4gIC8qKlxuICAgKiBJbm5lciBsb29wOiBydW4gdXAgdG8gYHN0ZXBUaHJvdWdocHV0YCBzZXF1ZW50aWFsIExMTSBjYWxscy5cbiAgICogUGFydHMgYXJlIHBlcnNpc3RlZCBhZnRlciBlYWNoIGNhbGwgc28gcHJvZ3Jlc3MgaXNuJ3QgbG9zdC5cbiAgICogU2Vzc2lvbiBpcyByZS1mZXRjaGVkIGJldHdlZW4gaXRlcmF0aW9ucyAoY29uZmlnIG1heSBjaGFuZ2UpLFxuICAgKiBwYXJhbGxlbGl6ZWQgd2l0aCBwYXJ0IHBlcnNpc3RlbmNlIG9uIG5vbi1maW5hbCBpdGVyYXRpb25zLlxuICAgKi9cbiAgY29uc3QgdXNhZ2VTdGVwczogU3RlcFVzYWdlW10gPSBbXTtcbiAgbGV0IGludGVybmFsU3RlcEluZGV4ID0gMDtcbiAgbGV0IGxvb3BGaW5pc2hSZWFzb246IEZpbmlzaFJlYXNvbiA9IFwidW5rbm93blwiIGFzIEZpbmlzaFJlYXNvbjtcbiAgbGV0IHdhc0Fib3J0ZWQgPSBmYWxzZTtcbiAgY29uc3QgYWxsUGVuZGluZ0FwcHJvdmFsczogeyBhcHByb3ZhbElkOiBzdHJpbmc7IHRvb2xOYW1lOiBzdHJpbmcgfVtdID0gW107XG4gIGxldCBjdXJyZW50TWF4U3RlcHM6IG51bWJlciB8IHVuZGVmaW5lZDtcblxuICB0cnkge1xuICAgIGZvciAobGV0IGl0ZXJhdGlvbiA9IDA7IGl0ZXJhdGlvbiA8IHN0ZXBUaHJvdWdocHV0OyBpdGVyYXRpb24rKykge1xuICAgICAgLy8gUmVzb2x2ZSBhZ2VudCBvcHRpb25zIGZyZXNoIGVhY2ggaXRlcmF0aW9uXG4gICAgICBjb25zdCByZXNvbHZlZE1vZGVsID0gYXdhaXQgcmVzb2x2ZU9wdGlvbihcbiAgICAgICAgYWdlbnQub3B0aW9ucy5tb2RlbCxcbiAgICAgICAgcmVzb2x2YWJsZUFyZ3NcbiAgICAgICk7XG4gICAgICBjb25zdCByZXNvbHZlZEdlbmVyYXRpb24gPSBhd2FpdCByZXNvbHZlT3B0aW9uKFxuICAgICAgICBhZ2VudC5vcHRpb25zLmdlbmVyYXRpb24sXG4gICAgICAgIHJlc29sdmFibGVBcmdzXG4gICAgICApO1xuICAgICAgY29uc3QgcmVzb2x2ZWRBY3RpdmVUb29scyA9IChhd2FpdCByZXNvbHZlT3B0aW9uKFxuICAgICAgICBhZ2VudC5vcHRpb25zLmFjdGl2ZVRvb2xzLFxuICAgICAgICByZXNvbHZhYmxlQXJnc1xuICAgICAgKSkgYXMgc3RyaW5nW10gfCB1bmRlZmluZWQ7XG5cbiAgICAgIGN1cnJlbnRNYXhTdGVwcyA9IHJlc29sdmVkR2VuZXJhdGlvbj8ubWF4U3RlcHM7XG5cbiAgICAgIGlmICghcmVzb2x2ZWRNb2RlbCkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXCJBZ2VudCBtb2RlbCBpcyBub3Qgc2V0XCIpO1xuICAgICAgfVxuXG4gICAgICBjb25zdCBzdGVwUGFydHM6IFVJTWVzc2FnZVtcInBhcnRzXCJdID0gW107XG4gICAgICBsZXQgc3RyZWFtVGV4dEZpbmlzaFJlYXNvbjogRmluaXNoUmVhc29uIHwgdW5kZWZpbmVkO1xuXG4gICAgICB0cnkge1xuICAgICAgICBjb25zdCBzdHJlYW0gPSBjcmVhdGVVSU1lc3NhZ2VTdHJlYW0oe1xuICAgICAgICAgIGV4ZWN1dGU6IGFzeW5jICh7IHdyaXRlciB9KSA9PiB7XG4gICAgICAgICAgICBjb25zdCB7IHJhd1Rvb2xzLCB0b29sc1dpdGhOZWVkc0FwcHJvdmFsLCB0b29sQ29udGV4dCB9ID0gZ2V0VG9vbHMoe1xuICAgICAgICAgICAgICBhZ2VudCxcbiAgICAgICAgICAgICAgc2FuZGJveCxcbiAgICAgICAgICAgICAgc2tpbGxzUmVmLFxuICAgICAgICAgICAgICB3cml0ZXIsXG4gICAgICAgICAgICAgIGFib3J0Q29udHJvbGxlcixcbiAgICAgICAgICAgICAgbmV4dFBhcnRJbmRleFJlZixcbiAgICAgICAgICAgICAgYXNzaXN0YW50TWVzc2FnZUlkLFxuICAgICAgICAgICAgICBzZXNzaW9uSWQsXG4gICAgICAgICAgICAgIHN0ZXBMb2csXG4gICAgICAgICAgICAgIGNvbnRleHQsXG4gICAgICAgICAgICAgIHN0YXRlLFxuICAgICAgICAgICAgICBtZXNzYWdlczogdWlNZXNzYWdlcyxcbiAgICAgICAgICAgICAgYWN0aXZlVG9vbHM6IHJlc29sdmVkQWN0aXZlVG9vbHMsXG4gICAgICAgICAgICB9KTtcblxuICAgICAgICAgICAgLy8gcHJvY2VzcyBhcHByb3ZhbHMgb25jZSAoZmlyc3QgaXRlcmF0aW9uIG9ubHkpXG4gICAgICAgICAgICBpZiAoaXRlcmF0aW9uID09PSAwKSB7XG4gICAgICAgICAgICAgIGF3YWl0IHByb2Nlc3NBcHByb3ZhbHMoe1xuICAgICAgICAgICAgICAgIGFsbFBhcnRzLFxuICAgICAgICAgICAgICAgIG1lc3NhZ2VzUmVzdWx0LFxuICAgICAgICAgICAgICAgIGNyZWF0ZWRBdCxcbiAgICAgICAgICAgICAgICByYXdUb29scyxcbiAgICAgICAgICAgICAgICBhYm9ydENvbnRyb2xsZXIsXG4gICAgICAgICAgICAgICAgdG9vbENvbnRleHQsXG4gICAgICAgICAgICAgICAgYWdlbnQsXG4gICAgICAgICAgICAgICAgc3RlcExvZyxcbiAgICAgICAgICAgICAgICB0aHJvdHRsZWRFbWl0U3RhdHVzLFxuICAgICAgICAgICAgICAgIHdyaXRlcixcbiAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIHRocm90dGxlZEVtaXRTdGF0dXMoe1xuICAgICAgICAgICAgICB3cml0ZXIsXG4gICAgICAgICAgICAgIHN0YXR1czogeyB0eXBlOiBcInRoaW5raW5nXCIgfSxcbiAgICAgICAgICAgIH0pO1xuXG4gICAgICAgICAgICBjb25zdCBpdGVyUmVzdWx0ID0gc3RyZWFtVGV4dEFpKHtcbiAgICAgICAgICAgICAgbWVzc2FnZXM6IFt7IHJvbGU6IFwidXNlclwiIGFzIGNvbnN0LCBjb250ZW50OiBcIi5cIiB9XSxcbiAgICAgICAgICAgICAgdG9vbHM6IHRvb2xzV2l0aE5lZWRzQXBwcm92YWwsXG4gICAgICAgICAgICAgIG1vZGVsOiByZXNvbHZlZE1vZGVsLFxuICAgICAgICAgICAgICBhYm9ydFNpZ25hbDogYWJvcnRDb250cm9sbGVyLnNpZ25hbCxcbiAgICAgICAgICAgICAgc3RvcFdoZW46IHN0ZXBDb3VudElzKDEpLFxuICAgICAgICAgICAgICB0ZW1wZXJhdHVyZTogcmVzb2x2ZWRHZW5lcmF0aW9uPy50ZW1wZXJhdHVyZSxcbiAgICAgICAgICAgICAgdG9wSzogcmVzb2x2ZWRHZW5lcmF0aW9uPy50b3BLLFxuICAgICAgICAgICAgICB0b3BQOiByZXNvbHZlZEdlbmVyYXRpb24/LnRvcFAsXG4gICAgICAgICAgICAgIGZyZXF1ZW5jeVBlbmFsdHk6IHJlc29sdmVkR2VuZXJhdGlvbj8uZnJlcXVlbmN5UGVuYWx0eSxcbiAgICAgICAgICAgICAgcHJlc2VuY2VQZW5hbHR5OiByZXNvbHZlZEdlbmVyYXRpb24/LnByZXNlbmNlUGVuYWx0eSxcbiAgICAgICAgICAgICAgbWF4T3V0cHV0VG9rZW5zOiByZXNvbHZlZEdlbmVyYXRpb24/Lm1heE91dHB1dFRva2VucyxcbiAgICAgICAgICAgICAgaGVhZGVyczogcmVzb2x2ZWRHZW5lcmF0aW9uPy5oZWFkZXJzLFxuICAgICAgICAgICAgICBleHBlcmltZW50YWxfY29udGV4dDoge1xuICAgICAgICAgICAgICAgIGNvbnRleHQsXG4gICAgICAgICAgICAgICAgc3RhdGUsXG4gICAgICAgICAgICAgICAgc2Vzc2lvbklkLFxuICAgICAgICAgICAgICAgIHNhbmRib3gsXG4gICAgICAgICAgICAgICAgbWVzc2FnZXM6IHVpTWVzc2FnZXMsXG4gICAgICAgICAgICAgIH0gc2F0aXNmaWVzIFRvb2xDb250ZXh0LFxuICAgICAgICAgICAgICBwcmVwYXJlU3RlcDogYXN5bmMgKHsgbW9kZWwgfSkgPT4ge1xuICAgICAgICAgICAgICAgIGNvbnN0IHNraWxsc0NvbnRleHQgPSBidWlsZFNraWxsc0NvbnRleHQoc2tpbGxzUmVmLmN1cnJlbnQpO1xuICAgICAgICAgICAgICAgIGNvbnN0IGN3ZFByb21wdCA9IHNhbmRib3hSZWNvcmQ/LnNldHVwPy5jd2RcbiAgICAgICAgICAgICAgICAgID8gYFlvdXIgd29ya2luZyBkaXJlY3RvcnkgaXMgJHtzYW5kYm94UmVjb3JkLnNldHVwLmN3ZH0uIEFsbCBmaWxlIHBhdGhzIHNob3VsZCBiZSBhYnNvbHV0ZS5gXG4gICAgICAgICAgICAgICAgICA6IFwiQWxsIGZpbGUgcGF0aHMgbXVzdCBiZSBhYnNvbHV0ZS5cIjtcblxuICAgICAgICAgICAgICAgIGNvbnN0IHJlc29sdmVkU3lzdGVtID0gYXdhaXQgcmVzb2x2ZVN5c3RlbShcbiAgICAgICAgICAgICAgICAgIGFnZW50Lm9wdGlvbnMuc3lzdGVtLFxuICAgICAgICAgICAgICAgICAgcmVzb2x2YWJsZUFyZ3NcbiAgICAgICAgICAgICAgICApO1xuXG4gICAgICAgICAgICAgICAgY29uc3Qgc3lzdGVtSGVhZGVyID0gam9pblByb21wdFNlY3Rpb25zKFxuICAgICAgICAgICAgICAgICAgQkFTRV9TWVNURU1fUFJPTVBULFxuICAgICAgICAgICAgICAgICAgY3dkUHJvbXB0LFxuICAgICAgICAgICAgICAgICAgcmVzb2x2ZWRTeXN0ZW1cbiAgICAgICAgICAgICAgICApO1xuICAgICAgICAgICAgICAgIGNvbnN0IHN5c3RlbUNvbnRleHQgPSBqb2luUHJvbXB0U2VjdGlvbnMoXG4gICAgICAgICAgICAgICAgICBza2lsbHNDb250ZXh0LFxuICAgICAgICAgICAgICAgICAgY3dkUHJvbXB0XG4gICAgICAgICAgICAgICAgKTtcblxuICAgICAgICAgICAgICAgIGNvbnN0IGNvbnZlcnRlZE1lc3NhZ2VzID0gYXdhaXQgY29udmVydFRvTW9kZWxNZXNzYWdlcyhcbiAgICAgICAgICAgICAgICAgIHVpTWVzc2FnZXMsXG4gICAgICAgICAgICAgICAgICB7IGlnbm9yZUluY29tcGxldGVUb29sQ2FsbHM6IHRydWUgfVxuICAgICAgICAgICAgICAgICk7XG5cbiAgICAgICAgICAgICAgICBjb25zdCBtb2RlbE1lc3NhZ2VzID0gW1xuICAgICAgICAgICAgICAgICAgLi4uKHN5c3RlbUhlYWRlci50cmltKClcbiAgICAgICAgICAgICAgICAgICAgPyBbeyByb2xlOiBcInN5c3RlbVwiIGFzIGNvbnN0LCBjb250ZW50OiBzeXN0ZW1IZWFkZXIgfV1cbiAgICAgICAgICAgICAgICAgICAgOiBbXSksXG4gICAgICAgICAgICAgICAgICAuLi4oc3lzdGVtQ29udGV4dC50cmltKClcbiAgICAgICAgICAgICAgICAgICAgPyBbeyByb2xlOiBcInN5c3RlbVwiIGFzIGNvbnN0LCBjb250ZW50OiBzeXN0ZW1Db250ZXh0IH1dXG4gICAgICAgICAgICAgICAgICAgIDogW10pLFxuICAgICAgICAgICAgICAgICAgLi4uY29udmVydGVkTWVzc2FnZXMsXG4gICAgICAgICAgICAgICAgXTtcblxuICAgICAgICAgICAgICAgIGNvbnN0IHByb21wdENhY2hpbmcgPSBhcHBseVByb21wdENhY2hpbmdUb01vZGVsUmVxdWVzdCh7XG4gICAgICAgICAgICAgICAgICBtb2RlbDogdHlwZW9mIG1vZGVsID09PSBcInN0cmluZ1wiID8gbW9kZWwgOiBtb2RlbC5tb2RlbElkLFxuICAgICAgICAgICAgICAgICAgc2Vzc2lvbklkLFxuICAgICAgICAgICAgICAgICAgbWVzc2FnZXM6IG1vZGVsTWVzc2FnZXMsXG4gICAgICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgICAgICBsZXQgYWN0aXZlVG9vbHMgPSByZXNvbHZlZEFjdGl2ZVRvb2xzID8/IHVuZGVmaW5lZDtcbiAgICAgICAgICAgICAgICBpZiAoc2tpbGxzUmVmLmN1cnJlbnQubGVuZ3RoID09PSAwICYmIGFjdGl2ZVRvb2xzKSB7XG4gICAgICAgICAgICAgICAgICBhY3RpdmVUb29scyA9IGFjdGl2ZVRvb2xzLmZpbHRlcihcbiAgICAgICAgICAgICAgICAgICAgKHQpID0+IHQgIT09IChcIlNraWxsXCIgc2F0aXNmaWVzIEJ1aWx0SW5Ub29sTmFtZSlcbiAgICAgICAgICAgICAgICAgICk7XG4gICAgICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgICAgICAgIG1lc3NhZ2VzOiBwcm9tcHRDYWNoaW5nLm1lc3NhZ2VzLFxuICAgICAgICAgICAgICAgICAgcHJvdmlkZXJPcHRpb25zOiBtZXJnZVByb3ZpZGVyT3B0aW9ucyh7XG4gICAgICAgICAgICAgICAgICAgIGN1cnJlbnQ6IHJlc29sdmVkR2VuZXJhdGlvbj8ucHJvdmlkZXJPcHRpb25zLFxuICAgICAgICAgICAgICAgICAgICBwYXRjaDogcHJvbXB0Q2FjaGluZy5wcm92aWRlck9wdGlvbnMsXG4gICAgICAgICAgICAgICAgICB9KSxcbiAgICAgICAgICAgICAgICAgIGFjdGl2ZVRvb2xzLFxuICAgICAgICAgICAgICAgIH07XG4gICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgIG9uU3RlcEZpbmlzaDogKHsgdXNhZ2UgfSkgPT4ge1xuICAgICAgICAgICAgICAgIGlmICh1c2FnZSkge1xuICAgICAgICAgICAgICAgICAgdXNhZ2VTdGVwcy5wdXNoKHtcbiAgICAgICAgICAgICAgICAgICAgc3RlcEluZGV4OiBzdGVwSW5kZXhPZmZzZXQgKyBpbnRlcm5hbFN0ZXBJbmRleCxcbiAgICAgICAgICAgICAgICAgICAgbW9kZWw6IHJlc29sdmVkTW9kZWwgPz8gXCJ1bmtub3duXCIsXG4gICAgICAgICAgICAgICAgICAgIGlucHV0VG9rZW5zOiB1c2FnZS5pbnB1dFRva2VucyA/PyAwLFxuICAgICAgICAgICAgICAgICAgICBvdXRwdXRUb2tlbnM6IHVzYWdlLm91dHB1dFRva2VucyA/PyAwLFxuICAgICAgICAgICAgICAgICAgICB0b3RhbFRva2VuczogdXNhZ2UudG90YWxUb2tlbnMgPz8gMCxcbiAgICAgICAgICAgICAgICAgICAgY2FjaGVSZWFkVG9rZW5zOlxuICAgICAgICAgICAgICAgICAgICAgIHVzYWdlLmlucHV0VG9rZW5EZXRhaWxzPy5jYWNoZVJlYWRUb2tlbnMgPz8gMCxcbiAgICAgICAgICAgICAgICAgICAgY2FjaGVXcml0ZVRva2VuczpcbiAgICAgICAgICAgICAgICAgICAgICB1c2FnZS5pbnB1dFRva2VuRGV0YWlscz8uY2FjaGVXcml0ZVRva2VucyA/PyAwLFxuICAgICAgICAgICAgICAgICAgICByZWFzb25pbmdUb2tlbnM6XG4gICAgICAgICAgICAgICAgICAgICAgdXNhZ2Uub3V0cHV0VG9rZW5EZXRhaWxzPy5yZWFzb25pbmdUb2tlbnMgPz8gMCxcbiAgICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICBpbnRlcm5hbFN0ZXBJbmRleCsrO1xuICAgICAgICAgICAgICB9LFxuICAgICAgICAgICAgfSk7XG5cbiAgICAgICAgICAgIHdyaXRlci5tZXJnZShcbiAgICAgICAgICAgICAgaXRlclJlc3VsdC50b1VJTWVzc2FnZVN0cmVhbSh7XG4gICAgICAgICAgICAgICAgZ2VuZXJhdGVNZXNzYWdlSWQ6ICgpID0+IGFzc2lzdGFudE1lc3NhZ2VJZCxcbiAgICAgICAgICAgICAgICBvbkZpbmlzaDogKHsgbWVzc2FnZXMgfSkgPT4ge1xuICAgICAgICAgICAgICAgICAgZm9yIChjb25zdCBtIG9mIG1lc3NhZ2VzKSB7XG4gICAgICAgICAgICAgICAgICAgIGlmIChtLnJvbGUgPT09IFwiYXNzaXN0YW50XCIpIHtcbiAgICAgICAgICAgICAgICAgICAgICBzdGVwUGFydHMucHVzaCguLi5tLnBhcnRzKTtcbiAgICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgICAgIH0sXG4gICAgICAgICAgICAgIH0pXG4gICAgICAgICAgICApO1xuXG4gICAgICAgICAgICBzdHJlYW1UZXh0RmluaXNoUmVhc29uID0gYXdhaXQgaXRlclJlc3VsdC5maW5pc2hSZWFzb247XG4gICAgICAgICAgfSxcbiAgICAgICAgfSk7XG5cbiAgICAgICAgYXdhaXQgc3RyZWFtLnBpcGVUbyh3cml0YWJsZSwge1xuICAgICAgICAgIHByZXZlbnRDbG9zZTogdHJ1ZSxcbiAgICAgICAgICBwcmV2ZW50QWJvcnQ6IHRydWUsXG4gICAgICAgIH0pO1xuICAgICAgfSBjYXRjaCAoZXJyKSB7XG4gICAgICAgIGlmIChhYm9ydENvbnRyb2xsZXIuc2lnbmFsLmFib3J0ZWQpIHtcbiAgICAgICAgICB3YXNBYm9ydGVkID0gdHJ1ZTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICB0aHJvdyBlcnI7XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgaWYgKGFib3J0Q29udHJvbGxlci5zaWduYWwuYWJvcnRlZCkge1xuICAgICAgICB3YXNBYm9ydGVkID0gdHJ1ZTtcbiAgICAgIH1cblxuICAgICAgLy8gTWFyayBpbnRlcnJ1cHRlZCB0b29sIHBhcnRzXG4gICAgICBpZiAod2FzQWJvcnRlZCkge1xuICAgICAgICBjb25zdCB0ZXJtaW5hbFN0YXRlcyA9IG5ldyBTZXQoW1xuICAgICAgICAgIFwib3V0cHV0LWF2YWlsYWJsZVwiLFxuICAgICAgICAgIFwib3V0cHV0LWVycm9yXCIsXG4gICAgICAgICAgXCJvdXRwdXQtZGVuaWVkXCIsXG4gICAgICAgICAgXCJkb25lXCIsXG4gICAgICAgIF0pO1xuICAgICAgICBmb3IgKGNvbnN0IHBhcnQgb2Ygc3RlcFBhcnRzKSB7XG4gICAgICAgICAgaWYgKFxuICAgICAgICAgICAgXCJ0eXBlXCIgaW4gcGFydCAmJlxuICAgICAgICAgICAgdHlwZW9mIHBhcnQudHlwZSA9PT0gXCJzdHJpbmdcIiAmJlxuICAgICAgICAgICAgcGFydC50eXBlLnN0YXJ0c1dpdGgoXCJ0b29sLVwiKSAmJlxuICAgICAgICAgICAgXCJzdGF0ZVwiIGluIHBhcnQgJiZcbiAgICAgICAgICAgICF0ZXJtaW5hbFN0YXRlcy5oYXMocGFydC5zdGF0ZSBhcyBzdHJpbmcpXG4gICAgICAgICAgKSB7XG4gICAgICAgICAgICAocGFydCBhcyB7IHN0YXRlOiBzdHJpbmcgfSkuc3RhdGUgPSBcIm91dHB1dC1lcnJvclwiO1xuICAgICAgICAgICAgKHBhcnQgYXMgeyBlcnJvclRleHQ/OiBzdHJpbmcgfSkuZXJyb3JUZXh0ID0gXCJpbnRlcnJ1cHRlZFwiO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICAvLyBIYW5kbGUgaW50ZXJydXB0ZWQgbGFzdFBhcnQgdHJ1bmNhdGlvblxuICAgICAgY29uc3QgbGFzdFBhcnQgPSBpbnRlcnJ1cHRlZExhc3RQYXJ0UmVmLmN1cnJlbnQ7XG5cbiAgICAgIGNvbnN0IG5ld1BhcnRSZWNvcmRzID0gc3RlcFBhcnRzXG4gICAgICAgIC5tYXAoKHVpUGFydCwgaSkgPT4ge1xuICAgICAgICAgIGNvbnN0IGluZGV4ID0gbmV4dFBhcnRJbmRleFJlZi5jdXJyZW50ICsgaTtcbiAgICAgICAgICBpZiAobGFzdFBhcnQgIT0gbnVsbCAmJiBpbmRleCA+IGxhc3RQYXJ0LmluZGV4KSB7XG4gICAgICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgICAgICB9XG4gICAgICAgICAgY29uc3QgdXNlQ2xpZW50UGFydCA9IGxhc3RQYXJ0ICE9IG51bGwgJiYgaW5kZXggPT09IGxhc3RQYXJ0LmluZGV4O1xuICAgICAgICAgIGNvbnN0IGNvbnRlbnQgPSAoXG4gICAgICAgICAgICB1c2VDbGllbnRQYXJ0ID8gbGFzdFBhcnQucGFydCA6IHVpUGFydFxuICAgICAgICAgICkgYXMgVHlwZWRVSU1lc3NhZ2U8VG9vbHM+W1wicGFydHNcIl1bbnVtYmVyXTtcbiAgICAgICAgICBjb25zdCBwYXJ0SWQgPVxuICAgICAgICAgICAgXCJhcHByb3ZhbFwiIGluIGNvbnRlbnQgJiYgY29udGVudC5hcHByb3ZhbD8uaWRcbiAgICAgICAgICAgICAgPyBhcHByb3ZhbFBhcnRJZChjb250ZW50LmFwcHJvdmFsLmlkKVxuICAgICAgICAgICAgICA6IGBwYXJ0XyR7YXNzaXN0YW50TWVzc2FnZUlkfV8ke2luZGV4fWA7XG4gICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgIGlkOiBwYXJ0SWQsXG4gICAgICAgICAgICBpbmRleCxcbiAgICAgICAgICAgIG1lc3NhZ2VJZDogYXNzaXN0YW50TWVzc2FnZUlkLFxuICAgICAgICAgICAgc2Vzc2lvbklkLFxuICAgICAgICAgICAgcGFydDogY29udGVudCxcbiAgICAgICAgICB9O1xuICAgICAgICB9KVxuICAgICAgICAuZmlsdGVyKChyKTogciBpcyBOb25OdWxsYWJsZTx0eXBlb2Ygcj4gPT4gciAhPT0gbnVsbCk7XG5cbiAgICAgIGNvbnN0IHBlcnNpc3RQcm9taXNlID0gUHJvbWlzZS5hbGwoW1xuICAgICAgICAuLi5uZXdQYXJ0UmVjb3Jkcy5tYXAoKHJlY29yZCkgPT5cbiAgICAgICAgICBhZ2VudC5zdG9yYWdlLnBhcnQuc2V0KHJlY29yZC5pZCwgcmVjb3JkKVxuICAgICAgICApLFxuICAgICAgICBhZ2VudC5zdG9yYWdlLnNlc3Npb24udXBkYXRlKHNlc3Npb25JZCwgeyBzdGF0ZSB9KSxcbiAgICAgIF0pO1xuXG4gICAgICBuZXh0UGFydEluZGV4UmVmLmN1cnJlbnQgPVxuICAgICAgICBsYXN0UGFydCAhPSBudWxsXG4gICAgICAgICAgPyBNYXRoLm1pbihcbiAgICAgICAgICAgICAgbmV4dFBhcnRJbmRleFJlZi5jdXJyZW50ICsgc3RlcFBhcnRzLmxlbmd0aCxcbiAgICAgICAgICAgICAgbGFzdFBhcnQuaW5kZXggKyAxXG4gICAgICAgICAgICApXG4gICAgICAgICAgOiBuZXh0UGFydEluZGV4UmVmLmN1cnJlbnQgKyBzdGVwUGFydHMubGVuZ3RoO1xuXG4gICAgICBjb25zdCBpdGVyUGVuZGluZ0FwcHJvdmFscyA9IHN0ZXBQYXJ0c1xuICAgICAgICAuZmlsdGVyKFxuICAgICAgICAgIChwKTogcCBpcyB0eXBlb2YgcCAmIHsgYXBwcm92YWw6IHsgaWQ6IHN0cmluZyB9IH0gPT5cbiAgICAgICAgICAgIFwic3RhdGVcIiBpbiBwICYmXG4gICAgICAgICAgICBwLnN0YXRlID09PSBcImFwcHJvdmFsLXJlcXVlc3RlZFwiICYmXG4gICAgICAgICAgICBcImFwcHJvdmFsXCIgaW4gcCAmJlxuICAgICAgICAgICAgISEocCBhcyB7IGFwcHJvdmFsPzogeyBpZDogc3RyaW5nIH0gfSkuYXBwcm92YWw/LmlkXG4gICAgICAgIClcbiAgICAgICAgLm1hcCgocCkgPT4gKHtcbiAgICAgICAgICBhcHByb3ZhbElkOiBwLmFwcHJvdmFsLmlkLFxuICAgICAgICAgIHRvb2xOYW1lOlxuICAgICAgICAgICAgXCJ0eXBlXCIgaW4gcCA/IFN0cmluZyhwLnR5cGUpLnJlcGxhY2UoXCJ0b29sLVwiLCBcIlwiKSA6IFwidW5rbm93blwiLFxuICAgICAgICB9KSk7XG5cbiAgICAgIGxvb3BGaW5pc2hSZWFzb24gPSB3YXNBYm9ydGVkXG4gICAgICAgID8gKFwic3RvcFwiIGFzIEZpbmlzaFJlYXNvbilcbiAgICAgICAgOiAoc3RyZWFtVGV4dEZpbmlzaFJlYXNvbiA/PyAoXCJ1bmtub3duXCIgYXMgRmluaXNoUmVhc29uKSk7XG5cbiAgICAgIC8vIERlY2lkZSB3aGV0aGVyIHRvIGJyZWFrIG9yIGNvbnRpbnVlXG4gICAgICBjb25zdCBzaG91bGRCcmVhayA9XG4gICAgICAgIGl0ZXJQZW5kaW5nQXBwcm92YWxzLmxlbmd0aCA+IDAgfHxcbiAgICAgICAgbG9vcEZpbmlzaFJlYXNvbiA9PT0gXCJzdG9wXCIgfHxcbiAgICAgICAgd2FzQWJvcnRlZCB8fFxuICAgICAgICBsYXN0UGFydCAhPSBudWxsIHx8XG4gICAgICAgIChjdXJyZW50TWF4U3RlcHMgIT0gbnVsbCAmJlxuICAgICAgICAgIHN0ZXBJbmRleE9mZnNldCArIHVzYWdlU3RlcHMubGVuZ3RoID49IGN1cnJlbnRNYXhTdGVwcykgfHxcbiAgICAgICAgaXRlcmF0aW9uID49IHN0ZXBUaHJvdWdocHV0IC0gMTtcblxuICAgICAgaWYgKHNob3VsZEJyZWFrKSB7XG4gICAgICAgIGF3YWl0IHBlcnNpc3RQcm9taXNlO1xuICAgICAgICBhbGxQYXJ0cy5wdXNoKC4uLm5ld1BhcnRSZWNvcmRzKTtcbiAgICAgICAgYWxsUGVuZGluZ0FwcHJvdmFscy5wdXNoKC4uLml0ZXJQZW5kaW5nQXBwcm92YWxzKTtcbiAgICAgICAgYnJlYWs7XG4gICAgICB9XG5cbiAgICAgIC8vIE5vbi1maW5hbCBpdGVyYXRpb246IHBlcnNpc3QgcGFydHMgYW5kIHJlYnVpbGQgbWVzc2FnZXNcbiAgICAgIGF3YWl0IHBlcnNpc3RQcm9taXNlO1xuICAgICAgYWxsUGFydHMucHVzaCguLi5uZXdQYXJ0UmVjb3Jkcyk7XG4gICAgICB1aU1lc3NhZ2VzID0gYXNzZW1ibGVVSU1lc3NhZ2VzKHtcbiAgICAgICAgbWVzc2FnZXM6IG1lc3NhZ2VzUmVzdWx0LFxuICAgICAgICBwYXJ0czogYWxsUGFydHMsXG4gICAgICAgIHVudGlsOiBjcmVhdGVkQXQsXG4gICAgICAgIGluY2x1ZGVRdWV1ZWQ6IHRydWUsXG4gICAgICAgIGV4Y2x1ZGVTdWJUb29sUGFydHM6IHRydWUsXG4gICAgICB9KTtcbiAgICB9XG4gIH0gZmluYWxseSB7XG4gICAgcG9sbENvbnRyb2xsZXIuYWJvcnQoKTtcbiAgICB0aHJvdHRsZWRFbWl0U3RhdHVzLmRpc3Bvc2UoKTtcbiAgfVxuXG4gIGlmIChzZXRTdGFydGVkUHJvbWlzZSkge1xuICAgIGF3YWl0IHNldFN0YXJ0ZWRQcm9taXNlO1xuICB9XG5cbiAgZG9uZVN0ZXAoe1xuICAgIGZpbmlzaFJlYXNvbjogbG9vcEZpbmlzaFJlYXNvbixcbiAgICB3YXNBYm9ydGVkLFxuICAgIGl0ZXJhdGlvbnM6IGludGVybmFsU3RlcEluZGV4LFxuICAgIHBlbmRpbmdBcHByb3ZhbHM6IGFsbFBlbmRpbmdBcHByb3ZhbHMubGVuZ3RoLFxuICB9KTtcblxuICByZXR1cm4ge1xuICAgIGZpbmlzaFJlYXNvbjogbG9vcEZpbmlzaFJlYXNvbixcbiAgICBsYXN0UGFydEluZGV4OiBuZXh0UGFydEluZGV4UmVmLmN1cnJlbnQsXG4gICAgdXNhZ2VTdGVwcyxcbiAgICBwZW5kaW5nQXBwcm92YWxzOiBhbGxQZW5kaW5nQXBwcm92YWxzLFxuICAgIG1heFN0ZXBzOiBjdXJyZW50TWF4U3RlcHMsXG4gICAgZGlzY292ZXJlZFNraWxsczogc2tpbGxzUmVmLmN1cnJlbnQsXG4gICAgY29udGV4dCxcbiAgfTtcbn1cblxuLyogLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLVxuICogU3RlcHNcbiAqIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi9cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIHN0cmVhbVRleHRTdGVwPFRvb2xzIGV4dGVuZHMgVG9vbFNldD4oXG4gIC4uLmFyZ3M6IFBhcmFtZXRlcnM8dHlwZW9mIHN0cmVhbVRleHQ8VG9vbHM+PlxuKSB7XG4gIFwidXNlIHN0ZXBcIjtcbiAgcmV0dXJuIGF3YWl0IHN0cmVhbVRleHQoLi4uYXJncyk7XG59XG5cbi8qIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1cbiAqIFV0aWxzXG4gKiAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSovXG5cbmFzeW5jIGZ1bmN0aW9uIHByb2Nlc3NBcHByb3ZhbHMoe1xuICBhbGxQYXJ0cyxcbiAgbWVzc2FnZXNSZXN1bHQsXG4gIGNyZWF0ZWRBdCxcbiAgcmF3VG9vbHMsXG4gIGFib3J0Q29udHJvbGxlcixcbiAgdG9vbENvbnRleHQsXG4gIGFnZW50LFxuICBzdGVwTG9nLFxuICB0aHJvdHRsZWRFbWl0U3RhdHVzLFxuICB3cml0ZXIsXG59OiB7XG4gIGFsbFBhcnRzOiBQYXJ0W107XG4gIG1lc3NhZ2VzUmVzdWx0OiBQYXJhbWV0ZXJzPHR5cGVvZiBhc3NlbWJsZVVJTWVzc2FnZXM+WzBdW1wibWVzc2FnZXNcIl07XG4gIGNyZWF0ZWRBdDogbnVtYmVyO1xuICByYXdUb29sczogVG9vbFNldDtcbiAgYWJvcnRDb250cm9sbGVyOiBBYm9ydENvbnRyb2xsZXI7XG4gIHRvb2xDb250ZXh0OiBUb29sQ29udGV4dDtcbiAgYWdlbnQ6IEFueUFnZW50O1xuICBzdGVwTG9nOiBMb2dnZXI7XG4gIHRocm90dGxlZEVtaXRTdGF0dXM6IFJldHVyblR5cGU8dHlwZW9mIGNyZWF0ZVRocm90dGxlZEVtaXRTdGF0dXM+O1xuICB3cml0ZXI6IFN0cmVhbVdyaXRlcjtcbn0pIHtcbiAgLy8gRXhlY3V0ZSBhcHByb3ZlZCB0b29scyBzbyB0aGUgbW9kZWwgc2VlcyBjb21wbGV0ZSB0b29sIGNhbGwgKyByZXN1bHQgcGFpcnNcbiAgY29uc3QgYXBwcm92ZWRQYXJ0cyA9IGFsbFBhcnRzLmZpbHRlcihcbiAgICAocCkgPT5cbiAgICAgIHAucGFydCAmJlxuICAgICAgXCJzdGF0ZVwiIGluIHAucGFydCAmJlxuICAgICAgKHAucGFydCBhcyB7IHN0YXRlOiBzdHJpbmcgfSkuc3RhdGUgPT09IFwiYXBwcm92YWwtcmVzcG9uZGVkXCIgJiZcbiAgICAgIFwiYXBwcm92YWxcIiBpbiBwLnBhcnQgJiZcbiAgICAgIChwLnBhcnQgYXMgeyBhcHByb3ZhbD86IHsgYXBwcm92ZWQ6IGJvb2xlYW4gfSB9KS5hcHByb3ZhbD8uYXBwcm92ZWQgPT09XG4gICAgICAgIHRydWVcbiAgKTtcblxuICBpZiAoYXBwcm92ZWRQYXJ0cy5sZW5ndGggPiAwKSB7XG4gICAgc3RlcExvZy5pbmZvKFwiZXhlY3V0aW5nIGFwcHJvdmVkIHRvb2xzXCIsIHtcbiAgICAgIGNvdW50OiBhcHByb3ZlZFBhcnRzLmxlbmd0aCxcbiAgICB9KTtcbiAgICB0aHJvdHRsZWRFbWl0U3RhdHVzKHtcbiAgICAgIHdyaXRlcixcbiAgICAgIHN0YXR1czogeyB0eXBlOiBcInByb2Nlc3NpbmctYXBwcm92YWxzXCIgfSxcbiAgICB9KTtcblxuICAgIGNvbnN0IHByZUV4ZWNNZXNzYWdlcyA9IGF3YWl0IGNvbnZlcnRUb01vZGVsTWVzc2FnZXMoXG4gICAgICBhc3NlbWJsZVVJTWVzc2FnZXMoe1xuICAgICAgICBtZXNzYWdlczogbWVzc2FnZXNSZXN1bHQsXG4gICAgICAgIHBhcnRzOiBhbGxQYXJ0cyxcbiAgICAgICAgdW50aWw6IGNyZWF0ZWRBdCxcbiAgICAgICAgaW5jbHVkZVF1ZXVlZDogdHJ1ZSxcbiAgICAgICAgZXhjbHVkZVN1YlRvb2xQYXJ0czogdHJ1ZSxcbiAgICAgIH0pLFxuICAgICAgeyBpZ25vcmVJbmNvbXBsZXRlVG9vbENhbGxzOiB0cnVlIH1cbiAgICApO1xuXG4gICAgYXdhaXQgUHJvbWlzZS5hbGwoXG4gICAgICBhcHByb3ZlZFBhcnRzLm1hcChhc3luYyAoYXApID0+IHtcbiAgICAgICAgaWYgKCFhcC5wYXJ0LnR5cGUuc3RhcnRzV2l0aChcInRvb2wtXCIpKSB7XG4gICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9XG4gICAgICAgIGNvbnN0IHBhcnQgPSBhcC5wYXJ0IGFzIEV4dHJhY3Q8XG4gICAgICAgICAgVUlNZXNzYWdlW1wicGFydHNcIl1bbnVtYmVyXSxcbiAgICAgICAgICB7IHR5cGU6IGB0b29sLSR7c3RyaW5nfWAgfVxuICAgICAgICA+O1xuICAgICAgICBjb25zdCB0b29sTmFtZSA9IHBhcnQudHlwZS5yZXBsYWNlKFwidG9vbC1cIiwgXCJcIik7XG4gICAgICAgIGNvbnN0IHRvb2xEZWYgPSByYXdUb29sc1t0b29sTmFtZV07XG5cbiAgICAgICAgaWYgKHRvb2xEZWY/LmV4ZWN1dGUgJiYgcGFydC5pbnB1dCAhPT0gdW5kZWZpbmVkKSB7XG4gICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIGNvbnN0IHRvb2xPdXRwdXQgPSBhd2FpdCB0b29sRGVmLmV4ZWN1dGUocGFydC5pbnB1dCwge1xuICAgICAgICAgICAgICB0b29sQ2FsbElkOiBwYXJ0LnRvb2xDYWxsSWQsXG4gICAgICAgICAgICAgIG1lc3NhZ2VzOiBwcmVFeGVjTWVzc2FnZXMsXG4gICAgICAgICAgICAgIGFib3J0U2lnbmFsOiBhYm9ydENvbnRyb2xsZXIuc2lnbmFsLFxuICAgICAgICAgICAgICBleHBlcmltZW50YWxfY29udGV4dDogdG9vbENvbnRleHQsXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICAgIHBhcnQuc3RhdGUgPSBcIm91dHB1dC1hdmFpbGFibGVcIjtcbiAgICAgICAgICAgIHBhcnQub3V0cHV0ID0gdG9vbE91dHB1dDtcbiAgICAgICAgICAgIHdyaXRlci53cml0ZSh7XG4gICAgICAgICAgICAgIHR5cGU6IFwidG9vbC1vdXRwdXQtYXZhaWxhYmxlXCIsXG4gICAgICAgICAgICAgIHRvb2xDYWxsSWQ6IHBhcnQudG9vbENhbGxJZCxcbiAgICAgICAgICAgICAgb3V0cHV0OiB0b29sT3V0cHV0LFxuICAgICAgICAgICAgfSk7XG4gICAgICAgICAgfSBjYXRjaCAoZXJyKSB7XG4gICAgICAgICAgICBwYXJ0LnN0YXRlID0gXCJvdXRwdXQtZXJyb3JcIjtcbiAgICAgICAgICAgIHBhcnQuZXJyb3JUZXh0ID0gZXJyIGluc3RhbmNlb2YgRXJyb3IgPyBlcnIubWVzc2FnZSA6IFN0cmluZyhlcnIpO1xuICAgICAgICAgICAgd3JpdGVyLndyaXRlKHtcbiAgICAgICAgICAgICAgdHlwZTogXCJ0b29sLW91dHB1dC1lcnJvclwiLFxuICAgICAgICAgICAgICB0b29sQ2FsbElkOiBwYXJ0LnRvb2xDYWxsSWQsXG4gICAgICAgICAgICAgIGVycm9yVGV4dDogcGFydC5lcnJvclRleHQsXG4gICAgICAgICAgICB9KTtcbiAgICAgICAgICB9XG4gICAgICAgICAgYXdhaXQgYWdlbnQuc3RvcmFnZS5wYXJ0LnNldChhcC5pZCwgeyAuLi5hcCwgcGFydCB9KTtcbiAgICAgICAgfVxuICAgICAgfSlcbiAgICApO1xuICB9XG5cbiAgLy8gQXV0by1kZW55IGFueSByZW1haW5pbmcgYXBwcm92YWwtcmVxdWVzdGVkIHBhcnRzIHNvIHRoZSBtb2RlbFxuICAvLyBzZWVzIGNvbXBsZXRlIHRvb2wgY2FsbCArIGRlbmllZCByZXN1bHQgcGFpcnMgKG5vdCBtaXNzaW5nIHJlc3VsdHMpLlxuICBjb25zdCBwZW5kaW5nQXBwcm92YWxQYXJ0cyA9IGFsbFBhcnRzLmZpbHRlcihcbiAgICAocCkgPT4gKHAucGFydCBhcyB7IHN0YXRlPzogc3RyaW5nIH0pLnN0YXRlID09PSBcImFwcHJvdmFsLXJlcXVlc3RlZFwiXG4gICk7XG4gIGlmIChwZW5kaW5nQXBwcm92YWxQYXJ0cy5sZW5ndGggPiAwKSB7XG4gICAgYXdhaXQgUHJvbWlzZS5hbGwoXG4gICAgICBwZW5kaW5nQXBwcm92YWxQYXJ0cy5tYXAoYXN5bmMgKHApID0+IHtcbiAgICAgICAgY29uc3QgZGVuaWVkID0ge1xuICAgICAgICAgIC4uLnAsXG4gICAgICAgICAgcGFydDoge1xuICAgICAgICAgICAgLi4ucC5wYXJ0LFxuICAgICAgICAgICAgc3RhdGU6IFwib3V0cHV0LWRlbmllZFwiLFxuICAgICAgICAgICAgYXBwcm92YWw6IHtcbiAgICAgICAgICAgICAgLi4uKChwLnBhcnQgYXMgeyBhcHByb3ZhbD86IG9iamVjdCB9KS5hcHByb3ZhbCA/PyB7fSksXG4gICAgICAgICAgICAgIGFwcHJvdmVkOiBmYWxzZSxcbiAgICAgICAgICAgICAgcmVhc29uOiBcImF1dG8tZGVuaWVkOiBuZXcgbWVzc2FnZSBzZW50XCIsXG4gICAgICAgICAgICB9LFxuICAgICAgICAgIH0gYXMgdHlwZW9mIHAucGFydCxcbiAgICAgICAgfTtcbiAgICAgICAgT2JqZWN0LmFzc2lnbihwLCBkZW5pZWQpO1xuICAgICAgICBhd2FpdCBhZ2VudC5zdG9yYWdlLnBhcnQuc2V0KHAuaWQsIGRlbmllZCk7XG4gICAgICB9KVxuICAgICk7XG4gIH1cbn1cblxudHlwZSBTdHJlYW1Xcml0ZXIgPSBVSU1lc3NhZ2VTdHJlYW1Xcml0ZXI7XG5cbmZ1bmN0aW9uIGVtaXRTdGF0dXMoe1xuICB3cml0ZXIsXG4gIHN0YXR1cyxcbiAgYWdlbnQsXG59OiB7XG4gIHdyaXRlcjogU3RyZWFtV3JpdGVyO1xuICBzdGF0dXM6IEFnZW50U3RhdHVzO1xuICBhZ2VudDogQW55QWdlbnQ7XG59KSB7XG4gIHdyaXRlci53cml0ZSh7IHR5cGU6IFwiZGF0YS1zdGF0dXNcIiwgZGF0YTogc3RhdHVzLCB0cmFuc2llbnQ6IHRydWUgfSk7XG4gIGFnZW50Lmhvb2tzPy5zdGF0dXM/LihzdGF0dXMpO1xufVxuXG5jb25zdCBTVEFUVVNfVEhST1RUTEVfTVMgPSA1MDA7XG5cbmZ1bmN0aW9uIGNyZWF0ZVRocm90dGxlZEVtaXRTdGF0dXMoYWdlbnQ6IEFueUFnZW50KSB7XG4gIGxldCB0aW1lcjogUmV0dXJuVHlwZTx0eXBlb2Ygc2V0VGltZW91dD4gfCBudWxsID0gbnVsbDtcbiAgbGV0IHBlbmRpbmc6IHtcbiAgICB3cml0ZXI6IFN0cmVhbVdyaXRlcjtcbiAgICBzdGF0dXM6IEFnZW50U3RhdHVzO1xuICB9IHwgbnVsbCA9IG51bGw7XG5cbiAgZnVuY3Rpb24gdGhyb3R0bGVkKHtcbiAgICB3cml0ZXIsXG4gICAgc3RhdHVzLFxuICB9OiB7XG4gICAgd3JpdGVyOiBTdHJlYW1Xcml0ZXI7XG4gICAgc3RhdHVzOiBBZ2VudFN0YXR1cztcbiAgfSkge1xuICAgIHBlbmRpbmcgPSB7IHdyaXRlciwgc3RhdHVzIH07XG4gICAgaWYgKHRpbWVyICE9PSBudWxsKSB7XG4gICAgICByZXR1cm47XG4gICAgfVxuICAgIGZsdXNoKCk7XG4gICAgdGltZXIgPSBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgIHRpbWVyID0gbnVsbDtcbiAgICAgIGlmIChwZW5kaW5nKSB7XG4gICAgICAgIGZsdXNoKCk7XG4gICAgICB9XG4gICAgfSwgU1RBVFVTX1RIUk9UVExFX01TKTtcbiAgfVxuXG4gIGZ1bmN0aW9uIGZsdXNoKCkge1xuICAgIGlmICghcGVuZGluZykge1xuICAgICAgcmV0dXJuO1xuICAgIH1cbiAgICBjb25zdCB7IHdyaXRlciwgc3RhdHVzIH0gPSBwZW5kaW5nO1xuICAgIHBlbmRpbmcgPSBudWxsO1xuICAgIGVtaXRTdGF0dXMoeyB3cml0ZXIsIHN0YXR1cywgYWdlbnQgfSk7XG4gIH1cblxuICB0aHJvdHRsZWQuZGlzcG9zZSA9ICgpID0+IHtcbiAgICBpZiAodGltZXIgIT09IG51bGwpIHtcbiAgICAgIGNsZWFyVGltZW91dCh0aW1lcik7XG4gICAgICB0aW1lciA9IG51bGw7XG4gICAgfVxuICAgIGZsdXNoKCk7XG4gIH07XG5cbiAgcmV0dXJuIHRocm90dGxlZDtcbn1cblxuYXN5bmMgZnVuY3Rpb24gcmVzb2x2ZU9wdGlvbjxUPihcbiAgdmFsdWU6IFJlc29sdmFibGU8VD4gfCB1bmRlZmluZWQsXG4gIGFyZ3M6IFJlc29sdmFibGVBcmdzXG4pOiBQcm9taXNlPFQgfCB1bmRlZmluZWQ+IHtcbiAgaWYgKHZhbHVlID09PSB1bmRlZmluZWQpIHtcbiAgICByZXR1cm4gdW5kZWZpbmVkO1xuICB9XG4gIGlmICh0eXBlb2YgdmFsdWUgPT09IFwiZnVuY3Rpb25cIikge1xuICAgIHJldHVybiBhd2FpdCAodmFsdWUgYXMgKGFyZ3M6IFJlc29sdmFibGVBcmdzKSA9PiBUIHwgUHJvbWlzZTxUPikoYXJncyk7XG4gIH1cbiAgcmV0dXJuIHZhbHVlO1xufVxuXG5hc3luYyBmdW5jdGlvbiByZXNvbHZlU3lzdGVtKFxuICBpbnB1dDogUmVzb2x2YWJsZTxzdHJpbmcgfCBzdHJpbmdbXSB8IHVuZGVmaW5lZD4gfCB1bmRlZmluZWQsXG4gIGFyZ3M6IFJlc29sdmFibGVBcmdzXG4pOiBQcm9taXNlPHN0cmluZyB8IG51bGw+IHtcbiAgY29uc3QgcmVzdWx0ID0gYXdhaXQgcmVzb2x2ZU9wdGlvbihpbnB1dCwgYXJncyk7XG4gIGlmICghcmVzdWx0KSB7XG4gICAgcmV0dXJuIG51bGw7XG4gIH1cbiAgcmV0dXJuIEFycmF5LmlzQXJyYXkocmVzdWx0KSA/IHJlc3VsdC5maWx0ZXIoQm9vbGVhbikuam9pbihcIlxcblwiKSA6IHJlc3VsdDtcbn1cblxuY29uc3QgQkFTRV9TWVNURU1fUFJPTVBUID1cbiAgXCJZb3UgYXJlIGFuIEFJIGFzc2lzdGFudCB3aXRoIGJhc2ljIHRvb2xzIHRvIGludGVyYWN0IHdpdGggeW91ciBlbnZpcm9ubWVudC4gRXhwbG9yZSBhbmQgd29yayBmcmVlbHkuXCI7XG5cbmZ1bmN0aW9uIGpvaW5Qcm9tcHRTZWN0aW9ucyhcbiAgLi4uc2VjdGlvbnM6IChzdHJpbmcgfCB1bmRlZmluZWQgfCBudWxsKVtdXG4pOiBzdHJpbmcge1xuICByZXR1cm4gc2VjdGlvbnMuZmlsdGVyKChzKSA9PiBzPy50cmltKCkpLmpvaW4oXCJcXG5cXG5cIik7XG59XG5cbmNvbnN0IGJhY2tncm91bmRQcm9jZXNzUHJvbXB0ID0gYCMjIEJhY2tncm91bmQgUHJvY2Vzc2VzXG5UaGVzZSBiYWNrZ3JvdW5kIHByb2Nlc3MgaW5zdHJ1Y3Rpb25zIGFyZSBmb3IgeW91IHRvIG1hbmlwdWxhdGUgdGhlIHByb2Nlc3NlcywgZG8gbm90IGJlIHRvIHZlcmJvc2UgdG8gdGhlIHVzZXIgYWJvdXQgdGhlIHJlc3BvbnNlIGRldGFpbHMgbGlrZSBcImhvdyB0byBkZWJ1ZyB0aGUgcHJvY2VzcyBvdXRwdXRcIiB0aGUgdXNlciB3aWxsIGhhdmUgYW4gVUkuXG5cblVzZSBcXGB3YWl0VW50aWw6IDBcXGAgZm9yIGRldiBzZXJ2ZXJzIHRoYXQgc2hvdWxkIHJ1biBpbmRlZmluaXRlbHkuXG5JdCBpcyBhIGdvb2QgcHJhY3RpY2UgdG8gY2hlY2sgdGhlIG91dHB1dCBsb2cgYWZ0ZXIgcnVubmluZyBwcm9jZXNzZXMgbGlrZSBkZXYgc2VydmVycyB0byBtYWtlIHN1cmUgdGhleSBzdGFydGVkIGNvcnJlY3RseS5cblxuVG8gcnVuIGEgYmFja2dyb3VuZCBwcm9jZXNzOlxuQmFzaCh7IGNvbW1hbmQ6IFwibnBtIHJ1biBkZXZcIiwgd2FpdFVudGlsOiAwIH0pXG5SZXR1cm5zIGltbWVkaWF0ZWx5IHdpdGggXFxgY29tbWFuZElkXFxgLiBUaGUgcHJvY2VzcyBrZWVwcyBydW5uaW5nIGluIHRoZSBzYW5kYm94LlxuXG5UbyBraWxsIGEgcHJvY2VzczpcbkJhc2goeyBjb21tYW5kOiBcInBzIGF1eCB8IGdyZXAgbm9kZVwiIH0pICAgLy8gRmluZCB0aGUgUElEXG5CYXNoKHsgY29tbWFuZDogXCJraWxsIHtwaWR9XCIgfSkgICAgICAgICAgICAvLyBHcmFjZWZ1bFxuQmFzaCh7IGNvbW1hbmQ6IFwia2lsbCAtOSB7cGlkfVwiIH0pICAgICAgICAgLy8gRm9yY2VcbmA7XG5cbmZ1bmN0aW9uIGJ1aWxkU2tpbGxzQ29udGV4dChza2lsbHM6IFNraWxsU3VtbWFyeVtdKTogc3RyaW5nIHtcbiAgaWYgKHNraWxscy5sZW5ndGggPT09IDApIHtcbiAgICByZXR1cm4gXCJcIjtcbiAgfVxuXG4gIGNvbnN0IHNraWxsTGluZXMgPSBza2lsbHNcbiAgICAubWFwKChzKSA9PiBgLSAke3MubmFtZX06ICR7cy5kZXNjcmlwdGlvbn1gKVxuICAgIC5qb2luKFwiXFxuXCIpO1xuXG4gIHJldHVybiBgIyMgQXZhaWxhYmxlIFNraWxsc1xuJHtza2lsbExpbmVzfVxuXG4ke2JhY2tncm91bmRQcm9jZXNzUHJvbXB0fVxuXG5Vc2UgdGhlIFNraWxsIHRvb2wgdG8gbG9hZCBhIHNraWxsJ3MgZnVsbCBpbnN0cnVjdGlvbnMgYmVmb3JlIGZvbGxvd2luZyBpdC5gO1xufVxuIiwgImltcG9ydCB0eXBlIHsgU2tpbGxFbnRyeSwgU2tpbGxJbnB1dCB9IGZyb20gXCIuL3R5cGVzXCI7XG5cbi8qKlxuICogUGFyc2VkIGZyb250bWF0dGVyIGZyb20gYSBTS0lMTC5tZCBmaWxlLlxuICovXG5leHBvcnQgdHlwZSBTa2lsbEZyb250bWF0dGVyID0ge1xuICBuYW1lOiBzdHJpbmc7XG4gIGRlc2NyaXB0aW9uOiBzdHJpbmc7XG59O1xuXG4vKipcbiAqIFBhcnNlcyBZQU1MIGZyb250bWF0dGVyIGZyb20gYSBTS0lMTC5tZCBmaWxlIGNvbnRlbnQuXG4gKiBGcm9udG1hdHRlciBtdXN0IGJlIGF0IHRoZSBzdGFydCBvZiB0aGUgZmlsZSwgZGVsaW1pdGVkIGJ5IGAtLS1gIG1hcmtlcnMuXG4gKlxuICogQGV4YW1wbGVcbiAqIGBgYG1hcmtkb3duXG4gKiAtLS1cbiAqIG5hbWU6IGNzdlxuICogZGVzY3JpcHRpb246IEFuYWx5emUgQ1NWIGRhdGFcbiAqIC0tLVxuICogIyBDU1YgU2tpbGxcbiAqIC4uLlxuICogYGBgXG4gKlxuICogQHJldHVybnMgUGFyc2VkIG5hbWUgYW5kIGRlc2NyaXB0aW9uLCBvciBudWxsIGlmIGZyb250bWF0dGVyIGlzIG1pc3NpbmcvaW52YWxpZFxuICovXG5leHBvcnQgZnVuY3Rpb24gcGFyc2VTa2lsbEZyb250bWF0dGVyKFxuICBjb250ZW50OiBzdHJpbmdcbik6IFNraWxsRnJvbnRtYXR0ZXIgfCBudWxsIHtcbiAgY29uc3QgdHJpbW1lZCA9IGNvbnRlbnQudHJpbSgpO1xuXG4gIGlmICghdHJpbW1lZC5zdGFydHNXaXRoKFwiLS0tXCIpKSB7XG4gICAgcmV0dXJuIG51bGw7XG4gIH1cblxuICBjb25zdCBlbmRNYXJrZXJJbmRleCA9IHRyaW1tZWQuaW5kZXhPZihcIi0tLVwiLCAzKTtcbiAgaWYgKGVuZE1hcmtlckluZGV4ID09PSAtMSkge1xuICAgIHJldHVybiBudWxsO1xuICB9XG5cbiAgY29uc3QgZnJvbnRtYXR0ZXJCbG9jayA9IHRyaW1tZWQuc2xpY2UoMywgZW5kTWFya2VySW5kZXgpLnRyaW0oKTtcbiAgY29uc3QgcGFyc2VkID0gcGFyc2VTaW1wbGVZYW1sKGZyb250bWF0dGVyQmxvY2spO1xuXG4gIGlmICghKHBhcnNlZC5uYW1lICYmIHBhcnNlZC5kZXNjcmlwdGlvbikpIHtcbiAgICByZXR1cm4gbnVsbDtcbiAgfVxuXG4gIHJldHVybiB7XG4gICAgbmFtZTogU3RyaW5nKHBhcnNlZC5uYW1lKSxcbiAgICBkZXNjcmlwdGlvbjogU3RyaW5nKHBhcnNlZC5kZXNjcmlwdGlvbiksXG4gIH07XG59XG5cbi8qKlxuICogUGFyc2VzIHNpbXBsZSBZQU1MIGtleS12YWx1ZSBwYWlycyAoc2luZ2xlLWxpbmUgdmFsdWVzIG9ubHkpLlxuICogVGhpcyBhdm9pZHMgYWRkaW5nIGEgZnVsbCBZQU1MIHBhcnNlciBkZXBlbmRlbmN5IGZvciBiYXNpYyBmcm9udG1hdHRlci5cbiAqL1xuZnVuY3Rpb24gcGFyc2VTaW1wbGVZYW1sKHlhbWw6IHN0cmluZyk6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4ge1xuICBjb25zdCByZXN1bHQ6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4gPSB7fTtcblxuICBmb3IgKGNvbnN0IGxpbmUgb2YgeWFtbC5zcGxpdChcIlxcblwiKSkge1xuICAgIGNvbnN0IHRyaW1tZWRMaW5lID0gbGluZS50cmltKCk7XG4gICAgaWYgKCF0cmltbWVkTGluZSB8fCB0cmltbWVkTGluZS5zdGFydHNXaXRoKFwiI1wiKSkge1xuICAgICAgY29udGludWU7XG4gICAgfVxuXG4gICAgY29uc3QgY29sb25JbmRleCA9IHRyaW1tZWRMaW5lLmluZGV4T2YoXCI6XCIpO1xuICAgIGlmIChjb2xvbkluZGV4ID09PSAtMSkge1xuICAgICAgY29udGludWU7XG4gICAgfVxuXG4gICAgY29uc3Qga2V5ID0gdHJpbW1lZExpbmUuc2xpY2UoMCwgY29sb25JbmRleCkudHJpbSgpO1xuICAgIGxldCB2YWx1ZSA9IHRyaW1tZWRMaW5lLnNsaWNlKGNvbG9uSW5kZXggKyAxKS50cmltKCk7XG5cbiAgICAvLyBSZW1vdmUgc3Vycm91bmRpbmcgcXVvdGVzIGlmIHByZXNlbnRcbiAgICBpZiAoXG4gICAgICAodmFsdWUuc3RhcnRzV2l0aCgnXCInKSAmJiB2YWx1ZS5lbmRzV2l0aCgnXCInKSkgfHxcbiAgICAgICh2YWx1ZS5zdGFydHNXaXRoKFwiJ1wiKSAmJiB2YWx1ZS5lbmRzV2l0aChcIidcIikpXG4gICAgKSB7XG4gICAgICB2YWx1ZSA9IHZhbHVlLnNsaWNlKDEsIC0xKTtcbiAgICB9XG5cbiAgICBpZiAoa2V5KSB7XG4gICAgICByZXN1bHRba2V5XSA9IHZhbHVlO1xuICAgIH1cbiAgfVxuXG4gIHJldHVybiByZXN1bHQ7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBub3JtYWxpemVTa2lsbHMoXG4gIHNraWxsczogU2tpbGxJbnB1dFtdIHwgbnVsbCB8IHVuZGVmaW5lZFxuKTogU2tpbGxFbnRyeVtdIHtcbiAgaWYgKCFza2lsbHMgfHwgc2tpbGxzLmxlbmd0aCA9PT0gMCkge1xuICAgIHJldHVybiBbXTtcbiAgfVxuXG4gIHJldHVybiBza2lsbHMubWFwKChza2lsbCkgPT4ge1xuICAgIGlmICh0eXBlb2Ygc2tpbGwgIT09IFwib2JqZWN0XCIgfHwgc2tpbGwgPT09IG51bGwpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgICAgXCJJbnZhbGlkIHNraWxscyBlbnRyeTogZXhwZWN0ZWQgYW4gb2JqZWN0IHdpdGggYSBgdHlwZWAgZmllbGQuXCJcbiAgICAgICk7XG4gICAgfVxuICAgIGlmIChcInR5cGVcIiBpbiBza2lsbCAmJiBza2lsbC50eXBlKSB7XG4gICAgICByZXR1cm4gc2tpbGwgYXMgU2tpbGxFbnRyeTtcbiAgICB9XG4gICAgdGhyb3cgbmV3IEVycm9yKFwiSW52YWxpZCBza2lsbHMgZW50cnk6IGV4cGVjdGVkIGB0eXBlYC5cIik7XG4gIH0pO1xufVxuIiwgImltcG9ydCB0eXBlIHsgQW55QWdlbnQgfSBmcm9tIFwiLi4vYWdlbnQvY2xpZW50XCI7XG5pbXBvcnQgdHlwZSB7IFNhbmRib3hJbnN0YW5jZSB9IGZyb20gXCIuLi9zYW5kYm94L2FkYXB0ZXJcIjtcbmltcG9ydCB7IGNyZWF0ZUxvZ2dlciwgdHlwZSBMb2dnZXIgfSBmcm9tIFwiLi4vdXRpbHMvbG9nZ2VyXCI7XG5pbXBvcnQgeyBwYXJzZVNraWxsRnJvbnRtYXR0ZXIgfSBmcm9tIFwiLi9wYXJzZXJcIjtcbmltcG9ydCB0eXBlIHsgU2tpbGxTdW1tYXJ5IH0gZnJvbSBcIi4vdHlwZXNcIjtcblxuLyoqXG4gKiBEaXNjb3ZlcnMgc2tpbGxzIGZyb20gZGlyZWN0b3JpZXMgaW5zaWRlIHRoZSBzYW5kYm94IGJ5IGZpbmRpbmcgYW5kIHBhcnNpbmcgU0tJTEwubWQgZmlsZXMuXG4gKiBTY2FucyBlYWNoIGRpcmVjdG9yeSBmb3Igc3ViZGlyZWN0b3JpZXMgY29udGFpbmluZyBTS0lMTC5tZCwgZXh0cmFjdHMgZnJvbnRtYXR0ZXIgbWV0YWRhdGEsXG4gKiBhbmQgcmV0dXJucyBzdW1tYXJpZXMgZm9yIHVzZSBpbiB0aGUgc3lzdGVtIHByb21wdC5cbiAqXG4gKiBAcmV0dXJucyBBcnJheSBvZiBza2lsbCBzdW1tYXJpZXMgKGRlZHVwbGljYXRlZCBieSBuYW1lLCBmaXJzdCBvY2N1cnJlbmNlIHdpbnMpXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBkaXNjb3ZlclNraWxsc0luU2FuZGJveChvcHRzOiB7XG4gIGFnZW50OiBBbnlBZ2VudDtcbiAgc2FuZGJveDogU2FuZGJveEluc3RhbmNlO1xuICBza2lsbHNEaXJzOiBzdHJpbmdbXTtcbiAgc2Vzc2lvbklkPzogc3RyaW5nO1xufSk6IFByb21pc2U8U2tpbGxTdW1tYXJ5W10+IHtcbiAgY29uc3QgeyBhZ2VudCwgc2FuZGJveCwgc2tpbGxzRGlycywgc2Vzc2lvbklkIH0gPSBvcHRzO1xuICBjb25zdCBiYXNlTG9nID0gY3JlYXRlTG9nZ2VyKHtcbiAgICBjb25maWc6IHsgLi4uYWdlbnQub3B0aW9ucy5sb2dnaW5nLCBuYW1lOiBhZ2VudC5uYW1lIH0sXG4gICAgc3Vic3lzdGVtOiBcInNraWxsc1wiLFxuICB9KTtcbiAgY29uc3QgbG9nID0gc2Vzc2lvbklkID8gYmFzZUxvZy53aXRoQ29udGV4dCh7IHNlc3Npb25JZCB9KSA6IGJhc2VMb2c7XG4gIGNvbnN0IGRvbmUgPSBsb2cudGltZShcImRpc2NvdmVyU2tpbGxzSW5TYW5kYm94XCIpO1xuXG4gIGNvbnN0IHN1bW1hcmllczogU2tpbGxTdW1tYXJ5W10gPSBbXTtcbiAgY29uc3Qgc2Vlbk5hbWVzID0gbmV3IFNldDxzdHJpbmc+KCk7XG5cbiAgZm9yIChjb25zdCBza2lsbHNEaXIgb2Ygc2tpbGxzRGlycykge1xuICAgIGNvbnN0IGRpckRvbmUgPSBsb2cudGltZShcInNjYW4gZGlyZWN0b3J5XCIsIHsgZGlyOiBza2lsbHNEaXIgfSk7XG4gICAgY29uc3QgZGlyU3VtbWFyaWVzID0gYXdhaXQgZGlzY292ZXJTa2lsbHNJbkRpcmVjdG9yeSh7XG4gICAgICBzYW5kYm94LFxuICAgICAgc2tpbGxzRGlyLFxuICAgICAgbG9nLFxuICAgIH0pO1xuICAgIGRpckRvbmUoeyBjb3VudDogZGlyU3VtbWFyaWVzLmxlbmd0aCB9KTtcblxuICAgIGZvciAoY29uc3Qgc3VtbWFyeSBvZiBkaXJTdW1tYXJpZXMpIHtcbiAgICAgIGlmICghc2Vlbk5hbWVzLmhhcyhzdW1tYXJ5Lm5hbWUpKSB7XG4gICAgICAgIHNlZW5OYW1lcy5hZGQoc3VtbWFyeS5uYW1lKTtcbiAgICAgICAgc3VtbWFyaWVzLnB1c2goc3VtbWFyeSk7XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgZG9uZSh7IHRvdGFsU2tpbGxzOiBzdW1tYXJpZXMubGVuZ3RoIH0pO1xuXG4gIHJldHVybiBzdW1tYXJpZXM7XG59XG5cbmFzeW5jIGZ1bmN0aW9uIGRpc2NvdmVyU2tpbGxzSW5EaXJlY3Rvcnkob3B0czoge1xuICBzYW5kYm94OiBTYW5kYm94SW5zdGFuY2U7XG4gIHNraWxsc0Rpcjogc3RyaW5nO1xuICBsb2c6IExvZ2dlcjtcbn0pOiBQcm9taXNlPFNraWxsU3VtbWFyeVtdPiB7XG4gIGNvbnN0IHsgc2FuZGJveCwgc2tpbGxzRGlyLCBsb2cgfSA9IG9wdHM7XG4gIGNvbnN0IHNraWxsUGF0aHMgPSBhd2FpdCBmaW5kU2tpbGxGaWxlcyh7IHNhbmRib3gsIHNraWxsc0RpciwgbG9nIH0pO1xuXG4gIGlmIChza2lsbFBhdGhzLmxlbmd0aCA9PT0gMCkge1xuICAgIHJldHVybiBbXTtcbiAgfVxuXG4gIGNvbnN0IHN1bW1hcmllczogU2tpbGxTdW1tYXJ5W10gPSBbXTtcblxuICBmb3IgKGNvbnN0IHNraWxsTWRQYXRoIG9mIHNraWxsUGF0aHMpIHtcbiAgICBjb25zdCBzdW1tYXJ5ID0gYXdhaXQgcGFyc2VTa2lsbEZpbGUoeyBzYW5kYm94LCBza2lsbE1kUGF0aCwgbG9nIH0pO1xuICAgIGlmIChzdW1tYXJ5KSB7XG4gICAgICBzdW1tYXJpZXMucHVzaChzdW1tYXJ5KTtcbiAgICB9XG4gIH1cblxuICByZXR1cm4gc3VtbWFyaWVzO1xufVxuXG5hc3luYyBmdW5jdGlvbiBmaW5kU2tpbGxGaWxlcyhvcHRzOiB7XG4gIHNhbmRib3g6IFNhbmRib3hJbnN0YW5jZTtcbiAgc2tpbGxzRGlyOiBzdHJpbmc7XG4gIGxvZzogTG9nZ2VyO1xufSk6IFByb21pc2U8c3RyaW5nW10+IHtcbiAgY29uc3QgeyBzYW5kYm94LCBza2lsbHNEaXIsIGxvZyB9ID0gb3B0cztcblxuICBjb25zdCBleGVjUmVzdWx0ID0gYXdhaXQgc2FuZGJveC5leGVjKHtcbiAgICBjb21tYW5kOiBcImZpbmRcIixcbiAgICBhcmdzOiBbc2tpbGxzRGlyLCBcIi1uYW1lXCIsIFwiU0tJTEwubWRcIiwgXCItdHlwZVwiLCBcImZcIl0sXG4gIH0pO1xuXG4gIGlmIChleGVjUmVzdWx0IGluc3RhbmNlb2YgRXJyb3IpIHtcbiAgICBsb2cud2FybihcImZhaWxlZCB0byBzY2FuIHNraWxscyBkaXJlY3RvcnlcIiwge1xuICAgICAgZGlyOiBza2lsbHNEaXIsXG4gICAgICBlcnJvcjogZXhlY1Jlc3VsdC5tZXNzYWdlLFxuICAgIH0pO1xuICAgIHJldHVybiBbXTtcbiAgfVxuXG4gIGNvbnN0IHsgc3Rkb3V0LCBleGl0Q29kZSB9ID0gYXdhaXQgZXhlY1Jlc3VsdC5yZXN1bHQ7XG5cbiAgaWYgKGV4aXRDb2RlICE9PSAwKSB7XG4gICAgbG9nLndhcm4oXCJza2lsbHMgZGlyZWN0b3J5IG5vdCBmb3VuZCBvciBpbmFjY2Vzc2libGVcIiwgeyBkaXI6IHNraWxsc0RpciB9KTtcbiAgICByZXR1cm4gW107XG4gIH1cblxuICByZXR1cm4gc3Rkb3V0XG4gICAgLnRyaW0oKVxuICAgIC5zcGxpdChcIlxcblwiKVxuICAgIC5maWx0ZXIoKHApID0+IHAubGVuZ3RoID4gMCk7XG59XG5cbmFzeW5jIGZ1bmN0aW9uIHBhcnNlU2tpbGxGaWxlKG9wdHM6IHtcbiAgc2FuZGJveDogU2FuZGJveEluc3RhbmNlO1xuICBza2lsbE1kUGF0aDogc3RyaW5nO1xuICBsb2c6IExvZ2dlcjtcbn0pOiBQcm9taXNlPFNraWxsU3VtbWFyeSB8IG51bGw+IHtcbiAgY29uc3QgeyBzYW5kYm94LCBza2lsbE1kUGF0aCwgbG9nIH0gPSBvcHRzO1xuICBjb25zdCBleGVjUmVzdWx0ID0gYXdhaXQgc2FuZGJveC5leGVjKHtcbiAgICBjb21tYW5kOiBcImNhdFwiLFxuICAgIGFyZ3M6IFtza2lsbE1kUGF0aF0sXG4gIH0pO1xuXG4gIGlmIChleGVjUmVzdWx0IGluc3RhbmNlb2YgRXJyb3IpIHtcbiAgICBsb2cud2FybihcImZhaWxlZCB0byByZWFkIHNraWxsIGZpbGVcIiwge1xuICAgICAgcGF0aDogc2tpbGxNZFBhdGgsXG4gICAgICBlcnJvcjogZXhlY1Jlc3VsdC5tZXNzYWdlLFxuICAgIH0pO1xuICAgIHJldHVybiBudWxsO1xuICB9XG5cbiAgY29uc3QgeyBzdGRvdXQsIGV4aXRDb2RlIH0gPSBhd2FpdCBleGVjUmVzdWx0LnJlc3VsdDtcblxuICBpZiAoZXhpdENvZGUgIT09IDApIHtcbiAgICBsb2cud2FybihcImNvdWxkIG5vdCByZWFkIHNraWxsIGZpbGVcIiwgeyBwYXRoOiBza2lsbE1kUGF0aCB9KTtcbiAgICByZXR1cm4gbnVsbDtcbiAgfVxuXG4gIGNvbnN0IHBhcnNlZCA9IHBhcnNlU2tpbGxGcm9udG1hdHRlcihzdGRvdXQpO1xuXG4gIGlmICghcGFyc2VkKSB7XG4gICAgbG9nLndhcm4oXCJpbnZhbGlkIG9yIG1pc3NpbmcgZnJvbnRtYXR0ZXJcIiwgeyBwYXRoOiBza2lsbE1kUGF0aCB9KTtcbiAgICByZXR1cm4gbnVsbDtcbiAgfVxuXG4gIHJldHVybiB7XG4gICAgbmFtZTogcGFyc2VkLm5hbWUsXG4gICAgZGVzY3JpcHRpb246IHBhcnNlZC5kZXNjcmlwdGlvbixcbiAgICBza2lsbE1kUGF0aCxcbiAgfTtcbn1cbiIsICJpbXBvcnQgdHlwZSB7IFNhbmRib3hJbnN0YW5jZSB9IGZyb20gXCIuLi9zYW5kYm94L2FkYXB0ZXJcIjtcbmltcG9ydCB0eXBlIHsgSG9zdFNraWxsSW5wdXQgfSBmcm9tIFwiLi90eXBlc1wiO1xuXG5hc3luYyBmdW5jdGlvbiBjb2xsZWN0RmlsZXMob3B0czoge1xuICBhYnNvbHV0ZVBhdGg6IHN0cmluZztcbiAgcmVsYXRpdmVQcmVmaXg/OiBzdHJpbmc7XG59KTogUHJvbWlzZTx7IHBhdGg6IHN0cmluZzsgY29udGVudDogQnVmZmVyIH1bXT4ge1xuICBjb25zdCB7IHJlYWRkaXIsIHJlYWRGaWxlLCBzdGF0IH0gPSBhd2FpdCBpbXBvcnQoXCJub2RlOmZzL3Byb21pc2VzXCIpO1xuICBjb25zdCB7IHJlc29sdmUsIHBvc2l4OiBwYXRoUG9zaXggfSA9IGF3YWl0IGltcG9ydChcIm5vZGU6cGF0aFwiKTtcblxuICBjb25zdCBmaWxlU3RhdCA9IGF3YWl0IHN0YXQob3B0cy5hYnNvbHV0ZVBhdGgpO1xuICBpZiAoZmlsZVN0YXQuaXNGaWxlKCkpIHtcbiAgICBjb25zdCBjb250ZW50ID0gYXdhaXQgcmVhZEZpbGUob3B0cy5hYnNvbHV0ZVBhdGgpO1xuICAgIHJldHVybiBbeyBwYXRoOiBvcHRzLnJlbGF0aXZlUHJlZml4ID8/IFwiU0tJTEwubWRcIiwgY29udGVudCB9XTtcbiAgfVxuXG4gIGlmICghZmlsZVN0YXQuaXNEaXJlY3RvcnkoKSkge1xuICAgIHJldHVybiBbXTtcbiAgfVxuXG4gIGNvbnN0IGVudHJpZXMgPSBhd2FpdCByZWFkZGlyKG9wdHMuYWJzb2x1dGVQYXRoLCB7IHdpdGhGaWxlVHlwZXM6IHRydWUgfSk7XG4gIGNvbnN0IGZpbGVzOiB7IHBhdGg6IHN0cmluZzsgY29udGVudDogQnVmZmVyIH1bXSA9IFtdO1xuXG4gIGZvciAoY29uc3QgZW50cnkgb2YgZW50cmllcykge1xuICAgIGNvbnN0IG5leHRBYnMgPSByZXNvbHZlKG9wdHMuYWJzb2x1dGVQYXRoLCBlbnRyeS5uYW1lKTtcbiAgICBjb25zdCBuZXh0UmVsID0gb3B0cy5yZWxhdGl2ZVByZWZpeFxuICAgICAgPyBwYXRoUG9zaXguam9pbihvcHRzLnJlbGF0aXZlUHJlZml4LCBlbnRyeS5uYW1lKVxuICAgICAgOiBlbnRyeS5uYW1lO1xuXG4gICAgaWYgKGVudHJ5LmlzRGlyZWN0b3J5KCkpIHtcbiAgICAgIGZpbGVzLnB1c2goXG4gICAgICAgIC4uLihhd2FpdCBjb2xsZWN0RmlsZXMoe1xuICAgICAgICAgIGFic29sdXRlUGF0aDogbmV4dEFicyxcbiAgICAgICAgICByZWxhdGl2ZVByZWZpeDogbmV4dFJlbCxcbiAgICAgICAgfSkpXG4gICAgICApO1xuICAgICAgY29udGludWU7XG4gICAgfVxuXG4gICAgaWYgKGVudHJ5LmlzRmlsZSgpKSB7XG4gICAgICBmaWxlcy5wdXNoKHtcbiAgICAgICAgcGF0aDogbmV4dFJlbCxcbiAgICAgICAgY29udGVudDogYXdhaXQgcmVhZEZpbGUobmV4dEFicyksXG4gICAgICB9KTtcbiAgICB9XG4gIH1cblxuICByZXR1cm4gZmlsZXM7XG59XG5cbmFzeW5jIGZ1bmN0aW9uIHN0YWJsZURpck5hbWUoaW5wdXRQYXRoOiBzdHJpbmcpOiBQcm9taXNlPHN0cmluZz4ge1xuICBjb25zdCB7IGNyZWF0ZUhhc2ggfSA9IGF3YWl0IGltcG9ydChcIm5vZGU6Y3J5cHRvXCIpO1xuICBjb25zdCBoYXNoID0gY3JlYXRlSGFzaChcInNoYTFcIikudXBkYXRlKGlucHV0UGF0aCkuZGlnZXN0KFwiaGV4XCIpLnNsaWNlKDAsIDEwKTtcbiAgcmV0dXJuIGBob3N0LSR7aGFzaH1gO1xufVxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gcmVzb2x2ZUhvc3RTa2lsbChvcHRzOiB7XG4gIHNhbmRib3g6IFNhbmRib3hJbnN0YW5jZTtcbiAgc2tpbGw6IEhvc3RTa2lsbElucHV0O1xufSk6IFByb21pc2U8c3RyaW5nPiB7XG4gIGNvbnN0IHsgcmVzb2x2ZSwgcG9zaXg6IHBhdGhQb3NpeCB9ID0gYXdhaXQgaW1wb3J0KFwibm9kZTpwYXRoXCIpO1xuXG4gIGNvbnN0IGhvc3RQYXRoID0gcmVzb2x2ZShwcm9jZXNzLmN3ZCgpLCBvcHRzLnNraWxsLnBhdGgpO1xuICBjb25zdCBmaWxlcyA9IGF3YWl0IGNvbGxlY3RGaWxlcyh7IGFic29sdXRlUGF0aDogaG9zdFBhdGggfSk7XG4gIGNvbnN0IGRpck5hbWUgPSBhd2FpdCBzdGFibGVEaXJOYW1lKGhvc3RQYXRoKTtcbiAgY29uc3QgZGVzdFBhdGggPSBwYXRoUG9zaXguam9pbihcIi5hZ2VudFwiLCBcInNraWxsc1wiLCBcIi5tYXRlcmlhbGl6ZWRcIiwgZGlyTmFtZSk7XG5cbiAgaWYgKGZpbGVzLmxlbmd0aCA9PT0gMCkge1xuICAgIHJldHVybiBkZXN0UGF0aDtcbiAgfVxuXG4gIGF3YWl0IG9wdHMuc2FuZGJveC53cml0ZUZpbGVzKHtcbiAgICBmaWxlcyxcbiAgICBkZXN0UGF0aCxcbiAgfSk7XG5cbiAgcmV0dXJuIGRlc3RQYXRoO1xufVxuIiwgImltcG9ydCB0eXBlIHsgU2FuZGJveEluc3RhbmNlIH0gZnJvbSBcIi4uL3NhbmRib3gvYWRhcHRlclwiO1xuaW1wb3J0IHR5cGUgeyBJbmxpbmVTa2lsbElucHV0IH0gZnJvbSBcIi4vdHlwZXNcIjtcblxuZnVuY3Rpb24gc2x1Z2lmeSh2YWx1ZTogc3RyaW5nKTogc3RyaW5nIHtcbiAgcmV0dXJuIHZhbHVlXG4gICAgLnRvTG93ZXJDYXNlKClcbiAgICAucmVwbGFjZSgvW15hLXowLTldKy9nLCBcIi1cIilcbiAgICAucmVwbGFjZSgvKF4tfC0kKS9nLCBcIlwiKTtcbn1cblxuYXN5bmMgZnVuY3Rpb24gc3RhYmxlSW5saW5lRGlyTmFtZShuYW1lOiBzdHJpbmcpOiBQcm9taXNlPHN0cmluZz4ge1xuICBjb25zdCB7IGNyZWF0ZUhhc2ggfSA9IGF3YWl0IGltcG9ydChcIm5vZGU6Y3J5cHRvXCIpO1xuICBjb25zdCBzbHVnID0gc2x1Z2lmeShuYW1lKSB8fCBcImlubGluZS1za2lsbFwiO1xuICBjb25zdCBoYXNoID0gY3JlYXRlSGFzaChcInNoYTFcIikudXBkYXRlKG5hbWUpLmRpZ2VzdChcImhleFwiKS5zbGljZSgwLCA4KTtcbiAgcmV0dXJuIGAke3NsdWd9LSR7aGFzaH1gO1xufVxuXG5mdW5jdGlvbiBidWlsZFNraWxsTWFya2Rvd24oc2tpbGw6IElubGluZVNraWxsSW5wdXQpOiBzdHJpbmcge1xuICByZXR1cm4gYC0tLVxubmFtZTogJHtza2lsbC5uYW1lfVxuZGVzY3JpcHRpb246ICR7c2tpbGwuZGVzY3JpcHRpb259XG4tLS1cblxuJHtza2lsbC5pbnN0cnVjdGlvbnMudHJpbSgpfVxuYDtcbn1cblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIHJlc29sdmVJbmxpbmVTa2lsbChvcHRzOiB7XG4gIHNhbmRib3g6IFNhbmRib3hJbnN0YW5jZTtcbiAgc2tpbGw6IElubGluZVNraWxsSW5wdXQ7XG59KTogUHJvbWlzZTxzdHJpbmc+IHtcbiAgY29uc3QgeyBwb3NpeDogcGF0aFBvc2l4IH0gPSBhd2FpdCBpbXBvcnQoXCJub2RlOnBhdGhcIik7XG5cbiAgY29uc3QgZGlyTmFtZSA9IGF3YWl0IHN0YWJsZUlubGluZURpck5hbWUob3B0cy5za2lsbC5uYW1lKTtcbiAgY29uc3QgZGVzdFBhdGggPSBwYXRoUG9zaXguam9pbihcIi5hZ2VudFwiLCBcInNraWxsc1wiLCBcIi5tYXRlcmlhbGl6ZWRcIiwgZGlyTmFtZSk7XG5cbiAgYXdhaXQgb3B0cy5zYW5kYm94LndyaXRlRmlsZXMoe1xuICAgIGRlc3RQYXRoLFxuICAgIGZpbGVzOiBbXG4gICAgICB7XG4gICAgICAgIHBhdGg6IFwiU0tJTEwubWRcIixcbiAgICAgICAgY29udGVudDogYnVpbGRTa2lsbE1hcmtkb3duKG9wdHMuc2tpbGwpLFxuICAgICAgfSxcbiAgICBdLFxuICB9KTtcblxuICByZXR1cm4gZGVzdFBhdGg7XG59XG4iLCAiaW1wb3J0IHR5cGUgeyBTYW5kYm94SW5zdGFuY2UgfSBmcm9tIFwiLi4vc2FuZGJveC9hZGFwdGVyXCI7XG5pbXBvcnQgdHlwZSB7IEdpdFNraWxsSW5wdXQgfSBmcm9tIFwiLi90eXBlc1wiO1xuXG5mdW5jdGlvbiBzaGVsbFF1b3RlKHZhbHVlOiBzdHJpbmcpOiBzdHJpbmcge1xuICByZXR1cm4gYCcke3ZhbHVlLnJlcGxhY2UoLycvZywgYCdcXFxcJydgKX0nYDtcbn1cblxuYXN5bmMgZnVuY3Rpb24gc3RhYmxlUmVwb0Rpck5hbWUoc2tpbGw6IEdpdFNraWxsSW5wdXQpOiBQcm9taXNlPHN0cmluZz4ge1xuICBjb25zdCB7IGNyZWF0ZUhhc2ggfSA9IGF3YWl0IGltcG9ydChcIm5vZGU6Y3J5cHRvXCIpO1xuICBjb25zdCBoYXNoSW5wdXQgPSBgJHtza2lsbC5yZXBvfTo6JHtza2lsbC5yZWYgPz8gXCJIRUFEXCJ9YDtcbiAgY29uc3QgaGFzaCA9IGNyZWF0ZUhhc2goXCJzaGExXCIpLnVwZGF0ZShoYXNoSW5wdXQpLmRpZ2VzdChcImhleFwiKS5zbGljZSgwLCAxMik7XG4gIHJldHVybiBgZ2l0LSR7aGFzaH1gO1xufVxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gcmVzb2x2ZVJlbW90ZVNraWxsKG9wdHM6IHtcbiAgc2FuZGJveDogU2FuZGJveEluc3RhbmNlO1xuICBza2lsbDogR2l0U2tpbGxJbnB1dDtcbn0pOiBQcm9taXNlPHN0cmluZz4ge1xuICBjb25zdCB7IHBvc2l4OiBwYXRoUG9zaXggfSA9IGF3YWl0IGltcG9ydChcIm5vZGU6cGF0aFwiKTtcblxuICBjb25zdCBkaXJOYW1lID0gYXdhaXQgc3RhYmxlUmVwb0Rpck5hbWUob3B0cy5za2lsbCk7XG4gIGNvbnN0IGNsb25lRGlyID0gcGF0aFBvc2l4LmpvaW4oXCIuYWdlbnRcIiwgXCJza2lsbHNcIiwgXCIubWF0ZXJpYWxpemVkXCIsIGRpck5hbWUpO1xuICBjb25zdCBxdW90ZWREaXIgPSBzaGVsbFF1b3RlKGNsb25lRGlyKTtcbiAgY29uc3QgcXVvdGVkUmVwbyA9IHNoZWxsUXVvdGUob3B0cy5za2lsbC5yZXBvKTtcbiAgY29uc3QgcXVvdGVkUmVmID0gc2hlbGxRdW90ZShvcHRzLnNraWxsLnJlZiA/PyBcIkhFQURcIik7XG5cbiAgY29uc3Qgc3luY0NvbW1hbmQgPSBbXG4gICAgXCJzZXQgLWV1byBwaXBlZmFpbFwiLFxuICAgIGBpZiBbICEgLWQgJHtxdW90ZWREaXJ9Ly5naXQgXTsgdGhlbmAsXG4gICAgYCAgcm0gLXJmICR7cXVvdGVkRGlyfWAsXG4gICAgYCAgbWtkaXIgLXAgJHtxdW90ZWREaXJ9YCxcbiAgICBgICBnaXQgY2xvbmUgLS1kZXB0aCAxICR7cXVvdGVkUmVwb30gJHtxdW90ZWREaXJ9YCxcbiAgICBcImZpXCIsXG4gICAgYGlmIFsgJHtxdW90ZWRSZWZ9ICE9ICdIRUFEJyBdOyB0aGVuYCxcbiAgICBgICBnaXQgLUMgJHtxdW90ZWREaXJ9IGZldGNoIC0tZGVwdGggMSBvcmlnaW4gJHtxdW90ZWRSZWZ9YCxcbiAgICBgICBnaXQgLUMgJHtxdW90ZWREaXJ9IGNoZWNrb3V0IEZFVENIX0hFQURgLFxuICAgIFwiZmlcIixcbiAgXS5qb2luKFwiXFxuXCIpO1xuXG4gIGNvbnN0IGV4ZWNSZXN1bHQgPSBhd2FpdCBvcHRzLnNhbmRib3guZXhlYyh7XG4gICAgY29tbWFuZDogXCJiYXNoXCIsXG4gICAgYXJnczogW1wiLWxjXCIsIHN5bmNDb21tYW5kXSxcbiAgfSk7XG4gIGNvbnN0IHsgZXhpdENvZGUsIHN0ZGVyciB9ID0gYXdhaXQgZXhlY1Jlc3VsdC5yZXN1bHQ7XG4gIGlmIChleGl0Q29kZSAhPT0gMCkge1xuICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgIGBGYWlsZWQgdG8gbWF0ZXJpYWxpemUgZ2l0IHNraWxsIGZyb20gJHtvcHRzLnNraWxsLnJlcG99OiAke3N0ZGVyciB8fCBcInVua25vd24gZXJyb3JcIn1gXG4gICAgKTtcbiAgfVxuXG4gIGxldCByZXNvbHZlZERpciA9IGNsb25lRGlyO1xuICBpZiAob3B0cy5za2lsbC5wYXRoKSB7XG4gICAgcmVzb2x2ZWREaXIgPSBwYXRoUG9zaXguam9pbihyZXNvbHZlZERpciwgb3B0cy5za2lsbC5wYXRoKTtcbiAgfVxuICBpZiAob3B0cy5za2lsbC5uYW1lKSB7XG4gICAgcmVzb2x2ZWREaXIgPSBwYXRoUG9zaXguam9pbihyZXNvbHZlZERpciwgb3B0cy5za2lsbC5uYW1lKTtcbiAgfVxuICByZXR1cm4gcmVzb2x2ZWREaXI7XG59XG4iLCAiaW1wb3J0IHR5cGUgeyBBbnlBZ2VudCB9IGZyb20gXCIuLi9hZ2VudC9jbGllbnRcIjtcbmltcG9ydCB0eXBlIHsgU2FuZGJveEluc3RhbmNlIH0gZnJvbSBcIi4uL3NhbmRib3gvYWRhcHRlclwiO1xuaW1wb3J0IHsgY3JlYXRlTG9nZ2VyIH0gZnJvbSBcIi4uL3V0aWxzL2xvZ2dlclwiO1xuaW1wb3J0IHsgbm9ybWFsaXplU2tpbGxzIH0gZnJvbSBcIi4vcGFyc2VyXCI7XG5pbXBvcnQgeyByZXNvbHZlSG9zdFNraWxsIH0gZnJvbSBcIi4vcmVzb2x2ZS1ob3N0XCI7XG5pbXBvcnQgeyByZXNvbHZlSW5saW5lU2tpbGwgfSBmcm9tIFwiLi9yZXNvbHZlLWlubGluZVwiO1xuaW1wb3J0IHsgcmVzb2x2ZVJlbW90ZVNraWxsIH0gZnJvbSBcIi4vcmVzb2x2ZS1yZW1vdGVcIjtcbmltcG9ydCB0eXBlIHsgU2tpbGxJbnB1dCB9IGZyb20gXCIuL3R5cGVzXCI7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiByZXNvbHZlU2tpbGxJbnB1dHNUb1NhbmRib3hEaXJzKG9wdHM6IHtcbiAgYWdlbnQ6IEFueUFnZW50O1xuICBzYW5kYm94OiBTYW5kYm94SW5zdGFuY2U7XG4gIHNraWxsczogU2tpbGxJbnB1dFtdO1xuICBzZXNzaW9uSWQ/OiBzdHJpbmc7XG59KTogUHJvbWlzZTxzdHJpbmdbXT4ge1xuICBjb25zdCBiYXNlTG9nID0gY3JlYXRlTG9nZ2VyKHtcbiAgICBjb25maWc6IHsgLi4ub3B0cy5hZ2VudC5vcHRpb25zLmxvZ2dpbmcsIG5hbWU6IG9wdHMuYWdlbnQubmFtZSB9LFxuICAgIHN1YnN5c3RlbTogXCJza2lsbHNcIixcbiAgfSk7XG4gIGNvbnN0IGxvZyA9IG9wdHMuc2Vzc2lvbklkXG4gICAgPyBiYXNlTG9nLndpdGhDb250ZXh0KHsgc2Vzc2lvbklkOiBvcHRzLnNlc3Npb25JZCB9KVxuICAgIDogYmFzZUxvZztcbiAgY29uc3QgZW50cmllcyA9IG5vcm1hbGl6ZVNraWxscyhvcHRzLnNraWxscyk7XG4gIGNvbnN0IHJlc29sdmVkOiBzdHJpbmdbXSA9IFtdO1xuXG4gIGZvciAoY29uc3QgZW50cnkgb2YgZW50cmllcykge1xuICAgIGlmIChlbnRyeS50eXBlID09PSBcInNhbmRib3hcIikge1xuICAgICAgcmVzb2x2ZWQucHVzaChlbnRyeS5wYXRoKTtcbiAgICAgIGNvbnRpbnVlO1xuICAgIH1cbiAgICBpZiAoZW50cnkudHlwZSA9PT0gXCJob3N0XCIpIHtcbiAgICAgIHJlc29sdmVkLnB1c2goXG4gICAgICAgIGF3YWl0IHJlc29sdmVIb3N0U2tpbGwoe1xuICAgICAgICAgIHNhbmRib3g6IG9wdHMuc2FuZGJveCxcbiAgICAgICAgICBza2lsbDogZW50cnksXG4gICAgICAgIH0pXG4gICAgICApO1xuICAgICAgY29udGludWU7XG4gICAgfVxuICAgIGlmIChlbnRyeS50eXBlID09PSBcImdpdFwiKSB7XG4gICAgICByZXNvbHZlZC5wdXNoKFxuICAgICAgICBhd2FpdCByZXNvbHZlUmVtb3RlU2tpbGwoe1xuICAgICAgICAgIHNhbmRib3g6IG9wdHMuc2FuZGJveCxcbiAgICAgICAgICBza2lsbDogZW50cnksXG4gICAgICAgIH0pXG4gICAgICApO1xuICAgICAgY29udGludWU7XG4gICAgfVxuICAgIGlmIChlbnRyeS50eXBlID09PSBcImlubGluZVwiKSB7XG4gICAgICByZXNvbHZlZC5wdXNoKFxuICAgICAgICBhd2FpdCByZXNvbHZlSW5saW5lU2tpbGwoe1xuICAgICAgICAgIHNhbmRib3g6IG9wdHMuc2FuZGJveCxcbiAgICAgICAgICBza2lsbDogZW50cnksXG4gICAgICAgIH0pXG4gICAgICApO1xuICAgICAgY29udGludWU7XG4gICAgfVxuICAgIGxvZy53YXJuKFwidW5rbm93biBza2lsbCBzb3VyY2UgZW50cnkgZW5jb3VudGVyZWRcIiwgeyBlbnRyeSB9KTtcbiAgfVxuXG4gIHJldHVybiByZXNvbHZlZDtcbn1cbiIsICJpbXBvcnQgdHlwZSB7IEpTT05WYWx1ZSwgTW9kZWxNZXNzYWdlIH0gZnJvbSBcImFpXCI7XG5cbnR5cGUgUHJvdmlkZXJPcHRpb25zID0gUmVjb3JkPHN0cmluZywgUmVjb3JkPHN0cmluZywgSlNPTlZhbHVlPj47XG5cbmNvbnN0IEFOVEhST1BJQ19NQVhfQ0FDSEVfQlJFQUtQT0lOVFNfUEVSX1JFUVVFU1QgPSA0O1xuXG5jb25zdCBDTEFVREVfTElLRV9NT0RFTF9NQVRDSEVSUyA9IFtcImNsYXVkZVwiLCBcImFudGhyb3BpY1wiXTtcblxuY29uc3QgQ0xBVURFX1BST01QVF9DQUNISU5HX1BST1ZJREVSX09QVElPTlM6IFByb3ZpZGVyT3B0aW9ucyA9IHtcbiAgYW50aHJvcGljOiB7XG4gICAgY2FjaGVDb250cm9sOiB7IHR5cGU6IFwiZXBoZW1lcmFsXCIgfSxcbiAgfSxcbiAgb3BlbnJvdXRlcjoge1xuICAgIGNhY2hlQ29udHJvbDogeyB0eXBlOiBcImVwaGVtZXJhbFwiIH0sXG4gIH0sXG4gIGJlZHJvY2s6IHtcbiAgICBjYWNoZVBvaW50OiB7IHR5cGU6IFwiZGVmYXVsdFwiIH0sXG4gIH0sXG4gIG9wZW5haUNvbXBhdGlibGU6IHtcbiAgICBjYWNoZV9jb250cm9sOiB7IHR5cGU6IFwiZXBoZW1lcmFsXCIgfSxcbiAgfSxcbiAgY29waWxvdDoge1xuICAgIGNvcGlsb3RfY2FjaGVfY29udHJvbDogeyB0eXBlOiBcImVwaGVtZXJhbFwiIH0sXG4gIH0sXG59O1xuXG5mdW5jdGlvbiBnZXRHYXRld2F5UHJvdmlkZXIobW9kZWw6IHN0cmluZyk6IHN0cmluZyB8IG51bGwge1xuICBjb25zdCBzbGFzaEluZGV4ID0gbW9kZWwuaW5kZXhPZihcIi9cIik7XG4gIGlmIChzbGFzaEluZGV4ID09PSAtMSkge1xuICAgIHJldHVybiBudWxsO1xuICB9XG4gIHJldHVybiBtb2RlbC5zbGljZSgwLCBzbGFzaEluZGV4KTtcbn1cblxuZnVuY3Rpb24gY291bnRBbnRocm9waWNDYWNoZUJyZWFrcG9pbnRzKG1lc3NhZ2VzOiBNb2RlbE1lc3NhZ2VbXSk6IG51bWJlciB7XG4gIGxldCBjb3VudCA9IDA7XG4gIGZvciAoY29uc3QgbWVzc2FnZSBvZiBtZXNzYWdlcykge1xuICAgIGNvbnN0IHByb3ZpZGVyT3B0aW9ucyA9IChtZXNzYWdlIGFzIHsgcHJvdmlkZXJPcHRpb25zPzogUHJvdmlkZXJPcHRpb25zIH0pXG4gICAgICAucHJvdmlkZXJPcHRpb25zO1xuICAgIGlmIChwcm92aWRlck9wdGlvbnM/LmFudGhyb3BpYz8uY2FjaGVDb250cm9sKSB7XG4gICAgICBjb3VudCArPSAxO1xuICAgIH1cbiAgICBpZiAoQXJyYXkuaXNBcnJheShtZXNzYWdlLmNvbnRlbnQpKSB7XG4gICAgICBmb3IgKGNvbnN0IHBhcnQgb2YgbWVzc2FnZS5jb250ZW50KSB7XG4gICAgICAgIGNvbnN0IHBhcnRQcm92aWRlck9wdGlvbnMgPSAoXG4gICAgICAgICAgcGFydCBhcyB7IHByb3ZpZGVyT3B0aW9ucz86IFByb3ZpZGVyT3B0aW9ucyB9XG4gICAgICAgICkucHJvdmlkZXJPcHRpb25zO1xuICAgICAgICBpZiAocGFydFByb3ZpZGVyT3B0aW9ucz8uYW50aHJvcGljPy5jYWNoZUNvbnRyb2wpIHtcbiAgICAgICAgICBjb3VudCArPSAxO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuICB9XG4gIHJldHVybiBjb3VudDtcbn1cblxuZnVuY3Rpb24gaXNDbGF1ZGVMaWtlTW9kZWwobW9kZWw6IHN0cmluZyk6IGJvb2xlYW4ge1xuICBjb25zdCBsb3dlciA9IG1vZGVsLnRvTG93ZXJDYXNlKCk7XG4gIHJldHVybiBDTEFVREVfTElLRV9NT0RFTF9NQVRDSEVSUy5zb21lKChtKSA9PiBsb3dlci5pbmNsdWRlcyhtKSk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBtZXJnZVByb3ZpZGVyT3B0aW9ucyhvcHRzOiB7XG4gIGN1cnJlbnQ6IFByb3ZpZGVyT3B0aW9ucyB8IHVuZGVmaW5lZDtcbiAgcGF0Y2g6IFByb3ZpZGVyT3B0aW9ucztcbn0pOiBQcm92aWRlck9wdGlvbnMge1xuICBjb25zdCBuZXh0OiBQcm92aWRlck9wdGlvbnMgPSB7IC4uLihvcHRzLmN1cnJlbnQgPz8ge30pIH07XG4gIGZvciAoY29uc3QgW2tleSwgdmFsdWVdIG9mIE9iamVjdC5lbnRyaWVzKG9wdHMucGF0Y2gpKSB7XG4gICAgbmV4dFtrZXldID0geyAuLi4obmV4dFtrZXldID8/IHt9KSwgLi4udmFsdWUgfTtcbiAgfVxuICByZXR1cm4gbmV4dDtcbn1cblxuZnVuY3Rpb24gaXNDYWNoZWFibGVDbGF1ZGVQYXJ0KHBhcnQ6IHVua25vd24pOiBib29sZWFuIHtcbiAgaWYgKCFwYXJ0IHx8IHR5cGVvZiBwYXJ0ICE9PSBcIm9iamVjdFwiKSB7XG4gICAgcmV0dXJuIGZhbHNlO1xuICB9XG5cbiAgaWYgKFwidHlwZVwiIGluIHBhcnQgJiYgKHBhcnQgYXMgeyB0eXBlPzogdW5rbm93biB9KS50eXBlID09PSBcInRleHRcIikge1xuICAgIGNvbnN0IHRleHQgPSAocGFydCBhcyB7IHRleHQ/OiB1bmtub3duIH0pLnRleHQ7XG4gICAgaWYgKHR5cGVvZiB0ZXh0ID09PSBcInN0cmluZ1wiKSB7XG4gICAgICByZXR1cm4gdGV4dC50cmltKCkubGVuZ3RoID4gMDtcbiAgICB9XG4gIH1cblxuICBpZiAoXCJ0eXBlXCIgaW4gcGFydCkge1xuICAgIGNvbnN0IHR5cGUgPSAocGFydCBhcyB7IHR5cGU/OiB1bmtub3duIH0pLnR5cGU7XG4gICAgaWYgKHR5cGUgPT09IFwidGhpbmtpbmdcIiB8fCB0eXBlID09PSBcInJlYXNvbmluZ1wiKSB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuICB9XG5cbiAgcmV0dXJuIHRydWU7XG59XG5cbmZ1bmN0aW9uIGhhc0NhY2hlYWJsZUNvbnRlbnQobWVzc2FnZTogTW9kZWxNZXNzYWdlKTogYm9vbGVhbiB7XG4gIGlmICh0eXBlb2YgbWVzc2FnZS5jb250ZW50ID09PSBcInN0cmluZ1wiKSB7XG4gICAgcmV0dXJuIG1lc3NhZ2UuY29udGVudC50cmltKCkubGVuZ3RoID4gMDtcbiAgfVxuXG4gIGlmIChBcnJheS5pc0FycmF5KG1lc3NhZ2UuY29udGVudCkpIHtcbiAgICByZXR1cm4gbWVzc2FnZS5jb250ZW50LnNvbWUoKHBhcnQpID0+IGlzQ2FjaGVhYmxlQ2xhdWRlUGFydChwYXJ0KSk7XG4gIH1cblxuICByZXR1cm4gZmFsc2U7XG59XG5cbmZ1bmN0aW9uIGZpbmRMYXN0Q2FjaGVhYmxlUGFydEluZGV4KGNvbnRlbnQ6IHVua25vd25bXSk6IG51bWJlciB8IG51bGwge1xuICBmb3IgKGxldCBpID0gY29udGVudC5sZW5ndGggLSAxOyBpID49IDA7IGkgLT0gMSkge1xuICAgIGlmIChpc0NhY2hlYWJsZUNsYXVkZVBhcnQoY29udGVudFtpXSkpIHtcbiAgICAgIHJldHVybiBpO1xuICAgIH1cbiAgfVxuICByZXR1cm4gbnVsbDtcbn1cblxuZnVuY3Rpb24gc2VsZWN0Q2xhdWRlQ2FjaGluZ1RhcmdldHMobWVzc2FnZXM6IE1vZGVsTWVzc2FnZVtdKTogbnVtYmVyW10ge1xuICBjb25zdCBzeXN0ZW1JbmRpY2VzOiBudW1iZXJbXSA9IFtdO1xuICBjb25zdCBmaW5hbEluZGljZXM6IG51bWJlcltdID0gW107XG5cbiAgZm9yIChsZXQgaSA9IDA7IGkgPCBtZXNzYWdlcy5sZW5ndGggJiYgc3lzdGVtSW5kaWNlcy5sZW5ndGggPCAyOyBpICs9IDEpIHtcbiAgICBjb25zdCBtZXNzYWdlID0gbWVzc2FnZXNbaV07XG4gICAgaWYgKG1lc3NhZ2Uucm9sZSA9PT0gXCJzeXN0ZW1cIiAmJiBoYXNDYWNoZWFibGVDb250ZW50KG1lc3NhZ2UpKSB7XG4gICAgICBzeXN0ZW1JbmRpY2VzLnB1c2goaSk7XG4gICAgfVxuICB9XG5cbiAgZm9yIChsZXQgaSA9IG1lc3NhZ2VzLmxlbmd0aCAtIDE7IGkgPj0gMCAmJiBmaW5hbEluZGljZXMubGVuZ3RoIDwgMjsgaSAtPSAxKSB7XG4gICAgY29uc3QgbWVzc2FnZSA9IG1lc3NhZ2VzW2ldO1xuICAgIGlmIChtZXNzYWdlLnJvbGUgIT09IFwic3lzdGVtXCIgJiYgaGFzQ2FjaGVhYmxlQ29udGVudChtZXNzYWdlKSkge1xuICAgICAgZmluYWxJbmRpY2VzLnB1c2goaSk7XG4gICAgfVxuICB9XG5cbiAgZmluYWxJbmRpY2VzLnJldmVyc2UoKTtcblxuICByZXR1cm4gWy4uLnN5c3RlbUluZGljZXMsIC4uLmZpbmFsSW5kaWNlc107XG59XG5cbmZ1bmN0aW9uIGFwcGx5Q2xhdWRlUHJvbXB0Q2FjaGluZyhvcHRzOiB7XG4gIG1lc3NhZ2VzOiBNb2RlbE1lc3NhZ2VbXTtcbiAgcHJvdmlkZXJJZDogc3RyaW5nIHwgbnVsbDtcbiAgbWF4QnJlYWtwb2ludHNQZXJSZXF1ZXN0OiBudW1iZXI7XG59KTogTW9kZWxNZXNzYWdlW10ge1xuICBjb25zdCBleGlzdGluZ0JyZWFrcG9pbnRzID0gY291bnRBbnRocm9waWNDYWNoZUJyZWFrcG9pbnRzKG9wdHMubWVzc2FnZXMpO1xuICBjb25zdCByZW1haW5pbmdCdWRnZXQgPSBNYXRoLm1heChcbiAgICAwLFxuICAgIG9wdHMubWF4QnJlYWtwb2ludHNQZXJSZXF1ZXN0IC0gZXhpc3RpbmdCcmVha3BvaW50c1xuICApO1xuXG4gIGlmIChyZW1haW5pbmdCdWRnZXQgPT09IDApIHtcbiAgICByZXR1cm4gb3B0cy5tZXNzYWdlcztcbiAgfVxuXG4gIGNvbnN0IHRhcmdldEluZGljZXMgPSBzZWxlY3RDbGF1ZGVDYWNoaW5nVGFyZ2V0cyhvcHRzLm1lc3NhZ2VzKS5zbGljZShcbiAgICAwLFxuICAgIHJlbWFpbmluZ0J1ZGdldFxuICApO1xuICBpZiAodGFyZ2V0SW5kaWNlcy5sZW5ndGggPT09IDApIHtcbiAgICByZXR1cm4gb3B0cy5tZXNzYWdlcztcbiAgfVxuXG4gIGNvbnN0IHVzZU1lc3NhZ2VMZXZlbE9wdGlvbnMgPVxuICAgIG9wdHMucHJvdmlkZXJJZCA9PT0gXCJhbnRocm9waWNcIiB8fFxuICAgIEJvb2xlYW4ob3B0cy5wcm92aWRlcklkPy5pbmNsdWRlcyhcImJlZHJvY2tcIikpO1xuXG4gIGNvbnN0IG5leHRNZXNzYWdlcyA9IG9wdHMubWVzc2FnZXMuc2xpY2UoKTtcblxuICBmb3IgKGNvbnN0IG1lc3NhZ2VJbmRleCBvZiB0YXJnZXRJbmRpY2VzKSB7XG4gICAgY29uc3QgbWVzc2FnZSA9IG5leHRNZXNzYWdlc1ttZXNzYWdlSW5kZXhdO1xuXG4gICAgY29uc3Qgc2hvdWxkVXNlQ29udGVudE9wdGlvbnMgPVxuICAgICAgIXVzZU1lc3NhZ2VMZXZlbE9wdGlvbnMgJiZcbiAgICAgIEFycmF5LmlzQXJyYXkobWVzc2FnZS5jb250ZW50KSAmJlxuICAgICAgbWVzc2FnZS5jb250ZW50Lmxlbmd0aCA+IDA7XG5cbiAgICBpZiAoc2hvdWxkVXNlQ29udGVudE9wdGlvbnMgJiYgQXJyYXkuaXNBcnJheShtZXNzYWdlLmNvbnRlbnQpKSB7XG4gICAgICBjb25zdCBwYXJ0SW5kZXggPSBmaW5kTGFzdENhY2hlYWJsZVBhcnRJbmRleChtZXNzYWdlLmNvbnRlbnQpO1xuICAgICAgaWYgKHBhcnRJbmRleCAhPT0gbnVsbCkge1xuICAgICAgICBjb25zdCBwYXJ0ID0gbWVzc2FnZS5jb250ZW50W3BhcnRJbmRleF07XG4gICAgICAgIGlmIChwYXJ0ICYmIHR5cGVvZiBwYXJ0ID09PSBcIm9iamVjdFwiKSB7XG4gICAgICAgICAgY29uc3QgcGFydFByb3ZpZGVyT3B0aW9ucyA9IChcbiAgICAgICAgICAgIHBhcnQgYXMgeyBwcm92aWRlck9wdGlvbnM/OiBQcm92aWRlck9wdGlvbnMgfVxuICAgICAgICAgICkucHJvdmlkZXJPcHRpb25zO1xuXG4gICAgICAgICAgY29uc3QgbmV4dENvbnRlbnQgPSBtZXNzYWdlLmNvbnRlbnQuc2xpY2UoKTtcbiAgICAgICAgICBuZXh0Q29udGVudFtwYXJ0SW5kZXhdID0ge1xuICAgICAgICAgICAgLi4uKHBhcnQgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4pLFxuICAgICAgICAgICAgcHJvdmlkZXJPcHRpb25zOiBtZXJnZVByb3ZpZGVyT3B0aW9ucyh7XG4gICAgICAgICAgICAgIGN1cnJlbnQ6IHBhcnRQcm92aWRlck9wdGlvbnMsXG4gICAgICAgICAgICAgIHBhdGNoOiBDTEFVREVfUFJPTVBUX0NBQ0hJTkdfUFJPVklERVJfT1BUSU9OUyxcbiAgICAgICAgICAgIH0pLFxuICAgICAgICAgIH0gYXMgKHR5cGVvZiBtZXNzYWdlLmNvbnRlbnQpW251bWJlcl07XG5cbiAgICAgICAgICBuZXh0TWVzc2FnZXNbbWVzc2FnZUluZGV4XSA9IHtcbiAgICAgICAgICAgIC4uLm1lc3NhZ2UsXG4gICAgICAgICAgICBjb250ZW50OiBuZXh0Q29udGVudCxcbiAgICAgICAgICB9IGFzIE1vZGVsTWVzc2FnZTtcbiAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cblxuICAgIGNvbnN0IG1lc3NhZ2VQcm92aWRlck9wdGlvbnMgPSAoXG4gICAgICBtZXNzYWdlIGFzIHsgcHJvdmlkZXJPcHRpb25zPzogUHJvdmlkZXJPcHRpb25zIH1cbiAgICApLnByb3ZpZGVyT3B0aW9ucztcblxuICAgIG5leHRNZXNzYWdlc1ttZXNzYWdlSW5kZXhdID0ge1xuICAgICAgLi4ubWVzc2FnZSxcbiAgICAgIHByb3ZpZGVyT3B0aW9uczogbWVyZ2VQcm92aWRlck9wdGlvbnMoe1xuICAgICAgICBjdXJyZW50OiBtZXNzYWdlUHJvdmlkZXJPcHRpb25zLFxuICAgICAgICBwYXRjaDogQ0xBVURFX1BST01QVF9DQUNISU5HX1BST1ZJREVSX09QVElPTlMsXG4gICAgICB9KSxcbiAgICB9O1xuICB9XG5cbiAgcmV0dXJuIG5leHRNZXNzYWdlcztcbn1cblxuZXhwb3J0IHR5cGUgUHJvbXB0Q2FjaGluZ1Jlc3VsdCA9IHtcbiAgbWVzc2FnZXM6IE1vZGVsTWVzc2FnZVtdO1xuICBwcm92aWRlck9wdGlvbnM6IFByb3ZpZGVyT3B0aW9ucztcbn07XG5cbmV4cG9ydCBmdW5jdGlvbiBhcHBseVByb21wdENhY2hpbmdUb01vZGVsUmVxdWVzdChvcHRzOiB7XG4gIG1vZGVsOiBzdHJpbmc7XG4gIHNlc3Npb25JZDogc3RyaW5nO1xuICBtZXNzYWdlczogTW9kZWxNZXNzYWdlW107XG4gIGFudGhyb3BpYz86IHsgbWF4QnJlYWtwb2ludHNQZXJSZXF1ZXN0PzogbnVtYmVyIH07XG4gIG9wZW5haT86IHsgc2V0UHJvbXB0Q2FjaGVLZXk/OiBib29sZWFuIH07XG59KTogUHJvbXB0Q2FjaGluZ1Jlc3VsdCB7XG4gIGNvbnN0IHByb3ZpZGVyID0gZ2V0R2F0ZXdheVByb3ZpZGVyKG9wdHMubW9kZWwpO1xuXG4gIGNvbnN0IHByb3ZpZGVyT3B0aW9uczogUHJvdmlkZXJPcHRpb25zID0ge307XG5cbiAgaWYgKFxuICAgIChwcm92aWRlciA9PT0gXCJvcGVuYWlcIiB8fCBwcm92aWRlciA9PT0gXCJhenVyZVwiKSAmJlxuICAgIG9wdHMub3BlbmFpPy5zZXRQcm9tcHRDYWNoZUtleSAhPT0gZmFsc2VcbiAgKSB7XG4gICAgcHJvdmlkZXJPcHRpb25zLm9wZW5haSA9IHsgcHJvbXB0Q2FjaGVLZXk6IG9wdHMuc2Vzc2lvbklkIH07XG4gIH1cblxuICBjb25zdCBtYXhCcmVha3BvaW50c1BlclJlcXVlc3QgPVxuICAgIG9wdHMuYW50aHJvcGljPy5tYXhCcmVha3BvaW50c1BlclJlcXVlc3QgPz9cbiAgICBBTlRIUk9QSUNfTUFYX0NBQ0hFX0JSRUFLUE9JTlRTX1BFUl9SRVFVRVNUO1xuXG4gIGNvbnN0IGNhY2hlZE1lc3NhZ2VzID0gaXNDbGF1ZGVMaWtlTW9kZWwob3B0cy5tb2RlbClcbiAgICA/IGFwcGx5Q2xhdWRlUHJvbXB0Q2FjaGluZyh7XG4gICAgICAgIG1lc3NhZ2VzOiBvcHRzLm1lc3NhZ2VzLFxuICAgICAgICBwcm92aWRlcklkOiBwcm92aWRlcixcbiAgICAgICAgbWF4QnJlYWtwb2ludHNQZXJSZXF1ZXN0LFxuICAgICAgfSlcbiAgICA6IG9wdHMubWVzc2FnZXM7XG5cbiAgcmV0dXJuIHtcbiAgICBtZXNzYWdlczogY2FjaGVkTWVzc2FnZXMsXG4gICAgcHJvdmlkZXJPcHRpb25zLFxuICB9O1xufVxuIiwgImltcG9ydCB0eXBlIHtcbiAgSW5mZXJVSVRvb2xzLFxuICBSZWFzb25pbmdVSVBhcnQsXG4gIFRleHRVSVBhcnQsXG4gIFRvb2xTZXQsXG4gIFVJTWVzc2FnZSxcbn0gZnJvbSBcImFpXCI7XG5pbXBvcnQgdHlwZSB7IEFnZW50RGF0YVR5cGVzIH0gZnJvbSBcIi4uL2FnZW50L3JlZ2lzdHJ5XCI7XG5pbXBvcnQgdHlwZSB7IE1lc3NhZ2UsIFBhcnQgfSBmcm9tIFwiLi4vc3RvcmFnZVwiO1xuXG4vKiogU3ViLXRvb2wgcGFydHMgKGZyb20gSmF2YVNjcmlwdCBtZXRhLXRvb2wpIHVzZSB0aGlzIHRvb2xDYWxsSWQgcHJlZml4LiAqL1xuZnVuY3Rpb24gaXNTdWJUb29sUGFydChwYXJ0OiBQYXJ0KTogYm9vbGVhbiB7XG4gIGNvbnN0IHAgPSBwYXJ0LnBhcnQgYXMgeyB0b29sQ2FsbElkPzogc3RyaW5nIH07XG4gIHJldHVybiB0eXBlb2YgcD8udG9vbENhbGxJZCA9PT0gXCJzdHJpbmdcIiAmJiBwLnRvb2xDYWxsSWQuc3RhcnRzV2l0aChcImpzX3RjX1wiKTtcbn1cblxuZnVuY3Rpb24gaXNTdHJlYW1pbmdDb250ZW50KFxuICBwYXJ0OiB1bmtub3duXG4pOiBwYXJ0IGlzIFRleHRVSVBhcnQgfCBSZWFzb25pbmdVSVBhcnQge1xuICByZXR1cm4gKFxuICAgIHR5cGVvZiBwYXJ0ID09PSBcIm9iamVjdFwiICYmXG4gICAgcGFydCAhPT0gbnVsbCAmJlxuICAgIFwidHlwZVwiIGluIHBhcnQgJiZcbiAgICAocGFydC50eXBlID09PSBcInRleHRcIiB8fCBwYXJ0LnR5cGUgPT09IFwicmVhc29uaW5nXCIpXG4gICk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBhcHBseUludGVycnVwdEN1dG9mZih7XG4gIHBhcnRzLFxuICBsYXN0UGFydCxcbn06IHtcbiAgcGFydHM6IFBhcnRbXTtcbiAgbGFzdFBhcnQ6IHsgaW5kZXg6IG51bWJlcjsgcGFydDogdW5rbm93biB9O1xufSk6IFBhcnRbXSB7XG4gIHJldHVybiBwYXJ0c1xuICAgIC5maWx0ZXIoKHApID0+IHAuaW5kZXggPD0gbGFzdFBhcnQuaW5kZXgpXG4gICAgLm1hcCgocCkgPT4ge1xuICAgICAgaWYgKHAuaW5kZXggIT09IGxhc3RQYXJ0LmluZGV4KSB7XG4gICAgICAgIHJldHVybiBwO1xuICAgICAgfVxuICAgICAgLyoqXG4gICAgICAgKiBPbmx5IHJlcGxhY2UgY29udGVudCBmb3Igc3RyZWFtaW5nIHBhcnRzICh0ZXh0LCByZWFzb25pbmcpIHdoZXJlIHRoZVxuICAgICAgICogY2xpZW50J3MgdHJ1bmNhdGVkIHZlcnNpb24gaXMgbW9yZSBhY2N1cmF0ZSB0aGFuIHRoZSBzZXJ2ZXIncyBmdWxsXG4gICAgICAgKiB2ZXJzaW9uLiBGb3Igc3RhdGVmdWwgcGFydHMgKHRvb2xzKSwgdGhlIHNlcnZlcidzIHZlcnNpb24gaXNcbiAgICAgICAqIGF1dGhvcml0YXRpdmUgXHUyMDE0IGl0IG1heSBoYXZlIGV2b2x2ZWQgKGUuZy4gYXBwcm92YWwtcmVxdWVzdGVkIFx1MjE5MlxuICAgICAgICogb3V0cHV0LWRlbmllZCkgYWZ0ZXIgdGhlIGNsaWVudCBzbmFwc2hvdCB3YXMgdGFrZW4uXG4gICAgICAgKi9cbiAgICAgIGlmIChpc1N0cmVhbWluZ0NvbnRlbnQobGFzdFBhcnQucGFydCkpIHtcbiAgICAgICAgcmV0dXJuIHsgLi4ucCwgcGFydDogbGFzdFBhcnQucGFydCB9IGFzIFBhcnQ7XG4gICAgICB9XG4gICAgICByZXR1cm4gcDtcbiAgICB9KTtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGFzc2VtYmxlVUlNZXNzYWdlczxUb29scyBleHRlbmRzIFRvb2xTZXQ+KG9wdHM6IHtcbiAgbWVzc2FnZXM6IE1lc3NhZ2VbXTtcbiAgcGFydHM6IFBhcnRbXTtcbiAgdW50aWw/OiBudW1iZXI7XG4gIGluY2x1ZGVRdWV1ZWQ/OiBib29sZWFuO1xuICAvKiogV2hlbiB0cnVlLCBmaWx0ZXJzIG91dCBzdWItdG9vbCBwYXJ0cyBzbyB0aGUgTExNIGRvZXNuJ3Qgc2VlIHRoZW0uICovXG4gIGV4Y2x1ZGVTdWJUb29sUGFydHM/OiBib29sZWFuO1xufSkge1xuICB0eXBlIFR5cGVkVUlNZXNzYWdlID0gVUlNZXNzYWdlPHVua25vd24sIEFnZW50RGF0YVR5cGVzLCBJbmZlclVJVG9vbHM8VG9vbHM+PjtcblxuICBsZXQgZmlsdGVyZWQgPSBvcHRzLm1lc3NhZ2VzO1xuXG4gIGlmIChvcHRzLnVudGlsICE9PSB1bmRlZmluZWQpIHtcbiAgICBjb25zdCB1bnRpbCA9IG9wdHMudW50aWw7XG4gICAgZmlsdGVyZWQgPSBmaWx0ZXJlZC5maWx0ZXIoKG0pID0+IG0uY3JlYXRlZEF0IDw9IHVudGlsKTtcbiAgfVxuXG4gIGlmICghb3B0cy5pbmNsdWRlUXVldWVkKSB7XG4gICAgZmlsdGVyZWQgPSBmaWx0ZXJlZC5maWx0ZXIoKG0pID0+IG0uc3RhcnRlZEF0ICE9PSBudWxsKTtcbiAgfVxuXG4gIC8vIFNvcnQgYnkgY3JlYXRlZEF0IHNvIG1lc3NhZ2VzIGFyZSBpbiBjaHJvbm9sb2dpY2FsIG9yZGVyIHJlZ2FyZGxlc3Mgb2ZcbiAgLy8gaG93IHRoZSBzdG9yYWdlIGxheWVyIHJldHVybnMgdGhlbSAoZS5nLiBsb2NhbCBGUyByZWFkcyBpbiByZWFkZGlyIG9yZGVyKS5cbiAgZmlsdGVyZWQgPSBbLi4uZmlsdGVyZWRdLnNvcnQoKGEsIGIpID0+IGEuY3JlYXRlZEF0IC0gYi5jcmVhdGVkQXQpO1xuXG4gIGNvbnN0IHBhcnRzID0gb3B0cy5leGNsdWRlU3ViVG9vbFBhcnRzXG4gICAgPyBvcHRzLnBhcnRzLmZpbHRlcigocCkgPT4gIWlzU3ViVG9vbFBhcnQocCkpXG4gICAgOiBvcHRzLnBhcnRzO1xuXG4gIGNvbnN0IHBhcnRzQnlNZXNzYWdlID0gbmV3IE1hcDxzdHJpbmcsIFBhcnRbXT4oKTtcbiAgZm9yIChjb25zdCBwYXJ0IG9mIHBhcnRzKSB7XG4gICAgY29uc3QgbWVzc2FnZUlkID0gU3RyaW5nKHBhcnQubWVzc2FnZUlkKTtcbiAgICBjb25zdCBleGlzdGluZyA9IHBhcnRzQnlNZXNzYWdlLmdldChtZXNzYWdlSWQpID8/IFtdO1xuICAgIGV4aXN0aW5nLnB1c2gocGFydCk7XG4gICAgcGFydHNCeU1lc3NhZ2Uuc2V0KG1lc3NhZ2VJZCwgZXhpc3RpbmcpO1xuICB9XG5cbiAgcmV0dXJuIGZpbHRlcmVkXG4gICAgLm1hcCgobSkgPT4ge1xuICAgICAgY29uc3QgbWVzc2FnZUlkID0gU3RyaW5nKG0uaWQpO1xuICAgICAgbGV0IG1lc3NhZ2VQYXJ0cyA9IHBhcnRzQnlNZXNzYWdlLmdldChtZXNzYWdlSWQpID8/IFtdO1xuICAgICAgbWVzc2FnZVBhcnRzLnNvcnQoKGEsIGIpID0+IGEuaW5kZXggLSBiLmluZGV4KTtcblxuICAgICAgaWYgKG0uaW50ZXJydXB0ZWRMYXN0UGFydCAhPSBudWxsKSB7XG4gICAgICAgIG1lc3NhZ2VQYXJ0cyA9IGFwcGx5SW50ZXJydXB0Q3V0b2ZmKHtcbiAgICAgICAgICBwYXJ0czogbWVzc2FnZVBhcnRzLFxuICAgICAgICAgIGxhc3RQYXJ0OiBtLmludGVycnVwdGVkTGFzdFBhcnQsXG4gICAgICAgIH0pO1xuICAgICAgfVxuXG4gICAgICByZXR1cm4ge1xuICAgICAgICBpZDogbWVzc2FnZUlkLFxuICAgICAgICByb2xlOiBtLnJvbGUsXG4gICAgICAgIHBhcnRzOiBtZXNzYWdlUGFydHMubWFwKChwKSA9PiBwLnBhcnQpLFxuICAgICAgICAuLi4obS5tZXRhZGF0YSAhPSBudWxsID8geyBtZXRhZGF0YTogbS5tZXRhZGF0YSB9IDoge30pLFxuICAgICAgfSBhcyBUeXBlZFVJTWVzc2FnZTtcbiAgICB9KVxuICAgIC5maWx0ZXIoKG0pID0+IG0ucGFydHMubGVuZ3RoID4gMCk7XG59XG4iLCAiaW1wb3J0IHR5cGUgeyBVSU1lc3NhZ2UgfSBmcm9tIFwiYWlcIjtcbmltcG9ydCB0eXBlIHsgU2Vzc2lvbkNvbnRleHQgfSBmcm9tIFwiLi9oYW5kbGVcIjtcblxuZXhwb3J0IHR5cGUgSW50ZXJydXB0T3B0aW9ucyA9IHtcbiAgbGFzdFBhcnQ/OiB7XG4gICAgaW5kZXg6IG51bWJlcjtcbiAgICBwYXJ0OiB1bmtub3duO1xuICB9O1xufTtcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGludGVycnVwdChcbiAgc2Vzc2lvbkNvbnRleHQ6IFNlc3Npb25Db250ZXh0LFxuICBvcHRzPzogSW50ZXJydXB0T3B0aW9uc1xuKSB7XG4gIFwidXNlIHN0ZXBcIjtcbiAgY29uc3QgbWVzc2FnZXMgPSBhd2FpdCBzZXNzaW9uQ29udGV4dC5hZ2VudC5zdG9yYWdlLm1lc3NhZ2UubGlzdEJ5U2Vzc2lvbihcbiAgICBzZXNzaW9uQ29udGV4dC5zZXNzaW9uSWRcbiAgKTtcblxuICBjb25zdCBub3cgPSBEYXRlLm5vdygpO1xuICBsZXQgcGVuZGluZ0Fzc2lzdGFudE1lc3NhZ2VzID0gbWVzc2FnZXMuZmlsdGVyKFxuICAgIChtKSA9PlxuICAgICAgbS5yb2xlID09PSBcImFzc2lzdGFudFwiICYmXG4gICAgICBtLmNvbXBsZXRlZEF0ID09PSBudWxsICYmXG4gICAgICBtLmludGVycnVwdGVkQXQgPT09IG51bGxcbiAgKTtcblxuICBpZiAocGVuZGluZ0Fzc2lzdGFudE1lc3NhZ2VzLmxlbmd0aCA9PT0gMCkge1xuICAgIGNvbnN0IE1BWF9SRVRST0FDVElWRV9NUyA9IDVfMDAwO1xuICAgIGNvbnN0IGxhdGVzdCA9IG1lc3NhZ2VzXG4gICAgICAuZmlsdGVyKFxuICAgICAgICAobSkgPT5cbiAgICAgICAgICBtLnJvbGUgPT09IFwiYXNzaXN0YW50XCIgJiZcbiAgICAgICAgICBtLmludGVycnVwdGVkQXQgPT09IG51bGwgJiZcbiAgICAgICAgICBtLmNvbXBsZXRlZEF0ICE9PSBudWxsICYmXG4gICAgICAgICAgbm93IC0gbS5jb21wbGV0ZWRBdCA8IE1BWF9SRVRST0FDVElWRV9NU1xuICAgICAgKVxuICAgICAgLnNvcnQoKGEsIGIpID0+IGIuY3JlYXRlZEF0IC0gYS5jcmVhdGVkQXQpXG4gICAgICAuYXQoMCk7XG4gICAgaWYgKGxhdGVzdCkge1xuICAgICAgcGVuZGluZ0Fzc2lzdGFudE1lc3NhZ2VzID0gW2xhdGVzdF07XG4gICAgfVxuICB9XG5cbiAgYXdhaXQgUHJvbWlzZS5hbGwoXG4gICAgcGVuZGluZ0Fzc2lzdGFudE1lc3NhZ2VzLm1hcCgobSkgPT5cbiAgICAgIHNlc3Npb25Db250ZXh0LmFnZW50LnN0b3JhZ2UubWVzc2FnZS5zZXQobS5pZCwge1xuICAgICAgICAuLi5tLFxuICAgICAgICBpbnRlcnJ1cHRlZEF0OiBub3csXG4gICAgICAgIGludGVycnVwdGVkTGFzdFBhcnQ6IG9wdHM/Lmxhc3RQYXJ0ID8/IG51bGwsXG4gICAgICB9KVxuICAgIClcbiAgKTtcblxuICBjb25zdCBtZXNzYWdlSWRzID0gbmV3IFNldChwZW5kaW5nQXNzaXN0YW50TWVzc2FnZXMubWFwKChtKSA9PiBtLmlkKSk7XG4gIGlmIChtZXNzYWdlSWRzLnNpemUgPT09IDApIHtcbiAgICByZXR1cm47XG4gIH1cblxuICBjb25zdCBhbGxQYXJ0cyA9IGF3YWl0IHNlc3Npb25Db250ZXh0LmFnZW50LnN0b3JhZ2UucGFydC5saXN0QnlTZXNzaW9uKFxuICAgIHNlc3Npb25Db250ZXh0LnNlc3Npb25JZFxuICApO1xuXG4gIGZvciAoY29uc3QgcCBvZiBhbGxQYXJ0cykge1xuICAgIGlmICghbWVzc2FnZUlkcy5oYXMocC5tZXNzYWdlSWQpKSB7XG4gICAgICBjb250aW51ZTtcbiAgICB9XG4gICAgaWYgKFxuICAgICAgcC5wYXJ0ICYmXG4gICAgICBcInN0YXRlXCIgaW4gcC5wYXJ0ICYmXG4gICAgICAocC5wYXJ0IGFzIHsgc3RhdGU6IHN0cmluZyB9KS5zdGF0ZSA9PT0gXCJhcHByb3ZhbC1yZXF1ZXN0ZWRcIiAmJlxuICAgICAgXCJhcHByb3ZhbFwiIGluIHAucGFydFxuICAgICkge1xuICAgICAgY29uc3QgYXBwcm92YWwgPSAocC5wYXJ0IGFzIHsgYXBwcm92YWw/OiB7IGlkOiBzdHJpbmcgfSB9KS5hcHByb3ZhbDtcbiAgICAgIGlmIChhcHByb3ZhbD8uaWQpIHtcbiAgICAgICAgYXdhaXQgc2Vzc2lvbkNvbnRleHQuYWdlbnQuc3RvcmFnZS5wYXJ0LnNldChwLmlkLCB7XG4gICAgICAgICAgLi4ucCxcbiAgICAgICAgICBwYXJ0OiB7XG4gICAgICAgICAgICAuLi5wLnBhcnQsXG4gICAgICAgICAgICBzdGF0ZTogXCJvdXRwdXQtZGVuaWVkXCIsXG4gICAgICAgICAgICBhcHByb3ZhbDoge1xuICAgICAgICAgICAgICAuLi5hcHByb3ZhbCxcbiAgICAgICAgICAgICAgYXBwcm92ZWQ6IGZhbHNlLFxuICAgICAgICAgICAgICByZWFzb246IFwiaW50ZXJydXB0ZWRcIixcbiAgICAgICAgICAgIH0sXG4gICAgICAgICAgfSBhcyBVSU1lc3NhZ2VbXCJwYXJ0c1wiXVtudW1iZXJdLFxuICAgICAgICB9KTtcbiAgICAgIH1cbiAgICB9XG4gIH1cbn1cbiIsICJpbXBvcnQgeyB0eXBlIFRvb2xTZXQsIHRvb2wgfSBmcm9tIFwiYWlcIjtcbmltcG9ydCB7IHogfSBmcm9tIFwiem9kXCI7XG5pbXBvcnQgdHlwZSB7IFNhbmRib3hJbnN0YW5jZSB9IGZyb20gXCIuLi9zYW5kYm94L2FkYXB0ZXJcIjtcbmltcG9ydCB7IGJ1aWx0SW5Ub29scyB9IGZyb20gXCIuXCI7XG5cbmV4cG9ydCBjb25zdCBKQVZBU0NSSVBUX1RPT0xfTkFNRSA9IFwiSmF2YVNjcmlwdFwiO1xuXG5leHBvcnQgdHlwZSBTdWJUb29sQ2FsbFJlc3VsdCA9IHsgcmVzdWx0PzogdW5rbm93bjsgZXJyb3I/OiBzdHJpbmcgfTtcbmV4cG9ydCB0eXBlIE9uU3ViVG9vbENhbGwgPSAoXG4gIHRvb2xOYW1lOiBzdHJpbmcsXG4gIGlucHV0OiB1bmtub3duLFxuICBleGVjdXRlOiAoKSA9PiBQcm9taXNlPHVua25vd24+XG4pID0+IFByb21pc2U8U3ViVG9vbENhbGxSZXN1bHQ+O1xuXG5jb25zdCBSRVFVRVNUX01BUktFUl9TVEFSVCA9IFwiX19UT09MX1JFUVVFU1RfX1wiO1xuY29uc3QgUkVRVUVTVF9NQVJLRVJfRU5EID0gXCJfX1RPT0xfUkVRVUVTVF9FTkRfX1wiO1xuY29uc3QgRVhFQ1VUSU9OX1RJTUVPVVRfTVMgPSA1ICogNjAgKiAxMDAwO1xuXG4vKipcbiAqIFRoZSBydW5uZXIgc2NyaXB0IGlzIHdyaXR0ZW4gdG8gdGhlIHNhbmRib3ggYW5kIGV4ZWN1dGVkIHZpYSBOb2RlLmpzLlxuICogSXQgcmVhZHMgdGhlIG1vZGVsJ3MgY29kZSBmcm9tIGNvZGUuanMsIHNldHMgdXAgY3R4LnRvb2xzIGFzIFJQQyBwcm94aWVzXG4gKiAocmVxdWVzdHMgc2VudCB2aWEgc3RkZXJyIG1hcmtlcnMsIHJlc3BvbnNlcyBwb2xsZWQgZnJvbSBmaWxlcyksIGV4ZWN1dGVzXG4gKiB0aGUgY29kZSwgYW5kIHdyaXRlcyB0aGUgcmVzdWx0IHRvIHN0ZG91dCBhcyBKU09OLlxuICovXG5jb25zdCBSVU5ORVJfU0NSSVBUID0gYFxuaW1wb3J0IHsgcmVhZEZpbGVTeW5jLCBleGlzdHNTeW5jLCB1bmxpbmtTeW5jLCBta2RpclN5bmMgfSBmcm9tICdub2RlOmZzJztcbmltcG9ydCB7IGpvaW4sIGRpcm5hbWUgfSBmcm9tICdub2RlOnBhdGgnO1xuaW1wb3J0IHsgZmlsZVVSTFRvUGF0aCB9IGZyb20gJ25vZGU6dXJsJztcblxuY29uc3QgU0NSSVBUX0RJUiA9IGRpcm5hbWUoZmlsZVVSTFRvUGF0aChpbXBvcnQubWV0YS51cmwpKTtcbmNvbnN0IFRPT0xfQ0FMTF9USU1FT1VUX01TID0gMzAwXzAwMDtcblxubGV0IHJlcUNvdW50ZXIgPSAwO1xuXG5hc3luYyBmdW5jdGlvbiBjYWxsVG9vbChuYW1lLCBpbnB1dCkge1xuICBjb25zdCBpZCA9ICdyZXFfJyArICgrK3JlcUNvdW50ZXIpICsgJ18nICsgRGF0ZS5ub3coKTtcbiAgY29uc3QgcmVzcG9uc2VGaWxlID0gam9pbihTQ1JJUFRfRElSLCBpZCArICcucmVzcG9uc2UuanNvbicpO1xuXG4gIGNvbnN0IHJlcXVlc3QgPSBKU09OLnN0cmluZ2lmeSh7IGlkLCB0b29sOiBuYW1lLCBpbnB1dCB9KTtcbiAgcHJvY2Vzcy5zdGRlcnIud3JpdGUoJyR7UkVRVUVTVF9NQVJLRVJfU1RBUlR9JyArIHJlcXVlc3QgKyAnJHtSRVFVRVNUX01BUktFUl9FTkR9JyArICdcXFxcbicpO1xuXG4gIGNvbnN0IHN0YXJ0ID0gRGF0ZS5ub3coKTtcbiAgd2hpbGUgKCFleGlzdHNTeW5jKHJlc3BvbnNlRmlsZSkpIHtcbiAgICBpZiAoRGF0ZS5ub3coKSAtIHN0YXJ0ID4gVE9PTF9DQUxMX1RJTUVPVVRfTVMpIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcignVG9vbCBjYWxsICcgKyBuYW1lICsgJyB0aW1lZCBvdXQgd2FpdGluZyBmb3IgcmVzcG9uc2UnKTtcbiAgICB9XG4gICAgYXdhaXQgbmV3IFByb21pc2UociA9PiBzZXRUaW1lb3V0KHIsIDMwKSk7XG4gIH1cblxuICBsZXQgcmF3O1xuICB0cnkge1xuICAgIHJhdyA9IHJlYWRGaWxlU3luYyhyZXNwb25zZUZpbGUsICd1dGY4Jyk7XG4gIH0gY2F0Y2gge1xuICAgIGF3YWl0IG5ldyBQcm9taXNlKHIgPT4gc2V0VGltZW91dChyLCA1MCkpO1xuICAgIHJhdyA9IHJlYWRGaWxlU3luYyhyZXNwb25zZUZpbGUsICd1dGY4Jyk7XG4gIH1cblxuICBjb25zdCByZXNwb25zZSA9IEpTT04ucGFyc2UocmF3KTtcbiAgdHJ5IHsgdW5saW5rU3luYyhyZXNwb25zZUZpbGUpOyB9IGNhdGNoIHt9XG5cbiAgaWYgKHJlc3BvbnNlLmVycm9yKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKHJlc3BvbnNlLmVycm9yKTtcbiAgfVxuICByZXR1cm4gcmVzcG9uc2UucmVzdWx0O1xufVxuXG5jb25zdCBjb2RlID0gcmVhZEZpbGVTeW5jKGpvaW4oU0NSSVBUX0RJUiwgJ2NvZGUuanMnKSwgJ3V0ZjgnKTtcbmNvbnN0IHRvb2xOYW1lcyA9IEpTT04ucGFyc2UocmVhZEZpbGVTeW5jKGpvaW4oU0NSSVBUX0RJUiwgJ3Rvb2xzLmpzb24nKSwgJ3V0ZjgnKSk7XG5cbmNvbnN0IGN0eCA9IHsgdG9vbHM6IHt9IH07XG5mb3IgKGNvbnN0IG5hbWUgb2YgdG9vbE5hbWVzKSB7XG4gIGN0eC50b29sc1tuYW1lXSA9IChpbnB1dCkgPT4gY2FsbFRvb2wobmFtZSwgaW5wdXQpO1xufVxuXG5jb25zdCBsb2dzID0gW107XG5jb25zdCBjdXN0b21Db25zb2xlID0ge1xuICBsb2c6ICguLi5hcmdzKSA9PiBsb2dzLnB1c2goYXJncy5tYXAoYSA9PiB0eXBlb2YgYSA9PT0gJ3N0cmluZycgPyBhIDogSlNPTi5zdHJpbmdpZnkoYSkpLmpvaW4oJyAnKSksXG4gIGVycm9yOiAoLi4uYXJncykgPT4gbG9ncy5wdXNoKCdbZXJyb3JdICcgKyBhcmdzLm1hcChhID0+IHR5cGVvZiBhID09PSAnc3RyaW5nJyA/IGEgOiBKU09OLnN0cmluZ2lmeShhKSkuam9pbignICcpKSxcbiAgd2FybjogKC4uLmFyZ3MpID0+IGxvZ3MucHVzaCgnW3dhcm5dICcgKyBhcmdzLm1hcChhID0+IHR5cGVvZiBhID09PSAnc3RyaW5nJyA/IGEgOiBKU09OLnN0cmluZ2lmeShhKSkuam9pbignICcpKSxcbn07XG5cbnRyeSB7XG4gIGNvbnN0IEFzeW5jRnVuY3Rpb24gPSBPYmplY3QuZ2V0UHJvdG90eXBlT2YoYXN5bmMgZnVuY3Rpb24oKSB7fSkuY29uc3RydWN0b3I7XG4gIGNvbnN0IGZuID0gbmV3IEFzeW5jRnVuY3Rpb24oJ2N0eCcsICdjb25zb2xlJywgY29kZSk7XG4gIGNvbnN0IHJlc3VsdCA9IGF3YWl0IGZuKGN0eCwgY3VzdG9tQ29uc29sZSk7XG5cbiAgY29uc3Qgb3V0cHV0ID0geyBzdWNjZXNzOiB0cnVlLCByZXN1bHQ6IHJlc3VsdCAhPT0gdW5kZWZpbmVkID8gcmVzdWx0IDogbnVsbCB9O1xuICBpZiAobG9ncy5sZW5ndGggPiAwKSBvdXRwdXQubG9ncyA9IGxvZ3M7XG4gIHByb2Nlc3Muc3Rkb3V0LndyaXRlKEpTT04uc3RyaW5naWZ5KG91dHB1dCkpO1xufSBjYXRjaCAoZXJyKSB7XG4gIGNvbnN0IG91dHB1dCA9IHtcbiAgICBzdWNjZXNzOiBmYWxzZSxcbiAgICBlcnJvcjogZXJyIGluc3RhbmNlb2YgRXJyb3IgPyBlcnIubWVzc2FnZSA6IFN0cmluZyhlcnIpLFxuICB9O1xuICBpZiAobG9ncy5sZW5ndGggPiAwKSBvdXRwdXQubG9ncyA9IGxvZ3M7XG4gIHByb2Nlc3Muc3Rkb3V0LndyaXRlKEpTT04uc3RyaW5naWZ5KG91dHB1dCkpO1xufVxuYDtcblxuLyoqXG4gKiBFeHRyYWN0cyBhIHBsYWluIEpTT04gU2NoZW1hIGZyb20gd2hhdGV2ZXIgZm9ybWF0IHRoZSBBSSBTREsgdG9vbCB1c2VzXG4gKiAoWm9kIHNjaGVtYSwgQUkgU0RLIHdyYXBwZWQgc2NoZW1hLCBvciByYXcgSlNPTiBTY2hlbWEpLlxuICovXG5leHBvcnQgZnVuY3Rpb24gZXh0cmFjdEpzb25TY2hlbWEoXG4gIHNjaGVtYTogdW5rbm93blxuKTogUmVjb3JkPHN0cmluZywgdW5rbm93bj4gfCBudWxsIHtcbiAgaWYgKCFzY2hlbWEgfHwgdHlwZW9mIHNjaGVtYSAhPT0gXCJvYmplY3RcIikge1xuICAgIHJldHVybiBudWxsO1xuICB9XG5cbiAgaWYgKFxuICAgIFwiX2RlZlwiIGluIHNjaGVtYSAmJlxuICAgIHR5cGVvZiAoc2NoZW1hIGFzIHsgX2RlZjogdW5rbm93biB9KS5fZGVmID09PSBcIm9iamVjdFwiXG4gICkge1xuICAgIHRyeSB7XG4gICAgICByZXR1cm4gei50b0pTT05TY2hlbWEoc2NoZW1hIGFzIHouWm9kVHlwZSkgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj47XG4gICAgfSBjYXRjaCB7XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG4gIH1cblxuICBjb25zdCBzID0gc2NoZW1hIGFzIFJlY29yZDxzdHJpbmcgfCBzeW1ib2wsIHVua25vd24+O1xuICBpZiAoU3ltYm9sLmZvcihcInZlcmNlbC5haS5zY2hlbWFcIikgaW4gcyAmJiBcImpzb25TY2hlbWFcIiBpbiBzKSB7XG4gICAgcmV0dXJuIHMuanNvblNjaGVtYSBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgfVxuXG4gIGlmIChcInR5cGVcIiBpbiBzIHx8IFwicHJvcGVydGllc1wiIGluIHMpIHtcbiAgICByZXR1cm4gcyBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgfVxuXG4gIHJldHVybiBudWxsO1xufVxuXG5leHBvcnQgZnVuY3Rpb24ganNvblNjaGVtYVRvVHMoXG4gIHNjaGVtYTogUmVjb3JkPHN0cmluZywgdW5rbm93bj4sXG4gIGluZGVudCA9IDBcbik6IHN0cmluZyB7XG4gIGlmICghc2NoZW1hKSB7XG4gICAgcmV0dXJuIFwidW5rbm93blwiO1xuICB9XG5cbiAgaWYgKHNjaGVtYS5lbnVtICYmIEFycmF5LmlzQXJyYXkoc2NoZW1hLmVudW0pKSB7XG4gICAgcmV0dXJuIHNjaGVtYS5lbnVtLm1hcCgodikgPT4gSlNPTi5zdHJpbmdpZnkodikpLmpvaW4oXCIgfCBcIik7XG4gIH1cblxuICBpZiAoc2NoZW1hLmFueU9mICYmIEFycmF5LmlzQXJyYXkoc2NoZW1hLmFueU9mKSkge1xuICAgIHJldHVybiBzY2hlbWEuYW55T2ZcbiAgICAgIC5tYXAoKHM6IFJlY29yZDxzdHJpbmcsIHVua25vd24+KSA9PiBqc29uU2NoZW1hVG9UcyhzLCBpbmRlbnQpKVxuICAgICAgLmpvaW4oXCIgfCBcIik7XG4gIH1cbiAgaWYgKHNjaGVtYS5vbmVPZiAmJiBBcnJheS5pc0FycmF5KHNjaGVtYS5vbmVPZikpIHtcbiAgICByZXR1cm4gc2NoZW1hLm9uZU9mXG4gICAgICAubWFwKChzOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPikgPT4ganNvblNjaGVtYVRvVHMocywgaW5kZW50KSlcbiAgICAgIC5qb2luKFwiIHwgXCIpO1xuICB9XG5cbiAgY29uc3QgdHlwZSA9IHNjaGVtYS50eXBlIGFzIHN0cmluZztcblxuICBzd2l0Y2ggKHR5cGUpIHtcbiAgICBjYXNlIFwic3RyaW5nXCI6XG4gICAgICByZXR1cm4gXCJzdHJpbmdcIjtcbiAgICBjYXNlIFwibnVtYmVyXCI6XG4gICAgY2FzZSBcImludGVnZXJcIjpcbiAgICAgIHJldHVybiBcIm51bWJlclwiO1xuICAgIGNhc2UgXCJib29sZWFuXCI6XG4gICAgICByZXR1cm4gXCJib29sZWFuXCI7XG4gICAgY2FzZSBcIm51bGxcIjpcbiAgICAgIHJldHVybiBcIm51bGxcIjtcbiAgICBjYXNlIFwiYXJyYXlcIjoge1xuICAgICAgY29uc3QgaXRlbXMgPSBzY2hlbWEuaXRlbXNcbiAgICAgICAgPyBqc29uU2NoZW1hVG9UcyhzY2hlbWEuaXRlbXMgYXMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4sIGluZGVudClcbiAgICAgICAgOiBcInVua25vd25cIjtcbiAgICAgIHJldHVybiBgQXJyYXk8JHtpdGVtc30+YDtcbiAgICB9XG4gICAgY2FzZSBcIm9iamVjdFwiOiB7XG4gICAgICBjb25zdCBwcm9wZXJ0aWVzID0gc2NoZW1hLnByb3BlcnRpZXMgYXNcbiAgICAgICAgfCBSZWNvcmQ8c3RyaW5nLCBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPj5cbiAgICAgICAgfCB1bmRlZmluZWQ7XG4gICAgICBpZiAoIXByb3BlcnRpZXMpIHtcbiAgICAgICAgcmV0dXJuIFwiUmVjb3JkPHN0cmluZywgdW5rbm93bj5cIjtcbiAgICAgIH1cbiAgICAgIGNvbnN0IHJlcXVpcmVkID0gbmV3IFNldCgoc2NoZW1hLnJlcXVpcmVkIGFzIHN0cmluZ1tdKSB8fCBbXSk7XG4gICAgICBjb25zdCBwYWQgPSBcIiAgXCIucmVwZWF0KGluZGVudCArIDEpO1xuICAgICAgY29uc3QgY2xvc2VQYWQgPSBcIiAgXCIucmVwZWF0KGluZGVudCk7XG4gICAgICBjb25zdCBwcm9wcyA9IE9iamVjdC5lbnRyaWVzKHByb3BlcnRpZXMpLm1hcCgoW2tleSwgdmFsdWVdKSA9PiB7XG4gICAgICAgIGNvbnN0IG9wdCA9IHJlcXVpcmVkLmhhcyhrZXkpID8gXCJcIiA6IFwiP1wiO1xuICAgICAgICByZXR1cm4gYCR7cGFkfSR7a2V5fSR7b3B0fTogJHtqc29uU2NoZW1hVG9Ucyh2YWx1ZSwgaW5kZW50ICsgMSl9YDtcbiAgICAgIH0pO1xuICAgICAgcmV0dXJuIGB7XFxuJHtwcm9wcy5qb2luKFwiO1xcblwiKX07XFxuJHtjbG9zZVBhZH19YDtcbiAgICB9XG4gICAgZGVmYXVsdDpcbiAgICAgIHJldHVybiBcInVua25vd25cIjtcbiAgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2VuZXJhdGVDb250ZXh0VHlwZVN0cmluZyh0b29sczogVG9vbFNldCk6IHN0cmluZyB7XG4gIGNvbnN0IHNpZ3M6IHN0cmluZ1tdID0gW107XG5cbiAgZm9yIChjb25zdCBbbmFtZSwgdF0gb2YgT2JqZWN0LmVudHJpZXModG9vbHMpKSB7XG4gICAgY29uc3QgcmF3ID0gdCBhcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbiAgICBjb25zdCBpbnB1dFNjaGVtYSA9IGV4dHJhY3RKc29uU2NoZW1hKHJhdy5pbnB1dFNjaGVtYSA/PyByYXcucGFyYW1ldGVycyk7XG4gICAgY29uc3Qgb3V0cHV0U2NoZW1hID0gZXh0cmFjdEpzb25TY2hlbWEocmF3Lm91dHB1dFNjaGVtYSk7XG5cbiAgICBjb25zdCBpbnB1dFR5cGUgPSBpbnB1dFNjaGVtYSA/IGpzb25TY2hlbWFUb1RzKGlucHV0U2NoZW1hLCAyKSA6IFwie31cIjtcbiAgICBjb25zdCBvdXRwdXRUeXBlID0gb3V0cHV0U2NoZW1hXG4gICAgICA/IGpzb25TY2hlbWFUb1RzKG91dHB1dFNjaGVtYSwgMilcbiAgICAgIDogXCJ1bmtub3duXCI7XG5cbiAgICBzaWdzLnB1c2goYCAgICAke25hbWV9KGlucHV0OiAke2lucHV0VHlwZX0pOiBQcm9taXNlPCR7b3V0cHV0VHlwZX0+YCk7XG4gIH1cblxuICByZXR1cm4gYHR5cGUgQ29udGV4dCA9IHtcXG4gIHRvb2xzOiB7XFxuJHtzaWdzLmpvaW4oXCI7XFxuXCIpfTtcXG4gIH1cXG59YDtcbn1cblxuYXN5bmMgZnVuY3Rpb24gZXhlY3V0ZUluU2FuZGJveCh7XG4gIGNvZGUsXG4gIHJwY0RpcixcbiAgYWJvcnRDb250cm9sbGVyLFxuICBzYW5kYm94LFxuICBhdmFpbGFibGVUb29scyxcbiAgb25TdWJUb29sQ2FsbCxcbiAgdG9vbENvbnRleHQsXG59OiB7XG4gIGNvZGU6IHN0cmluZztcbiAgcnBjRGlyOiBzdHJpbmc7XG4gIGFib3J0Q29udHJvbGxlcjogQWJvcnRDb250cm9sbGVyO1xuICBzYW5kYm94OiBTYW5kYm94SW5zdGFuY2U7XG4gIGF2YWlsYWJsZVRvb2xzOiBUb29sU2V0O1xuICBvblN1YlRvb2xDYWxsPzogT25TdWJUb29sQ2FsbDtcbiAgdG9vbENvbnRleHQ/OiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPjtcbn0pOiBQcm9taXNlPFJlY29yZDxzdHJpbmcsIHVua25vd24+PiB7XG4gIGNvbnN0IG1rZGlyUmVzdWx0ID0gYXdhaXQgc2FuZGJveC5leGVjKHtcbiAgICBjb21tYW5kOiBcIm1rZGlyXCIsXG4gICAgYXJnczogW1wiLXBcIiwgcnBjRGlyXSxcbiAgfSk7XG4gIGlmIChta2RpclJlc3VsdCBpbnN0YW5jZW9mIEVycm9yKSB7XG4gICAgcmV0dXJuIHsgc3VjY2VzczogZmFsc2UsIGVycm9yOiBta2RpclJlc3VsdC5tZXNzYWdlIH07XG4gIH1cbiAgYXdhaXQgbWtkaXJSZXN1bHQucmVzdWx0O1xuXG4gIGF3YWl0IHNhbmRib3gud3JpdGVGaWxlcyh7XG4gICAgZmlsZXM6IFtcbiAgICAgIHsgcGF0aDogXCJydW5uZXIubWpzXCIsIGNvbnRlbnQ6IFJVTk5FUl9TQ1JJUFQgfSxcbiAgICAgIHsgcGF0aDogXCJjb2RlLmpzXCIsIGNvbnRlbnQ6IGNvZGUgfSxcbiAgICAgIHtcbiAgICAgICAgcGF0aDogXCJ0b29scy5qc29uXCIsXG4gICAgICAgIGNvbnRlbnQ6IEpTT04uc3RyaW5naWZ5KE9iamVjdC5rZXlzKGF2YWlsYWJsZVRvb2xzKSksXG4gICAgICB9LFxuICAgIF0sXG4gICAgZGVzdFBhdGg6IHJwY0RpcixcbiAgfSk7XG5cbiAgY29uc3Qgbm9kZU1ham9yID0gTnVtYmVyLnBhcnNlSW50KHByb2Nlc3MudmVyc2lvbnMubm9kZS5zcGxpdChcIi5cIilbMF0sIDEwKTtcbiAgY29uc3QgcGVybWlzc2lvbkZsYWcgPVxuICAgIG5vZGVNYWpvciA+PSAyMiA/IFwiLS1wZXJtaXNzaW9uXCIgOiBcIi0tZXhwZXJpbWVudGFsLXBlcm1pc3Npb25cIjtcblxuICBjb25zdCBleGVjUmVzdWx0ID0gYXdhaXQgc2FuZGJveC5leGVjKHtcbiAgICBjb21tYW5kOiBcIm5vZGVcIixcbiAgICBhcmdzOiBbXG4gICAgICBwZXJtaXNzaW9uRmxhZyxcbiAgICAgIGAtLWFsbG93LWZzLXJlYWQ9JHtycGNEaXJ9YCxcbiAgICAgIGAtLWFsbG93LWZzLXdyaXRlPSR7cnBjRGlyfWAsXG4gICAgICBgJHtycGNEaXJ9L3J1bm5lci5tanNgLFxuICAgIF0sXG4gICAgc2lnbmFsOiBhYm9ydENvbnRyb2xsZXIuc2lnbmFsLFxuICB9KTtcblxuICBpZiAoZXhlY1Jlc3VsdCBpbnN0YW5jZW9mIEVycm9yKSB7XG4gICAgcmV0dXJuIHsgc3VjY2VzczogZmFsc2UsIGVycm9yOiBleGVjUmVzdWx0Lm1lc3NhZ2UgfTtcbiAgfVxuXG4gIGNvbnN0IGZhdGFsOiB7IGVycm9yOiBFcnJvciB8IG51bGwgfSA9IHsgZXJyb3I6IG51bGwgfTtcblxuICBjb25zdCBhYm9ydCA9IChlcnJvcjogRXJyb3IpID0+IHtcbiAgICBpZiAoIWZhdGFsLmVycm9yKSB7XG4gICAgICBmYXRhbC5lcnJvciA9IGVycm9yO1xuICAgICAgYWJvcnRDb250cm9sbGVyLmFib3J0KCk7XG4gICAgfVxuICB9O1xuXG4gIGNvbnN0IGhhbmRsZVRvb2xSZXF1ZXN0ID0gYXN5bmMgKHJlcXVlc3RKc29uOiBzdHJpbmcpID0+IHtcbiAgICBsZXQgcGFyc2VkOiB7IGlkOiBzdHJpbmc7IHRvb2w6IHN0cmluZzsgaW5wdXQ6IHVua25vd24gfTtcbiAgICB0cnkge1xuICAgICAgcGFyc2VkID0gSlNPTi5wYXJzZShyZXF1ZXN0SnNvbik7XG4gICAgfSBjYXRjaCB7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgY29uc3QgeyBpZCwgdG9vbDogdG9vbE5hbWUsIGlucHV0IH0gPSBwYXJzZWQ7XG5cbiAgICBsZXQgcmVzcG9uc2U6IHsgcmVzdWx0PzogdW5rbm93bjsgZXJyb3I/OiBzdHJpbmcgfTtcbiAgICB0cnkge1xuICAgICAgY29uc3QgdCA9IGF2YWlsYWJsZVRvb2xzW3Rvb2xOYW1lXTtcbiAgICAgIGlmICghdD8uZXhlY3V0ZSkge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgICAgYFRvb2wgJHt0b29sTmFtZX0gbm90IGZvdW5kIG9yIGhhcyBubyBleGVjdXRlIGZ1bmN0aW9uYFxuICAgICAgICApO1xuICAgICAgfVxuICAgICAgY29uc3QgZXhlYyA9IHQuZXhlY3V0ZS5iaW5kKHQpO1xuICAgICAgY29uc3QgZXhlY09wdGlvbnMgPSB7XG4gICAgICAgIHRvb2xDYWxsSWQ6IGBqc18ke3Rvb2xOYW1lfV8ke0RhdGUubm93KCl9YCxcbiAgICAgICAgbWVzc2FnZXM6IFtdLFxuICAgICAgICBleHBlcmltZW50YWxfY29udGV4dDogdG9vbENvbnRleHQsXG4gICAgICB9O1xuICAgICAgaWYgKG9uU3ViVG9vbENhbGwpIHtcbiAgICAgICAgcmVzcG9uc2UgPSBhd2FpdCBvblN1YlRvb2xDYWxsKHRvb2xOYW1lLCBpbnB1dCwgKCkgPT5cbiAgICAgICAgICBleGVjKGlucHV0LCBleGVjT3B0aW9ucylcbiAgICAgICAgKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IGV4ZWMoaW5wdXQsIGV4ZWNPcHRpb25zKTtcbiAgICAgICAgcmVzcG9uc2UgPSB7IHJlc3VsdCB9O1xuICAgICAgfVxuICAgIH0gY2F0Y2ggKGVycikge1xuICAgICAgY29uc3QgbXNnID0gZXJyIGluc3RhbmNlb2YgRXJyb3IgPyBlcnIubWVzc2FnZSA6IFN0cmluZyhlcnIpO1xuICAgICAgcmVzcG9uc2UgPSB7IGVycm9yOiBtc2cgfTtcbiAgICB9XG5cbiAgICB0cnkge1xuICAgICAgYXdhaXQgc2FuZGJveC53cml0ZUZpbGVzKHtcbiAgICAgICAgZmlsZXM6IFtcbiAgICAgICAgICB7XG4gICAgICAgICAgICBwYXRoOiBgJHtpZH0ucmVzcG9uc2UuanNvbmAsXG4gICAgICAgICAgICBjb250ZW50OiBKU09OLnN0cmluZ2lmeShyZXNwb25zZSksXG4gICAgICAgICAgfSxcbiAgICAgICAgXSxcbiAgICAgICAgZGVzdFBhdGg6IHJwY0RpcixcbiAgICAgIH0pO1xuICAgIH0gY2F0Y2ggKGVycikge1xuICAgICAgLyoqXG4gICAgICAgKiBJZiB3cml0aW5nIHRoZSByZXNwb25zZSBmaWxlIGZhaWxzLCB0aGUgcnVubmVyIHdpbGwgcG9sbCBmb3JldmVyXG4gICAgICAgKiB3YWl0aW5nIGZvciBpdC4gQWJvcnQgdGhlIHJ1bm5lciBwcm9jZXNzIHNvIHdlIGRvbid0IGRlYWRsb2NrLlxuICAgICAgICovXG4gICAgICBjb25zdCBtc2cgPSBlcnIgaW5zdGFuY2VvZiBFcnJvciA/IGVyci5tZXNzYWdlIDogU3RyaW5nKGVycik7XG4gICAgICBhYm9ydChuZXcgRXJyb3IoYEZhaWxlZCB0byB3cml0ZSBSUEMgcmVzcG9uc2UgZm9yICR7dG9vbE5hbWV9OiAke21zZ31gKSk7XG4gICAgfVxuICB9O1xuXG4gIGNvbnN0IHRvb2xDYWxsUHJvbWlzZXM6IFByb21pc2U8dm9pZD5bXSA9IFtdO1xuICBsZXQgc3RkZXJyQnVmZmVyID0gXCJcIjtcbiAgbGV0IG5vbk1hcmtlclN0ZGVyciA9IFwiXCI7XG5cbiAgY29uc3QgdGltZW91dCA9IHNldFRpbWVvdXQoKCkgPT4ge1xuICAgIGFib3J0KFxuICAgICAgbmV3IEVycm9yKFxuICAgICAgICBgSmF2YVNjcmlwdCBleGVjdXRpb24gdGltZWQgb3V0IGFmdGVyICR7RVhFQ1VUSU9OX1RJTUVPVVRfTVMgLyAxMDAwfXNgXG4gICAgICApXG4gICAgKTtcbiAgfSwgRVhFQ1VUSU9OX1RJTUVPVVRfTVMpO1xuXG4gIHRyeSB7XG4gICAgZm9yIGF3YWl0IChjb25zdCBlbnRyeSBvZiBleGVjUmVzdWx0LmxvZ3MoKSkge1xuICAgICAgaWYgKGZhdGFsLmVycm9yKSB7XG4gICAgICAgIGJyZWFrO1xuICAgICAgfVxuXG4gICAgICBpZiAoZW50cnkuc3RyZWFtID09PSBcInN0ZGVyclwiKSB7XG4gICAgICAgIHN0ZGVyckJ1ZmZlciArPSBlbnRyeS5kYXRhO1xuXG4gICAgICAgIHdoaWxlICh0cnVlKSB7XG4gICAgICAgICAgY29uc3Qgc3RhcnRJZHggPSBzdGRlcnJCdWZmZXIuaW5kZXhPZihSRVFVRVNUX01BUktFUl9TVEFSVCk7XG4gICAgICAgICAgaWYgKHN0YXJ0SWR4ID09PSAtMSkge1xuICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgY29uc3QgYmVmb3JlTWFya2VyID0gc3RkZXJyQnVmZmVyLnNsaWNlKDAsIHN0YXJ0SWR4KTtcbiAgICAgICAgICBpZiAoYmVmb3JlTWFya2VyLnRyaW0oKSkge1xuICAgICAgICAgICAgbm9uTWFya2VyU3RkZXJyICs9IGJlZm9yZU1hcmtlcjtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBjb25zdCBjb250ZW50U3RhcnQgPSBzdGFydElkeCArIFJFUVVFU1RfTUFSS0VSX1NUQVJULmxlbmd0aDtcbiAgICAgICAgICBjb25zdCBlbmRJZHggPSBzdGRlcnJCdWZmZXIuaW5kZXhPZihSRVFVRVNUX01BUktFUl9FTkQsIGNvbnRlbnRTdGFydCk7XG4gICAgICAgICAgaWYgKGVuZElkeCA9PT0gLTEpIHtcbiAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgIH1cblxuICAgICAgICAgIGNvbnN0IHJlcXVlc3RKc29uID0gc3RkZXJyQnVmZmVyLnNsaWNlKGNvbnRlbnRTdGFydCwgZW5kSWR4KTtcbiAgICAgICAgICBzdGRlcnJCdWZmZXIgPSBzdGRlcnJCdWZmZXIuc2xpY2UoZW5kSWR4ICsgUkVRVUVTVF9NQVJLRVJfRU5ELmxlbmd0aCk7XG5cbiAgICAgICAgICB0b29sQ2FsbFByb21pc2VzLnB1c2goaGFuZGxlVG9vbFJlcXVlc3QocmVxdWVzdEpzb24pKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cbiAgfSBmaW5hbGx5IHtcbiAgICBjbGVhclRpbWVvdXQodGltZW91dCk7XG4gIH1cblxuICBpZiAoc3RkZXJyQnVmZmVyLnRyaW0oKSkge1xuICAgIG5vbk1hcmtlclN0ZGVyciArPSBzdGRlcnJCdWZmZXI7XG4gIH1cblxuICBhd2FpdCBQcm9taXNlLmFsbFNldHRsZWQodG9vbENhbGxQcm9taXNlcyk7XG5cbiAgaWYgKGZhdGFsLmVycm9yKSB7XG4gICAgZXhlY1Jlc3VsdC5yZXN1bHQuY2F0Y2goKCkgPT4gdW5kZWZpbmVkKTtcbiAgICByZXR1cm4ge1xuICAgICAgc3VjY2VzczogZmFsc2UsXG4gICAgICBlcnJvcjogZmF0YWwuZXJyb3IubWVzc2FnZSxcbiAgICAgIC4uLihub25NYXJrZXJTdGRlcnIgPyB7IHN0ZGVycjogbm9uTWFya2VyU3RkZXJyLnNsaWNlKDAsIDIwMDApIH0gOiB7fSksXG4gICAgfTtcbiAgfVxuXG4gIGNvbnN0IHsgc3Rkb3V0LCBzdGRlcnIsIGV4aXRDb2RlIH0gPSBhd2FpdCBleGVjUmVzdWx0LnJlc3VsdDtcblxuICB0cnkge1xuICAgIHJldHVybiBKU09OLnBhcnNlKHN0ZG91dCk7XG4gIH0gY2F0Y2gge1xuICAgIHJldHVybiB7XG4gICAgICBzdWNjZXNzOiBmYWxzZSxcbiAgICAgIGVycm9yOiBgUnVubmVyIGZhaWxlZCAoZXhpdCAke2V4aXRDb2RlfSkuYCxcbiAgICAgIHN0ZGVycjogKG5vbk1hcmtlclN0ZGVyciArIHN0ZGVycikuc2xpY2UoMCwgMjAwMCksXG4gICAgICBzdGRvdXQ6IHN0ZG91dC5zbGljZSgwLCAxMDAwKSxcbiAgICB9O1xuICB9XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBjcmVhdGVKYXZhU2NyaXB0VG9vbChvcHRzOiB7XG4gIHRvb2xzOiBUb29sU2V0O1xuICBzYW5kYm94OiBTYW5kYm94SW5zdGFuY2U7XG4gIGFjdGl2ZVRvb2xzPzogc3RyaW5nW107XG4gIG9uU3ViVG9vbENhbGw/OiBPblN1YlRvb2xDYWxsO1xuICB0b29sQ29udGV4dD86IFJlY29yZDxzdHJpbmcsIHVua25vd24+O1xufSkge1xuICBjb25zdCB7IHNhbmRib3gsIG9uU3ViVG9vbENhbGwgfSA9IG9wdHM7XG4gIGNvbnN0IGFjdGl2ZVNldCA9IG9wdHMuYWN0aXZlVG9vbHMgPyBuZXcgU2V0KG9wdHMuYWN0aXZlVG9vbHMpIDogbnVsbDtcbiAgY29uc3QgYXZhaWxhYmxlVG9vbHM6IFRvb2xTZXQgPSB7fTtcbiAgZm9yIChjb25zdCBbbmFtZSwgdF0gb2YgT2JqZWN0LmVudHJpZXMob3B0cy50b29scykpIHtcbiAgICBpZiAoIWFjdGl2ZVNldCB8fCBhY3RpdmVTZXQuaGFzKG5hbWUpKSB7XG4gICAgICBhdmFpbGFibGVUb29sc1tuYW1lXSA9IHQ7XG4gICAgfVxuICB9XG4gIGNvbnN0IGNvbnRleHRUeXBlID0gZ2VuZXJhdGVDb250ZXh0VHlwZVN0cmluZyhhdmFpbGFibGVUb29scyk7XG5cbiAgcmV0dXJuIHRvb2woe1xuICAgIC4uLmJ1aWx0SW5Ub29scy5KYXZhU2NyaXB0LFxuICAgIGRlc2NyaXB0aW9uOiBgRXhlY3V0ZSBKYXZhU2NyaXB0IHRvIG9yY2hlc3RyYXRlIG11bHRpcGxlIHRvb2wgY2FsbHMgaW4gYSBzaW5nbGUgc3RlcC4gVXNlIHRoaXMgd2hlbiB5b3UgbmVlZCB0byBydW4gc2V2ZXJhbCB0b29scyBpbiBzZXF1ZW5jZSwgdHJhbnNmb3JtIGludGVybWVkaWF0ZSByZXN1bHRzLCBvciBwYXJhbGxlbGl6ZSBpbmRlcGVuZGVudCBvcGVyYXRpb25zIHdpdGggUHJvbWlzZS5hbGwuXG5cblRoZSBjb2RlIHJ1bnMgYXMgYW4gYXN5bmMgZnVuY3Rpb24gYm9keSB3aXRoIFxcYGN0eFxcYCBpbiBzY29wZTpcblxuXFxgXFxgXFxgdHlwZXNjcmlwdFxuJHtjb250ZXh0VHlwZX1cblxcYFxcYFxcYFxuXG5FeGFtcGxlczpcbi0gU2VxdWVudGlhbDogXFxgY29uc3QgZmlsZSA9IGF3YWl0IGN0eC50b29scy5SZWFkKHsgcGF0aDogXCJwYWNrYWdlLmpzb25cIiB9KTsgcmV0dXJuIEpTT04ucGFyc2UoZmlsZS5jb250ZW50KTtcXGBcbi0gUGFyYWxsZWw6IFxcYGNvbnN0IFthLCBiXSA9IGF3YWl0IFByb21pc2UuYWxsKFtjdHgudG9vbHMuUmVhZCh7IHBhdGg6IFwiYS50c1wiIH0pLCBjdHgudG9vbHMuUmVhZCh7IHBhdGg6IFwiYi50c1wiIH0pXSk7IHJldHVybiB7IGE6IGEuY29udGVudCwgYjogYi5jb250ZW50IH07XFxgXG4tIFRyYW5zZm9ybTogXFxgY29uc3QgZ3JlcCA9IGF3YWl0IGN0eC50b29scy5HcmVwKHsgcGF0dGVybjogXCJUT0RPXCIgfSk7IHJldHVybiBncmVwLm1hdGNoZXMuc3BsaXQoXCJcXFxcblwiKS5sZW5ndGg7XFxgXG4tIEJhc2g6IFxcYGNvbnN0IHIgPSBhd2FpdCBjdHgudG9vbHMuQmFzaCh7IGNvbW1hbmQ6IFwibm9kZSAtZSBcXFxcXCJjb25zb2xlLmxvZyhKU09OLnN0cmluZ2lmeSh7djoxfSkpXFxcXFwiXCIgfSk7IHJldHVybiBKU09OLnBhcnNlKHIuc3Rkb3V0KTtcXGBcblxuUmV0dXJuIGEgdmFsdWUgdG8gcGFzcyByZXN1bHRzIGJhY2suIFVzZSBjb25zb2xlLmxvZygpIGZvciBkZWJ1ZyBvdXRwdXQuXG5JbXBvcnRhbnQ6IEVhY2ggdG9vbCByZXR1cm5zIGEgc3RydWN0dXJlZCBvYmplY3QgbWF0Y2hpbmcgaXRzIG91dHB1dCBzY2hlbWEgXHUyMDE0IG5vdCBhIHJhdyBzdHJpbmcuIEFjY2VzcyB0aGUgc3BlY2lmaWMgZmllbGQgeW91IG5lZWQgKGUuZy4gQmFzaCdzIHJlc3VsdC5zdGRvdXQsIFJlYWQncyByZXN1bHQuY29udGVudCkgcmF0aGVyIHRoYW4gdHJ5aW5nIHRvIHBhcnNlIHRoZSBlbnRpcmUgcmVzdWx0LmAsXG4gICAgZXhlY3V0ZTogKHsgY29kZSB9KSA9PiB7XG4gICAgICBjb25zdCBydW5JZCA9IGBydW5fJHtEYXRlLm5vdygpfV8ke01hdGgucmFuZG9tKClcbiAgICAgICAgLnRvU3RyaW5nKDM2KVxuICAgICAgICAuc2xpY2UoMiwgOCl9YDtcbiAgICAgIGNvbnN0IHJwY0RpciA9IGAuYWdlbnQvanMtcnBjLyR7cnVuSWR9YDtcbiAgICAgIGNvbnN0IGFib3J0Q29udHJvbGxlciA9IG5ldyBBYm9ydENvbnRyb2xsZXIoKTtcblxuICAgICAgcmV0dXJuIGV4ZWN1dGVJblNhbmRib3goe1xuICAgICAgICBjb2RlLFxuICAgICAgICBycGNEaXIsXG4gICAgICAgIGFib3J0Q29udHJvbGxlcixcbiAgICAgICAgc2FuZGJveCxcbiAgICAgICAgYXZhaWxhYmxlVG9vbHMsXG4gICAgICAgIG9uU3ViVG9vbENhbGwsXG4gICAgICAgIHRvb2xDb250ZXh0OiBvcHRzLnRvb2xDb250ZXh0LFxuICAgICAgfSk7XG4gICAgfSxcbiAgfSk7XG59XG4iLCAiaW1wb3J0IHR5cGUgeyBUb29sIH0gZnJvbSBcImFpXCI7XG5pbXBvcnQgdHlwZSB7IEFueUFnZW50IH0gZnJvbSBcIi4uL2FnZW50L2NsaWVudFwiO1xuaW1wb3J0IHR5cGUgeyBCdWlsdEluVG9vbE5hbWUgfSBmcm9tIFwiLi9pbmRleFwiO1xuXG4vKiogQnVpbHQtaW4gZGVmYXVsdHMgZm9yIG5lZWRzQXBwcm92YWwgKHVzZXIgY2FuIG92ZXJyaWRlIHZpYSBhZ2VudCBjb25maWcpLiAqL1xuY29uc3QgYnVpbHRpbk5lZWRzQXBwcm92YWw6IFBhcnRpYWw8UmVjb3JkPEJ1aWx0SW5Ub29sTmFtZSwgYm9vbGVhbj4+ID0ge307XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBjaGVja05lZWRzQXBwcm92YWwoe1xuICBhZ2VudCxcbiAgdG9vbE5hbWUsXG4gIGlucHV0LFxuICB0b29sQ2FsbElkLFxuICBtZXNzYWdlcyxcbn06IHtcbiAgYWdlbnQ6IEFueUFnZW50O1xuICB0b29sTmFtZTogc3RyaW5nO1xuICBpbnB1dDogdW5rbm93bjtcbiAgdG9vbENhbGxJZDogc3RyaW5nO1xuICBtZXNzYWdlczogdW5rbm93bltdO1xufSk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICAvLyAxLiBQZXItdG9vbCBuZWVkc0FwcHJvdmFsIG9uIHRoZSBUb29sIGRlZmluaXRpb24gdGFrZXMgcHJpb3JpdHlcbiAgY29uc3QgdG9vbERlZiA9IGFnZW50Lm9wdGlvbnMudG9vbHM/Llt0b29sTmFtZV0gYXMgVG9vbCB8IHVuZGVmaW5lZDtcbiAgaWYgKHRvb2xEZWY/Lm5lZWRzQXBwcm92YWwgIT0gbnVsbCkge1xuICAgIGlmICh0eXBlb2YgdG9vbERlZi5uZWVkc0FwcHJvdmFsID09PSBcImJvb2xlYW5cIikge1xuICAgICAgcmV0dXJuIHRvb2xEZWYubmVlZHNBcHByb3ZhbDtcbiAgICB9XG4gICAgcmV0dXJuIGF3YWl0IHRvb2xEZWYubmVlZHNBcHByb3ZhbChpbnB1dCwge1xuICAgICAgdG9vbENhbGxJZCxcbiAgICAgIG1lc3NhZ2VzOiBtZXNzYWdlcyBhcyBuZXZlcltdLFxuICAgICAgZXhwZXJpbWVudGFsX2NvbnRleHQ6IHVuZGVmaW5lZCxcbiAgICB9KTtcbiAgfVxuXG4gIC8vIDIuIEFnZW50LWxldmVsIG5lZWRzQXBwcm92YWwgbWFwXG4gIGNvbnN0IGFnZW50RW50cnkgPSAoXG4gICAgYWdlbnQub3B0aW9ucy5uZWVkc0FwcHJvdmFsIGFzXG4gICAgICB8IFJlY29yZDxcbiAgICAgICAgICBzdHJpbmcsXG4gICAgICAgICAgYm9vbGVhbiB8ICgoLi4uYXJnczogdW5rbm93bltdKSA9PiBib29sZWFuIHwgUHJvbWlzZTxib29sZWFuPilcbiAgICAgICAgPlxuICAgICAgfCB1bmRlZmluZWRcbiAgKT8uW3Rvb2xOYW1lXTtcblxuICBpZiAoYWdlbnRFbnRyeSAhPSBudWxsKSB7XG4gICAgaWYgKHR5cGVvZiBhZ2VudEVudHJ5ID09PSBcImJvb2xlYW5cIikge1xuICAgICAgcmV0dXJuIGFnZW50RW50cnk7XG4gICAgfVxuICAgIHJldHVybiBhd2FpdCBhZ2VudEVudHJ5KGlucHV0LCB7XG4gICAgICB0b29sQ2FsbElkLFxuICAgICAgbWVzc2FnZXMsXG4gICAgICBleHBlcmltZW50YWxfY29udGV4dDogdW5kZWZpbmVkLFxuICAgIH0pO1xuICB9XG5cbiAgLy8gMy4gQnVpbHQtaW4gZGVmYXVsdHNcbiAgcmV0dXJuIGJ1aWx0aW5OZWVkc0FwcHJvdmFsW3Rvb2xOYW1lIGFzIEJ1aWx0SW5Ub29sTmFtZV0gPz8gZmFsc2U7XG59XG4iLCAiaW1wb3J0IHR5cGUge1xuICBJbmZlclRvb2xJbnB1dCxcbiAgSW5mZXJVSVRvb2xzLFxuICBNb2RlbE1lc3NhZ2UsXG4gIFRvb2xTZXQsXG4gIFVJTWVzc2FnZSxcbn0gZnJvbSBcImFpXCI7XG5pbXBvcnQgKiBhcyBlcnJvcmUgZnJvbSBcImVycm9yZVwiO1xuaW1wb3J0IHsgeiB9IGZyb20gXCJ6b2RcIjtcbmltcG9ydCB0eXBlIHsgQnVpbHRJblRvb2xOYW1lLCBidWlsdEluVG9vbHMsIFRvb2xDb250ZXh0IH0gZnJvbSBcIi4uL3Rvb2xzXCI7XG5pbXBvcnQgdHlwZSB7IEFueUFnZW50IH0gZnJvbSBcIi4vY2xpZW50XCI7XG5cbi8vIFx1MjUwMFx1MjUwMFx1MjUwMCBBZ2VudCBTdGF0dXMgXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXG5cbi8qKlxuICogVHJhbnNpZW50IHN0YXR1cyBpbmRpY2F0b3JzIGVtaXR0ZWQgb3ZlciB0aGUgc3RyZWFtIGR1cmluZyBsb25nLXJ1bm5pbmcgcGhhc2VzLlxuICogRGVsaXZlcmVkIHRvIHRoZSBjbGllbnQgYXMgYGRhdGEtc3RhdHVzYCBjaHVua3Mgd2l0aCBgdHJhbnNpZW50OiB0cnVlYCBcdTIwMTQgbm90IHBlcnNpc3RlZC5cbiAqL1xuZXhwb3J0IGNvbnN0IEFnZW50U3RhdHVzID0gei5kaXNjcmltaW5hdGVkVW5pb24oXCJ0eXBlXCIsIFtcbiAgei5vYmplY3QoeyB0eXBlOiB6LmxpdGVyYWwoXCJzYW5kYm94LXNldHVwXCIpIH0pLFxuICB6Lm9iamVjdCh7IHR5cGU6IHoubGl0ZXJhbChcInNhbmRib3gtc2V0dXAtY29sZFwiKSB9KSxcbiAgei5vYmplY3QoeyB0eXBlOiB6LmxpdGVyYWwoXCJsb2FkaW5nLXNraWxsc1wiKSB9KSxcbiAgei5vYmplY3QoeyB0eXBlOiB6LmxpdGVyYWwoXCJwcm9jZXNzaW5nLWFwcHJvdmFsc1wiKSB9KSxcbiAgei5vYmplY3QoeyB0eXBlOiB6LmxpdGVyYWwoXCJuZWVkcy1hcHByb3ZhbFwiKSB9KSxcbiAgei5vYmplY3QoeyB0eXBlOiB6LmxpdGVyYWwoXCJ0aGlua2luZ1wiKSB9KSxcbiAgei5vYmplY3QoeyB0eXBlOiB6LmxpdGVyYWwoXCJjdXN0b21cIiksIHN0YXR1czogei5zdHJpbmcoKSB9KSxcbl0pO1xuXG5leHBvcnQgdHlwZSBBZ2VudFN0YXR1cyA9IHouaW5mZXI8dHlwZW9mIEFnZW50U3RhdHVzPjtcblxuZXhwb3J0IHR5cGUgQWdlbnREYXRhVHlwZXMgPSB7IHN0YXR1czogQWdlbnRTdGF0dXMgfTtcblxuLy8gXHUyNTAwXHUyNTAwXHUyNTAwIEhvb2tzIFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFxuXG5leHBvcnQgdHlwZSBBZ2VudEhvb2tzID0ge1xuICBcInRvb2wuYmVmb3JlXCI/OiAob3B0czoge1xuICAgIG5hbWU6IHN0cmluZztcbiAgICBpbnB1dDogdW5rbm93bjtcbiAgICBjb250ZXh0OiBUb29sQ29udGV4dDtcbiAgfSkgPT4gUHJvbWlzZTx1bmRlZmluZWQgfCB7IGlucHV0OiB1bmtub3duIH0+O1xuICBcInRvb2wuYWZ0ZXJcIj86IChvcHRzOiB7XG4gICAgbmFtZTogc3RyaW5nO1xuICAgIGlucHV0OiB1bmtub3duO1xuICAgIHJlc3VsdDogdW5rbm93bjtcbiAgICBjb250ZXh0OiBUb29sQ29udGV4dDtcbiAgfSkgPT4gUHJvbWlzZTx1bmRlZmluZWQgfCB7IHJlc3VsdDogdW5rbm93biB9PjtcbiAgc3RhdHVzPzogKHN0YXR1czogQWdlbnRTdGF0dXMpID0+IHZvaWQgfCBQcm9taXNlPHZvaWQ+O1xufTtcblxuLy8gXHUyNTAwXHUyNTAwXHUyNTAwIE5lZWRzIEFwcHJvdmFsIFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFxuXG50eXBlIFRvb2xOYW1lPFRvb2xzPiA9IEV4dHJhY3Q8a2V5b2YgVG9vbHMsIHN0cmluZz4gfCBCdWlsdEluVG9vbE5hbWU7XG5cbnR5cGUgVG9vbElucHV0PFRvb2xzLCBLPiA9IEsgZXh0ZW5kcyBCdWlsdEluVG9vbE5hbWVcbiAgPyBJbmZlclRvb2xJbnB1dDwodHlwZW9mIGJ1aWx0SW5Ub29scylbS10+XG4gIDogSyBleHRlbmRzIGtleW9mIFRvb2xzXG4gICAgPyBUb29sc1tLXSBleHRlbmRzIGltcG9ydChcImFpXCIpLlRvb2xcbiAgICAgID8gSW5mZXJUb29sSW5wdXQ8VG9vbHNbS10+XG4gICAgICA6IHVua25vd25cbiAgICA6IHVua25vd247XG5cbmV4cG9ydCB0eXBlIE5lZWRzQXBwcm92YWxNYXA8VG9vbHM+ID0ge1xuICBbSyBpbiBUb29sTmFtZTxUb29scz5dPzpcbiAgICB8IGJvb2xlYW5cbiAgICB8ICgoXG4gICAgICAgIGlucHV0OiBUb29sSW5wdXQ8VG9vbHMsIEs+LFxuICAgICAgICBvcHRpb25zOiB7XG4gICAgICAgICAgdG9vbENhbGxJZDogc3RyaW5nO1xuICAgICAgICAgIG1lc3NhZ2VzOiBNb2RlbE1lc3NhZ2VbXTtcbiAgICAgICAgICBleHBlcmltZW50YWxfY29udGV4dDogdW5rbm93bjtcbiAgICAgICAgfVxuICAgICAgKSA9PiBib29sZWFuIHwgUHJvbWlzZTxib29sZWFuPik7XG59O1xuXG4vLyBcdTI1MDBcdTI1MDBcdTI1MDAgSW5mZXIgVUkgTWVzc2FnZSBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcblxuZXhwb3J0IHR5cGUgSW5mZXJVSU1lc3NhZ2U8QT4gPSBBIGV4dGVuZHMge1xuICB0b29sczogaW5mZXIgVCBleHRlbmRzIFRvb2xTZXQ7XG4gICRVSU1lc3NhZ2U6IFVJTWVzc2FnZTxpbmZlciBNPjtcbn1cbiAgPyBVSU1lc3NhZ2U8TSwgQWdlbnREYXRhVHlwZXMsIEluZmVyVUlUb29sczxUPj5cbiAgOiBBIGV4dGVuZHMgeyB0b29sczogaW5mZXIgVCBleHRlbmRzIFRvb2xTZXQgfVxuICAgID8gVUlNZXNzYWdlPHVua25vd24sIEFnZW50RGF0YVR5cGVzLCBJbmZlclVJVG9vbHM8VD4+XG4gICAgOiBVSU1lc3NhZ2U8dW5rbm93biwgQWdlbnREYXRhVHlwZXM+O1xuXG5leHBvcnQgdHlwZSBUeXBlZFVJTWVzc2FnZTxcbiAgVE1lc3NhZ2VNZXRhZGF0YSA9IHVua25vd24sXG4gIFRvb2xzIGV4dGVuZHMgVG9vbFNldCA9IFRvb2xTZXQsXG4+ID0gVUlNZXNzYWdlPFRNZXNzYWdlTWV0YWRhdGEsIEFnZW50RGF0YVR5cGVzLCBJbmZlclVJVG9vbHM8VG9vbHM+PjtcblxuLy8gXHUyNTAwXHUyNTAwXHUyNTAwIFJlZ2lzdHJ5IFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFxuXG5jbGFzcyBBZ2VudE5vdEZvdW5kRXJyb3IgZXh0ZW5kcyBlcnJvcmUuY3JlYXRlVGFnZ2VkRXJyb3Ioe1xuICBuYW1lOiBcIkFnZW50Tm90Rm91bmRFcnJvclwiLFxuICBtZXNzYWdlOiBgQWdlbnQgXCIkYWdlbnROYW1lXCIgbm90IGZvdW5kIGluIHJlZ2lzdHJ5LlxuXG5oaW50OiBhZ2VudCgpIG11c3QgYmUgY2FsbGVkIGJlZm9yZSB3b3JrZmxvdyBzdGVwcyBleGVjdXRlLlxuICAgICAgVmFsaWQgbG9jYXRpb25zOiBtb2R1bGUgc2NvcGUsIG9yIGluc2lkZSBhIFwidXNlIHdvcmtmbG93XCIgYm9keS5cblxuICAvLyBNb2R1bGUgc2NvcGUgKHN0YXRpYyBhZ2VudHMpXG4gIGV4cG9ydCBjb25zdCBteUFnZW50ID0gYWdlbnQoXCIkYWdlbnROYW1lXCIsIHsgLi4uIH0pO1xuXG4gIC8vIFdvcmtmbG93IGJvZHkgKGR5bmFtaWMgYWdlbnRzKVxuICBhc3luYyBmdW5jdGlvbiBteVdvcmtmbG93KHsgaW5wdXQgfSkge1xuICAgIFwidXNlIHdvcmtmbG93XCI7XG4gICAgY29uc3QgbXlBZ2VudCA9IGFnZW50KFwiJGFnZW50TmFtZVwiLCB7IC4uLiB9KTtcbiAgICAuLi5cbiAgfWAsXG59KSB7fVxuXG5jb25zdCBBR0VOVF9SRUdJU1RSWV9LRVkgPSBTeW1ib2wuZm9yKFwiZXhwZXJpbWVudGFsLWFnZW50LXJlZ2lzdHJ5XCIpO1xuXG5mdW5jdGlvbiBnZXRBZ2VudFJlZ2lzdHJ5KCk6IE1hcDxzdHJpbmcsIEFueUFnZW50PiB7XG4gIGNvbnN0IGcgPSBnbG9iYWxUaGlzIGFzIFJlY29yZDxzeW1ib2wsIHVua25vd24+O1xuICBsZXQgcmVnaXN0cnkgPSBnW0FHRU5UX1JFR0lTVFJZX0tFWV0gYXMgTWFwPHN0cmluZywgQW55QWdlbnQ+IHwgdW5kZWZpbmVkO1xuICBpZiAoIXJlZ2lzdHJ5KSB7XG4gICAgcmVnaXN0cnkgPSBuZXcgTWFwKCk7XG4gICAgZ1tBR0VOVF9SRUdJU1RSWV9LRVldID0gcmVnaXN0cnk7XG4gIH1cbiAgcmV0dXJuIHJlZ2lzdHJ5O1xufVxuXG5leHBvcnQgZnVuY3Rpb24gcmVnaXN0ZXJBZ2VudChuYW1lOiBzdHJpbmcsIGFnZW50OiBBbnlBZ2VudCkge1xuICBnZXRBZ2VudFJlZ2lzdHJ5KCkuc2V0KG5hbWUsIGFnZW50KTtcbn1cbmV4cG9ydCBmdW5jdGlvbiBnZXRBZ2VudDxSZXF1aXJlZCBleHRlbmRzIGJvb2xlYW4+KFxuICBuYW1lOiBzdHJpbmcsXG4gIHJlcXVpcmVkPzogUmVxdWlyZWRcbik6IFJlcXVpcmVkIGV4dGVuZHMgdHJ1ZSA/IEFueUFnZW50IDogQW55QWdlbnQgfCB1bmRlZmluZWQge1xuICBjb25zdCBhZ2VudCA9IGdldEFnZW50UmVnaXN0cnkoKS5nZXQobmFtZSk7XG4gIGlmIChyZXF1aXJlZCAmJiAhYWdlbnQpIHtcbiAgICB0aHJvdyBuZXcgQWdlbnROb3RGb3VuZEVycm9yKHsgYWdlbnROYW1lOiBuYW1lIH0pO1xuICB9XG4gIHJldHVybiBhZ2VudCBhcyBSZXF1aXJlZCBleHRlbmRzIHRydWUgPyBBbnlBZ2VudCA6IEFueUFnZW50IHwgdW5kZWZpbmVkO1xufVxuIiwgImltcG9ydCB7IFdPUktGTE9XX0RFU0VSSUFMSVpFLCBXT1JLRkxPV19TRVJJQUxJWkUgfSBmcm9tIFwiQHdvcmtmbG93L3NlcmRlXCI7XG5pbXBvcnQgdHlwZSB7IEdhdGV3YXlNb2RlbElkLCBJbmZlclVJVG9vbHMsIFRvb2xTZXQsIFVJTWVzc2FnZSB9IGZyb20gXCJhaVwiO1xuaW1wb3J0IHR5cGUgeyB6IH0gZnJvbSBcInpvZFwiO1xuaW1wb3J0IHR5cGUgeyBTYW5kYm94QmluZGluZywgU2FuZGJveEluc3RhbmNlIH0gZnJvbSBcIi4uL3NhbmRib3gvYWRhcHRlclwiO1xuaW1wb3J0IHR5cGUgeyBTYW5kYm94Q29udGV4dCB9IGZyb20gXCIuLi9zYW5kYm94L2hhbmRsZVwiO1xuaW1wb3J0IHsgc2FuZGJveEhhbmRsZSB9IGZyb20gXCIuLi9zYW5kYm94L2hhbmRsZVwiO1xuaW1wb3J0IHsgdHlwZSBTZXNzaW9uQ29udGV4dCwgc2Vzc2lvbkhhbmRsZSB9IGZyb20gXCIuLi9zZXNzaW9uL2hhbmRsZVwiO1xuaW1wb3J0IHR5cGUgeyBTa2lsbElucHV0IH0gZnJvbSBcIi4uL3NraWxscy90eXBlc1wiO1xuaW1wb3J0IHsgdHlwZSBTdG9yYWdlLCB0eXBlIFN0b3JhZ2VJbnB1dCwgdG9TdG9yYWdlIH0gZnJvbSBcIi4uL3N0b3JhZ2UvYWRhcHRlclwiO1xuaW1wb3J0IHsga3ZTdG9yYWdlRmFsbGJhY2sgfSBmcm9tIFwiLi4vc3RvcmFnZS9iaW5kaW5ncy9rdlwiO1xuaW1wb3J0IHsgbG9jYWxTdG9yYWdlIH0gZnJvbSBcIi4uL3N0b3JhZ2UvZW50cnlcIjtcbmltcG9ydCB7IHNob3VsZEZhbGxiYWNrVG9LVjIgfSBmcm9tIFwiLi4vc3RvcmFnZS9zaG91bGQtZmFsbGJhY2sta3YyXCI7XG5pbXBvcnQgdHlwZSB7IEdlbmVyYXRpb25PcHRpb25zIH0gZnJvbSBcIi4uL3N0b3JhZ2UvdHlwZXNcIjtcbmltcG9ydCB7IHR5cGUgQnVpbHRJblRvb2xOYW1lLCBidWlsdEluVG9vbHMgfSBmcm9tIFwiLi4vdG9vbHNcIjtcbmltcG9ydCB0eXBlIHsgTG9nZ2luZ0NvbmZpZyB9IGZyb20gXCIuLi91dGlscy9sb2dnZXJcIjtcbmltcG9ydCB7IGNyZWF0ZUxvZ2dlciB9IGZyb20gXCIuLi91dGlscy9sb2dnZXJcIjtcbmltcG9ydCB7XG4gIHR5cGUgQWdlbnREYXRhVHlwZXMsXG4gIHR5cGUgQWdlbnRIb29rcyxcbiAgZ2V0QWdlbnQsXG4gIHR5cGUgTmVlZHNBcHByb3ZhbE1hcCxcbiAgcmVnaXN0ZXJBZ2VudCxcbn0gZnJvbSBcIi4vcmVnaXN0cnlcIjtcblxuLy8gXHUyNTAwXHUyNTAwXHUyNTAwIFJlc29sdmFibGUgXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXG5cbmV4cG9ydCB0eXBlIFJlc29sdmFibGVBcmdzPFxuICBUQ29udGV4dCA9IFJlY29yZDxzdHJpbmcsIHVua25vd24+LFxuICBUb29scyBleHRlbmRzIFRvb2xTZXQgPSBUb29sU2V0LFxuICBUU3RhdGUgPSBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPixcbj4gPSB7XG4gIGNvbnRleHQ6IFRDb250ZXh0O1xuICBzZXNzaW9uSWQ6IHN0cmluZztcbiAgc2FuZGJveDogU2FuZGJveEluc3RhbmNlO1xuICBtZXNzYWdlczogVUlNZXNzYWdlPHVua25vd24sIEFnZW50RGF0YVR5cGVzLCBJbmZlclVJVG9vbHM8VG9vbHM+PltdO1xuICBzdGF0ZTogUmVhZG9ubHk8VFN0YXRlPjtcbn07XG5cbmV4cG9ydCB0eXBlIFJlc29sdmFibGU8XG4gIFQsXG4gIFRDb250ZXh0ID0gUmVjb3JkPHN0cmluZywgdW5rbm93bj4sXG4gIFRvb2xzIGV4dGVuZHMgVG9vbFNldCA9IFRvb2xTZXQsXG4gIFRTdGF0ZSA9IFJlY29yZDxzdHJpbmcsIHVua25vd24+LFxuPiA9IFQgfCAoKGFyZ3M6IFJlc29sdmFibGVBcmdzPFRDb250ZXh0LCBUb29scywgVFN0YXRlPikgPT4gVCB8IFByb21pc2U8VD4pO1xuXG4vLyBcdTI1MDBcdTI1MDBcdTI1MDAgU2Vzc2lvbiBEZWZhdWx0cyBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcdTI1MDBcblxuZXhwb3J0IHR5cGUgU2Vzc2lvbkRlZmF1bHRzPFxuICBUb29scyBleHRlbmRzIFRvb2xTZXQgPSBUb29sU2V0LFxuICBUQ29udGV4dCBleHRlbmRzIFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0gUmVjb3JkPHN0cmluZywgdW5rbm93bj4sXG4gIFRTdGF0ZSBleHRlbmRzIFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0gUmVjb3JkPHN0cmluZywgdW5rbm93bj4sXG4+ID0ge1xuICBtb2RlbD86IFJlc29sdmFibGU8R2F0ZXdheU1vZGVsSWQgfCB1bmRlZmluZWQsIFRDb250ZXh0LCBUb29scywgVFN0YXRlPjtcbiAgc3lzdGVtPzogUmVzb2x2YWJsZTxzdHJpbmcgfCBzdHJpbmdbXSB8IHVuZGVmaW5lZCwgVENvbnRleHQsIFRvb2xzLCBUU3RhdGU+O1xuICBnZW5lcmF0aW9uPzogUmVzb2x2YWJsZTxcbiAgICBHZW5lcmF0aW9uT3B0aW9ucyB8IHVuZGVmaW5lZCxcbiAgICBUQ29udGV4dCxcbiAgICBUb29scyxcbiAgICBUU3RhdGVcbiAgPjtcbiAgYWN0aXZlVG9vbHM/OiBSZXNvbHZhYmxlPFxuICAgIHJlYWRvbmx5IChFeHRyYWN0PGtleW9mIFRvb2xzLCBzdHJpbmc+IHwgQnVpbHRJblRvb2xOYW1lKVtdIHwgdW5kZWZpbmVkLFxuICAgIFRDb250ZXh0LFxuICAgIFRvb2xzLFxuICAgIFRTdGF0ZVxuICA+O1xuICBza2lsbHM/OiBSZXNvbHZhYmxlPFNraWxsSW5wdXRbXSB8IHVuZGVmaW5lZCwgVENvbnRleHQsIFRvb2xzLCBUU3RhdGU+O1xufTtcblxuLy8gXHUyNTAwXHUyNTAwXHUyNTAwIEFnZW50IE9wdGlvbnMgXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXG5cbmV4cG9ydCB0eXBlIEFnZW50T3B0aW9uczxcbiAgLy8gYmlvbWUtaWdub3JlIGxpbnQvc3VzcGljaW91cy9ub0V4cGxpY2l0QW55OiBkZWZhdWx0IGFsbG93cyB0eXBlLWVyYXNlZCB1c2FnZVxuICBUb29scyBleHRlbmRzIFRvb2xTZXQgPSBhbnksXG4gIFRTYW5kYm94QmluZGluZ3MgZXh0ZW5kcyBTYW5kYm94QmluZGluZyB8IFNhbmRib3hCaW5kaW5nW10gPSBTYW5kYm94QmluZGluZyxcbiAgVENvbnRleHQgZXh0ZW5kcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9IFJlY29yZDxzdHJpbmcsIG5ldmVyPixcbiAgVFN0YXRlIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4gPSBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPixcbiAgVE1lc3NhZ2VNZXRhZGF0YSA9IHVua25vd24sXG4+ID0gU2Vzc2lvbkRlZmF1bHRzPFRvb2xzLCBUQ29udGV4dCwgVFN0YXRlPiAmIHtcbiAgc3RvcmFnZT86IFN0b3JhZ2VJbnB1dDtcbiAgc2FuZGJveD86IFRTYW5kYm94QmluZGluZ3M7XG4gIHRvb2xzPzogVG9vbHM7XG4gIGhvb2tzPzogQWdlbnRIb29rcztcbiAgbmVlZHNBcHByb3ZhbD86IE5lZWRzQXBwcm92YWxNYXA8VG9vbHM+O1xuICBjb250ZXh0U2NoZW1hPzogei5ab2RUeXBlPFRDb250ZXh0PjtcbiAgc3RhdGVTY2hlbWE/OiB6LlpvZFR5cGU8VFN0YXRlPjtcbiAgbWVzc2FnZU1ldGFkYXRhU2NoZW1hPzogei5ab2RUeXBlPFRNZXNzYWdlTWV0YWRhdGE+O1xuICBsb2dnaW5nPzogT21pdDxMb2dnaW5nQ29uZmlnLCBcIm5hbWVcIj47XG59O1xuXG4vLyBcdTI1MDBcdTI1MDBcdTI1MDAgSW50ZXJuYWwgY29udGV4dCBjYWNoZXMgKGhpZGRlbiBmcm9tIHB1YmxpYyBBUEkpIFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFx1MjUwMFxuXG5jb25zdCBzZXNzaW9uQ29udGV4dHNNYXAgPSBuZXcgV2Vha01hcDxvYmplY3QsIE1hcDxzdHJpbmcsIFNlc3Npb25Db250ZXh0Pj4oKTtcbmNvbnN0IHNhbmRib3hDb250ZXh0c01hcCA9IG5ldyBXZWFrTWFwPG9iamVjdCwgTWFwPHN0cmluZywgU2FuZGJveENvbnRleHQ+PigpO1xuXG5leHBvcnQgZnVuY3Rpb24gZ2V0U2Vzc2lvbkNvbnRleHRzKFxuICBhZ2VudDogQW55QWdlbnRcbik6IE1hcDxzdHJpbmcsIFNlc3Npb25Db250ZXh0PiB7XG4gIGxldCBtYXAgPSBzZXNzaW9uQ29udGV4dHNNYXAuZ2V0KGFnZW50KTtcbiAgaWYgKCFtYXApIHtcbiAgICBtYXAgPSBuZXcgTWFwKCk7XG4gICAgc2Vzc2lvbkNvbnRleHRzTWFwLnNldChhZ2VudCwgbWFwKTtcbiAgfVxuICByZXR1cm4gbWFwO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0U2FuZGJveENvbnRleHRzKFxuICBhZ2VudDogQW55QWdlbnRcbik6IE1hcDxzdHJpbmcsIFNhbmRib3hDb250ZXh0PiB7XG4gIGxldCBtYXAgPSBzYW5kYm94Q29udGV4dHNNYXAuZ2V0KGFnZW50KTtcbiAgaWYgKCFtYXApIHtcbiAgICBtYXAgPSBuZXcgTWFwKCk7XG4gICAgc2FuZGJveENvbnRleHRzTWFwLnNldChhZ2VudCwgbWFwKTtcbiAgfVxuICByZXR1cm4gbWFwO1xufVxuXG4vLyBcdTI1MDBcdTI1MDBcdTI1MDAgQWdlbnQgQ2xhc3MgXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXHUyNTAwXG5cbi8vIGJpb21lLWlnbm9yZSBsaW50L3N1c3BpY2lvdXMvbm9FeHBsaWNpdEFueTogaW50ZW50aW9uYWxseSBlcmFzZWQgXHUyMDEzIHVzaW5nIGBhbnlgIGF2b2lkcyBjb250cmF2YXJpYW5jZSBpc3N1ZXMgd2l0aCB0b29sLXR5cGVkIG1ldGhvZHMgbGlrZSBgc2VuZGBcbmV4cG9ydCB0eXBlIEFueUFnZW50ID0gQWdlbnQ8YW55LCBhbnksIGFueSwgYW55LCBhbnk+O1xuXG5leHBvcnQgdHlwZSBJbmZlclNlc3Npb248QWdlbnQgZXh0ZW5kcyBBbnlBZ2VudCA9IEFueUFnZW50PiA9IFJldHVyblR5cGU8XG4gIEFnZW50W1wic2Vzc2lvblwiXVxuPjtcbmV4cG9ydCB0eXBlIFNlc3Npb25TZW5kQXJnczxcbiAgQWdlbnQgZXh0ZW5kcyBBbnlBZ2VudCxcbiAgU2Vzc2lvbiBleHRlbmRzIEluZmVyU2Vzc2lvbjxBZ2VudD4gPSBJbmZlclNlc3Npb248QWdlbnQ+LFxuPiA9IFBhcmFtZXRlcnM8U2Vzc2lvbltcInNlbmRcIl0+O1xuXG5leHBvcnQgY2xhc3MgQWdlbnQ8XG4gIFRNZXNzYWdlTWV0YWRhdGEgPSB1bmtub3duLFxuICBUb29scyBleHRlbmRzIFRvb2xTZXQgPSBUb29sU2V0LFxuICBUU2FuZGJveEJpbmRpbmdzIGV4dGVuZHMgU2FuZGJveEJpbmRpbmcgfCBTYW5kYm94QmluZGluZ1tdID0gU2FuZGJveEJpbmRpbmcsXG4gIFRDb250ZXh0IGV4dGVuZHMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4gPSBSZWNvcmQ8c3RyaW5nLCBuZXZlcj4sXG4gIFRTdGF0ZSBleHRlbmRzIFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0gUmVjb3JkPHN0cmluZywgdW5rbm93bj4sXG4+IHtcbiAgcmVhZG9ubHkgbmFtZTogc3RyaW5nO1xuICBvcHRpb25zOiBBZ2VudE9wdGlvbnM8XG4gICAgVG9vbHMsXG4gICAgVFNhbmRib3hCaW5kaW5ncyxcbiAgICBUQ29udGV4dCxcbiAgICBUU3RhdGUsXG4gICAgVE1lc3NhZ2VNZXRhZGF0YVxuICA+O1xuXG4gIGNvbnN0cnVjdG9yKFxuICAgIG5hbWU6IHN0cmluZyxcbiAgICBvcHRpb25zPzogQWdlbnRPcHRpb25zPFxuICAgICAgVG9vbHMsXG4gICAgICBUU2FuZGJveEJpbmRpbmdzLFxuICAgICAgVENvbnRleHQsXG4gICAgICBUU3RhdGUsXG4gICAgICBUTWVzc2FnZU1ldGFkYXRhXG4gICAgPlxuICApIHtcbiAgICB0aGlzLm5hbWUgPSBuYW1lO1xuICAgIHRoaXMub3B0aW9ucyA9IG9wdGlvbnMgPz8ge307XG4gICAgdGhpcy5zZXNzaW9uID0gKHNlc3Npb25IYW5kbGU8XG4gICAgICBUb29scyxcbiAgICAgIFRDb250ZXh0LFxuICAgICAgVFNhbmRib3hCaW5kaW5ncyxcbiAgICAgIFRNZXNzYWdlTWV0YWRhdGEsXG4gICAgICBUU3RhdGVcbiAgICA+KS5iaW5kKG51bGwsIHRoaXMgYXMgdW5rbm93biBhcyBBbnlBZ2VudCk7XG4gICAgdGhpcy5zYW5kYm94ID0gKHNhbmRib3hIYW5kbGU8VFNhbmRib3hCaW5kaW5ncz4pLmJpbmQoXG4gICAgICBudWxsLFxuICAgICAgdGhpcyBhcyB1bmtub3duIGFzIEFueUFnZW50XG4gICAgKTtcbiAgfVxuXG4gIHN0YXRpYyBbV09SS0ZMT1dfU0VSSUFMSVpFXShpbnN0YW5jZTogQW55QWdlbnQpIHtcbiAgICByZXR1cm4geyBuYW1lOiBpbnN0YW5jZS5uYW1lIH07XG4gIH1cblxuICBzdGF0aWMgW1dPUktGTE9XX0RFU0VSSUFMSVpFXShkYXRhOiB7IG5hbWU6IHN0cmluZyB9KSB7XG4gICAgcmV0dXJuIGdldEFnZW50KGRhdGEubmFtZSwgdHJ1ZSk7XG4gIH1cblxuICBwcml2YXRlIF9yZXNvbHZlZFN0b3JhZ2U6IFN0b3JhZ2UgfCB1bmRlZmluZWQ7XG5cbiAgZ2V0IHN0b3JhZ2UoKTogU3RvcmFnZSB7XG4gICAgaWYgKHRoaXMuX3Jlc29sdmVkU3RvcmFnZSkge1xuICAgICAgcmV0dXJuIHRoaXMuX3Jlc29sdmVkU3RvcmFnZTtcbiAgICB9XG5cbiAgICBpZiAodGhpcy5vcHRpb25zLnN0b3JhZ2UpIHtcbiAgICAgIHRoaXMuX3Jlc29sdmVkU3RvcmFnZSA9IHRvU3RvcmFnZSh0aGlzLm9wdGlvbnMuc3RvcmFnZSk7XG4gICAgICByZXR1cm4gdGhpcy5fcmVzb2x2ZWRTdG9yYWdlO1xuICAgIH1cblxuICAgIGNvbnN0IGxvZyA9IGNyZWF0ZUxvZ2dlcih7XG4gICAgICBjb25maWc6IHsgLi4udGhpcy5vcHRpb25zLmxvZ2dpbmcsIG5hbWU6IHRoaXMubmFtZSB9LFxuICAgICAgc3Vic3lzdGVtOiBcInN0b3JhZ2VcIixcbiAgICB9KTtcblxuICAgIGlmIChzaG91bGRGYWxsYmFja1RvS1YyKCkpIHtcbiAgICAgIGxvZy5pbmZvKFxuICAgICAgICBgTm8gc3RvcmFnZSBhZGFwdGVyIGNvbmZpZ3VyZWQgXHUyMDE0IHVzaW5nIEtWMiAobmFtZXNwYWNlOiBcIiR7dGhpcy5uYW1lfVwiKS5gXG4gICAgICApO1xuICAgICAgdGhpcy5fcmVzb2x2ZWRTdG9yYWdlID0gdG9TdG9yYWdlKFxuICAgICAgICBrdlN0b3JhZ2VGYWxsYmFjayh7IHByZWZpeDogdGhpcy5uYW1lIH0pXG4gICAgICApO1xuICAgICAgcmV0dXJuIHRoaXMuX3Jlc29sdmVkU3RvcmFnZTtcbiAgICB9XG5cbiAgICBsb2cud2FybihcbiAgICAgIFwiTm8gc3RvcmFnZSBhZGFwdGVyIGNvbmZpZ3VyZWQgXHUyMDE0IHVzaW5nIGxvY2FsU3RvcmFnZSgpLiBEYXRhIHdpbGwgbm90IHBlcnNpc3QgYWNyb3NzIHJlc3RhcnRzLlwiXG4gICAgKTtcbiAgICB0aGlzLl9yZXNvbHZlZFN0b3JhZ2UgPSB0b1N0b3JhZ2UobG9jYWxTdG9yYWdlKCkpO1xuICAgIHJldHVybiB0aGlzLl9yZXNvbHZlZFN0b3JhZ2U7XG4gIH1cblxuICBnZXQgdG9vbHMoKSB7XG4gICAgcmV0dXJuIHsgLi4uYnVpbHRJblRvb2xzLCAuLi50aGlzLm9wdGlvbnMudG9vbHMgfSBhcyB0eXBlb2YgYnVpbHRJblRvb2xzICZcbiAgICAgIFRvb2xzO1xuICB9XG5cbiAgZ2V0IGhvb2tzKCk6IEFnZW50SG9va3Mge1xuICAgIHJldHVybiB0aGlzLm9wdGlvbnMuaG9va3MgPz8ge307XG4gIH1cblxuICByZWFkb25seSBzZXNzaW9uO1xuXG4gIHJlYWRvbmx5IHNhbmRib3g7XG5cbiAgLyoqXG4gICAqIENyZWF0ZSBhIGRlcml2ZWQgYWdlbnQgd2l0aCBtZXJnZWQgb3B0aW9ucy4gVGhlIHJldHVybmVkIGFnZW50IGlzIE5PVFxuICAgKiByZWdpc3RlcmVkIGluIHRoZSB3b3JrZmxvdyByZWdpc3RyeSBcdTIwMTMgaXQncyBhIG9uZS1vZmYgaW5zdGFuY2UgdXNlZCBieVxuICAgKiB0aGUgZnJhbWV3b3JrIHRvIGxheWVyIGZzLXNjYW5uZWQgY29uZmlnIG9udG8gYW4gYWdlbnQudHMgZGVmaW5pdGlvbi5cbiAgICovXG4gIGV4dGVuZChcbiAgICBmbjogKG9wdGlvbnM6IEFnZW50T3B0aW9uczxUb29scywgVFNhbmRib3hCaW5kaW5ncywgVENvbnRleHQsIFRTdGF0ZSwgVE1lc3NhZ2VNZXRhZGF0YT4pID0+IEFnZW50T3B0aW9uczxUb29scywgVFNhbmRib3hCaW5kaW5ncywgVENvbnRleHQsIFRTdGF0ZSwgVE1lc3NhZ2VNZXRhZGF0YT5cbiAgKTogQWdlbnQ8VE1lc3NhZ2VNZXRhZGF0YSwgVG9vbHMsIFRTYW5kYm94QmluZGluZ3MsIFRDb250ZXh0LCBUU3RhdGU+IHtcbiAgICByZXR1cm4gbmV3IEFnZW50KHRoaXMubmFtZSwgZm4odGhpcy5vcHRpb25zKSk7XG4gIH1cblxuICAvKiogUGhhbnRvbSBwcm9wZXJ0eSBmb3IgdHlwZSBpbmZlcmVuY2UuIFVzZSBgdHlwZW9mIG15QWdlbnQuJFVJTWVzc2FnZWAgdG8gZ2V0IHRoZSB0eXBlZCBVSU1lc3NhZ2UuICovXG4gIGRlY2xhcmUgcmVhZG9ubHkgJFVJTWVzc2FnZTogVUlNZXNzYWdlPFxuICAgIFRNZXNzYWdlTWV0YWRhdGEsXG4gICAgQWdlbnREYXRhVHlwZXMsXG4gICAgSW5mZXJVSVRvb2xzPHR5cGVvZiBidWlsdEluVG9vbHMgJiBUb29scz5cbiAgPjtcbn1cblxuZXhwb3J0IGZ1bmN0aW9uIGFnZW50PFxuICBUb29scyBleHRlbmRzIFRvb2xTZXQgPSBUb29sU2V0LFxuICBUU2FuZGJveEJpbmRpbmdzIGV4dGVuZHMgU2FuZGJveEJpbmRpbmcgfCBTYW5kYm94QmluZGluZ1tdID0gU2FuZGJveEJpbmRpbmcsXG4gIFRDb250ZXh0IGV4dGVuZHMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4gPSBSZWNvcmQ8c3RyaW5nLCBuZXZlcj4sXG4gIFRTdGF0ZSBleHRlbmRzIFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0gUmVjb3JkPHN0cmluZywgdW5rbm93bj4sXG4gIFRNZXNzYWdlTWV0YWRhdGEgPSB1bmtub3duLFxuPihcbiAgbmFtZTogc3RyaW5nLFxuICBvcHRpb25zPzogQWdlbnRPcHRpb25zPFxuICAgIFRvb2xzLFxuICAgIFRTYW5kYm94QmluZGluZ3MsXG4gICAgVENvbnRleHQsXG4gICAgVFN0YXRlLFxuICAgIFRNZXNzYWdlTWV0YWRhdGFcbiAgPlxuKSB7XG4gIGNvbnN0IGluc3RhbmNlID0gbmV3IEFnZW50PFxuICAgIFRNZXNzYWdlTWV0YWRhdGEsXG4gICAgVG9vbHMsXG4gICAgVFNhbmRib3hCaW5kaW5ncyxcbiAgICBUQ29udGV4dCxcbiAgICBUU3RhdGVcbiAgPihuYW1lLCBvcHRpb25zKTtcbiAgcmVnaXN0ZXJBZ2VudChuYW1lLCBpbnN0YW5jZSk7XG4gIHJldHVybiBpbnN0YW5jZTtcbn1cblxuLy8gUmUtZXhwb3J0IGV2ZXJ5dGhpbmcgZnJvbSByZWdpc3RyeSBzbyBleGlzdGluZyBpbXBvcnRzIGZyb20gXCIuL2FnZW50L2NsaWVudFwiIGtlZXAgd29ya2luZy5cbmV4cG9ydCB7XG4gIHR5cGUgQWdlbnREYXRhVHlwZXMsXG4gIHR5cGUgQWdlbnRIb29rcyxcbiAgQWdlbnRTdGF0dXMsXG4gIGdldEFnZW50LFxuICB0eXBlIEluZmVyVUlNZXNzYWdlLFxuICB0eXBlIE5lZWRzQXBwcm92YWxNYXAsXG4gIHJlZ2lzdGVyQWdlbnQsXG4gIHR5cGUgVHlwZWRVSU1lc3NhZ2UsXG59IGZyb20gXCIuL3JlZ2lzdHJ5XCI7XG4iLCAiaW1wb3J0IHsgdWxpZCB9IGZyb20gXCJ1bGlkXCI7XG5pbXBvcnQgdHlwZSB7IEFueUFnZW50IH0gZnJvbSBcIi4uL2FnZW50L2NsaWVudFwiO1xuaW1wb3J0IHsgZ2V0U2FuZGJveENvbnRleHRzIH0gZnJvbSBcIi4uL2FnZW50L2NsaWVudFwiO1xuaW1wb3J0IHR5cGUgeyBTYW5kYm94QmluZGluZywgU2FuZGJveEluc3RhbmNlIH0gZnJvbSBcIi4vYWRhcHRlclwiO1xuaW1wb3J0IHsgZXhlYyB9IGZyb20gXCIuL2V4ZWNcIjtcbmltcG9ydCB7IGdldERvbWFpbiB9IGZyb20gXCIuL2dldC1kb21haW5cIjtcbmltcG9ydCB7IGdldFN0YXR1cyB9IGZyb20gXCIuL2dldC1zdGF0dXNcIjtcbmltcG9ydCB7IGtpbGwgfSBmcm9tIFwiLi9raWxsXCI7XG5pbXBvcnQgeyByZWFkRmlsZSB9IGZyb20gXCIuL3JlYWQtZmlsZVwiO1xuaW1wb3J0IHsgc2V0dXAgfSBmcm9tIFwiLi9zZXR1cFwiO1xuaW1wb3J0IHsgc25hcHNob3QgfSBmcm9tIFwiLi9zbmFwc2hvdFwiO1xuaW1wb3J0IHsgc3RhcnQgfSBmcm9tIFwiLi9zdGFydFwiO1xuaW1wb3J0IHsgc3RvcCB9IGZyb20gXCIuL3N0b3BcIjtcbmltcG9ydCB7IHVwZGF0ZU5ldHdvcmtQb2xpY3kgfSBmcm9tIFwiLi91cGRhdGUtbmV0d29yay1wb2xpY3lcIjtcbmltcG9ydCB7IHdyaXRlRmlsZXMgfSBmcm9tIFwiLi93cml0ZS1maWxlc1wiO1xuXG5leHBvcnQgdHlwZSBTYW5kYm94Q29udGV4dCA9IHtcbiAgYWdlbnQ6IEFueUFnZW50O1xuICBzYW5kYm94SWQ6IHN0cmluZyB8IG51bGw7XG4gIF9wZW5kaW5nU2V0dXA/OiBQcm9taXNlPFNhbmRib3hJbnN0YW5jZT47XG4gIF9jd2Q/OiBzdHJpbmc7XG4gIF9zZXNzaW9uSWQ/OiBzdHJpbmc7XG59O1xuXG5leHBvcnQgdHlwZSBTYW5kYm94T3B0aW9ucyA9IHtcbiAgaWQ/OiBzdHJpbmc7XG59O1xuXG5leHBvcnQgdHlwZSBTYW5kYm94SGFuZGxlPFxuICBUU2FuZGJveEJpbmRpbmdzIGV4dGVuZHMgU2FuZGJveEJpbmRpbmcgfCBTYW5kYm94QmluZGluZ1tdID0gU2FuZGJveEJpbmRpbmcsXG4+ID0gUmV0dXJuVHlwZTx0eXBlb2Ygc2FuZGJveEhhbmRsZTxUU2FuZGJveEJpbmRpbmdzPj47XG5cbmV4cG9ydCBmdW5jdGlvbiBzYW5kYm94SGFuZGxlPFxuICBUU2FuZGJveEJpbmRpbmdzIGV4dGVuZHMgU2FuZGJveEJpbmRpbmcgfCBTYW5kYm94QmluZGluZ1tdLFxuPihhZ2VudDogQW55QWdlbnQsIGlkPzogc3RyaW5nIHwgU2FuZGJveE9wdGlvbnMpIHtcbiAgY29uc3Qgc2FuZGJveElkID1cbiAgICB0eXBlb2YgaWQgPT09IFwic3RyaW5nXCIgPyBpZCA6IChpZD8uaWQgPz8gYHNhbmRib3hfJHt1bGlkKCl9YCk7XG5cbiAgY29uc3QgY29udGV4dHMgPSBnZXRTYW5kYm94Q29udGV4dHMoYWdlbnQpO1xuICBsZXQgc2FuZGJveENvbnRleHQgPSBjb250ZXh0cy5nZXQoc2FuZGJveElkKTtcbiAgaWYgKCFzYW5kYm94Q29udGV4dCkge1xuICAgIHNhbmRib3hDb250ZXh0ID0geyBhZ2VudCwgc2FuZGJveElkIH07XG4gICAgY29udGV4dHMuc2V0KHNhbmRib3hJZCwgc2FuZGJveENvbnRleHQpO1xuICB9XG5cbiAgcmV0dXJuIHtcbiAgICBpZDogc2FuZGJveElkLFxuICAgIGdldCBjd2QoKSB7XG4gICAgICByZXR1cm4gc2FuZGJveENvbnRleHQhLl9jd2QgPz8gXCIuXCI7XG4gICAgfSxcbiAgICBzZXR1cDogKHNldHVwPFRTYW5kYm94QmluZGluZ3M+KS5iaW5kKG51bGwsIHNhbmRib3hDb250ZXh0KSxcbiAgICBleGVjOiBleGVjLmJpbmQobnVsbCwgc2FuZGJveENvbnRleHQpLFxuICAgIHJlYWRGaWxlOiByZWFkRmlsZS5iaW5kKG51bGwsIHNhbmRib3hDb250ZXh0KSxcbiAgICB3cml0ZUZpbGVzOiB3cml0ZUZpbGVzLmJpbmQobnVsbCwgc2FuZGJveENvbnRleHQpLFxuICAgIGdldERvbWFpbjogZ2V0RG9tYWluLmJpbmQobnVsbCwgc2FuZGJveENvbnRleHQpLFxuICAgIHVwZGF0ZU5ldHdvcmtQb2xpY3k6IHVwZGF0ZU5ldHdvcmtQb2xpY3kuYmluZChudWxsLCBzYW5kYm94Q29udGV4dCksXG4gICAgc3RhcnQ6IHN0YXJ0LmJpbmQobnVsbCwgc2FuZGJveENvbnRleHQpLFxuICAgIHN0b3A6IHN0b3AuYmluZChudWxsLCBzYW5kYm94Q29udGV4dCksXG4gICAga2lsbDoga2lsbC5iaW5kKG51bGwsIHNhbmRib3hDb250ZXh0KSxcbiAgICBzbmFwc2hvdDogc25hcHNob3QuYmluZChudWxsLCBzYW5kYm94Q29udGV4dCksXG4gICAgZ2V0U3RhdHVzOiBnZXRTdGF0dXMuYmluZChudWxsLCBzYW5kYm94Q29udGV4dCksXG4gIH07XG59XG5cbnR5cGUgU2FuZGJveE1ldGhvZHM8XG4gIFRTYW5kYm94QmluZGluZ3MgZXh0ZW5kcyBTYW5kYm94QmluZGluZyB8IFNhbmRib3hCaW5kaW5nW10sXG4+ID0gT21pdDxTYW5kYm94SGFuZGxlPFRTYW5kYm94QmluZGluZ3M+LCBcImlkXCIgfCBcImN3ZFwiPjtcblxuZXhwb3J0IGZ1bmN0aW9uIGxhenlTYW5kYm94SGFuZGxlPFxuICBUU2FuZGJveEJpbmRpbmdzIGV4dGVuZHMgU2FuZGJveEJpbmRpbmcgfCBTYW5kYm94QmluZGluZ1tdLFxuPihcbiAgYWdlbnQ6IEFueUFnZW50LFxuICBzZXNzaW9uSWQ6IHN0cmluZ1xuKTogU2FuZGJveE1ldGhvZHM8VFNhbmRib3hCaW5kaW5ncz4ge1xuICBjb25zdCBzYW5kYm94Q29udGV4dDogU2FuZGJveENvbnRleHQgPSB7XG4gICAgYWdlbnQsXG4gICAgc2FuZGJveElkOiBudWxsLFxuICAgIF9zZXNzaW9uSWQ6IHNlc3Npb25JZCxcbiAgfTtcblxuICByZXR1cm4ge1xuICAgIHNldHVwOiAoc2V0dXA8VFNhbmRib3hCaW5kaW5ncz4pLmJpbmQobnVsbCwgc2FuZGJveENvbnRleHQpLFxuICAgIGV4ZWM6IGV4ZWMuYmluZChudWxsLCBzYW5kYm94Q29udGV4dCksXG4gICAgcmVhZEZpbGU6IHJlYWRGaWxlLmJpbmQobnVsbCwgc2FuZGJveENvbnRleHQpLFxuICAgIHdyaXRlRmlsZXM6IHdyaXRlRmlsZXMuYmluZChudWxsLCBzYW5kYm94Q29udGV4dCksXG4gICAgZ2V0RG9tYWluOiBnZXREb21haW4uYmluZChudWxsLCBzYW5kYm94Q29udGV4dCksXG4gICAgdXBkYXRlTmV0d29ya1BvbGljeTogdXBkYXRlTmV0d29ya1BvbGljeS5iaW5kKG51bGwsIHNhbmRib3hDb250ZXh0KSxcbiAgICBzdGFydDogc3RhcnQuYmluZChudWxsLCBzYW5kYm94Q29udGV4dCksXG4gICAgc3RvcDogc3RvcC5iaW5kKG51bGwsIHNhbmRib3hDb250ZXh0KSxcbiAgICBraWxsOiBraWxsLmJpbmQobnVsbCwgc2FuZGJveENvbnRleHQpLFxuICAgIHNuYXBzaG90OiBzbmFwc2hvdC5iaW5kKG51bGwsIHNhbmRib3hDb250ZXh0KSxcbiAgICBnZXRTdGF0dXM6IGdldFN0YXR1cy5iaW5kKG51bGwsIHNhbmRib3hDb250ZXh0KSxcbiAgfTtcbn1cbiIsICJleHBvcnQgY29uc3QgaXNWZXJjZWwgPSBhc3luYyAoKSA9PiB7XG4gIHRyeSB7XG4gICAgY29uc3QgeyBnZXRWZXJjZWxPaWRjVG9rZW4gfSA9IGF3YWl0IGltcG9ydChcIkB2ZXJjZWwvb2lkY1wiKTtcbiAgICByZXR1cm4gQm9vbGVhbihhd2FpdCBnZXRWZXJjZWxPaWRjVG9rZW4oKSk7XG4gIH0gY2F0Y2gge1xuICAgIHJldHVybiBmYWxzZTtcbiAgfVxufTtcbiIsICJpbXBvcnQgeyBpc1ZlcmNlbCB9IGZyb20gXCIuLi9hZ2VudC9pcy12ZXJjZWxcIjtcbmltcG9ydCB7IFNhbmRib3hFcnJvciwgU2FuZGJveE5vdEZvdW5kRXJyb3IgfSBmcm9tIFwiLi4vZXJyb3JzXCI7XG5pbXBvcnQgdHlwZSB7IFNhbmRib3ggfSBmcm9tIFwiLi4vc3RvcmFnZS90eXBlc1wiO1xuaW1wb3J0IHsgY3JlYXRlTG9nZ2VyIH0gZnJvbSBcIi4uL3V0aWxzL2xvZ2dlclwiO1xuaW1wb3J0IHR5cGUge1xuICBTYW5kYm94QmluZGluZyxcbiAgU2FuZGJveEluc3RhbmNlLFxuICBTYW5kYm94U2V0dXBGaWVsZHMsXG4gIFdpdGhDb25maWcsXG59IGZyb20gXCIuL2FkYXB0ZXJcIjtcbmltcG9ydCB0eXBlIHsgQnVpbHRJbkJpbmRpbmdzIH0gZnJvbSBcIi4vYmluZGluZ3NcIjtcbmltcG9ydCB0eXBlIHsgU2FuZGJveENvbnRleHQgfSBmcm9tIFwiLi9oYW5kbGVcIjtcblxuY29uc3QgUE9MTF9JTlRFUlZBTF9NUyA9IDUwO1xuY29uc3QgUE9MTF9USU1FT1VUX01TID0gNSAqIDYwICogMTAwMDtcblxudHlwZSBTZXR1cENvbW1vbkZpZWxkcyA9IFBhcnRpYWw8U2FuZGJveFNldHVwRmllbGRzPiAmIHtcbiAgY3dkPzogc3RyaW5nO1xuICBydW4/OiAoc2FuZGJveDogU2FuZGJveEluc3RhbmNlKSA9PiBQcm9taXNlPHZvaWQ+O1xuICBzaWduYWw/OiBBYm9ydFNpZ25hbDtcbn07XG5cbnR5cGUgVG9CaW5kaW5nVW5pb248VCBleHRlbmRzIFNhbmRib3hCaW5kaW5nIHwgU2FuZGJveEJpbmRpbmdbXT4gPVxuICBUIGV4dGVuZHMgU2FuZGJveEJpbmRpbmdbXSA/IFRbbnVtYmVyXSA6IFQ7XG5cbnR5cGUgU2V0dXBPcHRzRm9yQmluZGluZzxCIGV4dGVuZHMgU2FuZGJveEJpbmRpbmc+ID1cbiAgLy8gYmlvbWUtaWdub3JlIGxpbnQvc3VzcGljaW91cy9ub0V4cGxpY2l0QW55OiAuXG4gIEIgZXh0ZW5kcyBTYW5kYm94QmluZGluZzxpbmZlciBULCBpbmZlciBDLCBhbnk+XG4gICAgPyB7IHR5cGU6IFQgfSAmIFNldHVwQ29tbW9uRmllbGRzICYgV2l0aENvbmZpZzxDPlxuICAgIDogbmV2ZXI7XG5cbnR5cGUgSXNEZWZhdWx0PFQgZXh0ZW5kcyBTYW5kYm94QmluZGluZyB8IFNhbmRib3hCaW5kaW5nW10+ID1cbiAgU2FuZGJveEJpbmRpbmcgZXh0ZW5kcyBUID8gdHJ1ZSA6IGZhbHNlO1xuXG5leHBvcnQgdHlwZSBTZXR1cE9wdHM8XG4gIFRTYW5kYm94QmluZGluZ3MgZXh0ZW5kcyBTYW5kYm94QmluZGluZyB8IFNhbmRib3hCaW5kaW5nW10sXG4+ID1cbiAgfCBTZXR1cE9wdHNGb3JCaW5kaW5nPFxuICAgICAgSXNEZWZhdWx0PFRTYW5kYm94QmluZGluZ3M+IGV4dGVuZHMgdHJ1ZVxuICAgICAgICA/IEJ1aWx0SW5CaW5kaW5nc1xuICAgICAgICA6IFRvQmluZGluZ1VuaW9uPFRTYW5kYm94QmluZGluZ3M+XG4gICAgPlxuICB8IChTZXR1cENvbW1vbkZpZWxkcyAmIHsgdHlwZT86IG5ldmVyIH0pO1xuXG5leHBvcnQgdHlwZSBTZXR1cFJlc3VsdCA9IHtcbiAgZG9uZTogUHJvbWlzZTxTYW5kYm94SW5zdGFuY2U+O1xufTtcblxuZnVuY3Rpb24gaXNTaW5nbGVCaW5kaW5nKFxuICBzYW5kYm94OiBTYW5kYm94QmluZGluZyB8IFNhbmRib3hCaW5kaW5nW11cbik6IHNhbmRib3ggaXMgU2FuZGJveEJpbmRpbmcge1xuICByZXR1cm4gIUFycmF5LmlzQXJyYXkoc2FuZGJveCk7XG59XG5cbi8qKiBBdXRvLWRldGVjdCB0aGUgZGVmYXVsdCBiaW5kaW5nOiB2ZXJjZWwgXHUyMTkyIGRvY2tlciBcdTIxOTIgbG9jYWwgY2FzY2FkZS4gKi9cbmFzeW5jIGZ1bmN0aW9uIGdldERlZmF1bHRCaW5kaW5nKCk6IFByb21pc2U8U2FuZGJveEJpbmRpbmc+IHtcbiAgaWYgKGF3YWl0IGlzVmVyY2VsKCkpIHtcbiAgICBjb25zdCB7IHZlcmNlbFNhbmRib3ggfSA9IGF3YWl0IGltcG9ydChcIi4vYmluZGluZ3MvdmVyY2VsXCIpO1xuICAgIHJldHVybiB2ZXJjZWxTYW5kYm94KCk7XG4gIH1cblxuICBpZiAoYXdhaXQgaXNEb2NrZXJBdmFpbGFibGUoKSkge1xuICAgIGNvbnN0IHsgZG9ja2VyU2FuZGJveCB9ID0gYXdhaXQgaW1wb3J0KFwiLi9iaW5kaW5ncy9kb2NrZXJcIik7XG4gICAgcmV0dXJuIGRvY2tlclNhbmRib3goKTtcbiAgfVxuXG4gIGNvbnN0IHsgbG9jYWxTYW5kYm94IH0gPSBhd2FpdCBpbXBvcnQoXCIuL2JpbmRpbmdzL2xvY2FsXCIpO1xuICByZXR1cm4gbG9jYWxTYW5kYm94KCk7XG59XG5cbmFzeW5jIGZ1bmN0aW9uIGlzRG9ja2VyQXZhaWxhYmxlKCk6IFByb21pc2U8Ym9vbGVhbj4ge1xuICB0cnkge1xuICAgIGNvbnN0IHsgZXhlY1N5bmMgfSA9IGF3YWl0IGltcG9ydChcIm5vZGU6Y2hpbGRfcHJvY2Vzc1wiKTtcbiAgICBleGVjU3luYyhcImRvY2tlciBzYW5kYm94IC0taGVscFwiLCB7IHN0ZGlvOiBcImlnbm9yZVwiLCB0aW1lb3V0OiA1MDAwIH0pO1xuICAgIHJldHVybiB0cnVlO1xuICB9IGNhdGNoIHtcbiAgICByZXR1cm4gZmFsc2U7XG4gIH1cbn1cblxuYXN5bmMgZnVuY3Rpb24gcmVzb2x2ZUJpbmRpbmcoXG4gIHNhbmRib3g6IFNhbmRib3hCaW5kaW5nIHwgU2FuZGJveEJpbmRpbmdbXSB8IHVuZGVmaW5lZCxcbiAgdHlwZT86IHN0cmluZ1xuKTogUHJvbWlzZTx7IGJpbmRpbmc6IFNhbmRib3hCaW5kaW5nOyBiaW5kaW5nTmFtZTogc3RyaW5nIH0+IHtcbiAgaWYgKCFzYW5kYm94KSB7XG4gICAgY29uc3QgYmluZGluZyA9IGF3YWl0IGdldERlZmF1bHRCaW5kaW5nKCk7XG4gICAgcmV0dXJuIHsgYmluZGluZywgYmluZGluZ05hbWU6IGJpbmRpbmcudHlwZSB9O1xuICB9XG4gIGlmIChpc1NpbmdsZUJpbmRpbmcoc2FuZGJveCkpIHtcbiAgICByZXR1cm4geyBiaW5kaW5nOiBzYW5kYm94LCBiaW5kaW5nTmFtZTogc2FuZGJveC50eXBlIH07XG4gIH1cbiAgaWYgKHR5cGUpIHtcbiAgICBjb25zdCBtYXRjaCA9IHNhbmRib3guZmluZCgoYikgPT4gYi50eXBlID09PSB0eXBlKTtcbiAgICBpZiAoIW1hdGNoKSB7XG4gICAgICB0aHJvdyBuZXcgU2FuZGJveEVycm9yKHtcbiAgICAgICAgcmVhc29uOiBgVW5rbm93biBiaW5kaW5nIHR5cGU6IFwiJHt0eXBlfVwiLiBBdmFpbGFibGU6ICR7c2FuZGJveC5tYXAoKGIpID0+IGIudHlwZSkuam9pbihcIiwgXCIpfWAsXG4gICAgICB9KTtcbiAgICB9XG4gICAgcmV0dXJuIHsgYmluZGluZzogbWF0Y2gsIGJpbmRpbmdOYW1lOiBtYXRjaC50eXBlIH07XG4gIH1cbiAgaWYgKHNhbmRib3gubGVuZ3RoID09PSAwKSB7XG4gICAgdGhyb3cgbmV3IFNhbmRib3hFcnJvcih7IHJlYXNvbjogXCJObyBzYW5kYm94IGJpbmRpbmdzIGNvbmZpZ3VyZWRcIiB9KTtcbiAgfVxuICByZXR1cm4geyBiaW5kaW5nOiBzYW5kYm94WzBdLCBiaW5kaW5nTmFtZTogc2FuZGJveFswXS50eXBlIH07XG59XG5cbmFzeW5jIGZ1bmN0aW9uIHJlc29sdmVTYW5kYm94SWQoY3R4OiBTYW5kYm94Q29udGV4dCk6IFByb21pc2U8c3RyaW5nPiB7XG4gIGlmIChjdHguc2FuZGJveElkKSB7XG4gICAgcmV0dXJuIGN0eC5zYW5kYm94SWQ7XG4gIH1cbiAgaWYgKCFjdHguX3Nlc3Npb25JZCkge1xuICAgIHRocm93IG5ldyBFcnJvcihcbiAgICAgIFwiU2FuZGJveENvbnRleHQgaGFzIG5vIHNhbmRib3hJZCBhbmQgbm8gc2Vzc2lvbklkIHRvIHJlc29sdmUgZnJvbVwiXG4gICAgKTtcbiAgfVxuICBjb25zdCByZWNvcmQgPSBhd2FpdCBjdHguYWdlbnQuc3RvcmFnZS5zZXNzaW9uLmdldChjdHguX3Nlc3Npb25JZCk7XG4gIGN0eC5zYW5kYm94SWQgPSByZWNvcmQ/LnNhbmRib3hJZCA/PyBjdHguX3Nlc3Npb25JZDtcbiAgcmV0dXJuIGN0eC5zYW5kYm94SWQ7XG59XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBzZXR1cDxcbiAgVFNhbmRib3hCaW5kaW5ncyBleHRlbmRzIFNhbmRib3hCaW5kaW5nIHwgU2FuZGJveEJpbmRpbmdbXSxcbj4oXG4gIGN0eDogU2FuZGJveENvbnRleHQsXG4gIG9wdHM/OiBTZXR1cE9wdHM8VFNhbmRib3hCaW5kaW5ncz5cbik6IFByb21pc2U8U2V0dXBSZXN1bHQ+IHtcbiAgXCJ1c2Ugc3RlcFwiO1xuICBjb25zdCBzYW5kYm94SWQgPSBhd2FpdCByZXNvbHZlU2FuZGJveElkKGN0eCk7XG4gIGlmIChjdHguX3BlbmRpbmdTZXR1cCkge1xuICAgIHJldHVybiB7IGRvbmU6IGN0eC5fcGVuZGluZ1NldHVwIH07XG4gIH1cblxuICBjb25zdCBhZ2VudCA9IGN0eC5hZ2VudDtcbiAgY29uc3QgbG9nID0gY3JlYXRlTG9nZ2VyKHtcbiAgICBjb25maWc6IHsgLi4uYWdlbnQub3B0aW9ucy5sb2dnaW5nLCBuYW1lOiBhZ2VudC5uYW1lIH0sXG4gICAgc3Vic3lzdGVtOiBcInNhbmRib3g6c2V0dXBcIixcbiAgICBjb250ZXh0OiB7IHNhbmRib3hJZCB9LFxuICB9KTtcblxuICBjb25zdCBkb25lUmVzb2x2ZUJpbmRpbmcgPSBsb2cudGltZShcInJlc29sdmUgYmluZGluZ1wiLCB7XG4gICAgcmVxdWVzdGVkVHlwZTogb3B0cz8udHlwZSxcbiAgfSk7XG4gIGNvbnN0IHsgYmluZGluZywgYmluZGluZ05hbWUgfSA9IGF3YWl0IHJlc29sdmVCaW5kaW5nKFxuICAgIGFnZW50Lm9wdGlvbnMuc2FuZGJveCxcbiAgICBvcHRzPy50eXBlXG4gICk7XG4gIGRvbmVSZXNvbHZlQmluZGluZygpO1xuXG4gIGlmIChvcHRzPy5jd2QpIHtcbiAgICBjdHguX2N3ZCA9IG9wdHMuY3dkO1xuICB9XG5cbiAgY29uc3Qgc2V0dXBMb2cgPSBsb2cud2l0aENvbnRleHQoe1xuICAgIGJpbmRpbmc6IGJpbmRpbmdOYW1lLFxuICAgIHZlcnNpb246IG9wdHM/LnZlcnNpb24gPz8gYmluZGluZy5kZWZhdWx0cz8udmVyc2lvbixcbiAgfSk7XG5cbiAgLy8gYmlvbWUtaWdub3JlIGxpbnQvc3VzcGljaW91cy9ub0NvbmZ1c2luZ1ZvaWRUeXBlOiAuXG4gIGxldCByZXNvbHZlQmxvY2tpbmdQcm9taXNlID0gKHZhbHVlOiB2b2lkKSA9PiB2YWx1ZTtcbiAgLy8gYmlvbWUtaWdub3JlIGxpbnQvc3VzcGljaW91cy9ub0V4cGxpY2l0QW55OiAuXG4gIC8vIGJpb21lLWlnbm9yZSBsaW50L3N1c3BpY2lvdXMvbm9FbXB0eUJsb2NrU3RhdGVtZW50czogLlxuICBsZXQgcmVqZWN0QmxvY2tpbmdQcm9taXNlID0gKF9yZWFzb24/OiBhbnkpID0+IHt9O1xuICBjb25zdCBibG9ja2luZ1Byb21pc2UgPSBuZXcgUHJvbWlzZTx2b2lkPigocmVzb2x2ZSwgcmVqZWN0KSA9PiB7XG4gICAgcmVzb2x2ZUJsb2NraW5nUHJvbWlzZSA9IHJlc29sdmU7XG4gICAgcmVqZWN0QmxvY2tpbmdQcm9taXNlID0gcmVqZWN0O1xuICB9KTtcblxuICBjdHguX3BlbmRpbmdTZXR1cCA9IChhc3luYyAoKSA9PiB7XG4gICAgbGV0IHNsZWVwT2xkSW5zdGFuY2U6IFByb21pc2U8dm9pZD4gfCB1bmRlZmluZWQ7XG4gICAgY29uc3QgZG9uZVBlbmRpbmdTZXR1cCA9IHNldHVwTG9nLnRpbWUoXCJwZW5kaW5nIHNldHVwXCIpO1xuXG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IGRvbmVTdG9yYWdlR2V0ID0gc2V0dXBMb2cudGltZShcInN0b3JhZ2Uuc2FuZGJveC5nZXRcIik7XG4gICAgICBjb25zdCByZWNvcmQgPSBhd2FpdCBhZ2VudC5zdG9yYWdlLnNhbmRib3guZ2V0KHNhbmRib3hJZCk7XG4gICAgICBkb25lU3RvcmFnZUdldCgpO1xuXG4gICAgICBjb25zdCBiaW5kaW5nRGVmYXVsdHMgPSBiaW5kaW5nLmRlZmF1bHRzID8/IHt9O1xuICAgICAgY29uc3QgZWZmZWN0aXZlVmVyc2lvbiA9IG9wdHM/LnZlcnNpb24gPz8gYmluZGluZ0RlZmF1bHRzLnZlcnNpb24gPz8gbnVsbDtcbiAgICAgIGNvbnN0IGVmZmVjdGl2ZVJ1biA9IG9wdHM/LnJ1biA/PyBiaW5kaW5nRGVmYXVsdHMucnVuO1xuICAgICAgY29uc3QgZWZmZWN0aXZlTmV0d29ya1BvbGljeSA9XG4gICAgICAgIG9wdHM/Lm5ldHdvcmtQb2xpY3kgPz8gYmluZGluZ0RlZmF1bHRzLm5ldHdvcmtQb2xpY3kgPz8gbnVsbDtcbiAgICAgIGNvbnN0IGVmZmVjdGl2ZUN3ZCA9XG4gICAgICAgIGN0eC5fY3dkIHx8IG9wdHM/LmN3ZCB8fCBiaW5kaW5nRGVmYXVsdHMuY3dkIHx8IG51bGw7XG4gICAgICBpZiAoZWZmZWN0aXZlQ3dkKSB7XG4gICAgICAgIGN0eC5fY3dkID0gZWZmZWN0aXZlQ3dkO1xuICAgICAgfVxuICAgICAgY29uc3QgZWZmZWN0aXZlQ29uZmlnID0ge1xuICAgICAgICAuLi5iaW5kaW5nRGVmYXVsdHMuY29uZmlnLFxuICAgICAgICAvLyBiaW9tZS1pZ25vcmUgbGludC9zdXNwaWNpb3VzL25vRXhwbGljaXRBbnk6IG9wdHMgaXMgYSB1bmlvbjsgY29uZmlnIGV4aXN0cyBvbiBiaW5kaW5nLXNwZWNpZmljIGJyYW5jaGVzXG4gICAgICAgIC4uLihvcHRzIGFzIGFueSk/LmNvbmZpZyxcbiAgICAgIH07XG5cbiAgICAgIGlmIChyZWNvcmQpIHtcbiAgICAgICAgY29uc3QgdmVyc2lvbk1hdGNoZXMgPVxuICAgICAgICAgICFlZmZlY3RpdmVWZXJzaW9uIHx8IHJlY29yZC5zZXR1cD8udmVyc2lvbiA9PT0gZWZmZWN0aXZlVmVyc2lvbjtcblxuICAgICAgICBpZiAodmVyc2lvbk1hdGNoZXMgJiYgcmVjb3JkLnNldHVwPy5jb21wbGV0ZWRBdCkge1xuICAgICAgICAgIHJlc29sdmVCbG9ja2luZ1Byb21pc2UoKTtcbiAgICAgICAgICBjb25zdCBkb25lQ29ubmVjdEV4aXN0aW5nID0gc2V0dXBMb2cudGltZShcbiAgICAgICAgICAgIFwiY29ubmVjdCBleGlzdGluZyBzYW5kYm94XCIsXG4gICAgICAgICAgICB7XG4gICAgICAgICAgICAgIHNldHVwU3RhdGU6IFwiY29tcGxldGVkXCIsXG4gICAgICAgICAgICB9XG4gICAgICAgICAgKTtcbiAgICAgICAgICBjb25zdCBpbnN0YW5jZSA9IGF3YWl0IGJpbmRpbmcuY29ubmVjdCh7XG4gICAgICAgICAgICBhZ2VudCxcbiAgICAgICAgICAgIG1ldGFkYXRhOiByZWNvcmQuc2V0dXAubWV0YWRhdGEsXG4gICAgICAgICAgICBjd2Q6IHJlY29yZC5zZXR1cC5jd2QsXG4gICAgICAgICAgICBzaWduYWw6IG9wdHM/LnNpZ25hbCxcbiAgICAgICAgICAgIGxvZzogc2V0dXBMb2csXG4gICAgICAgICAgfSk7XG4gICAgICAgICAgZG9uZUNvbm5lY3RFeGlzdGluZygpO1xuICAgICAgICAgIGN0eC5fY3dkID0gaW5zdGFuY2UuY3dkO1xuICAgICAgICAgIHJldHVybiBpbnN0YW5jZTtcbiAgICAgICAgfVxuXG4gICAgICAgIGlmICh2ZXJzaW9uTWF0Y2hlcyAmJiAhcmVjb3JkLnNldHVwPy5jb21wbGV0ZWRBdCkge1xuICAgICAgICAgIHJlc29sdmVCbG9ja2luZ1Byb21pc2UoKTtcbiAgICAgICAgICBjb25zdCBkb25lUG9sbFVudGlsUmVhZHkgPSBzZXR1cExvZy50aW1lKFwicG9sbCB1bnRpbCByZWFkeVwiKTtcbiAgICAgICAgICBhd2FpdCBwb2xsVW50aWxSZWFkeShjdHgsIG9wdHM/LnNpZ25hbCk7XG4gICAgICAgICAgZG9uZVBvbGxVbnRpbFJlYWR5KCk7XG5cbiAgICAgICAgICBjb25zdCBkb25lU3RvcmFnZUdldFBvc3RQb2xsID0gc2V0dXBMb2cudGltZShcbiAgICAgICAgICAgIFwic3RvcmFnZS5zYW5kYm94LmdldCAocG9zdC1wb2xsKVwiXG4gICAgICAgICAgKTtcbiAgICAgICAgICBjb25zdCByZWFkeSA9IGF3YWl0IGFnZW50LnN0b3JhZ2Uuc2FuZGJveC5nZXQoc2FuZGJveElkKTtcbiAgICAgICAgICBkb25lU3RvcmFnZUdldFBvc3RQb2xsKCk7XG5cbiAgICAgICAgICBpZiAoIXJlYWR5KSB7XG4gICAgICAgICAgICB0aHJvdyBuZXcgU2FuZGJveE5vdEZvdW5kRXJyb3IoeyBpZDogc2FuZGJveElkIH0pO1xuICAgICAgICAgIH1cbiAgICAgICAgICBjb25zdCBkb25lQ29ubmVjdEV4aXN0aW5nID0gc2V0dXBMb2cudGltZShcbiAgICAgICAgICAgIFwiY29ubmVjdCBleGlzdGluZyBzYW5kYm94XCIsXG4gICAgICAgICAgICB7XG4gICAgICAgICAgICAgIHNldHVwU3RhdGU6IFwicG9zdC1wb2xsXCIsXG4gICAgICAgICAgICB9XG4gICAgICAgICAgKTtcbiAgICAgICAgICBjb25zdCBpbnN0YW5jZSA9IGF3YWl0IGJpbmRpbmcuY29ubmVjdCh7XG4gICAgICAgICAgICBhZ2VudCxcbiAgICAgICAgICAgIG1ldGFkYXRhOiByZWFkeS5zZXR1cC5tZXRhZGF0YSxcbiAgICAgICAgICAgIGN3ZDogcmVhZHkuc2V0dXAuY3dkLFxuICAgICAgICAgICAgc2lnbmFsOiBvcHRzPy5zaWduYWwsXG4gICAgICAgICAgICBsb2c6IHNldHVwTG9nLFxuICAgICAgICAgIH0pO1xuICAgICAgICAgIGRvbmVDb25uZWN0RXhpc3RpbmcoKTtcbiAgICAgICAgICBjdHguX2N3ZCA9IGluc3RhbmNlLmN3ZDtcbiAgICAgICAgICByZXR1cm4gaW5zdGFuY2U7XG4gICAgICAgIH1cblxuICAgICAgICBpZiAocmVjb3JkLnNldHVwLm1ldGFkYXRhKSB7XG4gICAgICAgICAgY29uc3Qgb2xkSW5zdGFuY2VNZXRhZGF0YSA9IHJlY29yZC5zZXR1cC5tZXRhZGF0YTtcbiAgICAgICAgICBzbGVlcE9sZEluc3RhbmNlID0gKGFzeW5jICgpID0+IHtcbiAgICAgICAgICAgIGNvbnN0IGRvbmVTdG9wT2xkSW5zdGFuY2UgPSBzZXR1cExvZy50aW1lKFxuICAgICAgICAgICAgICBcInN0b3Agb2xkIHNhbmRib3ggaW5zdGFuY2VcIlxuICAgICAgICAgICAgKTtcbiAgICAgICAgICAgIHRyeSB7XG4gICAgICAgICAgICAgIGNvbnN0IG9sZCA9IGF3YWl0IGJpbmRpbmcuY29ubmVjdCh7XG4gICAgICAgICAgICAgICAgYWdlbnQsXG4gICAgICAgICAgICAgICAgbWV0YWRhdGE6IG9sZEluc3RhbmNlTWV0YWRhdGEsXG4gICAgICAgICAgICAgICAgbG9nOiBzZXR1cExvZyxcbiAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICAgIGF3YWl0IG9sZC5zdG9wKCk7XG4gICAgICAgICAgICB9IGNhdGNoIHtcbiAgICAgICAgICAgICAgZG9uZVN0b3BPbGRJbnN0YW5jZSgpO1xuICAgICAgICAgICAgICByZXR1cm47XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBkb25lU3RvcE9sZEluc3RhbmNlKCk7XG4gICAgICAgICAgfSkoKTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBjb25zdCBub3cgPSBEYXRlLm5vdygpO1xuICAgICAgY29uc3Qgc2V0dXBfOiBTYW5kYm94W1wic2V0dXBcIl0gPSB7XG4gICAgICAgIGJpbmRpbmc6IGJpbmRpbmdOYW1lLFxuICAgICAgICB2ZXJzaW9uOiBlZmZlY3RpdmVWZXJzaW9uID8/IHJlY29yZD8uc2V0dXA/LnZlcnNpb24gPz8gbnVsbCxcbiAgICAgICAgbmV0d29ya1BvbGljeTpcbiAgICAgICAgICBlZmZlY3RpdmVOZXR3b3JrUG9saWN5ID8/IHJlY29yZD8uc2V0dXA/Lm5ldHdvcmtQb2xpY3kgPz8gbnVsbCxcbiAgICAgICAgY29tcGxldGVkQXQ6IG51bGwsXG4gICAgICAgIG1ldGFkYXRhOiBudWxsLFxuICAgICAgICBjd2Q6IG51bGwsXG4gICAgICB9O1xuXG4gICAgICBjb25zdCBuZXdSZWNvcmQ6IFNhbmRib3ggPSB7XG4gICAgICAgIGlkOiBzYW5kYm94SWQsXG4gICAgICAgIHNldHVwOiBzZXR1cF8sXG4gICAgICAgIGNyZWF0ZWRBdDogcmVjb3JkPy5jcmVhdGVkQXQgPz8gbm93LFxuICAgICAgICBsYXN0QWN0aXZlQXQ6IG5vdyxcbiAgICAgIH07XG5cbiAgICAgIGNvbnN0IGRvbmVTdG9yYWdlU2V0UGVuZGluZyA9IHNldHVwTG9nLnRpbWUoXG4gICAgICAgIFwic3RvcmFnZS5zYW5kYm94LnNldCAoc2V0dXAgcGVuZGluZylcIlxuICAgICAgKTtcbiAgICAgIGF3YWl0IGFnZW50LnN0b3JhZ2Uuc2FuZGJveC5zZXQoc2FuZGJveElkLCBuZXdSZWNvcmQpO1xuICAgICAgZG9uZVN0b3JhZ2VTZXRQZW5kaW5nKCk7XG5cbiAgICAgIHJlc29sdmVCbG9ja2luZ1Byb21pc2UoKTtcblxuICAgICAgY29uc3QgZG9uZUJpbmRpbmdDcmVhdGUgPSBzZXR1cExvZy50aW1lKFwiYmluZGluZy5jcmVhdGVcIik7XG4gICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBiaW5kaW5nLmNyZWF0ZSh7XG4gICAgICAgIGN3ZDogY3R4Ll9jd2QgPz8gXCIuXCIsXG4gICAgICAgIGFnZW50LFxuICAgICAgICBzZXR1cDoge1xuICAgICAgICAgIC4uLnNldHVwXyxcbiAgICAgICAgICBydW46IGVmZmVjdGl2ZVJ1bixcbiAgICAgICAgICBjb25maWc6IGVmZmVjdGl2ZUNvbmZpZyxcbiAgICAgICAgfSxcbiAgICAgICAgcmVjb3JkOiBuZXdSZWNvcmQsXG4gICAgICAgIHNpZ25hbDogb3B0cz8uc2lnbmFsLFxuICAgICAgICBsb2c6IHNldHVwTG9nLFxuICAgICAgfSk7XG4gICAgICBkb25lQmluZGluZ0NyZWF0ZSgpO1xuXG4gICAgICBzZXR1cF8uY29tcGxldGVkQXQgPSBEYXRlLm5vdygpO1xuICAgICAgc2V0dXBfLm1ldGFkYXRhID0gcmVzdWx0Lm1ldGFkYXRhO1xuICAgICAgc2V0dXBfLmN3ZCA9IHJlc3VsdC5pbnN0YW5jZS5jd2Q7XG5cbiAgICAgIGNvbnN0IGRvbmVTdG9yYWdlVXBkYXRlQ29tcGxldGUgPSBzZXR1cExvZy50aW1lKFxuICAgICAgICBcInN0b3JhZ2Uuc2FuZGJveC51cGRhdGUgKHNldHVwIGNvbXBsZXRlKVwiXG4gICAgICApO1xuICAgICAgYXdhaXQgYWdlbnQuc3RvcmFnZS5zYW5kYm94LnVwZGF0ZShzYW5kYm94SWQsIHtcbiAgICAgICAgc2V0dXA6IHNldHVwXyxcbiAgICAgICAgbGFzdEFjdGl2ZUF0OiBEYXRlLm5vdygpLFxuICAgICAgfSk7XG4gICAgICBkb25lU3RvcmFnZVVwZGF0ZUNvbXBsZXRlKCk7XG5cbiAgICAgIGN0eC5fY3dkID0gcmVzdWx0Lmluc3RhbmNlLmN3ZDtcbiAgICAgIHJldHVybiByZXN1bHQuaW5zdGFuY2U7XG4gICAgfSBjYXRjaCAoZXJyKSB7XG4gICAgICBjdHguX3BlbmRpbmdTZXR1cCA9IHVuZGVmaW5lZDtcbiAgICAgIHJlamVjdEJsb2NraW5nUHJvbWlzZShlcnIpO1xuICAgICAgdGhyb3cgZXJyO1xuICAgIH0gZmluYWxseSB7XG4gICAgICBjb25zdCBkb25lRmluYWxpemUgPSBzZXR1cExvZy50aW1lKFwiZmluYWxpemUgc2V0dXBcIik7XG4gICAgICBhd2FpdCBzbGVlcE9sZEluc3RhbmNlO1xuICAgICAgZG9uZUZpbmFsaXplKCk7XG4gICAgICBkb25lUGVuZGluZ1NldHVwKCk7XG4gICAgfVxuICB9KSgpO1xuXG4gIGNvbnN0IGRvbmVXYWl0Rm9yVW5ibG9jayA9IHNldHVwTG9nLnRpbWUoXCJ3YWl0IGZvciBzZXR1cCB1bmJsb2NrXCIpO1xuICBhd2FpdCBibG9ja2luZ1Byb21pc2U7XG4gIGRvbmVXYWl0Rm9yVW5ibG9jaygpO1xuICByZXR1cm4geyBkb25lOiBjdHguX3BlbmRpbmdTZXR1cCB9O1xufVxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gcmVzb2x2ZShcbiAgY3R4OiBTYW5kYm94Q29udGV4dCxcbiAgc2lnbmFsPzogQWJvcnRTaWduYWxcbik6IFByb21pc2U8U2FuZGJveEluc3RhbmNlPiB7XG4gIGNvbnN0IHNhbmRib3hJZCA9IGF3YWl0IHJlc29sdmVTYW5kYm94SWQoY3R4KTtcbiAgY29uc3QgYWdlbnQgPSBjdHguYWdlbnQ7XG4gIGNvbnN0IGxvZyA9IGNyZWF0ZUxvZ2dlcih7XG4gICAgY29uZmlnOiB7IC4uLmFnZW50Lm9wdGlvbnMubG9nZ2luZywgbmFtZTogYWdlbnQubmFtZSB9LFxuICAgIHN1YnN5c3RlbTogXCJzYW5kYm94OnJlc29sdmVcIixcbiAgICBjb250ZXh0OiB7IHNhbmRib3hJZCB9LFxuICB9KTtcblxuICBpZiAoY3R4Ll9wZW5kaW5nU2V0dXApIHtcbiAgICByZXR1cm4gYXdhaXQgY3R4Ll9wZW5kaW5nU2V0dXA7XG4gIH1cblxuICBsZXQgcmVjb3JkID0gYXdhaXQgYWdlbnQuc3RvcmFnZS5zYW5kYm94LmdldChzYW5kYm94SWQpO1xuXG4gIGlmICghcmVjb3JkKSB7XG4gICAgY29uc3QgeyBkb25lIH0gPSBhd2FpdCBzZXR1cChjdHgsIHsgc2lnbmFsIH0pO1xuICAgIHJldHVybiBhd2FpdCBkb25lO1xuICB9XG5cbiAgaWYgKCFyZWNvcmQuc2V0dXA/LmNvbXBsZXRlZEF0KSB7XG4gICAgYXdhaXQgcG9sbFVudGlsUmVhZHkoY3R4LCBzaWduYWwpO1xuICAgIHJlY29yZCA9IGF3YWl0IGFnZW50LnN0b3JhZ2Uuc2FuZGJveC5nZXQoc2FuZGJveElkKTtcbiAgfVxuXG4gIGlmICghcmVjb3JkKSB7XG4gICAgdGhyb3cgbmV3IFNhbmRib3hOb3RGb3VuZEVycm9yKHsgaWQ6IHNhbmRib3hJZCB9KTtcbiAgfVxuXG4gIGNvbnN0IHsgYmluZGluZyB9ID0gYXdhaXQgcmVzb2x2ZUJpbmRpbmcoXG4gICAgY3R4LmFnZW50Lm9wdGlvbnMuc2FuZGJveCxcbiAgICByZWNvcmQuc2V0dXAuYmluZGluZ1xuICApO1xuXG4gIGFnZW50LnN0b3JhZ2Uuc2FuZGJveFxuICAgIC51cGRhdGUoc2FuZGJveElkLCB7IGxhc3RBY3RpdmVBdDogRGF0ZS5ub3coKSB9KVxuICAgIC5jYXRjaCgoZXJyb3IpID0+IHtcbiAgICAgIGxvZy5lcnJvcihcIkZhaWxlZCB0byB1cGRhdGUgc2FuZGJveCBsYXN0QWN0aXZlQXRcIiwgeyBlcnJvciB9KTtcbiAgICB9KTtcblxuICBjb25zdCBpbnN0YW5jZSA9IGF3YWl0IGJpbmRpbmcuY29ubmVjdCh7XG4gICAgYWdlbnQsXG4gICAgbWV0YWRhdGE6IHJlY29yZC5zZXR1cC5tZXRhZGF0YSxcbiAgICBjd2Q6IHJlY29yZC5zZXR1cC5jd2QsXG4gICAgc2lnbmFsLFxuICAgIGxvZyxcbiAgfSk7XG4gIGN0eC5fY3dkID0gaW5zdGFuY2UuY3dkO1xuICByZXR1cm4gaW5zdGFuY2U7XG59XG5cbmFzeW5jIGZ1bmN0aW9uIHBvbGxVbnRpbFJlYWR5KFxuICBjdHg6IFNhbmRib3hDb250ZXh0LFxuICBzaWduYWw/OiBBYm9ydFNpZ25hbFxuKTogUHJvbWlzZTx2b2lkPiB7XG4gIGNvbnN0IGlkID0gY3R4LnNhbmRib3hJZCE7XG4gIGNvbnN0IGRlYWRsaW5lID0gRGF0ZS5ub3coKSArIFBPTExfVElNRU9VVF9NUztcblxuICB3aGlsZSAoRGF0ZS5ub3coKSA8IGRlYWRsaW5lKSB7XG4gICAgc2lnbmFsPy50aHJvd0lmQWJvcnRlZCgpO1xuXG4gICAgY29uc3QgcmVjb3JkID0gYXdhaXQgY3R4LmFnZW50LnN0b3JhZ2Uuc2FuZGJveC5nZXQoaWQpO1xuICAgIGlmIChyZWNvcmQ/LnNldHVwPy5jb21wbGV0ZWRBdCkge1xuICAgICAgcmV0dXJuO1xuICAgIH1cblxuICAgIGF3YWl0IG5ldyBQcm9taXNlKChyKSA9PiBzZXRUaW1lb3V0KHIsIFBPTExfSU5URVJWQUxfTVMpKTtcbiAgfVxuXG4gIHRocm93IG5ldyBTYW5kYm94RXJyb3Ioe1xuICAgIHJlYXNvbjogYFRpbWVkIG91dCB3YWl0aW5nIGZvciBzYW5kYm94IFwiJHtpZH1cIiBzZXR1cCB0byBjb21wbGV0ZWAsXG4gIH0pO1xufVxuIiwgImltcG9ydCB0eXBlIHsgU2FuZGJveEluc3RhbmNlIH0gZnJvbSBcIi4vYWRhcHRlclwiO1xuaW1wb3J0IHR5cGUgeyBTYW5kYm94Q29udGV4dCB9IGZyb20gXCIuL2hhbmRsZVwiO1xuaW1wb3J0IHsgcmVzb2x2ZSB9IGZyb20gXCIuL3NldHVwXCI7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBleGVjKFxuICBzYW5kYm94Q29udGV4dDogU2FuZGJveENvbnRleHQsXG4gIC4uLm9wdHM6IFBhcmFtZXRlcnM8U2FuZGJveEluc3RhbmNlW1wiZXhlY1wiXT5cbikge1xuICAvLyBub3RlOiAuZXhlYyBjYW4ndCBiZSBhIHN0ZXAgYmVjYXVzZSBpdCByZXR1cm5zIGxvZ3MgYW5kIGEgcmVzdWx0IHByb21pc2Ugd2hpY2ggY2FuJ3QgYmUgc2VyaWFsaXplZFxuICBjb25zdCBzYnggPSBhd2FpdCByZXNvbHZlKHNhbmRib3hDb250ZXh0KTtcbiAgcmV0dXJuIGF3YWl0IHNieC5leGVjKC4uLm9wdHMpO1xufVxuIiwgImltcG9ydCB0eXBlIHsgU2FuZGJveEluc3RhbmNlIH0gZnJvbSBcIi4vYWRhcHRlclwiO1xuaW1wb3J0IHR5cGUgeyBTYW5kYm94Q29udGV4dCB9IGZyb20gXCIuL2hhbmRsZVwiO1xuaW1wb3J0IHsgcmVzb2x2ZSB9IGZyb20gXCIuL3NldHVwXCI7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBnZXREb21haW4oXG4gIHNhbmRib3hDb250ZXh0OiBTYW5kYm94Q29udGV4dCxcbiAgLi4ub3B0czogUGFyYW1ldGVyczxTYW5kYm94SW5zdGFuY2VbXCJnZXREb21haW5cIl0+XG4pIHtcbiAgXCJ1c2Ugc3RlcFwiO1xuICBjb25zdCBzYnggPSBhd2FpdCByZXNvbHZlKHNhbmRib3hDb250ZXh0KTtcbiAgcmV0dXJuIGF3YWl0IHNieC5nZXREb21haW4oLi4ub3B0cyk7XG59XG4iLCAiaW1wb3J0IHR5cGUgeyBTYW5kYm94SW5zdGFuY2UgfSBmcm9tIFwiLi9hZGFwdGVyXCI7XG5pbXBvcnQgdHlwZSB7IFNhbmRib3hDb250ZXh0IH0gZnJvbSBcIi4vaGFuZGxlXCI7XG5pbXBvcnQgeyByZXNvbHZlIH0gZnJvbSBcIi4vc2V0dXBcIjtcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGdldFN0YXR1cyhcbiAgc2FuZGJveENvbnRleHQ6IFNhbmRib3hDb250ZXh0LFxuICAuLi5vcHRzOiBQYXJhbWV0ZXJzPFNhbmRib3hJbnN0YW5jZVtcImdldFN0YXR1c1wiXT5cbikge1xuICBcInVzZSBzdGVwXCI7XG4gIGNvbnN0IHNieCA9IGF3YWl0IHJlc29sdmUoc2FuZGJveENvbnRleHQpO1xuICByZXR1cm4gYXdhaXQgc2J4LmdldFN0YXR1cyguLi5vcHRzKTtcbn1cbiIsICJpbXBvcnQgdHlwZSB7IFNhbmRib3hJbnN0YW5jZSB9IGZyb20gXCIuL2FkYXB0ZXJcIjtcbmltcG9ydCB0eXBlIHsgU2FuZGJveENvbnRleHQgfSBmcm9tIFwiLi9oYW5kbGVcIjtcbmltcG9ydCB7IHJlc29sdmUgfSBmcm9tIFwiLi9zZXR1cFwiO1xuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24ga2lsbChcbiAgc2FuZGJveENvbnRleHQ6IFNhbmRib3hDb250ZXh0LFxuICAuLi5vcHRzOiBQYXJhbWV0ZXJzPFNhbmRib3hJbnN0YW5jZVtcImtpbGxcIl0+XG4pIHtcbiAgXCJ1c2Ugc3RlcFwiO1xuICBjb25zdCBzYnggPSBhd2FpdCByZXNvbHZlKHNhbmRib3hDb250ZXh0KTtcbiAgcmV0dXJuIGF3YWl0IHNieC5raWxsKC4uLm9wdHMpO1xufVxuIiwgImltcG9ydCB0eXBlIHsgU2FuZGJveEluc3RhbmNlIH0gZnJvbSBcIi4vYWRhcHRlclwiO1xuaW1wb3J0IHR5cGUgeyBTYW5kYm94Q29udGV4dCB9IGZyb20gXCIuL2hhbmRsZVwiO1xuaW1wb3J0IHsgcmVzb2x2ZSB9IGZyb20gXCIuL3NldHVwXCI7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiByZWFkRmlsZShcbiAgc2FuZGJveENvbnRleHQ6IFNhbmRib3hDb250ZXh0LFxuICAuLi5vcHRzOiBQYXJhbWV0ZXJzPFNhbmRib3hJbnN0YW5jZVtcInJlYWRGaWxlXCJdPlxuKSB7XG4gIFwidXNlIHN0ZXBcIjtcbiAgY29uc3Qgc2J4ID0gYXdhaXQgcmVzb2x2ZShzYW5kYm94Q29udGV4dCk7XG4gIHJldHVybiBhd2FpdCBzYngucmVhZEZpbGUoLi4ub3B0cyk7XG59XG4iLCAiaW1wb3J0IHR5cGUgeyBTYW5kYm94SW5zdGFuY2UgfSBmcm9tIFwiLi9hZGFwdGVyXCI7XG5pbXBvcnQgdHlwZSB7IFNhbmRib3hDb250ZXh0IH0gZnJvbSBcIi4vaGFuZGxlXCI7XG5pbXBvcnQgeyByZXNvbHZlIH0gZnJvbSBcIi4vc2V0dXBcIjtcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIHNuYXBzaG90KFxuICBzYW5kYm94Q29udGV4dDogU2FuZGJveENvbnRleHQsXG4gIC4uLm9wdHM6IFBhcmFtZXRlcnM8U2FuZGJveEluc3RhbmNlW1wic25hcHNob3RcIl0+XG4pIHtcbiAgXCJ1c2Ugc3RlcFwiO1xuICBjb25zdCBzYnggPSBhd2FpdCByZXNvbHZlKHNhbmRib3hDb250ZXh0KTtcbiAgcmV0dXJuIGF3YWl0IHNieC5zbmFwc2hvdCguLi5vcHRzKTtcbn1cbiIsICJpbXBvcnQgdHlwZSB7IFNhbmRib3hJbnN0YW5jZSB9IGZyb20gXCIuL2FkYXB0ZXJcIjtcbmltcG9ydCB0eXBlIHsgU2FuZGJveENvbnRleHQgfSBmcm9tIFwiLi9oYW5kbGVcIjtcbmltcG9ydCB7IHJlc29sdmUgfSBmcm9tIFwiLi9zZXR1cFwiO1xuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gc3RhcnQoXG4gIHNhbmRib3hDb250ZXh0OiBTYW5kYm94Q29udGV4dCxcbiAgLi4ub3B0czogUGFyYW1ldGVyczxTYW5kYm94SW5zdGFuY2VbXCJzdGFydFwiXT5cbikge1xuICBcInVzZSBzdGVwXCI7XG4gIGNvbnN0IHNieCA9IGF3YWl0IHJlc29sdmUoc2FuZGJveENvbnRleHQpO1xuICByZXR1cm4gYXdhaXQgc2J4LnN0YXJ0KC4uLm9wdHMpO1xufVxuIiwgImltcG9ydCB0eXBlIHsgU2FuZGJveEluc3RhbmNlIH0gZnJvbSBcIi4vYWRhcHRlclwiO1xuaW1wb3J0IHR5cGUgeyBTYW5kYm94Q29udGV4dCB9IGZyb20gXCIuL2hhbmRsZVwiO1xuaW1wb3J0IHsgcmVzb2x2ZSB9IGZyb20gXCIuL3NldHVwXCI7XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBzdG9wKFxuICBzYW5kYm94Q29udGV4dDogU2FuZGJveENvbnRleHQsXG4gIC4uLm9wdHM6IFBhcmFtZXRlcnM8U2FuZGJveEluc3RhbmNlW1wic3RvcFwiXT5cbikge1xuICBcInVzZSBzdGVwXCI7XG4gIGNvbnN0IHNieCA9IGF3YWl0IHJlc29sdmUoc2FuZGJveENvbnRleHQpO1xuICByZXR1cm4gYXdhaXQgc2J4LnN0b3AoLi4ub3B0cyk7XG59XG4iLCAiaW1wb3J0IHR5cGUgeyBTYW5kYm94SW5zdGFuY2UgfSBmcm9tIFwiLi9hZGFwdGVyXCI7XG5pbXBvcnQgdHlwZSB7IFNhbmRib3hDb250ZXh0IH0gZnJvbSBcIi4vaGFuZGxlXCI7XG5pbXBvcnQgeyByZXNvbHZlIH0gZnJvbSBcIi4vc2V0dXBcIjtcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIHVwZGF0ZU5ldHdvcmtQb2xpY3koXG4gIHNhbmRib3hDb250ZXh0OiBTYW5kYm94Q29udGV4dCxcbiAgLi4ub3B0czogUGFyYW1ldGVyczxTYW5kYm94SW5zdGFuY2VbXCJ1cGRhdGVOZXR3b3JrUG9saWN5XCJdPlxuKSB7XG4gIFwidXNlIHN0ZXBcIjtcbiAgY29uc3Qgc2J4ID0gYXdhaXQgcmVzb2x2ZShzYW5kYm94Q29udGV4dCk7XG4gIHJldHVybiBhd2FpdCBzYngudXBkYXRlTmV0d29ya1BvbGljeSguLi5vcHRzKTtcbn1cbiIsICJpbXBvcnQgdHlwZSB7IFNhbmRib3hJbnN0YW5jZSB9IGZyb20gXCIuL2FkYXB0ZXJcIjtcbmltcG9ydCB0eXBlIHsgU2FuZGJveENvbnRleHQgfSBmcm9tIFwiLi9oYW5kbGVcIjtcbmltcG9ydCB7IHJlc29sdmUgfSBmcm9tIFwiLi9zZXR1cFwiO1xuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gd3JpdGVGaWxlcyhcbiAgc2FuZGJveENvbnRleHQ6IFNhbmRib3hDb250ZXh0LFxuICAuLi5vcHRzOiBQYXJhbWV0ZXJzPFNhbmRib3hJbnN0YW5jZVtcIndyaXRlRmlsZXNcIl0+XG4pIHtcbiAgXCJ1c2Ugc3RlcFwiO1xuICBjb25zdCBzYnggPSBhd2FpdCByZXNvbHZlKHNhbmRib3hDb250ZXh0KTtcbiAgcmV0dXJuIGF3YWl0IHNieC53cml0ZUZpbGVzKC4uLm9wdHMpO1xufVxuIiwgImltcG9ydCB0eXBlIHsgVG9vbFNldCB9IGZyb20gXCJhaVwiO1xuaW1wb3J0IHsgdWxpZCB9IGZyb20gXCJ1bGlkXCI7XG5pbXBvcnQgdHlwZSB7IEFueUFnZW50IH0gZnJvbSBcIi4uL2FnZW50L2NsaWVudFwiO1xuaW1wb3J0IHsgZ2V0U2Vzc2lvbkNvbnRleHRzIH0gZnJvbSBcIi4uL2FnZW50L2NsaWVudFwiO1xuaW1wb3J0IHR5cGUgeyBTYW5kYm94QmluZGluZyB9IGZyb20gXCIuLi9zYW5kYm94L2FkYXB0ZXJcIjtcbmltcG9ydCB7IGxhenlTYW5kYm94SGFuZGxlIH0gZnJvbSBcIi4uL3NhbmRib3gvaGFuZGxlXCI7XG5pbXBvcnQgeyBoaXN0b3J5IH0gZnJvbSBcIi4vaGlzdG9yeVwiO1xuaW1wb3J0IHsgaW50ZXJydXB0IH0gZnJvbSBcIi4vaW50ZXJydXB0XCI7XG5pbXBvcnQgeyBzZW5kIH0gZnJvbSBcIi4vc2VuZFwiO1xuaW1wb3J0IHsgc3RyZWFtIH0gZnJvbSBcIi4vc3RyZWFtXCI7XG5pbXBvcnQgeyB1cGRhdGUgfSBmcm9tIFwiLi91cGRhdGVcIjtcbmltcG9ydCB7IHVzYWdlIH0gZnJvbSBcIi4vdXNhZ2VcIjtcblxudHlwZSBTZXNzaW9uT3B0aW9ucyA9IHtcbiAgaWQ/OiBzdHJpbmc7XG59O1xuXG5leHBvcnQgdHlwZSBTZXNzaW9uQ29udGV4dCA9IHtcbiAgYWdlbnQ6IEFueUFnZW50O1xuICBzZXNzaW9uSWQ6IHN0cmluZztcbn07XG5cbmV4cG9ydCBmdW5jdGlvbiBzZXNzaW9uSGFuZGxlPFxuICBUb29scyBleHRlbmRzIFRvb2xTZXQsXG4gIFRDb250ZXh0LFxuICBUU2FuZGJveEJpbmRpbmdzIGV4dGVuZHMgU2FuZGJveEJpbmRpbmcgfCBTYW5kYm94QmluZGluZ1tdID0gU2FuZGJveEJpbmRpbmcsXG4gIFRNZXNzYWdlTWV0YWRhdGEgPSB1bmtub3duLFxuICBUU3RhdGUgZXh0ZW5kcyBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPiA9IFJlY29yZDxzdHJpbmcsIHVua25vd24+LFxuPihhZ2VudDogQW55QWdlbnQsIGlkPzogc3RyaW5nIHwgU2Vzc2lvbk9wdGlvbnMpIHtcbiAgY29uc3Qgc2Vzc2lvbklkID1cbiAgICB0eXBlb2YgaWQgPT09IFwic3RyaW5nXCIgPyBpZCA6IChpZD8uaWQgPz8gYHNlc3Npb25fJHt1bGlkKCl9YCk7XG5cbiAgY29uc3QgY29udGV4dHMgPSBnZXRTZXNzaW9uQ29udGV4dHMoYWdlbnQpO1xuICBsZXQgc2Vzc2lvbkNvbnRleHQgPSBjb250ZXh0cy5nZXQoc2Vzc2lvbklkKTtcbiAgaWYgKCFzZXNzaW9uQ29udGV4dCkge1xuICAgIHNlc3Npb25Db250ZXh0ID0geyBhZ2VudCwgc2Vzc2lvbklkIH07XG4gICAgY29udGV4dHMuc2V0KHNlc3Npb25JZCwgc2Vzc2lvbkNvbnRleHQpO1xuICB9XG5cbiAgcmV0dXJuIHtcbiAgICBpZDogc2Vzc2lvbklkLFxuICAgIHNlbmQ6IChzZW5kPFRvb2xzLCBUQ29udGV4dCwgVE1lc3NhZ2VNZXRhZGF0YSwgVFN0YXRlPikuYmluZChcbiAgICAgIG51bGwsXG4gICAgICBzZXNzaW9uQ29udGV4dFxuICAgICksXG4gICAgc3RyZWFtOiBzdHJlYW0uYmluZChudWxsLCBzZXNzaW9uQ29udGV4dCksXG4gICAgaGlzdG9yeTogKGhpc3Rvcnk8VG9vbHM+KS5iaW5kKG51bGwsIHNlc3Npb25Db250ZXh0KSxcbiAgICBpbnRlcnJ1cHQ6IGludGVycnVwdC5iaW5kKG51bGwsIHNlc3Npb25Db250ZXh0KSxcbiAgICB1c2FnZTogdXNhZ2UuYmluZChudWxsLCBzZXNzaW9uQ29udGV4dCksXG4gICAgdXBkYXRlOiAodXBkYXRlPFRTdGF0ZT4pLmJpbmQobnVsbCwgc2Vzc2lvbkNvbnRleHQpLFxuICAgIHNhbmRib3g6IGxhenlTYW5kYm94SGFuZGxlPFRTYW5kYm94QmluZGluZ3M+KGFnZW50LCBzZXNzaW9uSWQpLFxuICB9O1xufVxuIiwgImV4cG9ydCB0eXBlIFN0ZXBVc2FnZSA9IHtcbiAgc3RlcEluZGV4OiBudW1iZXI7XG4gIG1vZGVsOiBzdHJpbmc7XG4gIGlucHV0VG9rZW5zOiBudW1iZXI7XG4gIG91dHB1dFRva2VuczogbnVtYmVyO1xuICB0b3RhbFRva2VuczogbnVtYmVyO1xuICBjYWNoZVJlYWRUb2tlbnM6IG51bWJlcjtcbiAgY2FjaGVXcml0ZVRva2VuczogbnVtYmVyO1xuICByZWFzb25pbmdUb2tlbnM6IG51bWJlcjtcbn07XG5cbmV4cG9ydCB0eXBlIFVzYWdlU3VtbWFyeSA9IHtcbiAgbW9kZWw6IHN0cmluZztcbiAgaW5wdXRUb2tlbnM6IG51bWJlcjtcbiAgb3V0cHV0VG9rZW5zOiBudW1iZXI7XG4gIHRvdGFsVG9rZW5zOiBudW1iZXI7XG4gIGNhY2hlUmVhZFRva2VuczogbnVtYmVyO1xuICBjYWNoZVdyaXRlVG9rZW5zOiBudW1iZXI7XG4gIHJlYXNvbmluZ1Rva2VuczogbnVtYmVyO1xuICBzdGVwQ291bnQ6IG51bWJlcjtcbn07XG5cbmV4cG9ydCB0eXBlIFNlc3Npb25Vc2FnZSA9IHtcbiAgdG90YWw6IFVzYWdlU3VtbWFyeSAmIHsgbWVzc2FnZUNvdW50OiBudW1iZXIgfTtcbiAgYnlNZXNzYWdlSWQ6IFJlY29yZDxzdHJpbmcsIFVzYWdlU3VtbWFyeSB8IG51bGw+O1xufTtcblxuZXhwb3J0IGZ1bmN0aW9uIGNvbXB1dGVTZXNzaW9uVXNhZ2UoXG4gIG1lc3NhZ2VzOiBBcnJheTx7XG4gICAgaWQ6IHN0cmluZztcbiAgICB1c2FnZTogeyBzdGVwczogU3RlcFVzYWdlW107IHN1bW1hcnk6IFVzYWdlU3VtbWFyeSB9IHwgbnVsbDtcbiAgfT5cbik6IFNlc3Npb25Vc2FnZSB7XG4gIGNvbnN0IGJ5TWVzc2FnZUlkOiBSZWNvcmQ8c3RyaW5nLCBVc2FnZVN1bW1hcnkgfCBudWxsPiA9IHt9O1xuICBmb3IgKGNvbnN0IG0gb2YgbWVzc2FnZXMpIHtcbiAgICBieU1lc3NhZ2VJZFttLmlkXSA9IG0udXNhZ2U/LnN1bW1hcnkgPz8gbnVsbDtcbiAgfVxuXG4gIGNvbnN0IHN1bW1hcmllcyA9IG1lc3NhZ2VzXG4gICAgLm1hcCgobSkgPT4gbS51c2FnZT8uc3VtbWFyeSlcbiAgICAuZmlsdGVyKChzKTogcyBpcyBVc2FnZVN1bW1hcnkgPT4gcyAhPT0gdW5kZWZpbmVkKTtcblxuICBjb25zdCB0b3RhbCA9IHtcbiAgICBtb2RlbDogc3VtbWFyaWVzWzBdPy5tb2RlbCA/PyBcInVua25vd25cIixcbiAgICBpbnB1dFRva2Vuczogc3VtbWFyaWVzLnJlZHVjZSgoYWNjLCBzKSA9PiBhY2MgKyBzLmlucHV0VG9rZW5zLCAwKSxcbiAgICBvdXRwdXRUb2tlbnM6IHN1bW1hcmllcy5yZWR1Y2UoKGFjYywgcykgPT4gYWNjICsgcy5vdXRwdXRUb2tlbnMsIDApLFxuICAgIHRvdGFsVG9rZW5zOiBzdW1tYXJpZXMucmVkdWNlKChhY2MsIHMpID0+IGFjYyArIHMudG90YWxUb2tlbnMsIDApLFxuICAgIGNhY2hlUmVhZFRva2Vuczogc3VtbWFyaWVzLnJlZHVjZSgoYWNjLCBzKSA9PiBhY2MgKyBzLmNhY2hlUmVhZFRva2VucywgMCksXG4gICAgY2FjaGVXcml0ZVRva2Vuczogc3VtbWFyaWVzLnJlZHVjZSgoYWNjLCBzKSA9PiBhY2MgKyBzLmNhY2hlV3JpdGVUb2tlbnMsIDApLFxuICAgIHJlYXNvbmluZ1Rva2Vuczogc3VtbWFyaWVzLnJlZHVjZSgoYWNjLCBzKSA9PiBhY2MgKyBzLnJlYXNvbmluZ1Rva2VucywgMCksXG4gICAgc3RlcENvdW50OiBzdW1tYXJpZXMucmVkdWNlKChhY2MsIHMpID0+IGFjYyArIHMuc3RlcENvdW50LCAwKSxcbiAgICBtZXNzYWdlQ291bnQ6IHN1bW1hcmllcy5sZW5ndGgsXG4gIH07XG5cbiAgcmV0dXJuIHsgdG90YWwsIGJ5TWVzc2FnZUlkIH07XG59XG4iLCAiaW1wb3J0IHR5cGUgeyBUb29sU2V0IH0gZnJvbSBcImFpXCI7XG5pbXBvcnQgeyBhc3NlbWJsZVVJTWVzc2FnZXMgfSBmcm9tIFwiLi4vdXRpbHMvdWlcIjtcbmltcG9ydCB7IGNvbXB1dGVTZXNzaW9uVXNhZ2UsIHR5cGUgU2Vzc2lvblVzYWdlIH0gZnJvbSBcIi4uL3V0aWxzL3VzYWdlXCI7XG5pbXBvcnQgdHlwZSB7IFNlc3Npb25Db250ZXh0IH0gZnJvbSBcIi4vaGFuZGxlXCI7XG5cbmV4cG9ydCB0eXBlIEhpc3RvcnlSZXN1bHQgPSB7XG4gIG1lc3NhZ2VzOiBSZXR1cm5UeXBlPHR5cGVvZiBhc3NlbWJsZVVJTWVzc2FnZXM+O1xuICBzdHJlYW1pbmdNZXNzYWdlSWQ6IHN0cmluZyB8IG51bGw7XG4gIHVzYWdlOiBTZXNzaW9uVXNhZ2U7XG59O1xuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gaGlzdG9yeTxUb29scyBleHRlbmRzIFRvb2xTZXQ+KFxuICBzZXNzaW9uQ29udGV4dDogU2Vzc2lvbkNvbnRleHRcbikge1xuICBcInVzZSBzdGVwXCI7XG4gIGNvbnN0IFtzZXNzaW9uLCBtZXNzYWdlcywgcGFydHNdID0gYXdhaXQgUHJvbWlzZS5hbGwoW1xuICAgIHNlc3Npb25Db250ZXh0LmFnZW50LnN0b3JhZ2Uuc2Vzc2lvbi5nZXQoc2Vzc2lvbkNvbnRleHQuc2Vzc2lvbklkKSxcbiAgICBzZXNzaW9uQ29udGV4dC5hZ2VudC5zdG9yYWdlLm1lc3NhZ2UubGlzdEJ5U2Vzc2lvbihcbiAgICAgIHNlc3Npb25Db250ZXh0LnNlc3Npb25JZFxuICAgICksXG4gICAgc2Vzc2lvbkNvbnRleHQuYWdlbnQuc3RvcmFnZS5wYXJ0Lmxpc3RCeVNlc3Npb24oc2Vzc2lvbkNvbnRleHQuc2Vzc2lvbklkKSxcbiAgXSk7XG5cbiAgY29uc3QgbGFzdEFzc2lzdGFudCA9XG4gICAgbWVzc2FnZXMuZmluZChcbiAgICAgIChtKSA9PiBtLmlkID09PSBzZXNzaW9uPy5sYXN0TWVzc2FnZUlkICYmIG0ucm9sZSA9PT0gXCJhc3Npc3RhbnRcIlxuICAgICkgPz8gbWVzc2FnZXMuZmlsdGVyKChtKSA9PiBtLnJvbGUgPT09IFwiYXNzaXN0YW50XCIpLmF0KC0xKTtcblxuICBjb25zdCBpc1N0cmVhbWluZyA9XG4gICAgbGFzdEFzc2lzdGFudD8uY29tcGxldGVkQXQgPT09IG51bGwgJiZcbiAgICBsYXN0QXNzaXN0YW50Py5pbnRlcnJ1cHRlZEF0ID09PSBudWxsO1xuICBjb25zdCBzdHJlYW1pbmdNZXNzYWdlSWQgPSBpc1N0cmVhbWluZyA/IGxhc3RBc3Npc3RhbnQuaWQgOiBudWxsO1xuICBjb25zdCB2aXNpYmxlVW50aWwgPSBsYXN0QXNzaXN0YW50Py5jcmVhdGVkQXQ7XG5cbiAgY29uc3QgYXNzZW1ibGVkID0gYXNzZW1ibGVVSU1lc3NhZ2VzPFRvb2xzPih7XG4gICAgbWVzc2FnZXMsXG4gICAgcGFydHMsXG4gICAgaW5jbHVkZVF1ZXVlZDogdmlzaWJsZVVudGlsICE9PSB1bmRlZmluZWQsXG4gICAgdW50aWw6IHZpc2libGVVbnRpbCxcbiAgfSk7XG5cbiAgY29uc3QgdXNhZ2UgPSBjb21wdXRlU2Vzc2lvblVzYWdlKG1lc3NhZ2VzKTtcblxuICByZXR1cm4geyBtZXNzYWdlczogYXNzZW1ibGVkLCBzdHJlYW1pbmdNZXNzYWdlSWQsIHVzYWdlIH07XG59XG4iLCAiaW1wb3J0IHR5cGUgeyBVSU1lc3NhZ2VDaHVuayB9IGZyb20gXCJhaVwiO1xuaW1wb3J0ICogYXMgZXJyb3JlIGZyb20gXCJlcnJvcmVcIjtcbmltcG9ydCB7IFNlc3Npb25Ob3RGb3VuZEVycm9yIH0gZnJvbSBcIi4uL2Vycm9yc1wiO1xuaW1wb3J0IHR5cGUgeyBTZXNzaW9uQ29udGV4dCB9IGZyb20gXCIuL2hhbmRsZVwiO1xuaW1wb3J0IHsgZ2V0QWN0aXZlUnVucywgcmVwbGF5QWN0aXZlUnVuIH0gZnJvbSBcIi4vc2VuZFwiO1xuXG5leHBvcnQgdHlwZSBTdHJlYW1PcHRpb25zID0ge1xuICBtZXNzYWdlSWQ/OiBzdHJpbmc7XG4gIHN0YXJ0SW5kZXg/OiBudW1iZXI7XG59O1xuXG5leHBvcnQgdHlwZSBXb3JrZmxvd1J1bkxpa2UgPSB7XG4gIGdldFJlYWRhYmxlKG9wdGlvbnM/OiB7IHN0YXJ0SW5kZXg/OiBudW1iZXIgfSk6IFJlYWRhYmxlU3RyZWFtO1xufTtcblxuZnVuY3Rpb24gaXNXb3JrZmxvd1J1bihhcmc6IHVua25vd24pOiBhcmcgaXMgV29ya2Zsb3dSdW5MaWtlIHtcbiAgcmV0dXJuIChcbiAgICB0eXBlb2YgYXJnID09PSBcIm9iamVjdFwiICYmXG4gICAgYXJnICE9PSBudWxsICYmXG4gICAgXCJnZXRSZWFkYWJsZVwiIGluIGFyZyAmJlxuICAgIHR5cGVvZiAoYXJnIGFzIFdvcmtmbG93UnVuTGlrZSkuZ2V0UmVhZGFibGUgPT09IFwiZnVuY3Rpb25cIlxuICApO1xufVxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gc3RyZWFtKFxuICBzZXNzaW9uQ29udGV4dDogU2Vzc2lvbkNvbnRleHQsXG4gIHJ1bk9yT3B0cz86IFdvcmtmbG93UnVuTGlrZSB8IFN0cmVhbU9wdGlvbnNcbik6IFByb21pc2U8UmVhZGFibGVTdHJlYW08VUlNZXNzYWdlQ2h1bms+PiB7XG4gIGlmIChpc1dvcmtmbG93UnVuKHJ1bk9yT3B0cykpIHtcbiAgICBjb25zdCByZWFkZXIgPSBydW5Pck9wdHMuZ2V0UmVhZGFibGUoKS5nZXRSZWFkZXIoKTtcbiAgICAvLyBhd2FpdCBmb3IgdGhlIGZpcnN0IGNodW5rIHRvIGJlIHJlYWR5IGJlZm9yZSBmaW5pc2hpbmcgdGhlIHdvcmtmbG93IHRvIGF2b2lkIHJhY2UgY29uZGl0aW9uc1xuICAgIGF3YWl0IHJlYWRlci5yZWFkKCkuY2F0Y2goKCkgPT4ge1xuICAgICAgdGhyb3cgbmV3IFN0cmVhbUVycm9yKHtcbiAgICAgICAgaWQ6IHNlc3Npb25Db250ZXh0LnNlc3Npb25JZCxcbiAgICAgICAgcmVhc29uOiBcInN0cmVhbSgpIGZhaWxlZCB0byByZWFkIGZpcnN0IGNodW5rIGZyb20gd29ya2Zsb3cgcnVuXCIsXG4gICAgICB9KTtcbiAgICB9KTtcbiAgICByZWFkZXIuY2FuY2VsKCk7XG5cbiAgICByZXR1cm4gcnVuT3JPcHRzLmdldFJlYWRhYmxlKCk7XG4gIH1cblxuICBjb25zdCBvcHRzID0gcnVuT3JPcHRzO1xuICBjb25zdCBzZXNzaW9uID0gYXdhaXQgc2Vzc2lvbkNvbnRleHQuYWdlbnQuc3RvcmFnZS5zZXNzaW9uLmdldChcbiAgICBzZXNzaW9uQ29udGV4dC5zZXNzaW9uSWRcbiAgKTtcbiAgaWYgKCFzZXNzaW9uKSB7XG4gICAgdGhyb3cgbmV3IFNlc3Npb25Ob3RGb3VuZEVycm9yKHsgaWQ6IHNlc3Npb25Db250ZXh0LnNlc3Npb25JZCB9KTtcbiAgfVxuICBjb25zdCBtZXNzYWdlSWQgPSBvcHRzPy5tZXNzYWdlSWQgPz8gc2Vzc2lvbi5sYXN0TWVzc2FnZUlkO1xuICBpZiAoIW1lc3NhZ2VJZCkge1xuICAgIHRocm93IG5ldyBTdHJlYW1FcnJvcih7XG4gICAgICBpZDogc2Vzc2lvbkNvbnRleHQuc2Vzc2lvbklkLFxuICAgICAgcmVhc29uOiBcIk5vIG1lc3NhZ2UgdG8gc3RyZWFtXCIsXG4gICAgfSk7XG4gIH1cblxuICBjb25zdCBhY3RpdmVSdW5zID0gZ2V0QWN0aXZlUnVucygpO1xuICBjb25zdCBhY3RpdmUgPSBhY3RpdmVSdW5zLmdldChtZXNzYWdlSWQpO1xuICBpZiAoIWFjdGl2ZSkge1xuICAgIGNvbnN0IG1lc3NhZ2UgPSBhd2FpdCBzZXNzaW9uQ29udGV4dC5hZ2VudC5zdG9yYWdlLm1lc3NhZ2UuZ2V0KG1lc3NhZ2VJZCk7XG4gICAgaWYgKG1lc3NhZ2U/LndvcmtmbG93UnVuSWQpIHtcbiAgICAgIGNvbnN0IHJlYWRhYmxlID0gYXdhaXQgaW1wb3J0KFwid29ya2Zsb3cvYXBpXCIpXG4gICAgICAgIC50aGVuKCh7IGdldFJ1biB9KSA9PiBnZXRSdW4obWVzc2FnZS53b3JrZmxvd1J1bklkISkuZ2V0UmVhZGFibGUoKSlcbiAgICAgICAgLmNhdGNoKCgpID0+IHVuZGVmaW5lZCk7XG4gICAgICBpZiAocmVhZGFibGUpIHtcbiAgICAgICAgcmV0dXJuIHJlYWRhYmxlO1xuICAgICAgfVxuICAgIH1cblxuICAgIHRocm93IG5ldyBTdHJlYW1FcnJvcih7XG4gICAgICBpZDogc2Vzc2lvbkNvbnRleHQuc2Vzc2lvbklkLFxuICAgICAgcmVhc29uOiBgTm8gYWN0aXZlIHJ1biBmb3IgbWVzc2FnZSAke21lc3NhZ2VJZH1gLFxuICAgIH0pO1xuICB9XG5cbiAgcmV0dXJuIHJlcGxheUFjdGl2ZVJ1bihhY3RpdmUsIG9wdHM/LnN0YXJ0SW5kZXgpO1xufVxuXG4vKiAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG4gKiBFcnJvcnNcbiAqIC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tKi9cblxuZXhwb3J0IGNsYXNzIFN0cmVhbUVycm9yIGV4dGVuZHMgZXJyb3JlLmNyZWF0ZVRhZ2dlZEVycm9yKHtcbiAgbmFtZTogXCJTdHJlYW1FcnJvclwiLFxuICBtZXNzYWdlOiBcIlN0cmVhbSBmYWlsZWQgZm9yIHNlc3Npb24gJGlkOiAkcmVhc29uXCIsXG59KSB7fVxuIiwgImltcG9ydCB0eXBlIHsgU2Vzc2lvbiB9IGZyb20gXCIuLi9zdG9yYWdlL3R5cGVzXCI7XG5pbXBvcnQgdHlwZSB7IFNlc3Npb25Db250ZXh0IH0gZnJvbSBcIi4vaGFuZGxlXCI7XG5cbmV4cG9ydCB0eXBlIFVwZGF0ZU9wdGlvbnM8XG4gIFRTdGF0ZSBleHRlbmRzIFJlY29yZDxzdHJpbmcsIHVua25vd24+ID0gUmVjb3JkPHN0cmluZywgdW5rbm93bj4sXG4+ID0gUGFydGlhbDxPbWl0PFNlc3Npb24sIFwiaWRcIiB8IFwiY3JlYXRlZEF0XCIgfCBcInVwZGF0ZWRBdFwiIHwgXCJzdGF0ZVwiPj4gJiB7XG4gIHN0YXRlPzogVFN0YXRlO1xufTtcblxuZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIHVwZGF0ZTxcbiAgVFN0YXRlIGV4dGVuZHMgUmVjb3JkPHN0cmluZywgdW5rbm93bj4gPSBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPixcbj4oXG4gIHNlc3Npb25Db250ZXh0OiBTZXNzaW9uQ29udGV4dCxcbiAgb3B0czogVXBkYXRlT3B0aW9uczxUU3RhdGU+XG4pIHtcbiAgXCJ1c2Ugc3RlcFwiO1xuICBjb25zdCB1cGRhdGVzID0gT2JqZWN0LmZyb21FbnRyaWVzKFxuICAgIE9iamVjdC5lbnRyaWVzKG9wdHMpLmZpbHRlcigoWywgdmFsdWVdKSA9PiB2YWx1ZSAhPT0gdW5kZWZpbmVkKVxuICApO1xuXG4gIGlmIChPYmplY3Qua2V5cyh1cGRhdGVzKS5sZW5ndGggPT09IDApIHtcbiAgICByZXR1cm47XG4gIH1cblxuICBhd2FpdCBzZXNzaW9uQ29udGV4dC5hZ2VudC5zdG9yYWdlLnNlc3Npb24udXBkYXRlKHNlc3Npb25Db250ZXh0LnNlc3Npb25JZCwge1xuICAgIC4uLnVwZGF0ZXMsXG4gICAgdXBkYXRlZEF0OiBEYXRlLm5vdygpLFxuICB9KTtcbn1cbiIsICJpbXBvcnQgeyBjb21wdXRlU2Vzc2lvblVzYWdlLCB0eXBlIFNlc3Npb25Vc2FnZSB9IGZyb20gXCIuLi91dGlscy91c2FnZVwiO1xuaW1wb3J0IHR5cGUgeyBTZXNzaW9uQ29udGV4dCB9IGZyb20gXCIuL2hhbmRsZVwiO1xuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gdXNhZ2UoXG4gIHNlc3Npb25Db250ZXh0OiBTZXNzaW9uQ29udGV4dFxuKTogUHJvbWlzZTxTZXNzaW9uVXNhZ2U+IHtcbiAgXCJ1c2Ugc3RlcFwiO1xuICBjb25zdCBtZXNzYWdlcyA9IGF3YWl0IHNlc3Npb25Db250ZXh0LmFnZW50LnN0b3JhZ2UubWVzc2FnZS5saXN0QnlTZXNzaW9uKFxuICAgIHNlc3Npb25Db250ZXh0LnNlc3Npb25JZFxuICApO1xuICByZXR1cm4gY29tcHV0ZVNlc3Np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SUFBSSxPQUFNLEtBQUssTUFBTSxHQUFHLEtBQUssT0FBTyxDQUFDO0FBQUEsRUFDOUM7QUFRRjtBQUVPLFNBQVMsTUFPZCxNQUNBLFNBT0E7QUFDQSxRQUFNLFdBQVcsSUFBSSxNQU1uQixNQUFNLE9BQU87QUFDZixnQkFBYyxNQUFNLFFBQVE7QUFDNUIsU0FBTztBQUNUOyIsCiAgIm5hbWVzIjogWyJ0b29sIiwgInVsaWQiLCAieiIsICJlcnJvcmUiLCAiYWdlbnQiLCAibG9nIiwgInJlYWRGaWxlIiwgInJlc29sdmUiLCAibG9nIiwgImFnZW50IiwgInJlc29sdmUiLCAibG9nIiwgInNhbmRib3giLCAic2FuZGJveFJlY29yZCIsICJzdHJlYW0iLCAidXNhZ2UiLCAiYWdlbnQiLCAibG9nIiwgInVzYWdlIiwgIm5vdyIsICJpZCIsICJ0b29sIiwgImV4ZWMiLCAidG9vbCIsICJhZ2VudCIsICJ0b29sIiwgInoiLCAiYWdlbnQiLCAidWxpZCIsICJzdGFydCIsICJzdGF0ZSIsICJlcnJvcmUiLCAieiIsICJhZ2VudCIsICJ1bGlkIiwgImFnZW50IiwgImxvZyIsICJyZXNvbHZlIiwgImFnZW50IiwgInVsaWQiLCAidWxpZCIsICJ1c2FnZSIsICJlcnJvcmUiLCAiYWdlbnQiLCAidWxpZCIsICJhZ2VudCIsICJsb2ciXQp9Cg==