@nlabs/lex 1.48.6 → 1.49.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.
- package/.storybook/main.ts +9 -2
- package/.vscode/settings.json +1 -6
- package/README.md +276 -4
- package/eslint.config.mjs +24 -0
- package/examples/lex.config.js +18 -8
- package/examples/serverless-example/README.md +109 -0
- package/examples/serverless-example/dist/handlers/echo.js +15 -0
- package/examples/serverless-example/dist/handlers/graphql.js +137 -0
- package/examples/serverless-example/dist/handlers/hello.js +15 -0
- package/examples/serverless-example/dist/handlers/test.js +17 -0
- package/examples/serverless-example/dist/handlers/websocket.js +14 -0
- package/examples/serverless-example/lex.config.mjs +74 -0
- package/jest.config.mjs +13 -12
- package/{dist → lib}/LexConfig.d.ts +14 -6
- package/lib/LexConfig.js +268 -0
- package/lib/commands/ai/ai.js +303 -0
- package/{dist → lib}/commands/build/build.d.ts +3 -0
- package/lib/commands/build/build.js +494 -0
- package/{dist → lib}/commands/clean/clean.js +1 -1
- package/lib/commands/compile/compile.js +241 -0
- package/lib/commands/copy/copy.js +38 -0
- package/{dist → lib}/commands/create/create.js +1 -1
- package/{dist → lib}/commands/dev/dev.d.ts +2 -0
- package/lib/commands/dev/dev.js +286 -0
- package/{dist → lib}/commands/init/init.js +1 -1
- package/{dist → lib}/commands/lint/lint.d.ts +4 -1
- package/lib/commands/lint/lint.js +993 -0
- package/{dist → lib}/commands/migrate/migrate.js +1 -1
- package/lib/commands/publish/publish.js +104 -0
- package/lib/commands/serverless/serverless.d.ts +17 -0
- package/lib/commands/serverless/serverless.js +662 -0
- package/lib/commands/storybook/storybook.js +249 -0
- package/lib/commands/test/test.js +428 -0
- package/lib/commands/update/update.js +128 -0
- package/lib/commands/versions/versions.js +41 -0
- package/{dist → lib}/create/changelog.js +1 -1
- package/{dist → lib}/index.d.ts +2 -0
- package/{dist → lib}/index.js +3 -1
- package/lib/lex.js +73 -0
- package/lib/storybook/index.d.ts +5 -0
- package/lib/types.js +1 -0
- package/lib/utils/aiService.d.ts +9 -0
- package/lib/utils/aiService.js +299 -0
- package/{dist → lib}/utils/app.d.ts +3 -0
- package/lib/utils/app.js +296 -0
- package/lib/utils/deepMerge.js +26 -0
- package/{dist → lib}/utils/file.d.ts +7 -3
- package/lib/utils/file.js +229 -0
- package/lib/utils/translations.d.ts +1 -0
- package/lib/utils/translations.js +74 -0
- package/package.json +62 -50
- package/postcss.config.js +5 -3
- package/tsconfig.build.json +2 -2
- package/webpack.config.js +229 -39
- package/dist/LexConfig.js +0 -286
- package/dist/commands/ai/ai.js +0 -303
- package/dist/commands/build/build.js +0 -404
- package/dist/commands/compile/compile.js +0 -234
- package/dist/commands/copy/copy.js +0 -38
- package/dist/commands/dev/dev.js +0 -74
- package/dist/commands/lint/lint.js +0 -811
- package/dist/commands/publish/publish.js +0 -104
- package/dist/commands/storybook/storybook.js +0 -249
- package/dist/commands/test/test.js +0 -429
- package/dist/commands/update/update.js +0 -132
- package/dist/commands/versions/versions.js +0 -41
- package/dist/lex.js +0 -70
- package/dist/utils/aiService.d.ts +0 -9
- package/dist/utils/aiService.js +0 -299
- package/dist/utils/app.js +0 -267
- package/dist/utils/deepMerge.js +0 -24
- package/dist/utils/file.js +0 -185
- package/emptyModule.js +0 -0
- package/eslint.config.js +0 -7
- /package/{dist → lib}/Button.stories.d.ts +0 -0
- /package/{dist → lib}/commands/ai/ai.d.ts +0 -0
- /package/{dist → lib}/commands/ai/index.d.ts +0 -0
- /package/{dist → lib}/commands/ai/index.js +0 -0
- /package/{dist → lib}/commands/clean/clean.d.ts +0 -0
- /package/{dist → lib}/commands/compile/compile.d.ts +0 -0
- /package/{dist → lib}/commands/config/config.d.ts +0 -0
- /package/{dist → lib}/commands/config/config.js +0 -0
- /package/{dist → lib}/commands/copy/copy.d.ts +0 -0
- /package/{dist → lib}/commands/create/create.d.ts +0 -0
- /package/{dist → lib}/commands/init/init.d.ts +0 -0
- /package/{dist → lib}/commands/link/link.d.ts +0 -0
- /package/{dist → lib}/commands/link/link.js +0 -0
- /package/{dist → lib}/commands/lint/autofix.d.ts +0 -0
- /package/{dist → lib}/commands/migrate/migrate.d.ts +0 -0
- /package/{dist → lib}/commands/publish/publish.d.ts +0 -0
- /package/{dist → lib}/commands/storybook/storybook.d.ts +0 -0
- /package/{dist → lib}/commands/test/test.d.ts +0 -0
- /package/{dist → lib}/commands/update/update.d.ts +0 -0
- /package/{dist → lib}/commands/upgrade/upgrade.d.ts +0 -0
- /package/{dist → lib}/commands/upgrade/upgrade.js +0 -0
- /package/{dist → lib}/commands/versions/versions.d.ts +0 -0
- /package/{dist → lib}/create/changelog.d.ts +0 -0
- /package/{dist → lib}/lex.d.ts +0 -0
- /package/{dist/types.js → lib/storybook/index.js} +0 -0
- /package/{dist → lib}/test-react/index.d.ts +0 -0
- /package/{dist → lib}/test-react/index.js +0 -0
- /package/{dist → lib}/types.d.ts +0 -0
- /package/{dist → lib}/utils/deepMerge.d.ts +0 -0
- /package/{dist → lib}/utils/log.d.ts +0 -0
- /package/{dist → lib}/utils/log.js +0 -0
- /package/{dist → lib}/utils/reactShim.d.ts +0 -0
- /package/{dist → lib}/utils/reactShim.js +0 -0
|
@@ -0,0 +1,993 @@
|
|
|
1
|
+
import { execa } from "execa";
|
|
2
|
+
import { existsSync, readFileSync, unlinkSync, writeFileSync } from "fs";
|
|
3
|
+
import { dirname, resolve as pathResolve, extname } from "path";
|
|
4
|
+
import { LexConfig } from "../../LexConfig.js";
|
|
5
|
+
import { createSpinner } from "../../utils/app.js";
|
|
6
|
+
import { resolveBinaryPath } from "../../utils/file.js";
|
|
7
|
+
import { log } from "../../utils/log.js";
|
|
8
|
+
let currentFilename;
|
|
9
|
+
let currentDirname;
|
|
10
|
+
try {
|
|
11
|
+
currentFilename = eval('require("url").fileURLToPath(import.meta.url)');
|
|
12
|
+
currentDirname = dirname(currentFilename);
|
|
13
|
+
} catch {
|
|
14
|
+
currentFilename = process.cwd();
|
|
15
|
+
currentDirname = process.cwd();
|
|
16
|
+
}
|
|
17
|
+
const createDefaultESLintConfig = (useTypescript, cwd) => {
|
|
18
|
+
const configPath = pathResolve(cwd, ".lex-temp-default-eslint.cjs");
|
|
19
|
+
const originalConfig = null;
|
|
20
|
+
const possiblePaths = [
|
|
21
|
+
pathResolve(currentDirname, "eslint.config.ts"),
|
|
22
|
+
pathResolve(currentDirname, "eslint.config.js"),
|
|
23
|
+
pathResolve(process.env.LEX_HOME || "./node_modules/@nlabs/lex", "eslint.config.ts"),
|
|
24
|
+
pathResolve(process.env.LEX_HOME || "./node_modules/@nlabs/lex", "eslint.config.js")
|
|
25
|
+
];
|
|
26
|
+
let foundConfig = "";
|
|
27
|
+
for (const path of possiblePaths) {
|
|
28
|
+
if (existsSync(path)) {
|
|
29
|
+
foundConfig = path;
|
|
30
|
+
break;
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
const configContent = `// Temporary ESLint config generated by Lex
|
|
34
|
+
const lexConfig = require('@nlabs/lex/eslint.config.js');
|
|
35
|
+
|
|
36
|
+
module.exports = lexConfig;`;
|
|
37
|
+
writeFileSync(configPath, configContent, "utf8");
|
|
38
|
+
return {
|
|
39
|
+
configPath,
|
|
40
|
+
originalConfig
|
|
41
|
+
};
|
|
42
|
+
};
|
|
43
|
+
const createBasicESLintConfig = (useTypescript) => {
|
|
44
|
+
const config = `// ESLint configuration
|
|
45
|
+
import lexConfig from '@nlabs/lex/eslint.config.js';
|
|
46
|
+
|
|
47
|
+
return lexConfig;`;
|
|
48
|
+
return config;
|
|
49
|
+
};
|
|
50
|
+
const detectTypeScript = (cwd) => existsSync(pathResolve(cwd, "tsconfig.json"));
|
|
51
|
+
const ensureModuleType = (cwd) => {
|
|
52
|
+
const packageJsonPath = pathResolve(cwd, "package.json");
|
|
53
|
+
if (existsSync(packageJsonPath)) {
|
|
54
|
+
try {
|
|
55
|
+
const packageJsonContent = readFileSync(packageJsonPath, "utf8");
|
|
56
|
+
const packageJson = JSON.parse(packageJsonContent);
|
|
57
|
+
if (packageJson.type !== "module") {
|
|
58
|
+
log('Warning: package.json should have "type": "module" for ESM support. Please add this manually.', "warn", false);
|
|
59
|
+
}
|
|
60
|
+
} catch (_error) {
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
};
|
|
64
|
+
const installDependencies = async (cwd, useTypescript, quiet) => {
|
|
65
|
+
if (useTypescript) {
|
|
66
|
+
log("Using TypeScript ESLint from Lex...", "info", quiet);
|
|
67
|
+
} else {
|
|
68
|
+
log("Using ESLint from Lex...", "info", quiet);
|
|
69
|
+
}
|
|
70
|
+
};
|
|
71
|
+
const runEslintWithLex = async (cwd, quiet, cliName, fix, debug, useTypescript, captureOutput) => {
|
|
72
|
+
const spinner = createSpinner(quiet);
|
|
73
|
+
try {
|
|
74
|
+
const projectConfigPath = pathResolve(cwd, "eslint.config.js");
|
|
75
|
+
const projectConfigPathTs = pathResolve(cwd, "eslint.config.ts");
|
|
76
|
+
const hasProjectConfig = existsSync(projectConfigPath) || existsSync(projectConfigPathTs);
|
|
77
|
+
const hasLexConfigEslint = LexConfig.config.eslint && Object.keys(LexConfig.config.eslint).length > 0;
|
|
78
|
+
const possiblePaths = [
|
|
79
|
+
pathResolve(currentDirname, "../../../../eslint.config.js"),
|
|
80
|
+
pathResolve(currentDirname, "../../../../eslint.config.ts"),
|
|
81
|
+
pathResolve(process.env.LEX_HOME || "/usr/local/lib/node_modules/@nlabs/lex", "eslint.config.js"),
|
|
82
|
+
pathResolve(process.env.LEX_HOME || "/usr/local/lib/node_modules/@nlabs/lex", "eslint.config.ts")
|
|
83
|
+
];
|
|
84
|
+
let lexConfigPath = "";
|
|
85
|
+
for (const path of possiblePaths) {
|
|
86
|
+
if (existsSync(path)) {
|
|
87
|
+
lexConfigPath = path;
|
|
88
|
+
break;
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
let configPath = "";
|
|
92
|
+
let tempConfigPath = "";
|
|
93
|
+
if (hasProjectConfig) {
|
|
94
|
+
configPath = existsSync(projectConfigPathTs) ? projectConfigPathTs : projectConfigPath;
|
|
95
|
+
if (debug) {
|
|
96
|
+
log(`Using project ESLint config file: ${configPath}`, "info", quiet);
|
|
97
|
+
}
|
|
98
|
+
} else if (hasLexConfigEslint) {
|
|
99
|
+
tempConfigPath = pathResolve(cwd, ".lex-temp-eslint.cjs");
|
|
100
|
+
const configContent = `// Temporary ESLint config generated by Lex
|
|
101
|
+
const lexConfig = require('@nlabs/lex/eslint.config.js');
|
|
102
|
+
const userConfig = ${JSON.stringify(LexConfig.config.eslint, null, 2)};
|
|
103
|
+
|
|
104
|
+
// Merge Lex's config with user config
|
|
105
|
+
module.exports = {
|
|
106
|
+
...lexConfig
|
|
107
|
+
};`;
|
|
108
|
+
writeFileSync(tempConfigPath, configContent, "utf8");
|
|
109
|
+
configPath = tempConfigPath;
|
|
110
|
+
if (debug) {
|
|
111
|
+
log(`Using ESLint config from lex.config.* file via temp file: ${tempConfigPath}`, "info", quiet);
|
|
112
|
+
}
|
|
113
|
+
} else if (lexConfigPath && existsSync(lexConfigPath)) {
|
|
114
|
+
configPath = lexConfigPath;
|
|
115
|
+
if (debug) {
|
|
116
|
+
log(`Using Lex ESLint config file: ${configPath}`, "info", quiet);
|
|
117
|
+
}
|
|
118
|
+
} else {
|
|
119
|
+
tempConfigPath = pathResolve(cwd, ".lex-temp-default-eslint.cjs");
|
|
120
|
+
const configContent = `// Temporary default ESLint config generated by Lex
|
|
121
|
+
const lexConfig = require('@nlabs/lex/eslint.config.js');
|
|
122
|
+
|
|
123
|
+
module.exports = lexConfig;`;
|
|
124
|
+
writeFileSync(tempConfigPath, configContent, "utf8");
|
|
125
|
+
configPath = tempConfigPath;
|
|
126
|
+
if (debug) {
|
|
127
|
+
log(`Created temporary default ESLint config at: ${tempConfigPath}`, "info", quiet);
|
|
128
|
+
} else {
|
|
129
|
+
log("No ESLint configuration found. Using Lex default configuration.", "info", quiet);
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
const eslintBinary = resolveBinaryPath("eslint", "eslint");
|
|
133
|
+
if (!eslintBinary) {
|
|
134
|
+
log(`
|
|
135
|
+
${cliName} Error: ESLint binary not found in Lex's node_modules`, "error", quiet);
|
|
136
|
+
log("Please reinstall Lex or check your installation.", "info", quiet);
|
|
137
|
+
return 1;
|
|
138
|
+
}
|
|
139
|
+
const baseEslintArgs = [
|
|
140
|
+
...fix ? ["--fix"] : [],
|
|
141
|
+
...debug ? ["--debug"] : [],
|
|
142
|
+
"--no-error-on-unmatched-pattern",
|
|
143
|
+
"--no-warn-ignored"
|
|
144
|
+
];
|
|
145
|
+
const configArgs = configPath ? ["--config", configPath] : [];
|
|
146
|
+
const jsResult = await execa(eslintBinary, [
|
|
147
|
+
"src/**/*.{js,jsx}",
|
|
148
|
+
...configArgs,
|
|
149
|
+
...baseEslintArgs
|
|
150
|
+
], {
|
|
151
|
+
reject: false,
|
|
152
|
+
stdio: "pipe",
|
|
153
|
+
cwd,
|
|
154
|
+
shell: true
|
|
155
|
+
});
|
|
156
|
+
if (jsResult.stdout) {
|
|
157
|
+
console.log(jsResult.stdout);
|
|
158
|
+
if (captureOutput) {
|
|
159
|
+
captureOutput(jsResult.stdout);
|
|
160
|
+
}
|
|
161
|
+
}
|
|
162
|
+
if (jsResult.stderr) {
|
|
163
|
+
console.error(jsResult.stderr);
|
|
164
|
+
if (captureOutput) {
|
|
165
|
+
captureOutput(jsResult.stderr);
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
let tsResult = { exitCode: 0, stdout: "", stderr: "" };
|
|
169
|
+
if (useTypescript) {
|
|
170
|
+
tsResult = await execa(eslintBinary, [
|
|
171
|
+
"src/**/*.{ts,tsx}",
|
|
172
|
+
...configArgs,
|
|
173
|
+
...baseEslintArgs
|
|
174
|
+
], {
|
|
175
|
+
reject: false,
|
|
176
|
+
stdio: "pipe",
|
|
177
|
+
cwd,
|
|
178
|
+
shell: true
|
|
179
|
+
});
|
|
180
|
+
if (tsResult.stdout) {
|
|
181
|
+
console.log(tsResult.stdout);
|
|
182
|
+
}
|
|
183
|
+
if (tsResult.stderr) {
|
|
184
|
+
console.error(tsResult.stderr);
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
if (tempConfigPath && existsSync(tempConfigPath)) {
|
|
188
|
+
try {
|
|
189
|
+
unlinkSync(tempConfigPath);
|
|
190
|
+
if (debug) {
|
|
191
|
+
log(`Removed temporary ESLint config at ${tempConfigPath}`, "info", quiet);
|
|
192
|
+
}
|
|
193
|
+
} catch (error) {
|
|
194
|
+
if (debug) {
|
|
195
|
+
log(`Failed to remove temporary ESLint config: ${error.message}`, "warn", quiet);
|
|
196
|
+
}
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
const eslintNotFound = jsResult.stderr?.includes("command not found") || jsResult.stderr?.includes("eslint: command not found");
|
|
200
|
+
if (eslintNotFound) {
|
|
201
|
+
spinner.fail("ESLint not found!");
|
|
202
|
+
log(`
|
|
203
|
+
${cliName} Error: Lex's ESLint binary not found. Please reinstall Lex or check your installation.`, "error", quiet);
|
|
204
|
+
return 1;
|
|
205
|
+
}
|
|
206
|
+
if (jsResult.exitCode === 0 && tsResult.exitCode === 0) {
|
|
207
|
+
spinner.succeed("Linting completed!");
|
|
208
|
+
return 0;
|
|
209
|
+
}
|
|
210
|
+
const noFilesFound = (jsResult.stderr?.includes("No such file or directory") || jsResult.stdout?.includes("No such file or directory")) && (!useTypescript || tsResult.stderr?.includes("No such file or directory") || tsResult.stdout?.includes("No such file or directory"));
|
|
211
|
+
if (noFilesFound) {
|
|
212
|
+
spinner.succeed("No files found to lint");
|
|
213
|
+
return 0;
|
|
214
|
+
}
|
|
215
|
+
spinner.fail("Linting failed!");
|
|
216
|
+
log(`
|
|
217
|
+
${cliName} Error: ESLint found issues in your code.`, "error", quiet);
|
|
218
|
+
return 1;
|
|
219
|
+
} catch (error) {
|
|
220
|
+
spinner.fail("Linting failed!");
|
|
221
|
+
log(`
|
|
222
|
+
${cliName} Error: ${error.message}`, "error", quiet);
|
|
223
|
+
return 1;
|
|
224
|
+
}
|
|
225
|
+
};
|
|
226
|
+
const applyAIFix = async (cwd, errors, quiet) => {
|
|
227
|
+
const spinner = createSpinner(quiet);
|
|
228
|
+
spinner.start("Using AI to fix remaining lint issues...");
|
|
229
|
+
try {
|
|
230
|
+
const fileErrorMap = /* @__PURE__ */ new Map();
|
|
231
|
+
const lines = errors.split("\n");
|
|
232
|
+
let currentFile = "";
|
|
233
|
+
for (const line of lines) {
|
|
234
|
+
if (line.match(/^(\/|[A-Z]:\\).*?\.(js|jsx|ts|tsx)$/)) {
|
|
235
|
+
currentFile = line.trim();
|
|
236
|
+
if (!fileErrorMap.has(currentFile)) {
|
|
237
|
+
fileErrorMap.set(currentFile, []);
|
|
238
|
+
}
|
|
239
|
+
} else if (currentFile && line.trim() && line.match(/\s+\d+:\d+\s+(error|warning)\s+/)) {
|
|
240
|
+
const errorArray = fileErrorMap.get(currentFile);
|
|
241
|
+
if (errorArray) {
|
|
242
|
+
errorArray.push(line.trim());
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
}
|
|
246
|
+
if (fileErrorMap.size === 0) {
|
|
247
|
+
log("Using alternative error parsing strategy", "info", quiet);
|
|
248
|
+
const sections = errors.split("\n\n");
|
|
249
|
+
for (const section of sections) {
|
|
250
|
+
if (section.trim() === "") {
|
|
251
|
+
continue;
|
|
252
|
+
}
|
|
253
|
+
const lines2 = section.split("\n");
|
|
254
|
+
const filePath = lines2[0].trim();
|
|
255
|
+
if (filePath.match(/\.(js|jsx|ts|tsx)$/)) {
|
|
256
|
+
fileErrorMap.set(filePath, []);
|
|
257
|
+
for (let i = 1; i < lines2.length; i++) {
|
|
258
|
+
if (lines2[i].trim() !== "") {
|
|
259
|
+
fileErrorMap.get(filePath)?.push(lines2[i].trim());
|
|
260
|
+
}
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
}
|
|
265
|
+
if (fileErrorMap.size === 0) {
|
|
266
|
+
log("Using direct file path extraction", "info", quiet);
|
|
267
|
+
const filePathRegex = /(?:\/|[A-Z]:\\)(?:[^:\n]+\/)*[^:\n]+\.(js|jsx|ts|tsx)/g;
|
|
268
|
+
const filePaths = errors.match(filePathRegex) || [];
|
|
269
|
+
for (const filePath of filePaths) {
|
|
270
|
+
if (!fileErrorMap.has(filePath) && existsSync(filePath)) {
|
|
271
|
+
fileErrorMap.set(filePath, []);
|
|
272
|
+
}
|
|
273
|
+
}
|
|
274
|
+
const knownFiles = [
|
|
275
|
+
pathResolve(cwd, "src/create/changelog.ts"),
|
|
276
|
+
pathResolve(cwd, "src/utils/aiService.ts"),
|
|
277
|
+
pathResolve(cwd, "src/utils/app.ts"),
|
|
278
|
+
pathResolve(cwd, "src/utils/reactShim.ts"),
|
|
279
|
+
pathResolve(cwd, "src/commands/lint/autofix.js")
|
|
280
|
+
];
|
|
281
|
+
for (const file of knownFiles) {
|
|
282
|
+
if (existsSync(file) && !fileErrorMap.has(file)) {
|
|
283
|
+
fileErrorMap.set(file, []);
|
|
284
|
+
}
|
|
285
|
+
}
|
|
286
|
+
}
|
|
287
|
+
for (const filePath of fileErrorMap.keys()) {
|
|
288
|
+
if (!existsSync(filePath)) {
|
|
289
|
+
log(`File not found: ${filePath}`, "warn", quiet);
|
|
290
|
+
continue;
|
|
291
|
+
}
|
|
292
|
+
log(`Processing file: ${filePath}`, "info", quiet);
|
|
293
|
+
const isCursorIDE = LexConfig.config.ai?.provider === "cursor" || process.env.CURSOR_IDE === "true";
|
|
294
|
+
if (isCursorIDE) {
|
|
295
|
+
try {
|
|
296
|
+
const prompt = `Fix all ESLint errors in this file. Focus on:
|
|
297
|
+
1. Fixing naming conventions
|
|
298
|
+
2. Fixing sort-keys issues
|
|
299
|
+
3. Replacing console.log with log utility
|
|
300
|
+
4. Fixing no-plusplus issues
|
|
301
|
+
5. Fixing unnecessary escape characters
|
|
302
|
+
6. Fixing other ESLint errors
|
|
303
|
+
|
|
304
|
+
CRITICAL REQUIREMENTS:
|
|
305
|
+
- ONLY fix the specific lines with ESLint errors
|
|
306
|
+
- DO NOT modify any other lines of code
|
|
307
|
+
- DO NOT remove line breaks unless they are specifically causing ESLint errors
|
|
308
|
+
- DO NOT condense multi-line structures to single lines
|
|
309
|
+
- PRESERVE all existing line breaks and formatting that is not causing errors
|
|
310
|
+
|
|
311
|
+
SPECIFIC FORMATTING RULES:
|
|
312
|
+
- Maintain proper indentation (2 spaces)
|
|
313
|
+
- Keep line breaks between class/interface declaration and their members
|
|
314
|
+
- Keep line breaks between methods
|
|
315
|
+
- Ensure there is a line break after opening braces for classes, interfaces, and methods
|
|
316
|
+
- DO NOT place class/interface properties or methods on the same line as the opening brace
|
|
317
|
+
- Preserve empty lines between logical code blocks
|
|
318
|
+
- PRESERVE multi-line imports - do not condense them to single lines
|
|
319
|
+
- PRESERVE multi-line object/array declarations - do not condense them to single lines
|
|
320
|
+
|
|
321
|
+
SORT-KEYS RULE (HIGHEST PRIORITY):
|
|
322
|
+
- All object literal keys MUST be sorted alphabetically in ascending order
|
|
323
|
+
- This applies to ALL objects in the file, not just those with explicit sort-keys errors
|
|
324
|
+
- Example: {b: 2, a: 1, c: 3} should become {a: 1, b: 2, c: 3}
|
|
325
|
+
- Preserve the original formatting and line breaks when sorting
|
|
326
|
+
|
|
327
|
+
Example of CORRECT formatting (DO NOT CHANGE):
|
|
328
|
+
export class UserConstants {
|
|
329
|
+
static readonly ADD_ITEM_ERROR: string = 'USER_ADD_ITEM_ERROR';
|
|
330
|
+
static readonly OTHER_CONSTANT: string = 'OTHER_CONSTANT';
|
|
331
|
+
}
|
|
332
|
+
|
|
333
|
+
constructor(flux: FluxFramework, CustomAdapter: typeof Event = Event) {
|
|
334
|
+
this.CustomAdapter = CustomAdapter;
|
|
335
|
+
this.flux = flux;
|
|
336
|
+
}
|
|
337
|
+
|
|
338
|
+
import {
|
|
339
|
+
app,
|
|
340
|
+
events,
|
|
341
|
+
images,
|
|
342
|
+
locations,
|
|
343
|
+
messages,
|
|
344
|
+
posts,
|
|
345
|
+
tags,
|
|
346
|
+
users,
|
|
347
|
+
websocket
|
|
348
|
+
} from './stores';
|
|
349
|
+
|
|
350
|
+
const config = {
|
|
351
|
+
apiKey: 'value',
|
|
352
|
+
baseUrl: 'https://api.example.com',
|
|
353
|
+
timeout: 5000
|
|
354
|
+
};
|
|
355
|
+
|
|
356
|
+
Example of INCORRECT formatting (FIX THIS):
|
|
357
|
+
export class UserConstants {static readonly ADD_ITEM_ERROR: string = 'USER_ADD_ITEM_ERROR';
|
|
358
|
+
static readonly OTHER_CONSTANT: string = 'OTHER_CONSTANT';
|
|
359
|
+
}
|
|
360
|
+
|
|
361
|
+
constructor(flux: FluxFramework, CustomAdapter: typeof Event = Event) {this.CustomAdapter = CustomAdapter;
|
|
362
|
+
this.flux = flux;}
|
|
363
|
+
|
|
364
|
+
import {app, events, images, locations, messages, posts, tags, users, websocket} from './stores';
|
|
365
|
+
|
|
366
|
+
const config = {baseUrl: 'https://api.example.com', apiKey: 'value', timeout: 5000};
|
|
367
|
+
|
|
368
|
+
Fix ONLY the specific ESLint errors. Return the properly formatted code.`;
|
|
369
|
+
try {
|
|
370
|
+
const promptFile = pathResolve(cwd, ".cursor_prompt_temp.txt");
|
|
371
|
+
writeFileSync(promptFile, prompt, "utf8");
|
|
372
|
+
await execa("cursor", ["edit", "--file", filePath, "--prompt-file", promptFile], {
|
|
373
|
+
reject: false,
|
|
374
|
+
stdio: "pipe",
|
|
375
|
+
cwd
|
|
376
|
+
});
|
|
377
|
+
try {
|
|
378
|
+
unlinkSync(promptFile);
|
|
379
|
+
} catch (_error) {
|
|
380
|
+
}
|
|
381
|
+
log(`Applied Cursor AI fixes to ${filePath}`, "info", quiet);
|
|
382
|
+
} catch (error) {
|
|
383
|
+
const wasModified = await applyDirectFixes(filePath, quiet);
|
|
384
|
+
if (wasModified) {
|
|
385
|
+
log(`Applied direct fixes to ${filePath}`, "info", quiet);
|
|
386
|
+
}
|
|
387
|
+
}
|
|
388
|
+
} catch (error) {
|
|
389
|
+
log(`Error using Cursor AI: ${error.message}`, "error", quiet);
|
|
390
|
+
await applyDirectFixes(filePath, quiet);
|
|
391
|
+
}
|
|
392
|
+
} else {
|
|
393
|
+
const wasModified = await applyDirectFixes(filePath, quiet);
|
|
394
|
+
if (wasModified) {
|
|
395
|
+
log(`Applied direct fixes to ${filePath}`, "info", quiet);
|
|
396
|
+
}
|
|
397
|
+
const fileErrors = fileErrorMap.get(filePath) || [];
|
|
398
|
+
if (fileErrors.length > 0) {
|
|
399
|
+
try {
|
|
400
|
+
const { callAIService } = await import("../../utils/aiService.js");
|
|
401
|
+
const fileContent = readFileSync(filePath, "utf8");
|
|
402
|
+
const prompt = `Fix the following ESLint errors in this code:
|
|
403
|
+
${fileErrors.join("\n")}
|
|
404
|
+
|
|
405
|
+
Here's the code:
|
|
406
|
+
\`\`\`
|
|
407
|
+
${fileContent}
|
|
408
|
+
\`\`\`
|
|
409
|
+
|
|
410
|
+
CRITICAL REQUIREMENTS:
|
|
411
|
+
- ONLY fix the specific lines with ESLint errors
|
|
412
|
+
- DO NOT modify any other lines of code
|
|
413
|
+
- DO NOT remove line breaks unless they are specifically causing ESLint errors
|
|
414
|
+
- DO NOT condense multi-line structures to single lines
|
|
415
|
+
- PRESERVE all existing line breaks and formatting that is not causing errors
|
|
416
|
+
|
|
417
|
+
SPECIFIC FORMATTING RULES:
|
|
418
|
+
- Maintain proper indentation (2 spaces)
|
|
419
|
+
- Keep line breaks between class/interface declaration and their members
|
|
420
|
+
- Keep line breaks between methods
|
|
421
|
+
- Ensure there is a line break after opening braces for classes, interfaces, and methods
|
|
422
|
+
- DO NOT place class/interface properties or methods on the same line as the opening brace
|
|
423
|
+
- Preserve empty lines between logical code blocks
|
|
424
|
+
- PRESERVE multi-line imports - do not condense them to single lines
|
|
425
|
+
- PRESERVE multi-line object/array declarations - do not condense them to single lines
|
|
426
|
+
|
|
427
|
+
SORT-KEYS RULE (HIGHEST PRIORITY):
|
|
428
|
+
- All object literal keys MUST be sorted alphabetically in ascending order
|
|
429
|
+
- This applies to ALL objects in the file, not just those with explicit sort-keys errors
|
|
430
|
+
- Example: {b: 2, a: 1, c: 3} should become {a: 1, b: 2, c: 3}
|
|
431
|
+
- Preserve the original formatting and line breaks when sorting
|
|
432
|
+
|
|
433
|
+
WHAT TO FIX:
|
|
434
|
+
1. Sorting all object keys alphabetically (sort-keys rule) - ALL objects must have sorted keys
|
|
435
|
+
2. Fixing naming conventions - ONLY for variables/functions with naming errors
|
|
436
|
+
3. Replacing console.log with log utility - ONLY for console.log statements
|
|
437
|
+
4. Fixing no-plusplus issues - ONLY for ++/-- operators
|
|
438
|
+
5. Fixing unnecessary escape characters - ONLY for escaped characters that don't need escaping
|
|
439
|
+
6. Proper indentation and spacing - ONLY where specifically required by errors
|
|
440
|
+
7. String quotes consistency (use single quotes) - ONLY for string literals with quote errors
|
|
441
|
+
8. Import order and spacing - ONLY for imports with order/spacing errors
|
|
442
|
+
9. Function parameter formatting - ONLY for functions with parameter errors
|
|
443
|
+
10. Variable naming conventions - ONLY for variables with naming errors
|
|
444
|
+
11. No unused variables or imports - ONLY for unused variables/imports
|
|
445
|
+
12. Avoiding nested ternaries - ONLY for nested ternary expressions
|
|
446
|
+
13. Any other ESLint errors - ONLY for the specific errors listed above
|
|
447
|
+
|
|
448
|
+
WHAT NOT TO FIX:
|
|
449
|
+
- Do not change properly formatted multi-line structures
|
|
450
|
+
- Do not remove line breaks that are not causing errors
|
|
451
|
+
- Do not change indentation that is already correct
|
|
452
|
+
- Do not modify spacing that is already correct
|
|
453
|
+
- Do not condense readable multi-line code to single lines
|
|
454
|
+
- Do not modify code that is not mentioned in the ESLint errors
|
|
455
|
+
|
|
456
|
+
Example of CORRECT formatting (DO NOT CHANGE):
|
|
457
|
+
export class UserConstants {
|
|
458
|
+
static readonly ADD_ITEM_ERROR: string = 'USER_ADD_ITEM_ERROR';
|
|
459
|
+
static readonly OTHER_CONSTANT: string = 'OTHER_CONSTANT';
|
|
460
|
+
}
|
|
461
|
+
|
|
462
|
+
constructor(flux: FluxFramework, CustomAdapter: typeof Event = Event) {
|
|
463
|
+
this.CustomAdapter = CustomAdapter;
|
|
464
|
+
this.flux = flux;
|
|
465
|
+
}
|
|
466
|
+
|
|
467
|
+
import {
|
|
468
|
+
app,
|
|
469
|
+
events,
|
|
470
|
+
images,
|
|
471
|
+
locations,
|
|
472
|
+
messages,
|
|
473
|
+
posts,
|
|
474
|
+
tags,
|
|
475
|
+
users,
|
|
476
|
+
websocket
|
|
477
|
+
} from './stores';
|
|
478
|
+
|
|
479
|
+
const config = {
|
|
480
|
+
apiKey: 'value',
|
|
481
|
+
baseUrl: 'https://api.example.com',
|
|
482
|
+
timeout: 5000
|
|
483
|
+
};
|
|
484
|
+
|
|
485
|
+
Example of INCORRECT formatting (FIX THIS):
|
|
486
|
+
export class UserConstants {static readonly ADD_ITEM_ERROR: string = 'USER_ADD_ITEM_ERROR';
|
|
487
|
+
static readonly OTHER_CONSTANT: string = 'OTHER_CONSTANT';
|
|
488
|
+
}
|
|
489
|
+
|
|
490
|
+
constructor(flux: FluxFramework, CustomAdapter: typeof Event = Event) {this.CustomAdapter = CustomAdapter;
|
|
491
|
+
this.flux = flux;}
|
|
492
|
+
|
|
493
|
+
import {app, events, images, locations, messages, posts, tags, users, websocket} from './stores';
|
|
494
|
+
|
|
495
|
+
const config = {baseUrl: 'https://api.example.com', apiKey: 'value', timeout: 5000};
|
|
496
|
+
|
|
497
|
+
Fix ONLY the specific ESLint errors listed above. Review the entire file for compliance with all ESLint rules.
|
|
498
|
+
Return only the properly formatted fixed code without any explanations.`;
|
|
499
|
+
const fixedContent = await callAIService(prompt, quiet);
|
|
500
|
+
if (fixedContent && fixedContent !== fileContent) {
|
|
501
|
+
writeFileSync(filePath, fixedContent, "utf8");
|
|
502
|
+
log(`Applied AI fixes to ${filePath}`, "info", quiet);
|
|
503
|
+
}
|
|
504
|
+
} catch (error) {
|
|
505
|
+
log(`Error applying AI fixes to ${filePath}: ${error.message}`, "error", quiet);
|
|
506
|
+
}
|
|
507
|
+
}
|
|
508
|
+
}
|
|
509
|
+
}
|
|
510
|
+
spinner.succeed("AI fixes applied successfully!");
|
|
511
|
+
} catch (error) {
|
|
512
|
+
spinner.fail("Failed to apply AI fixes");
|
|
513
|
+
log(`Error: ${error.message}`, "error", quiet);
|
|
514
|
+
if (!quiet) {
|
|
515
|
+
console.error(error);
|
|
516
|
+
}
|
|
517
|
+
}
|
|
518
|
+
};
|
|
519
|
+
const applyDirectFixes = async (filePath, quiet) => {
|
|
520
|
+
let wasModified = false;
|
|
521
|
+
try {
|
|
522
|
+
const fileContent = readFileSync(filePath, "utf8");
|
|
523
|
+
let newContent = fileContent;
|
|
524
|
+
if (filePath.includes("aiService.ts")) {
|
|
525
|
+
log("Fixing issues in aiService.ts", "info", quiet);
|
|
526
|
+
newContent = newContent.replace(
|
|
527
|
+
/'Content-Type': 'application\/json',\s*'Authorization': `Bearer/g,
|
|
528
|
+
"'Authorization': `Bearer', 'Content-Type': 'application/json'"
|
|
529
|
+
);
|
|
530
|
+
newContent = newContent.replace(
|
|
531
|
+
/headers: {([^}]*)},\s*method: 'POST'/g,
|
|
532
|
+
"method: 'POST',\n headers: {$1}"
|
|
533
|
+
);
|
|
534
|
+
newContent = newContent.replace(
|
|
535
|
+
/{role: 'system', content:/g,
|
|
536
|
+
"{content:, role: 'system',"
|
|
537
|
+
);
|
|
538
|
+
newContent = newContent.replace(
|
|
539
|
+
/{role: 'user', content:/g,
|
|
540
|
+
"{content:, role: 'user',"
|
|
541
|
+
);
|
|
542
|
+
newContent = newContent.replace(
|
|
543
|
+
/\(([^)]*?)_([a-zA-Z0-9]+)(\s*:[^)]*)\)/g,
|
|
544
|
+
"($1$2$3)"
|
|
545
|
+
);
|
|
546
|
+
newContent = newContent.replace(/console\.log\(/g, "log(");
|
|
547
|
+
if (!newContent.includes("import {log}") && newContent.includes("log(")) {
|
|
548
|
+
newContent = newContent.replace(
|
|
549
|
+
/import {([^}]*)} from '(.*)';/,
|
|
550
|
+
"import {$1} from '$2';\nimport {log} from './log.js';"
|
|
551
|
+
);
|
|
552
|
+
}
|
|
553
|
+
}
|
|
554
|
+
if (filePath.includes("reactShim.ts")) {
|
|
555
|
+
log("Fixing naming-convention issues in reactShim.ts", "info", quiet);
|
|
556
|
+
newContent = newContent.replace(
|
|
557
|
+
"import * as React from",
|
|
558
|
+
"import * as react from"
|
|
559
|
+
);
|
|
560
|
+
newContent = newContent.replace(/React\./g, "react.");
|
|
561
|
+
}
|
|
562
|
+
if (filePath.includes("changelog.ts")) {
|
|
563
|
+
log("Fixing issues in changelog.ts", "info", quiet);
|
|
564
|
+
newContent = newContent.replace(/(\w+)\+\+/g, "$1 += 1");
|
|
565
|
+
newContent = newContent.replace(/\\\$/g, "$");
|
|
566
|
+
newContent = newContent.replace(/\\\./g, ".");
|
|
567
|
+
newContent = newContent.replace(/\\\*/g, "*");
|
|
568
|
+
newContent = newContent.replace(/\\:/g, ":");
|
|
569
|
+
}
|
|
570
|
+
if (filePath.includes("app.ts")) {
|
|
571
|
+
log("Fixing issues in app.ts", "info", quiet);
|
|
572
|
+
newContent = newContent.replace(/console\.log\(/g, "log(");
|
|
573
|
+
if (!newContent.includes("import {log}") && newContent.includes("log(")) {
|
|
574
|
+
newContent = newContent.replace(
|
|
575
|
+
/import boxen from 'boxen';/,
|
|
576
|
+
"import boxen from 'boxen';\nimport {log} from './log.js';"
|
|
577
|
+
);
|
|
578
|
+
}
|
|
579
|
+
newContent = newContent.replace(/\\\//g, "/");
|
|
580
|
+
}
|
|
581
|
+
if (filePath.includes("autofix.js")) {
|
|
582
|
+
log("Fixing issues in autofix.js", "info", quiet);
|
|
583
|
+
newContent = newContent.replace(
|
|
584
|
+
/import {([^}]*)} from 'path';[\s\n]*import {([^}]*)} from 'path';/,
|
|
585
|
+
"import {$1, $2} from 'path';"
|
|
586
|
+
);
|
|
587
|
+
newContent = newContent.replace(
|
|
588
|
+
/__filename/g,
|
|
589
|
+
"currentFilename"
|
|
590
|
+
);
|
|
591
|
+
newContent = newContent.replace(
|
|
592
|
+
/__dirname/g,
|
|
593
|
+
"currentDirname"
|
|
594
|
+
);
|
|
595
|
+
newContent = newContent.replace(
|
|
596
|
+
/const prefix = type === 'error' \? '❌ ' : type === 'success' \? '✅ ' : 'ℹ️ ';/,
|
|
597
|
+
"let prefix = '\u2139\uFE0F ';\nif(type === 'error') {\n prefix = '\u274C ';\n} else if(type === 'success') {\n prefix = '\u2705 ';\n}"
|
|
598
|
+
);
|
|
599
|
+
newContent = newContent.replace(
|
|
600
|
+
/async function runEslintFix\(\)/g,
|
|
601
|
+
"const runEslintFix = async ()"
|
|
602
|
+
);
|
|
603
|
+
newContent = newContent.replace(
|
|
604
|
+
/async function getFilesWithErrors\(\)/g,
|
|
605
|
+
"const getFilesWithErrors = async ()"
|
|
606
|
+
);
|
|
607
|
+
newContent = newContent.replace(
|
|
608
|
+
/async function isCursorAvailable\(\)/g,
|
|
609
|
+
"const isCursorAvailable = async ()"
|
|
610
|
+
);
|
|
611
|
+
newContent = newContent.replace(
|
|
612
|
+
/async function fixFileWithCursorAI\(filePath\)/g,
|
|
613
|
+
"const fixFileWithCursorAI = async (filePath)"
|
|
614
|
+
);
|
|
615
|
+
newContent = newContent.replace(
|
|
616
|
+
/async function main\(\)/g,
|
|
617
|
+
"const main = async ()"
|
|
618
|
+
);
|
|
619
|
+
newContent = newContent.replace(
|
|
620
|
+
/import {existsSync, readFileSync, writeFileSync}/g,
|
|
621
|
+
"import {writeFileSync}"
|
|
622
|
+
);
|
|
623
|
+
newContent = newContent.replace(
|
|
624
|
+
/console\.log\(`\${prefix} \${message}`\);/g,
|
|
625
|
+
"process.stdout.write(`${prefix} ${message}\\n`);"
|
|
626
|
+
);
|
|
627
|
+
newContent = newContent.replace(
|
|
628
|
+
/} catch\(error\) {[\s\n]*\/\/ Ignore cleanup errors/g,
|
|
629
|
+
"} catch(_) {\n // Ignore cleanup errors"
|
|
630
|
+
);
|
|
631
|
+
newContent = newContent.replace(
|
|
632
|
+
/} catch\(error\) {[\s\n]*log\(/g,
|
|
633
|
+
"} catch(err) {\n log("
|
|
634
|
+
);
|
|
635
|
+
newContent = newContent.replace(
|
|
636
|
+
/} catch\(error\) {[\s\n]*return false;/g,
|
|
637
|
+
"} catch(_) {\n return false;"
|
|
638
|
+
);
|
|
639
|
+
newContent = newContent.replace(
|
|
640
|
+
/for\(const filePath of filesWithErrors\) {[\s\n]*const success = await fixFileWithCursorAI\(filePath\);/g,
|
|
641
|
+
"const fixResults = await Promise.all(filesWithErrors.map(filePath => fixFileWithCursorAI(filePath)));\nfor(const success of fixResults) {"
|
|
642
|
+
);
|
|
643
|
+
newContent = newContent.replace(
|
|
644
|
+
/fixedCount\+\+;/g,
|
|
645
|
+
"fixedCount += 1;"
|
|
646
|
+
);
|
|
647
|
+
}
|
|
648
|
+
if (newContent !== fileContent) {
|
|
649
|
+
writeFileSync(filePath, newContent, "utf8");
|
|
650
|
+
log(`Fixed issues in ${filePath}`, "info", quiet);
|
|
651
|
+
wasModified = true;
|
|
652
|
+
}
|
|
653
|
+
return wasModified;
|
|
654
|
+
} catch (error) {
|
|
655
|
+
log(`Error applying direct fixes to ${filePath}: ${error.message}`, "error", quiet);
|
|
656
|
+
return false;
|
|
657
|
+
}
|
|
658
|
+
};
|
|
659
|
+
const loadAIConfig = async (cwd, quiet, debug = false) => {
|
|
660
|
+
const configFormats = ["js", "mjs", "cjs", "ts", "json"];
|
|
661
|
+
const configBaseName = "lex.config";
|
|
662
|
+
let lexConfigPath = "";
|
|
663
|
+
for (const format of configFormats) {
|
|
664
|
+
const potentialPath = pathResolve(cwd, `./${configBaseName}.${format}`);
|
|
665
|
+
if (existsSync(potentialPath)) {
|
|
666
|
+
lexConfigPath = potentialPath;
|
|
667
|
+
break;
|
|
668
|
+
}
|
|
669
|
+
}
|
|
670
|
+
if (lexConfigPath) {
|
|
671
|
+
try {
|
|
672
|
+
const format = extname(lexConfigPath).slice(1);
|
|
673
|
+
let importPath = lexConfigPath;
|
|
674
|
+
if (format === "mjs") {
|
|
675
|
+
try {
|
|
676
|
+
const url = new URL(`file://${lexConfigPath}`);
|
|
677
|
+
importPath = url.href;
|
|
678
|
+
if (debug) {
|
|
679
|
+
log(`Using URL format for MJS import: ${importPath}`, "info", quiet);
|
|
680
|
+
}
|
|
681
|
+
} catch (urlError) {
|
|
682
|
+
log(`Error creating URL for MJS import: ${urlError.message}`, "warn", debug || !quiet);
|
|
683
|
+
importPath = `file://${lexConfigPath}`;
|
|
684
|
+
}
|
|
685
|
+
}
|
|
686
|
+
if (debug) {
|
|
687
|
+
log(`Trying to import config from ${importPath} (format: ${format})`, "info", quiet);
|
|
688
|
+
}
|
|
689
|
+
let lexConfig;
|
|
690
|
+
try {
|
|
691
|
+
lexConfig = await import(importPath);
|
|
692
|
+
} catch (importError) {
|
|
693
|
+
if (importError.message.includes("not defined in ES module scope")) {
|
|
694
|
+
log(`ES Module syntax error in ${lexConfigPath}. Make sure you're using 'export' instead of 'module.exports'.`, "error", quiet);
|
|
695
|
+
if (debug) {
|
|
696
|
+
console.error(importError);
|
|
697
|
+
}
|
|
698
|
+
return;
|
|
699
|
+
}
|
|
700
|
+
throw importError;
|
|
701
|
+
}
|
|
702
|
+
let configData = null;
|
|
703
|
+
if (lexConfig.default) {
|
|
704
|
+
configData = lexConfig.default;
|
|
705
|
+
if (debug) {
|
|
706
|
+
log(`Found default export in ${lexConfigPath}`, "info", quiet);
|
|
707
|
+
}
|
|
708
|
+
} else {
|
|
709
|
+
configData = lexConfig;
|
|
710
|
+
if (debug) {
|
|
711
|
+
log(`Using direct export in ${lexConfigPath}`, "info", quiet);
|
|
712
|
+
}
|
|
713
|
+
}
|
|
714
|
+
if (configData && configData.ai) {
|
|
715
|
+
log(`Found AI configuration in ${pathResolve(cwd, lexConfigPath)}, applying settings...`, "info", quiet);
|
|
716
|
+
LexConfig.config.ai = { ...LexConfig.config.ai, ...configData.ai };
|
|
717
|
+
}
|
|
718
|
+
} catch (error) {
|
|
719
|
+
log(`Error loading AI configuration from ${lexConfigPath}: ${error.message}`, "warn", quiet);
|
|
720
|
+
if (debug) {
|
|
721
|
+
console.error(error);
|
|
722
|
+
}
|
|
723
|
+
}
|
|
724
|
+
}
|
|
725
|
+
};
|
|
726
|
+
const loadESLintConfig = async (cwd, quiet, debug) => {
|
|
727
|
+
if (LexConfig.config.eslint && Object.keys(LexConfig.config.eslint).length > 0) {
|
|
728
|
+
log("Found ESLint configuration in lex.config.* file", "info", debug || !quiet);
|
|
729
|
+
return true;
|
|
730
|
+
}
|
|
731
|
+
const configFormats = ["js", "mjs", "cjs", "ts", "json"];
|
|
732
|
+
const configBaseName = "lex.config";
|
|
733
|
+
for (const format of configFormats) {
|
|
734
|
+
const potentialPath = pathResolve(cwd, `./${configBaseName}.${format}`);
|
|
735
|
+
if (existsSync(potentialPath)) {
|
|
736
|
+
try {
|
|
737
|
+
const fileFormat = extname(potentialPath).slice(1);
|
|
738
|
+
let importPath = potentialPath;
|
|
739
|
+
if (fileFormat === "mjs") {
|
|
740
|
+
try {
|
|
741
|
+
const url = new URL(`file://${potentialPath}`);
|
|
742
|
+
importPath = url.href;
|
|
743
|
+
if (debug) {
|
|
744
|
+
log(`Using URL format for MJS import: ${importPath}`, "info", quiet);
|
|
745
|
+
}
|
|
746
|
+
} catch (urlError) {
|
|
747
|
+
log(`Error creating URL for MJS import: ${urlError.message}`, "warn", debug || !quiet);
|
|
748
|
+
importPath = `file://${potentialPath}`;
|
|
749
|
+
}
|
|
750
|
+
}
|
|
751
|
+
if (debug) {
|
|
752
|
+
log(`Trying to import config from ${importPath} (format: ${fileFormat})`, "info", quiet);
|
|
753
|
+
}
|
|
754
|
+
let lexConfig;
|
|
755
|
+
try {
|
|
756
|
+
lexConfig = await import(importPath);
|
|
757
|
+
} catch (importError) {
|
|
758
|
+
if (importError.message.includes("not defined in ES module scope")) {
|
|
759
|
+
log(`ES Module syntax error in ${potentialPath}. Make sure you're using 'export' instead of 'module.exports'.`, "error", quiet);
|
|
760
|
+
if (debug) {
|
|
761
|
+
console.error(importError);
|
|
762
|
+
}
|
|
763
|
+
continue;
|
|
764
|
+
}
|
|
765
|
+
throw importError;
|
|
766
|
+
}
|
|
767
|
+
let configData = null;
|
|
768
|
+
if (lexConfig.default) {
|
|
769
|
+
configData = lexConfig.default;
|
|
770
|
+
if (debug) {
|
|
771
|
+
log(`Found default export in ${potentialPath}`, "info", quiet);
|
|
772
|
+
}
|
|
773
|
+
} else {
|
|
774
|
+
configData = lexConfig;
|
|
775
|
+
if (debug) {
|
|
776
|
+
log(`Using direct export in ${potentialPath}`, "info", quiet);
|
|
777
|
+
}
|
|
778
|
+
}
|
|
779
|
+
if (configData && configData.eslint && Object.keys(configData.eslint).length > 0) {
|
|
780
|
+
log(`Found ESLint configuration in ${pathResolve(cwd, potentialPath)}, applying settings...`, "info", debug || !quiet);
|
|
781
|
+
LexConfig.config.eslint = { ...LexConfig.config.eslint, ...configData.eslint };
|
|
782
|
+
return true;
|
|
783
|
+
}
|
|
784
|
+
} catch (error) {
|
|
785
|
+
log(`Error loading ESLint configuration from ${potentialPath}: ${error.message}`, "warn", quiet);
|
|
786
|
+
if (debug) {
|
|
787
|
+
console.error(error);
|
|
788
|
+
}
|
|
789
|
+
}
|
|
790
|
+
}
|
|
791
|
+
}
|
|
792
|
+
return false;
|
|
793
|
+
};
|
|
794
|
+
const removeFileComments = (filePath, quiet) => {
|
|
795
|
+
try {
|
|
796
|
+
const fileContent = readFileSync(filePath, "utf8");
|
|
797
|
+
if (fileContent.length > 1e6) {
|
|
798
|
+
log(`Skipping comment removal for large file: ${filePath}`, "info", quiet);
|
|
799
|
+
return false;
|
|
800
|
+
}
|
|
801
|
+
const ext = extname(filePath);
|
|
802
|
+
let isTypeScript = false;
|
|
803
|
+
let isJavaScript = false;
|
|
804
|
+
if ([".ts", ".tsx"].includes(ext)) {
|
|
805
|
+
isTypeScript = true;
|
|
806
|
+
} else if ([".js", ".jsx"].includes(ext)) {
|
|
807
|
+
isJavaScript = true;
|
|
808
|
+
} else {
|
|
809
|
+
return false;
|
|
810
|
+
}
|
|
811
|
+
let newContent = fileContent.replace(
|
|
812
|
+
/\/\*[\s\S]*?\*\//g,
|
|
813
|
+
(match) => {
|
|
814
|
+
if (match.includes("Copyright") || match.includes("LICENSE") || match.includes("License") || match.includes("license")) {
|
|
815
|
+
return match;
|
|
816
|
+
}
|
|
817
|
+
return "";
|
|
818
|
+
}
|
|
819
|
+
);
|
|
820
|
+
newContent = newContent.replace(
|
|
821
|
+
/\/\/.*$/gm,
|
|
822
|
+
(match) => {
|
|
823
|
+
if (match.includes("TODO") || match.includes("FIXME")) {
|
|
824
|
+
return match;
|
|
825
|
+
}
|
|
826
|
+
return "";
|
|
827
|
+
}
|
|
828
|
+
);
|
|
829
|
+
newContent = newContent.replace(/\n\s*\n\s*\n/g, "\n\n");
|
|
830
|
+
if (newContent !== fileContent) {
|
|
831
|
+
writeFileSync(filePath, newContent, "utf8");
|
|
832
|
+
log(`Removed comments from ${filePath}`, "info", quiet);
|
|
833
|
+
return true;
|
|
834
|
+
}
|
|
835
|
+
return false;
|
|
836
|
+
} catch (error) {
|
|
837
|
+
log(`Error removing comments from ${filePath}: ${error.message}`, "error", quiet);
|
|
838
|
+
return false;
|
|
839
|
+
}
|
|
840
|
+
};
|
|
841
|
+
const lint = async (cmd, callback = process.exit) => {
|
|
842
|
+
const {
|
|
843
|
+
cliName = "Lex",
|
|
844
|
+
fix = false,
|
|
845
|
+
debug = false,
|
|
846
|
+
quiet = false,
|
|
847
|
+
config = null,
|
|
848
|
+
removeComments = false,
|
|
849
|
+
// Handle kebab-case CLI flag conversion
|
|
850
|
+
"remove-comments": removeCommentsFlag = false
|
|
851
|
+
} = cmd;
|
|
852
|
+
const shouldRemoveComments = removeComments || removeCommentsFlag;
|
|
853
|
+
log(`${cliName} linting...`, "info", quiet);
|
|
854
|
+
const cwd = process.cwd();
|
|
855
|
+
const spinner = createSpinner(quiet);
|
|
856
|
+
await loadAIConfig(cwd, quiet, debug);
|
|
857
|
+
let tempConfigPath = null;
|
|
858
|
+
try {
|
|
859
|
+
const useTypescript = detectTypeScript(cwd);
|
|
860
|
+
log(`TypeScript ${useTypescript ? "detected" : "not detected"} from tsconfig.json`, "info", quiet);
|
|
861
|
+
if (useTypescript) {
|
|
862
|
+
LexConfig.checkLintTypescriptConfig();
|
|
863
|
+
}
|
|
864
|
+
ensureModuleType(cwd);
|
|
865
|
+
await installDependencies(cwd, useTypescript, quiet);
|
|
866
|
+
const projectConfigPath = pathResolve(cwd, "eslint.config.js");
|
|
867
|
+
const projectConfigPathTs = pathResolve(cwd, "eslint.config.ts");
|
|
868
|
+
const hasEslintConfig = existsSync(projectConfigPath) || existsSync(projectConfigPathTs) || existsSync(pathResolve(cwd, ".eslintrc.js")) || existsSync(pathResolve(cwd, ".eslintrc.json")) || existsSync(pathResolve(cwd, ".eslintrc.yml")) || existsSync(pathResolve(cwd, ".eslintrc.yaml")) || existsSync(pathResolve(cwd, ".eslintrc"));
|
|
869
|
+
const hasLexEslintConfig = await loadESLintConfig(cwd, quiet, debug);
|
|
870
|
+
if (hasLexEslintConfig) {
|
|
871
|
+
log("Using ESLint configuration from lex.config.* file", "info", quiet);
|
|
872
|
+
}
|
|
873
|
+
if (existsSync(pathResolve(cwd, ".eslintrc.json"))) {
|
|
874
|
+
unlinkSync(pathResolve(cwd, ".eslintrc.json"));
|
|
875
|
+
}
|
|
876
|
+
let lexConfigPath = "";
|
|
877
|
+
let shouldCreateTempConfig = false;
|
|
878
|
+
if (!hasEslintConfig && !hasLexEslintConfig) {
|
|
879
|
+
const possiblePaths = [
|
|
880
|
+
pathResolve(currentDirname, "../../../../eslint.config.ts"),
|
|
881
|
+
pathResolve(currentDirname, "../../../../eslint.config.js"),
|
|
882
|
+
pathResolve(process.env.LEX_HOME || "./node_modules/@nlabs/lex", "eslint.config.ts"),
|
|
883
|
+
pathResolve(process.env.LEX_HOME || "./node_modules/@nlabs/lex", "eslint.config.js")
|
|
884
|
+
];
|
|
885
|
+
for (const path of possiblePaths) {
|
|
886
|
+
if (existsSync(path)) {
|
|
887
|
+
lexConfigPath = path;
|
|
888
|
+
break;
|
|
889
|
+
}
|
|
890
|
+
}
|
|
891
|
+
if (debug) {
|
|
892
|
+
log(`Current directory: ${currentDirname}`, "info", quiet);
|
|
893
|
+
log(`Project config path: ${projectConfigPath}`, "info", quiet);
|
|
894
|
+
log(`Project config exists: ${hasEslintConfig}`, "info", quiet);
|
|
895
|
+
log(`Found Lex config: ${lexConfigPath}`, "info", quiet);
|
|
896
|
+
log(`Lex config exists: ${!!lexConfigPath && existsSync(lexConfigPath)}`, "info", quiet);
|
|
897
|
+
}
|
|
898
|
+
if (lexConfigPath && existsSync(lexConfigPath)) {
|
|
899
|
+
log("No ESLint configuration found in project. Using Lex's default configuration.", "info", quiet);
|
|
900
|
+
} else {
|
|
901
|
+
shouldCreateTempConfig = true;
|
|
902
|
+
}
|
|
903
|
+
}
|
|
904
|
+
if (config) {
|
|
905
|
+
const userConfigPath = pathResolve(cwd, config);
|
|
906
|
+
if (existsSync(userConfigPath)) {
|
|
907
|
+
log(`Using specified ESLint configuration: ${config}`, "info", quiet);
|
|
908
|
+
shouldCreateTempConfig = false;
|
|
909
|
+
} else {
|
|
910
|
+
log(`Specified ESLint configuration not found: ${config}. Using Lex's default configuration.`, "warn", quiet);
|
|
911
|
+
}
|
|
912
|
+
}
|
|
913
|
+
if (shouldCreateTempConfig) {
|
|
914
|
+
log("No ESLint configuration found. Creating a temporary configuration...", "info", quiet);
|
|
915
|
+
const configResult = createDefaultESLintConfig(useTypescript, cwd);
|
|
916
|
+
tempConfigPath = configResult.configPath;
|
|
917
|
+
}
|
|
918
|
+
let eslintOutput = "";
|
|
919
|
+
const captureOutput = (output) => {
|
|
920
|
+
eslintOutput += `${output}
|
|
921
|
+
`;
|
|
922
|
+
};
|
|
923
|
+
const result = await runEslintWithLex(cwd, quiet, cliName, true, debug, useTypescript, captureOutput);
|
|
924
|
+
if (shouldRemoveComments) {
|
|
925
|
+
spinner.start("Removing comments from files...");
|
|
926
|
+
const glob = await import("glob");
|
|
927
|
+
const files = glob.sync("{src,lib}/**/*.{js,jsx,ts,tsx}", {
|
|
928
|
+
cwd,
|
|
929
|
+
ignore: ["**/node_modules/**", "**/dist/**", "**/build/**"]
|
|
930
|
+
});
|
|
931
|
+
let processedCount = 0;
|
|
932
|
+
for (const file of files) {
|
|
933
|
+
const filePath = pathResolve(cwd, file);
|
|
934
|
+
if (removeFileComments(filePath, quiet)) {
|
|
935
|
+
processedCount++;
|
|
936
|
+
}
|
|
937
|
+
}
|
|
938
|
+
spinner.succeed(`Removed comments from ${processedCount} files`);
|
|
939
|
+
}
|
|
940
|
+
if (result !== 0 && fix) {
|
|
941
|
+
const aiConfigured = LexConfig.config.ai?.provider && LexConfig.config.ai.provider !== "none";
|
|
942
|
+
if (aiConfigured) {
|
|
943
|
+
log("Applying AI fixes to remaining issues...", "info", quiet);
|
|
944
|
+
await applyAIFix(cwd, eslintOutput, quiet);
|
|
945
|
+
const afterFixResult = await runEslintWithLex(cwd, quiet, cliName, false, debug, useTypescript);
|
|
946
|
+
callback(afterFixResult);
|
|
947
|
+
return afterFixResult;
|
|
948
|
+
}
|
|
949
|
+
log("ESLint could not fix all issues automatically.", "warn", quiet);
|
|
950
|
+
log("To enable AI-powered fixes, add AI configuration to your lex.config file:", "info", quiet);
|
|
951
|
+
log(`
|
|
952
|
+
// In lex.config.js (or lex.config.mjs, lex.config.cjs, etc.)
|
|
953
|
+
export default {
|
|
954
|
+
// Your existing config
|
|
955
|
+
ai: {
|
|
956
|
+
provider: 'cursor' // or 'openai', 'anthropic', etc.
|
|
957
|
+
// Additional provider-specific settings
|
|
958
|
+
}
|
|
959
|
+
};`, "info", quiet);
|
|
960
|
+
}
|
|
961
|
+
callback(result);
|
|
962
|
+
return result;
|
|
963
|
+
} catch (error) {
|
|
964
|
+
log(`
|
|
965
|
+
${cliName} Error: ${error.message}`, "error", quiet);
|
|
966
|
+
if (spinner) {
|
|
967
|
+
spinner.fail("Linting failed!");
|
|
968
|
+
}
|
|
969
|
+
callback(1);
|
|
970
|
+
return 1;
|
|
971
|
+
} finally {
|
|
972
|
+
const tempFilePaths = [
|
|
973
|
+
tempConfigPath,
|
|
974
|
+
pathResolve(cwd, ".lex-temp-eslint.cjs"),
|
|
975
|
+
pathResolve(cwd, ".lex-temp-default-eslint.cjs")
|
|
976
|
+
];
|
|
977
|
+
for (const filePath of tempFilePaths) {
|
|
978
|
+
if (filePath && existsSync(filePath)) {
|
|
979
|
+
try {
|
|
980
|
+
unlinkSync(filePath);
|
|
981
|
+
if (debug) {
|
|
982
|
+
log(`Cleaned up temporary ESLint config at ${filePath}`, "info", quiet);
|
|
983
|
+
}
|
|
984
|
+
} catch (_error) {
|
|
985
|
+
}
|
|
986
|
+
}
|
|
987
|
+
}
|
|
988
|
+
}
|
|
989
|
+
};
|
|
990
|
+
export {
|
|
991
|
+
lint
|
|
992
|
+
};
|
|
993
|
+
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vc3JjL2NvbW1hbmRzL2xpbnQvbGludC50cyJdLAogICJzb3VyY2VzQ29udGVudCI6IFsiLyoqXG4gKiBDb3B5cmlnaHQgKGMpIDIwMjItUHJlc2VudCwgTml0cm9nZW4gTGFicywgSW5jLlxuICogQ29weXJpZ2h0cyBsaWNlbnNlZCB1bmRlciB0aGUgTUlUIExpY2Vuc2UuIFNlZSB0aGUgYWNjb21wYW55aW5nIExJQ0VOU0UgZmlsZSBmb3IgdGVybXMuXG4gKi9cbmltcG9ydCB7ZXhlY2F9IGZyb20gJ2V4ZWNhJztcbmltcG9ydCB7ZXhpc3RzU3luYywgcmVhZEZpbGVTeW5jLCB1bmxpbmtTeW5jLCB3cml0ZUZpbGVTeW5jfSBmcm9tICdmcyc7XG5pbXBvcnQge2Rpcm5hbWUsIHJlc29sdmUgYXMgcGF0aFJlc29sdmUsIGV4dG5hbWV9IGZyb20gJ3BhdGgnO1xuXG5pbXBvcnQge0xleENvbmZpZ30gZnJvbSAnLi4vLi4vTGV4Q29uZmlnLmpzJztcbmltcG9ydCB7Y3JlYXRlU3Bpbm5lcn0gZnJvbSAnLi4vLi4vdXRpbHMvYXBwLmpzJztcbmltcG9ydCB7cmVzb2x2ZUJpbmFyeVBhdGh9IGZyb20gJy4uLy4uL3V0aWxzL2ZpbGUuanMnO1xuaW1wb3J0IHtsb2d9IGZyb20gJy4uLy4uL3V0aWxzL2xvZy5qcyc7XG5cbmxldCBjdXJyZW50RmlsZW5hbWU6IHN0cmluZztcbmxldCBjdXJyZW50RGlybmFtZTogc3RyaW5nO1xuXG50cnkge1xuICBjdXJyZW50RmlsZW5hbWUgPSBldmFsKCdyZXF1aXJlKFwidXJsXCIpLmZpbGVVUkxUb1BhdGgoaW1wb3J0Lm1ldGEudXJsKScpO1xuICBjdXJyZW50RGlybmFtZSA9IGRpcm5hbWUoY3VycmVudEZpbGVuYW1lKTtcbn0gY2F0Y2gge1xuICBjdXJyZW50RmlsZW5hbWUgPSBwcm9jZXNzLmN3ZCgpO1xuICBjdXJyZW50RGlybmFtZSA9IHByb2Nlc3MuY3dkKCk7XG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgTGludE9wdGlvbnMge1xuICByZWFkb25seSBjYWNoZT86IGJvb2xlYW47XG4gIHJlYWRvbmx5IGNhY2hlRmlsZT86IHN0cmluZztcbiAgcmVhZG9ubHkgY2FjaGVMb2NhdGlvbj86IHN0cmluZztcbiAgcmVhZG9ubHkgY2xpTmFtZT86IHN0cmluZztcbiAgcmVhZG9ubHkgY29sb3I/OiBib29sZWFuO1xuICByZWFkb25seSBjb25maWc/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IGRlYnVnPzogYm9vbGVhbjtcbiAgcmVhZG9ubHkgZW52Pzogc3RyaW5nO1xuICByZWFkb25seSBlbnZJbmZvPzogYm9vbGVhbjtcbiAgcmVhZG9ubHkgZXh0Pzogc3RyaW5nO1xuICByZWFkb25seSBmaXg/OiBib29sZWFuO1xuICByZWFkb25seSBmaXhEcnlSdW4/OiBib29sZWFuO1xuICByZWFkb25seSBmaXhUeXBlPzogc3RyaW5nO1xuICByZWFkb25seSBmb3JtYXQ/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IGdsb2JhbD86IHN0cmluZztcbiAgcmVhZG9ubHkgaWdub3JlUGF0aD86IHN0cmluZztcbiAgcmVhZG9ubHkgaWdub3JlUGF0dGVybj86IHN0cmluZztcbiAgcmVhZG9ubHkgaW5pdD86IGJvb2xlYW47XG4gIHJlYWRvbmx5IG1heFdhcm5pbmdzPzogc3RyaW5nO1xuICByZWFkb25seSBub0NvbG9yPzogYm9vbGVhbjtcbiAgcmVhZG9ubHkgbm9Fc2xpbnRyYz86IGJvb2xlYW47XG4gIHJlYWRvbmx5IG5vSWdub3JlPzogYm9vbGVhbjtcbiAgcmVhZG9ubHkgbm9JbmxpbmVDb25maWc/OiBib29sZWFuO1xuICByZWFkb25seSBvdXRwdXRGaWxlPzogc3RyaW5nO1xuICByZWFkb25seSBwYXJzZXI/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IHBhcnNlck9wdGlvbnM/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IHBsdWdpbj86IHN0cmluZztcbiAgcmVhZG9ubHkgcHJpbnRDb25maWc/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IHF1aWV0PzogYm9vbGVhbjtcbiAgcmVhZG9ubHkgcmVtb3ZlQ29tbWVudHM/OiBib29sZWFuO1xuICByZWFkb25seSByZXBvcnRVbnVzZWREaXNhYmxlRGlyZWN0aXZlcz86IGJvb2xlYW47XG4gIHJlYWRvbmx5IHJlc29sdmVQbHVnaW5zUmVsYXRpdmVUbz86IHN0cmluZztcbiAgcmVhZG9ubHkgcnVsZT86IHN0cmluZztcbiAgcmVhZG9ubHkgcnVsZXNkaXI/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IHN0ZGluPzogYm9vbGVhbjtcbiAgcmVhZG9ubHkgc3RkaW5GaWxlbmFtZT86IHN0cmluZztcbn1cblxuZXhwb3J0IHR5cGUgTGludENhbGxiYWNrID0gdHlwZW9mIHByb2Nlc3MuZXhpdDtcblxuaW50ZXJmYWNlIENvbmZpZ1Jlc3VsdCB7XG4gIGNvbmZpZ1BhdGg6IHN0cmluZztcbiAgb3JpZ2luYWxDb25maWc6IHN0cmluZyB8IG51bGw7XG59XG5cbmNvbnN0IGNyZWF0ZURlZmF1bHRFU0xpbnRDb25maWcgPSAodXNlVHlwZXNjcmlwdDogYm9vbGVhbiwgY3dkOiBzdHJpbmcpOiBDb25maWdSZXN1bHQgPT4ge1xuICAvLyBVc2UgYSB0ZW1wb3JhcnkgZmlsZSBwYXRoIGluc3RlYWQgb2YgY3JlYXRpbmcgaW4gdGhlIHByb2plY3QgZGlyZWN0b3J5XG4gIGNvbnN0IGNvbmZpZ1BhdGggPSBwYXRoUmVzb2x2ZShjd2QsICcubGV4LXRlbXAtZGVmYXVsdC1lc2xpbnQuY2pzJyk7XG4gIGNvbnN0IG9yaWdpbmFsQ29uZmlnID0gbnVsbDtcblxuICAvLyBUcnkgdG8gbG9jYXRlIExleCdzIEVTTGludCBjb25maWdcbiAgY29uc3QgcG9zc2libGVQYXRocyA9IFtcbiAgICBwYXRoUmVzb2x2ZShjdXJyZW50RGlybmFtZSwgJ2VzbGludC5jb25maWcudHMnKSxcbiAgICBwYXRoUmVzb2x2ZShjdXJyZW50RGlybmFtZSwgJ2VzbGludC5jb25maWcuanMnKSxcbiAgICBwYXRoUmVzb2x2ZShwcm9jZXNzLmVudi5MRVhfSE9NRSB8fCAnLi9ub2RlX21vZHVsZXMvQG5sYWJzL2xleCcsICdlc2xpbnQuY29uZmlnLnRzJyksXG4gICAgcGF0aFJlc29sdmUocHJvY2Vzcy5lbnYuTEVYX0hPTUUgfHwgJy4vbm9kZV9tb2R1bGVzL0BubGFicy9sZXgnLCAnZXNsaW50LmNvbmZpZy5qcycpXG4gIF07XG5cbiAgbGV0IGZvdW5kQ29uZmlnID0gJyc7XG5cbiAgZm9yKGNvbnN0IHBhdGggb2YgcG9zc2libGVQYXRocykge1xuICAgIGlmKGV4aXN0c1N5bmMocGF0aCkpIHtcbiAgICAgIGZvdW5kQ29uZmlnID0gcGF0aDtcbiAgICAgIGJyZWFrO1xuICAgIH1cbiAgfVxuXG4gIC8vIENyZWF0ZSBhIHRlbXBvcmFyeSBDb21tb25KUyBtb2R1bGUgdGhhdCByZXF1aXJlcyBMZXgncyBFU0xpbnQgY29uZmlnXG4gIGNvbnN0IGNvbmZpZ0NvbnRlbnQgPSBgLy8gVGVtcG9yYXJ5IEVTTGludCBjb25maWcgZ2VuZXJhdGVkIGJ5IExleFxuY29uc3QgbGV4Q29uZmlnID0gcmVxdWlyZSgnQG5sYWJzL2xleC9lc2xpbnQuY29uZmlnLmpzJyk7XG5cbm1vZHVsZS5leHBvcnRzID0gbGV4Q29uZmlnO2A7XG5cbiAgd3JpdGVGaWxlU3luYyhjb25maWdQYXRoLCBjb25maWdDb250ZW50LCAndXRmOCcpO1xuXG4gIHJldHVybiB7XG4gICAgY29uZmlnUGF0aCxcbiAgICBvcmlnaW5hbENvbmZpZ1xuICB9O1xufTtcblxuY29uc3QgY3JlYXRlQmFzaWNFU0xpbnRDb25maWcgPSAodXNlVHlwZXNjcmlwdDogYm9vbGVhbik6IHN0cmluZyA9PiB7XG4gIC8vIENyZWF0ZSBhIHNpbXBsaWZpZWQgY29uZmlnIHRoYXQgZXh0ZW5kcyBMZXgncyBFU0xpbnQgY29uZmlnXG4gIGNvbnN0IGNvbmZpZyA9IGAvLyBFU0xpbnQgY29uZmlndXJhdGlvblxuaW1wb3J0IGxleENvbmZpZyBmcm9tICdAbmxhYnMvbGV4L2VzbGludC5jb25maWcuanMnO1xuXG5yZXR1cm4gbGV4Q29uZmlnO2A7XG4gIHJldHVybiBjb25maWc7XG59O1xuXG4vKipcbiAqIENoZWNrIGlmIFR5cGVTY3JpcHQgaXMgYmVpbmcgdXNlZCBieSBsb29raW5nIGZvciB0c2NvbmZpZy5qc29uXG4gKi9cbmNvbnN0IGRldGVjdFR5cGVTY3JpcHQgPSAoY3dkOiBzdHJpbmcpOiBib29sZWFuID0+IGV4aXN0c1N5bmMocGF0aFJlc29sdmUoY3dkLCAndHNjb25maWcuanNvbicpKTtcblxuLyoqXG4gKiBFbnN1cmUgcGFja2FnZS5qc29uIGhhcyB0eXBlOiBtb2R1bGUgZm9yIEVTTSBzdXBwb3J0XG4gKi9cbmNvbnN0IGVuc3VyZU1vZHVsZVR5cGUgPSAoY3dkOiBzdHJpbmcpOiB2b2lkID0+IHtcbiAgY29uc3QgcGFja2FnZUpzb25QYXRoID0gcGF0aFJlc29sdmUoY3dkLCAncGFja2FnZS5qc29uJyk7XG5cbiAgaWYoZXhpc3RzU3luYyhwYWNrYWdlSnNvblBhdGgpKSB7XG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IHBhY2thZ2VKc29uQ29udGVudCA9IHJlYWRGaWxlU3luYyhwYWNrYWdlSnNvblBhdGgsICd1dGY4Jyk7XG4gICAgICBjb25zdCBwYWNrYWdlSnNvbiA9IEpTT04ucGFyc2UocGFja2FnZUpzb25Db250ZW50KTtcblxuICAgICAgLy8gSWYgdHlwZSBpcyBub3Qgc2V0IHRvIG1vZHVsZSwgd2FybiBpbnN0ZWFkIG9mIGF1dG8tbW9kaWZ5aW5nXG4gICAgICBpZihwYWNrYWdlSnNvbi50eXBlICE9PSAnbW9kdWxlJykge1xuICAgICAgICBsb2coJ1dhcm5pbmc6IHBhY2thZ2UuanNvbiBzaG91bGQgaGF2ZSBcInR5cGVcIjogXCJtb2R1bGVcIiBmb3IgRVNNIHN1cHBvcnQuIFBsZWFzZSBhZGQgdGhpcyBtYW51YWxseS4nLCAnd2FybicsIGZhbHNlKTtcbiAgICAgIH1cbiAgICB9IGNhdGNoIChfZXJyb3IpIHtcbiAgICAgIC8vIElnbm9yZSBlcnJvcnNcbiAgICB9XG4gIH1cbn07XG5cbmNvbnN0IGluc3RhbGxEZXBlbmRlbmNpZXMgPSBhc3luYyAoY3dkOiBzdHJpbmcsIHVzZVR5cGVzY3JpcHQ6IGJvb2xlYW4sIHF1aWV0OiBib29sZWFuKTogUHJvbWlzZTx2b2lkPiA9PiB7XG4gIGlmKHVzZVR5cGVzY3JpcHQpIHtcbiAgICBsb2coJ1VzaW5nIFR5cGVTY3JpcHQgRVNMaW50IGZyb20gTGV4Li4uJywgJ2luZm8nLCBxdWlldCk7XG4gIH0gZWxzZSB7XG4gICAgbG9nKCdVc2luZyBFU0xpbnQgZnJvbSBMZXguLi4nLCAnaW5mbycsIHF1aWV0KTtcbiAgfVxufTtcblxuY29uc3QgcnVuRXNsaW50V2l0aExleCA9IGFzeW5jIChcbiAgY3dkOiBzdHJpbmcsXG4gIHF1aWV0OiBib29sZWFuLFxuICBjbGlOYW1lOiBzdHJpbmcsXG4gIGZpeDogYm9vbGVhbixcbiAgZGVidWc6IGJvb2xlYW4sXG4gIHVzZVR5cGVzY3JpcHQ6IGJvb2xlYW4sXG4gIGNhcHR1cmVPdXRwdXQ/OiAob3V0cHV0OiBzdHJpbmcpID0+IHZvaWRcbik6IFByb21pc2U8bnVtYmVyPiA9PiB7XG4gIGNvbnN0IHNwaW5uZXIgPSBjcmVhdGVTcGlubmVyKHF1aWV0KTtcblxuICB0cnkge1xuICAgIGNvbnN0IHByb2plY3RDb25maWdQYXRoID0gcGF0aFJlc29sdmUoY3dkLCAnZXNsaW50LmNvbmZpZy5qcycpO1xuICAgIGNvbnN0IHByb2plY3RDb25maWdQYXRoVHMgPSBwYXRoUmVzb2x2ZShjd2QsICdlc2xpbnQuY29uZmlnLnRzJyk7XG4gICAgY29uc3QgaGFzUHJvamVjdENvbmZpZyA9IGV4aXN0c1N5bmMocHJvamVjdENvbmZpZ1BhdGgpIHx8IGV4aXN0c1N5bmMocHJvamVjdENvbmZpZ1BhdGhUcyk7XG4gICAgY29uc3QgaGFzTGV4Q29uZmlnRXNsaW50ID0gTGV4Q29uZmlnLmNvbmZpZy5lc2xpbnQgJiYgT2JqZWN0LmtleXMoTGV4Q29uZmlnLmNvbmZpZy5lc2xpbnQpLmxlbmd0aCA+IDA7XG5cbiAgICBjb25zdCBwb3NzaWJsZVBhdGhzID0gW1xuICAgICAgcGF0aFJlc29sdmUoY3VycmVudERpcm5hbWUsICcuLi8uLi8uLi8uLi9lc2xpbnQuY29uZmlnLmpzJyksXG4gICAgICBwYXRoUmVzb2x2ZShjdXJyZW50RGlybmFtZSwgJy4uLy4uLy4uLy4uL2VzbGludC5jb25maWcudHMnKSxcbiAgICAgIHBhdGhSZXNvbHZlKHByb2Nlc3MuZW52LkxFWF9IT01FIHx8ICcvdXNyL2xvY2FsL2xpYi9ub2RlX21vZHVsZXMvQG5sYWJzL2xleCcsICdlc2xpbnQuY29uZmlnLmpzJyksXG4gICAgICBwYXRoUmVzb2x2ZShwcm9jZXNzLmVudi5MRVhfSE9NRSB8fCAnL3Vzci9sb2NhbC9saWIvbm9kZV9tb2R1bGVzL0BubGFicy9sZXgnLCAnZXNsaW50LmNvbmZpZy50cycpXG4gICAgXTtcblxuICAgIGxldCBsZXhDb25maWdQYXRoID0gJyc7XG4gICAgZm9yKGNvbnN0IHBhdGggb2YgcG9zc2libGVQYXRocykge1xuICAgICAgaWYoZXhpc3RzU3luYyhwYXRoKSkge1xuICAgICAgICBsZXhDb25maWdQYXRoID0gcGF0aDtcbiAgICAgICAgYnJlYWs7XG4gICAgICB9XG4gICAgfVxuXG4gICAgbGV0IGNvbmZpZ1BhdGggPSAnJztcbiAgICBsZXQgdGVtcENvbmZpZ1BhdGggPSAnJztcblxuICAgIC8vIFByaW9yaXR5OlxuICAgIC8vIDEuIFByb2plY3QgZXNsaW50LmNvbmZpZyBmaWxlc1xuICAgIC8vIDIuIEVTTGludCBjb25maWcgaW4gbGV4LmNvbmZpZy4qIGZpbGVcbiAgICAvLyAzLiBMZXgncyBkZWZhdWx0IGVzbGludC5jb25maWcuanNcbiAgICAvLyA0LiBDcmVhdGUgYSB0ZW1wb3JhcnkgY29uZmlnIGZpbGVcbiAgICBpZihoYXNQcm9qZWN0Q29uZmlnKSB7XG4gICAgICBjb25maWdQYXRoID0gZXhpc3RzU3luYyhwcm9qZWN0Q29uZmlnUGF0aFRzKSA/IHByb2plY3RDb25maWdQYXRoVHMgOiBwcm9qZWN0Q29uZmlnUGF0aDtcbiAgICAgIGlmKGRlYnVnKSB7XG4gICAgICAgIGxvZyhgVXNpbmcgcHJvamVjdCBFU0xpbnQgY29uZmlnIGZpbGU6ICR7Y29uZmlnUGF0aH1gLCAnaW5mbycsIHF1aWV0KTtcbiAgICAgIH1cbiAgICB9IGVsc2UgaWYoaGFzTGV4Q29uZmlnRXNsaW50KSB7XG4gICAgICAvLyBXaGVuIHVzaW5nIGxleC5jb25maWcuZXNsaW50LCBjcmVhdGUgYSB0ZW1wb3JhcnkgSlMgY29uZmlnIGZpbGUgKG5vdCBKU09OKVxuICAgICAgLy8gdG8gYXZvaWQgRVNNIEpTT04gaW1wb3J0IGlzc3Vlc1xuICAgICAgdGVtcENvbmZpZ1BhdGggPSBwYXRoUmVzb2x2ZShjd2QsICcubGV4LXRlbXAtZXNsaW50LmNqcycpO1xuXG4gICAgICAvLyBDcmVhdGUgYSBDb21tb25KUyBtb2R1bGUgdGhhdCBleHRlbmRzIExleCdzIGVzbGludCBjb25maWdcbiAgICAgIGNvbnN0IGNvbmZpZ0NvbnRlbnQgPSBgLy8gVGVtcG9yYXJ5IEVTTGludCBjb25maWcgZ2VuZXJhdGVkIGJ5IExleFxuY29uc3QgbGV4Q29uZmlnID0gcmVxdWlyZSgnQG5sYWJzL2xleC9lc2xpbnQuY29uZmlnLmpzJyk7XG5jb25zdCB1c2VyQ29uZmlnID0gJHtKU09OLnN0cmluZ2lmeShMZXhDb25maWcuY29uZmlnLmVzbGludCwgbnVsbCwgMil9O1xuXG4vLyBNZXJnZSBMZXgncyBjb25maWcgd2l0aCB1c2VyIGNvbmZpZ1xubW9kdWxlLmV4cG9ydHMgPSB7XG4gIC4uLmxleENvbmZpZ1xufTtgO1xuXG4gICAgICB3cml0ZUZpbGVTeW5jKHRlbXBDb25maWdQYXRoLCBjb25maWdDb250ZW50LCAndXRmOCcpO1xuICAgICAgY29uZmlnUGF0aCA9IHRlbXBDb25maWdQYXRoO1xuXG4gICAgICBpZihkZWJ1Zykge1xuICAgICAgICBsb2coYFVzaW5nIEVTTGludCBjb25maWcgZnJvbSBsZXguY29uZmlnLiogZmlsZSB2aWEgdGVtcCBmaWxlOiAke3RlbXBDb25maWdQYXRofWAsICdpbmZvJywgcXVpZXQpO1xuICAgICAgfVxuICAgIH0gZWxzZSBpZihsZXhDb25maWdQYXRoICYmIGV4aXN0c1N5bmMobGV4Q29uZmlnUGF0aCkpIHtcbiAgICAgIGNvbmZpZ1BhdGggPSBsZXhDb25maWdQYXRoO1xuICAgICAgaWYoZGVidWcpIHtcbiAgICAgICAgbG9nKGBVc2luZyBMZXggRVNMaW50IGNvbmZpZyBmaWxlOiAke2NvbmZpZ1BhdGh9YCwgJ2luZm8nLCBxdWlldCk7XG4gICAgICB9XG4gICAgfSBlbHNlIHtcbiAgICAgIC8vIENyZWF0ZSBhIHRlbXBvcmFyeSBkZWZhdWx0IGNvbmZpZyBmaWxlIGlmIG5vIG90aGVyIGNvbmZpZyBpcyBmb3VuZFxuICAgICAgdGVtcENvbmZpZ1BhdGggPSBwYXRoUmVzb2x2ZShjd2QsICcubGV4LXRlbXAtZGVmYXVsdC1lc2xpbnQuY2pzJyk7XG5cbiAgICAgIC8vIENyZWF0ZSBhIGJhc2ljIEVTTGludCBjb25maWdcbiAgICAgIGNvbnN0IGNvbmZpZ0NvbnRlbnQgPSBgLy8gVGVtcG9yYXJ5IGRlZmF1bHQgRVNMaW50IGNvbmZpZyBnZW5lcmF0ZWQgYnkgTGV4XG5jb25zdCBsZXhDb25maWcgPSByZXF1aXJlKCdAbmxhYnMvbGV4L2VzbGludC5jb25maWcuanMnKTtcblxubW9kdWxlLmV4cG9ydHMgPSBsZXhDb25maWc7YDtcblxuICAgICAgd3JpdGVGaWxlU3luYyh0ZW1wQ29uZmlnUGF0aCwgY29uZmlnQ29udGVudCwgJ3V0ZjgnKTtcbiAgICAgIGNvbmZpZ1BhdGggPSB0ZW1wQ29uZmlnUGF0aDtcblxuICAgICAgaWYoZGVidWcpIHtcbiAgICAgICAgbG9nKGBDcmVhdGVkIHRlbXBvcmFyeSBkZWZhdWx0IEVTTGludCBjb25maWcgYXQ6ICR7dGVtcENvbmZpZ1BhdGh9YCwgJ2luZm8nLCBxdWlldCk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICBsb2coJ05vIEVTTGludCBjb25maWd1cmF0aW9uIGZvdW5kLiBVc2luZyBMZXggZGVmYXVsdCBjb25maWd1cmF0aW9uLicsICdpbmZvJywgcXVpZXQpO1xuICAgICAgfVxuICAgIH1cblxuICAgIGNvbnN0IGVzbGludEJpbmFyeSA9IHJlc29sdmVCaW5hcnlQYXRoKCdlc2xpbnQnLCAnZXNsaW50Jyk7XG5cbiAgICBpZighZXNsaW50QmluYXJ5KSB7XG4gICAgICBsb2coYFxcbiR7Y2xpTmFtZX0gRXJyb3I6IEVTTGludCBiaW5hcnkgbm90IGZvdW5kIGluIExleCdzIG5vZGVfbW9kdWxlc2AsICdlcnJvcicsIHF1aWV0KTtcbiAgICAgIGxvZygnUGxlYXNlIHJlaW5zdGFsbCBMZXggb3IgY2hlY2sgeW91ciBpbnN0YWxsYXRpb24uJywgJ2luZm8nLCBxdWlldCk7XG4gICAgICByZXR1cm4gMTtcbiAgICB9XG5cbiAgICAvLyBCYXNlIEVTTGludCBhcmd1bWVudHNcbiAgICBjb25zdCBiYXNlRXNsaW50QXJncyA9IFtcbiAgICAgIC4uLihmaXggPyBbJy0tZml4J10gOiBbXSksXG4gICAgICAuLi4oZGVidWcgPyBbJy0tZGVidWcnXSA6IFtdKSxcbiAgICAgICctLW5vLWVycm9yLW9uLXVubWF0Y2hlZC1wYXR0ZXJuJyxcbiAgICAgICctLW5vLXdhcm4taWdub3JlZCdcbiAgICBdO1xuXG4gICAgLy8gQWRkIGNvbmZpZyBwYXRoXG4gICAgY29uc3QgY29uZmlnQXJncyA9IGNvbmZpZ1BhdGggPyBbJy0tY29uZmlnJywgY29uZmlnUGF0aF0gOiBbXTtcblxuICAgIGNvbnN0IGpzUmVzdWx0ID0gYXdhaXQgZXhlY2EoZXNsaW50QmluYXJ5LCBbXG4gICAgICAnc3JjLyoqLyoue2pzLGpzeH0nLFxuICAgICAgLi4uY29uZmlnQXJncyxcbiAgICAgIC4uLmJhc2VFc2xpbnRBcmdzXG4gICAgXSwge1xuICAgICAgcmVqZWN0OiBmYWxzZSxcbiAgICAgIHN0ZGlvOiAncGlwZScsXG4gICAgICBjd2QsXG4gICAgICBzaGVsbDogdHJ1ZVxuICAgIH0pO1xuXG4gICAgaWYoanNSZXN1bHQuc3Rkb3V0KSB7XG4gICAgICBjb25zb2xlLmxvZyhqc1Jlc3VsdC5zdGRvdXQpO1xuICAgICAgaWYoY2FwdHVyZU91dHB1dCkge1xuICAgICAgICBjYXB0dXJlT3V0cHV0KGpzUmVzdWx0LnN0ZG91dCk7XG4gICAgICB9XG4gICAgfVxuXG4gICAgaWYoanNSZXN1bHQuc3RkZXJyKSB7XG4gICAgICBjb25zb2xlLmVycm9yKGpzUmVzdWx0LnN0ZGVycik7XG4gICAgICBpZihjYXB0dXJlT3V0cHV0KSB7XG4gICAgICAgIGNhcHR1cmVPdXRwdXQoanNSZXN1bHQuc3RkZXJyKTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICBsZXQgdHNSZXN1bHQ6IGFueSA9IHtleGl0Q29kZTogMCwgc3Rkb3V0OiAnJywgc3RkZXJyOiAnJ307XG4gICAgaWYodXNlVHlwZXNjcmlwdCkge1xuICAgICAgdHNSZXN1bHQgPSBhd2FpdCBleGVjYShlc2xpbnRCaW5hcnksIFtcbiAgICAgICAgJ3NyYy8qKi8qLnt0cyx0c3h9JyxcbiAgICAgICAgLi4uY29uZmlnQXJncyxcbiAgICAgICAgLi4uYmFzZUVzbGludEFyZ3NcbiAgICAgIF0sIHtcbiAgICAgICAgcmVqZWN0OiBmYWxzZSxcbiAgICAgICAgc3RkaW86ICdwaXBlJyxcbiAgICAgICAgY3dkLFxuICAgICAgICBzaGVsbDogdHJ1ZVxuICAgICAgfSk7XG5cbiAgICAgIGlmKHRzUmVzdWx0LnN0ZG91dCkge1xuICAgICAgICBjb25zb2xlLmxvZyh0c1Jlc3VsdC5zdGRvdXQpO1xuICAgICAgfVxuXG4gICAgICBpZih0c1Jlc3VsdC5zdGRlcnIpIHtcbiAgICAgICAgY29uc29sZS5lcnJvcih0c1Jlc3VsdC5zdGRlcnIpO1xuICAgICAgfVxuICAgIH1cblxuICAgIC8vIENsZWFuIHVwIHRlbXAgZmlsZSBpZiBjcmVhdGVkXG4gICAgaWYodGVtcENvbmZpZ1BhdGggJiYgZXhpc3RzU3luYyh0ZW1wQ29uZmlnUGF0aCkpIHtcbiAgICAgIHRyeSB7XG4gICAgICAgIHVubGlua1N5bmModGVtcENvbmZpZ1BhdGgpO1xuICAgICAgICBpZihkZWJ1Zykge1xuICAgICAgICAgIGxvZyhgUmVtb3ZlZCB0ZW1wb3JhcnkgRVNMaW50IGNvbmZpZyBhdCAke3RlbXBDb25maWdQYXRofWAsICdpbmZvJywgcXVpZXQpO1xuICAgICAgICB9XG4gICAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICAvLyBJZ25vcmUgZXJyb3JzIHdoZW4gY2xlYW5pbmcgdXBcbiAgICAgICAgaWYoZGVidWcpIHtcbiAgICAgICAgICBsb2coYEZhaWxlZCB0byByZW1vdmUgdGVtcG9yYXJ5IEVTTGludCBjb25maWc6ICR7ZXJyb3IubWVzc2FnZX1gLCAnd2FybicsIHF1aWV0KTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cblxuICAgIGNvbnN0IGVzbGludE5vdEZvdW5kID0ganNSZXN1bHQuc3RkZXJyPy5pbmNsdWRlcygnY29tbWFuZCBub3QgZm91bmQnKSB8fCBqc1Jlc3VsdC5zdGRlcnI/LmluY2x1ZGVzKCdlc2xpbnQ6IGNvbW1hbmQgbm90IGZvdW5kJyk7XG4gICAgaWYoZXNsaW50Tm90Rm91bmQpIHtcbiAgICAgIHNwaW5uZXIuZmFpbCgnRVNMaW50IG5vdCBmb3VuZCEnKTtcbiAgICAgIGxvZyhgXFxuJHtjbGlOYW1lfSBFcnJvcjogTGV4J3MgRVNMaW50IGJpbmFyeSBub3QgZm91bmQuIFBsZWFzZSByZWluc3RhbGwgTGV4IG9yIGNoZWNrIHlvdXIgaW5zdGFsbGF0aW9uLmAsICdlcnJvcicsIHF1aWV0KTtcbiAgICAgIHJldHVybiAxO1xuICAgIH1cblxuICAgIGlmKGpzUmVzdWx0LmV4aXRDb2RlID09PSAwICYmIHRzUmVzdWx0LmV4aXRDb2RlID09PSAwKSB7XG4gICAgICBzcGlubmVyLnN1Y2NlZWQoJ0xpbnRpbmcgY29tcGxldGVkIScpO1xuICAgICAgcmV0dXJuIDA7XG4gICAgfVxuXG4gICAgY29uc3Qgbm9GaWxlc0ZvdW5kID1cbiAgICAgIChqc1Jlc3VsdC5zdGRlcnI/LmluY2x1ZGVzKCdObyBzdWNoIGZpbGUgb3IgZGlyZWN0b3J5JykgfHwganNSZXN1bHQuc3Rkb3V0Py5pbmNsdWRlcygnTm8gc3VjaCBmaWxlIG9yIGRpcmVjdG9yeScpKSAmJlxuICAgICAgKCF1c2VUeXBlc2NyaXB0IHx8IHRzUmVzdWx0LnN0ZGVycj8uaW5jbHVkZXMoJ05vIHN1Y2ggZmlsZSBvciBkaXJlY3RvcnknKSB8fCB0c1Jlc3VsdC5zdGRvdXQ/LmluY2x1ZGVzKCdObyBzdWNoIGZpbGUgb3IgZGlyZWN0b3J5JykpO1xuXG4gICAgaWYobm9GaWxlc0ZvdW5kKSB7XG4gICAgICBzcGlubmVyLnN1Y2NlZWQoJ05vIGZpbGVzIGZvdW5kIHRvIGxpbnQnKTtcbiAgICAgIHJldHVybiAwO1xuICAgIH1cbiAgICBzcGlubmVyLmZhaWwoJ0xpbnRpbmcgZmFpbGVkIScpO1xuICAgIGxvZyhgXFxuJHtjbGlOYW1lfSBFcnJvcjogRVNMaW50IGZvdW5kIGlzc3VlcyBpbiB5b3VyIGNvZGUuYCwgJ2Vycm9yJywgcXVpZXQpO1xuICAgIHJldHVybiAxO1xuICB9IGNhdGNoIChlcnJvcikge1xuICAgIHNwaW5uZXIuZmFpbCgnTGludGluZyBmYWlsZWQhJyk7XG4gICAgbG9nKGBcXG4ke2NsaU5hbWV9IEVycm9yOiAke2Vycm9yLm1lc3NhZ2V9YCwgJ2Vycm9yJywgcXVpZXQpO1xuICAgIHJldHVybiAxO1xuICB9XG59O1xuXG5jb25zdCBhcHBseUFJRml4ID0gYXN5bmMgKFxuICBjd2Q6IHN0cmluZyxcbiAgZXJyb3JzOiBzdHJpbmcsXG4gIHF1aWV0OiBib29sZWFuXG4pOiBQcm9taXNlPHZvaWQ+ID0+IHtcbiAgY29uc3Qgc3Bpbm5lciA9IGNyZWF0ZVNwaW5uZXIocXVpZXQpO1xuICBzcGlubmVyLnN0YXJ0KCdVc2luZyBBSSB0byBmaXggcmVtYWluaW5nIGxpbnQgaXNzdWVzLi4uJyk7XG5cbiAgdHJ5IHtcbiAgICBjb25zdCBmaWxlRXJyb3JNYXAgPSBuZXcgTWFwPHN0cmluZywgc3RyaW5nW10+KCk7XG4gICAgY29uc3QgbGluZXMgPSBlcnJvcnMuc3BsaXQoJ1xcbicpO1xuICAgIGxldCBjdXJyZW50RmlsZSA9ICcnO1xuXG4gICAgZm9yKGNvbnN0IGxpbmUgb2YgbGluZXMpIHtcbiAgICAgIGlmKGxpbmUubWF0Y2goL14oXFwvfFtBLVpdOlxcXFwpLio/XFwuKGpzfGpzeHx0c3x0c3gpJC8pKSB7XG4gICAgICAgIGN1cnJlbnRGaWxlID0gbGluZS50cmltKCk7XG4gICAgICAgIGlmKCFmaWxlRXJyb3JNYXAuaGFzKGN1cnJlbnRGaWxlKSkge1xuICAgICAgICAgIGZpbGVFcnJvck1hcC5zZXQoY3VycmVudEZpbGUsIFtdKTtcbiAgICAgICAgfVxuICAgICAgfSBlbHNlIGlmKGN1cnJlbnRGaWxlICYmIGxpbmUudHJpbSgpICYmIGxpbmUubWF0Y2goL1xccytcXGQrOlxcZCtcXHMrKGVycm9yfHdhcm5pbmcpXFxzKy8pKSB7XG4gICAgICAgIGNvbnN0IGVycm9yQXJyYXkgPSBmaWxlRXJyb3JNYXAuZ2V0KGN1cnJlbnRGaWxlKTtcbiAgICAgICAgaWYoZXJyb3JBcnJheSkge1xuICAgICAgICAgIGVycm9yQXJyYXkucHVzaChsaW5lLnRyaW0oKSk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG5cbiAgICBpZihmaWxlRXJyb3JNYXAuc2l6ZSA9PT0gMCkge1xuICAgICAgbG9nKCdVc2luZyBhbHRlcm5hdGl2ZSBlcnJvciBwYXJzaW5nIHN0cmF0ZWd5JywgJ2luZm8nLCBxdWlldCk7XG5cbiAgICAgIGNvbnN0IHNlY3Rpb25zID0gZXJyb3JzLnNwbGl0KCdcXG5cXG4nKTtcblxuICAgICAgZm9yKGNvbnN0IHNlY3Rpb24gb2Ygc2VjdGlvbnMpIHtcbiAgICAgICAgaWYoc2VjdGlvbi50cmltKCkgPT09ICcnKSB7XG4gICAgICAgICAgY29udGludWU7XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBsaW5lcyA9IHNlY3Rpb24uc3BsaXQoJ1xcbicpO1xuICAgICAgICBjb25zdCBmaWxlUGF0aCA9IGxpbmVzWzBdLnRyaW0oKTtcblxuICAgICAgICBpZihmaWxlUGF0aC5tYXRjaCgvXFwuKGpzfGpzeHx0c3x0c3gpJC8pKSB7XG4gICAgICAgICAgZmlsZUVycm9yTWFwLnNldChmaWxlUGF0aCwgW10pO1xuXG4gICAgICAgICAgZm9yKGxldCBpID0gMTsgaSA8IGxpbmVzLmxlbmd0aDsgaSsrKSB7XG4gICAgICAgICAgICBpZihsaW5lc1tpXS50cmltKCkgIT09ICcnKSB7XG4gICAgICAgICAgICAgIGZpbGVFcnJvck1hcC5nZXQoZmlsZVBhdGgpPy5wdXNoKGxpbmVzW2ldLnRyaW0oKSk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICB9XG4gICAgfVxuXG4gICAgaWYoZmlsZUVycm9yTWFwLnNpemUgPT09IDApIHtcbiAgICAgIGxvZygnVXNpbmcgZGlyZWN0IGZpbGUgcGF0aCBleHRyYWN0aW9uJywgJ2luZm8nLCBxdWlldCk7XG5cbiAgICAgIGNvbnN0IGZpbGVQYXRoUmVnZXggPSAvKD86XFwvfFtBLVpdOlxcXFwpKD86W146XFxuXStcXC8pKlteOlxcbl0rXFwuKGpzfGpzeHx0c3x0c3gpL2c7XG4gICAgICBjb25zdCBmaWxlUGF0aHMgPSBlcnJvcnMubWF0Y2goZmlsZVBhdGhSZWdleCkgfHwgW107XG5cbiAgICAgIGZvcihjb25zdCBmaWxlUGF0aCBvZiBmaWxlUGF0aHMpIHtcbiAgICAgICAgaWYoIWZpbGVFcnJvck1hcC5oYXMoZmlsZVBhdGgpICYmIGV4aXN0c1N5bmMoZmlsZVBhdGgpKSB7XG4gICAgICAgICAgZmlsZUVycm9yTWFwLnNldChmaWxlUGF0aCwgW10pO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIGNvbnN0IGtub3duRmlsZXMgPSBbXG4gICAgICAgIHBhdGhSZXNvbHZlKGN3ZCwgJ3NyYy9jcmVhdGUvY2hhbmdlbG9nLnRzJyksXG4gICAgICAgIHBhdGhSZXNvbHZlKGN3ZCwgJ3NyYy91dGlscy9haVNlcnZpY2UudHMnKSxcbiAgICAgICAgcGF0aFJlc29sdmUoY3dkLCAnc3JjL3V0aWxzL2FwcC50cycpLFxuICAgICAgICBwYXRoUmVzb2x2ZShjd2QsICdzcmMvdXRpbHMvcmVhY3RTaGltLnRzJyksXG4gICAgICAgIHBhdGhSZXNvbHZlKGN3ZCwgJ3NyYy9jb21tYW5kcy9saW50L2F1dG9maXguanMnKVxuICAgICAgXTtcblxuICAgICAgZm9yKGNvbnN0IGZpbGUgb2Yga25vd25GaWxlcykge1xuICAgICAgICBpZihleGlzdHNTeW5jKGZpbGUpICYmICFmaWxlRXJyb3JNYXAuaGFzKGZpbGUpKSB7XG4gICAgICAgICAgZmlsZUVycm9yTWFwLnNldChmaWxlLCBbXSk7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG5cbiAgICBmb3IoY29uc3QgZmlsZVBhdGggb2YgZmlsZUVycm9yTWFwLmtleXMoKSkge1xuICAgICAgaWYoIWV4aXN0c1N5bmMoZmlsZVBhdGgpKSB7XG4gICAgICAgIGxvZyhgRmlsZSBub3QgZm91bmQ6ICR7ZmlsZVBhdGh9YCwgJ3dhcm4nLCBxdWlldCk7XG4gICAgICAgIGNvbnRpbnVlO1xuICAgICAgfVxuXG4gICAgICBsb2coYFByb2Nlc3NpbmcgZmlsZTogJHtmaWxlUGF0aH1gLCAnaW5mbycsIHF1aWV0KTtcblxuICAgICAgY29uc3QgaXNDdXJzb3JJREUgPSBMZXhDb25maWcuY29uZmlnLmFpPy5wcm92aWRlciA9PT0gJ2N1cnNvcicgfHwgcHJvY2Vzcy5lbnYuQ1VSU09SX0lERSA9PT0gJ3RydWUnO1xuXG4gICAgICBpZihpc0N1cnNvcklERSkge1xuICAgICAgICB0cnkge1xuICAgICAgICAgIGNvbnN0IHByb21wdCA9IGBGaXggYWxsIEVTTGludCBlcnJvcnMgaW4gdGhpcyBmaWxlLiBGb2N1cyBvbjpcbjEuIEZpeGluZyBuYW1pbmcgY29udmVudGlvbnNcbjIuIEZpeGluZyBzb3J0LWtleXMgaXNzdWVzXG4zLiBSZXBsYWNpbmcgY29uc29sZS5sb2cgd2l0aCBsb2cgdXRpbGl0eVxuNC4gRml4aW5nIG5vLXBsdXNwbHVzIGlzc3Vlc1xuNS4gRml4aW5nIHVubmVjZXNzYXJ5IGVzY2FwZSBjaGFyYWN0ZXJzXG42LiBGaXhpbmcgb3RoZXIgRVNMaW50IGVycm9yc1xuXG5DUklUSUNBTCBSRVFVSVJFTUVOVFM6XG4tIE9OTFkgZml4IHRoZSBzcGVjaWZpYyBsaW5lcyB3aXRoIEVTTGludCBlcnJvcnNcbi0gRE8gTk9UIG1vZGlmeSBhbnkgb3RoZXIgbGluZXMgb2YgY29kZVxuLSBETyBOT1QgcmVtb3ZlIGxpbmUgYnJlYWtzIHVubGVzcyB0aGV5IGFyZSBzcGVjaWZpY2FsbHkgY2F1c2luZyBFU0xpbnQgZXJyb3JzXG4tIERPIE5PVCBjb25kZW5zZSBtdWx0aS1saW5lIHN0cnVjdHVyZXMgdG8gc2luZ2xlIGxpbmVzXG4tIFBSRVNFUlZFIGFsbCBleGlzdGluZyBsaW5lIGJyZWFrcyBhbmQgZm9ybWF0dGluZyB0aGF0IGlzIG5vdCBjYXVzaW5nIGVycm9yc1xuXG5TUEVDSUZJQyBGT1JNQVRUSU5HIFJVTEVTOlxuLSBNYWludGFpbiBwcm9wZXIgaW5kZW50YXRpb24gKDIgc3BhY2VzKVxuLSBLZWVwIGxpbmUgYnJlYWtzIGJldHdlZW4gY2xhc3MvaW50ZXJmYWNlIGRlY2xhcmF0aW9uIGFuZCB0aGVpciBtZW1iZXJzXG4tIEtlZXAgbGluZSBicmVha3MgYmV0d2VlbiBtZXRob2RzXG4tIEVuc3VyZSB0aGVyZSBpcyBhIGxpbmUgYnJlYWsgYWZ0ZXIgb3BlbmluZyBicmFjZXMgZm9yIGNsYXNzZXMsIGludGVyZmFjZXMsIGFuZCBtZXRob2RzXG4tIERPIE5PVCBwbGFjZSBjbGFzcy9pbnRlcmZhY2UgcHJvcGVydGllcyBvciBtZXRob2RzIG9uIHRoZSBzYW1lIGxpbmUgYXMgdGhlIG9wZW5pbmcgYnJhY2Vcbi0gUHJlc2VydmUgZW1wdHkgbGluZXMgYmV0d2VlbiBsb2dpY2FsIGNvZGUgYmxvY2tzXG4tIFBSRVNFUlZFIG11bHRpLWxpbmUgaW1wb3J0cyAtIGRvIG5vdCBjb25kZW5zZSB0aGVtIHRvIHNpbmdsZSBsaW5lc1xuLSBQUkVTRVJWRSBtdWx0aS1saW5lIG9iamVjdC9hcnJheSBkZWNsYXJhdGlvbnMgLSBkbyBub3QgY29uZGVuc2UgdGhlbSB0byBzaW5nbGUgbGluZXNcblxuU09SVC1LRVlTIFJVTEUgKEhJR0hFU1QgUFJJT1JJVFkpOlxuLSBBbGwgb2JqZWN0IGxpdGVyYWwga2V5cyBNVVNUIGJlIHNvcnRlZCBhbHBoYWJldGljYWxseSBpbiBhc2NlbmRpbmcgb3JkZXJcbi0gVGhpcyBhcHBsaWVzIHRvIEFMTCBvYmplY3RzIGluIHRoZSBmaWxlLCBub3QganVzdCB0aG9zZSB3aXRoIGV4cGxpY2l0IHNvcnQta2V5cyBlcnJvcnNcbi0gRXhhbXBsZToge2I6IDIsIGE6IDEsIGM6IDN9IHNob3VsZCBiZWNvbWUge2E6IDEsIGI6IDIsIGM6IDN9XG4tIFByZXNlcnZlIHRoZSBvcmlnaW5hbCBmb3JtYXR0aW5nIGFuZCBsaW5lIGJyZWFrcyB3aGVuIHNvcnRpbmdcblxuRXhhbXBsZSBvZiBDT1JSRUNUIGZvcm1hdHRpbmcgKERPIE5PVCBDSEFOR0UpOlxuZXhwb3J0IGNsYXNzIFVzZXJDb25zdGFudHMge1xuICBzdGF0aWMgcmVhZG9ubHkgQUREX0lURU1fRVJST1I6IHN0cmluZyA9ICdVU0VSX0FERF9JVEVNX0VSUk9SJztcbiAgc3RhdGljIHJlYWRvbmx5IE9USEVSX0NPTlNUQU5UOiBzdHJpbmcgPSAnT1RIRVJfQ09OU1RBTlQnO1xufVxuXG5jb25zdHJ1Y3RvcihmbHV4OiBGbHV4RnJhbWV3b3JrLCBDdXN0b21BZGFwdGVyOiB0eXBlb2YgRXZlbnQgPSBFdmVudCkge1xuICB0aGlzLkN1c3RvbUFkYXB0ZXIgPSBDdXN0b21BZGFwdGVyO1xuICB0aGlzLmZsdXggPSBmbHV4O1xufVxuXG5pbXBvcnQge1xuICBhcHAsXG4gIGV2ZW50cyxcbiAgaW1hZ2VzLFxuICBsb2NhdGlvbnMsXG4gIG1lc3NhZ2VzLFxuICBwb3N0cyxcbiAgdGFncyxcbiAgdXNlcnMsXG4gIHdlYnNvY2tldFxufSBmcm9tICcuL3N0b3Jlcyc7XG5cbmNvbnN0IGNvbmZpZyA9IHtcbiAgYXBpS2V5OiAndmFsdWUnLFxuICBiYXNlVXJsOiAnaHR0cHM6Ly9hcGkuZXhhbXBsZS5jb20nLFxuICB0aW1lb3V0OiA1MDAwXG59O1xuXG5FeGFtcGxlIG9mIElOQ09SUkVDVCBmb3JtYXR0aW5nIChGSVggVEhJUyk6XG5leHBvcnQgY2xhc3MgVXNlckNvbnN0YW50cyB7c3RhdGljIHJlYWRvbmx5IEFERF9JVEVNX0VSUk9SOiBzdHJpbmcgPSAnVVNFUl9BRERfSVRFTV9FUlJPUic7XG4gIHN0YXRpYyByZWFkb25seSBPVEhFUl9DT05TVEFOVDogc3RyaW5nID0gJ09USEVSX0NPTlNUQU5UJztcbn1cblxuY29uc3RydWN0b3IoZmx1eDogRmx1eEZyYW1ld29yaywgQ3VzdG9tQWRhcHRlcjogdHlwZW9mIEV2ZW50ID0gRXZlbnQpIHt0aGlzLkN1c3RvbUFkYXB0ZXIgPSBDdXN0b21BZGFwdGVyO1xuICB0aGlzLmZsdXggPSBmbHV4O31cblxuaW1wb3J0IHthcHAsIGV2ZW50cywgaW1hZ2VzLCBsb2NhdGlvbnMsIG1lc3NhZ2VzLCBwb3N0cywgdGFncywgdXNlcnMsIHdlYnNvY2tldH0gZnJvbSAnLi9zdG9yZXMnO1xuXG5jb25zdCBjb25maWcgPSB7YmFzZVVybDogJ2h0dHBzOi8vYXBpLmV4YW1wbGUuY29tJywgYXBpS2V5OiAndmFsdWUnLCB0aW1lb3V0OiA1MDAwfTtcblxuRml4IE9OTFkgdGhlIHNwZWNpZmljIEVTTGludCBlcnJvcnMuIFJldHVybiB0aGUgcHJvcGVybHkgZm9ybWF0dGVkIGNvZGUuYDtcblxuICAgICAgICAgIHRyeSB7XG4gICAgICAgICAgICBjb25zdCBwcm9tcHRGaWxlID0gcGF0aFJlc29sdmUoY3dkLCAnLmN1cnNvcl9wcm9tcHRfdGVtcC50eHQnKTtcbiAgICAgICAgICAgIHdyaXRlRmlsZVN5bmMocHJvbXB0RmlsZSwgcHJvbXB0LCAndXRmOCcpO1xuXG4gICAgICAgICAgICAvLyBVc2UgQ3Vyc29yIENMSSB0byBmaXggdGhlIGZpbGVcbiAgICAgICAgICAgIGF3YWl0IGV4ZWNhKCdjdXJzb3InLCBbJ2VkaXQnLCAnLS1maWxlJywgZmlsZVBhdGgsICctLXByb21wdC1maWxlJywgcHJvbXB0RmlsZV0sIHtcbiAgICAgICAgICAgICAgcmVqZWN0OiBmYWxzZSxcbiAgICAgICAgICAgICAgc3RkaW86ICdwaXBlJyxcbiAgICAgICAgICAgICAgY3dkXG4gICAgICAgICAgICB9KTtcblxuICAgICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgICAgdW5saW5rU3luYyhwcm9tcHRGaWxlKTtcbiAgICAgICAgICAgIH0gY2F0Y2ggKF9lcnJvcikge1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBsb2coYEFwcGxpZWQgQ3Vyc29yIEFJIGZpeGVzIHRvICR7ZmlsZVBhdGh9YCwgJ2luZm8nLCBxdWlldCk7XG4gICAgICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgICAgICAgIGNvbnN0IHdhc01vZGlmaWVkID0gYXdhaXQgYXBwbHlEaXJlY3RGaXhlcyhmaWxlUGF0aCwgcXVpZXQpO1xuICAgICAgICAgICAgaWYod2FzTW9kaWZpZWQpIHtcbiAgICAgICAgICAgICAgbG9nKGBBcHBsaWVkIGRpcmVjdCBmaXhlcyB0byAke2ZpbGVQYXRofWAsICdpbmZvJywgcXVpZXQpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICAgICAgICBsb2coYEVycm9yIHVzaW5nIEN1cnNvciBBSTogJHtlcnJvci5tZXNzYWdlfWAsICdlcnJvcicsIHF1aWV0KTtcbiAgICAgICAgICBhd2FpdCBhcHBseURpcmVjdEZpeGVzKGZpbGVQYXRoLCBxdWlldCk7XG4gICAgICAgIH1cbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIGNvbnN0IHdhc01vZGlmaWVkID0gYXdhaXQgYXBwbHlEaXJlY3RGaXhlcyhmaWxlUGF0aCwgcXVpZXQpO1xuICAgICAgICBpZih3YXNNb2RpZmllZCkge1xuICAgICAgICAgIGxvZyhgQXBwbGllZCBkaXJlY3QgZml4ZXMgdG8gJHtmaWxlUGF0aH1gLCAnaW5mbycsIHF1aWV0KTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IGZpbGVFcnJvcnMgPSBmaWxlRXJyb3JNYXAuZ2V0KGZpbGVQYXRoKSB8fCBbXTtcbiAgICAgICAgaWYoZmlsZUVycm9ycy5sZW5ndGggPiAwKSB7XG4gICAgICAgICAgdHJ5IHtcbiAgICAgICAgICAgIGNvbnN0IHtjYWxsQUlTZXJ2aWNlfSA9IGF3YWl0IGltcG9ydCgnLi4vLi4vdXRpbHMvYWlTZXJ2aWNlLmpzJyk7XG5cbiAgICAgICAgICAgIGNvbnN0IGZpbGVDb250ZW50ID0gcmVhZEZpbGVTeW5jKGZpbGVQYXRoLCAndXRmOCcpO1xuXG4gICAgICAgICAgICBjb25zdCBwcm9tcHQgPSBgRml4IHRoZSBmb2xsb3dpbmcgRVNMaW50IGVycm9ycyBpbiB0aGlzIGNvZGU6XG4ke2ZpbGVFcnJvcnMuam9pbignXFxuJyl9XG5cbkhlcmUncyB0aGUgY29kZTpcblxcYFxcYFxcYFxuJHtmaWxlQ29udGVudH1cblxcYFxcYFxcYFxuXG5DUklUSUNBTCBSRVFVSVJFTUVOVFM6XG4tIE9OTFkgZml4IHRoZSBzcGVjaWZpYyBsaW5lcyB3aXRoIEVTTGludCBlcnJvcnNcbi0gRE8gTk9UIG1vZGlmeSBhbnkgb3RoZXIgbGluZXMgb2YgY29kZVxuLSBETyBOT1QgcmVtb3ZlIGxpbmUgYnJlYWtzIHVubGVzcyB0aGV5IGFyZSBzcGVjaWZpY2FsbHkgY2F1c2luZyBFU0xpbnQgZXJyb3JzXG4tIERPIE5PVCBjb25kZW5zZSBtdWx0aS1saW5lIHN0cnVjdHVyZXMgdG8gc2luZ2xlIGxpbmVzXG4tIFBSRVNFUlZFIGFsbCBleGlzdGluZyBsaW5lIGJyZWFrcyBhbmQgZm9ybWF0dGluZyB0aGF0IGlzIG5vdCBjYXVzaW5nIGVycm9yc1xuXG5TUEVDSUZJQyBGT1JNQVRUSU5HIFJVTEVTOlxuLSBNYWludGFpbiBwcm9wZXIgaW5kZW50YXRpb24gKDIgc3BhY2VzKVxuLSBLZWVwIGxpbmUgYnJlYWtzIGJldHdlZW4gY2xhc3MvaW50ZXJmYWNlIGRlY2xhcmF0aW9uIGFuZCB0aGVpciBtZW1iZXJzXG4tIEtlZXAgbGluZSBicmVha3MgYmV0d2VlbiBtZXRob2RzXG4tIEVuc3VyZSB0aGVyZSBpcyBhIGxpbmUgYnJlYWsgYWZ0ZXIgb3BlbmluZyBicmFjZXMgZm9yIGNsYXNzZXMsIGludGVyZmFjZXMsIGFuZCBtZXRob2RzXG4tIERPIE5PVCBwbGFjZSBjbGFzcy9pbnRlcmZhY2UgcHJvcGVydGllcyBvciBtZXRob2RzIG9uIHRoZSBzYW1lIGxpbmUgYXMgdGhlIG9wZW5pbmcgYnJhY2Vcbi0gUHJlc2VydmUgZW1wdHkgbGluZXMgYmV0d2VlbiBsb2dpY2FsIGNvZGUgYmxvY2tzXG4tIFBSRVNFUlZFIG11bHRpLWxpbmUgaW1wb3J0cyAtIGRvIG5vdCBjb25kZW5zZSB0aGVtIHRvIHNpbmdsZSBsaW5lc1xuLSBQUkVTRVJWRSBtdWx0aS1saW5lIG9iamVjdC9hcnJheSBkZWNsYXJhdGlvbnMgLSBkbyBub3QgY29uZGVuc2UgdGhlbSB0byBzaW5nbGUgbGluZXNcblxuU09SVC1LRVlTIFJVTEUgKEhJR0hFU1QgUFJJT1JJVFkpOlxuLSBBbGwgb2JqZWN0IGxpdGVyYWwga2V5cyBNVVNUIGJlIHNvcnRlZCBhbHBoYWJldGljYWxseSBpbiBhc2NlbmRpbmcgb3JkZXJcbi0gVGhpcyBhcHBsaWVzIHRvIEFMTCBvYmplY3RzIGluIHRoZSBmaWxlLCBub3QganVzdCB0aG9zZSB3aXRoIGV4cGxpY2l0IHNvcnQta2V5cyBlcnJvcnNcbi0gRXhhbXBsZToge2I6IDIsIGE6IDEsIGM6IDN9IHNob3VsZCBiZWNvbWUge2E6IDEsIGI6IDIsIGM6IDN9XG4tIFByZXNlcnZlIHRoZSBvcmlnaW5hbCBmb3JtYXR0aW5nIGFuZCBsaW5lIGJyZWFrcyB3aGVuIHNvcnRpbmdcblxuV0hBVCBUTyBGSVg6XG4xLiBTb3J0aW5nIGFsbCBvYmplY3Qga2V5cyBhbHBoYWJldGljYWxseSAoc29ydC1rZXlzIHJ1bGUpIC0gQUxMIG9iamVjdHMgbXVzdCBoYXZlIHNvcnRlZCBrZXlzXG4yLiBGaXhpbmcgbmFtaW5nIGNvbnZlbnRpb25zIC0gT05MWSBmb3IgdmFyaWFibGVzL2Z1bmN0aW9ucyB3aXRoIG5hbWluZyBlcnJvcnNcbjMuIFJlcGxhY2luZyBjb25zb2xlLmxvZyB3aXRoIGxvZyB1dGlsaXR5IC0gT05MWSBmb3IgY29uc29sZS5sb2cgc3RhdGVtZW50c1xuNC4gRml4aW5nIG5vLXBsdXNwbHVzIGlzc3VlcyAtIE9OTFkgZm9yICsrLy0tIG9wZXJhdG9yc1xuNS4gRml4aW5nIHVubmVjZXNzYXJ5IGVzY2FwZSBjaGFyYWN0ZXJzIC0gT05MWSBmb3IgZXNjYXBlZCBjaGFyYWN0ZXJzIHRoYXQgZG9uJ3QgbmVlZCBlc2NhcGluZ1xuNi4gUHJvcGVyIGluZGVudGF0aW9uIGFuZCBzcGFjaW5nIC0gT05MWSB3aGVyZSBzcGVjaWZpY2FsbHkgcmVxdWlyZWQgYnkgZXJyb3JzXG43LiBTdHJpbmcgcXVvdGVzIGNvbnNpc3RlbmN5ICh1c2Ugc2luZ2xlIHF1b3RlcykgLSBPTkxZIGZvciBzdHJpbmcgbGl0ZXJhbHMgd2l0aCBxdW90ZSBlcnJvcnNcbjguIEltcG9ydCBvcmRlciBhbmQgc3BhY2luZyAtIE9OTFkgZm9yIGltcG9ydHMgd2l0aCBvcmRlci9zcGFjaW5nIGVycm9yc1xuOS4gRnVuY3Rpb24gcGFyYW1ldGVyIGZvcm1hdHRpbmcgLSBPTkxZIGZvciBmdW5jdGlvbnMgd2l0aCBwYXJhbWV0ZXIgZXJyb3JzXG4xMC4gVmFyaWFibGUgbmFtaW5nIGNvbnZlbnRpb25zIC0gT05MWSBmb3IgdmFyaWFibGVzIHdpdGggbmFtaW5nIGVycm9yc1xuMTEuIE5vIHVudXNlZCB2YXJpYWJsZXMgb3IgaW1wb3J0cyAtIE9OTFkgZm9yIHVudXNlZCB2YXJpYWJsZXMvaW1wb3J0c1xuMTIuIEF2b2lkaW5nIG5lc3RlZCB0ZXJuYXJpZXMgLSBPTkxZIGZvciBuZXN0ZWQgdGVybmFyeSBleHByZXNzaW9uc1xuMTMuIEFueSBvdGhlciBFU0xpbnQgZXJyb3JzIC0gT05MWSBmb3IgdGhlIHNwZWNpZmljIGVycm9ycyBsaXN0ZWQgYWJvdmVcblxuV0hBVCBOT1QgVE8gRklYOlxuLSBEbyBub3QgY2hhbmdlIHByb3Blcmx5IGZvcm1hdHRlZCBtdWx0aS1saW5lIHN0cnVjdHVyZXNcbi0gRG8gbm90IHJlbW92ZSBsaW5lIGJyZWFrcyB0aGF0IGFyZSBub3QgY2F1c2luZyBlcnJvcnNcbi0gRG8gbm90IGNoYW5nZSBpbmRlbnRhdGlvbiB0aGF0IGlzIGFscmVhZHkgY29ycmVjdFxuLSBEbyBub3QgbW9kaWZ5IHNwYWNpbmcgdGhhdCBpcyBhbHJlYWR5IGNvcnJlY3Rcbi0gRG8gbm90IGNvbmRlbnNlIHJlYWRhYmxlIG11bHRpLWxpbmUgY29kZSB0byBzaW5nbGUgbGluZXNcbi0gRG8gbm90IG1vZGlmeSBjb2RlIHRoYXQgaXMgbm90IG1lbnRpb25lZCBpbiB0aGUgRVNMaW50IGVycm9yc1xuXG5FeGFtcGxlIG9mIENPUlJFQ1QgZm9ybWF0dGluZyAoRE8gTk9UIENIQU5HRSk6XG5leHBvcnQgY2xhc3MgVXNlckNvbnN0YW50cyB7XG4gIHN0YXRpYyByZWFkb25seSBBRERfSVRFTV9FUlJPUjogc3RyaW5nID0gJ1VTRVJfQUREX0lURU1fRVJST1InO1xuICBzdGF0aWMgcmVhZG9ubHkgT1RIRVJfQ09OU1RBTlQ6IHN0cmluZyA9ICdPVEhFUl9DT05TVEFOVCc7XG59XG5cbmNvbnN0cnVjdG9yKGZsdXg6IEZsdXhGcmFtZXdvcmssIEN1c3RvbUFkYXB0ZXI6IHR5cGVvZiBFdmVudCA9IEV2ZW50KSB7XG4gIHRoaXMuQ3VzdG9tQWRhcHRlciA9IEN1c3RvbUFkYXB0ZXI7XG4gIHRoaXMuZmx1eCA9IGZsdXg7XG59XG5cbmltcG9ydCB7XG4gIGFwcCxcbiAgZXZlbnRzLFxuICBpbWFnZXMsXG4gIGxvY2F0aW9ucyxcbiAgbWVzc2FnZXMsXG4gIHBvc3RzLFxuICB0YWdzLFxuICB1c2VycyxcbiAgd2Vic29ja2V0XG59IGZyb20gJy4vc3RvcmVzJztcblxuY29uc3QgY29uZmlnID0ge1xuICBhcGlLZXk6ICd2YWx1ZScsXG4gIGJhc2VVcmw6ICdodHRwczovL2FwaS5leGFtcGxlLmNvbScsXG4gIHRpbWVvdXQ6IDUwMDBcbn07XG5cbkV4YW1wbGUgb2YgSU5DT1JSRUNUIGZvcm1hdHRpbmcgKEZJWCBUSElTKTpcbmV4cG9ydCBjbGFzcyBVc2VyQ29uc3RhbnRzIHtzdGF0aWMgcmVhZG9ubHkgQUREX0lURU1fRVJST1I6IHN0cmluZyA9ICdVU0VSX0FERF9JVEVNX0VSUk9SJztcbiAgc3RhdGljIHJlYWRvbmx5IE9USEVSX0NPTlNUQU5UOiBzdHJpbmcgPSAnT1RIRVJfQ09OU1RBTlQnO1xufVxuXG5jb25zdHJ1Y3RvcihmbHV4OiBGbHV4RnJhbWV3b3JrLCBDdXN0b21BZGFwdGVyOiB0eXBlb2YgRXZlbnQgPSBFdmVudCkge3RoaXMuQ3VzdG9tQWRhcHRlciA9IEN1c3RvbUFkYXB0ZXI7XG4gIHRoaXMuZmx1eCA9IGZsdXg7fVxuXG5pbXBvcnQge2FwcCwgZXZlbnRzLCBpbWFnZXMsIGxvY2F0aW9ucywgbWVzc2FnZXMsIHBvc3RzLCB0YWdzLCB1c2Vycywgd2Vic29ja2V0fSBmcm9tICcuL3N0b3Jlcyc7XG5cbmNvbnN0IGNvbmZpZyA9IHtiYXNlVXJsOiAnaHR0cHM6Ly9hcGkuZXhhbXBsZS5jb20nLCBhcGlLZXk6ICd2YWx1ZScsIHRpbWVvdXQ6IDUwMDB9O1xuXG5GaXggT05MWSB0aGUgc3BlY2lmaWMgRVNMaW50IGVycm9ycyBsaXN0ZWQgYWJvdmUuIFJldmlldyB0aGUgZW50aXJlIGZpbGUgZm9yIGNvbXBsaWFuY2Ugd2l0aCBhbGwgRVNMaW50IHJ1bGVzLlxuUmV0dXJuIG9ubHkgdGhlIHByb3Blcmx5IGZvcm1hdHRlZCBmaXhlZCBjb2RlIHdpdGhvdXQgYW55IGV4cGxhbmF0aW9ucy5gO1xuXG4gICAgICAgICAgICBjb25zdCBmaXhlZENvbnRlbnQgPSBhd2FpdCBjYWxsQUlTZXJ2aWNlKHByb21wdCwgcXVpZXQpO1xuXG4gICAgICAgICAgICBpZihmaXhlZENvbnRlbnQgJiYgZml4ZWRDb250ZW50ICE9PSBmaWxlQ29udGVudCkge1xuICAgICAgICAgICAgICB3cml0ZUZpbGVTeW5jKGZpbGVQYXRoLCBmaXhlZENvbnRlbnQsICd1dGY4Jyk7XG4gICAgICAgICAgICAgIGxvZyhgQXBwbGllZCBBSSBmaXhlcyB0byAke2ZpbGVQYXRofWAsICdpbmZvJywgcXVpZXQpO1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICAgICAgICBsb2coYEVycm9yIGFwcGx5aW5nIEFJIGZpeGVzIHRvICR7ZmlsZVBhdGh9OiAke2Vycm9yLm1lc3NhZ2V9YCwgJ2Vycm9yJywgcXVpZXQpO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cblxuICAgIHNwaW5uZXIuc3VjY2VlZCgnQUkgZml4ZXMgYXBwbGllZCBzdWNjZXNzZnVsbHkhJyk7XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgc3Bpbm5lci5mYWlsKCdGYWlsZWQgdG8gYXBwbHkgQUkgZml4ZXMnKTtcbiAgICBsb2coYEVycm9yOiAke2Vycm9yLm1lc3NhZ2V9YCwgJ2Vycm9yJywgcXVpZXQpO1xuICAgIGlmKCFxdWlldCkge1xuICAgICAgY29uc29sZS5lcnJvcihlcnJvcik7XG4gICAgfVxuICB9XG59O1xuXG5jb25zdCBhcHBseURpcmVjdEZpeGVzID0gYXN5bmMgKGZpbGVQYXRoOiBzdHJpbmcsIHF1aWV0OiBib29sZWFuKTogUHJvbWlzZTxib29sZWFuPiA9PiB7XG4gIGxldCB3YXNNb2RpZmllZCA9IGZhbHNlO1xuXG4gIHRyeSB7XG4gICAgY29uc3QgZmlsZUNvbnRlbnQgPSByZWFkRmlsZVN5bmMoZmlsZVBhdGgsICd1dGY4Jyk7XG4gICAgbGV0IG5ld0NvbnRlbnQgPSBmaWxlQ29udGVudDtcblxuICAgIGlmKGZpbGVQYXRoLmluY2x1ZGVzKCdhaVNlcnZpY2UudHMnKSkge1xuICAgICAgbG9nKCdGaXhpbmcgaXNzdWVzIGluIGFpU2VydmljZS50cycsICdpbmZvJywgcXVpZXQpO1xuXG4gICAgICBuZXdDb250ZW50ID0gbmV3Q29udGVudC5yZXBsYWNlKFxuICAgICAgICAvJ0NvbnRlbnQtVHlwZSc6ICdhcHBsaWNhdGlvblxcL2pzb24nLFxccyonQXV0aG9yaXphdGlvbic6IGBCZWFyZXIvZyxcbiAgICAgICAgJ1xcJ0F1dGhvcml6YXRpb25cXCc6IGBCZWFyZXJcXCcsIFxcJ0NvbnRlbnQtVHlwZVxcJzogXFwnYXBwbGljYXRpb24vanNvblxcJydcbiAgICAgICk7XG5cbiAgICAgIG5ld0NvbnRlbnQgPSBuZXdDb250ZW50LnJlcGxhY2UoXG4gICAgICAgIC9oZWFkZXJzOiB7KFtefV0qKX0sXFxzKm1ldGhvZDogJ1BPU1QnL2csXG4gICAgICAgICdtZXRob2Q6IFxcJ1BPU1RcXCcsXFxuICAgICAgaGVhZGVyczogeyQxfSdcbiAgICAgICk7XG5cbiAgICAgIG5ld0NvbnRlbnQgPSBuZXdDb250ZW50LnJlcGxhY2UoXG4gICAgICAgIC97cm9sZTogJ3N5c3RlbScsIGNvbnRlbnQ6L2csXG4gICAgICAgICd7Y29udGVudDosIHJvbGU6IFxcJ3N5c3RlbVxcJywnXG4gICAgICApO1xuICAgICAgbmV3Q29udGVudCA9IG5ld0NvbnRlbnQucmVwbGFjZShcbiAgICAgICAgL3tyb2xlOiAndXNlcicsIGNvbnRlbnQ6L2csXG4gICAgICAgICd7Y29udGVudDosIHJvbGU6IFxcJ3VzZXJcXCcsJ1xuICAgICAgKTtcblxuICAgICAgbmV3Q29udGVudCA9IG5ld0NvbnRlbnQucmVwbGFjZShcbiAgICAgICAgL1xcKChbXildKj8pXyhbYS16QS1aMC05XSspKFxccyo6W14pXSopXFwpL2csXG4gICAgICAgICcoJDEkMiQzKSdcbiAgICAgICk7XG5cbiAgICAgIG5ld0NvbnRlbnQgPSBuZXdDb250ZW50LnJlcGxhY2UoL2NvbnNvbGVcXC5sb2dcXCgvZywgJ2xvZygnKTtcblxuICAgICAgaWYoIW5ld0NvbnRlbnQuaW5jbHVkZXMoJ2ltcG9ydCB7bG9nfScpICYmIG5ld0NvbnRlbnQuaW5jbHVkZXMoJ2xvZygnKSkge1xuICAgICAgICBuZXdDb250ZW50ID0gbmV3Q29udGVudC5yZXBsYWNlKFxuICAgICAgICAgIC9pbXBvcnQgeyhbXn1dKil9IGZyb20gJyguKiknOy8sXG4gICAgICAgICAgJ2ltcG9ydCB7JDF9IGZyb20gXFwnJDJcXCc7XFxuaW1wb3J0IHtsb2d9IGZyb20gXFwnLi9sb2cuanNcXCc7J1xuICAgICAgICApO1xuICAgICAgfVxuICAgIH1cblxuICAgIGlmKGZpbGVQYXRoLmluY2x1ZGVzKCdyZWFjdFNoaW0udHMnKSkge1xuICAgICAgbG9nKCdGaXhpbmcgbmFtaW5nLWNvbnZlbnRpb24gaXNzdWVzIGluIHJlYWN0U2hpbS50cycsICdpbmZvJywgcXVpZXQpO1xuXG4gICAgICBuZXdDb250ZW50ID0gbmV3Q29udGVudC5yZXBsYWNlKFxuICAgICAgICAnaW1wb3J0ICogYXMgUmVhY3QgZnJvbScsXG4gICAgICAgICdpbXBvcnQgKiBhcyByZWFjdCBmcm9tJ1xuICAgICAgKTtcblxuICAgICAgbmV3Q29udGVudCA9IG5ld0NvbnRlbnQucmVwbGFjZSgvUmVhY3RcXC4vZywgJ3JlYWN0LicpO1xuICAgIH1cblxuICAgIGlmKGZpbGVQYXRoLmluY2x1ZGVzKCdjaGFuZ2Vsb2cudHMnKSkge1xuICAgICAgbG9nKCdGaXhpbmcgaXNzdWVzIGluIGNoYW5nZWxvZy50cycsICdpbmZvJywgcXVpZXQpO1xuXG4gICAgICBuZXdDb250ZW50ID0gbmV3Q29udGVudC5yZXBsYWNlKC8oXFx3KylcXCtcXCsvZywgJyQxICs9IDEnKTtcblxuICAgICAgbmV3Q29udGVudCA9IG5ld0NvbnRlbnQucmVwbGFjZSgvXFxcXFxcJC9nLCAnJCcpO1xuICAgICAgbmV3Q29udGVudCA9IG5ld0NvbnRlbnQucmVwbGFjZSgvXFxcXFxcLi9nLCAnLicpO1xuICAgICAgbmV3Q29udGVudCA9IG5ld0NvbnRlbnQucmVwbGFjZSgvXFxcXFxcKi9nLCAnKicpO1xuICAgICAgbmV3Q29udGVudCA9IG5ld0NvbnRlbnQucmVwbGFjZSgvXFxcXDovZywgJzonKTtcbiAgICB9XG5cbiAgICBpZihmaWxlUGF0aC5pbmNsdWRlcygnYXBwLnRzJykpIHtcbiAgICAgIGxvZygnRml4aW5nIGlzc3VlcyBpbiBhcHAudHMnLCAnaW5mbycsIHF1aWV0KTtcblxuICAgICAgbmV3Q29udGVudCA9IG5ld0NvbnRlbnQucmVwbGFjZSgvY29uc29sZVxcLmxvZ1xcKC9nLCAnbG9nKCcpO1xuXG4gICAgICBpZighbmV3Q29udGVudC5pbmNsdWRlcygnaW1wb3J0IHtsb2d9JykgJiYgbmV3Q29udGVudC5pbmNsdWRlcygnbG9nKCcpKSB7XG4gICAgICAgIG5ld0NvbnRlbnQgPSBuZXdDb250ZW50LnJlcGxhY2UoXG4gICAgICAgICAgL2ltcG9ydCBib3hlbiBmcm9tICdib3hlbic7LyxcbiAgICAgICAgICAnaW1wb3J0IGJveGVuIGZyb20gXFwnYm94ZW5cXCc7XFxuaW1wb3J0IHtsb2d9IGZyb20gXFwnLi9sb2cuanNcXCc7J1xuICAgICAgICApO1xuICAgICAgfVxuXG4gICAgICBuZXdDb250ZW50ID0gbmV3Q29udGVudC5yZXBsYWNlKC9cXFxcXFwvL2csICcvJyk7XG4gICAgfVxuXG4gICAgaWYoZmlsZVBhdGguaW5jbHVkZXMoJ2F1dG9maXguanMnKSkge1xuICAgICAgbG9nKCdGaXhpbmcgaXNzdWVzIGluIGF1dG9maXguanMnLCAnaW5mbycsIHF1aWV0KTtcblxuICAgICAgbmV3Q29udGVudCA9IG5ld0NvbnRlbnQucmVwbGFjZShcbiAgICAgICAgL2ltcG9ydCB7KFtefV0qKX0gZnJvbSAncGF0aCc7W1xcc1xcbl0qaW1wb3J0IHsoW159XSopfSBmcm9tICdwYXRoJzsvLFxuICAgICAgICAnaW1wb3J0IHskMSwgJDJ9IGZyb20gXFwncGF0aFxcJzsnXG4gICAgICApO1xuXG4gICAgICBuZXdDb250ZW50ID0gbmV3Q29udGVudC5yZXBsYWNlKFxuICAgICAgICAvX19maWxlbmFtZS9nLFxuICAgICAgICAnY3VycmVudEZpbGVuYW1lJ1xuICAgICAgKTtcbiAgICAgIG5ld0NvbnRlbnQgPSBuZXdDb250ZW50LnJlcGxhY2UoXG4gICAgICAgIC9fX2Rpcm5hbWUvZyxcbiAgICAgICAgJ2N1cnJlbnREaXJuYW1lJ1xuICAgICAgKTtcblxuICAgICAgbmV3Q29udGVudCA9IG5ld0NvbnRlbnQucmVwbGFjZShcbiAgICAgICAgL2NvbnN0IHByZWZpeCA9IHR5cGUgPT09ICdlcnJvcicgXFw/ICdcdTI3NEMgJyA6IHR5cGUgPT09ICdzdWNjZXNzJyBcXD8gJ1x1MjcwNSAnIDogJ1x1MjEzOVx1RkUwRiAnOy8sXG4gICAgICAgICdsZXQgcHJlZml4ID0gXFwnXHUyMTM5XHVGRTBGIFxcJztcXG5pZih0eXBlID09PSBcXCdlcnJvclxcJykge1xcbiAgcHJlZml4ID0gXFwnXHUyNzRDIFxcJztcXG59IGVsc2UgaWYodHlwZSA9PT0gXFwnc3VjY2Vzc1xcJykge1xcbiAgcHJlZml4ID0gXFwnXHUyNzA1IFxcJztcXG59J1xuICAgICAgKTtcblxuICAgICAgbmV3Q29udGVudCA9IG5ld0NvbnRlbnQucmVwbGFjZShcbiAgICAgICAgL2FzeW5jIGZ1bmN0aW9uIHJ1bkVzbGludEZpeFxcKFxcKS9nLFxuICAgICAgICAnY29uc3QgcnVuRXNsaW50Rml4ID0gYXN5bmMgKCknXG4gICAgICApO1xuICAgICAgbmV3Q29udGVudCA9IG5ld0NvbnRlbnQucmVwbGFjZShcbiAgICAgICAgL2FzeW5jIGZ1bmN0aW9uIGdldEZpbGVzV2l0aEVycm9yc1xcKFxcKS9nLFxuICAgICAgICAnY29uc3QgZ2V0RmlsZXNXaXRoRXJyb3JzID0gYXN5bmMgKCknXG4gICAgICApO1xuICAgICAgbmV3Q29udGVudCA9IG5ld0NvbnRlbnQucmVwbGFjZShcbiAgICAgICAgL2FzeW5jIGZ1bmN0aW9uIGlzQ3Vyc29yQXZhaWxhYmxlXFwoXFwpL2csXG4gICAgICAgICdjb25zdCBpc0N1cnNvckF2YWlsYWJsZSA9IGFzeW5jICgpJ1xuICAgICAgKTtcbiAgICAgIG5ld0NvbnRlbnQgPSBuZXdDb250ZW50LnJlcGxhY2UoXG4gICAgICAgIC9hc3luYyBmdW5jdGlvbiBmaXhGaWxlV2l0aEN1cnNvckFJXFwoZmlsZVBhdGhcXCkvZyxcbiAgICAgICAgJ2NvbnN0IGZpeEZpbGVXaXRoQ3Vyc29yQUkgPSBhc3luYyAoZmlsZVBhdGgpJ1xuICAgICAgKTtcbiAgICAgIG5ld0NvbnRlbnQgPSBuZXdDb250ZW50LnJlcGxhY2UoXG4gICAgICAgIC9hc3luYyBmdW5jdGlvbiBtYWluXFwoXFwpL2csXG4gICAgICAgICdjb25zdCBtYWluID0gYXN5bmMgKCknXG4gICAgICApO1xuXG4gICAgICBuZXdDb250ZW50ID0gbmV3Q29udGVudC5yZXBsYWNlKFxuICAgICAgICAvaW1wb3J0IHtleGlzdHNTeW5jLCByZWFkRmlsZVN5bmMsIHdyaXRlRmlsZVN5bmN9L2csXG4gICAgICAgICdpbXBvcnQge3dyaXRlRmlsZVN5bmN9J1xuICAgICAgKTtcblxuICAgICAgbmV3Q29udGVudCA9IG5ld0NvbnRlbnQucmVwbGFjZShcbiAgICAgICAgL2NvbnNvbGVcXC5sb2dcXChgXFwke3ByZWZpeH0gXFwke21lc3NhZ2V9YFxcKTsvZyxcbiAgICAgICAgJ3Byb2Nlc3Muc3Rkb3V0LndyaXRlKGAke3ByZWZpeH0gJHttZXNzYWdlfVxcXFxuYCk7J1xuICAgICAgKTtcblxuICAgICAgbmV3Q29udGVudCA9IG5ld0NvbnRlbnQucmVwbGFjZShcbiAgICAgICAgL30gY2F0Y2hcXChlcnJvclxcKSB7W1xcc1xcbl0qXFwvXFwvIElnbm9yZSBjbGVhbnVwIGVycm9ycy9nLFxuICAgICAgICAnfSBjYXRjaChfKSB7XFxuICAgICAgLy8gSWdub3JlIGNsZWFudXAgZXJyb3JzJ1xuICAgICAgKTtcbiAgICAgIG5ld0NvbnRlbnQgPSBuZXdDb250ZW50LnJlcGxhY2UoXG4gICAgICAgIC99IGNhdGNoXFwoZXJyb3JcXCkge1tcXHNcXG5dKmxvZ1xcKC9nLFxuICAgICAgICAnfSBjYXRjaChlcnIpIHtcXG4gICAgbG9nKCdcbiAgICAgICk7XG4gICAgICBuZXdDb250ZW50ID0gbmV3Q29udGVudC5yZXBsYWNlKFxuICAgICAgICAvfSBjYXRjaFxcKGVycm9yXFwpIHtbXFxzXFxuXSpyZXR1cm4gZmFsc2U7L2csXG4gICAgICAgICd9IGNhdGNoKF8pIHtcXG4gICAgcmV0dXJuIGZhbHNlOydcbiAgICAgICk7XG5cbiAgICAgIG5ld0NvbnRlbnQgPSBuZXdDb250ZW50LnJlcGxhY2UoXG4gICAgICAgIC9mb3JcXChjb25zdCBmaWxlUGF0aCBvZiBmaWxlc1dpdGhFcnJvcnNcXCkge1tcXHNcXG5dKmNvbnN0IHN1Y2Nlc3MgPSBhd2FpdCBmaXhGaWxlV2l0aEN1cnNvckFJXFwoZmlsZVBhdGhcXCk7L2csXG4gICAgICAgICdjb25zdCBmaXhSZXN1bHRzID0gYXdhaXQgUHJvbWlzZS5hbGwoZmlsZXNXaXRoRXJyb3JzLm1hcChmaWxlUGF0aCA9PiBmaXhGaWxlV2l0aEN1cnNvckFJKGZpbGVQYXRoKSkpO1xcbmZvcihjb25zdCBzdWNjZXNzIG9mIGZpeFJlc3VsdHMpIHsnXG4gICAgICApO1xuXG4gICAgICBuZXdDb250ZW50ID0gbmV3Q29udGVudC5yZXBsYWNlKFxuICAgICAgICAvZml4ZWRDb3VudFxcK1xcKzsvZyxcbiAgICAgICAgJ2ZpeGVkQ291bnQgKz0gMTsnXG4gICAgICApO1xuICAgIH1cblxuICAgIGlmKG5ld0NvbnRlbnQgIT09IGZpbGVDb250ZW50KSB7XG4gICAgICB3cml0ZUZpbGVTeW5jKGZpbGVQYXRoLCBuZXdDb250ZW50LCAndXRmOCcpO1xuICAgICAgbG9nKGBGaXhlZCBpc3N1ZXMgaW4gJHtmaWxlUGF0aH1gLCAnaW5mbycsIHF1aWV0KTtcbiAgICAgIHdhc01vZGlmaWVkID0gdHJ1ZTtcbiAgICB9XG5cbiAgICByZXR1cm4gd2FzTW9kaWZpZWQ7XG4gIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgbG9nKGBFcnJvciBhcHBseWluZyBkaXJlY3QgZml4ZXMgdG8gJHtmaWxlUGF0aH06ICR7ZXJyb3IubWVzc2FnZX1gLCAnZXJyb3InLCBxdWlldCk7XG4gICAgcmV0dXJuIGZhbHNlO1xuICB9XG59O1xuXG5jb25zdCBsb2FkQUlDb25maWcgPSBhc3luYyAoY3dkOiBzdHJpbmcsIHF1aWV0OiBib29sZWFuLCBkZWJ1ZzogYm9vbGVhbiA9IGZhbHNlKTogUHJvbWlzZTx2b2lkPiA9PiB7XG4gIGNvbnN0IGNvbmZpZ0Zvcm1hdHMgPSBbJ2pzJywgJ21qcycsICdjanMnLCAndHMnLCAnanNvbiddO1xuICBjb25zdCBjb25maWdCYXNlTmFtZSA9ICdsZXguY29uZmlnJztcbiAgbGV0IGxleENvbmZpZ1BhdGggPSAnJztcblxuICBmb3IoY29uc3QgZm9ybWF0IG9mIGNvbmZpZ0Zvcm1hdHMpIHtcbiAgICBjb25zdCBwb3RlbnRpYWxQYXRoID0gcGF0aFJlc29sdmUoY3dkLCBgLi8ke2NvbmZpZ0Jhc2VOYW1lfS4ke2Zvcm1hdH1gKTtcbiAgICBpZihleGlzdHNTeW5jKHBvdGVudGlhbFBhdGgpKSB7XG4gICAgICBsZXhDb25maWdQYXRoID0gcG90ZW50aWFsUGF0aDtcbiAgICAgIGJyZWFrO1xuICAgIH1cbiAgfVxuXG4gIGlmKGxleENvbmZpZ1BhdGgpIHtcbiAgICB0cnkge1xuICAgICAgLy8gRm9yIE1KUyBmaWxlcywgd2UgbmVlZCB0byB1c2UgZHluYW1pYyBpbXBvcnQgd2l0aCBVUkwgZm9yIGNvbXBhdGliaWxpdHlcbiAgICAgIGNvbnN0IGZvcm1hdCA9IGV4dG5hbWUobGV4Q29uZmlnUGF0aCkuc2xpY2UoMSk7XG4gICAgICBsZXQgaW1wb3J0UGF0aCA9IGxleENvbmZpZ1BhdGg7XG5cbiAgICAgIC8vIFVzZSBVUkwgcHJvdG9jb2wgZm9yIEVTTSBpbXBvcnRzXG4gICAgICBpZihmb3JtYXQgPT09ICdtanMnKSB7XG4gICAgICAgIHRyeSB7XG4gICAgICAgICAgY29uc3QgdXJsID0gbmV3IFVSTChgZmlsZTovLyR7bGV4Q29uZmlnUGF0aH1gKTtcbiAgICAgICAgICBpbXBvcnRQYXRoID0gdXJsLmhyZWY7XG4gICAgICAgICAgaWYoZGVidWcpIHtcbiAgICAgICAgICAgIGxvZyhgVXNpbmcgVVJMIGZvcm1hdCBmb3IgTUpTIGltcG9ydDogJHtpbXBvcnRQYXRofWAsICdpbmZvJywgcXVpZXQpO1xuICAgICAgICAgIH1cbiAgICAgICAgfSBjYXRjaCAodXJsRXJyb3IpIHtcbiAgICAgICAgICBsb2coYEVycm9yIGNyZWF0aW5nIFVSTCBmb3IgTUpTIGltcG9ydDogJHt1cmxFcnJvci5tZXNzYWdlfWAsICd3YXJuJywgZGVidWcgfHwgIXF1aWV0KTtcbiAgICAgICAgICBpbXBvcnRQYXRoID0gYGZpbGU6Ly8ke2xleENvbmZpZ1BhdGh9YDtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBpZihkZWJ1Zykge1xuICAgICAgICBsb2coYFRyeWluZyB0byBpbXBvcnQgY29uZmlnIGZyb20gJHtpbXBvcnRQYXRofSAoZm9ybWF0OiAke2Zvcm1hdH0pYCwgJ2luZm8nLCBxdWlldCk7XG4gICAgICB9XG5cbiAgICAgIGxldCBsZXhDb25maWc7XG4gICAgICB0cnkge1xuICAgICAgICBsZXhDb25maWcgPSBhd2FpdCBpbXBvcnQoaW1wb3J0UGF0aCk7XG4gICAgICB9IGNhdGNoIChpbXBvcnRFcnJvcikge1xuICAgICAgICBpZihpbXBvcnRFcnJvci5tZXNzYWdlLmluY2x1ZGVzKCdub3QgZGVmaW5lZCBpbiBFUyBtb2R1bGUgc2NvcGUnKSkge1xuICAgICAgICAgIGxvZyhgRVMgTW9kdWxlIHN5bnRheCBlcnJvciBpbiAke2xleENvbmZpZ1BhdGh9LiBNYWtlIHN1cmUgeW91J3JlIHVzaW5nICdleHBvcnQnIGluc3RlYWQgb2YgJ21vZHVsZS5leHBvcnRzJy5gLCAnZXJyb3InLCBxdWlldCk7XG4gICAgICAgICAgaWYoZGVidWcpIHtcbiAgICAgICAgICAgIGNvbnNvbGUuZXJyb3IoaW1wb3J0RXJyb3IpO1xuICAgICAgICAgIH1cbiAgICAgICAgICByZXR1cm47XG4gICAgICAgIH1cbiAgICAgICAgdGhyb3cgaW1wb3J0RXJyb3I7XG4gICAgICB9XG5cbiAgICAgIC8vIEhhbmRsZSBib3RoIEVTTSAoZGVmYXVsdCBleHBvcnQpIGFuZCBDb21tb25KUyAobW9kdWxlLmV4cG9ydHMpXG4gICAgICBsZXQgY29uZmlnRGF0YSA9IG51bGw7XG4gICAgICBpZihsZXhDb25maWcuZGVmYXVsdCkge1xuICAgICAgICBjb25maWdEYXRhID0gbGV4Q29uZmlnLmRlZmF1bHQ7XG4gICAgICAgIGlmKGRlYnVnKSB7XG4gICAgICAgICAgbG9nKGBGb3VuZCBkZWZhdWx0IGV4cG9ydCBpbiAke2xleENvbmZpZ1BhdGh9YCwgJ2luZm8nLCBxdWlldCk7XG4gICAgICAgIH1cbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIC8vIEZvciBDb21tb25KUyBvciBvdGhlciBtb2R1bGUgc3lzdGVtc1xuICAgICAgICBjb25maWdEYXRhID0gbGV4Q29uZmlnO1xuICAgICAgICBpZihkZWJ1Zykge1xuICAgICAgICAgIGxvZyhgVXNpbmcgZGlyZWN0IGV4cG9ydCBpbiAke2xleENvbmZpZ1BhdGh9YCwgJ2luZm8nLCBxdWlldCk7XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgaWYoY29uZmlnRGF0YSAmJiBjb25maWdEYXRhLmFpKSB7XG4gICAgICAgIGxvZyhgRm91bmQgQUkgY29uZmlndXJhdGlvbiBpbiAke3BhdGhSZXNvbHZlKGN3ZCwgbGV4Q29uZmlnUGF0aCl9LCBhcHBseWluZyBzZXR0aW5ncy4uLmAsICdpbmZvJywgcXVpZXQpO1xuICAgICAgICBMZXhDb25maWcuY29uZmlnLmFpID0gey4uLkxleENvbmZpZy5jb25maWcuYWksIC4uLmNvbmZpZ0RhdGEuYWl9O1xuICAgICAgfVxuICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICBsb2coYEVycm9yIGxvYWRpbmcgQUkgY29uZmlndXJhdGlvbiBmcm9tICR7bGV4Q29uZmlnUGF0aH06ICR7ZXJyb3IubWVzc2FnZX1gLCAnd2FybicsIHF1aWV0KTtcbiAgICAgIGlmKGRlYnVnKSB7XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoZXJyb3IpO1xuICAgICAgfVxuICAgIH1cbiAgfVxufTtcblxuLyoqXG4gKiBMb2FkIEVTTGludCBjb25maWd1cmF0aW9uIGZyb20gbGV4LmNvbmZpZy4qIGZpbGVzXG4gKi9cbmNvbnN0IGxvYWRFU0xpbnRDb25maWcgPSBhc3luYyAoY3dkOiBzdHJpbmcsIHF1aWV0OiBib29sZWFuLCBkZWJ1ZzogYm9vbGVhbik6IFByb21pc2U8Ym9vbGVhbj4gPT4ge1xuICAvLyBDaGVjayBpZiBMZXhDb25maWcgYWxyZWFkeSBoYXMgRVNMaW50IGNvbmZpZ3VyYXRpb24gbG9hZGVkXG4gIGlmKExleENvbmZpZy5jb25maWcuZXNsaW50ICYmIE9iamVjdC5rZXlzKExleENvbmZpZy5jb25maWcuZXNsaW50KS5sZW5ndGggPiAwKSB7XG4gICAgbG9nKCdGb3VuZCBFU0xpbnQgY29uZmlndXJhdGlvbiBpbiBsZXguY29uZmlnLiogZmlsZScsICdpbmZvJywgZGVidWcgfHwgIXF1aWV0KTtcbiAgICByZXR1cm4gdHJ1ZTtcbiAgfVxuXG4gIC8vIFRyeSB0byBsb2FkIGZyb20gbGV4LmNvbmZpZy4qIGZpbGVzIGlmIG5vdCBhbHJlYWR5IGxvYWRlZFxuICBjb25zdCBjb25maWdGb3JtYXRzID0gWydqcycsICdtanMnLCAnY2pzJywgJ3RzJywgJ2pzb24nXTtcbiAgY29uc3QgY29uZmlnQmFzZU5hbWUgPSAnbGV4LmNvbmZpZyc7XG5cbiAgZm9yKGNvbnN0IGZvcm1hdCBvZiBjb25maWdGb3JtYXRzKSB7XG4gICAgY29uc3QgcG90ZW50aWFsUGF0aCA9IHBhdGhSZXNvbHZlKGN3ZCwgYC4vJHtjb25maWdCYXNlTmFtZX0uJHtmb3JtYXR9YCk7XG4gICAgaWYoZXhpc3RzU3luYyhwb3RlbnRpYWxQYXRoKSkge1xuICAgICAgdHJ5IHtcbiAgICAgICAgLy8gRm9yIE1KUyBmaWxlcywgd2UgbmVlZCB0byB1c2UgZHluYW1pYyBpbXBvcnQgd2l0aCBVUkwgZm9yIGNvbXBhdGliaWxpdHlcbiAgICAgICAgY29uc3QgZmlsZUZvcm1hdCA9IGV4dG5hbWUocG90ZW50aWFsUGF0aCkuc2xpY2UoMSk7XG4gICAgICAgIGxldCBpbXBvcnRQYXRoID0gcG90ZW50aWFsUGF0aDtcblxuICAgICAgICAvLyBVc2UgVVJMIHByb3RvY29sIGZvciBFU00gaW1wb3J0c1xuICAgICAgICBpZihmaWxlRm9ybWF0ID09PSAnbWpzJykge1xuICAgICAgICAgIHRyeSB7XG4gICAgICAgICAgICBjb25zdCB1cmwgPSBuZXcgVVJMKGBmaWxlOi8vJHtwb3RlbnRpYWxQYXRofWApO1xuICAgICAgICAgICAgaW1wb3J0UGF0aCA9IHVybC5ocmVmO1xuICAgICAgICAgICAgaWYoZGVidWcpIHtcbiAgICAgICAgICAgICAgbG9nKGBVc2luZyBVUkwgZm9ybWF0IGZvciBNSlMgaW1wb3J0OiAke2ltcG9ydFBhdGh9YCwgJ2luZm8nLCBxdWlldCk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfSBjYXRjaCAodXJsRXJyb3IpIHtcbiAgICAgICAgICAgIGxvZyhgRXJyb3IgY3JlYXRpbmcgVVJMIGZvciBNSlMgaW1wb3J0OiAke3VybEVycm9yLm1lc3NhZ2V9YCwgJ3dhcm4nLCBkZWJ1ZyB8fCAhcXVpZXQpO1xuICAgICAgICAgICAgaW1wb3J0UGF0aCA9IGBmaWxlOi8vJHtwb3RlbnRpYWxQYXRofWA7XG4gICAgICAgICAgfVxuICAgICAgICB9XG5cbiAgICAgICAgaWYoZGVidWcpIHtcbiAgICAgICAgICBsb2coYFRyeWluZyB0byBpbXBvcnQgY29uZmlnIGZyb20gJHtpbXBvcnRQYXRofSAoZm9ybWF0OiAke2ZpbGVGb3JtYXR9KWAsICdpbmZvJywgcXVpZXQpO1xuICAgICAgICB9XG5cbiAgICAgICAgbGV0IGxleENvbmZpZztcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICBsZXhDb25maWcgPSBhd2FpdCBpbXBvcnQoaW1wb3J0UGF0aCk7XG4gICAgICAgIH0gY2F0Y2ggKGltcG9ydEVycm9yKSB7XG4gICAgICAgICAgaWYoaW1wb3J0RXJyb3IubWVzc2FnZS5pbmNsdWRlcygnbm90IGRlZmluZWQgaW4gRVMgbW9kdWxlIHNjb3BlJykpIHtcbiAgICAgICAgICAgIGxvZyhgRVMgTW9kdWxlIHN5bnRheCBlcnJvciBpbiAke3BvdGVudGlhbFBhdGh9LiBNYWtlIHN1cmUgeW91J3JlIHVzaW5nICdleHBvcnQnIGluc3RlYWQgb2YgJ21vZHVsZS5leHBvcnRzJy5gLCAnZXJyb3InLCBxdWlldCk7XG4gICAgICAgICAgICBpZihkZWJ1Zykge1xuICAgICAgICAgICAgICBjb25zb2xlLmVycm9yKGltcG9ydEVycm9yKTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGNvbnRpbnVlO1xuICAgICAgICAgIH1cbiAgICAgICAgICB0aHJvdyBpbXBvcnRFcnJvcjtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIEhhbmRsZSBib3RoIEVTTSAoZGVmYXVsdCBleHBvcnQpIGFuZCBDb21tb25KUyAobW9kdWxlLmV4cG9ydHMpXG4gICAgICAgIGxldCBjb25maWdEYXRhID0gbnVsbDtcbiAgICAgICAgaWYobGV4Q29uZmlnLmRlZmF1bHQpIHtcbiAgICAgICAgICBjb25maWdEYXRhID0gbGV4Q29uZmlnLmRlZmF1bHQ7XG4gICAgICAgICAgaWYoZGVidWcpIHtcbiAgICAgICAgICAgIGxvZyhgRm91bmQgZGVmYXVsdCBleHBvcnQgaW4gJHtwb3RlbnRpYWxQYXRofWAsICdpbmZvJywgcXVpZXQpO1xuICAgICAgICAgIH1cbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAvLyBGb3IgQ29tbW9uSlMgb3Igb3RoZXIgbW9kdWxlIHN5c3RlbXNcbiAgICAgICAgICBjb25maWdEYXRhID0gbGV4Q29uZmlnO1xuICAgICAgICAgIGlmKGRlYnVnKSB7XG4gICAgICAgICAgICBsb2coYFVzaW5nIGRpcmVjdCBleHBvcnQgaW4gJHtwb3RlbnRpYWxQYXRofWAsICdpbmZvJywgcXVpZXQpO1xuICAgICAgICAgIH1cbiAgICAgICAgfVxuXG4gICAgICAgIGlmKGNvbmZpZ0RhdGEgJiYgY29uZmlnRGF0YS5lc2xpbnQgJiYgT2JqZWN0LmtleXMoY29uZmlnRGF0YS5lc2xpbnQpLmxlbmd0aCA+IDApIHtcbiAgICAgICAgICBsb2coYEZvdW5kIEVTTGludCBjb25maWd1cmF0aW9uIGluICR7cGF0aFJlc29sdmUoY3dkLCBwb3RlbnRpYWxQYXRoKX0sIGFwcGx5aW5nIHNldHRpbmdzLi4uYCwgJ2luZm8nLCBkZWJ1ZyB8fCAhcXVpZXQpO1xuICAgICAgICAgIExleENvbmZpZy5jb25maWcuZXNsaW50ID0gey4uLkxleENvbmZpZy5jb25maWcuZXNsaW50LCAuLi5jb25maWdEYXRhLmVzbGludH07XG4gICAgICAgICAgcmV0dXJuIHRydWU7XG4gICAgICAgIH1cbiAgICAgIH0gY2F0Y2ggKGVycm9yKSB7XG4gICAgICAgIGxvZyhgRXJyb3IgbG9hZGluZyBFU0xpbnQgY29uZmlndXJhdGlvbiBmcm9tICR7cG90ZW50aWFsUGF0aH06ICR7ZXJyb3IubWVzc2FnZX1gLCAnd2FybicsIHF1aWV0KTtcbiAgICAgICAgaWYoZGVidWcpIHtcbiAgICAgICAgICBjb25zb2xlLmVycm9yKGVycm9yKTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIHJldHVybiBmYWxzZTtcbn07XG5cbi8qKlxuICogUmVtb3ZlcyBjb21tZW50cyBmcm9tIGEgZmlsZSBleGNlcHQgZm9yIFRPRE9zIGFuZCBjb3B5cmlnaHQgbm90aWNlc1xuICovXG5jb25zdCByZW1vdmVGaWxlQ29tbWVudHMgPSAoZmlsZVBhdGg6IHN0cmluZywgcXVpZXQ6IGJvb2xlYW4pOiBib29sZWFuID0+IHtcbiAgdHJ5IHtcbiAgICBjb25zdCBmaWxlQ29udGVudCA9IHJlYWRGaWxlU3luYyhmaWxlUGF0aCwgJ3V0ZjgnKTtcblxuICAgIC8vIFNraXAgZmlsZXMgdGhhdCBhcmUgdG9vIGxhcmdlIHRvIGF2b2lkIHBlcmZvcm1hbmNlIGlzc3Vlc1xuICAgIGlmKGZpbGVDb250ZW50Lmxlbmd0aCA+IDEwMDAwMDApIHsgLy8gMU1CIGxpbWl0XG4gICAgICBsb2coYFNraXBwaW5nIGNvbW1lbnQgcmVtb3ZhbCBmb3IgbGFyZ2UgZmlsZTogJHtmaWxlUGF0aH1gLCAnaW5mbycsIHF1aWV0KTtcbiAgICAgIHJldHVybiBmYWxzZTtcbiAgICB9XG5cbiAgICAvLyBEZXRlcm1pbmUgZmlsZSB0eXBlIGZyb20gZXh0ZW5zaW9uXG4gICAgY29uc3QgZXh0ID0gZXh0bmFtZShmaWxlUGF0aCk7XG4gICAgbGV0IGlzVHlwZVNjcmlwdCA9IGZhbHNlO1xuICAgIGxldCBpc0phdmFTY3JpcHQgPSBmYWxzZTtcblxuICAgIGlmKFsnLnRzJywgJy50c3gnXS5pbmNsdWRlcyhleHQpKSB7XG4gICAgICBpc1R5cGVTY3JpcHQgPSB0cnVlO1xuICAgIH0gZWxzZSBpZihbJy5qcycsICcuanN4J10uaW5jbHVkZXMoZXh0KSkge1xuICAgICAgaXNKYXZhU2NyaXB0ID0gdHJ1ZTtcbiAgICB9IGVsc2Uge1xuICAgICAgLy8gU2tpcCB1bnN1cHBvcnRlZCBmaWxlIHR5cGVzXG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuXG4gICAgLy8gVXNlIHJlZ2V4IHRvIG1hdGNoIGRpZmZlcmVudCB0eXBlcyBvZiBjb21tZW50c1xuICAgIC8vIFByZXNlcnZlczpcbiAgICAvLyAxLiBDb3B5cmlnaHQgbm90aWNlcyAoLyogQ29weXJpZ2h0IC4uLiAqLylcbiAgICAvLyAyLiBUT0RPIGNvbW1lbnRzICgvLyBUT0RPOiAuLi4pXG4gICAgLy8gMy4gTGljZW5zZSBoZWFkZXJzICgvKiAuLi4gTGljZW5zZSAuLi4gKi8pXG5cbiAgICAvLyBIYW5kbGUgbXVsdGktbGluZSBjb21tZW50cyBmaXJzdCAtIHByZXNlcnZlIGNvcHlyaWdodC9saWNlbnNlIG5vdGljZXNcbiAgICBsZXQgbmV3Q29udGVudCA9IGZpbGVDb250ZW50LnJlcGxhY2UoXG4gICAgICAvXFwvXFwqW1xcc1xcU10qP1xcKlxcLy9nLFxuICAgICAgKG1hdGNoKSA9PiB7XG4gICAgICAgIGlmKG1hdGNoLmluY2x1ZGVzKCdDb3B5cmlnaHQnKSB8fFxuICAgICAgICAgIG1hdGNoLmluY2x1ZGVzKCdMSUNFTlNFJykgfHxcbiAgICAgICAgICBtYXRjaC5pbmNsdWRlcygnTGljZW5zZScpIHx8XG4gICAgICAgICAgbWF0Y2guaW5jbHVkZXMoJ2xpY2Vuc2UnKSkge1xuICAgICAgICAgIHJldHVybiBtYXRjaDtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gJyc7XG4gICAgICB9XG4gICAgKTtcblxuICAgIC8vIEhhbmRsZSBzaW5nbGUtbGluZSBjb21tZW50cyAtIHByZXNlcnZlIFRPRE9zXG4gICAgbmV3Q29udGVudCA9IG5ld0NvbnRlbnQucmVwbGFjZShcbiAgICAgIC9cXC9cXC8uKiQvZ20sXG4gICAgICAobWF0Y2gpID0+IHtcbiAgICAgICAgaWYobWF0Y2guaW5jbHVkZXMoJ1RPRE8nKSB8fCBtYXRjaC5pbmNsdWRlcygnRklYTUUnKSkge1xuICAgICAgICAgIHJldHVybiBtYXRjaDtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gJyc7XG4gICAgICB9XG4gICAgKTtcblxuICAgIC8vIENsZWFuIHVwIGFueSBtdWx0aXBsZSBibGFuayBsaW5lcyBjcmVhdGVkIGJ5IGNvbW1lbnQgcmVtb3ZhbFxuICAgIG5ld0NvbnRlbnQgPSBuZXdDb250ZW50LnJlcGxhY2UoL1xcblxccypcXG5cXHMqXFxuL2csICdcXG5cXG4nKTtcblxuICAgIC8vIElmIHRoZSBmaWxlIHdhcyBtb2RpZmllZCwgc2F2ZSBpdFxuICAgIGlmKG5ld0NvbnRlbnQgIT09IGZpbGVDb250ZW50KSB7XG4gICAgICB3cml0ZUZpbGVTeW5jKGZpbGVQYXRoLCBuZXdDb250ZW50LCAndXRmOCcpO1xuICAgICAgbG9nKGBSZW1vdmVkIGNvbW1lbnRzIGZyb20gJHtmaWxlUGF0aH1gLCAnaW5mbycsIHF1aWV0KTtcbiAgICAgIHJldHVybiB0cnVlO1xuICAgIH1cblxuICAgIHJldHVybiBmYWxzZTtcbiAgfSBjYXRjaCAoZXJyb3IpIHtcbiAgICBsb2coYEVycm9yIHJlbW92aW5nIGNvbW1lbnRzIGZyb20gJHtmaWxlUGF0aH06ICR7ZXJyb3IubWVzc2FnZX1gLCAnZXJyb3InLCBxdWlldCk7XG4gICAgcmV0dXJuIGZhbHNlO1xuICB9XG59O1xuXG5leHBvcnQgY29uc3QgbGludCA9IGFzeW5jIChjbWQ6IExpbnRPcHRpb25zICYgeydyZW1vdmUtY29tbWVudHMnPzogYm9vbGVhbn0sIGNhbGxiYWNrOiBMaW50Q2FsbGJhY2sgPSBwcm9jZXNzLmV4aXQpOiBQcm9taXNlPG51bWJlcj4gPT4ge1xuICBjb25zdCB7XG4gICAgY2xpTmFtZSA9ICdMZXgnLFxuICAgIGZpeCA9IGZhbHNlLFxuICAgIGRlYnVnID0gZmFsc2UsXG4gICAgcXVpZXQgPSBmYWxzZSxcbiAgICBjb25maWcgPSBudWxsLFxuICAgIHJlbW92ZUNvbW1lbnRzID0gZmFsc2UsXG4gICAgLy8gSGFuZGxlIGtlYmFiLWNhc2UgQ0xJIGZsYWcgY29udmVyc2lvblxuICAgICdyZW1vdmUtY29tbWVudHMnOiByZW1vdmVDb21tZW50c0ZsYWcgPSBmYWxzZVxuICB9ID0gY21kO1xuXG4gIC8vIFVzZSBlaXRoZXIgdGhlIGNhbWVsQ2FzZSBvciBrZWJhYi1jYXNlIHZlcnNpb24gb2YgdGhlIGZsYWdcbiAgY29uc3Qgc2hvdWxkUmVtb3ZlQ29tbWVudHMgPSByZW1vdmVDb21tZW50cyB8fCByZW1vdmVDb21tZW50c0ZsYWc7XG5cbiAgbG9nKGAke2NsaU5hbWV9IGxpbnRpbmcuLi5gLCAnaW5mbycsIHF1aWV0KTtcblxuICBjb25zdCBjd2QgPSBwcm9jZXNzLmN3ZCgpO1xuICBjb25zdCBzcGlubmVyID0gY3JlYXRlU3Bpbm5lcihxdWlldCk7XG5cbiAgYXdhaXQgbG9hZEFJQ29uZmlnKGN3ZCwgcXVpZXQsIGRlYnVnKTtcblxuICBsZXQgdGVtcENvbmZpZ1BhdGg6IHN0cmluZyB8IG51bGwgPSBudWxsO1xuXG4gIHRyeSB7XG4gICAgY29uc3QgdXNlVHlwZXNjcmlwdCA9IGRldGVjdFR5cGVTY3JpcHQoY3dkKTtcbiAgICBsb2coYFR5cGVTY3JpcHQgJHt1c2VUeXBlc2NyaXB0ID8gJ2RldGVjdGVkJyA6ICdub3QgZGV0ZWN0ZWQnfSBmcm9tIHRzY29uZmlnLmpzb25gLCAnaW5mbycsIHF1aWV0KTtcblxuICAgIGlmKHVzZVR5cGVzY3JpcHQpIHtcbiAgICAgIExleENvbmZpZy5jaGVja0xpbnRUeXBlc2NyaXB0Q29uZmlnKCk7XG4gICAgfVxuXG4gICAgZW5zdXJlTW9kdWxlVHlwZShjd2QpO1xuXG4gICAgYXdhaXQgaW5zdGFsbERlcGVuZGVuY2llcyhjd2QsIHVzZVR5cGVzY3JpcHQsIHF1aWV0KTtcblxuICAgIGNvbnN0IHByb2plY3RDb25maWdQYXRoID0gcGF0aFJlc29sdmUoY3dkLCAnZXNsaW50LmNvbmZpZy5qcycpO1xuICAgIGNvbnN0IHByb2plY3RDb25maWdQYXRoVHMgPSBwYXRoUmVzb2x2ZShjd2QsICdlc2xpbnQuY29uZmlnLnRzJyk7XG4gICAgY29uc3QgaGFzRXNsaW50Q29uZmlnID0gZXhpc3RzU3luYyhwcm9qZWN0Q29uZmlnUGF0aCkgfHxcbiAgICAgIGV4aXN0c1N5bmMocHJvamVjdENvbmZpZ1BhdGhUcykgfHxcbiAgICAgIGV4aXN0c1N5bmMocGF0aFJlc29sdmUoY3dkLCAnLmVzbGludHJjLmpzJykpIHx8XG4gICAgICBleGlzdHNTeW5jKHBhdGhSZXNvbHZlKGN3ZCwgJy5lc2xpbnRyYy5qc29uJykpIHx8XG4gICAgICBleGlzdHNTeW5jKHBhdGhSZXNvbHZlKGN3ZCwgJy5lc2xpbnRyYy55bWwnKSkgfHxcbiAgICAgIGV4aXN0c1N5bmMocGF0aFJlc29sdmUoY3dkLCAnLmVzbGludHJjLnlhbWwnKSkgfHxcbiAgICAgIGV4aXN0c1N5bmMocGF0aFJlc29sdmUoY3dkLCAnLmVzbGludHJjJykpO1xuXG4gICAgLy8gQ2hlY2sgZm9yIEVTTGludCBjb25maWcgaW4gbGV4LmNvbmZpZy4qXG4gICAgY29uc3QgaGFzTGV4RXNsaW50Q29uZmlnID0gYXdhaXQgbG9hZEVTTGludENvbmZpZyhjd2QsIHF1aWV0LCBkZWJ1Zyk7XG5cbiAgICBpZihoYXNMZXhFc2xpbnRDb25maWcpIHtcbiAgICAgIGxvZygnVXNpbmcgRVNMaW50IGNvbmZpZ3VyYXRpb24gZnJvbSBsZXguY29uZmlnLiogZmlsZScsICdpbmZvJywgcXVpZXQpO1xuICAgIH1cblxuICAgIGlmKGV4aXN0c1N5bmMocGF0aFJlc29sdmUoY3dkLCAnLmVzbGludHJjLmpzb24nKSkpIHtcbiAgICAgIHVubGlua1N5bmMocGF0aFJlc29sdmUoY3dkLCAnLmVzbGludHJjLmpzb24nKSk7XG4gICAgfVxuXG4gICAgbGV0IGxleENvbmZpZ1BhdGggPSAnJztcbiAgICBsZXQgc2hvdWxkQ3JlYXRlVGVtcENvbmZpZyA9IGZhbHNlO1xuXG4gICAgaWYoIWhhc0VzbGludENvbmZpZyAmJiAhaGFzTGV4RXNsaW50Q29uZmlnKSB7XG4gICAgICBjb25zdCBwb3NzaWJsZVBhdGhzID0gW1xuICAgICAgICBwYXRoUmVzb2x2ZShjdXJyZW50RGlybmFtZSwgJy4uLy4uLy4uLy4uL2VzbGludC5jb25maWcudHMnKSxcbiAgICAgICAgcGF0aFJlc29sdmUoY3VycmVudERpcm5hbWUsICcuLi8uLi8uLi8uLi9lc2xpbnQuY29uZmlnLmpzJyksXG4gICAgICAgIHBhdGhSZXNvbHZlKHByb2Nlc3MuZW52LkxFWF9IT01FIHx8ICcuL25vZGVfbW9kdWxlcy9AbmxhYnMvbGV4JywgJ2VzbGludC5jb25maWcudHMnKSxcbiAgICAgICAgcGF0aFJlc29sdmUocHJvY2Vzcy5lbnYuTEVYX0hPTUUgfHwgJy4vbm9kZV9tb2R1bGVzL0BubGFicy9sZXgnLCAnZXNsaW50LmNvbmZpZy5qcycpXG4gICAgICBdO1xuXG4gICAgICBmb3IoY29uc3QgcGF0aCBvZiBwb3NzaWJsZVBhdGhzKSB7XG4gICAgICAgIGlmKGV4aXN0c1N5bmMocGF0aCkpIHtcbiAgICAgICAgICBsZXhDb25maWdQYXRoID0gcGF0aDtcbiAgICAgICAgICBicmVhaztcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBpZihkZWJ1Zykge1xuICAgICAgICBsb2coYEN1cnJlbnQgZGlyZWN0b3J5OiAke2N1cnJlbnREaXJuYW1lfWAsICdpbmZvJywgcXVpZXQpO1xuICAgICAgICBsb2coYFByb2plY3QgY29uZmlnIHBhdGg6ICR7cHJvamVjdENvbmZpZ1BhdGh9YCwgJ2luZm8nLCBxdWlldCk7XG4gICAgICAgIGxvZyhgUHJvamVjdCBjb25maWcgZXhpc3RzOiAke2hhc0VzbGludENvbmZpZ31gLCAnaW5mbycsIHF1aWV0KTtcbiAgICAgICAgbG9nKGBGb3VuZCBMZXggY29uZmlnOiAke2xleENvbmZpZ1BhdGh9YCwgJ2luZm8nLCBxdWlldCk7XG4gICAgICAgIGxvZyhgTGV4IGNvbmZpZyBleGlzdHM6ICR7ISFsZXhDb25maWdQYXRoICYmIGV4aXN0c1N5bmMobGV4Q29uZmlnUGF0aCl9YCwgJ2luZm8nLCBxdWlldCk7XG4gICAgICB9XG5cbiAgICAgIGlmKGxleENvbmZpZ1BhdGggJiYgZXhpc3RzU3luYyhsZXhDb25maWdQYXRoKSkge1xuICAgICAgICBsb2coJ05vIEVTTGludCBjb25maWd1cmF0aW9uIGZvdW5kIGluIHByb2plY3QuIFVzaW5nIExleFxcJ3MgZGVmYXVsdCBjb25maWd1cmF0aW9uLicsICdpbmZvJywgcXVpZXQpO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgc2hvdWxkQ3JlYXRlVGVtcENvbmZpZyA9IHRydWU7XG4gICAgICB9XG4gICAgfVxuXG4gICAgaWYoY29uZmlnKSB7XG4gICAgICBjb25zdCB1c2VyQ29uZmlnUGF0aCA9IHBhdGhSZXNvbHZlKGN3ZCwgY29uZmlnKTtcbiAgICAgIGlmKGV4aXN0c1N5bmModXNlckNvbmZpZ1BhdGgpKSB7XG4gICAgICAgIGxvZyhgVXNpbmcgc3BlY2lmaWVkIEVTTGludCBjb25maWd1cmF0aW9uOiAke2NvbmZpZ31gLCAnaW5mbycsIHF1aWV0KTtcbiAgICAgICAgc2hvdWxkQ3JlYXRlVGVtcENvbmZpZyA9IGZhbHNlO1xuICAgICAgfSBlbHNlIHtcbiAgICAgICAgbG9nKGBTcGVjaWZpZWQgRVNMaW50IGNvbmZpZ3VyYXRpb24gbm90IGZvdW5kOiAke2NvbmZpZ30uIFVzaW5nIExleCdzIGRlZmF1bHQgY29uZmlndXJhdGlvbi5gLCAnd2FybicsIHF1aWV0KTtcbiAgICAgIH1cbiAgICB9XG5cbiAgICBpZihzaG91bGRDcmVhdGVUZW1wQ29uZmlnKSB7XG4gICAgICBsb2coJ05vIEVTTGludCBjb25maWd1cmF0aW9uIGZvdW5kLiBDcmVhdGluZyBhIHRlbXBvcmFyeSBjb25maWd1cmF0aW9uLi4uJywgJ2luZm8nLCBxdWlldCk7XG4gICAgICBjb25zdCBjb25maWdSZXN1bHQgPSBjcmVhdGVEZWZhdWx0RVNMaW50Q29uZmlnKHVzZVR5cGVzY3JpcHQsIGN3ZCk7XG4gICAgICB0ZW1wQ29uZmlnUGF0aCA9IGNvbmZpZ1Jlc3VsdC5jb25maWdQYXRoO1xuICAgIH1cblxuICAgIGxldCBlc2xpbnRPdXRwdXQgPSAnJztcbiAgICBjb25zdCBjYXB0dXJlT3V0cHV0ID0gKG91dHB1dDogc3RyaW5nKSA9PiB7XG4gICAgICBlc2xpbnRPdXRwdXQgKz0gYCR7b3V0cHV0fVxcbmA7XG4gICAgfTtcblxuICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IHJ1bkVzbGludFdpdGhMZXgoY3dkLCBxdWlldCwgY2xpTmFtZSwgdHJ1ZSwgZGVidWcsIHVzZVR5cGVzY3JpcHQsIGNhcHR1cmVPdXRwdXQpO1xuXG4gICAgLy8gUmVtb3ZlIGNvbW1lbnRzIGlmIGVuYWJsZWRcbiAgICBpZihzaG91bGRSZW1vdmVDb21tZW50cykge1xuICAgICAgc3Bpbm5lci5zdGFydCgnUmVtb3ZpbmcgY29tbWVudHMgZnJvbSBmaWxlcy4uLicpO1xuXG4gICAgICAvLyBHZXQgYWxsIC5qcywgLmpzeCwgLnRzLCAudHN4IGZpbGVzIGluIHRoZSBwcm9qZWN0XG4gICAgICBjb25zdCBnbG9iID0gYXdhaXQgaW1wb3J0KCdnbG9iJyk7XG4gICAgICBjb25zdCBmaWxlcyA9IGdsb2Iuc3luYygne3NyYyxsaWJ9LyoqLyoue2pzLGpzeCx0cyx0c3h9Jywge1xuICAgICAgICBjd2QsXG4gICAgICAgIGlnbm9yZTogWycqKi9ub2RlX21vZHVsZXMvKionLCAnKiovZGlzdC8qKicsICcqKi9idWlsZC8qKiddXG4gICAgICB9KTtcblxuICAgICAgbGV0IHByb2Nlc3NlZENvdW50ID0gMDtcbiAgICAgIGZvcihjb25zdCBmaWxlIG9mIGZpbGVzKSB7XG4gICAgICAgIGNvbnN0IGZpbGVQYXRoID0gcGF0aFJlc29sdmUoY3dkLCBmaWxlKTtcbiAgICAgICAgaWYocmVtb3ZlRmlsZUNvbW1lbnRzKGZpbGVQYXRoLCBxdWlldCkpIHtcbiAgICAgICAgICBwcm9jZXNzZWRDb3VudCsrO1xuICAgICAgICB9XG4gICAgICB9XG5cbiAgICAgIHNwaW5uZXIuc3VjY2VlZChgUmVtb3ZlZCBjb21tZW50cyBmcm9tICR7cHJvY2Vzc2VkQ291bnR9IGZpbGVzYCk7XG4gICAgfVxuXG4gICAgaWYocmVzdWx0ICE9PSAwICYmIGZpeCkge1xuICAgICAgY29uc3QgYWlDb25maWd1cmVkID0gTGV4Q29uZmlnLmNvbmZpZy5haT8ucHJvdmlkZXIgJiYgTGV4Q29uZmlnLmNvbmZpZy5haS5wcm92aWRlciAhPT0gJ25vbmUnO1xuXG4gICAgICBpZihhaUNvbmZpZ3VyZWQpIHtcbiAgICAgICAgbG9nKCdBcHBseWluZyBBSSBmaXhlcyB0byByZW1haW5pbmcgaXNzdWVzLi4uJywgJ2luZm8nLCBxdWlldCk7XG4gICAgICAgIGF3YWl0IGFwcGx5QUlGaXgoY3dkLCBlc2xpbnRPdXRwdXQsIHF1aWV0KTtcblxuICAgICAgICBjb25zdCBhZnRlckZpeFJlc3VsdCA9IGF3YWl0IHJ1bkVzbGludFdpdGhMZXgoY3dkLCBxdWlldCwgY2xpTmFtZSwgZmFsc2UsIGRlYnVnLCB1c2VUeXBlc2NyaXB0KTtcblxuICAgICAgICBjYWxsYmFjayhhZnRlckZpeFJlc3VsdCk7XG4gICAgICAgIHJldHVybiBhZnRlckZpeFJlc3VsdDtcbiAgICAgIH1cbiAgICAgIGxvZygnRVNMaW50IGNvdWxkIG5vdCBmaXggYWxsIGlzc3VlcyBhdXRvbWF0aWNhbGx5LicsICd3YXJuJywgcXVpZXQpO1xuICAgICAgbG9nKCdUbyBlbmFibGUgQUktcG93ZXJlZCBmaXhlcywgYWRkIEFJIGNvbmZpZ3VyYXRpb24gdG8geW91ciBsZXguY29uZmlnIGZpbGU6JywgJ2luZm8nLCBxdWlldCk7XG4gICAgICBsb2coYFxuLy8gSW4gbGV4LmNvbmZpZy5qcyAob3IgbGV4LmNvbmZpZy5tanMsIGxleC5jb25maWcuY2pzLCBldGMuKVxuZXhwb3J0IGRlZmF1bHQge1xuICAvLyBZb3VyIGV4aXN0aW5nIGNvbmZpZ1xuICBhaToge1xuICAgIHByb3ZpZGVyOiAnY3Vyc29yJyAvLyBvciAnb3BlbmFpJywgJ2FudGhyb3BpYycsIGV0Yy5cbiAgICAvLyBBZGRpdGlvbmFsIHByb3ZpZGVyLXNwZWNpZmljIHNldHRpbmdzXG4gIH1cbn07YCwgJ2luZm8nLCBxdWlldCk7XG4gICAgfVxuXG4gICAgY2FsbGJhY2socmVzdWx0KTtcbiAgICByZXR1cm4gcmVzdWx0O1xuICB9IGNhdGNoIChlcnJvcikge1xuICAgIGxvZyhgXFxuJHtjbGlOYW1lfSBFcnJvcjogJHtlcnJvci5tZXNzYWdlfWAsICdlcnJvcicsIHF1aWV0KTtcbiAgICBpZihzcGlubmVyKSB7XG4gICAgICBzcGlubmVyLmZhaWwoJ0xpbnRpbmcgZmFpbGVkIScpO1xuICAgIH1cbiAgICBjYWxsYmFjaygxKTtcbiAgICByZXR1cm4gMTtcbiAgfSBmaW5hbGx5IHtcbiAgICAvLyBDbGVhbiB1cCB0ZW1wb3JhcnkgY29uZmlnIGZpbGVzXG4gICAgY29uc3QgdGVtcEZpbGVQYXRocyA9IFtcbiAgICAgIHRlbXBDb25maWdQYXRoLFxuICAgICAgcGF0aFJlc29sdmUoY3dkLCAnLmxleC10ZW1wLWVzbGludC5janMnKSxcbiAgICAgIHBhdGhSZXNvbHZlKGN3ZCwgJy5sZXgtdGVtcC1kZWZhdWx0LWVzbGludC5janMnKVxuICAgIF07XG5cbiAgICBmb3IoY29uc3QgZmlsZVBhdGggb2YgdGVtcEZpbGVQYXRocykge1xuICAgICAgaWYoZmlsZVBhdGggJiYgZXhpc3RzU3luYyhmaWxlUGF0aCkpIHtcbiAgICAgICAgdHJ5IHtcbiAgICAgICAgICB1bmxpbmtTeW5jKGZpbGVQYXRoKTtcbiAgICAgICAgICBpZihkZWJ1Zykge1xuICAgICAgICAgICAgbG9nKGBDbGVhbmVkIHVwIHRlbXBvcmFyeSBFU0xpbnQgY29uZmlnIGF0ICR7ZmlsZVBhdGh9YCwgJ2luZm8nLCBxdWlldCk7XG4gICAgICAgICAgfVxuICAgICAgICB9IGNhdGNoIChfZXJyb3IpIHtcbiAgICAgICAgICAvLyBJZ25vcmUgZXJyb3JzXG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG4gIH1cbn07Il0sCiAgIm1hcHBpbmdzIjogIkFBSUEsU0FBUSxhQUFZO0FBQ3BCLFNBQVEsWUFBWSxjQUFjLFlBQVkscUJBQW9CO0FBQ2xFLFNBQVEsU0FBUyxXQUFXLGFBQWEsZUFBYztBQUV2RCxTQUFRLGlCQUFnQjtBQUN4QixTQUFRLHFCQUFvQjtBQUM1QixTQUFRLHlCQUF3QjtBQUNoQyxTQUFRLFdBQVU7QUFFbEIsSUFBSTtBQUNKLElBQUk7QUFFSixJQUFJO0FBQ0Ysb0JBQWtCLEtBQUssK0NBQStDO0FBQ3RFLG1CQUFpQixRQUFRLGVBQWU7QUFDMUMsUUFBUTtBQUNOLG9CQUFrQixRQUFRLElBQUk7QUFDOUIsbUJBQWlCLFFBQVEsSUFBSTtBQUMvQjtBQWdEQSxNQUFNLDRCQUE0QixDQUFDLGVBQXdCLFFBQThCO0FBRXZGLFFBQU0sYUFBYSxZQUFZLEtBQUssOEJBQThCO0FBQ2xFLFFBQU0saUJBQWlCO0FBR3ZCLFFBQU0sZ0JBQWdCO0FBQUEsSUFDcEIsWUFBWSxnQkFBZ0Isa0JBQWtCO0FBQUEsSUFDOUMsWUFBWSxnQkFBZ0Isa0JBQWtCO0FBQUEsSUFDOUMsWUFBWSxRQUFRLElBQUksWUFBWSw2QkFBNkIsa0JBQWtCO0FBQUEsSUFDbkYsWUFBWSxRQUFRLElBQUksWUFBWSw2QkFBNkIsa0JBQWtCO0FBQUEsRUFDckY7QUFFQSxNQUFJLGNBQWM7QUFFbEIsYUFBVSxRQUFRLGVBQWU7QUFDL0IsUUFBRyxXQUFXLElBQUksR0FBRztBQUNuQixvQkFBYztBQUNkO0FBQUEsSUFDRjtBQUFBLEVBQ0Y7QUFHQSxRQUFNLGdCQUFnQjtBQUFBO0FBQUE7QUFBQTtBQUt0QixnQkFBYyxZQUFZLGVBQWUsTUFBTTtBQUUvQyxTQUFPO0FBQUEsSUFDTDtBQUFBLElBQ0E7QUFBQSxFQUNGO0FBQ0Y7QUFFQSxNQUFNLDBCQUEwQixDQUFDLGtCQUFtQztBQUVsRSxRQUFNLFNBQVM7QUFBQTtBQUFBO0FBQUE7QUFJZixTQUFPO0FBQ1Q7QUFLQSxNQUFNLG1CQUFtQixDQUFDLFFBQXlCLFdBQVcsWUFBWSxLQUFLLGVBQWUsQ0FBQztBQUsvRixNQUFNLG1CQUFtQixDQUFDLFFBQXNCO0FBQzlDLFFBQU0sa0JBQWtCLFlBQVksS0FBSyxjQUFjO0FBRXZELE1BQUcsV0FBVyxlQUFlLEdBQUc7QUFDOUIsUUFBSTtBQUNGLFlBQU0scUJBQXFCLGFBQWEsaUJBQWlCLE1BQU07QUFDL0QsWUFBTSxjQUFjLEtBQUssTUFBTSxrQkFBa0I7QUFHakQsVUFBRyxZQUFZLFNBQVMsVUFBVTtBQUNoQyxZQUFJLGlHQUFpRyxRQUFRLEtBQUs7QUFBQSxNQUNwSDtBQUFBLElBQ0YsU0FBUyxRQUFRO0FBQUEsSUFFakI7QUFBQSxFQUNGO0FBQ0Y7QUFFQSxNQUFNLHNCQUFzQixPQUFPLEtBQWEsZUFBd0IsVUFBa0M7QUFDeEcsTUFBRyxlQUFlO0FBQ2hCLFFBQUksdUNBQXVDLFFBQVEsS0FBSztBQUFBLEVBQzFELE9BQU87QUFDTCxRQUFJLDRCQUE0QixRQUFRLEtBQUs7QUFBQSxFQUMvQztBQUNGO0FBRUEsTUFBTSxtQkFBbUIsT0FDdkIsS0FDQSxPQUNBLFNBQ0EsS0FDQSxPQUNBLGVBQ0Esa0JBQ29CO0FBQ3BCLFFBQU0sVUFBVSxjQUFjLEtBQUs7QUFFbkMsTUFBSTtBQUNGLFVBQU0sb0JBQW9CLFlBQVksS0FBSyxrQkFBa0I7QUFDN0QsVUFBTSxzQkFBc0IsWUFBWSxLQUFLLGtCQUFrQjtBQUMvRCxVQUFNLG1CQUFtQixXQUFXLGlCQUFpQixLQUFLLFdBQVcsbUJBQW1CO0FBQ3hGLFVBQU0scUJBQXFCLFVBQVUsT0FBTyxVQUFVLE9BQU8sS0FBSyxVQUFVLE9BQU8sTUFBTSxFQUFFLFNBQVM7QUFFcEcsVUFBTSxnQkFBZ0I7QUFBQSxNQUNwQixZQUFZLGdCQUFnQiw4QkFBOEI7QUFBQSxNQUMxRCxZQUFZLGdCQUFnQiw4QkFBOEI7QUFBQSxNQUMxRCxZQUFZLFFBQVEsSUFBSSxZQUFZLDBDQUEwQyxrQkFBa0I7QUFBQSxNQUNoRyxZQUFZLFFBQVEsSUFBSSxZQUFZLDBDQUEwQyxrQkFBa0I7QUFBQSxJQUNsRztBQUVBLFFBQUksZ0JBQWdCO0FBQ3BCLGVBQVUsUUFBUSxlQUFlO0FBQy9CLFVBQUcsV0FBVyxJQUFJLEdBQUc7QUFDbkIsd0JBQWdCO0FBQ2hCO0FBQUEsTUFDRjtBQUFBLElBQ0Y7QUFFQSxRQUFJLGFBQWE7QUFDakIsUUFBSSxpQkFBaUI7QUFPckIsUUFBRyxrQkFBa0I7QUFDbkIsbUJBQWEsV0FBVyxtQkFBbUIsSUFBSSxzQkFBc0I7QUFDckUsVUFBRyxPQUFPO0FBQ1IsWUFBSSxxQ0FBcUMsVUFBVSxJQUFJLFFBQVEsS0FBSztBQUFBLE1BQ3RFO0FBQUEsSUFDRixXQUFVLG9CQUFvQjtBQUc1Qix1QkFBaUIsWUFBWSxLQUFLLHNCQUFzQjtBQUd4RCxZQUFNLGdCQUFnQjtBQUFBO0FBQUEscUJBRVAsS0FBSyxVQUFVLFVBQVUsT0FBTyxRQUFRLE1BQU0sQ0FBQyxDQUFDO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQU8vRCxvQkFBYyxnQkFBZ0IsZUFBZSxNQUFNO0FBQ25ELG1CQUFhO0FBRWIsVUFBRyxPQUFPO0FBQ1IsWUFBSSw2REFBNkQsY0FBYyxJQUFJLFFBQVEsS0FBSztBQUFBLE1BQ2xHO0FBQUEsSUFDRixXQUFVLGlCQUFpQixXQUFXLGFBQWEsR0FBRztBQUNwRCxtQkFBYTtBQUNiLFVBQUcsT0FBTztBQUNSLFlBQUksaUNBQWlDLFVBQVUsSUFBSSxRQUFRLEtBQUs7QUFBQSxNQUNsRTtBQUFBLElBQ0YsT0FBTztBQUVMLHVCQUFpQixZQUFZLEtBQUssOEJBQThCO0FBR2hFLFlBQU0sZ0JBQWdCO0FBQUE7QUFBQTtBQUFBO0FBS3RCLG9CQUFjLGdCQUFnQixlQUFlLE1BQU07QUFDbkQsbUJBQWE7QUFFYixVQUFHLE9BQU87QUFDUixZQUFJLCtDQUErQyxjQUFjLElBQUksUUFBUSxLQUFLO0FBQUEsTUFDcEYsT0FBTztBQUNMLFlBQUksbUVBQW1FLFFBQVEsS0FBSztBQUFBLE1BQ3RGO0FBQUEsSUFDRjtBQUVBLFVBQU0sZUFBZSxrQkFBa0IsVUFBVSxRQUFRO0FBRXpELFFBQUcsQ0FBQyxjQUFjO0FBQ2hCLFVBQUk7QUFBQSxFQUFLLE9BQU8seURBQXlELFNBQVMsS0FBSztBQUN2RixVQUFJLG9EQUFvRCxRQUFRLEtBQUs7QUFDckUsYUFBTztBQUFBLElBQ1Q7QUFHQSxVQUFNLGlCQUFpQjtBQUFBLE1BQ3JCLEdBQUksTUFBTSxDQUFDLE9BQU8sSUFBSSxDQUFDO0FBQUEsTUFDdkIsR0FBSSxRQUFRLENBQUMsU0FBUyxJQUFJLENBQUM7QUFBQSxNQUMzQjtBQUFBLE1BQ0E7QUFBQSxJQUNGO0FBR0EsVUFBTSxhQUFhLGFBQWEsQ0FBQyxZQUFZLFVBQVUsSUFBSSxDQUFDO0FBRTVELFVBQU0sV0FBVyxNQUFNLE1BQU0sY0FBYztBQUFBLE1BQ3pDO0FBQUEsTUFDQSxHQUFHO0FBQUEsTUFDSCxHQUFHO0FBQUEsSUFDTCxHQUFHO0FBQUEsTUFDRCxRQUFRO0FBQUEsTUFDUixPQUFPO0FBQUEsTUFDUDtBQUFBLE1BQ0EsT0FBTztBQUFBLElBQ1QsQ0FBQztBQUVELFFBQUcsU0FBUyxRQUFRO0FBQ2xCLGNBQVEsSUFBSSxTQUFTLE1BQU07QUFDM0IsVUFBRyxlQUFlO0FBQ2hCLHNCQUFjLFNBQVMsTUFBTTtBQUFBLE1BQy9CO0FBQUEsSUFDRjtBQUVBLFFBQUcsU0FBUyxRQUFRO0FBQ2xCLGNBQVEsTUFBTSxTQUFTLE1BQU07QUFDN0IsVUFBRyxlQUFlO0FBQ2hCLHNCQUFjLFNBQVMsTUFBTTtBQUFBLE1BQy9CO0FBQUEsSUFDRjtBQUVBLFFBQUksV0FBZ0IsRUFBQyxVQUFVLEdBQUcsUUFBUSxJQUFJLFFBQVEsR0FBRTtBQUN4RCxRQUFHLGVBQWU7QUFDaEIsaUJBQVcsTUFBTSxNQUFNLGNBQWM7QUFBQSxRQUNuQztBQUFBLFFBQ0EsR0FBRztBQUFBLFFBQ0gsR0FBRztBQUFBLE1BQ0wsR0FBRztBQUFBLFFBQ0QsUUFBUTtBQUFBLFFBQ1IsT0FBTztBQUFBLFFBQ1A7QUFBQSxRQUNBLE9BQU87QUFBQSxNQUNULENBQUM7QUFFRCxVQUFHLFNBQVMsUUFBUTtBQUNsQixnQkFBUSxJQUFJLFNBQVMsTUFBTTtBQUFBLE1BQzdCO0FBRUEsVUFBRyxTQUFTLFFBQVE7QUFDbEIsZ0JBQVEsTUFBTSxTQUFTLE1BQU07QUFBQSxNQUMvQjtBQUFBLElBQ0Y7QUFHQSxRQUFHLGtCQUFrQixXQUFXLGNBQWMsR0FBRztBQUMvQyxVQUFJO0FBQ0YsbUJBQVcsY0FBYztBQUN6QixZQUFHLE9BQU87QUFDUixjQUFJLHNDQUFzQyxjQUFjLElBQUksUUFBUSxLQUFLO0FBQUEsUUFDM0U7QUFBQSxNQUNGLFNBQVMsT0FBTztBQUVkLFlBQUcsT0FBTztBQUNSLGNBQUksNkNBQTZDLE1BQU0sT0FBTyxJQUFJLFFBQVEsS0FBSztBQUFBLFFBQ2pGO0FBQUEsTUFDRjtBQUFBLElBQ0Y7QUFFQSxVQUFNLGlCQUFpQixTQUFTLFFBQVEsU0FBUyxtQkFBbUIsS0FBSyxTQUFTLFFBQVEsU0FBUywyQkFBMkI7QUFDOUgsUUFBRyxnQkFBZ0I7QUFDakIsY0FBUSxLQUFLLG1CQUFtQjtBQUNoQyxVQUFJO0FBQUEsRUFBSyxPQUFPLDJGQUEyRixTQUFTLEtBQUs7QUFDekgsYUFBTztBQUFBLElBQ1Q7QUFFQSxRQUFHLFNBQVMsYUFBYSxLQUFLLFNBQVMsYUFBYSxHQUFHO0FBQ3JELGNBQVEsUUFBUSxvQkFBb0I7QUFDcEMsYUFBTztBQUFBLElBQ1Q7QUFFQSxVQUFNLGdCQUNILFNBQVMsUUFBUSxTQUFTLDJCQUEyQixLQUFLLFNBQVMsUUFBUSxTQUFTLDJCQUEyQixPQUMvRyxDQUFDLGlCQUFpQixTQUFTLFFBQVEsU0FBUywyQkFBMkIsS0FBSyxTQUFTLFFBQVEsU0FBUywyQkFBMkI7QUFFcEksUUFBRyxjQUFjO0FBQ2YsY0FBUSxRQUFRLHdCQUF3QjtBQUN4QyxhQUFPO0FBQUEsSUFDVDtBQUNBLFlBQVEsS0FBSyxpQkFBaUI7QUFDOUIsUUFBSTtBQUFBLEVBQUssT0FBTyw2Q0FBNkMsU0FBUyxLQUFLO0FBQzNFLFdBQU87QUFBQSxFQUNULFNBQVMsT0FBTztBQUNkLFlBQVEsS0FBSyxpQkFBaUI7QUFDOUIsUUFBSTtBQUFBLEVBQUssT0FBTyxXQUFXLE1BQU0sT0FBTyxJQUFJLFNBQVMsS0FBSztBQUMxRCxXQUFPO0FBQUEsRUFDVDtBQUNGO0FBRUEsTUFBTSxhQUFhLE9BQ2pCLEtBQ0EsUUFDQSxVQUNrQjtBQUNsQixRQUFNLFVBQVUsY0FBYyxLQUFLO0FBQ25DLFVBQVEsTUFBTSwwQ0FBMEM7QUFFeEQsTUFBSTtBQUNGLFVBQU0sZUFBZSxvQkFBSSxJQUFzQjtBQUMvQyxVQUFNLFFBQVEsT0FBTyxNQUFNLElBQUk7QUFDL0IsUUFBSSxjQUFjO0FBRWxCLGVBQVUsUUFBUSxPQUFPO0FBQ3ZCLFVBQUcsS0FBSyxNQUFNLHFDQUFxQyxHQUFHO0FBQ3BELHNCQUFjLEtBQUssS0FBSztBQUN4QixZQUFHLENBQUMsYUFBYSxJQUFJLFdBQVcsR0FBRztBQUNqQyx1QkFBYSxJQUFJLGFBQWEsQ0FBQyxDQUFDO0FBQUEsUUFDbEM7QUFBQSxNQUNGLFdBQVUsZUFBZSxLQUFLLEtBQUssS0FBSyxLQUFLLE1BQU0saUNBQWlDLEdBQUc7QUFDckYsY0FBTSxhQUFhLGFBQWEsSUFBSSxXQUFXO0FBQy9DLFlBQUcsWUFBWTtBQUNiLHFCQUFXLEtBQUssS0FBSyxLQUFLLENBQUM7QUFBQSxRQUM3QjtBQUFBLE1BQ0Y7QUFBQSxJQUNGO0FBRUEsUUFBRyxhQUFhLFNBQVMsR0FBRztBQUMxQixVQUFJLDRDQUE0QyxRQUFRLEtBQUs7QUFFN0QsWUFBTSxXQUFXLE9BQU8sTUFBTSxNQUFNO0FBRXBDLGlCQUFVLFdBQVcsVUFBVTtBQUM3QixZQUFHLFFBQVEsS0FBSyxNQUFNLElBQUk7QUFDeEI7QUFBQSxRQUNGO0FBRUEsY0FBTUEsU0FBUSxRQUFRLE1BQU0sSUFBSTtBQUNoQyxjQUFNLFdBQVdBLE9BQU0sQ0FBQyxFQUFFLEtBQUs7QUFFL0IsWUFBRyxTQUFTLE1BQU0sb0JBQW9CLEdBQUc7QUFDdkMsdUJBQWEsSUFBSSxVQUFVLENBQUMsQ0FBQztBQUU3QixtQkFBUSxJQUFJLEdBQUcsSUFBSUEsT0FBTSxRQUFRLEtBQUs7QUFDcEMsZ0JBQUdBLE9BQU0sQ0FBQyxFQUFFLEtBQUssTUFBTSxJQUFJO0FBQ3pCLDJCQUFhLElBQUksUUFBUSxHQUFHLEtBQUtBLE9BQU0sQ0FBQyxFQUFFLEtBQUssQ0FBQztBQUFBLFlBQ2xEO0FBQUEsVUFDRjtBQUFBLFFBQ0Y7QUFBQSxNQUNGO0FBQUEsSUFDRjtBQUVBLFFBQUcsYUFBYSxTQUFTLEdBQUc7QUFDMUIsVUFBSSxxQ0FBcUMsUUFBUSxLQUFLO0FBRXRELFlBQU0sZ0JBQWdCO0FBQ3RCLFlBQU0sWUFBWSxPQUFPLE1BQU0sYUFBYSxLQUFLLENBQUM7QUFFbEQsaUJBQVUsWUFBWSxXQUFXO0FBQy9CLFlBQUcsQ0FBQyxhQUFhLElBQUksUUFBUSxLQUFLLFdBQVcsUUFBUSxHQUFHO0FBQ3RELHVCQUFhLElBQUksVUFBVSxDQUFDLENBQUM7QUFBQSxRQUMvQjtBQUFBLE1BQ0Y7QUFFQSxZQUFNLGFBQWE7QUFBQSxRQUNqQixZQUFZLEtBQUsseUJBQXlCO0FBQUEsUUFDMUMsWUFBWSxLQUFLLHdCQUF3QjtBQUFBLFFBQ3pDLFlBQVksS0FBSyxrQkFBa0I7QUFBQSxRQUNuQyxZQUFZLEtBQUssd0JBQXdCO0FBQUEsUUFDekMsWUFBWSxLQUFLLDhCQUE4QjtBQUFBLE1BQ2pEO0FBRUEsaUJBQVUsUUFBUSxZQUFZO0FBQzVCLFlBQUcsV0FBVyxJQUFJLEtBQUssQ0FBQyxhQUFhLElBQUksSUFBSSxHQUFHO0FBQzlDLHVCQUFhLElBQUksTUFBTSxDQUFDLENBQUM7QUFBQSxRQUMzQjtBQUFBLE1BQ0Y7QUFBQSxJQUNGO0FBRUEsZUFBVSxZQUFZLGFBQWEsS0FBSyxHQUFHO0FBQ3pDLFVBQUcsQ0FBQyxXQUFXLFFBQVEsR0FBRztBQUN4QixZQUFJLG1CQUFtQixRQUFRLElBQUksUUFBUSxLQUFLO0FBQ2hEO0FBQUEsTUFDRjtBQUVBLFVBQUksb0JBQW9CLFFBQVEsSUFBSSxRQUFRLEtBQUs7QUFFakQsWUFBTSxjQUFjLFVBQVUsT0FBTyxJQUFJLGFBQWEsWUFBWSxRQUFRLElBQUksZUFBZTtBQUU3RixVQUFHLGFBQWE7QUFDZCxZQUFJO0FBQ0YsZ0JBQU0sU0FBUztBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQTBFZixjQUFJO0FBQ0Ysa0JBQU0sYUFBYSxZQUFZLEtBQUsseUJBQXlCO0FBQzdELDBCQUFjLFlBQVksUUFBUSxNQUFNO0FBR3hDLGtCQUFNLE1BQU0sVUFBVSxDQUFDLFFBQVEsVUFBVSxVQUFVLGlCQUFpQixVQUFVLEdBQUc7QUFBQSxjQUMvRSxRQUFRO0FBQUEsY0FDUixPQUFPO0FBQUEsY0FDUDtBQUFBLFlBQ0YsQ0FBQztBQUVELGdCQUFJO0FBQ0YseUJBQVcsVUFBVTtBQUFBLFlBQ3ZCLFNBQVMsUUFBUTtBQUFBLFlBQ2pCO0FBRUEsZ0JBQUksOEJBQThCLFFBQVEsSUFBSSxRQUFRLEtBQUs7QUFBQSxVQUM3RCxTQUFTLE9BQU87QUFDZCxrQkFBTSxjQUFjLE1BQU0saUJBQWlCLFVBQVUsS0FBSztBQUMxRCxnQkFBRyxhQUFhO0FBQ2Qsa0JBQUksMkJBQTJCLFFBQVEsSUFBSSxRQUFRLEtBQUs7QUFBQSxZQUMxRDtBQUFBLFVBQ0Y7QUFBQSxRQUNGLFNBQVMsT0FBTztBQUNkLGNBQUksMEJBQTBCLE1BQU0sT0FBTyxJQUFJLFNBQVMsS0FBSztBQUM3RCxnQkFBTSxpQkFBaUIsVUFBVSxLQUFLO0FBQUEsUUFDeEM7QUFBQSxNQUNGLE9BQU87QUFDTCxjQUFNLGNBQWMsTUFBTSxpQkFBaUIsVUFBVSxLQUFLO0FBQzFELFlBQUcsYUFBYTtBQUNkLGNBQUksMkJBQTJCLFFBQVEsSUFBSSxRQUFRLEtBQUs7QUFBQSxRQUMxRDtBQUVBLGNBQU0sYUFBYSxhQUFhLElBQUksUUFBUSxLQUFLLENBQUM7QUFDbEQsWUFBRyxXQUFXLFNBQVMsR0FBRztBQUN4QixjQUFJO0FBQ0Ysa0JBQU0sRUFBQyxjQUFhLElBQUksTUFBTSxPQUFPLDBCQUEwQjtBQUUvRCxrQkFBTSxjQUFjLGFBQWEsVUFBVSxNQUFNO0FBRWpELGtCQUFNLFNBQVM7QUFBQSxFQUN6QixXQUFXLEtBQUssSUFBSSxDQUFDO0FBQUE7QUFBQTtBQUFBO0FBQUEsRUFJckIsV0FBVztBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBNkZELGtCQUFNLGVBQWUsTUFBTSxjQUFjLFFBQVEsS0FBSztBQUV0RCxnQkFBRyxnQkFBZ0IsaUJBQWlCLGFBQWE7QUFDL0MsNEJBQWMsVUFBVSxjQUFjLE1BQU07QUFDNUMsa0JBQUksdUJBQXVCLFFBQVEsSUFBSSxRQUFRLEtBQUs7QUFBQSxZQUN0RDtBQUFBLFVBQ0YsU0FBUyxPQUFPO0FBQ2QsZ0JBQUksOEJBQThCLFFBQVEsS0FBSyxNQUFNLE9BQU8sSUFBSSxTQUFTLEtBQUs7QUFBQSxVQUNoRjtBQUFBLFFBQ0Y7QUFBQSxNQUNGO0FBQUEsSUFDRjtBQUVBLFlBQVEsUUFBUSxnQ0FBZ0M7QUFBQSxFQUNsRCxTQUFTLE9BQU87QUFDZCxZQUFRLEtBQUssMEJBQTBCO0FBQ3ZDLFFBQUksVUFBVSxNQUFNLE9BQU8sSUFBSSxTQUFTLEtBQUs7QUFDN0MsUUFBRyxDQUFDLE9BQU87QUFDVCxjQUFRLE1BQU0sS0FBSztBQUFBLElBQ3JCO0FBQUEsRUFDRjtBQUNGO0FBRUEsTUFBTSxtQkFBbUIsT0FBTyxVQUFrQixVQUFxQztBQUNyRixNQUFJLGNBQWM7QUFFbEIsTUFBSTtBQUNGLFVBQU0sY0FBYyxhQUFhLFVBQVUsTUFBTTtBQUNqRCxRQUFJLGFBQWE7QUFFakIsUUFBRyxTQUFTLFNBQVMsY0FBYyxHQUFHO0FBQ3BDLFVBQUksaUNBQWlDLFFBQVEsS0FBSztBQUVsRCxtQkFBYSxXQUFXO0FBQUEsUUFDdEI7QUFBQSxRQUNBO0FBQUEsTUFDRjtBQUVBLG1CQUFhLFdBQVc7QUFBQSxRQUN0QjtBQUFBLFFBQ0E7QUFBQSxNQUNGO0FBRUEsbUJBQWEsV0FBVztBQUFBLFFBQ3RCO0FBQUEsUUFDQTtBQUFBLE1BQ0Y7QUFDQSxtQkFBYSxXQUFXO0FBQUEsUUFDdEI7QUFBQSxRQUNBO0FBQUEsTUFDRjtBQUVBLG1CQUFhLFdBQVc7QUFBQSxRQUN0QjtBQUFBLFFBQ0E7QUFBQSxNQUNGO0FBRUEsbUJBQWEsV0FBVyxRQUFRLG1CQUFtQixNQUFNO0FBRXpELFVBQUcsQ0FBQyxXQUFXLFNBQVMsY0FBYyxLQUFLLFdBQVcsU0FBUyxNQUFNLEdBQUc7QUFDdEUscUJBQWEsV0FBVztBQUFBLFVBQ3RCO0FBQUEsVUFDQTtBQUFBLFFBQ0Y7QUFBQSxNQUNGO0FBQUEsSUFDRjtBQUVBLFFBQUcsU0FBUyxTQUFTLGNBQWMsR0FBRztBQUNwQyxVQUFJLG1EQUFtRCxRQUFRLEtBQUs7QUFFcEUsbUJBQWEsV0FBVztBQUFBLFFBQ3RCO0FBQUEsUUFDQTtBQUFBLE1BQ0Y7QUFFQSxtQkFBYSxXQUFXLFFBQVEsWUFBWSxRQUFRO0FBQUEsSUFDdEQ7QUFFQSxRQUFHLFNBQVMsU0FBUyxjQUFjLEdBQUc7QUFDcEMsVUFBSSxpQ0FBaUMsUUFBUSxLQUFLO0FBRWxELG1CQUFhLFdBQVcsUUFBUSxjQUFjLFNBQVM7QUFFdkQsbUJBQWEsV0FBVyxRQUFRLFNBQVMsR0FBRztBQUM1QyxtQkFBYSxXQUFXLFFBQVEsU0FBUyxHQUFHO0FBQzVDLG1CQUFhLFdBQVcsUUFBUSxTQUFTLEdBQUc7QUFDNUMsbUJBQWEsV0FBVyxRQUFRLFFBQVEsR0FBRztBQUFBLElBQzdDO0FBRUEsUUFBRyxTQUFTLFNBQVMsUUFBUSxHQUFHO0FBQzlCLFVBQUksMkJBQTJCLFFBQVEsS0FBSztBQUU1QyxtQkFBYSxXQUFXLFFBQVEsbUJBQW1CLE1BQU07QUFFekQsVUFBRyxDQUFDLFdBQVcsU0FBUyxjQUFjLEtBQUssV0FBVyxTQUFTLE1BQU0sR0FBRztBQUN0RSxxQkFBYSxXQUFXO0FBQUEsVUFDdEI7QUFBQSxVQUNBO0FBQUEsUUFDRjtBQUFBLE1BQ0Y7QUFFQSxtQkFBYSxXQUFXLFFBQVEsU0FBUyxHQUFHO0FBQUEsSUFDOUM7QUFFQSxRQUFHLFNBQVMsU0FBUyxZQUFZLEdBQUc7QUFDbEMsVUFBSSwrQkFBK0IsUUFBUSxLQUFLO0FBRWhELG1CQUFhLFdBQVc7QUFBQSxRQUN0QjtBQUFBLFFBQ0E7QUFBQSxNQUNGO0FBRUEsbUJBQWEsV0FBVztBQUFBLFFBQ3RCO0FBQUEsUUFDQTtBQUFBLE1BQ0Y7QUFDQSxtQkFBYSxXQUFXO0FBQUEsUUFDdEI7QUFBQSxRQUNBO0FBQUEsTUFDRjtBQUVBLG1CQUFhLFdBQVc7QUFBQSxRQUN0QjtBQUFBLFFBQ0E7QUFBQSxNQUNGO0FBRUEsbUJBQWEsV0FBVztBQUFBLFFBQ3RCO0FBQUEsUUFDQTtBQUFBLE1BQ0Y7QUFDQSxtQkFBYSxXQUFXO0FBQUEsUUFDdEI7QUFBQSxRQUNBO0FBQUEsTUFDRjtBQUNBLG1CQUFhLFdBQVc7QUFBQSxRQUN0QjtBQUFBLFFBQ0E7QUFBQSxNQUNGO0FBQ0EsbUJBQWEsV0FBVztBQUFBLFFBQ3RCO0FBQUEsUUFDQTtBQUFBLE1BQ0Y7QUFDQSxtQkFBYSxXQUFXO0FBQUEsUUFDdEI7QUFBQSxRQUNBO0FBQUEsTUFDRjtBQUVBLG1CQUFhLFdBQVc7QUFBQSxRQUN0QjtBQUFBLFFBQ0E7QUFBQSxNQUNGO0FBRUEsbUJBQWEsV0FBVztBQUFBLFFBQ3RCO0FBQUEsUUFDQTtBQUFBLE1BQ0Y7QUFFQSxtQkFBYSxXQUFXO0FBQUEsUUFDdEI7QUFBQSxRQUNBO0FBQUEsTUFDRjtBQUNBLG1CQUFhLFdBQVc7QUFBQSxRQUN0QjtBQUFBLFFBQ0E7QUFBQSxNQUNGO0FBQ0EsbUJBQWEsV0FBVztBQUFBLFFBQ3RCO0FBQUEsUUFDQTtBQUFBLE1BQ0Y7QUFFQSxtQkFBYSxXQUFXO0FBQUEsUUFDdEI7QUFBQSxRQUNBO0FBQUEsTUFDRjtBQUVBLG1CQUFhLFdBQVc7QUFBQSxRQUN0QjtBQUFBLFFBQ0E7QUFBQSxNQUNGO0FBQUEsSUFDRjtBQUVBLFFBQUcsZUFBZSxhQUFhO0FBQzdCLG9CQUFjLFVBQVUsWUFBWSxNQUFNO0FBQzFDLFVBQUksbUJBQW1CLFFBQVEsSUFBSSxRQUFRLEtBQUs7QUFDaEQsb0JBQWM7QUFBQSxJQUNoQjtBQUVBLFdBQU87QUFBQSxFQUNULFNBQVMsT0FBTztBQUNkLFFBQUksa0NBQWtDLFFBQVEsS0FBSyxNQUFNLE9BQU8sSUFBSSxTQUFTLEtBQUs7QUFDbEYsV0FBTztBQUFBLEVBQ1Q7QUFDRjtBQUVBLE1BQU0sZUFBZSxPQUFPLEtBQWEsT0FBZ0IsUUFBaUIsVUFBeUI7QUFDakcsUUFBTSxnQkFBZ0IsQ0FBQyxNQUFNLE9BQU8sT0FBTyxNQUFNLE1BQU07QUFDdkQsUUFBTSxpQkFBaUI7QUFDdkIsTUFBSSxnQkFBZ0I7QUFFcEIsYUFBVSxVQUFVLGVBQWU7QUFDakMsVUFBTSxnQkFBZ0IsWUFBWSxLQUFLLEtBQUssY0FBYyxJQUFJLE1BQU0sRUFBRTtBQUN0RSxRQUFHLFdBQVcsYUFBYSxHQUFHO0FBQzVCLHNCQUFnQjtBQUNoQjtBQUFBLElBQ0Y7QUFBQSxFQUNGO0FBRUEsTUFBRyxlQUFlO0FBQ2hCLFFBQUk7QUFFRixZQUFNLFNBQVMsUUFBUSxhQUFhLEVBQUUsTUFBTSxDQUFDO0FBQzdDLFVBQUksYUFBYTtBQUdqQixVQUFHLFdBQVcsT0FBTztBQUNuQixZQUFJO0FBQ0YsZ0JBQU0sTUFBTSxJQUFJLElBQUksVUFBVSxhQUFhLEVBQUU7QUFDN0MsdUJBQWEsSUFBSTtBQUNqQixjQUFHLE9BQU87QUFDUixnQkFBSSxvQ0FBb0MsVUFBVSxJQUFJLFFBQVEsS0FBSztBQUFBLFVBQ3JFO0FBQUEsUUFDRixTQUFTLFVBQVU7QUFDakIsY0FBSSxzQ0FBc0MsU0FBUyxPQUFPLElBQUksUUFBUSxTQUFTLENBQUMsS0FBSztBQUNyRix1QkFBYSxVQUFVLGFBQWE7QUFBQSxRQUN0QztBQUFBLE1BQ0Y7QUFFQSxVQUFHLE9BQU87QUFDUixZQUFJLGdDQUFnQyxVQUFVLGFBQWEsTUFBTSxLQUFLLFFBQVEsS0FBSztBQUFBLE1BQ3JGO0FBRUEsVUFBSTtBQUNKLFVBQUk7QUFDRixvQkFBWSxNQUFNLE9BQU87QUFBQSxNQUMzQixTQUFTLGFBQWE7QUFDcEIsWUFBRyxZQUFZLFFBQVEsU0FBUyxnQ0FBZ0MsR0FBRztBQUNqRSxjQUFJLDZCQUE2QixhQUFhLGtFQUFrRSxTQUFTLEtBQUs7QUFDOUgsY0FBRyxPQUFPO0FBQ1Isb0JBQVEsTUFBTSxXQUFXO0FBQUEsVUFDM0I7QUFDQTtBQUFBLFFBQ0Y7QUFDQSxjQUFNO0FBQUEsTUFDUjtBQUdBLFVBQUksYUFBYTtBQUNqQixVQUFHLFVBQVUsU0FBUztBQUNwQixxQkFBYSxVQUFVO0FBQ3ZCLFlBQUcsT0FBTztBQUNSLGNBQUksMkJBQTJCLGFBQWEsSUFBSSxRQUFRLEtBQUs7QUFBQSxRQUMvRDtBQUFBLE1BQ0YsT0FBTztBQUVMLHFCQUFhO0FBQ2IsWUFBRyxPQUFPO0FBQ1IsY0FBSSwwQkFBMEIsYUFBYSxJQUFJLFFBQVEsS0FBSztBQUFBLFFBQzlEO0FBQUEsTUFDRjtBQUVBLFVBQUcsY0FBYyxXQUFXLElBQUk7QUFDOUIsWUFBSSw2QkFBNkIsWUFBWSxLQUFLLGFBQWEsQ0FBQywwQkFBMEIsUUFBUSxLQUFLO0FBQ3ZHLGtCQUFVLE9BQU8sS0FBSyxFQUFDLEdBQUcsVUFBVSxPQUFPLElBQUksR0FBRyxXQUFXLEdBQUU7QUFBQSxNQUNqRTtBQUFBLElBQ0YsU0FBUyxPQUFPO0FBQ2QsVUFBSSx1Q0FBdUMsYUFBYSxLQUFLLE1BQU0sT0FBTyxJQUFJLFFBQVEsS0FBSztBQUMzRixVQUFHLE9BQU87QUFDUixnQkFBUSxNQUFNLEtBQUs7QUFBQSxNQUNyQjtBQUFBLElBQ0Y7QUFBQSxFQUNGO0FBQ0Y7QUFLQSxNQUFNLG1CQUFtQixPQUFPLEtBQWEsT0FBZ0IsVUFBcUM7QUFFaEcsTUFBRyxVQUFVLE9BQU8sVUFBVSxPQUFPLEtBQUssVUFBVSxPQUFPLE1BQU0sRUFBRSxTQUFTLEdBQUc7QUFDN0UsUUFBSSxtREFBbUQsUUFBUSxTQUFTLENBQUMsS0FBSztBQUM5RSxXQUFPO0FBQUEsRUFDVDtBQUdBLFFBQU0sZ0JBQWdCLENBQUMsTUFBTSxPQUFPLE9BQU8sTUFBTSxNQUFNO0FBQ3ZELFFBQU0saUJBQWlCO0FBRXZCLGFBQVUsVUFBVSxlQUFlO0FBQ2pDLFVBQU0sZ0JBQWdCLFlBQVksS0FBSyxLQUFLLGNBQWMsSUFBSSxNQUFNLEVBQUU7QUFDdEUsUUFBRyxXQUFXLGFBQWEsR0FBRztBQUM1QixVQUFJO0FBRUYsY0FBTSxhQUFhLFFBQVEsYUFBYSxFQUFFLE1BQU0sQ0FBQztBQUNqRCxZQUFJLGFBQWE7QUFHakIsWUFBRyxlQUFlLE9BQU87QUFDdkIsY0FBSTtBQUNGLGtCQUFNLE1BQU0sSUFBSSxJQUFJLFVBQVUsYUFBYSxFQUFFO0FBQzdDLHlCQUFhLElBQUk7QUFDakIsZ0JBQUcsT0FBTztBQUNSLGtCQUFJLG9DQUFvQyxVQUFVLElBQUksUUFBUSxLQUFLO0FBQUEsWUFDckU7QUFBQSxVQUNGLFNBQVMsVUFBVTtBQUNqQixnQkFBSSxzQ0FBc0MsU0FBUyxPQUFPLElBQUksUUFBUSxTQUFTLENBQUMsS0FBSztBQUNyRix5QkFBYSxVQUFVLGFBQWE7QUFBQSxVQUN0QztBQUFBLFFBQ0Y7QUFFQSxZQUFHLE9BQU87QUFDUixjQUFJLGdDQUFnQyxVQUFVLGFBQWEsVUFBVSxLQUFLLFFBQVEsS0FBSztBQUFBLFFBQ3pGO0FBRUEsWUFBSTtBQUNKLFlBQUk7QUFDRixzQkFBWSxNQUFNLE9BQU87QUFBQSxRQUMzQixTQUFTLGFBQWE7QUFDcEIsY0FBRyxZQUFZLFFBQVEsU0FBUyxnQ0FBZ0MsR0FBRztBQUNqRSxnQkFBSSw2QkFBNkIsYUFBYSxrRUFBa0UsU0FBUyxLQUFLO0FBQzlILGdCQUFHLE9BQU87QUFDUixzQkFBUSxNQUFNLFdBQVc7QUFBQSxZQUMzQjtBQUNBO0FBQUEsVUFDRjtBQUNBLGdCQUFNO0FBQUEsUUFDUjtBQUdBLFlBQUksYUFBYTtBQUNqQixZQUFHLFVBQVUsU0FBUztBQUNwQix1QkFBYSxVQUFVO0FBQ3ZCLGNBQUcsT0FBTztBQUNSLGdCQUFJLDJCQUEyQixhQUFhLElBQUksUUFBUSxLQUFLO0FBQUEsVUFDL0Q7QUFBQSxRQUNGLE9BQU87QUFFTCx1QkFBYTtBQUNiLGNBQUcsT0FBTztBQUNSLGdCQUFJLDBCQUEwQixhQUFhLElBQUksUUFBUSxLQUFLO0FBQUEsVUFDOUQ7QUFBQSxRQUNGO0FBRUEsWUFBRyxjQUFjLFdBQVcsVUFBVSxPQUFPLEtBQUssV0FBVyxNQUFNLEVBQUUsU0FBUyxHQUFHO0FBQy9FLGNBQUksaUNBQWlDLFlBQVksS0FBSyxhQUFhLENBQUMsMEJBQTBCLFFBQVEsU0FBUyxDQUFDLEtBQUs7QUFDckgsb0JBQVUsT0FBTyxTQUFTLEVBQUMsR0FBRyxVQUFVLE9BQU8sUUFBUSxHQUFHLFdBQVcsT0FBTTtBQUMzRSxpQkFBTztBQUFBLFFBQ1Q7QUFBQSxNQUNGLFNBQVMsT0FBTztBQUNkLFlBQUksMkNBQTJDLGFBQWEsS0FBSyxNQUFNLE9BQU8sSUFBSSxRQUFRLEtBQUs7QUFDL0YsWUFBRyxPQUFPO0FBQ1Isa0JBQVEsTUFBTSxLQUFLO0FBQUEsUUFDckI7QUFBQSxNQUNGO0FBQUEsSUFDRjtBQUFBLEVBQ0Y7QUFFQSxTQUFPO0FBQ1Q7QUFLQSxNQUFNLHFCQUFxQixDQUFDLFVBQWtCLFVBQTRCO0FBQ3hFLE1BQUk7QUFDRixVQUFNLGNBQWMsYUFBYSxVQUFVLE1BQU07QUFHakQsUUFBRyxZQUFZLFNBQVMsS0FBUztBQUMvQixVQUFJLDRDQUE0QyxRQUFRLElBQUksUUFBUSxLQUFLO0FBQ3pFLGFBQU87QUFBQSxJQUNUO0FBR0EsVUFBTSxNQUFNLFFBQVEsUUFBUTtBQUM1QixRQUFJLGVBQWU7QUFDbkIsUUFBSSxlQUFlO0FBRW5CLFFBQUcsQ0FBQyxPQUFPLE1BQU0sRUFBRSxTQUFTLEdBQUcsR0FBRztBQUNoQyxxQkFBZTtBQUFBLElBQ2pCLFdBQVUsQ0FBQyxPQUFPLE1BQU0sRUFBRSxTQUFTLEdBQUcsR0FBRztBQUN2QyxxQkFBZTtBQUFBLElBQ2pCLE9BQU87QUFFTCxhQUFPO0FBQUEsSUFDVDtBQVNBLFFBQUksYUFBYSxZQUFZO0FBQUEsTUFDM0I7QUFBQSxNQUNBLENBQUMsVUFBVTtBQUNULFlBQUcsTUFBTSxTQUFTLFdBQVcsS0FDM0IsTUFBTSxTQUFTLFNBQVMsS0FDeEIsTUFBTSxTQUFTLFNBQVMsS0FDeEIsTUFBTSxTQUFTLFNBQVMsR0FBRztBQUMzQixpQkFBTztBQUFBLFFBQ1Q7QUFDQSxlQUFPO0FBQUEsTUFDVDtBQUFBLElBQ0Y7QUFHQSxpQkFBYSxXQUFXO0FBQUEsTUFDdEI7QUFBQSxNQUNBLENBQUMsVUFBVTtBQUNULFlBQUcsTUFBTSxTQUFTLE1BQU0sS0FBSyxNQUFNLFNBQVMsT0FBTyxHQUFHO0FBQ3BELGlCQUFPO0FBQUEsUUFDVDtBQUNBLGVBQU87QUFBQSxNQUNUO0FBQUEsSUFDRjtBQUdBLGlCQUFhLFdBQVcsUUFBUSxpQkFBaUIsTUFBTTtBQUd2RCxRQUFHLGVBQWUsYUFBYTtBQUM3QixvQkFBYyxVQUFVLFlBQVksTUFBTTtBQUMxQyxVQUFJLHlCQUF5QixRQUFRLElBQUksUUFBUSxLQUFLO0FBQ3RELGFBQU87QUFBQSxJQUNUO0FBRUEsV0FBTztBQUFBLEVBQ1QsU0FBUyxPQUFPO0FBQ2QsUUFBSSxnQ0FBZ0MsUUFBUSxLQUFLLE1BQU0sT0FBTyxJQUFJLFNBQVMsS0FBSztBQUNoRixXQUFPO0FBQUEsRUFDVDtBQUNGO0FBRU8sTUFBTSxPQUFPLE9BQU8sS0FBa0QsV0FBeUIsUUFBUSxTQUEwQjtBQUN0SSxRQUFNO0FBQUEsSUFDSixVQUFVO0FBQUEsSUFDVixNQUFNO0FBQUEsSUFDTixRQUFRO0FBQUEsSUFDUixRQUFRO0FBQUEsSUFDUixTQUFTO0FBQUEsSUFDVCxpQkFBaUI7QUFBQTtBQUFBLElBRWpCLG1CQUFtQixxQkFBcUI7QUFBQSxFQUMxQyxJQUFJO0FBR0osUUFBTSx1QkFBdUIsa0JBQWtCO0FBRS9DLE1BQUksR0FBRyxPQUFPLGVBQWUsUUFBUSxLQUFLO0FBRTFDLFFBQU0sTUFBTSxRQUFRLElBQUk7QUFDeEIsUUFBTSxVQUFVLGNBQWMsS0FBSztBQUVuQyxRQUFNLGFBQWEsS0FBSyxPQUFPLEtBQUs7QUFFcEMsTUFBSSxpQkFBZ0M7QUFFcEMsTUFBSTtBQUNGLFVBQU0sZ0JBQWdCLGlCQUFpQixHQUFHO0FBQzFDLFFBQUksY0FBYyxnQkFBZ0IsYUFBYSxjQUFjLHVCQUF1QixRQUFRLEtBQUs7QUFFakcsUUFBRyxlQUFlO0FBQ2hCLGdCQUFVLDBCQUEwQjtBQUFBLElBQ3RDO0FBRUEscUJBQWlCLEdBQUc7QUFFcEIsVUFBTSxvQkFBb0IsS0FBSyxlQUFlLEtBQUs7QUFFbkQsVUFBTSxvQkFBb0IsWUFBWSxLQUFLLGtCQUFrQjtBQUM3RCxVQUFNLHNCQUFzQixZQUFZLEtBQUssa0JBQWtCO0FBQy9ELFVBQU0sa0JBQWtCLFdBQVcsaUJBQWlCLEtBQ2xELFdBQVcsbUJBQW1CLEtBQzlCLFdBQVcsWUFBWSxLQUFLLGNBQWMsQ0FBQyxLQUMzQyxXQUFXLFlBQVksS0FBSyxnQkFBZ0IsQ0FBQyxLQUM3QyxXQUFXLFlBQVksS0FBSyxlQUFlLENBQUMsS0FDNUMsV0FBVyxZQUFZLEtBQUssZ0JBQWdCLENBQUMsS0FDN0MsV0FBVyxZQUFZLEtBQUssV0FBVyxDQUFDO0FBRzFDLFVBQU0scUJBQXFCLE1BQU0saUJBQWlCLEtBQUssT0FBTyxLQUFLO0FBRW5FLFFBQUcsb0JBQW9CO0FBQ3JCLFVBQUkscURBQXFELFFBQVEsS0FBSztBQUFBLElBQ3hFO0FBRUEsUUFBRyxXQUFXLFlBQVksS0FBSyxnQkFBZ0IsQ0FBQyxHQUFHO0FBQ2pELGlCQUFXLFlBQVksS0FBSyxnQkFBZ0IsQ0FBQztBQUFBLElBQy9DO0FBRUEsUUFBSSxnQkFBZ0I7QUFDcEIsUUFBSSx5QkFBeUI7QUFFN0IsUUFBRyxDQUFDLG1CQUFtQixDQUFDLG9CQUFvQjtBQUMxQyxZQUFNLGdCQUFnQjtBQUFBLFFBQ3BCLFlBQVksZ0JBQWdCLDhCQUE4QjtBQUFBLFFBQzFELFlBQVksZ0JBQWdCLDhCQUE4QjtBQUFBLFFBQzFELFlBQVksUUFBUSxJQUFJLFlBQVksNkJBQTZCLGtCQUFrQjtBQUFBLFFBQ25GLFlBQVksUUFBUSxJQUFJLFlBQVksNkJBQTZCLGtCQUFrQjtBQUFBLE1BQ3JGO0FBRUEsaUJBQVUsUUFBUSxlQUFlO0FBQy9CLFlBQUcsV0FBVyxJQUFJLEdBQUc7QUFDbkIsMEJBQWdCO0FBQ2hCO0FBQUEsUUFDRjtBQUFBLE1BQ0Y7QUFFQSxVQUFHLE9BQU87QUFDUixZQUFJLHNCQUFzQixjQUFjLElBQUksUUFBUSxLQUFLO0FBQ3pELFlBQUksd0JBQXdCLGlCQUFpQixJQUFJLFFBQVEsS0FBSztBQUM5RCxZQUFJLDBCQUEwQixlQUFlLElBQUksUUFBUSxLQUFLO0FBQzlELFlBQUkscUJBQXFCLGFBQWEsSUFBSSxRQUFRLEtBQUs7QUFDdkQsWUFBSSxzQkFBc0IsQ0FBQyxDQUFDLGlCQUFpQixXQUFXLGFBQWEsQ0FBQyxJQUFJLFFBQVEsS0FBSztBQUFBLE1BQ3pGO0FBRUEsVUFBRyxpQkFBaUIsV0FBVyxhQUFhLEdBQUc7QUFDN0MsWUFBSSxnRkFBaUYsUUFBUSxLQUFLO0FBQUEsTUFDcEcsT0FBTztBQUNMLGlDQUF5QjtBQUFBLE1BQzNCO0FBQUEsSUFDRjtBQUVBLFFBQUcsUUFBUTtBQUNULFlBQU0saUJBQWlCLFlBQVksS0FBSyxNQUFNO0FBQzlDLFVBQUcsV0FBVyxjQUFjLEdBQUc7QUFDN0IsWUFBSSx5Q0FBeUMsTUFBTSxJQUFJLFFBQVEsS0FBSztBQUNwRSxpQ0FBeUI7QUFBQSxNQUMzQixPQUFPO0FBQ0wsWUFBSSw2Q0FBNkMsTUFBTSx3Q0FBd0MsUUFBUSxLQUFLO0FBQUEsTUFDOUc7QUFBQSxJQUNGO0FBRUEsUUFBRyx3QkFBd0I7QUFDekIsVUFBSSx3RUFBd0UsUUFBUSxLQUFLO0FBQ3pGLFlBQU0sZUFBZSwwQkFBMEIsZUFBZSxHQUFHO0FBQ2pFLHVCQUFpQixhQUFhO0FBQUEsSUFDaEM7QUFFQSxRQUFJLGVBQWU7QUFDbkIsVUFBTSxnQkFBZ0IsQ0FBQyxXQUFtQjtBQUN4QyxzQkFBZ0IsR0FBRyxNQUFNO0FBQUE7QUFBQSxJQUMzQjtBQUVBLFVBQU0sU0FBUyxNQUFNLGlCQUFpQixLQUFLLE9BQU8sU0FBUyxNQUFNLE9BQU8sZUFBZSxhQUFhO0FBR3BHLFFBQUcsc0JBQXNCO0FBQ3ZCLGNBQVEsTUFBTSxpQ0FBaUM7QUFHL0MsWUFBTSxPQUFPLE1BQU0sT0FBTyxNQUFNO0FBQ2hDLFlBQU0sUUFBUSxLQUFLLEtBQUssa0NBQWtDO0FBQUEsUUFDeEQ7QUFBQSxRQUNBLFFBQVEsQ0FBQyxzQkFBc0IsY0FBYyxhQUFhO0FBQUEsTUFDNUQsQ0FBQztBQUVELFVBQUksaUJBQWlCO0FBQ3JCLGlCQUFVLFFBQVEsT0FBTztBQUN2QixjQUFNLFdBQVcsWUFBWSxLQUFLLElBQUk7QUFDdEMsWUFBRyxtQkFBbUIsVUFBVSxLQUFLLEdBQUc7QUFDdEM7QUFBQSxRQUNGO0FBQUEsTUFDRjtBQUVBLGNBQVEsUUFBUSx5QkFBeUIsY0FBYyxRQUFRO0FBQUEsSUFDakU7QUFFQSxRQUFHLFdBQVcsS0FBSyxLQUFLO0FBQ3RCLFlBQU0sZUFBZSxVQUFVLE9BQU8sSUFBSSxZQUFZLFVBQVUsT0FBTyxHQUFHLGFBQWE7QUFFdkYsVUFBRyxjQUFjO0FBQ2YsWUFBSSw0Q0FBNEMsUUFBUSxLQUFLO0FBQzdELGNBQU0sV0FBVyxLQUFLLGNBQWMsS0FBSztBQUV6QyxjQUFNLGlCQUFpQixNQUFNLGlCQUFpQixLQUFLLE9BQU8sU0FBUyxPQUFPLE9BQU8sYUFBYTtBQUU5RixpQkFBUyxjQUFjO0FBQ3ZCLGVBQU87QUFBQSxNQUNUO0FBQ0EsVUFBSSxrREFBa0QsUUFBUSxLQUFLO0FBQ25FLFVBQUksNkVBQTZFLFFBQVEsS0FBSztBQUM5RixVQUFJO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQTtBQUFBO0FBQUE7QUFBQSxLQVFMLFFBQVEsS0FBSztBQUFBLElBQ2Q7QUFFQSxhQUFTLE1BQU07QUFDZixXQUFPO0FBQUEsRUFDVCxTQUFTLE9BQU87QUFDZCxRQUFJO0FBQUEsRUFBSyxPQUFPLFdBQVcsTUFBTSxPQUFPLElBQUksU0FBUyxLQUFLO0FBQzFELFFBQUcsU0FBUztBQUNWLGNBQVEsS0FBSyxpQkFBaUI7QUFBQSxJQUNoQztBQUNBLGFBQVMsQ0FBQztBQUNWLFdBQU87QUFBQSxFQUNULFVBQUU7QUFFQSxVQUFNLGdCQUFnQjtBQUFBLE1BQ3BCO0FBQUEsTUFDQSxZQUFZLEtBQUssc0JBQXNCO0FBQUEsTUFDdkMsWUFBWSxLQUFLLDhCQUE4QjtBQUFBLElBQ2pEO0FBRUEsZUFBVSxZQUFZLGVBQWU7QUFDbkMsVUFBRyxZQUFZLFdBQVcsUUFBUSxHQUFHO0FBQ25DLFlBQUk7QUFDRixxQkFBVyxRQUFRO0FBQ25CLGNBQUcsT0FBTztBQUNSLGdCQUFJLHlDQUF5QyxRQUFRLElBQUksUUFBUSxLQUFLO0FBQUEsVUFDeEU7QUFBQSxRQUNGLFNBQVMsUUFBUTtBQUFBLFFBRWpCO0FBQUEsTUFDRjtBQUFBLElBQ0Y7QUFBQSxFQUNGO0FBQ0Y7IiwKICAibmFtZXMiOiBbImxpbmVzIl0KfQo=
|