UNPKG

@nlabs/lex

Version:
494 lines (488 loc) 56.9 kB
import GraphqlLoaderPlugin from "@luckycatfactory/esbuild-graphql-loader"; import { execa } from "execa"; import { existsSync, readFileSync } from "fs"; import { sync as globSync } from "glob"; import { dirname, resolve as pathResolve } from "path"; import { LexConfig, getTypeScriptConfigPath } from "../../LexConfig.js"; import { checkLinkedModules, copyConfiguredFiles, createSpinner, handleWebpackProgress, removeFiles } from "../../utils/app.js"; import { getDirName, relativeNodePath, resolveWebpackPaths, getLexPackageJsonPath } from "../../utils/file.js"; import { log } from "../../utils/log.js"; import { processTranslations } from "../../utils/translations.js"; import { aiFunction } from "../ai/ai.js"; import boxen from "boxen"; import chalk from "chalk"; let currentFilename; let currentDirname; try { currentFilename = eval('require("url").fileURLToPath(import.meta.url)'); currentDirname = dirname(currentFilename); } catch { currentFilename = process.cwd(); currentDirname = process.cwd(); } const displayBuildStatus = (bundler, outputPath, quiet, stats) => { if (quiet) return; let statsInfo = ""; if (stats && stats.modules && stats.assets) { statsInfo = ` ${chalk.green("Modules:")} ${chalk.cyan(stats.modules)} ${chalk.green("Assets:")} ${chalk.cyan(stats.assets)} ${chalk.green("Size:")} ${chalk.cyan(stats.size)} `; } const statusBox = boxen( `${chalk.cyan.bold("\u{1F3D7}\uFE0F Build Completed Successfully")} ${chalk.green("Bundler:")} ${chalk.cyan(bundler)} ${chalk.green("Output:")} ${chalk.underline(outputPath)}${statsInfo} ${chalk.yellow("Ready for deployment! \u{1F680}")}`, { padding: 1, margin: 1, borderStyle: "round", borderColor: "green", backgroundColor: "#1a1a1a" } ); console.log("\n" + statusBox + "\n"); }; const buildWithEsBuild = async (spinner, commandOptions, callback) => { const { cliName = "Lex", format = "cjs", outputPath, quiet, sourcePath, watch } = commandOptions; const { outputFullPath, sourceFullPath, targetEnvironment, useGraphQl, useTypescript } = LexConfig.config; const sourceDir = sourcePath ? pathResolve(process.cwd(), `./${sourcePath}`) : sourceFullPath || ""; const loader = { ".js": "js" }; if (useTypescript) { loader[".ts"] = "ts"; loader[".tsx"] = "tsx"; } const plugins = []; if (useGraphQl) { plugins.push(GraphqlLoaderPlugin()); } const globOptions = { cwd: sourceDir, dot: false, nodir: true, nosort: true }; const tsFiles = globSync(`${sourceDir}/**/!(*.spec|*.test).ts*`, globOptions); const jsFiles = globSync(`${sourceDir}/**/!(*.spec|*.test).js`, globOptions); const sourceFiles = [...tsFiles, ...jsFiles]; const packageJsonData = readFileSync(pathResolve(process.cwd(), "./package.json")); const packageJson = JSON.parse(packageJsonData.toString()); const external = [ ...Object.keys(packageJson.dependencies || {}), ...Object.keys(packageJson.peerDependencies || {}) ]; const dirName = getDirName(); const dirPath = pathResolve(dirName, "../.."); const outputDir = outputPath || outputFullPath || ""; const esbuildPath = relativeNodePath("esbuild/bin/esbuild", dirPath); const esbuildConfig = LexConfig.config.esbuild || {}; const esbuildOptions = [ ...sourceFiles, "--bundle", "--color=true", `--format=${format}`, `--outdir=${outputDir}`, `--platform=${esbuildConfig.platform || "node"}`, `--target=${esbuildConfig.target || (targetEnvironment === "node" ? "node20" : "es2020")}`, `--sourcemap=${esbuildConfig.sourcemap || "inline"}` ]; if (esbuildConfig.minify !== false) { esbuildOptions.push("--minify"); } if (esbuildConfig.treeShaking !== false) { esbuildOptions.push("--tree-shaking=true"); } if (esbuildConfig.drop && esbuildConfig.drop.length > 0) { esbuildConfig.drop.forEach((item) => { esbuildOptions.push(`--drop:${item}`); }); } if (esbuildConfig.pure && esbuildConfig.pure.length > 0) { esbuildConfig.pure.forEach((item) => { esbuildOptions.push(`--pure:${item}`); }); } if (esbuildConfig.legalComments) { esbuildOptions.push(`--legal-comments=${esbuildConfig.legalComments}`); } if (esbuildConfig.splitting !== false) { esbuildOptions.push("--splitting"); } if (esbuildConfig.metafile) { esbuildOptions.push("--metafile"); } if (esbuildConfig.banner) { Object.entries(esbuildConfig.banner).forEach(([type, content]) => { esbuildOptions.push(`--banner:${type}=${content}`); }); } if (esbuildConfig.footer) { Object.entries(esbuildConfig.footer).forEach(([type, content]) => { esbuildOptions.push(`--footer:${type}=${content}`); }); } if (esbuildConfig.define) { Object.entries(esbuildConfig.define).forEach(([key, value]) => { esbuildOptions.push(`--define:${key}=${value}`); }); } if (external.length) { esbuildOptions.push(`--external:${external.join(",")}`); } if (plugins.length) { esbuildOptions.push(`--plugins=${plugins.join(",")}`); } if (watch) { esbuildOptions.push("--watch"); } try { await execa(esbuildPath, esbuildOptions, { encoding: "utf8" }); spinner.succeed("Build completed successfully!"); displayBuildStatus("esbuild", outputDir, quiet); } catch (error) { log(` ${cliName} Error: ${error.message}`, "error", quiet); if (!quiet) { console.error(error); } spinner.fail("Code build failed."); if (commandOptions.assist) { spinner.start("AI is analyzing the error..."); try { await aiFunction({ prompt: `Fix this esbuild error: ${error.message} Error details: ${error.toString()}`, task: "help", context: true, quiet }); spinner.succeed("AI analysis complete"); } catch (aiError) { spinner.fail("Could not generate AI assistance"); if (!quiet) { console.error("AI assistance error:", aiError); } } } callback(1); return 1; } callback(0); return 0; }; const buildWithWebpack = async (spinner, cmd, callback) => { const { analyze, cliName = "Lex", config, configName, defineProcessEnvNodeEnv, devtool, disableInterpret, entry, env, failOnWarnings, json, merge, mode, name, nodeEnv, noDevtool, noStats, noTarget, noWatch, noWatchOptionsStdin, outputPath, quiet = false, stats, target, watch, watchOptionsStdin } = cmd; console.log("entry:", entry, "type:", typeof entry); console.log("outputPath:", outputPath, "type:", typeof outputPath); const entryValue = Array.isArray(entry) ? entry[0] : entry; let webpackConfig; if (config) { const isRelativeConfig = config.substr(0, 2) === "./"; webpackConfig = isRelativeConfig ? pathResolve(process.cwd(), config) : config; } else { const projectConfigPath = pathResolve(process.cwd(), "webpack.config.js"); const projectConfigPathTs = pathResolve(process.cwd(), "webpack.config.ts"); const hasProjectConfig = existsSync(projectConfigPath) || existsSync(projectConfigPathTs); if (hasProjectConfig) { webpackConfig = existsSync(projectConfigPathTs) ? projectConfigPathTs : projectConfigPath; } else { const { webpackConfig: resolvedConfig } = resolveWebpackPaths(currentDirname); webpackConfig = resolvedConfig; } } console.log("webpackConfig path:", webpackConfig); console.log("webpackConfig exists:", existsSync(webpackConfig)); if (!existsSync(webpackConfig)) { const lexPackagePath = getLexPackageJsonPath(); const lexPackageDir = dirname(lexPackagePath); const lexWebpackConfig = pathResolve(lexPackageDir, "webpack.config.js"); if (existsSync(lexWebpackConfig)) { webpackConfig = lexWebpackConfig; console.log("Using Lex webpack config:", webpackConfig); } else { log(` ${cliName} Error: Could not find webpack.config.js`, "error", quiet); spinner.fail("Build failed."); callback(1); return 1; } } const webpackOptions = [ "--color", "--progress", "--config", webpackConfig ]; if (analyze) webpackOptions.push("--analyze"); if (configName) webpackOptions.push("--configName", configName); if (defineProcessEnvNodeEnv) webpackOptions.push("--defineProcessEnvNodeEnv", defineProcessEnvNodeEnv); if (devtool) webpackOptions.push("--devtool", devtool); if (disableInterpret) webpackOptions.push("--disableInterpret"); if (entryValue) webpackOptions.push("--entry", entryValue.toString()); if (env) webpackOptions.push("--env", env); if (failOnWarnings) webpackOptions.push("--failOnWarnings"); if (json) webpackOptions.push("--json", json); if (mode) webpackOptions.push("--mode", mode); if (merge) webpackOptions.push("--merge"); if (name) webpackOptions.push("--name", name); if (noDevtool) webpackOptions.push("--noDevtool"); if (noStats) webpackOptions.push("--noStats"); if (noTarget) webpackOptions.push("--noTarget"); if (noWatch) webpackOptions.push("--noWatch"); if (noWatchOptionsStdin) webpackOptions.push("--noWatchOptionsStdin"); if (nodeEnv) webpackOptions.push("--nodeEnv", nodeEnv); if (outputPath) webpackOptions.push("--output-path", outputPath.toString()); if (stats) webpackOptions.push("--stats", stats); if (target) webpackOptions.push("--target", target); if (watch) webpackOptions.push("--watch"); if (watchOptionsStdin) webpackOptions.push("--watchOptionsStdin"); try { const { webpackPath } = resolveWebpackPaths(currentDirname); let finalWebpackOptions; if (webpackPath === "npx") { finalWebpackOptions = ["webpack", ...webpackOptions]; } else { finalWebpackOptions = [...webpackOptions]; } console.log("webpackPath:", webpackPath); console.log("finalWebpackOptions:", JSON.stringify(finalWebpackOptions)); console.log("finalWebpackOptions type:", Array.isArray(finalWebpackOptions) ? "Array" : typeof finalWebpackOptions); const childProcess = execa(webpackPath, finalWebpackOptions, { encoding: "utf8", stdio: "pipe" }); let buildCompleted = false; let buildStats = { modules: 0, assets: 0, size: "0 B" }; childProcess.stdout?.on("data", (data) => { const output = data.toString(); handleWebpackProgress(output, spinner, quiet, "\u{1F3D7}\uFE0F", "Webpack Building"); if (!buildCompleted && output.includes("compiled successfully")) { buildCompleted = true; spinner.succeed("Build completed successfully!"); const moduleMatch = output.match(/(\d+) modules/); const assetMatch = output.match(/(\d+) assets/); const sizeMatch = output.match(/assets by status ([\d.]+ \w+)/) || output.match(/assets by path.*?([\d.]+ \w+)/); if (moduleMatch) buildStats.modules = parseInt(moduleMatch[1], 10); if (assetMatch) buildStats.assets = parseInt(assetMatch[1], 10); if (sizeMatch) buildStats.size = sizeMatch[1]; displayBuildStatus("webpack", LexConfig.config.outputFullPath || "lib", quiet, buildStats); } }); childProcess.stderr?.on("data", (data) => { const output = data.toString(); handleWebpackProgress(output, spinner, quiet, "\u{1F3D7}\uFE0F", "Webpack Building"); if (!buildCompleted && output.includes("compiled successfully")) { buildCompleted = true; spinner.succeed("Build completed successfully!"); const moduleMatch = output.match(/(\d+) modules/); const assetMatch = output.match(/(\d+) assets/); const sizeMatch = output.match(/assets by status ([\d.]+ \w+)/) || output.match(/assets by path.*?([\d.]+ \w+)/); if (moduleMatch) buildStats.modules = parseInt(moduleMatch[1], 10); if (assetMatch) buildStats.assets = parseInt(assetMatch[1], 10); if (sizeMatch) buildStats.size = sizeMatch[1]; displayBuildStatus("webpack", LexConfig.config.outputFullPath || "lib", quiet, buildStats); } }); await childProcess; if (!buildCompleted) { spinner.succeed("Build completed successfully!"); displayBuildStatus("webpack", LexConfig.config.outputFullPath || "lib", quiet, buildStats); } callback(0); return 0; } catch (error) { log(` ${cliName} Error: ${error.message}`, "error", quiet); spinner.fail("Build failed."); if (cmd.assist) { spinner.start("AI is analyzing the webpack error..."); try { await aiFunction({ prompt: `Fix this webpack build error: ${error.message} Error details: ${error.toString()} Configuration used: ${JSON.stringify(webpackOptions, null, 2)}`, task: "help", context: true, quiet }); spinner.succeed("AI analysis complete"); } catch (aiError) { spinner.fail("Could not generate AI assistance"); if (!quiet) { console.error("AI assistance error:", aiError); } } } callback(1); return 1; } }; const build = async (cmd, callback = () => ({})) => { const { bundler = "webpack", cliName = "Lex", quiet = false, remove = false, test = false, translations = false, variables = "{}" } = cmd; const spinner = createSpinner(quiet); log(`${cliName} building...`, "info", quiet); await LexConfig.parseConfig(cmd); const { outputFullPath, useTypescript } = LexConfig.config; checkLinkedModules(); let variablesObj = { NODE_ENV: "production" }; if (variables) { try { variablesObj = JSON.parse(variables); } catch (error) { log(` ${cliName} Error: Environment variables option is not a valid JSON object.`, "error", quiet); callback(1); return 1; } } process.env = { ...process.env, ...variablesObj }; if (test) { log("Test mode: Build environment loaded, exiting", "info", quiet); callback(0); return 0; } if (translations) { spinner.start("Processing translations..."); try { const sourcePath = LexConfig.config.sourceFullPath || process.cwd(); const outputPath = LexConfig.config.outputFullPath || "lib"; await processTranslations(sourcePath, outputPath, quiet); spinner.succeed("Translations processed successfully!"); } catch (translationError) { log(` ${cliName} Error: Failed to process translations: ${translationError.message}`, "error", quiet); spinner.fail("Failed to process translations."); callback(1); return 1; } } spinner.start("Building code..."); if (remove) { await removeFiles(outputFullPath || ""); } if (useTypescript) { const compileConfigPath = getTypeScriptConfigPath("tsconfig.build.json"); if (existsSync(compileConfigPath)) { log("Using tsconfig.build.json for build...", "info", quiet); } else { LexConfig.checkCompileTypescriptConfig(); } } let buildResult = 0; if (bundler === "esbuild") { buildResult = await buildWithEsBuild(spinner, cmd, (status) => { buildResult = status; }); } else { buildResult = await buildWithWebpack(spinner, cmd, (status) => { buildResult = status; }); } if (buildResult === 0 && cmd.analyze) { spinner.start("AI is analyzing the build output for optimization opportunities..."); try { const stats = { outputPath: LexConfig.config.outputFullPath, entryPoints: bundler === "esbuild" ? `Source files: ${LexConfig.config.sourceFullPath}/**/*.{ts,js}` : LexConfig.config.webpack?.entry || "Unknown entry points" }; await aiFunction({ prompt: `Analyze this build for optimization opportunities: Build Type: ${bundler} Format: ${cmd.format || "default"} Environment: ${LexConfig.config.targetEnvironment} ${JSON.stringify(stats, null, 2)} What are the key optimization opportunities for this build configuration? Consider: 1. Bundle size optimization strategies 2. Code splitting recommendations 3. Tree-shaking improvements 4. Performance enhancements 5. Dependency optimizations`, task: "optimize", context: true, quiet }); spinner.succeed("AI build analysis complete"); } catch (aiError) { spinner.fail("Could not generate AI optimization analysis"); if (!quiet) { console.error("AI analysis error:", aiError); } } } if (buildResult === 0) { try { await copyConfiguredFiles(spinner, LexConfig.config, quiet); } catch (copyError) { log(` ${cliName} Error: Failed to copy configured files: ${copyError.message}`, "error", quiet); callback(1); return 1; } } callback(buildResult); return buildResult; }; var build_default = build; export { build, buildWithEsBuild, buildWithWebpack, build_default as default }; //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vc3JjL2NvbW1hbmRzL2J1aWxkL2J1aWxkLnRzIl0sCiAgInNvdXJjZXNDb250ZW50IjogWyIvKipcbiAqIENvcHlyaWdodCAoYykgMjAxOC1QcmVzZW50LCBOaXRyb2dlbiBMYWJzLCBJbmMuXG4gKiBDb3B5cmlnaHRzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gU2VlIHRoZSBhY2NvbXBhbnlpbmcgTElDRU5TRSBmaWxlIGZvciB0ZXJtcy5cbiAqL1xuaW1wb3J0IEdyYXBocWxMb2FkZXJQbHVnaW4gZnJvbSAnQGx1Y2t5Y2F0ZmFjdG9yeS9lc2J1aWxkLWdyYXBocWwtbG9hZGVyJztcbmltcG9ydCB7ZXhlY2F9IGZyb20gJ2V4ZWNhJztcbmltcG9ydCB7ZXhpc3RzU3luYywgcmVhZEZpbGVTeW5jfSBmcm9tICdmcyc7XG5pbXBvcnQge3N5bmMgYXMgZ2xvYlN5bmN9IGZyb20gJ2dsb2InO1xuaW1wb3J0IHtkaXJuYW1lLCByZXNvbHZlIGFzIHBhdGhSZXNvbHZlfSBmcm9tICdwYXRoJztcblxuaW1wb3J0IHtMZXhDb25maWcsIGdldFR5cGVTY3JpcHRDb25maWdQYXRofSBmcm9tICcuLi8uLi9MZXhDb25maWcuanMnO1xuaW1wb3J0IHtjaGVja0xpbmtlZE1vZHVsZXMsIGNvcHlDb25maWd1cmVkRmlsZXMsIGNyZWF0ZVNwaW5uZXIsIGNyZWF0ZVByb2dyZXNzQmFyLCBoYW5kbGVXZWJwYWNrUHJvZ3Jlc3MsIHJlbW92ZUZpbGVzfSBmcm9tICcuLi8uLi91dGlscy9hcHAuanMnO1xuaW1wb3J0IHtnZXREaXJOYW1lLCByZWxhdGl2ZU5vZGVQYXRoLCByZXNvbHZlV2VicGFja1BhdGhzLCBnZXRMZXhQYWNrYWdlSnNvblBhdGh9IGZyb20gJy4uLy4uL3V0aWxzL2ZpbGUuanMnO1xuaW1wb3J0IHtsb2d9IGZyb20gJy4uLy4uL3V0aWxzL2xvZy5qcyc7XG5pbXBvcnQge3Byb2Nlc3NUcmFuc2xhdGlvbnN9IGZyb20gJy4uLy4uL3V0aWxzL3RyYW5zbGF0aW9ucy5qcyc7XG5pbXBvcnQge2FpRnVuY3Rpb259IGZyb20gJy4uL2FpL2FpLmpzJztcbmltcG9ydCBib3hlbiBmcm9tICdib3hlbic7XG5pbXBvcnQgY2hhbGsgZnJvbSAnY2hhbGsnO1xuXG5sZXQgY3VycmVudEZpbGVuYW1lOiBzdHJpbmc7XG5sZXQgY3VycmVudERpcm5hbWU6IHN0cmluZztcblxudHJ5IHtcbiAgY3VycmVudEZpbGVuYW1lID0gZXZhbCgncmVxdWlyZShcInVybFwiKS5maWxlVVJMVG9QYXRoKGltcG9ydC5tZXRhLnVybCknKTtcbiAgY3VycmVudERpcm5hbWUgPSBkaXJuYW1lKGN1cnJlbnRGaWxlbmFtZSk7XG59IGNhdGNoIHtcbiAgY3VycmVudEZpbGVuYW1lID0gcHJvY2Vzcy5jd2QoKTtcbiAgY3VycmVudERpcm5hbWUgPSBwcm9jZXNzLmN3ZCgpO1xufVxuXG5leHBvcnQgaW50ZXJmYWNlIEJ1aWxkT3B0aW9ucyB7XG4gIHJlYWRvbmx5IGFzc2lzdD86IGJvb2xlYW47XG4gIHJlYWRvbmx5IGFuYWx5emU/OiBib29sZWFuO1xuICByZWFkb25seSBidW5kbGVyPzogJ3dlYnBhY2snIHwgJ2VzYnVpbGQnO1xuICByZWFkb25seSBjbGlOYW1lPzogc3RyaW5nO1xuICByZWFkb25seSBlbnRyeT86IHN0cmluZzsgLy8gPC0tIGFkZCBlbnRyeVxuICByZWFkb25seSBmb3JtYXQ/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IG91dHB1dFBhdGg/OiBzdHJpbmc7IC8vIDwtLSBhbHJlYWR5IHByZXNlbnRcbiAgcmVhZG9ubHkgcXVpZXQ/OiBib29sZWFuO1xuICByZWFkb25seSByZW1vdmU/OiBib29sZWFuO1xuICByZWFkb25seSBzb3VyY2VQYXRoPzogc3RyaW5nO1xuICByZWFkb25seSB0ZXN0PzogYm9vbGVhbjtcbiAgcmVhZG9ubHkgdHJhbnNsYXRpb25zPzogYm9vbGVhbjtcbiAgcmVhZG9ubHkgdmFyaWFibGVzPzogc3RyaW5nO1xuICByZWFkb25seSB3YXRjaD86IGJvb2xlYW47XG59XG5cbmV4cG9ydCB0eXBlIEJ1aWxkQ2FsbGJhY2sgPSAoc3RhdHVzOiBudW1iZXIpID0+IHZvaWQ7XG5cbmNvbnN0IGRpc3BsYXlCdWlsZFN0YXR1cyA9IChidW5kbGVyOiBzdHJpbmcsIG91dHB1dFBhdGg6IHN0cmluZywgcXVpZXQ6IGJvb2xlYW4sIHN0YXRzPzoge21vZHVsZXM/OiBudW1iZXI7IGFzc2V0cz86IG51bWJlcjsgc2l6ZT86IHN0cmluZ30pID0+IHtcbiAgaWYocXVpZXQpIHJldHVybjtcblxuICBsZXQgc3RhdHNJbmZvID0gJyc7XG4gIGlmKHN0YXRzICYmIHN0YXRzLm1vZHVsZXMgJiYgc3RhdHMuYXNzZXRzKSB7XG4gICAgc3RhdHNJbmZvID0gYFxcbiR7Y2hhbGsuZ3JlZW4oJ01vZHVsZXM6Jyl9ICAgICR7Y2hhbGsuY3lhbihzdGF0cy5tb2R1bGVzKX1cXG5gICtcbiAgICAgIGAke2NoYWxrLmdyZWVuKCdBc3NldHM6Jyl9ICAgICAke2NoYWxrLmN5YW4oc3RhdHMuYXNzZXRzKX1cXG5gICtcbiAgICAgIGAke2NoYWxrLmdyZWVuKCdTaXplOicpfSAgICAgICAke2NoYWxrLmN5YW4oc3RhdHMuc2l6ZSl9XFxuYDtcbiAgfVxuXG4gIGNvbnN0IHN0YXR1c0JveCA9IGJveGVuKFxuICAgIGAke2NoYWxrLmN5YW4uYm9sZCgnXHVEODNDXHVERkQ3XHVGRTBGICBCdWlsZCBDb21wbGV0ZWQgU3VjY2Vzc2Z1bGx5Jyl9XFxuXFxuYCArXG4gICAgYCR7Y2hhbGsuZ3JlZW4oJ0J1bmRsZXI6Jyl9ICAgICR7Y2hhbGsuY3lhbihidW5kbGVyKX1cXG5gICtcbiAgICBgJHtjaGFsay5ncmVlbignT3V0cHV0OicpfSAgICAgJHtjaGFsay51bmRlcmxpbmUob3V0cHV0UGF0aCl9JHtzdGF0c0luZm99XFxuYCArXG4gICAgYCR7Y2hhbGsueWVsbG93KCdSZWFkeSBmb3IgZGVwbG95bWVudCEgXHVEODNEXHVERTgwJyl9YCxcbiAgICB7XG4gICAgICBwYWRkaW5nOiAxLFxuICAgICAgbWFyZ2luOiAxLFxuICAgICAgYm9yZGVyU3R5bGU6ICdyb3VuZCcsXG4gICAgICBib3JkZXJDb2xvcjogJ2dyZWVuJyxcbiAgICAgIGJhY2tncm91bmRDb2xvcjogJyMxYTFhMWEnXG4gICAgfVxuICApO1xuXG4gIGNvbnNvbGUubG9nKCdcXG4nICsgc3RhdHVzQm94ICsgJ1xcbicpO1xufTtcblxuZXhwb3J0IGNvbnN0IGJ1aWxkV2l0aEVzQnVpbGQgPSBhc3luYyAoc3Bpbm5lciwgY29tbWFuZE9wdGlvbnM6IEJ1aWxkT3B0aW9ucywgY2FsbGJhY2s6IEJ1aWxkQ2FsbGJhY2spID0+IHtcbiAgY29uc3Qge1xuICAgIGNsaU5hbWUgPSAnTGV4JyxcbiAgICBmb3JtYXQgPSAnY2pzJyxcbiAgICBvdXRwdXRQYXRoLFxuICAgIHF1aWV0LFxuICAgIHNvdXJjZVBhdGgsXG4gICAgd2F0Y2hcbiAgfSA9IGNvbW1hbmRPcHRpb25zO1xuICBjb25zdCB7XG4gICAgb3V0cHV0RnVsbFBhdGgsXG4gICAgc291cmNlRnVsbFBhdGgsXG4gICAgdGFyZ2V0RW52aXJvbm1lbnQsXG4gICAgdXNlR3JhcGhRbCxcbiAgICB1c2VUeXBlc2NyaXB0XG4gIH0gPSBMZXhDb25maWcuY29uZmlnO1xuICBjb25zdCBzb3VyY2VEaXI6IHN0cmluZyA9IHNvdXJjZVBhdGggPyBwYXRoUmVzb2x2ZShwcm9jZXNzLmN3ZCgpLCBgLi8ke3NvdXJjZVBhdGh9YCkgOiBzb3VyY2VGdWxsUGF0aCB8fCAnJztcbiAgY29uc3QgbG9hZGVyID0ge1xuICAgICcuanMnOiAnanMnXG4gIH07XG5cbiAgaWYodXNlVHlwZXNjcmlwdCkge1xuICAgIGxvYWRlclsnLnRzJ10gPSAndHMnO1xuICAgIGxvYWRlclsnLnRzeCddID0gJ3RzeCc7XG4gIH1cblxuICBjb25zdCBwbHVnaW5zID0gW107XG5cbiAgaWYodXNlR3JhcGhRbCkge1xuICAgIHBsdWdpbnMucHVzaCgoR3JhcGhxbExvYWRlclBsdWdpbiBhcyB1bmtub3duIGFzICgpID0+IG5ldmVyKSgpKTtcbiAgfVxuXG4gIGNvbnN0IGdsb2JPcHRpb25zID0ge1xuICAgIGN3ZDogc291cmNlRGlyLFxuICAgIGRvdDogZmFsc2UsXG4gICAgbm9kaXI6IHRydWUsXG4gICAgbm9zb3J0OiB0cnVlXG4gIH07XG4gIGNvbnN0IHRzRmlsZXM6IHN0cmluZ1tdID0gZ2xvYlN5bmMoYCR7c291cmNlRGlyfS8qKi8hKCouc3BlY3wqLnRlc3QpLnRzKmAsIGdsb2JPcHRpb25zKTtcbiAgY29uc3QganNGaWxlczogc3RyaW5nW10gPSBnbG9iU3luYyhgJHtzb3VyY2VEaXJ9LyoqLyEoKi5zcGVjfCoudGVzdCkuanNgLCBnbG9iT3B0aW9ucyk7XG4gIGNvbnN0IHNvdXJjZUZpbGVzOiBzdHJpbmdbXSA9IFsuLi50c0ZpbGVzLCAuLi5qc0ZpbGVzXTtcblxuICBjb25zdCBwYWNrYWdlSnNvbkRhdGEgPSByZWFkRmlsZVN5bmMocGF0aFJlc29sdmUocHJvY2Vzcy5jd2QoKSwgJy4vcGFja2FnZS5qc29uJykpO1xuICBjb25zdCBwYWNrYWdlSnNvbiA9IEpTT04ucGFyc2UocGFja2FnZUpzb25EYXRhLnRvU3RyaW5nKCkpO1xuICBjb25zdCBleHRlcm5hbCA9IFtcbiAgICAuLi5PYmplY3Qua2V5cyhwYWNrYWdlSnNvbi5kZXBlbmRlbmNpZXMgfHwge30pLFxuICAgIC4uLk9iamVjdC5rZXlzKHBhY2thZ2VKc29uLnBlZXJEZXBlbmRlbmNpZXMgfHwge30pXG4gIF07XG5cbiAgY29uc3QgZGlyTmFtZSA9IGdldERpck5hbWUoKTtcbiAgY29uc3QgZGlyUGF0aDogc3RyaW5nID0gcGF0aFJlc29sdmUoZGlyTmFtZSwgJy4uLy4uJyk7XG4gIGNvbnN0IG91dHB1dERpcjogc3RyaW5nID0gb3V0cHV0UGF0aCB8fCBvdXRwdXRGdWxsUGF0aCB8fCAnJztcbiAgY29uc3QgZXNidWlsZFBhdGg6IHN0cmluZyA9IHJlbGF0aXZlTm9kZVBhdGgoJ2VzYnVpbGQvYmluL2VzYnVpbGQnLCBkaXJQYXRoKTtcbiAgY29uc3QgZXNidWlsZENvbmZpZyA9IExleENvbmZpZy5jb25maWcuZXNidWlsZCB8fCB7fTtcblxuICBjb25zdCBlc2J1aWxkT3B0aW9uczogc3RyaW5nW10gPSBbXG4gICAgLi4uc291cmNlRmlsZXMsXG4gICAgJy0tYnVuZGxlJyxcbiAgICAnLS1jb2xvcj10cnVlJyxcbiAgICBgLS1mb3JtYXQ9JHtmb3JtYXR9YCxcbiAgICBgLS1vdXRkaXI9JHtvdXRwdXREaXJ9YCxcbiAgICBgLS1wbGF0Zm9ybT0ke2VzYnVpbGRDb25maWcucGxhdGZvcm0gfHwgJ25vZGUnfWAsXG4gICAgYC0tdGFyZ2V0PSR7ZXNidWlsZENvbmZpZy50YXJnZXQgfHwgKHRhcmdldEVudmlyb25tZW50ID09PSAnbm9kZScgPyAnbm9kZTIwJyA6ICdlczIwMjAnKX1gLFxuICAgIGAtLXNvdXJjZW1hcD0ke2VzYnVpbGRDb25maWcuc291cmNlbWFwIHx8ICdpbmxpbmUnfWBcbiAgXTtcblxuICBpZihlc2J1aWxkQ29uZmlnLm1pbmlmeSAhPT0gZmFsc2UpIHtcbiAgICBlc2J1aWxkT3B0aW9ucy5wdXNoKCctLW1pbmlmeScpO1xuICB9XG5cbiAgaWYoZXNidWlsZENvbmZpZy50cmVlU2hha2luZyAhPT0gZmFsc2UpIHtcbiAgICBlc2J1aWxkT3B0aW9ucy5wdXNoKCctLXRyZWUtc2hha2luZz10cnVlJyk7XG4gIH1cblxuICBpZihlc2J1aWxkQ29uZmlnLmRyb3AgJiYgZXNidWlsZENvbmZpZy5kcm9wLmxlbmd0aCA+IDApIHtcbiAgICBlc2J1aWxkQ29uZmlnLmRyb3AuZm9yRWFjaChpdGVtID0+IHtcbiAgICAgIGVzYnVpbGRPcHRpb25zLnB1c2goYC0tZHJvcDoke2l0ZW19YCk7XG4gICAgfSk7XG4gIH1cblxuICBpZihlc2J1aWxkQ29uZmlnLnB1cmUgJiYgZXNidWlsZENvbmZpZy5wdXJlLmxlbmd0aCA+IDApIHtcbiAgICBlc2J1aWxkQ29uZmlnLnB1cmUuZm9yRWFjaChpdGVtID0+IHtcbiAgICAgIGVzYnVpbGRPcHRpb25zLnB1c2goYC0tcHVyZToke2l0ZW19YCk7XG4gICAgfSk7XG4gIH1cblxuICBpZihlc2J1aWxkQ29uZmlnLmxlZ2FsQ29tbWVudHMpIHtcbiAgICBlc2J1aWxkT3B0aW9ucy5wdXNoKGAtLWxlZ2FsLWNvbW1lbnRzPSR7ZXNidWlsZENvbmZpZy5sZWdhbENvbW1lbnRzfWApO1xuICB9XG5cbiAgaWYoZXNidWlsZENvbmZpZy5zcGxpdHRpbmcgIT09IGZhbHNlKSB7XG4gICAgZXNidWlsZE9wdGlvbnMucHVzaCgnLS1zcGxpdHRpbmcnKTtcbiAgfVxuXG4gIGlmKGVzYnVpbGRDb25maWcubWV0YWZpbGUpIHtcbiAgICBlc2J1aWxkT3B0aW9ucy5wdXNoKCctLW1ldGFmaWxlJyk7XG4gIH1cblxuICBpZihlc2J1aWxkQ29uZmlnLmJhbm5lcikge1xuICAgIE9iamVjdC5lbnRyaWVzKGVzYnVpbGRDb25maWcuYmFubmVyKS5mb3JFYWNoKChbdHlwZSwgY29udGVudF0pID0+IHtcbiAgICAgIGVzYnVpbGRPcHRpb25zLnB1c2goYC0tYmFubmVyOiR7dHlwZX09JHtjb250ZW50fWApO1xuICAgIH0pO1xuICB9XG5cbiAgaWYoZXNidWlsZENvbmZpZy5mb290ZXIpIHtcbiAgICBPYmplY3QuZW50cmllcyhlc2J1aWxkQ29uZmlnLmZvb3RlcikuZm9yRWFjaCgoW3R5cGUsIGNvbnRlbnRdKSA9PiB7XG4gICAgICBlc2J1aWxkT3B0aW9ucy5wdXNoKGAtLWZvb3Rlcjoke3R5cGV9PSR7Y29udGVudH1gKTtcbiAgICB9KTtcbiAgfVxuXG4gIGlmKGVzYnVpbGRDb25maWcuZGVmaW5lKSB7XG4gICAgT2JqZWN0LmVudHJpZXMoZXNidWlsZENvbmZpZy5kZWZpbmUpLmZvckVhY2goKFtrZXksIHZhbHVlXSkgPT4ge1xuICAgICAgZXNidWlsZE9wdGlvbnMucHVzaChgLS1kZWZpbmU6JHtrZXl9PSR7dmFsdWV9YCk7XG4gICAgfSk7XG4gIH1cblxuICBpZihleHRlcm5hbC5sZW5ndGgpIHtcbiAgICBlc2J1aWxkT3B0aW9ucy5wdXNoKGAtLWV4dGVybmFsOiR7ZXh0ZXJuYWwuam9pbignLCcpfWApO1xuICB9XG5cbiAgaWYocGx1Z2lucy5sZW5ndGgpIHtcbiAgICBlc2J1aWxkT3B0aW9ucy5wdXNoKGAtLXBsdWdpbnM9JHtwbHVnaW5zLmpvaW4oJywnKX1gKTtcbiAgfVxuICBpZih3YXRjaCkge1xuICAgIGVzYnVpbGRPcHRpb25zLnB1c2goJy0td2F0Y2gnKTtcbiAgfVxuXG4gIHRyeSB7XG4gICAgYXdhaXQgZXhlY2EoZXNidWlsZFBhdGgsIGVzYnVpbGRPcHRpb25zLCB7ZW5jb2Rpbmc6ICd1dGY4J30pO1xuXG4gICAgc3Bpbm5lci5zdWNjZWVkKCdCdWlsZCBjb21wbGV0ZWQgc3VjY2Vzc2Z1bGx5IScpO1xuICAgIGRpc3BsYXlCdWlsZFN0YXR1cygnZXNidWlsZCcsIG91dHB1dERpciwgcXVpZXQpO1xuICB9IGNhdGNoKGVycm9yKSB7XG4gICAgbG9nKGBcXG4ke2NsaU5hbWV9IEVycm9yOiAke2Vycm9yLm1lc3NhZ2V9YCwgJ2Vycm9yJywgcXVpZXQpO1xuXG4gICAgaWYoIXF1aWV0KSB7XG4gICAgICBjb25zb2xlLmVycm9yKGVycm9yKTtcbiAgICB9XG5cbiAgICBzcGlubmVyLmZhaWwoJ0NvZGUgYnVpbGQgZmFpbGVkLicpO1xuXG4gICAgaWYoY29tbWFuZE9wdGlvbnMuYXNzaXN0KSB7XG4gICAgICBzcGlubmVyLnN0YXJ0KCdBSSBpcyBhbmFseXppbmcgdGhlIGVycm9yLi4uJyk7XG5cbiAgICAgIHRyeSB7XG4gICAgICAgIGF3YWl0IGFpRnVuY3Rpb24oe1xuICAgICAgICAgIHByb21wdDogYEZpeCB0aGlzIGVzYnVpbGQgZXJyb3I6ICR7ZXJyb3IubWVzc2FnZX1cXG5cXG5FcnJvciBkZXRhaWxzOlxcbiR7ZXJyb3IudG9TdHJpbmcoKX1gLFxuICAgICAgICAgIHRhc2s6ICdoZWxwJyxcbiAgICAgICAgICBjb250ZXh0OiB0cnVlLFxuICAgICAgICAgIHF1aWV0XG4gICAgICAgIH0pO1xuXG4gICAgICAgIHNwaW5uZXIuc3VjY2VlZCgnQUkgYW5hbHlzaXMgY29tcGxldGUnKTtcbiAgICAgIH0gY2F0Y2goYWlFcnJvcikge1xuICAgICAgICBzcGlubmVyLmZhaWwoJ0NvdWxkIG5vdCBnZW5lcmF0ZSBBSSBhc3Npc3RhbmNlJyk7XG4gICAgICAgIGlmKCFxdWlldCkge1xuICAgICAgICAgIGNvbnNvbGUuZXJyb3IoJ0FJIGFzc2lzdGFuY2UgZXJyb3I6JywgYWlFcnJvcik7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG5cbiAgICBjYWxsYmFjaygxKTtcbiAgICByZXR1cm4gMTtcbiAgfVxuXG4gIGNhbGxiYWNrKDApO1xuICByZXR1cm4gMDtcbn07XG5cbmV4cG9ydCBjb25zdCBidWlsZFdpdGhXZWJwYWNrID0gYXN5bmMgKHNwaW5uZXIsIGNtZCwgY2FsbGJhY2spID0+IHtcbiAgY29uc3Qge1xuICAgIGFuYWx5emUsXG4gICAgY2xpTmFtZSA9ICdMZXgnLFxuICAgIGNvbmZpZyxcbiAgICBjb25maWdOYW1lLFxuICAgIGRlZmluZVByb2Nlc3NFbnZOb2RlRW52LFxuICAgIGRldnRvb2wsXG4gICAgZGlzYWJsZUludGVycHJldCxcbiAgICBlbnRyeSxcbiAgICBlbnYsXG4gICAgZmFpbE9uV2FybmluZ3MsXG4gICAganNvbixcbiAgICBtZXJnZSxcbiAgICBtb2RlLFxuICAgIG5hbWUsXG4gICAgbm9kZUVudixcbiAgICBub0RldnRvb2wsXG4gICAgbm9TdGF0cyxcbiAgICBub1RhcmdldCxcbiAgICBub1dhdGNoLFxuICAgIG5vV2F0Y2hPcHRpb25zU3RkaW4sXG4gICAgb3V0cHV0UGF0aCxcbiAgICBxdWlldCA9IGZhbHNlLFxuICAgIHN0YXRzLFxuICAgIHRhcmdldCxcbiAgICB3YXRjaCxcbiAgICB3YXRjaE9wdGlvbnNTdGRpblxuICB9ID0gY21kO1xuXG4gIGNvbnNvbGUubG9nKCdlbnRyeTonLCBlbnRyeSwgJ3R5cGU6JywgdHlwZW9mIGVudHJ5KTtcbiAgY29uc29sZS5sb2coJ291dHB1dFBhdGg6Jywgb3V0cHV0UGF0aCwgJ3R5cGU6JywgdHlwZW9mIG91dHB1dFBhdGgpO1xuXG4gIGNvbnN0IGVudHJ5VmFsdWUgPSBBcnJheS5pc0FycmF5KGVudHJ5KSA/IGVudHJ5WzBdIDogZW50cnk7XG5cbiAgbGV0IHdlYnBhY2tDb25maWc6IHN0cmluZztcblxuICBpZihjb25maWcpIHtcbiAgICBjb25zdCBpc1JlbGF0aXZlQ29uZmlnOiBib29sZWFuID0gY29uZmlnLnN1YnN0cigwLCAyKSA9PT0gJy4vJztcbiAgICB3ZWJwYWNrQ29uZmlnID0gaXNSZWxhdGl2ZUNvbmZpZyA/IHBhdGhSZXNvbHZlKHByb2Nlc3MuY3dkKCksIGNvbmZpZykgOiBjb25maWc7XG4gIH0gZWxzZSB7XG4gICAgY29uc3QgcHJvamVjdENvbmZpZ1BhdGggPSBwYXRoUmVzb2x2ZShwcm9jZXNzLmN3ZCgpLCAnd2VicGFjay5jb25maWcuanMnKTtcbiAgICBjb25zdCBwcm9qZWN0Q29uZmlnUGF0aFRzID0gcGF0aFJlc29sdmUocHJvY2Vzcy5jd2QoKSwgJ3dlYnBhY2suY29uZmlnLnRzJyk7XG4gICAgY29uc3QgaGFzUHJvamVjdENvbmZpZyA9IGV4aXN0c1N5bmMocHJvamVjdENvbmZpZ1BhdGgpIHx8IGV4aXN0c1N5bmMocHJvamVjdENvbmZpZ1BhdGhUcyk7XG5cbiAgICBpZihoYXNQcm9qZWN0Q29uZmlnKSB7XG4gICAgICB3ZWJwYWNrQ29uZmlnID0gZXhpc3RzU3luYyhwcm9qZWN0Q29uZmlnUGF0aFRzKSA/IHByb2plY3RDb25maWdQYXRoVHMgOiBwcm9qZWN0Q29uZmlnUGF0aDtcbiAgICB9IGVsc2Uge1xuICAgICAgY29uc3Qge3dlYnBhY2tDb25maWc6IHJlc29sdmVkQ29uZmlnfSA9IHJlc29sdmVXZWJwYWNrUGF0aHMoY3VycmVudERpcm5hbWUpO1xuICAgICAgd2VicGFja0NvbmZpZyA9IHJlc29sdmVkQ29uZmlnO1xuICAgIH1cbiAgfVxuXG4gIGNvbnNvbGUubG9nKCd3ZWJwYWNrQ29uZmlnIHBhdGg6Jywgd2VicGFja0NvbmZpZyk7XG4gIGNvbnNvbGUubG9nKCd3ZWJwYWNrQ29uZmlnIGV4aXN0czonLCBleGlzdHNTeW5jKHdlYnBhY2tDb25maWcpKTtcblxuICBpZighZXhpc3RzU3luYyh3ZWJwYWNrQ29uZmlnKSkge1xuICAgIGNvbnN0IGxleFBhY2thZ2VQYXRoID0gZ2V0TGV4UGFja2FnZUpzb25QYXRoKCk7XG4gICAgY29uc3QgbGV4UGFja2FnZURpciA9IGRpcm5hbWUobGV4UGFja2FnZVBhdGgpO1xuICAgIGNvbnN0IGxleFdlYnBhY2tDb25maWcgPSBwYXRoUmVzb2x2ZShsZXhQYWNrYWdlRGlyLCAnd2VicGFjay5jb25maWcuanMnKTtcblxuICAgIGlmKGV4aXN0c1N5bmMobGV4V2VicGFja0NvbmZpZykpIHtcbiAgICAgIHdlYnBhY2tDb25maWcgPSBsZXhXZWJwYWNrQ29uZmlnO1xuICAgICAgY29uc29sZS5sb2coJ1VzaW5nIExleCB3ZWJwYWNrIGNvbmZpZzonLCB3ZWJwYWNrQ29uZmlnKTtcbiAgICB9IGVsc2Uge1xuICAgICAgbG9nKGBcXG4ke2NsaU5hbWV9IEVycm9yOiBDb3VsZCBub3QgZmluZCB3ZWJwYWNrLmNvbmZpZy5qc2AsICdlcnJvcicsIHF1aWV0KTtcbiAgICAgIHNwaW5uZXIuZmFpbCgnQnVpbGQgZmFpbGVkLicpO1xuICAgICAgY2FsbGJhY2soMSk7XG4gICAgICByZXR1cm4gMTtcbiAgICB9XG4gIH1cblxuICBjb25zdCB3ZWJwYWNrT3B0aW9uczogc3RyaW5nW10gPSBbXG4gICAgJy0tY29sb3InLFxuICAgICctLXByb2dyZXNzJyxcbiAgICAnLS1jb25maWcnLCB3ZWJwYWNrQ29uZmlnXG4gIF07XG5cbiAgaWYoYW5hbHl6ZSkgd2VicGFja09wdGlvbnMucHVzaCgnLS1hbmFseXplJyk7XG4gIGlmKGNvbmZpZ05hbWUpIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tY29uZmlnTmFtZScsIGNvbmZpZ05hbWUpO1xuICBpZihkZWZpbmVQcm9jZXNzRW52Tm9kZUVudikgd2VicGFja09wdGlvbnMucHVzaCgnLS1kZWZpbmVQcm9jZXNzRW52Tm9kZUVudicsIGRlZmluZVByb2Nlc3NFbnZOb2RlRW52KTtcbiAgaWYoZGV2dG9vbCkgd2VicGFja09wdGlvbnMucHVzaCgnLS1kZXZ0b29sJywgZGV2dG9vbCk7XG4gIGlmKGRpc2FibGVJbnRlcnByZXQpIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tZGlzYWJsZUludGVycHJldCcpO1xuICAvLyBQYXNzIGVudHJ5IGRpcmVjdGx5IGFzIENMSSBmbGFnXG4gIGlmKGVudHJ5VmFsdWUpIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tZW50cnknLCBlbnRyeVZhbHVlLnRvU3RyaW5nKCkpO1xuICBpZihlbnYpIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tZW52JywgZW52KTtcbiAgaWYoZmFpbE9uV2FybmluZ3MpIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tZmFpbE9uV2FybmluZ3MnKTtcbiAgaWYoanNvbikgd2VicGFja09wdGlvbnMucHVzaCgnLS1qc29uJywganNvbik7XG4gIGlmKG1vZGUpIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tbW9kZScsIG1vZGUpO1xuICBpZihtZXJnZSkgd2VicGFja09wdGlvbnMucHVzaCgnLS1tZXJnZScpO1xuICBpZihuYW1lKSB3ZWJwYWNrT3B0aW9ucy5wdXNoKCctLW5hbWUnLCBuYW1lKTtcbiAgaWYobm9EZXZ0b29sKSB3ZWJwYWNrT3B0aW9ucy5wdXNoKCctLW5vRGV2dG9vbCcpO1xuICBpZihub1N0YXRzKSB3ZWJwYWNrT3B0aW9ucy5wdXNoKCctLW5vU3RhdHMnKTtcbiAgaWYobm9UYXJnZXQpIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tbm9UYXJnZXQnKTtcbiAgaWYobm9XYXRjaCkgd2VicGFja09wdGlvbnMucHVzaCgnLS1ub1dhdGNoJyk7XG4gIGlmKG5vV2F0Y2hPcHRpb25zU3RkaW4pIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tbm9XYXRjaE9wdGlvbnNTdGRpbicpO1xuICBpZihub2RlRW52KSB3ZWJwYWNrT3B0aW9ucy5wdXNoKCctLW5vZGVFbnYnLCBub2RlRW52KTtcbiAgaWYob3V0cHV0UGF0aCkgd2VicGFja09wdGlvbnMucHVzaCgnLS1vdXRwdXQtcGF0aCcsIG91dHB1dFBhdGgudG9TdHJpbmcoKSk7IC8vIENvbnZlcnQgdG8gc3RyaW5nXG4gIGlmKHN0YXRzKSB3ZWJwYWNrT3B0aW9ucy5wdXNoKCctLXN0YXRzJywgc3RhdHMpO1xuICBpZih0YXJnZXQpIHdlYnBhY2tPcHRpb25zLnB1c2goJy0tdGFyZ2V0JywgdGFyZ2V0KTtcbiAgaWYod2F0Y2gpIHdlYnBhY2tPcHRpb25zLnB1c2goJy0td2F0Y2gnKTtcbiAgaWYod2F0Y2hPcHRpb25zU3RkaW4pIHdlYnBhY2tPcHRpb25zLnB1c2goJy0td2F0Y2hPcHRpb25zU3RkaW4nKTtcblxuICB0cnkge1xuICAgIGNvbnN0IHt3ZWJwYWNrUGF0aH0gPSByZXNvbHZlV2VicGFja1BhdGhzKGN1cnJlbnREaXJuYW1lKTtcblxuICAgIC8vIEZpeDogRW5zdXJlIGZpbmFsV2VicGFja09wdGlvbnMgaXMgYWx3YXlzIGFuIGFycmF5IG9mIHN0cmluZ3NcbiAgICBsZXQgZmluYWxXZWJwYWNrT3B0aW9uczogc3RyaW5nW107XG4gICAgaWYod2VicGFja1BhdGggPT09ICducHgnKSB7XG4gICAgICBmaW5hbFdlYnBhY2tPcHRpb25zID0gWyd3ZWJwYWNrJywgLi4ud2VicGFja09wdGlvbnNdO1xuICAgIH0gZWxzZSB7XG4gICAgICBmaW5hbFdlYnBhY2tPcHRpb25zID0gWy4uLndlYnBhY2tPcHRpb25zXTtcbiAgICB9XG5cbiAgICBjb25zb2xlLmxvZygnd2VicGFja1BhdGg6Jywgd2VicGFja1BhdGgpO1xuICAgIGNvbnNvbGUubG9nKCdmaW5hbFdlYnBhY2tPcHRpb25zOicsIEpTT04uc3RyaW5naWZ5KGZpbmFsV2VicGFja09wdGlvbnMpKTtcbiAgICBjb25zb2xlLmxvZygnZmluYWxXZWJwYWNrT3B0aW9ucyB0eXBlOicsIEFycmF5LmlzQXJyYXkoZmluYWxXZWJwYWNrT3B0aW9ucykgPyAnQXJyYXknIDogdHlwZW9mIGZpbmFsV2VicGFja09wdGlvbnMpO1xuXG4gICAgLy8gTWFrZSBzdXJlIHdlJ3JlIHBhc3NpbmcgYW4gYXJyYXkgb2Ygc3RyaW5ncyB0byBleGVjYVxuICAgIGNvbnN0IGNoaWxkUHJvY2VzcyA9IGV4ZWNhKHdlYnBhY2tQYXRoLCBmaW5hbFdlYnBhY2tPcHRpb25zLCB7ZW5jb2Rpbmc6ICd1dGY4Jywgc3RkaW86ICdwaXBlJ30pO1xuXG4gICAgbGV0IGJ1aWxkQ29tcGxldGVkID0gZmFsc2U7XG4gICAgbGV0IGJ1aWxkU3RhdHMgPSB7XG4gICAgICBtb2R1bGVzOiAwLFxuICAgICAgYXNzZXRzOiAwLFxuICAgICAgc2l6ZTogJzAgQidcbiAgICB9O1xuXG4gICAgY2hpbGRQcm9jZXNzLnN0ZG91dD8ub24oJ2RhdGEnLCAoZGF0YTogQnVmZmVyKSA9PiB7XG4gICAgICBjb25zdCBvdXRwdXQgPSBkYXRhLnRvU3RyaW5nKCk7XG5cbiAgICAgIGhhbmRsZVdlYnBhY2tQcm9ncmVzcyhvdXRwdXQsIHNwaW5uZXIsIHF1aWV0LCAnXHVEODNDXHVERkQ3XHVGRTBGJywgJ1dlYnBhY2sgQnVpbGRpbmcnKTtcblxuICAgICAgaWYoIWJ1aWxkQ29tcGxldGVkICYmIG91dHB1dC5pbmNsdWRlcygnY29tcGlsZWQgc3VjY2Vzc2Z1bGx5JykpIHtcbiAgICAgICAgYnVpbGRDb21wbGV0ZWQgPSB0cnVlO1xuICAgICAgICBzcGlubmVyLnN1Y2NlZWQoJ0J1aWxkIGNvbXBsZXRlZCBzdWNjZXNzZnVsbHkhJyk7XG5cbiAgICAgICAgY29uc3QgbW9kdWxlTWF0Y2ggPSBvdXRwdXQubWF0Y2goLyhcXGQrKSBtb2R1bGVzLyk7XG4gICAgICAgIGNvbnN0IGFzc2V0TWF0Y2ggPSBvdXRwdXQubWF0Y2goLyhcXGQrKSBhc3NldHMvKTtcbiAgICAgICAgY29uc3Qgc2l6ZU1hdGNoID0gb3V0cHV0Lm1hdGNoKC9hc3NldHMgYnkgc3RhdHVzIChbXFxkLl0rIFxcdyspLykgfHwgb3V0cHV0Lm1hdGNoKC9hc3NldHMgYnkgcGF0aC4qPyhbXFxkLl0rIFxcdyspLyk7XG5cbiAgICAgICAgaWYobW9kdWxlTWF0Y2gpIGJ1aWxkU3RhdHMubW9kdWxlcyA9IHBhcnNlSW50KG1vZHVsZU1hdGNoWzFdLCAxMCk7XG4gICAgICAgIGlmKGFzc2V0TWF0Y2gpIGJ1aWxkU3RhdHMuYXNzZXRzID0gcGFyc2VJbnQoYXNzZXRNYXRjaFsxXSwgMTApO1xuICAgICAgICBpZihzaXplTWF0Y2gpIGJ1aWxkU3RhdHMuc2l6ZSA9IHNpemVNYXRjaFsxXTtcblxuICAgICAgICBkaXNwbGF5QnVpbGRTdGF0dXMoJ3dlYnBhY2snLCBMZXhDb25maWcuY29uZmlnLm91dHB1dEZ1bGxQYXRoIHx8ICdsaWInLCBxdWlldCwgYnVpbGRTdGF0cyk7XG4gICAgICB9XG4gICAgfSk7XG5cbiAgICBjaGlsZFByb2Nlc3Muc3RkZXJyPy5vbignZGF0YScsIChkYXRhOiBCdWZmZXIpID0+IHtcbiAgICAgIGNvbnN0IG91dHB1dCA9IGRhdGEudG9TdHJpbmcoKTtcblxuICAgICAgaGFuZGxlV2VicGFja1Byb2dyZXNzKG91dHB1dCwgc3Bpbm5lciwgcXVpZXQsICdcdUQ4M0NcdURGRDdcdUZFMEYnLCAnV2VicGFjayBCdWlsZGluZycpO1xuXG4gICAgICBpZighYnVpbGRDb21wbGV0ZWQgJiYgb3V0cHV0LmluY2x1ZGVzKCdjb21waWxlZCBzdWNjZXNzZnVsbHknKSkge1xuICAgICAgICBidWlsZENvbXBsZXRlZCA9IHRydWU7XG4gICAgICAgIHNwaW5uZXIuc3VjY2VlZCgnQnVpbGQgY29tcGxldGVkIHN1Y2Nlc3NmdWxseSEnKTtcblxuICAgICAgICBjb25zdCBtb2R1bGVNYXRjaCA9IG91dHB1dC5tYXRjaCgvKFxcZCspIG1vZHVsZXMvKTtcbiAgICAgICAgY29uc3QgYXNzZXRNYXRjaCA9IG91dHB1dC5tYXRjaCgvKFxcZCspIGFzc2V0cy8pO1xuICAgICAgICBjb25zdCBzaXplTWF0Y2ggPSBvdXRwdXQubWF0Y2goL2Fzc2V0cyBieSBzdGF0dXMgKFtcXGQuXSsgXFx3KykvKSB8fCBvdXRwdXQubWF0Y2goL2Fzc2V0cyBieSBwYXRoLio/KFtcXGQuXSsgXFx3KykvKTtcblxuICAgICAgICBpZihtb2R1bGVNYXRjaCkgYnVpbGRTdGF0cy5tb2R1bGVzID0gcGFyc2VJbnQobW9kdWxlTWF0Y2hbMV0sIDEwKTtcbiAgICAgICAgaWYoYXNzZXRNYXRjaCkgYnVpbGRTdGF0cy5hc3NldHMgPSBwYXJzZUludChhc3NldE1hdGNoWzFdLCAxMCk7XG4gICAgICAgIGlmKHNpemVNYXRjaCkgYnVpbGRTdGF0cy5zaXplID0gc2l6ZU1hdGNoWzFdO1xuXG4gICAgICAgIGRpc3BsYXlCdWlsZFN0YXR1cygnd2VicGFjaycsIExleENvbmZpZy5jb25maWcub3V0cHV0RnVsbFBhdGggfHwgJ2xpYicsIHF1aWV0LCBidWlsZFN0YXRzKTtcbiAgICAgIH1cbiAgICB9KTtcblxuICAgIGF3YWl0IGNoaWxkUHJvY2VzcztcblxuICAgIGlmKCFidWlsZENvbXBsZXRlZCkge1xuICAgICAgc3Bpbm5lci5zdWNjZWVkKCdCdWlsZCBjb21wbGV0ZWQgc3VjY2Vzc2Z1bGx5IScpO1xuICAgICAgZGlzcGxheUJ1aWxkU3RhdHVzKCd3ZWJwYWNrJywgTGV4Q29uZmlnLmNvbmZpZy5vdXRwdXRGdWxsUGF0aCB8fCAnbGliJywgcXVpZXQsIGJ1aWxkU3RhdHMpO1xuICAgIH1cblxuICAgIGNhbGxiYWNrKDApO1xuICAgIHJldHVybiAwO1xuICB9IGNhdGNoKGVycm9yKSB7XG4gICAgbG9nKGBcXG4ke2NsaU5hbWV9IEVycm9yOiAke2Vycm9yLm1lc3NhZ2V9YCwgJ2Vycm9yJywgcXVpZXQpO1xuXG4gICAgc3Bpbm5lci5mYWlsKCdCdWlsZCBmYWlsZWQuJyk7XG5cbiAgICBpZihjbWQuYXNzaXN0KSB7XG4gICAgICBzcGlubmVyLnN0YXJ0KCdBSSBpcyBhbmFseXppbmcgdGhlIHdlYnBhY2sgZXJyb3IuLi4nKTtcblxuICAgICAgdHJ5IHtcbiAgICAgICAgYXdhaXQgYWlGdW5jdGlvbih7XG4gICAgICAgICAgcHJvbXB0OiBgRml4IHRoaXMgd2VicGFjayBidWlsZCBlcnJvcjogJHtlcnJvci5tZXNzYWdlfVxcblxcbkVycm9yIGRldGFpbHM6XFxuJHtlcnJvci50b1N0cmluZygpfVxcblxcbkNvbmZpZ3VyYXRpb24gdXNlZDpcXG4ke0pTT04uc3RyaW5naWZ5KHdlYnBhY2tPcHRpb25zLCBudWxsLCAyKX1gLFxuICAgICAgICAgIHRhc2s6ICdoZWxwJyxcbiAgICAgICAgICBjb250ZXh0OiB0cnVlLFxuICAgICAgICAgIHF1aWV0XG4gICAgICAgIH0pO1xuXG4gICAgICAgIHNwaW5uZXIuc3VjY2VlZCgnQUkgYW5hbHlzaXMgY29tcGxldGUnKTtcbiAgICAgIH0gY2F0Y2goYWlFcnJvcikge1xuICAgICAgICBzcGlubmVyLmZhaWwoJ0NvdWxkIG5vdCBnZW5lcmF0ZSBBSSBhc3Npc3RhbmNlJyk7XG4gICAgICAgIGlmKCFxdWlldCkge1xuICAgICAgICAgIGNvbnNvbGUuZXJyb3IoJ0FJIGFzc2lzdGFuY2UgZXJyb3I6JywgYWlFcnJvcik7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG5cbiAgICBjYWxsYmFjaygxKTtcbiAgICByZXR1cm4gMTtcbiAgfVxufTtcblxuZXhwb3J0IGNvbnN0IGJ1aWxkID0gYXN5bmMgKGNtZDogQnVpbGRPcHRpb25zLCBjYWxsYmFjazogQnVpbGRDYWxsYmFjayA9ICgpID0+ICh7fSkpOiBQcm9taXNlPG51bWJlcj4gPT4ge1xuICBjb25zdCB7XG4gICAgYnVuZGxlciA9ICd3ZWJwYWNrJyxcbiAgICBjbGlOYW1lID0gJ0xleCcsXG4gICAgcXVpZXQgPSBmYWxzZSxcbiAgICByZW1vdmUgPSBmYWxzZSxcbiAgICB0ZXN0ID0gZmFsc2UsXG4gICAgdHJhbnNsYXRpb25zID0gZmFsc2UsXG4gICAgdmFyaWFibGVzID0gJ3t9J1xuICB9ID0gY21kO1xuXG4gIGNvbnN0IHNwaW5uZXIgPSBjcmVhdGVTcGlubmVyKHF1aWV0KTtcblxuICBsb2coYCR7Y2xpTmFtZX0gYnVpbGRpbmcuLi5gLCAnaW5mbycsIHF1aWV0KTtcblxuICBhd2FpdCBMZXhDb25maWcucGFyc2VDb25maWcoY21kKTtcblxuICBjb25zdCB7b3V0cHV0RnVsbFBhdGgsIHVzZVR5cGVzY3JpcHR9ID0gTGV4Q29uZmlnLmNvbmZpZztcblxuICBjaGVja0xpbmtlZE1vZHVsZXMoKTtcblxuICBsZXQgdmFyaWFibGVzT2JqOiBvYmplY3QgPSB7Tk9ERV9FTlY6ICdwcm9kdWN0aW9uJ307XG5cbiAgaWYodmFyaWFibGVzKSB7XG4gICAgdHJ5IHtcbiAgICAgIHZhcmlhYmxlc09iaiA9IEpTT04ucGFyc2UodmFyaWFibGVzKTtcbiAgICB9IGNhdGNoKGVycm9yKSB7XG4gICAgICBsb2coYFxcbiR7Y2xpTmFtZX0gRXJyb3I6IEVudmlyb25tZW50IHZhcmlhYmxlcyBvcHRpb24gaXMgbm90IGEgdmFsaWQgSlNPTiBvYmplY3QuYCwgJ2Vycm9yJywgcXVpZXQpO1xuXG4gICAgICBjYWxsYmFjaygxKTtcbiAgICAgIHJldHVybiAxO1xuICAgIH1cbiAgfVxuXG4gIHByb2Nlc3MuZW52ID0gey4uLnByb2Nlc3MuZW52LCAuLi52YXJpYWJsZXNPYmp9O1xuXG4gIC8vIElmIGluIHRlc3QgbW9kZSwgZXhpdCBlYXJseVxuICBpZih0ZXN0KSB7XG4gICAgbG9nKCdUZXN0IG1vZGU6IEJ1aWxkIGVudmlyb25tZW50IGxvYWRlZCwgZXhpdGluZycsICdpbmZvJywgcXVpZXQpO1xuICAgIGNhbGxiYWNrKDApO1xuICAgIHJldHVybiAwO1xuICB9XG5cbiAgLy8gUHJvY2VzcyB0cmFuc2xhdGlvbnMgaWYgZmxhZyBpcyBlbmFibGVkIChiZWZvcmUgYnVpbGRpbmcpXG4gIGlmKHRyYW5zbGF0aW9ucykge1xuICAgIHNwaW5uZXIuc3RhcnQoJ1Byb2Nlc3NpbmcgdHJhbnNsYXRpb25zLi4uJyk7XG5cbiAgICB0cnkge1xuICAgICAgY29uc3Qgc291cmNlUGF0aCA9IExleENvbmZpZy5jb25maWcuc291cmNlRnVsbFBhdGggfHwgcHJvY2Vzcy5jd2QoKTtcbiAgICAgIGNvbnN0IG91dHB1dFBhdGggPSBMZXhDb25maWcuY29uZmlnLm91dHB1dEZ1bGxQYXRoIHx8ICdsaWInO1xuXG4gICAgICBhd2FpdCBwcm9jZXNzVHJhbnNsYXRpb25zKHNvdXJjZVBhdGgsIG91dHB1dFBhdGgsIHF1aWV0KTtcbiAgICAgIHNwaW5uZXIuc3VjY2VlZCgnVHJhbnNsYXRpb25zIHByb2Nlc3NlZCBzdWNjZXNzZnVsbHkhJyk7XG4gICAgfSBjYXRjaCh0cmFuc2xhdGlvbkVycm9yKSB7XG4gICAgICBsb2coYFxcbiR7Y2xpTmFtZX0gRXJyb3I6IEZhaWxlZCB0byBwcm9jZXNzIHRyYW5zbGF0aW9uczogJHt0cmFuc2xhdGlvbkVycm9yLm1lc3NhZ2V9YCwgJ2Vycm9yJywgcXVpZXQpO1xuICAgICAgc3Bpbm5lci5mYWlsKCdGYWlsZWQgdG8gcHJvY2VzcyB0cmFuc2xhdGlvbnMuJyk7XG4gICAgICBjYWxsYmFjaygxKTtcbiAgICAgIHJldHVybiAxO1xuICAgIH1cbiAgfVxuXG4gIHNwaW5uZXIuc3RhcnQoJ0J1aWxkaW5nIGNvZGUuLi4nKTtcblxuICBpZihyZW1vdmUpIHtcbiAgICBhd2FpdCByZW1vdmVGaWxlcyhvdXRwdXRGdWxsUGF0aCB8fCAnJyk7XG4gIH1cblxuICBpZih1c2VUeXBlc2NyaXB0KSB7XG4gICAgY29uc3QgY29tcGlsZUNvbmZpZ1BhdGggPSBnZXRUeXBlU2NyaXB0Q29uZmlnUGF0aCgndHNjb25maWcuYnVpbGQuanNvbicpO1xuICAgIGlmKGV4aXN0c1N5bmMoY29tcGlsZUNvbmZpZ1BhdGgpKSB7XG4gICAgICBsb2coJ1VzaW5nIHRzY29uZmlnLmJ1aWxkLmpzb24gZm9yIGJ1aWxkLi4uJywgJ2luZm8nLCBxdWlldCk7XG4gICAgfSBlbHNlIHtcbiAgICAgIExleENvbmZpZy5jaGVja0NvbXBpbGVUeXBlc2NyaXB0Q29uZmlnKCk7XG4gICAgfVxuICB9XG5cbiAgbGV0IGJ1aWxkUmVzdWx0ID0gMDtcblxuICBpZihidW5kbGVyID09PSAnZXNidWlsZCcpIHtcbiAgICBidWlsZFJlc3VsdCA9IGF3YWl0IGJ1aWxkV2l0aEVzQnVpbGQoc3Bpbm5lciwgY21kLCAoc3RhdHVzKSA9PiB7XG4gICAgICBidWlsZFJlc3VsdCA9IHN0YXR1cztcbiAgICB9KTtcbiAgfSBlbHNlIHtcbiAgICBidWlsZFJlc3VsdCA9IGF3YWl0IGJ1aWxkV2l0aFdlYnBhY2soc3Bpbm5lciwgY21kLCAoc3RhdHVzKSA9PiB7XG4gICAgICBidWlsZFJlc3VsdCA9IHN0YXR1cztcbiAgICB9KTtcbiAgfVxuXG4gIGlmKGJ1aWxkUmVzdWx0ID09PSAwICYmIGNtZC5hbmFseXplKSB7XG4gICAgc3Bpbm5lci5zdGFydCgnQUkgaXMgYW5hbHl6aW5nIHRoZSBidWlsZCBvdXRwdXQgZm9yIG9wdGltaXphdGlvbiBvcHBvcnR1bml0aWVzLi4uJyk7XG5cbiAgICB0cnkge1xuICAgICAgY29uc3Qgc3RhdHMgPSB7XG4gICAgICAgIG91dHB1dFBhdGg6IExleENvbmZpZy5jb25maWcub3V0cHV0RnVsbFBhdGgsXG4gICAgICAgIGVudHJ5UG9pbnRzOiBidW5kbGVyID09PSAnZXNidWlsZCcgP1xuICAgICAgICAgIGBTb3VyY2UgZmlsZXM6ICR7TGV4Q29uZmlnLmNvbmZpZy5zb3VyY2VGdWxsUGF0aH0vKiovKi57dHMsanN9YCA6XG4gICAgICAgICAgTGV4Q29uZmlnLmNvbmZpZy53ZWJwYWNrPy5lbnRyeSB8fCAnVW5rbm93biBlbnRyeSBwb2ludHMnXG4gICAgICB9O1xuXG4gICAgICBhd2FpdCBhaUZ1bmN0aW9uKHtcbiAgICAgICAgcHJvbXB0OiBgQW5hbHl6ZSB0aGlzIGJ1aWxkIGZvciBvcHRpbWl6YXRpb24gb3Bwb3J0dW5pdGllczpcblxuQnVpbGQgVHlwZTogJHtidW5kbGVyfVxuRm9ybWF0OiAke2NtZC5mb3JtYXQgfHwgJ2RlZmF1bHQnfVxuRW52aXJvbm1lbnQ6ICR7TGV4Q29uZmlnLmNvbmZpZy50YXJnZXRFbnZpcm9ubWVudH1cbiR7SlNPTi5zdHJpbmdpZnkoc3RhdHMsIG51bGwsIDIpfVxuXG5XaGF0IGFyZSB0aGUga2V5IG9wdGltaXphdGlvbiBvcHBvcnR1bml0aWVzIGZvciB0aGlzIGJ1aWxkIGNvbmZpZ3VyYXRpb24/IENvbnNpZGVyOlxuMS4gQnVuZGxlIHNpemUgb3B0aW1pemF0aW9uIHN0cmF0ZWdpZXNcbjIuIENvZGUgc3BsaXR0aW5nIHJlY29tbWVuZGF0aW9uc1xuMy4gVHJlZS1zaGFraW5nIGltcHJvdmVtZW50c1xuNC4gUGVyZm9ybWFuY2UgZW5oYW5jZW1lbnRzXG41LiBEZXBlbmRlbmN5IG9wdGltaXphdGlvbnNgLFxuICAgICAgICB0YXNrOiAnb3B0aW1pemUnLFxuICAgICAgICBjb250ZXh0OiB0cnVlLFxuICAgICAgICBxdWlldFxuICAgICAgfSk7XG5cbiAgICAgIHNwaW5uZXIuc3VjY2VlZCgnQUkgYnVpbGQgYW5hbHlzaXMgY29tcGxldGUnKTtcbiAgICB9IGNhdGNoKGFpRXJyb3IpIHtcbiAgICAgIHNwaW5uZXIuZmFpbCgnQ291bGQgbm90IGdlbmVyYXRlIEFJIG9wdGltaXphdGlvbiBhbmFseXNpcycpO1xuICAgICAgaWYoIXF1aWV0KSB7XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoJ0FJIGFuYWx5c2lzIGVycm9yOicsIGFpRXJyb3IpO1xuICAgICAgfVxuICAgIH1cbiAgfVxuXG4gIGlmKGJ1aWxkUmVzdWx0ID09PSAwKSB7XG4gICAgdHJ5IHtcbiAgICAgIGF3YWl0IGNvcHlDb25maWd1cmVkRmlsZXMoc3Bpbm5lciwgTGV4Q29uZmlnLmNvbmZpZywgcXVpZXQpO1xuICAgIH0gY2F0Y2goY29weUVycm9yKSB7XG4gICAgICBsb2coYFxcbiR7Y2xpTmFtZX0gRXJyb3I6IEZhaWxlZCB0byBjb3B5IGNvbmZpZ3VyZWQgZmlsZXM6ICR7Y29weUVycm9yLm1lc3NhZ2V9YCwgJ2Vycm9yJywgcXVpZXQpO1xuICAgICAgY2FsbGJhY2soMSk7XG4gICAgICByZXR1cm4gMTtcbiAgICB9XG4gIH1cblxuICBjYWxsYmFjayhidWlsZFJlc3VsdCk7XG4gIHJldHVybiBidWlsZFJlc3VsdDtcbn07XG5cbmV4cG9ydCBkZWZhdWx0IGJ1aWxkO1xuIl0sCiAgIm1hcHBpbmdzIjogIkFBSUEsT0FBTyx5QkFBeUI7QUFDaEMsU0FBUSxhQUFZO0FBQ3BCLFNBQVEsWUFBWSxvQkFBbUI7QUFDdkMsU0FBUSxRQUFRLGdCQUFlO0FBQy9CLFNBQVEsU0FBUyxXQUFXLG1CQUFrQjtBQUU5QyxTQUFRLFdBQVcsK0JBQThCO0FBQ2pELFNBQVEsb0JBQW9CLHFCQUFxQixlQUFrQyx1QkFBdUIsbUJBQWtCO0FBQzVILFNBQVEsWUFBWSxrQkFBa0IscUJBQXFCLDZCQUE0QjtBQUN2RixTQUFRLFdBQVU7QUFDbEIsU0FBUSwyQkFBMEI7QUFDbEMsU0FBUSxrQkFBaUI7QUFDekIsT0FBTyxXQUFXO0FBQ2xCLE9BQU8sV0FBVztBQUVsQixJQUFJO0FBQ0osSUFBSTtBQUVKLElBQUk7QUFDRixvQkFBa0IsS0FBSywrQ0FBK0M7QUFDdEUsbUJBQWlCLFFBQVEsZUFBZTtBQUMxQyxRQUFRO0FBQ04sb0JBQWtCLFFBQVEsSUFBSTtBQUM5QixtQkFBaUIsUUFBUSxJQUFJO0FBQy9CO0FBcUJBLE1BQU0scUJBQXFCLENBQUMsU0FBaUIsWUFBb0IsT0FBZ0IsVUFBK0Q7QUFDOUksTUFBRyxNQUFPO0FBRVYsTUFBSSxZQUFZO0FBQ2hCLE1BQUcsU0FBUyxNQUFNLFdBQVcsTUFBTSxRQUFRO0FBQ3pDLGdCQUFZO0FBQUEsRUFBSyxNQUFNLE1BQU0sVUFBVSxDQUFDLE9BQU8sTUFBTSxLQUFLLE1BQU0sT0FBTyxDQUFDO0FBQUEsRUFDbkUsTUFBTSxNQUFNLFNBQVMsQ0FBQyxRQUFRLE1BQU0sS0FBSyxNQUFNLE1BQU0sQ0FBQztBQUFBLEVBQ3RELE1BQU0sTUFBTSxPQUFPLENBQUMsVUFBVSxNQUFNLEtBQUssTUFBTSxJQUFJ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