@quantiya/codevibe-claude-plugin 1.0.10 → 1.0.12

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (80) hide show
  1. package/.claude-plugin/plugin.json +1 -1
  2. package/README.md +75 -245
  3. package/dist/server.js +16 -1162
  4. package/node_modules/@quantiya/codevibe-core/README.md +15 -6
  5. package/node_modules/@quantiya/codevibe-core/bin/codevibe.js +1 -1
  6. package/node_modules/@quantiya/codevibe-core/dist/index.js +216 -67
  7. package/node_modules/@quantiya/codevibe-core/package.json +12 -9
  8. package/node_modules/node-abi/abi_registry.json +7 -0
  9. package/node_modules/node-abi/package.json +1 -1
  10. package/package.json +11 -21
  11. package/dist/appsync-client.d.ts +0 -67
  12. package/dist/appsync-client.d.ts.map +0 -1
  13. package/dist/appsync-client.js +0 -858
  14. package/dist/appsync-client.js.map +0 -1
  15. package/dist/auth-cli.d.ts +0 -18
  16. package/dist/auth-cli.d.ts.map +0 -1
  17. package/dist/auth-cli.js +0 -472
  18. package/dist/auth-cli.js.map +0 -1
  19. package/dist/command-executor.d.ts +0 -20
  20. package/dist/command-executor.d.ts.map +0 -1
  21. package/dist/command-executor.js +0 -127
  22. package/dist/command-executor.js.map +0 -1
  23. package/dist/config.d.ts +0 -25
  24. package/dist/config.d.ts.map +0 -1
  25. package/dist/config.js +0 -106
  26. package/dist/config.js.map +0 -1
  27. package/dist/crypto-service.d.ts +0 -115
  28. package/dist/crypto-service.d.ts.map +0 -1
  29. package/dist/crypto-service.js +0 -278
  30. package/dist/crypto-service.js.map +0 -1
  31. package/dist/http-api.d.ts +0 -35
  32. package/dist/http-api.d.ts.map +0 -1
  33. package/dist/http-api.js +0 -334
  34. package/dist/http-api.js.map +0 -1
  35. package/dist/key-manager.d.ts +0 -87
  36. package/dist/key-manager.d.ts.map +0 -1
  37. package/dist/key-manager.js +0 -287
  38. package/dist/key-manager.js.map +0 -1
  39. package/dist/logger.d.ts +0 -2
  40. package/dist/logger.d.ts.map +0 -1
  41. package/dist/logger.js +0 -18
  42. package/dist/logger.js.map +0 -1
  43. package/dist/prompt-responder.d.ts +0 -22
  44. package/dist/prompt-responder.d.ts.map +0 -1
  45. package/dist/prompt-responder.js +0 -132
  46. package/dist/prompt-responder.js.map +0 -1
  47. package/dist/server.d.ts +0 -9
  48. package/dist/server.d.ts.map +0 -1
  49. package/dist/server.js.map +0 -1
  50. package/dist/token-storage.d.ts +0 -39
  51. package/dist/token-storage.d.ts.map +0 -1
  52. package/dist/token-storage.js +0 -169
  53. package/dist/token-storage.js.map +0 -1
  54. package/dist/types.d.ts +0 -110
  55. package/dist/types.d.ts.map +0 -1
  56. package/dist/types.js +0 -17
  57. package/dist/types.js.map +0 -1
  58. package/node_modules/@quantiya/codevibe-core/dist/appsync/appsync-client.js +0 -576
  59. package/node_modules/@quantiya/codevibe-core/dist/appsync/index.js +0 -10
  60. package/node_modules/@quantiya/codevibe-core/dist/appsync/queries.js +0 -189
  61. package/node_modules/@quantiya/codevibe-core/dist/auth/auth-cli.js +0 -217
  62. package/node_modules/@quantiya/codevibe-core/dist/auth/auth-service.js +0 -464
  63. package/node_modules/@quantiya/codevibe-core/dist/auth/fetch-helpers.js +0 -165
  64. package/node_modules/@quantiya/codevibe-core/dist/auth/index.js +0 -9
  65. package/node_modules/@quantiya/codevibe-core/dist/config/config.js +0 -123
  66. package/node_modules/@quantiya/codevibe-core/dist/config/index.js +0 -8
  67. package/node_modules/@quantiya/codevibe-core/dist/crypto/crypto-service.js +0 -284
  68. package/node_modules/@quantiya/codevibe-core/dist/crypto/index.js +0 -9
  69. package/node_modules/@quantiya/codevibe-core/dist/keychain/index.js +0 -8
  70. package/node_modules/@quantiya/codevibe-core/dist/keychain/keychain-manager.js +0 -375
  71. package/node_modules/@quantiya/codevibe-core/dist/logger/index.js +0 -8
  72. package/node_modules/@quantiya/codevibe-core/dist/logger/logger.js +0 -142
  73. package/node_modules/@quantiya/codevibe-core/dist/prompt-parser.js +0 -236
  74. package/node_modules/@quantiya/codevibe-core/dist/session/index.js +0 -7
  75. package/node_modules/@quantiya/codevibe-core/dist/session/session-resume.js +0 -151
  76. package/node_modules/@quantiya/codevibe-core/dist/types/auth.js +0 -3
  77. package/node_modules/@quantiya/codevibe-core/dist/types/encryption.js +0 -3
  78. package/node_modules/@quantiya/codevibe-core/dist/types/events.js +0 -28
  79. package/node_modules/@quantiya/codevibe-core/dist/types/index.js +0 -22
  80. package/node_modules/@quantiya/codevibe-core/dist/types/session.js +0 -22
