@nlabs/lex 1.56.2 → 1.58.0

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.
@@ -1,843 +0,0 @@
1
- /**
2
- * Copyright (c) 2018-Present, Nitrogen Labs, Inc.
3
- * Copyrights licensed under the MIT License. See the accompanying LICENSE file for terms.
4
- */ import boxen from 'boxen';
5
- import chalk from 'chalk';
6
- import express from 'express';
7
- import { readFileSync, existsSync, mkdirSync, writeFileSync } from 'fs';
8
- import { homedir } from 'os';
9
- import { resolve as pathResolve, join, isAbsolute } from 'path';
10
- import { pathToFileURL } from 'url';
11
- import { WebSocketServer } from 'ws';
12
- import { LexConfig, getPackageDir } from '../../LexConfig.js';
13
- import { createSpinner, removeFiles } from '../../utils/app.js';
14
- import { log } from '../../utils/log.js';
15
- const getCacheDir = ()=>{
16
- const cacheDir = join(homedir(), '.lex-cache');
17
- if (!existsSync(cacheDir)) {
18
- mkdirSync(cacheDir, {
19
- recursive: true
20
- });
21
- }
22
- return cacheDir;
23
- };
24
- const getCachePath = ()=>join(getCacheDir(), 'public-ip.json');
25
- const readPublicIpCache = ()=>{
26
- const cachePath = getCachePath();
27
- if (!existsSync(cachePath)) {
28
- return null;
29
- }
30
- try {
31
- const cacheData = readFileSync(cachePath, 'utf8');
32
- const cache = JSON.parse(cacheData);
33
- // Check if cache is older than 1 week
34
- const oneWeekMs = 7 * 24 * 60 * 60 * 1000;
35
- if (Date.now() - cache.timestamp > oneWeekMs) {
36
- return null;
37
- }
38
- return cache;
39
- } catch {
40
- return null;
41
- }
42
- };
43
- const writePublicIpCache = (ip)=>{
44
- const cachePath = getCachePath();
45
- const cache = {
46
- ip,
47
- timestamp: Date.now()
48
- };
49
- writeFileSync(cachePath, JSON.stringify(cache, null, 2));
50
- };
51
- const fetchPublicIp = (forceRefresh = false)=>new Promise((resolve)=>{
52
- if (!forceRefresh) {
53
- const cached = readPublicIpCache();
54
- if (cached) {
55
- resolve(cached.ip);
56
- return;
57
- }
58
- }
59
- // Use fetch instead of https
60
- fetch('https://api.ipify.org').then((res)=>res.text()).then((data)=>{
61
- const ip = data.trim();
62
- if (ip) {
63
- writePublicIpCache(ip);
64
- }
65
- resolve(ip);
66
- }).catch(()=>resolve(undefined));
67
- });
68
- const resolvePublicIpForDisplay = async (forceRefresh = false)=>{
69
- const timeoutMs = 1500;
70
- const timeoutPromise = new Promise((resolve)=>{
71
- setTimeout(()=>resolve(undefined), timeoutMs);
72
- });
73
- try {
74
- return await Promise.race([
75
- fetchPublicIp(forceRefresh),
76
- timeoutPromise
77
- ]);
78
- } catch {
79
- return undefined;
80
- }
81
- };
82
- const displayServerStatus = (httpPort, httpsPort, wsPort, host, quiet, publicIp)=>{
83
- if (quiet) {
84
- return;
85
- }
86
- const httpUrl = `http://${host}:${httpPort}`;
87
- const httpsUrl = `https://${host}:${httpsPort}`;
88
- const wsUrl = `ws://${host}:${wsPort}`;
89
- const wssUrl = `wss://${host}:${wsPort}`;
90
- let urlLines = `${chalk.green('HTTP:')} ${chalk.underline(httpUrl)}\n`;
91
- urlLines += `${chalk.green('HTTPS:')} ${chalk.underline(httpsUrl)}\n`;
92
- urlLines += `${chalk.green('WebSocket:')} ${chalk.underline(wsUrl)}\n`;
93
- urlLines += `${chalk.green('WSS:')} ${chalk.underline(wssUrl)}\n`;
94
- if (publicIp) {
95
- urlLines += `\n${chalk.green('Public:')} ${chalk.underline(`http://${publicIp}:${httpPort}`)}\n`;
96
- }
97
- const statusBox = boxen(`${chalk.cyan.bold('🚀 Serverless Development Server Running')}\n\n${urlLines}\n` + `${chalk.yellow('Press Ctrl+C to stop the server')}`, {
98
- backgroundColor: '#1a1a1a',
99
- borderColor: 'cyan',
100
- borderStyle: 'round',
101
- margin: 1,
102
- padding: 1
103
- });
104
- console.log(`\n${statusBox}\n`);
105
- };
106
- const loadHandler = async (handlerPath, outputDir)=>{
107
- try {
108
- console.log(`[Serverless] Parsing handler path: ${handlerPath}`);
109
- // Parse AWS Lambda handler format: "path/to/file.exportName" or "file.exportName"
110
- // Examples: "index.handler", "handlers/api.handler", "src/index.default"
111
- const handlerParts = handlerPath.split('.');
112
- console.log('[Serverless] Handler parts after split:', handlerParts);
113
- let filePath;
114
- let exportName = null;
115
- if (handlerParts.length > 1) {
116
- // AWS Lambda format: "file.exportName"
117
- // Take the last part as export name, rest as file path
118
- // Handle cases like "index.handler" -> file: "index", export: "handler"
119
- // Or "handlers/api.handler" -> file: "handlers/api", export: "handler"
120
- exportName = handlerParts[handlerParts.length - 1] || null;
121
- filePath = handlerParts.slice(0, -1).join('.');
122
- console.log(`[Serverless] Parsed AWS Lambda format - filePath: "${filePath}", exportName: "${exportName}"`);
123
- } else {
124
- // Simple format: just the file path
125
- filePath = handlerPath;
126
- console.log(`[Serverless] Simple format - filePath: "${filePath}"`);
127
- }
128
- // Ensure filePath doesn't have the export name in it
129
- if (filePath.includes('.handler') || filePath.includes('.default')) {
130
- console.error(`[Serverless] WARNING: filePath still contains export name! filePath: "${filePath}"`);
131
- // Try to fix it - remove the last part if it looks like an export name
132
- const pathParts = filePath.split('.');
133
- if (pathParts.length > 1) {
134
- const lastPart = pathParts[pathParts.length - 1];
135
- if ([
136
- 'handler',
137
- 'default',
138
- 'index'
139
- ].includes(lastPart) && !exportName) {
140
- exportName = lastPart;
141
- filePath = pathParts.slice(0, -1).join('.');
142
- console.log(`[Serverless] Fixed - filePath: "${filePath}", exportName: "${exportName}"`);
143
- }
144
- }
145
- }
146
- // Handle both relative paths and absolute paths
147
- let fullPath;
148
- if (isAbsolute(filePath)) {
149
- fullPath = filePath;
150
- } else {
151
- fullPath = pathResolve(outputDir, filePath);
152
- }
153
- console.log(`[Serverless] Resolved fullPath (before extensions): ${fullPath}`);
154
- // Try different extensions if file doesn't exist
155
- if (!existsSync(fullPath)) {
156
- const extensions = [
157
- '.js',
158
- '.mjs',
159
- '.cjs'
160
- ];
161
- const pathWithoutExt = fullPath.replace(/\.(js|mjs|cjs)$/, '');
162
- console.log(`[Serverless] Trying extensions. Base path: ${pathWithoutExt}`);
163
- for (const ext of extensions){
164
- const candidatePath = pathWithoutExt + ext;
165
- console.log(`[Serverless] Checking: ${candidatePath} (exists: ${existsSync(candidatePath)})`);
166
- if (existsSync(candidatePath)) {
167
- fullPath = candidatePath;
168
- console.log(`[Serverless] Found file with extension: ${fullPath}`);
169
- break;
170
- }
171
- }
172
- } else {
173
- console.log(`[Serverless] File exists without trying extensions: ${fullPath}`);
174
- }
175
- console.log(`[Serverless] Final fullPath: ${fullPath}`);
176
- console.log(`[Serverless] Export name: ${exportName || 'default/handler'}`);
177
- console.log(`[Serverless] File exists: ${existsSync(fullPath)}`);
178
- if (!existsSync(fullPath)) {
179
- console.error(`[Serverless] Handler file not found: ${fullPath}`);
180
- console.error(`[Serverless] Output directory: ${outputDir}`);
181
- console.error(`[Serverless] Handler path from config: ${handlerPath}`);
182
- throw new Error(`Handler file not found: ${fullPath}`);
183
- }
184
- // Dynamic import of the handler with better error handling
185
- // Add .js extension if importing TypeScript compiled output
186
- const importPath = fullPath.endsWith('.ts') ? fullPath.replace(/\.ts$/, '.js') : fullPath;
187
- try {
188
- // Convert to file:// URL for ES module imports (required for absolute paths)
189
- // Use pathToFileURL to ensure proper file:// URL format
190
- const importUrl = pathToFileURL(importPath).href;
191
- console.log(`[Serverless] Importing handler from: ${importUrl}`);
192
- console.log(`[Serverless] File path: ${importPath}`);
193
- // Use import() with the file URL
194
- // Note: If the handler file has import errors (like missing dependencies),
195
- // those will surface here, but that's a handler code issue, not a loader issue
196
- const handlerModule = await import(importUrl);
197
- console.log(`[Serverless] Handler module loaded successfully. Exports: ${Object.keys(handlerModule).join(', ')}`);
198
- // Get the handler based on export name or try defaults
199
- let handler;
200
- if (exportName) {
201
- handler = handlerModule[exportName] || handlerModule.default?.[exportName] || handlerModule['module.exports']?.[exportName];
202
- if (!handler) {
203
- console.error(`[Serverless] Export "${exportName}" not found in module. Available exports: ${Object.keys(handlerModule).join(', ')}`);
204
- return null;
205
- }
206
- } else {
207
- // Try default, handler, or the module itself
208
- handler = handlerModule.default || handlerModule.handler || handlerModule;
209
- }
210
- console.log(`[Serverless] Handler found: ${typeof handler}, isFunction: ${typeof handler === 'function'}`);
211
- if (typeof handler !== 'function') {
212
- console.error(`[Serverless] Handler is not a function. Type: ${typeof handler}, Value:`, handler);
213
- return null;
214
- }
215
- return handler;
216
- } catch (importError) {
217
- console.error(`[Serverless] Import error for handler ${handlerPath}:`, importError.message);
218
- console.error('[Serverless] Import error stack:', importError.stack);
219
- // Check if this is a dependency resolution error (common with ES modules)
220
- if (importError.message && importError.message.includes('Cannot find module')) {
221
- console.error('[Serverless] This appears to be a dependency resolution error.');
222
- console.error('[Serverless] The handler file exists, but one of its imports is failing.');
223
- console.error('[Serverless] Check that all dependencies in the handler file are properly installed.');
224
- console.error(`[Serverless] Handler file: ${importPath}`);
225
- console.error('[Serverless] Make sure the handler and its dependencies are compiled correctly.');
226
- }
227
- return null;
228
- }
229
- } catch (error) {
230
- console.error(`[Serverless] Error loading handler ${handlerPath}:`, error.message);
231
- console.error('[Serverless] Error stack:', error.stack);
232
- return null;
233
- }
234
- };
235
- const captureConsoleLogs = (handler, quiet)=>{
236
- if (quiet) {
237
- return handler;
238
- }
239
- return async (event, context)=>{
240
- // Capture console.log, console.error, etc.
241
- const originalConsoleLog = console.log;
242
- const originalConsoleError = console.error;
243
- const originalConsoleWarn = console.warn;
244
- const originalConsoleInfo = console.info;
245
- const logs = [];
246
- console.log = (...args)=>{
247
- logs.push(`[LOG] ${args.join(' ')}`);
248
- originalConsoleLog(...args);
249
- };
250
- console.error = (...args)=>{
251
- logs.push(`[ERROR] ${args.join(' ')}`);
252
- originalConsoleError(...args);
253
- };
254
- console.warn = (...args)=>{
255
- logs.push(`[WARN] ${args.join(' ')}`);
256
- originalConsoleWarn(...args);
257
- };
258
- console.info = (...args)=>{
259
- logs.push(`[INFO] ${args.join(' ')}`);
260
- originalConsoleInfo(...args);
261
- };
262
- try {
263
- const result = await handler(event, context);
264
- // Output captured logs
265
- if (logs.length > 0) {
266
- console.log(chalk.gray('--- Handler Console Output ---'));
267
- logs.forEach((log)=>console.log(chalk.gray(log)));
268
- console.log(chalk.gray('--- End Handler Console Output ---'));
269
- }
270
- return result;
271
- } finally{
272
- // Restore original console methods
273
- console.log = originalConsoleLog;
274
- console.error = originalConsoleError;
275
- console.warn = originalConsoleWarn;
276
- console.info = originalConsoleInfo;
277
- }
278
- };
279
- };
280
- const createExpressServer = async (config, outputDir, httpPort, host, quiet, debug)=>{
281
- const app = express();
282
- // Enable CORS
283
- app.use((req, res, next)=>{
284
- res.header('Access-Control-Allow-Origin', '*');
285
- res.header('Access-Control-Allow-Methods', 'GET, POST, PUT, DELETE, PATCH, OPTIONS');
286
- res.header('Access-Control-Allow-Headers', '*');
287
- res.header('Access-Control-Allow-Credentials', 'true');
288
- if (req.method === 'OPTIONS') {
289
- res.sendStatus(200);
290
- } else {
291
- next();
292
- }
293
- });
294
- // Parse JSON bodies
295
- app.use(express.json());
296
- // Load GraphQL handler
297
- const loadGraphQLSchema = async ()=>{
298
- try {
299
- // Try to find a GraphQL handler
300
- let graphqlHandler = null;
301
- if (config.functions) {
302
- for (const [functionName, functionConfig] of Object.entries(config.functions)){
303
- if (functionConfig.events) {
304
- for (const event of functionConfig.events){
305
- if (event.http && event.http.path) {
306
- // Look for GraphQL endpoints
307
- if (event.http.path === '/public' || event.http.path === '/graphql') {
308
- graphqlHandler = await loadHandler(functionConfig.handler, outputDir);
309
- break;
310
- }
311
- }
312
- }
313
- }
314
- if (graphqlHandler) {
315
- break;
316
- }
317
- }
318
- }
319
- if (graphqlHandler) {
320
- log('Found GraphQL handler', 'info', quiet);
321
- return graphqlHandler;
322
- }
323
- return null;
324
- } catch (error) {
325
- log(`Error loading GraphQL handler: ${error.message}`, 'error', quiet);
326
- return null;
327
- }
328
- };
329
- // Set up GraphQL handler for GraphQL requests
330
- try {
331
- const graphqlHandler = await loadGraphQLSchema();
332
- if (graphqlHandler) {
333
- // Find the GraphQL path from the serverless config
334
- let graphqlPath = '/graphql'; // default fallback
335
- if (config.functions) {
336
- for (const [_functionName, functionConfig] of Object.entries(config.functions)){
337
- if (functionConfig.events) {
338
- for (const event of functionConfig.events){
339
- if (event?.http?.path) {
340
- graphqlPath = event.http.path;
341
- break;
342
- }
343
- }
344
- }
345
- if (graphqlPath !== '/graphql') {
346
- break;
347
- }
348
- }
349
- }
350
- // Set up GraphQL endpoint with enhanced console.log capture
351
- app.use(graphqlPath, async (req, res)=>{
352
- // GraphQL Debug Logging
353
- if (debug && req.body && req.body.query) {
354
- log('🔍 GraphQL Debug Mode: Analyzing request...', 'info', false);
355
- log(`📝 GraphQL Query: ${req.body.query}`, 'info', false);
356
- if (req.body.variables) {
357
- log(`📊 GraphQL Variables: ${JSON.stringify(req.body.variables, null, 2)}`, 'info', false);
358
- }
359
- if (req.body.operationName) {
360
- log(`🏷️ GraphQL Operation: ${req.body.operationName}`, 'info', false);
361
- }
362
- }
363
- // Enhanced console.log capture
364
- const originalConsoleLog = console.log;
365
- const logs = [];
366
- console.log = (...args)=>{
367
- const logMessage = args.map((arg)=>typeof arg === 'object' ? JSON.stringify(arg, null, 2) : String(arg)).join(' ');
368
- logs.push(logMessage);
369
- originalConsoleLog(`[GraphQL] ${logMessage}`);
370
- };
371
- // Create context for the handler
372
- const context = {
373
- awsRequestId: 'test-request-id',
374
- functionName: 'graphql',
375
- functionVersion: '$LATEST',
376
- getRemainingTimeInMillis: ()=>30000,
377
- invokedFunctionArn: 'arn:aws:lambda:us-east-1:123456789012:function:graphql',
378
- logGroupName: '/aws/lambda/graphql',
379
- logStreamName: 'test-log-stream',
380
- req,
381
- res
382
- };
383
- // Wrap handler with console log capture
384
- const wrappedHandler = captureConsoleLogs(graphqlHandler, quiet);
385
- try {
386
- // Call the handler with GraphQL parameters
387
- const result = await wrappedHandler({
388
- body: JSON.stringify(req.body),
389
- headers: req.headers,
390
- httpMethod: 'POST',
391
- path: graphqlPath,
392
- queryStringParameters: {}
393
- }, context);
394
- // Restore console.log
395
- console.log = originalConsoleLog;
396
- // Handle the result
397
- if (result && typeof result === 'object' && result.statusCode) {
398
- res.status(result.statusCode);
399
- if (result.headers) {
400
- Object.entries(result.headers).forEach(([key, value])=>{
401
- res.setHeader(key, String(value));
402
- });
403
- }
404
- res.send(result.body);
405
- } else {
406
- res.json(result);
407
- }
408
- } catch (error) {
409
- // Restore console.log
410
- console.log = originalConsoleLog;
411
- log(`GraphQL handler error: ${error.message}`, 'error', false);
412
- res.status(500).json({
413
- error: error.message
414
- });
415
- }
416
- });
417
- log(`GraphQL endpoint available at http://${host}:${httpPort}${graphqlPath}`, 'info', quiet);
418
- }
419
- } catch (error) {
420
- log(`Error setting up GraphQL: ${error.message}`, 'error', quiet);
421
- }
422
- // Fallback for non-GraphQL routes - handle all remaining routes
423
- app.use('/', async (req, res)=>{
424
- try {
425
- const url = req.url || '/';
426
- const method = req.method || 'GET';
427
- const pathname = req.path || url.split('?')[0]; // Extract pathname without query string
428
- // Always log requests (not affected by quiet flag for debugging)
429
- console.log(`[Serverless] ${method} ${url} (pathname: ${pathname})`);
430
- // Find matching function
431
- let matchedFunction = null;
432
- const functions = config.functions || {};
433
- if (config.functions) {
434
- const functionNames = Object.keys(functions);
435
- console.log(`[Serverless] Available functions: ${functionNames.join(', ')}`);
436
- console.log('[Serverless] Config functions:', JSON.stringify(functions, null, 2));
437
- for (const [functionName, functionConfig] of Object.entries(functions)){
438
- if (functionConfig.events) {
439
- for (const event of functionConfig.events){
440
- if (event.http) {
441
- const eventPath = event.http.path || '/';
442
- const eventMethod = (event.http.method || 'GET').toUpperCase();
443
- const requestMethod = method.toUpperCase();
444
- console.log(`[Serverless] Checking function ${functionName}: path="${eventPath}", method="${eventMethod}" against pathname="${pathname}", method="${requestMethod}"`);
445
- // Improved path matching - compare pathname without query string
446
- // Normalize paths (remove trailing slashes for comparison)
447
- const normalizedEventPath = eventPath.replace(/\/$/, '') || '/';
448
- const normalizedPathname = pathname.replace(/\/$/, '') || '/';
449
- if (normalizedEventPath === normalizedPathname && eventMethod === requestMethod) {
450
- matchedFunction = functionName;
451
- console.log(`[Serverless] ✓ Matched function: ${matchedFunction}`);
452
- break;
453
- }
454
- }
455
- }
456
- }
457
- if (matchedFunction) {
458
- break;
459
- }
460
- }
461
- } else {
462
- console.log('[Serverless] No functions found in config');
463
- }
464
- if (matchedFunction && functions[matchedFunction]) {
465
- // Resolve handler path relative to output directory
466
- const handlerPath = functions[matchedFunction].handler;
467
- console.log(`[Serverless] Loading handler: ${handlerPath} from outputDir: ${outputDir}`);
468
- const handler = await loadHandler(handlerPath, outputDir);
469
- if (handler) {
470
- console.log(`[Serverless] Handler loaded successfully, type: ${typeof handler}`);
471
- const wrappedHandler = captureConsoleLogs(handler, quiet);
472
- const event = {
473
- body: req.body,
474
- headers: req.headers,
475
- httpMethod: method,
476
- path: url,
477
- queryStringParameters: req.query
478
- };
479
- const context = {
480
- awsRequestId: 'test-request-id',
481
- functionName: matchedFunction,
482
- functionVersion: '$LATEST',
483
- getRemainingTimeInMillis: ()=>30000,
484
- invokedFunctionArn: `arn:aws:lambda:us-east-1:123456789012:function:${matchedFunction}`,
485
- logGroupName: `/aws/lambda/${matchedFunction}`,
486
- logStreamName: 'test-log-stream',
487
- memoryLimitInMB: '128'
488
- };
489
- try {
490
- console.log('[Serverless] Calling handler with event:', JSON.stringify(event, null, 2));
491
- const result = await wrappedHandler(event, context);
492
- console.log('[Serverless] Handler returned:', JSON.stringify(result, null, 2));
493
- if (result && typeof result === 'object' && result.statusCode) {
494
- res.status(result.statusCode);
495
- if (result.headers) {
496
- Object.entries(result.headers).forEach(([key, value])=>{
497
- res.setHeader(key, String(value));
498
- });
499
- }
500
- res.send(result.body);
501
- } else {
502
- res.json(result);
503
- }
504
- } catch (error) {
505
- console.error('[Serverless] Handler error:', error.message);
506
- console.error('[Serverless] Handler error stack:', error.stack);
507
- log(`Handler error: ${error.message}`, 'error', false);
508
- res.status(500).json({
509
- error: error.message
510
- });
511
- }
512
- } else {
513
- console.error(`[Serverless] Handler not found for function: ${matchedFunction}`);
514
- console.error(`[Serverless] Handler path: ${handlerPath}, Output dir: ${outputDir}`);
515
- res.status(404).json({
516
- error: 'Handler not found'
517
- });
518
- }
519
- } else {
520
- console.error(`[Serverless] Function not found for pathname: ${pathname}, method: ${method}`);
521
- console.error(`[Serverless] Available functions: ${config.functions ? Object.keys(functions).join(', ') : 'none'}`);
522
- res.status(404).json({
523
- error: 'Function not found'
524
- });
525
- }
526
- } catch (error) {
527
- log(`Route handling error: ${error.message}`, 'error', false);
528
- res.status(500).json({
529
- error: error.message
530
- });
531
- }
532
- });
533
- return app;
534
- };
535
- const createWebSocketServer = (config, outputDir, wsPort, quiet, debug)=>{
536
- const wss = new WebSocketServer({
537
- port: wsPort
538
- });
539
- wss.on('connection', async (ws, req)=>{
540
- log(`WebSocket connection established: ${req.url}`, 'info', false);
541
- ws.on('message', async (message)=>{
542
- try {
543
- const data = JSON.parse(message.toString());
544
- // Find matching WebSocket function
545
- let matchedFunction = null;
546
- const functions = config.functions || {};
547
- if (config.functions) {
548
- let defaultFunction = null;
549
- for (const [functionName, functionConfig] of Object.entries(functions)){
550
- if (functionConfig.events) {
551
- for (const event of functionConfig.events){
552
- if (event.websocket) {
553
- const route = event.websocket.route || '$connect';
554
- if (route === data.action) {
555
- matchedFunction = functionName;
556
- break;
557
- }
558
- if (route === '$default' && !defaultFunction) {
559
- defaultFunction = functionName;
560
- }
561
- }
562
- }
563
- }
564
- if (matchedFunction) {
565
- break;
566
- }
567
- }
568
- if (!matchedFunction) {
569
- matchedFunction = defaultFunction;
570
- }
571
- }
572
- if (matchedFunction && functions[matchedFunction]) {
573
- const handler = await loadHandler(functions[matchedFunction].handler, outputDir);
574
- if (handler) {
575
- // Wrap handler with console log capture
576
- const wrappedHandler = captureConsoleLogs(handler, quiet);
577
- const event = {
578
- body: data.body || null,
579
- requestContext: {
580
- apiGateway: {
581
- endpoint: `ws://localhost:${wsPort}`
582
- },
583
- connectionId: 'test-connection-id',
584
- routeKey: data.action || '$default'
585
- }
586
- };
587
- const context = {
588
- awsRequestId: 'test-request-id',
589
- functionName: matchedFunction,
590
- functionVersion: '$LATEST',
591
- getRemainingTimeInMillis: ()=>30000,
592
- invokedFunctionArn: `arn:aws:lambda:us-east-1:123456789012:function:${matchedFunction}`,
593
- logGroupName: `/aws/lambda/${matchedFunction}`,
594
- logStreamName: 'test-log-stream',
595
- memoryLimitInMB: '128'
596
- };
597
- const result = await wrappedHandler(event, context);
598
- // Handle Lambda response format for WebSocket
599
- if (result && typeof result === 'object' && result.statusCode) {
600
- // This is a Lambda response object, extract the body
601
- const body = result.body || '';
602
- ws.send(body);
603
- } else {
604
- // This is a direct response, stringify it
605
- ws.send(JSON.stringify(result));
606
- }
607
- } else {
608
- ws.send(JSON.stringify({
609
- error: 'Handler not found'
610
- }));
611
- }
612
- } else {
613
- ws.send(JSON.stringify({
614
- error: 'WebSocket function not found'
615
- }));
616
- }
617
- } catch (error) {
618
- log(`WebSocket error: ${error.message}`, 'error', false);
619
- ws.send(JSON.stringify({
620
- error: error.message
621
- }));
622
- }
623
- });
624
- ws.on('close', ()=>{
625
- log('WebSocket connection closed', 'info', false);
626
- });
627
- });
628
- return wss;
629
- };
630
- const loadEnvFile = (envPath)=>{
631
- const envVars = {};
632
- if (!existsSync(envPath)) {
633
- return envVars;
634
- }
635
- try {
636
- const envContent = readFileSync(envPath, 'utf8');
637
- const lines = envContent.split('\n');
638
- for (const line of lines){
639
- const trimmedLine = line.trim();
640
- // Skip empty lines and comments
641
- if (!trimmedLine || trimmedLine.startsWith('#')) {
642
- continue;
643
- }
644
- // Parse KEY=value format
645
- const equalIndex = trimmedLine.indexOf('=');
646
- if (equalIndex > 0) {
647
- const key = trimmedLine.substring(0, equalIndex).trim();
648
- const value = trimmedLine.substring(equalIndex + 1).trim();
649
- // Remove quotes if present
650
- const cleanValue = value.replace(/^["']|["']$/g, '');
651
- if (key) {
652
- envVars[key] = cleanValue;
653
- }
654
- }
655
- }
656
- } catch (error) {
657
- log(`Warning: Could not load .env file at ${envPath}: ${error.message}`, 'warn', false);
658
- }
659
- return envVars;
660
- };
661
- export const serverless = async (cmd, callback = ()=>({}))=>{
662
- const { cliName = 'Lex', config, debug = false, host: cliHost, httpPort: cliHttpPort, httpsPort: cliHttpsPort, quiet = false, remove = false, test = false, usePublicIp, variables, wsPort: cliWsPort } = cmd;
663
- const spinner = createSpinner(quiet);
664
- log(`${cliName} starting serverless development server...`, 'info', quiet);
665
- await LexConfig.parseConfig(cmd);
666
- const { outputFullPath } = LexConfig.config;
667
- // Load environment variables from .env files
668
- const envPaths = [
669
- pathResolve(process.cwd(), '.env'),
670
- pathResolve(process.cwd(), '.env.local'),
671
- pathResolve(process.cwd(), '.env.development')
672
- ];
673
- let envVars = {};
674
- // Load from .env files in order (later files override earlier ones)
675
- for (const envPath of envPaths){
676
- const fileEnvVars = loadEnvFile(envPath);
677
- if (Object.keys(fileEnvVars).length > 0) {
678
- log(`Loaded environment variables from: ${envPath}`, 'info', quiet);
679
- }
680
- envVars = {
681
- ...envVars,
682
- ...fileEnvVars
683
- };
684
- }
685
- // Start with default NODE_ENV and loaded .env variables
686
- let variablesObj = {
687
- NODE_ENV: 'development',
688
- ...envVars
689
- };
690
- // Override with command line variables if provided
691
- if (variables) {
692
- try {
693
- const cliVars = JSON.parse(variables);
694
- variablesObj = {
695
- ...variablesObj,
696
- ...cliVars
697
- };
698
- } catch (_error) {
699
- log(`\n${cliName} Error: Environment variables option is not a valid JSON object.`, 'error', quiet);
700
- callback(1);
701
- return 1;
702
- }
703
- }
704
- process.env = {
705
- ...process.env,
706
- ...variablesObj
707
- };
708
- // If in test mode, exit early after loading environment variables
709
- if (test) {
710
- log('Test mode: Environment variables loaded, exiting', 'info', quiet);
711
- callback(0);
712
- return 0;
713
- }
714
- if (remove) {
715
- spinner.start('Cleaning output directory...');
716
- await removeFiles(outputFullPath || '');
717
- spinner.succeed('Successfully cleaned output directory!');
718
- }
719
- // Load serverless configuration
720
- let serverlessConfig = {};
721
- try {
722
- // Use getPackageDir to handle npm workspaces correctly
723
- const packageDir = getPackageDir();
724
- // Try multiple config file formats
725
- const configFormats = config ? [
726
- config
727
- ] : [
728
- pathResolve(packageDir, 'lex.config.mjs'),
729
- pathResolve(packageDir, 'lex.config.js'),
730
- pathResolve(packageDir, 'lex.config.cjs'),
731
- pathResolve(packageDir, 'lex.config.ts')
732
- ];
733
- let configPath = null;
734
- for (const candidatePath of configFormats){
735
- if (existsSync(candidatePath)) {
736
- configPath = candidatePath;
737
- break;
738
- }
739
- }
740
- if (configPath) {
741
- log(`Loading serverless config from: ${configPath}`, 'info', quiet);
742
- const configModule = await import(configPath);
743
- serverlessConfig = configModule.default?.serverless || configModule.serverless || {};
744
- log('Serverless config loaded successfully', 'info', quiet);
745
- const functionNames = Object.keys(serverlessConfig.functions || {});
746
- log(`Loaded functions: ${functionNames.length > 0 ? functionNames.join(', ') : 'none'}`, 'info', quiet);
747
- // Debug: Print full config if debug mode
748
- if (debug) {
749
- log(`Full serverless config: ${JSON.stringify(serverlessConfig, null, 2)}`, 'info', false);
750
- }
751
- } else {
752
- log(`No serverless config found. Tried: ${configFormats.join(', ')}`, 'warn', quiet);
753
- }
754
- } catch (error) {
755
- log(`Error loading serverless config: ${error.message}`, 'error', quiet);
756
- if (debug) {
757
- log(`Config error stack: ${error.stack}`, 'error', false);
758
- }
759
- // Don't exit, continue with empty config
760
- }
761
- // Merge config with command line options
762
- // Determine effective host/ports with correct precedence: CLI > config > defaults
763
- const configOffline = serverlessConfig.custom?.['serverless-offline'] || {};
764
- const effectiveHost = cliHost ?? configOffline.host ?? 'localhost';
765
- const toNumber = (v, fallback)=>{
766
- if (v === undefined || v === null || v === '') {
767
- return fallback;
768
- }
769
- const n = typeof v === 'number' ? v : parseInt(String(v));
770
- return Number.isFinite(n) ? n : fallback;
771
- };
772
- const effectiveHttpPort = toNumber(cliHttpPort ?? configOffline.httpPort, 3100);
773
- const effectiveHttpsPort = toNumber(cliHttpsPort ?? configOffline.httpsPort, 3101);
774
- const effectiveWsPort = toNumber(cliWsPort ?? configOffline.wsPort, 3102);
775
- const finalConfig = {
776
- ...serverlessConfig,
777
- custom: {
778
- 'serverless-offline': {
779
- cors: serverlessConfig.custom?.['serverless-offline']?.cors !== false,
780
- host: effectiveHost,
781
- httpPort: effectiveHttpPort,
782
- httpsPort: effectiveHttpsPort,
783
- wsPort: effectiveWsPort
784
- }
785
- }
786
- };
787
- const outputDir = outputFullPath || 'lib';
788
- log(`Using output directory: ${outputDir}`, 'info', quiet);
789
- try {
790
- spinner.start('Starting serverless development server...');
791
- const httpPort = finalConfig.custom['serverless-offline'].httpPort;
792
- const wsPort = finalConfig.custom['serverless-offline'].wsPort;
793
- const host = finalConfig.custom['serverless-offline'].host;
794
- log(`Creating HTTP server on ${host}:${httpPort}`, 'info', quiet);
795
- log(`Creating WebSocket server on port ${wsPort}`, 'info', quiet);
796
- // Create Express server
797
- const expressApp = await createExpressServer(finalConfig, outputDir, httpPort, host, quiet, debug);
798
- // Create WebSocket server
799
- const wsServer = createWebSocketServer(finalConfig, outputDir, wsPort, quiet, debug);
800
- // Handle server errors
801
- wsServer.on('error', (error)=>{
802
- log(`WebSocket server error: ${error.message}`, 'error', quiet);
803
- spinner.fail('Failed to start WebSocket server.');
804
- callback(1);
805
- return;
806
- });
807
- // Start Express server
808
- const server = expressApp.listen(httpPort, host, ()=>{
809
- spinner.succeed('Serverless development server started.');
810
- void resolvePublicIpForDisplay(usePublicIp).then((publicIp)=>{
811
- displayServerStatus(httpPort, finalConfig.custom['serverless-offline'].httpsPort, wsPort, host, quiet, publicIp);
812
- });
813
- });
814
- // Handle Express server errors
815
- server.on('error', (error)=>{
816
- log(`Express server error: ${error.message}`, 'error', quiet);
817
- spinner.fail('Failed to start Express server.');
818
- callback(1);
819
- return;
820
- });
821
- // Handle graceful shutdown
822
- const shutdown = ()=>{
823
- log('\nShutting down serverless development server...', 'info', quiet);
824
- server.close();
825
- wsServer.close();
826
- callback(0);
827
- };
828
- process.on('SIGINT', shutdown);
829
- process.on('SIGTERM', shutdown);
830
- // Keep the process alive
831
- process.stdin.resume();
832
- log('Serverless development server is running. Press Ctrl+C to stop.', 'info', quiet);
833
- // Don't call callback here, let the process stay alive
834
- return 0;
835
- } catch (error) {
836
- log(`\n${cliName} Error: ${error.message}`, 'error', quiet);
837
- spinner.fail('Failed to start serverless development server.');
838
- callback(1);
839
- return 1;
840
- }
841
- };
842
-
843
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9jb21tYW5kcy9zZXJ2ZXJsZXNzL3NlcnZlcmxlc3MudHMiXSwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTgtUHJlc2VudCwgTml0cm9nZW4gTGFicywgSW5jLlxuICogQ29weXJpZ2h0cyBsaWNlbnNlZCB1bmRlciB0aGUgTUlUIExpY2Vuc2UuIFNlZSB0aGUgYWNjb21wYW55aW5nIExJQ0VOU0UgZmlsZSBmb3IgdGVybXMuXG4gKi9cbmltcG9ydCBib3hlbiBmcm9tICdib3hlbic7XG5pbXBvcnQgY2hhbGsgZnJvbSAnY2hhbGsnO1xuaW1wb3J0IGV4cHJlc3MgZnJvbSAnZXhwcmVzcyc7XG5pbXBvcnQge3JlYWRGaWxlU3luYywgZXhpc3RzU3luYywgbWtkaXJTeW5jLCB3cml0ZUZpbGVTeW5jfSBmcm9tICdmcyc7XG5pbXBvcnQge2hvbWVkaXJ9IGZyb20gJ29zJztcbmltcG9ydCB7cmVzb2x2ZSBhcyBwYXRoUmVzb2x2ZSwgam9pbiwgaXNBYnNvbHV0ZX0gZnJvbSAncGF0aCc7XG5pbXBvcnQge3BhdGhUb0ZpbGVVUkx9IGZyb20gJ3VybCc7XG5pbXBvcnQge1dlYlNvY2tldFNlcnZlcn0gZnJvbSAnd3MnO1xuXG5pbXBvcnQge0xleENvbmZpZywgZ2V0UGFja2FnZURpcn0gZnJvbSAnLi4vLi4vTGV4Q29uZmlnLmpzJztcbmltcG9ydCB7Y3JlYXRlU3Bpbm5lciwgcmVtb3ZlRmlsZXN9IGZyb20gJy4uLy4uL3V0aWxzL2FwcC5qcyc7XG5pbXBvcnQge2xvZ30gZnJvbSAnLi4vLi4vdXRpbHMvbG9nLmpzJztcblxuZXhwb3J0IGludGVyZmFjZSBTZXJ2ZXJsZXNzT3B0aW9ucyB7XG4gIHJlYWRvbmx5IGNsaU5hbWU/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IGNvbmZpZz86IHN0cmluZztcbiAgcmVhZG9ubHkgZGVidWc/OiBib29sZWFuO1xuICByZWFkb25seSBob3N0Pzogc3RyaW5nO1xuICByZWFkb25seSBodHRwUG9ydD86IG51bWJlcjtcbiAgcmVhZG9ubHkgaHR0cHNQb3J0PzogbnVtYmVyO1xuICByZWFkb25seSBxdWlldD86IGJvb2xlYW47XG4gIHJlYWRvbmx5IHJlbW92ZT86IGJvb2xlYW47XG4gIHJlYWRvbmx5IHRlc3Q/OiBib29sZWFuO1xuICByZWFkb25seSB1c2VQdWJsaWNJcD86IGJvb2xlYW47XG4gIHJlYWRvbmx5IHZhcmlhYmxlcz86IHN0cmluZztcbiAgcmVhZG9ubHkgd3NQb3J0PzogbnVtYmVyO1xufVxuXG5leHBvcnQgdHlwZSBTZXJ2ZXJsZXNzQ2FsbGJhY2sgPSAoc3RhdHVzOiBudW1iZXIpID0+IHZvaWQ7XG5cbmludGVyZmFjZSBQdWJsaWNJcENhY2hlIHtcbiAgaXA6IHN0cmluZztcbiAgdGltZXN0YW1wOiBudW1iZXI7XG59XG5cbmludGVyZmFjZSBTZXJ2ZXJsZXNzSGFuZGxlciB7XG4gIHJlYWRvbmx5IGhhbmRsZXI6IHN0cmluZztcbiAgcmVhZG9ubHkgZXZlbnRzPzogQXJyYXk8e1xuICAgIHJlYWRvbmx5IGh0dHA/OiB7XG4gICAgICByZWFkb25seSBjb3JzPzogYm9vbGVhbjtcbiAgICAgIHJlYWRvbmx5IG1ldGhvZD86IHN0cmluZztcbiAgICAgIHJlYWRvbmx5IHBhdGg/OiBzdHJpbmc7XG4gICAgfTtcbiAgICByZWFkb25seSB3ZWJzb2NrZXQ/OiB7XG4gICAgICByZWFkb25seSByb3V0ZT86IHN0cmluZztcbiAgICB9O1xuICB9Pjtcbn1cblxuaW50ZXJmYWNlIFNlcnZlcmxlc3NDb25maWcge1xuICByZWFkb25seSBjdXN0b20/OiB7XG4gICAgcmVhZG9ubHkgJ3NlcnZlcmxlc3Mtb2ZmbGluZSc/OiB7XG4gICAgICByZWFkb25seSBjb3JzPzogYm9vbGVhbjtcbiAgICAgIHJlYWRvbmx5IGhvc3Q/OiBzdHJpbmc7XG4gICAgICByZWFkb25seSBodHRwUG9ydD86IG51bWJlcjtcbiAgICAgIHJlYWRvbmx5IGh0dHBzUG9ydD86IG51bWJlcjtcbiAgICAgIHJlYWRvbmx5IHdzUG9ydD86IG51bWJlcjtcbiAgICB9O1xuICB9O1xuICByZWFkb25seSBmdW5jdGlvbnM/OiBSZWNvcmQ8c3RyaW5nLCBTZXJ2ZXJsZXNzSGFuZGxlcj47XG59XG5cbmludGVyZmFjZSBXZWJTb2NrZXRDbGllbnRMaWtlIHtcbiAgb24oZXZlbnQ6IHN0cmluZywgbGlzdGVuZXI6ICguLi5hcmdzOiBhbnlbXSkgPT4gdm9pZCk6IHZvaWQ7XG4gIHNlbmQoZGF0YTogc3RyaW5nKTogdm9pZDtcbn1cblxuY29uc3QgZ2V0Q2FjaGVEaXIgPSAoKTogc3RyaW5nID0+IHtcbiAgY29uc3QgY2FjaGVEaXIgPSBqb2luKGhvbWVkaXIoKSwgJy5sZXgtY2FjaGUnKTtcbiAgaWYoIWV4aXN0c1N5bmMoY2FjaGVEaXIpKSB7XG4gICAgbWtkaXJTeW5jKGNhY2hlRGlyLCB7cmVjdXJzaXZlOiB0cnVlfSk7XG4gIH1cbiAgcmV0dXJuIGNhY2hlRGlyO1xufTtcblxuY29uc3QgZ2V0Q2FjaGVQYXRoID0gKCk6IHN0cmluZyA9PiBqb2luKGdldENhY2hlRGlyKCksICdwdWJsaWMtaXAuanNvbicpO1xuXG5jb25zdCByZWFkUHVibGljSXBDYWNoZSA9ICgpOiBQdWJsaWNJcENhY2hlIHwgbnVsbCA9PiB7XG4gIGNvbnN0IGNhY2hlUGF0aCA9IGdldENhY2hlUGF0aCgpO1xuICBpZighZXhpc3RzU3luYyhjYWNoZVBhdGgpKSB7XG4gICAgcmV0dXJuIG51bGw7XG4gIH1cblxuICB0cnkge1xuICAgIGNvbnN0IGNhY2hlRGF0YSA9IHJlYWRGaWxlU3luYyhjYWNoZVBhdGgsICd1dGY4Jyk7XG4gICAgY29uc3QgY2FjaGU6IFB1YmxpY0lwQ2FjaGUgPSBKU09OLnBhcnNlKGNhY2hlRGF0YSk7XG5cbiAgICAvLyBDaGVjayBpZiBjYWNoZSBpcyBvbGRlciB0aGFuIDEgd2Vla1xuICAgIGNvbnN0IG9uZVdlZWtNcyA9IDcgKiAyNCAqIDYwICogNjAgKiAxMDAwO1xuICAgIGlmKERhdGUubm93KCkgLSBjYWNoZS50aW1lc3RhbXAgPiBvbmVXZWVrTXMpIHtcbiAgICAgIHJldHVybiBudWxsO1xuICAgIH1cblxuICAgIHJldHVybiBjYWNoZTtcbiAgfSBjYXRjaHtcbiAgICByZXR1cm4gbnVsbDtcbiAgfVxufTtcblxuY29uc3Qgd3JpdGVQdWJsaWNJcENhY2hlID0gKGlwOiBzdHJpbmcpOiB2b2lkID0+IHtcbiAgY29uc3QgY2FjaGVQYXRoID0gZ2V0Q2FjaGVQYXRoKCk7XG4gIGNvbnN0IGNhY2hlOiBQdWJsaWNJcENhY2hlID0ge1xuICAgIGlwLFxuICAgIHRpbWVzdGFtcDogRGF0ZS5ub3coKVxuICB9O1xuICB3cml0ZUZpbGVTeW5jKGNhY2hlUGF0aCwgSlNPTi5zdHJpbmdpZnkoY2FjaGUsIG51bGwsIDIpKTtcbn07XG5cbmNvbnN0IGZldGNoUHVibGljSXAgPSAoZm9yY2VSZWZyZXNoOiBib29sZWFuID0gZmFsc2UpOiBQcm9taXNlPHN0cmluZyB8IHVuZGVmaW5lZD4gPT4gbmV3IFByb21pc2UoKHJlc29sdmUpID0+IHtcbiAgaWYoIWZvcmNlUmVmcmVzaCkge1xuICAgIGNvbnN0IGNhY2hlZCA9IHJlYWRQdWJsaWNJcENhY2hlKCk7XG4gICAgaWYoY2FjaGVkKSB7XG4gICAgICByZXNvbHZlKGNhY2hlZC5pcCk7XG4gICAgICByZXR1cm47XG4gICAgfVxuICB9XG5cbiAgLy8gVXNlIGZldGNoIGluc3RlYWQgb2YgaHR0cHNcbiAgZmV0Y2goJ2h0dHBzOi8vYXBpLmlwaWZ5Lm9yZycpXG4gICAgLnRoZW4oKHJlcykgPT4gcmVzLnRleHQoKSlcbiAgICAudGhlbigoZGF0YSkgPT4ge1xuICAgICAgY29uc3QgaXAgPSBkYXRhLnRyaW0oKTtcbiAgICAgIGlmKGlwKSB7XG4gICAgICAgIHdyaXRlUHVibGljSXBDYWNoZShpcCk7XG4gICAgICB9XG4gICAgICByZXNvbHZlKGlwKTtcbiAgICB9KVxuICAgIC5jYXRjaCgoKSA9PiByZXNvbHZlKHVuZGVmaW5lZCkpO1xufSk7XG5cbmNvbnN0IHJlc29sdmVQdWJsaWNJcEZvckRpc3BsYXkgPSBhc3luYyAoZm9yY2VSZWZyZXNoOiBib29sZWFuID0gZmFsc2UpOiBQcm9taXNlPHN0cmluZyB8IHVuZGVmaW5lZD4gPT4ge1xuICBjb25zdCB0aW1lb3V0TXMgPSAxNTAwO1xuICBjb25zdCB0aW1lb3V0UHJvbWlzZSA9IG5ldyBQcm9taXNlPHVuZGVmaW5lZD4oKHJlc29sdmUpID0+IHtcbiAgICBzZXRUaW1lb3V0KCgpID0+IHJlc29sdmUodW5kZWZpbmVkKSwgdGltZW91dE1zKTtcbiAgfSk7XG5cbiAgdHJ5IHtcbiAgICByZXR1cm4gYXdhaXQgUHJvbWlzZS5yYWNlKFtmZXRjaFB1YmxpY0lwKGZvcmNlUmVmcmVzaCksIHRpbWVvdXRQcm9taXNlXSk7XG4gIH0gY2F0Y2h7XG4gICAgcmV0dXJuIHVuZGVmaW5lZDtcbiAgfVxufTtcblxuY29uc3QgZGlzcGxheVNlcnZlclN0YXR1cyA9IChcbiAgaHR0cFBvcnQ6IG51bWJlcixcbiAgaHR0cHNQb3J0OiBudW1iZXIsXG4gIHdzUG9ydDogbnVtYmVyLFxuICBob3N0OiBzdHJpbmcsXG4gIHF1aWV0OiBib29sZWFuLFxuICBwdWJsaWNJcD86IHN0cmluZ1xuKSA9PiB7XG4gIGlmKHF1aWV0KSB7XG4gICAgcmV0dXJuO1xuICB9XG5cbiAgY29uc3QgaHR0cFVybCA9IGBodHRwOi8vJHtob3N0fToke2h0dHBQb3J0fWA7XG4gIGNvbnN0IGh0dHBzVXJsID0gYGh0dHBzOi8vJHtob3N0fToke2h0dHBzUG9ydH1gO1xuICBjb25zdCB3c1VybCA9IGB3czovLyR7aG9zdH06JHt3c1BvcnR9YDtcbiAgY29uc3Qgd3NzVXJsID0gYHdzczovLyR7aG9zdH06JHt3c1BvcnR9YDtcblxuICBsZXQgdXJsTGluZXMgPSBgJHtjaGFsay5ncmVlbignSFRUUDonKX0gICAgICAke2NoYWxrLnVuZGVybGluZShodHRwVXJsKX1cXG5gO1xuICB1cmxMaW5lcyArPSBgJHtjaGFsay5ncmVlbignSFRUUFM6Jyl9ICAgICAke2NoYWxrLnVuZGVybGluZShodHRwc1VybCl9XFxuYDtcbiAgdXJsTGluZXMgKz0gYCR7Y2hhbGsuZ3JlZW4oJ1dlYlNvY2tldDonKX0gJHtjaGFsay51bmRlcmxpbmUod3NVcmwpfVxcbmA7XG4gIHVybExpbmVzICs9IGAke2NoYWxrLmdyZWVuKCdXU1M6Jyl9ICAgICAgICR7Y2hhbGsudW5kZXJsaW5lKHdzc1VybCl9XFxuYDtcblxuICBpZihwdWJsaWNJcCkge1xuICAgIHVybExpbmVzICs9IGBcXG4ke2NoYWxrLmdyZWVuKCdQdWJsaWM6Jyl9ICAgICR7Y2hhbGsudW5kZXJsaW5lKGBodHRwOi8vJHtwdWJsaWNJcH06JHtodHRwUG9ydH1gKX1cXG5gO1xuICB9XG5cbiAgY29uc3Qgc3RhdHVzQm94ID0gYm94ZW4oXG4gICAgYCR7Y2hhbGsuY3lhbi5ib2xkKCfwn5qAIFNlcnZlcmxlc3MgRGV2ZWxvcG1lbnQgU2VydmVyIFJ1bm5pbmcnKX1cXG5cXG4ke3VybExpbmVzfVxcbmAgK1xuICAgIGAke2NoYWxrLnllbGxvdygnUHJlc3MgQ3RybCtDIHRvIHN0b3AgdGhlIHNlcnZlcicpfWAsXG4gICAge1xuICAgICAgYmFja2dyb3VuZENvbG9yOiAnIzFhMWExYScsXG4gICAgICBib3JkZXJDb2xvcjogJ2N5YW4nLFxuICAgICAgYm9yZGVyU3R5bGU6ICdyb3VuZCcsXG4gICAgICBtYXJnaW46IDEsXG4gICAgICBwYWRkaW5nOiAxXG4gICAgfVxuICApO1xuXG4gIGNvbnNvbGUubG9nKGBcXG4ke3N0YXR1c0JveH1cXG5gKTtcbn07XG5cbmNvbnN0IGxvYWRIYW5kbGVyID0gYXN5bmMgKGhhbmRsZXJQYXRoOiBzdHJpbmcsIG91dHB1dERpcjogc3RyaW5nKSA9PiB7XG4gIHRyeSB7XG4gICAgY29uc29sZS5sb2coYFtTZXJ2ZXJsZXNzXSBQYXJzaW5nIGhhbmRsZXIgcGF0aDogJHtoYW5kbGVyUGF0aH1gKTtcblxuICAgIC8vIFBhcnNlIEFXUyBMYW1iZGEgaGFuZGxlciBmb3JtYXQ6IFwicGF0aC90by9maWxlLmV4cG9ydE5hbWVcIiBvciBcImZpbGUuZXhwb3J0TmFtZVwiXG4gICAgLy8gRXhhbXBsZXM6IFwiaW5kZXguaGFuZGxlclwiLCBcImhhbmRsZXJzL2FwaS5oYW5kbGVyXCIsIFwic3JjL2luZGV4LmRlZmF1bHRcIlxuICAgIGNvbnN0IGhhbmRsZXJQYXJ0cyA9IGhhbmRsZXJQYXRoLnNwbGl0KCcuJyk7XG4gICAgY29uc29sZS5sb2coJ1tTZXJ2ZXJsZXNzXSBIYW5kbGVyIHBhcnRzIGFmdGVyIHNwbGl0OicsIGhhbmRsZXJQYXJ0cyk7XG5cbiAgICBsZXQgZmlsZVBhdGg6IHN0cmluZztcbiAgICBsZXQgZXhwb3J0TmFtZTogc3RyaW5nIHwgbnVsbCA9IG51bGw7XG5cbiAgICBpZihoYW5kbGVyUGFydHMubGVuZ3RoID4gMSkge1xuICAgICAgLy8gQVdTIExhbWJkYSBmb3JtYXQ6IFwiZmlsZS5leHBvcnROYW1lXCJcbiAgICAgIC8vIFRha2UgdGhlIGxhc3QgcGFydCBhcyBleHBvcnQgbmFtZSwgcmVzdCBhcyBmaWxlIHBhdGhcbiAgICAgIC8vIEhhbmRsZSBjYXNlcyBsaWtlIFwiaW5kZXguaGFuZGxlclwiIC0+IGZpbGU6IFwiaW5kZXhcIiwgZXhwb3J0OiBcImhhbmRsZXJcIlxuICAgICAgLy8gT3IgXCJoYW5kbGVycy9hcGkuaGFuZGxlclwiIC0+IGZpbGU6IFwiaGFuZGxlcnMvYXBpXCIsIGV4cG9ydDogXCJoYW5kbGVyXCJcbiAgICAgIGV4cG9ydE5hbWUgPSBoYW5kbGVyUGFydHNbaGFuZGxlclBhcnRzLmxlbmd0aCAtIDFdIHx8IG51bGw7XG4gICAgICBmaWxlUGF0aCA9IGhhbmRsZXJQYXJ0cy5zbGljZSgwLCAtMSkuam9pbignLicpO1xuICAgICAgY29uc29sZS5sb2coYFtTZXJ2ZXJsZXNzXSBQYXJzZWQgQVdTIExhbWJkYSBmb3JtYXQgLSBmaWxlUGF0aDogXCIke2ZpbGVQYXRofVwiLCBleHBvcnROYW1lOiBcIiR7ZXhwb3J0TmFtZX1cImApO1xuICAgIH0gZWxzZSB7XG4gICAgICAvLyBTaW1wbGUgZm9ybWF0OiBqdXN0IHRoZSBmaWxlIHBhdGhcbiAgICAgIGZpbGVQYXRoID0gaGFuZGxlclBhdGg7XG4gICAgICBjb25zb2xlLmxvZyhgW1NlcnZlcmxlc3NdIFNpbXBsZSBmb3JtYXQgLSBmaWxlUGF0aDogXCIke2ZpbGVQYXRofVwiYCk7XG4gICAgfVxuXG4gICAgLy8gRW5zdXJlIGZpbGVQYXRoIGRvZXNuJ3QgaGF2ZSB0aGUgZXhwb3J0IG5hbWUgaW4gaXRcbiAgICBpZihmaWxlUGF0aC5pbmNsdWRlcygnLmhhbmRsZXInKSB8fCBmaWxlUGF0aC5pbmNsdWRlcygnLmRlZmF1bHQnKSkge1xuICAgICAgY29uc29sZS5lcnJvcihgW1NlcnZlcmxlc3NdIFdBUk5JTkc6IGZpbGVQYXRoIHN0aWxsIGNvbnRhaW5zIGV4cG9ydCBuYW1lISBmaWxlUGF0aDogXCIke2ZpbGVQYXRofVwiYCk7XG4gICAgICAvLyBUcnkgdG8gZml4IGl0IC0gcmVtb3ZlIHRoZSBsYXN0IHBhcnQgaWYgaXQgbG9va3MgbGlrZSBhbiBleHBvcnQgbmFtZVxuICAgICAgY29uc3QgcGF0aFBhcnRzID0gZmlsZVBhdGguc3BsaXQoJy4nKTtcbiAgICAgIGlmKHBhdGhQYXJ0cy5sZW5ndGggPiAxKSB7XG4gICAgICAgIGNvbnN0IGxhc3RQYXJ0ID0gcGF0aFBhcnRzW3BhdGhQYXJ0cy5sZW5ndGggLSAxXTtcbiAgICAgICAgaWYoWydoYW5kbGVyJywgJ2RlZmF1bHQnLCAnaW5kZXgnXS5pbmNsdWRlcyhsYXN0UGFydCkgJiYgIWV4cG9ydE5hbWUpIHtcbiAgICAgICAgICBleHBvcnROYW1lID0gbGFzdFBhcnQ7XG4gICAgICAgICAgZmlsZVBhdGggPSBwYXRoUGFydHMuc2xpY2UoMCwgLTEpLmpvaW4oJy4nKTtcbiAgICAgICAgICBjb25zb2xlLmxvZyhgW1NlcnZlcmxlc3NdIEZpeGVkIC0gZmlsZVBhdGg6IFwiJHtmaWxlUGF0aH1cIiwgZXhwb3J0TmFtZTogXCIke2V4cG9ydE5hbWV9XCJgKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cblxuICAgIC8vIEhhbmRsZSBib3RoIHJlbGF0aXZlIHBhdGhzIGFuZCBhYnNvbHV0ZSBwYXRoc1xuICAgIGxldCBmdWxsUGF0aDogc3RyaW5nO1xuICAgIGlmKGlzQWJzb2x1dGUoZmlsZVBhdGgpKSB7XG4gICAgICBmdWxsUGF0aCA9IGZpbGVQYXRoO1xuICAgIH0gZWxzZSB7XG4gICAgICBmdWxsUGF0aCA9IHBhdGhSZXNvbHZlKG91dHB1dERpciwgZmlsZVBhdGgpO1xuICAgIH1cbiAgICBjb25zb2xlLmxvZyhgW1NlcnZlcmxlc3NdIFJlc29sdmVkIGZ1bGxQYXRoIChiZWZvcmUgZXh0ZW5zaW9ucyk6ICR7ZnVsbFBhdGh9YCk7XG5cbiAgICAvLyBUcnkgZGlmZmVyZW50IGV4dGVuc2lvbnMgaWYgZmlsZSBkb2Vzbid0IGV4aXN0XG4gICAgaWYoIWV4aXN0c1N5bmMoZnVsbFBhdGgpKSB7XG4gICAgICBjb25zdCBleHRlbnNpb25zID0gWycuanMnLCAnLm1qcycsICcuY2pzJ107XG4gICAgICBjb25zdCBwYXRoV2l0aG91dEV4dCA9IGZ1bGxQYXRoLnJlcGxhY2UoL1xcLihqc3xtanN8Y2pzKSQvLCAnJyk7XG4gICAgICBjb25zb2xlLmxvZyhgW1NlcnZlcmxlc3NdIFRyeWluZyBleHRlbnNpb25zLiBCYXNlIHBhdGg6ICR7cGF0aFdpdGhvdXRFeHR9YCk7XG4gICAgICBmb3IoY29uc3QgZXh0IG9mIGV4dGVuc2lvbnMpIHtcbiAgICAgICAgY29uc3QgY2FuZGlkYXRlUGF0aCA9IHBhdGhXaXRob3V0RXh0ICsgZXh0O1xuICAgICAgICBjb25zb2xlLmxvZyhgW1NlcnZlcmxlc3NdIENoZWNraW5nOiAke2NhbmRpZGF0ZVBhdGh9IChleGlzdHM6ICR7ZXhpc3RzU3luYyhjYW5kaWRhdGVQYXRoKX0pYCk7XG4gICAgICAgIGlmKGV4aXN0c1N5bmMoY2FuZGlkYXRlUGF0aCkpIHtcbiAgICAgICAgICBmdWxsUGF0aCA9IGNhbmRpZGF0ZVBhdGg7XG4gICAgICAgICAgY29uc29sZS5sb2coYFtTZXJ2ZXJsZXNzXSBGb3VuZCBmaWxlIHdpdGggZXh0ZW5zaW9uOiAke2Z1bGxQYXRofWApO1xuICAgICAgICAgIGJyZWFrO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnNvbGUubG9nKGBbU2VydmVybGVzc10gRmlsZSBleGlzdHMgd2l0aG91dCB0cnlpbmcgZXh0ZW5zaW9uczogJHtmdWxsUGF0aH1gKTtcbiAgICB9XG5cbiAgICBjb25zb2xlLmxvZyhgW1NlcnZlcmxlc3NdIEZpbmFsIGZ1bGxQYXRoOiAke2Z1bGxQYXRofWApO1xuICAgIGNvbnNvbGUubG9nKGBbU2VydmVybGVzc10gRXhwb3J0IG5hbWU6ICR7ZXhwb3J0TmFtZSB8fCAnZGVmYXVsdC9oYW5kbGVyJ31gKTtcbiAgICBjb25zb2xlLmxvZyhgW1NlcnZlcmxlc3NdIEZpbGUgZXhpc3RzOiAke2V4aXN0c1N5bmMoZnVsbFBhdGgpfWApO1xuXG4gICAgaWYoIWV4aXN0c1N5bmMoZnVsbFBhdGgpKSB7XG4gICAgICBjb25zb2xlLmVycm9yKGBbU2VydmVybGVzc10gSGFuZGxlciBmaWxlIG5vdCBmb3VuZDogJHtmdWxsUGF0aH1gKTtcbiAgICAgIGNvbnNvbGUuZXJyb3IoYFtTZXJ2ZXJsZXNzXSBPdXRwdXQgZGlyZWN0b3J5OiAke291dHB1dERpcn1gKTtcbiAgICAgIGNvbnNvbGUuZXJyb3IoYFtTZXJ2ZXJsZXNzXSBIYW5kbGVyIHBhdGggZnJvbSBjb25maWc6ICR7aGFuZGxlclBhdGh9YCk7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoYEhhbmRsZXIgZmlsZSBub3QgZm91bmQ6ICR7ZnVsbFBhdGh9YCk7XG4gICAgfVxuXG4gICAgLy8gRHluYW1pYyBpbXBvcnQgb2YgdGhlIGhhbmRsZXIgd2l0aCBiZXR0ZXIgZXJyb3IgaGFuZGxpbmdcbiAgICAvLyBBZGQgLmpzIGV4dGVuc2lvbiBpZiBpbXBvcnRpbmcgVHlwZVNjcmlwdCBjb21waWxlZCBvdXRwdXRcbiAgICBjb25zdCBpbXBvcnRQYXRoID0gZnVsbFBhdGguZW5kc1dpdGgoJy50cycpID8gZnVsbFBhdGgucmVwbGFjZSgvXFwudHMkLywgJy5qcycpIDogZnVsbFBhdGg7XG5cbiAgICB0cnkge1xuICAgICAgLy8gQ29udmVydCB0byBmaWxlOi8vIFVSTCBmb3IgRVMgbW9kdWxlIGltcG9ydHMgKHJlcXVpcmVkIGZvciBhYnNvbHV0ZSBwYXRocylcbiAgICAgIC8vIFVzZSBwYXRoVG9GaWxlVVJMIHRvIGVuc3VyZSBwcm9wZXIgZmlsZTovLyBVUkwgZm9ybWF0XG4gICAgICBjb25zdCBpbXBvcnRVcmwgPSBwYXRoVG9GaWxlVVJMKGltcG9ydFBhdGgpLmhyZWY7XG4gICAgICBjb25zb2xlLmxvZyhgW1NlcnZlcmxlc3NdIEltcG9ydGluZyBoYW5kbGVyIGZyb206ICR7aW1wb3J0VXJsfWApO1xuICAgICAgY29uc29sZS5sb2coYFtTZXJ2ZXJsZXNzXSBGaWxlIHBhdGg6ICR7aW1wb3J0UGF0aH1gKTtcblxuICAgICAgLy8gVXNlIGltcG9ydCgpIHdpdGggdGhlIGZpbGUgVVJMXG4gICAgICAvLyBOb3RlOiBJZiB0aGUgaGFuZGxlciBmaWxlIGhhcyBpbXBvcnQgZXJyb3JzIChsaWtlIG1pc3NpbmcgZGVwZW5kZW5jaWVzKSxcbiAgICAgIC8vIHRob3NlIHdpbGwgc3VyZmFjZSBoZXJlLCBidXQgdGhhdCdzIGEgaGFuZGxlciBjb2RlIGlzc3VlLCBub3QgYSBsb2FkZXIgaXNzdWVcbiAgICAgIGNvbnN0IGhhbmRsZXJNb2R1bGUgPSBhd2FpdCBpbXBvcnQoaW1wb3J0VXJsKTtcbiAgICAgIGNvbnNvbGUubG9nKGBbU2VydmVybGVzc10gSGFuZGxlciBtb2R1bGUgbG9hZGVkIHN1Y2Nlc3NmdWxseS4gRXhwb3J0czogJHtPYmplY3Qua2V5cyhoYW5kbGVyTW9kdWxlKS5qb2luKCcsICcpfWApO1xuXG4gICAgICAvLyBHZXQgdGhlIGhhbmRsZXIgYmFzZWQgb24gZXhwb3J0IG5hbWUgb3IgdHJ5IGRlZmF1bHRzXG4gICAgICBsZXQgaGFuZGxlcjogYW55O1xuICAgICAgaWYoZXhwb3J0TmFtZSkge1xuICAgICAgICBoYW5kbGVyID0gaGFuZGxlck1vZHVsZVtleHBvcnROYW1lXVxuICAgICAgICAgIHx8IGhhbmRsZXJNb2R1bGUuZGVmYXVsdD8uW2V4cG9ydE5hbWVdXG4gICAgICAgICAgfHwgaGFuZGxlck1vZHVsZVsnbW9kdWxlLmV4cG9ydHMnXT8uW2V4cG9ydE5hbWVdO1xuICAgICAgICBpZighaGFuZGxlcikge1xuICAgICAgICAgIGNvbnNvbGUuZXJyb3IoYFtTZXJ2ZXJsZXNzXSBFeHBvcnQgXCIke2V4cG9ydE5hbWV9XCIgbm90IGZvdW5kIGluIG1vZHVsZS4gQXZhaWxhYmxlIGV4cG9ydHM6ICR7T2JqZWN0LmtleXMoaGFuZGxlck1vZHVsZSkuam9pbignLCAnKX1gKTtcbiAgICAgICAgICByZXR1cm4gbnVsbDtcbiAgICAgICAgfVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgLy8gVHJ5IGRlZmF1bHQsIGhhbmRsZXIsIG9yIHRoZSBtb2R1bGUgaXRzZWxmXG4gICAgICAgIGhhbmRsZXIgPSBoYW5kbGVyTW9kdWxlLmRlZmF1bHQgfHwgaGFuZGxlck1vZHVsZS5oYW5kbGVyIHx8IGhhbmRsZXJNb2R1bGU7XG4gICAgICB9XG5cbiAgICAgIGNvbnNvbGUubG9nKGBbU2VydmVybGVzc10gSGFuZGxlciBmb3VuZDogJHt0eXBlb2YgaGFuZGxlcn0sIGlzRnVuY3Rpb246ICR7dHlwZW9mIGhhbmRsZXIgPT09ICdmdW5jdGlvbid9YCk7XG5cbiAgICAgIGlmKHR5cGVvZiBoYW5kbGVyICE9PSAnZnVuY3Rpb24nKSB7XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoYFtTZXJ2ZXJsZXNzXSBIYW5kbGVyIGlzIG5vdCBhIGZ1bmN0aW9uLiBUeXBlOiAke3R5cGVvZiBoYW5kbGVyfSwgVmFsdWU6YCwgaGFuZGxlcik7XG4gICAgICAgIHJldHVybiBudWxsO1xuICAgICAgfVxuXG4gICAgICByZXR1cm4gaGFuZGxlcjtcbiAgICB9IGNhdGNoKGltcG9ydEVycm9yOiBhbnkpIHtcbiAgICAgIGNvbnNvbGUuZXJyb3IoYFtTZXJ2ZXJsZXNzXSBJbXBvcnQgZXJyb3IgZm9yIGhhbmRsZXIgJHtoYW5kbGVyUGF0aH06YCwgaW1wb3J0RXJyb3IubWVzc2FnZSk7XG4gICAgICBjb25zb2xlLmVycm9yKCdbU2VydmVybGVzc10gSW1wb3J0IGVycm9yIHN0YWNrOicsIGltcG9ydEVycm9yLnN0YWNrKTtcblxuICAgICAgLy8gQ2hlY2sgaWYgdGhpcyBpcyBhIGRlcGVuZGVuY3kgcmVzb2x1dGlvbiBlcnJvciAoY29tbW9uIHdpdGggRVMgbW9kdWxlcylcbiAgICAgIGlmKGltcG9ydEVycm9yLm1lc3NhZ2UgJiYgaW1wb3J0RXJyb3IubWVzc2FnZS5pbmNsdWRlcygnQ2Fubm90IGZpbmQgbW9kdWxlJykpIHtcbiAgICAgICAgY29uc29sZS5lcnJvcignW1NlcnZlcmxlc3NdIFRoaXMgYXBwZWFycyB0byBiZSBhIGRlcGVuZGVuY3kgcmVzb2x1dGlvbiBlcnJvci4nKTtcbiAgICAgICAgY29uc29sZS5lcnJvcignW1NlcnZlcmxlc3NdIFRoZSBoYW5kbGVyIGZpbGUgZXhpc3RzLCBidXQgb25lIG9mIGl0cyBpbXBvcnRzIGlzIGZhaWxpbmcuJyk7XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoJ1tTZXJ2ZXJsZXNzXSBDaGVjayB0aGF0IGFsbCBkZXBlbmRlbmNpZXMgaW4gdGhlIGhhbmRsZXIgZmlsZSBhcmUgcHJvcGVybHkgaW5zdGFsbGVkLicpO1xuICAgICAgICBjb25zb2xlLmVycm9yKGBbU2VydmVybGVzc10gSGFuZGxlciBmaWxlOiAke2ltcG9ydFBhdGh9YCk7XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoJ1tTZXJ2ZXJsZXNzXSBNYWtlIHN1cmUgdGhlIGhhbmRsZXIgYW5kIGl0cyBkZXBlbmRlbmNpZXMgYXJlIGNvbXBpbGVkIGNvcnJlY3RseS4nKTtcbiAgICAgIH1cblxuICAgICAgcmV0dXJuIG51bGw7XG4gICAgfVxuICB9IGNhdGNoKGVycm9yOiBhbnkpIHtcbiAgICBjb25zb2xlLmVycm9yKGBbU2VydmVybGVzc10gRXJyb3IgbG9hZGluZyBoYW5kbGVyICR7aGFuZGxlclBhdGh9OmAsIGVycm9yLm1lc3NhZ2UpO1xuICAgIGNvbnNvbGUuZXJyb3IoJ1tTZXJ2ZXJsZXNzXSBFcnJvciBzdGFjazonLCBlcnJvci5zdGFjayk7XG4gICAgcmV0dXJuIG51bGw7XG4gIH1cbn07XG5cbmNvbnN0IGNhcHR1cmVDb25zb2xlTG9ncyA9IChoYW5kbGVyOiAoZXZlbnQ6IGFueSwgY29udGV4dDogYW55KSA9PiBQcm9taXNlPGFueT4sIHF1aWV0OiBib29sZWFuKSA9PiB7XG4gIGlmKHF1aWV0KSB7XG4gICAgcmV0dXJuIGhhbmRsZXI7XG4gIH1cblxuICByZXR1cm4gYXN5bmMgKGV2ZW50OiBhbnksIGNvbnRleHQ6IGFueSkgPT4ge1xuICAgIC8vIENhcHR1cmUgY29uc29sZS5sb2csIGNvbnNvbGUuZXJyb3IsIGV0Yy5cbiAgICBjb25zdCBvcmlnaW5hbENvbnNvbGVMb2cgPSBjb25zb2xlLmxvZztcbiAgICBjb25zdCBvcmlnaW5hbENvbnNvbGVFcnJvciA9IGNvbnNvbGUuZXJyb3I7XG4gICAgY29uc3Qgb3JpZ2luYWxDb25zb2xlV2FybiA9IGNvbnNvbGUud2FybjtcbiAgICBjb25zdCBvcmlnaW5hbENvbnNvbGVJbmZvID0gY29uc29sZS5pbmZvO1xuXG4gICAgY29uc3QgbG9nczogc3RyaW5nW10gPSBbXTtcblxuICAgIGNvbnNvbGUubG9nID0gKC4uLmFyZ3M6IGFueVtdKSA9PiB7XG4gICAgICBsb2dzLnB1c2goYFtMT0ddICR7YXJncy5qb2luKCcgJyl9YCk7XG4gICAgICBvcmlnaW5hbENvbnNvbGVMb2coLi4uYXJncyk7XG4gICAgfTtcblxuICAgIGNvbnNvbGUuZXJyb3IgPSAoLi4uYXJnczogYW55W10pID0+IHtcbiAgICAgIGxvZ3MucHVzaChgW0VSUk9SXSAke2FyZ3Muam9pbignICcpfWApO1xuICAgICAgb3JpZ2luYWxDb25zb2xlRXJyb3IoLi4uYXJncyk7XG4gICAgfTtcblxuICAgIGNvbnNvbGUud2FybiA9ICguLi5hcmdzOiBhbnlbXSkgPT4ge1xuICAgICAgbG9ncy5wdXNoKGBbV0FSTl0gJHthcmdzLmpvaW4oJyAnKX1gKTtcbiAgICAgIG9yaWdpbmFsQ29uc29sZVdhcm4oLi4uYXJncyk7XG4gICAgfTtcblxuICAgIGNvbnNvbGUuaW5mbyA9ICguLi5hcmdzOiBhbnlbXSkgPT4ge1xuICAgICAgbG9ncy5wdXNoKGBbSU5GT10gJHthcmdzLmpvaW4oJyAnKX1gKTtcbiAgICAgIG9yaWdpbmFsQ29uc29sZUluZm8oLi4uYXJncyk7XG4gICAgfTtcblxuICAgIHRyeSB7XG4gICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBoYW5kbGVyKGV2ZW50LCBjb250ZXh0KTtcblxuICAgICAgLy8gT3V0cHV0IGNhcHR1cmVkIGxvZ3NcbiAgICAgIGlmKGxvZ3MubGVuZ3RoID4gMCkge1xuICAgICAgICBjb25zb2xlLmxvZyhjaGFsay5ncmF5KCctLS0gSGFuZGxlciBDb25zb2xlIE91dHB1dCAtLS0nKSk7XG4gICAgICAgIGxvZ3MuZm9yRWFjaCgobG9nKSA9PiBjb25zb2xlLmxvZyhjaGFsay5ncmF5KGxvZykpKTtcbiAgICAgICAgY29uc29sZS5sb2coY2hhbGsuZ3JheSgnLS0tIEVuZCBIYW5kbGVyIENvbnNvbGUgT3V0cHV0IC0tLScpKTtcbiAgICAgIH1cblxuICAgICAgcmV0dXJuIHJlc3VsdDtcbiAgICB9IGZpbmFsbHkge1xuICAgICAgLy8gUmVzdG9yZSBvcmlnaW5hbCBjb25zb2xlIG1ldGhvZHNcbiAgICAgIGNvbnNvbGUubG9nID0gb3JpZ2luYWxDb25zb2xlTG9nO1xuICAgICAgY29uc29sZS5lcnJvciA9IG9yaWdpbmFsQ29uc29sZUVycm9yO1xuICAgICAgY29uc29sZS53YXJuID0gb3JpZ2luYWxDb25zb2xlV2FybjtcbiAgICAgIGNvbnNvbGUuaW5mbyA9IG9yaWdpbmFsQ29uc29sZUluZm87XG4gICAgfVxuICB9O1xufTtcblxuY29uc3QgY3JlYXRlRXhwcmVzc1NlcnZlciA9IGFzeW5jIChcbiAgY29uZmlnOiBTZXJ2ZXJsZXNzQ29uZmlnLFxuICBvdXRwdXREaXI6IHN0cmluZyxcbiAgaHR0cFBvcnQ6IG51bWJlcixcbiAgaG9zdDogc3RyaW5nLFxuICBxdWlldDogYm9vbGVhbixcbiAgZGVidWc6IGJvb2xlYW5cbikgPT4ge1xuICBjb25zdCBhcHAgPSBleHByZXNzKCk7XG5cbiAgLy8gRW5hYmxlIENPUlNcbiAgYXBwLnVzZSgocmVxLCByZXMsIG5leHQpID0+IHtcbiAgICByZXMuaGVhZGVyKCdBY2Nlc3MtQ29udHJvbC1BbGxvdy1PcmlnaW4nLCAnKicpO1xuICAgIHJlcy5oZWFkZXIoJ0FjY2Vzcy1Db250cm9sLUFsbG93LU1ldGhvZHMnLCAnR0VULCBQT1NULCBQVVQsIERFTEVURSwgUEFUQ0gsIE9QVElPTlMnKTtcbiAgICByZXMuaGVhZGVyKCdBY2Nlc3MtQ29udHJvbC1BbGxvdy1IZWFkZXJzJywgJyonKTtcbiAgICByZXMuaGVhZGVyKCdBY2Nlc3MtQ29udHJvbC1BbGxvdy1DcmVkZW50aWFscycsICd0cnVlJyk7XG5cbiAgICBpZihyZXEubWV0aG9kID09PSAnT1BUSU9OUycpIHtcbiAgICAgIHJlcy5zZW5kU3RhdHVzKDIwMCk7XG4gICAgfSBlbHNlIHtcbiAgICAgIG5leHQoKTtcbiAgICB9XG4gIH0pO1xuXG4gIC8vIFBhcnNlIEpTT04gYm9kaWVzXG4gIGFwcC51c2UoZXhwcmVzcy5qc29uKCkpO1xuXG4gIC8vIExvYWQgR3JhcGhRTCBoYW5kbGVyXG4gIGNvbnN0IGxvYWRHcmFwaFFMU2NoZW1hID0gYXN5bmMgKCkgPT4ge1xuICAgIHRyeSB7XG4gICAgICAvLyBUcnkgdG8gZmluZCBhIEdyYXBoUUwgaGFuZGxlclxuICAgICAgbGV0IGdyYXBocWxIYW5kbGVyID0gbnVsbDtcblxuICAgICAgaWYoY29uZmlnLmZ1bmN0aW9ucykge1xuICAgICAgICBmb3IoY29uc3QgW2Z1bmN0aW9uTmFtZSwgZnVuY3Rpb25Db25maWddIG9mIE9iamVjdC5lbnRyaWVzKGNvbmZpZy5mdW5jdGlvbnMpKSB7XG4gICAgICAgICAgaWYoZnVuY3Rpb25Db25maWcuZXZlbnRzKSB7XG4gICAgICAgICAgICBmb3IoY29uc3QgZXZlbnQgb2YgZnVuY3Rpb25Db25maWcuZXZlbnRzKSB7XG4gICAgICAgICAgICAgIGlmKGV2ZW50Lmh0dHAgJiYgZXZlbnQuaHR0cC5wYXRoKSB7XG4gICAgICAgICAgICAgICAgLy8gTG9vayBmb3IgR3JhcGhRTCBlbmRwb2ludHNcbiAgICAgICAgICAgICAgICBpZihldmVudC5odHRwLnBhdGggPT09ICcvcHVibGljJyB8fCBldmVudC5odHRwLnBhdGggPT09ICcvZ3JhcGhxbCcpIHtcbiAgICAgICAgICAgICAgICAgIGdyYXBocWxIYW5kbGVyID0gYXdhaXQgbG9hZEhhbmRsZXIoZnVuY3Rpb25Db25maWcuaGFuZGxlciwgb3V0cHV0RGlyKTtcbiAgICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgICBpZihncmFwaHFsSGFuZGxlcikge1xuICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIGlmKGdyYXBocWxIYW5kbGVyKSB7XG4gICAgICAgIGxvZygnRm91bmQgR3JhcGhRTCBoYW5kbGVyJywgJ2luZm8nLCBxdWlldCk7XG4gICAgICAgIHJldHVybiBncmFwaHFsSGFuZGxlcjtcbiAgICAgIH1cbiAgICAgIHJldHVybiBudWxsO1xuICAgIH0gY2F0Y2goZXJyb3IpIHtcbiAgICAgIGxvZyhgRXJyb3IgbG9hZGluZyBHcmFwaFFMIGhhbmRsZXI6ICR7ZXJyb3IubWVzc2FnZX1gLCAnZXJyb3InLCBxdWlldCk7XG4gICAgICByZXR1cm4gbnVsbDtcbiAgICB9XG4gIH07XG5cbiAgLy8gU2V0IHVwIEdyYXBoUUwgaGFuZGxlciBmb3IgR3JhcGhRTCByZXF1ZXN0c1xuICB0cnkge1xuICAgIGNvbnN0IGdyYXBocWxIYW5kbGVyID0gYXdhaXQgbG9hZEdyYXBoUUxTY2hlbWEoKTtcbiAgICBpZihncmFwaHFsSGFuZGxlcikge1xuICAgICAgLy8gRmluZCB0aGUgR3JhcGhRTCBwYXRoIGZyb20gdGhlIHNlcnZlcmxlc3MgY29uZmlnXG4gICAgICBsZXQgZ3JhcGhxbFBhdGggPSAnL2dyYXBocWwnOyAvLyBkZWZhdWx0IGZhbGxiYWNrXG5cbiAgICAgIGlmKGNvbmZpZy5mdW5jdGlvbnMpIHtcbiAgICAgICAgZm9yKGNvbnN0IFtfZnVuY3Rpb25OYW1lLCBmdW5jdGlvbkNvbmZpZ10gb2YgT2JqZWN0LmVudHJpZXMoY29uZmlnLmZ1bmN0aW9ucykpIHtcbiAgICAgICAgICBpZihmdW5jdGlvbkNvbmZpZy5ldmVudHMpIHtcbiAgICAgICAgICAgIGZvcihjb25zdCBldmVudCBvZiBmdW5jdGlvbkNvbmZpZy5ldmVudHMpIHtcbiAgICAgICAgICAgICAgaWYoZXZlbnQ/Lmh0dHA/LnBhdGgpIHtcbiAgICAgICAgICAgICAgICBncmFwaHFsUGF0aCA9IGV2ZW50Lmh0dHAucGF0aDtcbiAgICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgICBpZihncmFwaHFsUGF0aCAhPT0gJy9ncmFwaHFsJykge1xuICAgICAgICAgICAgYnJlYWs7XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIC8vIFNldCB1cCBHcmFwaFFMIGVuZHBvaW50IHdpdGggZW5oYW5jZWQgY29uc29sZS5sb2cgY2FwdHVyZVxuICAgICAgYXBwLnVzZShncmFwaHFsUGF0aCwgYXN5bmMgKHJlcSwgcmVzKSA9PiB7XG4gICAgICAgIC8vIEdyYXBoUUwgRGVidWcgTG9nZ2luZ1xuICAgICAgICBpZihkZWJ1ZyAmJiByZXEuYm9keSAmJiByZXEuYm9keS5xdWVyeSkge1xuICAgICAgICAgIGxvZygn8J+UjSBHcmFwaFFMIERlYnVnIE1vZGU6IEFuYWx5emluZyByZXF1ZXN0Li4uJywgJ2luZm8nLCBmYWxzZSk7XG4gICAgICAgICAgbG9nKGDwn5OdIEdyYXBoUUwgUXVlcnk6ICR7cmVxLmJvZHkucXVlcnl9YCwgJ2luZm8nLCBmYWxzZSk7XG4gICAgICAgICAgaWYocmVxLmJvZHkudmFyaWFibGVzKSB7XG4gICAgICAgICAgICBsb2coYPCfk4ogR3JhcGhRTCBWYXJpYWJsZXM6ICR7SlNPTi5zdHJpbmdpZnkocmVxLmJvZHkudmFyaWFibGVzLCBudWxsLCAyKX1gLCAnaW5mbycsIGZhbHNlKTtcbiAgICAgICAgICB9XG4gICAgICAgICAgaWYocmVxLmJvZHkub3BlcmF0aW9uTmFtZSkge1xuICAgICAgICAgICAgbG9nKGDwn4+377iPICBHcmFwaFFMIE9wZXJhdGlvbjogJHtyZXEuYm9keS5vcGVyYXRpb25OYW1lfWAsICdpbmZvJywgZmFsc2UpO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIC8vIEVuaGFuY2VkIGNvbnNvbGUubG9nIGNhcHR1cmVcbiAgICAgICAgY29uc3Qgb3JpZ2luYWxDb25zb2xlTG9nID0gY29uc29sZS5sb2c7XG4gICAgICAgIGNvbnN0IGxvZ3M6IHN0cmluZ1tdID0gW107XG5cbiAgICAgICAgY29uc29sZS5sb2cgPSAoLi4uYXJncykgPT4ge1xuICAgICAgICAgIGNvbnN0IGxvZ01lc3NhZ2UgPSBhcmdzLm1hcCgoYXJnKSA9PlxuICAgICAgICAgICAgKHR5cGVvZiBhcmcgPT09ICdvYmplY3QnID8gSlNPTi5zdHJpbmdpZnkoYXJnLCBudWxsLCAyKSA6IFN0cmluZyhhcmcpKVxuICAgICAgICAgICkuam9pbignICcpO1xuICAgICAgICAgIGxvZ3MucHVzaChsb2dNZXNzYWdlKTtcbiAgICAgICAgICBvcmlnaW5hbENvbnNvbGVMb2coYFtHcmFwaFFMXSAke2xvZ01lc3NhZ2V9YCk7XG4gICAgICAgIH07XG5cbiAgICAgICAgLy8gQ3JlYXRlIGNvbnRleHQgZm9yIHRoZSBoYW5kbGVyXG4gICAgICAgIGNvbnN0IGNvbnRleHQgPSB7XG4gICAgICAgICAgYXdzUmVxdWVzdElkOiAndGVzdC1yZXF1ZXN0LWlkJyxcbiAgICAgICAgICBmdW5jdGlvbk5hbWU6ICdncmFwaHFsJyxcbiAgICAgICAgICBmdW5jdGlvblZlcnNpb246ICckTEFURVNUJyxcbiAgICAgICAgICBnZXRSZW1haW5pbmdUaW1lSW5NaWxsaXM6ICgpID0+IDMwMDAwLFxuICAgICAgICAgIGludm9rZWRGdW5jdGlvbkFybjogJ2Fybjphd3M6bGFtYmRhOnVzLWVhc3QtMToxMjM0NTY3ODkwMTI6ZnVuY3Rpb246Z3JhcGhxbCcsXG4gICAgICAgICAgbG9nR3JvdXBOYW1lOiAnL2F3cy9sYW1iZGEvZ3JhcGhxbCcsXG4gICAgICAgICAgbG9nU3RyZWFtTmFtZTogJ3Rlc3QtbG9nLXN0cmVhbScsXG4gICAgICAgICAgcmVxLFxuICAgICAgICAgIHJlc1xuICAgICAgICB9O1xuXG4gICAgICAgIC8vIFdyYXAgaGFuZGxlciB3aXRoIGNvbnNvbGUgbG9nIGNhcHR1cmVcbiAgICAgICAgY29uc3Qgd3JhcHBlZEhhbmRsZXIgPSBjYXB0dXJlQ29uc29sZUxvZ3MoZ3JhcGhxbEhhbmRsZXIsIHF1aWV0KTtcblxuICAgICAgICB0cnkge1xuICAgICAgICAgIC8vIENhbGwgdGhlIGhhbmRsZXIgd2l0aCBHcmFwaFFMIHBhcmFtZXRlcnNcbiAgICAgICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCB3cmFwcGVkSGFuZGxlcih7XG4gICAgICAgICAgICBib2R5OiBKU09OLnN0cmluZ2lmeShyZXEuYm9keSksXG4gICAgICAgICAgICBoZWFkZXJzOiByZXEuaGVhZGVycyxcbiAgICAgICAgICAgIGh0dHBNZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgICAgIHBhdGg6IGdyYXBocWxQYXRoLFxuICAgICAgICAgICAgcXVlcnlTdHJpbmdQYXJhbWV0ZXJzOiB7fVxuICAgICAgICAgIH0sIGNvbnRleHQpO1xuXG4gICAgICAgICAgLy8gUmVzdG9yZSBjb25zb2xlLmxvZ1xuICAgICAgICAgIGNvbnNvbGUubG9nID0gb3JpZ2luYWxDb25zb2xlTG9nO1xuXG4gICAgICAgICAgLy8gSGFuZGxlIHRoZSByZXN1bHRcbiAgICAgICAgICBpZihyZXN1bHQgJiYgdHlwZW9mIHJlc3VsdCA9PT0gJ29iamVjdCcgJiYgcmVzdWx0LnN0YXR1c0NvZGUpIHtcbiAgICAgICAgICAgIHJlcy5zdGF0dXMocmVzdWx0LnN0YXR1c0NvZGUpO1xuICAgICAgICAgICAgaWYocmVzdWx0LmhlYWRlcnMpIHtcbiAgICAgICAgICAgICAgT2JqZWN0LmVudHJpZXMocmVzdWx0LmhlYWRlcnMpLmZvckVhY2goKFtrZXksIHZhbHVlXSkgPT4ge1xuICAgICAgICAgICAgICAgIHJlcy5zZXRIZWFkZXIoa2V5LCBTdHJpbmcodmFsdWUpKTtcbiAgICAgICAgICAgICAgfSk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICByZXMuc2VuZChyZXN1bHQuYm9keSk7XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHJlcy5qc29uKHJlc3VsdCk7XG4gICAgICAgICAgfVxuICAgICAgICB9IGNhdGNoKGVycm9yKSB7XG4gICAgICAgICAgLy8gUmVzdG9yZSBjb25zb2xlLmxvZ1xuICAgICAgICAgIGNvbnNvbGUubG9nID0gb3JpZ2luYWxDb25zb2xlTG9nO1xuICAgICAgICAgIGxvZyhgR3JhcGhRTCBoYW5kbGVyIGVycm9yOiAke2Vycm9yLm1lc3NhZ2V9YCwgJ2Vycm9yJywgZmFsc2UpO1xuICAgICAgICAgIHJlcy5zdGF0dXMoNTAwKS5qc29uKHtlcnJvcjogZXJyb3IubWVzc2FnZX0pO1xuICAgICAgICB9XG4gICAgICB9KTtcblxuICAgICAgbG9nKGBHcmFwaFFMIGVuZHBvaW50IGF2YWlsYWJsZSBhdCBodHRwOi8vJHtob3N0fToke2h0dHBQb3J0fSR7Z3JhcGhxbFBhdGh9YCwgJ2luZm8nLCBxdWlldCk7XG4gICAgfVxuICB9IGNhdGNoKGVycm9yKSB7XG4gICAgbG9nKGBFcnJvciBzZXR0aW5nIHVwIEdyYXBoUUw6ICR7ZXJyb3IubWVzc2FnZX1gLCAnZXJyb3InLCBxdWlldCk7XG4gIH1cblxuICAvLyBGYWxsYmFjayBmb3Igbm9uLUdyYXBoUUwgcm91dGVzIC0gaGFuZGxlIGFsbCByZW1haW5pbmcgcm91dGVzXG4gIGFwcC51c2UoJy8nLCBhc3luYyAocmVxLCByZXMpID0+IHtcbiAgICB0cnkge1xuICAgICAgY29uc3QgdXJsID0gcmVxLnVybCB8fCAnLyc7XG4gICAgICBjb25zdCBtZXRob2QgPSByZXEubWV0aG9kIHx8ICdHRVQnO1xuICAgICAgY29uc3QgcGF0aG5hbWUgPSByZXEucGF0aCB8fCB1cmwuc3BsaXQoJz8nKVswXTsgLy8gRXh0cmFjdCBwYXRobmFtZSB3aXRob3V0IHF1ZXJ5IHN0cmluZ1xuXG4gICAgICAvLyBBbHdheXMgbG9nIHJlcXVlc3RzIChub3QgYWZmZWN0ZWQgYnkgcXVpZXQgZmxhZyBmb3IgZGVidWdnaW5nKVxuICAgICAgY29uc29sZS5sb2coYFtTZXJ2ZXJsZXNzXSAke21ldGhvZH0gJHt1cmx9IChwYXRobmFtZTogJHtwYXRobmFtZX0pYCk7XG5cbiAgICAgIC8vIEZpbmQgbWF0Y2hpbmcgZnVuY3Rpb25cbiAgICAgIGxldCBtYXRjaGVkRnVuY3Rpb24gPSBudWxsO1xuICAgICAgY29uc3QgZnVuY3Rpb25zID0gY29uZmlnLmZ1bmN0aW9ucyB8fCB7fTtcblxuICAgICAgaWYoY29uZmlnLmZ1bmN0aW9ucykge1xuICAgICAgICBjb25zdCBmdW5jdGlvbk5hbWVzID0gT2JqZWN0LmtleXMoZnVuY3Rpb25zKTtcbiAgICAgICAgY29uc29sZS5sb2coYFtTZXJ2ZXJsZXNzXSBBdmFpbGFibGUgZnVuY3Rpb25zOiAke2Z1bmN0aW9uTmFtZXMuam9pbignLCAnKX1gKTtcbiAgICAgICAgY29uc29sZS5sb2coJ1tTZXJ2ZXJsZXNzXSBDb25maWcgZnVuY3Rpb25zOicsIEpTT04uc3RyaW5naWZ5KGZ1bmN0aW9ucywgbnVsbCwgMikpO1xuXG4gICAgICAgIGZvcihjb25zdCBbZnVuY3Rpb25OYW1lLCBmdW5jdGlvbkNvbmZpZ10gb2YgT2JqZWN0LmVudHJpZXMoZnVuY3Rpb25zKSkge1xuICAgICAgICAgIGlmKGZ1bmN0aW9uQ29uZmlnLmV2ZW50cykge1xuICAgICAgICAgICAgZm9yKGNvbnN0IGV2ZW50IG9mIGZ1bmN0aW9uQ29uZmlnLmV2ZW50cykge1xuICAgICAgICAgICAgICBpZihldmVudC5odHRwKSB7XG4gICAgICAgICAgICAgICAgY29uc3QgZXZlbnRQYXRoID0gZXZlbnQuaHR0cC5wYXRoIHx8ICcvJztcbiAgICAgICAgICAgICAgICBjb25zdCBldmVudE1ldGhvZCA9IChldmVudC5odHRwLm1ldGhvZCB8fCAnR0VUJykudG9VcHBlckNhc2UoKTtcbiAgICAgICAgICAgICAgICBjb25zdCByZXF1ZXN0TWV0aG9kID0gbWV0aG9kLnRvVXBwZXJDYXNlKCk7XG5cbiAgICAgICAgICAgICAgICBjb25zb2xlLmxvZyhgW1NlcnZlcmxlc3NdIENoZWNraW5nIGZ1bmN0aW9uICR7ZnVuY3Rpb25OYW1lfTogcGF0aD1cIiR7ZXZlbnRQYXRofVwiLCBtZXRob2Q9XCIke2V2ZW50TWV0aG9kfVwiIGFnYWluc3QgcGF0aG5hbWU9XCIke3BhdGhuYW1lfVwiLCBtZXRob2Q9XCIke3JlcXVlc3RNZXRob2R9XCJgKTtcblxuICAgICAgICAgICAgICAgIC8vIEltcHJvdmVkIHBhdGggbWF0Y2hpbmcgLSBjb21wYXJlIHBhdGhuYW1lIHdpdGhvdXQgcXVlcnkgc3RyaW5nXG4gICAgICAgICAgICAgICAgLy8gTm9ybWFsaXplIHBhdGhzIChyZW1vdmUgdHJhaWxpbmcgc2xhc2hlcyBmb3IgY29tcGFyaXNvbilcbiAgICAgICAgICAgICAgICBjb25zdCBub3JtYWxpemVkRXZlbnRQYXRoID0gZXZlbnRQYXRoLnJlcGxhY2UoL1xcLyQvLCAnJykgfHwgJy8nO1xuICAgICAgICAgICAgICAgIGNvbnN0IG5vcm1hbGl6ZWRQYXRobmFtZSA9IHBhdGhuYW1lLnJlcGxhY2UoL1xcLyQvLCAnJykgfHwgJy8nO1xuXG4gICAgICAgICAgICAgICAgaWYobm9ybWFsaXplZEV2ZW50UGF0aCA9PT0gbm9ybWFsaXplZFBhdGhuYW1lICYmIGV2ZW50TWV0aG9kID09PSByZXF1ZXN0TWV0aG9kKSB7XG4gICAgICAgICAgICAgICAgICBtYXRjaGVkRnVuY3Rpb24gPSBmdW5jdGlvbk5hbWU7XG4gICAgICAgICAgICAgICAgICBjb25zb2xlLmxvZyhgW1NlcnZlcmxlc3NdIOKckyBNYXRjaGVkIGZ1bmN0aW9uOiAke21hdGNoZWRGdW5jdGlvbn1gKTtcbiAgICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgICBpZihtYXRjaGVkRnVuY3Rpb24pIHtcbiAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfSBlbHNlIHtcbiAgICAgICAgY29uc29sZS5sb2coJ1tTZXJ2ZXJsZXNzXSBObyBmdW5jdGlvbnMgZm91bmQgaW4gY29uZmlnJyk7XG4gICAgICB9XG5cbiAgICAgIGlmKG1hdGNoZWRGdW5jdGlvbiAmJiBmdW5jdGlvbnNbbWF0Y2hlZEZ1bmN0aW9uXSkge1xuICAgICAgICAvLyBSZXNvbHZlIGhhbmRsZXIgcGF0aCByZWxhdGl2ZSB0byBvdXRwdXQgZGlyZWN0b3J5XG4gICAgICAgIGNvbnN0IGhhbmRsZXJQYXRoID0gZnVuY3Rpb25zW21hdGNoZWRGdW5jdGlvbl0uaGFuZGxlcjtcbiAgICAgICAgY29uc29sZS5sb2coYFtTZXJ2ZXJsZXNzXSBMb2FkaW5nIGhhbmRsZXI6ICR7aGFuZGxlclBhdGh9IGZyb20gb3V0cHV0RGlyOiAke291dHB1dERpcn1gKTtcbiAgICAgICAgY29uc3QgaGFuZGxlciA9IGF3YWl0IGxvYWRIYW5kbGVyKGhhbmRsZXJQYXRoLCBvdXRwdXREaXIpO1xuXG4gICAgICAgIGlmKGhhbmRsZXIpIHtcbiAgICAgICAgICBjb25zb2xlLmxvZyhgW1NlcnZlcmxlc3NdIEhhbmRsZXIgbG9hZGVkIHN1Y2Nlc3NmdWxseSwgdHlwZTogJHt0eXBlb2YgaGFuZGxlcn1gKTtcbiAgICAgICAgICBjb25zdCB3cmFwcGVkSGFuZGxlciA9IGNhcHR1cmVDb25zb2xlTG9ncyhoYW5kbGVyLCBxdWlldCk7XG5cbiAgICAgICAgICBjb25zdCBldmVudCA9IHtcbiAgICAgICAgICAgIGJvZHk6IHJlcS5ib2R5LFxuICAgICAgICAgICAgaGVhZGVyczogcmVxLmhlYWRlcnMsXG4gICAgICAgICAgICBodHRwTWV0aG9kOiBtZXRob2QsXG4gICAgICAgICAgICBwYXRoOiB1cmwsXG4gICAgICAgICAgICBxdWVyeVN0cmluZ1BhcmFtZXRlcnM6IHJlcS5xdWVyeVxuICAgICAgICAgIH07XG5cbiAgICAgICAgICBjb25zdCBjb250ZXh0ID0ge1xuICAgICAgICAgICAgYXdzUmVxdWVzdElkOiAndGVzdC1yZXF1ZXN0LWlkJyxcbiAgICAgICAgICAgIGZ1bmN0aW9uTmFtZTogbWF0Y2hlZEZ1bmN0aW9uLFxuICAgICAgICAgICAgZnVuY3Rpb25WZXJzaW9uOiAnJExBVEVTVCcsXG4gICAgICAgICAgICBnZXRSZW1haW5pbmdUaW1lSW5NaWxsaXM6ICgpID0+IDMwMDAwLFxuICAgICAgICAgICAgaW52b2tlZEZ1bmN0aW9uQXJuOiBgYXJuOmF3czpsYW1iZGE6dXMtZWFzdC0xOjEyMzQ1Njc4OTAxMjpmdW5jdGlvbjoke21hdGNoZWRGdW5jdGlvbn1gLFxuICAgICAgICAgICAgbG9nR3JvdXBOYW1lOiBgL2F3cy9sYW1iZGEvJHttYXRjaGVkRnVuY3Rpb259YCxcbiAgICAgICAgICAgIGxvZ1N0cmVhbU5hbWU6ICd0ZXN0LWxvZy1zdHJlYW0nLFxuICAgICAgICAgICAgbWVtb3J5TGltaXRJbk1COiAnMTI4J1xuICAgICAgICAgIH07XG5cbiAgICAgICAgICB0cnkge1xuICAgICAgICAgICAgY29uc29sZS5sb2coJ1tTZXJ2ZXJsZXNzXSBDYWxsaW5nIGhhbmRsZXIgd2l0aCBldmVudDonLCBKU09OLnN0cmluZ2lmeShldmVudCwgbnVsbCwgMikpO1xuICAgICAgICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgd3JhcHBlZEhhbmRsZXIoZXZlbnQsIGNvbnRleHQpO1xuICAgICAgICAgICAgY29uc29sZS5sb2coJ1tTZXJ2ZXJsZXNzXSBIYW5kbGVyIHJldHVybmVkOicsIEpTT04uc3RyaW5naWZ5KHJlc3VsdCwgbnVsbCwgMikpO1xuXG4gICAgICAgICAgICBpZihyZXN1bHQgJiYgdHlwZW9mIHJlc3VsdCA9PT0gJ29iamVjdCcgJiYgcmVzdWx0LnN0YXR1c0NvZGUpIHtcbiAgICAgICAgICAgICAgcmVzLnN0YXR1cyhyZXN1bHQuc3RhdHVzQ29kZSk7XG4gICAgICAgICAgICAgIGlmKHJlc3VsdC5oZWFkZXJzKSB7XG4gICAgICAgICAgICAgICAgT2JqZWN0LmVudHJpZXMocmVzdWx0LmhlYWRlcnMpLmZvckVhY2goKFtrZXksIHZhbHVlXSkgPT4ge1xuICAgICAgICAgICAgICAgICAgcmVzLnNldEhlYWRlcihrZXksIFN0cmluZyh2YWx1ZSkpO1xuICAgICAgICAgICAgICAgIH0pO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIHJlcy5zZW5kKHJlc3VsdC5ib2R5KTtcbiAgICAgICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgICAgIHJlcy5qc29uKHJlc3VsdCk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfSBjYXRjaChlcnJvcjogYW55KSB7XG4gICAgICAgICAgICBjb25zb2xlLmVycm9yKCdbU2VydmVybGVzc10gSGFuZGxlciBlcnJvcjonLCBlcnJvci5tZXNzYWdlKTtcbiAgICAgICAgICAgIGNvbnNvbGUuZXJyb3IoJ1tTZXJ2ZXJsZXNzXSBIYW5kbGVyIGVycm9yIHN0YWNrOicsIGVycm9yLnN0YWNrKTtcbiAgICAgICAgICAgIGxvZyhgSGFuZGxlciBlcnJvcjogJHtlcnJvci5tZXNzYWdlfWAsICdlcnJvcicsIGZhbHNlKTtcbiAgICAgICAgICAgIHJlcy5zdGF0dXMoNTAwKS5qc29uKHtlcnJvcjogZXJyb3IubWVzc2FnZX0pO1xuICAgICAgICAgIH1cbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBjb25zb2xlLmVycm9yKGBbU2VydmVybGVzc10gSGFuZGxlciBub3QgZm91bmQgZm9yIGZ1bmN0aW9uOiAke21hdGNoZWRGdW5jdGlvbn1gKTtcbiAgICAgICAgICBjb25zb2xlLmVycm9yKGBbU2VydmVybGVzc10gSGFuZGxlciBwYXRoOiAke2hhbmRsZXJQYXRofSwgT3V0cHV0IGRpcjogJHtvdXRwdXREaXJ9YCk7XG4gICAgICAgICAgcmVzLnN0YXR1cyg0MDQpLmpzb24oe2Vycm9yOiAnSGFuZGxlciBub3QgZm91bmQnfSk7XG4gICAgICAgIH1cbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoYFtTZXJ2ZXJsZXNzXSBGdW5jdGlvbiBub3QgZm91bmQgZm9yIHBhdGhuYW1lOiAke3BhdGhuYW1lfSwgbWV0aG9kOiAke21ldGhvZH1gKTtcbiAgICAgICAgY29uc29sZS5lcnJvcihgW1NlcnZlcmxlc3NdIEF2YWlsYWJsZSBmdW5jdGlvbnM6ICR7Y29uZmlnLmZ1bmN0aW9ucyA/IE9iamVjdC5rZXlzKGZ1bmN0aW9ucykuam9pbignLCAnKSA6ICdub25lJ31gKTtcbiAgICAgICAgcmVzLnN0YXR1cyg0MDQpLmpzb24oe2Vycm9yOiAnRnVuY3Rpb24gbm90IGZvdW5kJ30pO1xuICAgICAgfVxuICAgIH0gY2F0Y2goZXJyb3IpIHtcbiAgICAgIGxvZyhgUm91dGUgaGFuZGxpbmcgZXJyb3I6ICR7ZXJyb3IubWVzc2FnZX1gLCAnZXJyb3InLCBmYWxzZSk7XG4gICAgICByZXMuc3RhdHVzKDUwMCkuanNvbih7ZXJyb3I6IGVycm9yLm1lc3NhZ2V9KTtcbiAgICB9XG4gIH0pO1xuXG4gIHJldHVybiBhcHA7XG59O1xuXG5jb25zdCBjcmVhdGVXZWJTb2NrZXRTZXJ2ZXIgPSAoXG4gIGNvbmZpZzogU2VydmVybGVzc0NvbmZpZyxcbiAgb3V0cHV0RGlyOiBzdHJpbmcsXG4gIHdzUG9ydDogbnVtYmVyLFxuICBxdWlldDogYm9vbGVhbixcbiAgZGVidWc6IGJvb2xlYW5cbikgPT4ge1xuICBjb25zdCB3c3MgPSBuZXcgV2ViU29ja2V0U2VydmVyKHtwb3J0OiB3c1BvcnR9KTtcblxuICB3c3Mub24oJ2Nvbm5lY3Rpb24nLCBhc3luYyAod3M6IFdlYlNvY2tldENsaWVudExpa2UsIHJlcTogYW55KSA9PiB7XG4gICAgbG9nKGBXZWJTb2NrZXQgY29ubmVjdGlvbiBlc3RhYmxpc2hlZDogJHtyZXEudXJsfWAsICdpbmZvJywgZmFsc2UpO1xuXG4gICAgd3Mub24oJ21lc3NhZ2UnLCBhc3luYyAobWVzc2FnZTogYW55KSA9PiB7XG4gICAgICB0cnkge1xuICAgICAgICBjb25zdCBkYXRhID0gSlNPTi5wYXJzZShtZXNzYWdlLnRvU3RyaW5nKCkpO1xuXG4gICAgICAgIC8vIEZpbmQgbWF0Y2hpbmcgV2ViU29ja2V0IGZ1bmN0aW9uXG4gICAgICAgIGxldCBtYXRjaGVkRnVuY3Rpb24gPSBudWxsO1xuICAgICAgICBjb25zdCBmdW5jdGlvbnMgPSBjb25maWcuZnVuY3Rpb25zIHx8IHt9O1xuXG4gICAgICAgIGlmKGNvbmZpZy5mdW5jdGlvbnMpIHtcbiAgICAgICAgICBsZXQgZGVmYXVsdEZ1bmN0aW9uOiBzdHJpbmcgfCBudWxsID0gbnVsbDtcblxuICAgICAgICAgIGZvcihjb25zdCBbZnVuY3Rpb25OYW1lLCBmdW5jdGlvbkNvbmZpZ10gb2YgT2JqZWN0LmVudHJpZXMoZnVuY3Rpb25zKSkge1xuICAgICAgICAgICAgaWYoZnVuY3Rpb25Db25maWcuZXZlbnRzKSB7XG4gICAgICAgICAgICAgIGZvcihjb25zdCBldmVudCBvZiBmdW5jdGlvbkNvbmZpZy5ldmVudHMpIHtcbiAgICAgICAgICAgICAgICBpZihldmVudC53ZWJzb2NrZXQpIHtcbiAgICAgICAgICAgICAgICAgIGNvbnN0IHJvdXRlID0gZXZlbnQud2Vic29ja2V0LnJvdXRlIHx8ICckY29ubmVjdCc7XG5cbiAgICAgICAgICAgICAgICAgIGlmKHJvdXRlID09PSBkYXRhLmFjdGlvbikge1xuICAgICAgICAgICAgICAgICAgICBtYXRjaGVkRnVuY3Rpb24gPSBmdW5jdGlvbk5hbWU7XG4gICAgICAgICAgICAgICAgICAgIGJyZWFrO1xuICAgICAgICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICAgICAgICBpZihyb3V0ZSA9PT0gJyRkZWZhdWx0JyAmJiAhZGVmYXVsdEZ1bmN0aW9uKSB7XG4gICAgICAgICAgICAgICAgICAgIGRlZmF1bHRGdW5jdGlvbiA9IGZ1bmN0aW9uTmFtZTtcbiAgICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICB9XG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGlmKG1hdGNoZWRGdW5jdGlvbikge1xuICAgICAgICAgICAgICBicmVhaztcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG5cbiAgICAgICAgICBpZighbWF0Y2hlZEZ1bmN0aW9uKSB7XG4gICAgICAgICAgICBtYXRjaGVkRnVuY3Rpb24gPSBkZWZhdWx0RnVuY3Rpb247XG4gICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgaWYobWF0Y2hlZEZ1bmN0aW9uICYmIGZ1bmN0aW9uc1ttYXRjaGVkRnVuY3Rpb25dKSB7XG4gICAgICAgICAgY29uc3QgaGFuZGxlciA9IGF3YWl0IGxvYWRIYW5kbGVyKGZ1bmN0aW9uc1ttYXRjaGVkRnVuY3Rpb25dLmhhbmRsZXIsIG91dHB1dERpcik7XG5cbiAgICAgICAgICBpZihoYW5kbGVyKSB7XG4gICAgICAgICAgICAvLyBXcmFwIGhhbmRsZXIgd2l0aCBjb25zb2xlIGxvZyBjYXB0dXJlXG4gICAgICAgICAgICBjb25zdCB3cmFwcGVkSGFuZGxlciA9IGNhcHR1cmVDb25zb2xlTG9ncyhoYW5kbGVyLCBxdWlldCk7XG4gICAgICAgICAgICBjb25zdCBldmVudCA9IHtcbiAgICAgICAgICAgICAgYm9keTogZGF0YS5ib2R5IHx8IG51bGwsXG4gICAgICAgICAgICAgIHJlcXVlc3RDb250ZXh0OiB7XG4gICAgICAgICAgICAgICAgYXBpR2F0ZXdheToge1xuICAgICAgICAgICAgICAgICAgZW5kcG9pbnQ6IGB3czovL2xvY2FsaG9zdDoke3dzUG9ydH1gXG4gICAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICAgICBjb25uZWN0aW9uSWQ6ICd0ZXN0LWNvbm5lY3Rpb24taWQnLFxuICAgICAgICAgICAgICAgIHJvdXRlS2V5OiBkYXRhLmFjdGlvbiB8fCAnJGRlZmF1bHQnXG4gICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH07XG5cbiAgICAgICAgICAgIGNvbnN0IGNvbnRleHQgPSB7XG4gICAgICAgICAgICAgIGF3c1JlcXVlc3RJZDogJ3Rlc3QtcmVxdWVzdC1pZCcsXG4gICAgICAgICAgICAgIGZ1bmN0aW9uTmFtZTogbWF0Y2hlZEZ1bmN0aW9uLFxuICAgICAgICAgICAgICBmdW5jdGlvblZlcnNpb246ICckTEFURVNUJyxcbiAgICAgICAgICAgICAgZ2V0UmVtYWluaW5nVGltZUluTWlsbGlzOiAoKSA9PiAzMDAwMCxcbiAgICAgICAgICAgICAgaW52b2tlZEZ1bmN0aW9uQXJuOiBgYXJuOmF3czpsYW1iZGE6dXMtZWFzdC0xOjEyMzQ1Njc4OTAxMjpmdW5jdGlvbjoke21hdGNoZWRGdW5jdGlvbn1gLFxuICAgICAgICAgICAgICBsb2dHcm91cE5hbWU6IGAvYXdzL2xhbWJkYS8ke21hdGNoZWRGdW5jdGlvbn1gLFxuICAgICAgICAgICAgICBsb2dTdHJlYW1OYW1lOiAndGVzdC1sb2ctc3RyZWFtJyxcbiAgICAgICAgICAgICAgbWVtb3J5TGltaXRJbk1COiAnMTI4J1xuICAgICAgICAgICAgfTtcblxuICAgICAgICAgICAgY29uc3QgcmVzdWx0ID0gYXdhaXQgd3JhcHBlZEhhbmRsZXIoZXZlbnQsIGNvbnRleHQpO1xuXG4gICAgICAgICAgICAvLyBIYW5kbGUgTGFtYmRhIHJlc3BvbnNlIGZvcm1hdCBmb3IgV2ViU29ja2V0XG4gICAgICAgICAgICBpZihyZXN1bHQgJiYgdHlwZW9mIHJlc3VsdCA9PT0gJ29iamVjdCcgJiYgcmVzdWx0LnN0YXR1c0NvZGUpIHtcbiAgICAgICAgICAgICAgLy8gVGhpcyBpcyBhIExhbWJkYSByZXNwb25zZSBvYmplY3QsIGV4dHJhY3QgdGhlIGJvZHlcbiAgICAgICAgICAgICAgY29uc3QgYm9keSA9IHJlc3VsdC5ib2R5IHx8ICcnO1xuICAgICAgICAgICAgICB3cy5zZW5kKGJvZHkpO1xuICAgICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgICAgLy8gVGhpcyBpcyBhIGRpcmVjdCByZXNwb25zZSwgc3RyaW5naWZ5IGl0XG4gICAgICAgICAgICAgIHdzLnNlbmQoSlNPTi5zdHJpbmdpZnkocmVzdWx0KSk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHdzLnNlbmQoSlNPTi5zdHJpbmdpZnkoe2Vycm9yOiAnSGFuZGxlciBub3QgZm91bmQnfSkpO1xuICAgICAgICAgIH1cbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICB3cy5zZW5kKEpTT04uc3RyaW5naWZ5KHtlcnJvcjogJ1dlYlNvY2tldCBmdW5jdGlvbiBub3QgZm91bmQnfSkpO1xuICAgICAgICB9XG4gICAgICB9IGNhdGNoKGVycm9yKSB7XG4gICAgICAgIGxvZyhgV2ViU29ja2V0IGVycm9yOiAke2Vycm9yLm1lc3NhZ2V9YCwgJ2Vycm9yJywgZmFsc2UpO1xuICAgICAgICB3cy5zZW5kKEpTT04uc3RyaW5naWZ5KHtlcnJvcjogZXJyb3IubWVzc2FnZX0pKTtcbiAgICAgIH1cbiAgICB9KTtcblxuICAgIHdzLm9uKCdjbG9zZScsICgpID0+IHtcbiAgICAgIGxvZygnV2ViU29ja2V0IGNvbm5lY3Rpb24gY2xvc2VkJywgJ2luZm8nLCBmYWxzZSk7XG4gICAgfSk7XG4gIH0pO1xuXG4gIHJldHVybiB3c3M7XG59O1xuXG5jb25zdCBsb2FkRW52RmlsZSA9IChlbnZQYXRoOiBzdHJpbmcpOiBSZWNvcmQ8c3RyaW5nLCBzdHJpbmc+ID0+IHtcbiAgY29uc3QgZW52VmFyczogUmVjb3JkPHN0cmluZywgc3RyaW5nPiA9IHt9O1xuXG4gIGlmKCFleGlzdHNTeW5jKGVudlBhdGgpKSB7XG4gICAgcmV0dXJuIGVudlZhcnM7XG4gIH1cblxuICB0cnkge1xuICAgIGNvbnN0IGVudkNvbnRlbnQgPSByZWFkRmlsZVN5bmMoZW52UGF0aCwgJ3V0ZjgnKTtcbiAgICBjb25zdCBsaW5lcyA9IGVudkNvbnRlbnQuc3BsaXQoJ1xcbicpO1xuXG4gICAgZm9yKGNvbnN0IGxpbmUgb2YgbGluZXMpIHtcbiAgICAgIGNvbnN0IHRyaW1tZWRMaW5lID0gbGluZS50cmltKCk7XG5cbiAgICAgIC8vIFNraXAgZW1wdHkgbGluZXMgYW5kIGNvbW1lbnRzXG4gICAgICBpZighdHJpbW1lZExpbmUgfHwgdHJpbW1lZExpbmUuc3RhcnRzV2l0aCgnIycpKSB7XG4gICAgICAgIGNvbnRpbnVlO1xuICAgICAgfVxuXG4gICAgICAvLyBQYXJzZSBLRVk9dmFsdWUgZm9ybWF0XG4gICAgICBjb25zdCBlcXVhbEluZGV4ID0gdHJpbW1lZExpbmUuaW5kZXhPZignPScpO1xuICAgICAgaWYoZXF1YWxJbmRleCA+IDApIHtcbiAgICAgICAgY29uc3Qga2V5ID0gdHJpbW1lZExpbmUuc3Vic3RyaW5nKDAsIGVxdWFsSW5kZXgpLnRyaW0oKTtcbiAgICAgICAgY29uc3QgdmFsdWUgPSB0cmltbWVkTGluZS5zdWJzdHJpbmcoZXF1YWxJbmRleCArIDEpLnRyaW0oKTtcblxuICAgICAgICAvLyBSZW1vdmUgcXVvdGVzIGlmIHByZXNlbnRcbiAgICAgICAgY29uc3QgY2xlYW5WYWx1ZSA9IHZhbHVlLnJlcGxhY2UoL15bXCInXXxbXCInXSQvZywgJycpO1xuXG4gICAgICAgIGlmKGtleSkge1xuICAgICAgICAgIGVudlZhcnNba2V5XSA9IGNsZWFuVmFsdWU7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG4gIH0gY2F0Y2goZXJyb3IpIHtcbiAgICBsb2coYFdhcm5pbmc6IENvdWxkIG5vdCBsb2FkIC5lbnYgZmlsZSBhdCAke2VudlBhdGh9OiAke2Vycm9yLm1lc3NhZ2V9YCwgJ3dhcm4nLCBmYWxzZSk7XG4gIH1cblxuICByZXR1cm4gZW52VmFycztcbn07XG5cbmV4cG9ydCBjb25zdCBzZXJ2ZXJsZXNzID0gYXN5bmMgKFxuICBjbWQ6IFNlcnZlcmxlc3NPcHRpb25zLFxuICBjYWxsYmFjazogU2VydmVybGVzc0NhbGxiYWNrID0gKCkgPT4gKHt9KVxuKTogUHJvbWlzZTxudW1iZXI+ID0+IHtcbiAgY29uc3Qge1xuICAgIGNsaU5hbWUgPSAnTGV4JyxcbiAgICBjb25maWcsXG4gICAgZGVidWcgPSBmYWxzZSxcbiAgICBob3N0OiBjbGlIb3N0LFxuICAgIGh0dHBQb3J0OiBjbGlIdHRwUG9ydCxcbiAgICBodHRwc1BvcnQ6IGNsaUh0dHBzUG9ydCxcbiAgICBxdWlldCA9IGZhbHNlLFxuICAgIHJlbW92ZSA9IGZhbHNlLFxuICAgIHRlc3QgPSBmYWxzZSxcbiAgICB1c2VQdWJsaWNJcCxcbiAgICB2YXJpYWJsZXMsXG4gICAgd3NQb3J0OiBjbGlXc1BvcnRcbiAgfSA9IGNtZDtcblxuICBjb25zdCBzcGlubmVyID0gY3JlYXRlU3Bpbm5lcihxdWlldCk7XG5cbiAgbG9nKGAke2NsaU5hbWV9IHN0YXJ0aW5nIHNlcnZlcmxlc3MgZGV2ZWxvcG1lbnQgc2VydmVyLi4uYCwgJ2luZm8nLCBxdWlldCk7XG5cbiAgYXdhaXQgTGV4Q29uZmlnLnBhcnNlQ29uZmlnKGNtZCk7XG5cbiAgY29uc3Qge291dHB1dEZ1bGxQYXRofSA9IExleENvbmZpZy5jb25maWc7XG5cbiAgLy8gTG9hZCBlbnZpcm9ubWVudCB2YXJpYWJsZXMgZnJvbSAuZW52IGZpbGVzXG4gIGNvbnN0IGVudlBhdGhzID0gW1xuICAgIHBhdGhSZXNvbHZlKHByb2Nlc3MuY3dkKCksICcuZW52JyksXG4gICAgcGF0aFJlc29sdmUocHJvY2Vzcy5jd2QoKSwgJy5lbnYubG9jYWwnKSxcbiAgICBwYXRoUmVzb2x2ZShwcm9jZXNzLmN3ZCgpLCAnLmVudi5kZXZlbG9wbWVudCcpXG4gIF07XG5cbiAgbGV0IGVudlZhcnM6IFJlY29yZDxzdHJpbmcsIHN0cmluZz4gPSB7fTtcblxuICAvLyBMb2FkIGZyb20gLmVudiBmaWxlcyBpbiBvcmRlciAobGF0ZXIgZmlsZXMgb3ZlcnJpZGUgZWFybGllciBvbmVzKVxuICBmb3IoY29uc3QgZW52UGF0aCBvZiBlbnZQYXRocykge1xuICAgIGNvbnN0IGZpbGVFbnZWYXJzID0gbG9hZEVudkZpbGUoZW52UGF0aCk7XG4gICAgaWYoT2JqZWN0LmtleXMoZmlsZUVudlZhcnMpLmxlbmd0aCA+IDApIHtcbiAgICAgIGxvZyhgTG9hZGVkIGVudmlyb25tZW50IHZhcmlhYmxlcyBmcm9tOiAke2VudlBhdGh9YCwgJ2luZm8nLCBxdWlldCk7XG4gICAgfVxuICAgIGVudlZhcnMgPSB7Li4uZW52VmFycywgLi4uZmlsZUVudlZhcnN9O1xuICB9XG5cbiAgLy8gU3RhcnQgd2l0aCBkZWZhdWx0IE5PREVfRU5WIGFuZCBsb2FkZWQgLmVudiB2YXJpYWJsZXNcbiAgbGV0IHZhcmlhYmxlc09iajogb2JqZWN0ID0ge05PREVfRU5WOiAnZGV2ZWxvcG1lbnQnLCAuLi5lbnZWYXJzfTtcblxuICAvLyBPdmVycmlkZSB3aXRoIGNvbW1hbmQgbGluZSB2YXJpYWJsZXMgaWYgcHJvdmlkZWRcbiAgaWYodmFyaWFibGVzKSB7XG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IGNsaVZhcnMgPSBKU09OLnBhcnNlKHZhcmlhYmxlcyk7XG4gICAgICB2YXJpYWJsZXNPYmogPSB7Li4udmFyaWFibGVzT2JqLCAuLi5jbGlWYXJzfTtcbiAgICB9IGNhdGNoKF9lcnJvcikge1xuICAgICAgbG9nKGBcXG4ke2NsaU5hbWV9IEVycm9yOiBFbnZpcm9ubWVudCB2YXJpYWJsZXMgb3B0aW9uIGlzIG5vdCBhIHZhbGlkIEpTT04gb2JqZWN0LmAsICdlcnJvcicsIHF1aWV0KTtcbiAgICAgIGNhbGxiYWNrKDEpO1xuICAgICAgcmV0dXJuIDE7XG4gICAgfVxuICB9XG5cbiAgcHJvY2Vzcy5lbnYgPSB7Li4ucHJvY2Vzcy5lbnYsIC4uLnZhcmlhYmxlc09ian07XG5cbiAgLy8gSWYgaW4gdGVzdCBtb2RlLCBleGl0IGVhcmx5IGFmdGVyIGxvYWRpbmcgZW52aXJvbm1lbnQgdmFyaWFibGVzXG4gIGlmKHRlc3QpIHtcbiAgICBsb2coJ1Rlc3QgbW9kZTogRW52aXJvbm1lbnQgdmFyaWFibGVzIGxvYWRlZCwgZXhpdGluZycsICdpbmZvJywgcXVpZXQpO1xuICAgIGNhbGxiYWNrKDApO1xuICAgIHJldHVybiAwO1xuICB9XG5cbiAgaWYocmVtb3ZlKSB7XG4gICAgc3Bpbm5lci5zdGFydCgnQ2xlYW5pbmcgb3V0cHV0IGRpcmVjdG9yeS4uLicpO1xuICAgIGF3YWl0IHJlbW92ZUZpbGVzKG91dHB1dEZ1bGxQYXRoIHx8ICcnKTtcbiAgICBzcGlubmVyLnN1Y2NlZWQoJ1N1Y2Nlc3NmdWxseSBjbGVhbmVkIG91dHB1dCBkaXJlY3RvcnkhJyk7XG4gIH1cblxuICAvLyBMb2FkIHNlcnZlcmxlc3MgY29uZmlndXJhdGlvblxuICBsZXQgc2VydmVybGVzc0NvbmZpZzogU2VydmVybGVzc0NvbmZpZyA9IHt9O1xuXG4gIHRyeSB7XG4gICAgLy8gVXNlIGdldFBhY2thZ2VEaXIgdG8gaGFuZGxlIG5wbSB3b3Jrc3BhY2VzIGNvcnJlY3RseVxuICAgIGNvbnN0IHBhY2thZ2VEaXIgPSBnZXRQYWNrYWdlRGlyKCk7XG5cbiAgICAvLyBUcnkgbXVsdGlwbGUgY29uZmlnIGZpbGUgZm9ybWF0c1xuICAgIGNvbnN0IGNvbmZpZ0Zvcm1hdHMgPSBjb25maWcgPyBbY29uZmlnXSA6IFtcbiAgICAgIHBhdGhSZXNvbHZlKHBhY2thZ2VEaXIsICdsZXguY29uZmlnLm1qcycpLFxuICAgICAgcGF0aFJlc29sdmUocGFja2FnZURpciwgJ2xleC5jb25maWcuanMnKSxcbiAgICAgIHBhdGhSZXNvbHZlKHBhY2thZ2VEaXIsICdsZXguY29uZmlnLmNqcycpLFxuICAgICAgcGF0aFJlc29sdmUocGFja2FnZURpciwgJ2xleC5jb25maWcudHMnKVxuICAgIF07XG5cbiAgICBsZXQgY29uZmlnUGF0aDogc3RyaW5nIHwgbnVsbCA9IG51bGw7XG4gICAgZm9yKGNvbnN0IGNhbmRpZGF0ZVBhdGggb2YgY29uZmlnRm9ybWF0cykge1xuICAgICAgaWYoZXhpc3RzU3luYyhjYW5kaWRhdGVQYXRoKSkge1xuICAgICAgICBjb25maWdQYXRoID0gY2FuZGlkYXRlUGF0aDtcbiAgICAgICAgYnJlYWs7XG4gICAgICB9XG4gICAgfVxuXG4gICAgaWYoY29uZmlnUGF0aCkge1xuICAgICAgbG9nKGBMb2FkaW5nIHNlcnZlcmxlc3MgY29uZmlnIGZyb206ICR7Y29uZmlnUGF0aH1gLCAnaW5mbycsIHF1aWV0KTtcbiAgICAgIGNvbnN0IGNvbmZpZ01vZHVsZSA9IGF3YWl0IGltcG9ydChjb25maWdQYXRoKTtcbiAgICAgIHNlcnZlcmxlc3NDb25maWcgPSBjb25maWdNb2R1bGUuZGVmYXVsdD8uc2VydmVybGVzcyB8fCBjb25maWdNb2R1bGUuc2VydmVybGVzcyB8fCB7fTtcbiAgICAgIGxvZygnU2VydmVybGVzcyBjb25maWcgbG9hZGVkIHN1Y2Nlc3NmdWxseScsICdpbmZvJywgcXVpZXQpO1xuICAgICAgY29uc3QgZnVuY3Rpb25OYW1lcyA9IE9iamVjdC5rZXlzKHNlcnZlcmxlc3NDb25maWcuZnVuY3Rpb25zIHx8IHt9KTtcbiAgICAgIGxvZyhgTG9hZGVkIGZ1bmN0aW9uczogJHtmdW5jdGlvbk5hbWVzLmxlbmd0aCA+IDAgPyBmdW5jdGlvbk5hbWVzLmpvaW4oJywgJykgOiAnbm9uZSd9YCwgJ2luZm8nLCBxdWlldCk7XG5cbiAgICAgIC8vIERlYnVnOiBQcmludCBmdWxsIGNvbmZpZyBpZiBkZWJ1ZyBtb2RlXG4gICAgICBpZihkZWJ1Zykge1xuICAgICAgICBsb2coYEZ1bGwgc2VydmVybGVzcyBjb25maWc6ICR7SlNPTi5zdHJpbmdpZnkoc2VydmVybGVzc0NvbmZpZywgbnVsbCwgMil9YCwgJ2luZm8nLCBmYWxzZSk7XG4gICAgICB9XG4gICAgfSBlbHNlIHtcbiAgICAgIGxvZyhgTm8gc2VydmVybGVzcyBjb25maWcgZm91bmQuIFRyaWVkOiAke2NvbmZpZ0Zvcm1hdHMuam9pbignLCAnKX1gLCAnd2FybicsIHF1aWV0KTtcbiAgICB9XG4gIH0gY2F0Y2goZXJyb3IpIHtcbiAgICBsb2coYEVycm9yIGxvYWRpbmcgc2VydmVybGVzcyBjb25maWc6ICR7ZXJyb3IubWVzc2FnZX1gLCAnZXJyb3InLCBxdWlldCk7XG4gICAgaWYoZGVidWcpIHtcbiAgICAgIGxvZyhgQ29uZmlnIGVycm9yIHN0YWNrOiAke2Vycm9yLnN0YWNrfWAsICdlcnJvcicsIGZhbHNlKTtcbiAgICB9XG4gICAgLy8gRG9uJ3QgZXhpdCwgY29udGludWUgd2l0aCBlbXB0eSBjb25maWdcbiAgfVxuXG4gIC8vIE1lcmdlIGNvbmZpZyB3aXRoIGNvbW1hbmQgbGluZSBvcHRpb25zXG4gIC8vIERldGVybWluZSBlZmZlY3RpdmUgaG9zdC9wb3J0cyB3aXRoIGNvcnJlY3QgcHJlY2VkZW5jZTogQ0xJID4gY29uZmlnID4gZGVmYXVsdHNcbiAgY29uc3QgY29uZmlnT2ZmbGluZSA9IHNlcnZlcmxlc3NDb25maWcuY3VzdG9tPy5bJ3NlcnZlcmxlc3Mtb2ZmbGluZSddIHx8IHt9O1xuICBjb25zdCBlZmZlY3RpdmVIb3N0ID0gKGNsaUhvc3QgPz8gY29uZmlnT2ZmbGluZS5ob3N0ID8/ICdsb2NhbGhvc3QnKSBhcyBzdHJpbmc7XG4gIGNvbnN0IHRvTnVtYmVyID0gKHY6IGFueSwgZmFsbGJhY2s6IG51bWJlcik6IG51bWJlciA9PiB7XG4gICAgaWYodiA9PT0gdW5kZWZpbmVkIHx8IHYgPT09IG51bGwgfHwgdiA9PT0gJycpIHtcbiAgICAgIHJldHVybiBmYWxsYmFjaztcbiAgICB9XG5cbiAgICBjb25zdCBuID0gdHlwZW9mIHYgPT09ICdudW1iZXInID8gdiA6IHBhcnNlSW50KFN0cmluZyh2KSk7XG4gICAgcmV0dXJuIE51bWJlci5pc0Zpbml0ZShuKSA/IG4gOiBmYWxsYmFjaztcbiAgfTtcbiAgY29uc3QgZWZmZWN0aXZlSHR0cFBvcnQgPSB0b051bWJlcihjbGlIdHRwUG9ydCA/PyBjb25maWdPZmZsaW5lLmh0dHBQb3J0LCAzMTAwKTtcbiAgY29uc3QgZWZmZWN0aXZlSHR0cHNQb3J0ID0gdG9OdW1iZXIoY2xpSHR0cHNQb3J0ID8/IGNvbmZpZ09mZmxpbmUuaHR0cHNQb3J0LCAzMTAxKTtcbiAgY29uc3QgZWZmZWN0aXZlV3NQb3J0ID0gdG9OdW1iZXIoY2xpV3NQb3J0ID8/IGNvbmZpZ09mZmxpbmUud3NQb3J0LCAzMTAyKTtcblxuICBjb25zdCBmaW5hbENvbmZpZzogU2VydmVybGVzc0NvbmZpZyA9IHtcbiAgICAuLi5zZXJ2ZXJsZXNzQ29uZmlnLFxuICAgIGN1c3RvbToge1xuICAgICAgJ3NlcnZlcmxlc3Mtb2ZmbGluZSc6IHtcbiAgICAgICAgY29yczogc2VydmVybGVzc0NvbmZpZy5jdXN0b20/Llsnc2VydmVybGVzcy1vZmZsaW5lJ10/LmNvcnMgIT09IGZhbHNlLFxuICAgICAgICBob3N0OiBlZmZlY3RpdmVIb3N0LFxuICAgICAgICBodHRwUG9ydDogZWZmZWN0aXZlSHR0cFBvcnQsXG4gICAgICAgIGh0dHBzUG9ydDogZWZmZWN0aXZlSHR0cHNQb3J0LFxuICAgICAgICB3c1BvcnQ6IGVmZmVjdGl2ZVdzUG9ydFxuICAgICAgfVxuICAgIH1cbiAgfTtcblxuICBjb25zdCBvdXRwdXREaXIgPSBvdXRwdXRGdWxsUGF0aCB8fCAnbGliJztcbiAgbG9nKGBVc2luZyBvdXRwdXQgZGlyZWN0b3J5OiAke291dHB1dERpcn1gLCAnaW5mbycsIHF1aWV0KTtcblxuICB0cnkge1xuICAgIHNwaW5uZXIuc3RhcnQoJ1N0YXJ0aW5nIHNlcnZlcmxlc3MgZGV2ZWxvcG1lbnQgc2VydmVyLi4uJyk7XG5cbiAgICBjb25zdCBodHRwUG9ydCA9IGZpbmFsQ29uZmlnLmN1c3RvbSFbJ3NlcnZlcmxlc3Mtb2ZmbGluZSddIS5odHRwUG9ydCE7XG4gICAgY29uc3Qgd3NQb3J0ID0gZmluYWxDb25maWcuY3VzdG9tIVsnc2VydmVybGVzcy1vZmZsaW5lJ10hLndzUG9ydCE7XG4gICAgY29uc3QgaG9zdCA9IGZpbmFsQ29uZmlnLmN1c3RvbSFbJ3NlcnZlcmxlc3Mtb2ZmbGluZSddIS5ob3N0ITtcblxuICAgIGxvZyhgQ3JlYXRpbmcgSFRUUCBzZXJ2ZXIgb24gJHtob3N0fToke2h0dHBQb3J0fWAsICdpbmZvJywgcXVpZXQpO1xuICAgIGxvZyhgQ3JlYXRpbmcgV2ViU29ja2V0IHNlcnZlciBvbiBwb3J0ICR7d3NQb3J0fWAsICdpbmZvJywgcXVpZXQpO1xuXG4gICAgLy8gQ3JlYXRlIEV4cHJlc3Mgc2VydmVyXG4gICAgY29uc3QgZXhwcmVzc0FwcCA9IGF3YWl0IGNyZWF0ZUV4cHJlc3NTZXJ2ZXIoXG4gICAgICBmaW5hbENvbmZpZyxcbiAgICAgIG91dHB1dERpcixcbiAgICAgIGh0dHBQb3J0LFxuICAgICAgaG9zdCxcbiAgICAgIHF1aWV0LFxuICAgICAgZGVidWdcbiAgICApO1xuXG4gICAgLy8gQ3JlYXRlIFdlYlNvY2tldCBzZXJ2ZXJcbiAgICBjb25zdCB3c1NlcnZlciA9IGNyZWF0ZVdlYlNvY2tldFNlcnZlcihcbiAgICAgIGZpbmFsQ29uZmlnLFxuICAgICAgb3V0cHV0RGlyLFxuICAgICAgd3NQb3J0LFxuICAgICAgcXVpZXQsXG4gICAgICBkZWJ1Z1xuICAgICk7XG5cbiAgICAvLyBIYW5kbGUgc2VydmVyIGVycm9yc1xuICAgIHdzU2VydmVyLm9uKCdlcnJvcicsIChlcnJvcjogRXJyb3IpID0+IHtcbiAgICAgIGxvZyhgV2ViU29ja2V0IHNlcnZlciBlcnJvcjogJHtlcnJvci5tZXNzYWdlfWAsICdlcnJvcicsIHF1aWV0KTtcbiAgICAgIHNwaW5uZXIuZmFpbCgnRmFpbGVkIHRvIHN0YXJ0IFdlYlNvY2tldCBzZXJ2ZXIuJyk7XG4gICAgICBjYWxsYmFjaygxKTtcbiAgICAgIHJldHVybjtcbiAgICB9KTtcblxuICAgIC8vIFN0YXJ0IEV4cHJlc3Mgc2VydmVyXG4gICAgY29uc3Qgc2VydmVyID0gZXhwcmVzc0FwcC5saXN0ZW4oaHR0cFBvcnQsIGhvc3QsICgpID0+IHtcbiAgICAgIHNwaW5uZXIuc3VjY2VlZCgnU2VydmVybGVzcyBkZXZlbG9wbWVudCBzZXJ2ZXIgc3RhcnRlZC4nKTtcbiAgICAgIHZvaWQgcmVzb2x2ZVB1YmxpY0lwRm9yRGlzcGxheSh1c2VQdWJsaWNJcCkudGhlbigocHVibGljSXApID0+IHtcbiAgICAgICAgZGlzcGxheVNlcnZlclN0YXR1cyhcbiAgICAgICAgICBodHRwUG9ydCxcbiAgICAgICAgICBmaW5hbENvbmZpZy5jdXN0b20hWydzZXJ2ZXJsZXNzLW9mZmxpbmUnXSEuaHR0cHNQb3J0ISxcbiAgICAgICAgICB3c1BvcnQsXG4gICAgICAgICAgaG9zdCxcbiAgICAgICAgICBxdWlldCxcbiAgICAgICAgICBwdWJsaWNJcFxuICAgICAgICApO1xuICAgICAgfSk7XG4gICAgfSk7XG5cbiAgICAvLyBIYW5kbGUgRXhwcmVzcyBzZXJ2ZXIgZXJyb3JzXG4gICAgc2VydmVyLm9uKCdlcnJvcicsIChlcnJvcikgPT4ge1xuICAgICAgbG9nKGBFeHByZXNzIHNlcnZlciBlcnJvcjogJHtlcnJvci5tZXNzYWdlfWAsICdlcnJvcicsIHF1aWV0KTtcbiAgICAgIHNwaW5uZXIuZmFpbCgnRmFpbGVkIHRvIHN0YXJ0IEV4cHJlc3Mgc2VydmVyLicpO1xuICAgICAgY2FsbGJhY2soMSk7XG4gICAgICByZXR1cm47XG4gICAgfSk7XG5cbiAgICAvLyBIYW5kbGUgZ3JhY2VmdWwgc2h1dGRvd25cbiAgICBjb25zdCBzaHV0ZG93biA9ICgpID0+IHtcbiAgICAgIGxvZygnXFxuU2h1dHRpbmcgZG93biBzZXJ2ZXJsZXNzIGRldmVsb3BtZW50IHNlcnZlci4uLicsICdpbmZvJywgcXVpZXQpO1xuICAgICAgc2VydmVyLmNsb3NlKCk7XG4gICAgICB3c1NlcnZlci5jbG9zZSgpO1xuICAgICAgY2FsbGJhY2soMCk7XG4gICAgfTtcblxuICAgIHByb2Nlc3Mub24oJ1NJR0lOVCcsIHNodXRkb3duKTtcbiAgICBwcm9jZXNzLm9uKCdTSUdURVJNJywgc2h1dGRvd24pO1xuXG4gICAgLy8gS2VlcCB0aGUgcHJvY2VzcyBhbGl2ZVxuICAgIHByb2Nlc3Muc3RkaW4ucmVzdW1lKCk7XG5cbiAgICBsb2coJ1NlcnZlcmxlc3MgZGV2ZWxvcG1lbnQgc2VydmVyIGlzIHJ1bm5pbmcuIFByZXNzIEN0cmwrQyB0byBzdG9wLicsICdpbmZvJywgcXVpZXQpO1xuXG4gICAgLy8gRG9uJ3QgY2FsbCBjYWxsYmFjayBoZXJlLCBsZXQgdGhlIHByb2Nlc3Mgc3RheSBhbGl2ZVxuICAgIHJldHVybiAwO1xuICB9IGNhdGNoKGVycm9yKSB7XG4gICAgbG9nKGBcXG4ke2NsaU5hbWV9IEVycm9yOiAke2Vycm9yLm1lc3NhZ2V9YCwgJ2Vycm9yJywgcXVpZXQpO1xuICAgIHNwaW5uZXIuZmFpbCgnRmFpbGVkIHRvIHN0YXJ0IHNlcnZlcmxlc3MgZGV2ZWxvcG1lbnQgc2VydmVyLicpO1xuICAgIGNhbGxiYWNrKDEpO1xuICAgIHJldHVybiAxO1xuICB9XG59O1xuIl0sIm5hbWVzIjpbImJveGVuIiwiY2hhbGsiLCJleHByZXNzIiwicmVhZEZpbGVTeW5jIiwiZXhpc3RzU3luYyIsIm1rZGlyU3luYyIsIndyaXRlRmlsZVN5bmMiLCJob21lZGlyIiwicmVzb2x2ZSIsInBhdGhSZXNvbHZlIiwiam9pbiIsImlzQWJzb2x1dGUiLCJwYXRoVG9GaWxlVVJMIiwiV2ViU29ja2V0U2VydmVyIiwiTGV4Q29uZmlnIiwiZ2V0UGFja2FnZURpciIsImNyZWF0ZVNwaW5uZXIiLCJyZW1vdmVGaWxlcyIsImxvZyIsImdldENhY2hlRGlyIiwiY2FjaGVEaXIiLCJyZWN1cnNpdmUiLCJnZXRDYWNoZVBhdGgiLCJyZWFkUHVibGljSXBDYWNoZSIsImNhY2hlUGF0aCIsImNhY2hlRGF0YSIsImNhY2hlIiwiSlNPTiIsInBhcnNlIiwib25lV2Vla01zIiwiRGF0ZSIsIm5vdyIsInRpbWVzdGFtcCIsIndyaXRlUHVibGljSXBDYWNoZSIsImlwIiwic3RyaW5naWZ5IiwiZmV0Y2hQdWJsaWNJcCIsImZvcmNlUmVmcmVzaCIsIlByb21pc2UiLCJjYWNoZWQiLCJmZXRjaCIsInRoZW4iLCJyZXMiLCJ0ZXh0IiwiZGF0YSIsInRyaW0iLCJjYXRjaCIsInVuZGVmaW5lZCIsInJlc29sdmVQdWJsaWNJcEZvckRpc3BsYXkiLCJ0aW1lb3V0TXMiLCJ0aW1lb3V0UHJvbWlzZSIsInNldFRpbWVvdXQiLCJyYWNlIiwiZGlzcGxheVNlcnZlclN0YXR1cyIsImh0dHBQb3J0IiwiaHR0cHNQb3J0Iiwid3NQb3J0IiwiaG9zdCIsInF1aWV0IiwicHVibGljSXAiLCJodHRwVXJsIiwiaHR0cHNVcmwiLCJ3c1VybCIsIndzc1VybCIsInVybExpbmVzIiwiZ3JlZW4iLCJ1bmRlcmxpbmUiLCJzdGF0dXNCb3giLCJjeWFuIiwiYm9sZCIsInllbGxvdyIsImJhY2tncm91bmRDb2xvciIsImJvcmRlckNvbG9yIiwiYm9yZGVyU3R5bGUiLCJtYXJnaW4iLCJwYWRkaW5nIiwiY29uc29sZSIsImxvYWRIYW5kbGVyIiwiaGFuZGxlclBhdGgiLCJvdXRwdXREaXIiLCJoYW5kbGVyUGFydHMiLCJzcGxpdCIsImZpbGVQYXRoIiwiZXhwb3J0TmFtZSIsImxlbmd0aCIsInNsaWNlIiwiaW5jbHVkZXMiLCJlcnJvciIsInBhdGhQYXJ0cyIsImxhc3RQYXJ0IiwiZnVsbFBhdGgiLCJleHRlbnNpb25zIiwicGF0aFdpdGhvdXRFeHQiLCJyZXBsYWNlIiwiZXh0IiwiY2FuZGlkYXRlUGF0aCIsIkVycm9yIiwiaW1wb3J0UGF0aCIsImVuZHNXaXRoIiwiaW1wb3J0VXJsIiwiaHJlZiIsImhhbmRsZXJNb2R1bGUiLCJPYmplY3QiLCJrZXlzIiwiaGFuZGxlciIsImRlZmF1bHQiLCJpbXBvcnRFcnJvciIsIm1lc3NhZ2UiLCJzdGFjayIsImNhcHR1cmVDb25zb2xlTG9ncyIsImV2ZW50IiwiY29udGV4dCIsIm9yaWdpbmFsQ29uc29sZUxvZyIsIm9yaWdpbmFsQ29uc29sZUVycm9yIiwib3JpZ2luYWxDb25zb2xlV2FybiIsIndhcm4iLCJvcmlnaW5hbENvbnNvbGVJbmZvIiwiaW5mbyIsImxvZ3MiLCJhcmdzIiwicHVzaCIsInJlc3VsdCIsImdyYXkiLCJmb3JFYWNoIiwiY3JlYXRlRXhwcmVzc1NlcnZlciIsImNvbmZpZyIsImRlYnVnIiwiYXBwIiwidXNlIiwicmVxIiwibmV4dCIsImhlYWRlciIsIm1ldGhvZCIsInNlbmRTdGF0dXMiLCJqc29uIiwibG9hZEdyYXBoUUxTY2hlbWEiLCJncmFwaHFsSGFuZGxlciIsImZ1bmN0aW9ucyIsImZ1bmN0aW9uTmFtZSIsImZ1bmN0aW9uQ29uZmlnIiwiZW50cmllcyIsImV2ZW50cyIsImh0dHAiLCJwYXRoIiwiZ3JhcGhxbFBhdGgiLCJfZnVuY3Rpb25OYW1lIiwiYm9keSIsInF1ZXJ5IiwidmFyaWFibGVzIiwib3BlcmF0aW9uTmFtZSIsImxvZ01lc3NhZ2UiLCJtYXAiLCJhcmciLCJTdHJpbmciLCJhd3NSZXF1ZXN0SWQiLCJmdW5jdGlvblZlcnNpb24iLCJnZXRSZW1haW5pbmdUaW1lSW5NaWxsaXMiLCJpbnZva2VkRnVuY3Rpb25Bcm4iLCJsb2dHcm91cE5hbWUiLCJsb2dTdHJlYW1OYW1lIiwid3JhcHBlZEhhbmRsZXIiLCJoZWFkZXJzIiwiaHR0cE1ldGhvZCIsInF1ZXJ5U3RyaW5nUGFyYW1ldGVycyIsInN0YXR1c0NvZGUiLCJzdGF0dXMiLCJrZXkiLCJ2YWx1ZSIsInNldEhlYWRlciIsInNlbmQiLCJ1cmwiLCJwYXRobmFtZSIsIm1hdGNoZWRGdW5jdGlvbiIsImZ1bmN0aW9uTmFtZXMiLCJldmVudFBhdGgiLCJldmVudE1ldGhvZCIsInRvVXBwZXJDYXNlIiwicmVxdWVzdE1ldGhvZCIsIm5vcm1hbGl6ZWRFdmVudFBhdGgiLCJub3JtYWxpemVkUGF0aG5hbWUiLCJtZW1vcnlMaW1pdEluTUIiLCJjcmVhdGVXZWJTb2NrZXRTZXJ2ZXIiLCJ3c3MiLCJwb3J0Iiwib24iLCJ3cyIsInRvU3RyaW5nIiwiZGVmYXVsdEZ1bmN0aW9uIiwid2Vic29ja2V0Iiwicm91dGUiLCJhY3Rpb24iLCJyZXF1ZXN0Q29udGV4dCIsImFwaUdhdGV3YXkiLCJlbmRwb2ludCIsImNvbm5lY3Rpb25JZCIsInJvdXRlS2V5IiwibG9hZEVudkZpbGUiLCJlbnZQYXRoIiwiZW52VmFycyIsImVudkNvbnRlbnQiLCJsaW5lcyIsImxpbmUiLCJ0cmltbWVkTGluZSIsInN0YXJ0c1dpdGgiLCJlcXVhbEluZGV4IiwiaW5kZXhPZiIsInN1YnN0cmluZyIsImNsZWFuVmFsdWUiLCJzZXJ2ZXJsZXNzIiwiY21kIiwiY2FsbGJhY2siLCJjbGlOYW1lIiwiY2xpSG9zdCIsImNsaUh0dHBQb3J0IiwiY2xpSHR0cHNQb3J0IiwicmVtb3ZlIiwidGVzdCIsInVzZVB1YmxpY0lwIiwiY2xpV3NQb3J0Iiwic3Bpbm5lciIsInBhcnNlQ29uZmlnIiwib3V0cHV0RnVsbFBhdGgiLCJlbnZQYXRocyIsInByb2Nlc3MiLCJjd2QiLCJmaWxlRW52VmFycyIsInZhcmlhYmxlc09iaiIsIk5PREVfRU5WIiwiY2xpVmFycyIsIl9lcnJvciIsImVudiIsInN0YXJ0Iiwic3VjY2VlZCIsInNlcnZlcmxlc3NDb25maWciLCJwYWNrYWdlRGlyIiwiY29uZmlnRm9ybWF0cyIsImNvbmZpZ1BhdGgiLCJjb25maWdNb2R1bGUiLCJjb25maWdPZmZsaW5lIiwiY3VzdG9tIiwiZWZmZWN0aXZlSG9zdCIsInRvTnVtYmVyIiwidiIsImZhbGxiYWNrIiwibiIsInBhcnNlSW50IiwiTnVtYmVyIiwiaXNGaW5pdGUiLCJlZmZlY3RpdmVIdHRwUG9ydCIsImVmZmVjdGl2ZUh0dHBzUG9ydCIsImVmZmVjdGl2ZVdzUG9ydCIsImZpbmFsQ29uZmlnIiwiY29ycyIsImV4cHJlc3NBcHAiLCJ3c1NlcnZlciIsImZhaWwiLCJzZXJ2ZXIiLCJsaXN0ZW4iLCJzaHV0ZG93biIsImNsb3NlIiwic3RkaW4iLCJyZXN1bWUiXSwibWFwcGluZ3MiOiJBQUFBOzs7Q0FHQyxHQUNELE9BQU9BLFdBQVcsUUFBUTtBQUMxQixPQUFPQyxXQUFXLFFBQVE7QUFDMUIsT0FBT0MsYUFBYSxVQUFVO0FBQzlCLFNBQVFDLFlBQVksRUFBRUMsVUFBVSxFQUFFQyxTQUFTLEVBQUVDLGFBQWEsUUFBTyxLQUFLO0FBQ3RFLFNBQVFDLE9BQU8sUUFBTyxLQUFLO0FBQzNCLFNBQVFDLFdBQVdDLFdBQVcsRUFBRUMsSUFBSSxFQUFFQyxVQUFVLFFBQU8sT0FBTztBQUM5RCxTQUFRQyxhQUFhLFFBQU8sTUFBTTtBQUNsQyxTQUFRQyxlQUFlLFFBQU8sS0FBSztBQUVuQyxTQUFRQyxTQUFTLEVBQUVDLGFBQWEsUUFBTyxxQkFBcUI7QUFDNUQsU0FBUUMsYUFBYSxFQUFFQyxXQUFXLFFBQU8scUJBQXFCO0FBQzlELFNBQVFDLEdBQUcsUUFBTyxxQkFBcUI7QUF3RHZDLE1BQU1DLGNBQWM7SUFDbEIsTUFBTUMsV0FBV1YsS0FBS0gsV0FBVztJQUNqQyxJQUFHLENBQUNILFdBQVdnQixXQUFXO1FBQ3hCZixVQUFVZSxVQUFVO1lBQUNDLFdBQVc7UUFBSTtJQUN0QztJQUNBLE9BQU9EO0FBQ1Q7QUFFQSxNQUFNRSxlQUFlLElBQWNaLEtBQUtTLGVBQWU7QUFFdkQsTUFBTUksb0JBQW9CO0lBQ3hCLE1BQU1DLFlBQVlGO0lBQ2xCLElBQUcsQ0FBQ2xCLFdBQVdvQixZQUFZO1FBQ3pCLE9BQU87SUFDVDtJQUVBLElBQUk7UUFDRixNQUFNQyxZQUFZdEIsYUFBYXFCLFdBQVc7UUFDMUMs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