@nlabs/lex
Version:
287 lines (285 loc) • 33.4 kB
JavaScript
import boxen from "boxen";
import chalk from "chalk";
import { execa } from "execa";
import { existsSync, readFileSync, writeFileSync, mkdirSync } from "fs";
import https from "https";
import { networkInterfaces, homedir } from "os";
import { dirname, resolve as pathResolve, join } from "path";
import { LexConfig } from "../../LexConfig.js";
import { createSpinner, handleWebpackProgress, removeFiles } from "../../utils/app.js";
import { resolveWebpackPaths } from "../../utils/file.js";
import { log } from "../../utils/log.js";
import { processTranslations } from "../../utils/translations.js";
let currentFilename;
let currentDirname;
try {
currentFilename = eval('require("url").fileURLToPath(import.meta.url)');
currentDirname = dirname(currentFilename);
} catch {
currentFilename = process.cwd();
currentDirname = process.cwd();
}
const getCacheDir = () => {
const cacheDir = join(homedir(), ".lex-cache");
if (!existsSync(cacheDir)) {
mkdirSync(cacheDir, { recursive: true });
}
return cacheDir;
};
const getCachePath = () => join(getCacheDir(), "public-ip.json");
const readPublicIpCache = () => {
const cachePath = getCachePath();
if (!existsSync(cachePath)) {
return null;
}
try {
const cacheData = readFileSync(cachePath, "utf8");
const cache = JSON.parse(cacheData);
const oneWeekMs = 7 * 24 * 60 * 60 * 1e3;
if (Date.now() - cache.timestamp > oneWeekMs) {
return null;
}
return cache;
} catch {
return null;
}
};
const writePublicIpCache = (ip) => {
const cachePath = getCachePath();
const cache = {
ip,
timestamp: Date.now()
};
writeFileSync(cachePath, JSON.stringify(cache, null, 2));
};
const fetchPublicIp = (forceRefresh = false) => new Promise((resolve) => {
if (!forceRefresh) {
const cached = readPublicIpCache();
if (cached) {
resolve(cached.ip);
return;
}
}
https.get("https://api.ipify.org", (res) => {
let data = "";
res.on("data", (chunk) => data += chunk);
res.on("end", () => {
const ip = data.trim();
if (ip) {
writePublicIpCache(ip);
}
resolve(ip);
});
}).on("error", () => resolve(void 0));
});
const getNetworkAddresses = () => {
const interfaces = networkInterfaces();
const addresses = {
local: "localhost",
private: null,
public: null
};
for (const name of Object.keys(interfaces)) {
const networkInterface = interfaces[name];
if (!networkInterface) {
continue;
}
for (const iface of networkInterface) {
if (iface.family === "IPv4" && !iface.internal) {
const ip = iface.address;
if (ip.startsWith("10.") || ip.startsWith("192.168.") || ip.startsWith("172.")) {
if (!addresses.private) {
addresses.private = ip;
}
} else {
if (!addresses.public) {
addresses.public = ip;
}
}
}
}
}
return addresses;
};
const displayServerStatus = (port = 7001, quiet, publicIp) => {
if (quiet) {
return;
}
const addresses = getNetworkAddresses();
const localUrl = `http://localhost:${port}`;
const privateUrl = addresses.private ? `http://${addresses.private}:${port}` : null;
let publicUrl = null;
if (publicIp) {
publicUrl = `http://${publicIp}:${port}`;
} else if (addresses.public) {
publicUrl = `http://${addresses.public}:${port}`;
}
let urlLines = `${chalk.green("Local:")} ${chalk.underline(localUrl)}
`;
if (privateUrl) {
urlLines += `${chalk.green("Private:")} ${chalk.underline(privateUrl)}
`;
}
if (publicUrl) {
urlLines += `${chalk.green("Public:")} ${chalk.underline(publicUrl)}
`;
}
const statusBox = boxen(
`${chalk.cyan.bold("\u{1F680} Development Server Running")}
${urlLines}
${chalk.yellow("Press Ctrl+C to stop the server")}`,
{
backgroundColor: "#1a1a1a",
borderColor: "cyan",
borderStyle: "round",
margin: 1,
padding: 1
}
);
console.log(`
${statusBox}
`);
};
const dev = async (cmd, callback = () => ({})) => {
const { bundleAnalyzer, cliName = "Lex", config, open = false, quiet, remove, translations = false, usePublicIp, variables } = cmd;
const spinner = createSpinner(quiet);
log(`${cliName} start development server...`, "info", quiet);
await LexConfig.parseConfig(cmd);
const { outputFullPath, useTypescript } = LexConfig.config;
let variablesObj = { NODE_ENV: "development" };
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 (useTypescript) {
LexConfig.checkTypescriptConfig();
}
if (remove) {
spinner.start("Cleaning output directory...");
await removeFiles(outputFullPath || "");
spinner.succeed("Successfully cleaned output directory!");
}
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;
}
}
let webpackConfig;
if (config) {
const isRelativeConfig = config.substr(0, 2) === "./";
webpackConfig = isRelativeConfig ? pathResolve(process.cwd(), config) : config;
} else {
const { webpackConfig: resolvedConfig } = resolveWebpackPaths(currentDirname);
webpackConfig = resolvedConfig;
}
const { webpackPath } = resolveWebpackPaths(currentDirname);
const webpackOptions = [
"--color",
"--watch",
"--config",
webpackConfig
];
if (bundleAnalyzer) {
webpackOptions.push("--bundleAnalyzer");
}
try {
const finalWebpackOptions = webpackPath === "npx" ? ["webpack", ...webpackOptions] : webpackOptions;
spinner.start("Starting development server...");
const childProcess = execa(webpackPath, finalWebpackOptions, {
encoding: "utf8",
env: {
LEX_QUIET: quiet,
WEBPACK_DEV_OPEN: open
},
stdio: "pipe"
});
let serverStarted = false;
let detectedPort = 7001;
childProcess.stdout?.on("data", (data) => {
const output = data.toString();
handleWebpackProgress(output, spinner, quiet, "\u{1F680}", "Webpack Building");
if (!serverStarted && (output.includes("Local:") || output.includes("webpack compiled") || output.includes("webpack-plugin-serve") || output.includes("http://localhost") || output.includes("listening on port"))) {
serverStarted = true;
spinner.succeed("Development server started.");
const portMatch = output.match(/Local:\s*http:\/\/[^:]+:(\d+)/) || output.match(/http:\/\/localhost:(\d+)/) || output.match(/port:\s*(\d+)/) || output.match(/listening on port (\d+)/) || output.match(/WebpackPluginServe listening on port (\d+)/);
if (portMatch) {
detectedPort = parseInt(portMatch[1]);
}
displayServerStatus(detectedPort, quiet);
fetchPublicIp(usePublicIp).then((publicIp) => {
if (publicIp) {
displayServerStatus(detectedPort, quiet, publicIp);
}
});
}
});
childProcess.stderr?.on("data", (data) => {
const output = data.toString();
handleWebpackProgress(output, spinner, quiet, "\u{1F680}", "Webpack Building");
if (!serverStarted && (output.includes("Local:") || output.includes("webpack compiled") || output.includes("webpack-plugin-serve") || output.includes("http://localhost") || output.includes("listening on port"))) {
serverStarted = true;
spinner.succeed("Development server started.");
const portMatch = output.match(/Local:\s*http:\/\/[^:]+:(\d+)/) || output.match(/http:\/\/localhost:(\d+)/) || output.match(/port:\s*(\d+)/) || output.match(/listening on port (\d+)/) || output.match(/WebpackPluginServe listening on port (\d+)/);
if (portMatch) {
detectedPort = parseInt(portMatch[1]);
}
displayServerStatus(detectedPort, quiet);
fetchPublicIp(usePublicIp).then((publicIp) => {
if (publicIp) {
displayServerStatus(detectedPort, quiet, publicIp);
}
});
}
});
setTimeout(() => {
if (!serverStarted) {
spinner.succeed("Development server started.");
displayServerStatus(detectedPort, quiet);
fetchPublicIp(usePublicIp).then((publicIp) => {
if (publicIp) {
displayServerStatus(detectedPort, quiet, publicIp);
}
});
}
}, 5e3);
await childProcess;
if (!serverStarted) {
spinner.succeed("Development server started.");
displayServerStatus(detectedPort, quiet);
fetchPublicIp(usePublicIp).then((publicIp) => {
if (publicIp) {
displayServerStatus(detectedPort, quiet, publicIp);
}
});
}
callback(0);
return 0;
} catch (error) {
log(`
${cliName} Error: ${error.message}`, "error", quiet);
spinner.fail("There was an error while running Webpack.");
callback(1);
return 1;
}
};
export {
dev
};
//# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vc3JjL2NvbW1hbmRzL2Rldi9kZXYudHMiXSwKICAic291cmNlc0NvbnRlbnQiOiBbIi8qKlxuICogQ29weXJpZ2h0IChjKSAyMDE4LVByZXNlbnQsIE5pdHJvZ2VuIExhYnMsIEluYy5cbiAqIENvcHlyaWdodHMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiBTZWUgdGhlIGFjY29tcGFueWluZyBMSUNFTlNFIGZpbGUgZm9yIHRlcm1zLlxuICovXG5pbXBvcnQgYm94ZW4gZnJvbSAnYm94ZW4nO1xuaW1wb3J0IGNoYWxrIGZyb20gJ2NoYWxrJztcbmltcG9ydCB7ZXhlY2F9IGZyb20gJ2V4ZWNhJztcbmltcG9ydCB7ZXhpc3RzU3luYywgcmVhZEZpbGVTeW5jLCB3cml0ZUZpbGVTeW5jLCBta2RpclN5bmN9IGZyb20gJ2ZzJztcbmltcG9ydCBodHRwcyBmcm9tICdodHRwcyc7XG5pbXBvcnQge25ldHdvcmtJbnRlcmZhY2VzLCBob21lZGlyfSBmcm9tICdvcyc7XG5pbXBvcnQge2Rpcm5hbWUsIHJlc29sdmUgYXMgcGF0aFJlc29sdmUsIGpvaW59IGZyb20gJ3BhdGgnO1xuXG5pbXBvcnQge0xleENvbmZpZ30gZnJvbSAnLi4vLi4vTGV4Q29uZmlnLmpzJztcbmltcG9ydCB7Y3JlYXRlU3Bpbm5lciwgaGFuZGxlV2VicGFja1Byb2dyZXNzLCByZW1vdmVGaWxlc30gZnJvbSAnLi4vLi4vdXRpbHMvYXBwLmpzJztcbmltcG9ydCB7cmVzb2x2ZVdlYnBhY2tQYXRoc30gZnJvbSAnLi4vLi4vdXRpbHMvZmlsZS5qcyc7XG5pbXBvcnQge2xvZ30gZnJvbSAnLi4vLi4vdXRpbHMvbG9nLmpzJztcbmltcG9ydCB7cHJvY2Vzc1RyYW5zbGF0aW9uc30gZnJvbSAnLi4vLi4vdXRpbHMvdHJhbnNsYXRpb25zLmpzJztcblxubGV0IGN1cnJlbnRGaWxlbmFtZTogc3RyaW5nO1xubGV0IGN1cnJlbnREaXJuYW1lOiBzdHJpbmc7XG5cbnRyeSB7XG4gIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBuby1ldmFsXG4gIGN1cnJlbnRGaWxlbmFtZSA9IGV2YWwoJ3JlcXVpcmUoXCJ1cmxcIikuZmlsZVVSTFRvUGF0aChpbXBvcnQubWV0YS51cmwpJyk7XG4gIGN1cnJlbnREaXJuYW1lID0gZGlybmFtZShjdXJyZW50RmlsZW5hbWUpO1xufSBjYXRjaCB7XG4gIGN1cnJlbnRGaWxlbmFtZSA9IHByb2Nlc3MuY3dkKCk7XG4gIGN1cnJlbnREaXJuYW1lID0gcHJvY2Vzcy5jd2QoKTtcbn1cblxuZXhwb3J0IGludGVyZmFjZSBEZXZPcHRpb25zIHtcbiAgcmVhZG9ubHkgYnVuZGxlQW5hbHl6ZXI/OiBib29sZWFuO1xuICByZWFkb25seSBjbGlOYW1lPzogc3RyaW5nO1xuICByZWFkb25seSBjb25maWc/OiBzdHJpbmc7XG4gIHJlYWRvbmx5IG9wZW4/OiBib29sZWFuO1xuICByZWFkb25seSBxdWlldD86IGJvb2xlYW47XG4gIHJlYWRvbmx5IHJlbW92ZT86IGJvb2xlYW47XG4gIHJlYWRvbmx5IHRyYW5zbGF0aW9ucz86IGJvb2xlYW47XG4gIHJlYWRvbmx5IHVzZVB1YmxpY0lwPzogYm9vbGVhbjtcbiAgcmVhZG9ubHkgdmFyaWFibGVzPzogc3RyaW5nO1xufVxuXG5leHBvcnQgdHlwZSBEZXZDYWxsYmFjayA9IChzdGF0dXM6IG51bWJlcikgPT4gdm9pZDtcblxuaW50ZXJmYWNlIFB1YmxpY0lwQ2FjaGUge1xuICBpcDogc3RyaW5nO1xuICB0aW1lc3RhbXA6IG51bWJlcjtcbn1cblxuY29uc3QgZ2V0Q2FjaGVEaXIgPSAoKTogc3RyaW5nID0+IHtcbiAgY29uc3QgY2FjaGVEaXIgPSBqb2luKGhvbWVkaXIoKSwgJy5sZXgtY2FjaGUnKTtcbiAgaWYoIWV4aXN0c1N5bmMoY2FjaGVEaXIpKSB7XG4gICAgbWtkaXJTeW5jKGNhY2hlRGlyLCB7cmVjdXJzaXZlOiB0cnVlfSk7XG4gIH1cbiAgcmV0dXJuIGNhY2hlRGlyO1xufTtcblxuY29uc3QgZ2V0Q2FjaGVQYXRoID0gKCk6IHN0cmluZyA9PiBqb2luKGdldENhY2hlRGlyKCksICdwdWJsaWMtaXAuanNvbicpO1xuXG5jb25zdCByZWFkUHVibGljSXBDYWNoZSA9ICgpOiBQdWJsaWNJcENhY2hlIHwgbnVsbCA9PiB7XG4gIGNvbnN0IGNhY2hlUGF0aCA9IGdldENhY2hlUGF0aCgpO1xuICBpZighZXhpc3RzU3luYyhjYWNoZVBhdGgpKSB7XG4gICAgcmV0dXJuIG51bGw7XG4gIH1cblxuICB0cnkge1xuICAgIGNvbnN0IGNhY2hlRGF0YSA9IHJlYWRGaWxlU3luYyhjYWNoZVBhdGgsICd1dGY4Jyk7XG4gICAgY29uc3QgY2FjaGU6IFB1YmxpY0lwQ2FjaGUgPSBKU09OLnBhcnNlKGNhY2hlRGF0YSk7XG5cbiAgICAvLyBDaGVjayBpZiBjYWNoZSBpcyBvbGRlciB0aGFuIDEgd2VlayAoNyBkYXlzICogMjQgaG91cnMgKiA2MCBtaW51dGVzICogNjAgc2Vjb25kcyAqIDEwMDAgbWlsbGlzZWNvbmRzKVxuICAgIGNvbnN0IG9uZVdlZWtNcyA9IDcgKiAyNCAqIDYwICogNjAgKiAxMDAwO1xuICAgIGlmKERhdGUubm93KCkgLSBjYWNoZS50aW1lc3RhbXAgPiBvbmVXZWVrTXMpIHtcbiAgICAgIHJldHVybiBudWxsO1xuICAgIH1cblxuICAgIHJldHVybiBjYWNoZTtcbiAgfSBjYXRjaCB7XG4gICAgcmV0dXJuIG51bGw7XG4gIH1cbn07XG5cbmNvbnN0IHdyaXRlUHVibGljSXBDYWNoZSA9IChpcDogc3RyaW5nKTogdm9pZCA9PiB7XG4gIGNvbnN0IGNhY2hlUGF0aCA9IGdldENhY2hlUGF0aCgpO1xuICBjb25zdCBjYWNoZTogUHVibGljSXBDYWNoZSA9IHtcbiAgICBpcCxcbiAgICB0aW1lc3RhbXA6IERhdGUubm93KClcbiAgfTtcbiAgd3JpdGVGaWxlU3luYyhjYWNoZVBhdGgsIEpTT04uc3RyaW5naWZ5KGNhY2hlLCBudWxsLCAyKSk7XG59O1xuXG5jb25zdCBmZXRjaFB1YmxpY0lwID0gKGZvcmNlUmVmcmVzaDogYm9vbGVhbiA9IGZhbHNlKTogUHJvbWlzZTxzdHJpbmcgfCB1bmRlZmluZWQ+ID0+IG5ldyBQcm9taXNlKChyZXNvbHZlKSA9PiB7XG4gIC8vIENoZWNrIGNhY2hlIGZpcnN0IHVubGVzcyBmb3JjZSByZWZyZXNoIGlzIHJlcXVlc3RlZFxuICBpZighZm9yY2VSZWZyZXNoKSB7XG4gICAgY29uc3QgY2FjaGVkID0gcmVhZFB1YmxpY0lwQ2FjaGUoKTtcbiAgICBpZihjYWNoZWQpIHtcbiAgICAgIHJlc29sdmUoY2FjaGVkLmlwKTtcbiAgICAgIHJldHVybjtcbiAgICB9XG4gIH1cblxuICBodHRwcy5nZXQoJ2h0dHBzOi8vYXBpLmlwaWZ5Lm9yZycsIChyZXMpID0+IHtcbiAgICBsZXQgZGF0YSA9ICcnO1xuICAgIHJlcy5vbignZGF0YScsIChjaHVuaykgPT4gKGRhdGEgKz0gY2h1bmspKTtcbiAgICByZXMub24oJ2VuZCcsICgpID0+IHtcbiAgICAgIGNvbnN0IGlwID0gZGF0YS50cmltKCk7XG4gICAgICBpZihpcCkge1xuICAgICAgICB3cml0ZVB1YmxpY0lwQ2FjaGUoaXApO1xuICAgICAgfVxuICAgICAgcmVzb2x2ZShpcCk7XG4gICAgfSk7XG4gIH0pLm9uKCdlcnJvcicsICgpID0+IHJlc29sdmUodW5kZWZpbmVkKSk7XG59KTtcblxuY29uc3QgZ2V0TmV0d29ya0FkZHJlc3NlcyA9ICgpID0+IHtcbiAgY29uc3QgaW50ZXJmYWNlcyA9IG5ldHdvcmtJbnRlcmZhY2VzKCk7XG4gIGNvbnN0IGFkZHJlc3NlcyA9IHtcbiAgICBsb2NhbDogJ2xvY2FsaG9zdCcsXG4gICAgcHJpdmF0ZTogbnVsbCxcbiAgICBwdWJsaWM6IG51bGxcbiAgfTtcblxuICBmb3IoY29uc3QgbmFtZSBvZiBPYmplY3Qua2V5cyhpbnRlcmZhY2VzKSkge1xuICAgIGNvbnN0IG5ldHdvcmtJbnRlcmZhY2UgPSBpbnRlcmZhY2VzW25hbWVdO1xuICAgIGlmKCFuZXR3b3JrSW50ZXJmYWNlKSB7XG4gICAgICBjb250aW51ZTtcbiAgICB9XG5cbiAgICBmb3IoY29uc3QgaWZhY2Ugb2YgbmV0d29ya0ludGVyZmFjZSkge1xuICAgICAgaWYoaWZhY2UuZmFtaWx5ID09PSAnSVB2NCcgJiYgIWlmYWNlLmludGVybmFsKSB7XG4gICAgICAgIGNvbnN0IGlwID0gaWZhY2UuYWRkcmVzcztcblxuICAgICAgICAvLyBQcml2YXRlIElQIHJhbmdlc1xuICAgICAgICBpZihpcC5zdGFydHNXaXRoKCcxMC4nKSB8fCBpcC5zdGFydHNXaXRoKCcxOTIuMTY4LicpIHx8IGlwLnN0YXJ0c1dpdGgoJzE3Mi4nKSkge1xuICAgICAgICAgIGlmKCFhZGRyZXNzZXMucHJpdmF0ZSkge1xuICAgICAgICAgICAgYWRkcmVzc2VzLnByaXZhdGUgPSBpcDtcbiAgICAgICAgICB9XG4gICAgICAgIH0gZWxzZSB7XG4gICAgICAgICAgLy8gUHVibGljIElQIChub3QgaW4gcHJpdmF0ZSByYW5nZXMpXG4gICAgICAgICAgaWYoIWFkZHJlc3Nlcy5wdWJsaWMpIHtcbiAgICAgICAgICAgIGFkZHJlc3Nlcy5wdWJsaWMgPSBpcDtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG4gIH1cblxuICByZXR1cm4gYWRkcmVzc2VzO1xufTtcblxuY29uc3QgZGlzcGxheVNlcnZlclN0YXR1cyA9IChwb3J0OiBudW1iZXIgPSA3MDAxLCBxdWlldDogYm9vbGVhbiwgcHVibGljSXA/OiBzdHJpbmcpID0+IHtcbiAgaWYocXVpZXQpIHtcbiAgICByZXR1cm47XG4gIH1cblxuICBjb25zdCBhZGRyZXNzZXMgPSBnZXROZXR3b3JrQWRkcmVzc2VzKCk7XG4gIGNvbnN0IGxvY2FsVXJsID0gYGh0dHA6Ly9sb2NhbGhvc3Q6JHtwb3J0fWA7XG4gIGNvbnN0IHByaXZhdGVVcmwgPSBhZGRyZXNzZXMucHJpdmF0ZSA/IGBodHRwOi8vJHthZGRyZXNzZXMucHJpdmF0ZX06JHtwb3J0fWAgOiBudWxsO1xuICBsZXQgcHVibGljVXJsID0gbnVsbDtcbiAgaWYocHVibGljSXApIHtcbiAgICBwdWJsaWNVcmwgPSBgaHR0cDovLyR7cHVibGljSXB9OiR7cG9ydH1gO1xuICB9IGVsc2UgaWYoYWRkcmVzc2VzLnB1YmxpYykge1xuICAgIHB1YmxpY1VybCA9IGBodHRwOi8vJHthZGRyZXNzZXMucHVibGljfToke3BvcnR9YDtcbiAgfVxuXG4gIGxldCB1cmxMaW5lcyA9IGAke2NoYWxrLmdyZWVuKCdMb2NhbDonKX0gICAgICR7Y2hhbGsudW5kZXJsaW5lKGxvY2FsVXJsKX1cXG5gO1xuXG4gIGlmKHByaXZhdGVVcmwpIHtcbiAgICB1cmxMaW5lcyArPSBgJHtjaGFsay5ncmVlbignUHJpdmF0ZTonKX0gICAke2NoYWxrLnVuZGVybGluZShwcml2YXRlVXJsKX1cXG5gO1xuICB9XG5cbiAgaWYocHVibGljVXJsKSB7XG4gICAgdXJsTGluZXMgKz0gYCR7Y2hhbGsuZ3JlZW4oJ1B1YmxpYzonKX0gICAgJHtjaGFsay51bmRlcmxpbmUocHVibGljVXJsKX1cXG5gO1xuICB9XG5cbiAgY29uc3Qgc3RhdHVzQm94ID0gYm94ZW4oXG4gICAgYCR7Y2hhbGsuY3lhbi5ib2xkKCdcdUQ4M0RcdURFODAgRGV2ZWxvcG1lbnQgU2VydmVyIFJ1bm5pbmcnKX1cXG5cXG4ke3VybExpbmVzfVxcbmAgK1xuICAgIGAke2NoYWxrLnllbGxvdygnUHJlc3MgQ3RybCtDIHRvIHN0b3AgdGhlIHNlcnZlcicpfWAsXG4gICAge1xuICAgICAgYmFja2dyb3VuZENvbG9yOiAnIzFhMWExYScsXG4gICAgICBib3JkZXJDb2xvcjogJ2N5YW4nLFxuICAgICAgYm9yZGVyU3R5bGU6ICdyb3VuZCcsXG4gICAgICBtYXJnaW46IDEsXG4gICAgICBwYWRkaW5nOiAxXG4gICAgfVxuICApO1xuXG4gIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBuby1jb25zb2xlXG4gIGNvbnNvbGUubG9nKGBcXG4ke3N0YXR1c0JveH1cXG5gKTtcbn07XG5cbmV4cG9ydCBjb25zdCBkZXYgPSBhc3luYyAoY21kOiBEZXZPcHRpb25zLCBjYWxsYmFjazogRGV2Q2FsbGJhY2sgPSAoKSA9PiAoe30pKTogUHJvbWlzZTxudW1iZXI+ID0+IHtcbiAgY29uc3Qge2J1bmRsZUFuYWx5emVyLCBjbGlOYW1lID0gJ0xleCcsIGNvbmZpZywgb3BlbiA9IGZhbHNlLCBxdWlldCwgcmVtb3ZlLCB0cmFuc2xhdGlvbnMgPSBmYWxzZSwgdXNlUHVibGljSXAsIHZhcmlhYmxlc30gPSBjbWQ7XG5cbiAgY29uc3Qgc3Bpbm5lciA9IGNyZWF0ZVNwaW5uZXIocXVpZXQpO1xuXG4gIGxvZyhgJHtjbGlOYW1lfSBzdGFydCBkZXZlbG9wbWVudCBzZXJ2ZXIuLi5gLCAnaW5mbycsIHF1aWV0KTtcblxuICBhd2FpdCBMZXhDb25maWcucGFyc2VDb25maWcoY21kKTtcblxuICBjb25zdCB7b3V0cHV0RnVsbFBhdGgsIHVzZVR5cGVzY3JpcHR9ID0gTGV4Q29uZmlnLmNvbmZpZztcblxuICBsZXQgdmFyaWFibGVzT2JqOiBvYmplY3QgPSB7Tk9ERV9FTlY6ICdkZXZlbG9wbWVudCd9O1xuXG4gIGlmKHZhcmlhYmxlcykge1xuICAgIHRyeSB7XG4gICAgICB2YXJpYWJsZXNPYmogPSBKU09OLnBhcnNlKHZhcmlhYmxlcyk7XG4gICAgfSBjYXRjaCAoX2Vycm9yKSB7XG4gICAgICBsb2coYFxcbiR7Y2xpTmFtZX0gRXJyb3I6IEVudmlyb25tZW50IHZhcmlhYmxlcyBvcHRpb24gaXMgbm90IGEgdmFsaWQgSlNPTiBvYmplY3QuYCwgJ2Vycm9yJywgcXVpZXQpO1xuICAgICAgY2FsbGJhY2soMSk7XG4gICAgICByZXR1cm4gMTtcbiAgICB9XG4gIH1cblxuICBwcm9jZXNzLmVudiA9IHsuLi5wcm9jZXNzLmVudiwgLi4udmFyaWFibGVzT2JqfTtcblxuICBpZih1c2VUeXBlc2NyaXB0KSB7XG4gICAgTGV4Q29uZmlnLmNoZWNrVHlwZXNjcmlwdENvbmZpZygpO1xuICB9XG5cbiAgaWYocmVtb3ZlKSB7XG4gICAgc3Bpbm5lci5zdGFydCgnQ2xlYW5pbmcgb3V0cHV0IGRpcmVjdG9yeS4uLicpO1xuXG4gICAgYXdhaXQgcmVtb3ZlRmlsZXMob3V0cHV0RnVsbFBhdGggfHwgJycpO1xuXG4gICAgc3Bpbm5lci5zdWNjZWVkKCdTdWNjZXNzZnVsbHkgY2xlYW5lZCBvdXRwdXQgZGlyZWN0b3J5IScpO1xuICB9XG5cbiAgLy8gUHJvY2VzcyB0cmFuc2xhdGlvbnMgaWYgZmxhZyBpcyBlbmFibGVkIChiZWZvcmUgc3RhcnRpbmcgZGV2IHNlcnZlcilcbiAgaWYodHJhbnNsYXRpb25zKSB7XG4gICAgc3Bpbm5lci5zdGFydCgnUHJvY2Vzc2luZyB0cmFuc2xhdGlvbnMuLi4nKTtcblxuICAgIHRyeSB7XG4gICAgICBjb25zdCBzb3VyY2VQYXRoID0gTGV4Q29uZmlnLmNvbmZpZy5zb3VyY2VGdWxsUGF0aCB8fCBwcm9jZXNzLmN3ZCgpO1xuICAgICAgY29uc3Qgb3V0cHV0UGF0aCA9IExleENvbmZpZy5jb25maWcub3V0cHV0RnVsbFBhdGggfHwgJ2xpYic7XG5cbiAgICAgIGF3YWl0IHByb2Nlc3NUcmFuc2xhdGlvbnMoc291cmNlUGF0aCwgb3V0cHV0UGF0aCwgcXVpZXQpO1xuICAgICAgc3Bpbm5lci5zdWNjZWVkKCdUcmFuc2xhdGlvbnMgcHJvY2Vzc2VkIHN1Y2Nlc3NmdWxseSEnKTtcbiAgICB9IGNhdGNoICh0cmFuc2xhdGlvbkVycm9yKSB7XG4gICAgICBsb2coYFxcbiR7Y2xpTmFtZX0gRXJyb3I6IEZhaWxlZCB0byBwcm9jZXNzIHRyYW5zbGF0aW9uczogJHt0cmFuc2xhdGlvbkVycm9yLm1lc3NhZ2V9YCwgJ2Vycm9yJywgcXVpZXQpO1xuICAgICAgc3Bpbm5lci5mYWlsKCdGYWlsZWQgdG8gcHJvY2VzcyB0cmFuc2xhdGlvbnMuJyk7XG4gICAgICBjYWxsYmFjaygxKTtcbiAgICAgIHJldHVybiAxO1xuICAgIH1cbiAgfVxuXG4gIGxldCB3ZWJwYWNrQ29uZmlnOiBzdHJpbmc7XG5cbiAgaWYoY29uZmlnKSB7XG4gICAgY29uc3QgaXNSZWxhdGl2ZUNvbmZpZzogYm9vbGVhbiA9IGNvbmZpZy5zdWJzdHIoMCwgMikgPT09ICcuLyc7XG4gICAgd2VicGFja0NvbmZpZyA9IGlzUmVsYXRpdmVDb25maWcgPyBwYXRoUmVzb2x2ZShwcm9jZXNzLmN3ZCgpLCBjb25maWcpIDogY29uZmlnO1xuICB9IGVsc2Uge1xuICAgIGNvbnN0IHt3ZWJwYWNrQ29uZmlnOiByZXNvbHZlZENvbmZpZ30gPSByZXNvbHZlV2VicGFja1BhdGhzKGN1cnJlbnREaXJuYW1lKTtcbiAgICB3ZWJwYWNrQ29uZmlnID0gcmVzb2x2ZWRDb25maWc7XG4gIH1cblxuICBjb25zdCB7d2VicGFja1BhdGh9ID0gcmVzb2x2ZVdlYnBhY2tQYXRocyhjdXJyZW50RGlybmFtZSk7XG5cbiAgY29uc3Qgd2VicGFja09wdGlvbnM6IHN0cmluZ1tdID0gW1xuICAgICctLWNvbG9yJyxcbiAgICAnLS13YXRjaCcsXG4gICAgJy0tY29uZmlnJywgd2VicGFja0NvbmZpZ1xuICBdO1xuXG4gIGlmKGJ1bmRsZUFuYWx5emVyKSB7XG4gICAgd2VicGFja09wdGlvbnMucHVzaCgnLS1idW5kbGVBbmFseXplcicpO1xuICB9XG5cbiAgdHJ5IHtcbiAgICBjb25zdCBmaW5hbFdlYnBhY2tPcHRpb25zID0gd2VicGFja1BhdGggPT09ICducHgnID8gWyd3ZWJwYWNrJywgLi4ud2VicGFja09wdGlvbnNdIDogd2VicGFja09wdGlvbnM7XG5cbiAgICBzcGlubmVyLnN0YXJ0KCdTdGFydGluZyBkZXZlbG9wbWVudCBzZXJ2ZXIuLi4nKTtcblxuICAgIGNvbnN0IGNoaWxkUHJvY2VzcyA9IGV4ZWNhKHdlYnBhY2tQYXRoLCBmaW5hbFdlYnBhY2tPcHRpb25zLCB7XG4gICAgICBlbmNvZGluZzogJ3V0ZjgnLFxuICAgICAgZW52OiB7XG4gICAgICAgIExFWF9RVUlFVDogcXVpZXQsXG4gICAgICAgIFdFQlBBQ0tfREVWX09QRU46IG9wZW5cbiAgICAgIH0sXG4gICAgICBzdGRpbzogJ3BpcGUnXG4gICAgfSBhcyBhbnkpO1xuXG4gICAgbGV0IHNlcnZlclN0YXJ0ZWQgPSBmYWxzZTtcbiAgICBsZXQgZGV0ZWN0ZWRQb3J0ID0gNzAwMTtcblxuICAgIGNoaWxkUHJvY2Vzcy5zdGRvdXQ/Lm9uKCdkYXRhJywgKGRhdGE6IEJ1ZmZlcikgPT4ge1xuICAgICAgY29uc3Qgb3V0cHV0ID0gZGF0YS50b1N0cmluZygpO1xuXG4gICAgICBoYW5kbGVXZWJwYWNrUHJvZ3Jlc3Mob3V0cHV0LCBzcGlubmVyLCBxdWlldCwgJ1x1RDgzRFx1REU4MCcsICdXZWJwYWNrIEJ1aWxkaW5nJyk7XG5cbiAgICAgIGlmKCFzZXJ2ZXJTdGFydGVkICYmIChvdXRwdXQuaW5jbHVkZXMoJ0xvY2FsOicpIHx8IG91dHB1dC5pbmNsdWRlcygnd2VicGFjayBjb21waWxlZCcpIHx8IG91dHB1dC5pbmNsdWRlcygnd2VicGFjay1wbHVnaW4tc2VydmUnKSB8fCBvdXRwdXQuaW5jbHVkZXMoJ2h0dHA6Ly9sb2NhbGhvc3QnKSB8fCBvdXRwdXQuaW5jbHVkZXMoJ2xpc3RlbmluZyBvbiBwb3J0JykpKSB7XG4gICAgICAgIHNlcnZlclN0YXJ0ZWQgPSB0cnVlO1xuICAgICAgICBzcGlubmVyLnN1Y2NlZWQoJ0RldmVsb3BtZW50IHNlcnZlciBzdGFydGVkLicpO1xuXG4gICAgICAgIC8vIFRyeSBtdWx0aXBsZSBwYXR0ZXJucyB0byBkZXRlY3QgdGhlIHBvcnRcbiAgICAgICAgY29uc3QgcG9ydE1hdGNoID0gb3V0cHV0Lm1hdGNoKC9Mb2NhbDpcXHMqaHR0cDpcXC9cXC9bXjpdKzooXFxkKykvKSB8fFxuICAgICAgICAgIG91dHB1dC5tYXRjaCgvaHR0cDpcXC9cXC9sb2NhbGhvc3Q6KFxcZCspLykgfHxcbiAgICAgICAgICBvdXRwdXQubWF0Y2goL3BvcnQ6XFxzKihcXGQrKS8pIHx8XG4gICAgICAgICAgb3V0cHV0Lm1hdGNoKC9saXN0ZW5pbmcgb24gcG9ydCAoXFxkKykvKSB8fFxuICAgICAgICAgIG91dHB1dC5tYXRjaCgvV2VicGFja1BsdWdpblNlcnZlIGxpc3RlbmluZyBvbiBwb3J0IChcXGQrKS8pO1xuICAgICAgICBpZihwb3J0TWF0Y2gpIHtcbiAgICAgICAgICBkZXRlY3RlZFBvcnQgPSBwYXJzZUludChwb3J0TWF0Y2hbMV0pO1xuICAgICAgICB9XG5cbiAgICAgICAgZGlzcGxheVNlcnZlclN0YXR1cyhkZXRlY3RlZFBvcnQsIHF1aWV0KTtcbiAgICAgICAgZmV0Y2hQdWJsaWNJcCh1c2VQdWJsaWNJcCkudGhlbigocHVibGljSXApID0+IHtcbiAgICAgICAgICBpZihwdWJsaWNJcCkge1xuICAgICAgICAgICAgZGlzcGxheVNlcnZlclN0YXR1cyhkZXRlY3RlZFBvcnQsIHF1aWV0LCBwdWJsaWNJcCk7XG4gICAgICAgICAgfVxuICAgICAgICB9KTtcbiAgICAgIH1cbiAgICB9KTtcblxuICAgIGNoaWxkUHJvY2Vzcy5zdGRlcnI/Lm9uKCdkYXRhJywgKGRhdGE6IEJ1ZmZlcikgPT4ge1xuICAgICAgY29uc3Qgb3V0cHV0ID0gZGF0YS50b1N0cmluZygpO1xuXG4gICAgICBoYW5kbGVXZWJwYWNrUHJvZ3Jlc3Mob3V0cHV0LCBzcGlubmVyLCBxdWlldCwgJ1x1RDgzRFx1REU4MCcsICdXZWJwYWNrIEJ1aWxkaW5nJyk7XG5cbiAgICAgIGlmKCFzZXJ2ZXJTdGFydGVkICYmIChvdXRwdXQuaW5jbHVkZXMoJ0xvY2FsOicpIHx8IG91dHB1dC5pbmNsdWRlcygnd2VicGFjayBjb21waWxlZCcpIHx8IG91dHB1dC5pbmNsdWRlcygnd2VicGFjay1wbHVnaW4tc2VydmUnKSB8fCBvdXRwdXQuaW5jbHVkZXMoJ2h0dHA6Ly9sb2NhbGhvc3QnKSB8fCBvdXRwdXQuaW5jbHVkZXMoJ2xpc3RlbmluZyBvbiBwb3J0JykpKSB7XG4gICAgICAgIHNlcnZlclN0YXJ0ZWQgPSB0cnVlO1xuICAgICAgICBzcGlubmVyLnN1Y2NlZWQoJ0RldmVsb3BtZW50IHNlcnZlciBzdGFydGVkLicpO1xuXG4gICAgICAgIC8vIFRyeSBtdWx0aXBsZSBwYXR0ZXJucyB0byBkZXRlY3QgdGhlIHBvcnRcbiAgICAgICAgY29uc3QgcG9ydE1hdGNoID0gb3V0cHV0Lm1hdGNoKC9Mb2NhbDpcXHMqaHR0cDpcXC9cXC9bXjpdKzooXFxkKykvKSB8fFxuICAgICAgICAgIG91dHB1dC5tYXRjaCgvaHR0cDpcXC9cXC9sb2NhbGhvc3Q6KFxcZCspLykgfHxcbiAgICAgICAgICBvdXRwdXQubWF0Y2goL3BvcnQ6XFxzKihcXGQrKS8pIHx8XG4gICAgICAgICAgb3V0cHV0Lm1hdGNoKC9saXN0ZW5pbmcgb24gcG9ydCAoXFxkKykvKSB8fFxuICAgICAgICAgIG91dHB1dC5tYXRjaCgvV2VicGFja1BsdWdpblNlcnZlIGxpc3RlbmluZyBvbiBwb3J0IChcXGQrKS8pO1xuICAgICAgICBpZihwb3J0TWF0Y2gpIHtcbiAgICAgICAgICBkZXRlY3RlZFBvcnQgPSBwYXJzZUludChwb3J0TWF0Y2hbMV0pO1xuICAgICAgICB9XG5cbiAgICAgICAgZGlzcGxheVNlcnZlclN0YXR1cyhkZXRlY3RlZFBvcnQsIHF1aWV0KTtcbiAgICAgICAgZmV0Y2hQdWJsaWNJcCh1c2VQdWJsaWNJcCkudGhlbigocHVibGljSXApID0+IHtcbiAgICAgICAgICBpZihwdWJsaWNJcCkge1xuICAgICAgICAgICAgZGlzcGxheVNlcnZlclN0YXR1cyhkZXRlY3RlZFBvcnQsIHF1aWV0LCBwdWJsaWNJcCk7XG4gICAgICAgICAgfVxuICAgICAgICB9KTtcbiAgICAgIH1cbiAgICB9KTtcblxuICAgIHNldFRpbWVvdXQoKCkgPT4ge1xuICAgICAgaWYoIXNlcnZlclN0YXJ0ZWQpIHtcbiAgICAgICAgc3Bpbm5lci5zdWNjZWVkKCdEZXZlbG9wbWVudCBzZXJ2ZXIgc3RhcnRlZC4nKTtcbiAgICAgICAgZGlzcGxheVNlcnZlclN0YXR1cyhkZXRlY3RlZFBvcnQsIHF1aWV0KTtcbiAgICAgICAgZmV0Y2hQdWJsaWNJcCh1c2VQdWJsaWNJcCkudGhlbigocHVibGljSXApID0+IHtcbiAgICAgICAgICBpZihwdWJsaWNJcCkge1xuICAgICAgICAgICAgZGlzcGxheVNlcnZlclN0YXR1cyhkZXRlY3RlZFBvcnQsIHF1aWV0LCBwdWJsaWNJcCk7XG4gICAgICAgICAgfVxuICAgICAgICB9KTtcbiAgICAgIH1cbiAgICB9LCA1MDAwKTtcblxuICAgIGF3YWl0IGNoaWxkUHJvY2VzcztcblxuICAgIGlmKCFzZXJ2ZXJTdGFydGVkKSB7XG4gICAgICBzcGlubmVyLnN1Y2NlZWQoJ0RldmVsb3BtZW50IHNlcnZlciBzdGFydGVkLicpO1xuICAgICAgZGlzcGxheVNlcnZlclN0YXR1cyhkZXRlY3RlZFBvcnQsIHF1aWV0KTtcbiAgICAgIGZldGNoUHVibGljSXAodXNlUHVibGljSXApLnRoZW4oKHB1YmxpY0lwKSA9PiB7XG4gICAgICAgIGlmKHB1YmxpY0lwKSB7XG4gICAgICAgICAgZGlzcGxheVNlcnZlclN0YXR1cyhkZXRlY3RlZFBvcnQsIHF1aWV0LCBwdWJsaWNJcCk7XG4gICAgICAgIH1cbiAgICAgIH0pO1xuICAgIH1cblxuICAgIGNhbGxiYWNrKDApO1xuICAgIHJldHVybiAwO1xuICB9IGNhdGNoIChlcnJvcikge1xuICAgIGxvZyhgXFxuJHtjbGlOYW1lfSBFcnJvcjogJHtlcnJvci5tZXNzYWdlfWAsICdlcnJvcicsIHF1aWV0KTtcblxuICAgIHNwaW5uZXIuZmFpbCgnVGhlcmUgd2FzIGFuIGVycm9yIHdoaWxlIHJ1bm5pbmcgV2VicGFjay4nKTtcblxuICAgIGNhbGxiYWNrKDEpO1xuICAgIHJldHVybiAxO1xuICB9XG59OyJdLAogICJtYXBwaW5ncyI6ICJBQUlBLE9BQU8sV0FBVztBQUNsQixPQUFPLFdBQVc7QUFDbEIsU0FBUSxhQUFZO0FBQ3BCLFNBQVEsWUFBWSxjQUFjLGVBQWUsaUJBQWdCO0FBQ2pFLE9BQU8sV0FBVztBQUNsQixTQUFRLG1CQUFtQixlQUFjO0FBQ3pDLFNBQVEsU0FBUyxXQUFXLGFBQWEsWUFBVztBQUVwRCxTQUFR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