@@ -1,464 +0,0 @@
1
- "use strict";
2
- //
3
- // auth-service.ts
4
- // CodeVibe Core
5
- //
6
- // OAuth authentication service
7
- //
8
- var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
9
- if (k2 === undefined) k2 = k;
10
- var desc = Object.getOwnPropertyDescriptor(m, k);
11
- if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
12
- desc = { enumerable: true, get: function() { return m[k]; } };
13
- }
14
- Object.defineProperty(o, k2, desc);
15
- }) : (function(o, m, k, k2) {
16
- if (k2 === undefined) k2 = k;
17
- o[k2] = m[k];
18
- }));
19
- var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
20
- Object.defineProperty(o, "default", { enumerable: true, value: v });
21
- }) : function(o, v) {
22
- o["default"] = v;
23
- });
24
- var __importStar = (this && this.__importStar) || (function () {
25
- var ownKeys = function(o) {
26
- ownKeys = Object.getOwnPropertyNames || function (o) {
27
- var ar = [];
28
- for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
29
- return ar;
30
- };
31
- return ownKeys(o);
32
- };
33
- return function (mod) {
34
- if (mod && mod.__esModule) return mod;
35
- var result = {};
36
- if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
37
- __setModuleDefault(result, mod);
38
- return result;
39
- };
40
- })();
41
- Object.defineProperty(exports, "__esModule", { value: true });
42
- exports.authService = exports.AuthService = void 0;
43
- const crypto = __importStar(require("crypto"));
44
- const fs = __importStar(require("fs"));
45
- const http = __importStar(require("http"));
46
- const child_process_1 = require("child_process");
47
- const config_1 = require("../config");
48
- const keychain_1 = require("../keychain");
49
- const logger_1 = require("../logger");
50
- const fetch_helpers_1 = require("./fetch-helpers");
51
- const CALLBACK_PORT = 8080;
52
- const CALLBACK_PATH = '/callback';
53
- const REDIRECT_URI = `http://localhost:${CALLBACK_PORT}${CALLBACK_PATH}`;
54
- /**
55
- * Authentication service for OAuth flows
56
- */
57
- class AuthService {
58
- constructor() { }
59
- static getInstance() {
60
- if (!AuthService.instance) {
61
- AuthService.instance = new AuthService();
62
- }
63
- return AuthService.instance;
64
- }
65
- /**
66
- * Open URL in the user's default browser. Cross-platform: macOS, Linux,
67
- * WSL, Windows. Always prints the URL to stdout first as a fallback —
68
- * if no browser-opening command is available, the user can copy-paste.
69
- *
70
- * On WSL, prefers opening the Windows host browser via WSL interop
71
- * (wslview → cmd.exe → powershell.exe) before falling back to xdg-open.
72
- */
73
- openBrowser(url) {
74
- // Always print the URL first so users can copy-paste if automatic
75
- // browser-opening fails (especially important on headless WSL setups).
76
- console.log('');
77
- console.log('Opening your browser for sign-in...');
78
- console.log('If your browser does not open automatically, visit this URL manually:');
79
- console.log(` ${url}`);
80
- console.log('');
81
- const attempts = this.getBrowserCommands();
82
- this.tryBrowserCommand(attempts, url, 0);
83
- }
84
- /**
85
- * Returns the list of browser-opening commands to try in order, based on
86
- * the current platform. On WSL, returns WSL interop commands first so the
87
- * Windows browser opens (which is what users actually want on WSL).
88
- */
89
- getBrowserCommands() {
90
- const platform = process.platform;
91
- if (platform === 'darwin') {
92
- return [{ cmd: 'open', fixedArgs: [] }];
93
- }
94
- if (platform === 'win32') {
95
- // cmd's start needs an empty title as the first arg so that URLs
96
- // containing special characters aren't misinterpreted as the title.
97
- return [{ cmd: 'cmd', fixedArgs: ['/c', 'start', ''] }];
98
- }
99
- // Linux or WSL
100
- const attempts = [];
101
- if (this.isRunningInWSL()) {
102
- // Prefer opening the Windows host browser via WSL interop.
103
- attempts.push({ cmd: 'wslview', fixedArgs: [] });
104
- attempts.push({ cmd: 'cmd.exe', fixedArgs: ['/c', 'start', ''] });
105
- attempts.push({
106
- cmd: 'powershell.exe',
107
- fixedArgs: ['-NoProfile', '-Command', 'Start-Process'],
108
- });
109
- }
110
- // Fall back to generic Linux openers
111
- attempts.push({ cmd: 'xdg-open', fixedArgs: [] });
112
- return attempts;
113
- }
114
- /**
115
- * Detect whether we're running inside WSL (1 or 2). Returns false on
116
- * any non-Linux platform or if /proc/sys/kernel/osrelease is not readable.
117
- */
118
- isRunningInWSL() {
119
- if (process.platform !== 'linux')
120
- return false;
121
- try {
122
- const osRelease = fs.readFileSync('/proc/sys/kernel/osrelease', 'utf8');
123
- return /microsoft|wsl/i.test(osRelease);
124
- }
125
- catch {
126
- return false;
127
- }
128
- }
129
- /**
130
- * Try each browser-opening command in order. Advances to the next fallback on:
131
- * - Spawn failure (ENOENT / the command not being installed)
132
- * - Synchronous throw from spawn()
133
- * - Runtime failure where the command spawns but exits with a non-zero
134
- * code (e.g. wslview when WSL interop is disabled, xdg-open when no
135
- * default browser is registered, cmd.exe failing to launch start)
136
- * - Process terminated by signal
137
- *
138
- * Stays on the current command when:
139
- * - Process exits with code 0 (success)
140
- * - Process is still running after 3 seconds (assumed success — opener
141
- * is doing real work like launching a slow app, not hung)
142
- *
143
- * If all attempts exhaust, logs at debug level — the user still has the
144
- * sign-in URL printed to stdout as a copy-paste fallback.
145
- */
146
- tryBrowserCommand(attempts, url, index) {
147
- if (index >= attempts.length) {
148
- logger_1.logger.debug('[AuthService] No browser-opening command succeeded. ' +
149
- 'User must open the sign-in URL manually (printed to stdout above).');
150
- return;
151
- }
152
- const attempt = attempts[index];
153
- const args = [...attempt.fixedArgs, url];
154
- let settled = false;
155
- // Helper to advance exactly once — guards against both 'error' and 'exit'
156
- // events firing for the same child, or the timeout racing with an event.
157
- const advance = (reason) => {
158
- if (settled)
159
- return;
160
- settled = true;
161
- logger_1.logger.debug(`[AuthService] Browser command '${attempt.cmd}' ${reason}; trying next fallback`);
162
- this.tryBrowserCommand(attempts, url, index + 1);
163
- };
164
- // Helper to mark this attempt as successful so no future event handlers
165
- // fire tryBrowserCommand on the next index.
166
- const stopTrying = (reason) => {
167
- if (settled)
168
- return;
169
- settled = true;
170
- logger_1.logger.debug(`[AuthService] Browser command '${attempt.cmd}' ${reason}`);
171
- };
172
- let child;
173
- try {
174
- child = (0, child_process_1.spawn)(attempt.cmd, args, {
175
- detached: true,
176
- stdio: 'ignore',
177
- });
178
- }
179
- catch (err) {
180
- advance(`threw synchronously: ${err?.message || err}`);
181
- return;
182
- }
183
- // Fires when the OS cannot spawn the process at all (most commonly ENOENT
184
- // for "command not found"). The process never actually ran.
185
- child.on('error', (err) => {
186
- advance(`failed to spawn: ${err?.message || err}`);
187
- });
188
- // Fires after the spawned process exits. code === 0 is success (opener
189
- // delegated cleanly and returned). Non-zero code or a signal means the
190
- // opener failed at runtime — we fall through to the next fallback.
191
- child.on('exit', (code, signal) => {
192
- if (code === 0) {
193
- stopTrying('exited successfully');
194
- }
195
- else if (signal) {
196
- advance(`terminated by signal ${signal}`);
197
- }
198
- else {
199
- advance(`exited with code ${code}`);
200
- }
201
- });
202
- // If the opener is still running after 3 seconds, it is almost certainly
203
- // doing real work (launching a slow browser, waiting for Windows interop)
204
- // rather than hung. Stop trying more fallbacks so we don't spawn multiple
205
- // browsers. The child keeps running independently because detached:true.
206
- const timer = setTimeout(() => {
207
- stopTrying('still running after 3s, assuming success');
208
- }, 3000);
209
- // Don't let the timer itself keep the Node event loop alive.
210
- timer.unref();
211
- // Detach the child from the parent's event loop so Node doesn't block
212
- // on its exit. Exit handler still fires if/when the child actually exits.
213
- child.unref();
214
- }
215
- /**
216
- * Generate state for CSRF protection
217
- */
218
- generateState() {
219
- return crypto.randomBytes(32).toString('hex');
220
- }
221
- /**
222
- * Build authorization URL
223
- */
224
- buildAuthUrl(state) {
225
- const config = (0, config_1.getConfig)();
226
- const params = new URLSearchParams({
227
- client_id: config.aws.cognitoClientId,
228
- response_type: 'code',
229
- scope: 'email openid profile',
230
- redirect_uri: REDIRECT_URI,
231
- state: state,
232
- });
233
- return `https://${config.aws.cognitoDomain}/oauth2/authorize?${params.toString()}`;
234
- }
235
- /**
236
- * Exchange authorization code for tokens
237
- */
238
- async exchangeCodeForTokens(code) {
239
- const config = (0, config_1.getConfig)();
240
- const tokenUrl = `https://${config.aws.cognitoDomain}/oauth2/token`;
241
- const params = new URLSearchParams({
242
- grant_type: 'authorization_code',
243
- client_id: config.aws.cognitoClientId,
244
- code: code,
245
- redirect_uri: REDIRECT_URI,
246
- });
247
- const response = await (0, fetch_helpers_1.fetchWithDiagnostics)(tokenUrl, {
248
- method: 'POST',
249
- headers: { 'Content-Type': 'application/x-www-form-urlencoded' },
250
- body: params.toString(),
251
- }, 'Token exchange');
252
- if (!response.ok) {
253
- const errorText = await response.text();
254
- throw new Error(`Token exchange failed: ${response.status} ${errorText}`);
255
- }
256
- const data = await response.json();
257
- return {
258
- accessToken: data.access_token,
259
- idToken: data.id_token,
260
- refreshToken: data.refresh_token,
261
- expiresIn: data.expires_in,
262
- };
263
- }
264
- /**
265
- * Decode JWT payload
266
- */
267
- decodeJwt(token) {
268
- const parts = token.split('.');
269
- if (parts.length !== 3)
270
- throw new Error('Invalid JWT');
271
- return JSON.parse(Buffer.from(parts[1], 'base64').toString('utf-8'));
272
- }
273
- /**
274
- * Refresh tokens
275
- */
276
- async refreshTokens(refreshToken) {
277
- const config = (0, config_1.getConfig)();
278
- const tokenUrl = `https://${config.aws.cognitoDomain}/oauth2/token`;
279
- const params = new URLSearchParams({
280
- grant_type: 'refresh_token',
281
- client_id: config.aws.cognitoClientId,
282
- refresh_token: refreshToken,
283
- });
284
- const response = await (0, fetch_helpers_1.fetchWithDiagnostics)(tokenUrl, {
285
- method: 'POST',
286
- headers: { 'Content-Type': 'application/x-www-form-urlencoded' },
287
- body: params.toString(),
288
- }, 'Token refresh');
289
- if (!response.ok) {
290
- throw new Error(`Token refresh failed: ${response.status}`);
291
- }
292
- const data = await response.json();
293
- return {
294
- accessToken: data.access_token,
295
- idToken: data.id_token,
296
- expiresIn: data.expires_in,
297
- };
298
- }
299
- /**
300
- * Login via OAuth browser flow
301
- */
302
- async login() {
303
- // Check if already logged in
304
- const existing = await keychain_1.keychainManager.getTokens((0, config_1.getEnvironment)());
305
- if (existing && !keychain_1.keychainManager.isTokenExpired(existing)) {
306
- return existing;
307
- }
308
- const state = this.generateState();
309
- const authUrl = this.buildAuthUrl(state);
310
- return new Promise((resolve, reject) => {
311
- const server = http.createServer(async (req, res) => {
312
- if (!req.url?.startsWith(CALLBACK_PATH)) {
313
- res.writeHead(404);
314
- res.end('Not found');
315
- return;
316
- }
317
- try {
318
- const url = new URL(req.url, `http://localhost:${CALLBACK_PORT}`);
319
- const code = url.searchParams.get('code');
320
- const returnedState = url.searchParams.get('state');
321
- const error = url.searchParams.get('error');
322
- if (error) {
323
- throw new Error(`OAuth error: ${error}`);
324
- }
325
- if (returnedState !== state) {
326
- throw new Error('State mismatch');
327
- }
328
- if (!code) {
329
- throw new Error('No authorization code');
330
- }
331
- const tokens = await this.exchangeCodeForTokens(code);
332
- const idPayload = this.decodeJwt(tokens.idToken);
333
- const tokenData = {
334
- accessToken: tokens.accessToken,
335
- idToken: tokens.idToken,
336
- refreshToken: tokens.refreshToken,
337
- expiresAt: Date.now() + (tokens.expiresIn * 1000),
338
- userId: idPayload.sub,
339
- email: idPayload.email || 'unknown',
340
- };
341
- await keychain_1.keychainManager.setTokens(tokenData, (0, config_1.getEnvironment)());
342
- res.writeHead(200, { 'Content-Type': 'text/html; charset=utf-8' });
343
- res.end(`
344
- <!DOCTYPE html>
345
- <html>
346
- <head><title>Success</title></head>
347
- <body style="font-family: system-ui; max-width: 600px; margin: 50px auto; text-align: center;">
348
- <h1 style="color: #22c55e;">&#10003; Authentication Successful</h1>
349
- <p>You can close this window.</p>
350
- </body>
351
- </html>
352
- `);
353
- setTimeout(() => {
354
- server.close(() => resolve(tokenData));
355
- }, 500);
356
- }
357
- catch (err) {
358
- // HTML-escape the error message so newlines and special chars
359
- // render safely. Multi-line diagnostic messages (from
360
- // fetchWithDiagnostics) need <pre> to preserve formatting.
361
- const safeMessage = String(err?.message || err)
362
- .replace(/&/g, '&amp;')
363
- .replace(/</g, '&lt;')
364
- .replace(/>/g, '&gt;');
365
- res.writeHead(400, { 'Content-Type': 'text/html; charset=utf-8' });
366
- res.end(`
367
- <!DOCTYPE html>
368
- <html>
369
- <head><title>Error</title></head>
370
- <body style="font-family: system-ui; max-width: 720px; margin: 50px auto; padding: 0 16px;">
371
- <h1 style="color: #ef4444; text-align: center;">&#10007; Authentication Failed</h1>
372
- <pre style="background: #f4f4f5; padding: 16px; border-radius: 8px; white-space: pre-wrap; word-wrap: break-word; font-size: 13px; line-height: 1.5;">${safeMessage}</pre>
373
- <p style="text-align: center; color: #71717a; margin-top: 24px;">You can close this window and try again in your terminal.</p>
374
- </body>
375
- </html>
376
- `);
377
- setTimeout(() => {
378
- server.close(() => reject(err));
379
- }, 500);
380
- }
381
- });
382
- server.on('error', (err) => {
383
- if (err.code === 'EADDRINUSE') {
384
- reject(new Error(`Port ${CALLBACK_PORT} is in use`));
385
- }
386
- else {
387
- reject(err);
388
- }
389
- });
390
- server.listen(CALLBACK_PORT, 'localhost', () => {
391
- logger_1.logger.info('[AuthService] Callback server started');
392
- this.openBrowser(authUrl);
393
- });
394
- // Timeout
395
- setTimeout(() => {
396
- server.close(() => reject(new Error('Login timeout')));
397
- }, 2 * 60 * 1000);
398
- });
399
- }
400
- /**
401
- * Logout
402
- */
403
- async logout() {
404
- const config = (0, config_1.getConfig)();
405
- const deleted = await keychain_1.keychainManager.deleteTokens((0, config_1.getEnvironment)());
406
- if (deleted) {
407
- // Start temporary server to handle signout callback
408
- return new Promise((resolve) => {
409
- const server = http.createServer((req, res) => {
410
- if (req.url?.startsWith('/signout')) {
411
- res.writeHead(200, { 'Content-Type': 'text/html; charset=utf-8' });
412
- res.end(`
413
- <!DOCTYPE html>
414
- <html>
415
- <head><title>Signed Out</title></head>
416
- <body style="font-family: system-ui; max-width: 600px; margin: 50px auto; text-align: center;">
417
- <h1 style="color: #22c55e;">&#10003; Signed Out</h1>
418
- <p>You can close this window.</p>
419
- </body>
420
- </html>
421
- `);
422
- setTimeout(() => {
423
- server.close(() => resolve(true));
424
- }, 500);
425
- }
426
- else {
427
- res.writeHead(404);
428
- res.end('Not found');
429
- }
430
- });
431
- server.on('error', () => {
432
- resolve(true); // Still return true since tokens were deleted
433
- });
434
- server.listen(CALLBACK_PORT, 'localhost', () => {
435
- const logoutUrl = `https://${config.aws.cognitoDomain}/logout?` +
436
- `client_id=${config.aws.cognitoClientId}&` +
437
- `logout_uri=${encodeURIComponent(REDIRECT_URI.replace('/callback', '/signout'))}`;
438
- this.openBrowser(logoutUrl);
439
- });
440
- // Timeout after 30 seconds
441
- setTimeout(() => {
442
- server.close(() => resolve(true));
443
- }, 30 * 1000);
444
- });
445
- }
446
- return deleted;
447
- }
448
- /**
449
- * Get current auth status
450
- */
451
- async getStatus() {
452
- const tokens = await keychain_1.keychainManager.getTokens((0, config_1.getEnvironment)());
453
- if (!tokens) {
454
- return { authenticated: false };
455
- }
456
- return {
457
- authenticated: !keychain_1.keychainManager.isTokenExpired(tokens),
458
- tokens,
459
- };
460
- }
461
- }
462
- exports.AuthService = AuthService;
463
- exports.authService = AuthService.getInstance();
464
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXV0aC1zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL2F1dGgvYXV0aC1zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQSxFQUFFO0FBQ0Ysa0JBQWtCO0FBQ2xCLGdCQUFnQjtBQUNoQixFQUFFO0FBQ0YsK0JBQStCO0FBQy9CLEVBQUU7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQUVGLCtDQUFpQztBQUNqQyx1Q0FBeUI7QUFDekIsMkNBQTZCO0FBQzdCLGlEQUFzQztBQUN0QyxzQ0FBc0Q7QUFDdEQsMENBQThDO0FBQzlDLHNDQUFtQztBQUVuQyxtREFBdUQ7QUFFdkQsTUFBTSxhQUFhLEdBQUcsSUFBSSxDQUFDO0FBQzNCLE1BQU0sYUFBYSxHQUFHLFdBQVcsQ0FBQztBQUNsQyxNQUFNLFlBQVksR0FBRyxvQkFBb0IsYUFBYSxHQUFHLGFBQWEsRUFBRSxDQUFDO0FBRXpFOztHQUVHO0FBQ0gsTUFBYSxXQUFXO0lBR3RCLGdCQUF1QixDQUFDO0lBRXhCLE1BQU0sQ0FBQyxXQUFXO1FBQ2hCLElBQUksQ0FBQyxXQUFXLENBQUMsUUFBUSxFQUFFLENBQUM7WUFDMUIsV0FBVyxDQUFDLFFBQVEsR0FBRyxJQUFJLFdBQVcsRUFBRSxDQUFDO1FBQzNDLENBQUM7UUFDRCxPQUFPLFdBQVcsQ0FBQyxRQUFRLENBQUM7SUFDOUIsQ0FBQztJQUVEOzs7Ozs7O09BT0c7SUFDSyxXQUFXLENBQUMsR0FBVztRQUM3QixrRUFBa0U7UUFDbEUsdUVBQXVFO1FBQ3ZFLE9BQU8sQ0FBQyxHQUFHLENBQUMsRUFBRSxDQUFDLENBQUM7UUFDaEIsT0FBTyxDQUFDLEdBQUcsQ0FBQyxxQ0FBcUMsQ0FBQyxDQUFDO1FBQ25ELE9BQU8sQ0FBQyxHQUFHLENBQUMsdUVBQXVFLENBQUMsQ0FBQztRQUNyRixPQUFPLENBQUMsR0FBRyxDQUFDLEtBQUssR0FBRyxFQUFFLENBQUMsQ0FBQztRQUN4QixPQUFP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Q0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDO2dCQUNwQyxDQUFDLEVBQUUsRUFBRSxHQUFHLElBQUksQ0FBQyxDQUFDO1lBQ2hCLENBQUMsQ0FBQyxDQUFDO1FBQ0wsQ0FBQztRQUVELE9BQU8sT0FBTyxDQUFDO0lBQ2pCLENBQUM7SUFFRDs7T0FFRztJQUNILEtBQUssQ0FBQyxTQUFTO1FBQ2IsTUFBTSxNQUFNLEdBQUcsTUFBTSwwQkFBZSxDQUFDLFNBQVMsQ0FBQyxJQUFBLHVCQUFjLEdBQUUsQ0FBQyxDQUFDO1FBQ2pFLElBQUksQ0FBQyxNQUFNLEVBQUUsQ0FBQztZQUNaLE9BQU8sRUFBRSxhQUFhLEVBQUUsS0FBSyxFQUFFLENBQUM7UUFDbEMsQ0FBQztRQUVELE9BQU87WUFDTCxhQUFhLEVBQUUsQ0FBQywwQkFBZSxDQUFDLGNBQWMsQ0FBQyxNQUFNLENBQUM7WUFDdEQsTUFBTTtTQUNQLENBQUM7SUFDSixDQUFDO0NBQ0Y7QUE1ZUQsa0NBNGVDO0FBRVksUUFBQSxXQUFXLEdBQUcsV0FBVyxDQUFDLFdBQVcsRUFBRSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLy9cbi8vIGF1dGgtc2VydmljZS50c1xuLy8gQ29kZVZpYmUgQ29yZVxuLy9cbi8vIE9BdXRoIGF1dGhlbnRpY2F0aW9uIHNlcnZpY2Vcbi8vXG5cbmltcG9ydCAqIGFzIGNyeXB0byBmcm9tICdjcnlwdG8nO1xuaW1wb3J0ICogYXMgZnMgZnJvbSAnZnMnO1xuaW1wb3J0ICogYXMgaHR0cCBmcm9tICdodHRwJztcbmltcG9ydCB7IHNwYXduIH0gZnJvbSAnY2hpbGRfcHJvY2Vzcyc7XG5pbXBvcnQgeyBnZXRDb25maWcsIGdldEVudmlyb25tZW50IH0gZnJvbSAnLi4vY29uZmlnJztcbmltcG9ydCB7IGtleWNoYWluTWFuYWdlciB9IGZyb20gJy4uL2tleWNoYWluJztcbmltcG9ydCB7IGxvZ2dlciB9IGZyb20gJy4uL2xvZ2dlcic7XG5pbXBvcnQgeyBUb2tlbkRhdGEgfSBmcm9tICcuLi90eXBlcyc7XG5pbXBvcnQgeyBmZXRjaFdpdGhEaWFnbm9zdGljcyB9IGZyb20gJy4vZmV0Y2gtaGVscGVycyc7XG5cbmNvbnN0IENBTExCQUNLX1BPUlQgPSA4MDgwO1xuY29uc3QgQ0FMTEJBQ0tfUEFUSCA9ICcvY2FsbGJhY2snO1xuY29uc3QgUkVESVJFQ1RfVVJJID0gYGh0dHA6Ly9sb2NhbGhvc3Q6JHtDQUxMQkFDS19QT1JUfSR7Q0FMTEJBQ0tfUEFUSH1gO1xuXG4vKipcbiAqIEF1dGhlbnRpY2F0aW9uIHNlcnZpY2UgZm9yIE9BdXRoIGZsb3dzXG4gKi9cbmV4cG9ydCBjbGFzcyBBdXRoU2VydmljZSB7XG4gIHByaXZhdGUgc3RhdGljIGluc3RhbmNlOiBBdXRoU2VydmljZTtcblxuICBwcml2YXRlIGNvbnN0cnVjdG9yKCkge31cblxuICBzdGF0aWMgZ2V0SW5zdGFuY2UoKTogQXV0aFNlcnZpY2Uge1xuICAgIGlmICghQXV0aFNlcnZpY2UuaW5zdGFuY2UpIHtcbiAgICAgIEF1dGhTZXJ2aWNlLmluc3RhbmNlID0gbmV3IEF1dGhTZXJ2aWNlKCk7XG4gICAgfVxuICAgIHJldHVybiBBdXRoU2VydmljZS5pbnN0YW5jZTtcbiAgfVxuXG4gIC8qKlxuICAgKiBPcGVuIFVSTCBpbiB0aGUgdXNlcidzIGRlZmF1bHQgYnJvd3Nlci4gQ3Jvc3MtcGxhdGZvcm06IG1hY09TLCBMaW51eCxcbiAgICogV1NMLCBXaW5kb3dzLiBBbHdheXMgcHJpbnRzIHRoZSBVUkwgdG8gc3Rkb3V0IGZpcnN0IGFzIGEgZmFsbGJhY2sg4oCUXG4gICAqIGlmIG5vIGJyb3dzZXItb3BlbmluZyBjb21tYW5kIGlzIGF2YWlsYWJsZSwgdGhlIHVzZXIgY2FuIGNvcHktcGFzdGUuXG4gICAqXG4gICAqIE9uIFdTTCwgcHJlZmVycyBvcGVuaW5nIHRoZSBXaW5kb3dzIGhvc3QgYnJvd3NlciB2aWEgV1NMIGludGVyb3BcbiAgICogKHdzbHZpZXcg4oaSIGNtZC5leGUg4oaSIHBvd2Vyc2hlbGwuZXhlKSBiZWZvcmUgZmFsbGluZyBiYWNrIHRvIHhkZy1vcGVuLlxuICAgKi9cbiAgcHJpdmF0ZSBvcGVuQnJvd3Nlcih1cmw6IHN0cmluZyk6IHZvaWQge1xuICAgIC8vIEFsd2F5cyBwcmludCB0aGUgVVJMIGZpcnN0IHNvIHVzZXJzIGNhbiBjb3B5LXBhc3RlIGlmIGF1dG9tYXRpY1xuICAgIC8vIGJyb3dzZXItb3BlbmluZyBmYWlscyAoZXNwZWNpYWxseSBpbXBvcnRhbnQgb24gaGVhZGxlc3MgV1NMIHNldHVwcykuXG4gICAgY29uc29sZS5sb2coJycpO1xuICAgIGNvbnNvbGUubG9nKCdPcGVuaW5nIHlvdXIgYnJvd3NlciBmb3Igc2lnbi1pbi4uLicpO1xuICAgIGNvbnNvbGUubG9nKCdJZiB5b3VyIGJyb3dzZXIgZG9lcyBub3Qgb3BlbiBhdXRvbWF0aWNhbGx5LCB2aXNpdCB0aGlzIFVSTCBtYW51YWxseTonKTtcbiAgICBjb25zb2xlLmxvZyhgICAke3VybH1gKTtcbiAgICBjb25zb2xlLmxvZygnJyk7XG5cbiAgICBjb25zdCBhdHRlbXB0cyA9IHRoaXMuZ2V0QnJvd3NlckNvbW1hbmRzKCk7XG4gICAgdGhpcy50cnlCcm93c2VyQ29tbWFuZChhdHRlbXB0cywgdXJsLCAwKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZXR1cm5zIHRoZSBsaXN0IG9mIGJyb3dzZXItb3BlbmluZyBjb21tYW5kcyB0byB0cnkgaW4gb3JkZXIsIGJhc2VkIG9uXG4gICAqIHRoZSBjdXJyZW50IHBsYXRmb3JtLiBPbiBXU0wsIHJldHVybnMgV1NMIGludGVyb3AgY29tbWFuZHMgZmlyc3Qgc28gdGhlXG4gICAqIFdpbmRvd3MgYnJvd3NlciBvcGVucyAod2hpY2ggaXMgd2hhdCB1c2VycyBhY3R1YWxseSB3YW50IG9uIFdTTCkuXG4gICAqL1xuICBwcml2YXRlIGdldEJyb3dzZXJDb21tYW5kcygpOiBBcnJheTx7IGNtZDogc3RyaW5nOyBmaXhlZEFyZ3M6IHN0cmluZ1tdIH0+IHtcbiAgICBjb25zdCBwbGF0Zm9ybSA9IHByb2Nlc3MucGxhdGZvcm07XG5cbiAgICBpZiAocGxhdGZvcm0gPT09ICdkYXJ3aW4nKSB7XG4gICAgICByZXR1cm4gW3sgY21kOiAnb3BlbicsIGZpeGVkQXJnczogW10gfV07XG4gICAgfVxuXG4gICAgaWYgKHBsYXRmb3JtID09PSAnd2luMzInKSB7XG4gICAgICAvLyBjbWQncyBzdGFydCBuZWVkcyBhbiBlbXB0eSB0aXRsZSBhcyB0aGUgZmlyc3QgYXJnIHNvIHRoYXQgVVJMc1xuICAgICAgLy8gY29udGFpbmluZyBzcGVjaWFsIGNoYXJhY3RlcnMgYXJlbid0IG1pc2ludGVycHJldGVkIGFzIHRoZSB0aXRsZS5cbiAgICAgIHJldHVybiBbeyBjbWQ6ICdjbWQnLCBmaXhlZEFyZ3M6IFsnL2MnLCAnc3RhcnQnLCAnJ10gfV07XG4gICAgfVxuXG4gICAgLy8gTGludXggb3IgV1NMXG4gICAgY29uc3QgYXR0ZW1wdHM6IEFycmF5PHsgY21kOiBzdHJpbmc7IGZpeGVkQXJnczogc3RyaW5nW10gfT4gPSBbXTtcblxuICAgIGlmICh0aGlzLmlzUnVubmluZ0luV1NMKCkpIHtcbiAgICAgIC8vIFByZWZlciBvcGVuaW5nIHRoZSBXaW5kb3dzIGhvc3QgYnJvd3NlciB2aWEgV1NMIGludGVyb3AuXG4gICAgICBhdHRlbXB0cy5wdXNoKHsgY21kOiAnd3NsdmlldycsIGZpeGVkQXJnczogW10gfSk7XG4gICAgICBhdHRlbXB0cy5wdXNoKHsgY21kOiAnY21kLmV4ZScsIGZpeGVkQXJnczogWycvYycsICdzdGFydCcsICcnXSB9KTtcbiAgICAgIGF0dGVtcHRzLnB1c2goe1xuICAgICAgICBjbWQ6ICdwb3dlcnNoZWxsLmV4ZScsXG4gICAgICAgIGZpeGVkQXJnczogWyctTm9Qcm9maWxlJywgJy1Db21tYW5kJywgJ1N0YXJ0LVByb2Nlc3MnXSxcbiAgICAgIH0pO1xuICAgIH1cblxuICAgIC8vIEZhbGwgYmFjayB0byBnZW5lcmljIExpbnV4IG9wZW5lcnNcbiAgICBhdHRlbXB0cy5wdXNoKHsgY21kOiAneGRnLW9wZW4nLCBmaXhlZEFyZ3M6IFtdIH0pO1xuXG4gICAgcmV0dXJuIGF0dGVtcHRzO1xuICB9XG5cbiAgLyoqXG4gICAqIERldGVjdCB3aGV0aGVyIHdlJ3JlIHJ1bm5pbmcgaW5zaWRlIFdTTCAoMSBvciAyKS4gUmV0dXJucyBmYWxzZSBvblxuICAgKiBhbnkgbm9uLUxpbnV4IHBsYXRmb3JtIG9yIGlmIC9wcm9jL3N5cy9rZXJuZWwvb3NyZWxlYXNlIGlzIG5vdCByZWFkYWJsZS5cbiAgICovXG4gIHByaXZhdGUgaXNSdW5uaW5nSW5XU0woKTogYm9vbGVhbiB7XG4gICAgaWYgKHByb2Nlc3MucGxhdGZvcm0gIT09ICdsaW51eCcpIHJldHVybiBmYWxzZTtcbiAgICB0cnkge1xuICAgICAgY29uc3Qgb3NSZWxlYXNlID0gZnMucmVhZEZpbGVTeW5jKCcvcHJvYy9zeXMva2VybmVsL29zcmVsZWFzZScsICd1dGY4Jyk7XG4gICAgICByZXR1cm4gL21pY3Jvc29mdHx3c2wvaS50ZXN0KG9zUmVsZWFzZSk7XG4gICAgfSBjYXRjaCB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuICB9XG5cbiAgLyoqXG4gICAqIFRyeSBlYWNoIGJyb3dzZXItb3BlbmluZyBjb21tYW5kIGluIG9yZGVyLiBBZHZhbmNlcyB0byB0aGUgbmV4dCBmYWxsYmFjayBvbjpcbiAgICogICAtIFNwYXduIGZhaWx1cmUgKEVOT0VOVCAvIHRoZSBjb21tYW5kIG5vdCBiZWluZyBpbnN0YWxsZWQpXG4gICAqICAgLSBTeW5jaHJvbm91cyB0aHJvdyBmcm9tIHNwYXduKClcbiAgICogICAtIFJ1bnRpbWUgZmFpbHVyZSB3aGVyZSB0aGUgY29tbWFuZCBzcGF3bnMgYnV0IGV4aXRzIHdpdGggYSBub24temVyb1xuICAgKiAgICAgY29kZSAoZS5nLiB3c2x2aWV3IHdoZW4gV1NMIGludGVyb3AgaXMgZGlzYWJsZWQsIHhkZy1vcGVuIHdoZW4gbm9cbiAgICogICAgIGRlZmF1bHQgYnJvd3NlciBpcyByZWdpc3RlcmVkLCBjbWQuZXhlIGZhaWxpbmcgdG8gbGF1bmNoIHN0YXJ0KVxuICAgKiAgIC0gUHJvY2VzcyB0ZXJtaW5hdGVkIGJ5IHNpZ25hbFxuICAgKlxuICAgKiBTdGF5cyBvbiB0aGUgY3VycmVudCBjb21tYW5kIHdoZW46XG4gICAqICAgLSBQcm9jZXNzIGV4aXRzIHdpdGggY29kZSAwIChzdWNjZXNzKVxuICAgKiAgIC0gUHJvY2VzcyBpcyBzdGlsbCBydW5uaW5nIGFmdGVyIDMgc2Vjb25kcyAoYXNzdW1lZCBzdWNjZXNzIOKAlCBvcGVuZXJcbiAgICogICAgIGlzIGRvaW5nIHJlYWwgd29yayBsaWtlIGxhdW5jaGluZyBhIHNsb3cgYXBwLCBub3QgaHVuZylcbiAgICpcbiAgICogSWYgYWxsIGF0dGVtcHRzIGV4aGF1c3QsIGxvZ3MgYXQgZGVidWcgbGV2ZWwg4oCUIHRoZSB1c2VyIHN0aWxsIGhhcyB0aGVcbiAgICogc2lnbi1pbiBVUkwgcHJpbnRlZCB0byBzdGRvdXQgYXMgYSBjb3B5LXBhc3RlIGZhbGxiYWNrLlxuICAgKi9cbiAgcHJpdmF0ZSB0cnlCcm93c2VyQ29tbWFuZChcbiAgICBhdHRlbXB0czogQXJyYXk8eyBjbWQ6IHN0cmluZzsgZml4ZWRBcmdzOiBzdHJpbmdbXSB9PixcbiAgICB1cmw6IHN0cmluZyxcbiAgICBpbmRleDogbnVtYmVyXG4gICk6IHZvaWQge1xuICAgIGlmIChpbmRleCA+PSBhdHRlbXB0cy5sZW5ndGgpIHtcbiAgICAgIGxvZ2dlci5kZWJ1ZyhcbiAgICAgICAgJ1tBdXRoU2VydmljZV0gTm8gYnJvd3Nlci1vcGVuaW5nIGNvbW1hbmQgc3VjY2VlZGVkLiAnICtcbiAgICAgICAgICAnVXNlciBtdXN0IG9wZW4gdGhlIHNpZ24taW4gVVJMIG1hbnVhbGx5IChwcmludGVkIHRvIHN0ZG91dCBhYm92ZSkuJ1xuICAgICAgKTtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBjb25zdCBhdHRlbXB0ID0gYXR0ZW1wdHNbaW5kZXhdO1xuICAgIGNvbnN0IGFyZ3MgPSBbLi4uYXR0ZW1wdC5maXhlZEFyZ3MsIHVybF07XG4gICAgbGV0IHNldHRsZWQgPSBmYWxzZTtcblxuICAgIC8vIEhlbHBlciB0byBhZHZhbmNlIGV4YWN0bHkgb25jZSDigJQgZ3VhcmRzIGFnYWluc3QgYm90aCAnZXJyb3InIGFuZCAnZXhpdCdcbiAgICAvLyBldmVudHMgZmlyaW5nIGZvciB0aGUgc2FtZSBjaGlsZCwgb3IgdGhlIHRpbWVvdXQgcmFjaW5nIHdpdGggYW4gZXZlbnQuXG4gICAgY29uc3QgYWR2YW5jZSA9IChyZWFzb246IHN0cmluZykgPT4ge1xuICAgICAgaWYgKHNldHRsZWQpIHJldHVybjtcbiAgICAgIHNldHRsZWQgPSB0cnVlO1xuICAgICAgbG9nZ2VyLmRlYnVnKFxuICAgICAgICBgW0F1dGhTZXJ2aWNlXSBCcm93c2VyIGNvbW1hbmQgJyR7YXR0ZW1wdC5jbWR9JyAke3JlYXNvbn07IHRyeWluZyBuZXh0IGZhbGxiYWNrYFxuICAgICAgKTtcbiAgICAgIHRoaXMudHJ5QnJvd3NlckNvbW1hbmQoYXR0ZW1wdHMsIHVybCwgaW5kZXggKyAxKTtcbiAgICB9O1xuXG4gICAgLy8gSGVscGVyIHRvIG1hcmsgdGhpcyBhdHRlbXB0IGFzIHN1Y2Nlc3NmdWwgc28gbm8gZnV0dXJlIGV2ZW50IGhhbmRsZXJzXG4gICAgLy8gZmlyZSB0cnlCcm93c2VyQ29tbWFuZCBvbiB0aGUgbmV4dCBpbmRleC5cbiAgICBjb25zdCBzdG9wVHJ5aW5nID0gKHJlYXNvbjogc3RyaW5nKSA9PiB7XG4gICAgICBpZiAoc2V0dGxlZCkgcmV0dXJuO1xuICAgICAgc2V0dGxlZCA9IHRydWU7XG4gICAgICBsb2dnZXIuZGVidWcoYFtBdXRoU2VydmljZV0gQnJvd3NlciBjb21tYW5kICcke2F0dGVtcHQuY21kfScgJHtyZWFzb259YCk7XG4gICAgfTtcblxuICAgIGxldCBjaGlsZDtcbiAgICB0cnkge1xuICAgICAgY2hpbGQgPSBzcGF3bihhdHRlbXB0LmNtZCwgYXJncywge1xuICAgICAgICBkZXRhY2hlZDogdHJ1ZSxcbiAgICAgICAgc3RkaW86ICdpZ25vcmUnLFxuICAgICAgfSk7XG4gICAgfSBjYXRjaCAoZXJyOiBhbnkpIHtcbiAgICAgIGFkdmFuY2UoYHRocmV3IHN5bmNocm9ub3VzbHk6ICR7ZXJyPy5tZXNzYWdlIHx8IGVycn1gKTtcbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICAvLyBGaXJlcyB3aGVuIHRoZSBPUyBjYW5ub3Qgc3Bhd24gdGhlIHByb2Nlc3MgYXQgYWxsIChtb3N0IGNvbW1vbmx5IEVOT0VOVFxuICAgIC8vIGZvciBcImNvbW1hbmQgbm90IGZvdW5kXCIpLiBUaGUgcHJvY2VzcyBuZXZlciBhY3R1YWxseSByYW4uXG4gICAgY2hpbGQub24oJ2Vycm9yJywgKGVycjogYW55KSA9PiB7XG4gICAgICBhZHZhbmNlKGBmYWlsZWQgdG8gc3Bhd246ICR7ZXJyPy5tZXNzYWdlIHx8IGVycn1gKTtcbiAgICB9KTtcblxuICAgIC8vIEZpcmVzIGFmdGVyIHRoZSBzcGF3bmVkIHByb2Nlc3MgZXhpdHMuIGNvZGUgPT09IDAgaXMgc3VjY2VzcyAob3BlbmVyXG4gICAgLy8gZGVsZWdhdGVkIGNsZWFubHkgYW5kIHJldHVybmVkKS4gTm9uLXplcm8gY29kZSBvciBhIHNpZ25hbCBtZWFucyB0aGVcbiAgICAvLyBvcGVuZXIgZmFpbGVkIGF0IHJ1bnRpbWUg4oCUIHdlIGZhbGwgdGhyb3VnaCB0byB0aGUgbmV4dCBmYWxsYmFjay5cbiAgICBjaGlsZC5vbignZXhpdCcsIChjb2RlOiBudW1iZXIgfCBudWxsLCBzaWduYWw6IHN0cmluZyB8IG51bGwpID0+IHtcbiAgICAgIGlmIChjb2RlID09PSAwKSB7XG4gICAgICAgIHN0b3BUcnlpbmcoJ2V4aXRlZCBzdWNjZXNzZnVsbHknKTtcbiAgICAgIH0gZWxzZSBpZiAoc2lnbmFsKSB7XG4gICAgICAgIGFkdmFuY2UoYHRlcm1pbmF0ZWQgYnkgc2lnbmFsICR7c2lnbmFsfWApO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgYWR2YW5jZShgZXhpdGVkIHdpdGggY29kZSAke2NvZGV9YCk7XG4gICAgICB9XG4gICAgfSk7XG5cbiAgICAvLyBJZiB0aGUgb3BlbmVyIGlzIHN0aWxsIHJ1bm5pbmcgYWZ0ZXIgMyBzZWNvbmRzLCBpdCBpcyBhbG1vc3QgY2VydGFpbmx5XG4gICAgLy8gZG9pbmcgcmVhbCB3b3JrIChsYXVuY2hpbmcgYSBzbG93IGJyb3dzZXIsIHdhaXRpbmcgZm9yIFdpbmRvd3MgaW50ZXJvcClcbiAgICAvLyByYXRoZXIgdGhhbiBodW5nLiBTdG9wIHRyeWluZyBtb3JlIGZhbGxiYWNrcyBzbyB3ZSBkb24ndCBzcGF3biBtdWx0aXBsZVxuICAgIC8vIGJyb3dzZXJzLiBUaGUgY2hpbGQga2VlcHMgcnVubmluZyBpbmRlcGVuZGVudGx5IGJlY2F1c2UgZGV0YWNoZWQ6dHJ1ZS5cbiAgICBjb25zdCB0aW1lciA9IHNldFRpbWVvdXQoKCkgPT4ge1xuICAgICAgc3RvcFRyeWluZygnc3RpbGwgcnVubmluZyBhZnRlciAzcywgYXNzdW1pbmcgc3VjY2VzcycpO1xuICAgIH0sIDMwMDApO1xuICAgIC8vIERvbid0IGxldCB0aGUgdGltZXIgaXRzZWxmIGtlZXAgdGhlIE5vZGUgZXZlbnQgbG9vcCBhbGl2ZS5cbiAgICB0aW1lci51bnJlZigpO1xuXG4gICAgLy8gRGV0YWNoIHRoZSBjaGlsZCBmcm9tIHRoZSBwYXJlbnQncyBldmVudCBsb29wIHNvIE5vZGUgZG9lc24ndCBibG9ja1xuICAgIC8vIG9uIGl0cyBleGl0LiBFeGl0IGhhbmRsZXIgc3RpbGwgZmlyZXMgaWYvd2hlbiB0aGUgY2hpbGQgYWN0dWFsbHkgZXhpdHMuXG4gICAgY2hpbGQudW5yZWYoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZW5lcmF0ZSBzdGF0ZSBmb3IgQ1NSRiBwcm90ZWN0aW9uXG4gICAqL1xuICBwcml2YXRlIGdlbmVyYXRlU3RhdGUoKTogc3RyaW5nIHtcbiAgICByZXR1cm4gY3J5cHRvLnJhbmRvbUJ5dGVzKDMyKS50b1N0cmluZygnaGV4Jyk7XG4gIH1cblxuICAvKipcbiAgICogQnVpbGQgYXV0aG9yaXphdGlvbiBVUkxcbiAgICovXG4gIHByaXZhdGUgYnVpbGRBdXRoVXJsKHN0YXRlOiBzdHJpbmcpOiBzdHJpbmcge1xuICAgIGNvbnN0IGNvbmZpZyA9IGdldENvbmZpZygpO1xuICAgIGNvbnN0IHBhcmFtcyA9IG5ldyBVUkxTZWFyY2hQYXJhbXMoe1xuICAgICAgY2xpZW50X2lkOiBjb25maWcuYXdzLmNvZ25pdG9DbGllbnRJZCxcbiAgICAgIHJlc3BvbnNlX3R5cGU6ICdjb2RlJyxcbiAgICAgIHNjb3BlOiAnZW1haWwgb3BlbmlkIHByb2ZpbGUnLFxuICAgICAgcmVkaXJlY3RfdXJpOiBSRURJUkVDVF9VUkksXG4gICAgICBzdGF0ZTogc3RhdGUsXG4gICAgfSk7XG5cbiAgICByZXR1cm4gYGh0dHBzOi8vJHtjb25maWcuYXdzLmNvZ25pdG9Eb21haW59L29hdXRoMi9hdXRob3JpemU/JHtwYXJhbXMudG9TdHJpbmcoKX1gO1xuICB9XG5cbiAgLyoqXG4gICAqIEV4Y2hhbmdlIGF1dGhvcml6YXRpb24gY29kZSBmb3IgdG9rZW5zXG4gICAqL1xuICBwcml2YXRlIGFzeW5jIGV4Y2hhbmdlQ29kZUZvclRva2Vucyhjb2RlOiBzdHJpbmcpOiBQcm9taXNlPHtcbiAgICBhY2Nlc3NUb2tlbjogc3RyaW5nO1xuICAgIGlkVG9rZW46IHN0cmluZztcbiAgICByZWZyZXNoVG9rZW46IHN0cmluZztcbiAgICBleHBpcmVzSW46IG51bWJlcjtcbiAgfT4ge1xuICAgIGNvbnN0IGNvbmZpZyA9IGdldENvbmZpZygpO1xuICAgIGNvbnN0IHRva2VuVXJsID0gYGh0dHBzOi8vJHtjb25maWcuYXdzLmNvZ25pdG9Eb21haW59L29hdXRoMi90b2tlbmA7XG5cbiAgICBjb25zdCBwYXJhbXMgPSBuZXcgVVJMU2VhcmNoUGFyYW1zKHtcbiAgICAgIGdyYW50X3R5cGU6ICdhdXRob3JpemF0aW9uX2NvZGUnLFxuICAgICAgY2xpZW50X2lkOiBjb25maWcuYXdzLmNvZ25pdG9DbGllbnRJZCxcbiAgICAgIGNvZGU6IGNvZGUsXG4gICAgICByZWRpcmVjdF91cmk6IFJFRElSRUNUX1VSSSxcbiAgICB9KTtcblxuICAgIGNvbnN0IHJlc3BvbnNlID0gYXdhaXQgZmV0Y2hXaXRoRGlhZ25vc3RpY3MoXG4gICAgICB0b2tlblVybCxcbiAgICAgIHtcbiAgICAgICAgbWV0aG9kOiAnUE9TVCcsXG4gICAgICAgIGhlYWRlcnM6IHsgJ0NvbnRlbnQtVHlwZSc6ICdhcHBsaWNhdGlvbi94LXd3dy1mb3JtLXVybGVuY29kZWQnIH0sXG4gICAgICAgIGJvZHk6IHBhcmFtcy50b1N0cmluZygpLFxuICAgICAgfSxcbiAgICAgICdUb2tlbiBleGNoYW5nZSdcbiAgICApO1xuXG4gICAgaWYgKCFyZXNwb25zZS5vaykge1xuICAgICAgY29uc3QgZXJyb3JUZXh0ID0gYXdhaXQgcmVzcG9uc2UudGV4dCgpO1xuICAgICAgdGhyb3cgbmV3IEVycm9yKGBUb2tlbiBleGNoYW5nZSBmYWlsZWQ6ICR7cmVzcG9uc2Uuc3RhdHVzfSAke2Vycm9yVGV4dH1gKTtcbiAgICB9XG5cbiAgICBjb25zdCBkYXRhID0gYXdhaXQgcmVzcG9uc2UuanNvbigpIGFzIHtcbiAgICAgIGFjY2Vzc190b2tlbjogc3RyaW5nO1xuICAgICAgaWRfdG9rZW46IHN0cmluZztcbiAgICAgIHJlZnJlc2hfdG9rZW46IHN0cmluZztcbiAgICAgIGV4cGlyZXNfaW46IG51bWJlcjtcbiAgICB9O1xuXG4gICAgcmV0dXJuIHtcbiAgICAgIGFjY2Vzc1Rva2VuOiBkYXRhLmFjY2Vzc190b2tlbixcbiAgICAgIGlkVG9rZW46IGRhdGEuaWRfdG9rZW4sXG4gICAgICByZWZyZXNoVG9rZW46IGRhdGEucmVmcmVzaF90b2tlbixcbiAgICAgIGV4cGlyZXNJbjogZGF0YS5leHBpcmVzX2luLFxuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogRGVjb2RlIEpXVCBwYXlsb2FkXG4gICAqL1xuICBwcml2YXRlIGRlY29kZUp3dCh0b2tlbjogc3RyaW5nKTogYW55IHtcbiAgICBjb25zdCBwYXJ0cyA9IHRva2VuLnNwbGl0KCcuJyk7XG4gICAgaWYgKHBhcnRzLmxlbmd0aCAhPT0gMykgdGhyb3cgbmV3IEVycm9yKCdJbnZhbGlkIEpXVCcpO1xuICAgIHJldHVybiBKU09OLnBhcnNlKEJ1ZmZlci5mcm9tKHBhcnRzWzFdLCAnYmFzZTY0JykudG9TdHJpbmcoJ3V0Zi04JykpO1xuICB9XG5cbiAgLyoqXG4gICAqIFJlZnJlc2ggdG9rZW5zXG4gICAqL1xuICBhc3luYyByZWZyZXNoVG9rZW5zKHJlZnJlc2hUb2tlbjogc3RyaW5nKTogUHJvbWlzZTx7XG4gICAgYWNjZXNzVG9rZW46IHN0cmluZztcbiAgICBpZFRva2VuOiBzdHJpbmc7XG4gICAgZXhwaXJlc0luOiBudW1iZXI7XG4gIH0+IHtcbiAgICBjb25zdCBjb25maWcgPSBnZXRDb25maWcoKTtcbiAgICBjb25zdCB0b2tlblVybCA9IGBodHRwczovLyR7Y29uZmlnLmF3cy5jb2duaXRvRG9tYWlufS9vYXV0aDIvdG9rZW5gO1xuXG4gICAgY29uc3QgcGFyYW1zID0gbmV3IFVSTFNlYXJjaFBhcmFtcyh7XG4gICAgICBncmFudF90eXBlOiAncmVmcmVzaF90b2tlbicsXG4gICAgICBjbGllbnRfaWQ6IGNvbmZpZy5hd3MuY29nbml0b0NsaWVudElkLFxuICAgICAgcmVmcmVzaF90b2tlbjogcmVmcmVzaFRva2VuLFxuICAgIH0pO1xuXG4gICAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCBmZXRjaFdpdGhEaWFnbm9zdGljcyhcbiAgICAgIHRva2VuVXJsLFxuICAgICAge1xuICAgICAgICBtZXRob2Q6ICdQT1NUJyxcbiAgICAgICAgaGVhZGVyczogeyAnQ29udGVudC1UeXBlJzogJ2FwcGxpY2F0aW9uL3gtd3d3LWZvcm0tdXJsZW5jb2RlZCcgfSxcbiAgICAgICAgYm9keTogcGFyYW1zLnRvU3RyaW5nKCksXG4gICAgICB9LFxuICAgICAgJ1Rva2VuIHJlZnJlc2gnXG4gICAgKTtcblxuICAgIGlmICghcmVzcG9uc2Uub2spIHtcbiAgICAgIHRocm93IG5ldyBFcnJvcihgVG9rZW4gcmVmcmVzaCBmYWlsZWQ6ICR7cmVzcG9uc2Uuc3RhdHVzfWApO1xuICAgIH1cblxuICAgIGNvbnN0IGRhdGEgPSBhd2FpdCByZXNwb25zZS5qc29uKCkgYXMge1xuICAgICAgYWNjZXNzX3Rva2VuOiBzdHJpbmc7XG4gICAgICBpZF90b2tlbjogc3RyaW5nO1xuICAgICAgZXhwaXJlc19pbjogbnVtYmVyO1xuICAgIH07XG5cbiAgICByZXR1cm4ge1xuICAgICAgYWNjZXNzVG9rZW46IGRhdGEuYWNjZXNzX3Rva2VuLFxuICAgICAgaWRUb2tlbjogZGF0YS5pZF90b2tlbixcbiAgICAgIGV4cGlyZXNJbjogZGF0YS5leHBpcmVzX2luLFxuICAgIH07XG4gIH1cblxuICAvKipcbiAgICogTG9naW4gdmlhIE9BdXRoIGJyb3dzZXIgZmxvd1xuICAgKi9cbiAgYXN5bmMgbG9naW4oKTogUHJvbWlzZTxUb2tlbkRhdGEgfCBudWxsPiB7XG4gICAgLy8gQ2hlY2sgaWYgYWxyZWFkeSBsb2dnZWQgaW5cbiAgICBjb25zdCBleGlzdGluZyA9IGF3YWl0IGtleWNoYWluTWFuYWdlci5nZXRUb2tlbnMoZ2V0RW52aXJvbm1lbnQoKSk7XG4gICAgaWYgKGV4aXN0aW5nICYmICFrZXljaGFpbk1hbmFnZXIuaXNUb2tlbkV4cGlyZWQoZXhpc3RpbmcpKSB7XG4gICAgICByZXR1cm4gZXhpc3Rpbmc7XG4gICAgfVxuXG4gICAgY29uc3Qgc3RhdGUgPSB0aGlzLmdlbmVyYXRlU3RhdGUoKTtcbiAgICBjb25zdCBhdXRoVXJsID0gdGhpcy5idWlsZEF1dGhVcmwoc3RhdGUpO1xuXG4gICAgcmV0dXJuIG5ldyBQcm9taXNlKChyZXNvbHZlLCByZWplY3QpID0+IHtcbiAgICAgIGNvbnN0IHNlcnZlciA9IGh0dHAuY3JlYXRlU2VydmVyKGFzeW5jIChyZXEsIHJlcykgPT4ge1xuICAgICAgICBpZiAoIXJlcS51cmw/LnN0YXJ0c1dpdGgoQ0FMTEJBQ0tfUEFUSCkpIHtcbiAgICAgICAgICByZXMud3JpdGVIZWFkKDQwNCk7XG4gICAgICAgICAgcmVzLmVuZCgnTm90IGZvdW5kJyk7XG4gICAgICAgICAgcmV0dXJuO1xuICAgICAgICB9XG5cbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICBjb25zdCB1cmwgPSBuZXcgVVJMKHJlcS51cmwsIGBodHRwOi8vbG9jYWxob3N0OiR7Q0FMTEJBQ0tfUE9SVH1gKTtcbiAgICAgICAgICBjb25zdCBjb2RlID0gdXJsLnNlYXJjaFBhcmFtcy5nZXQoJ2NvZGUnKTtcbiAgICAgICAgICBjb25zdCByZXR1cm5lZFN0YXRlID0gdXJsLnNlYXJjaFBhcmFtcy5nZXQoJ3N0YXRlJyk7XG4gICAgICAgICAgY29uc3QgZXJyb3IgPSB1cmwuc2VhcmNoUGFyYW1zLmdldCgnZXJyb3InKTtcblxuICAgICAgICAgIGlmIChlcnJvcikge1xuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBPQXV0aCBlcnJvcjogJHtlcnJvcn1gKTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBpZiAocmV0dXJuZWRTdGF0ZSAhPT0gc3RhdGUpIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcignU3RhdGUgbWlzbWF0Y2gnKTtcbiAgICAgICAgICB9XG5cbiAgICAgICAgICBpZiAoIWNvZGUpIHtcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcignTm8gYXV0aG9yaXphdGlvbiBjb2RlJyk7XG4gICAgICAgICAgfVxuXG4gICAgICAgICAgY29uc3QgdG9rZW5zID0gYXdhaXQgdGhpcy5leGNoYW5nZUNvZGVGb3JUb2tlbnMoY29kZSk7XG4gICAgICAgICAgY29uc3QgaWRQYXlsb2FkID0gdGhpcy5kZWNvZGVKd3QodG9rZW5zLmlkVG9rZW4pO1xuXG4gICAgICAgICAgY29uc3QgdG9rZW5EYXRhOiBUb2tlbkRhdGEgPSB7XG4gICAgICAgICAgICBhY2Nlc3NUb2tlbjogdG9rZW5zLmFjY2Vzc1Rva2VuLFxuICAgICAgICAgICAgaWRUb2tlbjogdG9rZW5zLmlkVG9rZW4sXG4gICAgICAgICAgICByZWZyZXNoVG9rZW46IHRva2Vucy5yZWZyZXNoVG9rZW4sXG4gICAgICAgICAgICBleHBpcmVzQXQ6IERhdGUubm93KCkgKyAodG9rZW5zLmV4cGlyZXNJbiAqIDEwMDApLFxuICAgICAgICAgICAgdXNlcklkOiBpZFBheWxvYWQuc3ViLFxuICAgICAgICAgICAgZW1haWw6IGlkUGF5bG9hZC5lbWFpbCB8fCAndW5rbm93bicsXG4gICAgICAgICAgfTtcblxuICAgICAgICAgIGF3YWl0IGtleWNoYWluTWFuYWdlci5zZXRUb2tlbnModG9rZW5EYXRhLCBnZXRFbnZpcm9ubWVudCgpKTtcblxuICAgICAgICAgIHJlcy53cml0ZUhlYWQoMjAwLCB7ICdDb250ZW50LVR5cGUnOiAndGV4dC9odG1sOyBjaGFyc2V0PXV0Zi04JyB9KTtcbiAgICAgICAgICByZXMuZW5kKGBcbiAgICAgICAgICAgIDwhRE9DVFlQRSBodG1sPlxuICAgICAgICAgICAgPGh0bWw+XG4gICAgICAgICAgICA8aGVhZD48dGl0bGU+U3VjY2VzczwvdGl0bGU+PC9oZWFkPlxuICAgICAgICAgICAgPGJvZHkgc3R5bGU9XCJmb250LWZhbWlseTogc3lzdGVtLXVpOyBtYXgtd2lkdGg6IDYwMHB4OyBtYXJnaW46IDUwcHggYXV0bzsgdGV4dC1hbGlnbjogY2VudGVyO1wiPlxuICAgICAgICAgICAgICA8aDEgc3R5bGU9XCJjb2xvcjogIzIyYzU1ZTtcIj4mIzEwMDAzOyBBdXRoZW50aWNhdGlvbiBTdWNjZXNzZnVsPC9oMT5cbiAgICAgICAgICAgICAgPHA+WW91IGNhbiBjbG9zZSB0aGlzIHdpbmRvdy48L3A+XG4gICAgICAgICAgICA8L2JvZHk+XG4gICAgICAgICAgICA8L2h0bWw+XG4gICAgICAgICAgYCk7XG5cbiAgICAgICAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgICAgIHNlcnZlci5jbG9zZSgoKSA9PiByZXNvbHZlKHRva2VuRGF0YSkpO1xuICAgICAgICAgIH0sIDUwMCk7XG4gICAgICAgIH0gY2F0Y2ggKGVycjogYW55KSB7XG4gICAgICAgICAgLy8gSFRNTC1lc2NhcGUgdGhlIGVycm9yIG1lc3NhZ2Ugc28gbmV3bGluZXMgYW5kIHNwZWNpYWwgY2hhcnNcbiAgICAgICAgICAvLyByZW5kZXIgc2FmZWx5LiBNdWx0aS1saW5lIGRpYWdub3N0aWMgbWVzc2FnZXMgKGZyb21cbiAgICAgICAgICAvLyBmZXRjaFdpdGhEaWFnbm9zdGljcykgbmVlZCA8cHJlPiB0byBwcmVzZXJ2ZSBmb3JtYXR0aW5nLlxuICAgICAgICAgIGNvbnN0IHNhZmVNZXNzYWdlID0gU3RyaW5nKGVycj8ubWVzc2FnZSB8fCBlcnIpXG4gICAgICAgICAgICAucmVwbGFjZSgvJi9nLCAnJmFtcDsnKVxuICAgICAgICAgICAgLnJlcGxhY2UoLzwvZywgJyZsdDsnKVxuICAgICAgICAgICAgLnJlcGxhY2UoLz4vZywgJyZndDsnKTtcblxuICAgICAgICAgIHJlcy53cml0ZUhlYWQoNDAwLCB7ICdDb250ZW50LVR5cGUnOiAndGV4dC9odG1sOyBjaGFyc2V0PXV0Zi04JyB9KTtcbiAgICAgICAgICByZXMuZW5kKGBcbiAgICAgICAgICAgIDwhRE9DVFlQRSBodG1sPlxuICAgICAgICAgICAgPGh0bWw+XG4gICAgICAgICAgICA8aGVhZD48dGl0bGU+RXJyb3I8L3RpdGxlPjwvaGVhZD5cbiAgICAgICAgICAgIDxib2R5IHN0eWxlPVwiZm9udC1mYW1pbHk6IHN5c3RlbS11aTsgbWF4LXdpZHRoOiA3MjBweDsgbWFyZ2luOiA1MHB4IGF1dG87IHBhZGRpbmc6IDAgMTZweDtcIj5cbiAgICAgICAgICAgICAgPGgxIHN0eWxlPVwiY29sb3I6ICNlZjQ0NDQ7IHRleHQtYWxpZ246IGNlbnRlcjtcIj4mIzEwMDA3OyBBdXRoZW50aWNhdGlvbiBGYWlsZWQ8L2gxPlxuICAgICAgICAgICAgICA8cHJlIHN0eWxlPVwiYmFja2dyb3VuZDogI2Y0ZjRmNTsgcGFkZGluZzogMTZweDsgYm9yZGVyLXJhZGl1czogOHB4OyB3aGl0ZS1zcGFjZTogcHJlLXdyYXA7IHdvcmQtd3JhcDogYnJlYWstd29yZDsgZm9udC1zaXplOiAxM3B4OyBsaW5lLWhlaWdodDogMS41O1wiPiR7c2FmZU1lc3NhZ2V9PC9wcmU+XG4gICAgICAgICAgICAgIDxwIHN0eWxlPVwidGV4dC1hbGlnbjogY2VudGVyOyBjb2xvcjogIzcxNzE3YTsgbWFyZ2luLXRvcDogMjRweDtcIj5Zb3UgY2FuIGNsb3NlIHRoaXMgd2luZG93IGFuZCB0cnkgYWdhaW4gaW4geW91ciB0ZXJtaW5hbC48L3A+XG4gICAgICAgICAgICA8L2JvZHk+XG4gICAgICAgICAgICA8L2h0bWw+XG4gICAgICAgICAgYCk7XG5cbiAgICAgICAgICBzZXRUaW1lb3V0KCgpID0+IHtcbiAgICAgICAgICAgIHNlcnZlci5jbG9zZSgoKSA9PiByZWplY3QoZXJyKSk7XG4gICAgICAgICAgfSwgNTAwKTtcbiAgICAgICAgfVxuICAgICAgfSk7XG5cbiAgICAgIHNlcnZlci5vbignZXJyb3InLCAoZXJyOiBhbnkpID0+IHtcbiAgICAgICAgaWYgKGVyci5jb2RlID09PSAnRUFERFJJTlVTRScpIHtcbiAgICAgICAgICByZWplY3QobmV3IEVycm9yKGBQb3J0ICR7Q0FMTEJBQ0tfUE9SVH0gaXMgaW4gdXNlYCkpO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIHJlamVjdChlcnIpO1xuICAgICAgICB9XG4gICAgICB9KTtcblxuICAgICAgc2VydmVyLmxpc3RlbihDQUxMQkFDS19QT1JULCAnbG9jYWxob3N0JywgKCkgPT4ge1xuICAgICAgICBsb2dnZXIuaW5mbygnW0F1dGhTZXJ2aWNlXSBDYWxsYmFjayBzZXJ2ZXIgc3RhcnRlZCcpO1xuICAgICAgICB0aGlzLm9wZW5Ccm93c2VyKGF1dGhVcmwpO1xuICAgICAgfSk7XG5cbiAgICAgIC8vIFRpbWVvdXRcbiAgICAgIHNldFRpbWVvdXQoKCkgPT4ge1xuICAgICAgICBzZXJ2ZXIuY2xvc2UoKCkgPT4gcmVqZWN0KG5ldyBFcnJvcignTG9naW4gdGltZW91dCcpKSk7XG4gICAgICB9LCAyICogNjAgKiAxMDAwKTtcbiAgICB9KTtcbiAgfVxuXG4gIC8qKlxuICAgKiBMb2dvdXRcbiAgICovXG4gIGFzeW5jIGxvZ291dCgpOiBQcm9taXNlPGJvb2xlYW4+IHtcbiAgICBjb25zdCBjb25maWcgPSBnZXRDb25maWcoKTtcbiAgICBjb25zdCBkZWxldGVkID0gYXdhaXQga2V5Y2hhaW5NYW5hZ2VyLmRlbGV0ZVRva2VucyhnZXRFbnZpcm9ubWVudCgpKTtcblxuICAgIGlmIChkZWxldGVkKSB7XG4gICAgICAvLyBTdGFydCB0ZW1wb3Jhcnkgc2VydmVyIHRvIGhhbmRsZSBzaWdub3V0IGNhbGxiYWNrXG4gICAgICByZXR1cm4gbmV3IFByb21pc2UoKHJlc29sdmUpID0+IHtcbiAgICAgICAgY29uc3Qgc2VydmVyID0gaHR0cC5jcmVhdGVTZXJ2ZXIoKHJlcSwgcmVzKSA9PiB7XG4gICAgICAgICAgaWYgKHJlcS51cmw/LnN0YXJ0c1dpdGgoJy9zaWdub3V0JykpIHtcbiAgICAgICAgICAgIHJlcy53cml0ZUhlYWQoMjAwLCB7ICdDb250ZW50LVR5cGUnOiAndGV4dC9odG1sOyBjaGFyc2V0PXV0Zi04JyB9KTtcbiAgICAgICAgICAgIHJlcy5lbmQoYFxuICAgICAgICAgICAgICA8IURPQ1RZUEUgaHRtbD5cbiAgICAgICAgICAgICAgPGh0bWw+XG4gICAgICAgICAgICAgIDxoZWFkPjx0aXRsZT5TaWduZWQgT3V0PC90aXRsZT48L2hlYWQ+XG4gICAgICAgICAgICAgIDxib2R5IHN0eWxlPVwiZm9udC1mYW1pbHk6IHN5c3RlbS11aTsgbWF4LXdpZHRoOiA2MDBweDsgbWFyZ2luOiA1MHB4IGF1dG87IHRleHQtYWxpZ246IGNlbnRlcjtcIj5cbiAgICAgICAgICAgICAgICA8aDEgc3R5bGU9XCJjb2xvcjogIzIyYzU1ZTtcIj4mIzEwMDAzOyBTaWduZWQgT3V0PC9oMT5cbiAgICAgICAgICAgICAgICA8cD5Zb3UgY2FuIGNsb3NlIHRoaXMgd2luZG93LjwvcD5cbiAgICAgICAgICAgICAgPC9ib2R5PlxuICAgICAgICAgICAgICA8L2h0bWw+XG4gICAgICAgICAgICBgKTtcblxuICAgICAgICAgICAgc2V0VGltZW91dCgoKSA9PiB7XG4gICAgICAgICAgICAgIHNlcnZlci5jbG9zZSgoKSA9PiByZXNvbHZlKHRydWUpKTtcbiAgICAgICAgICAgIH0sIDUwMCk7XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIHJlcy53cml0ZUhlYWQoNDA0KTtcbiAgICAgICAgICAgIHJlcy5lbmQoJ05vdCBmb3VuZCcpO1xuICAgICAgICAgIH1cbiAgICAgICAgfSk7XG5cbiAgICAgICAgc2VydmVyLm9uKCdlcnJvcicsICgpID0+IHtcbiAgICAgICAgICByZXNvbHZlKHRydWUpOyAvLyBTdGlsbCByZXR1cm4gdHJ1ZSBzaW5jZSB0b2tlbnMgd2VyZSBkZWxldGVkXG4gICAgICAgIH0pO1xuXG4gICAgICAgIHNlcnZlci5saXN0ZW4oQ0FMTEJBQ0tfUE9SVCwgJ2xvY2FsaG9zdCcsICgpID0+IHtcbiAgICAgICAgICBjb25zdCBsb2dvdXRVcmwgPSBgaHR0cHM6Ly8ke2NvbmZpZy5hd3MuY29nbml0b0RvbWFpbn0vbG9nb3V0P2AgK1xuICAgICAgICAgICAgYGNsaWVudF9pZD0ke2NvbmZpZy5hd3MuY29nbml0b0NsaWVudElkfSZgICtcbiAgICAgICAgICAgIGBsb2dvdXRfdXJpPSR7ZW5jb2RlVVJJQ29tcG9uZW50KFJFRElSRUNUX1VSSS5yZXBsYWNlKCcvY2FsbGJhY2snLCAnL3NpZ25vdXQnKSl9YDtcblxuICAgICAgICAgIHRoaXMub3BlbkJyb3dzZXIobG9nb3V0VXJsKTtcbiAgICAgICAgfSk7XG5cbiAgICAgICAgLy8gVGltZW91dCBhZnRlciAzMCBzZWNvbmRzXG4gICAgICAgIHNldFRpbWVvdXQoKCkgPT4ge1xuICAgICAgICAgIHNlcnZlci5jbG9zZSgoKSA9PiByZXNvbHZlKHRydWUpKTtcbiAgICAgICAgfSwgMzAgKiAxMDAwKTtcbiAgICAgIH0pO1xuICAgIH1cblxuICAgIHJldHVybiBkZWxldGVkO1xuICB9XG5cbiAgLyoqXG4gICAqIEdldCBjdXJyZW50IGF1dGggc3RhdHVzXG4gICAqL1xuICBhc3luYyBnZXRTdGF0dXMoKTogUHJvbWlzZTx7IGF1dGhlbnRpY2F0ZWQ6IGJvb2xlYW47IHRva2Vucz86IFRva2VuRGF0YSB9PiB7XG4gICAgY29uc3QgdG9rZW5zID0gYXdhaXQga2V5Y2hhaW5NYW5hZ2VyLmdldFRva2VucyhnZXRFbnZpcm9ubWVudCgpKTtcbiAgICBpZiAoIXRva2Vucykge1xuICAgICAgcmV0dXJuIHsgYXV0aGVudGljYXRlZDogZmFsc2UgfTtcbiAgICB9XG5cbiAgICByZXR1cm4ge1xuICAgICAgYXV0aGVudGljYXRlZDogIWtleWNoYWluTWFuYWdlci5pc1Rva2VuRXhwaXJlZCh0b2tlbnMpLFxuICAgICAgdG9rZW5zLFxuICAgIH07XG4gIH1cbn1cblxuZXhwb3J0IGNvbnN0IGF1dGhTZXJ2aWNlID0gQXV0aFNlcnZpY2UuZ2V0SW5zdGFuY2UoKTtcbiJdfQ==