cdk-local-lambda 0.0.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.
Files changed (53) hide show
  1. package/LICENSE +202 -0
  2. package/README.md +94 -0
  3. package/lib/aspect/docker-function-hook.d.ts +18 -0
  4. package/lib/aspect/docker-function-hook.js +31 -0
  5. package/lib/aspect/live-lambda-aspect.d.ts +85 -0
  6. package/lib/aspect/live-lambda-aspect.js +277 -0
  7. package/lib/aspect/live-lambda-bootstrap.d.ts +17 -0
  8. package/lib/aspect/live-lambda-bootstrap.js +260 -0
  9. package/lib/aspect/nodejs-function-hook.d.ts +20 -0
  10. package/lib/aspect/nodejs-function-hook.js +27 -0
  11. package/lib/bootstrap-stack/bootstrap-stack.d.ts +60 -0
  12. package/lib/bootstrap-stack/bootstrap-stack.js +338 -0
  13. package/lib/cli/appsync/client.d.ts +30 -0
  14. package/lib/cli/appsync/client.js +227 -0
  15. package/lib/cli/cdk-app.d.ts +7 -0
  16. package/lib/cli/cdk-app.js +25 -0
  17. package/lib/cli/commands/bootstrap.d.ts +9 -0
  18. package/lib/cli/commands/bootstrap.js +50 -0
  19. package/lib/cli/commands/local.d.ts +40 -0
  20. package/lib/cli/commands/local.js +1172 -0
  21. package/lib/cli/daemon.d.ts +22 -0
  22. package/lib/cli/daemon.js +18 -0
  23. package/lib/cli/docker/container.d.ts +116 -0
  24. package/lib/cli/docker/container.js +414 -0
  25. package/lib/cli/docker/types.d.ts +71 -0
  26. package/lib/cli/docker/types.js +5 -0
  27. package/lib/cli/docker/watcher.d.ts +44 -0
  28. package/lib/cli/docker/watcher.js +115 -0
  29. package/lib/cli/index.d.ts +9 -0
  30. package/lib/cli/index.js +26 -0
  31. package/lib/cli/runtime-api/server.d.ts +102 -0
  32. package/lib/cli/runtime-api/server.js +396 -0
  33. package/lib/cli/runtime-api/types.d.ts +149 -0
  34. package/lib/cli/runtime-api/types.js +10 -0
  35. package/lib/cli/runtime-wrapper/nodejs-runtime.d.ts +16 -0
  36. package/lib/cli/runtime-wrapper/nodejs-runtime.js +248 -0
  37. package/lib/cli/watcher/file-watcher.d.ts +32 -0
  38. package/lib/cli/watcher/file-watcher.js +57 -0
  39. package/lib/functions/bridge/appsync-client.d.ts +73 -0
  40. package/lib/functions/bridge/appsync-client.js +345 -0
  41. package/lib/functions/bridge/handler.d.ts +17 -0
  42. package/lib/functions/bridge/handler.js +79 -0
  43. package/lib/functions/bridge/ssm-config.d.ts +19 -0
  44. package/lib/functions/bridge/ssm-config.js +45 -0
  45. package/lib/functions/bridge-builder/handler.d.ts +12 -0
  46. package/lib/functions/bridge-builder/handler.js +181 -0
  47. package/lib/functions/bridge-docker/runtime.d.ts +9 -0
  48. package/lib/functions/bridge-docker/runtime.js +127 -0
  49. package/lib/index.d.ts +24 -0
  50. package/lib/index.js +28 -0
  51. package/lib/shared/types.d.ts +102 -0
  52. package/lib/shared/types.js +125 -0
  53. package/package.json +111 -0
