@nlabs/lex 1.48.7 → 1.49.1

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 (105) hide show
  1. package/.storybook/main.ts +9 -2
  2. package/.vscode/settings.json +1 -6
  3. package/README.md +249 -0
  4. package/eslint.config.mjs +24 -0
  5. package/examples/lex.config.js +18 -8
  6. package/examples/serverless-example/README.md +109 -0
  7. package/examples/serverless-example/dist/handlers/echo.js +15 -0
  8. package/examples/serverless-example/dist/handlers/graphql.js +137 -0
  9. package/examples/serverless-example/dist/handlers/hello.js +15 -0
  10. package/examples/serverless-example/dist/handlers/test.js +17 -0
  11. package/examples/serverless-example/dist/handlers/websocket.js +14 -0
  12. package/examples/serverless-example/lex.config.mjs +74 -0
  13. package/jest.config.mjs +13 -12
  14. package/{dist → lib}/LexConfig.d.ts +7 -6
  15. package/lib/LexConfig.js +268 -0
  16. package/lib/commands/ai/ai.js +303 -0
  17. package/{dist → lib}/commands/build/build.d.ts +3 -0
  18. package/lib/commands/build/build.js +494 -0
  19. package/{dist → lib}/commands/clean/clean.js +1 -1
  20. package/lib/commands/compile/compile.js +241 -0
  21. package/lib/commands/copy/copy.js +38 -0
  22. package/{dist → lib}/commands/create/create.js +1 -1
  23. package/{dist → lib}/commands/dev/dev.d.ts +2 -0
  24. package/lib/commands/dev/dev.js +286 -0
  25. package/{dist → lib}/commands/init/init.js +1 -1
  26. package/lib/commands/lint/lint.js +962 -0
  27. package/{dist → lib}/commands/migrate/migrate.js +1 -1
  28. package/lib/commands/publish/publish.js +104 -0
  29. package/lib/commands/serverless/serverless.d.ts +17 -0
  30. package/lib/commands/serverless/serverless.js +662 -0
  31. package/lib/commands/storybook/storybook.js +249 -0
  32. package/lib/commands/test/test.js +428 -0
  33. package/lib/commands/update/update.js +128 -0
  34. package/{dist → lib}/create/changelog.js +1 -1
  35. package/{dist → lib}/index.d.ts +1 -0
  36. package/{dist → lib}/index.js +2 -1
  37. package/lib/lex.js +73 -0
  38. package/lib/utils/aiService.d.ts +9 -0
  39. package/lib/utils/aiService.js +299 -0
  40. package/{dist → lib}/utils/app.d.ts +3 -0
  41. package/lib/utils/app.js +296 -0
  42. package/{dist → lib}/utils/file.d.ts +7 -3
  43. package/lib/utils/file.js +229 -0
  44. package/lib/utils/translations.d.ts +1 -0
  45. package/lib/utils/translations.js +74 -0
  46. package/package.json +60 -54
  47. package/postcss.config.js +5 -3
  48. package/tsconfig.build.json +2 -2
  49. package/webpack.config.js +229 -39
  50. package/dist/LexConfig.js +0 -287
  51. package/dist/commands/ai/ai.js +0 -303
  52. package/dist/commands/build/build.js +0 -404
  53. package/dist/commands/compile/compile.js +0 -234
  54. package/dist/commands/copy/copy.js +0 -38
  55. package/dist/commands/dev/dev.js +0 -74
  56. package/dist/commands/lint/lint.js +0 -993
  57. package/dist/commands/publish/publish.js +0 -104
  58. package/dist/commands/storybook/storybook.js +0 -249
  59. package/dist/commands/test/test.js +0 -429
  60. package/dist/commands/update/update.js +0 -132
  61. package/dist/lex.js +0 -70
  62. package/dist/utils/aiService.d.ts +0 -9
  63. package/dist/utils/aiService.js +0 -299
  64. package/dist/utils/app.js +0 -267
  65. package/dist/utils/file.js +0 -185
  66. package/emptyModule.js +0 -0
  67. package/eslint.config.js +0 -3
  68. /package/{dist → lib}/Button.stories.d.ts +0 -0
  69. /package/{dist → lib}/commands/ai/ai.d.ts +0 -0
  70. /package/{dist → lib}/commands/ai/index.d.ts +0 -0
  71. /package/{dist → lib}/commands/ai/index.js +0 -0
  72. /package/{dist → lib}/commands/clean/clean.d.ts +0 -0
  73. /package/{dist → lib}/commands/compile/compile.d.ts +0 -0
  74. /package/{dist → lib}/commands/config/config.d.ts +0 -0
  75. /package/{dist → lib}/commands/config/config.js +0 -0
  76. /package/{dist → lib}/commands/copy/copy.d.ts +0 -0
  77. /package/{dist → lib}/commands/create/create.d.ts +0 -0
  78. /package/{dist → lib}/commands/init/init.d.ts +0 -0
  79. /package/{dist → lib}/commands/link/link.d.ts +0 -0
  80. /package/{dist → lib}/commands/link/link.js +0 -0
  81. /package/{dist → lib}/commands/lint/autofix.d.ts +0 -0
  82. /package/{dist → lib}/commands/lint/lint.d.ts +0 -0
  83. /package/{dist → lib}/commands/migrate/migrate.d.ts +0 -0
  84. /package/{dist → lib}/commands/publish/publish.d.ts +0 -0
  85. /package/{dist → lib}/commands/storybook/storybook.d.ts +0 -0
  86. /package/{dist → lib}/commands/test/test.d.ts +0 -0
  87. /package/{dist → lib}/commands/update/update.d.ts +0 -0
  88. /package/{dist → lib}/commands/upgrade/upgrade.d.ts +0 -0
  89. /package/{dist → lib}/commands/upgrade/upgrade.js +0 -0
  90. /package/{dist → lib}/commands/versions/versions.d.ts +0 -0
  91. /package/{dist → lib}/commands/versions/versions.js +0 -0
  92. /package/{dist → lib}/create/changelog.d.ts +0 -0
  93. /package/{dist → lib}/lex.d.ts +0 -0
  94. /package/{dist → lib}/storybook/index.d.ts +0 -0
  95. /package/{dist → lib}/storybook/index.js +0 -0
  96. /package/{dist → lib}/test-react/index.d.ts +0 -0
  97. /package/{dist → lib}/test-react/index.js +0 -0
  98. /package/{dist → lib}/types.d.ts +0 -0
  99. /package/{dist → lib}/types.js +0 -0
  100. /package/{dist → lib}/utils/deepMerge.d.ts +0 -0
  101. /package/{dist → lib}/utils/deepMerge.js +0 -0
  102. /package/{dist → lib}/utils/log.d.ts +0 -0
  103. /package/{dist → lib}/utils/log.js +0 -0
  104. /package/{dist → lib}/utils/reactShim.d.ts +0 -0
  105. /package/{dist → lib}/utils/reactShim.js +0 -0
