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.
- package/LICENSE +202 -0
- package/README.md +94 -0
- package/lib/aspect/docker-function-hook.d.ts +18 -0
- package/lib/aspect/docker-function-hook.js +31 -0
- package/lib/aspect/live-lambda-aspect.d.ts +85 -0
- package/lib/aspect/live-lambda-aspect.js +277 -0
- package/lib/aspect/live-lambda-bootstrap.d.ts +17 -0
- package/lib/aspect/live-lambda-bootstrap.js +260 -0
- package/lib/aspect/nodejs-function-hook.d.ts +20 -0
- package/lib/aspect/nodejs-function-hook.js +27 -0
- package/lib/bootstrap-stack/bootstrap-stack.d.ts +60 -0
- package/lib/bootstrap-stack/bootstrap-stack.js +338 -0
- package/lib/cli/appsync/client.d.ts +30 -0
- package/lib/cli/appsync/client.js +227 -0
- package/lib/cli/cdk-app.d.ts +7 -0
- package/lib/cli/cdk-app.js +25 -0
- package/lib/cli/commands/bootstrap.d.ts +9 -0
- package/lib/cli/commands/bootstrap.js +50 -0
- package/lib/cli/commands/local.d.ts +40 -0
- package/lib/cli/commands/local.js +1172 -0
- package/lib/cli/daemon.d.ts +22 -0
- package/lib/cli/daemon.js +18 -0
- package/lib/cli/docker/container.d.ts +116 -0
- package/lib/cli/docker/container.js +414 -0
- package/lib/cli/docker/types.d.ts +71 -0
- package/lib/cli/docker/types.js +5 -0
- package/lib/cli/docker/watcher.d.ts +44 -0
- package/lib/cli/docker/watcher.js +115 -0
- package/lib/cli/index.d.ts +9 -0
- package/lib/cli/index.js +26 -0
- package/lib/cli/runtime-api/server.d.ts +102 -0
- package/lib/cli/runtime-api/server.js +396 -0
- package/lib/cli/runtime-api/types.d.ts +149 -0
- package/lib/cli/runtime-api/types.js +10 -0
- package/lib/cli/runtime-wrapper/nodejs-runtime.d.ts +16 -0
- package/lib/cli/runtime-wrapper/nodejs-runtime.js +248 -0
- package/lib/cli/watcher/file-watcher.d.ts +32 -0
- package/lib/cli/watcher/file-watcher.js +57 -0
- package/lib/functions/bridge/appsync-client.d.ts +73 -0
- package/lib/functions/bridge/appsync-client.js +345 -0
- package/lib/functions/bridge/handler.d.ts +17 -0
- package/lib/functions/bridge/handler.js +79 -0
- package/lib/functions/bridge/ssm-config.d.ts +19 -0
- package/lib/functions/bridge/ssm-config.js +45 -0
- package/lib/functions/bridge-builder/handler.d.ts +12 -0
- package/lib/functions/bridge-builder/handler.js +181 -0
- package/lib/functions/bridge-docker/runtime.d.ts +9 -0
- package/lib/functions/bridge-docker/runtime.js +127 -0
- package/lib/index.d.ts +24 -0
- package/lib/index.js +28 -0
- package/lib/shared/types.d.ts +102 -0
- package/lib/shared/types.js +125 -0
- 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
|