@@ -0,0 +1,1172 @@
1
+ /**
2
+ * Local command for running Lambda functions locally using Docker.
3
+ *
4
+ * This command:
5
+ * 1. Starts CDK watch with CDK_LIVE=true and hotswap
6
+ * 2. Discovers Lambda functions with live-lambda:handler tag
7
+ * 3. Connects to AppSync Events
8
+ * 4. Subscribes to invocation channels
9
+ * 5. For each function, maintains ONE Docker container that handles all invocations
10
+ * 6. Sends responses back via AppSync
11
+ * 7. Re-discovers functions after each CDK deploy
12
+ */
13
+ import { spawn } from "node:child_process";
14
+ import * as path from "node:path";
15
+ import { fileURLToPath } from "node:url";
16
+ import { LambdaClient, ListFunctionsCommand, ListTagsCommand, } from "@aws-sdk/client-lambda";
17
+ import { GetParameterCommand, SSMClient } from "@aws-sdk/client-ssm";
18
+ import { Command, Options } from "@effect/cli";
19
+ import { Command as PlatformCommand, } from "@effect/platform";
20
+ import { BunContext } from "@effect/platform-bun";
21
+ import { Duration, Effect, Exit, Fiber, Logger, LogLevel, Queue, Ref, Schedule, Scope, Stream, } from "effect";
22
+ import { BOOTSTRAP_STACK_NAME, BOOTSTRAP_VERSION, buildChannelName, LIVE_LAMBDA_DOCKER_TAG, LIVE_LAMBDA_TAG, SSM_BASE_PATH, } from "../../shared/types.js";
23
+ import { makeAppSyncClient } from "../appsync/client.js";
24
+ import { buildExtensionWrapperCommand, Docker, DockerLive, makeLambdaContainerConfig, } from "../docker/container.js";
25
+ import { watchDockerContexts, } from "../docker/watcher.js";
26
+ import { notifyExtensionsInvoke, queueInvocation, startRuntimeApiServer, waitForResponse, } from "../runtime-api/server.js";
27
+ /**
28
+ * Default idle timeout before proactively stopping containers.
29
+ * Set slightly below the poll timeout (240s) to stop container before
30
+ * the Lambda RIC gets a 503 error.
31
+ */
32
+ const DEFAULT_IDLE_TIMEOUT_MS = 230_000; // 230 seconds (~4 minutes)
33
+ /**
34
+ * Check if bootstrap stack version matches the expected version.
35
+ * Returns true if version matches, false if missing or mismatched.
36
+ */
37
+ const checkBootstrapVersion = (qualifier) => Effect.gen(function* () {
38
+ const ssmClient = new SSMClient({});
39
+ const basePath = `${SSM_BASE_PATH}/${qualifier}`;
40
+ const result = yield* Effect.tryPromise({
41
+ try: async () => {
42
+ const response = await ssmClient.send(new GetParameterCommand({ Name: `${basePath}/version` }));
43
+ return response.Parameter?.Value;
44
+ },
45
+ catch: () => null,
46
+ }).pipe(Effect.catchAll(() => Effect.succeed(null)));
47
+ if (result === null) {
48
+ yield* Effect.logInfo("Bootstrap stack version parameter not found");
49
+ return false;
50
+ }
51
+ if (result !== BOOTSTRAP_VERSION) {
52
+ yield* Effect.logInfo(`Bootstrap stack version mismatch: found ${result}, expected ${BOOTSTRAP_VERSION}`);
53
+ return false;
54
+ }
55
+ return true;
56
+ });
57
+ /**
58
+ * Run the bootstrap stack deployment.
59
+ */
60
+ const runBootstrap = (options) => Effect.gen(function* () {
61
+ yield* Effect.logInfo("Running bootstrap stack deployment...");
62
+ // Resolve the CDK app path relative to this module
63
+ const __filename = fileURLToPath(import.meta.url);
64
+ const __dirname = path.dirname(__filename);
65
+ const cdkAppPath = path.resolve(__dirname, "..", "cdk-app.js");
66
+ const args = [
67
+ "cdk",
68
+ "deploy",
69
+ BOOTSTRAP_STACK_NAME,
70
+ "--require-approval",
71
+ "never",
72
+ "--app",
73
+ `bun ${cdkAppPath}`,
74
+ ];
75
+ if (options.profile) {
76
+ args.push("--profile", options.profile);
77
+ }
78
+ const env = {
79
+ ...process.env,
80
+ };
81
+ if (options.region) {
82
+ env.AWS_REGION = options.region;
83
+ env.CDK_DEFAULT_REGION = options.region;
84
+ }
85
+ yield* Effect.logInfo(`Running: npx ${args.join(" ")}`);
86
+ const command = PlatformCommand.make("npx", ...args).pipe(PlatformCommand.env(env), PlatformCommand.stdin("inherit"));
87
+ const proc = yield* PlatformCommand.start(command);
88
+ // Stream output to console
89
+ yield* Stream.merge(proc.stdout, proc.stderr).pipe(Stream.decodeText(), Stream.splitLines, Stream.runForEach((line) => Effect.sync(() => {
90
+ process.stdout.write(`${line}\n`);
91
+ })));
92
+ const exitCode = yield* proc.exitCode;
93
+ if (exitCode !== 0) {
94
+ return yield* Effect.fail(new Error(`Bootstrap deployment failed with exit code ${exitCode}`));
95
+ }
96
+ yield* Effect.logInfo("Bootstrap stack deployed successfully!");
97
+ }).pipe(Effect.scoped);
98
+ /**
99
+ * Ensure bootstrap stack is deployed with correct version.
100
+ * Automatically deploys if missing or outdated.
101
+ */
102
+ const ensureBootstrap = (options) => Effect.gen(function* () {
103
+ yield* Effect.logDebug("[Local] Checking bootstrap stack version...");
104
+ const versionOk = yield* checkBootstrapVersion(options.qualifier);
105
+ if (!versionOk) {
106
+ yield* Effect.logInfo("[Local] Bootstrap stack needs to be deployed or updated.");
107
+ yield* runBootstrap({ profile: options.profile, region: options.region });
108
+ }
109
+ else {
110
+ yield* Effect.logDebug("[Local] Bootstrap stack version OK.");
111
+ }
112
+ });
113
+ /**
114
+ * Read AppSync endpoints from SSM.
115
+ */
116
+ const getAppSyncEndpoints = (qualifier) => Effect.gen(function* () {
117
+ const ssmClient = new SSMClient({});
118
+ const basePath = `${SSM_BASE_PATH}/${qualifier}`;
119
+ const getParam = (name) => Effect.tryPromise({
120
+ try: async () => {
121
+ const result = await ssmClient.send(new GetParameterCommand({ Name: `${basePath}/${name}` }));
122
+ return result.Parameter?.Value ?? "";
123
+ },
124
+ catch: (error) => new Error(`Failed to get SSM parameter ${name}: ${String(error)}`),
125
+ });
126
+ const httpEndpoint = yield* getParam("http-endpoint");
127
+ const realtimeEndpoint = yield* getParam("realtime-endpoint");
128
+ return { httpEndpoint, realtimeEndpoint };
129
+ });
130
+ /**
131
+ * CloudFormation stack name tag (set automatically by CDK)
132
+ */
133
+ const CFN_STACK_NAME_TAG = "aws:cloudformation:stack-name";
134
+ /**
135
+ * Discover Lambda functions with live-lambda tags.
136
+ * @param stackFilter - Optional list of stack names to filter by
137
+ */
138
+ const discoverFunctions = (stackFilter) => Effect.gen(function* () {
139
+ const lambdaClient = new LambdaClient({});
140
+ const functions = [];
141
+ // List all functions
142
+ let nextMarker;
143
+ do {
144
+ const listResponse = yield* Effect.tryPromise({
145
+ try: () => lambdaClient.send(new ListFunctionsCommand({ Marker: nextMarker })),
146
+ catch: (error) => new Error(`Failed to list functions: ${String(error)}`),
147
+ });
148
+ for (const fn of listResponse.Functions ?? []) {
149
+ // Get tags for each function
150
+ const tagsResult = yield* Effect.tryPromise({
151
+ try: () => lambdaClient.send(new ListTagsCommand({ Resource: fn.FunctionArn })),
152
+ catch: () => new Error(`Failed to get tags for ${fn.FunctionName}`),
153
+ }).pipe(Effect.catchAll(() => Effect.succeed({ Tags: {} })));
154
+ const tags = tagsResult.Tags ?? {};
155
+ // Check for live-lambda tag (handler) or docker-context tag
156
+ if (tags[LIVE_LAMBDA_TAG] || tags[LIVE_LAMBDA_DOCKER_TAG]) {
157
+ // Filter by stack name if specified
158
+ if (stackFilter && stackFilter.length > 0) {
159
+ const stackName = tags[CFN_STACK_NAME_TAG];
160
+ if (!stackName || !stackFilter.includes(stackName)) {
161
+ continue;
162
+ }
163
+ }
164
+ // Determine architecture from Lambda config
165
+ const architectures = fn.Architectures ?? ["x86_64"];
166
+ const architecture = architectures.includes("arm64")
167
+ ? "arm64"
168
+ : "x86_64";
169
+ const discovered = {
170
+ functionName: fn.FunctionName,
171
+ functionArn: fn.FunctionArn,
172
+ localHandler: tags[LIVE_LAMBDA_TAG] ?? "",
173
+ dockerContextPath: tags[LIVE_LAMBDA_DOCKER_TAG],
174
+ memoryMB: fn.MemorySize ?? 128,
175
+ architecture,
176
+ };
177
+ functions.push(discovered);
178
+ }
179
+ }
180
+ nextMarker = listResponse.NextMarker;
181
+ } while (nextMarker);
182
+ return functions;
183
+ });
184
+ /**
185
+ * Start a long-running Docker container for a function.
186
+ * Builds the image from the local Docker context path, then runs it.
187
+ * The container continuously polls our Runtime API for invocations.
188
+ * Uses Effect.forkDaemon to ensure the container runs independently with context.
189
+ */
190
+ const startFunctionContainer = (fn, port, projectRoot, additionalEnv, invocationContexts) => Effect.gen(function* () {
191
+ if (!fn.dockerContextPath) {
192
+ return yield* Effect.fail(new Error(`Function ${fn.functionName} has no Docker context path`));
193
+ }
194
+ const docker = yield* Docker;
195
+ const dockerRuntime = yield* docker.getRuntimeInfo();
196
+ const runtimeApiHost = dockerRuntime.isDockerDesktop
197
+ ? "host.docker.internal"
198
+ : "runtime.api";
199
+ // Generate a local image name from function name
200
+ const imageName = `live-lambda-${fn.functionName.toLowerCase().replace(/[^a-z0-9-]/g, "-")}`;
201
+ // Resolve the context path relative to project root
202
+ const contextPath = fn.dockerContextPath.startsWith("/")
203
+ ? fn.dockerContextPath
204
+ : `${projectRoot}/${fn.dockerContextPath}`;
205
+ // Determine platform from architecture
206
+ const platform = fn.architecture === "arm64" ? "linux/arm64" : "linux/amd64";
207
+ // Build the Docker image from local context
208
+ yield* docker
209
+ .build({
210
+ contextPath,
211
+ imageName,
212
+ platform,
213
+ })
214
+ .pipe(Effect.scoped);
215
+ // Inspect the image to get original entrypoint/cmd for extension wrapper
216
+ const imageConfig = yield* docker.inspect(imageName).pipe(Effect.scoped);
217
+ const extensionWrapper = buildExtensionWrapperCommand(imageConfig.entrypoint, imageConfig.cmd);
218
+ const containerConfig = makeLambdaContainerConfig({
219
+ imageUri: imageName,
220
+ runtimeApiHost,
221
+ runtimeApiPort: port,
222
+ functionName: fn.functionName,
223
+ functionVersion: "$LATEST",
224
+ memoryMB: fn.memoryMB,
225
+ timeoutSeconds: 3600, // Long timeout - container stays running
226
+ platform,
227
+ additionalEnv,
228
+ invocationContexts,
229
+ });
230
+ // Apply extension wrapper to start extensions before the app
231
+ containerConfig.entrypoint = extensionWrapper.entrypoint;
232
+ containerConfig.command = extensionWrapper.command;
233
+ yield* Effect.logInfo(`Starting container for ${fn.functionName} on port ${port}`);
234
+ // Use Effect.forkDaemon to run the container independently with context preserved
235
+ // Effect.scoped provides the scope needed by docker.run
236
+ const fiber = yield* docker.run(containerConfig).pipe(Effect.scoped, Effect.map(() => undefined), Effect.forkDaemon);
237
+ return fiber;
238
+ }).pipe(Effect.provide(DockerLive));
239
+ // Type guards for response types
240
+ const isLambdaResponse = (r) => "body" in r && !("errorType" in r);
241
+ const isLambdaError = (r) => "requestId" in r && "errorType" in r;
242
+ const isLambdaInitError = (r) => !("requestId" in r) && "errorType" in r;
243
+ /**
244
+ * Reset the idle timer for a container.
245
+ * After the timeout expires with no new responses, the container is stopped.
246
+ * This prevents the confusing 503 error from the Lambda RIC when the poll times out.
247
+ */
248
+ const resetIdleTimer = (container, containersRef, idleTimeoutMs) => Effect.gen(function* () {
249
+ // Cancel existing timer if any
250
+ if (container.idleTimerFiber) {
251
+ yield* Fiber.interrupt(container.idleTimerFiber).pipe(Effect.catchAll(() => Effect.void));
252
+ container.idleTimerFiber = null;
253
+ }
254
+ // Start new timer - use never for error type since we catch all errors
255
+ const timerFiber = yield* Effect.gen(function* () {
256
+ yield* Effect.sleep(Duration.millis(idleTimeoutMs));
257
+ // Timer expired - stop container proactively
258
+ yield* Effect.logInfo(`[Local] Stopping idle container ${container.fn.functionName} (will restart on next invocation)`);
259
+ // Stop the container using Docker with fast timeout (1s)
260
+ // The container may be blocked on long-polling, so we need to force kill
261
+ const docker = yield* Docker;
262
+ yield* docker.stop(container.containerName, 1).pipe(Effect.scoped, Effect.catchAll(() => Effect.void));
263
+ // Remove from map so next invocation creates fresh container
264
+ const currentContainers = yield* Ref.get(containersRef);
265
+ currentContainers.delete(container.fn.functionName);
266
+ yield* Ref.set(containersRef, currentContainers);
267
+ }).pipe(Effect.provide(DockerLive), Effect.catchAll(() => Effect.void), Effect.forkDaemon);
268
+ container.idleTimerFiber = timerFiber;
269
+ });
270
+ /**
271
+ * Process responses from a container and dispatch to waiting callers.
272
+ * After each response, resets the idle timer to proactively stop the container
273
+ * before the poll timeout causes confusing Lambda RIC errors.
274
+ */
275
+ const processContainerResponses = (container, containersRef, idleTimeoutMs, client, invocationContexts) => Effect.gen(function* () {
276
+ const responseChannel = buildChannelName.response(container.fn.functionName);
277
+ // Continuously process responses from the container
278
+ while (true) {
279
+ const result = yield* waitForResponse(container.runtimeState);
280
+ let response;
281
+ if (isLambdaResponse(result)) {
282
+ response = {
283
+ type: "response",
284
+ requestId: result.requestId,
285
+ result: result.body,
286
+ };
287
+ }
288
+ else if (isLambdaError(result)) {
289
+ response = {
290
+ type: "response",
291
+ requestId: result.requestId,
292
+ error: {
293
+ errorType: result.errorType,
294
+ errorMessage: result.errorMessage,
295
+ stackTrace: result.stackTrace,
296
+ },
297
+ };
298
+ }
299
+ else if (isLambdaInitError(result)) {
300
+ yield* Effect.logError(`[Local] Init error: ${result.errorType}: ${result.errorMessage}`);
301
+ continue;
302
+ }
303
+ else {
304
+ continue;
305
+ }
306
+ // Send response back via AppSync
307
+ yield* client.publishResponse(responseChannel, response);
308
+ yield* Effect.logDebug(`[Local] Sent response for ${response.requestId}`);
309
+ // Print end marker with timing
310
+ const ctx = invocationContexts.get(response.requestId);
311
+ if (ctx) {
312
+ const durationMs = Date.now() - ctx.start;
313
+ if (response.error) {
314
+ console.log(`[${ctx.num}] \u2514\u2500\u2500 \u2717 ${response.error.errorType}: ${response.error.errorMessage} (${durationMs}ms) \u2500\u2500`);
315
+ }
316
+ else {
317
+ console.log(`[${ctx.num}] \u2514\u2500\u2500 \u2713 Done (${durationMs}ms) \u2500\u2500`);
318
+ }
319
+ invocationContexts.delete(response.requestId);
320
+ }
321
+ // Reset idle timer - container will be stopped if no new invocations arrive
322
+ yield* resetIdleTimer(container, containersRef, idleTimeoutMs);
323
+ }
324
+ });
325
+ /**
326
+ * Start a Node.js worker process for a function.
327
+ * Spawns Bun to run the runtime wrapper with the handler path.
328
+ * The worker continuously polls our Runtime API for invocations.
329
+ */
330
+ const startNodejsWorker = (fn, port, projectRoot, env, invocationContexts) => Effect.gen(function* () {
331
+ if (!fn.localHandler) {
332
+ return yield* Effect.fail(new Error(`Function ${fn.functionName} has no local handler path`));
333
+ }
334
+ // Resolve the runtime wrapper path relative to this module
335
+ const __filename = fileURLToPath(import.meta.url);
336
+ const __dirname = path.dirname(__filename);
337
+ const runtimeWrapperPath = path.resolve(__dirname, "..", "runtime-wrapper", "nodejs-runtime.js");
338
+ // Get absolute path to bun for pure env isolation (no PATH dependency)
339
+ const bunPath = Bun.which("bun");
340
+ if (!bunPath) {
341
+ return yield* Effect.fail(new Error("Could not find 'bun' executable in PATH"));
342
+ }
343
+ // Build environment for the worker process
344
+ // Pure isolation: only env from bridge + local overrides, no local PATH/HOME
345
+ const workerEnv = {
346
+ // Start with env vars from the bridge Lambda (AWS credentials, user-defined vars)
347
+ ...env,
348
+ // Local overrides (these are set by the daemon, not from bridge)
349
+ AWS_LAMBDA_RUNTIME_API: `localhost:${port}`,
350
+ _HANDLER: fn.localHandler,
351
+ LAMBDA_TASK_ROOT: projectRoot,
352
+ // Memory limit for context object
353
+ AWS_LAMBDA_FUNCTION_MEMORY_SIZE: String(fn.memoryMB),
354
+ };
355
+ yield* Effect.logDebug(`[Local] Starting Node.js worker for ${fn.functionName} on port ${port}`);
356
+ yield* Effect.logDebug(`[Local] Handler: ${fn.localHandler}`);
357
+ // Spawn Bun with --watch to automatically restart when handler files change
358
+ // This enables hot-reload without needing to restart the daemon
359
+ // Use absolute bun path for pure env isolation (no PATH dependency)
360
+ const workerProcess = spawn(bunPath, ["--watch", runtimeWrapperPath], {
361
+ cwd: projectRoot,
362
+ env: workerEnv,
363
+ stdio: ["ignore", "pipe", "pipe"],
364
+ });
365
+ // Pattern to parse Lambda log format: TIMESTAMP\tREQUEST_ID\tLEVEL\tMESSAGE
366
+ // Lambda uses tabs between fields. Captures: [1] = request ID, [2] = level + message
367
+ const lambdaLogPattern = /^(\d{4}-\d{2}-\d{2}T[\d:.]+Z)[\t\s]+([0-9a-f-]{36})[\t\s]+(.*)$/i;
368
+ // Helper to format log line with invocation prefix
369
+ const formatLine = (rawLine) => {
370
+ // Strip carriage returns that can cause terminal corruption
371
+ const line = rawLine.replace(/\r/g, "");
372
+ const match = lambdaLogPattern.exec(line);
373
+ if (match) {
374
+ const requestId = match[2];
375
+ const ctx = invocationContexts.get(requestId);
376
+ if (ctx) {
377
+ // Strip timestamp and request ID, keep just LEVEL MESSAGE
378
+ return { prefix: `[${ctx.num}]`, content: match[3] };
379
+ }
380
+ }
381
+ return { prefix: "[Worker]", content: line };
382
+ };
383
+ // Forward stdout/stderr with invocation number prefix
384
+ workerProcess.stdout?.on("data", (data) => {
385
+ const lines = data.toString().trim().split("\n");
386
+ for (const rawLine of lines) {
387
+ const { prefix, content } = formatLine(rawLine);
388
+ console.log(`${prefix} ${content}`);
389
+ }
390
+ });
391
+ workerProcess.stderr?.on("data", (data) => {
392
+ const lines = data.toString().trim().split("\n");
393
+ for (const rawLine of lines) {
394
+ const { prefix, content } = formatLine(rawLine);
395
+ console.error(`${prefix} ${content}`);
396
+ }
397
+ });
398
+ workerProcess.on("error", (err) => {
399
+ Effect.runSync(Effect.logError(`Worker error: ${err.message}`).pipe(Effect.annotateLogs("function", fn.functionName)));
400
+ });
401
+ workerProcess.on("close", (code) => {
402
+ Effect.runSync(Effect.logInfo(`Worker exited with code ${code}`).pipe(Effect.annotateLogs("function", fn.functionName)));
403
+ });
404
+ return workerProcess;
405
+ });
406
+ /**
407
+ * Process responses from a Node.js worker and dispatch to waiting callers.
408
+ */
409
+ const processWorkerResponses = (worker, client, invocationContexts) => Effect.gen(function* () {
410
+ const responseChannel = buildChannelName.response(worker.fn.functionName);
411
+ // Continuously process responses from the worker
412
+ while (true) {
413
+ const result = yield* waitForResponse(worker.runtimeState);
414
+ let response;
415
+ if (isLambdaResponse(result)) {
416
+ response = {
417
+ type: "response",
418
+ requestId: result.requestId,
419
+ result: result.body,
420
+ };
421
+ }
422
+ else if (isLambdaError(result)) {
423
+ response = {
424
+ type: "response",
425
+ requestId: result.requestId,
426
+ error: {
427
+ errorType: result.errorType,
428
+ errorMessage: result.errorMessage,
429
+ stackTrace: result.stackTrace,
430
+ },
431
+ };
432
+ }
433
+ else if (isLambdaInitError(result)) {
434
+ yield* Effect.logError(`[Local] Worker init error: ${result.errorType}: ${result.errorMessage}`);
435
+ continue;
436
+ }
437
+ else {
438
+ continue;
439
+ }
440
+ // Send response back via AppSync
441
+ yield* client.publishResponse(responseChannel, response);
442
+ yield* Effect.logDebug(`[Local] Sent response for ${response.requestId}`);
443
+ // Print end marker with timing
444
+ const ctx = invocationContexts.get(response.requestId);
445
+ if (ctx) {
446
+ const durationMs = Date.now() - ctx.start;
447
+ if (response.error) {
448
+ console.log(`[${ctx.num}] \u2514\u2500\u2500 \u2717 ${response.error.errorType}: ${response.error.errorMessage} (${durationMs}ms) \u2500\u2500`);
449
+ }
450
+ else {
451
+ console.log(`[${ctx.num}] \u2514\u2500\u2500 \u2713 Done (${durationMs}ms) \u2500\u2500`);
452
+ }
453
+ invocationContexts.delete(response.requestId);
454
+ }
455
+ }
456
+ });
457
+ /**
458
+ * Environment variables to completely filter out from invocation env.
459
+ * These are not relevant for local execution.
460
+ */
461
+ const ENV_VARS_TO_FILTER = new Set(["AWS_LAMBDA_LOG_STREAM_NAME"]);
462
+ /**
463
+ * Environment variables to ignore when calculating if env has changed.
464
+ * These change frequently but don't require a container restart.
465
+ *
466
+ * TODO: AWS credentials (ACCESS_KEY_ID, SECRET_ACCESS_KEY, SESSION_TOKEN)
467
+ * do expire and will need refreshing. For now we ignore them to avoid
468
+ * unnecessary restarts, but we should implement credential refresh logic
469
+ * that updates the container's credentials without a full restart.
470
+ */
471
+ const ENV_VARS_TO_IGNORE_IN_DIFF = new Set([
472
+ "AWS_ACCESS_KEY_ID",
473
+ "AWS_SECRET_ACCESS_KEY",
474
+ "AWS_SESSION_TOKEN",
475
+ ]);
476
+ /**
477
+ * Filter out irrelevant env vars from invocation environment.
478
+ */
479
+ const sanitizeInvocationEnv = (env) => {
480
+ const result = {};
481
+ for (const [key, value] of Object.entries(env)) {
482
+ if (!ENV_VARS_TO_FILTER.has(key)) {
483
+ result[key] = value;
484
+ }
485
+ }
486
+ return result;
487
+ };
488
+ /**
489
+ * Compute which environment variables have changed between two env objects.
490
+ * Returns a human-readable summary of the changes.
491
+ * Ignores env vars in ENV_VARS_TO_IGNORE_IN_DIFF.
492
+ */
493
+ const getEnvDiff = (oldEnv, newEnv) => {
494
+ const changes = [];
495
+ // Check for added or modified keys
496
+ for (const key of Object.keys(newEnv)) {
497
+ if (ENV_VARS_TO_IGNORE_IN_DIFF.has(key))
498
+ continue;
499
+ if (!(key in oldEnv)) {
500
+ changes.push(`+${key}`);
501
+ }
502
+ else if (oldEnv[key] !== newEnv[key]) {
503
+ changes.push(`~${key}`);
504
+ }
505
+ }
506
+ // Check for removed keys
507
+ for (const key of Object.keys(oldEnv)) {
508
+ if (ENV_VARS_TO_IGNORE_IN_DIFF.has(key))
509
+ continue;
510
+ if (!(key in newEnv)) {
511
+ changes.push(`-${key}`);
512
+ }
513
+ }
514
+ return changes.join(", ");
515
+ };
516
+ /**
517
+ * Ensure a Node.js worker is started for a function.
518
+ * If the worker already exists, return it.
519
+ * Otherwise, create the Runtime API server, add to workers map, and start the worker.
520
+ */
521
+ const ensureWorkerStarted = (fn, invocationEnv, workersRef, serverScope, projectRoot, appSyncClient, invocationContexts) => Effect.gen(function* () {
522
+ // Check if worker already exists
523
+ const currentWorkers = yield* Ref.get(workersRef);
524
+ const existing = currentWorkers.get(fn.functionName);
525
+ if (existing) {
526
+ // Check if env vars have changed (e.g., after CDK redeploy)
527
+ const envDiff = getEnvDiff(existing.env, invocationEnv);
528
+ if (envDiff) {
529
+ yield* Effect.logInfo(`[Local] Environment changed for ${fn.functionName}, restarting worker... (${envDiff})`);
530
+ // Kill the old worker
531
+ yield* Effect.try(() => existing.workerProcess.kill("SIGTERM")).pipe(Effect.catchAll(() => Effect.void));
532
+ // Remove from map so we create a new one below
533
+ currentWorkers.delete(fn.functionName);
534
+ yield* Ref.set(workersRef, currentWorkers);
535
+ }
536
+ else {
537
+ return existing;
538
+ }
539
+ }
540
+ // Worker doesn't exist - start it lazily
541
+ yield* Effect.logDebug(`[Local] Starting Node.js worker for first invocation of ${fn.functionName}...`);
542
+ // Create Runtime API server on ephemeral port
543
+ const { port, state: runtimeState } = yield* startRuntimeApiServer({
544
+ functionMetadata: {
545
+ functionName: fn.functionName,
546
+ functionVersion: "$LATEST",
547
+ handler: fn.localHandler,
548
+ },
549
+ }).pipe(Effect.provideService(Scope.Scope, serverScope));
550
+ // Create worker object (without process initially - will be set after start)
551
+ // We add to map BEFORE starting worker to handle concurrent invocations
552
+ const worker = {
553
+ fn,
554
+ runtimeState,
555
+ port,
556
+ workerProcess: undefined, // Will be set shortly
557
+ env: invocationEnv,
558
+ };
559
+ // Add to map immediately to prevent race conditions
560
+ currentWorkers.set(fn.functionName, worker);
561
+ yield* Ref.set(workersRef, currentWorkers);
562
+ // Start the worker process - remove from map on failure
563
+ const workerProcess = yield* startNodejsWorker(fn, port, projectRoot, invocationEnv, invocationContexts).pipe(Effect.catchAll((error) => Effect.gen(function* () {
564
+ // Remove broken worker from map on failure
565
+ yield* Effect.logError(`[Local] Failed to start worker for ${fn.functionName}: ${error}`);
566
+ const current = yield* Ref.get(workersRef);
567
+ current.delete(fn.functionName);
568
+ yield* Ref.set(workersRef, current);
569
+ return yield* Effect.fail(error);
570
+ })));
571
+ // Update worker with the process
572
+ worker.workerProcess = workerProcess;
573
+ // Start processing responses in the background
574
+ Effect.runFork(processWorkerResponses(worker, appSyncClient, invocationContexts));
575
+ return worker;
576
+ });
577
+ /**
578
+ * Ensure a container is started for a function.
579
+ * If the container already exists, return it.
580
+ * If the env vars have changed, restart the container.
581
+ * Otherwise, create the Runtime API server, add to containers map, and start the container.
582
+ */
583
+ const ensureContainerStarted = (fn, invocationEnv, containersRef, serverScope, projectRoot, idleTimeoutMs, pollTimeoutMs, appSyncClient, invocationContexts) => Effect.gen(function* () {
584
+ // Check if container already exists
585
+ const currentContainers = yield* Ref.get(containersRef);
586
+ const existing = currentContainers.get(fn.functionName);
587
+ if (existing) {
588
+ // Check if env vars have changed (e.g., after CDK redeploy)
589
+ const envDiff = getEnvDiff(existing.env, invocationEnv);
590
+ if (envDiff) {
591
+ // IMPORTANT: Update env immediately (before any yields) to prevent
592
+ // other concurrent invocations from also detecting the change and
593
+ // triggering duplicate restarts
594
+ existing.env = invocationEnv;
595
+ // If already restarting for env change, just return the existing container
596
+ // The invocation will be queued and picked up by the new container
597
+ if (existing.isRebuilding) {
598
+ yield* Effect.logDebug(`[Local] Environment change restart already in progress for ${fn.functionName}, queueing invocation`);
599
+ return existing;
600
+ }
601
+ yield* Effect.logInfo(`[Local] Environment changed for ${fn.functionName}, restarting container... (${envDiff})`);
602
+ existing.isRebuilding = true;
603
+ // Stop the Docker container forcefully (timeout=1s)
604
+ // This sends SIGTERM and then SIGKILL after 1 second
605
+ // We must do this BEFORE interrupting the fiber, because the fiber is
606
+ // blocked waiting for the docker process to exit
607
+ yield* Effect.gen(function* () {
608
+ const docker = yield* Docker;
609
+ yield* docker.stop(existing.containerName, 1).pipe(Effect.scoped);
610
+ }).pipe(Effect.provide(DockerLive), Effect.catchAll((error) => Effect.logDebug(`[Local] Error stopping container (may already be stopped): ${error}`)));
611
+ // Now interrupt the fiber (it should complete quickly since container stopped)
612
+ yield* Fiber.interrupt(existing.containerFiber).pipe(Effect.catchAll(() => Effect.void));
613
+ // Restart the container with new environment, reusing existing RuntimeAPI
614
+ const newFiber = yield* startFunctionContainer(fn, existing.port, projectRoot, invocationEnv, invocationContexts).pipe(Effect.catchAll((error) => Effect.gen(function* () {
615
+ yield* Effect.logError(`[Local] Failed to restart container for ${fn.functionName}: ${error}`);
616
+ existing.isRebuilding = false;
617
+ return yield* Effect.fail(error);
618
+ })));
619
+ // Update container with new fiber
620
+ existing.containerFiber = newFiber;
621
+ existing.isRebuilding = false;
622
+ return existing;
623
+ }
624
+ else {
625
+ return existing;
626
+ }
627
+ }
628
+ // Container doesn't exist - start it lazily
629
+ yield* Effect.logInfo(`[Local] Starting container for first invocation of ${fn.functionName}...`);
630
+ // Create Runtime API server on ephemeral port
631
+ // Use poll timeout that's shorter than idle timeout so container exits naturally
632
+ const { port, state: runtimeState } = yield* startRuntimeApiServer({
633
+ pollTimeoutMs,
634
+ functionMetadata: {
635
+ functionName: fn.functionName,
636
+ functionVersion: "$LATEST",
637
+ handler: "index.handler",
638
+ },
639
+ }).pipe(Effect.provideService(Scope.Scope, serverScope));
640
+ // Generate container name and image name
641
+ const containerName = `lambda-${fn.functionName.replace(/[^a-zA-Z0-9]/g, "-")}`;
642
+ const imageName = `live-lambda-${fn.functionName.toLowerCase().replace(/[^a-z0-9-]/g, "-")}`;
643
+ // Create container object (without fiber initially - will be set after start)
644
+ // We add to map BEFORE starting container to handle concurrent invocations
645
+ const container = {
646
+ fn,
647
+ runtimeState,
648
+ port,
649
+ containerFiber: undefined, // Will be set shortly
650
+ containerName,
651
+ imageName,
652
+ isRebuilding: false,
653
+ pendingResponses: new Map(),
654
+ idleTimerFiber: null,
655
+ env: invocationEnv,
656
+ };
657
+ // Add to map immediately to prevent race conditions
658
+ currentContainers.set(fn.functionName, container);
659
+ yield* Ref.set(containersRef, currentContainers);
660
+ // Build and start the container - remove from map on failure
661
+ const containerFiber = yield* startFunctionContainer(fn, port, projectRoot, invocationEnv, invocationContexts).pipe(Effect.catchAll((error) => Effect.gen(function* () {
662
+ // Remove broken container from map on failure
663
+ yield* Effect.logError(`[Local] Failed to start container for ${fn.functionName}: ${error}`);
664
+ const current = yield* Ref.get(containersRef);
665
+ current.delete(fn.functionName);
666
+ yield* Ref.set(containersRef, current);
667
+ return yield* Effect.fail(error);
668
+ })));
669
+ // Update container with the fiber
670
+ container.containerFiber = containerFiber;
671
+ // Start processing responses in the background using forkDaemon
672
+ yield* processContainerResponses(container, containersRef, idleTimeoutMs, appSyncClient, invocationContexts).pipe(Effect.forkDaemon);
673
+ return container;
674
+ });
675
+ /**
676
+ * Rebuild a Docker container after file changes.
677
+ * Invocations arriving during rebuild will be queued and picked up by the new container.
678
+ */
679
+ const rebuildDockerContainer = (functionId, containersRef, projectRoot) => Effect.gen(function* () {
680
+ const containers = yield* Ref.get(containersRef);
681
+ const container = containers.get(functionId);
682
+ if (!container) {
683
+ yield* Effect.logInfo(`[Local] Cannot rebuild ${functionId} - container not found`);
684
+ return;
685
+ }
686
+ // Mark as rebuilding - invocations will still queue but we log it
687
+ container.isRebuilding = true;
688
+ yield* Effect.logDebug(`[Local] Rebuilding container for ${functionId}...`);
689
+ const docker = yield* Docker;
690
+ // Stop the existing container using Docker service
691
+ const containerId = container.containerName;
692
+ yield* Effect.logInfo(`[Local] Stopping container with name prefix: ${containerId}`);
693
+ const stopCount = yield* docker.stop(containerId).pipe(Effect.scoped, Effect.catchAll((error) => Effect.gen(function* () {
694
+ yield* Effect.logInfo(`Note: Container stop had issue: ${error}`);
695
+ return 0;
696
+ })));
697
+ yield* Effect.logDebug(`[Local] Stopped ${stopCount} container(s)`);
698
+ // Resolve the context path
699
+ const fn = container.fn;
700
+ const contextPath = fn.dockerContextPath?.startsWith("/")
701
+ ? fn.dockerContextPath
702
+ : `${projectRoot}/${fn.dockerContextPath}`;
703
+ // Determine platform from architecture
704
+ const platform = fn.architecture === "arm64" ? "linux/arm64" : "linux/amd64";
705
+ // Rebuild the Docker image
706
+ yield* docker
707
+ .build({
708
+ contextPath,
709
+ imageName: container.imageName,
710
+ platform,
711
+ })
712
+ .pipe(Effect.scoped);
713
+ // Inspect the image to get original entrypoint/cmd for extension wrapper
714
+ const imageConfig = yield* docker
715
+ .inspect(container.imageName)
716
+ .pipe(Effect.scoped);
717
+ const extensionWrapper = buildExtensionWrapperCommand(imageConfig.entrypoint, imageConfig.cmd);
718
+ // Restart the container by triggering container startup
719
+ // The existing fiber will have exited when we stopped the container
720
+ // We need to start a new one
721
+ const dockerRuntime = yield* docker.getRuntimeInfo();
722
+ const runtimeApiHost = dockerRuntime.isDockerDesktop
723
+ ? "host.docker.internal"
724
+ : "runtime.api";
725
+ yield* Effect.logInfo(`[Local] New container will connect to Runtime API at ${runtimeApiHost}:${container.port}`);
726
+ const containerConfig = makeLambdaContainerConfig({
727
+ imageUri: container.imageName,
728
+ runtimeApiHost,
729
+ runtimeApiPort: container.port,
730
+ functionName: fn.functionName,
731
+ functionVersion: "$LATEST",
732
+ memoryMB: fn.memoryMB,
733
+ timeoutSeconds: 3600,
734
+ platform,
735
+ });
736
+ // Apply extension wrapper to start extensions before the app
737
+ containerConfig.entrypoint = extensionWrapper.entrypoint;
738
+ containerConfig.command = extensionWrapper.command;
739
+ // Start the new container
740
+ yield* Effect.logDebug(`[Local] Starting new container for ${functionId}...`);
741
+ // Use Effect.forkDaemon to run the container independently
742
+ const newFiber = yield* docker.run(containerConfig).pipe(Effect.scoped, Effect.tap((result) => Effect.gen(function* () {
743
+ // Only suppress errors for expected exit codes from docker stop:
744
+ // 0: clean, 143: SIGTERM, 137: SIGKILL
745
+ const code = result.exitCode;
746
+ if (code !== 0 && code !== 143 && code !== 137) {
747
+ yield* Effect.logError(`Container for ${functionId} exited with code ${code}`);
748
+ yield* Effect.logError(`stderr: ${result.stderr}`);
749
+ }
750
+ })), Effect.map(() => undefined), Effect.catchAll((error) => Effect.logError(`Container error for ${functionId}: ${error}`)), Effect.forkDaemon);
751
+ // Update the container state with the new fiber
752
+ container.containerFiber = newFiber;
753
+ // Wait a moment for the container to start and begin polling
754
+ yield* Effect.sleep("2 seconds");
755
+ container.isRebuilding = false;
756
+ yield* Effect.logDebug(`[Local] Container rebuilt for ${functionId}`);
757
+ }).pipe(Effect.provide(DockerLive));
758
+ /**
759
+ * Handle incoming invocations for a Docker container function by queueing them.
760
+ * Starts the container lazily if not already running.
761
+ * Invocations are queued even if a rebuild is in progress - the new container will pick them up.
762
+ */
763
+ const handleDockerInvocation = (fn, invocation, containersRef, serverScope, projectRoot, idleTimeoutMs, pollTimeoutMs, appSyncClient, invocationCounter, invocationContexts) => Effect.gen(function* () {
764
+ // Ensure container is started (lazy startup on first invocation)
765
+ // Use env from invocation (captured from bridge Lambda on first invoke)
766
+ const container = yield* ensureContainerStarted(fn, sanitizeInvocationEnv(invocation.env ?? {}), containersRef, serverScope, projectRoot, idleTimeoutMs, pollTimeoutMs, appSyncClient, invocationContexts);
767
+ // Assign invocation number and track context for logging
768
+ const invocationNum = yield* Ref.updateAndGet(invocationCounter, (n) => n + 1);
769
+ const startTime = Date.now();
770
+ invocationContexts.set(invocation.requestId, {
771
+ num: invocationNum,
772
+ start: startTime,
773
+ fn: fn.functionName,
774
+ isDocker: true,
775
+ });
776
+ // Print start marker
777
+ console.log(`\n[${invocationNum}] \u250c\u2500\u2500 [Docker] ${fn.functionName} \u2500\u2500`);
778
+ // Log if queuing during a rebuild
779
+ if (container.isRebuilding) {
780
+ yield* Effect.logDebug(`[Local] Queueing invocation ${invocation.requestId} for ${fn.functionName} (rebuild in progress, will be picked up by new container)`);
781
+ }
782
+ else {
783
+ yield* Effect.logDebug(`[Local] Queueing invocation ${invocation.requestId} for ${fn.functionName}`);
784
+ }
785
+ const lambdaInvocation = {
786
+ requestId: invocation.requestId,
787
+ event: invocation.event,
788
+ invokedFunctionArn: invocation.context.invokedFunctionArn,
789
+ deadlineMs: Date.now() + invocation.context.getRemainingTimeInMillis,
790
+ functionName: fn.functionName,
791
+ functionVersion: invocation.context.functionVersion,
792
+ memoryLimitMB: fn.memoryMB,
793
+ logGroupName: invocation.context.logGroupName,
794
+ logStreamName: invocation.context.logStreamName,
795
+ };
796
+ yield* Effect.logDebug(`[Local] Queueing to Runtime API on port ${container.port}`);
797
+ // Notify extensions about the invocation (for Lambda Web Adapter support)
798
+ yield* notifyExtensionsInvoke(container.runtimeState, lambdaInvocation);
799
+ yield* queueInvocation(container.runtimeState, lambdaInvocation);
800
+ yield* Effect.logDebug(`[Local] Invocation queued successfully`);
801
+ });
802
+ /**
803
+ * Handle incoming invocations for a Node.js function by queueing them.
804
+ * Starts the worker lazily if not already running.
805
+ */
806
+ const handleNodejsInvocation = (fn, invocation, workersRef, serverScope, projectRoot, appSyncClient, invocationCounter, invocationContexts) => Effect.gen(function* () {
807
+ // Get env vars from invocation (forwarded from bridge Lambda)
808
+ const invocationEnv = sanitizeInvocationEnv(invocation.env ?? {});
809
+ // Ensure worker is started (lazy startup on first invocation)
810
+ const worker = yield* ensureWorkerStarted(fn, invocationEnv, workersRef, serverScope, projectRoot, appSyncClient, invocationContexts);
811
+ // Assign invocation number and track context for logging
812
+ const invocationNum = yield* Ref.updateAndGet(invocationCounter, (n) => n + 1);
813
+ const startTime = Date.now();
814
+ invocationContexts.set(invocation.requestId, {
815
+ num: invocationNum,
816
+ start: startTime,
817
+ fn: fn.functionName,
818
+ isDocker: false,
819
+ });
820
+ // Print start marker
821
+ console.log(`\n[${invocationNum}] \u250c\u2500\u2500 ${fn.functionName} \u2500\u2500`);
822
+ yield* Effect.logDebug(`[Local] Queueing invocation ${invocation.requestId} for ${fn.functionName}`);
823
+ const lambdaInvocation = {
824
+ requestId: invocation.requestId,
825
+ event: invocation.event,
826
+ invokedFunctionArn: invocation.context.invokedFunctionArn,
827
+ deadlineMs: Date.now() + invocation.context.getRemainingTimeInMillis,
828
+ functionName: fn.functionName,
829
+ functionVersion: invocation.context.functionVersion,
830
+ memoryLimitMB: fn.memoryMB,
831
+ logGroupName: invocation.context.logGroupName,
832
+ logStreamName: invocation.context.logStreamName,
833
+ };
834
+ yield* queueInvocation(worker.runtimeState, lambdaInvocation);
835
+ });
836
+ /**
837
+ * Start CDK watch process with CDK_LIVE=true using Effect's Command.
838
+ * Returns the process and a queue of events.
839
+ */
840
+ const startCdkWatch = (options, scope) => Effect.gen(function* () {
841
+ const args = [
842
+ "cdk",
843
+ "watch",
844
+ "--hotswap-fallback",
845
+ "--no-logs",
846
+ "--method=direct",
847
+ ];
848
+ if (options.stacks && options.stacks.length > 0) {
849
+ args.push(...options.stacks);
850
+ }
851
+ else if (options.all) {
852
+ args.push("--all");
853
+ }
854
+ if (options.profile) {
855
+ args.push("--profile", options.profile);
856
+ }
857
+ const env = {
858
+ ...process.env,
859
+ CDK_LIVE: "true",
860
+ };
861
+ if (options.region) {
862
+ env.AWS_REGION = options.region;
863
+ env.CDK_DEFAULT_REGION = options.region;
864
+ }
865
+ yield* Effect.logDebug(`Starting: npx ${args.join(" ")}`);
866
+ const command = PlatformCommand.make("npx", ...args).pipe(PlatformCommand.env(env), PlatformCommand.runInShell(true), PlatformCommand.stdin("inherit"));
867
+ // Extend the process resource lifetime to the provided scope
868
+ const proc = yield* PlatformCommand.start(command).pipe(Scope.extend(scope));
869
+ // Event queue for callers to subscribe to
870
+ const events = yield* Queue.unbounded();
871
+ // State tracking for deploy status messages
872
+ let isDeploying = false;
873
+ let isFirstDeploy = true;
874
+ let outputBuffer = "";
875
+ const discoveredStacks = new Set();
876
+ // Patterns to detect CDK watch behavior
877
+ const deployCompletePattern = /✅\s+\S+|Deployment time:/;
878
+ const deployStartPattern = /Deploying|hotswap|Hotswapping|Bundling/i;
879
+ const noChangesPattern = /no changes|identical|up to date/i;
880
+ const errorPattern = /error|failed|Error|Failed|ERR!/i;
881
+ const stackNamePattern = /^(\S+):\s*deploying|✅\s+(\S+)/;
882
+ // Merge stdout and stderr, decode to text, split into lines
883
+ const outputStream = Stream.merge(proc.stdout, proc.stderr).pipe(Stream.decodeText(), Stream.splitLines);
884
+ // Fork stream processing in background
885
+ yield* outputStream.pipe(Stream.runForEach((line) => Effect.gen(function* () {
886
+ outputBuffer += line + "\n";
887
+ // Log all output at debug level
888
+ yield* Effect.logDebug(`[CDK] ${line}`);
889
+ // Try to extract stack name
890
+ const match = stackNamePattern.exec(line.trim());
891
+ if (match) {
892
+ const stackName = match[1] || match[2];
893
+ if (stackName && !discoveredStacks.has(stackName)) {
894
+ discoveredStacks.add(stackName);
895
+ yield* Queue.offer(events, {
896
+ _tag: "StackDiscovered",
897
+ stackName,
898
+ });
899
+ }
900
+ }
901
+ // Detect deploy start
902
+ if (!isDeploying && deployStartPattern.test(line)) {
903
+ isDeploying = true;
904
+ if (!isFirstDeploy) {
905
+ yield* Effect.logInfo("[CDK] Deploying...");
906
+ }
907
+ }
908
+ // Detect errors - output immediately
909
+ if (errorPattern.test(line)) {
910
+ yield* Effect.sync(() => process.stderr.write(line + "\n"));
911
+ }
912
+ // Check for deploy completion (only trigger once per deploy cycle)
913
+ if (isDeploying && deployCompletePattern.test(line)) {
914
+ isDeploying = false;
915
+ isFirstDeploy = false;
916
+ outputBuffer = "";
917
+ yield* Effect.logInfo("[CDK] Deploy complete");
918
+ // Small delay to ensure AWS has propagated the changes
919
+ yield* Effect.sleep("1 second");
920
+ yield* Queue.offer(events, { _tag: "DeployComplete" });
921
+ }
922
+ // Check for no changes
923
+ if (noChangesPattern.test(line)) {
924
+ isDeploying = false;
925
+ outputBuffer = "";
926
+ }
927
+ })),
928
+ // Log errors and exit code when stream ends
929
+ Effect.tapError((error) => Effect.logError(`[CDK] CDK watch error: ${error}`)), Effect.ensuring(proc.exitCode.pipe(Effect.flatMap((code) => code !== 0
930
+ ? Effect.logError(`[CDK] CDK watch exited with code ${code}`)
931
+ : Effect.logDebug(`[CDK] CDK watch exited with code ${code}`)), Effect.catchAll(() => Effect.void))), Effect.fork);
932
+ return { process: proc, events };
933
+ });
934
+ /**
935
+ * Common CLI options.
936
+ */
937
+ const profileOption = Options.text("profile").pipe(Options.optional, Options.withDescription("AWS profile to use"));
938
+ const regionOption = Options.text("region").pipe(Options.optional, Options.withDescription("AWS region"));
939
+ const qualifierOption = Options.text("qualifier").pipe(Options.withDefault("hnb659fds"), Options.withDescription("CDK bootstrap qualifier"));
940
+ const stacksOption = Options.text("stacks").pipe(Options.optional, Options.withDescription("Stack names to deploy (comma-separated, default: all)"));
941
+ const allStacksOption = Options.boolean("all").pipe(Options.withDefault(false), Options.withDescription("Deploy all stacks (like cdk deploy --all)"));
942
+ const debugOption = Options.boolean("debug").pipe(Options.withDefault(false), Options.withDescription("Enable debug logging"));
943
+ const idleTimeoutOption = Options.integer("idle-timeout").pipe(Options.withDefault(DEFAULT_IDLE_TIMEOUT_MS), Options.withDescription("Idle timeout in ms before stopping containers (default: 230000)"));
944
+ const pollTimeoutOption = Options.integer("poll-timeout").pipe(Options.optional, Options.withDescription("Poll timeout in ms for Runtime API (default: idle-timeout - 10s). Must be shorter than idle-timeout."));
945
+ /**
946
+ * Local command definition.
947
+ */
948
+ export const localCommand = Command.make("local", {
949
+ profile: profileOption,
950
+ region: regionOption,
951
+ qualifier: qualifierOption,
952
+ stacks: stacksOption,
953
+ all: allStacksOption,
954
+ debug: debugOption,
955
+ idleTimeout: idleTimeoutOption,
956
+ pollTimeout: pollTimeoutOption,
957
+ }, ({ profile, region, qualifier, stacks, all, debug, idleTimeout, pollTimeout, }) => {
958
+ const logLevel = debug ? LogLevel.Debug : LogLevel.Info;
959
+ return Effect.gen(function* () {
960
+ yield* Effect.logInfo("[Local] Starting local Lambda development...");
961
+ const profileValue = profile._tag === "Some" ? profile.value : undefined;
962
+ const regionValue = region._tag === "Some" ? region.value : undefined;
963
+ const stacksFromOption = stacks._tag === "Some"
964
+ ? stacks.value.split(",").map((s) => s.trim())
965
+ : undefined;
966
+ // Poll timeout should be shorter than idle timeout so containers exit naturally
967
+ // before we try to stop them. Default: idle timeout - 10 seconds.
968
+ const effectivePollTimeout = pollTimeout._tag === "Some"
969
+ ? pollTimeout.value
970
+ : Math.max(idleTimeout - 10_000, 5_000); // At least 5 seconds
971
+ if (profileValue) {
972
+ process.env.AWS_PROFILE = profileValue;
973
+ }
974
+ if (regionValue) {
975
+ process.env.AWS_REGION = regionValue;
976
+ }
977
+ // Bootstrap check must complete first
978
+ yield* ensureBootstrap({
979
+ qualifier,
980
+ profile: profileValue,
981
+ region: regionValue,
982
+ });
983
+ // Stack filter - populated from CDK watch output, used to filter Lambda discovery
984
+ // Using object so closure in startOrUpdateDaemon sees updates
985
+ const filterState = { stacks: stacksFromOption ?? [] };
986
+ // Create a long-lived scope for all Runtime API servers and CDK watch process
987
+ // Resources will run until this scope is closed (when the program ends)
988
+ const serverScope = yield* Scope.make();
989
+ // Start CDK watch immediately after bootstrap
990
+ // Stack names are discovered from CDK watch output (no need for separate cdk ls)
991
+ yield* Effect.logInfo("[CDK] Deploying...");
992
+ const { process: cdkWatchProcess, events: cdkEvents } = yield* startCdkWatch({
993
+ profile: profileValue,
994
+ region: regionValue,
995
+ stacks: stacksFromOption,
996
+ all,
997
+ }, serverScope);
998
+ // Track running Docker containers by function name
999
+ const containers = yield* Ref.make(new Map());
1000
+ // Track running Node.js workers by function name
1001
+ const workers = yield* Ref.make(new Map());
1002
+ // Track registered functions (for lazy container/worker startup)
1003
+ const registeredFunctions = yield* Ref.make(new Map());
1004
+ // Track Docker functions with active file watchers
1005
+ const watchedDockerFunctions = new Set();
1006
+ // Invocation tracking for improved logging output
1007
+ const invocationCounter = yield* Ref.make(0);
1008
+ // Use a plain Map for invocation contexts so it can be accessed from stream callbacks
1009
+ const invocationContexts = new Map();
1010
+ // Project root is the current working directory (where CDK app lives)
1011
+ const projectRoot = process.cwd();
1012
+ let appSyncClient = null;
1013
+ // Track if we've logged the "Watching" message (only log once)
1014
+ const logState = { hasLoggedWatching: false, hasDiscoveredOnce: false };
1015
+ // Function to start/update the daemon with discovered functions
1016
+ const startOrUpdateDaemon = Effect.gen(function* () {
1017
+ // Only show "Discovering functions..." on first run
1018
+ if (!logState.hasDiscoveredOnce) {
1019
+ yield* Effect.logInfo("[Local] Discovering functions...");
1020
+ }
1021
+ // Get AppSync endpoints (may need to wait for first deploy)
1022
+ if (!appSyncClient) {
1023
+ yield* Effect.logDebug("[Local] Reading AppSync endpoints from SSM...");
1024
+ const endpoints = yield* getAppSyncEndpoints(qualifier).pipe(Effect.retry({ times: 10, schedule: Schedule.spaced("3 seconds") }));
1025
+ yield* Effect.logDebug(`[Local] HTTP endpoint: ${endpoints.httpEndpoint}`);
1026
+ appSyncClient = makeAppSyncClient(endpoints);
1027
+ }
1028
+ // Discover functions (filtered to stacks in this CDK project)
1029
+ yield* Effect.logDebug("[Local] Discovering Lambda functions...");
1030
+ const functions = yield* discoverFunctions(filterState.stacks);
1031
+ if (functions.length === 0) {
1032
+ yield* Effect.logInfo("[Local] No functions found with live-lambda tags yet. Have you patched your CDK project (bootstrap) and added the LiveLambdaAspect?");
1033
+ return;
1034
+ }
1035
+ const currentRegistered = yield* Ref.get(registeredFunctions);
1036
+ // Collect new functions to register
1037
+ const newFunctions = [];
1038
+ for (const fn of functions) {
1039
+ const isDocker = Boolean(fn.dockerContextPath);
1040
+ const isNodejs = !isDocker && Boolean(fn.localHandler);
1041
+ if (!isDocker && !isNodejs) {
1042
+ yield* Effect.logDebug(`[Local] Skipping ${fn.functionName} - no Docker context or local handler`);
1043
+ continue;
1044
+ }
1045
+ if (currentRegistered.has(fn.functionName)) {
1046
+ yield* Effect.logDebug(`[Local] Already watching ${fn.functionName}`);
1047
+ continue;
1048
+ }
1049
+ newFunctions.push({ fn, isDocker });
1050
+ }
1051
+ // Print summary of discovered functions
1052
+ if (newFunctions.length > 0) {
1053
+ const summary = newFunctions
1054
+ .map(({ fn, isDocker }) => {
1055
+ const mode = isDocker ? "docker" : "node";
1056
+ const shortName = fn.functionName.includes("-")
1057
+ ? fn.functionName.split("-").slice(-2, -1)[0] || fn.functionName
1058
+ : fn.functionName;
1059
+ return `${shortName} (${mode})`;
1060
+ })
1061
+ .join(", ");
1062
+ // Use different message for first discovery vs subsequent
1063
+ const prefix = logState.hasDiscoveredOnce
1064
+ ? "[Local] Functions added:"
1065
+ : "[Local] Functions:";
1066
+ yield* Effect.logInfo(`${prefix} ${summary}`);
1067
+ }
1068
+ // Register functions and set up subscriptions
1069
+ for (const { fn, isDocker } of newFunctions) {
1070
+ yield* Effect.logDebug(`[Local] Registering ${fn.functionName} (${isDocker ? "Docker" : "Node.js"})`);
1071
+ // Register the function
1072
+ currentRegistered.set(fn.functionName, fn);
1073
+ // Subscribe to invocations for this function
1074
+ const invocationChannel = buildChannelName.invocation(fn.functionName);
1075
+ yield* Effect.logDebug(`[Local] Subscribing to invocations for ${fn.functionName}`);
1076
+ // Subscribe using forkDaemon to run independently with context preserved
1077
+ // Route to Docker or Node.js handler based on function type
1078
+ if (isDocker) {
1079
+ yield* appSyncClient
1080
+ .subscribeToInvocations(invocationChannel)
1081
+ .pipe(Stream.runForEach((invocation) => handleDockerInvocation(fn, invocation, containers, serverScope, projectRoot, idleTimeout, effectivePollTimeout, appSyncClient, invocationCounter, invocationContexts).pipe(Effect.catchAll((error) => Effect.logError(`[Local] Docker invocation error: ${error}`)))), Effect.forkDaemon);
1082
+ }
1083
+ else {
1084
+ yield* appSyncClient
1085
+ .subscribeToInvocations(invocationChannel)
1086
+ .pipe(Stream.runForEach((invocation) => handleNodejsInvocation(fn, invocation, workers, serverScope, projectRoot, appSyncClient, invocationCounter, invocationContexts).pipe(Effect.catchAll((error) => Effect.logError(`[Local] Node.js invocation error: ${error}`)))), Effect.forkDaemon);
1087
+ }
1088
+ }
1089
+ yield* Ref.set(registeredFunctions, currentRegistered);
1090
+ // Start file watchers for new Docker functions
1091
+ const newDockerFunctions = [];
1092
+ for (const fn of functions) {
1093
+ if (fn.dockerContextPath &&
1094
+ !watchedDockerFunctions.has(fn.functionName)) {
1095
+ // Resolve the context path
1096
+ const contextPath = fn.dockerContextPath.startsWith("/")
1097
+ ? fn.dockerContextPath
1098
+ : `${projectRoot}/${fn.dockerContextPath}`;
1099
+ newDockerFunctions.push({
1100
+ functionId: fn.functionName,
1101
+ dockerContextPath: contextPath,
1102
+ });
1103
+ watchedDockerFunctions.add(fn.functionName);
1104
+ }
1105
+ }
1106
+ // Start watching new Docker contexts
1107
+ if (newDockerFunctions.length > 0) {
1108
+ yield* Effect.logDebug(`[Local] Starting file watchers for ${newDockerFunctions.length} Docker function(s)...`);
1109
+ // Fork a daemon fiber to handle file change events with context preserved
1110
+ yield* watchDockerContexts(newDockerFunctions, 500).pipe(Stream.runForEach((event) => Effect.gen(function* () {
1111
+ yield* Effect.logDebug(`[Local] File changed in ${event.functionId}: ${event.filePath}`);
1112
+ yield* rebuildDockerContainer(event.functionId, containers, projectRoot).pipe(Effect.catchAll((error) => Effect.logError(`[Local] Rebuild failed for ${event.functionId}: ${error}`)));
1113
+ })), Effect.forkDaemon);
1114
+ }
1115
+ if (!logState.hasLoggedWatching) {
1116
+ logState.hasLoggedWatching = true;
1117
+ yield* Effect.logInfo("[Local] Ready for invocations");
1118
+ }
1119
+ // Mark that we've completed first discovery
1120
+ logState.hasDiscoveredOnce = true;
1121
+ });
1122
+ // Handle CDK watch events (stack discovery and deploy completion)
1123
+ yield* Queue.take(cdkEvents).pipe(Effect.flatMap((event) => Effect.gen(function* () {
1124
+ switch (event._tag) {
1125
+ case "StackDiscovered":
1126
+ // Add discovered stack to filter (if not using --stacks)
1127
+ if (!stacksFromOption &&
1128
+ !filterState.stacks.includes(event.stackName)) {
1129
+ filterState.stacks.push(event.stackName);
1130
+ yield* Effect.logDebug(`[Local] Discovered stack: ${event.stackName}`);
1131
+ }
1132
+ break;
1133
+ case "DeployComplete":
1134
+ // Run daemon update on deploy completion
1135
+ yield* startOrUpdateDaemon.pipe(Effect.catchAll((error) => Effect.logError(`Failed to update daemon: ${error}`)));
1136
+ break;
1137
+ }
1138
+ })), Effect.forever, Effect.fork);
1139
+ // Handle cleanup on exit
1140
+ const cleanup = async () => {
1141
+ await Effect.runPromise(Effect.gen(function* () {
1142
+ yield* Effect.logInfo("\nShutting down...");
1143
+ // Stop CDK watch
1144
+ yield* cdkWatchProcess
1145
+ .kill("SIGTERM")
1146
+ .pipe(Effect.catchAll(() => Effect.void));
1147
+ // Stop all Docker containers using the Docker service
1148
+ const docker = yield* Docker;
1149
+ const currentContainers = yield* Ref.get(containers);
1150
+ for (const [name, container] of currentContainers) {
1151
+ yield* Effect.logInfo(`Stopping container: ${name}`);
1152
+ yield* docker.stop(container.containerName).pipe(Effect.scoped, Effect.catchAll(() => Effect.void));
1153
+ }
1154
+ // Stop all Node.js workers
1155
+ const currentWorkers = yield* Ref.get(workers);
1156
+ for (const [name, worker] of currentWorkers) {
1157
+ yield* Effect.logInfo(`Stopping worker: ${name}`);
1158
+ yield* Effect.try(() => worker.workerProcess.kill("SIGTERM")).pipe(Effect.catchAll(() => Effect.void));
1159
+ }
1160
+ // Close the server scope to clean up Runtime API servers
1161
+ yield* Scope.close(serverScope, Exit.void);
1162
+ }).pipe(Effect.provide(DockerLive), Effect.provide(BunContext.layer), Effect.provide(Logger.pretty), Effect.provide(Logger.minimumLogLevel(logLevel))));
1163
+ process.exit(0);
1164
+ };
1165
+ process.on("SIGINT", cleanup);
1166
+ process.on("SIGTERM", cleanup);
1167
+ yield* Effect.logInfo("[Local] Press Ctrl+C to stop");
1168
+ // Keep the process running
1169
+ yield* Effect.never;
1170
+ }).pipe(Effect.provide(Logger.pretty), Effect.provide(Logger.minimumLogLevel(logLevel)));
1171
+ }).pipe(Command.withDescription("Start local Lambda development with CDK watch and Docker containers"));
1172
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibG9jYWwuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvY2xpL2NvbW1hbmRzL2xvY2FsLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7Ozs7OztHQVdHO0FBRUgsT0FBTyxFQUFxQixLQUFLLEVBQUUsTUFBTSxvQkFBb0IsQ0FBQTtBQUM3RCxPQUFPLEtBQUssSUFBSSxNQUFNLFdBQVcsQ0FBQTtBQUNqQyxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sVUFBVSxDQUFBO0FBQ3hDLE9BQU8sRUFDTCxZQUFZLEVBQ1osb0JBQW9CLEVBQ3BCLGVBQWUsR0FDaEIsTUFBTSx3QkFBd0IsQ0FBQTtBQUMvQixPQUFPLEVBQUUsbUJBQW1CLEVBQUUsU0FBUyxFQUFFLE1BQU0scUJBQXFCLENBQUE7QUFDcEUsT0FBTyxFQUFFLE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSxhQUFhLENBQUE7QUFDOUMsT0FBTyxFQUVMLE9BQU8sSUFBSSxlQUFlLEdBQzNCLE1BQU0sa0JBQWtCLENBQUE7QUFFekIsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLHNCQUFzQixDQUFBO0FBQ2pELE9BQU8sRUFDTCxRQUFRLEVBQ1IsTUFBTSxFQUNOLElBQUksRUFDSixLQUFLLEVBQ0wsTUFBTSxFQUNOLFFBQVEsRUFDUixLQUFLLEVBQ0wsR0FBRyxFQUNILFFBQVEsRUFDUixLQUFLLEVBQ0wsTUFBTSxHQUNQLE1BQU0sUUFBUSxDQUFBO0FBQ2YsT0FBTyxFQUNMLG9CQUFvQixFQUNwQixpQkFBaUIsRUFDakIsZ0JBQWdCLEVBRWhCLHNCQUFzQixFQUN0QixlQUFlLEVBRWYsYUFBYSxHQUNkLE1BQU0sdUJBQXVCLENBQUE7QUFDOUIsT0FBTyxFQUFFLGlCQUFpQixFQUFFLE1BQU0sc0JBQXNCLENBQUE7QUFDeEQsT0FBTyxFQUNMLDRCQUE0QixFQUM1QixNQUFNLEVBQ04sVUFBVSxFQUNWLHlCQUF5QixHQUMxQixNQUFNLHdCQUF3QixDQUFBO0FB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QyxNQUFNLENBQUMsTUFBTSxDQUFDLEVBQzdCLE1BQU0sQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLGVBQWUsQ0FBQyxRQUFRLENBQUMsQ0FBQyxDQUNqRCxDQUNGLENBQUE7WUFFRCxPQUFPLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxDQUFBO1FBQ2pCLENBQUMsQ0FBQTtRQUVELE9BQU8sQ0FBQyxFQUFFLENBQUMsUUFBUSxFQUFFLE9BQU8sQ0FBQyxDQUFBO1FBQzdCLE9BQU8sQ0FBQyxFQUFFLENBQUMsU0FBUyxFQUFFLE9BQU8sQ0FBQyxDQUFBO1FBRTlCLEtBQUssQ0FBQyxDQUFDLE1BQU0sQ0FBQyxPQUFPLENBQUMsOEJBQThCLENBQUMsQ0FBQTtRQUVyRCwyQkFBMkI7UUFDM0IsS0FBSyxDQUFDLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQTtJQUNyQixDQUFDLENBQUMsQ0FBQyxJQUFJLENBQ0wsTUFBTSxDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUMsTUFBTSxDQUFDLEVBQzdCLE1BQU0sQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDLGVBQWUsQ0FBQyxRQUFRLENBQUMsQ0FBQyxDQUNqRCxDQUFBO0FBQ0gsQ0FBQyxDQUNGLENBQUMsSUFBSSxDQUNKLE9BQU8sQ0FBQyxlQUFlLENBQ3JCLHFFQUFxRSxDQUN0RSxDQUNGLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIExvY2FsIGNvbW1hbmQgZm9yIHJ1bm5pbmcgTGFtYmRhIGZ1bmN0aW9ucyBsb2NhbGx5IHVzaW5nIERvY2tlci5cbiAqXG4gKiBUaGlzIGNvbW1hbmQ6XG4gKiAxLiBTdGFydHMgQ0RLIHdhdGNoIHdpdGggQ0RLX0xJVkU9dHJ1ZSBhbmQgaG90c3dhcFxuICogMi4gRGlzY292ZXJzIExhbWJkYSBmdW5jdGlvbnMgd2l0aCBsaXZlLWxhbWJkYTpoYW5kbGVyIHRhZ1xuICogMy4gQ29ubmVjdHMgdG8gQXBwU3luYyBFdmVudHNcbiAqIDQuIFN1YnNjcmliZXMgdG8gaW52b2NhdGlvbiBjaGFubmVsc1xuICogNS4gRm9yIGVhY2ggZnVuY3Rpb24sIG1haW50YWlucyBPTkUgRG9ja2VyIGNvbnRhaW5lciB0aGF0IGhhbmRsZXMgYWxsIGludm9jYXRpb25zXG4gKiA2LiBTZW5kcyByZXNwb25zZXMgYmFjayB2aWEgQXBwU3luY1xuICogNy4gUmUtZGlzY292ZXJzIGZ1bmN0aW9ucyBhZnRlciBlYWNoIENESyBkZXBsb3lcbiAqL1xuXG5pbXBvcnQgeyB0eXBlIENoaWxkUHJvY2Vzcywgc3Bhd24gfSBmcm9tIFwibm9kZTpjaGlsZF9wcm9jZXNzXCJcbmltcG9ydCAqIGFzIHBhdGggZnJvbSBcIm5vZGU6cGF0aFwiXG5pbXBvcnQgeyBmaWxlVVJMVG9QYXRoIH0gZnJvbSBcIm5vZGU6dXJsXCJcbmltcG9ydCB7XG4gIExhbWJkYUNsaWVudCxcbiAgTGlzdEZ1bmN0aW9uc0NvbW1hbmQsXG4gIExpc3RUYWdzQ29tbWFuZCxcbn0gZnJvbSBcIkBhd3Mtc2RrL2NsaWVudC1sYW1iZGFcIlxuaW1wb3J0IHsgR2V0UGFyYW1ldGVyQ29tbWFuZCwgU1NNQ2xpZW50IH0gZnJvbSBcIkBhd3Mtc2RrL2NsaWVudC1zc21cIlxuaW1wb3J0IHsgQ29tbWFuZCwgT3B0aW9ucyB9IGZyb20gXCJAZWZmZWN0L2NsaVwiXG5pbXBvcnQge1xuICB0eXBlIENvbW1hbmRFeGVjdXRvcixcbiAgQ29tbWFuZCBhcyBQbGF0Zm9ybUNvbW1hbmQsXG59IGZyb20gXCJAZWZmZWN0L3BsYXRmb3JtXCJcbmltcG9ydCB0eXBlIHsgUHJvY2VzcyBhcyBFZmZlY3RQcm9jZXNzIH0gZnJvbSBcIkBlZmZlY3QvcGxhdGZvcm0vQ29tbWFuZEV4ZWN1dG9yXCJcbmltcG9ydCB7IEJ1bkNvbnRleHQgfSBmcm9tIFwiQGVmZmVjdC9wbGF0Zm9ybS1idW5cIlxuaW1wb3J0IHtcbiAgRHVyYXRpb24sXG4gIEVmZmVjdCxcbiAgRXhpdCxcbiAgRmliZXIsXG4gIExvZ2dlcixcbiAgTG9nTGV2ZWwsXG4gIFF1ZXVlLFxuICBSZWYsXG4gIFNjaGVkdWxlLFxuICBTY29wZSxcbiAgU3RyZWFtLFxufSBmcm9tIFwiZWZmZWN0XCJcbmltcG9ydCB7XG4gIEJPT1RTVFJBUF9TVEFDS19OQU1FLFxuICBCT09UU1RSQVBfVkVSU0lPTixcbiAgYnVpbGRDaGFubmVsTmFtZSxcbiAgdHlwZSBJbnZvY2F0aW9uTWVzc2FnZSxcbiAgTElWRV9MQU1CREFfRE9DS0VSX1RBRyxcbiAgTElWRV9MQU1CREFfVEFHLFxuICB0eXBlIFJlc3BvbnNlTWVzc2FnZSxcbiAgU1NNX0JBU0VfUEFUSCxcbn0gZnJvbSBcIi4uLy4uL3NoYXJlZC90eXBlcy5qc1wiXG5pbXBvcnQgeyBtYWtlQXBwU3luY0NsaWVudCB9IGZyb20gXCIuLi9hcHBzeW5jL2NsaWVudC5qc1wiXG5pbXBvcnQge1xuICBidWlsZEV4dGVuc2lvbldyYXBwZXJDb21tYW5kLFxuICBEb2NrZXIsXG4gIERvY2tlckxpdmUsXG4gIG1ha2VMYW1iZGFDb250YWluZXJDb25maWcsXG59IGZyb20gXCIuLi9kb2NrZXIvY29udGFpbmVyLmpzXCJcbmltcG9ydCB7XG4gIHR5cGUgV2F0Y2hlZERvY2tlckZ1bmN0aW9uLFxuICB3YXRjaERvY2tlckNvbnRleHRzLFxufSBmcm9tIFwiLi4vZG9ja2VyL3dhdGNoZXIuanNcIlxuaW1wb3J0IHtcbiAgbm90aWZ5RXh0ZW5zaW9uc0ludm9rZSxcbiAgcXVldWVJbnZvY2F0aW9uLFxuICB0eXBlIFJ1bnRpbWVBcGlTdGF0ZSxcbiAgc3RhcnRSdW50aW1lQXBpU2VydmVyLFxuICB3YWl0Rm9yUmVzcG9uc2UsXG59IGZyb20gXCIuLi9ydW50aW1lLWFwaS9zZXJ2ZXIuanNcIlxuaW1wb3J0IHR5cGUge1xuICBMYW1iZGFFcnJvcixcbiAgTGFtYmRhSW5pdEVycm9yLFxuICBMYW1iZGFJbnZvY2F0aW9uLFxuICBMYW1iZGFSZXNwb25zZSxcbn0gZnJvbSBcIi4uL3J1bnRpbWUtYXBpL3R5cGVzLmpzXCJcblxuLyoqXG4gKiBEaXNjb3ZlcmVkIExhbWJkYSBmdW5jdGlvbiBpbmZvLlxuICovXG5pbnRlcmZhY2UgRGlzY292ZXJlZEZ1bmN0aW9uIHtcbiAgZnVuY3Rpb25OYW1lOiBzdHJpbmdcbiAgZnVuY3Rpb25Bcm46IHN0cmluZ1xuICBsb2NhbEhhbmRsZXI6IHN0cmluZ1xuICAvKiogTG9jYWwgRG9ja2VyIGNvbnRleHQgcGF0aCBmb3IgRG9ja2VySW1hZ2VGdW5jdGlvbiAqL1xuICBkb2NrZXJDb250ZXh0UGF0aD86IHN0cmluZ1xuICBtZW1vcnlNQjogbnVtYmVyXG4gIGFyY2hpdGVjdHVyZT86IFwiYXJtNjRcIiB8IFwieDg2XzY0XCJcbn1cblxuLyoqXG4gKiBEZWZhdWx0IGlkbGUgdGltZW91dCBiZWZvcmUgcHJvYWN0aXZlbHkgc3RvcHBpbmcgY29udGFpbmVycy5cbiAqIFNldCBzbGlnaHRseSBiZWxvdyB0aGUgcG9sbCB0aW1lb3V0ICgyNDBzKSB0byBzdG9wIGNvbnRhaW5lciBiZWZvcmVcbiAqIHRoZSBMYW1iZGEgUklDIGdldHMgYSA1MDMgZXJyb3IuXG4gKi9cbmNvbnN0IERFRkFVTFRfSURMRV9USU1FT1VUX01TID0gMjMwXzAwMCAvLyAyMzAgc2Vjb25kcyAofjQgbWludXRlcylcblxuLyoqXG4gKiBTdGF0ZSBmb3IgYSBydW5uaW5nIGZ1bmN0aW9uIGNvbnRhaW5lci5cbiAqL1xuaW50ZXJmYWNlIEZ1bmN0aW9uQ29udGFpbmVyIHtcbiAgZm46IERpc2NvdmVyZWRGdW5jdGlvblxuICBydW50aW1lU3RhdGU6IFJ1bnRpbWVBcGlTdGF0ZVxuICAvKiogVGhlIHBvcnQgdGhlIFJ1bnRpbWUgQVBJIHNlcnZlciBpcyBsaXN0ZW5pbmcgb24gKi9cbiAgcG9ydDogbnVtYmVyXG4gIGNvbnRhaW5lckZpYmVyOiBGaWJlci5SdW50aW1lRmliZXI8dm9pZCwgRXJyb3I+XG4gIGNvbnRhaW5lck5hbWU6IHN0cmluZ1xuICAvKiogRG9ja2VyIGltYWdlIG5hbWUgZm9yIHJlYnVpbGRzICovXG4gIGltYWdlTmFtZTogc3RyaW5nXG4gIC8qKiBXaGV0aGVyIHRoZSBjb250YWluZXIgaXMgY3VycmVudGx5IGJlaW5nIHJlYnVpbHQgKGludm9jYXRpb25zIHdpbGwgcXVldWUpICovXG4gIGlzUmVidWlsZGluZzogYm9vbGVhblxuICAvKiogTWFwIG9mIHJlcXVlc3RJZCAtPiByZXNwb25zZSByZXNvbHZlciAqL1xuICBwZW5kaW5nUmVzcG9uc2VzOiBNYXA8XG4gICAgc3RyaW5nLFxuICAgIHtcbiAgICAgIHJlc29sdmU6IChyZXNwb25zZTogUmVzcG9uc2VNZXNzYWdlKSA9PiB2b2lkXG4gICAgfVxuICA+XG4gIC8qKiBGaWJlciBmb3IgdGhlIGlkbGUgc2h1dGRvd24gdGltZXIsIGNhbmNlbGxlZCB3aGVuIG5ldyByZXNwb25zZXMgYXJyaXZlICovXG4gIGlkbGVUaW1lckZpYmVyOiBGaWJlci5SdW50aW1lRmliZXI8dm9pZCwgbmV2ZXI+IHwgbnVsbFxuICAvKiogRW52aXJvbm1lbnQgdmFyaWFibGVzIGNhcHR1cmVkIGZyb20gZmlyc3QgaW52b2NhdGlvbiAqL1xuICBlbnY6IFJlY29yZDxzdHJpbmcsIHN0cmluZz5cbn1cblxuLyoqXG4gKiBTdGF0ZSBmb3IgYSBydW5uaW5nIE5vZGUuanMgd29ya2VyIHByb2Nlc3MuXG4gKi9cbmludGVyZmFjZSBOb2RlanNXb3JrZXIge1xuICBmbjogRGlzY292ZXJlZEZ1bmN0aW9uXG4gIHJ1bnRpbWVTdGF0ZTogUnVudGltZUFwaVN0YXRlXG4gIC8qKiBUaGUgcG9ydCB0aGUgUnVudGltZSBBUEkgc2VydmVyIGlzIGxpc3RlbmluZyBvbiAqL1xuICBwb3J0OiBudW1iZXJcbiAgLyoqIFRoZSBzcGF3bmVkIEJ1biBwcm9jZXNzICovXG4gIHdvcmtlclByb2Nlc3M6IENoaWxkUHJvY2Vzc1xuICAvKiogRW52aXJvbm1lbnQgdmFyaWFibGVzIGNhcHR1cmVkIGZyb20gZmlyc3QgaW52b2NhdGlvbiAqL1xuICBlbnY6IFJlY29yZDxzdHJpbmcsIHN0cmluZz5cbn1cblxuLyoqXG4gKiBJbnZvY2F0aW9uIGNvbnRleHQgZm9yIHRyYWNraW5nIGFuZCBsb2dnaW5nLlxuICovXG5leHBvcnQgaW50ZXJmYWNlIEludm9jYXRpb25Db250ZXh0IHtcbiAgLyoqIFNlcXVlbnRpYWwgaW52b2NhdGlvbiBudW1iZXIgKi9cbiAgbnVtOiBudW1iZXJcbiAgLyoqIFN0YXJ0IHRpbWVzdGFtcCBpbiBtaWxsaXNlY29uZHMgKi9cbiAgc3RhcnQ6IG51bWJlclxuICAvKiogRnVuY3Rpb24gbmFtZSAqL1xuICBmbjogc3RyaW5nXG4gIC8qKiBXaGV0aGVyIHRoaXMgaXMgYSBEb2NrZXIgY29udGFpbmVyIGludm9jYXRpb24gKi9cbiAgaXNEb2NrZXI6IGJvb2xlYW5cbn1cblxuLyoqXG4gKiBDaGVjayBpZiBib290c3RyYXAgc3RhY2sgdmVyc2lvbiBtYXRjaGVzIHRoZSBleHBlY3RlZCB2ZXJzaW9uLlxuICogUmV0dXJucyB0cnVlIGlmIHZlcnNpb24gbWF0Y2hlcywgZmFsc2UgaWYgbWlzc2luZyBvciBtaXNtYXRjaGVkLlxuICovXG5jb25zdCBjaGVja0Jvb3RzdHJhcFZlcnNpb24gPSAocXVhbGlmaWVyOiBzdHJpbmcpID0+XG4gIEVmZmVjdC5nZW4oZnVuY3Rpb24qICgpIHtcbiAgICBjb25zdCBzc21DbGllbnQgPSBuZXcgU1NNQ2xpZW50KHt9KVxuICAgIGNvbnN0IGJhc2VQYXRoID0gYCR7U1NNX0JBU0VfUEFUSH0vJHtxdWFsaWZpZXJ9YFxuXG4gICAgY29uc3QgcmVzdWx0ID0geWllbGQqIEVmZmVjdC50cnlQcm9taXNlKHtcbiAgICAgIHRyeTogYXN5bmMgKCkgPT4ge1xuICAgICAgICBjb25zdCByZXNwb25zZSA9IGF3YWl0IHNzbUNsaWVudC5zZW5kKFxuICAgICAgICAgIG5ldyBHZXRQYXJhbWV0ZXJDb21tYW5kKHsgTmFtZTogYCR7YmFzZVBhdGh9L3ZlcnNpb25gIH0pLFxuICAgICAgICApXG4gICAgICAgIHJldHVybiByZXNwb25zZS5QYXJhbWV0ZXI/LlZhbHVlXG4gICAgICB9LFxuICAgICAgY2F0Y2g6ICgpID0+IG51bGwsXG4gICAgfSkucGlwZShFZmZlY3QuY2F0Y2hBbGwoKCkgPT4gRWZmZWN0LnN1Y2NlZWQobnVsbCkpKVxuXG4gICAgaWYgKHJlc3VsdCA9PT0gbnVsbCkge1xuICAgICAgeWllbGQqIEVmZmVjdC5sb2dJbmZvKFwiQm9vdHN0cmFwIHN0YWNrIHZlcnNpb24gcGFyYW1ldGVyIG5vdCBmb3VuZFwiKVxuICAgICAgcmV0dXJuIGZhbHNlXG4gICAgfVxuXG4gICAgaWYgKHJlc3VsdCAhPT0gQk9PVFNUUkFQX1ZFUlNJT04pIHtcbiAgICAgIHlpZWxkKiBFZmZlY3QubG9nSW5mbyhcbiAgICAgICAgYEJvb3RzdHJhcCBzdGFjayB2ZXJzaW9uIG1pc21hdGNoOiBmb3VuZCAke3Jlc3VsdH0sIGV4cGVjdGVkICR7Qk9PVFNUUkFQX1ZFUlNJT059YCxcbiAgICAgIClcbiAgICAgIHJldHVybiBmYWxzZVxuICAgIH1cblxuICAgIHJldHVybiB0cnVlXG4gIH0pXG5cbi8qKlxuICogUnVuIHRoZSBib290c3RyYXAgc3RhY2sgZGVwbG95bWVudC5cbiAqL1xuY29uc3QgcnVuQm9vdHN0cmFwID0gKG9wdGlvbnM6IHsgcHJvZmlsZT86IHN0cmluZzsgcmVnaW9uPzogc3RyaW5nIH0pID0+XG4gIEVmZmVjdC5nZW4oZnVuY3Rpb24qICgpIHtcbiAgICB5aWVsZCogRWZmZWN0LmxvZ0luZm8oXCJSdW5uaW5nIGJvb3RzdHJhcCBzdGFjayBkZXBsb3ltZW50Li4uXCIpXG5cbiAgICAvLyBSZXNvbHZlIHRoZSBDREsgYXBwIHBhdGggcmVsYXRpdmUgdG8gdGhpcyBtb2R1bGVcbiAgICBjb25zdCBfX2ZpbGVuYW1lID0gZmlsZVVSTFRvUGF0aChpbXBvcnQubWV0YS51cmwpXG4gICAgY29uc3QgX19kaXJuYW1lID0gcGF0aC5kaXJuYW1lKF9fZmlsZW5hbWUpXG4gICAgY29uc3QgY2RrQXBwUGF0aCA9IHBhdGgucmVzb2x2ZShfX2Rpcm5hbWUsIFwiLi5cIiwgXCJjZGstYXBwLmpzXCIpXG5cbiAgICBjb25zdCBhcmdzID0gW1xuICAgICAgXCJjZGtcIixcbiAgICAgIFwiZGVwbG95XCIsXG4gICAgICBCT09UU1RSQVBfU1RBQ0tfTkFNRSxcbiAgICAgIFwiLS1yZXF1aXJlLWFwcHJvdmFsXCIsXG4gICAgICBcIm5ldmVyXCIsXG4gICAgICBcIi0tYXBwXCIsXG4gICAgICBgYnVuICR7Y2RrQXBwUGF0aH1gLFxuICAgIF1cblxuICAgIGlmIChvcHRpb25zLnByb2ZpbGUpIHtcbiAgICAgIGFyZ3MucHVzaChcIi0tcHJvZmlsZVwiLCBvcHRpb25zLnByb2ZpbGUpXG4gICAgfVxuXG4gICAgY29uc3QgZW52OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+ID0ge1xuICAgICAgLi4ucHJvY2Vzcy5lbnYsXG4gICAgfSBhcyBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+XG4gICAgaWYgKG9wdGlvbnMucmVnaW9uKSB7XG4gICAgICBlbnYuQVdTX1JFR0lPTiA9IG9wdGlvbnMucmVnaW9uXG4gICAgICBlbnYuQ0RLX0RFRkFVTFRfUkVHSU9OID0gb3B0aW9ucy5yZWdpb25cbiAgICB9XG5cbiAgICB5aWVsZCogRWZmZWN0LmxvZ0luZm8oYFJ1bm5pbmc6IG5weCAke2FyZ3Muam9pbihcIiBcIil9YClcblxuICAgIGNvbnN0IGNvbW1hbmQgPSBQbGF0Zm9ybUNvbW1hbmQubWFrZShcIm5weFwiLCAuLi5hcmdzKS5waXBlKFxuICAgICAgUGxhdGZvcm1Db21tYW5kLmVudihlbnYpLFxuICAgICAgUGxhdGZvcm1Db21tYW5kLnN0ZGluKFwiaW5oZXJpdFwiKSxcbiAgICApXG5cbiAgICBjb25zdCBwcm9jID0geWllbGQqIFBsYXRmb3JtQ29tbWFuZC5zdGFydChjb21tYW5kKVxuXG4gICAgLy8gU3RyZWFtIG91dHB1dCB0byBjb25zb2xlXG4gICAgeWllbGQqIFN0cmVhbS5tZXJnZShwcm9jLnN0ZG91dCwgcHJvYy5zdGRlcnIpLnBpcGUoXG4gICAgICBTdHJlYW0uZGVjb2RlVGV4dCgpLFxuICAgICAgU3RyZWFtLnNwbGl0TGluZXMsXG4gICAgICBTdHJlYW0ucnVuRm9yRWFjaCgobGluZSkgPT5cbiAgICAgICAgRWZmZWN0LnN5bmMoKCkgPT4ge1xuICAgICAgICAgIHByb2Nlc3Muc3Rkb3V0LndyaXRlKGAke2xpbmV9XFxuYClcbiAgICAgICAgfSksXG4gICAgICApLFxuICAgIClcblxuICAgIGNvbnN0IGV4aXRDb2RlID0geWllbGQqIHByb2MuZXhpdENvZGVcbiAgICBpZiAoZXhpdENvZGUgIT09IDApIHtcbiAgICAgIHJldHVybiB5aWVsZCogRWZmZWN0LmZhaWwoXG4gICAgICAgIG5ldyBFcnJvcihgQm9vdHN0cmFwIGRlcGxveW1lbnQgZmFpbGVkIHdpdGggZXhpdCBjb2RlICR7ZXhpdENvZGV9YCksXG4gICAgICApXG4gICAgfVxuXG4gICAgeWllbGQqIEVmZmVjdC5sb2dJbmZvKFwiQm9vdHN0cmFwIHN0YWNrIGRlcGxveWVkIHN1Y2Nlc3NmdWxseSFcIilcbiAgfSkucGlwZShFZmZlY3Quc2NvcGVkKVxuXG4vKipcbiAqIEVuc3VyZSBib290c3RyYXAgc3RhY2sgaXMgZGVwbG95ZWQgd2l0aCBjb3JyZWN0IHZlcnNpb24uXG4gKiBBdXRvbWF0aWNhbGx5IGRlcGxveXMgaWYgbWlzc2luZyBvciBvdXRkYXRlZC5cbiAqL1xuY29uc3QgZW5zdXJlQm9vdHN0cmFwID0gKG9wdGlvbnM6IHtcbiAgcXVhbGlmaWVyOiBzdHJpbmdcbiAgcHJvZmlsZT86IHN0cmluZ1xuICByZWdpb24/OiBzdHJpbmdcbn0pID0+XG4gIEVmZmVjdC5nZW4oZnVuY3Rpb24qICgpIHtcbiAgICB5aWVsZCogRWZmZWN0LmxvZ0RlYnVnKFwiW0xvY2FsXSBDaGVja2luZyBib290c3RyYXAgc3RhY2sgdmVyc2lvbi4uLlwiKVxuXG4gICAgY29uc3QgdmVyc2lvbk9rID0geWllbGQqIGNoZWNrQm9vdHN0cmFwVmVyc2lvbihvcHRpb25zLnF1YWxpZmllcilcblxuICAgIGlmICghdmVyc2lvbk9rKSB7XG4gICAgICB5aWVsZCogRWZmZWN0LmxvZ0luZm8oXG4gICAgICAgIFwiW0xvY2FsXSBCb290c3RyYXAgc3RhY2sgbmVlZHMgdG8gYmUgZGVwbG95ZWQgb3IgdXBkYXRlZC5cIixcbiAgICAgIClcbiAgICAgIHlpZWxkKiBydW5Cb290c3RyYXAoeyBwcm9maWxlOiBvcHRpb25zLnByb2ZpbGUsIHJlZ2lvbjogb3B0aW9ucy5yZWdpb24gfSlcbiAgICB9IGVsc2Uge1xuICAgICAgeWllbGQqIEVmZmVjdC5sb2dEZWJ1ZyhcIltMb2NhbF0gQm9vdHN0cmFwIHN0YWNrIHZlcnNpb24gT0suXCIpXG4gICAgfVxuICB9KVxuXG4vKipcbiAqIFJlYWQgQXBwU3luYyBlbmRwb2ludHMgZnJvbSBTU00uXG4gKi9cbmNvbnN0IGdldEFwcFN5bmNFbmRwb2ludHMgPSAocXVhbGlmaWVyOiBzdHJpbmcpID0+XG4gIEVmZmVjdC5nZW4oZnVuY3Rpb24qICgpIHtcbiAgICBjb25zdCBzc21DbGllbnQgPSBuZXcgU1NNQ2xpZW50KHt9KVxuICAgIGNvbnN0IGJhc2VQYXRoID0gYCR7U1NNX0JBU0VfUEFUSH0vJHtxdWFsaWZpZXJ9YFxuXG4gICAgY29uc3QgZ2V0UGFyYW0gPSAobmFtZTogc3RyaW5nKSA9PlxuICAgICAgRWZmZWN0LnRyeVByb21pc2Uoe1xuICAgICAgICB0cnk6IGFzeW5jICgpID0+IHtcbiAgICAgICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBzc21DbGllbnQuc2VuZChcbiAgICAgICAgICAgIG5ldyBHZXRQYXJhbWV0ZXJDb21tYW5kKHsgTmFtZTogYCR7YmFzZVBhdGh9LyR7bmFtZX1gIH0pLFxuICAgICAgICAgIClcbiAgICAgICAgICByZXR1cm4gcmVzdWx0LlBhcmFtZXRlcj8uVmFsdWUgPz8gXCJcIlxuICAgICAgICB9LFxuICAgICAgICBjYXRjaDogKGVycm9yKSA9PlxuICAgICAgICAgIG5ldyBFcnJvcihgRmFpbGVkIHRvIGdldCBTU00gcGFyYW1ldGVyICR7bmFtZX06ICR7U3RyaW5nKGVycm9yKX1gKSxcbiAgICAgIH0pXG5cbiAgICBjb25zdCBodHRwRW5kcG9pbnQgPSB5aWVsZCogZ2V0UGFyYW0oXCJodHRwLWVuZHBvaW50XCIpXG4gICAgY29uc3QgcmVhbHRpbWVFbmRwb2ludCA9IHlpZWxkKiBnZXRQYXJhbShcInJlYWx0aW1lLWVuZHBvaW50XCIpXG5cbiAgICByZXR1cm4geyBodHRwRW5kcG9pbnQsIHJlYWx0aW1lRW5kcG9pbnQgfVxuICB9KVxuXG4vKipcbiAqIENsb3VkRm9ybWF0aW9uIHN0YWNrIG5hbWUgdGFnIChzZXQgYXV0b21hdGljYWxseSBieSBDREspXG4gKi9cbmNvbnN0IENGTl9TVEFDS19OQU1FX1RBRyA9IFwiYXdzOmNsb3VkZm9ybWF0aW9uOnN0YWNrLW5hbWVcIlxuXG4vKipcbiAqIERpc2NvdmVyIExhbWJkYSBmdW5jdGlvbnMgd2l0aCBsaXZlLWxhbWJkYSB0YWdzLlxuICogQHBhcmFtIHN0YWNrRmlsdGVyIC0gT3B0aW9uYWwgbGlzdCBvZiBzdGFjayBuYW1lcyB0byBmaWx0ZXIgYnlcbiAqL1xuY29uc3QgZGlzY292ZXJGdW5jdGlvbnMgPSAoc3RhY2tGaWx0ZXI/OiBzdHJpbmdbXSkgPT5cbiAgRWZmZWN0LmdlbihmdW5jdGlvbiogKCkge1xuICAgIGNvbnN0IGxhbWJkYUNsaWVudCA9IG5ldyBMYW1iZGFDbGllbnQoe30pXG4gICAgY29uc3QgZnVuY3Rpb25zOiBEaXNjb3ZlcmVkRnVuY3Rpb25bXSA9IFtdXG5cbiAgICAvLyBMaXN0IGFsbCBmdW5jdGlvbnNcbiAgICBsZXQgbmV4dE1hcmtlcjogc3RyaW5nIHwgdW5kZWZpbmVkXG4gICAgZG8ge1xuICAgICAgY29uc3QgbGlzdFJlc3BvbnNlID0geWllbGQqIEVmZmVjdC50cnlQcm9taXNlKHtcbiAgICAgICAgdHJ5OiAoKSA9PlxuICAgICAgICAgIGxhbWJkYUNsaWVudC5zZW5kKG5ldyBMaXN0RnVuY3Rpb25zQ29tbWFuZCh7IE1hcmtlcjogbmV4dE1hcmtlciB9KSksXG4gICAgICAgIGNhdGNoOiAoZXJyb3IpID0+XG4gICAgICAgICAgbmV3IEVycm9yKGBGYWlsZWQgdG8gbGlzdCBmdW5jdGlvbnM6ICR7U3RyaW5nKGVycm9yKX1gKSxcbiAgICAgIH0pXG5cbiAgICAgIGZvciAoY29uc3QgZm4gb2YgbGlzdFJlc3BvbnNlLkZ1bmN0aW9ucyA/PyBbXSkge1xuICAgICAgICAvLyBHZXQgdGFncyBmb3IgZWFjaCBmdW5jdGlvblxuICAgICAgICBjb25zdCB0YWdzUmVzdWx0ID0geWllbGQqIEVmZmVjdC50cnlQcm9taXNlKHtcbiAgICAgICAgICB0cnk6ICgpID0+XG4gICAgICAgICAgICBsYW1iZGFDbGllbnQuc2VuZChcbiAgICAgICAgICAgICAgbmV3IExpc3RUYWdzQ29tbWFuZCh7IFJlc291cmNlOiBmbi5GdW5jdGlvbkFybiB9KSxcbiAgICAgICAgICAgICksXG4gICAgICAgICAgY2F0Y2g6ICgpID0+IG5ldyBFcnJvcihgRmFpbGVkIHRvIGdldCB0YWdzIGZvciAke2ZuLkZ1bmN0aW9uTmFtZX1gKSxcbiAgICAgICAgfSkucGlwZShcbiAgICAgICAgICBFZmZlY3QuY2F0Y2hBbGwoKCkgPT5cbiAgICAgICAgICAgIEVmZmVjdC5zdWNjZWVkKHsgVGFnczoge30gYXMgUmVjb3JkPHN0cmluZywgc3RyaW5nPiB9KSxcbiAgICAgICAgICApLFxuICAgICAgICApXG5cbiAgICAgICAgY29uc3QgdGFncyA9IHRhZ3NSZXN1bHQuVGFncyA/PyB7fVxuXG4gICAgICAgIC8vIENoZWNrIGZvciBsaXZlLWxhbWJkYSB0YWcgKGhhbmRsZXIpIG9yIGRvY2tlci1jb250ZXh0IHRhZ1xuICAgICAgICBpZiAodGFnc1tMSVZFX0xBTUJEQV9UQUddIHx8IHRhZ3NbTElWRV9MQU1CREFfRE9DS0VSX1RBR10pIHtcbiAgICAgICAgICAvLyBGaWx0ZXIgYnkgc3RhY2sgbmFtZSBpZiBzcGVjaWZpZWRcbiAgICAgICAgICBpZiAoc3RhY2tGaWx0ZXIgJiYgc3RhY2tGaWx0ZXIubGVuZ3RoID4gMCkge1xuICAgICAgICAgICAgY29uc3Qgc3RhY2tOYW1lID0gdGFnc1tDRk5fU1RBQ0tfTkFNRV9UQUddXG4gICAgICAgICAgICBpZiAoIXN0YWNrTmFtZSB8fCAhc3RhY2tGaWx0ZXIuaW5jbHVkZXMoc3RhY2tOYW1lKSkge1xuICAgICAgICAgICAgICBjb250aW51ZVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIERldGVybWluZSBhcmNoaXRlY3R1cmUgZnJvbSBMYW1iZGEgY29uZmlnXG4gICAgICAgICAgY29uc3QgYXJjaGl0ZWN0dXJlcyA9IGZuLkFyY2hpdGVjdHVyZXMgPz8gW1wieDg2XzY0XCJdXG4gICAgICAgICAgY29uc3QgYXJjaGl0ZWN0dXJlID0gYXJjaGl0ZWN0dXJlcy5pbmNsdWRlcyhcImFybTY0XCIpXG4gICAgICAgICAgICA/IChcImFybTY0XCIgYXMgY29uc3QpXG4gICAgICAgICAgICA6IChcIng4Nl82NFwiIGFzIGNvbnN0KVxuXG4gICAgICAgICAgY29uc3QgZGlzY292ZXJlZDogRGlzY292ZXJlZEZ1bmN0aW9uID0ge1xuICAgICAgICAgICAgZnVuY3Rpb25OYW1lOiBmbi5GdW5jdGlvbk5hbWUhLFxuICAgICAgICAgICAgZnVuY3Rpb25Bcm46IGZuLkZ1bmN0aW9uQXJuISxcbiAgICAgICAgICAgIGxvY2FsSGFuZGxlcjogdGFnc1tMSVZFX0xBTUJEQV9UQUddID8/IFwiXCIsXG4gICAgICAgICAgICBkb2NrZXJDb250ZXh0UGF0aDogdGFnc1tMSVZFX0xBTUJEQV9ET0NLRVJfVEFHXSxcbiAgICAgICAgICAgIG1lbW9yeU1COiBmbi5NZW1vcnlTaXplID8/IDEyOCxcbiAgICAgICAgICAgIGFyY2hpdGVjdHVyZSxcbiAgICAgICAgICB9XG4gICAgICAgICAgZnVuY3Rpb25zLnB1c2goZGlzY292ZXJlZClcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBuZXh0TWFya2VyID0gbGlzdFJlc3BvbnNlLk5leHRNYXJrZXJcbiAgICB9IHdoaWxlIChuZXh0TWFya2VyKVxuXG4gICAgcmV0dXJuIGZ1bmN0aW9uc1xuICB9KVxuXG4vKipcbiAqIFN0YXJ0IGEgbG9uZy1ydW5uaW5nIERvY2tlciBjb250YWluZXIgZm9yIGEgZnVuY3Rpb24uXG4gKiBCdWlsZHMgdGhlIGltYWdlIGZyb20gdGhlIGxvY2FsIERvY2tlciBjb250ZXh0IHBhdGgsIHRoZW4gcnVucyBpdC5cbiAqIFRoZSBjb250YWluZXIgY29udGludW91c2x5IHBvbGxzIG91ciBSdW50aW1lIEFQSSBmb3IgaW52b2NhdGlvbnMuXG4gKiBVc2VzIEVmZmVjdC5mb3JrRGFlbW9uIHRvIGVuc3VyZSB0aGUgY29udGFpbmVyIHJ1bnMgaW5kZXBlbmRlbnRseSB3aXRoIGNvbnRleHQuXG4gKi9cbmNvbnN0IHN0YXJ0RnVuY3Rpb25Db250YWluZXIgPSAoXG4gIGZuOiBEaXNjb3ZlcmVkRnVuY3Rpb24sXG4gIHBvcnQ6IG51bWJlcixcbiAgcHJvamVjdFJvb3Q6IHN0cmluZyxcbiAgYWRkaXRpb25hbEVudjogUmVjb3JkPHN0cmluZywgc3RyaW5nPixcbiAgaW52b2NhdGlvbkNvbnRleHRzPzogTWFwPHN0cmluZywgeyBudW06IG51bWJlciB9Pixcbik6IEVmZmVjdC5FZmZlY3Q8XG4gIEZpYmVyLlJ1bnRpbWVGaWJlcjx2b2lkLCBFcnJvcj4sXG4gIEVycm9yLFxuICBDb21tYW5kRXhlY3V0b3IuQ29tbWFuZEV4ZWN1dG9yXG4+ID0+XG4gIEVmZmVjdC5nZW4oZnVuY3Rpb24qICgpIHtcbiAgICBpZiAoIWZuLmRvY2tlckNvbnRleHRQYXRoKSB7XG4gICAgICByZXR1cm4geWllbGQqIEVmZmVjdC5mYWlsKFxuICAgICAgICBuZXcgRXJyb3IoYEZ1bmN0aW9uICR7Zm4uZnVuY3Rpb25OYW1lfSBoYXMgbm8gRG9ja2VyIGNvbnRleHQgcGF0aGApLFxuICAgICAgKVxuICAgIH1cblxuICAgIGNvbnN0IGRvY2tlciA9IHlpZWxkKiBEb2NrZXJcblxuICAgIGNvbnN0IGRvY2tlclJ1bnRpbWUgPSB5aWVsZCogZG9ja2VyLmdldFJ1bnRpbWVJbmZvKClcbiAgICBjb25zdCBydW50aW1lQXBpSG9zdCA9IGRvY2tlclJ1bnRpbWUuaXNEb2NrZXJEZXNrdG9wXG4gICAgICA/IFwiaG9zdC5kb2NrZXIuaW50ZXJuYWxcIlxuICAgICAgOiBcInJ1bnRpbWUuYXBpXCJcblxuICAgIC8vIEdlbmVyYXRlIGEgbG9jYWwgaW1hZ2UgbmFtZSBmcm9tIGZ1bmN0aW9uIG5hbWVcbiAgICBjb25zdCBpbWFnZU5hbWUgPSBgbGl2ZS1sYW1iZGEtJHtmbi5mdW5jdGlvbk5hbWUudG9Mb3dlckNhc2UoKS5yZXBsYWNlKC9bXmEtejAtOS1dL2csIFwiLVwiKX1gXG5cbiAgICAvLyBSZXNvbHZlIHRoZSBjb250ZXh0IHBhdGggcmVsYXRpdmUgdG8gcHJvamVjdCByb290XG4gICAgY29uc3QgY29udGV4dFBhdGggPSBmbi5kb2NrZXJDb250ZXh0UGF0aC5zdGFydHNXaXRoKFwiL1wiKVxuICAgICAgPyBmbi5kb2NrZXJDb250ZXh0UGF0aFxuICAgICAgOiBgJHtwcm9qZWN0Um9vdH0vJHtmbi5kb2NrZXJDb250ZXh0UGF0aH1gXG5cbiAgICAvLyBEZXRlcm1pbmUgcGxhdGZvcm0gZnJvbSBhcmNoaXRlY3R1cmVcbiAgICBjb25zdCBwbGF0Zm9ybSA9IGZuLmFyY2hpdGVjdHVyZSA9PT0gXCJhcm02NFwiID8gXCJsaW51eC9hcm02NFwiIDogXCJsaW51eC9hbWQ2NFwiXG5cbiAgICAvLyBCdWlsZCB0aGUgRG9ja2VyIGltYWdlIGZyb20gbG9jYWwgY29udGV4dFxuICAgIHlpZWxkKiBkb2NrZXJcbiAgICAgIC5idWlsZCh7XG4gICAgICAgIGNvbnRleHRQYXRoLFxuICAgICAgICBpbWFnZU5hbWUsXG4gICAgICAgIHBsYXRmb3JtLFxuICAgICAgfSlcbiAgICAgIC5waXBlKEVmZmVjdC5zY29wZWQpXG5cbiAgICAvLyBJbnNwZWN0IHRoZSBpbWFnZSB0byBnZXQgb3JpZ2luYWwgZW50cnlwb2ludC9jbWQgZm9yIGV4dGVuc2lvbiB3cmFwcGVyXG4gICAgY29uc3QgaW1hZ2VDb25maWcgPSB5aWVsZCogZG9ja2VyLmluc3BlY3QoaW1hZ2VOYW1lKS5waXBlKEVmZmVjdC5zY29wZWQpXG4gICAgY29uc3QgZXh0ZW5zaW9uV3JhcHBlciA9IGJ1aWxkRXh0ZW5zaW9uV3JhcHBlckNvbW1hbmQoXG4gICAgICBpbWFnZUNvbmZpZy5lbnRyeXBvaW50LFxuICAgICAgaW1hZ2VDb25maWcuY21kLFxuICAgIClcblxuICAgIGNvbnN0IGNvbnRhaW5lckNvbmZpZyA9IG1ha2VMYW1iZGFDb250YWluZXJDb25maWcoe1xuICAgICAgaW1hZ2VVcmk6IGltYWdlTmFtZSxcbiAgICAgIHJ1bnRpbWVBcGlIb3N0LFxuICAgICAgcnVudGltZUFwaVBvcnQ6IHBvcnQsXG4gICAgICBmdW5jdGlvbk5hbWU6IGZuLmZ1bmN0aW9uTmFtZSxcbiAgICAgIGZ1bmN0aW9uVmVyc2lvbjogXCIkTEFURVNUXCIsXG4gICAgICBtZW1vcnlNQjogZm4ubWVtb3J5TUIsXG4gICAgICB0aW1lb3V0U2Vjb25kczogMzYwMCwgLy8gTG9uZyB0aW1lb3V0IC0gY29udGFpbmVyIHN0YXlzIHJ1bm5pbmdcbiAgICAgIHBsYXRmb3JtLFxuICAgICAgYWRkaXRpb25hbEVudixcbiAgICAgIGludm9jYXRpb25Db250ZXh0cyxcbiAgICB9KVxuXG4gICAgLy8gQXBwbHkgZXh0ZW5zaW9uIHdyYXBwZXIgdG8gc3RhcnQgZXh0ZW5zaW9ucyBiZWZvcmUgdGhlIGFwcFxuICAgIGNvbnRhaW5lckNvbmZpZy5lbnRyeXBvaW50ID0gZXh0ZW5zaW9uV3JhcHBlci5lbnRyeXBvaW50XG4gICAgY29udGFpbmVyQ29uZmlnLmNvbW1hbmQgPSBleHRlbnNpb25XcmFwcGVyLmNvbW1hbmRcblxuICAgIHlpZWxkKiBFZmZlY3QubG9nSW5mbyhcbiAgICAgIGBTdGFydGluZyBjb250YWluZXIgZm9yICR7Zm4uZnVuY3Rpb25OYW1lfSBvbiBwb3J0ICR7cG9ydH1gLFxuICAgIClcblxuICAgIC8vIFVzZSBFZmZlY3QuZm9ya0RhZW1vbiB0byBydW4gdGhlIGNvbnRhaW5lciBpbmRlcGVuZGVudGx5IHdpdGggY29udGV4dCBwcmVzZXJ2ZWRcbiAgICAvLyBFZmZlY3Quc2NvcGVkIHByb3ZpZGVzIHRoZSBzY29wZSBuZWVkZWQgYnkgZG9ja2VyLnJ1blxuICAgIGNvbnN0IGZpYmVyID0geWllbGQqIGRvY2tlci5ydW4oY29udGFpbmVyQ29uZmlnKS5waXBlKFxuICAgICAgRWZmZWN0LnNjb3BlZCxcbiAgICAgIEVmZmVjdC5tYXAoKCkgPT4gdW5kZWZpbmVkIGFzIHZvaWQpLFxuICAgICAgRWZmZWN0LmZvcmtEYWVtb24sXG4gICAgKVxuXG4gICAgcmV0dXJuIGZpYmVyXG4gIH0pLnBpcGUoRWZmZWN0LnByb3ZpZGUoRG9ja2VyTGl2ZSkpXG5cbi8vIFR5cGUgZ3VhcmRzIGZvciByZXNwb25zZSB0eXBlc1xuY29uc3QgaXNMYW1iZGFSZXNwb25zZSA9IChcbiAgcjogTGFtYmRhUmVzcG9uc2UgfCBMYW1iZGFFcnJvciB8IExhbWJkYUluaXRFcnJvcixcbik6IHIgaXMgTGFtYmRhUmVzcG9uc2UgPT4gXCJib2R5XCIgaW4gciAmJiAhKFwiZXJyb3JUeXBlXCIgaW4gcilcblxuY29uc3QgaXNMYW1iZGFFcnJvciA9IChcbiAgcjogTGFtYmRhUmVzcG9uc2UgfCBMYW1iZGFFcnJvciB8IExhbWJkYUluaXRFcnJvcixcbik6IHIgaXMgTGFtYmRhRXJyb3IgPT4gXCJyZXF1ZXN0SWRcIiBpbiByICYmIFwiZXJyb3JUeXBlXCIgaW4gclxuXG5jb25zdCBpc0xhbWJkYUluaXRFcnJvciA9IChcbiAgcjogTGFtYmRhUmVzcG9uc2UgfCBMYW1iZGFFcnJvciB8IExhbWJkYUluaXRFcnJvcixcbik6IHIgaXMgTGFtYmRhSW5pdEVycm9yID0+ICEoXCJyZXF1ZXN0SWRcIiBpbiByKSAmJiBcImVycm9yVHlwZVwiIGluIHJcblxuLyoqXG4gKiBSZXNldCB0aGUgaWRsZSB0aW1lciBmb3IgYSBjb250YWluZXIuXG4gKiBBZnRlciB0aGUgdGltZW91dCBleHBpcmVzIHdpdGggbm8gbmV3IHJlc3BvbnNlcywgdGhlIGNvbnRhaW5lciBpcyBzdG9wcGVkLlxuICogVGhpcyBwcmV2ZW50cyB0aGUgY29uZnVzaW5nIDUwMyBlcnJvciBmcm9tIHRoZSBMYW1iZGEgUklDIHdoZW4gdGhlIHBvbGwgdGltZXMgb3V0LlxuICovXG5jb25zdCByZXNldElkbGVUaW1lciA9IChcbiAgY29udGFpbmVyOiBGdW5jdGlvbkNvbnRhaW5lcixcbiAgY29udGFpbmVyc1JlZjogUmVmLlJlZjxNYXA8c3RyaW5nLCBGdW5jdGlvbkNvbnRhaW5lcj4+LFxuICBpZGxlVGltZW91dE1zOiBudW1iZXIsXG4pOiBFZmZlY3QuRWZmZWN0PHZvaWQsIG5ldmVyLCBDb21tYW5kRXhlY3V0b3IuQ29tbWFuZEV4ZWN1dG9yPiA9PlxuICBFZmZlY3QuZ2VuKGZ1bmN0aW9uKiAoKSB7XG4gICAgLy8gQ2FuY2VsIGV4aXN0aW5nIHRpbWVyIGlmIGFueVxuICAgIGlmIChjb250YWluZXIuaWRsZVRpbWVyRmliZXIpIHtcbiAgICAgIHlpZWxkKiBGaWJlci5pbnRlcnJ1cHQoY29udGFpbmVyLmlkbGVUaW1lckZpYmVyKS5waXBlKFxuICAgICAgICBFZmZlY3QuY2F0Y2hBbGwoKCkgPT4gRWZmZWN0LnZvaWQpLFxuICAgICAgKVxuICAgICAgY29udGFpbmVyLmlkbGVUaW1lckZpYmVyID0gbnVsbFxuICAgIH1cblxuICAgIC8vIFN0YXJ0IG5ldyB0aW1lciAtIHVzZSBuZXZlciBmb3IgZXJyb3IgdHlwZSBzaW5jZSB3ZSBjYXRjaCBhbGwgZXJyb3JzXG4gICAgY29uc3QgdGltZXJGaWJlcjogRmliZXIuUnVudGltZUZpYmVyPHZvaWQsIG5ldmVyPiA9IHlpZWxkKiBFZmZlY3QuZ2VuKFxuICAgICAgZnVuY3Rpb24qICgpIHtcbiAgICAgICAgeWllbGQqIEVmZmVjdC5zbGVlcChEdXJhdGlvbi5taWxsaXMoaWRsZVRpbWVvdXRNcykpXG5cbiAgICAgICAgLy8gVGltZXIgZXhwaXJlZCAtIHN0b3AgY29udGFpbmVyIHByb2FjdGl2ZWx5XG4gICAgICAgIHlpZWxkKiBFZmZlY3QubG9nSW5mbyhcbiAgICAgICAgICBgW0xvY2FsXSBTdG9wcGluZyBpZGxlIGNvbnRhaW5lciAke2NvbnRhaW5lci5mbi5mdW5jdGlvbk5hbWV9ICh3aWxsIHJlc3RhcnQgb24gbmV4dCBpbnZvY2F0aW9uKWAsXG4gICAgICAgIClcblxuICAgICAgICAvLyBTdG9wIHRoZSBjb250YWluZXIgdXNpbmcgRG9ja2VyIHdpdGggZmFzdCB0aW1lb3V0ICgxcylcbiAgICAgICAgLy8gVGhlIGNvbnRhaW5lciBtYXkgYmUgYmxvY2tlZCBvbiBsb25nLXBvbGxpbmcsIHNvIHdlIG5lZWQgdG8gZm9yY2Uga2lsbFxuICAgICAgICBjb25zdCBkb2NrZXIgPSB5aWVsZCogRG9ja2VyXG4gICAgICAgIHlpZWxkKiBkb2NrZXIuc3RvcChjb250YWluZXIuY29udGFpbmVyTmFtZSwgMSkucGlwZShcbiAgICAgICAgICBFZmZlY3Quc2NvcGVkLFxuICAgICAgICAgIEVmZmVjdC5jYXRjaEFsbCgoKSA9PiBFZmZlY3Qudm9pZCksXG4gICAgICAgIClcblxuICAgICAgICAvLyBSZW1vdmUgZnJvbSBtYXAgc28gbmV4dCBpbnZvY2F0aW9uIGNyZWF0ZXMgZnJlc2ggY29udGFpbmVyXG4gICAgICAgIGNvbnN0IGN1cnJlbnRDb250YWluZXJzID0geWllbGQqIFJlZi5nZXQoY29udGFpbmVyc1JlZilcbiAgICAgICAgY3VycmVudENvbnRhaW5lcnMuZGVsZXRlKGNvbnRhaW5lci5mbi5mdW5jdGlvbk5hbWUpXG4gICAgICAgIHlpZWxkKiBSZWYuc2V0KGNvbnRhaW5lcnNSZWYsIGN1cnJlbnRDb250YWluZXJzKVxuICAgICAgfSxcbiAgICApLnBpcGUoXG4gICAgICBFZmZlY3QucHJvdmlkZShEb2NrZXJMaXZlKSxcbiAgICAgIEVmZmVjdC5jYXRjaEFsbCgoKSA9PiBFZmZlY3Qudm9pZCksXG4gICAgICBFZmZlY3QuZm9ya0RhZW1vbixcbiAgICApXG5cbiAgICBjb250YWluZXIuaWRsZVRpbWVyRmliZXIgPSB0aW1lckZpYmVyXG4gIH0pXG5cbi8qKlxuICogUHJvY2VzcyByZXNwb25zZXMgZnJvbSBhIGNvbnRhaW5lciBhbmQgZGlzcGF0Y2ggdG8gd2FpdGluZyBjYWxsZXJzLlxuICogQWZ0ZXIgZWFjaCByZXNwb25zZSwgcmVzZXRzIHRoZSBpZGxlIHRpbWVyIHRvIHByb2FjdGl2ZWx5IHN0b3AgdGhlIGNvbnRhaW5lclxuICogYmVmb3JlIHRoZSBwb2xsIHRpbWVvdXQgY2F1c2VzIGNvbmZ1c2luZyBMYW1iZGEgUklDIGVycm9ycy5cbiAqL1xuY29uc3QgcHJvY2Vzc0NvbnRhaW5lclJlc3BvbnNlcyA9IChcbiAgY29udGFpbmVyOiBGdW5jdGlvbkNvbnRhaW5lcixcbiAgY29udGFpbmVyc1JlZjogUmVmLlJlZjxNYXA8c3RyaW5nLCBGdW5jdGlvbkNvbnRhaW5lcj4+LFxuICBpZGxlVGltZW91dE1zOiBudW1iZXIsXG4gIGNsaWVudDogUmV0dXJuVHlwZTx0eXBlb2YgbWFrZUFwcFN5bmNDbGllbnQ+LFxuICBpbnZvY2F0aW9uQ29udGV4dHM6IE1hcDxzdHJpbmcsIEludm9jYXRpb25Db250ZXh0Pixcbik6IEVmZmVjdC5FZmZlY3Q8dm9pZCwgRXJyb3IsIENvbW1hbmRFeGVjdXRvci5Db21tYW5kRXhlY3V0b3I+ID0+XG4gIEVmZmVjdC5nZW4oZnVuY3Rpb24qICgpIHtcbiAgICBjb25zdCByZXNwb25zZUNoYW5uZWwgPSBidWlsZENoYW5uZWxOYW1lLnJlc3BvbnNlKGNvbnRhaW5lci5mbi5mdW5jdGlvbk5hbWUpXG5cbiAgICAvLyBDb250aW51b3VzbHkgcHJvY2VzcyByZXNwb25zZXMgZnJvbSB0aGUgY29udGFpbmVyXG4gICAgd2hpbGUgKHRydWUpIHtcbiAgICAgIGNvbnN0IHJlc3VsdCA9IHlpZWxkKiB3YWl0Rm9yUmVzcG9uc2UoY29udGFpbmVyLnJ1bnRpbWVTdGF0ZSlcblxuICAgICAgbGV0IHJlc3BvbnNlOiBSZXNwb25zZU1lc3NhZ2VcbiAgICAgIGlmIChpc0xhbWJkYVJlc3BvbnNlKHJlc3VsdCkpIHtcbiAgICAgICAgcmVzcG9uc2UgPSB7XG4gICAgICAgICAgdHlwZTogXCJyZXNwb25zZVwiLFxuICAgICAgICAgIHJlcXVlc3RJZDogcmVzdWx0LnJlcXVlc3RJZCxcbiAgICAgICAgICByZXN1bHQ6IHJlc3VsdC5ib2R5LFxuICAgICAgICB9XG4gICAgICB9IGVsc2UgaWYgKGlzTGFtYmRhRXJyb3IocmVzdWx0KSkge1xuICAgICAgICByZXNwb25zZSA9IHtcbiAgICAgICAgICB0eXBlOiBcInJlc3BvbnNlXCIsXG4gICAgICAgICAgcmVxdWVzdElkOiByZXN1bHQucmVxdWVzdElkLFxuICAgICAgICAgIGVycm9yOiB7XG4gICAgICAgICAgICBlcnJvclR5cGU6IHJlc3VsdC5lcnJvclR5cGUsXG4gICAgICAgICAgICBlcnJvck1lc3NhZ2U6IHJlc3VsdC5lcnJvck1lc3NhZ2UsXG4gICAgICAgICAgICBzdGFja1RyYWNlOiByZXN1bHQuc3RhY2tUcmFjZSxcbiAgICAgICAgICB9LFxuICAgICAgICB9XG4gICAgICB9IGVsc2UgaWYgKGlzTGFtYmRhSW5pdEVycm9yKHJlc3VsdCkpIHtcbiAgICAgICAgeWllbGQqIEVmZmVjdC5sb2dFcnJvcihcbiAgICAgICAgICBgW0xvY2FsXSBJbml0IGVycm9yOiAke3Jlc3VsdC5lcnJvclR5cGV9OiAke3Jlc3VsdC5lcnJvck1lc3NhZ2V9YCxcbiAgICAgICAgKVxuICAgICAgICBjb250aW51ZVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgY29udGludWVcbiAgICAgIH1cblxuICAgICAgLy8gU2VuZCByZXNwb25zZSBiYWNrIHZpYSBBcHBTeW5jXG4gICAgICB5aWVsZCogY2xpZW50LnB1Ymxpc2hSZXNwb25zZShyZXNwb25zZUNoYW5uZWwsIHJlc3BvbnNlKVxuICAgICAgeWllbGQqIEVmZmVjdC5sb2dEZWJ1ZyhgW0xvY2FsXSBTZW50IHJlc3BvbnNlIGZvciAke3Jlc3BvbnNlLnJlcXVlc3RJZH1gKVxuXG4gICAgICAvLyBQcmludCBlbmQgbWFya2VyIHdpdGggdGltaW5nXG4gICAgICBjb25zdCBjdHggPSBpbnZvY2F0aW9uQ29udGV4dHMuZ2V0KHJlc3BvbnNlLnJlcXVlc3RJZClcbiAgICAgIGlmIChjdHgpIHtcbiAgICAgICAgY29uc3QgZHVyYXRpb25NcyA9IERhdGUubm93KCkgLSBjdHguc3RhcnRcbiAgICAgICAgaWYgKHJlc3BvbnNlLmVycm9yKSB7XG4gICAgICAgICAgY29uc29sZS5sb2coXG4gICAgICAgICAgICBgWyR7Y3R4Lm51bX1dIFxcdTI1MTRcXHUyNTAwXFx1MjUwMCBcXHUyNzE3ICR7cmVzcG9uc2UuZXJyb3IuZXJyb3JUeXBlfTogJHtyZXNwb25zZS5lcnJvci5lcnJvck1lc3NhZ2V9ICgke2R1cmF0aW9uTXN9bXMpIFxcdTI1MDBcXHUyNTAwYCxcbiAgICAgICAgICApXG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgY29uc29sZS5sb2coXG4gICAgICAgICAgICBgWyR7Y3R4Lm51bX1dIFxcdTI1MTRcXHUyNTAwXFx1MjUwMCBcXHUyNzEzIERvbmUgKCR7ZHVyYXRpb25Nc31tcykgXFx1MjUwMFxcdTI1MDBgLFxuICAgICAgICAgIClcbiAgICAgICAgfVxuICAgICAgICBpbnZvY2F0aW9uQ29udGV4dHMuZGVsZXRlKHJlc3BvbnNlLnJlcXVlc3RJZClcbiAgICAgIH1cblxuICAgICAgLy8gUmVzZXQgaWRsZSB0aW1lciAtIGNvbnRhaW5lciB3aWxsIGJlIHN0b3BwZWQgaWYgbm8gbmV3IGludm9jYXRpb25zIGFycml2ZVxuICAgICAgeWllbGQqIHJlc2V0SWRsZVRpbWVyKGNvbnRhaW5lciwgY29udGFpbmVyc1JlZiwgaWRsZVRpbWVvdXRNcylcbiAgICB9XG4gIH0pXG5cbi8qKlxuICogU3RhcnQgYSBOb2RlLmpzIHdvcmtlciBwcm9jZXNzIGZvciBhIGZ1bmN0aW9uLlxuICogU3Bhd25zIEJ1biB0byBydW4gdGhlIHJ1bnRpbWUgd3JhcHBlciB3aXRoIHRoZSBoYW5kbGVyIHBhdGguXG4gKiBUaGUgd29ya2VyIGNvbnRpbnVvdXNseSBwb2xscyBvdXIgUnVudGltZSBBUEkgZm9yIGludm9jYXRpb25zLlxuICovXG5jb25zdCBzdGFydE5vZGVqc1dvcmtlciA9IChcbiAgZm46IERpc2NvdmVyZWRGdW5jdGlvbixcbiAgcG9ydDogbnVtYmVyLFxuICBwcm9qZWN0Um9vdDogc3RyaW5nLFxuICBlbnY6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4sXG4gIGludm9jYXRpb25Db250ZXh0czogTWFwPHN0cmluZywgSW52b2NhdGlvbkNvbnRleHQ+LFxuKTogRWZmZWN0LkVmZmVjdDxDaGlsZFByb2Nlc3MsIEVycm9yPiA9PlxuICBFZmZlY3QuZ2VuKGZ1bmN0aW9uKiAoKSB7XG4gICAgaWYgKCFmbi5sb2NhbEhhbmRsZXIpIHtcbiAgICAgIHJldHVybiB5aWVsZCogRWZmZWN0LmZhaWwoXG4gICAgICAgIG5ldyBFcnJvcihgRnVuY3Rpb24gJHtmbi5mdW5jdGlvbk5hbWV9IGhhcyBubyBsb2NhbCBoYW5kbGVyIHBhdGhgKSxcbiAgICAgIClcbiAgICB9XG5cbiAgICAvLyBSZXNvbHZlIHRoZSBydW50aW1lIHdyYXBwZXIgcGF0aCByZWxhdGl2ZSB0byB0aGlzIG1vZHVsZVxuICAgIGNvbnN0IF9fZmlsZW5hbWUgPSBmaWxlVVJMVG9QYXRoKGltcG9ydC5tZXRhLnVybClcbiAgICBjb25zdCBfX2Rpcm5hbWUgPSBwYXRoLmRpcm5hbWUoX19maWxlbmFtZSlcbiAgICBjb25zdCBydW50aW1lV3JhcHBlclBhdGggPSBwYXRoLnJlc29sdmUoXG4gICAgICBfX2Rpcm5hbWUsXG4gICAgICBcIi4uXCIsXG4gICAgICBcInJ1bnRpbWUtd3JhcHBlclwiLFxuICAgICAgXCJub2RlanMtcnVudGltZS5qc1wiLFxuICAgIClcblxuICAgIC8vIEdldCBhYnNvbHV0ZSBwYXRoIHRvIGJ1biBmb3IgcHVyZSBlbnYgaXNvbGF0aW9uIChubyBQQVRIIGRlcGVuZGVuY3kpXG4gICAgY29uc3QgYnVuUGF0aCA9IEJ1bi53aGljaChcImJ1blwiKVxuICAgIGlmICghYnVuUGF0aCkge1xuICAgICAgcmV0dXJuIHlpZWxkKiBFZmZlY3QuZmFpbChcbiAgICAgICAgbmV3IEVycm9yKFwiQ291bGQgbm90IGZpbmQgJ2J1bicgZXhlY3V0YWJsZSBpbiBQQVRIXCIpLFxuICAgICAgKVxuICAgIH1cblxuICAgIC8vIEJ1aWxkIGVudmlyb25tZW50IGZvciB0aGUgd29ya2VyIHByb2Nlc3NcbiAgICAvLyBQdXJlIGlzb2xhdGlvbjogb25seSBlbnYgZnJvbSBicmlkZ2UgKyBsb2NhbCBvdmVycmlkZXMsIG5vIGxvY2FsIFBBVEgvSE9NRVxuICAgIGNvbnN0IHdvcmtlckVudjogTm9kZUpTLlByb2Nlc3NFbnYgPSB7XG4gICAgICAvLyBTdGFydCB3aXRoIGVudiB2YXJzIGZyb20gdGhlIGJyaWRnZSBMYW1iZGEgKEFXUyBjcmVkZW50aWFscywgdXNlci1kZWZpbmVkIHZhcnMpXG4gICAgICAuLi5lbnYsXG4gICAgICAvLyBMb2NhbCBvdmVycmlkZXMgKHRoZXNlIGFyZSBzZXQgYnkgdGhlIGRhZW1vbiwgbm90IGZyb20gYnJpZGdlKVxuICAgICAgQVdTX0xBTUJEQV9SVU5USU1FX0FQSTogYGxvY2FsaG9zdDoke3BvcnR9YCxcbiAgICAgIF9IQU5ETEVSOiBmbi5sb2NhbEhhbmRsZXIsXG4gICAgICBMQU1CREFfVEFTS19ST09UOiBwcm9qZWN0Um9vdCxcbiAgICAgIC8vIE1lbW9yeSBsaW1pdCBmb3IgY29udGV4dCBvYmplY3RcbiAgICAgIEFXU19MQU1CREFfRlVOQ1RJT05fTUVNT1JZX1NJWkU6IFN0cmluZyhmbi5tZW1vcnlNQiksXG4gICAgfVxuXG4gICAgeWllbGQqIEVmZmVjdC5sb2dEZWJ1ZyhcbiAgICAgIGBbTG9jYWxdIFN0YXJ0aW5nIE5vZGUuanMgd29ya2VyIGZvciAke2ZuLmZ1bmN0aW9uTmFtZX0gb24gcG9ydCAke3BvcnR9YCxcbiAgICApXG4gICAgeWllbGQqIEVmZmVjdC5sb2dEZWJ1ZyhgW0xvY2FsXSBIYW5kbGVyOiAke2ZuLmxvY2FsSGFuZGxlcn1gKVxuXG4gICAgLy8gU3Bhd24gQnVuIHdpdGggLS13YXRjaCB0byBhdXRvbWF0aWNhbGx5IHJlc3RhcnQgd2hlbiBoYW5kbGVyIGZpbGVzIGNoYW5nZVxuICAgIC8vIFRoaXMgZW5hYmxlcyBob3QtcmVsb2FkIHdpdGhvdXQgbmVlZGluZyB0byByZXN0YXJ0IHRoZSBkYWVtb25cbiAgICAvLyBVc2UgYWJzb2x1dGUgYnVuIHBhdGggZm9yIHB1cmUgZW52IGlzb2xhdGlvbiAobm8gUEFUSCBkZXBlbmRlbmN5KVxuICAgIGNvbnN0IHdvcmtlclByb2Nlc3MgPSBzcGF3bihidW5QYXRoLCBbXCItLXdhdGNoXCIsIHJ1bnRpbWVXcmFwcGVyUGF0aF0sIHtcbiAgICAgIGN3ZDogcHJvamVjdFJvb3QsXG4gICAgICBlbnY6IHdvcmtlckVudixcbiAgICAgIHN0ZGlvOiBbXCJpZ25vcmVcIiwgXCJwaXBlXCIsIFwicGlwZVwiXSxcbiAgICB9KVxuXG4gICAgLy8gUGF0dGVybiB0byBwYXJzZSBMYW1iZGEgbG9nIGZvcm1hdDogVElNRVNUQU1QXFx0UkVRVUVTVF9JRFxcdExFVkVMXFx0TUVTU0FHRVxuICAgIC8vIExhbWJkYSB1c2VzIHRhYnMgYmV0d2VlbiBmaWVsZHMuIENhcHR1cmVzOiBbMV0gPSByZXF1ZXN0IElELCBbMl0gPSBsZXZlbCArIG1lc3NhZ2VcbiAgICBjb25zdCBsYW1iZGFMb2dQYXR0ZXJuID1cbiAgICAgIC9eKFxcZHs0fS1cXGR7Mn0tXFxkezJ9VFtcXGQ6Ll0rWilbXFx0XFxzXSsoWzAtOWEtZi1dezM2fSlbXFx0XFxzXSsoLiopJC9pXG5cbiAgICAvLyBIZWxwZXIgdG8gZm9ybWF0IGxvZyBsaW5lIHdpdGggaW52b2NhdGlvbiBwcmVmaXhcbiAgICBjb25zdCBmb3JtYXRMaW5lID0gKFxuICAgICAgcmF3TGluZTogc3RyaW5nLFxuICAgICk6IHsgcHJlZml4OiBzdHJpbmc7IGNvbnRlbnQ6IHN0cmluZyB9ID0+IHtcbiAgICAgIC8vIFN0cmlwIGNhcnJpYWdlIHJldHVybnMgdGhhdCBjYW4gY2F1c2UgdGVybWluYWwgY29ycnVwdGlvblxuICAgICAgY29uc3QgbGluZSA9IHJhd0xpbmUucmVwbGFjZSgvXFxyL2csIFwiXCIpXG4gICAgICBjb25zdCBtYXRjaCA9IGxhbWJkYUxvZ1BhdHRlcm4uZXhlYyhsaW5lKVxuICAgICAgaWYgKG1hdGNoKSB7XG4gICAgICAgIGNvbnN0IHJlcXVlc3RJZCA9IG1hdGNoWzJdXG4gICAgICAgIGNvbnN0IGN0eCA9IGludm9jYXRpb25Db250ZXh0cy5nZXQocmVxdWVzdElkKVxuICAgICAgICBpZiAoY3R4KSB7XG4gICAgICAgICAgLy8gU3RyaXAgdGltZXN0YW1wIGFuZCByZXF1ZXN0IElELCBrZWVwIGp1c3QgTEVWRUwgTUVTU0FHRVxuICAgICAgICAgIHJldHVybiB7IHByZWZpeDogYFske2N0eC5udW19XWAsIGNvbnRlbnQ6IG1hdGNoWzNdIH1cbiAgICAgICAgfVxuICAgICAgfVxuICAgICAgcmV0dXJuIHsgcHJlZml4OiBcIltXb3JrZXJdXCIsIGNvbnRlbnQ6IGxpbmUgfVxuICAgIH1cblxuICAgIC8vIEZvcndhcmQgc3Rkb3V0L3N0ZGVyciB3aXRoIGludm9jYXRpb24gbnVtYmVyIHByZWZpeFxuICAgIHdvcmtlclByb2Nlc3Muc3Rkb3V0Py5vbihcImRhdGFcIiwgKGRhdGE6IEJ1ZmZlcikgPT4ge1xuICAgICAgY29uc3QgbGluZXMgPSBkYXRhLnRvU3RyaW5nKCkudHJpbSgpLnNwbGl0KFwiXFxuXCIpXG4gICAgICBmb3IgKGNvbnN0IHJhd0xpbmUgb2YgbGluZXMpIHtcbiAgICAgICAgY29uc3QgeyBwcmVmaXgsIGNvbnRlbnQgfSA9IGZvcm1hdExpbmUocmF3TGluZSlcbiAgICAgICAgY29uc29sZS5sb2coYCR7cHJlZml4fSAke2NvbnRlbnR9YClcbiAgICAgIH1cbiAgICB9KVxuXG4gICAgd29ya2VyUHJvY2Vzcy5zdGRlcnI/Lm9uKFwiZGF0YVwiLCAoZGF0YTogQnVmZmVyKSA9PiB7XG4gICAgICBjb25zdCBsaW5lcyA9IGRhdGEudG9TdHJpbmcoKS50cmltKCkuc3BsaXQoXCJcXG5cIilcbiAgICAgIGZvciAoY29uc3QgcmF3TGluZSBvZiBsaW5lcykge1xuICAgICAgICBjb25zdCB7IHByZWZpeCwgY29udGVudCB9ID0gZm9ybWF0TGluZShyYXdMaW5lKVxuICAgICAgICBjb25zb2xlLmVycm9yKGAke3ByZWZpeH0gJHtjb250ZW50fWApXG4gICAgICB9XG4gICAgfSlcblxuICAgIHdvcmtlclByb2Nlc3Mub24oXCJlcnJvclwiLCAoZXJyKSA9PiB7XG4gICAgICBFZmZlY3QucnVuU3luYyhcbiAgICAgICAgRWZmZWN0LmxvZ0Vycm9yKGBXb3JrZXIgZXJyb3I6ICR7ZXJyLm1lc3NhZ2V9YCkucGlwZShcbiAgICAgICAgICBFZmZlY3QuYW5ub3RhdGVMb2dzKFwiZnVuY3Rpb25cIiwgZm4uZnVuY3Rpb25OYW1lKSxcbiAgICAgICAgKSxcbiAgICAgIClcbiAgICB9KVxuXG4gICAgd29ya2VyUHJvY2Vzcy5vbihcImNsb3NlXCIsIChjb2RlKSA9PiB7XG4gICAgICBFZmZlY3QucnVuU3luYyhcbiAgICAgICAgRWZmZWN0LmxvZ0luZm8oYFdvcmtlciBleGl0ZWQgd2l0aCBjb2RlICR7Y29kZX1gKS5waXBlKFxuICAgICAgICAgIEVmZmVjdC5hbm5vdGF0ZUxvZ3MoXCJmdW5jdGlvblwiLCBmbi5mdW5jdGlvbk5hbWUpLFxuICAgICAgICApLFxuICAgICAgKVxuICAgIH0pXG5cbiAgICByZXR1cm4gd29ya2VyUHJvY2Vzc1xuICB9KVxuXG4vKipcbiAqIFByb2Nlc3MgcmVzcG9uc2VzIGZyb20gYSBOb2RlLmpzIHdvcmtlciBhbmQgZGlzcGF0Y2ggdG8gd2FpdGluZyBjYWxsZXJzLlxuICovXG5jb25zdCBwcm9jZXNzV29ya2VyUmVzcG9uc2VzID0gKFxuICB3b3JrZXI6IE5vZGVqc1dvcmtlcixcbiAgY2xpZW50OiBSZXR1cm5UeXBlPHR5cGVvZiBtYWtlQXBwU3luY0NsaWVudD4sXG4gIGludm9jYXRpb25Db250ZXh0czogTWFwPHN0cmluZywgSW52b2NhdGlvbkNvbnRleHQ+LFxuKSA9PlxuICBFZmZlY3QuZ2VuKGZ1bmN0aW9uKiAoKSB7XG4gICAgY29uc3QgcmVzcG9uc2VDaGFubmVsID0gYnVpbGRDaGFubmVsTmFtZS5yZXNwb25zZSh3b3JrZXIuZm4uZnVuY3Rpb25OYW1lKVxuXG4gICAgLy8gQ29udGludW91c2x5IHByb2Nlc3MgcmVzcG9uc2VzIGZyb20gdGhlIHdvcmtlclxuICAgIHdoaWxlICh0cnVlKSB7XG4gICAgICBjb25zdCByZXN1bHQgPSB5aWVsZCogd2FpdEZvclJlc3BvbnNlKHdvcmtlci5ydW50aW1lU3RhdGUpXG5cbiAgICAgIGxldCByZXNwb25zZTogUmVzcG9uc2VNZXNzYWdlXG4gICAgICBpZiAoaXNMYW1iZGFSZXNwb25zZShyZXN1bHQpKSB7XG4gICAgICAgIHJlc3BvbnNlID0ge1xuICAgICAgICAgIHR5cGU6IFwicmVzcG9uc2VcIixcbiAgICAgICAgICByZXF1ZXN0SWQ6IHJlc3VsdC5yZXF1ZXN0SWQsXG4gICAgICAgICAgcmVzdWx0OiByZXN1bHQuYm9keSxcbiAgICAgICAgfVxuICAgICAgfSBlbHNlIGlmIChpc0xhbWJkYUVycm9yKHJlc3VsdCkpIHtcbiAgICAgICAgcmVzcG9uc2UgPSB7XG4gICAgICAgICAgdHlwZTogXCJyZXNwb25zZVwiLFxuICAgICAgICAgIHJlcXVlc3RJZDogcmVzdWx0LnJlcXVlc3RJZCxcbiAgICAgICAgICBlcnJvcjoge1xuICAgICAgICAgICAgZXJyb3JUeXBlOiByZXN1bHQuZXJyb3JUeXBlLFxuICAgICAgICAgICAgZXJyb3JNZXNzYWdlOiByZXN1bHQuZXJyb3JNZXNzYWdlLFxuICAgICAgICAgICAgc3RhY2tUcmFjZTogcmVzdWx0LnN0YWNrVHJhY2UsXG4gICAgICAgICAgfSxcbiAgICAgICAgfVxuICAgICAgfSBlbHNlIGlmIChpc0xhbWJkYUluaXRFcnJvcihyZXN1bHQpKSB7XG4gICAgICAgIHlpZWxkKiBFZmZlY3QubG9nRXJyb3IoXG4gICAgICAgICAgYFtMb2NhbF0gV29ya2VyIGluaXQgZXJyb3I6ICR7cmVzdWx0LmVycm9yVHlwZX06ICR7cmVzdWx0LmVycm9yTWVzc2FnZX1gLFxuICAgICAgICApXG4gICAgICAgIGNvbnRpbnVlXG4gICAgICB9IGVsc2Uge1xuICAgICAgICBjb250aW51ZVxuICAgICAgfVxuXG4gICAgICAvLyBTZW5kIHJlc3BvbnNlIGJhY2sgdmlhIEFwcFN5bmNcbiAgICAgIHlpZWxkKiBjbGllbnQucHVibGlzaFJlc3BvbnNlKHJlc3BvbnNlQ2hhbm5lbCwgcmVzcG9uc2UpXG4gICAgICB5aWVsZCogRWZmZWN0LmxvZ0RlYnVnKGBbTG9jYWxdIFNlbnQgcmVzcG9uc2UgZm9yICR7cmVzcG9uc2UucmVxdWVzdElkfWApXG5cbiAgICAgIC8vIFByaW50IGVuZCBtYXJrZXIgd2l0aCB0aW1pbmdcbiAgICAgIGNvbnN0IGN0eCA9IGludm9jYXRpb25Db250ZXh0cy5nZXQocmVzcG9uc2UucmVxdWVzdElkKVxuICAgICAgaWYgKGN0eCkge1xuICAgICAgICBjb25zdCBkdXJhdGlvbk1zID0gRGF0ZS5ub3coKSAtIGN0eC5zdGFydFxuICAgICAgICBpZiAocmVzcG9uc2UuZXJyb3IpIHtcbiAgICAgICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgIGBbJHtjdHgubnVtfV0gXFx1MjUxNFxcdTI1MDBcXHUyNTAwIFxcdTI3MTcgJHtyZXNwb25zZS5lcnJvci5lcnJvclR5cGV9OiAke3Jlc3BvbnNlLmVycm9yLmVycm9yTWVzc2FnZX0gKCR7ZHVyYXRpb25Nc31tcykgXFx1MjUwMFxcdTI1MDBgLFxuICAgICAgICAgIClcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBjb25zb2xlLmxvZyhcbiAgICAgICAgICAgIGBbJHtjdHgubnVtfV0gXFx1MjUxNFxcdTI1MDBcXHUyNTAwIFxcdTI3MTMgRG9uZSAoJHtkdXJhdGlvbk1zfW1zKSBcXHUyNTAwXFx1MjUwMGAsXG4gICAgICAgICAgKVxuICAgICAgICB9XG4gICAgICAgIGludm9jYXRpb25Db250ZXh0cy5kZWxldGUocmVzcG9uc2UucmVxdWVzdElkKVxuICAgICAgfVxuICAgIH1cbiAgfSlcblxuLyoqXG4gKiBFbnZpcm9ubWVudCB2YXJpYWJsZXMgdG8gY29tcGxldGVseSBmaWx0ZXIgb3V0IGZyb20gaW52b2NhdGlvbiBlbnYuXG4gKiBUaGVzZSBhcmUgbm90IHJlbGV2YW50IGZvciBsb2NhbCBleGVjdXRpb24uXG4gKi9cbmNvbnN0IEVOVl9WQVJTX1RPX0ZJTFRFUiA9IG5ldyBTZXQoW1wiQVdTX0xBTUJEQV9MT0dfU1RSRUFNX05BTUVcIl0pXG5cbi8qKlxuICogRW52aXJvbm1lbnQgdmFyaWFibGVzIHRvIGlnbm9yZSB3aGVuIGNhbGN1bGF0aW5nIGlmIGVudiBoYXMgY2hhbmdlZC5cbiAqIFRoZXNlIGNoYW5nZSBmcmVxdWVudGx5IGJ1dCBkb24ndCByZXF1aXJlIGEgY29udGFpbmVyIHJlc3RhcnQuXG4gKlxuICogVE9ETzogQVdTIGNyZWRlbnRpYWxzIChBQ0NFU1NfS0VZX0lELCBTRUNSRVRfQUNDRVNTX0tFWSwgU0VTU0lPTl9UT0tFTilcbiAqIGRvIGV4cGlyZSBhbmQgd2lsbCBuZWVkIHJlZnJlc2hpbmcuIEZvciBub3cgd2UgaWdub3JlIHRoZW0gdG8gYXZvaWRcbiAqIHVubmVjZXNzYXJ5IHJlc3RhcnRzLCBidXQgd2Ugc2hvdWxkIGltcGxlbWVudCBjcmVkZW50aWFsIHJlZnJlc2ggbG9naWNcbiAqIHRoYXQgdXBkYXRlcyB0aGUgY29udGFpbmVyJ3MgY3JlZGVudGlhbHMgd2l0aG91dCBhIGZ1bGwgcmVzdGFydC5cbiAqL1xuY29uc3QgRU5WX1ZBUlNfVE9fSUdOT1JFX0lOX0RJRkYgPSBuZXcgU2V0KFtcbiAgXCJBV1NfQUNDRVNTX0tFWV9JRFwiLFxuICBcIkFXU19TRUNSRVRfQUNDRVNTX0tFWVwiLFxuICBcIkFXU19TRVNTSU9OX1RPS0VOXCIsXG5dKVxuXG4vKipcbiAqIEZpbHRlciBvdXQgaXJyZWxldmFudCBlbnYgdmFycyBmcm9tIGludm9jYXRpb24gZW52aXJvbm1lbnQuXG4gKi9cbmNvbnN0IHNhbml0aXplSW52b2NhdGlvbkVudiA9IChcbiAgZW52OiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+LFxuKTogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9PiB7XG4gIGNvbnN0IHJlc3VsdDogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9IHt9XG4gIGZvciAoY29uc3QgW2tleSwgdmFsdWVdIG9mIE9iamVjdC5lbnRyaWVzKGVudikpIHtcbiAgICBpZiAoIUVOVl9WQVJTX1RPX0ZJTFRFUi5oYXMoa2V5KSkge1xuICAgICAgcmVzdWx0W2tleV0gPSB2YWx1ZVxuICAgIH1cbiAgfVxuICByZXR1cm4gcmVzdWx0XG59XG5cbi8qKlxuICogQ29tcHV0ZSB3aGljaCBlbnZpcm9ubWVudCB2YXJpYWJsZXMgaGF2ZSBjaGFuZ2VkIGJldHdlZW4gdHdvIGVudiBvYmplY3RzLlxuICogUmV0dXJucyBhIGh1bWFuLXJlYWRhYmxlIHN1bW1hcnkgb2YgdGhlIGNoYW5nZXMuXG4gKiBJZ25vcmVzIGVudiB2YXJzIGluIEVOVl9WQVJTX1RPX0lHTk9SRV9JTl9ESUZGLlxuICovXG5jb25zdCBnZXRFbnZEaWZmID0gKFxuICBvbGRFbnY6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4sXG4gIG5ld0VudjogUmVjb3JkPHN0cmluZywgc3RyaW5nPixcbik6IHN0cmluZyA9PiB7XG4gIGNvbnN0IGNoYW5nZXM6IHN0cmluZ1tdID0gW11cblxuICAvLyBDaGVjayBmb3IgYWRkZWQgb3IgbW9kaWZpZWQga2V5c1xuICBmb3IgKGNvbnN0IGtleSBvZiBPYmplY3Qua2V5cyhuZXdFbnYpKSB7XG4gICAgaWYgKEVOVl9WQVJTX1RPX0lHTk9SRV9JTl9ESUZGLmhhcyhrZXkpKSBjb250aW51ZVxuICAgIGlmICghKGtleSBpbiBvbGRFbnYpKSB7XG4gICAgICBjaGFuZ2VzLnB1c2goYCske2tleX1gKVxuICAgIH0gZWxzZSBpZiAob2xkRW52W2tleV0gIT09IG5ld0VudltrZXldKSB7XG4gICAgICBjaGFuZ2VzLnB1c2goYH4ke2tleX1gKVxuICAgIH1cbiAgfVxuXG4gIC8vIENoZWNrIGZvciByZW1vdmVkIGtleXNcbiAgZm9yIChjb25zdCBrZXkgb2YgT2JqZWN0LmtleXMob2xkRW52KSkge1xuICAgIGlmIChFTlZfVkFSU19UT19JR05PUkVfSU5fRElGRi5oYXMoa2V5KSkgY29udGludWVcbiAgICBpZiAoIShrZXkgaW4gbmV3RW52KSkge1xuICAgICAgY2hhbmdlcy5wdXNoKGAtJHtrZXl9YClcbiAgICB9XG4gIH1cblxuICByZXR1cm4gY2hhbmdlcy5qb2luKFwiLCBcIilcbn1cblxuLyoqXG4gKiBFbnN1cmUgYSBOb2RlLmpzIHdvcmtlciBpcyBzdGFydGVkIGZvciBhIGZ1bmN0aW9uLlxuICogSWYgdGhlIHdvcmtlciBhbHJlYWR5IGV4aXN0cywgcmV0dXJuIGl0LlxuICogT3RoZXJ3aXNlLCBjcmVhdGUgdGhlIFJ1bnRpbWUgQVBJIHNlcnZlciwgYWRkIHRvIHdvcmtlcnMgbWFwLCBhbmQgc3RhcnQgdGhlIHdvcmtlci5cbiAqL1xuY29uc3QgZW5zdXJlV29ya2VyU3RhcnRlZCA9IChcbiAgZm46IERpc2NvdmVyZWRGdW5jdGlvbixcbiAgaW52b2NhdGlvbkVudjogUmVjb3JkPHN0cmluZywgc3RyaW5nPixcbiAgd29ya2Vyc1JlZjogUmVmLlJlZjxNYXA8c3RyaW5nLCBOb2RlanNXb3JrZXI+PixcbiAgc2VydmVyU2NvcGU6IFNjb3BlLlNjb3BlLFxuICBwcm9qZWN0Um9vdDogc3RyaW5nLFxuICBhcHBTeW5jQ2xpZW50OiBSZXR1cm5UeXBlPHR5cGVvZiBtYWtlQXBwU3luY0NsaWVudD4sXG4gIGludm9jYXRpb25Db250ZXh0czogTWFwPHN0cmluZywgSW52b2NhdGlvbkNvbnRleHQ+LFxuKTogRWZmZWN0LkVmZmVjdDxOb2RlanNXb3JrZXIsIEVycm9yPiA9PlxuICBFZmZlY3QuZ2VuKGZ1bmN0aW9uKiAoKSB7XG4gICAgLy8gQ2hlY2sgaWYgd29ya2VyIGFscmVhZHkgZXhpc3RzXG4gICAgY29uc3QgY3VycmVudFdvcmtlcnMgPSB5aWVsZCogUmVmLmdldCh3b3JrZXJzUmVmKVxuICAgIGNvbnN0IGV4aXN0aW5nID0gY3VycmVudFdvcmtlcnMuZ2V0KGZuLmZ1bmN0aW9uTmFtZSlcbiAgICBpZiAoZXhpc3RpbmcpIHtcbiAgICAgIC8vIENoZWNrIGlmIGVudiB2YXJzIGhhdmUgY2hhbmdlZCAoZS5nLiwgYWZ0ZXIgQ0RLIHJlZGVwbG95KVxuICAgICAgY29uc3QgZW52RGlmZiA9IGdldEVudkRpZmYoZXhpc3RpbmcuZW52LCBpbnZvY2F0aW9uRW52KVxuICAgICAgaWYgKGVudkRpZmYpIHtcbiAgICAgICAgeWllbGQqIEVmZmVjdC5sb2dJbmZvKFxuICAgICAgICAgIGBbTG9jYWxdIEVudmlyb25tZW50IGNoYW5nZWQgZm9yICR7Zm4uZnVuY3Rpb25OYW1lfSwgcmVzdGFydGluZyB3b3JrZXIuLi4gKCR7ZW52RGlmZn0pYCxcbiAgICAgICAgKVxuICAgICAgICAvLyBLaWxsIHRoZSBvbGQgd29ya2VyXG4gICAgICAgIHlpZWxkKiBFZmZlY3QudHJ5KCgpID0+IGV4aXN0aW5nLndvcmtlclByb2Nlc3Mua2lsbChcIlNJR1RFUk1cIikpLnBpcGUoXG4gICAgICAgICAgRWZmZWN0LmNhdGNoQWxsKCgpID0+IEVmZmVjdC52b2lkKSxcbiAgICAgICAgKVxuICAgICAgICAvLyBSZW1vdmUgZnJvbSBtYXAgc28gd2UgY3JlYXRlIGEgbmV3IG9uZSBiZWxvd1xuICAgICAgICBjdXJyZW50V29ya2Vycy5kZWxldGUoZm4uZnVuY3Rpb25OYW1lKVxuICAgICAgICB5aWVsZCogUmVmLnNldCh3b3JrZXJzUmVmLCBjdXJyZW50V29ya2VycylcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHJldHVybiBleGlzdGluZ1xuICAgICAgfVxuICAgIH1cblxuICAgIC8vIFdvcmtlciBkb2Vzbid0IGV4aXN0IC0gc3RhcnQgaXQgbGF6aWx5XG4gICAgeWllbGQqIEVmZmVjdC5sb2dEZWJ1ZyhcbiAgICAgIGBbTG9jYWxdIFN0YXJ0aW5nIE5vZGUuanMgd29ya2VyIGZvciBmaXJzdCBpbnZvY2F0aW9uIG9mICR7Zm4uZnVuY3Rpb25OYW1lfS4uLmAsXG4gICAgKVxuXG4gICAgLy8gQ3JlYXRlIFJ1bnRpbWUgQVBJIHNlcnZlciBvbiBlcGhlbWVyYWwgcG9ydFxuICAgIGNvbnN0IHsgcG9ydCwgc3RhdGU6IHJ1bnRpbWVTdGF0ZSB9ID0geWllbGQqIHN0YXJ0UnVudGltZUFwaVNlcnZlcih7XG4gICAgICBmdW5jdGlvbk1ldGFkYXRhOiB7XG4gICAgICAgIGZ1bmN0aW9uTmFtZTogZm4uZnVuY3Rpb25OYW1lLFxuICAgICAgICBmdW5jdGlvblZlcnNpb246IFwiJExBVEVTVFwiLFxuICAgICAgICBoYW5kbGVyOiBmbi5sb2NhbEhhbmRsZXIsXG4gICAgICB9LFxuICAgIH0pLnBpcGUoRWZmZWN0LnByb3ZpZGVTZXJ2aWNlKFNjb3BlLlNjb3BlLCBzZXJ2ZXJTY29wZSkpXG5cbiAgICAvLyBDcmVhdGUgd29ya2VyIG9iamVjdCAod2l0aG91dCBwcm9jZXNzIGluaXRpYWxseSAtIHdpbGwgYmUgc2V0IGFmdGVyIHN0YXJ0KVxuICAgIC8vIFdlIGFkZCB0byBtYXAgQkVGT1JFIHN0YXJ0aW5nIHdvcmtlciB0byBoYW5kbGUgY29uY3VycmVudCBpbnZvY2F0aW9uc1xuICAgIGNvbnN0IHdvcmtlcjogTm9kZWpzV29ya2VyID0ge1xuICAgICAgZm4sXG4gICAgICBydW50aW1lU3RhdGUsXG4gICAgICBwb3J0LFxuICAgICAgd29ya2VyUHJvY2VzczogdW5kZWZpbmVkIGFzIHVua25vd24gYXMgQ2hpbGRQcm9jZXNzLCAvLyBXaWxsIGJlIHNldCBzaG9ydGx5XG4gICAgICBlbnY6IGludm9jYXRpb25FbnYsXG4gICAgfVxuXG4gICAgLy8gQWRkIHRvIG1hcCBpbW1lZGlhdGVseSB0byBwcmV2ZW50IHJhY2UgY29uZGl0aW9uc1xuICAgIGN1cnJlbnRXb3JrZXJzLnNldChmbi5mdW5jdGlvbk5hbWUsIHdvcmtlcilcbiAgICB5aWVsZCogUmVmLnNldCh3b3JrZXJzUmVmLCBjdXJyZW50V29ya2VycylcblxuICAgIC8vIFN0YXJ0IHRoZSB3b3JrZXIgcHJvY2VzcyAtIHJlbW92ZSBmcm9tIG1hcCBvbiBmYWlsdXJlXG4gICAgY29uc3Qgd29ya2VyUHJvY2VzcyA9IHlpZWxkKiBzdGFydE5vZGVqc1dvcmtlcihcbiAgICAgIGZuLFxuICAgICAgcG9ydCxcbiAgICAgIHByb2plY3RSb290LFxuICAgICAgaW52b2NhdGlvbkVudixcbiAgICAgIGludm9jYXRpb25Db250ZXh0cyxcbiAgICApLnBpcGUoXG4gICAgICBFZmZlY3QuY2F0Y2hBbGwoKGVycm9yKSA9PlxuICAgICAgICBFZmZlY3QuZ2VuKGZ1bmN0aW9uKiAoKSB7XG4gICAgICAgICAgLy8gUmVtb3ZlIGJyb2tlbiB3b3JrZXIgZnJvbSBtYXAgb24gZmFpbHVyZVxuICAgICAgICAgIHlpZWxkKiBFZmZlY3QubG9nRXJyb3IoXG4gICAgICAgICAgICBgW0xvY2FsXSBGYWlsZWQgdG8gc3RhcnQgd29ya2VyIGZvciAke2ZuLmZ1bmN0aW9uTmFtZX06ICR7ZXJyb3J9YCxcbiAgICAgICAgICApXG4gICAgICAgICAgY29uc3QgY3VycmVudCA9IHlpZWxkKiBSZWYuZ2V0KHdvcmtlcnNSZWYpXG4gICAgICAgICAgY3VycmVudC5kZWxldGUoZm4uZnVuY3Rpb25OYW1lKVxuICAgICAgICAgIHlpZWxkKiBSZWYuc2V0KHdvcmtlcnNSZWYsIGN1cnJlbnQpXG4gICAgICAgICAgcmV0dXJuIHlpZWxkKiBFZmZlY3QuZmFpbChlcnJvcilcbiAgICAgICAgfSksXG4gICAgICApLFxuICAgIClcblxuICAgIC8vIFVwZGF0ZSB3b3JrZXIgd2l0aCB0aGUgcHJvY2Vzc1xuICAgIHdvcmtlci53b3JrZXJQcm9jZXNzID0gd29ya2VyUHJvY2Vzc1xuXG4gICAgLy8gU3RhcnQgcHJvY2Vzc2luZyByZXNwb25zZXMgaW4gdGhlIGJhY2tncm91bmRcbiAgICBFZmZlY3QucnVuRm9yayhcbiAgICAgIHByb2Nlc3NXb3JrZXJSZXNwb25zZXMod29ya2VyLCBhcHBTeW5jQ2xpZW50LCBpbnZvY2F0aW9uQ29udGV4dHMpLFxuICAgIClcblxuICAgIHJldHVybiB3b3JrZXJcbiAgfSlcblxuLyoqXG4gKiBFbnN1cmUgYSBjb250YWluZXIgaXMgc3RhcnRlZCBmb3IgYSBmdW5jdGlvbi5cbiAqIElmIHRoZSBjb250YWluZXIgYWxyZWFkeSBleGlzdHMsIHJldHVybiBpdC5cbiAqIElmIHRoZSBlbnYgdmFycyBoYXZlIGNoYW5nZWQsIHJlc3RhcnQgdGhlIGNvbnRhaW5lci5cbiAqIE90aGVyd2lzZSwgY3JlYXRlIHRoZSBSdW50aW1lIEFQSSBzZXJ2ZXIsIGFkZCB0byBjb250YWluZXJzIG1hcCwgYW5kIHN0YXJ0IHRoZSBjb250YWluZXIuXG4gKi9cbmNvbnN0IGVuc3VyZUNvbnRhaW5lclN0YXJ0ZWQgPSAoXG4gIGZuOiBEaXNjb3ZlcmVkRnVuY3Rpb24sXG4gIGludm9jYXRpb25FbnY6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4sXG4gIGNvbnRhaW5lcnNSZWY6IFJlZi5SZWY8TWFwPHN0cmluZywgRnVuY3Rpb25Db250YWluZXI+PixcbiAgc2VydmVyU2NvcGU6IFNjb3BlLlNjb3BlLFxuICBwcm9qZWN0Um9vdDogc3RyaW5nLFxuICBpZGxlVGltZW91dE1zOiBudW1iZXIsXG4gIHBvbGxUaW1lb3V0TXM6IG51bWJlcixcbiAgYXBwU3luY0NsaWVudDogUmV0dXJuVHlwZTx0eXBlb2YgbWFrZUFwcFN5bmNDbGllbnQ+LFxuICBpbnZvY2F0aW9uQ29udGV4dHM6IE1hcDxzdHJpbmcsIEludm9jYXRpb25Db250ZXh0Pixcbik6IEVmZmVjdC5FZmZlY3Q8RnVuY3Rpb25Db250YWluZXIsIEVycm9yLCBDb21tYW5kRXhlY3V0b3IuQ29tbWFuZEV4ZWN1dG9yPiA9PlxuICBFZmZlY3QuZ2VuKGZ1bmN0aW9uKiAoKSB7XG4gICAgLy8gQ2hlY2sgaWYgY29udGFpbmVyIGFscmVhZHkgZXhpc3RzXG4gICAgY29uc3QgY3VycmVudENvbnRhaW5lcnMgPSB5aWVsZCogUmVmLmdldChjb250YWluZXJzUmVmKVxuICAgIGNvbnN0IGV4aXN0aW5nID0gY3VycmVudENvbnRhaW5lcnMuZ2V0KGZuLmZ1bmN0aW9uTmFtZSlcbiAgICBpZiAoZXhpc3RpbmcpIHtcbiAgICAgIC8vIENoZWNrIGlmIGVudiB2YXJzIGhhdmUgY2hhbmdlZCAoZS5nLiwgYWZ0ZXIgQ0RLIHJlZGVwbG95KVxuICAgICAgY29uc3QgZW52RGlmZiA9IGdldEVudkRpZmYoZXhpc3RpbmcuZW52LCBpbnZvY2F0aW9uRW52KVxuICAgICAgaWYgKGVudkRpZmYpIHtcbiAgICAgICAgLy8gSU1QT1JUQU5UOiBVcGRhdGUgZW52IGltbWVkaWF0ZWx5IChiZWZvcmUgYW55IHlpZWxkcykgdG8gcHJldmVudFxuICAgICAgICAvLyBvdGhlciBjb25jdXJyZW50IGludm9jYXRpb25zIGZyb20gYWxzbyBkZXRlY3RpbmcgdGhlIGNoYW5nZSBhbmRcbiAgICAgICAgLy8gdHJpZ2dlcmluZyBkdXBsaWNhdGUgcmVzdGFydHNcbiAgICAgICAgZXhpc3RpbmcuZW52ID0gaW52b2NhdGlvbkVudlxuXG4gICAgICAgIC8vIElmIGFscmVhZHkgcmVzdGFydGluZyBmb3IgZW52IGNoYW5nZSwganVzdCByZXR1cm4gdGhlIGV4aXN0aW5nIGNvbnRhaW5lclxuICAgICAgICAvLyBUaGUgaW52b2NhdGlvbiB3aWxsIGJlIHF1ZXVlZCBhbmQgcGlja2VkIHVwIGJ5IHRoZSBuZXcgY29udGFpbmVyXG4gICAgICAgIGlmIChleGlzdGluZy5pc1JlYnVpbGRpbmcpIHtcbiAgICAgICAgICB5aWVsZCogRWZmZWN0LmxvZ0RlYnVnKFxuICAgICAgICAgICAgYFtMb2NhbF0gRW52aXJvbm1lbnQgY2hhbmdlIHJlc3RhcnQgYWxyZWFkeSBpbiBwcm9ncmVzcyBmb3IgJHtmbi5mdW5jdGlvbk5hbWV9LCBxdWV1ZWluZyBpbnZvY2F0aW9uYCxcbiAgICAgICAgICApXG4gICAgICAgICAgcmV0dXJuIGV4aXN0aW5nXG4gICAgICAgIH1cblxuICAgICAgICB5aWVsZCogRWZmZWN0LmxvZ0luZm8oXG4gICAgICAgICAgYFtMb2NhbF0gRW52aXJvbm1lbnQgY2hhbmdlZCBmb3IgJHtmbi5mdW5jdGlvbk5hbWV9LCByZXN0YXJ0aW5nIGNvbnRhaW5lci4uLiAoJHtlbnZEaWZmfSlgLFxuICAgICAgICApXG4gICAgICAgIGV4aXN0aW5nLmlzUmVidWlsZGluZyA9IHRydWVcblxuICAgICAgICAvLyBTdG9wIHRoZSBEb2NrZXIgY29udGFpbmVyIGZvcmNlZnVsbHkgKHRpbWVvdXQ9MXMpXG4gICAgICAgIC8vIFRoaXMgc2VuZHMgU0lHVEVSTSBhbmQgdGhlbiBTSUdLSUxMIGFmdGVyIDEgc2Vjb25kXG4gICAgICAgIC8vIFdlIG11c3QgZG8gdGhpcyBCRUZPUkUgaW50ZXJydXB0aW5nIHRoZSBmaWJlciwgYmVjYXVzZSB0aGUgZmliZXIgaXNcbiAgICAgICAgLy8gYmxvY2tlZCB3YWl0aW5nIGZvciB0aGUgZG9ja2VyIHByb2Nlc3MgdG8gZXhpdFxuICAgICAgICB5aWVsZCogRWZmZWN0LmdlbihmdW5jdGlvbiogKCkge1xuICAgICAgICAgIGNvbnN0IGRvY2tlciA9IHlpZWxkKiBEb2NrZXJcbiAgICAgICAgICB5aWVsZCogZG9ja2VyLnN0b3AoZXhpc3RpbmcuY29udGFpbmVyTmFtZSwgMSkucGlwZShFZmZlY3Quc2NvcGVkKVxuICAgICAgICB9KS5waXBlKFxuICAgICAgICAgIEVmZmVjdC5wcm92aWRlKERvY2tlckxpdmUpLFxuICAgICAgICAgIEVmZmVjdC5jYXRjaEFsbCgoZXJyb3IpID0+XG4gICAgICAgICAgICBFZmZlY3QubG9nRGVidWcoXG4gICAgICAgICAgICAgIGBbTG9jYWxdIEVycm9yIHN0b3BwaW5nIGNvbnRhaW5lciAobWF5IGFscmVhZHkgYmUgc3RvcHBlZCk6ICR7ZXJyb3J9YCxcbiAgICAgICAgICAgICksXG4gICAgICAgICAgKSxcbiAgICAgICAgKVxuXG4gICAgICAgIC8vIE5vdyBpbnRlcnJ1cHQgdGhlIGZpYmVyIChpdCBzaG91bGQgY29tcGxldGUgcXVpY2tseSBzaW5jZSBjb250YWluZXIgc3RvcHBlZClcbiAgICAgICAgeWllbGQqIEZpYmVyLmludGVycnVwdChleGlzdGluZy5jb250YWluZXJGaWJlcikucGlwZShcbiAgICAgICAgICBFZmZlY3QuY2F0Y2hBbGwoKCkgPT4gRWZmZWN0LnZvaWQpLFxuICAgICAgICApXG5cbiAgICAgICAgLy8gUmVzdGFydCB0aGUgY29udGFpbmVyIHdpdGggbmV3IGVudmlyb25tZW50LCByZXVzaW5nIGV4aXN0aW5nIFJ1bnRpbWVBUElcbiAgICAgICAgY29uc3QgbmV3RmliZXIgPSB5aWVsZCogc3RhcnRGdW5jdGlvbkNvbnRhaW5lcihcbiAgICAgICAgICBmbixcbiAgICAgICAgICBleGlzdGluZy5wb3J0LFxuICAgICAgICAgIHByb2plY3RSb290LFxuICAgICAgICAgIGludm9jYXRpb25FbnYsXG4gICAgICAgICAgaW52b2NhdGlvbkNvbnRleHRzLFxuICAgICAgICApLnBpcGUoXG4gICAgICAgICAgRWZmZWN0LmNhdGNoQWxsKChlcnJvcikgPT5cbiAgICAgICAgICAgIEVmZmVjdC5nZW4oZnVuY3Rpb24qICgpIHtcbiAgICAgICAgICAgICAgeWllbGQqIEVmZmVjdC5sb2dFcnJvcihcbiAgICAgICAgICAgICAgICBgW0xvY2FsXSBGYWlsZWQgdG8gcmVzdGFydCBjb250YWluZXIgZm9yICR7Zm4uZnVuY3Rpb25OYW1lfTogJHtlcnJvcn1gLFxuICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgIGV4aXN0aW5nLmlzUmVidWlsZGluZyA9IGZhbHNlXG4gICAgICAgICAgICAgIHJldHVybiB5aWVsZCogRWZmZWN0LmZhaWwoZXJyb3IpXG4gICAgICAgICAgICB9KSxcbiAgICAgICAgICApLFxuICAgICAgICApXG5cbiAgICAgICAgLy8gVXBkYXRlIGNvbnRhaW5lciB3aXRoIG5ldyBmaWJlclxuICAgICAgICBleGlzdGluZy5jb250YWluZXJGaWJlciA9IG5ld0ZpYmVyXG4gICAgICAgIGV4aXN0aW5nLmlzUmVidWlsZGluZyA9IGZhbHNlXG5cbiAgICAgICAgcmV0dXJuIGV4aXN0aW5nXG4gICAgICB9IGVsc2Uge1xuICAgICAgICByZXR1cm4gZXhpc3RpbmdcbiAgICAgIH1cbiAgICB9XG5cbiAgICAvLyBDb250YWluZXIgZG9lc24ndCBleGlzdCAtIHN0YXJ0IGl0IGxhemlseVxuICAgIHlpZWxkKiBFZmZlY3QubG9nSW5mbyhcbiAgICAgIGBbTG9jYWxdIFN0YXJ0aW5nIGNvbnRhaW5lciBmb3IgZmlyc3QgaW52b2NhdGlvbiBvZiAke2ZuLmZ1bmN0aW9uTmFtZX0uLi5gLFxuICAgIClcblxuICAgIC8vIENyZWF0ZSBSdW50aW1lIEFQSSBzZXJ2ZXIgb24gZXBoZW1lcmFsIHBvcnRcbiAgICAvLyBVc2UgcG9sbCB0aW1lb3V0IHRoYXQncyBzaG9ydGVyIHRoYW4gaWRsZSB0aW1lb3V0IHNvIGNvbnRhaW5lciBleGl0cyBuYXR1cmFsbHlcbiAgICBjb25zdCB7IHBvcnQsIHN0YXRlOiBydW50aW1lU3RhdGUgfSA9IHlpZWxkKiBzdGFydFJ1bnRpbWVBcGlTZXJ2ZXIoe1xuICAgICAgcG9sbFRpbWVvdXRNcyxcbiAgICAgIGZ1bmN0aW9uTWV0YWRhdGE6IHtcbiAgICAgICAgZnVuY3Rpb25OYW1lOiBmbi5mdW5jdGlvbk5hbWUsXG4gICAgICAgIGZ1bmN0aW9uVmVyc2lvbjogXCIkTEFURVNUXCIsXG4gICAgICAgIGhhbmRsZXI6IFwiaW5kZXguaGFuZGxlclwiLFxuICAgICAgfSxcbiAgICB9KS5waXBlKEVmZmVjdC5wcm92aWRlU2VydmljZShTY29wZS5TY29wZSwgc2VydmVyU2NvcGUpKVxuXG4gICAgLy8gR2VuZXJhdGUgY29udGFpbmVyIG5hbWUgYW5kIGltYWdlIG5hbWVcbiAgICBjb25zdCBjb250YWluZXJOYW1lID0gYGxhbWJkYS0ke2ZuLmZ1bmN0aW9uTmFtZS5yZXBsYWNlKC9bXmEtekEtWjAtOV0vZywgXCItXCIpfWBcbiAgICBjb25zdCBpbWFnZU5hbWUgPSBgbGl2ZS1sYW1iZGEtJHtmbi5mdW5jdGlvbk5hbWUudG9Mb3dlckNhc2UoKS5yZXBsYWNlKC9bXmEtejAtOS1dL2csIFwiLVwiKX1gXG5cbiAgICAvLyBDcmVhdGUgY29udGFpbmVyIG9iamVjdCAod2l0aG91dCBmaWJlciBpbml0aWFsbHkgLSB3aWxsIGJlIHNldCBhZnRlciBzdGFydClcbiAgICAvLyBXZSBhZGQgdG8gbWFwIEJFRk9SRSBzdGFydGluZyBjb250YWluZXIgdG8gaGFuZGxlIGNvbmN1cnJlbnQgaW52b2NhdGlvbnNcbiAgICBjb25zdCBjb250YWluZXI6IEZ1bmN0aW9uQ29udGFpbmVyID0ge1xuICAgICAgZm4sXG4gICAgICBydW50aW1lU3RhdGUsXG4gICAgICBwb3J0LFxuICAgICAgY29udGFpbmVyRmliZXI6IHVuZGVmaW5lZCBhcyB1bmtub3duIGFzIEZpYmVyLlJ1bnRpbWVGaWJlcjx2b2lkLCBFcnJvcj4sIC8vIFdpbGwgYmUgc2V0IHNob3J0bHlcbiAgICAgIGNvbnRhaW5lck5hbWUsXG4gICAgICBpbWFnZU5hbWUsXG4gICAgICBpc1JlYnVpbGRpbmc6IGZhbHNlLFxuICAgICAgcGVuZGluZ1Jlc3BvbnNlczogbmV3IE1hcCgpLFxuICAgICAgaWRsZVRpbWVyRmliZXI6IG51bGwsXG4gICAgICBlbnY6IGludm9jYXRpb25FbnYsXG4gICAgfVxuXG4gICAgLy8gQWRkIHRvIG1hcCBpbW1lZGlhdGVseSB0byBwcmV2ZW50IHJhY2UgY29uZGl0aW9uc1xuICAgIGN1cnJlbnRDb250YWluZXJzLnNldChmbi5mdW5jdGlvbk5hbWUsIGNvbnRhaW5lcilcbiAgICB5aWVsZCogUmVmLnNldChjb250YWluZXJzUmVmLCBjdXJyZW50Q29udGFpbmVycylcblxuICAgIC8vIEJ1aWxkIGFuZCBzdGFydCB0aGUgY29udGFpbmVyIC0gcmVtb3ZlIGZyb20gbWFwIG9uIGZhaWx1cmVcbiAgICBjb25zdCBjb250YWluZXJGaWJlciA9IHlpZWxkKiBzdGFydEZ1bmN0aW9uQ29udGFpbmVyKFxuICAgICAgZm4sXG4gICAgICBwb3J0LFxuICAgICAgcHJvamVjdFJvb3QsXG4gICAgICBpbnZvY2F0aW9uRW52LFxuICAgICAgaW52b2NhdGlvbkNvbnRleHRzLFxuICAgICkucGlwZShcbiAgICAgIEVmZmVjdC5jYXRjaEFsbCgoZXJyb3IpID0+XG4gICAgICAgIEVmZmVjdC5nZW4oZnVuY3Rpb24qICgpIHtcbiAgICAgICAgICAvLyBSZW1vdmUgYnJva2VuIGNvbnRhaW5lciBmcm9tIG1hcCBvbiBmYWlsdXJlXG4gICAgICAgICAgeWllbGQqIEVmZmVjdC5sb2dFcnJvcihcbiAgICAgICAgICAgIGBbTG9jYWxdIEZhaWxlZCB0byBzdGFydCBjb250YWluZXIgZm9yICR7Zm4uZnVuY3Rpb25OYW1lfTogJHtlcnJvcn1gLFxuICAgICAgICAgIClcbiAgICAgICAgICBjb25zdCBjdXJyZW50ID0geWllbGQqIFJlZi5nZXQoY29udGFpbmVyc1JlZilcbiAgICAgICAgICBjdXJyZW50LmRlbGV0ZShmbi5mdW5jdGlvbk5hbWUpXG4gICAgICAgICAgeWllbGQqIFJlZi5zZXQoY29udGFpbmVyc1JlZiwgY3VycmVudClcbiAgICAgICAgICByZXR1cm4geWllbGQqIEVmZmVjdC5mYWlsKGVycm9yKVxuICAgICAgICB9KSxcbiAgICAgICksXG4gICAgKVxuXG4gICAgLy8gVXBkYXRlIGNvbnRhaW5lciB3aXRoIHRoZSBmaWJlclxuICAgIGNvbnRhaW5lci5jb250YWluZXJGaWJlciA9IGNvbnRhaW5lckZpYmVyXG5cbiAgICAvLyBTdGFydCBwcm9jZXNzaW5nIHJlc3BvbnNlcyBpbiB0aGUgYmFja2dyb3VuZCB1c2luZyBmb3JrRGFlbW9uXG4gICAgeWllbGQqIHByb2Nlc3NDb250YWluZXJSZXNwb25zZXMoXG4gICAgICBjb250YWluZXIsXG4gICAgICBjb250YWluZXJzUmVmLFxuICAgICAgaWRsZVRpbWVvdXRNcyxcbiAgICAgIGFwcFN5bmNDbGllbnQsXG4gICAgICBpbnZvY2F0aW9uQ29udGV4dHMsXG4gICAgKS5waXBlKEVmZmVjdC5mb3JrRGFlbW9uKVxuXG4gICAgcmV0dXJuIGNvbnRhaW5lclxuICB9KVxuXG4vKipcbiAqIFJlYnVpbGQgYSBEb2NrZXIgY29udGFpbmVyIGFmdGVyIGZpbGUgY2hhbmdlcy5cbiAqIEludm9jYXRpb25zIGFycml2aW5nIGR1cmluZyByZWJ1aWxkIHdpbGwgYmUgcXVldWVkIGFuZCBwaWNrZWQgdXAgYnkgdGhlIG5ldyBjb250YWluZXIuXG4gKi9cbmNvbnN0IHJlYnVpbGREb2NrZXJDb250YWluZXIgPSAoXG4gIGZ1bmN0aW9uSWQ6IHN0cmluZyxcbiAgY29udGFpbmVyc1JlZjogUmVmLlJlZjxNYXA8c3RyaW5nLCBGdW5jdGlvbkNvbnRhaW5lcj4+LFxuICBwcm9qZWN0Um9vdDogc3RyaW5nLFxuKTogRWZmZWN0LkVmZmVjdDx2b2lkLCBFcnJvciwgQ29tbWFuZEV4ZWN1dG9yLkNvbW1hbmRFeGVjdXRvcj4gPT5cbiAgRWZmZWN0LmdlbihmdW5jdGlvbiogKCkge1xuICAgIGNvbnN0IGNvbnRhaW5lcnMgPSB5aWVsZCogUmVmLmdldChjb250YWluZXJzUmVmKVxuICAgIGNvbnN0IGNvbnRhaW5lciA9IGNvbnRhaW5lcnMuZ2V0KGZ1bmN0aW9uSWQpXG5cbiAgICBpZiAoIWNvbnRhaW5lcikge1xuICAgICAgeWllbGQqIEVmZmVjdC5sb2dJbmZvKFxuICAgICAgICBgW0xvY2FsXSBDYW5ub3QgcmVidWlsZCAke2Z1bmN0aW9uSWR9IC0gY29udGFpbmVyIG5vdCBmb3VuZGAsXG4gICAgICApXG4gICAgICByZXR1cm5cbiAgICB9XG5cbiAgICAvLyBNYXJrIGFzIHJlYnVpbGRpbmcgLSBpbnZvY2F0aW9ucyB3aWxsIHN0aWxsIHF1ZXVlIGJ1dCB3ZSBsb2cgaXRcbiAgICBjb250YWluZXIuaXNSZWJ1aWxkaW5nID0gdHJ1ZVxuXG4gICAgeWllbGQqIEVmZmVjdC5sb2dEZWJ1ZyhgW0xvY2FsXSBSZWJ1aWxkaW5nIGNvbnRhaW5lciBmb3IgJHtmdW5jdGlvbklkfS4uLmApXG5cbiAgICBjb25zdCBkb2NrZXIgPSB5aWVsZCogRG9ja2VyXG5cbiAgICAvLyBTdG9wIHRoZSBleGlzdGluZyBjb250YWluZXIgdXNpbmcgRG9ja2VyIHNlcnZpY2VcbiAgICBjb25zdCBjb250YWluZXJJZCA9IGNvbnRhaW5lci5jb250YWluZXJOYW1lXG4gICAgeWllbGQqIEVmZmVjdC5sb2dJbmZvKFxuICAgICAgYFtMb2NhbF0gU3RvcHBpbmcgY29udGFpbmVyIHdpdGggbmFtZSBwcmVmaXg6ICR7Y29udGFpbmVySWR9YCxcbiAgICApXG5cbiAgICBjb25zdCBzdG9wQ291bnQgPSB5aWVsZCogZG9ja2VyLnN0b3AoY29udGFpbmVySWQpLnBpcGUoXG4gICAgICBFZmZlY3Quc2NvcGVkLFxuICAgICAgRWZmZWN0LmNhdGNoQWxsKChlcnJvcikgPT5cbiAgICAgICAgRWZmZWN0LmdlbihmdW5jdGlvbiogKCkge1xuICAgICAgICAgIHlpZWxkKiBFZmZlY3QubG9nSW5mbyhgTm90ZTogQ29udGFpbmVyIHN0b3AgaGFkIGlzc3VlOiAke2Vycm9yfWApXG4gICAgICAgICAgcmV0dXJuIDBcbiAgICAgICAgfSksXG4gICAgICApLFxuICAgIClcbiAgICB5aWVsZCogRWZmZWN0LmxvZ0RlYnVnKGBbTG9jYWxdIFN0b3BwZWQgJHtzdG9wQ291bnR9IGNvbnRhaW5lcihzKWApXG5cbiAgICAvLyBSZXNvbHZlIHRoZSBjb250ZXh0IHBhdGhcbiAgICBjb25zdCBmbiA9IGNvbnRhaW5lci5mblxuICAgIGNvbnN0IGNvbnRleHRQYXRoID0gZm4uZG9ja2VyQ29udGV4dFBhdGg/LnN0YXJ0c1dpdGgoXCIvXCIpXG4gICAgICA/IGZuLmRvY2tlckNvbnRleHRQYXRoXG4gICAgICA6IGAke3Byb2plY3RSb290fS8ke2ZuLmRvY2tlckNvbnRleHRQYXRofWBcblxuICAgIC8vIERldGVybWluZSBwbGF0Zm9ybSBmcm9tIGFyY2hpdGVjdHVyZVxuICAgIGNvbnN0IHBsYXRmb3JtID0gZm4uYXJjaGl0ZWN0dXJlID09PSBcImFybTY0XCIgPyBcImxpbnV4L2FybTY0XCIgOiBcImxpbnV4L2FtZDY0XCJcblxuICAgIC8vIFJlYnVpbGQgdGhlIERvY2tlciBpbWFnZVxuICAgIHlpZWxkKiBkb2NrZXJcbiAgICAgIC5idWlsZCh7XG4gICAgICAgIGNvbnRleHRQYXRoLFxuICAgICAgICBpbWFnZU5hbWU6IGNvbnRhaW5lci5pbWFnZU5hbWUsXG4gICAgICAgIHBsYXRmb3JtLFxuICAgICAgfSlcbiAgICAgIC5waXBlKEVmZmVjdC5zY29wZWQpXG5cbiAgICAvLyBJbnNwZWN0IHRoZSBpbWFnZSB0byBnZXQgb3JpZ2luYWwgZW50cnlwb2ludC9jbWQgZm9yIGV4dGVuc2lvbiB3cmFwcGVyXG4gICAgY29uc3QgaW1hZ2VDb25maWcgPSB5aWVsZCogZG9ja2VyXG4gICAgICAuaW5zcGVjdChjb250YWluZXIuaW1hZ2VOYW1lKVxuICAgICAgLnBpcGUoRWZmZWN0LnNjb3BlZClcbiAgICBjb25zdCBleHRlbnNpb25XcmFwcGVyID0gYnVpbGRFeHRlbnNpb25XcmFwcGVyQ29tbWFuZChcbiAgICAgIGltYWdlQ29uZmlnLmVudHJ5cG9pbnQsXG4gICAgICBpbWFnZUNvbmZpZy5jbWQsXG4gICAgKVxuXG4gICAgLy8gUmVzdGFydCB0aGUgY29udGFpbmVyIGJ5IHRyaWdnZXJpbmcgY29udGFpbmVyIHN0YXJ0dXBcbiAgICAvLyBUaGUgZXhpc3RpbmcgZmliZXIgd2lsbCBoYXZlIGV4aXRlZCB3aGVuIHdlIHN0b3BwZWQgdGhlIGNvbnRhaW5lclxuICAgIC8vIFdlIG5lZWQgdG8gc3RhcnQgYSBuZXcgb25lXG4gICAgY29uc3QgZG9ja2VyUnVudGltZSA9IHlpZWxkKiBkb2NrZXIuZ2V0UnVudGltZUluZm8oKVxuICAgIGNvbnN0IHJ1bnRpbWVBcGlIb3N0ID0gZG9ja2VyUnVudGltZS5pc0RvY2tlckRlc2t0b3BcbiAgICAgID8gXCJob3N0LmRvY2tlci5pbnRlcm5hbFwiXG4gICAgICA6IFwicnVudGltZS5hcGlcIlxuXG4gICAgeWllbGQqIEVmZmVjdC5sb2dJbmZvKFxuICAgICAgYFtMb2NhbF0gTmV3IGNvbnRhaW5lciB3aWxsIGNvbm5lY3QgdG8gUnVudGltZSBBUEkgYXQgJHtydW50aW1lQXBpSG9zdH06JHtjb250YWluZXIucG9ydH1gLFxuICAgIClcblxuICAgIGNvbnN0IGNvbnRhaW5lckNvbmZpZyA9IG1ha2VMYW1iZGFDb250YWluZXJDb25maWcoe1xuICAgICAgaW1hZ2VVcmk6IGNvbnRhaW5lci5pbWFnZU5hbWUsXG4gICAgICBydW50aW1lQXBpSG9zdCxcbiAgICAgIHJ1bnRpbWVBcGlQb3J0OiBjb250YWluZXIucG9ydCxcbiAgICAgIGZ1bmN0aW9uTmFtZTogZm4uZnVuY3Rpb25OYW1lLFxuICAgICAgZnVuY3Rpb25WZXJzaW9uOiBcIiRMQVRFU1RcIixcbiAgICAgIG1lbW9yeU1COiBmbi5tZW1vcnlNQixcbiAgICAgIHRpbWVvdXRTZWNvbmRzOiAzNjAwLFxuICAgICAgcGxhdGZvcm0sXG4gICAgfSlcblxuICAgIC8vIEFwcGx5IGV4dGVuc2lvbiB3cmFwcGVyIHRvIHN0YXJ0IGV4dGVuc2lvbnMgYmVmb3JlIHRoZSBhcHBcbiAgICBjb250YWluZXJDb25maWcuZW50cnlwb2ludCA9IGV4dGVuc2lvbldyYXBwZXIuZW50cnlwb2ludFxuICAgIGNvbnRhaW5lckNvbmZpZy5jb21tYW5kID0gZXh0ZW5zaW9uV3JhcHBlci5jb21tYW5kXG5cbiAgICAvLyBTdGFydCB0aGUgbmV3IGNvbnRhaW5lclxuICAgIHlpZWxkKiBFZmZlY3QubG9nRGVidWcoXG4gICAgICBgW0xvY2FsXSBTdGFydGluZyBuZXcgY29udGFpbmVyIGZvciAke2Z1bmN0aW9uSWR9Li4uYCxcbiAgICApXG5cbiAgICAvLyBVc2UgRWZmZWN0LmZvcmtEYWVtb24gdG8gcnVuIHRoZSBjb250YWluZXIgaW5kZXBlbmRlbnRseVxuICAgIGNvbnN0IG5ld0ZpYmVyID0geWllbGQqIGRvY2tlci5ydW4oY29udGFpbmVyQ29uZmlnKS5waXBlKFxuICAgICAgRWZmZWN0LnNjb3BlZCxcbiAgICAgIEVmZmVjdC50YXAoKHJlc3VsdCkgPT5cbiAgICAgICAgRWZmZWN0LmdlbihmdW5jdGlvbiogKCkge1xuICAgICAgICAgIC8vIE9ubHkgc3VwcHJlc3MgZXJyb3JzIGZvciBleHBlY3RlZCBleGl0IGNvZGVzIGZyb20gZG9ja2VyIHN0b3A6XG4gICAgICAgICAgLy8gMDogY2xlYW4sIDE0MzogU0lHVEVSTSwgMTM3OiBTSUdLSUxMXG4gICAgICAgICAgY29uc3QgY29kZSA9IHJlc3VsdC5leGl0Q29kZVxuICAgICAgICAgIGlmIChjb2RlICE9PSAwICYmIGNvZGUgIT09IDE0MyAmJiBjb2RlICE9PSAxMzcpIHtcbiAgICAgICAgICAgIHlpZWxkKiBFZmZlY3QubG9nRXJyb3IoXG4gICAgICAgICAgICAgIGBDb250YWluZXIgZm9yICR7ZnVuY3Rpb25JZH0gZXhpdGVkIHdpdGggY29kZSAke2NvZGV9YCxcbiAgICAgICAgICAgIClcbiAgICAgICAgICAgIHlpZWxkKiBFZmZlY3QubG9nRXJyb3IoYHN0ZGVycjogJHtyZXN1bHQuc3RkZXJyfWApXG4gICAgICAgICAgfVxuICAgICAgICB9KSxcbiAgICAgICksXG4gICAgICBFZmZlY3QubWFwKCgpID0+IHVuZGVmaW5lZCBhcyB2b2lkKSxcbiAgICAgIEVmZmVjdC5jYXRjaEFsbCgoZXJyb3IpID0+XG4gICAgICAgIEVmZmVjdC5sb2dFcnJvcihgQ29udGFpbmVyIGVycm9yIGZvciAke2Z1bmN0aW9uSWR9OiAke2Vycm9yfWApLFxuICAgICAgKSxcbiAgICAgIEVmZmVjdC5mb3JrRGFlbW9uLFxuICAgIClcblxuICAgIC8vIFVwZGF0ZSB0aGUgY29udGFpbmVyIHN0YXRlIHdpdGggdGhlIG5ldyBmaWJlclxuICAgIGNvbnRhaW5lci5jb250YWluZXJGaWJlciA9IG5ld0ZpYmVyXG5cbiAgICAvLyBXYWl0IGEgbW9tZW50IGZvciB0aGUgY29udGFpbmVyIHRvIHN0YXJ0IGFuZCBiZWdpbiBwb2xsaW5nXG4gICAgeWllbGQqIEVmZmVjdC5zbGVlcChcIjIgc2Vjb25kc1wiKVxuXG4gICAgY29udGFpbmVyLmlzUmVidWlsZGluZyA9IGZhbHNlXG4gICAgeWllbGQqIEVmZmVjdC5sb2dEZWJ1ZyhgW0xvY2FsXSBDb250YWluZXIgcmVidWlsdCBmb3IgJHtmdW5jdGlvbklkfWApXG4gIH0pLnBpcGUoRWZmZWN0LnByb3ZpZGUoRG9ja2VyTGl2ZSkpXG5cbi8qKlxuICogSGFuZGxlIGluY29taW5nIGludm9jYXRpb25zIGZvciBhIERvY2tlciBjb250YWluZXIgZnVuY3Rpb24gYnkgcXVldWVpbmcgdGhlbS5cbiAqIFN0YXJ0cyB0aGUgY29udGFpbmVyIGxhemlseSBpZiBub3QgYWxyZWFkeSBydW5uaW5nLlxuICogSW52b2NhdGlvbnMgYXJlIHF1ZXVlZCBldmVuIGlmIGEgcmVidWlsZCBpcyBpbiBwcm9ncmVzcyAtIHRoZSBuZXcgY29udGFpbmVyIHdpbGwgcGljayB0aGVtIHVwLlxuICovXG5jb25zdCBoYW5kbGVEb2NrZXJJbnZvY2F0aW9uID0gKFxuICBmbjogRGlzY292ZXJlZEZ1bmN0aW9uLFxuICBpbnZvY2F0aW9uOiBJbnZvY2F0aW9uTWVzc2FnZSxcbiAgY29udGFpbmVyc1JlZjogUmVmLlJlZjxNYXA8c3RyaW5nLCBGdW5jdGlvbkNvbnRhaW5lcj4+LFxuICBzZXJ2ZXJTY29wZTogU2NvcGUuU2NvcGUsXG4gIHByb2plY3RSb290OiBzdHJpbmcsXG4gIGlkbGVUaW1lb3V0TXM6IG51bWJlcixcbiAgcG9sbFRpbWVvdXRNczogbnVtYmVyLFxuICBhcHBTeW5jQ2xpZW50OiBSZXR1cm5UeXBlPHR5cGVvZiBtYWtlQXBwU3luY0NsaWVudD4sXG4gIGludm9jYXRpb25Db3VudGVyOiBSZWYuUmVmPG51bWJlcj4sXG4gIGludm9jYXRpb25Db250ZXh0czogTWFwPHN0cmluZywgSW52b2NhdGlvbkNvbnRleHQ+LFxuKTogRWZmZWN0LkVmZmVjdDx2b2lkLCBFcnJvciwgQ29tbWFuZEV4ZWN1dG9yLkNvbW1hbmRFeGVjdXRvcj4gPT5cbiAgRWZmZWN0LmdlbihmdW5jdGlvbiogKCkge1xuICAgIC8vIEVuc3VyZSBjb250YWluZXIgaXMgc3RhcnRlZCAobGF6eSBzdGFydHVwIG9uIGZpcnN0IGludm9jYXRpb24pXG4gICAgLy8gVXNlIGVudiBmcm9tIGludm9jYXRpb24gKGNhcHR1cmVkIGZyb20gYnJpZGdlIExhbWJkYSBvbiBmaXJzdCBpbnZva2UpXG4gICAgY29uc3QgY29udGFpbmVyID0geWllbGQqIGVuc3VyZUNvbnRhaW5lclN0YXJ0ZWQoXG4gICAgICBmbixcbiAgICAgIHNhbml0aXplSW52b2NhdGlvbkVudihpbnZvY2F0aW9uLmVudiA/PyB7fSksXG4gICAgICBjb250YWluZXJzUmVmLFxuICAgICAgc2VydmVyU2NvcGUsXG4gICAgICBwcm9qZWN0Um9vdCxcbiAgICAgIGlkbGVUaW1lb3V0TXMsXG4gICAgICBwb2xsVGltZW91dE1zLFxuICAgICAgYXBwU3luY0NsaWVudCxcbiAgICAgIGludm9jYXRpb25Db250ZXh0cyxcbiAgICApXG5cbiAgICAvLyBBc3NpZ24gaW52b2NhdGlvbiBudW1iZXIgYW5kIHRyYWNrIGNvbnRleHQgZm9yIGxvZ2dpbmdcbiAgICBjb25zdCBpbnZvY2F0aW9uTnVtID0geWllbGQqIFJlZi51cGRhdGVBbmRHZXQoXG4gICAgICBpbnZvY2F0aW9uQ291bnRlcixcbiAgICAgIChuKSA9PiBuICsgMSxcbiAgICApXG4gICAgY29uc3Qgc3RhcnRUaW1lID0gRGF0ZS5ub3coKVxuICAgIGludm9jYXRpb25Db250ZXh0cy5zZXQoaW52b2NhdGlvbi5yZXF1ZXN0SWQsIHtcbiAgICAgIG51bTogaW52b2NhdGlvbk51bSxcbiAgICAgIHN0YXJ0OiBzdGFydFRpbWUsXG4gICAgICBmbjogZm4uZnVuY3Rpb25OYW1lLFxuICAgICAgaXNEb2NrZXI6IHRydWUsXG4gICAgfSlcblxuICAgIC8vIFByaW50IHN0YXJ0IG1hcmtlclxuICAgIGNvbnNvbGUubG9nKFxuICAgICAgYFxcblske2ludm9jYXRpb25OdW19XSBcXHUyNTBjXFx1MjUwMFxcdTI1MDAgW0RvY2tlcl0gJHtmbi5mdW5jdGlvbk5hbWV9IFxcdTI1MDBcXHUyNTAwYCxcbiAgICApXG5cbiAgICAvLyBMb2cgaWYgcXVldWluZyBkdXJpbmcgYSByZWJ1aWxkXG4gICAgaWYgKGNvbnRhaW5lci5pc1JlYnVpbGRpbmcpIHtcbiAgICAgIHlpZWxkKiBFZmZlY3QubG9nRGVidWcoXG4gICAgICAgIGBbTG9jYWxdIFF1ZXVlaW5nIGludm9jYXRpb24gJHtpbnZvY2F0aW9uLnJlcXVlc3RJZH0gZm9yICR7Zm4uZnVuY3Rpb25OYW1lfSAocmVidWlsZCBpbiBwcm9ncmVzcywgd2lsbCBiZSBwaWNrZWQgdXAgYnkgbmV3IGNvbnRhaW5lcilgLFxuICAgICAgKVxuICAgIH0gZWxzZSB7XG4gICAgICB5aWVsZCogRWZmZWN0LmxvZ0RlYnVnKFxuICAgICAgICBgW0xvY2FsXSBRdWV1ZWluZyBpbnZvY2F0aW9uICR7aW52b2NhdGlvbi5yZXF1ZXN0SWR9IGZvciAke2ZuLmZ1bmN0aW9uTmFtZX1gLFxuICAgICAgKVxuICAgIH1cblxuICAgIGNvbnN0IGxhbWJkYUludm9jYXRpb246IExhbWJkYUludm9jYXRpb24gPSB7XG4gICAgICByZXF1ZXN0SWQ6IGludm9jYXRpb24ucmVxdWVzdElkLFxuICAgICAgZXZlbnQ6IGludm9jYXRpb24uZXZlbnQsXG4gICAgICBpbnZva2VkRnVuY3Rpb25Bcm46IGludm9jYXRpb24uY29udGV4dC5pbnZva2VkRnVuY3Rpb25Bcm4sXG4gICAgICBkZWFkbGluZU1zOiBEYXRlLm5vdygpICsgaW52b2NhdGlvbi5jb250ZXh0LmdldFJlbWFpbmluZ1RpbWVJbk1pbGxpcyxcbiAgICAgIGZ1bmN0aW9uTmFtZTogZm4uZnVuY3Rpb25OYW1lLFxuICAgICAgZnVuY3Rpb25WZXJzaW9uOiBpbnZvY2F0aW9uLmNvbnRleHQuZnVuY3Rpb25WZXJzaW9uLFxuICAgICAgbWVtb3J5TGltaXRNQjogZm4ubWVtb3J5TUIsXG4gICAgICBsb2dHcm91cE5hbWU6IGludm9jYXRpb24uY29udGV4dC5sb2dHcm91cE5hbWUsXG4gICAgICBsb2dTdHJlYW1OYW1lOiBpbnZvY2F0aW9uLmNvbnRleHQubG9nU3RyZWFtTmFtZSxcbiAgICB9XG5cbiAgICB5aWVsZCogRWZmZWN0LmxvZ0RlYnVnKFxuICAgICAgYFtMb2NhbF0gUXVldWVpbmcgdG8gUnVudGltZSBBUEkgb24gcG9ydCAke2NvbnRhaW5lci5wb3J0fWAsXG4gICAgKVxuICAgIC8vIE5vdGlmeSBleHRlbnNpb25zIGFib3V0IHRoZSBpbnZvY2F0aW9uIChmb3IgTGFtYmRhIFdlYiBBZGFwdGVyIHN1cHBvcnQpXG4gICAgeWllbGQqIG5vdGlmeUV4dGVuc2lvbnNJbnZva2UoY29udGFpbmVyLnJ1bnRpbWVTdGF0ZSwgbGFtYmRhSW52b2NhdGlvbilcbiAgICB5aWVsZCogcXVldWVJbnZvY2F0aW9uKGNvbnRhaW5lci5ydW50aW1lU3RhdGUsIGxhbWJkYUludm9jYXRpb24pXG4gICAgeWllbGQqIEVmZmVjdC5sb2dEZWJ1ZyhgW0xvY2FsXSBJbnZvY2F0aW9uIHF1ZXVlZCBzdWNjZXNzZnVsbHlgKVxuICB9KVxuXG4vKipcbiAqIEhhbmRsZSBpbmNvbWluZyBpbnZvY2F0aW9ucyBmb3IgYSBOb2RlLmpzIGZ1bmN0aW9uIGJ5IHF1ZXVlaW5nIHRoZW0uXG4gKiBTdGFydHMgdGhlIHdvcmtlciBsYXppbHkgaWYgbm90IGFscmVhZHkgcnVubmluZy5cbiAqL1xuY29uc3QgaGFuZGxlTm9kZWpzSW52b2NhdGlvbiA9IChcbiAgZm46IERpc2NvdmVyZWRGdW5jdGlvbixcbiAgaW52b2NhdGlvbjogSW52b2NhdGlvbk1lc3NhZ2UsXG4gIHdvcmtlcnNSZWY6IFJlZi5SZWY8TWFwPHN0cmluZywgTm9kZWpzV29ya2VyPj4sXG4gIHNlcnZlclNjb3BlOiBTY29wZS5TY29wZSxcbiAgcHJvamVjdFJvb3Q6IHN0cmluZyxcbiAgYXBwU3luY0NsaWVudDogUmV0dXJuVHlwZTx0eXBlb2YgbWFrZUFwcFN5bmNDbGllbnQ+LFxuICBpbnZvY2F0aW9uQ291bnRlcjogUmVmLlJlZjxudW1iZXI+LFxuICBpbnZvY2F0aW9uQ29udGV4dHM6IE1hcDxzdHJpbmcsIEludm9jYXRpb25Db250ZXh0Pixcbik6IEVmZmVjdC5FZmZlY3Q8dm9pZCwgRXJyb3I+ID0+XG4gIEVmZmVjdC5nZW4oZnVuY3Rpb24qICgpIHtcbiAgICAvLyBHZXQgZW52IHZhcnMgZnJvbSBpbnZvY2F0aW9uIChmb3J3YXJkZWQgZnJvbSBicmlkZ2UgTGFtYmRhKVxuICAgIGNvbnN0IGludm9jYXRpb25FbnYgPSBzYW5pdGl6ZUludm9jYXRpb25FbnYoaW52b2NhdGlvbi5lbnYgPz8ge30pXG5cbiAgICAvLyBFbnN1cmUgd29ya2VyIGlzIHN0YXJ0ZWQgKGxhenkgc3RhcnR1cCBvbiBmaXJzdCBpbnZvY2F0aW9uKVxuICAgIGNvbnN0IHdvcmtlciA9IHlpZWxkKiBlbnN1cmVXb3JrZXJTdGFydGVkKFxuICAgICAgZm4sXG4gICAgICBpbnZvY2F0aW9uRW52LFxuICAgICAgd29ya2Vyc1JlZixcbiAgICAgIHNlcnZlclNjb3BlLFxuICAgICAgcHJvamVjdFJvb3QsXG4gICAgICBhcHBTeW5jQ2xpZW50LFxuICAgICAgaW52b2NhdGlvbkNvbnRleHRzLFxuICAgIClcblxuICAgIC8vIEFzc2lnbiBpbnZvY2F0aW9uIG51bWJlciBhbmQgdHJhY2sgY29udGV4dCBmb3IgbG9nZ2luZ1xuICAgIGNvbnN0IGludm9jYXRpb25OdW0gPSB5aWVsZCogUmVmLnVwZGF0ZUFuZEdldChcbiAgICAgIGludm9jYXRpb25Db3VudGVyLFxuICAgICAgKG4pID0+IG4gKyAxLFxuICAgIClcbiAgICBjb25zdCBzdGFydFRpbWUgPSBEYXRlLm5vdygpXG4gICAgaW52b2NhdGlvbkNvbnRleHRzLnNldChpbnZvY2F0aW9uLnJlcXVlc3RJZCwge1xuICAgICAgbnVtOiBpbnZvY2F0aW9uTnVtLFxuICAgICAgc3RhcnQ6IHN0YXJ0VGltZSxcbiAgICAgIGZuOiBmbi5mdW5jdGlvbk5hbWUsXG4gICAgICBpc0RvY2tlcjogZmFsc2UsXG4gICAgfSlcblxuICAgIC8vIFByaW50IHN0YXJ0IG1hcmtlclxuICAgIGNvbnNvbGUubG9nKFxuICAgICAgYFxcblske2ludm9jYXRpb25OdW19XSBcXHUyNTBjXFx1MjUwMFxcdTI1MDAgJHtmbi5mdW5jdGlvbk5hbWV9IFxcdTI1MDBcXHUyNTAwYCxcbiAgICApXG5cbiAgICB5aWVsZCogRWZmZWN0LmxvZ0RlYnVnKFxuICAgICAgYFtMb2NhbF0gUXVldWVpbmcgaW52b2NhdGlvbiAke2ludm9jYXRpb24ucmVxdWVzdElkfSBmb3IgJHtmbi5mdW5jdGlvbk5hbWV9YCxcbiAgICApXG5cbiAgICBjb25zdCBsYW1iZGFJbnZvY2F0aW9uOiBMYW1iZGFJbnZvY2F0aW9uID0ge1xuICAgICAgcmVxdWVzdElkOiBpbnZvY2F0aW9uLnJlcXVlc3RJZCxcbiAgICAgIGV2ZW50OiBpbnZvY2F0aW9uLmV2ZW50LFxuICAgICAgaW52b2tlZEZ1bmN0aW9uQXJuOiBpbnZvY2F0aW9uLmNvbnRleHQuaW52b2tlZEZ1bmN0aW9uQXJuLFxuICAgICAgZGVhZGxpbmVNczogRGF0ZS5ub3coKSArIGludm9jYXRpb24uY29udGV4dC5nZXRSZW1haW5pbmdUaW1lSW5NaWxsaXMsXG4gICAgICBmdW5jdGlvbk5hbWU6IGZuLmZ1bmN0aW9uTmFtZSxcbiAgICAgIGZ1bmN0aW9uVmVyc2lvbjogaW52b2NhdGlvbi5jb250ZXh0LmZ1bmN0aW9uVmVyc2lvbixcbiAgICAgIG1lbW9yeUxpbWl0TUI6IGZuLm1lbW9yeU1CLFxuICAgICAgbG9nR3JvdXBOYW1lOiBpbnZvY2F0aW9uLmNvbnRleHQubG9nR3JvdXBOYW1lLFxuICAgICAgbG9nU3RyZWFtTmFtZTogaW52b2NhdGlvbi5jb250ZXh0LmxvZ1N0cmVhbU5hbWUsXG4gICAgfVxuXG4gICAgeWllbGQqIHF1ZXVlSW52b2NhdGlvbih3b3JrZXIucnVudGltZVN0YXRlLCBsYW1iZGFJbnZvY2F0aW9uKVxuICB9KVxuXG4vKipcbiAqIENESyB3YXRjaCBldmVudCB0eXBlcy5cbiAqL1xudHlwZSBDZGtXYXRjaEV2ZW50ID1cbiAgfCB7IHJlYWRvbmx5IF90YWc6IFwiU3RhY2tEaXNjb3ZlcmVkXCI7IHJlYWRvbmx5IHN0YWNrTmFtZTogc3RyaW5nIH1cbiAgfCB7IHJlYWRvbmx5IF90YWc6IFwiRGVwbG95Q29tcGxldGVcIiB9XG5cbi8qKlxuICogU3RhcnQgQ0RLIHdhdGNoIHByb2Nlc3Mgd2l0aCBDREtfTElWRT10cnVlIHVzaW5nIEVmZmVjdCdzIENvbW1hbmQuXG4gKiBSZXR1cm5zIHRoZSBwcm9jZXNzIGFuZCBhIHF1ZXVlIG9mIGV2ZW50cy5cbiAqL1xuY29uc3Qgc3RhcnRDZGtXYXRjaCA9IChcbiAgb3B0aW9uczoge1xuICAgIHByb2ZpbGU/OiBzdHJpbmdcbiAgICByZWdpb24/OiBzdHJpbmdcbiAgICBzdGFja3M/OiBzdHJpbmdbXVxuICAgIGFsbD86IGJvb2xlYW5cbiAgfSxcbiAgc2NvcGU6IFNjb3BlLlNjb3BlLFxuKTogRWZmZWN0LkVmZmVjdDxcbiAgeyBwcm9jZXNzOiBFZmZlY3RQcm9jZXNzOyBldmVudHM6IFF1ZXVlLlF1ZXVlPENka1dhdGNoRXZlbnQ+IH0sXG4gIEVycm9yLFxuICBDb21tYW5kRXhlY3V0b3IuQ29tbWFuZEV4ZWN1dG9yXG4+ID0+XG4gIEVmZmVjdC5nZW4oZnVuY3Rpb24qICgpIHtcbiAgICBjb25zdCBhcmdzID0gW1xuICAgICAgXCJjZGtcIixcbiAgICAgIFwid2F0Y2hcIixcbiAgICAgIFwiLS1ob3Rzd2FwLWZhbGxiYWNrXCIsXG4gICAgICBcIi0tbm8tbG9nc1wiLFxuICAgICAgXCItLW1ldGhvZD1kaXJlY3RcIixcbiAgICBdXG5cbiAgICBpZiAob3B0aW9ucy5zdGFja3MgJiYgb3B0aW9ucy5zdGFja3MubGVuZ3RoID4gMCkge1xuICAgICAgYXJncy5wdXNoKC4uLm9wdGlvbnMuc3RhY2tzKVxuICAgIH0gZWxzZSBpZiAob3B0aW9ucy5hbGwpIHtcbiAgICAgIGFyZ3MucHVzaChcIi0tYWxsXCIpXG4gICAgfVxuXG4gICAgaWYgKG9wdGlvbnMucHJvZmlsZSkge1xuICAgICAgYXJncy5wdXNoKFwiLS1wcm9maWxlXCIsIG9wdGlvbnMucHJvZmlsZSlcbiAgICB9XG5cbiAgICBjb25zdCBlbnY6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4gPSB7XG4gICAgICAuLi5wcm9jZXNzLmVudixcbiAgICAgIENES19MSVZFOiBcInRydWVcIixcbiAgICB9IGFzIFJlY29yZDxzdHJpbmcsIHN0cmluZz5cblxuICAgIGlmIChvcHRpb25zLnJlZ2lvbikge1xuICAgICAgZW52LkFXU19SRUdJT04gPSBvcHRpb25zLnJlZ2lvblxuICAgICAgZW52LkNES19ERUZBVUxUX1JFR0lPTiA9IG9wdGlvbnMucmVnaW9uXG4gICAgfVxuXG4gICAgeWllbGQqIEVmZmVjdC5sb2dEZWJ1ZyhgU3RhcnRpbmc6IG5weCAke2FyZ3Muam9pbihcIiBcIil9YClcblxuICAgIGNvbnN0IGNvbW1hbmQgPSBQbGF0Zm9ybUNvbW1hbmQubWFrZShcIm5weFwiLCAuLi5hcmdzKS5waXBlKFxuICAgICAgUGxhdGZvcm1Db21tYW5kLmVudihlbnYpLFxuICAgICAgUGxhdGZvcm1Db21tYW5kLnJ1bkluU2hlbGwodHJ1ZSksXG4gICAgICBQbGF0Zm9ybUNvbW1hbmQuc3RkaW4oXCJpbmhlcml0XCIpLFxuICAgIClcblxuICAgIC8vIEV4dGVuZCB0aGUgcHJvY2VzcyByZXNvdXJjZSBsaWZldGltZSB0byB0aGUgcHJvdmlkZWQgc2NvcGVcbiAgICBjb25zdCBwcm9jID0geWllbGQqIFBsYXRmb3JtQ29tbWFuZC5zdGFydChjb21tYW5kKS5waXBlKFNjb3BlLmV4dGVuZChzY29wZSkpXG5cbiAgICAvLyBFdmVudCBxdWV1ZSBmb3IgY2FsbGVycyB0byBzdWJzY3JpYmUgdG9cbiAgICBjb25zdCBldmVudHMgPSB5aWVsZCogUXVldWUudW5ib3VuZGVkPENka1dhdGNoRXZlbnQ+KClcblxuICAgIC8vIFN0YXRlIHRyYWNraW5nIGZvciBkZXBsb3kgc3RhdHVzIG1lc3NhZ2VzXG4gICAgbGV0IGlzRGVwbG95aW5nID0gZmFsc2VcbiAgICBsZXQgaXNGaXJzdERlcGxveSA9IHRydWVcbiAgICBsZXQgb3V0cHV0QnVmZmVyID0gXCJcIlxuICAgIGNvbnN0IGRpc2NvdmVyZWRTdGFja3MgPSBuZXcgU2V0PHN0cmluZz4oKVxuXG4gICAgLy8gUGF0dGVybnMgdG8gZGV0ZWN0IENESyB3YXRjaCBiZWhhdmlvclxuICAgIGNvbnN0IGRlcGxveUNvbXBsZXRlUGF0dGVybiA9IC/inIVcXHMrXFxTK3xEZXBsb3ltZW50IHRpbWU6L1xuICAgIGNvbnN0IGRlcGxveVN0YXJ0UGF0dGVybiA9IC9EZXBsb3lpbmd8aG90c3dhcHxIb3Rzd2FwcGluZ3xCdW5kbGluZy9pXG4gICAgY29uc3Qgbm9DaGFuZ2VzUGF0dGVybiA9IC9ubyBjaGFuZ2VzfGlkZW50aWNhbHx1cCB0byBkYXRlL2lcbiAgICBjb25zdCBlcnJvclBhdHRlcm4gPSAvZXJyb3J8ZmFpbGVkfEVycm9yfEZhaWxlZHxFUlIhL2lcbiAgICBjb25zdCBzdGFja05hbWVQYXR0ZXJuID0gL14oXFxTKyk6XFxzKmRlcGxveWluZ3zinIVcXHMrKFxcUyspL1xuXG4gICAgLy8gTWVyZ2Ugc3Rkb3V0IGFuZCBzdGRlcnIsIGRlY29kZSB0byB0ZXh0LCBzcGxpdCBpbnRvIGxpbmVzXG4gICAgY29uc3Qgb3V0cHV0U3RyZWFtID0gU3RyZWFtLm1lcmdlKHByb2Muc3Rkb3V0LCBwcm9jLnN0ZGVycikucGlwZShcbiAgICAgIFN0cmVhbS5kZWNvZGVUZXh0KCksXG4gICAgICBTdHJlYW0uc3BsaXRMaW5lcyxcbiAgICApXG5cbiAgICAvLyBGb3JrIHN0cmVhbSBwcm9jZXNzaW5nIGluIGJhY2tncm91bmRcbiAgICB5aWVsZCogb3V0cHV0U3RyZWFtLnBpcGUoXG4gICAgICBTdHJlYW0ucnVuRm9yRWFjaCgobGluZSkgPT5cbiAgICAgICAgRWZmZWN0LmdlbihmdW5jdGlvbiogKCkge1xuICAgICAgICAgIG91dHB1dEJ1ZmZlciArPSBsaW5lICsgXCJcXG5cIlxuXG4gICAgICAgICAgLy8gTG9nIGFsbCBvdXRwdXQgYXQgZGVidWcgbGV2ZWxcbiAgICAgICAgICB5aWVsZCogRWZmZWN0LmxvZ0RlYnVnKGBbQ0RLXSAke2xpbmV9YClcblxuICAgICAgICAgIC8vIFRyeSB0byBleHRyYWN0IHN0YWNrIG5hbWVcbiAgICAgICAgICBjb25zdCBtYXRjaCA9IHN0YWNrTmFtZVBhdHRlcm4uZXhlYyhsaW5lLnRyaW0oKSlcbiAgICAgICAgICBpZiAobWF0Y2gpIHtcbiAgICAgICAgICAgIGNvbnN0IHN0YWNrTmFtZSA9IG1hdGNoWzFdIHx8IG1hdGNoWzJdXG4gICAgICAgICAgICBpZiAoc3RhY2tOYW1lICYmICFkaXNjb3ZlcmVkU3RhY2tzLmhhcyhzdGFja05hbWUpKSB7XG4gICAgICAgICAgICAgIGRpc2NvdmVyZWRTdGFja3MuYWRkKHN0YWNrTmFtZSlcbiAgICAgICAgICAgICAgeWllbGQqIFF1ZXVlLm9mZmVyKGV2ZW50cywge1xuICAgICAgICAgICAgICAgIF90YWc6IFwiU3RhY2tEaXNjb3ZlcmVkXCIsXG4gICAgICAgICAgICAgICAgc3RhY2tOYW1lLFxuICAgICAgICAgICAgICB9KVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIERldGVjdCBkZXBsb3kgc3RhcnRcbiAgICAgICAgICBpZiAoIWlzRGVwbG95aW5nICYmIGRlcGxveVN0YXJ0UGF0dGVybi50ZXN0KGxpbmUpKSB7XG4gICAgICAgICAgICBpc0RlcGxveWluZyA9IHRydWVcbiAgICAgICAgICAgIGlmICghaXNGaXJzdERlcGxveSkge1xuICAgICAgICAgICAgICB5aWVsZCogRWZmZWN0LmxvZ0luZm8oXCJbQ0RLXSBEZXBsb3lpbmcuLi5cIilcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG5cbiAgICAgICAgICAvLyBEZXRlY3QgZXJyb3JzIC0gb3V0cHV0IGltbWVkaWF0ZWx5XG4gICAgICAgICAgaWYgKGVycm9yUGF0dGVybi50ZXN0KGxpbmUpKSB7XG4gICAgICAgICAgICB5aWVsZCogRWZmZWN0LnN5bmMoKCkgPT4gcHJvY2Vzcy5zdGRlcnIud3JpdGUobGluZSArIFwiXFxuXCIpKVxuICAgICAgICAgIH1cblxuICAgICAgICAgIC8vIENoZWNrIGZvciBkZXBsb3kgY29tcGxldGlvbiAob25seSB0cmlnZ2VyIG9uY2UgcGVyIGRlcGxveSBjeWNsZSlcbiAgICAgICAgICBpZiAoaXNEZXBsb3lpbmcgJiYgZGVwbG95Q29tcGxldGVQYXR0ZXJuLnRlc3QobGluZSkpIHtcbiAgICAgICAgICAgIGlzRGVwbG95aW5nID0gZmFsc2VcbiAgICAgICAgICAgIGlzRmlyc3REZXBsb3kgPSBmYWxzZVxuICAgICAgICAgICAgb3V0cHV0QnVmZmVyID0gXCJcIlxuICAgICAgICAgICAgeWllbGQqIEVmZmVjdC5sb2dJbmZvKFwiW0NES10gRGVwbG95IGNvbXBsZXRlXCIpXG4gICAgICAgICAgICAvLyBTbWFsbCBkZWxheSB0byBlbnN1cmUgQVdTIGhhcyBwcm9wYWdhdGVkIHRoZSBjaGFuZ2VzXG4gICAgICAgICAgICB5aWVsZCogRWZmZWN0LnNsZWVwKFwiMSBzZWNvbmRcIilcbiAgICAgICAgICAgIHlpZWxkKiBRdWV1ZS5vZmZlcihldmVudHMsIHsgX3RhZzogXCJEZXBsb3lDb21wbGV0ZVwiIH0pXG4gICAgICAgICAgfVxuXG4gICAgICAgICAgLy8gQ2hlY2sgZm9yIG5vIGNoYW5nZXNcbiAgICAgICAgICBpZiAobm9DaGFuZ2VzUGF0dGVybi50ZXN0KGxpbmUpKSB7XG4gICAgICAgICAgICBpc0RlcGxveWluZyA9IGZhbHNlXG4gICAgICAgICAgICBvdXRwdXRCdWZmZXIgPSBcIlwiXG4gICAgICAgICAgfVxuICAgICAgICB9KSxcbiAgICAgICksXG4gICAgICAvLyBMb2cgZXJyb3JzIGFuZCBleGl0IGNvZGUgd2hlbiBzdHJlYW0gZW5kc1xuICAgICAgRWZmZWN0LnRhcEVycm9yKChlcnJvcikgPT5cbiAgICAgICAgRWZmZWN0LmxvZ0Vycm9yKGBbQ0RLXSBDREsgd2F0Y2ggZXJyb3I6ICR7ZXJyb3J9YCksXG4gICAgICApLFxuICAgICAgRWZmZWN0LmVuc3VyaW5nKFxuICAgICAgICBwcm9jLmV4aXRDb2RlLnBpcGUoXG4gICAgICAgICAgRWZmZWN0LmZsYXRNYXAoKGNvZGUpID0+XG4gICAgICAgICAgICBjb2RlICE9PSAwXG4gICAgICAgICAgICAgID8gRWZmZWN0LmxvZ0Vycm9yKGBbQ0RLXSBDREsgd2F0Y2ggZXhpdGVkIHdpdGggY29kZSAke2NvZGV9YClcbiAgICAgICAgICAgICAgOiBFZmZlY3QubG9nRGVidWcoYFtDREtdIENESyB3YXRjaCBleGl0ZWQgd2l0aCBjb2RlICR7Y29kZX1gKSxcbiAgICAgICAgICApLFxuICAgICAgICAgIEVmZmVjdC5jYXRjaEFsbCgoKSA9PiBFZmZlY3Qudm9pZCksXG4gICAgICAgICksXG4gICAgICApLFxuICAgICAgRWZmZWN0LmZvcmssXG4gICAgKVxuXG4gICAgcmV0dXJuIHsgcHJvY2VzczogcHJvYywgZXZlbnRzIH1cbiAgfSlcblxuLyoqXG4gKiBDb21tb24gQ0xJIG9wdGlvbnMuXG4gKi9cbmNvbnN0IHByb2ZpbGVPcHRpb24gPSBPcHRpb25zLnRleHQoXCJwcm9maWxlXCIpLnBpcGUoXG4gIE9wdGlvbnMub3B0aW9uYWwsXG4gIE9wdGlvbnMud2l0aERlc2NyaXB0aW9uKFwiQVdTIHByb2ZpbGUgdG8gdXNlXCIpLFxuKVxuXG5jb25zdCByZWdpb25PcHRpb24gPSBPcHRpb25zLnRleHQoXCJyZWdpb25cIikucGlwZShcbiAgT3B0aW9ucy5vcHRpb25hbCxcbiAgT3B0aW9ucy53aXRoRGVzY3JpcHRpb24oXCJBV1MgcmVnaW9uXCIpLFxuKVxuXG5jb25zdCBxdWFsaWZpZXJPcHRpb24gPSBPcHRpb25zLnRleHQoXCJxdWFsaWZpZXJcIikucGlwZShcbiAgT3B0aW9ucy53aXRoRGVmYXVsdChcImhuYjY1OWZkc1wiKSxcbiAgT3B0aW9ucy53aXRoRGVzY3JpcHRpb24oXCJDREsgYm9vdHN0cmFwIHF1YWxpZmllclwiKSxcbilcblxuY29uc3Qgc3RhY2tzT3B0aW9uID0gT3B0aW9ucy50ZXh0KFwic3RhY2tzXCIpLnBpcGUoXG4gIE9wdGlvbnMub3B0aW9uYWwsXG4gIE9wdGlvbnMud2l0aERlc2NyaXB0aW9uKFxuICAgIFwiU3RhY2sgbmFtZXMgdG8gZGVwbG95IChjb21tYS1zZXBhcmF0ZWQsIGRlZmF1bHQ6IGFsbClcIixcbiAgKSxcbilcblxuY29uc3QgYWxsU3RhY2tzT3B0aW9uID0gT3B0aW9ucy5ib29sZWFuKFwiYWxsXCIpLnBpcGUoXG4gIE9wdGlvbnMud2l0aERlZmF1bHQoZmFsc2UpLFxuICBPcHRpb25zLndpdGhEZXNjcmlwdGlvbihcIkRlcGxveSBhbGwgc3RhY2tzIChsaWtlIGNkayBkZXBsb3kgLS1hbGwpXCIpLFxuKVxuXG5jb25zdCBkZWJ1Z09wdGlvbiA9IE9wdGlvbnMuYm9vbGVhbihcImRlYnVnXCIpLnBpcGUoXG4gIE9wdGlvbnMud2l0aERlZmF1bHQoZmFsc2UpLFxuICBPcHRpb25zLndpdGhEZXNjcmlwdGlvbihcIkVuYWJsZSBkZWJ1ZyBsb2dnaW5nXCIpLFxuKVxuXG5jb25zdCBpZGxlVGltZW91dE9wdGlvbiA9IE9wdGlvbnMuaW50ZWdlcihcImlkbGUtdGltZW91dFwiKS5waXBlKFxuICBPcHRpb25zLndpdGhEZWZhdWx0KERFRkFVTFRfSURMRV9USU1FT1VUX01TKSxcbiAgT3B0aW9ucy53aXRoRGVzY3JpcHRpb24oXG4gICAgXCJJZGxlIHRpbWVvdXQgaW4gbXMgYmVmb3JlIHN0b3BwaW5nIGNvbnRhaW5lcnMgKGRlZmF1bHQ6IDIzMDAwMClcIixcbiAgKSxcbilcblxuY29uc3QgcG9sbFRpbWVvdXRPcHRpb24gPSBPcHRpb25zLmludGVnZXIoXCJwb2xsLXRpbWVvdXRcIikucGlwZShcbiAgT3B0aW9ucy5vcHRpb25hbCxcbiAgT3B0aW9ucy53aXRoRGVzY3JpcHRpb24oXG4gICAgXCJQb2xsIHRpbWVvdXQgaW4gbXMgZm9yIFJ1bnRpbWUgQVBJIChkZWZhdWx0OiBpZGxlLXRpbWVvdXQgLSAxMHMpLiBNdXN0IGJlIHNob3J0ZXIgdGhhbiBpZGxlLXRpbWVvdXQuXCIsXG4gICksXG4pXG5cbi8qKlxuICogTG9jYWwgY29tbWFuZCBkZWZpbml0aW9uLlxuICovXG5leHBvcnQgY29uc3QgbG9jYWxDb21tYW5kID0gQ29tbWFuZC5tYWtlKFxuICBcImxvY2FsXCIsXG4gIHtcbiAgICBwcm9maWxlOiBwcm9maWxlT3B0aW9uLFxuICAgIHJlZ2lvbjogcmVnaW9uT3B0aW9uLFxuICAgIHF1YWxpZmllcjogcXVhbGlmaWVyT3B0aW9uLFxuICAgIHN0YWNrczogc3RhY2tzT3B0aW9uLFxuICAgIGFsbDogYWxsU3RhY2tzT3B0aW9uLFxuICAgIGRlYnVnOiBkZWJ1Z09wdGlvbixcbiAgICBpZGxlVGltZW91dDogaWRsZVRpbWVvdXRPcHRpb24sXG4gICAgcG9sbFRpbWVvdXQ6IHBvbGxUaW1lb3V0T3B0aW9uLFxuICB9LFxuICAoe1xuICAgIHByb2ZpbGUsXG4gICAgcmVnaW9uLFxuICAgIHF1YWxpZmllcixcbiAgICBzdGFja3MsXG4gICAgYWxsLFxuICAgIGRlYnVnLFxuICAgIGlkbGVUaW1lb3V0LFxuICAgIHBvbGxUaW1lb3V0LFxuICB9KSA9PiB7XG4gICAgY29uc3QgbG9nTGV2ZWwgPSBkZWJ1ZyA/IExvZ0xldmVsLkRlYnVnIDogTG9nTGV2ZWwuSW5mb1xuICAgIHJldHVybiBFZmZlY3QuZ2VuKGZ1bmN0aW9uKiAoKSB7XG4gICAgICB5aWVsZCogRWZmZWN0LmxvZ0luZm8oXCJbTG9jYWxdIFN0YXJ0aW5nIGxvY2FsIExhbWJkYSBkZXZlbG9wbWVudC4uLlwiKVxuXG4gICAgICBjb25zdCBwcm9maWxlVmFsdWUgPSBwcm9maWxlLl90YWcgPT09IFwiU29tZVwiID8gcHJvZmlsZS52YWx1ZSA6IHVuZGVmaW5lZFxuICAgICAgY29uc3QgcmVnaW9uVmFsdWUgPSByZWdpb24uX3RhZyA9PT0gXCJTb21lXCIgPyByZWdpb24udmFsdWUgOiB1bmRlZmluZWRcbiAgICAgIGNvbnN0IHN0YWNrc0Zyb21PcHRpb24gPVxuICAgICAgICBzdGFja3MuX3RhZyA9PT0gXCJTb21lXCJcbiAgICAgICAgICA/IHN0YWNrcy52YWx1ZS5zcGxpdChcIixcIikubWFwKChzKSA9PiBzLnRyaW0oKSlcbiAgICAgICAgICA6IHVuZGVmaW5lZFxuXG4gICAgICAvLyBQb2xsIHRpbWVvdXQgc2hvdWxkIGJlIHNob3J0ZXIgdGhhbiBpZGxlIHRpbWVvdXQgc28gY29udGFpbmVycyBleGl0IG5hdHVyYWxseVxuICAgICAgLy8gYmVmb3JlIHdlIHRyeSB0byBzdG9wIHRoZW0uIERlZmF1bHQ6IGlkbGUgdGltZW91dCAtIDEwIHNlY29uZHMuXG4gICAgICBjb25zdCBlZmZlY3RpdmVQb2xsVGltZW91dCA9XG4gICAgICAgIHBvbGxUaW1lb3V0Ll90YWcgPT09IFwiU29tZVwiXG4gICAgICAgICAgPyBwb2xsVGltZW91dC52YWx1ZVxuICAgICAgICAgIDogTWF0aC5tYXgoaWRsZVRpbWVvdXQgLSAxMF8wMDAsIDVfMDAwKSAvLyBBdCBsZWFzdCA1IHNlY29uZHNcblxuICAgICAgaWYgKHByb2ZpbGVWYWx1ZSkge1xuICAgICAgICBwcm9jZXNzLmVudi5BV1NfUFJPRklMRSA9IHByb2ZpbGVWYWx1ZVxuICAgICAgfVxuICAgICAgaWYgKHJlZ2lvblZhbHVlKSB7XG4gICAgICAgIHByb2Nlc3MuZW52LkFXU19SRUdJT04gPSByZWdpb25WYWx1ZVxuICAgICAgfVxuXG4gICAgICAvLyBCb290c3RyYXAgY2hlY2sgbXVzdCBjb21wbGV0ZSBmaXJzdFxuICAgICAgeWllbGQqIGVuc3VyZUJvb3RzdHJhcCh7XG4gICAgICAgIHF1YWxpZmllcixcbiAgICAgICAgcHJvZmlsZTogcHJvZmlsZVZhbHVlLFxuICAgICAgICByZWdpb246IHJlZ2lvblZhbHVlLFxuICAgICAgfSlcblxuICAgICAgLy8gU3RhY2sgZmlsdGVyIC0gcG9wdWxhdGVkIGZyb20gQ0RLIHdhdGNoIG91dHB1dCwgdXNlZCB0byBmaWx0ZXIgTGFtYmRhIGRpc2NvdmVyeVxuICAgICAgLy8gVXNpbmcgb2JqZWN0IHNvIGNsb3N1cmUgaW4gc3RhcnRPclVwZGF0ZURhZW1vbiBzZWVzIHVwZGF0ZXNcbiAgICAgIGNvbnN0IGZpbHRlclN0YXRlID0geyBzdGFja3M6IHN0YWNrc0Zyb21PcHRpb24gPz8gKFtdIGFzIHN0cmluZ1tdKSB9XG5cbiAgICAgIC8vIENyZWF0ZSBhIGxvbmctbGl2ZWQgc2NvcGUgZm9yIGFsbCBSdW50aW1lIEFQSSBzZXJ2ZXJzIGFuZCBDREsgd2F0Y2ggcHJvY2Vzc1xuICAgICAgLy8gUmVzb3VyY2VzIHdpbGwgcnVuIHVudGlsIHRoaXMgc2NvcGUgaXMgY2xvc2VkICh3aGVuIHRoZSBwcm9ncmFtIGVuZHMpXG4gICAgICBjb25zdCBzZXJ2ZXJTY29wZSA9IHlpZWxkKiBTY29wZS5tYWtlKClcblxuICAgICAgLy8gU3RhcnQgQ0RLIHdhdGNoIGltbWVkaWF0ZWx5IGFmdGVyIGJvb3RzdHJhcFxuICAgICAgLy8gU3RhY2sgbmFtZXMgYXJlIGRpc2NvdmVyZWQgZnJvbSBDREsgd2F0Y2ggb3V0cHV0IChubyBuZWVkIGZvciBzZXBhcmF0ZSBjZGsgbHMpXG4gICAgICB5aWVsZCogRWZmZWN0LmxvZ0luZm8oXCJbQ0RLXSBEZXBsb3lpbmcuLi5cIilcblxuICAgICAgY29uc3QgeyBwcm9jZXNzOiBjZGtXYXRjaFByb2Nlc3MsIGV2ZW50czogY2RrRXZlbnRzIH0gPVxuICAgICAgICB5aWVsZCogc3RhcnRDZGtXYXRjaChcbiAgICAgICAgICB7XG4gICAgICAgICAgICBwcm9maWxlOiBwcm9maWxlVmFsdWUsXG4gICAgICAgICAgICByZWdpb246IHJlZ2lvblZhbHVlLFxuICAgICAgICAgICAgc3RhY2tzOiBzdGFja3NGcm9tT3B0aW9uLFxuICAgICAgICAgICAgYWxsLFxuICAgICAgICAgIH0sXG4gICAgICAgICAgc2VydmVyU2NvcGUsXG4gICAgICAgIClcblxuICAgICAgLy8gVHJhY2sgcnVubmluZyBEb2NrZXIgY29udGFpbmVycyBieSBmdW5jdGlvbiBuYW1lXG4gICAgICBjb25zdCBjb250YWluZXJzID0geWllbGQqIFJlZi5tYWtlPE1hcDxzdHJpbmcsIEZ1bmN0aW9uQ29udGFpbmVyPj4oXG4gICAgICAgIG5ldyBNYXAoKSxcbiAgICAgIClcblxuICAgICAgLy8gVHJhY2sgcnVubmluZyBOb2RlLmpzIHdvcmtlcnMgYnkgZnVuY3Rpb24gbmFtZVxuICAgICAgY29uc3Qgd29ya2VycyA9IHlpZWxkKiBSZWYubWFrZTxNYXA8c3RyaW5nLCBOb2RlanNXb3JrZXI+PihuZXcgTWFwKCkpXG5cbiAgICAgIC8vIFRyYWNrIHJlZ2lzdGVyZWQgZnVuY3Rpb25zIChmb3IgbGF6eSBjb250YWluZXIvd29ya2VyIHN0YXJ0dXApXG4gICAgICBjb25zdCByZWdpc3RlcmVkRnVuY3Rpb25zID0geWllbGQqIFJlZi5tYWtlPFxuICAgICAgICBNYXA8c3RyaW5nLCBEaXNjb3ZlcmVkRnVuY3Rpb24+XG4gICAgICA+KG5ldyBNYXAoKSlcblxuICAgICAgLy8gVHJhY2sgRG9ja2VyIGZ1bmN0aW9ucyB3aXRoIGFjdGl2ZSBmaWxlIHdhdGNoZXJzXG4gICAgICBjb25zdCB3YXRjaGVkRG9ja2VyRnVuY3Rpb25zID0gbmV3IFNldDxzdHJpbmc+KClcblxuICAgICAgLy8gSW52b2NhdGlvbiB0cmFja2luZyBmb3IgaW1wcm92ZWQgbG9nZ2luZyBvdXRwdXRcbiAgICAgIGNvbnN0IGludm9jYXRpb25Db3VudGVyID0geWllbGQqIFJlZi5tYWtlPG51bWJlcj4oMClcbiAgICAgIC8vIFVzZSBhIHBsYWluIE1hcCBmb3IgaW52b2NhdGlvbiBjb250ZXh0cyBzbyBpdCBjYW4gYmUgYWNjZXNzZWQgZnJvbSBzdHJlYW0gY2FsbGJhY2tzXG4gICAgICBjb25zdCBpbnZvY2F0aW9uQ29udGV4dHMgPSBuZXcgTWFwPHN0cmluZywgSW52b2NhdGlvbkNvbnRleHQ+KClcblxuICAgICAgLy8gUHJvamVjdCByb290IGlzIHRoZSBjdXJyZW50IHdvcmtpbmcgZGlyZWN0b3J5ICh3aGVyZSBDREsgYXBwIGxpdmVzKVxuICAgICAgY29uc3QgcHJvamVjdFJvb3QgPSBwcm9jZXNzLmN3ZCgpXG5cbiAgICAgIGxldCBhcHBTeW5jQ2xpZW50OiBSZXR1cm5UeXBlPHR5cGVvZiBtYWtlQXBwU3luY0NsaWVudD4gfCBudWxsID0gbnVsbFxuXG4gICAgICAvLyBUcmFjayBpZiB3ZSd2ZSBsb2dnZWQgdGhlIFwiV2F0Y2hpbmdcIiBtZXNzYWdlIChvbmx5IGxvZyBvbmNlKVxuICAgICAgY29uc3QgbG9nU3RhdGUgPSB7IGhhc0xvZ2dlZFdhdGNoaW5nOiBmYWxzZSwgaGFzRGlzY292ZXJlZE9uY2U6IGZhbHNlIH1cblxuICAgICAgLy8gRnVuY3Rpb24gdG8gc3RhcnQvdXBkYXRlIHRoZSBkYWVtb24gd2l0aCBkaXNjb3ZlcmVkIGZ1bmN0aW9uc1xuICAgICAgY29uc3Qgc3RhcnRPclVwZGF0ZURhZW1vbiA9IEVmZmVjdC5nZW4oZnVuY3Rpb24qICgpIHtcbiAgICAgICAgLy8gT25seSBzaG93IFwiRGlzY292ZXJpbmcgZnVuY3Rpb25zLi4uXCIgb24gZmlyc3QgcnVuXG4gICAgICAgIGlmICghbG9nU3RhdGUuaGFzRGlzY292ZXJlZE9uY2UpIHtcbiAgICAgICAgICB5aWVsZCogRWZmZWN0LmxvZ0luZm8oXCJbTG9jYWxdIERpc2NvdmVyaW5nIGZ1bmN0aW9ucy4uLlwiKVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gR2V0IEFwcFN5bmMgZW5kcG9pbnRzIChtYXkgbmVlZCB0byB3YWl0IGZvciBmaXJzdCBkZXBsb3kpXG4gICAgICAgIGlmICghYXBwU3luY0NsaWVudCkge1xuICAgICAgICAgIHlpZWxkKiBFZmZlY3QubG9nRGVidWcoXG4gICAgICAgICAgICBcIltMb2NhbF0gUmVhZGluZyBBcHBTeW5jIGVuZHBvaW50cyBmcm9tIFNTTS4uLlwiLFxuICAgICAgICAgIClcbiAgICAgICAgICBjb25zdCBlbmRwb2ludHMgPSB5aWVsZCogZ2V0QXBwU3luY0VuZHBvaW50cyhxdWFsaWZpZXIpLnBpcGUoXG4gICAgICAgICAgICBFZmZlY3QucmV0cnkoeyB0aW1lczogMTAsIHNjaGVkdWxlOiBTY2hlZHVsZS5zcGFjZWQoXCIzIHNlY29uZHNcIikgfSksXG4gICAgICAgICAgKVxuICAgICAgICAgIHlpZWxkKiBFZmZlY3QubG9nRGVidWcoXG4gICAgICAgICAgICBgW0xvY2FsXSBIVFRQIGVuZHBvaW50OiAke2VuZHBvaW50cy5odHRwRW5kcG9pbnR9YCxcbiAgICAgICAgICApXG4gICAgICAgICAgYXBwU3luY0NsaWVudCA9IG1ha2VBcHBTeW5jQ2xpZW50KGVuZHBvaW50cylcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIERpc2NvdmVyIGZ1bmN0aW9ucyAoZmlsdGVyZWQgdG8gc3RhY2tzIGluIHRoaXMgQ0RLIHByb2plY3QpXG4gICAgICAgIHlpZWxkKiBFZmZlY3QubG9nRGVidWcoXCJbTG9jYWxdIERpc2NvdmVyaW5nIExhbWJkYSBmdW5jdGlvbnMuLi5cIilcbiAgICAgICAgY29uc3QgZnVuY3Rpb25zID0geWllbGQqIGRpc2NvdmVyRnVuY3Rpb25zKGZpbHRlclN0YXRlLnN0YWNrcylcblxuICAgICAgICBpZiAoZnVuY3Rpb25zLmxlbmd0aCA9PT0gMCkge1xuICAgICAgICAgIHlpZWxkKiBFZmZlY3QubG9nSW5mbyhcbiAgICAgICAgICAgIFwiW0xvY2FsXSBObyBmdW5jdGlvbnMgZm91bmQgd2l0aCBsaXZlLWxhbWJkYSB0YWdzIHlldC4gSGF2ZSB5b3UgcGF0Y2hlZCB5b3VyIENESyBwcm9qZWN0IChib290c3RyYXApIGFuZCBhZGRlZCB0aGUgTGl2ZUxhbWJkYUFzcGVjdD9cIixcbiAgICAgICAgICApXG4gICAgICAgICAgcmV0dXJuXG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBjdXJyZW50UmVnaXN0ZXJlZCA9IHlpZWxkKiBSZWYuZ2V0KHJlZ2lzdGVyZWRGdW5jdGlvbnMpXG5cbiAgICAgICAgLy8gQ29sbGVjdCBuZXcgZnVuY3Rpb25zIHRvIHJlZ2lzdGVyXG4gICAgICAgIGNvbnN0IG5ld0Z1bmN0aW9uczogQXJyYXk8e1xuICAgICAgICAgIGZuOiBEaXNjb3ZlcmVkRnVuY3Rpb25cbiAgICAgICAgICBpc0RvY2tlcjogYm9vbGVhblxuICAgICAgICB9PiA9IFtdXG5cbiAgICAgICAgZm9yIChjb25zdCBmbiBvZiBmdW5jdGlvbnMpIHtcbiAgICAgICAgICBjb25zdCBpc0RvY2tlciA9IEJvb2xlYW4oZm4uZG9ja2VyQ29udGV4dFBhdGgpXG4gICAgICAgICAgY29uc3QgaXNOb2RlanMgPSAhaXNEb2NrZXIgJiYgQm9vbGVhbihmbi5sb2NhbEhhbmRsZXIpXG5cbiAgICAgICAgICBpZiAoIWlzRG9ja2VyICYmICFpc05vZGVqcykge1xuICAgICAgICAgICAgeWllbGQqIEVmZmVjdC5sb2dEZWJ1ZyhcbiAgICAgICAgICAgICAgYFtMb2NhbF0gU2tpcHBpbmcgJHtmbi5mdW5jdGlvbk5hbWV9IC0gbm8gRG9ja2VyIGNvbnRleHQgb3IgbG9jYWwgaGFuZGxlcmAsXG4gICAgICAgICAgICApXG4gICAgICAgICAgICBjb250aW51ZVxuICAgICAgICAgIH1cblxuICAgICAgICAgIGlmIChjdXJyZW50UmVnaXN0ZXJlZC5oYXMoZm4uZnVuY3Rpb25OYW1lKSkge1xuICAgICAgICAgICAgeWllbGQqIEVmZmVjdC5sb2dEZWJ1ZyhcbiAgICAgICAgICAgICAgYFtMb2NhbF0gQWxyZWFkeSB3YXRjaGluZyAke2ZuLmZ1bmN0aW9uTmFtZX1gLFxuICAgICAgICAgICAgKVxuICAgICAgICAgICAgY29udGludWVcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBuZXdGdW5jdGlvbnMucHVzaCh7IGZuLCBpc0RvY2tlciB9KVxuICAgICAgICB9XG5cbiAgICAgICAgLy8gUHJpbnQgc3VtbWFyeSBvZiBkaXNjb3ZlcmVkIGZ1bmN0aW9uc1xuICAgICAgICBpZiAobmV3RnVuY3Rpb25zLmxlbmd0aCA+IDApIHtcbiAgICAgICAgICBjb25zdCBzdW1tYXJ5ID0gbmV3RnVuY3Rpb25zXG4gICAgICAgICAgICAubWFwKCh7IGZuLCBpc0RvY2tlciB9KSA9PiB7XG4gICAgICAgICAgICAgIGNvbnN0IG1vZGUgPSBpc0RvY2tlciA/IFwiZG9ja2VyXCIgOiBcIm5vZGVcIlxuICAgICAgICAgICAgICBjb25zdCBzaG9ydE5hbWUgPSBmbi5mdW5jdGlvbk5hbWUuaW5jbHVkZXMoXCItXCIpXG4gICAgICAgICAgICAgICAgPyBmbi5mdW5jdGlvbk5hbWUuc3BsaXQoXCItXCIpLnNsaWNlKC0yLCAtMSlbMF0gfHwgZm4uZnVuY3Rpb25OYW1lXG4gICAgICAgICAgICAgICAgOiBmbi5mdW5jdGlvbk5hbWVcbiAgICAgICAgICAgICAgcmV0dXJuIGAke3Nob3J0TmFtZX0gKCR7bW9kZX0pYFxuICAgICAgICAgICAgfSlcbiAgICAgICAgICAgIC5qb2luKFwiLCBcIilcbiAgICAgICAgICAvLyBVc2UgZGlmZmVyZW50IG1lc3NhZ2UgZm9yIGZpcnN0IGRpc2NvdmVyeSB2cyBzdWJzZXF1ZW50XG4gICAgICAgICAgY29uc3QgcHJlZml4ID0gbG9nU3RhdGUuaGFzRGlzY292ZXJlZE9uY2VcbiAgICAgICAgICAgID8gXCJbTG9jYWxdIEZ1bmN0aW9ucyBhZGRlZDpcIlxuICAgICAgICAgICAgOiBcIltMb2NhbF0gRnVuY3Rpb25zOlwiXG4gICAgICAgICAgeWllbGQqIEVmZmVjdC5sb2dJbmZvKGAke3ByZWZpeH0gJHtzdW1tYXJ5fWApXG4gICAgICAgIH1cblxuICAgICAgICAvLyBSZWdpc3RlciBmdW5jdGlvbnMgYW5kIHNldCB1cCBzdWJzY3JpcHRpb25zXG4gICAgICAgIGZvciAoY29uc3QgeyBmbiwgaXNEb2NrZXIgfSBvZiBuZXdGdW5jdGlvbnMpIHtcbiAgICAgICAgICB5aWVsZCogRWZmZWN0LmxvZ0RlYnVnKFxuICAgICAgICAgICAgYFtMb2NhbF0gUmVnaXN0ZXJpbmcgJHtmbi5mdW5jdGlvbk5hbWV9ICgke2lzRG9ja2VyID8gXCJEb2NrZXJcIiA6IFwiTm9kZS5qc1wifSlgLFxuICAgICAgICAgIClcblxuICAgICAgICAgIC8vIFJlZ2lzdGVyIHRoZSBmdW5jdGlvblxuICAgICAgICAgIGN1cnJlbnRSZWdpc3RlcmVkLnNldChmbi5mdW5jdGlvbk5hbWUsIGZuKVxuXG4gICAgICAgICAgLy8gU3Vic2NyaWJlIHRvIGludm9jYXRpb25zIGZvciB0aGlzIGZ1bmN0aW9uXG4gICAgICAgICAgY29uc3QgaW52b2NhdGlvbkNoYW5uZWwgPSBidWlsZENoYW5uZWxOYW1lLmludm9jYXRpb24oZm4uZnVuY3Rpb25OYW1lKVxuICAgICAgICAgIHlpZWxkKiBFZmZlY3QubG9nRGVidWcoXG4gICAgICAgICAgICBgW0xvY2FsXSBTdWJzY3JpYmluZyB0byBpbnZvY2F0aW9ucyBmb3IgJHtmbi5mdW5jdGlvbk5hbWV9YCxcbiAgICAgICAgICApXG5cbiAgICAgICAgICAvLyBTdWJzY3JpYmUgdXNpbmcgZm9ya0RhZW1vbiB0byBydW4gaW5kZXBlbmRlbnRseSB3aXRoIGNvbnRleHQgcHJlc2VydmVkXG4gICAgICAgICAgLy8gUm91dGUgdG8gRG9ja2VyIG9yIE5vZGUuanMgaGFuZGxlciBiYXNlZCBvbiBmdW5jdGlvbiB0eXBlXG4gICAgICAgICAgaWYgKGlzRG9ja2VyKSB7XG4gICAgICAgICAgICB5aWVsZCogYXBwU3luY0NsaWVudCFcbiAgICAgICAgICAgICAgLnN1YnNjcmliZVRvSW52b2NhdGlvbnMoaW52b2NhdGlvbkNoYW5uZWwpXG4gICAgICAgICAgICAgIC5waXBlKFxuICAgICAgICAgICAgICAgIFN0cmVhbS5ydW5Gb3JFYWNoKChpbnZvY2F0aW9uKSA9PlxuICAgICAgICAgICAgICAgICAgaGFuZGxlRG9ja2VySW52b2NhdGlvbihcbiAgICAgICAgICAgICAgICAgICAgZm4sXG4gICAgICAgICAgICAgICAgICAgIGludm9jYXRpb24sXG4gICAgICAgICAgICAgICAgICAgIGNvbnRhaW5lcnMsXG4gICAgICAgICAgICAgICAgICAgIHNlcnZlclNjb3BlLFxuICAgICAgICAgICAgICAgICAgICBwcm9qZWN0Um9vdCxcbiAgICAgICAgICAgICAgICAgICAgaWRsZVRpbWVvdXQsXG4gICAgICAgICAgICAgICAgICAgIGVmZmVjdGl2ZVBvbGxUaW1lb3V0LFxuICAgICAgICAgICAgICAgICAgICBhcHBTeW5jQ2xpZW50ISxcbiAgICAgICAgICAgICAgICAgICAgaW52b2NhdGlvbkNvdW50ZXIsXG4gICAgICAgICAgICAgICAgICAgIGludm9jYXRpb25Db250ZXh0cyxcbiAgICAgICAgICAgICAgICAgICkucGlwZShcbiAgICAgICAgICAgICAgICAgICAgRWZmZWN0LmNhdGNoQWxsKChlcnJvcikgPT5cbiAgICAgICAgICAgICAgICAgICAgICBFZmZlY3QubG9nRXJyb3IoXG4gICAgICAgICAgICAgICAgICAgICAgICBgW0xvY2FsXSBEb2NrZXIgaW52b2NhdGlvbiBlcnJvcjogJHtlcnJvcn1gLFxuICAgICAgICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICAgRWZmZWN0LmZvcmtEYWVtb24sXG4gICAgICAgICAgICAgIClcbiAgICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgICAgeWllbGQqIGFwcFN5bmNDbGllbnQhXG4gICAgICAgICAgICAgIC5zdWJzY3JpYmVUb0ludm9jYXRpb25zKGludm9jYXRpb25DaGFubmVsKVxuICAgICAgICAgICAgICAucGlwZShcbiAgICAgICAgICAgICAgICBTdHJlYW0ucnVuRm9yRWFjaCgoaW52b2NhdGlvbikgPT5cbiAgICAgICAgICAgICAgICAgIGhhbmRsZU5vZGVqc0ludm9jYXRpb24oXG4gICAgICAgICAgICAgICAgICAgIGZuLFxuICAgICAgICAgICAgICAgICAgICBpbnZvY2F0aW9uLFxuICAgICAgICAgICAgICAgICAgICB3b3JrZXJzLFxuICAgICAgICAgICAgICAgICAgICBzZXJ2ZXJTY29wZSxcbiAgICAgICAgICAgICAgICAgICAgcHJvamVjdFJvb3QsXG4gICAgICAgICAgICAgICAgICAgIGFwcFN5bmNDbGllbnQhLFxuICAgICAgICAgICAgICAgICAgICBpbnZvY2F0aW9uQ291bnRlcixcbiAgICAgICAgICAgICAgICAgICAgaW52b2NhdGlvbkNvbnRleHRzLFxuICAgICAgICAgICAgICAgICAgKS5waXBlKFxuICAgICAgICAgICAgICAgICAgICBFZmZlY3QuY2F0Y2hBbGwoKGVycm9yKSA9PlxuICAgICAgICAgICAgICAgICAgICAgIEVmZmVjdC5sb2dFcnJvcihcbiAgICAgICAgICAgICAgICAgICAgICAgIGBbTG9jYWxdIE5vZGUuanMgaW52b2NhdGlvbiBlcnJvcjogJHtlcnJvcn1gLFxuICAgICAgICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICAgRWZmZWN0LmZvcmtEYWVtb24sXG4gICAgICAgICAgICAgIClcbiAgICAgICAgICB9XG4gICAgICAgIH1cblxuICAgICAgICB5aWVsZCogUmVmLnNldChyZWdpc3RlcmVkRnVuY3Rpb25zLCBjdXJyZW50UmVnaXN0ZXJlZClcblxuICAgICAgICAvLyBTdGFydCBmaWxlIHdhdGNoZXJzIGZvciBuZXcgRG9ja2VyIGZ1bmN0aW9uc1xuICAgICAgICBjb25zdCBuZXdEb2NrZXJGdW5jdGlvbnM6IFdhdGNoZWREb2NrZXJGdW5jdGlvbltdID0gW11cbiAgICAgICAgZm9yIChjb25zdCBmbiBvZiBmdW5jdGlvbnMpIHtcbiAgICAgICAgICBpZiAoXG4gICAgICAgICAgICBmbi5kb2NrZXJDb250ZXh0UGF0aCAmJlxuICAgICAgICAgICAgIXdhdGNoZWREb2NrZXJGdW5jdGlvbnMuaGFzKGZuLmZ1bmN0aW9uTmFtZSlcbiAgICAgICAgICApIHtcbiAgICAgICAgICAgIC8vIFJlc29sdmUgdGhlIGNvbnRleHQgcGF0aFxuICAgICAgICAgICAgY29uc3QgY29udGV4dFBhdGggPSBmbi5kb2NrZXJDb250ZXh0UGF0aC5zdGFydHNXaXRoKFwiL1wiKVxuICAgICAgICAgICAgICA/IGZuLmRvY2tlckNvbnRleHRQYXRoXG4gICAgICAgICAgICAgIDogYCR7cHJvamVjdFJvb3R9LyR7Zm4uZG9ja2VyQ29udGV4dFBhdGh9YFxuXG4gICAgICAgICAgICBuZXdEb2NrZXJGdW5jdGlvbnMucHVzaCh7XG4gICAgICAgICAgICAgIGZ1bmN0aW9uSWQ6IGZuLmZ1bmN0aW9uTmFtZSxcbiAgICAgICAgICAgICAgZG9ja2VyQ29udGV4dFBhdGg6IGNvbnRleHRQYXRoLFxuICAgICAgICAgICAgfSlcbiAgICAgICAgICAgIHdhdGNoZWREb2NrZXJGdW5jdGlvbnMuYWRkKGZuLmZ1bmN0aW9uTmFtZSlcbiAgICAgICAgICB9XG4gICAgICAgIH1cblxuICAgICAgICAvLyBTdGFydCB3YXRjaGluZyBuZXcgRG9ja2VyIGNvbnRleHRzXG4gICAgICAgIGlmIChuZXdEb2NrZXJGdW5jdGlvbnMubGVuZ3RoID4gMCkge1xuICAgICAgICAgIHlpZWxkKiBFZmZlY3QubG9nRGVidWcoXG4gICAgICAgICAgICBgW0xvY2FsXSBTdGFydGluZyBmaWxlIHdhdGNoZXJzIGZvciAke25ld0RvY2tlckZ1bmN0aW9ucy5sZW5ndGh9IERvY2tlciBmdW5jdGlvbihzKS4uLmAsXG4gICAgICAgICAgKVxuXG4gICAgICAgICAgLy8gRm9yayBhIGRhZW1vbiBmaWJlciB0byBoYW5kbGUgZmlsZSBjaGFuZ2UgZXZlbnRzIHdpdGggY29udGV4dCBwcmVzZXJ2ZWRcbiAgICAgICAgICB5aWVsZCogd2F0Y2hEb2NrZXJDb250ZXh0cyhuZXdEb2NrZXJGdW5jdGlvbnMsIDUwMCkucGlwZShcbiAgICAgICAgICAgIFN0cmVhbS5ydW5Gb3JFYWNoKChldmVudCkgPT5cbiAgICAgICAgICAgICAgRWZmZWN0LmdlbihmdW5jdGlvbiogKCkge1xuICAgICAgICAgICAgICAgIHlpZWxkKiBFZmZlY3QubG9nRGVidWcoXG4gICAgICAgICAgICAgICAgICBgW0xvY2FsXSBGaWxlIGNoYW5nZWQgaW4gJHtldmVudC5mdW5jdGlvbklkfTogJHtldmVudC5maWxlUGF0aH1gLFxuICAgICAgICAgICAgICAgIClcbiAgICAgICAgICAgICAgICB5aWVsZCogcmVidWlsZERvY2tlckNvbnRhaW5lcihcbiAgICAgICAgICAgICAgICAgIGV2ZW50LmZ1bmN0aW9uSWQsXG4gICAgICAgICAgICAgICAgICBjb250YWluZXJzLFxuICAgICAgICAgICAgICAgICAgcHJvamVjdFJvb3QsXG4gICAgICAgICAgICAgICAgKS5waXBlKFxuICAgICAgICAgICAgICAgICAgRWZmZWN0LmNhdGNoQWxsKChlcnJvcikgPT5cbiAgICAgICAgICAgICAgICAgICAgRWZmZWN0LmxvZ0Vycm9yKFxuICAgICAgICAgICAgICAgICAgICAgIGBbTG9jYWxdIFJlYnVpbGQgZmFpbGVkIGZvciAke2V2ZW50LmZ1bmN0aW9uSWR9OiAke2Vycm9yfWAsXG4gICAgICAgICAgICAgICAgICAgICksXG4gICAgICAgICAgICAgICAgICApLFxuICAgICAgICAgICAgICAgIClcbiAgICAgICAgICAgICAgfSksXG4gICAgICAgICAgICApLFxuICAgICAgICAgICAgRWZmZWN0LmZvcmtEYWVtb24sXG4gICAgICAgICAgKVxuICAgICAgICB9XG5cbiAgICAgICAgaWYgKCFsb2dTdGF0ZS5oYXNMb2dnZWRXYXRjaGluZykge1xuICAgICAgICAgIGxvZ1N0YXRlLmhhc0xvZ2dlZFdhdGNoaW5nID0gdHJ1ZVxuICAgICAgICAgIHlpZWxkKiBFZmZlY3QubG9nSW5mbyhcIltMb2NhbF0gUmVhZHkgZm9yIGludm9jYXRpb25zXCIpXG4gICAgICAgIH1cblxuICAgICAgICAvLyBNYXJrIHRoYXQgd2UndmUgY29tcGxldGVkIGZpcnN0IGRpc2NvdmVyeVxuICAgICAgICBsb2dTdGF0ZS5oYXNEaXNjb3ZlcmVkT25jZSA9IHRydWVcbiAgICAgIH0pXG5cbiAgICAgIC8vIEhhbmRsZSBDREsgd2F0Y2ggZXZlbnRzIChzdGFjayBkaXNjb3ZlcnkgYW5kIGRlcGxveSBjb21wbGV0aW9uKVxuICAgICAgeWllbGQqIFF1ZXVlLnRha2UoY2RrRXZlbnRzKS5waXBlKFxuICAgICAgICBFZmZlY3QuZmxhdE1hcCgoZXZlbnQpID0+XG4gICAgICAgICAgRWZmZWN0LmdlbihmdW5jdGlvbiogKCkge1xuICAgICAgICAgICAgc3dpdGNoIChldmVudC5fdGFnKSB7XG4gICAgICAgICAgICAgIGNhc2UgXCJTdGFja0Rpc2NvdmVyZWRcIjpcbiAgICAgICAgICAgICAgICAvLyBBZGQgZGlzY292ZXJlZCBzdGFjayB0byBmaWx0ZXIgKGlmIG5vdCB1c2luZyAtLXN0YWNrcylcbiAgICAgICAgICAgICAgICBpZiAoXG4gICAgICAgICAgICAgICAgICAhc3RhY2tzRnJvbU9wdGlvbiAmJlxuICAgICAgICAgICAgICAgICAgIWZpbHRlclN0YXRlLnN0YWNrcy5pbmNsdWRlcyhldmVudC5zdGFja05hbWUpXG4gICAgICAgICAgICAgICAgKSB7XG4gICAgICAgICAgICAgICAgICBmaWx0ZXJTdGF0ZS5zdGFja3MucHVzaChldmVudC5zdGFja05hbWUpXG4gICAgICAgICAgICAgICAgICB5aWVsZCogRWZmZWN0LmxvZ0RlYnVnKFxuICAgICAgICAgICAgICAgICAgICBgW0xvY2FsXSBEaXNjb3ZlcmVkIHN0YWNrOiAke2V2ZW50LnN0YWNrTmFtZX1gLFxuICAgICAgICAgICAgICAgICAgKVxuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICBicmVha1xuICAgICAgICAgICAgICBjYXNlIFwiRGVwbG95Q29tcGxldGVcIjpcbiAgICAgICAgICAgICAgICAvLyBSdW4gZGFlbW9uIHVwZGF0ZSBvbiBkZXBsb3kgY29tcGxldGlvblxuICAgICAgICAgICAgICAgIHlpZWxkKiBzdGFydE9yVXBkYXRlRGFlbW9uLnBpcGUoXG4gICAgICAgICAgICAgICAgICBFZmZlY3QuY2F0Y2hBbGwoKGVycm9yKSA9PlxuICAgICAgICAgICAgICAgICAgICBFZmZlY3QubG9nRXJyb3IoYEZhaWxlZCB0byB1cGRhdGUgZGFlbW9uOiAke2Vycm9yfWApLFxuICAgICAgICAgICAgICAgICAgKSxcbiAgICAgICAgICAgICAgICApXG4gICAgICAgICAgICAgICAgYnJlYWtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9KSxcbiAgICAgICAgKSxcbiAgICAgICAgRWZmZWN0LmZvcmV2ZXIsXG4gICAgICAgIEVmZmVjdC5mb3JrLFxuICAgICAgKVxuXG4gICAgICAvLyBIYW5kbGUgY2xlYW51cCBvbiBleGl0XG4gICAgICBjb25zdCBjbGVhbnVwID0gYXN5bmMgKCkgPT4ge1xuICAgICAgICBhd2FpdCBFZmZlY3QucnVuUHJvbWlzZShcbiAgICAgICAgICBFZmZlY3QuZ2VuKGZ1bmN0aW9uKiAoKSB7XG4gICAgICAgICAgICB5aWVsZCogRWZmZWN0LmxvZ0luZm8oXCJcXG5TaHV0dGluZyBkb3duLi4uXCIpXG5cbiAgICAgICAgICAgIC8vIFN0b3AgQ0RLIHdhdGNoXG4gICAgICAgICAgICB5aWVsZCogY2RrV2F0Y2hQcm9jZXNzXG4gICAgICAgICAgICAgIC5raWxsKFwiU0lHVEVSTVwiKVxuICAgICAgICAgICAgICAucGlwZShFZmZlY3QuY2F0Y2hBbGwoKCkgPT4gRWZmZWN0LnZvaWQpKVxuXG4gICAgICAgICAgICAvLyBTdG9wIGFsbCBEb2NrZXIgY29udGFpbmVycyB1c2luZyB0aGUgRG9ja2VyIHNlcnZpY2VcbiAgICAgICAgICAgIGNvbnN0IGRvY2tlciA9IHlpZWxkKiBEb2NrZXJcbiAgICAgICAgICAgIGNvbnN0IGN1cnJlbnRDb250YWluZXJzID0geWllbGQqIFJlZi5nZXQoY29udGFpbmVycylcbiAgICAgICAgICAgIGZvciAoY29uc3QgW25hbWUsIGNvbnRhaW5lcl0gb2YgY3VycmVudENvbnRhaW5lcnMpIHtcbiAgICAgICAgICAgICAgeWllbGQqIEVmZmVjdC5sb2dJbmZvKGBTdG9wcGluZyBjb250YWluZXI6ICR7bmFtZX1gKVxuICAgICAgICAgICAgICB5aWVsZCogZG9ja2VyLnN0b3AoY29udGFpbmVyLmNvbnRhaW5lck5hbWUpLnBpcGUoXG4gICAgICAgICAgICAgICAgRWZmZWN0LnNjb3BlZCxcbiAgICAgICAgICAgICAgICBFZmZlY3QuY2F0Y2hBbGwoKCkgPT4gRWZmZWN0LnZvaWQpLFxuICAgICAgICAgICAgICApXG4gICAgICAgICAgICB9XG5cbiAgICAgICAgICAgIC8vIFN0b3AgYWxsIE5vZGUuanMgd29ya2Vyc1xuICAgICAgICAgICAgY29uc3QgY3VycmVudFdvcmtlcnMgPSB5aWVsZCogUmVmLmdldCh3b3JrZXJzKVxuICAgICAgICAgICAgZm9yIChjb25zdCBbbmFtZSwgd29ya2VyXSBvZiBjdXJyZW50V29ya2Vycykge1xuICAgICAgICAgICAgICB5aWVsZCogRWZmZWN0LmxvZ0luZm8oYFN0b3BwaW5nIHdvcmtlcjogJHtuYW1lfWApXG4gICAgICAgICAgICAgIHlpZWxkKiBFZmZlY3QudHJ5KCgpID0+XG4gICAgICAgICAgICAgICAgd29ya2VyLndvcmtlclByb2Nlc3Mua2lsbChcIlNJR1RFUk1cIiksXG4gICAgICAgICAgICAgICkucGlwZShFZmZlY3QuY2F0Y2hBbGwoKCkgPT4gRWZmZWN0LnZvaWQpKVxuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAvLyBDbG9zZSB0aGUgc2VydmVyIHNjb3BlIHRvIGNsZWFuIHVwIFJ1bnRpbWUgQVBJIHNlcnZlcnNcbiAgICAgICAgICAgIHlpZWxkKiBTY29wZS5jbG9zZShzZXJ2ZXJTY29wZSwgRXhpdC52b2lkKVxuICAgICAgICAgIH0pLnBpcGUoXG4gICAgICAgICAgICBFZmZlY3QucHJvdmlkZShEb2NrZXJMaXZlKSxcbiAgICAgICAgICAgIEVmZmVjdC5wcm92aWRlKEJ1bkNvbnRleHQubGF5ZXIpLFxuICAgICAgICAgICAgRWZmZWN0LnByb3ZpZGUoTG9nZ2VyLnByZXR0eSksXG4gICAgICAgICAgICBFZmZlY3QucHJvdmlkZShMb2dnZXIubWluaW11bUxvZ0xldmVsKGxvZ0xldmVsKSksXG4gICAgICAgICAgKSxcbiAgICAgICAgKVxuXG4gICAgICAgIHByb2Nlc3MuZXhpdCgwKVxuICAgICAgfVxuXG4gICAgICBwcm9jZXNzLm9uKFwiU0lHSU5UXCIsIGNsZWFudXApXG4gICAgICBwcm9jZXNzLm9uKFwiU0lHVEVSTVwiLCBjbGVhbnVwKVxuXG4gICAgICB5aWVsZCogRWZmZWN0LmxvZ0luZm8oXCJbTG9jYWxdIFByZXNzIEN0cmwrQyB0byBzdG9wXCIpXG5cbiAgICAgIC8vIEtlZXAgdGhlIHByb2Nlc3MgcnVubmluZ1xuICAgICAgeWllbGQqIEVmZmVjdC5uZXZlclxuICAgIH0pLnBpcGUoXG4gICAgICBFZmZlY3QucHJvdmlkZShMb2dnZXIucHJldHR5KSxcbiAgICAgIEVmZmVjdC5wcm92aWRlKExvZ2dlci5taW5pbXVtTG9nTGV2ZWwobG9nTGV2ZWwpKSxcbiAgICApXG4gIH0sXG4pLnBpcGUoXG4gIENvbW1hbmQud2l0aERlc2NyaXB0aW9uKFxuICAgIFwiU3RhcnQgbG9jYWwgTGFtYmRhIGRldmVsb3BtZW50IHdpdGggQ0RLIHdhdGNoIGFuZCBEb2NrZXIgY29udGFpbmVyc1wiLFxuICApLFxuKVxuIl19