@@ -1,404 +0,0 @@
1
- import GraphqlLoaderPlugin from "@luckycatfactory/esbuild-graphql-loader";
2
- import { execa } from "execa";
3
- import { existsSync, readFileSync } from "fs";
4
- import { sync as globSync } from "glob";
5
- import { resolve as pathResolve } from "path";
6
- import { LexConfig, getTypeScriptConfigPath } from "../../LexConfig.js";
7
- import { checkLinkedModules, copyConfiguredFiles, createSpinner, removeFiles } from "../../utils/app.js";
8
- import { getDirName, relativeNodePath } from "../../utils/file.js";
9
- import { log } from "../../utils/log.js";
10
- import { aiFunction } from "../ai/ai.js";
11
- const buildWithEsBuild = async (spinner, commandOptions, callback) => {
12
- const {
13
- cliName = "Lex",
14
- format = "cjs",
15
- outputPath,
16
- quiet,
17
- sourcePath,
18
- watch
19
- } = commandOptions;
20
- const {
21
- outputFullPath,
22
- sourceFullPath,
23
- targetEnvironment,
24
- useGraphQl,
25
- useTypescript
26
- } = LexConfig.config;
27
- const sourceDir = sourcePath ? pathResolve(process.cwd(), `./${sourcePath}`) : sourceFullPath || "";
28
- const loader = {
29
- ".js": "js"
30
- };
31
- if (useTypescript) {
32
- loader[".ts"] = "ts";
33
- loader[".tsx"] = "tsx";
34
- }
35
- const plugins = [];
36
- if (useGraphQl) {
37
- plugins.push(GraphqlLoaderPlugin());
38
- }
39
- const globOptions = {
40
- cwd: sourceDir,
41
- dot: false,
42
- nodir: true,
43
- nosort: true
44
- };
45
- const tsFiles = globSync(`${sourceDir}/**/!(*.spec|*.test).ts*`, globOptions);
46
- const jsFiles = globSync(`${sourceDir}/**/!(*.spec|*.test).js`, globOptions);
47
- const sourceFiles = [...tsFiles, ...jsFiles];
48
- const packageJsonData = readFileSync(pathResolve(process.cwd(), "./package.json"));
49
- const packageJson = JSON.parse(packageJsonData.toString());
50
- const external = [
51
- ...Object.keys(packageJson.dependencies || {}),
52
- ...Object.keys(packageJson.peerDependencies || {})
53
- ];
54
- const dirName = getDirName();
55
- const dirPath = pathResolve(dirName, "../..");
56
- const outputDir = outputPath || outputFullPath || "";
57
- const esbuildPath = relativeNodePath("esbuild/bin/esbuild", dirPath);
58
- const esbuildConfig = LexConfig.config.esbuild || {};
59
- const esbuildOptions = [
60
- ...sourceFiles,
61
- "--bundle",
62
- "--color=true",
63
- `--format=${format}`,
64
- `--outdir=${outputDir}`,
65
- `--platform=${esbuildConfig.platform || "node"}`,
66
- `--target=${esbuildConfig.target || (targetEnvironment === "node" ? "node20" : "es2020")}`,
67
- `--sourcemap=${esbuildConfig.sourcemap || "inline"}`
68
- ];
69
- if (esbuildConfig.minify !== false) {
70
- esbuildOptions.push("--minify");
71
- }
72
- if (esbuildConfig.treeShaking !== false) {
73
- esbuildOptions.push("--tree-shaking=true");
74
- }
75
- if (esbuildConfig.drop && esbuildConfig.drop.length > 0) {
76
- esbuildConfig.drop.forEach((item) => {
77
- esbuildOptions.push(`--drop:${item}`);
78
- });
79
- }
80
- if (esbuildConfig.pure && esbuildConfig.pure.length > 0) {
81
- esbuildConfig.pure.forEach((item) => {
82
- esbuildOptions.push(`--pure:${item}`);
83
- });
84
- }
85
- if (esbuildConfig.legalComments) {
86
- esbuildOptions.push(`--legal-comments=${esbuildConfig.legalComments}`);
87
- }
88
- if (esbuildConfig.splitting !== false) {
89
- esbuildOptions.push("--splitting");
90
- }
91
- if (esbuildConfig.metafile) {
92
- esbuildOptions.push("--metafile");
93
- }
94
- if (esbuildConfig.banner) {
95
- Object.entries(esbuildConfig.banner).forEach(([type, content]) => {
96
- esbuildOptions.push(`--banner:${type}=${content}`);
97
- });
98
- }
99
- if (esbuildConfig.footer) {
100
- Object.entries(esbuildConfig.footer).forEach(([type, content]) => {
101
- esbuildOptions.push(`--footer:${type}=${content}`);
102
- });
103
- }
104
- if (esbuildConfig.define) {
105
- Object.entries(esbuildConfig.define).forEach(([key, value]) => {
106
- esbuildOptions.push(`--define:${key}=${value}`);
107
- });
108
- }
109
- if (external.length) {
110
- esbuildOptions.push(`--external:${external.join(",")}`);
111
- }
112
- if (plugins.length) {
113
- esbuildOptions.push(`--plugins=${plugins.join(",")}`);
114
- }
115
- if (watch) {
116
- esbuildOptions.push("--watch");
117
- }
118
- try {
119
- await execa(esbuildPath, esbuildOptions, { encoding: "utf8" });
120
- spinner.succeed("Build completed successfully!");
121
- } catch (error) {
122
- log(`
123
- ${cliName} Error: ${error.message}`, "error", quiet);
124
- if (!quiet) {
125
- console.error(error);
126
- }
127
- spinner.fail("Code build failed.");
128
- if (commandOptions.assist) {
129
- spinner.start("AI is analyzing the error...");
130
- try {
131
- await aiFunction({
132
- prompt: `Fix this esbuild error: ${error.message}
133
-
134
- Error details:
135
- ${error.toString()}`,
136
- task: "help",
137
- context: true,
138
- quiet
139
- });
140
- spinner.succeed("AI analysis complete");
141
- } catch (aiError) {
142
- spinner.fail("Could not generate AI assistance");
143
- if (!quiet) {
144
- console.error("AI assistance error:", aiError);
145
- }
146
- }
147
- }
148
- callback(1);
149
- return 1;
150
- }
151
- callback(0);
152
- return 0;
153
- };
154
- const buildWithWebpack = async (spinner, cmd, callback) => {
155
- const {
156
- analyze,
157
- cliName = "Lex",
158
- config,
159
- configName,
160
- defineProcessEnvNodeEnv,
161
- devtool,
162
- disableInterpret,
163
- entry,
164
- env,
165
- failOnWarnings,
166
- json,
167
- merge,
168
- mode,
169
- name,
170
- nodeEnv,
171
- noDevtool,
172
- noStats,
173
- noTarget,
174
- noWatch,
175
- noWatchOptionsStdin,
176
- outputPath,
177
- quiet = false,
178
- stats,
179
- target,
180
- watch,
181
- watchOptionsStdin
182
- } = cmd;
183
- let webpackConfig;
184
- const dirName = getDirName();
185
- if (config) {
186
- const isRelativeConfig = config.substr(0, 2) === "./";
187
- webpackConfig = isRelativeConfig ? pathResolve(process.cwd(), config) : config;
188
- } else {
189
- webpackConfig = pathResolve(dirName, "../../webpack.config.js");
190
- }
191
- const webpackOptions = [
192
- "--color",
193
- "--progress",
194
- "--config",
195
- webpackConfig
196
- ];
197
- if (analyze) {
198
- webpackOptions.push("--analyze");
199
- }
200
- if (configName) {
201
- webpackOptions.push("--configName", configName);
202
- }
203
- if (defineProcessEnvNodeEnv) {
204
- webpackOptions.push("--defineProcessEnvNodeEnv", defineProcessEnvNodeEnv);
205
- }
206
- if (devtool) {
207
- webpackOptions.push("--devtool", devtool);
208
- }
209
- if (disableInterpret) {
210
- webpackOptions.push("--disableInterpret");
211
- }
212
- if (entry) {
213
- webpackOptions.push("--entry", entry);
214
- }
215
- if (env) {
216
- webpackOptions.push("--env", env);
217
- }
218
- if (failOnWarnings) {
219
- webpackOptions.push("--failOnWarnings");
220
- }
221
- if (json) {
222
- webpackOptions.push("--json", json);
223
- }
224
- if (mode) {
225
- webpackOptions.push("--mode", mode);
226
- }
227
- if (merge) {
228
- webpackOptions.push("--merge");
229
- }
230
- if (name) {
231
- webpackOptions.push("--name", name);
232
- }
233
- if (noDevtool) {
234
- webpackOptions.push("--noDevtool");
235
- }
236
- if (noStats) {
237
- webpackOptions.push("--noStats");
238
- }
239
- if (noTarget) {
240
- webpackOptions.push("--noTarget");
241
- }
242
- if (noWatch) {
243
- webpackOptions.push("--noWatch");
244
- }
245
- if (noWatchOptionsStdin) {
246
- webpackOptions.push("--noWatchOptionsStdin");
247
- }
248
- if (nodeEnv) {
249
- webpackOptions.push("--nodeEnv", nodeEnv);
250
- }
251
- if (outputPath) {
252
- webpackOptions.push("--outputPath", outputPath);
253
- }
254
- if (stats) {
255
- webpackOptions.push("--stats", stats);
256
- }
257
- if (target) {
258
- webpackOptions.push("--target", target);
259
- }
260
- if (watch) {
261
- webpackOptions.push("--watch");
262
- }
263
- if (watchOptionsStdin) {
264
- webpackOptions.push("--watchOptionsStdin");
265
- }
266
- const dirPath = pathResolve(dirName, "../..");
267
- try {
268
- const webpackPath = relativeNodePath("webpack-cli/bin/cli.js", dirPath);
269
- await execa(webpackPath, webpackOptions, { encoding: "utf8", stdio: "inherit" });
270
- spinner.succeed("Build completed successfully!");
271
- callback(0);
272
- return 0;
273
- } catch (error) {
274
- log(`
275
- ${cliName} Error: ${error.message}`, "error", quiet);
276
- spinner.fail("Build failed.");
277
- if (cmd.assist) {
278
- spinner.start("AI is analyzing the webpack error...");
279
- try {
280
- await aiFunction({
281
- prompt: `Fix this webpack build error: ${error.message}
282
-
283
- Error details:
284
- ${error.toString()}
285
-
286
- Configuration used:
287
- ${JSON.stringify(webpackOptions, null, 2)}`,
288
- task: "help",
289
- context: true,
290
- quiet
291
- });
292
- spinner.succeed("AI analysis complete");
293
- } catch (aiError) {
294
- spinner.fail("Could not generate AI assistance");
295
- if (!quiet) {
296
- console.error("AI assistance error:", aiError);
297
- }
298
- }
299
- }
300
- callback(1);
301
- return 1;
302
- }
303
- };
304
- const build = async (cmd, callback = () => ({})) => {
305
- const {
306
- bundler = "webpack",
307
- cliName = "Lex",
308
- quiet = false,
309
- remove = false,
310
- variables = "{}"
311
- } = cmd;
312
- const spinner = createSpinner(quiet);
313
- log(`${cliName} building...`, "info", quiet);
314
- await LexConfig.parseConfig(cmd);
315
- const { outputFullPath, useTypescript } = LexConfig.config;
316
- checkLinkedModules();
317
- let variablesObj = { NODE_ENV: "production" };
318
- if (variables) {
319
- try {
320
- variablesObj = JSON.parse(variables);
321
- } catch (error) {
322
- log(`
323
- ${cliName} Error: Environment variables option is not a valid JSON object.`, "error", quiet);
324
- callback(1);
325
- return 1;
326
- }
327
- }
328
- process.env = { ...process.env, ...variablesObj };
329
- spinner.start("Building code...");
330
- if (remove) {
331
- await removeFiles(outputFullPath || "");
332
- }
333
- if (useTypescript) {
334
- const compileConfigPath = getTypeScriptConfigPath("tsconfig.build.json");
335
- if (existsSync(compileConfigPath)) {
336
- log("Using tsconfig.build.json for build...", "info", quiet);
337
- } else {
338
- LexConfig.checkCompileTypescriptConfig();
339
- }
340
- }
341
- let buildResult = 0;
342
- if (bundler === "esbuild") {
343
- buildResult = await buildWithEsBuild(spinner, cmd, (status) => {
344
- buildResult = status;
345
- });
346
- } else {
347
- buildResult = await buildWithWebpack(spinner, cmd, (status) => {
348
- buildResult = status;
349
- });
350
- }
351
- if (buildResult === 0 && cmd.analyze) {
352
- spinner.start("AI is analyzing the build output for optimization opportunities...");
353
- try {
354
- const stats = {
355
- outputPath: LexConfig.config.outputFullPath,
356
- entryPoints: bundler === "esbuild" ? `Source files: ${LexConfig.config.sourceFullPath}/**/*.{ts,js}` : LexConfig.config.webpack?.entry || "Unknown entry points"
357
- };
358
- await aiFunction({
359
- prompt: `Analyze this build for optimization opportunities:
360
-
361
- Build Type: ${bundler}
362
- Format: ${cmd.format || "default"}
363
- Environment: ${LexConfig.config.targetEnvironment}
364
- ${JSON.stringify(stats, null, 2)}
365
-
366
- What are the key optimization opportunities for this build configuration? Consider:
367
- 1. Bundle size optimization strategies
368
- 2. Code splitting recommendations
369
- 3. Tree-shaking improvements
370
- 4. Performance enhancements
371
- 5. Dependency optimizations`,
372
- task: "optimize",
373
- context: true,
374
- quiet
375
- });
376
- spinner.succeed("AI build analysis complete");
377
- } catch (aiError) {
378
- spinner.fail("Could not generate AI optimization analysis");
379
- if (!quiet) {
380
- console.error("AI analysis error:", aiError);
381
- }
382
- }
383
- }
384
- if (buildResult === 0) {
385
- try {
386
- await copyConfiguredFiles(spinner, LexConfig.config, quiet);
387
- } catch (copyError) {
388
- log(`
389
- ${cliName} Error: Failed to copy configured files: ${copyError.message}`, "error", quiet);
390
- callback(1);
391
- return 1;
392
- }
393
- }
394
- callback(buildResult);
395
- return buildResult;
396
- };
397
- var build_default = build;
398
- export {
399
- build,
400
- buildWithEsBuild,
401
- buildWithWebpack,
402
- build_default as default
403
- };
404
- //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vc3JjL2NvbW1hbmRzL2J1aWxkL2J1aWxkLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKipcbiAqIENvcHlyaWdodCAoYykgMjAxOC1QcmVzZW50LCBOaXRyb2dlbiBMYWJzLCBJbmMuXG4gKiBDb3B5cmlnaHRzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gU2VlIHRoZSBhY2NvbXBhbnlpbmcgTElDRU5TRSBmaWxlIGZvciB0ZXJtcy5cbiAqL1xuaW1wb3J0IEdyYXBocWxMb2FkZXJQbHVnaW4gZnJvbSAnQGx1Y2t5Y2F0ZmFjdG9yeS9lc2J1aWxkLWdyYXBocWwtbG9hZGVyJztcbmltcG9ydCB7ZXhlY2F9IGZyb20gJ2V4ZWNhJztcbmltcG9ydCB7ZXhpc3RzU3luYywgcmVhZEZpbGVTeW5jfSBmcm9tICdmcyc7XG5pbXBvcnQge3N5bmMgYXMgZ2xvYlN5bmN9IGZyb20gJ2dsb2InO1xuaW1wb3J0IHtyZXNvbHZlIGFzIHBhdGhSZXNvbHZlfSBmcm9tICdwYXRoJztcblxuaW1wb3J0IHtMZXhDb25maWcsIGdldFR5cGVTY3JpcHRDb25maWdQYXRofSBmcm9tICcuLi8uLi9MZXhDb25maWcuanMnO1xuaW1wb3J0IHtjaGVja0xpbmtlZE1vZHVsZXMsIGNvcHlDb25maWd1cmVkRmlsZXMsIGNyZWF0ZVNwaW5uZXIsIHJlbW92ZUZpbGVzfSBmcm9tICcuLi8uLi91dGlscy9hcHAuanMnO1xuaW1wb3J0IHtnZXREaXJOYW1lLCByZWxhdGl2ZU5vZGVQYXRofSBmcm9tICcuLi8uLi91dGlscy9maWxlLmpzJztcbmltcG9ydCB7bG9nfSBmcm9tICcuLi8uLi91dGlscy9sb2cuanMnO1xuaW1wb3J0IHthaUZ1bmN0aW9ufSBmcm9tICcuLi9haS9haS5qcyc7XG5cbmV4cG9ydCBpbnRlcmZhY2UgQnVpbGRPcHRpb25zIHtcbiAgcmVhZG9ubHkgYXNzaXN0PzogYm9vbGVhbjtcbiAgcmVhZG9ubHkgYW5hbHl6ZT86IGJvb2xlYW47XG4gIHJlYWRvbmx5IGJ1bmRsZXI/OiAnd2VicGFjaycgfCAnZXNidWlsZCc7XG4gIHJlYWRvbmx5IGNsaU5hbWU/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IGZvcm1hdD86IHN0cmluZztcbiAgcmVhZG9ubHkgb3V0cHV0UGF0aD86IHN0cmluZztcbiAgcmVhZG9ubHkgcXVpZXQ/OiBib29sZWFuO1xuICByZWFkb25seSByZW1vdmU/OiBib29sZWFuO1xuICByZWFkb25seSBzb3VyY2VQYXRoPzogc3RyaW5nO1xuICByZWFkb25seSB2YXJpYWJsZXM/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IHdhdGNoPzogYm9vbGVhbjtcbn1cblxuZXhwb3J0IHR5cGUgQnVpbGRDYWxsYmFjayA9IChzdGF0dXM6IG51bWJlcikgPT4gdm9pZDtcblxuZXhwb3J0IGNvbnN0IGJ1aWxkV2l0aEVzQnVpbGQgPSBhc3luYyAoc3Bpbm5lciwgY29tbWFuZE9wdGlvbnM6IEJ1aWxkT3B0aW9ucywgY2FsbGJhY2s6IEJ1aWxkQ2FsbGJhY2spID0+IHtcbiAgY29uc3Qge1xuICAgIGNsaU5hbWUgPSAnTGV4JyxcbiAgICBmb3JtYXQgPSAnY2pzJyxcbiAgICBvdXRwdXRQYXRoLFxuICAgIHF1aWV0LFxuICAgIHNvdXJjZVBhdGgsXG4gICAgd2F0Y2hcbiAgfSA9IGNvbW1hbmRPcHRpb25zO1xuICBjb25zdCB7XG4gICAgb3V0cHV0RnVsbFBhdGgsXG4gICAgc291cmNlRnVsbFBhdGgsXG4gICAgdGFyZ2V0RW52aXJvbm1lbnQsXG4gICAgdXNlR3JhcGhRbCxcbiAgICB1c2VUeXBlc2NyaXB0XG4gIH0gPSBMZXhDb25maWcuY29uZmlnO1xuICBjb25zdCBzb3VyY2VEaXI6IHN0cmluZyA9IHNvdXJjZVBhdGggPyBwYXRoUmVzb2x2ZShwcm9jZXNzLmN3ZCgpLCBgLi8ke3NvdXJjZVBhdGh9YCkgOiBzb3VyY2VGdWxsUGF0aCB8fCAnJztcbiAgY29uc3QgbG9hZGVyID0ge1xuICAgICcuanMnOiAnanMnXG4gIH07XG5cbiAgaWYodXNlVHlwZXNjcmlwdCkge1xuICAgIGxvYWRlclsnLnRzJ10gPSAndHMnO1xuICAgIGxvYWRlclsnLnRzeCddID0gJ3RzeCc7XG4gIH1cblxuICBjb25zdCBwbHVnaW5zID0gW107XG5cbiAgaWYodXNlR3JhcGhRbCkge1xuICAgIHBsdWdpbnMucHVzaCgoR3JhcGhxbExvYWRlclBsdWdpbiBhcyB1bmtub3duIGFzICgpID0+IG5ldmVyKSgpKTtcbiAgfVxuXG4gIGNvbnN0IGdsb2JPcHRpb25zID0ge1xuICAgIGN3ZDogc291cmNlRGlyLFxuICAgIGRvdDogZmFsc2UsXG4gICAgbm9kaXI6IHRydWUsXG4gICAgbm9zb3J0OiB0cnVlXG4gIH07XG4gIGNvbnN0IHRzRmlsZXM6IHN0cmluZ1tdID0gZ2xvYlN5bmMoYCR7c291cmNlRGlyfS8qKi8hKCouc3BlY3wqLnRlc3QpLnRzKmAsIGdsb2JPcHRpb25zKTtcbiAgY29uc3QganNGaWxlczogc3RyaW5nW10gPSBnbG9iU3luYyhgJHtzb3VyY2VEaXJ9LyoqLyEoKi5zcGVjfCoudGVzdCkuanNgLCBnbG9iT3B0aW9ucyk7XG4gIGNvbnN0IHNvdXJjZUZpbGVzOiBzdHJpbmdbXSA9IFsuLi50c0ZpbGVzLCAuLi5qc0ZpbGVzXTtcblxuICBjb25zdCBwYWNrYWdlSnNvbkRhdGEgPSByZWFkRmlsZVN5bmMocGF0aFJlc29sdmUocHJvY2Vzcy5jd2QoKSwgJy4vcGFja2FnZS5qc29uJykpO1xuICBjb25zdCBwYWNrYWdlSnNvbiA9IEpTT04ucGFyc2UocGFja2FnZUpzb25EYXRhLnRvU3RyaW5nKCkpO1xuICBjb25zdCBleHRlcm5hbCA9IFtcbiAgICAuLi5PYmplY3Qua2V5cyhwYWNrYWdlSnNvbi5kZXBlbmRlbmNpZXMgfHwge30pLFxuICAgIC4uLk9iamVjdC5rZXlzKHBhY2thZ2VKc29uLnBlZXJEZXBlbmRlbmNpZXMgfHwge30pXG4gIF07XG5cbiAgY29uc3QgZGlyTmFtZSA9IGdldERpck5hbWUoKTtcbiAgY29uc3QgZGlyUGF0aDogc3RyaW5nID0gcGF0aFJlc29sdmUoZGlyTmFtZSwgJy4uLy4uJyk7XG4gIGNvbnN0IG91dHB1dERpcjogc3RyaW5nID0gb3V0cHV0UGF0aCB8fCBvdXRwdXRGdWxsUGF0aCB8fCAnJztcbiAgY29uc3QgZXNidWlsZFBhdGg6IHN0cmluZyA9IHJlbGF0aXZlTm9kZVBhdGgoJ2VzYnVpbGQvYmluL2VzYnVpbGQnLCBkaXJQYXRoKTtcbiAgY29uc3QgZXNidWlsZENvbmZpZyA9IExleENvbmZpZy5jb25maWcuZXNidWlsZCB8fCB7fTtcblxuICBjb25zdCBlc2J1aWxkT3B0aW9uczogc3RyaW5nW10gPSBbXG4gICAgLi4uc291cmNlRmlsZXMsXG4gICAgJy0tYnVuZGxlJyxcbiAgICAnLS1jb2xvcj10cnVlJyxcbiAgICBgLS1mb3JtYXQ9JHtmb3JtYXR9YCxcbiAgICBgLS1vdXRkaXI9JHtvdXRwdXREaXJ9YCxcbiAgICBgLS1wbGF0Zm9ybT0ke2VzYnVpbGRDb25maWcucGxhdGZvcm0gfHwgJ25vZGUnfWAsXG4gICAgYC0tdGFyZ2V0PSR7ZXNidWlsZENvbmZpZy50YXJnZXQgfHwgKHRhcmdldEVudmlyb25tZW50ID09PSAnbm9kZScgPyAnbm9kZTIwJyA6ICdlczIwMjAnKX1gLFxuICAgIGAtLXNvdXJjZW1hcD0ke2VzYnVpbGRDb25maWcuc291cmNlbWFwIHx8ICdpbmxpbmUnfWBcbiAgXTtcblxuICBpZiAoZXNidWlsZENvbmZpZy5taW5pZnkgIT09IGZhbHNlKSB7XG4gICAgZXNidWlsZE9wdGlvbnMucHVzaCgnLS1taW5pZnknKTtcbiAgfVxuXG4gIGlmIChlc2J1aWxkQ29uZmlnLnRyZWVTaGFraW5nICE9PSBmYWxzZSkge1xuICAgIGVzYnVpbGRPcHRpb25zLnB1c2goJy0tdHJlZS1zaGFraW5nPXRydWUnKTtcbiAgfVxuXG4gICAgaWYgKGVzYnVpbGRDb25maWcuZHJvcCAmJiBlc2J1aWxkQ29uZmlnLmRyb3AubGVuZ3RoID4gMCkge1xuICAgIGVzYnVpbGRDb25maWcuZHJvcC5mb3JFYWNoKGl0ZW0gPT4ge1xuICAgICAgZXNidWlsZE9wdGlvbnMucHVzaChgLS1kcm9wOiR7aXRlbX1gKTtcbiAgICB9KTtcbiAgfVxuXG4gIGlmIChlc2J1aWxkQ29uZmlnLnB1cmUgJiYgZXNidWlsZENvbmZpZy5wdXJlLmxlbmd0aCA+IDApIHtcbiAgICBlc2J1aWxkQ29uZmlnLnB1cmUuZm9yRWFjaChpdGVtID0+IHtcbiAgICAgIGVzYnVpbGRPcHRpb25zLnB1c2goYC0tcHVyZToke2l0ZW19YCk7XG4gICAgfSk7XG4gIH1cblxuICBpZiAoZXNidWlsZENvbmZpZy5sZWdhbENvbW1lbnRzKSB7XG4gICAgZXNidWlsZE9wdGlvbnMucHVzaChgLS1sZWdhbC1jb21tZW50cz0ke2VzYnVpbGRDb25maWcubGVnYWxDb21tZW50c31gKTtcbiAgfVxuXG4gIGlmIChlc2J1aWxkQ29uZmlnLnNwbGl0dGluZyAhPT0gZmFsc2UpIHtcbiAgICBlc2J1aWxkT3B0aW9ucy5wdXNoKCctLXNwbGl0dGluZycpO1xuICB9XG5cbiAgaWYgKGVzYnVpbGRDb25maWcubWV0YWZpbGUpIHtcbiAgICBlc2J1aWxkT3B0aW9ucy5wdXNoKCctLW1ldGFmaWxlJyk7XG4gIH1cblxuICBpZiAoZXNidWlsZENvbmZpZy5iYW5uZXIpIHtcbiAgICBPYmplY3QuZW50cmllcyhlc2J1aWxkQ29uZmlnLmJhbm5lcikuZm9yRWFjaCgoW3R5cGUsIGNvbnRlbnRdKSA9PiB7XG4gICAgICBlc2J1aWxkT3B0aW9ucy5wdXNoKGAtLWJhbm5lcjoke3R5cGV9PSR7Y29udGVudH1gKTtcbiAgICB9KTtcbiAgfVxuXG4gIGlmIChlc2J1aWxkQ29uZmlnLmZvb3Rlcikge1xuICAgIE9iamVjdC5lbnRyaWVzKGVzYnVpbGRDb25maWcuZm9vdGVyKS5mb3JFYWNoKChbdHlwZSwgY29udGVudF0pID0+IHtcbiAgICAgIGVzYnVpbGRPcHRpb25zLnB1c2goYC0tZm9vdGVyOiR7dHlwZX09JHtjb250ZW50fWApO1xuICAgIH0pO1xuICB9XG5cbiAgaWYgKGVzYnVpbGRDb25maWcuZGVmaW5lKSB7XG4gICAgT2JqZWN0LmVudHJpZXMoZXNidWlsZENvbmZpZy5kZWZpbmUpLmZvckVhY2goKFtrZXksIHZhbHVlXSkgPT4ge1xuICAgICAgZXNidWlsZE9wdGlvbnMucHVzaChgLS1kZWZpbmU6JHtrZXl9PSR7dmFsdWV9YCk7XG4gICAgfSk7XG4gIH1cblxuICBpZihleHRlcm5hbC5sZW5ndGgpIHtcbiAgICBlc2J1aWxkT3B0aW9ucy5wdXNoKGAtLWV4dGVybmFsOiR7ZXh0ZXJuYWwuam9pbignLCcpfWApO1xuICB9XG5cbiAgaWYocGx1Z2lucy5sZW5ndGgpIHtcbiAgICBlc2J1aWxkT3B0aW9ucy5wdXNoKGAtLXBsdWdpbnM9JHtwbHVnaW5zLmpvaW4oJywnKX1gKTtcbiAgfVxuICBpZih3YXRjaCkge1xuICAgIGVzYnVpbGRPcHRpb25zLnB1c2goJy0td2F0Y2gnKTtcbiAgfVxuXG4gIHRyeSB7XG4gICAgYXdhaXQgZXhlY2EoZXNidWlsZFBhdGgsIGVzYnVpbGRPcHRpb25zLCB7ZW5jb2Rpbmc6ICd1dGY4J30pO1xuXG4gICAgc3Bpbm5lci5zdWNjZWVkKCdCdWlsZCBjb21wbGV0ZWQgc3VjY2Vzc2Z1bGx5IScpO1xuICB9IGNhdGNoKGVycm9yKSB7XG4gICAgbG9nKGBcXG4ke2NsaU5hbWV9IEVycm9yOiAke2Vycm9yLm1lc3NhZ2V9YCwgJ2Vycm9yJywgcXVpZXQpO1xuXG4gICAgaWYoIXF1aWV0KSB7XG4gICAgICBjb25zb2xlLmVycm9yKGVycm9yKTtcbiAgICB9XG5cbiAgICBzcGlubmVyLmZhaWwoJ0NvZGUgYnVpbGQgZmFpbGVkLicpO1xuXG4gICAgaWYoY29tbWFuZE9wdGlvbnMuYXNzaXN0KSB7XG4gICAgICBzcGlubmVyLnN0YXJ0KCdBSSBpcyBhbmFseXppbmcgdGhlIGVycm9yLi4uJyk7XG5cbiAgICAgIHRyeSB7XG4gICAgICAgIGF3YWl0IGFpRnVuY3Rpb24oe1xuICAgICAgICAgIHByb21wdDogYEZpeCB0aGlzIGVzYnVpbGQgZXJyb3I6ICR7ZXJyb3IubWVzc2FnZX1cXG5cXG5FcnJvciBkZXRhaWxzOlxcbiR7ZXJyb3IudG9TdHJpbmcoKX1gLFxuICAgICAgICAgIHRhc2s6ICdoZWxwJyxcbiAgICAgICAgICBjb250ZXh0OiB0cnVlLFxuICAgICAgICAgIHF1aWV0XG4gICAgICAgIH0pO1xuXG4gICAgICAgIHNwaW5uZXIuc3VjY2VlZCgnQUkgYW5hbHlzaXMgY29tcGxldGUnKTtcbiAgICAgIH0gY2F0Y2goYWlFcnJvcikge1xuICAgICAgICBzcGlubmVyLmZhaWwoJ0NvdWxkIG5vdCBnZW5lcmF0ZSBBSSBhc3Npc3RhbmNlJyk7XG4gICAgICAgIGlmKCFxdWlldCkge1xuICAgICAgICAgIGNvbnNvbGUuZXJyb3IoJ0FJIGFzc2lzdGFuY2UgZXJyb3I6JywgYWlFcnJvcik7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG5cbiAgICBjYWxsYmFjaygxKTtcbiAgICByZXR1cm4gMTtcbiAgfVxuXG4gIGNhbGxiYWNrKDApO1xuICByZXR1cm4gMDtcbn07XG5cbmV4cG9ydCBjb25zdCBidWlsZFdpdGhXZWJwYWNrID0gYXN5bmMgKHNwaW5uZXIsIGNtZCwgY2FsbGJhY2spID0+IHtcbiAgY29uc3Qge1xuICAgIGFuYWx5emUsXG4gICAgY2xpTmFtZSA9ICdMZXgnLFxuICAgIGNvbmZpZyxcbiAgICBjb25maWdOYW1lLFxuICAgIGRlZmluZVByb2Nlc3NFbnZOb2RlRW52LFxuICAgIGRldnRvb2wsXG4gICAgZGlzYWJsZUludGVycHJldCxcbiAgICBlbnRyeSxcbiAgICBlbnYsXG4gICAgZmFpbE9uV2FybmluZ3MsXG4gICAganNvbixcbiAgICBtZXJnZSxcbiAgICBtb2RlLFxuICAgIG5hbWUsXG4gICAgbm9kZUVudixcbiAgICBub0RldnRvb2wsXG4gICAgbm9TdGF0cyxcbiAgICBub1RhcmdldCxcbiAgICBub1dhdGNoLFxuICAgIG5vV2F0Y2hPcHRpb25zU3RkaW4sXG4gICAgb3V0cHV0UGF0aCxcbiAgICBxdWlldCA9IGZhbHNlLFxuICAgIHN0YXRzLFxuICAgIHRhcmdldCxcbiAgICB3YXRjaCxcbiAgICB3YXRjaE9wdGlvbnNTdGRpblxuICB9ID0gY21kO1xuXG4gIGxldCB3ZWJwYWNrQ29uZmlnOiBzdHJpbmc7XG4gIGNvbnN0IGRpck5hbWUgPSBnZXREaXJOYW1lKCk7XG5cbiAgaWYoY29uZmlnKSB7XG4gICAgY29uc3QgaXNSZWxhdGl2ZUNvbmZpZzogYm9vbGVhbiA9IGNvbmZpZy5zdWJzdHIoMCwgMikgPT09ICcuLyc7XG4gICAgd2VicGFja0NvbmZpZyA9IGlzUmVsYXRpdmVDb25maWcgPyBwYXRoUmVzb2x2ZShwcm9jZXNzLmN3ZCgpLCBjb25maWcpIDogY29uZmlnO1xuICB9IGVsc2Uge1xuICAgIHdlYnBhY2tDb25maWcgPSBwYXRoUmVzb2x2ZShkaXJOYW1lLCAnLi4vLi4vd2VicGFjay5jb25maWcuanMnKTtcbiAgfVxuXG4gIGNvbnN0IHdlYnBhY2tPcHRpb25zOiBzdHJpbmdbXSA9IFtcbiAgICAnLS1jb2xvcicsXG4gICAgJy0tcHJvZ3Jlc3MnLFxuICAgICctLWNvbmZpZycsIHdlYnBhY2tDb25maWdcbiAgXTtcblxuICBpZihhbmFseXplKSB7XG4gICAgd2VicGFja09wdGlvbnMucHVzaCgnLS1hbmFseXplJyk7XG4gIH1cblxuICBpZihjb25maWdOYW1lKSB7XG4gICAgd2VicGFja09wdGlvbnMucHVzaCgnLS1jb25maWdOYW1lJywgY29uZmlnTmFtZSk7XG4gIH1cblxuICBpZihkZWZpbmVQcm9jZXNzRW52Tm9kZUVudikge1xuICAgIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tZGVmaW5lUHJvY2Vzc0Vudk5vZGVFbnYnLCBkZWZpbmVQcm9jZXNzRW52Tm9kZUVudik7XG4gIH1cblxuICBpZihkZXZ0b29sKSB7XG4gICAgd2VicGFja09wdGlvbnMucHVzaCgnLS1kZXZ0b29sJywgZGV2dG9vbCk7XG4gIH1cblxuICBpZihkaXNhYmxlSW50ZXJwcmV0KSB7XG4gICAgd2VicGFja09wdGlvbnMucHVzaCgnLS1kaXNhYmxlSW50ZXJwcmV0Jyk7XG4gIH1cblxuICBpZihlbnRyeSkge1xuICAgIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tZW50cnknLCBlbnRyeSk7XG4gIH1cblxuICBpZihlbnYpIHtcbiAgICB3ZWJwYWNrT3B0aW9ucy5wdXNoKCctLWVudicsIGVudik7XG4gIH1cblxuICBpZihmYWlsT25XYXJuaW5ncykge1xuICAgIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tZmFpbE9uV2FybmluZ3MnKTtcbiAgfVxuXG4gIGlmKGpzb24pIHtcbiAgICB3ZWJwYWNrT3B0aW9ucy5wdXNoKCctLWpzb24nLCBqc29uKTtcbiAgfVxuXG4gIGlmKG1vZGUpIHtcbiAgICB3ZWJwYWNrT3B0aW9ucy5wdXNoKCctLW1vZGUnLCBtb2RlKTtcbiAgfVxuXG4gIGlmKG1lcmdlKSB7XG4gICAgd2VicGFja09wdGlvbnMucHVzaCgnLS1tZXJnZScpO1xuICB9XG5cbiAgaWYobmFtZSkge1xuICAgIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tbmFtZScsIG5hbWUpO1xuICB9XG5cbiAgaWYobm9EZXZ0b29sKSB7XG4gICAgd2VicGFja09wdGlvbnMucHVzaCgnLS1ub0RldnRvb2wnKTtcbiAgfVxuXG4gIGlmKG5vU3RhdHMpIHtcbiAgICB3ZWJwYWNrT3B0aW9ucy5wdXNoKCctLW5vU3RhdHMnKTtcbiAgfVxuXG4gIGlmKG5vVGFyZ2V0KSB7XG4gICAgd2VicGFja09wdGlvbnMucHVzaCgnLS1ub1RhcmdldCcpO1xuICB9XG5cbiAgaWYobm9XYXRjaCkge1xuICAgIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tbm9XYXRjaCcpO1xuICB9XG5cbiAgaWYobm9XYXRjaE9wdGlvbnNTdGRpbikge1xuICAgIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tbm9XYXRjaE9wdGlvbnNTdGRpbicpO1xuICB9XG5cbiAgaWYobm9kZUVudikge1xuICAgIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tbm9kZUVudicsIG5vZGVFbnYpO1xuICB9XG5cbiAgaWYob3V0cHV0UGF0aCkge1xuICAgIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tb3V0cHV0UGF0aCcsIG91dHB1dFBhdGgpO1xuICB9XG5cbiAgaWYoc3RhdHMpIHtcbiAgICB3ZWJwYWNrT3B0aW9ucy5wdXNoKCctLXN0YXRzJywgc3RhdHMpO1xuICB9XG5cbiAgaWYodGFyZ2V0KSB7XG4gICAgd2VicGFja09wdGlvbnMucHVzaCgnLS10YXJnZXQnLCB0YXJnZXQpO1xuICB9XG5cbiAgaWYod2F0Y2gpIHtcbiAgICB3ZWJwYWNrT3B0aW9ucy5wdXNoKCctLXdhdGNoJyk7XG4gIH1cblxuICBpZih3YXRjaE9wdGlvbnNTdGRpbikge1xuICAgIHdlYnBhY2tPcHRpb25zLnB1c2goJy0td2F0Y2hPcHRpb25zU3RkaW4nKTtcbiAgfVxuXG4gIGNvbnN0IGRpclBhdGg6IHN0cmluZyA9IHBhdGhSZXNvbHZlKGRpck5hbWUsICcuLi8uLicpO1xuXG4gIHRyeSB7XG4gICAgY29uc3Qgd2VicGFja1BhdGg6IHN0cmluZyA9IHJlbGF0aXZlTm9kZVBhdGgoJ3dlYnBhY2stY2xpL2Jpbi9jbGkuanMnLCBkaXJQYXRoKTtcbiAgICBhd2FpdCBleGVjYSh3ZWJwYWNrUGF0aCwgd2VicGFja09wdGlvbnMsIHtlbmNvZGluZzogJ3V0ZjgnLCBzdGRpbzogJ2luaGVyaXQnfSk7XG5cbiAgICBzcGlubmVyLnN1Y2NlZWQoJ0J1aWxkIGNvbXBsZXRlZCBzdWNjZXNzZnVsbHkhJyk7XG5cbiAgICBjYWxsYmFjaygwKTtcbiAgICByZXR1cm4gMDtcbiAgfSBjYXRjaChlcnJvcikge1xuICAgIGxvZyhgXFxuJHtjbGlOYW1lfSBFcnJvcjogJHtlcnJvci5tZXNzYWdlfWAsICdlcnJvcicsIHF1aWV0KTtcblxuICAgIHNwaW5uZXIuZmFpbCgnQnVpbGQgZmFpbGVkLicpO1xuXG4gICAgaWYoY21kLmFzc2lzdCkge1xuICAgICAgc3Bpbm5lci5zdGFydCgnQUkgaXMgYW5hbHl6aW5nIHRoZSB3ZWJwYWNrIGVycm9yLi4uJyk7XG5cbiAgICAgIHRyeSB7XG4gICAgICAgIGF3YWl0IGFpRnVuY3Rpb24oe1xuICAgICAgICAgIHByb21wdDogYEZpeCB0aGlzIHdlYnBhY2sgYnVpbGQgZXJyb3I6ICR7ZXJyb3IubWVzc2FnZX1cXG5cXG5FcnJvciBkZXRhaWxzOlxcbiR7ZXJyb3IudG9TdHJpbmcoKX1cXG5cXG5Db25maWd1cmF0aW9uIHVzZWQ6XFxuJHtKU09OLnN0cmluZ2lmeSh3ZWJwYWNrT3B0aW9ucywgbnVsbCwgMil9YCxcbiAgICAgICAgICB0YXNrOiAnaGVscCcsXG4gICAgICAgICAgY29udGV4dDogdHJ1ZSxcbiAgICAgICAgICBxdWlldFxuICAgICAgICB9KTtcblxuICAgICAgICBzcGlubmVyLnN1Y2NlZWQoJ0FJIGFuYWx5c2lzIGNvbXBsZXRlJyk7XG4gICAgICB9IGNhdGNoKGFpRXJyb3IpIHtcbiAgICAgICAgc3Bpbm5lci5mYWlsKCdDb3VsZCBub3QgZ2VuZXJhdGUgQUkgYXNzaXN0YW5jZScpO1xuICAgICAgICBpZighcXVpZXQpIHtcbiAgICAgICAgICBjb25zb2xlLmVycm9yKCdBSSBhc3Npc3RhbmNlIGVycm9yOicsIGFpRXJyb3IpO1xuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuXG4gICAgY2FsbGJhY2soMSk7XG4gICAgcmV0dXJuIDE7XG4gIH1cbn07XG5cbmV4cG9ydCBjb25zdCBidWlsZCA9IGFzeW5jIChjbWQ6IEJ1aWxkT3B0aW9ucywgY2FsbGJhY2s6IEJ1aWxkQ2FsbGJhY2sgPSAoKSA9PiAoe30pKTogUHJvbWlzZTxudW1iZXI+ID0+IHtcbiAgY29uc3Qge1xuICAgIGJ1bmRsZXIgPSAnd2VicGFjaycsXG4gICAgY2xpTmFtZSA9ICdMZXgnLFxuICAgIHF1aWV0ID0gZmFsc2UsXG4gICAgcmVtb3ZlID0gZmFsc2UsXG4gICAgdmFyaWFibGVzID0gJ3t9J1xuICB9ID0gY21kO1xuXG4gIGNvbnN0IHNwaW5uZXIgPSBjcmVhdGVTcGlubmVyKHF1aWV0KTtcblxuICBsb2coYCR7Y2xpTmFtZX0gYnVpbGRpbmcuLi5gLCAnaW5mbycsIHF1aWV0KTtcblxuICBhd2FpdCBMZXhDb25maWcucGFyc2VDb25maWcoY21kKTtcblxuICBjb25zdCB7b3V0cHV0RnVsbFBhdGgsIHVzZVR5cGVzY3JpcHR9ID0gTGV4Q29uZmlnLmNvbmZpZztcblxuICBjaGVja0xpbmtlZE1vZHVsZXMoKTtcblxuICBsZXQgdmFyaWFibGVzT2JqOiBvYmplY3QgPSB7Tk9ERV9FTlY6ICdwcm9kdWN0aW9uJ307XG5cbiAgaWYodmFyaWFibGVzKSB7XG4gICAgdHJ5IHtcbiAgICAgIHZhcmlhYmxlc09iaiA9IEpTT04ucGFyc2UodmFyaWFibGVzKTtcbiAgICB9IGNhdGNoKGVycm9yKSB7XG4gICAgICBsb2coYFxcbiR7Y2xpTmFtZX0gRXJyb3I6IEVudmlyb25tZW50IHZhcmlhYmxlcyBvcHRpb24gaXMgbm90IGEgdmFsaWQgSlNPTiBvYmplY3QuYCwgJ2Vycm9yJywgcXVpZXQpO1xuXG4gICAgICBjYWxsYmFjaygxKTtcbiAgICAgIHJldHVybiAxO1xuICAgIH1cbiAgfVxuXG4gIHByb2Nlc3MuZW52ID0gey4uLnByb2Nlc3MuZW52LCAuLi52YXJpYWJsZXNPYmp9O1xuXG4gIHNwaW5uZXIuc3RhcnQoJ0J1aWxkaW5nIGNvZGUuLi4nKTtcblxuICBpZihyZW1vdmUpIHtcbiAgICBhd2FpdCByZW1vdmVGaWxlcyhvdXRwdXRGdWxsUGF0aCB8fCAnJyk7XG4gIH1cblxuICBpZih1c2VUeXBlc2NyaXB0KSB7XG4gICAgLy8gVXNlIHRoZSBjb21waWxlLXNwZWNpZmljIFR5cGVTY3JpcHQgY29uZmlnIGZvciBidWlsZGluZ1xuICAgIGNvbnN0IGNvbXBpbGVDb25maWdQYXRoID0gZ2V0VHlwZVNjcmlwdENvbmZpZ1BhdGgoJ3RzY29uZmlnLmJ1aWxkLmpzb24nKTtcbiAgICBpZihleGlzdHNTeW5jKGNvbXBpbGVDb25maWdQYXRoKSkge1xuICAgICAgbG9nKCdVc2luZyB0c2NvbmZpZy5idWlsZC5qc29uIGZvciBidWlsZC4uLicsICdpbmZvJywgcXVpZXQpO1xuICAgIH0gZWxzZSB7XG4gICAgICBMZXhDb25maWcuY2hlY2tDb21waWxlVHlwZXNjcmlwdENvbmZpZygpO1xuICAgIH1cbiAgfVxuXG4gIGxldCBidWlsZFJlc3VsdCA9IDA7XG5cbiAgaWYoYnVuZGxlciA9PT0gJ2VzYnVpbGQnKSB7XG4gICAgYnVpbGRSZXN1bHQgPSBhd2FpdCBidWlsZFdpdGhFc0J1aWxkKHNwaW5uZXIsIGNtZCwgKHN0YXR1cykgPT4ge1xuICAgICAgYnVpbGRSZXN1bHQgPSBzdGF0dXM7XG4gICAgfSk7XG4gIH0gZWxzZSB7XG4gICAgYnVpbGRSZXN1bHQgPSBhd2FpdCBidWlsZFdpdGhXZWJwYWNrKHNwaW5uZXIsIGNtZCwgKHN0YXR1cykgPT4ge1xuICAgICAgYnVpbGRSZXN1bHQgPSBzdGF0dXM7XG4gICAgfSk7XG4gIH1cblxuICBpZihidWlsZFJlc3VsdCA9PT0gMCAmJiBjbWQuYW5hbHl6ZSkge1xuICAgIHNwaW5uZXIuc3RhcnQoJ0FJIGlzIGFuYWx5emluZyB0aGUgYnVpbGQgb3V0cHV0IGZvciBvcHRpbWl6YXRpb24gb3Bwb3J0dW5pdGllcy4uLicpO1xuXG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IHN0YXRzID0ge1xuICAgICAgICBvdXRwdXRQYXRoOiBMZXhDb25maWcuY29uZmlnLm91dHB1dEZ1bGxQYXRoLFxuICAgICAgICBlbnRyeVBvaW50czogYnVuZGxlciA9PT0gJ2VzYnVpbGQnID9cbiAgICAgICAgICBgU291cmNlIGZpbGVzOiAke0xleENvbmZpZy5jb25maWcuc291cmNlRnVsbFBhdGh9LyoqLyoue3RzLGpzfWAgOlxuICAgICAgICAgIExleENvbmZpZy5jb25maWcud2VicGFjaz8uZW50cnkgfHwgJ1Vua25vd24gZW50cnkgcG9pbnRzJ1xuICAgICAgfTtcblxuICAgICAgYXdhaXQgYWlGdW5jdGlvbih7XG4gICAgICAgIHByb21wdDogYEFuYWx5emUgdGhpcyBidWlsZCBmb3Igb3B0aW1pemF0aW9uIG9wcG9ydHVuaXRpZXM6XG5cbkJ1aWxkIFR5cGU6ICR7YnVuZGxlcn1cbkZvcm1hdDogJHtjbWQuZm9ybWF0IHx8ICdkZWZhdWx0J31cbkVudmlyb25tZW50OiAke0xleENvbmZpZy5jb25maWcudGFyZ2V0RW52aXJvbm1lbnR9XG4ke0pTT04uc3RyaW5naWZ5KHN0YXRzLCBudWxsLCAyKX1cblxuV2hhdCBhcmUgdGhlIGtleSBvcHRpbWl6YXRpb24gb3Bwb3J0dW5pdGllcyBmb3IgdGhpcyBidWlsZCBjb25maWd1cmF0aW9uPyBDb25zaWRlcjpcbjEuIEJ1bmRsZSBzaXplIG9wdGltaXphdGlvbiBzdHJhdGVnaWVzXG4yLiBDb2RlIHNwbGl0dGluZyByZWNvbW1lbmRhdGlvbnNcbjMuIFRyZWUtc2hha2luZyBpbXByb3ZlbWVudHNcbjQuIFBlcmZvcm1hbmNlIGVuaGFuY2VtZW50c1xuNS4gRGVwZW5kZW5jeSBvcHRpbWl6YXRpb25zYCxcbiAgICAgICAgdGFzazogJ29wdGltaXplJyxcbiAgICAgICAgY29udGV4dDogdHJ1ZSxcbiAgICAgICAgcXVpZXRcbiAgICAgIH0pO1xuXG4gICAgICBzcGlubmVyLnN1Y2NlZWQoJ0FJIGJ1aWxkIGFuYWx5c2lzIGNvbXBsZXRlJyk7XG4gICAgfSBjYXRjaChhaUVycm9yKSB7XG4gICAgICBzcGlubmVyLmZhaWwoJ0NvdWxkIG5vdCBnZW5lcmF0ZSBBSSBvcHRpbWl6YXRpb24gYW5hbHlzaXMnKTtcbiAgICAgIGlmKCFxdWlldCkge1xuICAgICAgICBjb25zb2xlLmVycm9yKCdBSSBhbmFseXNpcyBlcnJvcjonLCBhaUVycm9yKTtcbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICAvLyBDb3B5IGNvbmZpZ3VyZWQgZmlsZXMgYWZ0ZXIgc3VjY2Vzc2Z1bCBidWlsZFxuICBpZihidWlsZFJlc3VsdCA9PT0gMCkge1xuICAgIHRyeSB7XG4gICAgICBhd2FpdCBjb3B5Q29uZmlndXJlZEZpbGVzKHNwaW5uZXIsIExleENvbmZpZy5jb25maWcsIHF1aWV0KTtcbiAgICB9IGNhdGNoKGNvcHlFcnJvcikge1xuICAgICAgbG9nKGBcXG4ke2NsaU5hbWV9IEVycm9yOiBGYWlsZWQgdG8gY29weSBjb25maWd1cmVkIGZpbGVzOiAke2NvcHlFcnJvci5tZXNzYWdlfWAsICdlcnJvcicsIHF1aWV0KTtcbiAgICAgIGNhbGxiYWNrKDEpO1xuICAgICAgcmV0dXJuIDE7XG4gICAgfVxuICB9XG5cbiAgY2FsbGJhY2soYnVpbGRSZXN1bHQpO1xuICByZXR1cm4gYnVpbGRSZXN1bHQ7XG59O1xuXG5leHBvcnQgZGVmYXVsdCBidWlsZDtcbiJdLAogICJtYXBwaW5ncyI6ICJBQUlBLE9BQU8seUJBQXlCO0FBQ2hDLFNBQVEsYUFBWTtBQUNwQixTQUFRLFlBQVksb0JBQW1CO0FBQ3ZDLFNBQVEsUUFBUSxnQkFBZTtBQUMvQixTQUFRLFdBQVcsbUJBQWtCO0FBRXJDLFNBQVEsV0FBVywrQkFBOEI7QUFDakQsU0FBUSxvQkFBb0IscUJBQXFCLGVBQWUsbUJBQWtCO0FBQ2xGLFNBQVEsWUFBWSx3QkFBdUI7QUFDM0MsU0FBUSxXQUFVO0FBQ2xCLFNBQVEsa0JBQWlCO0FBa0JsQixNQUFNLG1CQUFtQixPQUFPLFNBQVMsZ0JBQThCLGFBQTRCO0FBQ3hHLFFBQU07QUFBQSxJQUNKLFVBQVU7QUFBQSxJQUNWLFNBQVM7QUFBQSxJQUNUO0FBQUEsSUFDQTtBQUFBLElBQ0E7QUFBQSxJQUNBO0FBQUEsRUFDRixJQUFJO0FBQ0osUUFBTTtBQUFBLElBQ0o7QUFBQSxJQUNBO0FBQUEsSUFDQTtBQUFBLElBQ0E7QUFBQSxJQUNBO0FBQUEsRUFDRixJQUFJLFVBQVU7QUFDZCxRQUFNLFlBQW9CLGFBQWEsWUFBWSxRQUFRLElBQUksR0FBRyxLQUFLLFVBQVUsRUFBRSxJQUFJLGtCQUFrQjtBQUN6RyxRQUFNLFNBQVM7QUFBQSxJQUNiLE9BQU87QUFBQSxFQUNUO0FBRUEsTUFBRyxlQUFlO0FBQ2hCLFdBQU8sS0FBSyxJQUFJO0FBQ2hCLFdBQU8sTUFBTSxJQUFJO0FBQUEsRUFDbkI7QUFFQSxRQUFNLFVBQVUsQ0FBQztB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