UNPKG

@nlabs/lex

Version:
287 lines (285 loc) 33.4 kB
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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