diff --git a/dist/index.cjs b/dist/index.cjs index db5d8a91..b9179202 100644 --- a/dist/index.cjs +++ b/dist/index.cjs @@ -1,2 +1,2 @@ -"use strict";require("colors");var e=require("fs"),t=require("path"),o=require("body-parser"),r=require("cors"),i=require("express"),n=require("multer"),s=require("http"),a=require("https"),l=require("dotenv"),c=require("express-rate-limit"),p=require("url"),u=require("https-proxy-agent"),d=require("uuid"),h=require("tarn"),g=require("puppeteer"),f=require("node:path"),m=require("node:crypto");require("prompts");var v="undefined"!=typeof document?document.currentScript:null;function y(e){var t=Object.create(null);return e&&Object.keys(e).forEach((function(o){if("default"!==o){var r=Object.getOwnPropertyDescriptor(e,o);Object.defineProperty(t,o,r.get?r:{enumerable:!0,get:function(){return e[o]}})}})),t.default=e,Object.freeze(t)}var b=y(p);l.config();const w={puppeteer:{args:{value:[],type:"string[]",description:"Array of arguments to send to puppeteer."}},highcharts:{version:{value:"latest",envLink:"HIGHCHARTS_VERSION",type:"string",description:"Highcharts version to use."},cdnURL:{value:"https://code.highcharts.com/",envLink:"HIGHCHARTS_CDN",type:"string",description:"The CDN URL of Highcharts scripts to use."},coreScripts:{envLink:"HIGHCHARTS_CORE_SCRIPTS",value:["highcharts","highcharts-more","highcharts-3d"],type:"string[]",description:"Highcharts core scripts to fetch."},modules:{envLink:"HIGHCHARTS_MODULES",value:["stock","map","gantt","exporting","export-data","parallel-coordinates","accessibility","annotations-advanced","boost-canvas","boost","data","draggable-points","static-scale","broken-axis","heatmap","tilemap","timeline","treemap","item-series","drilldown","histogram-bellcurve","bullet","funnel","funnel3d","pyramid3d","networkgraph","pareto","pattern-fill","pictorial","price-indicator","sankey","arc-diagram","dependency-wheel","series-label","solid-gauge","sonification","stock-tools","streamgraph","sunburst","variable-pie","variwide","vector","venn","windbarb","wordcloud","xrange","no-data-to-display","drag-panes","debugger","dumbbell","lollipop","cylinder","organization","dotplot","marker-clusters","hollowcandlestick","heikinashi"],type:"string[]",description:"Highcharts modules to fetch."},indicators:{envLink:"HIGHCHARTS_INDICATORS",value:["indicators-all"],type:"string[]",description:"Highcharts indicators to fetch."},scripts:{value:["https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.29.4/moment.min.js","https://cdnjs.cloudflare.com/ajax/libs/moment-timezone/0.5.34/moment-timezone-with-data.min.js"],type:"string[]",description:"Additional direct scripts/optional dependencies (e.g. moment.js)."},forceFetch:{envLink:"HIGHCHARTS_FORCE_FETCH",value:!1,type:"boolean",description:"Should all the scripts be refetched after rerunning the server."}},export:{infile:{value:!1,type:"string",description:"The input file name along with a type (json or svg). It can be a correct JSON or SVG file."},instr:{value:!1,type:"string",description:"An input in a form of a stringified JSON or SVG file. Overrides the --infile."},options:{value:!1,type:"string",description:"An alias for the --instr option."},outfile:{value:!1,type:"string",description:"The output filename along with a type (jpeg, png, pdf or svg). Ignores the --type flag."},type:{envLink:"EXPORT_DEFAULT_TYPE",value:"png",type:"string",description:"The format of the file to export to. Can be jpeg, png, pdf or svg."},constr:{envLink:"EXPORT_DEFAULT_CONSTR",value:"chart",type:"string",description:"The constructor to use. Can be chart, stockChart, mapChart or ganttChart."},defaultHeight:{envLink:"EXPORT_DEFAULT_HEIGHT",value:400,type:"number",description:"The default height of the exported chart. Used when not found any value set."},defaultWidth:{envLink:"EXPORT_DEFAULT_WIDTH",value:600,type:"number",description:"The default width of the exported chart. Used when not found any value set."},defaultScale:{envLink:"EXPORT_DEFAULT_SCALE",value:1,type:"number",description:"The default scale of the exported chart. Ranges between 1 and 5."},height:{type:"number",value:!1,description:"The default height of the exported chart. Overrides the option in the chart settings."},width:{type:"number",value:!1,description:"The width of the exported chart. Overrides the option in the chart settings."},scale:{value:!1,type:"number",description:"The scale of the exported chart. Ranges between 1 and 5."},globalOptions:{value:!1,type:"string",description:"A stringified JSON or a filename with options to be passed into the Highcharts.setOptions."},themeOptions:{value:!1,type:"string",description:"A stringified JSON or a filename with theme options to be passed into the Highcharts.setOptions."},batch:{value:!1,type:"string",description:'Starts a batch job. A string that contains input/output pairs: "in=out;in=out;..".'}},customCode:{allowCodeExecution:{envLink:"HIGHCHARTS_ALLOW_CODE_EXECUTION",value:!1,type:"boolean",description:"If set to true, allow for the execution of arbitrary code when exporting."},allowFileResources:{envLink:"HIGHCHARTS_ALLOW_FILE_RESOURCES",value:!0,type:"boolean",description:"Allow injecting resources from the filesystem. Has no effect when running as a server."},customCode:{value:!1,type:"string",description:"A function to be called before chart initialization. Can be a filename with the js extension."},callback:{value:!1,type:"string",description:"A JavaScript file with a function to run on construction."},resources:{value:!1,type:"string",description:"An additional resource in a form of stringified JSON. It can contain files, js and css sections."},loadConfig:{value:!1,type:"string",description:"A file that contains a pre-defined config to use."},createConfig:{value:!1,type:"string",description:"Allows to set options through a prompt and save in a provided config file."}},server:{enable:{envLink:"HIGHCHARTS_SERVER_ENABLE",value:!1,type:"boolean",cliName:"enableServer",description:"If set to true, starts a server on 0.0.0.0."},host:{envLink:"HIGHCHARTS_SERVER_HOST",value:"0.0.0.0",type:"string",description:"The hostname of the server. Also starts a server listening on the supplied hostname."},port:{envLink:"HIGHCHARTS_SERVER_PORT",value:7801,type:"number",description:"The port to use for the server. Defaults to 7801."},ssl:{enable:{envLink:"HIGHCHARTS_SERVER_SSL_ENABLE",value:!1,type:"boolean",cliName:"enableSsl",description:"Enables the SSL protocol."},force:{envLink:"HIGHCHARTS_SERVER_SSL_FORCE",value:!1,type:"boolean",cliName:"sslForced",description:"If set to true, forces the server to only serve over HTTPS."},port:{envLink:"HIGHCHARTS_SERVER_SSL_PORT",value:443,type:"number",cliName:"sslPort",description:"The port on which to run the SSL server."},certPath:{envLink:"HIGHCHARTS_SSL_CERT_PATH",value:"",type:"string",description:"The path to the SSL certificate/key."}},rateLimiting:{enable:{envLink:"HIGHCHARTS_RATE_LIMIT_ENABLE",value:!1,type:"boolean",cliName:"enableRateLimiting",description:"Enables rate limiting."},maxRequests:{envLink:"HIGHCHARTS_RATE_LIMIT_MAX",value:10,type:"number",description:"Max requests allowed in a one minute."},window:{envLink:"HIGHCHARTS_RATE_LIMIT_WINDOW",value:1,type:"number",description:"The time window in minutes for rate limiting."},delay:{envLink:"HIGHCHARTS_RATE_LIMIT_DELAY",value:0,type:"number",description:"The amount to delay each successive request before hitting the max."},trustProxy:{envLink:"HIGHCHARTS_RATE_LIMIT_TRUST_PROXY",value:!1,type:"boolean",description:"Set this to true if behind a load balancer."},skipKey:{envLink:"HIGHCHARTS_RATE_LIMIT_SKIP_KEY",value:"",type:"number|string",description:"Allows bypassing the rate limiter and should be provided with skipToken argument."},skipToken:{envLink:"HIGHCHARTS_RATE_LIMIT_SKIP_TOKEN",value:"",type:"number|string",description:"Allows bypassing the rate limiter and should be provided with skipKey argument."}}},pool:{initialWorkers:{envLink:"HIGHCHARTS_POOL_MIN_WORKERS",value:4,type:"number",description:"The number of initial workers to spawn."},maxWorkers:{envLink:"HIGHCHARTS_POOL_MAX_WORKERS",value:8,type:"number",description:"The number of max workers to spawn."},workLimit:{envLink:"HIGHCHARTS_POOL_WORK_LIMIT",value:40,type:"number",description:"The pieces of work that can be performed before restarting process."},queueSize:{envLink:"HIGHCHARTS_POOL_QUEUE_SIZE",value:5,type:"number",description:"The size of the request overflow queue."},timeoutThreshold:{envLink:"HIGHCHARTS_POOL_TIMEOUT",value:5e3,type:"number",description:"The number of milliseconds before timing out."},acquireTimeout:{envLink:"HIGHCHARTS_POOL_ACQUIRE_TIMEOUT",value:5e3,type:"number",description:"The number of milliseconds to wait for acquiring a resource."},reaper:{envLink:"HIGHCHARTS_POOL_ENABLE_REAPER",value:!0,type:"boolean",description:"Whether or not to evict workers after a certain time period."},benchmarking:{envLink:"HIGHCHARTS_POOL_BENCHMARKING",value:!1,type:"boolean",description:"Enable benchmarking."},listenToProcessExits:{envLink:"HIGHCHARTS_POOL_LISTEN_TO_PROCESS_EXITS",value:!0,type:"boolean",description:"Set to false in order to skip attaching process.exit handlers."}},logging:{level:{envLink:"HIGHCHARTS_LOG_LEVEL",value:4,type:"number",cliName:"logLevel",description:"The log level (0: silent, 1: error, 2: warning, 3: notice, 4: verbose)."},file:{envLink:"HIGHCHARTS_LOG_FILE",value:"highcharts-export-server.log",type:"string",cliName:"logFile",description:"A name of a log file. The --logDest also needs to be set to enable file logging."},dest:{envLink:"HIGHCHARTS_LOG_DEST",value:"log/",type:"string",cliName:"logDest",description:"The path to store log files. Also enables file logging."}},ui:{enable:{envLink:"HIGHCHARTS_UI_ENABLE",value:!1,type:"boolean",cliName:"enableUi",description:"Enables the UI for the export server."},route:{envLink:"HIGHCHARTS_UI_ROUTE",value:"/",type:"string",cliName:"uiRoute",description:"The route to attach the UI to."}},other:{noLogo:{envLink:"HIGHCHARTS_NO_LOGO",value:!1,type:"boolean",description:"Skip printing the logo on a startup. Will be replaced by a simple text."}},payload:{}};w.puppeteer.args.value.join(","),w.highcharts.version.value,w.highcharts.cdnURL.value,w.highcharts.modules.value,w.highcharts.scripts.value.join(","),w.highcharts.forceFetch.value,w.export.type.value,w.export.constr.value,w.export.defaultHeight.value,w.export.defaultWidth.value,w.export.defaultScale.value,w.customCode.allowCodeExecution.value,w.customCode.allowFileResources.value,w.server.enable.value,w.server.host.value,w.server.port.value,w.server.ssl.enable.value,w.server.ssl.force.value,w.server.ssl.port.value,w.server.ssl.certPath.value,w.server.rateLimiting.enable.value,w.server.rateLimiting.maxRequests.value,w.server.rateLimiting.window.value,w.server.rateLimiting.delay.value,w.server.rateLimiting.trustProxy.value,w.server.rateLimiting.skipKey.value,w.server.rateLimiting.skipToken.value,w.pool.initialWorkers.value,w.pool.maxWorkers.value,w.pool.workLimit.value,w.pool.queueSize.value,w.pool.timeoutThreshold.value,w.pool.acquireTimeout.value,w.pool.reaper.value,w.pool.benchmarking.value,w.pool.listenToProcessExits.value,w.logging.level.value,w.logging.file.value,w.logging.dest.value,w.ui.enable.value,w.ui.route.value,w.other.noLogo.value;const x=["options","globalOptions","themeOptions","resources","payload"],T={},k=(e,t="")=>{Object.keys(e).forEach((o=>{if(!["puppeteer","highcharts"].includes(o)){const r=e[o];void 0===r.value?k(r,`${t}.${o}`):T[r.cliName||o]=`${t}.${o}`.substring(1)}}))};k(w);let S={toConsole:!0,toFile:!1,pathCreated:!1,levelsDesc:[{title:"error",color:"red"},{title:"warning",color:"yellow"},{title:"notice",color:"blue"},{title:"verbose",color:"gray"}],listeners:[]};for(const[e,t]of Object.entries(w.logging))S[e]=t.value;const H=(...t)=>{const[o,...r]=t,{level:i,levelsDesc:n}=S;if(0===o||o>i||i>n.length)return;const s=`${(new Date).toString().split("(")[0].trim()} [${n[o-1].title}] -`;S.listeners.forEach((e=>{e(s,r.join(" "))})),S.toFile&&(S.pathCreated||(!e.existsSync(S.dest)&&e.mkdirSync(S.dest),S.pathCreated=!0),e.appendFile(`${S.dest}${S.file}`,[s].concat(r).join(" ")+"\n",(e=>{e&&(console.log(`[logger] Unable to write to log file: ${e}`),S.toFile=!1)}))),S.toConsole&&console.log.apply(void 0,[s.toString()[S.levelsDesc[o-1].color]].concat(r))},E=p.fileURLToPath(new URL("../.","undefined"==typeof document?require("url").pathToFileURL(__filename).href:v&&v.src||new URL("index.cjs",document.baseURI).href)),R=(e,t=/\s\s+/g,o=" ")=>e.replaceAll(t,o).trim(),L=(e,t)=>{const o=["png","jpeg","pdf","svg"];if(t){const r=t.split(".").pop();o.includes(r)&&e!==r&&(e=r)}return{"image/png":"png","image/jpeg":"jpeg","application/pdf":"pdf","image/svg+xml":"svg"}[e]||o.find((t=>t===e))||"png"},C=(t=!1,o)=>{const r=["js","css","files"];let i=t,n=!1;if(o&&t.endsWith(".json"))try{t?t&&t.endsWith(".json")?i=O(e.readFileSync(t,"utf8")):(i=O(t),!0===i&&(i=O(e.readFileSync("resources.json","utf8")))):i=O(e.readFileSync("resources.json","utf8"))}catch(e){return H(3,"[cli] No resources found.")}else i=O(t),o||delete i.files;for(const e in i)r.includes(e)?n||(n=!0):delete i[e];return n?(i.files&&(i.files=i.files.map((e=>e.trim())),(!i.files||i.files.length<=0)&&delete i.files),i):H(3,"[cli] No resources found.")};function O(e,t){try{const o=JSON.parse("string"!=typeof e?JSON.stringify(e):e);return"string"!=typeof o&&t?JSON.stringify(o):o}catch(e){return!1}}const _=e=>{if(null===e||"object"!=typeof e)return e;const t=Array.isArray(e)?[]:{};for(const o in e)Object.prototype.hasOwnProperty.call(e,o)&&(t[o]=_(e[o]));return t},A=(e,t)=>JSON.stringify(e,((e,o)=>("string"==typeof o&&((o=o.trim()).startsWith("function(")||o.startsWith("function ("))&&o.endsWith("}")&&(o=t?`EXP_FUN${(o+"").replaceAll(/\n|\t|\r/g," ")}EXP_FUN`:void 0),"function"==typeof o?`EXP_FUN${(o+"").replaceAll(/\n|\t|\r/g," ")}EXP_FUN`:o))).replaceAll(/"EXP_FUN|EXP_FUN"/g,"");function $(){console.log("Usage of CLI arguments:".bold,"\n------",`\nFor more detailed information visit readme at: ${"https://github.com/highcharts/node-export-server#readme".bold.yellow}.`);const e=t=>{for(const[o,r]of Object.entries(t))if(Object.prototype.hasOwnProperty.call(r,"value")){let e=` --${r.cliName||o} ${("<"+r.type+">").green} `;if(e.length<48)for(let t=e.length;t<48;t++)e+=".";console.log(e,r.description,`[Default: ${r.value.toString().bold}]`.blue)}else e(r)};Object.keys(w).forEach((t=>{["puppeteer","highcharts"].includes(t)||(console.log(`\n${t.toUpperCase()}`.red),e(w[t]))})),console.log("\n")}const I=e=>!["false","undefined","null","NaN","0",""].includes(e)&&!!e,j=(t,o)=>{if(t&&"string"==typeof t)return(t=t.trim()).endsWith(".js")?!!o&&j(e.readFileSync(t,"utf8")):t.startsWith("function()")||t.startsWith("function ()")||t.startsWith("()=>")||t.startsWith("() =>")?`(${t})()`:t.replace(/;$/,"")};var P=(e,t)=>{const o="Too many requests, you have been rate limited. Please try again later.",r={max:t.maxRequests||30,window:t.window||1,delay:t.delay||0,trustProxy:t.trustProxy||!1,skipKey:t.skipKey||!1,skipToken:t.skipToken||!1};r.trustProxy&&e.enable("trust proxy");const i=c({windowMs:60*r.window*1e3,max:r.max,delayMs:r.delay,handler:(e,t)=>{t.format({json:()=>{t.status(429).send({message:o})},default:()=>{t.status(429).send(o)}})},skip:e=>!1!==r.skipKey&&!1!==r.skipToken&&e.query.key===r.skipKey&&e.query.access_token===r.skipToken&&(H(4,"[rate-limiting] Skipping rate limiter."),!0)});e.use(i),H(3,R(`[rate-limiting] Enabled rate limiting: ${r.max} requests\n per ${r.window} minute per IP, trusting proxy:\n ${r.trustProxy}.`))};async function N(e,t={}){return new Promise(((o,r)=>{const i=(e=>e.startsWith("https")?a:s)(e);i.get(e,t,(e=>{let t="";e.on("data",(e=>{t+=e})),e.on("end",(()=>{t||r("Nothing was fetched from the URL."),e.text=t,o(e)}))})).on("error",(e=>{r(e)}))}))}l.config();const F=t.join(E,".cache"),U={cdnURL:"https://code.highcharts.com/",activeManifest:{},sources:"",hcVersion:""};let q=!1;const G=()=>U.hcVersion=U.sources.substr(0,U.sources.indexOf("*/")).replace("/*","").replace("*/","").replace(/\n/g,"").trim(),W=async(e,t)=>{try{e.endsWith(".js")&&(e=e.substring(0,e.length-3)),H(4,`[cache] Fetching script - ${e}.js`);const o=t?{agent:t,timeout:+process.env.PROXY_SERVER_TIMEOUT||5e3}:{},r=await N(`${e}.js`,o);if(200===r.statusCode)return r.text;throw`${r.statusCode}`}catch(t){throw H(1,`[cache] Error fetching script ${e}.js: ${t}.`),t}},M=async(t,o)=>{const{coreScripts:r,modules:i,indicators:n,scripts:s}=t,a="latest"!==t.version&&t.version?`${t.version}/`:"";H(3,"[cache] Updating cache to Highcharts ",a);const l=[...r.map((e=>`${a}${e}`)),...i.map((e=>"map"===e?`maps/${a}modules/${e}`:`${a}modules/${e}`)),...n.map((e=>`stock/${a}indicators/${e}`))];let c;const p=process.env.PROXY_SERVER_HOST,d=process.env.PROXY_SERVER_PORT;p&&d&&(c=new u({host:p,port:+d}));const h={};try{return U.sources=(await Promise.all([...l.map((async e=>{const o=await W(`${t.cdnURL||U.cdnURL}${e}`,c);return"string"==typeof o&&(h[e.replace(/(.*)\/|(.*)modules\/|stock\/(.*)indicators\/|maps\/(.*)modules\//gi,"")]=1),o})),...s.map((e=>W(e,c)))])).join(";\n"),G(),e.writeFileSync(o,U.sources),h}catch(e){H(1,"[cache] Unable to update local Highcharts cache.")}},D=async o=>{let r;const i=t.join(F,"manifest.json"),n=t.join(F,"sources.js");if(q=o,!e.existsSync(F)&&e.mkdirSync(F),!e.existsSync(i)||o.forceFetch)H(3,"[cache] Fetching and caching Highcharts dependencies."),r=await M(o,n);else{let t=!1;const s=JSON.parse(e.readFileSync(i));if(s.modules&&Array.isArray(s.modules)){const e={};s.modules.forEach((t=>e[t]=1)),s.modules=e}const{modules:a,coreScripts:l,indicators:c}=o,p=a.length+l.length+c.length;s.version!==o.version?(H(3,"[cache] Highcharts version mismatch in cache, need to re-fetch."),t=!0):Object.keys(s.modules||{}).length!==p?(H(3,"[cache] Cache and requested modules does not match, need to re-fetch."),t=!0):t=(o.modules||[]).some((e=>{if(!s.modules[e])return H(3,`[cache] The ${e} missing in cache, need to re-fetch.`),!0})),t?r=await M(o,n):(H(3,"[cache] Dependency cache is up to date, proceeding."),U.sources=e.readFileSync(n,"utf8"),r=s.modules,G())}await(async(o,r)=>{const i={version:o.version,modules:r||{}};U.activeManifest=i,H(4,"[cache] writing new manifest");try{e.writeFileSync(t.join(F,"manifest.json"),JSON.stringify(i),"utf8")}catch(e){H(1,`[cache] Error writing cache manifest: ${e}.`)}})(o,r)};var V=async e=>!!q&&await D(Object.assign(q,{version:e})),z=()=>U,J=()=>U.hcVersion;const K=m.randomBytes(64).toString("base64url"),X=f.join("tmp",`puppeteer-${K}`),B=[`--user-data-dir=${f.join(X,"profile")}`,"--autoplay-policy=user-gesture-required","--disable-background-networking","--disable-background-timer-throttling","--disable-backgrounding-occluded-windows","--disable-breakpad","--disable-client-side-phishing-detection","--disable-component-update","--disable-default-apps","--disable-dev-shm-usage","--disable-domain-reliability","--disable-extensions","--disable-features=AudioServiceOutOfProcess","--disable-hang-monitor","--disable-ipc-flooding-protection","--disable-notifications","--disable-offer-store-unmasked-wallet-cards","--disable-popup-blocking","--disable-print-preview","--disable-prompt-on-repost","--disable-renderer-backgrounding","--disable-session-crashed-bubble","--disable-setuid-sandbox","--disable-speech-api","--disable-sync","--hide-crash-restore-bubble","--hide-scrollbars","--ignore-gpu-blacklist","--metrics-recording-only","--mute-audio","--no-default-browser-check","--no-first-run","--no-pings","--no-sandbox","--no-zygote","--password-store=basic","--use-mock-keychain"],Y=b.fileURLToPath(new URL(".","undefined"==typeof document?require("url").pathToFileURL(__filename).href:v&&v.src||new URL("index.cjs",document.baseURI).href)),Q=e.readFileSync(Y+"/../templates/template.html","utf8");let Z;const ee=async()=>{if(!Z)return!1;const e=await Z.newPage();return await e.setContent(Q),await e.addScriptTag({path:Y+"/../.cache/sources.js"}),await e.evaluate((()=>window.setupHighcharts())),e.on("pageerror",(async t=>{H(1,"[page error]",t),await e.$eval("#container",((e,t)=>{window._displayErrors&&(e.innerHTML=t)}),`

Chart input data error

${t.toString()}`)})),e},te=async()=>{Z.connected&&await Z.close()};const oe=b.fileURLToPath(new URL(".","undefined"==typeof document?require("url").pathToFileURL(__filename).href:v&&v.src||new URL("index.cjs",document.baseURI).href)),re=async(e,t,o)=>await e.evaluate(((e,t)=>window.triggerExport(e,t)),t,o);var ie=async(o,r,i)=>{const n=[],s=async e=>{for(const e of n)await e.dispose();await e.evaluate((()=>{const[,...e]=document.getElementsByTagName("script"),[,...t]=document.getElementsByTagName("style"),[...o]=document.getElementsByTagName("link");for(const r of[...e,...t,...o])r.remove()}))};try{const a=()=>{};H(4,"[export] Determining export path.");const l=i.export;await o.evaluate((()=>requestAnimationFrame((()=>{}))));const c=l?.options?.chart?.displayErrors&&z().activeManifest.modules.debugger;await o.evaluate((e=>window._displayErrors=e),c);const p=()=>{};let u;if(r.indexOf&&(r.indexOf("=0||r.indexOf("=0)){if(H(4,"[export] Treating as SVG."),"svg"===l.type)return r;u=!0;const e=()=>{};await o.setContent((e=>`\n\n\n \n \n Highcarts Export\n \n \n \n
\n ${e}\n
\n \n\n\n`)(r)),e()}else if(H(4,"[export] Treating as config."),l.strInj){const e=()=>{};await re(o,{chart:{height:l.height,width:l.width}},i),e()}else{r.chart.height=l.height,r.chart.width=l.width;const e=()=>{};await re(o,r,i),e()}p();const d=()=>{},h=i.customCode.resources;if(h){if(h.js&&n.push(await o.addScriptTag({content:h.js})),h.files)for(const t of h.files)try{const r=!t.startsWith("http");n.push(await o.addScriptTag(r?{content:e.readFileSync(t,"utf8")}:{url:t}))}catch(e){H(4,"[export] JS file not found.")}const r=()=>{};if(h.css){let e=h.css.match(/@import\s*([^;]*);/g);if(e)for(let r of e)r&&(r=r.replace("url(","").replace("@import","").replace(/"/g,"").replace(/'/g,"").replace(/;/,"").replace(/\)/g,"").trim(),r.startsWith("http")?n.push(await o.addStyleTag({url:r})):i.customCode.allowFileResources&&n.push(await o.addStyleTag({path:t.join(oe,r)})));n.push(await o.addStyleTag({content:h.css.replace(/@import\s*([^;]*);/g,"")||" "}))}r()}d();const g=u?await o.$eval("#chart-container svg:first-of-type",(async(e,t)=>({chartHeight:e.height.baseVal.value*t,chartWidth:e.width.baseVal.value*t})),parseFloat(l.scale)):await o.evaluate((async()=>{const{chartHeight:e,chartWidth:t}=window.Highcharts.charts[0];return{chartHeight:e,chartWidth:t}})),f=()=>{},m=Math.ceil(g?.chartHeight||l.height),v=Math.ceil(g?.chartWidth||l.width);await o.setViewport({height:m,width:v,deviceScaleFactor:u?1:parseFloat(l.scale)});const y=u?e=>{document.body.style.zoom=e,document.body.style.margin="0px"}:()=>{document.body.style.zoom=1};await o.evaluate(y,parseFloat(l.scale));const{height:b,width:w,x:x,y:T}=await(e=>e.$eval("#chart-container",(e=>{const{x:t,y:o,width:r,height:i}=e.getBoundingClientRect();return{x:t,y:o,width:r,height:Math.trunc(i>1?i:500)}})))(o);let k;u||await o.setViewport({width:Math.round(w),height:Math.round(b),deviceScaleFactor:parseFloat(l.scale)}),f();const S=()=>{};if("svg"===l.type)k=await(async e=>await e.$eval("#container svg:first-of-type",(e=>e.outerHTML)))(o);else if("png"===l.type||"jpeg"===l.type)k=await(async(e,t,o,r)=>await Promise.race([e.screenshot({type:t,encoding:o,clip:r,omitBackground:!0}),new Promise(((e,t)=>setTimeout((()=>t(new Error("Rasterization timeout"))),1500)))]))(o,l.type,"base64",{width:v,height:m,x:x,y:T});else{if("pdf"!==l.type)throw`Unsupported output format ${l.type}`;k=await(async(e,t,o,r)=>await e.pdf({height:t+1,width:o,encoding:r}))(o,m,v,"base64")}return await o.evaluate((()=>{const e=Highcharts.charts;if(e.length)for(const t of e)t&&t.destroy(),Highcharts.charts.shift()})),S(),a(),await s(o),k}catch(e){return await s(o),H(1,`[export] Error encountered during export: ${e}`),e}};let ne,se=0,ae=0,le=0,ce=0,pe=0,ue={},de=!1;const he={create:async()=>{const e=d.v4();let t=!1;const o=(new Date).getTime();try{if(t=await ee(),!t||t.isClosed())throw"invalid page";H(3,`[pool] Successfully created a worker ${e} - took ${(new Date).getTime()-o} ms.`)}catch(e){throw H(1,`[pool] Error creating a new page in pool entry creation! ${e}`),"Error creating page"}return{id:e,page:t,workCount:Math.round(Math.random()*(ue.workLimit/2))}},validate:e=>!(ue.workLimit&&++e.workCount>ue.workLimit)||(H(3,"[pool] Worker failed validation:",`exceeded work limit (limit is ${ue.workLimit})`),!1),destroy:e=>{H(3,`[pool] Destroying pool entry ${e.id}.`),e.page&&e.page.close()},log:(e,t)=>console.log(`${t}: ${e}`)},ge=async e=>{ne=e.puppeteerArgs;try{await(async e=>{const t=[...B,...e||[]];if(!Z){let e=0;const o=async()=>{try{H(3,"[browser] attempting to get a browser instance (try",e+")"),Z=await g.launch({headless:"new",args:t,userDataDir:"./tmp/"})}catch(t){H(0,"[browser]",t),++e<25?(H(3,"[browser] failed:",t),await new Promise((e=>setTimeout(e,4e3))),await o()):H(0,"Max retries reached")}};try{await o()}catch(e){return H(0,"[browser] Unable to open browser"),!1}if(!Z)return H(0,"[browser] Unable to open browser"),!1}return Z})(ne)}catch(e){H(0,"[pool|browser]",e)}if(ue=e&&e.pool?{...e.pool}:{},H(3,"[pool] Initializing pool:",`min ${ue.initialWorkers}, max ${ue.maxWorkers}.`),de)return H(4,"[pool] Already initialized, please kill it before creating a new one.");ue.listenToProcessExits&&(H(4,"[pool] Attaching exit listeners to the process."),process.on("exit",(async()=>{await fe()})),process.on("SIGINT",((e,t)=>{H(4,`The ${e} event with code: ${t}.`),process.exit(1)})),process.on("SIGTERM",((e,t)=>{H(4,`The ${e} event with code: ${t}.`),process.exit(1)})),process.on("uncaughtException",(async(e,t)=>{H(4,`The ${t} error, message: ${e.message}.`)})));try{de=new h.Pool({...he,min:ue.initialWorkers,max:ue.maxWorkers,createRetryIntervalMillis:200,createTimeoutMillis:ue.acquireTimeout,acquireTimeoutMillis:ue.acquireTimeout,destroyTimeoutMillis:ue.acquireTimeout,idleTimeoutMillis:ue.timeoutThreshold,reapIntervalMillis:1e3,propagateCreateError:!1}),de.on("createFail",((e,t)=>{H(1,`[pool] Error when creating worker of an event id ${e}:`,t)})),de.on("acquireFail",((e,t)=>{H(1,`[pool] Error when acquiring worker of an event id ${e}:`,t)})),de.on("destroyFail",((e,t,o)=>{H(1,`[pool] Error when destroying worker of an id ${t.id}, event id ${e}:`,o)})),de.on("release",(e=>{H(4,`[pool] Releasing a worker of an id ${e.id}`)})),de.on("destroySuccess",((e,t)=>{H(4,`[pool] Destroyed a worker of an id ${t.id}`)}));const e=[];for(let t=0;t{de.release(e)})),H(3,`[pool] The pool is ready with ${ue.initialWorkers} initial resources waiting.`)}catch(e){throw H(1,`[pool] Couldn't create the worker pool ${e}`),e}};async function fe(){return H(3,"[pool] Killing all workers."),de.destroyed?(await te(),!0):(await de.destroy(),await te(),!0)}const me=async(e,t)=>{let o;const r=e=>{throw++ce,o&&de.release(o),"In pool.postWork: "+e};if(H(4,"[pool] Work received, starting to process."),ue.benchmarking&&ve(),++ae,!de)return H(1,"[pool] Work received, but pool has not been started."),r("Pool is not inited but work was posted to it!");try{H(4,"[pool] Acquiring worker"),o=await de.acquire().promise}catch(e){return r(`[pool] Error when acquiring available entry: ${e}`)}if(H(4,"[pool] Acquired worker handle"),!o.page)return r("Resolved worker page is invalid: pool setup is wonky");try{let i=(new Date).getTime();H(4,`[pool] Starting work on pool entry ${o.id}.`);const n=await ie(o.page,e,t);if(n instanceof Error)return"Rasterization timeout"===n.message&&(o.page.close(),o.page=await ee()),r(n);de.release(o);const s=(new Date).getTime()-i;return le+=s,pe=le/++se,H(4,`[pool] Work completed in ${s} ms.`),{data:n,options:t}}catch(e){r(`Error trying to perform puppeteer export: ${e}.`)}};function ve(){const{min:e,max:t,size:o,available:r,borrowed:i,pending:n,spareResourceCapacity:s}=de;H(4,`[pool] The minimum number of resources allowed by pool: ${e}.`),H(4,`[pool] The maximum number of resources allowed by pool: ${t}.`),H(4,`[pool] The number of all resources in pool (free or in use): ${o}.`),H(4,`[pool] The number of resources that are currently available: ${r}.`),H(4,`[pool] The number of resources that are currently acquired: ${i}.`),H(4,`[pool] The number of callers waiting to acquire a resource: ${n}.`),H(4,`[pool] The number of how many more resources can the pool manage/create: ${s}.`)}var ye=()=>({min:de.min,max:de.max,size:de.size,available:de.available,borrowed:de.borrowed,pending:de.pending,spareResourceCapacity:de.spareResourceCapacity}),be=()=>ae,we=()=>ce,xe=()=>pe,Te=()=>se;const ke=process.env.npm_package_version,Se=new Date;let He={};const Ee=()=>He,Re=(e,t,o=[])=>{const r=_(e);for(const[e,n]of Object.entries(t))r[e]="object"!=typeof(i=n)||Array.isArray(i)||null===i||o.includes(e)||void 0===r[e]?void 0!==n?n:r[e]:Re(r[e],n,o);var i;return r};function Le(e,t={},o=""){Object.keys(e).forEach((r=>{if(!["puppeteer","highcharts"].includes(r)){const i=e[r],n=t&&t[r];let s;void 0===i.value?Le(i,n,`${o}.${r}`):(void 0!==n&&(i.value=n),i.envLink&&("boolean"===i.type?i.value=I([process.env[i.envLink],i.value].find((e=>e||"false"===e))):"number"===i.type?(s=+process.env[i.envLink],i.value=s>=0?s:i.value):i.type.indexOf("]")>=0&&process.env[i.envLink]?i.value=process.env[i.envLink].split(","):i.value=process.env[i.envLink]||i.value))}}))}function Ce(e){let t={};for(const[o,r]of Object.entries(e))t[o]=Object.prototype.hasOwnProperty.call(r,"value")?r.value:Ce(r);return t}let Oe=!1;const _e=async(t,o)=>{H(4,"[chart] Starting exporting process.");const r=((e,t={})=>{let o={};return e.svg?(o=_(t),o.export.type=e.type||e.export.type,o.export.scale=e.scale||e.export.scale,o.export.outfile=e.outfile||e.export.outfile,o.payload={svg:e.svg}):o=Re(t,e,x),o.export.outfile=o.export?.outfile||`chart.${o.export?.type||"png"}`,o})(t,Ee()),i=r.export;return r.payload?.svg&&""!==r.payload.svg?je(r.payload.svg.trim(),r,o):i.infile&&i.infile.length?(H(4,"[chart] Attempting to export from an input file."),e.readFile(i.infile,"utf8",((e,t)=>e?H(1,`[chart] Error loading input file: ${e}.`):(r.export.instr=t,je(r.export.instr.trim(),r,o))))):i.instr&&""!==i.instr||i.options&&""!==i.options?(H(4,"[chart] Attempting to export from a raw input."),I(r.customCode?.allowCodeExecution)?Ie(r,o):"string"==typeof i.instr?je(i.instr.trim(),r,o):$e(r,i.instr||i.options,o)):(H(1,R(`[chart] No input specified.\n ${JSON.stringify(i,void 0," ")}.`)),o&&o(!1,{error:!0,message:"No input specified."}))},Ae=e=>{const{chart:t,exporting:o}=e.export?.options||O(e.export?.instr),r=O(e.export?.globalOptions);let i=e.export?.scale||o?.scale||r?.exporting?.scale||e.export?.defaultScale||1;return i=Math.max(.1,Math.min(i,5)),i=((e,t=1)=>{const o=Math.pow(10,t||0);return Math.round(+e*o)/o})(i,2),{height:e.export?.height||o?.sourceHeight||t?.height||r?.exporting?.sourceHeight||r?.chart?.height||e.export?.defaultHeight||400,width:e.export?.width||o?.sourceWidth||t?.width||r?.exporting?.sourceWidth||r?.chart?.width||e.export?.defaultWidth||600,scale:i}},$e=(t,o,r,i)=>{let{export:n,customCode:s}=t;const a="boolean"==typeof s.allowCodeExecution?s.allowCodeExecution:Oe;if(s){if("string"==typeof t.customCode.resources)t.customCode.resources=C(t.customCode.resources,I(t.customCode.allowFileResources));else if(!t.customCode.resources)try{const o=e.readFileSync("resources.json","utf8");t.customCode.resources=C(o,I(t.customCode.allowFileResources))}catch(e){H(3,"[chart] The default resources.json file not found.")}}else s=t.customCode={};if(!a&&s){if(s.callback||s.resources||s.customCode)return r&&r(!1,{error:!0,message:R("The callback, resources and customCode have been disabled for this\n server.")});s.callback=!1,s.resources=!1,s.customCode=!1}if(o&&(o.chart=o.chart||{},o.exporting=o.exporting||{},o.exporting.enabled=!1),n.constr=n.constr||"chart",n.type=L(n.type,n.outfile),"svg"===n.type&&(n.width=!1),["globalOptions","themeOptions"].forEach((t=>{try{n&&n[t]&&("string"==typeof n[t]&&n[t].endsWith(".json")?n[t]=O(e.readFileSync(n[t],"utf8"),!0):n[t]=O(n[t],!0))}catch(e){n[t]={},H(1,`[chart] The ${t} not found.`)}})),s.allowCodeExecution&&(s.customCode=j(s.customCode,s.allowFileResources)),s&&s.callback&&s.callback?.indexOf("{")<0)if(s.allowFileResources)try{s.callback=e.readFileSync(s.callback,"utf8")}catch(e){H(2,`[chart] Error loading callback: ${e}.`),s.callback=!1}else s.callback=!1;t.export={...t.export,...Ae(t)},me(n.strInj||o||i,t).then((e=>r(e))).catch((e=>(H(0,"[chart] When posting work:",e),r(!1,e))))},Ie=(e,t)=>{try{let o,r=e.export.instr||e.export.options;return"string"!=typeof r&&(o=r=A(r,e.customCode?.allowCodeExecution)),o=r.replaceAll(/\t|\n|\r/g,"").trim(),";"===o[o.length-1]&&(o=o.substring(0,o.length-1)),e.export.strInj=o,$e(e,!1,t)}catch(o){const r=R(`Malformed input detected for ${e.export?.requestId||"?"}:\n Please make sure that your JSON/JavaScript options\n are sent using the "options" attribute, and that if you're using\n SVG, it is unescaped.`);return H(1,r),t&&t(!1,JSON.stringify({error:!0,message:r}))}},je=(e,t,o)=>{const{allowCodeExecution:r}=t.customCode;if(e.indexOf("=0||e.indexOf("=0)return H(4,"[chart] Parsing input as SVG."),$e(t,!1,o,e);try{const r=JSON.parse(e.replaceAll(/\t|\n|\r/g," "));return $e(t,r,o)}catch(e){return I(r)?Ie(t,o):o&&o(!1,{error:!0,message:R("Only JSON configurations and SVG is allowed for this server. If\n this is your server, JavaScript exporting can be enabled by starting\n the server with the --allowCodeExecution flag.")})}},Pe={png:"image/png",jpeg:"image/jpeg",gif:"image/gif",pdf:"application/pdf",svg:"image/svg+xml"};let Ne=0;const Fe=[],Ue=[],qe=(e,t,o,r)=>{let i=!0;const{id:n,uniqueId:s,type:a,body:l}=r;return e.some((e=>{if(e){let r=e(t,o,n,s,a,l);return void 0!==r&&!0!==r&&(i=r),!0}})),i},Ge=(e,t)=>{(()=>{const e=process.hrtime.bigint()})();const o=Ee(),r=e.body,i=++Ne,n=d.v4().replace(/-/g,"");let s=L(r.type);if(!r)return t.status(400).send(R("Body is required. Sending a body? Make sure your Content-type header\n is correct. Accepted is application/json and multipart/form-data."));let a=O(r.infile||r.options||r.data);if(!a&&!r.svg)return H(2,R(`Request ${n} from ${e.headers["x-forwarded-for"]||e.connection.remoteAddress} was incorrect. Check your payload.`)),t.status(400).send(R("No correct chart data found. Please make sure you are using\n application/json or multipart/form-data headers, and that the chart\n data is in the 'infile', 'options' or 'data' attribute if sending\n JSON or in the 'svg' if sending SVG."));let l=!1;if(l=qe(Fe,e,t,{id:i,uniqueId:n,type:s,body:r}),!0!==l)return t.send(l);let c=!1;e.socket.on("close",(()=>{c=!0})),H(4,`[export] Got an incoming HTTP request ${n}.`),r.constr="string"==typeof r.constr&&r.constr||"chart";const p={export:{instr:a,type:s,constr:r.constr[0].toLowerCase()+r.constr.substr(1),height:r.height,width:r.width,scale:r.scale||o.export.scale,globalOptions:O(r.globalOptions,!0),themeOptions:O(r.themeOptions,!0)},customCode:{allowCodeExecution:Oe,allowFileResources:!1,resources:O(r.resources,!0),callback:r.callback,customCode:r.customCode}};a&&(p.export.instr=A(a,p.customCode.allowCodeExecution));const u=Re(o,p);if(u.export.options=a,u.payload={svg:r.svg||!1,b64:r.b64||!1,dataOptions:O(r.dataOptions,!0),noDownload:r.noDownload||!1,requestId:n},r.svg&&(h=u.payload.svg,["localhost","(10).(.*).(.*).(.*)","(127).(.*).(.*).(.*)","(172).(1[6-9]|2[0-9]|3[0-1]).(.*).(.*)","(192).(168).(.*).(.*)"].some((e=>h.match(`xlink:href="(?:(http://|https://))?${e}`)))))return t.status(400).send("SVG potentially contain at least one forbidden URL in xlink:href element.");var h;_e(u,((o,a)=>(e.socket.removeAllListeners("close"),c?H(3,R("[export] The client closed the connection before the chart was done\n processing.")):a?(H(1,R(`[export] Work: ${n} could not be completed, sending:\n ${a}`)),t.status(400).send(a.message)):o&&o.data?(s=o.options.export.type,qe(Ue,e,t,{id:i,body:o.data}),o.data?r.b64?"pdf"===s?t.send(Buffer.from(o.data,"utf8").toString("base64")):t.send(o.data):(t.header("Content-Type",Pe[s]||"image/png"),r.noDownload||t.attachment(`${e.params.filename||"chart"}.${s||"png"}`),"svg"===s?t.send(o.data):t.send(Buffer.from(o.data,"base64"))):void 0):(H(1,R(`[export] Unexpected return from chart generation, please check your\n data Request: ${n} is ${o.data}.`)),t.status(400).send("Unexpected return from chart generation, please check your data.")))))};const We=i();We.disable("x-powered-by"),We.use(r());const Me=n.memoryStorage(),De=n({storage:Me,limits:{fieldsSize:"50MB"}});We.use(De.any()),We.use(o.json({limit:"50mb"})),We.use(o.urlencoded({extended:!0,limit:"50mb"})),We.use(o.urlencoded({extended:!1,limit:"50mb"}));const Ve=e=>H(1,`[server] Socket error: ${e}`),ze=e=>{e.on("clientError",Ve),e.on("error",Ve),e.on("connection",(e=>e.on("error",(e=>Ve(e)))))},Je=async o=>{if(!o.enable)return!1;if(!o.ssl.enable&&!o.ssl.force){const e=s.createServer(We);ze(e),e.listen(o.port,o.host),H(3,`[server] Started HTTP server on ${o.host}:${o.port}.`)}if(o.ssl.enable){let r,i;try{r=await e.promises.readFile(t.posix.join(o.ssl.certPath,"server.key"),"utf8"),i=await e.promises.readFile(t.posix.join(o.ssl.certPath,"server.crt"),"utf8")}catch(e){H(1,`[server] Unable to load key/certificate from ${o.ssl.certPath}.`)}if(r&&i){const e=a.createServer(We);ze(e),e.listen(o.ssl.port,o.host),H(3,`[server] Started HTTPS server on ${o.host}:${o.ssl.port}.`)}}o.rateLimiting&&o.rateLimiting.enable&&![0,NaN].includes(o.rateLimiting.maxRequests)&&P(We,o.rateLimiting),We.use(i.static(t.posix.join(E,"public"))),(e=>{!!e&&e.get("/health",((e,t)=>{t.send({status:"OK",bootTime:Se,uptime:Math.floor(((new Date).getTime()-Se.getTime())/1e3/60)+" minutes",version:ke,highchartsVersion:J(),averageProcessingTime:xe(),performedExports:Te(),failedExports:we(),exportAttempts:be(),sucessRatio:Te()/be()*100,pool:ye()})}))})(We),(e=>{e.post("/",Ge),e.post("/:filename",Ge)})(We),(e=>{!!e&&e.get("/",((e,o)=>{o.sendFile(t.join(E,"public","index.html"))}))})(We),(e=>{!!e&&e.post("/change_hc_version/:newVersion",(async(e,t)=>{const o=process.env.HIGHCHARTS_ADMIN_TOKEN;if(!o||!o.length)return t.send({error:!0,message:"Server not configured to do run-time version changes: HIGHCHARTS_ADMIN_TOKEN not set"});const r=e.get("hc-auth");if(!r||r!==o)return t.send({error:!0,message:"Invalid or missing token: set token in the hc-auth header"});const i=e.params.newVersion;if(i){try{await V(i)}catch(e){t.send({error:!0,message:e})}t.send({version:J()})}else t.send({error:!0,message:"No new version supplied"})}))})(We)};var Ke={startServer:Je,getExpress:()=>i,getApp:()=>We,use:(e,...t)=>{We.use(e,...t)},get:(e,...t)=>{We.get(e,...t)},post:(e,...t)=>{We.post(e,...t)},enableRateLimiting:e=>P(We,e)},Xe={log:H,mapToNewConfig:e=>{const t={};for(const[o,r]of Object.entries(e)){const e=T[o]?T[o].split("."):[];e.reduce(((t,o,i)=>t[o]=e.length-1===i?r:t[o]||{}),t)}return t},setOptions:(t,o)=>(o?.length&&(He=function(t){const o=t.findIndex((e=>"loadConfig"===e.replace(/-/g,"")));if(o>-1&&t[o+1]){const r=t[o+1];try{if(r&&r.endsWith(".json"))return JSON.parse(e.readFileSync(r))}catch(e){H(1,`[config] Unable to load config from the ${r}: ${e}`)}}return{}}(o)),Le(w,He),He=Ce(w),t&&(He=Re(He,t,x)),o?.length&&(He=function(e,t,o){for(let o=0;o(i.length-1===a&&void 0!==n[s]&&(t[++o]?n[s]=t[o]||n[s]:(console.log(`Missing argument value for ${r}!`.red,"\n"),e=$())),n[s])),e)}return e}(He,o)),He),singleExport:t=>{t.export.instr=t.export.instr||t.export.options,_e(t,((t,o)=>{o&&(H(1,`[cli] ${o.message}`),process.exit(1));const{outfile:r,type:i}=t.options.export;e.writeFileSync(r||`chart.${i}`,"svg"!==i?Buffer.from(t.data,"base64"):t.data),fe()}))},startExport:_e,batchExport:t=>{const o=[];for(let r of t.export.batch.split(";"))r=r.split("="),2===r.length&&o.push(new Promise(((o,i)=>{_e({...t,export:{...t.export,infile:r[0],outfile:r[1]}},((t,r)=>{if(r)return i(r);e.writeFileSync(t.options.export.outfile,Buffer.from(t.data,"base64")),o()}))})));Promise.all(o).then((()=>{fe()})).catch((e=>{H(1,`[chart] Error encountered during batch export: ${e}`),fe()}))},server:Ke,startServer:Je,killPool:fe,initPool:async(e={})=>{var t,o;return t=e.customCode&&e.customCode.allowCodeExecution,Oe=I(t),(o=e.logging&&parseInt(e.logging.level))>=0&&o<=S.levelsDesc.length&&(S.level=o),e.logging&&e.logging.dest&&((e,t)=>{if(S={...S,dest:e||S.dest,file:t||S.file,toFile:!0},0===S.dest.length)return H(1,"[logger] File logging init: no path supplied.");S.dest.endsWith("/")||(S.dest+="/")})(e.logging.dest,e.logging.file||"highcharts-export-server.log"),await D(e.highcharts||{version:"latest"}),await ge({pool:e.pool||{initialWorkers:1,maxWorkers:1},puppeteerArgs:e.puppeteer?.args||[]}),e}};module.exports=Xe; -//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.cjs","sources":["../lib/schemas/config.js","../lib/logger.js","../lib/utils.js","../lib/server/rate_limit.js","../lib/fetch.js","../lib/cache.js","../lib/browser.js","../lib/export.js","../lib/benchmark.js","../templates/svg_export/svg_export.js","../lib/pool.js","../lib/server/routes/health.js","../lib/config.js","../lib/chart.js","../lib/server/routes/export.js","../lib/server/server.js","../lib/server/routes/ui.js","../lib/server/routes/change_hc_version.js","../lib/index.js"],"sourcesContent":["/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\n// Load .env into environment variables\nimport dotenv from 'dotenv';\n\ndotenv.config();\n\n// This is the configuration object with all options and their default values,\n// also from the .env file if one exists\nexport const defaultConfig = {\n  puppeteer: {\n    args: {\n      value: [],\n      type: 'string[]',\n      description: 'Array of arguments to send to puppeteer.'\n    }\n  },\n  highcharts: {\n    version: {\n      value: 'latest',\n      envLink: 'HIGHCHARTS_VERSION',\n      type: 'string',\n      description: 'Highcharts version to use.'\n    },\n    cdnURL: {\n      value: 'https://code.highcharts.com/',\n      envLink: 'HIGHCHARTS_CDN',\n      type: 'string',\n      description: 'The CDN URL of Highcharts scripts to use.'\n    },\n    coreScripts: {\n      envLink: 'HIGHCHARTS_CORE_SCRIPTS',\n      value: ['highcharts', 'highcharts-more', 'highcharts-3d'],\n      type: 'string[]',\n      description: 'Highcharts core scripts to fetch.'\n    },\n    modules: {\n      envLink: 'HIGHCHARTS_MODULES',\n      value: [\n        'stock',\n        'map',\n        'gantt',\n        'exporting',\n        'export-data',\n        'parallel-coordinates',\n        'accessibility',\n        'annotations-advanced',\n        'boost-canvas',\n        'boost',\n        'data',\n        'draggable-points',\n        'static-scale',\n        'broken-axis',\n        'heatmap',\n        'tilemap',\n        'timeline',\n        'treemap',\n        'item-series',\n        'drilldown',\n        'histogram-bellcurve',\n        'bullet',\n        'funnel',\n        'funnel3d',\n        'pyramid3d',\n        'networkgraph',\n        'pareto',\n        'pattern-fill',\n        'pictorial',\n        'price-indicator',\n        'sankey',\n        'arc-diagram',\n        'dependency-wheel',\n        'series-label',\n        'solid-gauge',\n        'sonification',\n        'stock-tools',\n        'streamgraph',\n        'sunburst',\n        'variable-pie',\n        'variwide',\n        'vector',\n        'venn',\n        'windbarb',\n        'wordcloud',\n        'xrange',\n        'no-data-to-display',\n        'drag-panes',\n        'debugger',\n        'dumbbell',\n        'lollipop',\n        'cylinder',\n        'organization',\n        'dotplot',\n        'marker-clusters',\n        'hollowcandlestick',\n        'heikinashi'\n      ],\n      type: 'string[]',\n      description: 'Highcharts modules to fetch.'\n    },\n    indicators: {\n      envLink: 'HIGHCHARTS_INDICATORS',\n      value: ['indicators-all'],\n      type: 'string[]',\n      description: 'Highcharts indicators to fetch.'\n    },\n    scripts: {\n      value: [\n        'https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.29.4/moment.min.js',\n        'https://cdnjs.cloudflare.com/ajax/libs/moment-timezone/0.5.34/moment-timezone-with-data.min.js'\n      ],\n      type: 'string[]',\n      description:\n        'Additional direct scripts/optional dependencies (e.g. moment.js).'\n    },\n    forceFetch: {\n      envLink: 'HIGHCHARTS_FORCE_FETCH',\n      value: false,\n      type: 'boolean',\n      description:\n        'Should all the scripts be refetched after rerunning the server.'\n    }\n  },\n  export: {\n    infile: {\n      value: false,\n      type: 'string',\n      description:\n        'The input file name along with a type (json or svg). It can be a correct JSON or SVG file.'\n    },\n    instr: {\n      value: false,\n      type: 'string',\n      description:\n        'An input in a form of a stringified JSON or SVG file. Overrides the --infile.'\n    },\n    options: {\n      value: false,\n      type: 'string',\n      description: 'An alias for the --instr option.'\n    },\n    outfile: {\n      value: false,\n      type: 'string',\n      description:\n        'The output filename along with a type (jpeg, png, pdf or svg). Ignores the --type flag.'\n    },\n    type: {\n      envLink: 'EXPORT_DEFAULT_TYPE',\n      value: 'png',\n      type: 'string',\n      description:\n        'The format of the file to export to. Can be jpeg, png, pdf or svg.'\n    },\n    constr: {\n      envLink: 'EXPORT_DEFAULT_CONSTR',\n      value: 'chart',\n      type: 'string',\n      description:\n        'The constructor to use. Can be chart, stockChart, mapChart or ganttChart.'\n    },\n    defaultHeight: {\n      envLink: 'EXPORT_DEFAULT_HEIGHT',\n      value: 400,\n      type: 'number',\n      description:\n        'The default height of the exported chart. Used when not found any value set.'\n    },\n    defaultWidth: {\n      envLink: 'EXPORT_DEFAULT_WIDTH',\n      value: 600,\n      type: 'number',\n      description:\n        'The default width of the exported chart. Used when not found any value set.'\n    },\n    defaultScale: {\n      envLink: 'EXPORT_DEFAULT_SCALE',\n      value: 1,\n      type: 'number',\n      description:\n        'The default scale of the exported chart. Ranges between 1 and 5.'\n    },\n    height: {\n      type: 'number',\n      value: false,\n      description:\n        'The default height of the exported chart. Overrides the option in the chart settings.'\n    },\n    width: {\n      type: 'number',\n      value: false,\n      description:\n        'The width of the exported chart. Overrides the option in the chart settings.'\n    },\n    scale: {\n      value: false,\n      type: 'number',\n      description: 'The scale of the exported chart. Ranges between 1 and 5.'\n    },\n    globalOptions: {\n      value: false,\n      type: 'string',\n      description:\n        'A stringified JSON or a filename with options to be passed into the Highcharts.setOptions.'\n    },\n    themeOptions: {\n      value: false,\n      type: 'string',\n      description:\n        'A stringified JSON or a filename with theme options to be passed into the Highcharts.setOptions.'\n    },\n    batch: {\n      value: false,\n      type: 'string',\n      description:\n        'Starts a batch job. A string that contains input/output pairs: \"in=out;in=out;..\".'\n    }\n  },\n  customCode: {\n    allowCodeExecution: {\n      envLink: 'HIGHCHARTS_ALLOW_CODE_EXECUTION',\n      value: false,\n      type: 'boolean',\n      description:\n        'If set to true, allow for the execution of arbitrary code when exporting.'\n    },\n    allowFileResources: {\n      envLink: 'HIGHCHARTS_ALLOW_FILE_RESOURCES',\n      value: true,\n      type: 'boolean',\n      description:\n        'Allow injecting resources from the filesystem. Has no effect when running as a server.'\n    },\n    customCode: {\n      value: false,\n      type: 'string',\n      description:\n        'A function to be called before chart initialization. Can be a filename with the js extension.'\n    },\n    callback: {\n      value: false,\n      type: 'string',\n      description: 'A JavaScript file with a function to run on construction.'\n    },\n    resources: {\n      value: false,\n      type: 'string',\n      description:\n        'An additional resource in a form of stringified JSON. It can contain files, js and css sections.'\n    },\n    loadConfig: {\n      value: false,\n      type: 'string',\n      description: 'A file that contains a pre-defined config to use.'\n    },\n    createConfig: {\n      value: false,\n      type: 'string',\n      description:\n        'Allows to set options through a prompt and save in a provided config file.'\n    }\n  },\n  server: {\n    enable: {\n      envLink: 'HIGHCHARTS_SERVER_ENABLE',\n      value: false,\n      type: 'boolean',\n      cliName: 'enableServer',\n      description: 'If set to true, starts a server on 0.0.0.0.'\n    },\n    host: {\n      envLink: 'HIGHCHARTS_SERVER_HOST',\n      value: '0.0.0.0',\n      type: 'string',\n      description:\n        'The hostname of the server. Also starts a server listening on the supplied hostname.'\n    },\n    port: {\n      envLink: 'HIGHCHARTS_SERVER_PORT',\n      value: 7801,\n      type: 'number',\n      description: 'The port to use for the server. Defaults to 7801.'\n    },\n    ssl: {\n      enable: {\n        envLink: 'HIGHCHARTS_SERVER_SSL_ENABLE',\n        value: false,\n        type: 'boolean',\n        cliName: 'enableSsl',\n        description: 'Enables the SSL protocol.'\n      },\n      force: {\n        envLink: 'HIGHCHARTS_SERVER_SSL_FORCE',\n        value: false,\n        type: 'boolean',\n        cliName: 'sslForced',\n        description:\n          'If set to true, forces the server to only serve over HTTPS.'\n      },\n      port: {\n        envLink: 'HIGHCHARTS_SERVER_SSL_PORT',\n        value: 443,\n        type: 'number',\n        cliName: 'sslPort',\n        description: 'The port on which to run the SSL server.'\n      },\n      certPath: {\n        envLink: 'HIGHCHARTS_SSL_CERT_PATH',\n        value: '',\n        type: 'string',\n        description: 'The path to the SSL certificate/key.'\n      }\n    },\n    rateLimiting: {\n      enable: {\n        envLink: 'HIGHCHARTS_RATE_LIMIT_ENABLE',\n        value: false,\n        type: 'boolean',\n        cliName: 'enableRateLimiting',\n        description: 'Enables rate limiting.'\n      },\n      maxRequests: {\n        envLink: 'HIGHCHARTS_RATE_LIMIT_MAX',\n        value: 10,\n        type: 'number',\n        description: 'Max requests allowed in a one minute.'\n      },\n      window: {\n        envLink: 'HIGHCHARTS_RATE_LIMIT_WINDOW',\n        value: 1,\n        type: 'number',\n        description: 'The time window in minutes for rate limiting.'\n      },\n      delay: {\n        envLink: 'HIGHCHARTS_RATE_LIMIT_DELAY',\n        value: 0,\n        type: 'number',\n        description:\n          'The amount to delay each successive request before hitting the max.'\n      },\n      trustProxy: {\n        envLink: 'HIGHCHARTS_RATE_LIMIT_TRUST_PROXY',\n        value: false,\n        type: 'boolean',\n        description: 'Set this to true if behind a load balancer.'\n      },\n      skipKey: {\n        envLink: 'HIGHCHARTS_RATE_LIMIT_SKIP_KEY',\n        value: '',\n        type: 'number|string',\n        description:\n          'Allows bypassing the rate limiter and should be provided with skipToken argument.'\n      },\n      skipToken: {\n        envLink: 'HIGHCHARTS_RATE_LIMIT_SKIP_TOKEN',\n        value: '',\n        type: 'number|string',\n        description:\n          'Allows bypassing the rate limiter and should be provided with skipKey argument.'\n      }\n    }\n  },\n  pool: {\n    initialWorkers: {\n      envLink: 'HIGHCHARTS_POOL_MIN_WORKERS',\n      value: 4,\n      type: 'number',\n      description: 'The number of initial workers to spawn.'\n    },\n    maxWorkers: {\n      envLink: 'HIGHCHARTS_POOL_MAX_WORKERS',\n      value: 8,\n      type: 'number',\n      description: 'The number of max workers to spawn.'\n    },\n    workLimit: {\n      envLink: 'HIGHCHARTS_POOL_WORK_LIMIT',\n      value: 40,\n      type: 'number',\n      description:\n        'The pieces of work that can be performed before restarting process.'\n    },\n    queueSize: {\n      envLink: 'HIGHCHARTS_POOL_QUEUE_SIZE',\n      value: 5,\n      type: 'number',\n      description: 'The size of the request overflow queue.'\n    },\n    timeoutThreshold: {\n      envLink: 'HIGHCHARTS_POOL_TIMEOUT',\n      value: 5000,\n      type: 'number',\n      description: 'The number of milliseconds before timing out.'\n    },\n    acquireTimeout: {\n      envLink: 'HIGHCHARTS_POOL_ACQUIRE_TIMEOUT',\n      value: 5000,\n      type: 'number',\n      description:\n        'The number of milliseconds to wait for acquiring a resource.'\n    },\n    reaper: {\n      envLink: 'HIGHCHARTS_POOL_ENABLE_REAPER',\n      value: true,\n      type: 'boolean',\n      description:\n        'Whether or not to evict workers after a certain time period.'\n    },\n    benchmarking: {\n      envLink: 'HIGHCHARTS_POOL_BENCHMARKING',\n      value: false,\n      type: 'boolean',\n      description: 'Enable benchmarking.'\n    },\n    listenToProcessExits: {\n      envLink: 'HIGHCHARTS_POOL_LISTEN_TO_PROCESS_EXITS',\n      value: true,\n      type: 'boolean',\n      description:\n        'Set to false in order to skip attaching process.exit handlers.'\n    }\n  },\n  logging: {\n    level: {\n      envLink: 'HIGHCHARTS_LOG_LEVEL',\n      value: 4,\n      type: 'number',\n      cliName: 'logLevel',\n      description:\n        'The log level (0: silent, 1: error, 2: warning, 3: notice, 4: verbose).'\n    },\n    file: {\n      envLink: 'HIGHCHARTS_LOG_FILE',\n      value: 'highcharts-export-server.log',\n      type: 'string',\n      cliName: 'logFile',\n      description:\n        'A name of a log file. The --logDest also needs to be set to enable file logging.'\n    },\n    dest: {\n      envLink: 'HIGHCHARTS_LOG_DEST',\n      value: 'log/',\n      type: 'string',\n      cliName: 'logDest',\n      description: 'The path to store log files. Also enables file logging.'\n    }\n  },\n  ui: {\n    enable: {\n      envLink: 'HIGHCHARTS_UI_ENABLE',\n      value: false,\n      type: 'boolean',\n      cliName: 'enableUi',\n      description: 'Enables the UI for the export server.'\n    },\n    route: {\n      envLink: 'HIGHCHARTS_UI_ROUTE',\n      value: '/',\n      type: 'string',\n      cliName: 'uiRoute',\n      description: 'The route to attach the UI to.'\n    }\n  },\n  other: {\n    noLogo: {\n      envLink: 'HIGHCHARTS_NO_LOGO',\n      value: false,\n      type: 'boolean',\n      description:\n        'Skip printing the logo on a startup. Will be replaced by a simple text.'\n    }\n  },\n  payload: {}\n};\n\n// The config descriptions object for the prompts functionality. It contains\n// information like:\n// * Type of a prompt\n// * Name of an option\n// * Short description of a chosen option\n// * Initial value\nexport const promptsConfig = {\n  puppeteer: [\n    {\n      type: 'list',\n      name: 'args',\n      message: 'Puppeteer arguments',\n      initial: defaultConfig.puppeteer.args.value.join(','),\n      separator: ','\n    }\n  ],\n  highcharts: [\n    {\n      type: 'text',\n      name: 'version',\n      message: 'Highcharts version',\n      initial: defaultConfig.highcharts.version.value\n    },\n    {\n      type: 'text',\n      name: 'cdnURL',\n      message: 'The url of CDN',\n      initial: defaultConfig.highcharts.cdnURL.value\n    },\n    {\n      type: 'multiselect',\n      name: 'modules',\n      message: 'Available modules',\n      instructions: 'Space: Select specific, A: Select all, Enter: Confirm.',\n      choices: defaultConfig.highcharts.modules.value\n    },\n    {\n      type: 'list',\n      name: 'scripts',\n      message: 'Custom scripts',\n      initial: defaultConfig.highcharts.scripts.value.join(','),\n      separator: ','\n    },\n    {\n      type: 'toggle',\n      name: 'forceFetch',\n      message: 'Should refetch all the scripts after each server rerun',\n      initial: defaultConfig.highcharts.forceFetch.value\n    }\n  ],\n  export: [\n    {\n      type: 'select',\n      name: 'type',\n      message: 'The default type of a file to export to',\n      hint: `Default: ${defaultConfig.export.type.value}`,\n      initial: 0,\n      choices: ['png', 'jpeg', 'pdf', 'svg']\n    },\n    {\n      type: 'select',\n      name: 'constr',\n      message: 'The default constructor for Highcharts to use',\n      hint: `Default: ${defaultConfig.export.constr.value}`,\n      initial: 0,\n      choices: ['chart', 'stockChart', 'mapChart', 'ganttChart']\n    },\n    {\n      type: 'number',\n      name: 'defaultHeight',\n      message: 'The default fallback height of the exported chart',\n      initial: defaultConfig.export.defaultHeight.value\n    },\n    {\n      type: 'number',\n      name: 'defaultWidth',\n      message: 'The default fallback width of the exported chart',\n      initial: defaultConfig.export.defaultWidth.value\n    },\n    {\n      type: 'number',\n      name: 'defaultScale',\n      message: 'The default fallback scale of the exported chart',\n      initial: defaultConfig.export.defaultScale.value,\n      min: 0.1,\n      max: 5\n    }\n  ],\n  customCode: [\n    {\n      type: 'toggle',\n      name: 'allowCodeExecution',\n      message: 'Allow to execute custom code',\n      initial: defaultConfig.customCode.allowCodeExecution.value\n    },\n    {\n      type: 'toggle',\n      name: 'allowFileResources',\n      message: 'Allow file resources',\n      initial: defaultConfig.customCode.allowFileResources.value\n    }\n  ],\n  server: [\n    {\n      type: 'toggle',\n      name: 'enable',\n      message: 'Starts a server on 0.0.0.0',\n      initial: defaultConfig.server.enable.value\n    },\n    {\n      type: 'text',\n      name: 'host',\n      message: 'A hostname of a server',\n      initial: defaultConfig.server.host.value\n    },\n    {\n      type: 'number',\n      name: 'port',\n      message: 'A port of a server',\n      initial: defaultConfig.server.port.value\n    },\n    {\n      type: 'toggle',\n      name: 'ssl.enable',\n      message: 'Enable SSL protocol',\n      initial: defaultConfig.server.ssl.enable.value\n    },\n    {\n      type: 'toggle',\n      name: 'ssl.force',\n      message: 'Force to only serve over HTTPS',\n      initial: defaultConfig.server.ssl.force.value\n    },\n    {\n      type: 'number',\n      name: 'ssl.port',\n      message: 'Port on which to run the SSL server',\n      initial: defaultConfig.server.ssl.port.value\n    },\n    {\n      type: 'text',\n      name: 'ssl.certPath',\n      message: 'A path where to find the SSL certificate/key',\n      initial: defaultConfig.server.ssl.certPath.value\n    },\n    {\n      type: 'toggle',\n      name: 'rateLimiting.enable',\n      message: 'Enable rate limiting',\n      initial: defaultConfig.server.rateLimiting.enable.value\n    },\n    {\n      type: 'number',\n      name: 'rateLimiting.maxRequests',\n      message: 'Max requests allowed in a one minute',\n      initial: defaultConfig.server.rateLimiting.maxRequests.value\n    },\n    {\n      type: 'number',\n      name: 'rateLimiting.window',\n      message: 'The time window in minutes for rate limiting',\n      initial: defaultConfig.server.rateLimiting.window.value\n    },\n    {\n      type: 'number',\n      name: 'rateLimiting.delay',\n      message:\n        'The amount to delay each successive request before hitting the max',\n      initial: defaultConfig.server.rateLimiting.delay.value\n    },\n    {\n      type: 'toggle',\n      name: 'rateLimiting.trustProxy',\n      message: 'Set this to true if behind a load balancer',\n      initial: defaultConfig.server.rateLimiting.trustProxy.value\n    },\n    {\n      type: 'text',\n      name: 'rateLimiting.skipKey',\n      message:\n        'Allows bypassing the rate limiter and should be provided with skipToken argument',\n      initial: defaultConfig.server.rateLimiting.skipKey.value\n    },\n    {\n      type: 'text',\n      name: 'rateLimiting.skipToken',\n      message:\n        'Allows bypassing the rate limiter and should be provided with skipKey argument',\n      initial: defaultConfig.server.rateLimiting.skipToken.value\n    }\n  ],\n  pool: [\n    {\n      type: 'number',\n      name: 'initialWorkers',\n      message: 'The number of initial workers to spawn',\n      initial: defaultConfig.pool.initialWorkers.value\n    },\n    {\n      type: 'number',\n      name: 'maxWorkers',\n      message: 'The number of max workers to spawn',\n      initial: defaultConfig.pool.maxWorkers.value\n    },\n    {\n      type: 'number',\n      name: 'workLimit',\n      message:\n        'The pieces of work that can be performed before restarting a puppeteer process',\n      initial: defaultConfig.pool.workLimit.value\n    },\n    {\n      type: 'number',\n      name: 'queueSize',\n      message: 'The size of the request overflow queue',\n      initial: defaultConfig.pool.queueSize.value\n    },\n    {\n      type: 'number',\n      name: 'timeoutThreshold',\n      message: 'The number of seconds before timing out',\n      initial: defaultConfig.pool.timeoutThreshold.value\n    },\n    {\n      type: 'number',\n      name: 'acquireTimeout',\n      message: 'The number of milliseconds to wait for acquiring a resource',\n      initial: defaultConfig.pool.acquireTimeout.value\n    },\n    {\n      type: 'toggle',\n      name: 'reaper',\n      message: 'The reaper to remove hanging processes',\n      initial: defaultConfig.pool.reaper.value\n    },\n    {\n      type: 'toggle',\n      name: 'benchmarking',\n      message: 'Set benchmarking',\n      initial: defaultConfig.pool.benchmarking.value\n    },\n    {\n      type: 'toggle',\n      name: 'listenToProcessExits',\n      message: 'Set to false in order to skip attaching process.exit handlers',\n      initial: defaultConfig.pool.listenToProcessExits.value\n    }\n  ],\n  logging: [\n    {\n      type: 'number',\n      name: 'level',\n      message:\n        'The log level (0: silent, 1: error, 2: warning, 3: notice, 4: verbose)',\n      initial: defaultConfig.logging.level.value,\n      round: 0,\n      min: 0,\n      max: 4\n    },\n    {\n      type: 'text',\n      name: 'file',\n      message:\n        'A name of a log file. The --logDest also needs to be set to enable file logging',\n      initial: defaultConfig.logging.file.value\n    },\n    {\n      type: 'text',\n      name: 'dest',\n      message: 'A path to log files. It enables file logging',\n      initial: defaultConfig.logging.dest.value\n    }\n  ],\n  ui: [\n    {\n      type: 'toggle',\n      name: 'enable',\n      message: 'Enable UI for the export server',\n      initial: defaultConfig.ui.enable.value\n    },\n    {\n      type: 'text',\n      name: 'route',\n      message: 'A route to attach the UI to',\n      initial: defaultConfig.ui.route.value\n    }\n  ],\n  other: [\n    {\n      type: 'toggle',\n      name: 'noLogo',\n      message:\n        'Skip printing the logo on a startup. Will be replaced by a simple text',\n      initial: defaultConfig.other.noLogo.value\n    }\n  ]\n};\n\n// Absolute props that, in case of merging recursively, need to be force merged\nexport const absoluteProps = [\n  'options',\n  'globalOptions',\n  'themeOptions',\n  'resources',\n  'payload'\n];\n\n// Argument nesting level of all export server options\nexport const nestedArgs = {};\n\n/**\n * Creates nested arguments chain for all options\n *\n * @param {object} obj - The object based on which the initial configuration be\n * made.\n * @param {string } propChain - Required for creating a string chain of\n * properties for nested arguments.\n */\nconst createNestedArgs = (obj, propChain = '') => {\n  Object.keys(obj).forEach((k) => {\n    if (!['puppeteer', 'highcharts'].includes(k)) {\n      const entry = obj[k];\n      if (typeof entry.value === 'undefined') {\n        // Go deeper in the nested arguments\n        createNestedArgs(entry, `${propChain}.${k}`);\n      } else {\n        // Create the chain of nested arguments\n        nestedArgs[entry.cliName || k] = `${propChain}.${k}`.substring(1);\n      }\n    }\n  });\n};\n\ncreateNestedArgs(defaultConfig);\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { appendFile, existsSync, mkdirSync } from 'fs';\n\nimport { defaultConfig } from './schemas/config.js';\n\n// The default logging config\nlet logging = {\n  // Flags for logging status\n  toConsole: true,\n  toFile: false,\n  pathCreated: false,\n  // Log levels\n  levelsDesc: [\n    {\n      title: 'error',\n      color: 'red'\n    },\n    {\n      title: 'warning',\n      color: 'yellow'\n    },\n    {\n      title: 'notice',\n      color: 'blue'\n    },\n    {\n      title: 'verbose',\n      color: 'gray'\n    }\n  ],\n  // Log listeners\n  listeners: []\n};\n\n// Gather init logging options\nfor (const [key, option] of Object.entries(defaultConfig.logging)) {\n  logging[key] = option.value;\n}\n\n/**\n * Logs a message. Accepts a variable amount of arguments. Arguments after\n * `level` will be passed directly to console.log, and/or will be joined\n * and appended to the log file.\n *\n * @param {any} args - An array of arguments where the first is the log level\n * and the rest are strings to build a message with.\n */\nexport const log = (...args) => {\n  const [newLevel, ...texts] = args;\n\n  // Current logging options\n  const { level, levelsDesc } = logging;\n\n  // Check if log level is within a correct range\n  if (newLevel === 0 || newLevel > level || level > levelsDesc.length) {\n    return;\n  }\n\n  // Get rid of the GMT text information\n  const newDate = new Date().toString().split('(')[0].trim();\n\n  // Create a message's prefix\n  const prefix = `${newDate} [${levelsDesc[newLevel - 1].title}] -`;\n\n  // Call available log listeners\n  logging.listeners.forEach((fn) => {\n    fn(prefix, texts.join(' '));\n  });\n\n  // Log to file\n  if (logging.toFile) {\n    if (!logging.pathCreated) {\n      // Create if does not exist\n      !existsSync(logging.dest) && mkdirSync(logging.dest);\n\n      // We now assume the path is available, e.g. it's the responsibility\n      // of the user to create the path with the correct access rights.\n      logging.pathCreated = true;\n    }\n\n    // Add the content to a file\n    appendFile(\n      `${logging.dest}${logging.file}`,\n      [prefix].concat(texts).join(' ') + '\\n',\n      (error) => {\n        if (error) {\n          console.log(`[logger] Unable to write to log file: ${error}`);\n          logging.toFile = false;\n        }\n      }\n    );\n  }\n\n  // Log to console\n  if (logging.toConsole) {\n    console.log.apply(\n      undefined,\n      [prefix.toString()[logging.levelsDesc[newLevel - 1].color]].concat(texts)\n    );\n  }\n};\n\n/**\n * Sets the file logging configuration.\n *\n * @param {string} logDest - A path to log to.\n * @param {string} logFile - The name of the log file.\n */\nexport const enableFileLogging = (logDest, logFile) => {\n  // Update logging options\n  logging = {\n    ...logging,\n    dest: logDest || logging.dest,\n    file: logFile || logging.file,\n    toFile: true\n  };\n\n  if (logging.dest.length === 0) {\n    return log(1, '[logger] File logging init: no path supplied.');\n  }\n\n  if (!logging.dest.endsWith('/')) {\n    logging.dest += '/';\n  }\n};\n\n/**\n * Adds a log listener.\n *\n * @param {function} fn - The function to call when getting a log event.\n */\nexport const listen = (fn) => {\n  logging.listeners.push(fn);\n};\n\n/**\n * Sets the current log level. Log levels are:\n * - 0 = no logging\n * - 1 = error\n * - 2 = warning\n * - 3 = notice\n * - 4 = verbose\n *\n * @param {number} newLevel - The new log level (0 - 4).\n */\nexport const setLogLevel = (newLevel) => {\n  if (newLevel >= 0 && newLevel <= logging.levelsDesc.length) {\n    logging.level = newLevel;\n  }\n};\n\n/**\n * Enables or disables logging to the stdout.\n *\n * @param {boolean} enabled - Whether log to console or not.\n */\nexport const toggleSTDOut = (enabled) => {\n  logging.toConsole = enabled;\n};\n\nexport default {\n  log,\n  enableFileLogging,\n  listen,\n  setLogLevel,\n  toggleSTDOut\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { readFileSync } from 'fs';\nimport { fileURLToPath } from 'url';\n\nimport { defaultConfig } from '../lib/schemas/config.js';\nimport { log } from './logger.js';\n\nconst MAX_BACKOFF_ATTEMPTS = 6;\n\nexport const __dirname = fileURLToPath(new URL('../.', import.meta.url));\n\n/**\n * Clears text from whitespaces with a regex rule.\n *\n * @param {string} rule - The rule for clearing a string, default to /\\s\\s+/g.\n * @return {string} - Cleared text.\n */\nexport const clearText = (text, rule = /\\s\\s+/g, replacer = ' ') =>\n  text.replaceAll(rule, replacer).trim();\n\n/**\n * Delays calling the function by time calculated based on the backoff\n * algorithm.\n *\n * @param {function} fn - A function to try to call with the backoff algorithm\n * on.\n * @param {number} attempt - The number of an attempt, where the first one is 0.\n */\nexport const expBackoff = async (fn, attempt = 0, ...args) => {\n  try {\n    // Try to call the function\n    return await fn(...args);\n  } catch (error) {\n    // Calculate delay in ms\n    const delayInMs = 2 ** attempt * 1000;\n\n    // If the attempt exceeds the maximum attempts of reapeat, throw an error\n    if (++attempt >= MAX_BACKOFF_ATTEMPTS) {\n      throw error;\n    }\n\n    // Wait given amount of time\n    await new Promise((response) => setTimeout(response, delayInMs));\n    log(\n      3,\n      `[pool] Waited ${delayInMs}ms until next call for the resource id: ${args[0]}.`\n    );\n\n    // Try again\n    return expBackoff(fn, attempt, ...args);\n  }\n};\n\n/**\n * Fixes to supported type format if MIME.\n *\n * @param {string} type - Type to be corrected.\n * @param {string} outfile - Name of the outfile.\n */\nexport const fixType = (type, outfile) => {\n  // MIME types\n  const mimeTypes = {\n    'image/png': 'png',\n    'image/jpeg': 'jpeg',\n    'application/pdf': 'pdf',\n    'image/svg+xml': 'svg'\n  };\n\n  // Formats\n  const formats = ['png', 'jpeg', 'pdf', 'svg'];\n\n  // Check if type and outfile's extensions are the same\n  if (outfile) {\n    const outType = outfile.split('.').pop();\n\n    // Check if extension has a correct type\n    if (formats.includes(outType) && type !== outType) {\n      type = outType;\n    }\n  }\n\n  // Return a correct type\n  return mimeTypes[type] || formats.find((t) => t === type) || 'png';\n};\n\n/**\n * Handles the provided resources.\n *\n * @param {string} resources - The stringified resources.\n * @param {string} allowFileResources - Decide if resources from file are\n * allowed.\n */\nexport const handleResources = (resources = false, allowFileResources) => {\n  const allowedProps = ['js', 'css', 'files'];\n\n  let handledResources = resources;\n  let correctResources = false;\n\n  // Try to load resources from a file\n  if (allowFileResources && resources.endsWith('.json')) {\n    try {\n      if (!resources) {\n        handledResources = isCorrectJSON(\n          readFileSync('resources.json', 'utf8')\n        );\n      } else if (resources && resources.endsWith('.json')) {\n        handledResources = isCorrectJSON(readFileSync(resources, 'utf8'));\n      } else {\n        handledResources = isCorrectJSON(resources);\n        if (handledResources === true) {\n          handledResources = isCorrectJSON(\n            readFileSync('resources.json', 'utf8')\n          );\n        }\n      }\n    } catch (notice) {\n      return log(3, `[cli] No resources found.`);\n    }\n  } else {\n    // Try to get JSON\n    handledResources = isCorrectJSON(resources);\n\n    // Get rid of the files section\n    if (!allowFileResources) {\n      delete handledResources.files;\n    }\n  }\n\n  // Filter from unnecessary properties\n  for (const propName in handledResources) {\n    if (!allowedProps.includes(propName)) {\n      delete handledResources[propName];\n    } else if (!correctResources) {\n      correctResources = true;\n    }\n  }\n\n  // Check if at least one of allowed properties is present\n  if (!correctResources) {\n    return log(3, `[cli] No resources found.`);\n  }\n\n  // Handle files section\n  if (handledResources.files) {\n    handledResources.files = handledResources.files.map((item) => item.trim());\n    if (!handledResources.files || handledResources.files.length <= 0) {\n      delete handledResources.files;\n    }\n  }\n\n  // Return resources\n  return handledResources;\n};\n\n/**\n * Checks if provided data is or can be a correct JSON.\n *\n * @param {any} data - Data to be checked.\n * @param {boolean} toString - If true, return stringified representation.\n */\nexport function isCorrectJSON(data, toString) {\n  try {\n    // Get the string representation if not already before parsing\n    const parsedData = JSON.parse(\n      typeof data !== 'string' ? JSON.stringify(data) : data\n    );\n\n    // Return a stringified representation of a JSON if required\n    if (typeof parsedData !== 'string' && toString) {\n      return JSON.stringify(parsedData);\n    }\n\n    // Return a JSON\n    return parsedData;\n  } catch (error) {\n    return false;\n  }\n}\n\n/**\n * Checks if item is an object.\n *\n * @param {any} item - Item to be checked.\n */\nexport const isObject = (item) =>\n  typeof item === 'object' && !Array.isArray(item) && item !== null;\n\n/**\n * Checks if string contains private range urls.\n *\n * @export utils\n * @param item {string} item to be checked\n */\nexport const isPrivateRangeUrlFound = (item) => {\n  return [\n    'localhost',\n    '(10).(.*).(.*).(.*)',\n    '(127).(.*).(.*).(.*)',\n    '(172).(1[6-9]|2[0-9]|3[0-1]).(.*).(.*)',\n    '(192).(168).(.*).(.*)'\n  ].some((ipRegEx) =>\n    item.match(`xlink:href=\"(?:(http://|https://))?${ipRegEx}`)\n  );\n};\n\n/**\n * Creates and returns a deep copy of the given object.\n *\n * @param {object} object - Object to copy.\n * @return {object} - Deep copy of the object.\n */\nexport const deepCopy = (obj) => {\n  if (obj === null || typeof obj !== 'object') {\n    return obj;\n  }\n\n  const copy = Array.isArray(obj) ? [] : {};\n\n  for (const key in obj) {\n    if (Object.prototype.hasOwnProperty.call(obj, key)) {\n      copy[key] = deepCopy(obj[key]);\n    }\n  }\n\n  return copy;\n};\n\n/**\n * Stringifies object with options. Possible to preserve functions with\n * allowFunctions flag.\n *\n * @param {object} options - Options to stringify.\n * @param {boolean} allowFunctions - Flag for keeping functions.\n */\nexport const optionsStringify = (options, allowFunctions) => {\n  const replacerCallback = (name, value) => {\n    if (typeof value === 'string') {\n      value = value.trim();\n\n      // If allowFunctions is set to true, preserve functions\n      if (\n        (value.startsWith('function(') || value.startsWith('function (')) &&\n        value.endsWith('}')\n      ) {\n        value = allowFunctions\n          ? `EXP_FUN${(value + '').replaceAll(/\\n|\\t|\\r/g, ' ')}EXP_FUN`\n          : undefined;\n      }\n    }\n\n    return typeof value === 'function'\n      ? `EXP_FUN${(value + '').replaceAll(/\\n|\\t|\\r/g, ' ')}EXP_FUN`\n      : value;\n  };\n\n  // Stringify options and if required, replace special functions marks\n  return JSON.stringify(options, replacerCallback).replaceAll(\n    /\"EXP_FUN|EXP_FUN\"/g,\n    ''\n  );\n};\n\n/**\n * Prints the export server logo.\n *\n * @param {boolean} noLogo - Whether to display logo or text.\n */\nexport const printLogo = (noLogo) => {\n  // Get package version either from env or from package.json\n  const packageVersion =\n    process.env.npm_package_version ||\n    JSON.parse(readFileSync(new URL('../package.json', import.meta.url)))\n      .version;\n\n  // Print text only\n  if (noLogo) {\n    console.log(`Starting highcharts export server v${packageVersion}...`);\n    return;\n  }\n\n  // Print the logo\n  console.log(\n    readFileSync(__dirname + '/msg/startup.msg').toString().bold.yellow,\n    `v${packageVersion}`\n  );\n};\n\n/**\n * Prints the CLI usage. If required, it can list properties recursively\n */\nexport function printUsage() {\n  const pad = 48;\n  const readme = 'https://github.com/highcharts/node-export-server#readme';\n\n  // Display readme information\n  console.log(\n    'Usage of CLI arguments:'.bold,\n    '\\n------',\n    `\\nFor more detailed information visit readme at: ${readme.bold.yellow}.`\n  );\n\n  const cycleCategories = (categories) => {\n    for (const [name, option] of Object.entries(categories)) {\n      // If category has more levels, go further\n      if (!Object.prototype.hasOwnProperty.call(option, 'value')) {\n        cycleCategories(option);\n      } else {\n        let descName = `  --${option.cliName || name} ${\n          ('<' + option.type + '>').green\n        } `;\n        if (descName.length < pad) {\n          for (let i = descName.length; i < pad; i++) {\n            descName += '.';\n          }\n        }\n\n        // Display correctly aligned messages\n        console.log(\n          descName,\n          option.description,\n          `[Default: ${option.value.toString().bold}]`.blue\n        );\n      }\n    }\n  };\n\n  // Cycle through options of each categories and display the usage info\n  Object.keys(defaultConfig).forEach((category) => {\n    // Only puppeteer and highcharts categories cannot be configured through CLI\n    if (!['puppeteer', 'highcharts'].includes(category)) {\n      console.log(`\\n${category.toUpperCase()}`.red);\n      cycleCategories(defaultConfig[category]);\n    }\n  });\n  console.log('\\n');\n}\n\n/**\n * Rounds number to passed precision.\n *\n * @param {number} value - Number to round.\n * @param {number} precision - A precision of rounding.\n */\nexport const roundNumber = (value, precision = 1) => {\n  const multiplier = Math.pow(10, precision || 0);\n  return Math.round(+value * multiplier) / multiplier;\n};\n\n/**\n * Casts the item to boolean.\n *\n * @param {any} item - Item to be cast.\n */\nexport const toBoolean = (item) =>\n  ['false', 'undefined', 'null', 'NaN', '0', ''].includes(item)\n    ? false\n    : !!item;\n\n/**\n * If necessary, places a custom code inside a function.\n *\n * @param {any} customCode - The customCode.\n */\nexport const wrapAround = (customCode, allowFileResources) => {\n  if (customCode && typeof customCode === 'string') {\n    customCode = customCode.trim();\n\n    if (customCode.endsWith('.js')) {\n      return allowFileResources\n        ? wrapAround(readFileSync(customCode, 'utf8'))\n        : false;\n    } else if (\n      customCode.startsWith('function()') ||\n      customCode.startsWith('function ()') ||\n      customCode.startsWith('()=>') ||\n      customCode.startsWith('() =>')\n    ) {\n      return `(${customCode})()`;\n    }\n    return customCode.replace(/;$/, '');\n  }\n};\n\n/**\n * Utility to measure time.\n */\nexport const measureTime = () => {\n  const start = process.hrtime.bigint();\n  return () => Number(process.hrtime.bigint() - start) / 1000000;\n};\n\nexport default {\n  __dirname,\n  clearText,\n  expBackoff,\n  fixType,\n  handleResources,\n  isCorrectJSON,\n  isObject,\n  isPrivateRangeUrlFound,\n  optionsStringify,\n  printLogo,\n  printUsage,\n  roundNumber,\n  toBoolean,\n  wrapAround,\n  measureTime\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport rateLimit from 'express-rate-limit';\n\nimport { clearText } from '../utils.js';\nimport { log } from '../logger.js';\n\n/**\n * Enables rate limiting for a given app.\n *\n * @param {object} app - The express app.\n * @param {object} limitConfig - The options for the rate limiting.\n */\nexport default (app, limitConfig) => {\n  const msg =\n    'Too many requests, you have been rate limited. Please try again later.';\n\n  // Options for the rate limiter\n  const rateOptions = {\n    max: limitConfig.maxRequests || 30,\n    window: limitConfig.window || 1,\n    delay: limitConfig.delay || 0,\n    trustProxy: limitConfig.trustProxy || false,\n    skipKey: limitConfig.skipKey || false,\n    skipToken: limitConfig.skipToken || false\n  };\n\n  // Set if behind a proxy\n  if (rateOptions.trustProxy) {\n    app.enable('trust proxy');\n  }\n\n  // Create a limiter\n  const limiter = rateLimit({\n    windowMs: rateOptions.window * 60 * 1000,\n    // Limit each IP to 100 requests per windowMs\n    max: rateOptions.max,\n    // Disable delaying, full speed until the max limit is reached\n    delayMs: rateOptions.delay,\n    handler: (request, response) => {\n      response.format({\n        json: () => {\n          response.status(429).send({ message: msg });\n        },\n        default: () => {\n          response.status(429).send(msg);\n        }\n      });\n    },\n    skip: (request) => {\n      // Allow bypassing the limiter if a valid key/token has been sent\n      if (\n        rateOptions.skipKey !== false &&\n        rateOptions.skipToken !== false &&\n        request.query.key === rateOptions.skipKey &&\n        request.query.access_token === rateOptions.skipToken\n      ) {\n        log(4, '[rate-limiting] Skipping rate limiter.');\n        return true;\n      }\n      return false;\n    }\n  });\n\n  // Use a limiter as a middleware\n  app.use(limiter);\n\n  log(\n    3,\n    clearText(\n      `[rate-limiting] Enabled rate limiting: ${rateOptions.max} requests\n      per ${rateOptions.window} minute per IP, trusting proxy:\n      ${rateOptions.trustProxy}.`\n    )\n  );\n};\n","/**\n * This module exports two functions: fetch (for GET requests) and post (for POST requests).\n */\n\nimport http from 'http';\nimport https from 'https';\n\n/**\n * Determines the protocol of the given URL (either `http` or `https`).\n *\n * @function\n * @param {string} url - The URL whose protocol needs to be determined.\n * @returns {Object} Returns the `https` module if the URL starts with 'https',\n * otherwise returns the `http` module.\n * @private\n *\n * @example\n *\n * const protocol = getProtocol('https://example.com');\n * console.log(protocol); // Outputs the 'https' module\n */\nconst getProtocol = (url) => {\n  return url.startsWith('https') ? https : http;\n};\n\n/**\n * Sends a GET request to the specified URL with optional request options.\n *\n * @function\n * @async\n * @param {string} url - The URL to fetch.\n * @param {Object} [requestOptions={}] - Optional request options and headers.\n * @returns {Promise<Object>} Returns a promise that resolves with the response object.\n * The response object contains a `.text` property with the raw response data.\n * @throws {Error} Throws an error if the request fails or if no data is fetched from the URL.\n *\n * @example\n *\n * async function getData() {\n *   try {\n *     const response = await fetch('https://api.example.com/data');\n *     console.log(response.text);\n *   } catch (error) {\n *     console.error('Error fetching data:', error);\n *   }\n * }\n *\n * getData();\n */\nasync function fetch(url, requestOptions = {}) {\n  return new Promise((resolve, reject) => {\n    const protocol = getProtocol(url);\n\n    protocol\n      .get(url, requestOptions, (res) => {\n        let data = '';\n\n        // A chunk of data has been received.\n        res.on('data', (chunk) => {\n          data += chunk;\n        });\n\n        // The whole response has been received.\n        res.on('end', () => {\n          if (!data) {\n            reject('Nothing was fetched from the URL.');\n          }\n\n          res.text = data;\n          resolve(res);\n        });\n      })\n      .on('error', (error) => {\n        reject(error);\n      });\n  });\n}\n\n/**\n * Sends a POST request to the specified URL with the given body and request options.\n *\n * @function\n * @async\n * @param {string} url - The URL to which the request should be sent.\n * @param {Object} [body={}] - The data to be sent as the request body, in JSON format.\n * @param {Object} [requestOptions={}] - Optional request options and headers.\n * @returns {Promise<Object>} - Returns a promise that resolves with the parsed JSON response.\n * @throws {Error} Throws an error if the request fails or if the response cannot be parsed.\n *\n * @example\n *\n * async function sendData() {\n *   const dataToSend = {\n *     key1: 'value1',\n *     key2: 'value2',\n *   };\n *   try {\n *     const response = await post('https://api.example.com/data', dataToSend);\n *     console.log(response);\n *   } catch (error) {\n *     console.error('Error sending data:', error);\n *   }\n * }\n *\n * sendData();\n */\nasync function post(url, body = {}, requestOptions = {}) {\n  return new Promise((resolve, reject) => {\n    const protocol = getProtocol(url);\n    const data = JSON.stringify(body);\n\n    // Set default headers and merge with requestOptions\n    const options = Object.assign(\n      {\n        method: 'POST',\n        headers: {\n          'Content-Type': 'application/json',\n          'Content-Length': data.length\n        }\n      },\n      requestOptions\n    );\n\n    const req = protocol\n      .request(url, options, (res) => {\n        let responseData = '';\n\n        // A chunk of data has been received.\n        res.on('data', (chunk) => {\n          responseData += chunk;\n        });\n\n        // The whole response has been received.\n        res.on('end', () => {\n          try {\n            res.text = responseData;\n            resolve(res);\n          } catch (error) {\n            reject(error);\n          }\n        });\n      })\n      .on('error', (error) => {\n        reject(error);\n      });\n\n    // Write the request body and end the request.\n    req.write(data);\n    req.end();\n  });\n}\n\nexport default fetch;\nexport { fetch, post };\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\n// The cache manager manages the Highcharts library and its dependencies.\n// The cache itself is stored in .cache, and is checked by the config system\n// before starting the service\n\nimport { existsSync, mkdirSync, readFileSync, writeFileSync } from 'fs';\nimport { join } from 'path';\n\nimport dotenv from 'dotenv';\nimport HttpsProxyAgent from 'https-proxy-agent';\nimport { fetch } from './fetch.js';\n\nimport { log } from './logger.js';\nimport { __dirname } from '../lib/utils.js';\n\ndotenv.config();\n\nconst cachePath = join(__dirname, '.cache');\n\nconst cache = {\n  cdnURL: 'https://code.highcharts.com/',\n  activeManifest: {},\n  sources: '',\n  hcVersion: ''\n};\n\n// TODO: The config should be accesssible globally so we don't have to do this sort of thing..\nlet appliedConfig = false;\n\n/**\n * Extracts the Highcharts version from the cache\n */\nconst extractVersion = () =>\n  (cache.hcVersion = cache.sources\n    .substr(0, cache.sources.indexOf('*/'))\n    .replace('/*', '')\n    .replace('*/', '')\n    .replace(/\\n/g, '')\n    .trim());\n\n/**\n * Saves the Highcharts part of a config to a manifest file in the cache\n *\n * @param {object} config - Highcharts related configuration object.\n * @param {object} fetchedModules - An object that contains mapped names of\n * fetched Highcharts modules to use.\n */\nconst saveConfigToManifest = async (config, fetchedModules) => {\n  const newManifest = {\n    version: config.version,\n    modules: fetchedModules || {}\n  };\n\n  // Update cache object with the current modules\n  cache.activeManifest = newManifest;\n\n  log(4, '[cache] writing new manifest');\n\n  try {\n    writeFileSync(\n      join(cachePath, 'manifest.json'),\n      JSON.stringify(newManifest),\n      'utf8'\n    );\n  } catch (error) {\n    log(1, `[cache] Error writing cache manifest: ${error}.`);\n  }\n};\n\n/**\n * Fetches a single script.\n *\n * @param {string} script - A path to script to get.\n * @param {object} proxyAgent - The proxy agent to use for a request.\n */\nconst fetchScript = async (script, proxyAgent) => {\n  try {\n    // Get rid of the .js from the custom strings\n    if (script.endsWith('.js')) {\n      script = script.substring(0, script.length - 3);\n    }\n\n    log(4, `[cache] Fetching script - ${script}.js`);\n\n    // If exists, add proxy agent to request options\n    const requestOptions = proxyAgent\n      ? {\n          agent: proxyAgent,\n          timeout: +process.env['PROXY_SERVER_TIMEOUT'] || 5000\n        }\n      : {};\n\n    // Fetch the script\n    const response = await fetch(`${script}.js`, requestOptions);\n\n    // If OK, return its text representation\n    if (response.statusCode === 200) {\n      return response.text;\n    }\n\n    throw `${response.statusCode}`;\n  } catch (error) {\n    log(1, `[cache] Error fetching script ${script}.js: ${error}.`);\n    throw error;\n  }\n};\n\n/**\n * Updates the Highcharts cache.\n *\n * @param {object} config - Highcharts related configuration object.\n * @param {string} sourcePath - A path to the file where save updated sources.\n * @return {object} An object that contains mapped names of fetched Highcharts\n * modules to use.\n */\nconst updateCache = async (config, sourcePath) => {\n  const { coreScripts, modules, indicators, scripts: customScripts } = config;\n  const hcVersion =\n    config.version === 'latest' || !config.version ? '' : `${config.version}/`;\n\n  log(3, '[cache] Updating cache to Highcharts ', hcVersion);\n\n  // Gather all scripts to fetch\n  const allScripts = [\n    ...coreScripts.map((c) => `${hcVersion}${c}`),\n    ...modules.map((m) =>\n      m === 'map' ? `maps/${hcVersion}modules/${m}` : `${hcVersion}modules/${m}`\n    ),\n    ...indicators.map((i) => `stock/${hcVersion}indicators/${i}`)\n  ];\n\n  // Configure proxy if exists\n  let proxyAgent;\n  const proxyHost = process.env['PROXY_SERVER_HOST'];\n  const proxyPort = process.env['PROXY_SERVER_PORT'];\n\n  if (proxyHost && proxyPort) {\n    proxyAgent = new HttpsProxyAgent({\n      host: proxyHost,\n      port: +proxyPort\n    });\n  }\n\n  const fetchedModules = {};\n  try {\n    cache.sources = // TODO: convert to for loop\n      (\n        await Promise.all([\n          ...allScripts.map(async (script) => {\n            const text = await fetchScript(\n              `${config.cdnURL || cache.cdnURL}${script}`,\n              proxyAgent\n            );\n\n            // If fetched correctly, set it\n            if (typeof text === 'string') {\n              fetchedModules[\n                script.replace(\n                  /(.*)\\/|(.*)modules\\/|stock\\/(.*)indicators\\/|maps\\/(.*)modules\\//gi,\n                  ''\n                )\n              ] = 1;\n            }\n\n            return text;\n          }),\n          ...customScripts.map((script) => fetchScript(script, proxyAgent))\n        ])\n      ).join(';\\n');\n    extractVersion();\n\n    // Save the fetched modules into caches' source JSON\n    writeFileSync(sourcePath, cache.sources);\n    return fetchedModules;\n  } catch (error) {\n    log(1, '[cache] Unable to update local Highcharts cache.');\n  }\n};\n\nexport const updateVersion = async (newVersion) =>\n  appliedConfig\n    ? await checkCache(\n        Object.assign(appliedConfig, {\n          version: newVersion\n        })\n      )\n    : false;\n\n/**\n * Fetches any missing Highcharts and dependencies\n *\n * @param {object} config - Highcharts related configuration object.\n */\nexport const checkCache = async (config) => {\n  let fetchedModules;\n  // Prepare paths to manifest and sources from the .cache folder\n  const manifestPath = join(cachePath, 'manifest.json');\n  const sourcePath = join(cachePath, 'sources.js');\n\n  // TODO: deal with trying to switch to the running version\n  // const activeVersion = appliedConfig ? appliedConfig.version : false;\n\n  appliedConfig = config;\n\n  // Create the .cache destination if it doesn't exist already\n  !existsSync(cachePath) && mkdirSync(cachePath);\n\n  // Fetch all the scripts either if manifest.json does not exist\n  // or if the forceFetch option is enabled\n  if (!existsSync(manifestPath) || config.forceFetch) {\n    log(3, '[cache] Fetching and caching Highcharts dependencies.');\n    fetchedModules = await updateCache(config, sourcePath);\n  } else {\n    let requestUpdate = false;\n\n    // Read the manifest JSON\n    const manifest = JSON.parse(readFileSync(manifestPath));\n\n    // Check if the modules is an array, if so, we rewrite it to a map to make\n    // it easier to resolve modules.\n    if (manifest.modules && Array.isArray(manifest.modules)) {\n      const moduleMap = {};\n      manifest.modules.forEach((m) => (moduleMap[m] = 1));\n      manifest.modules = moduleMap;\n    }\n\n    const { modules, coreScripts, indicators } = config;\n    const numberOfModules =\n      modules.length + coreScripts.length + indicators.length;\n\n    // Compare the loaded config with the contents in .cache.\n    // If there are changes, fetch requested modules and products,\n    // and bake them into a giant blob. Save the blob.\n    if (manifest.version !== config.version) {\n      log(3, '[cache] Highcharts version mismatch in cache, need to re-fetch.');\n      requestUpdate = true;\n    } else if (Object.keys(manifest.modules || {}).length !== numberOfModules) {\n      log(\n        3,\n        '[cache] Cache and requested modules does not match, need to re-fetch.'\n      );\n      requestUpdate = true;\n    } else {\n      // Check each module, if anything is missing refetch everything\n      requestUpdate = (config.modules || []).some((moduleName) => {\n        if (!manifest.modules[moduleName]) {\n          log(\n            3,\n            `[cache] The ${moduleName} missing in cache, need to re-fetch.`\n          );\n          return true;\n        }\n      });\n    }\n\n    if (requestUpdate) {\n      fetchedModules = await updateCache(config, sourcePath);\n    } else {\n      log(3, '[cache] Dependency cache is up to date, proceeding.');\n\n      // Load the sources\n      cache.sources = readFileSync(sourcePath, 'utf8');\n\n      // Get current modules map\n      fetchedModules = manifest.modules;\n      extractVersion();\n    }\n  }\n\n  // Finally, save the new manifest, which is basically our current config\n  // in a slightly different format\n  await saveConfigToManifest(config, fetchedModules);\n};\n\nexport default {\n  checkCache,\n  updateVersion,\n  getCache: () => cache,\n  highcharts: () => cache.sources,\n  version: () => cache.hcVersion\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport puppeteer from 'puppeteer';\nimport fs from 'fs';\nimport * as url from 'url';\nimport { log } from './logger.js';\nimport path from 'node:path';\n\n// Workaround for https://bugs.chromium.org/p/chromium/issues/detail?id=1463328\n// Not ideal - leaves trash in the FS\nimport { randomBytes } from 'node:crypto';\nconst RANDOM_PID = randomBytes(64).toString('base64url');\nconst PUPPETEER_DIR = path.join('tmp', `puppeteer-${RANDOM_PID}`);\nconst DATA_DIR = path.join(PUPPETEER_DIR, 'profile');\n\n// The minimal args to speed up the browser\nconst minimalArgs = [\n  `--user-data-dir=${DATA_DIR}`,\n  '--autoplay-policy=user-gesture-required',\n  '--disable-background-networking',\n  '--disable-background-timer-throttling',\n  '--disable-backgrounding-occluded-windows',\n  '--disable-breakpad',\n  '--disable-client-side-phishing-detection',\n  '--disable-component-update',\n  '--disable-default-apps',\n  '--disable-dev-shm-usage',\n  '--disable-domain-reliability',\n  '--disable-extensions',\n  '--disable-features=AudioServiceOutOfProcess',\n  '--disable-hang-monitor',\n  '--disable-ipc-flooding-protection',\n  '--disable-notifications',\n  '--disable-offer-store-unmasked-wallet-cards',\n  '--disable-popup-blocking',\n  '--disable-print-preview',\n  '--disable-prompt-on-repost',\n  '--disable-renderer-backgrounding',\n  '--disable-session-crashed-bubble',\n  '--disable-setuid-sandbox',\n  '--disable-speech-api',\n  '--disable-sync',\n  '--hide-crash-restore-bubble',\n  '--hide-scrollbars',\n  '--ignore-gpu-blacklist',\n  '--metrics-recording-only',\n  '--mute-audio',\n  '--no-default-browser-check',\n  '--no-first-run',\n  '--no-pings',\n  '--no-sandbox',\n  '--no-zygote',\n  '--password-store=basic',\n  '--use-mock-keychain'\n];\n\nconst __dirname = url.fileURLToPath(new URL('.', import.meta.url));\n\nconst template = fs.readFileSync(\n  __dirname + '/../templates/template.html',\n  'utf8'\n);\n\nlet browser;\n\nexport const newPage = async () => {\n  if (!browser) return false;\n\n  const p = await browser.newPage();\n\n  await p.setContent(template);\n  await p.addScriptTag({ path: __dirname + '/../.cache/sources.js' });\n  // eslint-disable-next-line no-undef\n  await p.evaluate(() => window.setupHighcharts());\n\n  p.on('pageerror', async (err) => {\n    // TODO: Consider adding a switch here that turns on log(0) logging\n    // on page errors.\n    log(1, '[page error]', err);\n    await p.$eval(\n      '#container',\n      (element, errorMessage) => {\n        // eslint-disable-next-line no-undef\n        if (window._displayErrors) {\n          element.innerHTML = errorMessage;\n        }\n      },\n      `<h1>Chart input data error</h1>${err.toString()}`\n    );\n  });\n\n  return p;\n};\n\nexport const create = async (puppeteerArgs) => {\n  const allArgs = [...minimalArgs, ...(puppeteerArgs || [])];\n\n  // Create a browser\n  if (!browser) {\n    let tryCount = 0;\n\n    const open = async () => {\n      try {\n        log(\n          3,\n          '[browser] attempting to get a browser instance (try',\n          tryCount + ')'\n        );\n\n        browser = await puppeteer.launch({\n          headless: 'new',\n          args: allArgs,\n          userDataDir: './tmp/'\n        });\n      } catch (e) {\n        log(0, '[browser]', e);\n        if (++tryCount < 25) {\n          log(3, '[browser] failed:', e);\n          await new Promise((response) => setTimeout(response, 4000));\n          await open();\n        } else {\n          log(0, 'Max retries reached');\n        }\n      }\n    };\n\n    try {\n      await open();\n    } catch (e) {\n      log(0, '[browser] Unable to open browser');\n      return false;\n    }\n\n    if (!browser) {\n      log(0, '[browser] Unable to open browser');\n      return false;\n    }\n  }\n\n  // Return a browser promise\n  return browser;\n};\n\nexport const get = async () => {\n  if (!browser) {\n    throw 'No valid browser has been created';\n  }\n\n  return browser;\n};\n\nexport const close = async () => {\n  // Close the browser when connnected\n  if (browser.connected) {\n    await browser.close();\n  }\n};\n\nexport default {\n  get,\n  close,\n  newPage\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\n// TODO: remove this temp benchmark stuff. I had this idea of doing a general benchmarking\n// system, but it adds so much bloat in the code that it shouldn't be there.\n\nimport benchmark from './benchmark.js';\nimport cache from './cache.js';\nimport { log } from './logger.js';\nimport svgTemplate from './../templates/svg_export/svg_export.js';\n\nimport { readFileSync } from 'fs';\nimport path from 'path';\nimport * as url from 'url';\n\nconst __basedir = url.fileURLToPath(new URL('.', import.meta.url));\n\n// const jsonTemplate = require('./../templates/json_export/json_export.js');\n\n/**\n * Gets the clip region for the chart DOM node.\n *\n * @param {object} page - A page of a browser instance.\n * @return {object} - A clipped region.\n */\nconst getClipRegion = (page) =>\n  page.$eval('#chart-container', (element) => {\n    const { x, y, width, height } = element.getBoundingClientRect();\n    return {\n      x,\n      y,\n      width,\n      height: Math.trunc(height > 1 ? height : 500)\n    };\n  });\n\n/**\n * Rasterizes the page to an image (PNG or JPEG)\n *\n * @param {object} page - A page of a browser instance.\n * @param {string} type - The type of a result image.\n * @param {string} encoding - The type of encoding used.\n * @param {string} clip - The clip region.\n * @returns {string} - A string representation of a screenshot.\n */\nconst createImage = async (page, type, encoding, clip) =>\n  await Promise.race([\n    page.screenshot({\n      type,\n      encoding,\n      clip,\n\n      // #447 - always render on a transparent page\n      // this will not affect users who do not explicitly set\n      // chart.backgroundColor to a color with opacity lower than 1\n      omitBackground: true\n    }),\n    new Promise((resolve, reject) =>\n      setTimeout(() => reject(new Error('Rasterization timeout')), 1500)\n    )\n  ]);\n\n/**\n * Turns page into a PDF.\n *\n * @param {object} page - A page of a browser instance.\n * @param {number} height - The height of a chart.\n * @param {number} width - The width of a chart.\n * @param {string} encoding - The type of encoding used.\n * @return {object} - A buffer with PDF representation.\n */\nconst createPDF = async (page, height, width, encoding) =>\n  await page.pdf({\n    // This will remove an extra empty page in PDF exports\n    height: height + 1,\n    width,\n    encoding\n  });\n\n/**\n * Exports as a SVG.\n *\n * @param {object} page - A page of a browser instance.\n * @return {object} - The outerHTML element with the SVG representation.\n */\nconst createSVG = async (page) =>\n  await page.$eval(\n    '#container svg:first-of-type',\n    (element) => element.outerHTML\n  );\n\n/** Load config into a page and render a chart */\nconst setAsConfig = async (page, chart, options) =>\n  await page.evaluate(\n    // eslint-disable-next-line no-undef\n    (chart, options) => window.triggerExport(chart, options),\n    chart,\n    options\n  );\n\n/** Load SVG into a page */\n// const setAsSVG = async (page, svgStr) => true;\n\n/**\n * Does an export for a given browser.\n *\n * @param {object} browser - A browser instance.\n * @param {object} chart - Chart's options.\n * @param {object} options - All options object.\n * @return {object} - The data returned from one of the methods for exporting\n * a specific type of an image.\n */\nexport default async (page, chart, options) => {\n  /**\n   * Keeps track of all resources added on the page with addXXXTag. etc\n   * It's VITAL that all added resources ends up here so we can clear things\n   * out when doing a new export in the same page!\n   */\n  const injectedResources = [];\n\n  /** Clear out all state set on the page with addScriptTag/addStyleTag. */\n  const clearInjected = async (page) => {\n    for (const res of injectedResources) {\n      await res.dispose();\n    }\n\n    // Reset all CSS and script tags\n    await page.evaluate(() => {\n      // eslint-disable-next-line no-undef\n      const [, ...scriptsToRemove] = document.getElementsByTagName('script');\n      // eslint-disable-next-line no-undef\n      const [, ...stylesToRemove] = document.getElementsByTagName('style');\n      // eslint-disable-next-line no-undef\n      const [...linksToRemove] = document.getElementsByTagName('link');\n\n      // Remove tags\n      for (const element of [\n        ...scriptsToRemove,\n        ...stylesToRemove,\n        ...linksToRemove\n      ]) {\n        element.remove();\n      }\n    });\n  };\n\n  try {\n    const exportBench = benchmark('Puppeteer');\n\n    log(4, '[export] Determining export path.');\n\n    const exportOptions = options.export;\n\n    // Force a rAF\n    // See https://github.com/puppeteer/puppeteer/issues/7507\n    // eslint-disable-next-line no-undef\n    await page.evaluate(() => requestAnimationFrame(() => {}));\n\n    // Decide whether display error or debbuger wrapper around it\n    const displayErrors =\n      exportOptions?.options?.chart?.displayErrors &&\n      cache.getCache().activeManifest.modules.debugger;\n\n    // eslint-disable-next-line no-undef\n    await page.evaluate((d) => (window._displayErrors = d), displayErrors);\n\n    const svgBench = benchmark('SVG handling');\n\n    let isSVG;\n\n    if (\n      chart.indexOf &&\n      (chart.indexOf('<svg') >= 0 || chart.indexOf('<?xml') >= 0)\n    ) {\n      // SVG INPUT HANDLING\n\n      log(4, '[export] Treating as SVG.');\n\n      // If input is also svg, just return it\n      if (exportOptions.type === 'svg') {\n        return chart;\n      }\n\n      isSVG = true;\n      const setPageBench = benchmark('Setting content');\n      await page.setContent(svgTemplate(chart));\n      setPageBench();\n    } else {\n      // JSON Config handling\n\n      log(4, '[export] Treating as config.');\n\n      // Need to perform straight inject\n      if (exportOptions.strInj) {\n        // Injection based configuration export\n        const setPageBench = benchmark('Setting page content (inject)');\n\n        await setAsConfig(\n          page,\n          {\n            chart: {\n              height: exportOptions.height,\n              width: exportOptions.width\n            }\n          },\n          options\n        );\n\n        setPageBench();\n      } else {\n        // Basic configuration export\n\n        chart.chart.height = exportOptions.height;\n        chart.chart.width = exportOptions.width;\n\n        const setContentBench = benchmark('Setting page content (config)');\n        await setAsConfig(page, chart, options);\n        setContentBench();\n      }\n    }\n\n    svgBench();\n    const resBench = benchmark('Applying resources');\n\n    // Use resources\n    const resources = options.customCode.resources;\n    if (resources) {\n      // Load custom JS code\n      if (resources.js) {\n        injectedResources.push(\n          await page.addScriptTag({\n            content: resources.js\n          })\n        );\n      }\n\n      // Load scripts from all custom files\n      if (resources.files) {\n        for (const file of resources.files) {\n          try {\n            const isLocal = !file.startsWith('http') ? true : false;\n\n            // Add each custom script from resources' files\n            injectedResources.push(\n              await page.addScriptTag(\n                isLocal\n                  ? {\n                      content: readFileSync(file, 'utf8')\n                    }\n                  : {\n                      url: file\n                    }\n              )\n            );\n          } catch (notice) {\n            log(4, '[export] JS file not found.');\n          }\n        }\n      }\n\n      const cssBench = benchmark('Loading css');\n\n      // Load CSS\n      if (resources.css) {\n        let cssImports = resources.css.match(/@import\\s*([^;]*);/g);\n        if (cssImports) {\n          // Handle css section\n          for (let cssImportPath of cssImports) {\n            if (cssImportPath) {\n              cssImportPath = cssImportPath\n                .replace('url(', '')\n                .replace('@import', '')\n                .replace(/\"/g, '')\n                .replace(/'/g, '')\n                .replace(/;/, '')\n                .replace(/\\)/g, '')\n                .trim();\n\n              // Add each custom css from resources\n              if (cssImportPath.startsWith('http')) {\n                injectedResources.push(\n                  await page.addStyleTag({\n                    url: cssImportPath\n                  })\n                );\n              } else if (options.customCode.allowFileResources) {\n                injectedResources.push(\n                  await page.addStyleTag({\n                    path: path.join(__basedir, cssImportPath)\n                  })\n                );\n              }\n            }\n          }\n        }\n\n        // The rest of the CSS section will be content by now\n        injectedResources.push(\n          await page.addStyleTag({\n            content: resources.css.replace(/@import\\s*([^;]*);/g, '') || ' '\n          })\n        );\n      }\n\n      cssBench();\n    }\n\n    resBench();\n\n    // Get the real chart size\n    const size = isSVG\n      ? await page.$eval(\n          '#chart-container svg:first-of-type',\n          async (element, scale) => {\n            return {\n              chartHeight: element.height.baseVal.value * scale,\n              chartWidth: element.width.baseVal.value * scale\n            };\n          },\n          parseFloat(exportOptions.scale)\n        )\n      : await page.evaluate(async () => {\n          // eslint-disable-next-line no-undef\n          const { chartHeight, chartWidth } = window.Highcharts.charts[0];\n          return {\n            chartHeight,\n            chartWidth\n          };\n        });\n\n    const vpBench = benchmark('Setting viewport');\n\n    // Set final height and width for viewport\n    const viewportHeight = Math.ceil(size?.chartHeight || exportOptions.height);\n    const viewportWidth = Math.ceil(size?.chartWidth || exportOptions.width);\n\n    // Set the viewport for the first time\n    // NOTE: the call to setViewport is expensive - can we get away with only\n    // calling it once, e.g. moving this one into the isSVG condition below?\n    await page.setViewport({\n      height: viewportHeight,\n      width: viewportWidth,\n      deviceScaleFactor: isSVG ? 1 : parseFloat(exportOptions.scale)\n    });\n\n    // Prepare a zoom callback for the next evaluate call\n    const zoomCallback = isSVG\n      ? // In case of SVG the zoom must be set directly for body\n        (scale) => {\n          // Set the zoom as scale\n          // eslint-disable-next-line no-undef\n          document.body.style.zoom = scale;\n\n          // Set the margin to 0px\n          // eslint-disable-next-line no-undef\n          document.body.style.margin = '0px';\n        }\n      : // No need for such scale manipulation in case of other types of exports\n        () => {\n          // Reset the zoom for other exports than to SVGs\n          // eslint-disable-next-line no-undef\n          document.body.style.zoom = 1;\n        };\n\n    // Set the zoom accordingly\n    await page.evaluate(zoomCallback, parseFloat(exportOptions.scale));\n\n    // Get the clip region for the page\n    const { height, width, x, y } = await getClipRegion(page);\n\n    if (!isSVG) {\n      // Set the final viewport now that we have the real height\n      await page.setViewport({\n        width: Math.round(width),\n        height: Math.round(height),\n        deviceScaleFactor: parseFloat(exportOptions.scale)\n      });\n    }\n\n    vpBench();\n\n    let data;\n\n    const expBenchmark = benchmark('Rasterizing chart');\n\n    // RASTERIZATION\n    if (exportOptions.type === 'svg') {\n      // SVG\n      data = await createSVG(page);\n    } else if (exportOptions.type === 'png' || exportOptions.type === 'jpeg') {\n      // PNG or JPEG\n      data = await createImage(page, exportOptions.type, 'base64', {\n        width: viewportWidth,\n        height: viewportHeight,\n        x,\n        y\n      });\n    } else if (exportOptions.type === 'pdf') {\n      // PDF\n      data = await createPDF(page, viewportHeight, viewportWidth, 'base64');\n    } else {\n      throw `Unsupported output format ${exportOptions.type}`;\n    }\n\n    // Destroy old charts after the export is done\n    await page.evaluate(() => {\n      // eslint-disable-next-line no-undef\n      const oldCharts = Highcharts.charts;\n\n      // Check in any already existing charts\n      if (oldCharts.length) {\n        // Destroy old charts\n        for (const oldChart of oldCharts) {\n          oldChart && oldChart.destroy();\n          // eslint-disable-next-line no-undef\n          Highcharts.charts.shift();\n        }\n      }\n    });\n\n    expBenchmark();\n    exportBench();\n\n    await clearInjected(page);\n\n    return data;\n  } catch (error) {\n    await clearInjected(page);\n    log(1, `[export] Error encountered during export: ${error}`);\n\n    return error;\n  }\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2022, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { log } from './logger.js';\nconst timers = {};\n\n// TODO: Read from config\nlet enabled = false;\n\nexport default (id) => {\n  if (!enabled) {\n    return () => {};\n  }\n\n  timers[id] = new Date();\n  return () => {\n    log(\n      3,\n      `[benchmark] - ${id}: ${new Date().getTime() - timers[id].getTime()}ms`\n    );\n  };\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport cssTemplate from './css.js';\n\nexport default (chart) => `\n<!DOCTYPE html>\n<html lang='en-US'>\n  <head>\n    <meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\">\n    <title>Highcarts Export</title>\n  </head>\n  <style>\n    ${cssTemplate()}\n  </style>\n  <body>\n    <div id=\"chart-container\">\n      ${chart}\n    </div>\n  </body>\n</html>\n\n`;\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { v4 as uuid } from 'uuid';\nimport { Pool } from 'tarn';\nimport {\n  close,\n  newPage as browserNewPage,\n  create as createBrowser\n} from './browser.js';\nimport { log } from './logger.js';\n\nimport puppeteerExport from './export.js';\n\nlet performedExports = 0;\nlet exportAttempts = 0;\nlet timeSpent = 0;\nlet droppedExports = 0;\nlet spentAverage = 0;\nlet poolConfig = {};\n\n// The pool instance\nlet pool = false;\n\n// Custom puppeteer arguments\nlet puppeteerArgs;\n\nconst factory = {\n  /**\n   * Creates a new worker.\n   *\n   * @return {object} - An object with the id of a resource, the work count and\n   * a reference to the browser page.\n   */\n  create: async () => {\n    const id = uuid();\n    let page = false;\n\n    const s = new Date().getTime();\n\n    try {\n      page = await browserNewPage();\n\n      if (!page || page.isClosed()) {\n        throw 'invalid page';\n      }\n\n      log(\n        3,\n        `[pool] Successfully created a worker ${id} - took ${\n          new Date().getTime() - s\n        } ms.`\n      );\n    } catch (error) {\n      log(\n        1,\n        `[pool] Error creating a new page in pool entry creation! ${error}`\n      );\n\n      throw 'Error creating page';\n    }\n\n    return {\n      id,\n      page,\n      // Try to distribute the initial work count\n      workCount: Math.round(Math.random() * (poolConfig.workLimit / 2))\n    };\n  },\n\n  /**\n   * Validates a worker.\n   *\n   * @param {object} workerHandle - A browser's instance.\n   *\n   * @return {boolean} - Bool that indicates if a resource is valid or not.\n   */\n  validate: (workerHandle) => {\n    if (\n      poolConfig.workLimit &&\n      ++workerHandle.workCount > poolConfig.workLimit\n    ) {\n      log(\n        3,\n        `[pool] Worker failed validation:`,\n        `exceeded work limit (limit is ${poolConfig.workLimit})`\n      );\n      return false;\n    }\n    return true;\n  },\n\n  /**\n   * Destroys a worker.\n   *\n   * @param {object} workerHandle - A browser's instance.\n   */\n  destroy: (workerHandle) => {\n    log(3, `[pool] Destroying pool entry ${workerHandle.id}.`);\n\n    if (workerHandle.page) {\n      // We don't really need to wait around for this.\n      workerHandle.page.close();\n    }\n  },\n\n  // Logger function\n  log: (message, logLevel) => console.log(`${logLevel}: ${message}`)\n};\n\n/**\n * Inits the pool of resources.\n *\n * @param {object} config - Pool configuration along with custom puppeteer\n * arguments for the puppeteer.launch function.\n */\nexport const init = async (config) => {\n  // The newest puppeteer arguments for the browser creation\n  puppeteerArgs = config.puppeteerArgs;\n\n  // Wait until we've sucessfully created a browser instance.\n  try {\n    await createBrowser(puppeteerArgs);\n  } catch (e) {\n    log(0, '[pool|browser]', e);\n  }\n\n  // For the module scope usage\n  poolConfig = config && config.pool ? { ...config.pool } : {};\n\n  log(\n    3,\n    '[pool] Initializing pool:',\n    `min ${poolConfig.initialWorkers}, max ${poolConfig.maxWorkers}.`\n  );\n\n  if (pool) {\n    return log(\n      4,\n      '[pool] Already initialized, please kill it before creating a new one.'\n    );\n  }\n\n  // Attach process' exit listeners\n  if (poolConfig.listenToProcessExits) {\n    attachProcessExitListeners();\n  }\n\n  try {\n    // Create a pool along with a minimal number of resources\n    pool = new Pool({\n      // Get the create/validate/destroy/log functions\n      ...factory,\n      min: poolConfig.initialWorkers,\n      max: poolConfig.maxWorkers,\n      createRetryIntervalMillis: 200,\n      createTimeoutMillis: poolConfig.acquireTimeout,\n      acquireTimeoutMillis: poolConfig.acquireTimeout,\n      destroyTimeoutMillis: poolConfig.acquireTimeout,\n      idleTimeoutMillis: poolConfig.timeoutThreshold,\n      reapIntervalMillis: 1000, // poolConfig.reaper ? 120000 : 0, for now\n      propagateCreateError: false\n    });\n\n    // Set events\n    pool.on('createFail', (eventId, err) => {\n      log(\n        1,\n        `[pool] Error when creating worker of an event id ${eventId}:`,\n        err\n      );\n    });\n\n    pool.on('acquireFail', (eventId, err) => {\n      log(\n        1,\n        `[pool] Error when acquiring worker of an event id ${eventId}:`,\n        err\n      );\n    });\n\n    pool.on('destroyFail', (eventId, resource, err) => {\n      log(\n        1,\n        `[pool] Error when destroying worker of an id ${resource.id}, event id ${eventId}:`,\n        err\n      );\n    });\n\n    pool.on('release', (resource) => {\n      log(4, `[pool] Releasing a worker of an id ${resource.id}`);\n    });\n\n    pool.on('destroySuccess', (eventId, resource) => {\n      log(4, `[pool] Destroyed a worker of an id ${resource.id}`);\n    });\n\n    const initialResources = [];\n    // Create an initial number of resources\n    for (let i = 0; i < poolConfig.initialWorkers; i++) {\n      initialResources.push(await pool.acquire().promise);\n    }\n\n    // Release the initial number of resources back to the pool\n    initialResources.forEach((resource) => {\n      pool.release(resource);\n    });\n\n    log(\n      3,\n      `[pool] The pool is ready with ${poolConfig.initialWorkers} initial resources waiting.`\n    );\n  } catch (error) {\n    log(1, `[pool] Couldn't create the worker pool ${error}`);\n    throw error;\n  }\n};\n\n/**\n * Attaches process' exit listeners.\n */\nexport function attachProcessExitListeners() {\n  log(4, '[pool] Attaching exit listeners to the process.');\n\n  // Kill all pool resources on exit\n  process.on('exit', async () => {\n    await killPool();\n  });\n\n  // Handler for the SIGINT\n  process.on('SIGINT', (name, code) => {\n    log(4, `The ${name} event with code: ${code}.`);\n    process.exit(1);\n  });\n\n  // Handler for the SIGTERM\n  process.on('SIGTERM', (name, code) => {\n    log(4, `The ${name} event with code: ${code}.`);\n    process.exit(1);\n  });\n\n  // Handler for the uncaughtException\n  process.on('uncaughtException', async (error, name) => {\n    log(4, `The ${name} error, message: ${error.message}.`);\n  });\n}\n\n/**\n * Kills the pool and flush the browser instance.\n */\nexport async function killPool() {\n  log(3, '[pool] Killing all workers.');\n\n  // Return true when the pool is already destroyed\n  if (pool.destroyed) {\n    // Close the browser instance if still connected\n    await close();\n    return true;\n  }\n\n  // If still alive, destroy the pool of pages before closing a browser\n  await pool.destroy();\n\n  // Close the browser instance\n  await close();\n  return true;\n}\n\n/**\n * Posts work to the pool.\n *\n * @param {object} chart - Chart's options.\n * @param {object} options - All options object.\n */\nexport const postWork = async (chart, options) => {\n  let workerHandle;\n\n  // Handle fail conditions\n  const fail = (msg) => {\n    ++droppedExports;\n\n    if (workerHandle) {\n      pool.release(workerHandle);\n    }\n\n    throw 'In pool.postWork: ' + msg;\n  };\n\n  log(4, '[pool] Work received, starting to process.');\n\n  if (poolConfig.benchmarking) {\n    getPoolInfo();\n  }\n\n  ++exportAttempts;\n\n  if (!pool) {\n    log(1, '[pool] Work received, but pool has not been started.');\n    return fail('Pool is not inited but work was posted to it!');\n  }\n\n  // Acquire the worker along with the id of resource and work count\n  try {\n    log(4, '[pool] Acquiring worker');\n    workerHandle = await pool.acquire().promise;\n  } catch (error) {\n    return fail(`[pool] Error when acquiring available entry: ${error}`);\n  }\n\n  log(4, '[pool] Acquired worker handle');\n\n  if (!workerHandle.page) {\n    return fail('Resolved worker page is invalid: pool setup is wonky');\n  }\n\n  try {\n    // Save the start time\n    let workStart = new Date().getTime();\n\n    log(4, `[pool] Starting work on pool entry ${workerHandle.id}.`);\n\n    // Perform an export on a puppeteer level\n    const result = await puppeteerExport(workerHandle.page, chart, options);\n\n    // Check if it's an error\n    if (result instanceof Error) {\n      // TODO: If the export failed because puppeteer timed out, we need to force kill the worker so we get a new page. That needs to be handled better than this hack.\n      if (result.message === 'Rasterization timeout') {\n        workerHandle.page.close();\n        workerHandle.page = await browserNewPage();\n      }\n\n      return fail(result);\n    }\n\n    // Release the resource back to the pool\n    pool.release(workerHandle);\n\n    // Used for statistics in averageTime and processedWorkCount, which\n    // in turn is used by the /health route.\n    const workEnd = new Date().getTime();\n    const exportTime = workEnd - workStart;\n    timeSpent += exportTime;\n    spentAverage = timeSpent / ++performedExports;\n\n    log(4, `[pool] Work completed in ${exportTime} ms.`);\n\n    // Otherwise return the result\n    return {\n      data: result,\n      options\n    };\n  } catch (error) {\n    fail(`Error trying to perform puppeteer export: ${error}.`);\n  }\n};\n\n/**\n * Gets the pool.\n */\nexport function getPool() {\n  return pool;\n}\n\nexport const getPoolInfoJSON = () => ({\n  min: pool.min,\n  max: pool.max,\n  size: pool.size,\n  available: pool.available,\n  borrowed: pool.borrowed,\n  pending: pool.pending,\n  spareResourceCapacity: pool.spareResourceCapacity\n});\n\n/**\n * Gets the pool's information.\n */\nexport function getPoolInfo() {\n  const {\n    min,\n    max,\n    size,\n    available,\n    borrowed,\n    pending,\n    spareResourceCapacity\n  } = pool;\n\n  log(4, `[pool] The minimum number of resources allowed by pool: ${min}.`);\n  log(4, `[pool] The maximum number of resources allowed by pool: ${max}.`);\n  log(\n    4,\n    `[pool] The number of all resources in pool (free or in use): ${size}.`\n  );\n  log(\n    4,\n    `[pool] The number of resources that are currently available: ${available}.`\n  );\n  log(\n    4,\n    `[pool] The number of resources that are currently acquired: ${borrowed}.`\n  );\n  log(\n    4,\n    `[pool] The number of callers waiting to acquire a resource: ${pending}.`\n  );\n  log(\n    4,\n    `[pool] The number of how many more resources can the pool manage/create: ${spareResourceCapacity}.`\n  );\n}\n\nexport default {\n  init,\n  killPool,\n  postWork,\n  getPool,\n  getPoolInfo,\n  getPoolInfoJSON,\n  workAttempts: () => exportAttempts,\n  droppedWork: () => droppedExports,\n  averageTime: () => spentAverage,\n  processedWorkCount: () => performedExports\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport cache from '../../cache.js';\nimport pool from '../../pool.js';\n\nconst packageVersion = process.env.npm_package_version;\nconst serverStartTime = new Date();\n\n/**\n * Adds the /health route which outputs basic stats for the server\n */\nexport default (app) =>\n  !app\n    ? false\n    : app.get('/health', (request, response) => {\n        response.send({\n          status: 'OK',\n          bootTime: serverStartTime,\n          uptime:\n            Math.floor(\n              (new Date().getTime() - serverStartTime.getTime()) / 1000 / 60\n            ) + ' minutes',\n          version: packageVersion,\n          highchartsVersion: cache.version(),\n          averageProcessingTime: pool.averageTime(),\n          performedExports: pool.processedWorkCount(),\n          failedExports: pool.droppedWork(),\n          exportAttempts: pool.workAttempts(),\n          sucessRatio: (pool.processedWorkCount() / pool.workAttempts()) * 100,\n          // eslint-disable-next-line import/no-named-as-default-member\n          pool: pool.getPoolInfoJSON()\n        });\n      });\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { existsSync, readFileSync, promises as fsPromises } from 'fs';\n\nimport prompts from 'prompts';\n\nimport { log } from './logger.js';\nimport { deepCopy, isObject, printUsage, toBoolean } from './utils.js';\nimport {\n  absoluteProps,\n  defaultConfig,\n  nestedArgs,\n  promptsConfig\n} from './schemas/config.js';\n\nlet generalOptions = {};\n\n/**\n * Getter for the general options.\n *\n * @return {object} - General options object.\n */\nexport const getOptions = () => generalOptions;\n\n/**\n * Initializes and sets the general options for the server instace.\n *\n * @param {object} userOptions - Additional user options (e.g. from the node\n * module usage).\n * @param {string[]} args - CLI arguments.\n * @return {object} - General options object.\n */\nexport const setOptions = (userOptions, args) => {\n  // Only for the CLI usage\n  if (args?.length) {\n    // Get the additional options from the custom JSON file\n    generalOptions = loadConfigFile(args);\n  }\n\n  // Update the default config with a correct option values\n  updateDefaultConfig(defaultConfig, generalOptions);\n\n  // Set values for server's options and returns them\n  generalOptions = initOptions(defaultConfig);\n\n  // Apply user options if there are any\n  if (userOptions) {\n    // Merge user options\n    generalOptions = mergeConfigOptions(\n      generalOptions,\n      userOptions,\n      absoluteProps\n    );\n  }\n\n  // Only for the CLI usage\n  if (args?.length) {\n    // Pair provided arguments\n    generalOptions = pairArgumentValue(generalOptions, args, defaultConfig);\n  }\n\n  // Return final general options\n  return generalOptions;\n};\n\n/**\n * Displays a prompt for the manual configuration.\n *\n * @param {string} configFileName - The name of a configuration file.\n */\nexport const manualConfig = async (configFileName) => {\n  // Prepare a config object\n  let configFile = {};\n\n  // Check if provided config file exists\n  if (existsSync(configFileName)) {\n    configFile = JSON.parse(readFileSync(configFileName, 'utf8'));\n  }\n\n  // Question about a configuration category\n  const onSubmit = async (p, categories) => {\n    let questionsCounter = 0;\n    let allQuestions = [];\n\n    // Create a corresponding property in the manualConfig object\n    for (const section of categories) {\n      // Mark each option with a section\n      promptsConfig[section] = promptsConfig[section].map((option) => ({\n        ...option,\n        section\n      }));\n\n      // Collect the questions\n      allQuestions = [...allQuestions, ...promptsConfig[section]];\n    }\n\n    await prompts(allQuestions, {\n      onSubmit: async (prompt, answer) => {\n        // Get the default modules\n        if (prompt.name === 'modules') {\n          answer = answer.length\n            ? answer.map((module) => prompt.choices[module])\n            : prompt.choices;\n\n          configFile[prompt.section][prompt.name] = answer;\n        } else {\n          configFile[prompt.section] = recursiveProps(\n            Object.assign({}, configFile[prompt.section] || {}),\n            prompt.name.split('.'),\n            answer\n          );\n        }\n\n        if (++questionsCounter === allQuestions.length) {\n          try {\n            await fsPromises.writeFile(\n              configFileName,\n              JSON.stringify(configFile, null, 2),\n              'utf8'\n            );\n          } catch (error) {\n            log(1, `[config] Error while creating config.json: ${error}`);\n          }\n          return true;\n        }\n      }\n    });\n\n    return true;\n  };\n\n  // Find the categories\n  const choices = Object.keys(promptsConfig).map((choice) => ({\n    title: `${choice} options`,\n    value: choice\n  }));\n\n  // Category prompt\n  return prompts(\n    {\n      type: 'multiselect',\n      name: 'category',\n      message: 'Which category do you want to configure?',\n      hint: 'Space: Select specific, A: Select all, Enter: Confirm.',\n      instructions: '',\n      choices\n    },\n    { onSubmit }\n  );\n};\n\n/**\n * Maps the old options to the new config structure.\n *\n * @param {object} oldOptions - Options to be mapped.\n */\nexport const mapToNewConfig = (oldOptions) => {\n  const newOptions = {};\n  // Cycle through old-structured options\n  for (const [key, value] of Object.entries(oldOptions)) {\n    const propertiesChain = nestedArgs[key] ? nestedArgs[key].split('.') : [];\n\n    // Populate object in correct properties levels\n    propertiesChain.reduce(\n      (obj, prop, index) =>\n        (obj[prop] =\n          propertiesChain.length - 1 === index ? value : obj[prop] || {}),\n      newOptions\n    );\n  }\n  return newOptions;\n};\n\n/**\n * Merges the new options to the options object. It omits undefined values.\n *\n * @param {object} options - Old options.\n * @param {object} newOptions - New options.\n * @param {string[]} absoluteProps - Array of object names that should be force\n * merged.\n */\nexport const mergeConfigOptions = (options, newOptions, absoluteProps = []) => {\n  const mergedOptions = deepCopy(options);\n\n  for (const [key, value] of Object.entries(newOptions)) {\n    mergedOptions[key] =\n      isObject(value) &&\n      !absoluteProps.includes(key) &&\n      mergedOptions[key] !== undefined\n        ? mergeConfigOptions(mergedOptions[key], value, absoluteProps)\n        : value !== undefined\n        ? value\n        : mergedOptions[key];\n  }\n\n  return mergedOptions;\n};\n\n/**\n * Initializes options for the `startExport` method by merging user options\n * with the general options.\n *\n * @param {any} exportOptions - User options for exporting.\n * @param {any} generalOptions - General options are used for the export server.\n * @return {object} - User options merged with default options.\n */\nexport const initExportSettings = (exportOptions, generalOptions = {}) => {\n  let options = {};\n\n  if (exportOptions.svg) {\n    options = deepCopy(generalOptions);\n    options.export.type = exportOptions.type || exportOptions.export.type;\n    options.export.scale = exportOptions.scale || exportOptions.export.scale;\n    options.export.outfile =\n      exportOptions.outfile || exportOptions.export.outfile;\n    options.payload = {\n      svg: exportOptions.svg\n    };\n  } else {\n    options = mergeConfigOptions(\n      generalOptions,\n      exportOptions,\n      // Omit going down recursively with the belows\n      absoluteProps\n    );\n  }\n\n  options.export.outfile =\n    options.export?.outfile || `chart.${options.export?.type || 'png'}`;\n  return options;\n};\n\n/**\n * Loads the configuration from a custom JSON file.\n *\n * @param {string[]} args - CLI arguments.\n * @return {object} - Options object from the JSON file.\n */\nfunction loadConfigFile(args) {\n  // Check if the --loadConfig option was used\n  const configIndex = args.findIndex(\n    (arg) => arg.replace(/-/g, '') === 'loadConfig'\n  );\n\n  // Check if the --loadConfig has a value\n  if (configIndex > -1 && args[configIndex + 1]) {\n    const fileName = args[configIndex + 1];\n    try {\n      // Check if an additional config file is a correct JSON file\n      if (fileName && fileName.endsWith('.json')) {\n        // Load an optional custom JSON config file\n        return JSON.parse(readFileSync(fileName));\n      }\n    } catch (error) {\n      log(1, `[config] Unable to load config from the ${fileName}: ${error}`);\n    }\n  }\n\n  // No additional options to return\n  return {};\n}\n\n/**\n * Setting correct values of the options from the default config.\n *\n * @param {object} configObj - The config object based on which the initial\n * configuration be made.\n * @param {object} customObj - The custom object which can contain additional\n * option values to set.\n * @param {string} propChain - Required for creating a string chain of\n * properties for nested arguments.\n */\nfunction updateDefaultConfig(configObj, customObj = {}, propChain = '') {\n  Object.keys(configObj).forEach((key) => {\n    if (!['puppeteer', 'highcharts'].includes(key)) {\n      const entry = configObj[key];\n      const customValue = customObj && customObj[key];\n      let numEnvVal;\n\n      if (typeof entry.value === 'undefined') {\n        updateDefaultConfig(entry, customValue, `${propChain}.${key}`);\n      } else {\n        // If a value from a custom JSON exists, it take precedence\n        if (customValue !== undefined) {\n          entry.value = customValue;\n        }\n\n        // If a value from an env variable exists, it take precedence\n        if (entry.envLink) {\n          // Load the env var\n          if (entry.type === 'boolean') {\n            entry.value = toBoolean(\n              [process.env[entry.envLink], entry.value].find(\n                (el) => el || el === 'false'\n              )\n            );\n          } else if (entry.type === 'number') {\n            numEnvVal = +process.env[entry.envLink];\n            entry.value = numEnvVal >= 0 ? numEnvVal : entry.value;\n          } else if (\n            entry.type.indexOf(']') >= 0 &&\n            process.env[entry.envLink]\n          ) {\n            entry.value = process.env[entry.envLink].split(',');\n          } else {\n            entry.value = process.env[entry.envLink] || entry.value;\n          }\n        }\n      }\n    }\n  });\n}\n\n/**\n * Inits options recursively.\n *\n * @param {any} items - Items to update options from.\n * @return {object} - Updated options object.\n */\nfunction initOptions(items) {\n  let options = {};\n  for (const [name, item] of Object.entries(items)) {\n    options[name] = Object.prototype.hasOwnProperty.call(item, 'value')\n      ? item.value\n      : initOptions(item);\n  }\n  return options;\n}\n\n/**\n * Pairs argument with a corresponding value.\n *\n * @param {object} options - All server options.\n * @param {string[]} args - Array of arguments from a user.\n * @param {object} defaultConfig - The default config object.\n */\nfunction pairArgumentValue(options, args, defaultConfig) {\n  for (let i = 0; i < args.length; i++) {\n    let option = args[i].replace(/-/g, '');\n\n    // Find the right place for property's value\n    const propertiesChain = nestedArgs[option]\n      ? nestedArgs[option].split('.')\n      : [];\n\n    propertiesChain.reduce((obj, prop, index) => {\n      if (propertiesChain.length - 1 === index) {\n        // Finds an option and set a corresponding value\n        if (typeof obj[prop] !== 'undefined') {\n          if (args[++i]) {\n            obj[prop] = args[i] || obj[prop];\n          } else {\n            console.log(`Missing argument value for ${option}!`.red, '\\n');\n            options = printUsage(defaultConfig);\n          }\n        }\n      }\n      return obj[prop];\n    }, options);\n  }\n\n  return options;\n}\n\n/**\n * Recursively sets a property in a correct indentation level based on the\n * array of nested properties names.\n *\n * @param {object} objectToUpdate - Object where a property must be set on a\n * correct level.\n * @param  {string[]}nestedNames - Array of nasted names that indicates\n * indentation level.\n * @param {any} value - A value to assign to the property.\n * @return {object} - Updated options object.\n */\nfunction recursiveProps(objectToUpdate, nestedNames, value) {\n  while (nestedNames.length > 1) {\n    const propName = nestedNames.shift();\n\n    // Create a property in object if it doesn't exist\n    if (!Object.prototype.hasOwnProperty.call(objectToUpdate, propName)) {\n      objectToUpdate[propName] = {};\n    }\n\n    // Call function again if there still names to go\n    objectToUpdate[propName] = recursiveProps(\n      Object.assign({}, objectToUpdate[propName]),\n      nestedNames,\n      value\n    );\n\n    return objectToUpdate;\n  }\n\n  // Assign the final value\n  objectToUpdate[nestedNames[0]] = value;\n  return objectToUpdate;\n}\n\nexport default {\n  getOptions,\n  setOptions,\n  manualConfig,\n  mapToNewConfig,\n  mergeConfigOptions,\n  initExportSettings\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { readFile, readFileSync, writeFileSync } from 'fs';\n\nimport { log } from './logger.js';\nimport { killPool, postWork } from './pool.js';\nimport {\n  clearText,\n  fixType,\n  handleResources,\n  isCorrectJSON,\n  optionsStringify,\n  roundNumber,\n  toBoolean,\n  wrapAround\n} from './utils.js';\nimport { initExportSettings, getOptions } from './config.js';\n\nlet allowCodeExecution = false;\n\nexport const startExport = async (settings, endCallback) => {\n  // Starting exporting process message\n  log(4, '[chart] Starting exporting process.');\n\n  // Initialize options\n  const options = initExportSettings(settings, getOptions());\n\n  // Get the export options\n  const exportOptions = options.export;\n\n  // If SVG is an input (argument can be sent only by the request)\n  if (options.payload?.svg && options.payload.svg !== '') {\n    return exportAsString(options.payload.svg.trim(), options, endCallback);\n  }\n\n  // Export using options from the file\n  if (exportOptions.infile && exportOptions.infile.length) {\n    log(4, '[chart] Attempting to export from an input file.');\n\n    // Try to read the file\n    return readFile(exportOptions.infile, 'utf8', (error, infile) => {\n      if (error) {\n        return log(1, `[chart] Error loading input file: ${error}.`);\n      }\n\n      // Get the string representation\n      options.export.instr = infile;\n      return exportAsString(options.export.instr.trim(), options, endCallback);\n    });\n  }\n\n  // Export with options from the raw representation\n  if (\n    (exportOptions.instr && exportOptions.instr !== '') ||\n    (exportOptions.options && exportOptions.options !== '')\n  ) {\n    log(4, '[chart] Attempting to export from a raw input.');\n\n    // Perform a direct inject when forced\n    if (toBoolean(options.customCode?.allowCodeExecution)) {\n      return doStraightInject(options, endCallback);\n    }\n\n    // Either try to parse to JSON first or do the direct export\n    return typeof exportOptions.instr === 'string'\n      ? exportAsString(exportOptions.instr.trim(), options, endCallback)\n      : doExport(\n          options,\n          exportOptions.instr || exportOptions.options,\n          endCallback\n        );\n  }\n\n  // No input specified, pass an error message to the callback\n  log(\n    1,\n    clearText(\n      `[chart] No input specified.\n      ${JSON.stringify(exportOptions, undefined, '  ')}.`\n    )\n  );\n\n  return (\n    endCallback &&\n    endCallback(false, {\n      error: true,\n      message: 'No input specified.'\n    })\n  );\n};\n\nexport const batchExport = (options) => {\n  const batchFunctions = [];\n\n  // Split and pair the --batch arguments\n  for (let pair of options.export.batch.split(';')) {\n    pair = pair.split('=');\n    if (pair.length === 2) {\n      batchFunctions.push(\n        new Promise((resolve, reject) => {\n          startExport(\n            {\n              ...options,\n              export: {\n                ...options.export,\n                infile: pair[0],\n                outfile: pair[1]\n              }\n            },\n            (info, error) => {\n              // Throw an error\n              if (error) {\n                return reject(error);\n              }\n\n              // Save the base64 from a buffer to a correct image file\n              writeFileSync(\n                info.options.export.outfile,\n                Buffer.from(info.data, 'base64')\n              );\n\n              resolve();\n            }\n          );\n        })\n      );\n    }\n  }\n\n  // Kill the pool after all exports are done\n  Promise.all(batchFunctions)\n    .then(() => {\n      killPool();\n    })\n    .catch((error) => {\n      log(1, `[chart] Error encountered during batch export: ${error}`);\n      killPool();\n    });\n};\n\nexport const singleExport = (options) => {\n  // Use instr or its alias, options\n  options.export.instr = options.export.instr || options.export.options;\n\n  // Perform an export\n  startExport(options, (info, error) => {\n    // Exit process when error\n    if (error) {\n      log(1, `[cli] ${error.message}`);\n      process.exit(1);\n    }\n\n    const { outfile, type } = info.options.export;\n\n    // Save the base64 from a buffer to a correct image file\n    writeFileSync(\n      outfile || `chart.${type}`,\n      type !== 'svg' ? Buffer.from(info.data, 'base64') : info.data\n    );\n\n    // Kill the pool\n    killPool();\n  });\n};\n\n/**\n * Function for choosing chart size and scale based on options prioritization.\n *\n * @param {object} options - All options object.\n * @return {object} - An object with updated size and scale for a chart.\n */\nexport const findChartSize = (options) => {\n  const { chart, exporting } =\n    options.export?.options || isCorrectJSON(options.export?.instr);\n\n  // See if globalOptions holds chart or exporting size\n  const globalOptions = isCorrectJSON(options.export?.globalOptions);\n\n  // Secure scale value\n  let scale =\n    options.export?.scale ||\n    exporting?.scale ||\n    globalOptions?.exporting?.scale ||\n    options.export?.defaultScale ||\n    1;\n\n  // the scale cannot be lower than 0.1 and cannot be higher than 5.0\n  scale = Math.max(0.1, Math.min(scale, 5.0));\n\n  // we want to round the numbers like 0.23234 -> 0.23\n  scale = roundNumber(scale, 2);\n\n  // Find chart size and scale\n  return {\n    height:\n      options.export?.height ||\n      exporting?.sourceHeight ||\n      chart?.height ||\n      globalOptions?.exporting?.sourceHeight ||\n      globalOptions?.chart?.height ||\n      options.export?.defaultHeight ||\n      400,\n    width:\n      options.export?.width ||\n      exporting?.sourceWidth ||\n      chart?.width ||\n      globalOptions?.exporting?.sourceWidth ||\n      globalOptions?.chart?.width ||\n      options.export?.defaultWidth ||\n      600,\n    scale\n  };\n};\n\n/**\n * Function for final options preparation before export.\n *\n * @param {object} options - All options object.\n * @param {object} chartJson - Chart JSON.\n * @param {function} endCallback - The end callback.\n * @param {string} svg - The SVG representation.\n */\nconst doExport = (options, chartJson, endCallback, svg) => {\n  let { export: exportOptions, customCode: customCodeOptions } = options;\n\n  const allowCodeExecutionScoped =\n    typeof customCodeOptions.allowCodeExecution === 'boolean'\n      ? customCodeOptions.allowCodeExecution\n      : allowCodeExecution;\n\n  if (!customCodeOptions) {\n    customCodeOptions = options.customCode = {};\n  } else if (typeof options.customCode.resources === 'string') {\n    // Process resources\n    options.customCode.resources = handleResources(\n      options.customCode.resources,\n      toBoolean(options.customCode.allowFileResources)\n    );\n  } else if (!options.customCode.resources) {\n    try {\n      const resources = readFileSync('resources.json', 'utf8');\n      options.customCode.resources = handleResources(\n        resources,\n        toBoolean(options.customCode.allowFileResources)\n      );\n    } catch (err) {\n      log(3, `[chart] The default resources.json file not found.`);\n    }\n  }\n\n  // If the allowCodeExecution flag isn't set, we should refuse the usage\n  // of callback, resources, and custom code. Additionally, the worker will\n  // refuse to run arbitrary JavaScript. Prioritized should be the scoped\n  // option, then we should take a look at the overall pool option.\n  if (!allowCodeExecutionScoped && customCodeOptions) {\n    if (\n      customCodeOptions.callback ||\n      customCodeOptions.resources ||\n      customCodeOptions.customCode\n    ) {\n      // Send back a friendly message saying that the exporter does not support\n      // these settings.\n      return (\n        endCallback &&\n        endCallback(false, {\n          error: true,\n          message: clearText(\n            `The callback, resources and customCode have been disabled for this\n            server.`\n          )\n        })\n      );\n    }\n\n    // Reset all additional custom code\n    customCodeOptions.callback = false;\n    customCodeOptions.resources = false;\n    customCodeOptions.customCode = false;\n  }\n\n  // Clean properties to keep it lean and mean\n  if (chartJson) {\n    chartJson.chart = chartJson.chart || {};\n    chartJson.exporting = chartJson.exporting || {};\n    chartJson.exporting.enabled = false;\n  }\n\n  exportOptions.constr = exportOptions.constr || 'chart';\n  exportOptions.type = fixType(exportOptions.type, exportOptions.outfile);\n  if (exportOptions.type === 'svg') {\n    exportOptions.width = false;\n  }\n\n  // Prepare global and theme options\n  ['globalOptions', 'themeOptions'].forEach((optionsName) => {\n    try {\n      if (exportOptions && exportOptions[optionsName]) {\n        if (\n          typeof exportOptions[optionsName] === 'string' &&\n          exportOptions[optionsName].endsWith('.json')\n        ) {\n          exportOptions[optionsName] = isCorrectJSON(\n            readFileSync(exportOptions[optionsName], 'utf8'),\n            true\n          );\n        } else {\n          exportOptions[optionsName] = isCorrectJSON(\n            exportOptions[optionsName],\n            true\n          );\n        }\n      }\n    } catch (error) {\n      exportOptions[optionsName] = {};\n      log(1, `[chart] The ${optionsName} not found.`);\n    }\n  });\n\n  // Prepare customCode\n  if (customCodeOptions.allowCodeExecution) {\n    customCodeOptions.customCode = wrapAround(\n      customCodeOptions.customCode,\n      customCodeOptions.allowFileResources\n    );\n  }\n\n  // Get the callback\n  if (\n    customCodeOptions &&\n    customCodeOptions.callback &&\n    customCodeOptions.callback?.indexOf('{') < 0\n  ) {\n    // The allowFileResources is always set to false for HTTP requests to avoid\n    // injecting arbitrary files from the fs\n    if (customCodeOptions.allowFileResources) {\n      try {\n        customCodeOptions.callback = readFileSync(\n          customCodeOptions.callback,\n          'utf8'\n        );\n      } catch (error) {\n        log(2, `[chart] Error loading callback: ${error}.`);\n        customCodeOptions.callback = false;\n      }\n    } else {\n      customCodeOptions.callback = false;\n    }\n  }\n\n  // Size search\n  options.export = {\n    ...options.export,\n    ...findChartSize(options)\n  };\n\n  // Post the work to the pool\n  postWork(exportOptions.strInj || chartJson || svg, options)\n    .then((result) => endCallback(result))\n    .catch((error) => {\n      log(0, '[chart] When posting work:', error);\n      return endCallback(false, error);\n    });\n};\n\n/**\n * Function for straight injecting the code.\n * Dangerous and must be used deliberately by someone who sets up a server\n * (see  --allowCodeExecution).\n *\n * @param {object} options - All options object.\n * @param {function} endCallback - The function to call when exporting is done.\n */\nconst doStraightInject = (options, endCallback) => {\n  try {\n    let strInj;\n    let instr = options.export.instr || options.export.options;\n\n    if (typeof instr !== 'string') {\n      // Try to stringify options\n      strInj = instr = optionsStringify(\n        instr,\n        options.customCode?.allowCodeExecution\n      );\n    }\n    strInj = instr.replaceAll(/\\t|\\n|\\r/g, '').trim();\n\n    // Get rid of the ;\n    if (strInj[strInj.length - 1] === ';') {\n      strInj = strInj.substring(0, strInj.length - 1);\n    }\n\n    // Save as stright inject string\n    options.export.strInj = strInj;\n    return doExport(options, false, endCallback);\n  } catch (error) {\n    const message = clearText(\n      `Malformed input detected for ${options.export?.requestId || '?'}:\n      Please make sure that your JSON/JavaScript options\n      are sent using the \"options\" attribute, and that if you're using\n      SVG, it is unescaped.`\n    );\n\n    log(1, message);\n    return (\n      endCallback &&\n      endCallback(\n        false,\n        JSON.stringify({\n          error: true,\n          message\n        })\n      )\n    );\n  }\n};\n\n/**\n * Prepares an input before exporting.\n *\n * @param {string} stringToExport - String representation of SVG/export options.\n * @param {object} options - All options object.\n * @param {function} endCallback - The function to call when exporting is done.\n */\nconst exportAsString = (stringToExport, options, endCallback) => {\n  const { allowCodeExecution } = options.customCode;\n\n  // Check if it is SVG\n  if (\n    stringToExport.indexOf('<svg') >= 0 ||\n    stringToExport.indexOf('<?xml') >= 0\n  ) {\n    log(4, '[chart] Parsing input as SVG.');\n    return doExport(options, false, endCallback, stringToExport);\n  }\n\n  try {\n    // Try to parse to JSON and call the doExport function\n    const chartJSON = JSON.parse(stringToExport.replaceAll(/\\t|\\n|\\r/g, ' '));\n\n    // If a correct JSON, do the export\n    return doExport(options, chartJSON, endCallback);\n  } catch (error) {\n    // Not a valid JSON\n    if (toBoolean(allowCodeExecution)) {\n      return doStraightInject(options, endCallback);\n    } else {\n      // Do not allow straight injection without the allowCodeExecution flag\n      return (\n        endCallback &&\n        endCallback(false, {\n          error: true,\n          message: clearText(\n            `Only JSON configurations and SVG is allowed for this server. If\n            this is your server, JavaScript exporting can be enabled by starting\n            the server with the --allowCodeExecution flag.`\n          )\n        })\n      );\n    }\n  }\n};\n\nexport const getAllowCodeExecution = () => allowCodeExecution;\n\nexport const setAllowCodeExecution = (value) => {\n  allowCodeExecution = toBoolean(value);\n};\n\n/**\n * Starts an exporting process\n *\n * @param {object} settings - Settings for export.\n * @param {function} endCallback - The function to call when exporting is done.\n */\nexport default {\n  batchExport,\n  singleExport,\n  getAllowCodeExecution,\n  setAllowCodeExecution,\n  startExport,\n  findChartSize\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { v4 as uuid } from 'uuid';\n\nimport { getAllowCodeExecution, startExport } from '../../chart.js';\nimport { getOptions, mergeConfigOptions } from '../../config.js';\nimport { log } from '../../logger.js';\nimport {\n  clearText,\n  fixType,\n  isCorrectJSON,\n  isPrivateRangeUrlFound,\n  optionsStringify,\n  measureTime\n} from '../../utils.js';\n\n// Reversed MIME types\nconst reversedMime = {\n  png: 'image/png',\n  jpeg: 'image/jpeg',\n  gif: 'image/gif',\n  pdf: 'application/pdf',\n  svg: 'image/svg+xml'\n};\n\n// The requests counter\nlet requestsCounter = 0;\n\nconst benchmark = false;\n\n// The array of callbacks to call before a request\nconst beforeRequest = [];\n\n// The array of callbacks to call after a request\nconst afterRequest = [];\n\n/**\n * Calls callbacks.\n *\n * @param {Array} callbacks - An array of callbacks.\n * @param {object} request - The request.\n * @param {object} response - The response.\n * @param {object} data - The data to send to callbacks.\n * @return {object} - The result from a callback.\n */\nconst doCallbacks = (callbacks, request, response, data) => {\n  let result = true;\n  const { id, uniqueId, type, body } = data;\n\n  callbacks.some((callback) => {\n    if (callback) {\n      let callResponse = callback(request, response, id, uniqueId, type, body);\n\n      if (callResponse !== undefined && callResponse !== true) {\n        result = callResponse;\n      }\n\n      return true;\n    }\n  });\n\n  return result;\n};\n\n/**\n * Handles an export.\n *\n * @param {object} request - The request.\n * @param {object} response - The response.\n */\nconst exportHandler = (request, response) => {\n  // Start counting time\n  const stopCounter = measureTime();\n\n  // Get the current server's general options\n  const defaultOptions = getOptions();\n\n  // Init default options\n  if (benchmark) {\n    console.log('Init default options:', stopCounter(), 'ms.');\n  }\n\n  const body = request.body;\n  const id = ++requestsCounter;\n  const uniqueId = uuid().replace(/-/g, '');\n  let type = fixType(body.type);\n\n  // Fix type\n  if (benchmark) {\n    console.log('Fix type:', stopCounter(), 'ms.');\n  }\n\n  // Throw 'Bad Request' if there's no body\n  if (!body) {\n    return response.status(400).send(\n      clearText(\n        `Body is required. Sending a body? Make sure your Content-type header\n        is correct. Accepted is application/json and multipart/form-data.`\n      )\n    );\n  }\n\n  // All of the below can be used\n  let instr = isCorrectJSON(body.infile || body.options || body.data);\n\n  // Is correct JSON\n  if (benchmark) {\n    console.log('Is correct JSON:', stopCounter(), 'ms.');\n  }\n\n  // Throw 'Bad Request' if there's no JSON or SVG to export\n  if (!instr && !body.svg) {\n    log(\n      2,\n      clearText(\n        `Request ${uniqueId} from ${\n          request.headers['x-forwarded-for'] || request.connection.remoteAddress\n        } was incorrect. Check your payload.`\n      )\n    );\n\n    return response.status(400).send(\n      clearText(\n        `No correct chart data found. Please make sure you are using\n        application/json or multipart/form-data headers, and that the chart\n        data is in the 'infile', 'options' or 'data' attribute if sending\n        JSON or in the 'svg' if sending SVG.`\n      )\n    );\n  }\n\n  let callResponse = false;\n\n  // Call the before request functions\n  callResponse = doCallbacks(beforeRequest, request, response, {\n    id,\n    uniqueId,\n    type,\n    body\n  });\n\n  // Do callbacks\n  if (benchmark) {\n    console.log('Do callbacks:', stopCounter(), 'ms.');\n  }\n\n  // Block the request if one of a callbacks failed\n  if (callResponse !== true) {\n    return response.send(callResponse);\n  }\n\n  let connectionAborted = false;\n\n  // In case the connection is closed, force to abort further actions\n  request.socket.on('close', () => {\n    connectionAborted = true;\n  });\n\n  log(4, `[export] Got an incoming HTTP request ${uniqueId}.`);\n\n  body.constr = (typeof body.constr === 'string' && body.constr) || 'chart';\n\n  // Gather and organize options from the payload\n  const requestOptions = {\n    export: {\n      instr,\n      type,\n      constr: body.constr[0].toLowerCase() + body.constr.substr(1),\n      height: body.height,\n      width: body.width,\n      scale: body.scale || defaultOptions.export.scale,\n      globalOptions: isCorrectJSON(body.globalOptions, true),\n      themeOptions: isCorrectJSON(body.themeOptions, true)\n    },\n    customCode: {\n      allowCodeExecution: getAllowCodeExecution(),\n      allowFileResources: false,\n      resources: isCorrectJSON(body.resources, true),\n      callback: body.callback,\n      customCode: body.customCode\n    }\n  };\n\n  // Organize options\n  if (benchmark) {\n    console.log('Organize options:', stopCounter(), 'ms.');\n  }\n\n  if (instr) {\n    // Stringify JSON with options\n    requestOptions.export.instr = optionsStringify(\n      instr,\n      requestOptions.customCode.allowCodeExecution\n    );\n\n    // Stringify JSON with options\n    if (benchmark) {\n      console.log('Stringify JSON with options:', stopCounter(), 'ms.');\n    }\n  }\n\n  // Merge the request options into default ones\n  const options = mergeConfigOptions(defaultOptions, requestOptions);\n\n  // Merge config options\n  if (benchmark) {\n    console.log('Merge config options:', stopCounter(), 'ms.');\n  }\n\n  // Save the JSON if exists\n  options.export.options = instr;\n\n  // Lastly, add the server specific arguments into options as payload\n  options.payload = {\n    svg: body.svg || false,\n    b64: body.b64 || false,\n    dataOptions: isCorrectJSON(body.dataOptions, true),\n    noDownload: body.noDownload || false,\n    requestId: uniqueId\n  };\n\n  // Setting payload\n  if (benchmark) {\n    console.log('Setting payload:', stopCounter(), 'ms.');\n  }\n\n  // Test xlink:href elements from payload's SVG\n  if (body.svg && isPrivateRangeUrlFound(options.payload.svg)) {\n    return response\n      .status(400)\n      .send(\n        'SVG potentially contain at least one forbidden URL in xlink:href element.'\n      );\n  }\n\n  // Check URL range\n  if (benchmark) {\n    console.log('Check URL range:', stopCounter(), 'ms.');\n  }\n\n  // Start the export process\n  startExport(options, (info, error) => {\n    // Remove the close event from the socket\n    request.socket.removeAllListeners('close');\n\n    // After Puppeteer exporting\n    if (benchmark) {\n      console.log('After Puppeteer exporting:', stopCounter(), 'ms.', '\\n');\n    }\n\n    // If the connection was closed, do nothing\n    if (connectionAborted) {\n      return log(\n        3,\n        clearText(\n          `[export] The client closed the connection before the chart was done\n          processing.`\n        )\n      );\n    }\n\n    // If error, return it\n    if (error) {\n      log(\n        1,\n        clearText(\n          `[export] Work: ${uniqueId} could not be completed, sending:\n          ${error}`\n        )\n      );\n      return response.status(400).send(error.message);\n    }\n\n    // If data is missing, return the error\n    if (!info || !info.data) {\n      log(\n        1,\n        clearText(\n          `[export] Unexpected return from chart generation, please check your\n          data Request: ${uniqueId} is ${info.data}.`\n        )\n      );\n      return response\n        .status(400)\n        .send(\n          'Unexpected return from chart generation, please check your data.'\n        );\n    }\n\n    // Get the type from options\n    type = info.options.export.type;\n\n    // The after request callbacks\n    doCallbacks(afterRequest, request, response, { id, body: info.data });\n\n    if (info.data) {\n      // If only base64 is required, return it\n      if (body.b64) {\n        // Check if it is already base64 or a raw SVG\n        if (type === 'pdf') {\n          return response.send(\n            Buffer.from(info.data, 'utf8').toString('base64')\n          );\n        }\n        return response.send(info.data);\n      }\n\n      // Set correct content type\n      response.header('Content-Type', reversedMime[type] || 'image/png');\n\n      // Decide whether to download or not chart file\n      if (!body.noDownload) {\n        response.attachment(\n          `${request.params.filename || 'chart'}.${type || 'png'}`\n        );\n      }\n\n      // If SVG, return plain content\n      return type === 'svg'\n        ? response.send(info.data)\n        : response.send(Buffer.from(info.data, 'base64'));\n    }\n  });\n};\n\nexport default (app) => {\n  app.post('/', exportHandler);\n  app.post('/:filename', exportHandler);\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { promises as fsPromises } from 'fs';\nimport { posix } from 'path';\n\nimport bodyParser from 'body-parser';\nimport cors from 'cors';\nimport express from 'express';\nimport multer from 'multer';\nimport http from 'http';\nimport https from 'https';\n\nimport { log } from '../logger.js';\nimport rateLimit from './rate_limit.js';\nimport { __dirname } from '../utils.js';\n\nimport healthRoute from './routes/health.js';\nimport exportRoutes from './routes/export.js';\nimport vswitchRoute from './routes/change_hc_version.js';\nimport uiRoute from './routes/ui.js';\n\n// Create express app\nconst app = express();\n\n// Disable the X-Powered-By header\napp.disable('x-powered-by');\n\n// Enable CORS support\napp.use(cors());\n\n// Enable parsing of form data (files) with Multer package\nconst storage = multer.memoryStorage();\nconst upload = multer({\n  storage,\n  limits: {\n    fieldsSize: '50MB'\n  }\n});\n\napp.use(upload.any());\n\n// Enable body parser\napp.use(bodyParser.json({ limit: '50mb' }));\napp.use(bodyParser.urlencoded({ extended: true, limit: '50mb' }));\napp.use(bodyParser.urlencoded({ extended: false, limit: '50mb' }));\n\n/**\n * Error handler function.\n *\n * @param {object} error - An error object.\n * @return {string} - An error message.\n */\nconst errorHandler = (error) => log(1, `[server] Socket error: ${error}`);\n\n/**\n * Attaches error handlers for a server.\n *\n * @param {object} server - The http/https server.\n */\nconst attachErrorHandlers = (server) => {\n  server.on('clientError', errorHandler);\n  server.on('error', errorHandler);\n  server.on('connection', (socket) =>\n    socket.on('error', (error) => errorHandler(error, socket))\n  );\n};\n\nexport const startServer = async (serverConfig) => {\n  // Stop if not enabled\n  if (!serverConfig.enable) {\n    return false;\n  }\n\n  // // Get the pool\n  // const pool = getPool();\n\n  // // Try to create browser instance before starting the server\n  // const resource = await pool.acquire();\n\n  // // If not found, throw an error\n  // if (!resource.browser) {\n  //   log(1, `[server] Could not acquire browser instance.`);\n  //   process.exit(1);\n  // }\n\n  // // Release the resource\n  // pool.release(resource);\n\n  // Listen HTTP server\n  if (!serverConfig.ssl.enable && !serverConfig.ssl.force) {\n    // Main server instance (HTTP)\n    const httpServer = http.createServer(app);\n    // Attach error handlers and listen to the server\n    attachErrorHandlers(httpServer);\n    // Listen\n    httpServer.listen(serverConfig.port, serverConfig.host);\n\n    log(\n      3,\n      `[server] Started HTTP server on ${serverConfig.host}:${serverConfig.port}.`\n    );\n  }\n\n  // Listen HTTPS server\n  if (serverConfig.ssl.enable) {\n    // Set up an SSL server also\n    let key, cert;\n\n    try {\n      // Get the SSL key\n      key = await fsPromises.readFile(\n        posix.join(serverConfig.ssl.certPath, 'server.key'),\n        'utf8'\n      );\n\n      // Get the SSL certificate\n      cert = await fsPromises.readFile(\n        posix.join(serverConfig.ssl.certPath, 'server.crt'),\n        'utf8'\n      );\n    } catch (error) {\n      log(\n        1,\n        `[server] Unable to load key/certificate from ${serverConfig.ssl.certPath}.`\n      );\n    }\n\n    if (key && cert) {\n      // Main server instance (HTTPS)\n      const httpsServer = https.createServer(app);\n      // Attach error handlers and listen to the server\n      attachErrorHandlers(httpsServer);\n      // Listen\n      httpsServer.listen(serverConfig.ssl.port, serverConfig.host);\n\n      log(\n        3,\n        `[server] Started HTTPS server on ${serverConfig.host}:${serverConfig.ssl.port}.`\n      );\n    }\n  }\n\n  // Enable the rate limiter if config says so\n  if (\n    serverConfig.rateLimiting &&\n    serverConfig.rateLimiting.enable &&\n    ![0, NaN].includes(serverConfig.rateLimiting.maxRequests)\n  ) {\n    rateLimit(app, serverConfig.rateLimiting);\n  }\n\n  // Set up static folder's route\n  app.use(express.static(posix.join(__dirname, 'public')));\n\n  // Set up routes\n  healthRoute(app);\n  exportRoutes(app);\n  uiRoute(app);\n  vswitchRoute(app);\n};\n\n/**\n * Returns the express instance.\n */\nexport const getExpress = () => {\n  return express;\n};\n\n/**\n * Returns the app instance.\n */\nexport const getApp = () => {\n  return app;\n};\n\n/**\n * Adds a middleware to the server.\n *\n * @param {object} path - An endpoint path to add middlewares to.\n * @param {Array} middlewares - An unlimited number of middlewares to use\n * against the specific endpoint.\n */\nexport const use = (path, ...middlewares) => {\n  app.use(path, ...middlewares);\n};\n\n/**\n * Adds a get route to the server.\n *\n * @param {object} path - An endpoint path to add middlewares to.\n * @param {Array} middlewares - An unlimited number of middlewares to use\n * against the specific endpoint for GET method.\n */\nexport const get = (path, ...middlewares) => {\n  app.get(path, ...middlewares);\n};\n\n/**\n * Adds a post route to the server.\n *\n * @param {object} path - An endpoint path to add middlewares to.\n * @param {Array} middlewares - An unlimited number of middlewares to use\n * against the specific endpoint for POST method.\n */\nexport const post = (path, ...middlewares) => {\n  app.post(path, ...middlewares);\n};\n\n/**\n * Forcefully enables rate limiting.\n *\n * @param {object} limitConfig - The options object for the rate limiter\n * configuration.\n */\nexport const enableRateLimiting = (limitConfig) => {\n  return rateLimit(app, limitConfig);\n};\n\nexport default {\n  startServer,\n  getExpress,\n  getApp,\n  use,\n  get,\n  post,\n  enableRateLimiting\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { join } from 'path';\n\nimport { __dirname } from '../../utils.js';\n/**\n * Adds the / route for a UI when enabled for the export server\n */\nexport default (app) =>\n  !app\n    ? false\n    : app.get('/', (request, response) => {\n        response.sendFile(join(__dirname, 'public', 'index.html'));\n      });\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport cache from '../../cache.js';\n\n/**\n * Adds a route that can be used to change the HC version on the server\n * TODO: Add auth token and connect to API\n */\nexport default (app) =>\n  !app\n    ? false\n    : app.post('/change_hc_version/:newVersion', async (request, response) => {\n        const ctoken = process.env.HIGHCHARTS_ADMIN_TOKEN;\n\n        if (!ctoken || !ctoken.length) {\n          return response.send({\n            error: true,\n            message:\n              'Server not configured to do run-time version changes: HIGHCHARTS_ADMIN_TOKEN not set'\n          });\n        }\n\n        const token = request.get('hc-auth');\n\n        if (!token || token !== ctoken) {\n          return response.send({\n            error: true,\n            message: 'Invalid or missing token: set token in the hc-auth header'\n          });\n        }\n\n        const newVersion = request.params.newVersion;\n\n        if (newVersion) {\n          try {\n            // eslint-disable-next-line import/no-named-as-default-member\n            await cache.updateVersion(newVersion);\n          } catch (e) {\n            response.send({\n              error: true,\n              message: e\n            });\n          }\n\n          response.send({\n            version: cache.version()\n          });\n        } else {\n          response.send({\n            error: true,\n            message: 'No new version supplied'\n          });\n        }\n      });\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\n// Add the main directory in the global object\nimport 'colors';\n\nimport server, { startServer } from './server/server.js';\nimport {\n  setAllowCodeExecution,\n  batchExport,\n  singleExport,\n  startExport\n} from './chart.js';\nimport { mapToNewConfig, setOptions } from './config.js';\nimport { log, setLogLevel, enableFileLogging } from './logger.js';\nimport { killPool, init } from './pool.js';\nimport { checkCache } from './cache.js';\n\nexport default {\n  log,\n  mapToNewConfig,\n  setOptions,\n  singleExport,\n  startExport,\n  batchExport,\n  server,\n  startServer,\n  killPool,\n  initPool: async (options = {}) => {\n    // Set the allowCodeExecution per export module scope\n    setAllowCodeExecution(\n      options.customCode && options.customCode.allowCodeExecution\n    );\n\n    // Set the log level\n    setLogLevel(options.logging && parseInt(options.logging.level));\n\n    // Set the log file path and name\n    if (options.logging && options.logging.dest) {\n      enableFileLogging(\n        options.logging.dest,\n        options.logging.file || 'highcharts-export-server.log'\n      );\n    }\n\n    // Check if cache needs to be updated\n    await checkCache(options.highcharts || { version: 'latest' });\n\n    // Init the pool\n    await init({\n      pool: options.pool || {\n        initialWorkers: 1,\n        maxWorkers: 1\n      },\n      puppeteerArgs: options.puppeteer?.args || []\n    });\n\n    // Return updated options\n    return options;\n  }\n};\n"],"names":["dotenv","config","defaultConfig","puppeteer","args","value","type","description","highcharts","version","envLink","cdnURL","coreScripts","modules","indicators","scripts","forceFetch","export","infile","instr","options","outfile","constr","defaultHeight","defaultWidth","defaultScale","height","width","scale","globalOptions","themeOptions","batch","customCode","allowCodeExecution","allowFileResources","callback","resources","loadConfig","createConfig","server","enable","cliName","host","port","ssl","force","certPath","rateLimiting","maxRequests","window","delay","trustProxy","skipKey","skipToken","pool","initialWorkers","maxWorkers","workLimit","queueSize","timeoutThreshold","acquireTimeout","reaper","benchmarking","listenToProcessExits","logging","level","file","dest","ui","route","other","noLogo","payload","join","absoluteProps","nestedArgs","createNestedArgs","obj","propChain","Object","keys","forEach","k","includes","entry","substring","toConsole","toFile","pathCreated","levelsDesc","title","color","listeners","key","option","entries","log","newLevel","texts","length","prefix","Date","toString","split","trim","fn","existsSync","mkdirSync","appendFile","concat","error","console","apply","undefined","__dirname","fileURLToPath","URL","document","require","pathToFileURL","__filename","href","_documentCurrentScript","src","baseURI","clearText","text","rule","replacer","replaceAll","fixType","formats","outType","pop","find","t","handleResources","allowedProps","handledResources","correctResources","endsWith","isCorrectJSON","readFileSync","notice","files","propName","map","item","data","parsedData","JSON","parse","stringify","deepCopy","copy","Array","isArray","prototype","hasOwnProperty","call","optionsStringify","allowFunctions","name","startsWith","printUsage","bold","yellow","cycleCategories","categories","descName","green","i","blue","category","toUpperCase","red","toBoolean","wrapAround","replace","rateLimit","app","limitConfig","msg","rateOptions","max","limiter","windowMs","delayMs","handler","request","response","format","json","status","send","message","default","skip","query","access_token","use","async","fetch","url","requestOptions","Promise","resolve","reject","protocol","https","http","getProtocol","get","res","on","chunk","cachePath","cache","activeManifest","sources","hcVersion","appliedConfig","extractVersion","substr","indexOf","fetchScript","script","proxyAgent","agent","timeout","process","env","statusCode","updateCache","sourcePath","customScripts","allScripts","c","m","proxyHost","proxyPort","HttpsProxyAgent","fetchedModules","all","writeFileSync","checkCache","manifestPath","requestUpdate","manifest","moduleMap","numberOfModules","some","moduleName","newManifest","saveConfigToManifest","cache$1","newVersion","assign","RANDOM_PID","randomBytes","PUPPETEER_DIR","path","minimalArgs","template","fs","browser","newPage","p","setContent","addScriptTag","evaluate","setupHighcharts","err","$eval","element","errorMessage","_displayErrors","innerHTML","close","connected","__basedir","setAsConfig","page","chart","triggerExport","puppeteerExport","injectedResources","clearInjected","dispose","scriptsToRemove","getElementsByTagName","stylesToRemove","linksToRemove","remove","exportBench","exportOptions","requestAnimationFrame","displayErrors","debugger","d","svgBench","isSVG","setPageBench","svgTemplate","strInj","setContentBench","resBench","js","push","content","isLocal","cssBench","css","cssImports","match","cssImportPath","addStyleTag","size","chartHeight","baseVal","chartWidth","parseFloat","Highcharts","charts","vpBench","viewportHeight","Math","ceil","viewportWidth","setViewport","deviceScaleFactor","zoomCallback","body","style","zoom","margin","x","y","getBoundingClientRect","trunc","getClipRegion","round","expBenchmark","outerHTML","createSVG","encoding","clip","race","screenshot","omitBackground","setTimeout","Error","createImage","pdf","createPDF","oldCharts","oldChart","destroy","shift","puppeteerArgs","performedExports","exportAttempts","timeSpent","droppedExports","spentAverage","poolConfig","factory","create","id","uuid","s","getTime","browserNewPage","isClosed","workCount","random","validate","workerHandle","logLevel","init","allArgs","tryCount","open","launch","headless","userDataDir","e","createBrowser","killPool","code","exit","Pool","min","createRetryIntervalMillis","createTimeoutMillis","acquireTimeoutMillis","destroyTimeoutMillis","idleTimeoutMillis","reapIntervalMillis","propagateCreateError","eventId","resource","initialResources","acquire","promise","release","destroyed","postWork","fail","getPoolInfo","workStart","result","exportTime","available","borrowed","pending","spareResourceCapacity","pool$1","packageVersion","npm_package_version","serverStartTime","generalOptions","getOptions","mergeConfigOptions","newOptions","mergedOptions","updateDefaultConfig","configObj","customObj","customValue","numEnvVal","el","initOptions","items","startExport","settings","endCallback","svg","initExportSettings","exportAsString","readFile","doStraightInject","doExport","findChartSize","exporting","precision","multiplier","pow","roundNumber","sourceHeight","sourceWidth","chartJson","customCodeOptions","allowCodeExecutionScoped","enabled","optionsName","then","catch","requestId","stringToExport","chartJSON","reversedMime","png","jpeg","gif","requestsCounter","beforeRequest","afterRequest","doCallbacks","callbacks","uniqueId","callResponse","exportHandler","start","hrtime","bigint","measureTime","defaultOptions","headers","connection","remoteAddress","connectionAborted","socket","toLowerCase","b64","dataOptions","noDownload","ipRegEx","info","removeAllListeners","Buffer","from","header","attachment","params","filename","express","disable","cors","storage","multer","memoryStorage","upload","limits","fieldsSize","any","bodyParser","limit","urlencoded","extended","errorHandler","attachErrorHandlers","startServer","serverConfig","httpServer","createServer","listen","cert","fsPromises","posix","httpsServer","NaN","static","bootTime","uptime","floor","highchartsVersion","averageProcessingTime","failedExports","sucessRatio","healthRoute","post","exportRoutes","sendFile","uiRoute","ctoken","HIGHCHARTS_ADMIN_TOKEN","token","vswitchRoute","getExpress","getApp","middlewares","enableRateLimiting","index","mapToNewConfig","oldOptions","propertiesChain","reduce","prop","setOptions","userOptions","configIndex","findIndex","arg","fileName","loadConfigFile","pairArgumentValue","singleExport","batchExport","batchFunctions","pair","initPool","parseInt","logDest","logFile","enableFileLogging"],"mappings":"6uBAiBAA,EAAOC,SAIA,MAAMC,EAAgB,CAC3BC,UAAW,CACTC,KAAM,CACJC,MAAO,GACPC,KAAM,WACNC,YAAa,6CAGjBC,WAAY,CACVC,QAAS,CACPJ,MAAO,SACPK,QAAS,qBACTJ,KAAM,SACNC,YAAa,8BAEfI,OAAQ,CACNN,MAAO,+BACPK,QAAS,iBACTJ,KAAM,SACNC,YAAa,6CAEfK,YAAa,CACXF,QAAS,0BACTL,MAAO,CAAC,aAAc,kBAAmB,iBACzCC,KAAM,WACNC,YAAa,qCAEfM,QAAS,CACPH,QAAS,qBACTL,MAAO,CACL,QACA,MACA,QACA,YACA,cACA,uBACA,gBACA,uBACA,eACA,QACA,OACA,mBACA,eACA,cACA,UACA,UACA,WACA,UACA,cACA,YACA,sBACA,SACA,SACA,WACA,YACA,eACA,SACA,eACA,YACA,kBACA,SACA,cACA,mBACA,eACA,cACA,eACA,cACA,cACA,WACA,eACA,WACA,SACA,OACA,WACA,YACA,SACA,qBACA,aACA,WACA,WACA,WACA,WACA,eACA,UACA,kBACA,oBACA,cAEFC,KAAM,WACNC,YAAa,gCAEfO,WAAY,CACVJ,QAAS,wBACTL,MAAO,CAAC,kBACRC,KAAM,WACNC,YAAa,mCAEfQ,QAAS,CACPV,MAAO,CACL,wEACA,kGAEFC,KAAM,WACNC,YACE,qEAEJS,WAAY,CACVN,QAAS,yBACTL,OAAO,EACPC,KAAM,UACNC,YACE,oEAGNU,OAAQ,CACNC,OAAQ,CACNb,OAAO,EACPC,KAAM,SACNC,YACE,8FAEJY,MAAO,CACLd,OAAO,EACPC,KAAM,SACNC,YACE,iFAEJa,QAAS,CACPf,OAAO,EACPC,KAAM,SACNC,YAAa,oCAEfc,QAAS,CACPhB,OAAO,EACPC,KAAM,SACNC,YACE,2FAEJD,KAAM,CACJI,QAAS,sBACTL,MAAO,MACPC,KAAM,SACNC,YACE,sEAEJe,OAAQ,CACNZ,QAAS,wBACTL,MAAO,QACPC,KAAM,SACNC,YACE,6EAEJgB,cAAe,CACbb,QAAS,wBACTL,MAAO,IACPC,KAAM,SACNC,YACE,gFAEJiB,aAAc,CAC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"} +"use strict";require("colors");var e=require("fs"),t=require("path"),o=require("body-parser"),r=require("cors"),i=require("express"),n=require("multer"),s=require("http"),a=require("https"),l=require("dotenv"),c=require("express-rate-limit"),p=require("url"),u=require("https-proxy-agent"),d=require("uuid"),h=require("tarn"),g=require("puppeteer"),f=require("node:path"),m=require("node:crypto");require("prompts");var v="undefined"!=typeof document?document.currentScript:null;function y(e){var t=Object.create(null);return e&&Object.keys(e).forEach((function(o){if("default"!==o){var r=Object.getOwnPropertyDescriptor(e,o);Object.defineProperty(t,o,r.get?r:{enumerable:!0,get:function(){return e[o]}})}})),t.default=e,Object.freeze(t)}var b=y(p);l.config();const w={puppeteer:{args:{value:[],type:"string[]",description:"Array of arguments to send to puppeteer."}},highcharts:{version:{value:"latest",envLink:"HIGHCHARTS_VERSION",type:"string",description:"Highcharts version to use."},cdnURL:{value:"https://code.highcharts.com/",envLink:"HIGHCHARTS_CDN",type:"string",description:"The CDN URL of Highcharts scripts to use."},coreScripts:{envLink:"HIGHCHARTS_CORE_SCRIPTS",value:["highcharts","highcharts-more","highcharts-3d"],type:"string[]",description:"Highcharts core scripts to fetch."},modules:{envLink:"HIGHCHARTS_MODULES",value:["stock","map","gantt","exporting","export-data","parallel-coordinates","accessibility","annotations-advanced","boost-canvas","boost","data","draggable-points","static-scale","broken-axis","heatmap","tilemap","timeline","treemap","treegraph","item-series","drilldown","histogram-bellcurve","bullet","funnel","funnel3d","pyramid3d","networkgraph","pareto","pattern-fill","pictorial","price-indicator","sankey","arc-diagram","dependency-wheel","series-label","solid-gauge","sonification","stock-tools","streamgraph","sunburst","variable-pie","variwide","vector","venn","windbarb","wordcloud","xrange","no-data-to-display","drag-panes","debugger","dumbbell","lollipop","cylinder","organization","dotplot","marker-clusters","hollowcandlestick","heikinashi"],type:"string[]",description:"Highcharts modules to fetch."},indicators:{envLink:"HIGHCHARTS_INDICATORS",value:["indicators-all"],type:"string[]",description:"Highcharts indicators to fetch."},scripts:{value:["https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.29.4/moment.min.js","https://cdnjs.cloudflare.com/ajax/libs/moment-timezone/0.5.34/moment-timezone-with-data.min.js"],type:"string[]",description:"Additional direct scripts/optional dependencies (e.g. moment.js)."},forceFetch:{envLink:"HIGHCHARTS_FORCE_FETCH",value:!1,type:"boolean",description:"Should all the scripts be refetched after rerunning the server."}},export:{infile:{value:!1,type:"string",description:"The input file name along with a type (json or svg). It can be a correct JSON or SVG file."},instr:{value:!1,type:"string",description:"An input in a form of a stringified JSON or SVG file. Overrides the --infile."},options:{value:!1,type:"string",description:"An alias for the --instr option."},outfile:{value:!1,type:"string",description:"The output filename along with a type (jpeg, png, pdf or svg). Ignores the --type flag."},type:{envLink:"EXPORT_DEFAULT_TYPE",value:"png",type:"string",description:"The format of the file to export to. Can be jpeg, png, pdf or svg."},constr:{envLink:"EXPORT_DEFAULT_CONSTR",value:"chart",type:"string",description:"The constructor to use. Can be chart, stockChart, mapChart or ganttChart."},defaultHeight:{envLink:"EXPORT_DEFAULT_HEIGHT",value:400,type:"number",description:"The default height of the exported chart. Used when not found any value set."},defaultWidth:{envLink:"EXPORT_DEFAULT_WIDTH",value:600,type:"number",description:"The default width of the exported chart. Used when not found any value set."},defaultScale:{envLink:"EXPORT_DEFAULT_SCALE",value:1,type:"number",description:"The default scale of the exported chart. Ranges between 1 and 5."},height:{type:"number",value:!1,description:"The default height of the exported chart. Overrides the option in the chart settings."},width:{type:"number",value:!1,description:"The width of the exported chart. Overrides the option in the chart settings."},scale:{value:!1,type:"number",description:"The scale of the exported chart. Ranges between 1 and 5."},globalOptions:{value:!1,type:"string",description:"A stringified JSON or a filename with options to be passed into the Highcharts.setOptions."},themeOptions:{value:!1,type:"string",description:"A stringified JSON or a filename with theme options to be passed into the Highcharts.setOptions."},batch:{value:!1,type:"string",description:'Starts a batch job. A string that contains input/output pairs: "in=out;in=out;..".'}},customCode:{allowCodeExecution:{envLink:"HIGHCHARTS_ALLOW_CODE_EXECUTION",value:!1,type:"boolean",description:"If set to true, allow for the execution of arbitrary code when exporting."},allowFileResources:{envLink:"HIGHCHARTS_ALLOW_FILE_RESOURCES",value:!0,type:"boolean",description:"Allow injecting resources from the filesystem. Has no effect when running as a server."},customCode:{value:!1,type:"string",description:"A function to be called before chart initialization. Can be a filename with the js extension."},callback:{value:!1,type:"string",description:"A JavaScript file with a function to run on construction."},resources:{value:!1,type:"string",description:"An additional resource in a form of stringified JSON. It can contain files, js and css sections."},loadConfig:{value:!1,type:"string",description:"A file that contains a pre-defined config to use."},createConfig:{value:!1,type:"string",description:"Allows to set options through a prompt and save in a provided config file."}},server:{enable:{envLink:"HIGHCHARTS_SERVER_ENABLE",value:!1,type:"boolean",cliName:"enableServer",description:"If set to true, starts a server on 0.0.0.0."},host:{envLink:"HIGHCHARTS_SERVER_HOST",value:"0.0.0.0",type:"string",description:"The hostname of the server. Also starts a server listening on the supplied hostname."},port:{envLink:"HIGHCHARTS_SERVER_PORT",value:7801,type:"number",description:"The port to use for the server. Defaults to 7801."},ssl:{enable:{envLink:"HIGHCHARTS_SERVER_SSL_ENABLE",value:!1,type:"boolean",cliName:"enableSsl",description:"Enables the SSL protocol."},force:{envLink:"HIGHCHARTS_SERVER_SSL_FORCE",value:!1,type:"boolean",cliName:"sslForced",description:"If set to true, forces the server to only serve over HTTPS."},port:{envLink:"HIGHCHARTS_SERVER_SSL_PORT",value:443,type:"number",cliName:"sslPort",description:"The port on which to run the SSL server."},certPath:{envLink:"HIGHCHARTS_SSL_CERT_PATH",value:"",type:"string",description:"The path to the SSL certificate/key."}},rateLimiting:{enable:{envLink:"HIGHCHARTS_RATE_LIMIT_ENABLE",value:!1,type:"boolean",cliName:"enableRateLimiting",description:"Enables rate limiting."},maxRequests:{envLink:"HIGHCHARTS_RATE_LIMIT_MAX",value:10,type:"number",description:"Max requests allowed in a one minute."},window:{envLink:"HIGHCHARTS_RATE_LIMIT_WINDOW",value:1,type:"number",description:"The time window in minutes for rate limiting."},delay:{envLink:"HIGHCHARTS_RATE_LIMIT_DELAY",value:0,type:"number",description:"The amount to delay each successive request before hitting the max."},trustProxy:{envLink:"HIGHCHARTS_RATE_LIMIT_TRUST_PROXY",value:!1,type:"boolean",description:"Set this to true if behind a load balancer."},skipKey:{envLink:"HIGHCHARTS_RATE_LIMIT_SKIP_KEY",value:"",type:"number|string",description:"Allows bypassing the rate limiter and should be provided with skipToken argument."},skipToken:{envLink:"HIGHCHARTS_RATE_LIMIT_SKIP_TOKEN",value:"",type:"number|string",description:"Allows bypassing the rate limiter and should be provided with skipKey argument."}}},pool:{initialWorkers:{envLink:"HIGHCHARTS_POOL_MIN_WORKERS",value:4,type:"number",description:"The number of initial workers to spawn."},maxWorkers:{envLink:"HIGHCHARTS_POOL_MAX_WORKERS",value:8,type:"number",description:"The number of max workers to spawn."},workLimit:{envLink:"HIGHCHARTS_POOL_WORK_LIMIT",value:40,type:"number",description:"The pieces of work that can be performed before restarting process."},queueSize:{envLink:"HIGHCHARTS_POOL_QUEUE_SIZE",value:5,type:"number",description:"The size of the request overflow queue."},timeoutThreshold:{envLink:"HIGHCHARTS_POOL_TIMEOUT",value:5e3,type:"number",description:"The number of milliseconds before timing out."},acquireTimeout:{envLink:"HIGHCHARTS_POOL_ACQUIRE_TIMEOUT",value:5e3,type:"number",description:"The number of milliseconds to wait for acquiring a resource."},reaper:{envLink:"HIGHCHARTS_POOL_ENABLE_REAPER",value:!0,type:"boolean",description:"Whether or not to evict workers after a certain time period."},benchmarking:{envLink:"HIGHCHARTS_POOL_BENCHMARKING",value:!1,type:"boolean",description:"Enable benchmarking."},listenToProcessExits:{envLink:"HIGHCHARTS_POOL_LISTEN_TO_PROCESS_EXITS",value:!0,type:"boolean",description:"Set to false in order to skip attaching process.exit handlers."}},logging:{level:{envLink:"HIGHCHARTS_LOG_LEVEL",value:4,type:"number",cliName:"logLevel",description:"The log level (0: silent, 1: error, 2: warning, 3: notice, 4: verbose)."},file:{envLink:"HIGHCHARTS_LOG_FILE",value:"highcharts-export-server.log",type:"string",cliName:"logFile",description:"A name of a log file. The --logDest also needs to be set to enable file logging."},dest:{envLink:"HIGHCHARTS_LOG_DEST",value:"log/",type:"string",cliName:"logDest",description:"The path to store log files. Also enables file logging."}},ui:{enable:{envLink:"HIGHCHARTS_UI_ENABLE",value:!1,type:"boolean",cliName:"enableUi",description:"Enables the UI for the export server."},route:{envLink:"HIGHCHARTS_UI_ROUTE",value:"/",type:"string",cliName:"uiRoute",description:"The route to attach the UI to."}},other:{noLogo:{envLink:"HIGHCHARTS_NO_LOGO",value:!1,type:"boolean",description:"Skip printing the logo on a startup. Will be replaced by a simple text."}},payload:{}};w.puppeteer.args.value.join(","),w.highcharts.version.value,w.highcharts.cdnURL.value,w.highcharts.modules.value,w.highcharts.scripts.value.join(","),w.highcharts.forceFetch.value,w.export.type.value,w.export.constr.value,w.export.defaultHeight.value,w.export.defaultWidth.value,w.export.defaultScale.value,w.customCode.allowCodeExecution.value,w.customCode.allowFileResources.value,w.server.enable.value,w.server.host.value,w.server.port.value,w.server.ssl.enable.value,w.server.ssl.force.value,w.server.ssl.port.value,w.server.ssl.certPath.value,w.server.rateLimiting.enable.value,w.server.rateLimiting.maxRequests.value,w.server.rateLimiting.window.value,w.server.rateLimiting.delay.value,w.server.rateLimiting.trustProxy.value,w.server.rateLimiting.skipKey.value,w.server.rateLimiting.skipToken.value,w.pool.initialWorkers.value,w.pool.maxWorkers.value,w.pool.workLimit.value,w.pool.queueSize.value,w.pool.timeoutThreshold.value,w.pool.acquireTimeout.value,w.pool.reaper.value,w.pool.benchmarking.value,w.pool.listenToProcessExits.value,w.logging.level.value,w.logging.file.value,w.logging.dest.value,w.ui.enable.value,w.ui.route.value,w.other.noLogo.value;const x=["options","globalOptions","themeOptions","resources","payload"],T={},k=(e,t="")=>{Object.keys(e).forEach((o=>{if(!["puppeteer","highcharts"].includes(o)){const r=e[o];void 0===r.value?k(r,`${t}.${o}`):T[r.cliName||o]=`${t}.${o}`.substring(1)}}))};k(w);let S={toConsole:!0,toFile:!1,pathCreated:!1,levelsDesc:[{title:"error",color:"red"},{title:"warning",color:"yellow"},{title:"notice",color:"blue"},{title:"verbose",color:"gray"}],listeners:[]};for(const[e,t]of Object.entries(w.logging))S[e]=t.value;const H=(...t)=>{const[o,...r]=t,{level:i,levelsDesc:n}=S;if(0===o||o>i||i>n.length)return;const s=`${(new Date).toString().split("(")[0].trim()} [${n[o-1].title}] -`;S.listeners.forEach((e=>{e(s,r.join(" "))})),S.toFile&&(S.pathCreated||(!e.existsSync(S.dest)&&e.mkdirSync(S.dest),S.pathCreated=!0),e.appendFile(`${S.dest}${S.file}`,[s].concat(r).join(" ")+"\n",(e=>{e&&(console.log(`[logger] Unable to write to log file: ${e}`),S.toFile=!1)}))),S.toConsole&&console.log.apply(void 0,[s.toString()[S.levelsDesc[o-1].color]].concat(r))},E=p.fileURLToPath(new URL("../.","undefined"==typeof document?require("url").pathToFileURL(__filename).href:v&&v.src||new URL("index.cjs",document.baseURI).href)),R=(e,t=/\s\s+/g,o=" ")=>e.replaceAll(t,o).trim(),L=(e,t)=>{const o=["png","jpeg","pdf","svg"];if(t){const r=t.split(".").pop();o.includes(r)&&e!==r&&(e=r)}return{"image/png":"png","image/jpeg":"jpeg","application/pdf":"pdf","image/svg+xml":"svg"}[e]||o.find((t=>t===e))||"png"},C=(t=!1,o)=>{const r=["js","css","files"];let i=t,n=!1;if(o&&t.endsWith(".json"))try{t?t&&t.endsWith(".json")?i=O(e.readFileSync(t,"utf8")):(i=O(t),!0===i&&(i=O(e.readFileSync("resources.json","utf8")))):i=O(e.readFileSync("resources.json","utf8"))}catch(e){return H(3,"[cli] No resources found.")}else i=O(t),o||delete i.files;for(const e in i)r.includes(e)?n||(n=!0):delete i[e];return n?(i.files&&(i.files=i.files.map((e=>e.trim())),(!i.files||i.files.length<=0)&&delete i.files),i):H(3,"[cli] No resources found.")};function O(e,t){try{const o=JSON.parse("string"!=typeof e?JSON.stringify(e):e);return"string"!=typeof o&&t?JSON.stringify(o):o}catch(e){return!1}}const _=e=>{if(null===e||"object"!=typeof e)return e;const t=Array.isArray(e)?[]:{};for(const o in e)Object.prototype.hasOwnProperty.call(e,o)&&(t[o]=_(e[o]));return t},A=(e,t)=>JSON.stringify(e,((e,o)=>("string"==typeof o&&((o=o.trim()).startsWith("function(")||o.startsWith("function ("))&&o.endsWith("}")&&(o=t?`EXP_FUN${(o+"").replaceAll(/\n|\t|\r/g," ")}EXP_FUN`:void 0),"function"==typeof o?`EXP_FUN${(o+"").replaceAll(/\n|\t|\r/g," ")}EXP_FUN`:o))).replaceAll(/"EXP_FUN|EXP_FUN"/g,"");function $(){console.log("Usage of CLI arguments:".bold,"\n------",`\nFor more detailed information visit readme at: ${"https://github.com/highcharts/node-export-server#readme".bold.yellow}.`);const e=t=>{for(const[o,r]of Object.entries(t))if(Object.prototype.hasOwnProperty.call(r,"value")){let e=` --${r.cliName||o} ${("<"+r.type+">").green} `;if(e.length<48)for(let t=e.length;t<48;t++)e+=".";console.log(e,r.description,`[Default: ${r.value.toString().bold}]`.blue)}else e(r)};Object.keys(w).forEach((t=>{["puppeteer","highcharts"].includes(t)||(console.log(`\n${t.toUpperCase()}`.red),e(w[t]))})),console.log("\n")}const I=e=>!["false","undefined","null","NaN","0",""].includes(e)&&!!e,j=(t,o)=>{if(t&&"string"==typeof t)return(t=t.trim()).endsWith(".js")?!!o&&j(e.readFileSync(t,"utf8")):t.startsWith("function()")||t.startsWith("function ()")||t.startsWith("()=>")||t.startsWith("() =>")?`(${t})()`:t.replace(/;$/,"")};var P=(e,t)=>{const o="Too many requests, you have been rate limited. Please try again later.",r={max:t.maxRequests||30,window:t.window||1,delay:t.delay||0,trustProxy:t.trustProxy||!1,skipKey:t.skipKey||!1,skipToken:t.skipToken||!1};r.trustProxy&&e.enable("trust proxy");const i=c({windowMs:60*r.window*1e3,max:r.max,delayMs:r.delay,handler:(e,t)=>{t.format({json:()=>{t.status(429).send({message:o})},default:()=>{t.status(429).send(o)}})},skip:e=>!1!==r.skipKey&&!1!==r.skipToken&&e.query.key===r.skipKey&&e.query.access_token===r.skipToken&&(H(4,"[rate-limiting] Skipping rate limiter."),!0)});e.use(i),H(3,R(`[rate-limiting] Enabled rate limiting: ${r.max} requests\n per ${r.window} minute per IP, trusting proxy:\n ${r.trustProxy}.`))};async function N(e,t={}){return new Promise(((o,r)=>{const i=(e=>e.startsWith("https")?a:s)(e);i.get(e,t,(e=>{let t="";e.on("data",(e=>{t+=e})),e.on("end",(()=>{t||r("Nothing was fetched from the URL."),e.text=t,o(e)}))})).on("error",(e=>{r(e)}))}))}l.config();const F=t.join(E,".cache"),U={cdnURL:"https://code.highcharts.com/",activeManifest:{},sources:"",hcVersion:""};let q=!1;const G=()=>U.hcVersion=U.sources.substr(0,U.sources.indexOf("*/")).replace("/*","").replace("*/","").replace(/\n/g,"").trim(),W=async(e,t)=>{try{e.endsWith(".js")&&(e=e.substring(0,e.length-3)),H(4,`[cache] Fetching script - ${e}.js`);const o=t?{agent:t,timeout:+process.env.PROXY_SERVER_TIMEOUT||5e3}:{},r=await N(`${e}.js`,o);if(200===r.statusCode)return r.text;throw`${r.statusCode}`}catch(t){throw H(1,`[cache] Error fetching script ${e}.js: ${t}.`),t}},M=async(t,o)=>{const{coreScripts:r,modules:i,indicators:n,scripts:s}=t,a="latest"!==t.version&&t.version?`${t.version}/`:"";H(3,"[cache] Updating cache to Highcharts ",a);const l=[...r.map((e=>`${a}${e}`)),...i.map((e=>"map"===e?`maps/${a}modules/${e}`:`${a}modules/${e}`)),...n.map((e=>`stock/${a}indicators/${e}`))];let c;const p=process.env.PROXY_SERVER_HOST,d=process.env.PROXY_SERVER_PORT;p&&d&&(c=new u({host:p,port:+d}));const h={};try{return U.sources=(await Promise.all([...l.map((async e=>{const o=await W(`${t.cdnURL||U.cdnURL}${e}`,c);return"string"==typeof o&&(h[e.replace(/(.*)\/|(.*)modules\/|stock\/(.*)indicators\/|maps\/(.*)modules\//gi,"")]=1),o})),...s.map((e=>W(e,c)))])).join(";\n"),G(),e.writeFileSync(o,U.sources),h}catch(e){H(1,"[cache] Unable to update local Highcharts cache.")}},D=async o=>{let r;const i=t.join(F,"manifest.json"),n=t.join(F,"sources.js");if(q=o,!e.existsSync(F)&&e.mkdirSync(F),!e.existsSync(i)||o.forceFetch)H(3,"[cache] Fetching and caching Highcharts dependencies."),r=await M(o,n);else{let t=!1;const s=JSON.parse(e.readFileSync(i));if(s.modules&&Array.isArray(s.modules)){const e={};s.modules.forEach((t=>e[t]=1)),s.modules=e}const{modules:a,coreScripts:l,indicators:c}=o,p=a.length+l.length+c.length;s.version!==o.version?(H(3,"[cache] Highcharts version mismatch in cache, need to re-fetch."),t=!0):Object.keys(s.modules||{}).length!==p?(H(3,"[cache] Cache and requested modules does not match, need to re-fetch."),t=!0):t=(o.modules||[]).some((e=>{if(!s.modules[e])return H(3,`[cache] The ${e} missing in cache, need to re-fetch.`),!0})),t?r=await M(o,n):(H(3,"[cache] Dependency cache is up to date, proceeding."),U.sources=e.readFileSync(n,"utf8"),r=s.modules,G())}await(async(o,r)=>{const i={version:o.version,modules:r||{}};U.activeManifest=i,H(4,"[cache] writing new manifest");try{e.writeFileSync(t.join(F,"manifest.json"),JSON.stringify(i),"utf8")}catch(e){H(1,`[cache] Error writing cache manifest: ${e}.`)}})(o,r)};var V=async e=>!!q&&await D(Object.assign(q,{version:e})),z=()=>U,J=()=>U.hcVersion;const K=m.randomBytes(64).toString("base64url"),X=f.join("tmp",`puppeteer-${K}`),B=[`--user-data-dir=${f.join(X,"profile")}`,"--autoplay-policy=user-gesture-required","--disable-background-networking","--disable-background-timer-throttling","--disable-backgrounding-occluded-windows","--disable-breakpad","--disable-client-side-phishing-detection","--disable-component-update","--disable-default-apps","--disable-dev-shm-usage","--disable-domain-reliability","--disable-extensions","--disable-features=AudioServiceOutOfProcess","--disable-hang-monitor","--disable-ipc-flooding-protection","--disable-notifications","--disable-offer-store-unmasked-wallet-cards","--disable-popup-blocking","--disable-print-preview","--disable-prompt-on-repost","--disable-renderer-backgrounding","--disable-session-crashed-bubble","--disable-setuid-sandbox","--disable-speech-api","--disable-sync","--hide-crash-restore-bubble","--hide-scrollbars","--ignore-gpu-blacklist","--metrics-recording-only","--mute-audio","--no-default-browser-check","--no-first-run","--no-pings","--no-sandbox","--no-zygote","--password-store=basic","--use-mock-keychain"],Y=b.fileURLToPath(new URL(".","undefined"==typeof document?require("url").pathToFileURL(__filename).href:v&&v.src||new URL("index.cjs",document.baseURI).href)),Q=e.readFileSync(Y+"/../templates/template.html","utf8");let Z;const ee=async()=>{if(!Z)return!1;const e=await Z.newPage();return await e.setContent(Q),await e.addScriptTag({path:Y+"/../.cache/sources.js"}),await e.evaluate((()=>window.setupHighcharts())),e.on("pageerror",(async t=>{H(1,"[page error]",t),await e.$eval("#container",((e,t)=>{window._displayErrors&&(e.innerHTML=t)}),`

Chart input data error

${t.toString()}`)})),e},te=async()=>{Z.connected&&await Z.close()};const oe=b.fileURLToPath(new URL(".","undefined"==typeof document?require("url").pathToFileURL(__filename).href:v&&v.src||new URL("index.cjs",document.baseURI).href)),re=async(e,t,o)=>await e.evaluate(((e,t)=>window.triggerExport(e,t)),t,o);var ie=async(o,r,i)=>{const n=[],s=async e=>{for(const e of n)await e.dispose();await e.evaluate((()=>{const[,...e]=document.getElementsByTagName("script"),[,...t]=document.getElementsByTagName("style"),[...o]=document.getElementsByTagName("link");for(const r of[...e,...t,...o])r.remove()}))};try{const a=()=>{};H(4,"[export] Determining export path.");const l=i.export;await o.evaluate((()=>requestAnimationFrame((()=>{}))));const c=l?.options?.chart?.displayErrors&&z().activeManifest.modules.debugger;await o.evaluate((e=>window._displayErrors=e),c);const p=()=>{};let u;if(r.indexOf&&(r.indexOf("=0||r.indexOf("=0)){if(H(4,"[export] Treating as SVG."),"svg"===l.type)return r;u=!0;const e=()=>{};await o.setContent((e=>`\n\n\n \n \n Highcarts Export\n \n \n \n
\n ${e}\n
\n \n\n\n`)(r)),e()}else if(H(4,"[export] Treating as config."),l.strInj){const e=()=>{};await re(o,{chart:{height:l.height,width:l.width}},i),e()}else{r.chart.height=l.height,r.chart.width=l.width;const e=()=>{};await re(o,r,i),e()}p();const d=()=>{},h=i.customCode.resources;if(h){if(h.js&&n.push(await o.addScriptTag({content:h.js})),h.files)for(const t of h.files)try{const r=!t.startsWith("http");n.push(await o.addScriptTag(r?{content:e.readFileSync(t,"utf8")}:{url:t}))}catch(e){H(4,"[export] JS file not found.")}const r=()=>{};if(h.css){let e=h.css.match(/@import\s*([^;]*);/g);if(e)for(let r of e)r&&(r=r.replace("url(","").replace("@import","").replace(/"/g,"").replace(/'/g,"").replace(/;/,"").replace(/\)/g,"").trim(),r.startsWith("http")?n.push(await o.addStyleTag({url:r})):i.customCode.allowFileResources&&n.push(await o.addStyleTag({path:t.join(oe,r)})));n.push(await o.addStyleTag({content:h.css.replace(/@import\s*([^;]*);/g,"")||" "}))}r()}d();const g=u?await o.$eval("#chart-container svg:first-of-type",(async(e,t)=>({chartHeight:e.height.baseVal.value*t,chartWidth:e.width.baseVal.value*t})),parseFloat(l.scale)):await o.evaluate((async()=>{const{chartHeight:e,chartWidth:t}=window.Highcharts.charts[0];return{chartHeight:e,chartWidth:t}})),f=()=>{},m=Math.ceil(g?.chartHeight||l.height),v=Math.ceil(g?.chartWidth||l.width);await o.setViewport({height:m,width:v,deviceScaleFactor:u?1:parseFloat(l.scale)});const y=u?e=>{document.body.style.zoom=e,document.body.style.margin="0px"}:()=>{document.body.style.zoom=1};await o.evaluate(y,parseFloat(l.scale));const{height:b,width:w,x:x,y:T}=await(e=>e.$eval("#chart-container",(e=>{const{x:t,y:o,width:r,height:i}=e.getBoundingClientRect();return{x:t,y:o,width:r,height:Math.trunc(i>1?i:500)}})))(o);let k;u||await o.setViewport({width:Math.round(w),height:Math.round(b),deviceScaleFactor:parseFloat(l.scale)}),f();const S=()=>{};if("svg"===l.type)k=await(async e=>await e.$eval("#container svg:first-of-type",(e=>e.outerHTML)))(o);else if("png"===l.type||"jpeg"===l.type)k=await(async(e,t,o,r)=>await Promise.race([e.screenshot({type:t,encoding:o,clip:r,omitBackground:!0}),new Promise(((e,t)=>setTimeout((()=>t(new Error("Rasterization timeout"))),1500)))]))(o,l.type,"base64",{width:v,height:m,x:x,y:T});else{if("pdf"!==l.type)throw`Unsupported output format ${l.type}`;k=await(async(e,t,o,r)=>await e.pdf({height:t+1,width:o,encoding:r}))(o,m,v,"base64")}return await o.evaluate((()=>{const e=Highcharts.charts;if(e.length)for(const t of e)t&&t.destroy(),Highcharts.charts.shift()})),S(),a(),await s(o),k}catch(e){return await s(o),H(1,`[export] Error encountered during export: ${e}`),e}};let ne,se=0,ae=0,le=0,ce=0,pe=0,ue={},de=!1;const he={create:async()=>{const e=d.v4();let t=!1;const o=(new Date).getTime();try{if(t=await ee(),!t||t.isClosed())throw"invalid page";H(3,`[pool] Successfully created a worker ${e} - took ${(new Date).getTime()-o} ms.`)}catch(e){throw H(1,`[pool] Error creating a new page in pool entry creation! ${e}`),"Error creating page"}return{id:e,page:t,workCount:Math.round(Math.random()*(ue.workLimit/2))}},validate:e=>!(ue.workLimit&&++e.workCount>ue.workLimit)||(H(3,"[pool] Worker failed validation:",`exceeded work limit (limit is ${ue.workLimit})`),!1),destroy:e=>{H(3,`[pool] Destroying pool entry ${e.id}.`),e.page&&e.page.close()},log:(e,t)=>console.log(`${t}: ${e}`)},ge=async e=>{ne=e.puppeteerArgs;try{await(async e=>{const t=[...B,...e||[]];if(!Z){let e=0;const o=async()=>{try{H(3,"[browser] attempting to get a browser instance (try",e+")"),Z=await g.launch({headless:"new",args:t,userDataDir:"./tmp/"})}catch(t){H(0,"[browser]",t),++e<25?(H(3,"[browser] failed:",t),await new Promise((e=>setTimeout(e,4e3))),await o()):H(0,"Max retries reached")}};try{await o()}catch(e){return H(0,"[browser] Unable to open browser"),!1}if(!Z)return H(0,"[browser] Unable to open browser"),!1}return Z})(ne)}catch(e){H(0,"[pool|browser]",e)}if(ue=e&&e.pool?{...e.pool}:{},H(3,"[pool] Initializing pool:",`min ${ue.initialWorkers}, max ${ue.maxWorkers}.`),de)return H(4,"[pool] Already initialized, please kill it before creating a new one.");ue.listenToProcessExits&&(H(4,"[pool] Attaching exit listeners to the process."),process.on("exit",(async()=>{await fe()})),process.on("SIGINT",((e,t)=>{H(4,`The ${e} event with code: ${t}.`),process.exit(1)})),process.on("SIGTERM",((e,t)=>{H(4,`The ${e} event with code: ${t}.`),process.exit(1)})),process.on("uncaughtException",(async(e,t)=>{H(4,`The ${t} error, message: ${e.message}.`)})));try{de=new h.Pool({...he,min:ue.initialWorkers,max:ue.maxWorkers,createRetryIntervalMillis:200,createTimeoutMillis:ue.acquireTimeout,acquireTimeoutMillis:ue.acquireTimeout,destroyTimeoutMillis:ue.acquireTimeout,idleTimeoutMillis:ue.timeoutThreshold,reapIntervalMillis:1e3,propagateCreateError:!1}),de.on("createFail",((e,t)=>{H(1,`[pool] Error when creating worker of an event id ${e}:`,t)})),de.on("acquireFail",((e,t)=>{H(1,`[pool] Error when acquiring worker of an event id ${e}:`,t)})),de.on("destroyFail",((e,t,o)=>{H(1,`[pool] Error when destroying worker of an id ${t.id}, event id ${e}:`,o)})),de.on("release",(e=>{H(4,`[pool] Releasing a worker of an id ${e.id}`)})),de.on("destroySuccess",((e,t)=>{H(4,`[pool] Destroyed a worker of an id ${t.id}`)}));const e=[];for(let t=0;t{de.release(e)})),H(3,`[pool] The pool is ready with ${ue.initialWorkers} initial resources waiting.`)}catch(e){throw H(1,`[pool] Couldn't create the worker pool ${e}`),e}};async function fe(){return H(3,"[pool] Killing all workers."),de.destroyed?(await te(),!0):(await de.destroy(),await te(),!0)}const me=async(e,t)=>{let o;const r=e=>{throw++ce,o&&de.release(o),"In pool.postWork: "+e};if(H(4,"[pool] Work received, starting to process."),ue.benchmarking&&ve(),++ae,!de)return H(1,"[pool] Work received, but pool has not been started."),r("Pool is not inited but work was posted to it!");try{H(4,"[pool] Acquiring worker"),o=await de.acquire().promise}catch(e){return r(`[pool] Error when acquiring available entry: ${e}`)}if(H(4,"[pool] Acquired worker handle"),!o.page)return r("Resolved worker page is invalid: pool setup is wonky");try{let i=(new Date).getTime();H(4,`[pool] Starting work on pool entry ${o.id}.`);const n=await ie(o.page,e,t);if(n instanceof Error)return"Rasterization timeout"===n.message&&(o.page.close(),o.page=await ee()),r(n);de.release(o);const s=(new Date).getTime()-i;return le+=s,pe=le/++se,H(4,`[pool] Work completed in ${s} ms.`),{data:n,options:t}}catch(e){r(`Error trying to perform puppeteer export: ${e}.`)}};function ve(){const{min:e,max:t,size:o,available:r,borrowed:i,pending:n,spareResourceCapacity:s}=de;H(4,`[pool] The minimum number of resources allowed by pool: ${e}.`),H(4,`[pool] The maximum number of resources allowed by pool: ${t}.`),H(4,`[pool] The number of all resources in pool (free or in use): ${o}.`),H(4,`[pool] The number of resources that are currently available: ${r}.`),H(4,`[pool] The number of resources that are currently acquired: ${i}.`),H(4,`[pool] The number of callers waiting to acquire a resource: ${n}.`),H(4,`[pool] The number of how many more resources can the pool manage/create: ${s}.`)}var ye=()=>({min:de.min,max:de.max,size:de.size,available:de.available,borrowed:de.borrowed,pending:de.pending,spareResourceCapacity:de.spareResourceCapacity}),be=()=>ae,we=()=>ce,xe=()=>pe,Te=()=>se;const ke=process.env.npm_package_version,Se=new Date;let He={};const Ee=()=>He,Re=(e,t,o=[])=>{const r=_(e);for(const[e,n]of Object.entries(t))r[e]="object"!=typeof(i=n)||Array.isArray(i)||null===i||o.includes(e)||void 0===r[e]?void 0!==n?n:r[e]:Re(r[e],n,o);var i;return r};function Le(e,t={},o=""){Object.keys(e).forEach((r=>{if(!["puppeteer","highcharts"].includes(r)){const i=e[r],n=t&&t[r];let s;void 0===i.value?Le(i,n,`${o}.${r}`):(void 0!==n&&(i.value=n),i.envLink&&("boolean"===i.type?i.value=I([process.env[i.envLink],i.value].find((e=>e||"false"===e))):"number"===i.type?(s=+process.env[i.envLink],i.value=s>=0?s:i.value):i.type.indexOf("]")>=0&&process.env[i.envLink]?i.value=process.env[i.envLink].split(","):i.value=process.env[i.envLink]||i.value))}}))}function Ce(e){let t={};for(const[o,r]of Object.entries(e))t[o]=Object.prototype.hasOwnProperty.call(r,"value")?r.value:Ce(r);return t}let Oe=!1;const _e=async(t,o)=>{H(4,"[chart] Starting exporting process.");const r=((e,t={})=>{let o={};return e.svg?(o=_(t),o.export.type=e.type||e.export.type,o.export.scale=e.scale||e.export.scale,o.export.outfile=e.outfile||e.export.outfile,o.payload={svg:e.svg}):o=Re(t,e,x),o.export.outfile=o.export?.outfile||`chart.${o.export?.type||"png"}`,o})(t,Ee()),i=r.export;return r.payload?.svg&&""!==r.payload.svg?je(r.payload.svg.trim(),r,o):i.infile&&i.infile.length?(H(4,"[chart] Attempting to export from an input file."),e.readFile(i.infile,"utf8",((e,t)=>e?H(1,`[chart] Error loading input file: ${e}.`):(r.export.instr=t,je(r.export.instr.trim(),r,o))))):i.instr&&""!==i.instr||i.options&&""!==i.options?(H(4,"[chart] Attempting to export from a raw input."),I(r.customCode?.allowCodeExecution)?Ie(r,o):"string"==typeof i.instr?je(i.instr.trim(),r,o):$e(r,i.instr||i.options,o)):(H(1,R(`[chart] No input specified.\n ${JSON.stringify(i,void 0," ")}.`)),o&&o(!1,{error:!0,message:"No input specified."}))},Ae=e=>{const{chart:t,exporting:o}=e.export?.options||O(e.export?.instr),r=O(e.export?.globalOptions);let i=e.export?.scale||o?.scale||r?.exporting?.scale||e.export?.defaultScale||1;return i=Math.max(.1,Math.min(i,5)),i=((e,t=1)=>{const o=Math.pow(10,t||0);return Math.round(+e*o)/o})(i,2),{height:e.export?.height||o?.sourceHeight||t?.height||r?.exporting?.sourceHeight||r?.chart?.height||e.export?.defaultHeight||400,width:e.export?.width||o?.sourceWidth||t?.width||r?.exporting?.sourceWidth||r?.chart?.width||e.export?.defaultWidth||600,scale:i}},$e=(t,o,r,i)=>{let{export:n,customCode:s}=t;const a="boolean"==typeof s.allowCodeExecution?s.allowCodeExecution:Oe;if(s){if("string"==typeof t.customCode.resources)t.customCode.resources=C(t.customCode.resources,I(t.customCode.allowFileResources));else if(!t.customCode.resources)try{const o=e.readFileSync("resources.json","utf8");t.customCode.resources=C(o,I(t.customCode.allowFileResources))}catch(e){H(3,"[chart] The default resources.json file not found.")}}else s=t.customCode={};if(!a&&s){if(s.callback||s.resources||s.customCode)return r&&r(!1,{error:!0,message:R("The callback, resources and customCode have been disabled for this\n server.")});s.callback=!1,s.resources=!1,s.customCode=!1}if(o&&(o.chart=o.chart||{},o.exporting=o.exporting||{},o.exporting.enabled=!1),n.constr=n.constr||"chart",n.type=L(n.type,n.outfile),"svg"===n.type&&(n.width=!1),["globalOptions","themeOptions"].forEach((t=>{try{n&&n[t]&&("string"==typeof n[t]&&n[t].endsWith(".json")?n[t]=O(e.readFileSync(n[t],"utf8"),!0):n[t]=O(n[t],!0))}catch(e){n[t]={},H(1,`[chart] The ${t} not found.`)}})),s.allowCodeExecution&&(s.customCode=j(s.customCode,s.allowFileResources)),s&&s.callback&&s.callback?.indexOf("{")<0)if(s.allowFileResources)try{s.callback=e.readFileSync(s.callback,"utf8")}catch(e){H(2,`[chart] Error loading callback: ${e}.`),s.callback=!1}else s.callback=!1;t.export={...t.export,...Ae(t)},me(n.strInj||o||i,t).then((e=>r(e))).catch((e=>(H(0,"[chart] When posting work:",e),r(!1,e))))},Ie=(e,t)=>{try{let o,r=e.export.instr||e.export.options;return"string"!=typeof r&&(o=r=A(r,e.customCode?.allowCodeExecution)),o=r.replaceAll(/\t|\n|\r/g,"").trim(),";"===o[o.length-1]&&(o=o.substring(0,o.length-1)),e.export.strInj=o,$e(e,!1,t)}catch(o){const r=R(`Malformed input detected for ${e.export?.requestId||"?"}:\n Please make sure that your JSON/JavaScript options\n are sent using the "options" attribute, and that if you're using\n SVG, it is unescaped.`);return H(1,r),t&&t(!1,JSON.stringify({error:!0,message:r}))}},je=(e,t,o)=>{const{allowCodeExecution:r}=t.customCode;if(e.indexOf("=0||e.indexOf("=0)return H(4,"[chart] Parsing input as SVG."),$e(t,!1,o,e);try{const r=JSON.parse(e.replaceAll(/\t|\n|\r/g," "));return $e(t,r,o)}catch(e){return I(r)?Ie(t,o):o&&o(!1,{error:!0,message:R("Only JSON configurations and SVG is allowed for this server. If\n this is your server, JavaScript exporting can be enabled by starting\n the server with the --allowCodeExecution flag.")})}},Pe={png:"image/png",jpeg:"image/jpeg",gif:"image/gif",pdf:"application/pdf",svg:"image/svg+xml"};let Ne=0;const Fe=[],Ue=[],qe=(e,t,o,r)=>{let i=!0;const{id:n,uniqueId:s,type:a,body:l}=r;return e.some((e=>{if(e){let r=e(t,o,n,s,a,l);return void 0!==r&&!0!==r&&(i=r),!0}})),i},Ge=(e,t)=>{(()=>{const e=process.hrtime.bigint()})();const o=Ee(),r=e.body,i=++Ne,n=d.v4().replace(/-/g,"");let s=L(r.type);if(!r)return t.status(400).send(R("Body is required. Sending a body? Make sure your Content-type header\n is correct. Accepted is application/json and multipart/form-data."));let a=O(r.infile||r.options||r.data);if(!a&&!r.svg)return H(2,R(`Request ${n} from ${e.headers["x-forwarded-for"]||e.connection.remoteAddress} was incorrect. Check your payload.`)),t.status(400).send(R("No correct chart data found. Please make sure you are using\n application/json or multipart/form-data headers, and that the chart\n data is in the 'infile', 'options' or 'data' attribute if sending\n JSON or in the 'svg' if sending SVG."));let l=!1;if(l=qe(Fe,e,t,{id:i,uniqueId:n,type:s,body:r}),!0!==l)return t.send(l);let c=!1;e.socket.on("close",(()=>{c=!0})),H(4,`[export] Got an incoming HTTP request ${n}.`),r.constr="string"==typeof r.constr&&r.constr||"chart";const p={export:{instr:a,type:s,constr:r.constr[0].toLowerCase()+r.constr.substr(1),height:r.height,width:r.width,scale:r.scale||o.export.scale,globalOptions:O(r.globalOptions,!0),themeOptions:O(r.themeOptions,!0)},customCode:{allowCodeExecution:Oe,allowFileResources:!1,resources:O(r.resources,!0),callback:r.callback,customCode:r.customCode}};a&&(p.export.instr=A(a,p.customCode.allowCodeExecution));const u=Re(o,p);if(u.export.options=a,u.payload={svg:r.svg||!1,b64:r.b64||!1,dataOptions:O(r.dataOptions,!0),noDownload:r.noDownload||!1,requestId:n},r.svg&&(h=u.payload.svg,["localhost","(10).(.*).(.*).(.*)","(127).(.*).(.*).(.*)","(172).(1[6-9]|2[0-9]|3[0-1]).(.*).(.*)","(192).(168).(.*).(.*)"].some((e=>h.match(`xlink:href="(?:(http://|https://))?${e}`)))))return t.status(400).send("SVG potentially contain at least one forbidden URL in xlink:href element.");var h;_e(u,((o,a)=>(e.socket.removeAllListeners("close"),c?H(3,R("[export] The client closed the connection before the chart was done\n processing.")):a?(H(1,R(`[export] Work: ${n} could not be completed, sending:\n ${a}`)),t.status(400).send(a.message)):o&&o.data?(s=o.options.export.type,qe(Ue,e,t,{id:i,body:o.data}),o.data?r.b64?"pdf"===s?t.send(Buffer.from(o.data,"utf8").toString("base64")):t.send(o.data):(t.header("Content-Type",Pe[s]||"image/png"),r.noDownload||t.attachment(`${e.params.filename||"chart"}.${s||"png"}`),"svg"===s?t.send(o.data):t.send(Buffer.from(o.data,"base64"))):void 0):(H(1,R(`[export] Unexpected return from chart generation, please check your\n data Request: ${n} is ${o.data}.`)),t.status(400).send("Unexpected return from chart generation, please check your data.")))))};const We=i();We.disable("x-powered-by"),We.use(r());const Me=n.memoryStorage(),De=n({storage:Me,limits:{fieldsSize:"50MB"}});We.use(De.any()),We.use(o.json({limit:"50mb"})),We.use(o.urlencoded({extended:!0,limit:"50mb"})),We.use(o.urlencoded({extended:!1,limit:"50mb"}));const Ve=e=>H(1,`[server] Socket error: ${e}`),ze=e=>{e.on("clientError",Ve),e.on("error",Ve),e.on("connection",(e=>e.on("error",(e=>Ve(e)))))},Je=async o=>{if(!o.enable)return!1;if(!o.ssl.enable&&!o.ssl.force){const e=s.createServer(We);ze(e),e.listen(o.port,o.host),H(3,`[server] Started HTTP server on ${o.host}:${o.port}.`)}if(o.ssl.enable){let r,i;try{r=await e.promises.readFile(t.posix.join(o.ssl.certPath,"server.key"),"utf8"),i=await e.promises.readFile(t.posix.join(o.ssl.certPath,"server.crt"),"utf8")}catch(e){H(1,`[server] Unable to load key/certificate from ${o.ssl.certPath}.`)}if(r&&i){const e=a.createServer(We);ze(e),e.listen(o.ssl.port,o.host),H(3,`[server] Started HTTPS server on ${o.host}:${o.ssl.port}.`)}}o.rateLimiting&&o.rateLimiting.enable&&![0,NaN].includes(o.rateLimiting.maxRequests)&&P(We,o.rateLimiting),We.use(i.static(t.posix.join(E,"public"))),(e=>{!!e&&e.get("/health",((e,t)=>{t.send({status:"OK",bootTime:Se,uptime:Math.floor(((new Date).getTime()-Se.getTime())/1e3/60)+" minutes",version:ke,highchartsVersion:J(),averageProcessingTime:xe(),performedExports:Te(),failedExports:we(),exportAttempts:be(),sucessRatio:Te()/be()*100,pool:ye()})}))})(We),(e=>{e.post("/",Ge),e.post("/:filename",Ge)})(We),(e=>{!!e&&e.get("/",((e,o)=>{o.sendFile(t.join(E,"public","index.html"))}))})(We),(e=>{!!e&&e.post("/change_hc_version/:newVersion",(async(e,t)=>{const o=process.env.HIGHCHARTS_ADMIN_TOKEN;if(!o||!o.length)return t.send({error:!0,message:"Server not configured to do run-time version changes: HIGHCHARTS_ADMIN_TOKEN not set"});const r=e.get("hc-auth");if(!r||r!==o)return t.send({error:!0,message:"Invalid or missing token: set token in the hc-auth header"});const i=e.params.newVersion;if(i){try{await V(i)}catch(e){t.send({error:!0,message:e})}t.send({version:J()})}else t.send({error:!0,message:"No new version supplied"})}))})(We)};var Ke={startServer:Je,getExpress:()=>i,getApp:()=>We,use:(e,...t)=>{We.use(e,...t)},get:(e,...t)=>{We.get(e,...t)},post:(e,...t)=>{We.post(e,...t)},enableRateLimiting:e=>P(We,e)},Xe={log:H,mapToNewConfig:e=>{const t={};for(const[o,r]of Object.entries(e)){const e=T[o]?T[o].split("."):[];e.reduce(((t,o,i)=>t[o]=e.length-1===i?r:t[o]||{}),t)}return t},setOptions:(t,o)=>(o?.length&&(He=function(t){const o=t.findIndex((e=>"loadConfig"===e.replace(/-/g,"")));if(o>-1&&t[o+1]){const r=t[o+1];try{if(r&&r.endsWith(".json"))return JSON.parse(e.readFileSync(r))}catch(e){H(1,`[config] Unable to load config from the ${r}: ${e}`)}}return{}}(o)),Le(w,He),He=Ce(w),t&&(He=Re(He,t,x)),o?.length&&(He=function(e,t,o){for(let o=0;o(i.length-1===a&&void 0!==n[s]&&(t[++o]?n[s]=t[o]||n[s]:(console.log(`Missing argument value for ${r}!`.red,"\n"),e=$())),n[s])),e)}return e}(He,o)),He),singleExport:t=>{t.export.instr=t.export.instr||t.export.options,_e(t,((t,o)=>{o&&(H(1,`[cli] ${o.message}`),process.exit(1));const{outfile:r,type:i}=t.options.export;e.writeFileSync(r||`chart.${i}`,"svg"!==i?Buffer.from(t.data,"base64"):t.data),fe()}))},startExport:_e,batchExport:t=>{const o=[];for(let r of t.export.batch.split(";"))r=r.split("="),2===r.length&&o.push(new Promise(((o,i)=>{_e({...t,export:{...t.export,infile:r[0],outfile:r[1]}},((t,r)=>{if(r)return i(r);e.writeFileSync(t.options.export.outfile,Buffer.from(t.data,"base64")),o()}))})));Promise.all(o).then((()=>{fe()})).catch((e=>{H(1,`[chart] Error encountered during batch export: ${e}`),fe()}))},server:Ke,startServer:Je,killPool:fe,initPool:async(e={})=>{var t,o;return t=e.customCode&&e.customCode.allowCodeExecution,Oe=I(t),(o=e.logging&&parseInt(e.logging.level))>=0&&o<=S.levelsDesc.length&&(S.level=o),e.logging&&e.logging.dest&&((e,t)=>{if(S={...S,dest:e||S.dest,file:t||S.file,toFile:!0},0===S.dest.length)return H(1,"[logger] File logging init: no path supplied.");S.dest.endsWith("/")||(S.dest+="/")})(e.logging.dest,e.logging.file||"highcharts-export-server.log"),await D(e.highcharts||{version:"latest"}),await ge({pool:e.pool||{initialWorkers:1,maxWorkers:1},puppeteerArgs:e.puppeteer?.args||[]}),e}};module.exports=Xe; +//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"index.cjs","sources":["../lib/schemas/config.js","../lib/logger.js","../lib/utils.js","../lib/server/rate_limit.js","../lib/fetch.js","../lib/cache.js","../lib/browser.js","../lib/export.js","../lib/benchmark.js","../templates/svg_export/svg_export.js","../lib/pool.js","../lib/server/routes/health.js","../lib/config.js","../lib/chart.js","../lib/server/routes/export.js","../lib/server/server.js","../lib/server/routes/ui.js","../lib/server/routes/change_hc_version.js","../lib/index.js"],"sourcesContent":["/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\n// Load .env into environment variables\nimport dotenv from 'dotenv';\n\ndotenv.config();\n\n// This is the configuration object with all options and their default values,\n// also from the .env file if one exists\nexport const defaultConfig = {\n  puppeteer: {\n    args: {\n      value: [],\n      type: 'string[]',\n      description: 'Array of arguments to send to puppeteer.'\n    }\n  },\n  highcharts: {\n    version: {\n      value: 'latest',\n      envLink: 'HIGHCHARTS_VERSION',\n      type: 'string',\n      description: 'Highcharts version to use.'\n    },\n    cdnURL: {\n      value: 'https://code.highcharts.com/',\n      envLink: 'HIGHCHARTS_CDN',\n      type: 'string',\n      description: 'The CDN URL of Highcharts scripts to use.'\n    },\n    coreScripts: {\n      envLink: 'HIGHCHARTS_CORE_SCRIPTS',\n      value: ['highcharts', 'highcharts-more', 'highcharts-3d'],\n      type: 'string[]',\n      description: 'Highcharts core scripts to fetch.'\n    },\n    modules: {\n      envLink: 'HIGHCHARTS_MODULES',\n      value: [\n        'stock',\n        'map',\n        'gantt',\n        'exporting',\n        'export-data',\n        'parallel-coordinates',\n        'accessibility',\n        'annotations-advanced',\n        'boost-canvas',\n        'boost',\n        'data',\n        'draggable-points',\n        'static-scale',\n        'broken-axis',\n        'heatmap',\n        'tilemap',\n        'timeline',\n        'treemap',\n        'treegraph',\n        'item-series',\n        'drilldown',\n        'histogram-bellcurve',\n        'bullet',\n        'funnel',\n        'funnel3d',\n        'pyramid3d',\n        'networkgraph',\n        'pareto',\n        'pattern-fill',\n        'pictorial',\n        'price-indicator',\n        'sankey',\n        'arc-diagram',\n        'dependency-wheel',\n        'series-label',\n        'solid-gauge',\n        'sonification',\n        'stock-tools',\n        'streamgraph',\n        'sunburst',\n        'variable-pie',\n        'variwide',\n        'vector',\n        'venn',\n        'windbarb',\n        'wordcloud',\n        'xrange',\n        'no-data-to-display',\n        'drag-panes',\n        'debugger',\n        'dumbbell',\n        'lollipop',\n        'cylinder',\n        'organization',\n        'dotplot',\n        'marker-clusters',\n        'hollowcandlestick',\n        'heikinashi'\n      ],\n      type: 'string[]',\n      description: 'Highcharts modules to fetch.'\n    },\n    indicators: {\n      envLink: 'HIGHCHARTS_INDICATORS',\n      value: ['indicators-all'],\n      type: 'string[]',\n      description: 'Highcharts indicators to fetch.'\n    },\n    scripts: {\n      value: [\n        'https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.29.4/moment.min.js',\n        'https://cdnjs.cloudflare.com/ajax/libs/moment-timezone/0.5.34/moment-timezone-with-data.min.js'\n      ],\n      type: 'string[]',\n      description:\n        'Additional direct scripts/optional dependencies (e.g. moment.js).'\n    },\n    forceFetch: {\n      envLink: 'HIGHCHARTS_FORCE_FETCH',\n      value: false,\n      type: 'boolean',\n      description:\n        'Should all the scripts be refetched after rerunning the server.'\n    }\n  },\n  export: {\n    infile: {\n      value: false,\n      type: 'string',\n      description:\n        'The input file name along with a type (json or svg). It can be a correct JSON or SVG file.'\n    },\n    instr: {\n      value: false,\n      type: 'string',\n      description:\n        'An input in a form of a stringified JSON or SVG file. Overrides the --infile.'\n    },\n    options: {\n      value: false,\n      type: 'string',\n      description: 'An alias for the --instr option.'\n    },\n    outfile: {\n      value: false,\n      type: 'string',\n      description:\n        'The output filename along with a type (jpeg, png, pdf or svg). Ignores the --type flag.'\n    },\n    type: {\n      envLink: 'EXPORT_DEFAULT_TYPE',\n      value: 'png',\n      type: 'string',\n      description:\n        'The format of the file to export to. Can be jpeg, png, pdf or svg.'\n    },\n    constr: {\n      envLink: 'EXPORT_DEFAULT_CONSTR',\n      value: 'chart',\n      type: 'string',\n      description:\n        'The constructor to use. Can be chart, stockChart, mapChart or ganttChart.'\n    },\n    defaultHeight: {\n      envLink: 'EXPORT_DEFAULT_HEIGHT',\n      value: 400,\n      type: 'number',\n      description:\n        'The default height of the exported chart. Used when not found any value set.'\n    },\n    defaultWidth: {\n      envLink: 'EXPORT_DEFAULT_WIDTH',\n      value: 600,\n      type: 'number',\n      description:\n        'The default width of the exported chart. Used when not found any value set.'\n    },\n    defaultScale: {\n      envLink: 'EXPORT_DEFAULT_SCALE',\n      value: 1,\n      type: 'number',\n      description:\n        'The default scale of the exported chart. Ranges between 1 and 5.'\n    },\n    height: {\n      type: 'number',\n      value: false,\n      description:\n        'The default height of the exported chart. Overrides the option in the chart settings.'\n    },\n    width: {\n      type: 'number',\n      value: false,\n      description:\n        'The width of the exported chart. Overrides the option in the chart settings.'\n    },\n    scale: {\n      value: false,\n      type: 'number',\n      description: 'The scale of the exported chart. Ranges between 1 and 5.'\n    },\n    globalOptions: {\n      value: false,\n      type: 'string',\n      description:\n        'A stringified JSON or a filename with options to be passed into the Highcharts.setOptions.'\n    },\n    themeOptions: {\n      value: false,\n      type: 'string',\n      description:\n        'A stringified JSON or a filename with theme options to be passed into the Highcharts.setOptions.'\n    },\n    batch: {\n      value: false,\n      type: 'string',\n      description:\n        'Starts a batch job. A string that contains input/output pairs: \"in=out;in=out;..\".'\n    }\n  },\n  customCode: {\n    allowCodeExecution: {\n      envLink: 'HIGHCHARTS_ALLOW_CODE_EXECUTION',\n      value: false,\n      type: 'boolean',\n      description:\n        'If set to true, allow for the execution of arbitrary code when exporting.'\n    },\n    allowFileResources: {\n      envLink: 'HIGHCHARTS_ALLOW_FILE_RESOURCES',\n      value: true,\n      type: 'boolean',\n      description:\n        'Allow injecting resources from the filesystem. Has no effect when running as a server.'\n    },\n    customCode: {\n      value: false,\n      type: 'string',\n      description:\n        'A function to be called before chart initialization. Can be a filename with the js extension.'\n    },\n    callback: {\n      value: false,\n      type: 'string',\n      description: 'A JavaScript file with a function to run on construction.'\n    },\n    resources: {\n      value: false,\n      type: 'string',\n      description:\n        'An additional resource in a form of stringified JSON. It can contain files, js and css sections.'\n    },\n    loadConfig: {\n      value: false,\n      type: 'string',\n      description: 'A file that contains a pre-defined config to use.'\n    },\n    createConfig: {\n      value: false,\n      type: 'string',\n      description:\n        'Allows to set options through a prompt and save in a provided config file.'\n    }\n  },\n  server: {\n    enable: {\n      envLink: 'HIGHCHARTS_SERVER_ENABLE',\n      value: false,\n      type: 'boolean',\n      cliName: 'enableServer',\n      description: 'If set to true, starts a server on 0.0.0.0.'\n    },\n    host: {\n      envLink: 'HIGHCHARTS_SERVER_HOST',\n      value: '0.0.0.0',\n      type: 'string',\n      description:\n        'The hostname of the server. Also starts a server listening on the supplied hostname.'\n    },\n    port: {\n      envLink: 'HIGHCHARTS_SERVER_PORT',\n      value: 7801,\n      type: 'number',\n      description: 'The port to use for the server. Defaults to 7801.'\n    },\n    ssl: {\n      enable: {\n        envLink: 'HIGHCHARTS_SERVER_SSL_ENABLE',\n        value: false,\n        type: 'boolean',\n        cliName: 'enableSsl',\n        description: 'Enables the SSL protocol.'\n      },\n      force: {\n        envLink: 'HIGHCHARTS_SERVER_SSL_FORCE',\n        value: false,\n        type: 'boolean',\n        cliName: 'sslForced',\n        description:\n          'If set to true, forces the server to only serve over HTTPS.'\n      },\n      port: {\n        envLink: 'HIGHCHARTS_SERVER_SSL_PORT',\n        value: 443,\n        type: 'number',\n        cliName: 'sslPort',\n        description: 'The port on which to run the SSL server.'\n      },\n      certPath: {\n        envLink: 'HIGHCHARTS_SSL_CERT_PATH',\n        value: '',\n        type: 'string',\n        description: 'The path to the SSL certificate/key.'\n      }\n    },\n    rateLimiting: {\n      enable: {\n        envLink: 'HIGHCHARTS_RATE_LIMIT_ENABLE',\n        value: false,\n        type: 'boolean',\n        cliName: 'enableRateLimiting',\n        description: 'Enables rate limiting.'\n      },\n      maxRequests: {\n        envLink: 'HIGHCHARTS_RATE_LIMIT_MAX',\n        value: 10,\n        type: 'number',\n        description: 'Max requests allowed in a one minute.'\n      },\n      window: {\n        envLink: 'HIGHCHARTS_RATE_LIMIT_WINDOW',\n        value: 1,\n        type: 'number',\n        description: 'The time window in minutes for rate limiting.'\n      },\n      delay: {\n        envLink: 'HIGHCHARTS_RATE_LIMIT_DELAY',\n        value: 0,\n        type: 'number',\n        description:\n          'The amount to delay each successive request before hitting the max.'\n      },\n      trustProxy: {\n        envLink: 'HIGHCHARTS_RATE_LIMIT_TRUST_PROXY',\n        value: false,\n        type: 'boolean',\n        description: 'Set this to true if behind a load balancer.'\n      },\n      skipKey: {\n        envLink: 'HIGHCHARTS_RATE_LIMIT_SKIP_KEY',\n        value: '',\n        type: 'number|string',\n        description:\n          'Allows bypassing the rate limiter and should be provided with skipToken argument.'\n      },\n      skipToken: {\n        envLink: 'HIGHCHARTS_RATE_LIMIT_SKIP_TOKEN',\n        value: '',\n        type: 'number|string',\n        description:\n          'Allows bypassing the rate limiter and should be provided with skipKey argument.'\n      }\n    }\n  },\n  pool: {\n    initialWorkers: {\n      envLink: 'HIGHCHARTS_POOL_MIN_WORKERS',\n      value: 4,\n      type: 'number',\n      description: 'The number of initial workers to spawn.'\n    },\n    maxWorkers: {\n      envLink: 'HIGHCHARTS_POOL_MAX_WORKERS',\n      value: 8,\n      type: 'number',\n      description: 'The number of max workers to spawn.'\n    },\n    workLimit: {\n      envLink: 'HIGHCHARTS_POOL_WORK_LIMIT',\n      value: 40,\n      type: 'number',\n      description:\n        'The pieces of work that can be performed before restarting process.'\n    },\n    queueSize: {\n      envLink: 'HIGHCHARTS_POOL_QUEUE_SIZE',\n      value: 5,\n      type: 'number',\n      description: 'The size of the request overflow queue.'\n    },\n    timeoutThreshold: {\n      envLink: 'HIGHCHARTS_POOL_TIMEOUT',\n      value: 5000,\n      type: 'number',\n      description: 'The number of milliseconds before timing out.'\n    },\n    acquireTimeout: {\n      envLink: 'HIGHCHARTS_POOL_ACQUIRE_TIMEOUT',\n      value: 5000,\n      type: 'number',\n      description:\n        'The number of milliseconds to wait for acquiring a resource.'\n    },\n    reaper: {\n      envLink: 'HIGHCHARTS_POOL_ENABLE_REAPER',\n      value: true,\n      type: 'boolean',\n      description:\n        'Whether or not to evict workers after a certain time period.'\n    },\n    benchmarking: {\n      envLink: 'HIGHCHARTS_POOL_BENCHMARKING',\n      value: false,\n      type: 'boolean',\n      description: 'Enable benchmarking.'\n    },\n    listenToProcessExits: {\n      envLink: 'HIGHCHARTS_POOL_LISTEN_TO_PROCESS_EXITS',\n      value: true,\n      type: 'boolean',\n      description:\n        'Set to false in order to skip attaching process.exit handlers.'\n    }\n  },\n  logging: {\n    level: {\n      envLink: 'HIGHCHARTS_LOG_LEVEL',\n      value: 4,\n      type: 'number',\n      cliName: 'logLevel',\n      description:\n        'The log level (0: silent, 1: error, 2: warning, 3: notice, 4: verbose).'\n    },\n    file: {\n      envLink: 'HIGHCHARTS_LOG_FILE',\n      value: 'highcharts-export-server.log',\n      type: 'string',\n      cliName: 'logFile',\n      description:\n        'A name of a log file. The --logDest also needs to be set to enable file logging.'\n    },\n    dest: {\n      envLink: 'HIGHCHARTS_LOG_DEST',\n      value: 'log/',\n      type: 'string',\n      cliName: 'logDest',\n      description: 'The path to store log files. Also enables file logging.'\n    }\n  },\n  ui: {\n    enable: {\n      envLink: 'HIGHCHARTS_UI_ENABLE',\n      value: false,\n      type: 'boolean',\n      cliName: 'enableUi',\n      description: 'Enables the UI for the export server.'\n    },\n    route: {\n      envLink: 'HIGHCHARTS_UI_ROUTE',\n      value: '/',\n      type: 'string',\n      cliName: 'uiRoute',\n      description: 'The route to attach the UI to.'\n    }\n  },\n  other: {\n    noLogo: {\n      envLink: 'HIGHCHARTS_NO_LOGO',\n      value: false,\n      type: 'boolean',\n      description:\n        'Skip printing the logo on a startup. Will be replaced by a simple text.'\n    }\n  },\n  payload: {}\n};\n\n// The config descriptions object for the prompts functionality. It contains\n// information like:\n// * Type of a prompt\n// * Name of an option\n// * Short description of a chosen option\n// * Initial value\nexport const promptsConfig = {\n  puppeteer: [\n    {\n      type: 'list',\n      name: 'args',\n      message: 'Puppeteer arguments',\n      initial: defaultConfig.puppeteer.args.value.join(','),\n      separator: ','\n    }\n  ],\n  highcharts: [\n    {\n      type: 'text',\n      name: 'version',\n      message: 'Highcharts version',\n      initial: defaultConfig.highcharts.version.value\n    },\n    {\n      type: 'text',\n      name: 'cdnURL',\n      message: 'The url of CDN',\n      initial: defaultConfig.highcharts.cdnURL.value\n    },\n    {\n      type: 'multiselect',\n      name: 'modules',\n      message: 'Available modules',\n      instructions: 'Space: Select specific, A: Select all, Enter: Confirm.',\n      choices: defaultConfig.highcharts.modules.value\n    },\n    {\n      type: 'list',\n      name: 'scripts',\n      message: 'Custom scripts',\n      initial: defaultConfig.highcharts.scripts.value.join(','),\n      separator: ','\n    },\n    {\n      type: 'toggle',\n      name: 'forceFetch',\n      message: 'Should refetch all the scripts after each server rerun',\n      initial: defaultConfig.highcharts.forceFetch.value\n    }\n  ],\n  export: [\n    {\n      type: 'select',\n      name: 'type',\n      message: 'The default type of a file to export to',\n      hint: `Default: ${defaultConfig.export.type.value}`,\n      initial: 0,\n      choices: ['png', 'jpeg', 'pdf', 'svg']\n    },\n    {\n      type: 'select',\n      name: 'constr',\n      message: 'The default constructor for Highcharts to use',\n      hint: `Default: ${defaultConfig.export.constr.value}`,\n      initial: 0,\n      choices: ['chart', 'stockChart', 'mapChart', 'ganttChart']\n    },\n    {\n      type: 'number',\n      name: 'defaultHeight',\n      message: 'The default fallback height of the exported chart',\n      initial: defaultConfig.export.defaultHeight.value\n    },\n    {\n      type: 'number',\n      name: 'defaultWidth',\n      message: 'The default fallback width of the exported chart',\n      initial: defaultConfig.export.defaultWidth.value\n    },\n    {\n      type: 'number',\n      name: 'defaultScale',\n      message: 'The default fallback scale of the exported chart',\n      initial: defaultConfig.export.defaultScale.value,\n      min: 0.1,\n      max: 5\n    }\n  ],\n  customCode: [\n    {\n      type: 'toggle',\n      name: 'allowCodeExecution',\n      message: 'Allow to execute custom code',\n      initial: defaultConfig.customCode.allowCodeExecution.value\n    },\n    {\n      type: 'toggle',\n      name: 'allowFileResources',\n      message: 'Allow file resources',\n      initial: defaultConfig.customCode.allowFileResources.value\n    }\n  ],\n  server: [\n    {\n      type: 'toggle',\n      name: 'enable',\n      message: 'Starts a server on 0.0.0.0',\n      initial: defaultConfig.server.enable.value\n    },\n    {\n      type: 'text',\n      name: 'host',\n      message: 'A hostname of a server',\n      initial: defaultConfig.server.host.value\n    },\n    {\n      type: 'number',\n      name: 'port',\n      message: 'A port of a server',\n      initial: defaultConfig.server.port.value\n    },\n    {\n      type: 'toggle',\n      name: 'ssl.enable',\n      message: 'Enable SSL protocol',\n      initial: defaultConfig.server.ssl.enable.value\n    },\n    {\n      type: 'toggle',\n      name: 'ssl.force',\n      message: 'Force to only serve over HTTPS',\n      initial: defaultConfig.server.ssl.force.value\n    },\n    {\n      type: 'number',\n      name: 'ssl.port',\n      message: 'Port on which to run the SSL server',\n      initial: defaultConfig.server.ssl.port.value\n    },\n    {\n      type: 'text',\n      name: 'ssl.certPath',\n      message: 'A path where to find the SSL certificate/key',\n      initial: defaultConfig.server.ssl.certPath.value\n    },\n    {\n      type: 'toggle',\n      name: 'rateLimiting.enable',\n      message: 'Enable rate limiting',\n      initial: defaultConfig.server.rateLimiting.enable.value\n    },\n    {\n      type: 'number',\n      name: 'rateLimiting.maxRequests',\n      message: 'Max requests allowed in a one minute',\n      initial: defaultConfig.server.rateLimiting.maxRequests.value\n    },\n    {\n      type: 'number',\n      name: 'rateLimiting.window',\n      message: 'The time window in minutes for rate limiting',\n      initial: defaultConfig.server.rateLimiting.window.value\n    },\n    {\n      type: 'number',\n      name: 'rateLimiting.delay',\n      message:\n        'The amount to delay each successive request before hitting the max',\n      initial: defaultConfig.server.rateLimiting.delay.value\n    },\n    {\n      type: 'toggle',\n      name: 'rateLimiting.trustProxy',\n      message: 'Set this to true if behind a load balancer',\n      initial: defaultConfig.server.rateLimiting.trustProxy.value\n    },\n    {\n      type: 'text',\n      name: 'rateLimiting.skipKey',\n      message:\n        'Allows bypassing the rate limiter and should be provided with skipToken argument',\n      initial: defaultConfig.server.rateLimiting.skipKey.value\n    },\n    {\n      type: 'text',\n      name: 'rateLimiting.skipToken',\n      message:\n        'Allows bypassing the rate limiter and should be provided with skipKey argument',\n      initial: defaultConfig.server.rateLimiting.skipToken.value\n    }\n  ],\n  pool: [\n    {\n      type: 'number',\n      name: 'initialWorkers',\n      message: 'The number of initial workers to spawn',\n      initial: defaultConfig.pool.initialWorkers.value\n    },\n    {\n      type: 'number',\n      name: 'maxWorkers',\n      message: 'The number of max workers to spawn',\n      initial: defaultConfig.pool.maxWorkers.value\n    },\n    {\n      type: 'number',\n      name: 'workLimit',\n      message:\n        'The pieces of work that can be performed before restarting a puppeteer process',\n      initial: defaultConfig.pool.workLimit.value\n    },\n    {\n      type: 'number',\n      name: 'queueSize',\n      message: 'The size of the request overflow queue',\n      initial: defaultConfig.pool.queueSize.value\n    },\n    {\n      type: 'number',\n      name: 'timeoutThreshold',\n      message: 'The number of seconds before timing out',\n      initial: defaultConfig.pool.timeoutThreshold.value\n    },\n    {\n      type: 'number',\n      name: 'acquireTimeout',\n      message: 'The number of milliseconds to wait for acquiring a resource',\n      initial: defaultConfig.pool.acquireTimeout.value\n    },\n    {\n      type: 'toggle',\n      name: 'reaper',\n      message: 'The reaper to remove hanging processes',\n      initial: defaultConfig.pool.reaper.value\n    },\n    {\n      type: 'toggle',\n      name: 'benchmarking',\n      message: 'Set benchmarking',\n      initial: defaultConfig.pool.benchmarking.value\n    },\n    {\n      type: 'toggle',\n      name: 'listenToProcessExits',\n      message: 'Set to false in order to skip attaching process.exit handlers',\n      initial: defaultConfig.pool.listenToProcessExits.value\n    }\n  ],\n  logging: [\n    {\n      type: 'number',\n      name: 'level',\n      message:\n        'The log level (0: silent, 1: error, 2: warning, 3: notice, 4: verbose)',\n      initial: defaultConfig.logging.level.value,\n      round: 0,\n      min: 0,\n      max: 4\n    },\n    {\n      type: 'text',\n      name: 'file',\n      message:\n        'A name of a log file. The --logDest also needs to be set to enable file logging',\n      initial: defaultConfig.logging.file.value\n    },\n    {\n      type: 'text',\n      name: 'dest',\n      message: 'A path to log files. It enables file logging',\n      initial: defaultConfig.logging.dest.value\n    }\n  ],\n  ui: [\n    {\n      type: 'toggle',\n      name: 'enable',\n      message: 'Enable UI for the export server',\n      initial: defaultConfig.ui.enable.value\n    },\n    {\n      type: 'text',\n      name: 'route',\n      message: 'A route to attach the UI to',\n      initial: defaultConfig.ui.route.value\n    }\n  ],\n  other: [\n    {\n      type: 'toggle',\n      name: 'noLogo',\n      message:\n        'Skip printing the logo on a startup. Will be replaced by a simple text',\n      initial: defaultConfig.other.noLogo.value\n    }\n  ]\n};\n\n// Absolute props that, in case of merging recursively, need to be force merged\nexport const absoluteProps = [\n  'options',\n  'globalOptions',\n  'themeOptions',\n  'resources',\n  'payload'\n];\n\n// Argument nesting level of all export server options\nexport const nestedArgs = {};\n\n/**\n * Creates nested arguments chain for all options\n *\n * @param {object} obj - The object based on which the initial configuration be\n * made.\n * @param {string } propChain - Required for creating a string chain of\n * properties for nested arguments.\n */\nconst createNestedArgs = (obj, propChain = '') => {\n  Object.keys(obj).forEach((k) => {\n    if (!['puppeteer', 'highcharts'].includes(k)) {\n      const entry = obj[k];\n      if (typeof entry.value === 'undefined') {\n        // Go deeper in the nested arguments\n        createNestedArgs(entry, `${propChain}.${k}`);\n      } else {\n        // Create the chain of nested arguments\n        nestedArgs[entry.cliName || k] = `${propChain}.${k}`.substring(1);\n      }\n    }\n  });\n};\n\ncreateNestedArgs(defaultConfig);\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { appendFile, existsSync, mkdirSync } from 'fs';\n\nimport { defaultConfig } from './schemas/config.js';\n\n// The default logging config\nlet logging = {\n  // Flags for logging status\n  toConsole: true,\n  toFile: false,\n  pathCreated: false,\n  // Log levels\n  levelsDesc: [\n    {\n      title: 'error',\n      color: 'red'\n    },\n    {\n      title: 'warning',\n      color: 'yellow'\n    },\n    {\n      title: 'notice',\n      color: 'blue'\n    },\n    {\n      title: 'verbose',\n      color: 'gray'\n    }\n  ],\n  // Log listeners\n  listeners: []\n};\n\n// Gather init logging options\nfor (const [key, option] of Object.entries(defaultConfig.logging)) {\n  logging[key] = option.value;\n}\n\n/**\n * Logs a message. Accepts a variable amount of arguments. Arguments after\n * `level` will be passed directly to console.log, and/or will be joined\n * and appended to the log file.\n *\n * @param {any} args - An array of arguments where the first is the log level\n * and the rest are strings to build a message with.\n */\nexport const log = (...args) => {\n  const [newLevel, ...texts] = args;\n\n  // Current logging options\n  const { level, levelsDesc } = logging;\n\n  // Check if log level is within a correct range\n  if (newLevel === 0 || newLevel > level || level > levelsDesc.length) {\n    return;\n  }\n\n  // Get rid of the GMT text information\n  const newDate = new Date().toString().split('(')[0].trim();\n\n  // Create a message's prefix\n  const prefix = `${newDate} [${levelsDesc[newLevel - 1].title}] -`;\n\n  // Call available log listeners\n  logging.listeners.forEach((fn) => {\n    fn(prefix, texts.join(' '));\n  });\n\n  // Log to file\n  if (logging.toFile) {\n    if (!logging.pathCreated) {\n      // Create if does not exist\n      !existsSync(logging.dest) && mkdirSync(logging.dest);\n\n      // We now assume the path is available, e.g. it's the responsibility\n      // of the user to create the path with the correct access rights.\n      logging.pathCreated = true;\n    }\n\n    // Add the content to a file\n    appendFile(\n      `${logging.dest}${logging.file}`,\n      [prefix].concat(texts).join(' ') + '\\n',\n      (error) => {\n        if (error) {\n          console.log(`[logger] Unable to write to log file: ${error}`);\n          logging.toFile = false;\n        }\n      }\n    );\n  }\n\n  // Log to console\n  if (logging.toConsole) {\n    console.log.apply(\n      undefined,\n      [prefix.toString()[logging.levelsDesc[newLevel - 1].color]].concat(texts)\n    );\n  }\n};\n\n/**\n * Sets the file logging configuration.\n *\n * @param {string} logDest - A path to log to.\n * @param {string} logFile - The name of the log file.\n */\nexport const enableFileLogging = (logDest, logFile) => {\n  // Update logging options\n  logging = {\n    ...logging,\n    dest: logDest || logging.dest,\n    file: logFile || logging.file,\n    toFile: true\n  };\n\n  if (logging.dest.length === 0) {\n    return log(1, '[logger] File logging init: no path supplied.');\n  }\n\n  if (!logging.dest.endsWith('/')) {\n    logging.dest += '/';\n  }\n};\n\n/**\n * Adds a log listener.\n *\n * @param {function} fn - The function to call when getting a log event.\n */\nexport const listen = (fn) => {\n  logging.listeners.push(fn);\n};\n\n/**\n * Sets the current log level. Log levels are:\n * - 0 = no logging\n * - 1 = error\n * - 2 = warning\n * - 3 = notice\n * - 4 = verbose\n *\n * @param {number} newLevel - The new log level (0 - 4).\n */\nexport const setLogLevel = (newLevel) => {\n  if (newLevel >= 0 && newLevel <= logging.levelsDesc.length) {\n    logging.level = newLevel;\n  }\n};\n\n/**\n * Enables or disables logging to the stdout.\n *\n * @param {boolean} enabled - Whether log to console or not.\n */\nexport const toggleSTDOut = (enabled) => {\n  logging.toConsole = enabled;\n};\n\nexport default {\n  log,\n  enableFileLogging,\n  listen,\n  setLogLevel,\n  toggleSTDOut\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { readFileSync } from 'fs';\nimport { fileURLToPath } from 'url';\n\nimport { defaultConfig } from '../lib/schemas/config.js';\nimport { log } from './logger.js';\n\nconst MAX_BACKOFF_ATTEMPTS = 6;\n\nexport const __dirname = fileURLToPath(new URL('../.', import.meta.url));\n\n/**\n * Clears text from whitespaces with a regex rule.\n *\n * @param {string} rule - The rule for clearing a string, default to /\\s\\s+/g.\n * @return {string} - Cleared text.\n */\nexport const clearText = (text, rule = /\\s\\s+/g, replacer = ' ') =>\n  text.replaceAll(rule, replacer).trim();\n\n/**\n * Delays calling the function by time calculated based on the backoff\n * algorithm.\n *\n * @param {function} fn - A function to try to call with the backoff algorithm\n * on.\n * @param {number} attempt - The number of an attempt, where the first one is 0.\n */\nexport const expBackoff = async (fn, attempt = 0, ...args) => {\n  try {\n    // Try to call the function\n    return await fn(...args);\n  } catch (error) {\n    // Calculate delay in ms\n    const delayInMs = 2 ** attempt * 1000;\n\n    // If the attempt exceeds the maximum attempts of reapeat, throw an error\n    if (++attempt >= MAX_BACKOFF_ATTEMPTS) {\n      throw error;\n    }\n\n    // Wait given amount of time\n    await new Promise((response) => setTimeout(response, delayInMs));\n    log(\n      3,\n      `[pool] Waited ${delayInMs}ms until next call for the resource id: ${args[0]}.`\n    );\n\n    // Try again\n    return expBackoff(fn, attempt, ...args);\n  }\n};\n\n/**\n * Fixes to supported type format if MIME.\n *\n * @param {string} type - Type to be corrected.\n * @param {string} outfile - Name of the outfile.\n */\nexport const fixType = (type, outfile) => {\n  // MIME types\n  const mimeTypes = {\n    'image/png': 'png',\n    'image/jpeg': 'jpeg',\n    'application/pdf': 'pdf',\n    'image/svg+xml': 'svg'\n  };\n\n  // Formats\n  const formats = ['png', 'jpeg', 'pdf', 'svg'];\n\n  // Check if type and outfile's extensions are the same\n  if (outfile) {\n    const outType = outfile.split('.').pop();\n\n    // Check if extension has a correct type\n    if (formats.includes(outType) && type !== outType) {\n      type = outType;\n    }\n  }\n\n  // Return a correct type\n  return mimeTypes[type] || formats.find((t) => t === type) || 'png';\n};\n\n/**\n * Handles the provided resources.\n *\n * @param {string} resources - The stringified resources.\n * @param {string} allowFileResources - Decide if resources from file are\n * allowed.\n */\nexport const handleResources = (resources = false, allowFileResources) => {\n  const allowedProps = ['js', 'css', 'files'];\n\n  let handledResources = resources;\n  let correctResources = false;\n\n  // Try to load resources from a file\n  if (allowFileResources && resources.endsWith('.json')) {\n    try {\n      if (!resources) {\n        handledResources = isCorrectJSON(\n          readFileSync('resources.json', 'utf8')\n        );\n      } else if (resources && resources.endsWith('.json')) {\n        handledResources = isCorrectJSON(readFileSync(resources, 'utf8'));\n      } else {\n        handledResources = isCorrectJSON(resources);\n        if (handledResources === true) {\n          handledResources = isCorrectJSON(\n            readFileSync('resources.json', 'utf8')\n          );\n        }\n      }\n    } catch (notice) {\n      return log(3, `[cli] No resources found.`);\n    }\n  } else {\n    // Try to get JSON\n    handledResources = isCorrectJSON(resources);\n\n    // Get rid of the files section\n    if (!allowFileResources) {\n      delete handledResources.files;\n    }\n  }\n\n  // Filter from unnecessary properties\n  for (const propName in handledResources) {\n    if (!allowedProps.includes(propName)) {\n      delete handledResources[propName];\n    } else if (!correctResources) {\n      correctResources = true;\n    }\n  }\n\n  // Check if at least one of allowed properties is present\n  if (!correctResources) {\n    return log(3, `[cli] No resources found.`);\n  }\n\n  // Handle files section\n  if (handledResources.files) {\n    handledResources.files = handledResources.files.map((item) => item.trim());\n    if (!handledResources.files || handledResources.files.length <= 0) {\n      delete handledResources.files;\n    }\n  }\n\n  // Return resources\n  return handledResources;\n};\n\n/**\n * Checks if provided data is or can be a correct JSON.\n *\n * @param {any} data - Data to be checked.\n * @param {boolean} toString - If true, return stringified representation.\n */\nexport function isCorrectJSON(data, toString) {\n  try {\n    // Get the string representation if not already before parsing\n    const parsedData = JSON.parse(\n      typeof data !== 'string' ? JSON.stringify(data) : data\n    );\n\n    // Return a stringified representation of a JSON if required\n    if (typeof parsedData !== 'string' && toString) {\n      return JSON.stringify(parsedData);\n    }\n\n    // Return a JSON\n    return parsedData;\n  } catch (error) {\n    return false;\n  }\n}\n\n/**\n * Checks if item is an object.\n *\n * @param {any} item - Item to be checked.\n */\nexport const isObject = (item) =>\n  typeof item === 'object' && !Array.isArray(item) && item !== null;\n\n/**\n * Checks if string contains private range urls.\n *\n * @export utils\n * @param item {string} item to be checked\n */\nexport const isPrivateRangeUrlFound = (item) => {\n  return [\n    'localhost',\n    '(10).(.*).(.*).(.*)',\n    '(127).(.*).(.*).(.*)',\n    '(172).(1[6-9]|2[0-9]|3[0-1]).(.*).(.*)',\n    '(192).(168).(.*).(.*)'\n  ].some((ipRegEx) =>\n    item.match(`xlink:href=\"(?:(http://|https://))?${ipRegEx}`)\n  );\n};\n\n/**\n * Creates and returns a deep copy of the given object.\n *\n * @param {object} object - Object to copy.\n * @return {object} - Deep copy of the object.\n */\nexport const deepCopy = (obj) => {\n  if (obj === null || typeof obj !== 'object') {\n    return obj;\n  }\n\n  const copy = Array.isArray(obj) ? [] : {};\n\n  for (const key in obj) {\n    if (Object.prototype.hasOwnProperty.call(obj, key)) {\n      copy[key] = deepCopy(obj[key]);\n    }\n  }\n\n  return copy;\n};\n\n/**\n * Stringifies object with options. Possible to preserve functions with\n * allowFunctions flag.\n *\n * @param {object} options - Options to stringify.\n * @param {boolean} allowFunctions - Flag for keeping functions.\n */\nexport const optionsStringify = (options, allowFunctions) => {\n  const replacerCallback = (name, value) => {\n    if (typeof value === 'string') {\n      value = value.trim();\n\n      // If allowFunctions is set to true, preserve functions\n      if (\n        (value.startsWith('function(') || value.startsWith('function (')) &&\n        value.endsWith('}')\n      ) {\n        value = allowFunctions\n          ? `EXP_FUN${(value + '').replaceAll(/\\n|\\t|\\r/g, ' ')}EXP_FUN`\n          : undefined;\n      }\n    }\n\n    return typeof value === 'function'\n      ? `EXP_FUN${(value + '').replaceAll(/\\n|\\t|\\r/g, ' ')}EXP_FUN`\n      : value;\n  };\n\n  // Stringify options and if required, replace special functions marks\n  return JSON.stringify(options, replacerCallback).replaceAll(\n    /\"EXP_FUN|EXP_FUN\"/g,\n    ''\n  );\n};\n\n/**\n * Prints the export server logo.\n *\n * @param {boolean} noLogo - Whether to display logo or text.\n */\nexport const printLogo = (noLogo) => {\n  // Get package version either from env or from package.json\n  const packageVersion =\n    process.env.npm_package_version ||\n    JSON.parse(readFileSync(new URL('../package.json', import.meta.url)))\n      .version;\n\n  // Print text only\n  if (noLogo) {\n    console.log(`Starting highcharts export server v${packageVersion}...`);\n    return;\n  }\n\n  // Print the logo\n  console.log(\n    readFileSync(__dirname + '/msg/startup.msg').toString().bold.yellow,\n    `v${packageVersion}`\n  );\n};\n\n/**\n * Prints the CLI usage. If required, it can list properties recursively\n */\nexport function printUsage() {\n  const pad = 48;\n  const readme = 'https://github.com/highcharts/node-export-server#readme';\n\n  // Display readme information\n  console.log(\n    'Usage of CLI arguments:'.bold,\n    '\\n------',\n    `\\nFor more detailed information visit readme at: ${readme.bold.yellow}.`\n  );\n\n  const cycleCategories = (categories) => {\n    for (const [name, option] of Object.entries(categories)) {\n      // If category has more levels, go further\n      if (!Object.prototype.hasOwnProperty.call(option, 'value')) {\n        cycleCategories(option);\n      } else {\n        let descName = `  --${option.cliName || name} ${\n          ('<' + option.type + '>').green\n        } `;\n        if (descName.length < pad) {\n          for (let i = descName.length; i < pad; i++) {\n            descName += '.';\n          }\n        }\n\n        // Display correctly aligned messages\n        console.log(\n          descName,\n          option.description,\n          `[Default: ${option.value.toString().bold}]`.blue\n        );\n      }\n    }\n  };\n\n  // Cycle through options of each categories and display the usage info\n  Object.keys(defaultConfig).forEach((category) => {\n    // Only puppeteer and highcharts categories cannot be configured through CLI\n    if (!['puppeteer', 'highcharts'].includes(category)) {\n      console.log(`\\n${category.toUpperCase()}`.red);\n      cycleCategories(defaultConfig[category]);\n    }\n  });\n  console.log('\\n');\n}\n\n/**\n * Rounds number to passed precision.\n *\n * @param {number} value - Number to round.\n * @param {number} precision - A precision of rounding.\n */\nexport const roundNumber = (value, precision = 1) => {\n  const multiplier = Math.pow(10, precision || 0);\n  return Math.round(+value * multiplier) / multiplier;\n};\n\n/**\n * Casts the item to boolean.\n *\n * @param {any} item - Item to be cast.\n */\nexport const toBoolean = (item) =>\n  ['false', 'undefined', 'null', 'NaN', '0', ''].includes(item)\n    ? false\n    : !!item;\n\n/**\n * If necessary, places a custom code inside a function.\n *\n * @param {any} customCode - The customCode.\n */\nexport const wrapAround = (customCode, allowFileResources) => {\n  if (customCode && typeof customCode === 'string') {\n    customCode = customCode.trim();\n\n    if (customCode.endsWith('.js')) {\n      return allowFileResources\n        ? wrapAround(readFileSync(customCode, 'utf8'))\n        : false;\n    } else if (\n      customCode.startsWith('function()') ||\n      customCode.startsWith('function ()') ||\n      customCode.startsWith('()=>') ||\n      customCode.startsWith('() =>')\n    ) {\n      return `(${customCode})()`;\n    }\n    return customCode.replace(/;$/, '');\n  }\n};\n\n/**\n * Utility to measure time.\n */\nexport const measureTime = () => {\n  const start = process.hrtime.bigint();\n  return () => Number(process.hrtime.bigint() - start) / 1000000;\n};\n\nexport default {\n  __dirname,\n  clearText,\n  expBackoff,\n  fixType,\n  handleResources,\n  isCorrectJSON,\n  isObject,\n  isPrivateRangeUrlFound,\n  optionsStringify,\n  printLogo,\n  printUsage,\n  roundNumber,\n  toBoolean,\n  wrapAround,\n  measureTime\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport rateLimit from 'express-rate-limit';\n\nimport { clearText } from '../utils.js';\nimport { log } from '../logger.js';\n\n/**\n * Enables rate limiting for a given app.\n *\n * @param {object} app - The express app.\n * @param {object} limitConfig - The options for the rate limiting.\n */\nexport default (app, limitConfig) => {\n  const msg =\n    'Too many requests, you have been rate limited. Please try again later.';\n\n  // Options for the rate limiter\n  const rateOptions = {\n    max: limitConfig.maxRequests || 30,\n    window: limitConfig.window || 1,\n    delay: limitConfig.delay || 0,\n    trustProxy: limitConfig.trustProxy || false,\n    skipKey: limitConfig.skipKey || false,\n    skipToken: limitConfig.skipToken || false\n  };\n\n  // Set if behind a proxy\n  if (rateOptions.trustProxy) {\n    app.enable('trust proxy');\n  }\n\n  // Create a limiter\n  const limiter = rateLimit({\n    windowMs: rateOptions.window * 60 * 1000,\n    // Limit each IP to 100 requests per windowMs\n    max: rateOptions.max,\n    // Disable delaying, full speed until the max limit is reached\n    delayMs: rateOptions.delay,\n    handler: (request, response) => {\n      response.format({\n        json: () => {\n          response.status(429).send({ message: msg });\n        },\n        default: () => {\n          response.status(429).send(msg);\n        }\n      });\n    },\n    skip: (request) => {\n      // Allow bypassing the limiter if a valid key/token has been sent\n      if (\n        rateOptions.skipKey !== false &&\n        rateOptions.skipToken !== false &&\n        request.query.key === rateOptions.skipKey &&\n        request.query.access_token === rateOptions.skipToken\n      ) {\n        log(4, '[rate-limiting] Skipping rate limiter.');\n        return true;\n      }\n      return false;\n    }\n  });\n\n  // Use a limiter as a middleware\n  app.use(limiter);\n\n  log(\n    3,\n    clearText(\n      `[rate-limiting] Enabled rate limiting: ${rateOptions.max} requests\n      per ${rateOptions.window} minute per IP, trusting proxy:\n      ${rateOptions.trustProxy}.`\n    )\n  );\n};\n","/**\n * This module exports two functions: fetch (for GET requests) and post (for POST requests).\n */\n\nimport http from 'http';\nimport https from 'https';\n\n/**\n * Determines the protocol of the given URL (either `http` or `https`).\n *\n * @function\n * @param {string} url - The URL whose protocol needs to be determined.\n * @returns {Object} Returns the `https` module if the URL starts with 'https',\n * otherwise returns the `http` module.\n * @private\n *\n * @example\n *\n * const protocol = getProtocol('https://example.com');\n * console.log(protocol); // Outputs the 'https' module\n */\nconst getProtocol = (url) => {\n  return url.startsWith('https') ? https : http;\n};\n\n/**\n * Sends a GET request to the specified URL with optional request options.\n *\n * @function\n * @async\n * @param {string} url - The URL to fetch.\n * @param {Object} [requestOptions={}] - Optional request options and headers.\n * @returns {Promise<Object>} Returns a promise that resolves with the response object.\n * The response object contains a `.text` property with the raw response data.\n * @throws {Error} Throws an error if the request fails or if no data is fetched from the URL.\n *\n * @example\n *\n * async function getData() {\n *   try {\n *     const response = await fetch('https://api.example.com/data');\n *     console.log(response.text);\n *   } catch (error) {\n *     console.error('Error fetching data:', error);\n *   }\n * }\n *\n * getData();\n */\nasync function fetch(url, requestOptions = {}) {\n  return new Promise((resolve, reject) => {\n    const protocol = getProtocol(url);\n\n    protocol\n      .get(url, requestOptions, (res) => {\n        let data = '';\n\n        // A chunk of data has been received.\n        res.on('data', (chunk) => {\n          data += chunk;\n        });\n\n        // The whole response has been received.\n        res.on('end', () => {\n          if (!data) {\n            reject('Nothing was fetched from the URL.');\n          }\n\n          res.text = data;\n          resolve(res);\n        });\n      })\n      .on('error', (error) => {\n        reject(error);\n      });\n  });\n}\n\n/**\n * Sends a POST request to the specified URL with the given body and request options.\n *\n * @function\n * @async\n * @param {string} url - The URL to which the request should be sent.\n * @param {Object} [body={}] - The data to be sent as the request body, in JSON format.\n * @param {Object} [requestOptions={}] - Optional request options and headers.\n * @returns {Promise<Object>} - Returns a promise that resolves with the parsed JSON response.\n * @throws {Error} Throws an error if the request fails or if the response cannot be parsed.\n *\n * @example\n *\n * async function sendData() {\n *   const dataToSend = {\n *     key1: 'value1',\n *     key2: 'value2',\n *   };\n *   try {\n *     const response = await post('https://api.example.com/data', dataToSend);\n *     console.log(response);\n *   } catch (error) {\n *     console.error('Error sending data:', error);\n *   }\n * }\n *\n * sendData();\n */\nasync function post(url, body = {}, requestOptions = {}) {\n  return new Promise((resolve, reject) => {\n    const protocol = getProtocol(url);\n    const data = JSON.stringify(body);\n\n    // Set default headers and merge with requestOptions\n    const options = Object.assign(\n      {\n        method: 'POST',\n        headers: {\n          'Content-Type': 'application/json',\n          'Content-Length': data.length\n        }\n      },\n      requestOptions\n    );\n\n    const req = protocol\n      .request(url, options, (res) => {\n        let responseData = '';\n\n        // A chunk of data has been received.\n        res.on('data', (chunk) => {\n          responseData += chunk;\n        });\n\n        // The whole response has been received.\n        res.on('end', () => {\n          try {\n            res.text = responseData;\n            resolve(res);\n          } catch (error) {\n            reject(error);\n          }\n        });\n      })\n      .on('error', (error) => {\n        reject(error);\n      });\n\n    // Write the request body and end the request.\n    req.write(data);\n    req.end();\n  });\n}\n\nexport default fetch;\nexport { fetch, post };\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\n// The cache manager manages the Highcharts library and its dependencies.\n// The cache itself is stored in .cache, and is checked by the config system\n// before starting the service\n\nimport { existsSync, mkdirSync, readFileSync, writeFileSync } from 'fs';\nimport { join } from 'path';\n\nimport dotenv from 'dotenv';\nimport HttpsProxyAgent from 'https-proxy-agent';\nimport { fetch } from './fetch.js';\n\nimport { log } from './logger.js';\nimport { __dirname } from '../lib/utils.js';\n\ndotenv.config();\n\nconst cachePath = join(__dirname, '.cache');\n\nconst cache = {\n  cdnURL: 'https://code.highcharts.com/',\n  activeManifest: {},\n  sources: '',\n  hcVersion: ''\n};\n\n// TODO: The config should be accesssible globally so we don't have to do this sort of thing..\nlet appliedConfig = false;\n\n/**\n * Extracts the Highcharts version from the cache\n */\nconst extractVersion = () =>\n  (cache.hcVersion = cache.sources\n    .substr(0, cache.sources.indexOf('*/'))\n    .replace('/*', '')\n    .replace('*/', '')\n    .replace(/\\n/g, '')\n    .trim());\n\n/**\n * Saves the Highcharts part of a config to a manifest file in the cache\n *\n * @param {object} config - Highcharts related configuration object.\n * @param {object} fetchedModules - An object that contains mapped names of\n * fetched Highcharts modules to use.\n */\nconst saveConfigToManifest = async (config, fetchedModules) => {\n  const newManifest = {\n    version: config.version,\n    modules: fetchedModules || {}\n  };\n\n  // Update cache object with the current modules\n  cache.activeManifest = newManifest;\n\n  log(4, '[cache] writing new manifest');\n\n  try {\n    writeFileSync(\n      join(cachePath, 'manifest.json'),\n      JSON.stringify(newManifest),\n      'utf8'\n    );\n  } catch (error) {\n    log(1, `[cache] Error writing cache manifest: ${error}.`);\n  }\n};\n\n/**\n * Fetches a single script.\n *\n * @param {string} script - A path to script to get.\n * @param {object} proxyAgent - The proxy agent to use for a request.\n */\nconst fetchScript = async (script, proxyAgent) => {\n  try {\n    // Get rid of the .js from the custom strings\n    if (script.endsWith('.js')) {\n      script = script.substring(0, script.length - 3);\n    }\n\n    log(4, `[cache] Fetching script - ${script}.js`);\n\n    // If exists, add proxy agent to request options\n    const requestOptions = proxyAgent\n      ? {\n          agent: proxyAgent,\n          timeout: +process.env['PROXY_SERVER_TIMEOUT'] || 5000\n        }\n      : {};\n\n    // Fetch the script\n    const response = await fetch(`${script}.js`, requestOptions);\n\n    // If OK, return its text representation\n    if (response.statusCode === 200) {\n      return response.text;\n    }\n\n    throw `${response.statusCode}`;\n  } catch (error) {\n    log(1, `[cache] Error fetching script ${script}.js: ${error}.`);\n    throw error;\n  }\n};\n\n/**\n * Updates the Highcharts cache.\n *\n * @param {object} config - Highcharts related configuration object.\n * @param {string} sourcePath - A path to the file where save updated sources.\n * @return {object} An object that contains mapped names of fetched Highcharts\n * modules to use.\n */\nconst updateCache = async (config, sourcePath) => {\n  const { coreScripts, modules, indicators, scripts: customScripts } = config;\n  const hcVersion =\n    config.version === 'latest' || !config.version ? '' : `${config.version}/`;\n\n  log(3, '[cache] Updating cache to Highcharts ', hcVersion);\n\n  // Gather all scripts to fetch\n  const allScripts = [\n    ...coreScripts.map((c) => `${hcVersion}${c}`),\n    ...modules.map((m) =>\n      m === 'map' ? `maps/${hcVersion}modules/${m}` : `${hcVersion}modules/${m}`\n    ),\n    ...indicators.map((i) => `stock/${hcVersion}indicators/${i}`)\n  ];\n\n  // Configure proxy if exists\n  let proxyAgent;\n  const proxyHost = process.env['PROXY_SERVER_HOST'];\n  const proxyPort = process.env['PROXY_SERVER_PORT'];\n\n  if (proxyHost && proxyPort) {\n    proxyAgent = new HttpsProxyAgent({\n      host: proxyHost,\n      port: +proxyPort\n    });\n  }\n\n  const fetchedModules = {};\n  try {\n    cache.sources = // TODO: convert to for loop\n      (\n        await Promise.all([\n          ...allScripts.map(async (script) => {\n            const text = await fetchScript(\n              `${config.cdnURL || cache.cdnURL}${script}`,\n              proxyAgent\n            );\n\n            // If fetched correctly, set it\n            if (typeof text === 'string') {\n              fetchedModules[\n                script.replace(\n                  /(.*)\\/|(.*)modules\\/|stock\\/(.*)indicators\\/|maps\\/(.*)modules\\//gi,\n                  ''\n                )\n              ] = 1;\n            }\n\n            return text;\n          }),\n          ...customScripts.map((script) => fetchScript(script, proxyAgent))\n        ])\n      ).join(';\\n');\n    extractVersion();\n\n    // Save the fetched modules into caches' source JSON\n    writeFileSync(sourcePath, cache.sources);\n    return fetchedModules;\n  } catch (error) {\n    log(1, '[cache] Unable to update local Highcharts cache.');\n  }\n};\n\nexport const updateVersion = async (newVersion) =>\n  appliedConfig\n    ? await checkCache(\n        Object.assign(appliedConfig, {\n          version: newVersion\n        })\n      )\n    : false;\n\n/**\n * Fetches any missing Highcharts and dependencies\n *\n * @param {object} config - Highcharts related configuration object.\n */\nexport const checkCache = async (config) => {\n  let fetchedModules;\n  // Prepare paths to manifest and sources from the .cache folder\n  const manifestPath = join(cachePath, 'manifest.json');\n  const sourcePath = join(cachePath, 'sources.js');\n\n  // TODO: deal with trying to switch to the running version\n  // const activeVersion = appliedConfig ? appliedConfig.version : false;\n\n  appliedConfig = config;\n\n  // Create the .cache destination if it doesn't exist already\n  !existsSync(cachePath) && mkdirSync(cachePath);\n\n  // Fetch all the scripts either if manifest.json does not exist\n  // or if the forceFetch option is enabled\n  if (!existsSync(manifestPath) || config.forceFetch) {\n    log(3, '[cache] Fetching and caching Highcharts dependencies.');\n    fetchedModules = await updateCache(config, sourcePath);\n  } else {\n    let requestUpdate = false;\n\n    // Read the manifest JSON\n    const manifest = JSON.parse(readFileSync(manifestPath));\n\n    // Check if the modules is an array, if so, we rewrite it to a map to make\n    // it easier to resolve modules.\n    if (manifest.modules && Array.isArray(manifest.modules)) {\n      const moduleMap = {};\n      manifest.modules.forEach((m) => (moduleMap[m] = 1));\n      manifest.modules = moduleMap;\n    }\n\n    const { modules, coreScripts, indicators } = config;\n    const numberOfModules =\n      modules.length + coreScripts.length + indicators.length;\n\n    // Compare the loaded config with the contents in .cache.\n    // If there are changes, fetch requested modules and products,\n    // and bake them into a giant blob. Save the blob.\n    if (manifest.version !== config.version) {\n      log(3, '[cache] Highcharts version mismatch in cache, need to re-fetch.');\n      requestUpdate = true;\n    } else if (Object.keys(manifest.modules || {}).length !== numberOfModules) {\n      log(\n        3,\n        '[cache] Cache and requested modules does not match, need to re-fetch.'\n      );\n      requestUpdate = true;\n    } else {\n      // Check each module, if anything is missing refetch everything\n      requestUpdate = (config.modules || []).some((moduleName) => {\n        if (!manifest.modules[moduleName]) {\n          log(\n            3,\n            `[cache] The ${moduleName} missing in cache, need to re-fetch.`\n          );\n          return true;\n        }\n      });\n    }\n\n    if (requestUpdate) {\n      fetchedModules = await updateCache(config, sourcePath);\n    } else {\n      log(3, '[cache] Dependency cache is up to date, proceeding.');\n\n      // Load the sources\n      cache.sources = readFileSync(sourcePath, 'utf8');\n\n      // Get current modules map\n      fetchedModules = manifest.modules;\n      extractVersion();\n    }\n  }\n\n  // Finally, save the new manifest, which is basically our current config\n  // in a slightly different format\n  await saveConfigToManifest(config, fetchedModules);\n};\n\nexport default {\n  checkCache,\n  updateVersion,\n  getCache: () => cache,\n  highcharts: () => cache.sources,\n  version: () => cache.hcVersion\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport puppeteer from 'puppeteer';\nimport fs from 'fs';\nimport * as url from 'url';\nimport { log } from './logger.js';\nimport path from 'node:path';\n\n// Workaround for https://bugs.chromium.org/p/chromium/issues/detail?id=1463328\n// Not ideal - leaves trash in the FS\nimport { randomBytes } from 'node:crypto';\nconst RANDOM_PID = randomBytes(64).toString('base64url');\nconst PUPPETEER_DIR = path.join('tmp', `puppeteer-${RANDOM_PID}`);\nconst DATA_DIR = path.join(PUPPETEER_DIR, 'profile');\n\n// The minimal args to speed up the browser\nconst minimalArgs = [\n  `--user-data-dir=${DATA_DIR}`,\n  '--autoplay-policy=user-gesture-required',\n  '--disable-background-networking',\n  '--disable-background-timer-throttling',\n  '--disable-backgrounding-occluded-windows',\n  '--disable-breakpad',\n  '--disable-client-side-phishing-detection',\n  '--disable-component-update',\n  '--disable-default-apps',\n  '--disable-dev-shm-usage',\n  '--disable-domain-reliability',\n  '--disable-extensions',\n  '--disable-features=AudioServiceOutOfProcess',\n  '--disable-hang-monitor',\n  '--disable-ipc-flooding-protection',\n  '--disable-notifications',\n  '--disable-offer-store-unmasked-wallet-cards',\n  '--disable-popup-blocking',\n  '--disable-print-preview',\n  '--disable-prompt-on-repost',\n  '--disable-renderer-backgrounding',\n  '--disable-session-crashed-bubble',\n  '--disable-setuid-sandbox',\n  '--disable-speech-api',\n  '--disable-sync',\n  '--hide-crash-restore-bubble',\n  '--hide-scrollbars',\n  '--ignore-gpu-blacklist',\n  '--metrics-recording-only',\n  '--mute-audio',\n  '--no-default-browser-check',\n  '--no-first-run',\n  '--no-pings',\n  '--no-sandbox',\n  '--no-zygote',\n  '--password-store=basic',\n  '--use-mock-keychain'\n];\n\nconst __dirname = url.fileURLToPath(new URL('.', import.meta.url));\n\nconst template = fs.readFileSync(\n  __dirname + '/../templates/template.html',\n  'utf8'\n);\n\nlet browser;\n\nexport const newPage = async () => {\n  if (!browser) return false;\n\n  const p = await browser.newPage();\n\n  await p.setContent(template);\n  await p.addScriptTag({ path: __dirname + '/../.cache/sources.js' });\n  // eslint-disable-next-line no-undef\n  await p.evaluate(() => window.setupHighcharts());\n\n  p.on('pageerror', async (err) => {\n    // TODO: Consider adding a switch here that turns on log(0) logging\n    // on page errors.\n    log(1, '[page error]', err);\n    await p.$eval(\n      '#container',\n      (element, errorMessage) => {\n        // eslint-disable-next-line no-undef\n        if (window._displayErrors) {\n          element.innerHTML = errorMessage;\n        }\n      },\n      `<h1>Chart input data error</h1>${err.toString()}`\n    );\n  });\n\n  return p;\n};\n\nexport const create = async (puppeteerArgs) => {\n  const allArgs = [...minimalArgs, ...(puppeteerArgs || [])];\n\n  // Create a browser\n  if (!browser) {\n    let tryCount = 0;\n\n    const open = async () => {\n      try {\n        log(\n          3,\n          '[browser] attempting to get a browser instance (try',\n          tryCount + ')'\n        );\n\n        browser = await puppeteer.launch({\n          headless: 'new',\n          args: allArgs,\n          userDataDir: './tmp/'\n        });\n      } catch (e) {\n        log(0, '[browser]', e);\n        if (++tryCount < 25) {\n          log(3, '[browser] failed:', e);\n          await new Promise((response) => setTimeout(response, 4000));\n          await open();\n        } else {\n          log(0, 'Max retries reached');\n        }\n      }\n    };\n\n    try {\n      await open();\n    } catch (e) {\n      log(0, '[browser] Unable to open browser');\n      return false;\n    }\n\n    if (!browser) {\n      log(0, '[browser] Unable to open browser');\n      return false;\n    }\n  }\n\n  // Return a browser promise\n  return browser;\n};\n\nexport const get = async () => {\n  if (!browser) {\n    throw 'No valid browser has been created';\n  }\n\n  return browser;\n};\n\nexport const close = async () => {\n  // Close the browser when connnected\n  if (browser.connected) {\n    await browser.close();\n  }\n};\n\nexport default {\n  get,\n  close,\n  newPage\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\n// TODO: remove this temp benchmark stuff. I had this idea of doing a general benchmarking\n// system, but it adds so much bloat in the code that it shouldn't be there.\n\nimport benchmark from './benchmark.js';\nimport cache from './cache.js';\nimport { log } from './logger.js';\nimport svgTemplate from './../templates/svg_export/svg_export.js';\n\nimport { readFileSync } from 'fs';\nimport path from 'path';\nimport * as url from 'url';\n\nconst __basedir = url.fileURLToPath(new URL('.', import.meta.url));\n\n// const jsonTemplate = require('./../templates/json_export/json_export.js');\n\n/**\n * Gets the clip region for the chart DOM node.\n *\n * @param {object} page - A page of a browser instance.\n * @return {object} - A clipped region.\n */\nconst getClipRegion = (page) =>\n  page.$eval('#chart-container', (element) => {\n    const { x, y, width, height } = element.getBoundingClientRect();\n    return {\n      x,\n      y,\n      width,\n      height: Math.trunc(height > 1 ? height : 500)\n    };\n  });\n\n/**\n * Rasterizes the page to an image (PNG or JPEG)\n *\n * @param {object} page - A page of a browser instance.\n * @param {string} type - The type of a result image.\n * @param {string} encoding - The type of encoding used.\n * @param {string} clip - The clip region.\n * @returns {string} - A string representation of a screenshot.\n */\nconst createImage = async (page, type, encoding, clip) =>\n  await Promise.race([\n    page.screenshot({\n      type,\n      encoding,\n      clip,\n\n      // #447 - always render on a transparent page\n      // this will not affect users who do not explicitly set\n      // chart.backgroundColor to a color with opacity lower than 1\n      omitBackground: true\n    }),\n    new Promise((resolve, reject) =>\n      setTimeout(() => reject(new Error('Rasterization timeout')), 1500)\n    )\n  ]);\n\n/**\n * Turns page into a PDF.\n *\n * @param {object} page - A page of a browser instance.\n * @param {number} height - The height of a chart.\n * @param {number} width - The width of a chart.\n * @param {string} encoding - The type of encoding used.\n * @return {object} - A buffer with PDF representation.\n */\nconst createPDF = async (page, height, width, encoding) =>\n  await page.pdf({\n    // This will remove an extra empty page in PDF exports\n    height: height + 1,\n    width,\n    encoding\n  });\n\n/**\n * Exports as a SVG.\n *\n * @param {object} page - A page of a browser instance.\n * @return {object} - The outerHTML element with the SVG representation.\n */\nconst createSVG = async (page) =>\n  await page.$eval(\n    '#container svg:first-of-type',\n    (element) => element.outerHTML\n  );\n\n/** Load config into a page and render a chart */\nconst setAsConfig = async (page, chart, options) =>\n  await page.evaluate(\n    // eslint-disable-next-line no-undef\n    (chart, options) => window.triggerExport(chart, options),\n    chart,\n    options\n  );\n\n/** Load SVG into a page */\n// const setAsSVG = async (page, svgStr) => true;\n\n/**\n * Does an export for a given browser.\n *\n * @param {object} browser - A browser instance.\n * @param {object} chart - Chart's options.\n * @param {object} options - All options object.\n * @return {object} - The data returned from one of the methods for exporting\n * a specific type of an image.\n */\nexport default async (page, chart, options) => {\n  /**\n   * Keeps track of all resources added on the page with addXXXTag. etc\n   * It's VITAL that all added resources ends up here so we can clear things\n   * out when doing a new export in the same page!\n   */\n  const injectedResources = [];\n\n  /** Clear out all state set on the page with addScriptTag/addStyleTag. */\n  const clearInjected = async (page) => {\n    for (const res of injectedResources) {\n      await res.dispose();\n    }\n\n    // Reset all CSS and script tags\n    await page.evaluate(() => {\n      // eslint-disable-next-line no-undef\n      const [, ...scriptsToRemove] = document.getElementsByTagName('script');\n      // eslint-disable-next-line no-undef\n      const [, ...stylesToRemove] = document.getElementsByTagName('style');\n      // eslint-disable-next-line no-undef\n      const [...linksToRemove] = document.getElementsByTagName('link');\n\n      // Remove tags\n      for (const element of [\n        ...scriptsToRemove,\n        ...stylesToRemove,\n        ...linksToRemove\n      ]) {\n        element.remove();\n      }\n    });\n  };\n\n  try {\n    const exportBench = benchmark('Puppeteer');\n\n    log(4, '[export] Determining export path.');\n\n    const exportOptions = options.export;\n\n    // Force a rAF\n    // See https://github.com/puppeteer/puppeteer/issues/7507\n    // eslint-disable-next-line no-undef\n    await page.evaluate(() => requestAnimationFrame(() => {}));\n\n    // Decide whether display error or debbuger wrapper around it\n    const displayErrors =\n      exportOptions?.options?.chart?.displayErrors &&\n      cache.getCache().activeManifest.modules.debugger;\n\n    // eslint-disable-next-line no-undef\n    await page.evaluate((d) => (window._displayErrors = d), displayErrors);\n\n    const svgBench = benchmark('SVG handling');\n\n    let isSVG;\n\n    if (\n      chart.indexOf &&\n      (chart.indexOf('<svg') >= 0 || chart.indexOf('<?xml') >= 0)\n    ) {\n      // SVG INPUT HANDLING\n\n      log(4, '[export] Treating as SVG.');\n\n      // If input is also svg, just return it\n      if (exportOptions.type === 'svg') {\n        return chart;\n      }\n\n      isSVG = true;\n      const setPageBench = benchmark('Setting content');\n      await page.setContent(svgTemplate(chart));\n      setPageBench();\n    } else {\n      // JSON Config handling\n\n      log(4, '[export] Treating as config.');\n\n      // Need to perform straight inject\n      if (exportOptions.strInj) {\n        // Injection based configuration export\n        const setPageBench = benchmark('Setting page content (inject)');\n\n        await setAsConfig(\n          page,\n          {\n            chart: {\n              height: exportOptions.height,\n              width: exportOptions.width\n            }\n          },\n          options\n        );\n\n        setPageBench();\n      } else {\n        // Basic configuration export\n\n        chart.chart.height = exportOptions.height;\n        chart.chart.width = exportOptions.width;\n\n        const setContentBench = benchmark('Setting page content (config)');\n        await setAsConfig(page, chart, options);\n        setContentBench();\n      }\n    }\n\n    svgBench();\n    const resBench = benchmark('Applying resources');\n\n    // Use resources\n    const resources = options.customCode.resources;\n    if (resources) {\n      // Load custom JS code\n      if (resources.js) {\n        injectedResources.push(\n          await page.addScriptTag({\n            content: resources.js\n          })\n        );\n      }\n\n      // Load scripts from all custom files\n      if (resources.files) {\n        for (const file of resources.files) {\n          try {\n            const isLocal = !file.startsWith('http') ? true : false;\n\n            // Add each custom script from resources' files\n            injectedResources.push(\n              await page.addScriptTag(\n                isLocal\n                  ? {\n                      content: readFileSync(file, 'utf8')\n                    }\n                  : {\n                      url: file\n                    }\n              )\n            );\n          } catch (notice) {\n            log(4, '[export] JS file not found.');\n          }\n        }\n      }\n\n      const cssBench = benchmark('Loading css');\n\n      // Load CSS\n      if (resources.css) {\n        let cssImports = resources.css.match(/@import\\s*([^;]*);/g);\n        if (cssImports) {\n          // Handle css section\n          for (let cssImportPath of cssImports) {\n            if (cssImportPath) {\n              cssImportPath = cssImportPath\n                .replace('url(', '')\n                .replace('@import', '')\n                .replace(/\"/g, '')\n                .replace(/'/g, '')\n                .replace(/;/, '')\n                .replace(/\\)/g, '')\n                .trim();\n\n              // Add each custom css from resources\n              if (cssImportPath.startsWith('http')) {\n                injectedResources.push(\n                  await page.addStyleTag({\n                    url: cssImportPath\n                  })\n                );\n              } else if (options.customCode.allowFileResources) {\n                injectedResources.push(\n                  await page.addStyleTag({\n                    path: path.join(__basedir, cssImportPath)\n                  })\n                );\n              }\n            }\n          }\n        }\n\n        // The rest of the CSS section will be content by now\n        injectedResources.push(\n          await page.addStyleTag({\n            content: resources.css.replace(/@import\\s*([^;]*);/g, '') || ' '\n          })\n        );\n      }\n\n      cssBench();\n    }\n\n    resBench();\n\n    // Get the real chart size\n    const size = isSVG\n      ? await page.$eval(\n          '#chart-container svg:first-of-type',\n          async (element, scale) => {\n            return {\n              chartHeight: element.height.baseVal.value * scale,\n              chartWidth: element.width.baseVal.value * scale\n            };\n          },\n          parseFloat(exportOptions.scale)\n        )\n      : await page.evaluate(async () => {\n          // eslint-disable-next-line no-undef\n          const { chartHeight, chartWidth } = window.Highcharts.charts[0];\n          return {\n            chartHeight,\n            chartWidth\n          };\n        });\n\n    const vpBench = benchmark('Setting viewport');\n\n    // Set final height and width for viewport\n    const viewportHeight = Math.ceil(size?.chartHeight || exportOptions.height);\n    const viewportWidth = Math.ceil(size?.chartWidth || exportOptions.width);\n\n    // Set the viewport for the first time\n    // NOTE: the call to setViewport is expensive - can we get away with only\n    // calling it once, e.g. moving this one into the isSVG condition below?\n    await page.setViewport({\n      height: viewportHeight,\n      width: viewportWidth,\n      deviceScaleFactor: isSVG ? 1 : parseFloat(exportOptions.scale)\n    });\n\n    // Prepare a zoom callback for the next evaluate call\n    const zoomCallback = isSVG\n      ? // In case of SVG the zoom must be set directly for body\n        (scale) => {\n          // Set the zoom as scale\n          // eslint-disable-next-line no-undef\n          document.body.style.zoom = scale;\n\n          // Set the margin to 0px\n          // eslint-disable-next-line no-undef\n          document.body.style.margin = '0px';\n        }\n      : // No need for such scale manipulation in case of other types of exports\n        () => {\n          // Reset the zoom for other exports than to SVGs\n          // eslint-disable-next-line no-undef\n          document.body.style.zoom = 1;\n        };\n\n    // Set the zoom accordingly\n    await page.evaluate(zoomCallback, parseFloat(exportOptions.scale));\n\n    // Get the clip region for the page\n    const { height, width, x, y } = await getClipRegion(page);\n\n    if (!isSVG) {\n      // Set the final viewport now that we have the real height\n      await page.setViewport({\n        width: Math.round(width),\n        height: Math.round(height),\n        deviceScaleFactor: parseFloat(exportOptions.scale)\n      });\n    }\n\n    vpBench();\n\n    let data;\n\n    const expBenchmark = benchmark('Rasterizing chart');\n\n    // RASTERIZATION\n    if (exportOptions.type === 'svg') {\n      // SVG\n      data = await createSVG(page);\n    } else if (exportOptions.type === 'png' || exportOptions.type === 'jpeg') {\n      // PNG or JPEG\n      data = await createImage(page, exportOptions.type, 'base64', {\n        width: viewportWidth,\n        height: viewportHeight,\n        x,\n        y\n      });\n    } else if (exportOptions.type === 'pdf') {\n      // PDF\n      data = await createPDF(page, viewportHeight, viewportWidth, 'base64');\n    } else {\n      throw `Unsupported output format ${exportOptions.type}`;\n    }\n\n    // Destroy old charts after the export is done\n    await page.evaluate(() => {\n      // eslint-disable-next-line no-undef\n      const oldCharts = Highcharts.charts;\n\n      // Check in any already existing charts\n      if (oldCharts.length) {\n        // Destroy old charts\n        for (const oldChart of oldCharts) {\n          oldChart && oldChart.destroy();\n          // eslint-disable-next-line no-undef\n          Highcharts.charts.shift();\n        }\n      }\n    });\n\n    expBenchmark();\n    exportBench();\n\n    await clearInjected(page);\n\n    return data;\n  } catch (error) {\n    await clearInjected(page);\n    log(1, `[export] Error encountered during export: ${error}`);\n\n    return error;\n  }\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2022, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { log } from './logger.js';\nconst timers = {};\n\n// TODO: Read from config\nlet enabled = false;\n\nexport default (id) => {\n  if (!enabled) {\n    return () => {};\n  }\n\n  timers[id] = new Date();\n  return () => {\n    log(\n      3,\n      `[benchmark] - ${id}: ${new Date().getTime() - timers[id].getTime()}ms`\n    );\n  };\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport cssTemplate from './css.js';\n\nexport default (chart) => `\n<!DOCTYPE html>\n<html lang='en-US'>\n  <head>\n    <meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\">\n    <title>Highcarts Export</title>\n  </head>\n  <style>\n    ${cssTemplate()}\n  </style>\n  <body>\n    <div id=\"chart-container\">\n      ${chart}\n    </div>\n  </body>\n</html>\n\n`;\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { v4 as uuid } from 'uuid';\nimport { Pool } from 'tarn';\nimport {\n  close,\n  newPage as browserNewPage,\n  create as createBrowser\n} from './browser.js';\nimport { log } from './logger.js';\n\nimport puppeteerExport from './export.js';\n\nlet performedExports = 0;\nlet exportAttempts = 0;\nlet timeSpent = 0;\nlet droppedExports = 0;\nlet spentAverage = 0;\nlet poolConfig = {};\n\n// The pool instance\nlet pool = false;\n\n// Custom puppeteer arguments\nlet puppeteerArgs;\n\nconst factory = {\n  /**\n   * Creates a new worker.\n   *\n   * @return {object} - An object with the id of a resource, the work count and\n   * a reference to the browser page.\n   */\n  create: async () => {\n    const id = uuid();\n    let page = false;\n\n    const s = new Date().getTime();\n\n    try {\n      page = await browserNewPage();\n\n      if (!page || page.isClosed()) {\n        throw 'invalid page';\n      }\n\n      log(\n        3,\n        `[pool] Successfully created a worker ${id} - took ${\n          new Date().getTime() - s\n        } ms.`\n      );\n    } catch (error) {\n      log(\n        1,\n        `[pool] Error creating a new page in pool entry creation! ${error}`\n      );\n\n      throw 'Error creating page';\n    }\n\n    return {\n      id,\n      page,\n      // Try to distribute the initial work count\n      workCount: Math.round(Math.random() * (poolConfig.workLimit / 2))\n    };\n  },\n\n  /**\n   * Validates a worker.\n   *\n   * @param {object} workerHandle - A browser's instance.\n   *\n   * @return {boolean} - Bool that indicates if a resource is valid or not.\n   */\n  validate: (workerHandle) => {\n    if (\n      poolConfig.workLimit &&\n      ++workerHandle.workCount > poolConfig.workLimit\n    ) {\n      log(\n        3,\n        `[pool] Worker failed validation:`,\n        `exceeded work limit (limit is ${poolConfig.workLimit})`\n      );\n      return false;\n    }\n    return true;\n  },\n\n  /**\n   * Destroys a worker.\n   *\n   * @param {object} workerHandle - A browser's instance.\n   */\n  destroy: (workerHandle) => {\n    log(3, `[pool] Destroying pool entry ${workerHandle.id}.`);\n\n    if (workerHandle.page) {\n      // We don't really need to wait around for this.\n      workerHandle.page.close();\n    }\n  },\n\n  // Logger function\n  log: (message, logLevel) => console.log(`${logLevel}: ${message}`)\n};\n\n/**\n * Inits the pool of resources.\n *\n * @param {object} config - Pool configuration along with custom puppeteer\n * arguments for the puppeteer.launch function.\n */\nexport const init = async (config) => {\n  // The newest puppeteer arguments for the browser creation\n  puppeteerArgs = config.puppeteerArgs;\n\n  // Wait until we've sucessfully created a browser instance.\n  try {\n    await createBrowser(puppeteerArgs);\n  } catch (e) {\n    log(0, '[pool|browser]', e);\n  }\n\n  // For the module scope usage\n  poolConfig = config && config.pool ? { ...config.pool } : {};\n\n  log(\n    3,\n    '[pool] Initializing pool:',\n    `min ${poolConfig.initialWorkers}, max ${poolConfig.maxWorkers}.`\n  );\n\n  if (pool) {\n    return log(\n      4,\n      '[pool] Already initialized, please kill it before creating a new one.'\n    );\n  }\n\n  // Attach process' exit listeners\n  if (poolConfig.listenToProcessExits) {\n    attachProcessExitListeners();\n  }\n\n  try {\n    // Create a pool along with a minimal number of resources\n    pool = new Pool({\n      // Get the create/validate/destroy/log functions\n      ...factory,\n      min: poolConfig.initialWorkers,\n      max: poolConfig.maxWorkers,\n      createRetryIntervalMillis: 200,\n      createTimeoutMillis: poolConfig.acquireTimeout,\n      acquireTimeoutMillis: poolConfig.acquireTimeout,\n      destroyTimeoutMillis: poolConfig.acquireTimeout,\n      idleTimeoutMillis: poolConfig.timeoutThreshold,\n      reapIntervalMillis: 1000, // poolConfig.reaper ? 120000 : 0, for now\n      propagateCreateError: false\n    });\n\n    // Set events\n    pool.on('createFail', (eventId, err) => {\n      log(\n        1,\n        `[pool] Error when creating worker of an event id ${eventId}:`,\n        err\n      );\n    });\n\n    pool.on('acquireFail', (eventId, err) => {\n      log(\n        1,\n        `[pool] Error when acquiring worker of an event id ${eventId}:`,\n        err\n      );\n    });\n\n    pool.on('destroyFail', (eventId, resource, err) => {\n      log(\n        1,\n        `[pool] Error when destroying worker of an id ${resource.id}, event id ${eventId}:`,\n        err\n      );\n    });\n\n    pool.on('release', (resource) => {\n      log(4, `[pool] Releasing a worker of an id ${resource.id}`);\n    });\n\n    pool.on('destroySuccess', (eventId, resource) => {\n      log(4, `[pool] Destroyed a worker of an id ${resource.id}`);\n    });\n\n    const initialResources = [];\n    // Create an initial number of resources\n    for (let i = 0; i < poolConfig.initialWorkers; i++) {\n      initialResources.push(await pool.acquire().promise);\n    }\n\n    // Release the initial number of resources back to the pool\n    initialResources.forEach((resource) => {\n      pool.release(resource);\n    });\n\n    log(\n      3,\n      `[pool] The pool is ready with ${poolConfig.initialWorkers} initial resources waiting.`\n    );\n  } catch (error) {\n    log(1, `[pool] Couldn't create the worker pool ${error}`);\n    throw error;\n  }\n};\n\n/**\n * Attaches process' exit listeners.\n */\nexport function attachProcessExitListeners() {\n  log(4, '[pool] Attaching exit listeners to the process.');\n\n  // Kill all pool resources on exit\n  process.on('exit', async () => {\n    await killPool();\n  });\n\n  // Handler for the SIGINT\n  process.on('SIGINT', (name, code) => {\n    log(4, `The ${name} event with code: ${code}.`);\n    process.exit(1);\n  });\n\n  // Handler for the SIGTERM\n  process.on('SIGTERM', (name, code) => {\n    log(4, `The ${name} event with code: ${code}.`);\n    process.exit(1);\n  });\n\n  // Handler for the uncaughtException\n  process.on('uncaughtException', async (error, name) => {\n    log(4, `The ${name} error, message: ${error.message}.`);\n  });\n}\n\n/**\n * Kills the pool and flush the browser instance.\n */\nexport async function killPool() {\n  log(3, '[pool] Killing all workers.');\n\n  // Return true when the pool is already destroyed\n  if (pool.destroyed) {\n    // Close the browser instance if still connected\n    await close();\n    return true;\n  }\n\n  // If still alive, destroy the pool of pages before closing a browser\n  await pool.destroy();\n\n  // Close the browser instance\n  await close();\n  return true;\n}\n\n/**\n * Posts work to the pool.\n *\n * @param {object} chart - Chart's options.\n * @param {object} options - All options object.\n */\nexport const postWork = async (chart, options) => {\n  let workerHandle;\n\n  // Handle fail conditions\n  const fail = (msg) => {\n    ++droppedExports;\n\n    if (workerHandle) {\n      pool.release(workerHandle);\n    }\n\n    throw 'In pool.postWork: ' + msg;\n  };\n\n  log(4, '[pool] Work received, starting to process.');\n\n  if (poolConfig.benchmarking) {\n    getPoolInfo();\n  }\n\n  ++exportAttempts;\n\n  if (!pool) {\n    log(1, '[pool] Work received, but pool has not been started.');\n    return fail('Pool is not inited but work was posted to it!');\n  }\n\n  // Acquire the worker along with the id of resource and work count\n  try {\n    log(4, '[pool] Acquiring worker');\n    workerHandle = await pool.acquire().promise;\n  } catch (error) {\n    return fail(`[pool] Error when acquiring available entry: ${error}`);\n  }\n\n  log(4, '[pool] Acquired worker handle');\n\n  if (!workerHandle.page) {\n    return fail('Resolved worker page is invalid: pool setup is wonky');\n  }\n\n  try {\n    // Save the start time\n    let workStart = new Date().getTime();\n\n    log(4, `[pool] Starting work on pool entry ${workerHandle.id}.`);\n\n    // Perform an export on a puppeteer level\n    const result = await puppeteerExport(workerHandle.page, chart, options);\n\n    // Check if it's an error\n    if (result instanceof Error) {\n      // TODO: If the export failed because puppeteer timed out, we need to force kill the worker so we get a new page. That needs to be handled better than this hack.\n      if (result.message === 'Rasterization timeout') {\n        workerHandle.page.close();\n        workerHandle.page = await browserNewPage();\n      }\n\n      return fail(result);\n    }\n\n    // Release the resource back to the pool\n    pool.release(workerHandle);\n\n    // Used for statistics in averageTime and processedWorkCount, which\n    // in turn is used by the /health route.\n    const workEnd = new Date().getTime();\n    const exportTime = workEnd - workStart;\n    timeSpent += exportTime;\n    spentAverage = timeSpent / ++performedExports;\n\n    log(4, `[pool] Work completed in ${exportTime} ms.`);\n\n    // Otherwise return the result\n    return {\n      data: result,\n      options\n    };\n  } catch (error) {\n    fail(`Error trying to perform puppeteer export: ${error}.`);\n  }\n};\n\n/**\n * Gets the pool.\n */\nexport function getPool() {\n  return pool;\n}\n\nexport const getPoolInfoJSON = () => ({\n  min: pool.min,\n  max: pool.max,\n  size: pool.size,\n  available: pool.available,\n  borrowed: pool.borrowed,\n  pending: pool.pending,\n  spareResourceCapacity: pool.spareResourceCapacity\n});\n\n/**\n * Gets the pool's information.\n */\nexport function getPoolInfo() {\n  const {\n    min,\n    max,\n    size,\n    available,\n    borrowed,\n    pending,\n    spareResourceCapacity\n  } = pool;\n\n  log(4, `[pool] The minimum number of resources allowed by pool: ${min}.`);\n  log(4, `[pool] The maximum number of resources allowed by pool: ${max}.`);\n  log(\n    4,\n    `[pool] The number of all resources in pool (free or in use): ${size}.`\n  );\n  log(\n    4,\n    `[pool] The number of resources that are currently available: ${available}.`\n  );\n  log(\n    4,\n    `[pool] The number of resources that are currently acquired: ${borrowed}.`\n  );\n  log(\n    4,\n    `[pool] The number of callers waiting to acquire a resource: ${pending}.`\n  );\n  log(\n    4,\n    `[pool] The number of how many more resources can the pool manage/create: ${spareResourceCapacity}.`\n  );\n}\n\nexport default {\n  init,\n  killPool,\n  postWork,\n  getPool,\n  getPoolInfo,\n  getPoolInfoJSON,\n  workAttempts: () => exportAttempts,\n  droppedWork: () => droppedExports,\n  averageTime: () => spentAverage,\n  processedWorkCount: () => performedExports\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport cache from '../../cache.js';\nimport pool from '../../pool.js';\n\nconst packageVersion = process.env.npm_package_version;\nconst serverStartTime = new Date();\n\n/**\n * Adds the /health route which outputs basic stats for the server\n */\nexport default (app) =>\n  !app\n    ? false\n    : app.get('/health', (request, response) => {\n        response.send({\n          status: 'OK',\n          bootTime: serverStartTime,\n          uptime:\n            Math.floor(\n              (new Date().getTime() - serverStartTime.getTime()) / 1000 / 60\n            ) + ' minutes',\n          version: packageVersion,\n          highchartsVersion: cache.version(),\n          averageProcessingTime: pool.averageTime(),\n          performedExports: pool.processedWorkCount(),\n          failedExports: pool.droppedWork(),\n          exportAttempts: pool.workAttempts(),\n          sucessRatio: (pool.processedWorkCount() / pool.workAttempts()) * 100,\n          // eslint-disable-next-line import/no-named-as-default-member\n          pool: pool.getPoolInfoJSON()\n        });\n      });\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { existsSync, readFileSync, promises as fsPromises } from 'fs';\n\nimport prompts from 'prompts';\n\nimport { log } from './logger.js';\nimport { deepCopy, isObject, printUsage, toBoolean } from './utils.js';\nimport {\n  absoluteProps,\n  defaultConfig,\n  nestedArgs,\n  promptsConfig\n} from './schemas/config.js';\n\nlet generalOptions = {};\n\n/**\n * Getter for the general options.\n *\n * @return {object} - General options object.\n */\nexport const getOptions = () => generalOptions;\n\n/**\n * Initializes and sets the general options for the server instace.\n *\n * @param {object} userOptions - Additional user options (e.g. from the node\n * module usage).\n * @param {string[]} args - CLI arguments.\n * @return {object} - General options object.\n */\nexport const setOptions = (userOptions, args) => {\n  // Only for the CLI usage\n  if (args?.length) {\n    // Get the additional options from the custom JSON file\n    generalOptions = loadConfigFile(args);\n  }\n\n  // Update the default config with a correct option values\n  updateDefaultConfig(defaultConfig, generalOptions);\n\n  // Set values for server's options and returns them\n  generalOptions = initOptions(defaultConfig);\n\n  // Apply user options if there are any\n  if (userOptions) {\n    // Merge user options\n    generalOptions = mergeConfigOptions(\n      generalOptions,\n      userOptions,\n      absoluteProps\n    );\n  }\n\n  // Only for the CLI usage\n  if (args?.length) {\n    // Pair provided arguments\n    generalOptions = pairArgumentValue(generalOptions, args, defaultConfig);\n  }\n\n  // Return final general options\n  return generalOptions;\n};\n\n/**\n * Displays a prompt for the manual configuration.\n *\n * @param {string} configFileName - The name of a configuration file.\n */\nexport const manualConfig = async (configFileName) => {\n  // Prepare a config object\n  let configFile = {};\n\n  // Check if provided config file exists\n  if (existsSync(configFileName)) {\n    configFile = JSON.parse(readFileSync(configFileName, 'utf8'));\n  }\n\n  // Question about a configuration category\n  const onSubmit = async (p, categories) => {\n    let questionsCounter = 0;\n    let allQuestions = [];\n\n    // Create a corresponding property in the manualConfig object\n    for (const section of categories) {\n      // Mark each option with a section\n      promptsConfig[section] = promptsConfig[section].map((option) => ({\n        ...option,\n        section\n      }));\n\n      // Collect the questions\n      allQuestions = [...allQuestions, ...promptsConfig[section]];\n    }\n\n    await prompts(allQuestions, {\n      onSubmit: async (prompt, answer) => {\n        // Get the default modules\n        if (prompt.name === 'modules') {\n          answer = answer.length\n            ? answer.map((module) => prompt.choices[module])\n            : prompt.choices;\n\n          configFile[prompt.section][prompt.name] = answer;\n        } else {\n          configFile[prompt.section] = recursiveProps(\n            Object.assign({}, configFile[prompt.section] || {}),\n            prompt.name.split('.'),\n            answer\n          );\n        }\n\n        if (++questionsCounter === allQuestions.length) {\n          try {\n            await fsPromises.writeFile(\n              configFileName,\n              JSON.stringify(configFile, null, 2),\n              'utf8'\n            );\n          } catch (error) {\n            log(1, `[config] Error while creating config.json: ${error}`);\n          }\n          return true;\n        }\n      }\n    });\n\n    return true;\n  };\n\n  // Find the categories\n  const choices = Object.keys(promptsConfig).map((choice) => ({\n    title: `${choice} options`,\n    value: choice\n  }));\n\n  // Category prompt\n  return prompts(\n    {\n      type: 'multiselect',\n      name: 'category',\n      message: 'Which category do you want to configure?',\n      hint: 'Space: Select specific, A: Select all, Enter: Confirm.',\n      instructions: '',\n      choices\n    },\n    { onSubmit }\n  );\n};\n\n/**\n * Maps the old options to the new config structure.\n *\n * @param {object} oldOptions - Options to be mapped.\n */\nexport const mapToNewConfig = (oldOptions) => {\n  const newOptions = {};\n  // Cycle through old-structured options\n  for (const [key, value] of Object.entries(oldOptions)) {\n    const propertiesChain = nestedArgs[key] ? nestedArgs[key].split('.') : [];\n\n    // Populate object in correct properties levels\n    propertiesChain.reduce(\n      (obj, prop, index) =>\n        (obj[prop] =\n          propertiesChain.length - 1 === index ? value : obj[prop] || {}),\n      newOptions\n    );\n  }\n  return newOptions;\n};\n\n/**\n * Merges the new options to the options object. It omits undefined values.\n *\n * @param {object} options - Old options.\n * @param {object} newOptions - New options.\n * @param {string[]} absoluteProps - Array of object names that should be force\n * merged.\n */\nexport const mergeConfigOptions = (options, newOptions, absoluteProps = []) => {\n  const mergedOptions = deepCopy(options);\n\n  for (const [key, value] of Object.entries(newOptions)) {\n    mergedOptions[key] =\n      isObject(value) &&\n      !absoluteProps.includes(key) &&\n      mergedOptions[key] !== undefined\n        ? mergeConfigOptions(mergedOptions[key], value, absoluteProps)\n        : value !== undefined\n        ? value\n        : mergedOptions[key];\n  }\n\n  return mergedOptions;\n};\n\n/**\n * Initializes options for the `startExport` method by merging user options\n * with the general options.\n *\n * @param {any} exportOptions - User options for exporting.\n * @param {any} generalOptions - General options are used for the export server.\n * @return {object} - User options merged with default options.\n */\nexport const initExportSettings = (exportOptions, generalOptions = {}) => {\n  let options = {};\n\n  if (exportOptions.svg) {\n    options = deepCopy(generalOptions);\n    options.export.type = exportOptions.type || exportOptions.export.type;\n    options.export.scale = exportOptions.scale || exportOptions.export.scale;\n    options.export.outfile =\n      exportOptions.outfile || exportOptions.export.outfile;\n    options.payload = {\n      svg: exportOptions.svg\n    };\n  } else {\n    options = mergeConfigOptions(\n      generalOptions,\n      exportOptions,\n      // Omit going down recursively with the belows\n      absoluteProps\n    );\n  }\n\n  options.export.outfile =\n    options.export?.outfile || `chart.${options.export?.type || 'png'}`;\n  return options;\n};\n\n/**\n * Loads the configuration from a custom JSON file.\n *\n * @param {string[]} args - CLI arguments.\n * @return {object} - Options object from the JSON file.\n */\nfunction loadConfigFile(args) {\n  // Check if the --loadConfig option was used\n  const configIndex = args.findIndex(\n    (arg) => arg.replace(/-/g, '') === 'loadConfig'\n  );\n\n  // Check if the --loadConfig has a value\n  if (configIndex > -1 && args[configIndex + 1]) {\n    const fileName = args[configIndex + 1];\n    try {\n      // Check if an additional config file is a correct JSON file\n      if (fileName && fileName.endsWith('.json')) {\n        // Load an optional custom JSON config file\n        return JSON.parse(readFileSync(fileName));\n      }\n    } catch (error) {\n      log(1, `[config] Unable to load config from the ${fileName}: ${error}`);\n    }\n  }\n\n  // No additional options to return\n  return {};\n}\n\n/**\n * Setting correct values of the options from the default config.\n *\n * @param {object} configObj - The config object based on which the initial\n * configuration be made.\n * @param {object} customObj - The custom object which can contain additional\n * option values to set.\n * @param {string} propChain - Required for creating a string chain of\n * properties for nested arguments.\n */\nfunction updateDefaultConfig(configObj, customObj = {}, propChain = '') {\n  Object.keys(configObj).forEach((key) => {\n    if (!['puppeteer', 'highcharts'].includes(key)) {\n      const entry = configObj[key];\n      const customValue = customObj && customObj[key];\n      let numEnvVal;\n\n      if (typeof entry.value === 'undefined') {\n        updateDefaultConfig(entry, customValue, `${propChain}.${key}`);\n      } else {\n        // If a value from a custom JSON exists, it take precedence\n        if (customValue !== undefined) {\n          entry.value = customValue;\n        }\n\n        // If a value from an env variable exists, it take precedence\n        if (entry.envLink) {\n          // Load the env var\n          if (entry.type === 'boolean') {\n            entry.value = toBoolean(\n              [process.env[entry.envLink], entry.value].find(\n                (el) => el || el === 'false'\n              )\n            );\n          } else if (entry.type === 'number') {\n            numEnvVal = +process.env[entry.envLink];\n            entry.value = numEnvVal >= 0 ? numEnvVal : entry.value;\n          } else if (\n            entry.type.indexOf(']') >= 0 &&\n            process.env[entry.envLink]\n          ) {\n            entry.value = process.env[entry.envLink].split(',');\n          } else {\n            entry.value = process.env[entry.envLink] || entry.value;\n          }\n        }\n      }\n    }\n  });\n}\n\n/**\n * Inits options recursively.\n *\n * @param {any} items - Items to update options from.\n * @return {object} - Updated options object.\n */\nfunction initOptions(items) {\n  let options = {};\n  for (const [name, item] of Object.entries(items)) {\n    options[name] = Object.prototype.hasOwnProperty.call(item, 'value')\n      ? item.value\n      : initOptions(item);\n  }\n  return options;\n}\n\n/**\n * Pairs argument with a corresponding value.\n *\n * @param {object} options - All server options.\n * @param {string[]} args - Array of arguments from a user.\n * @param {object} defaultConfig - The default config object.\n */\nfunction pairArgumentValue(options, args, defaultConfig) {\n  for (let i = 0; i < args.length; i++) {\n    let option = args[i].replace(/-/g, '');\n\n    // Find the right place for property's value\n    const propertiesChain = nestedArgs[option]\n      ? nestedArgs[option].split('.')\n      : [];\n\n    propertiesChain.reduce((obj, prop, index) => {\n      if (propertiesChain.length - 1 === index) {\n        // Finds an option and set a corresponding value\n        if (typeof obj[prop] !== 'undefined') {\n          if (args[++i]) {\n            obj[prop] = args[i] || obj[prop];\n          } else {\n            console.log(`Missing argument value for ${option}!`.red, '\\n');\n            options = printUsage(defaultConfig);\n          }\n        }\n      }\n      return obj[prop];\n    }, options);\n  }\n\n  return options;\n}\n\n/**\n * Recursively sets a property in a correct indentation level based on the\n * array of nested properties names.\n *\n * @param {object} objectToUpdate - Object where a property must be set on a\n * correct level.\n * @param  {string[]}nestedNames - Array of nasted names that indicates\n * indentation level.\n * @param {any} value - A value to assign to the property.\n * @return {object} - Updated options object.\n */\nfunction recursiveProps(objectToUpdate, nestedNames, value) {\n  while (nestedNames.length > 1) {\n    const propName = nestedNames.shift();\n\n    // Create a property in object if it doesn't exist\n    if (!Object.prototype.hasOwnProperty.call(objectToUpdate, propName)) {\n      objectToUpdate[propName] = {};\n    }\n\n    // Call function again if there still names to go\n    objectToUpdate[propName] = recursiveProps(\n      Object.assign({}, objectToUpdate[propName]),\n      nestedNames,\n      value\n    );\n\n    return objectToUpdate;\n  }\n\n  // Assign the final value\n  objectToUpdate[nestedNames[0]] = value;\n  return objectToUpdate;\n}\n\nexport default {\n  getOptions,\n  setOptions,\n  manualConfig,\n  mapToNewConfig,\n  mergeConfigOptions,\n  initExportSettings\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { readFile, readFileSync, writeFileSync } from 'fs';\n\nimport { log } from './logger.js';\nimport { killPool, postWork } from './pool.js';\nimport {\n  clearText,\n  fixType,\n  handleResources,\n  isCorrectJSON,\n  optionsStringify,\n  roundNumber,\n  toBoolean,\n  wrapAround\n} from './utils.js';\nimport { initExportSettings, getOptions } from './config.js';\n\nlet allowCodeExecution = false;\n\nexport const startExport = async (settings, endCallback) => {\n  // Starting exporting process message\n  log(4, '[chart] Starting exporting process.');\n\n  // Initialize options\n  const options = initExportSettings(settings, getOptions());\n\n  // Get the export options\n  const exportOptions = options.export;\n\n  // If SVG is an input (argument can be sent only by the request)\n  if (options.payload?.svg && options.payload.svg !== '') {\n    return exportAsString(options.payload.svg.trim(), options, endCallback);\n  }\n\n  // Export using options from the file\n  if (exportOptions.infile && exportOptions.infile.length) {\n    log(4, '[chart] Attempting to export from an input file.');\n\n    // Try to read the file\n    return readFile(exportOptions.infile, 'utf8', (error, infile) => {\n      if (error) {\n        return log(1, `[chart] Error loading input file: ${error}.`);\n      }\n\n      // Get the string representation\n      options.export.instr = infile;\n      return exportAsString(options.export.instr.trim(), options, endCallback);\n    });\n  }\n\n  // Export with options from the raw representation\n  if (\n    (exportOptions.instr && exportOptions.instr !== '') ||\n    (exportOptions.options && exportOptions.options !== '')\n  ) {\n    log(4, '[chart] Attempting to export from a raw input.');\n\n    // Perform a direct inject when forced\n    if (toBoolean(options.customCode?.allowCodeExecution)) {\n      return doStraightInject(options, endCallback);\n    }\n\n    // Either try to parse to JSON first or do the direct export\n    return typeof exportOptions.instr === 'string'\n      ? exportAsString(exportOptions.instr.trim(), options, endCallback)\n      : doExport(\n          options,\n          exportOptions.instr || exportOptions.options,\n          endCallback\n        );\n  }\n\n  // No input specified, pass an error message to the callback\n  log(\n    1,\n    clearText(\n      `[chart] No input specified.\n      ${JSON.stringify(exportOptions, undefined, '  ')}.`\n    )\n  );\n\n  return (\n    endCallback &&\n    endCallback(false, {\n      error: true,\n      message: 'No input specified.'\n    })\n  );\n};\n\nexport const batchExport = (options) => {\n  const batchFunctions = [];\n\n  // Split and pair the --batch arguments\n  for (let pair of options.export.batch.split(';')) {\n    pair = pair.split('=');\n    if (pair.length === 2) {\n      batchFunctions.push(\n        new Promise((resolve, reject) => {\n          startExport(\n            {\n              ...options,\n              export: {\n                ...options.export,\n                infile: pair[0],\n                outfile: pair[1]\n              }\n            },\n            (info, error) => {\n              // Throw an error\n              if (error) {\n                return reject(error);\n              }\n\n              // Save the base64 from a buffer to a correct image file\n              writeFileSync(\n                info.options.export.outfile,\n                Buffer.from(info.data, 'base64')\n              );\n\n              resolve();\n            }\n          );\n        })\n      );\n    }\n  }\n\n  // Kill the pool after all exports are done\n  Promise.all(batchFunctions)\n    .then(() => {\n      killPool();\n    })\n    .catch((error) => {\n      log(1, `[chart] Error encountered during batch export: ${error}`);\n      killPool();\n    });\n};\n\nexport const singleExport = (options) => {\n  // Use instr or its alias, options\n  options.export.instr = options.export.instr || options.export.options;\n\n  // Perform an export\n  startExport(options, (info, error) => {\n    // Exit process when error\n    if (error) {\n      log(1, `[cli] ${error.message}`);\n      process.exit(1);\n    }\n\n    const { outfile, type } = info.options.export;\n\n    // Save the base64 from a buffer to a correct image file\n    writeFileSync(\n      outfile || `chart.${type}`,\n      type !== 'svg' ? Buffer.from(info.data, 'base64') : info.data\n    );\n\n    // Kill the pool\n    killPool();\n  });\n};\n\n/**\n * Function for choosing chart size and scale based on options prioritization.\n *\n * @param {object} options - All options object.\n * @return {object} - An object with updated size and scale for a chart.\n */\nexport const findChartSize = (options) => {\n  const { chart, exporting } =\n    options.export?.options || isCorrectJSON(options.export?.instr);\n\n  // See if globalOptions holds chart or exporting size\n  const globalOptions = isCorrectJSON(options.export?.globalOptions);\n\n  // Secure scale value\n  let scale =\n    options.export?.scale ||\n    exporting?.scale ||\n    globalOptions?.exporting?.scale ||\n    options.export?.defaultScale ||\n    1;\n\n  // the scale cannot be lower than 0.1 and cannot be higher than 5.0\n  scale = Math.max(0.1, Math.min(scale, 5.0));\n\n  // we want to round the numbers like 0.23234 -> 0.23\n  scale = roundNumber(scale, 2);\n\n  // Find chart size and scale\n  return {\n    height:\n      options.export?.height ||\n      exporting?.sourceHeight ||\n      chart?.height ||\n      globalOptions?.exporting?.sourceHeight ||\n      globalOptions?.chart?.height ||\n      options.export?.defaultHeight ||\n      400,\n    width:\n      options.export?.width ||\n      exporting?.sourceWidth ||\n      chart?.width ||\n      globalOptions?.exporting?.sourceWidth ||\n      globalOptions?.chart?.width ||\n      options.export?.defaultWidth ||\n      600,\n    scale\n  };\n};\n\n/**\n * Function for final options preparation before export.\n *\n * @param {object} options - All options object.\n * @param {object} chartJson - Chart JSON.\n * @param {function} endCallback - The end callback.\n * @param {string} svg - The SVG representation.\n */\nconst doExport = (options, chartJson, endCallback, svg) => {\n  let { export: exportOptions, customCode: customCodeOptions } = options;\n\n  const allowCodeExecutionScoped =\n    typeof customCodeOptions.allowCodeExecution === 'boolean'\n      ? customCodeOptions.allowCodeExecution\n      : allowCodeExecution;\n\n  if (!customCodeOptions) {\n    customCodeOptions = options.customCode = {};\n  } else if (typeof options.customCode.resources === 'string') {\n    // Process resources\n    options.customCode.resources = handleResources(\n      options.customCode.resources,\n      toBoolean(options.customCode.allowFileResources)\n    );\n  } else if (!options.customCode.resources) {\n    try {\n      const resources = readFileSync('resources.json', 'utf8');\n      options.customCode.resources = handleResources(\n        resources,\n        toBoolean(options.customCode.allowFileResources)\n      );\n    } catch (err) {\n      log(3, `[chart] The default resources.json file not found.`);\n    }\n  }\n\n  // If the allowCodeExecution flag isn't set, we should refuse the usage\n  // of callback, resources, and custom code. Additionally, the worker will\n  // refuse to run arbitrary JavaScript. Prioritized should be the scoped\n  // option, then we should take a look at the overall pool option.\n  if (!allowCodeExecutionScoped && customCodeOptions) {\n    if (\n      customCodeOptions.callback ||\n      customCodeOptions.resources ||\n      customCodeOptions.customCode\n    ) {\n      // Send back a friendly message saying that the exporter does not support\n      // these settings.\n      return (\n        endCallback &&\n        endCallback(false, {\n          error: true,\n          message: clearText(\n            `The callback, resources and customCode have been disabled for this\n            server.`\n          )\n        })\n      );\n    }\n\n    // Reset all additional custom code\n    customCodeOptions.callback = false;\n    customCodeOptions.resources = false;\n    customCodeOptions.customCode = false;\n  }\n\n  // Clean properties to keep it lean and mean\n  if (chartJson) {\n    chartJson.chart = chartJson.chart || {};\n    chartJson.exporting = chartJson.exporting || {};\n    chartJson.exporting.enabled = false;\n  }\n\n  exportOptions.constr = exportOptions.constr || 'chart';\n  exportOptions.type = fixType(exportOptions.type, exportOptions.outfile);\n  if (exportOptions.type === 'svg') {\n    exportOptions.width = false;\n  }\n\n  // Prepare global and theme options\n  ['globalOptions', 'themeOptions'].forEach((optionsName) => {\n    try {\n      if (exportOptions && exportOptions[optionsName]) {\n        if (\n          typeof exportOptions[optionsName] === 'string' &&\n          exportOptions[optionsName].endsWith('.json')\n        ) {\n          exportOptions[optionsName] = isCorrectJSON(\n            readFileSync(exportOptions[optionsName], 'utf8'),\n            true\n          );\n        } else {\n          exportOptions[optionsName] = isCorrectJSON(\n            exportOptions[optionsName],\n            true\n          );\n        }\n      }\n    } catch (error) {\n      exportOptions[optionsName] = {};\n      log(1, `[chart] The ${optionsName} not found.`);\n    }\n  });\n\n  // Prepare customCode\n  if (customCodeOptions.allowCodeExecution) {\n    customCodeOptions.customCode = wrapAround(\n      customCodeOptions.customCode,\n      customCodeOptions.allowFileResources\n    );\n  }\n\n  // Get the callback\n  if (\n    customCodeOptions &&\n    customCodeOptions.callback &&\n    customCodeOptions.callback?.indexOf('{') < 0\n  ) {\n    // The allowFileResources is always set to false for HTTP requests to avoid\n    // injecting arbitrary files from the fs\n    if (customCodeOptions.allowFileResources) {\n      try {\n        customCodeOptions.callback = readFileSync(\n          customCodeOptions.callback,\n          'utf8'\n        );\n      } catch (error) {\n        log(2, `[chart] Error loading callback: ${error}.`);\n        customCodeOptions.callback = false;\n      }\n    } else {\n      customCodeOptions.callback = false;\n    }\n  }\n\n  // Size search\n  options.export = {\n    ...options.export,\n    ...findChartSize(options)\n  };\n\n  // Post the work to the pool\n  postWork(exportOptions.strInj || chartJson || svg, options)\n    .then((result) => endCallback(result))\n    .catch((error) => {\n      log(0, '[chart] When posting work:', error);\n      return endCallback(false, error);\n    });\n};\n\n/**\n * Function for straight injecting the code.\n * Dangerous and must be used deliberately by someone who sets up a server\n * (see  --allowCodeExecution).\n *\n * @param {object} options - All options object.\n * @param {function} endCallback - The function to call when exporting is done.\n */\nconst doStraightInject = (options, endCallback) => {\n  try {\n    let strInj;\n    let instr = options.export.instr || options.export.options;\n\n    if (typeof instr !== 'string') {\n      // Try to stringify options\n      strInj = instr = optionsStringify(\n        instr,\n        options.customCode?.allowCodeExecution\n      );\n    }\n    strInj = instr.replaceAll(/\\t|\\n|\\r/g, '').trim();\n\n    // Get rid of the ;\n    if (strInj[strInj.length - 1] === ';') {\n      strInj = strInj.substring(0, strInj.length - 1);\n    }\n\n    // Save as stright inject string\n    options.export.strInj = strInj;\n    return doExport(options, false, endCallback);\n  } catch (error) {\n    const message = clearText(\n      `Malformed input detected for ${options.export?.requestId || '?'}:\n      Please make sure that your JSON/JavaScript options\n      are sent using the \"options\" attribute, and that if you're using\n      SVG, it is unescaped.`\n    );\n\n    log(1, message);\n    return (\n      endCallback &&\n      endCallback(\n        false,\n        JSON.stringify({\n          error: true,\n          message\n        })\n      )\n    );\n  }\n};\n\n/**\n * Prepares an input before exporting.\n *\n * @param {string} stringToExport - String representation of SVG/export options.\n * @param {object} options - All options object.\n * @param {function} endCallback - The function to call when exporting is done.\n */\nconst exportAsString = (stringToExport, options, endCallback) => {\n  const { allowCodeExecution } = options.customCode;\n\n  // Check if it is SVG\n  if (\n    stringToExport.indexOf('<svg') >= 0 ||\n    stringToExport.indexOf('<?xml') >= 0\n  ) {\n    log(4, '[chart] Parsing input as SVG.');\n    return doExport(options, false, endCallback, stringToExport);\n  }\n\n  try {\n    // Try to parse to JSON and call the doExport function\n    const chartJSON = JSON.parse(stringToExport.replaceAll(/\\t|\\n|\\r/g, ' '));\n\n    // If a correct JSON, do the export\n    return doExport(options, chartJSON, endCallback);\n  } catch (error) {\n    // Not a valid JSON\n    if (toBoolean(allowCodeExecution)) {\n      return doStraightInject(options, endCallback);\n    } else {\n      // Do not allow straight injection without the allowCodeExecution flag\n      return (\n        endCallback &&\n        endCallback(false, {\n          error: true,\n          message: clearText(\n            `Only JSON configurations and SVG is allowed for this server. If\n            this is your server, JavaScript exporting can be enabled by starting\n            the server with the --allowCodeExecution flag.`\n          )\n        })\n      );\n    }\n  }\n};\n\nexport const getAllowCodeExecution = () => allowCodeExecution;\n\nexport const setAllowCodeExecution = (value) => {\n  allowCodeExecution = toBoolean(value);\n};\n\n/**\n * Starts an exporting process\n *\n * @param {object} settings - Settings for export.\n * @param {function} endCallback - The function to call when exporting is done.\n */\nexport default {\n  batchExport,\n  singleExport,\n  getAllowCodeExecution,\n  setAllowCodeExecution,\n  startExport,\n  findChartSize\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { v4 as uuid } from 'uuid';\n\nimport { getAllowCodeExecution, startExport } from '../../chart.js';\nimport { getOptions, mergeConfigOptions } from '../../config.js';\nimport { log } from '../../logger.js';\nimport {\n  clearText,\n  fixType,\n  isCorrectJSON,\n  isPrivateRangeUrlFound,\n  optionsStringify,\n  measureTime\n} from '../../utils.js';\n\n// Reversed MIME types\nconst reversedMime = {\n  png: 'image/png',\n  jpeg: 'image/jpeg',\n  gif: 'image/gif',\n  pdf: 'application/pdf',\n  svg: 'image/svg+xml'\n};\n\n// The requests counter\nlet requestsCounter = 0;\n\nconst benchmark = false;\n\n// The array of callbacks to call before a request\nconst beforeRequest = [];\n\n// The array of callbacks to call after a request\nconst afterRequest = [];\n\n/**\n * Calls callbacks.\n *\n * @param {Array} callbacks - An array of callbacks.\n * @param {object} request - The request.\n * @param {object} response - The response.\n * @param {object} data - The data to send to callbacks.\n * @return {object} - The result from a callback.\n */\nconst doCallbacks = (callbacks, request, response, data) => {\n  let result = true;\n  const { id, uniqueId, type, body } = data;\n\n  callbacks.some((callback) => {\n    if (callback) {\n      let callResponse = callback(request, response, id, uniqueId, type, body);\n\n      if (callResponse !== undefined && callResponse !== true) {\n        result = callResponse;\n      }\n\n      return true;\n    }\n  });\n\n  return result;\n};\n\n/**\n * Handles an export.\n *\n * @param {object} request - The request.\n * @param {object} response - The response.\n */\nconst exportHandler = (request, response) => {\n  // Start counting time\n  const stopCounter = measureTime();\n\n  // Get the current server's general options\n  const defaultOptions = getOptions();\n\n  // Init default options\n  if (benchmark) {\n    console.log('Init default options:', stopCounter(), 'ms.');\n  }\n\n  const body = request.body;\n  const id = ++requestsCounter;\n  const uniqueId = uuid().replace(/-/g, '');\n  let type = fixType(body.type);\n\n  // Fix type\n  if (benchmark) {\n    console.log('Fix type:', stopCounter(), 'ms.');\n  }\n\n  // Throw 'Bad Request' if there's no body\n  if (!body) {\n    return response.status(400).send(\n      clearText(\n        `Body is required. Sending a body? Make sure your Content-type header\n        is correct. Accepted is application/json and multipart/form-data.`\n      )\n    );\n  }\n\n  // All of the below can be used\n  let instr = isCorrectJSON(body.infile || body.options || body.data);\n\n  // Is correct JSON\n  if (benchmark) {\n    console.log('Is correct JSON:', stopCounter(), 'ms.');\n  }\n\n  // Throw 'Bad Request' if there's no JSON or SVG to export\n  if (!instr && !body.svg) {\n    log(\n      2,\n      clearText(\n        `Request ${uniqueId} from ${\n          request.headers['x-forwarded-for'] || request.connection.remoteAddress\n        } was incorrect. Check your payload.`\n      )\n    );\n\n    return response.status(400).send(\n      clearText(\n        `No correct chart data found. Please make sure you are using\n        application/json or multipart/form-data headers, and that the chart\n        data is in the 'infile', 'options' or 'data' attribute if sending\n        JSON or in the 'svg' if sending SVG.`\n      )\n    );\n  }\n\n  let callResponse = false;\n\n  // Call the before request functions\n  callResponse = doCallbacks(beforeRequest, request, response, {\n    id,\n    uniqueId,\n    type,\n    body\n  });\n\n  // Do callbacks\n  if (benchmark) {\n    console.log('Do callbacks:', stopCounter(), 'ms.');\n  }\n\n  // Block the request if one of a callbacks failed\n  if (callResponse !== true) {\n    return response.send(callResponse);\n  }\n\n  let connectionAborted = false;\n\n  // In case the connection is closed, force to abort further actions\n  request.socket.on('close', () => {\n    connectionAborted = true;\n  });\n\n  log(4, `[export] Got an incoming HTTP request ${uniqueId}.`);\n\n  body.constr = (typeof body.constr === 'string' && body.constr) || 'chart';\n\n  // Gather and organize options from the payload\n  const requestOptions = {\n    export: {\n      instr,\n      type,\n      constr: body.constr[0].toLowerCase() + body.constr.substr(1),\n      height: body.height,\n      width: body.width,\n      scale: body.scale || defaultOptions.export.scale,\n      globalOptions: isCorrectJSON(body.globalOptions, true),\n      themeOptions: isCorrectJSON(body.themeOptions, true)\n    },\n    customCode: {\n      allowCodeExecution: getAllowCodeExecution(),\n      allowFileResources: false,\n      resources: isCorrectJSON(body.resources, true),\n      callback: body.callback,\n      customCode: body.customCode\n    }\n  };\n\n  // Organize options\n  if (benchmark) {\n    console.log('Organize options:', stopCounter(), 'ms.');\n  }\n\n  if (instr) {\n    // Stringify JSON with options\n    requestOptions.export.instr = optionsStringify(\n      instr,\n      requestOptions.customCode.allowCodeExecution\n    );\n\n    // Stringify JSON with options\n    if (benchmark) {\n      console.log('Stringify JSON with options:', stopCounter(), 'ms.');\n    }\n  }\n\n  // Merge the request options into default ones\n  const options = mergeConfigOptions(defaultOptions, requestOptions);\n\n  // Merge config options\n  if (benchmark) {\n    console.log('Merge config options:', stopCounter(), 'ms.');\n  }\n\n  // Save the JSON if exists\n  options.export.options = instr;\n\n  // Lastly, add the server specific arguments into options as payload\n  options.payload = {\n    svg: body.svg || false,\n    b64: body.b64 || false,\n    dataOptions: isCorrectJSON(body.dataOptions, true),\n    noDownload: body.noDownload || false,\n    requestId: uniqueId\n  };\n\n  // Setting payload\n  if (benchmark) {\n    console.log('Setting payload:', stopCounter(), 'ms.');\n  }\n\n  // Test xlink:href elements from payload's SVG\n  if (body.svg && isPrivateRangeUrlFound(options.payload.svg)) {\n    return response\n      .status(400)\n      .send(\n        'SVG potentially contain at least one forbidden URL in xlink:href element.'\n      );\n  }\n\n  // Check URL range\n  if (benchmark) {\n    console.log('Check URL range:', stopCounter(), 'ms.');\n  }\n\n  // Start the export process\n  startExport(options, (info, error) => {\n    // Remove the close event from the socket\n    request.socket.removeAllListeners('close');\n\n    // After Puppeteer exporting\n    if (benchmark) {\n      console.log('After Puppeteer exporting:', stopCounter(), 'ms.', '\\n');\n    }\n\n    // If the connection was closed, do nothing\n    if (connectionAborted) {\n      return log(\n        3,\n        clearText(\n          `[export] The client closed the connection before the chart was done\n          processing.`\n        )\n      );\n    }\n\n    // If error, return it\n    if (error) {\n      log(\n        1,\n        clearText(\n          `[export] Work: ${uniqueId} could not be completed, sending:\n          ${error}`\n        )\n      );\n      return response.status(400).send(error.message);\n    }\n\n    // If data is missing, return the error\n    if (!info || !info.data) {\n      log(\n        1,\n        clearText(\n          `[export] Unexpected return from chart generation, please check your\n          data Request: ${uniqueId} is ${info.data}.`\n        )\n      );\n      return response\n        .status(400)\n        .send(\n          'Unexpected return from chart generation, please check your data.'\n        );\n    }\n\n    // Get the type from options\n    type = info.options.export.type;\n\n    // The after request callbacks\n    doCallbacks(afterRequest, request, response, { id, body: info.data });\n\n    if (info.data) {\n      // If only base64 is required, return it\n      if (body.b64) {\n        // Check if it is already base64 or a raw SVG\n        if (type === 'pdf') {\n          return response.send(\n            Buffer.from(info.data, 'utf8').toString('base64')\n          );\n        }\n        return response.send(info.data);\n      }\n\n      // Set correct content type\n      response.header('Content-Type', reversedMime[type] || 'image/png');\n\n      // Decide whether to download or not chart file\n      if (!body.noDownload) {\n        response.attachment(\n          `${request.params.filename || 'chart'}.${type || 'png'}`\n        );\n      }\n\n      // If SVG, return plain content\n      return type === 'svg'\n        ? response.send(info.data)\n        : response.send(Buffer.from(info.data, 'base64'));\n    }\n  });\n};\n\nexport default (app) => {\n  app.post('/', exportHandler);\n  app.post('/:filename', exportHandler);\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { promises as fsPromises } from 'fs';\nimport { posix } from 'path';\n\nimport bodyParser from 'body-parser';\nimport cors from 'cors';\nimport express from 'express';\nimport multer from 'multer';\nimport http from 'http';\nimport https from 'https';\n\nimport { log } from '../logger.js';\nimport rateLimit from './rate_limit.js';\nimport { __dirname } from '../utils.js';\n\nimport healthRoute from './routes/health.js';\nimport exportRoutes from './routes/export.js';\nimport vswitchRoute from './routes/change_hc_version.js';\nimport uiRoute from './routes/ui.js';\n\n// Create express app\nconst app = express();\n\n// Disable the X-Powered-By header\napp.disable('x-powered-by');\n\n// Enable CORS support\napp.use(cors());\n\n// Enable parsing of form data (files) with Multer package\nconst storage = multer.memoryStorage();\nconst upload = multer({\n  storage,\n  limits: {\n    fieldsSize: '50MB'\n  }\n});\n\napp.use(upload.any());\n\n// Enable body parser\napp.use(bodyParser.json({ limit: '50mb' }));\napp.use(bodyParser.urlencoded({ extended: true, limit: '50mb' }));\napp.use(bodyParser.urlencoded({ extended: false, limit: '50mb' }));\n\n/**\n * Error handler function.\n *\n * @param {object} error - An error object.\n * @return {string} - An error message.\n */\nconst errorHandler = (error) => log(1, `[server] Socket error: ${error}`);\n\n/**\n * Attaches error handlers for a server.\n *\n * @param {object} server - The http/https server.\n */\nconst attachErrorHandlers = (server) => {\n  server.on('clientError', errorHandler);\n  server.on('error', errorHandler);\n  server.on('connection', (socket) =>\n    socket.on('error', (error) => errorHandler(error, socket))\n  );\n};\n\nexport const startServer = async (serverConfig) => {\n  // Stop if not enabled\n  if (!serverConfig.enable) {\n    return false;\n  }\n\n  // // Get the pool\n  // const pool = getPool();\n\n  // // Try to create browser instance before starting the server\n  // const resource = await pool.acquire();\n\n  // // If not found, throw an error\n  // if (!resource.browser) {\n  //   log(1, `[server] Could not acquire browser instance.`);\n  //   process.exit(1);\n  // }\n\n  // // Release the resource\n  // pool.release(resource);\n\n  // Listen HTTP server\n  if (!serverConfig.ssl.enable && !serverConfig.ssl.force) {\n    // Main server instance (HTTP)\n    const httpServer = http.createServer(app);\n    // Attach error handlers and listen to the server\n    attachErrorHandlers(httpServer);\n    // Listen\n    httpServer.listen(serverConfig.port, serverConfig.host);\n\n    log(\n      3,\n      `[server] Started HTTP server on ${serverConfig.host}:${serverConfig.port}.`\n    );\n  }\n\n  // Listen HTTPS server\n  if (serverConfig.ssl.enable) {\n    // Set up an SSL server also\n    let key, cert;\n\n    try {\n      // Get the SSL key\n      key = await fsPromises.readFile(\n        posix.join(serverConfig.ssl.certPath, 'server.key'),\n        'utf8'\n      );\n\n      // Get the SSL certificate\n      cert = await fsPromises.readFile(\n        posix.join(serverConfig.ssl.certPath, 'server.crt'),\n        'utf8'\n      );\n    } catch (error) {\n      log(\n        1,\n        `[server] Unable to load key/certificate from ${serverConfig.ssl.certPath}.`\n      );\n    }\n\n    if (key && cert) {\n      // Main server instance (HTTPS)\n      const httpsServer = https.createServer(app);\n      // Attach error handlers and listen to the server\n      attachErrorHandlers(httpsServer);\n      // Listen\n      httpsServer.listen(serverConfig.ssl.port, serverConfig.host);\n\n      log(\n        3,\n        `[server] Started HTTPS server on ${serverConfig.host}:${serverConfig.ssl.port}.`\n      );\n    }\n  }\n\n  // Enable the rate limiter if config says so\n  if (\n    serverConfig.rateLimiting &&\n    serverConfig.rateLimiting.enable &&\n    ![0, NaN].includes(serverConfig.rateLimiting.maxRequests)\n  ) {\n    rateLimit(app, serverConfig.rateLimiting);\n  }\n\n  // Set up static folder's route\n  app.use(express.static(posix.join(__dirname, 'public')));\n\n  // Set up routes\n  healthRoute(app);\n  exportRoutes(app);\n  uiRoute(app);\n  vswitchRoute(app);\n};\n\n/**\n * Returns the express instance.\n */\nexport const getExpress = () => {\n  return express;\n};\n\n/**\n * Returns the app instance.\n */\nexport const getApp = () => {\n  return app;\n};\n\n/**\n * Adds a middleware to the server.\n *\n * @param {object} path - An endpoint path to add middlewares to.\n * @param {Array} middlewares - An unlimited number of middlewares to use\n * against the specific endpoint.\n */\nexport const use = (path, ...middlewares) => {\n  app.use(path, ...middlewares);\n};\n\n/**\n * Adds a get route to the server.\n *\n * @param {object} path - An endpoint path to add middlewares to.\n * @param {Array} middlewares - An unlimited number of middlewares to use\n * against the specific endpoint for GET method.\n */\nexport const get = (path, ...middlewares) => {\n  app.get(path, ...middlewares);\n};\n\n/**\n * Adds a post route to the server.\n *\n * @param {object} path - An endpoint path to add middlewares to.\n * @param {Array} middlewares - An unlimited number of middlewares to use\n * against the specific endpoint for POST method.\n */\nexport const post = (path, ...middlewares) => {\n  app.post(path, ...middlewares);\n};\n\n/**\n * Forcefully enables rate limiting.\n *\n * @param {object} limitConfig - The options object for the rate limiter\n * configuration.\n */\nexport const enableRateLimiting = (limitConfig) => {\n  return rateLimit(app, limitConfig);\n};\n\nexport default {\n  startServer,\n  getExpress,\n  getApp,\n  use,\n  get,\n  post,\n  enableRateLimiting\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { join } from 'path';\n\nimport { __dirname } from '../../utils.js';\n/**\n * Adds the / route for a UI when enabled for the export server\n */\nexport default (app) =>\n  !app\n    ? false\n    : app.get('/', (request, response) => {\n        response.sendFile(join(__dirname, 'public', 'index.html'));\n      });\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport cache from '../../cache.js';\n\n/**\n * Adds a route that can be used to change the HC version on the server\n * TODO: Add auth token and connect to API\n */\nexport default (app) =>\n  !app\n    ? false\n    : app.post('/change_hc_version/:newVersion', async (request, response) => {\n        const ctoken = process.env.HIGHCHARTS_ADMIN_TOKEN;\n\n        if (!ctoken || !ctoken.length) {\n          return response.send({\n            error: true,\n            message:\n              'Server not configured to do run-time version changes: HIGHCHARTS_ADMIN_TOKEN not set'\n          });\n        }\n\n        const token = request.get('hc-auth');\n\n        if (!token || token !== ctoken) {\n          return response.send({\n            error: true,\n            message: 'Invalid or missing token: set token in the hc-auth header'\n          });\n        }\n\n        const newVersion = request.params.newVersion;\n\n        if (newVersion) {\n          try {\n            // eslint-disable-next-line import/no-named-as-default-member\n            await cache.updateVersion(newVersion);\n          } catch (e) {\n            response.send({\n              error: true,\n              message: e\n            });\n          }\n\n          response.send({\n            version: cache.version()\n          });\n        } else {\n          response.send({\n            error: true,\n            message: 'No new version supplied'\n          });\n        }\n      });\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\n// Add the main directory in the global object\nimport 'colors';\n\nimport server, { startServer } from './server/server.js';\nimport {\n  setAllowCodeExecution,\n  batchExport,\n  singleExport,\n  startExport\n} from './chart.js';\nimport { mapToNewConfig, setOptions } from './config.js';\nimport { log, setLogLevel, enableFileLogging } from './logger.js';\nimport { killPool, init } from './pool.js';\nimport { checkCache } from './cache.js';\n\nexport default {\n  log,\n  mapToNewConfig,\n  setOptions,\n  singleExport,\n  startExport,\n  batchExport,\n  server,\n  startServer,\n  killPool,\n  initPool: async (options = {}) => {\n    // Set the allowCodeExecution per export module scope\n    setAllowCodeExecution(\n      options.customCode && options.customCode.allowCodeExecution\n    );\n\n    // Set the log level\n    setLogLevel(options.logging && parseInt(options.logging.level));\n\n    // Set the log file path and name\n    if (options.logging && options.logging.dest) {\n      enableFileLogging(\n        options.logging.dest,\n        options.logging.file || 'highcharts-export-server.log'\n      );\n    }\n\n    // Check if cache needs to be updated\n    await checkCache(options.highcharts || { version: 'latest' });\n\n    // Init the pool\n    await init({\n      pool: options.pool || {\n        initialWorkers: 1,\n        maxWorkers: 1\n      },\n      puppeteerArgs: options.puppeteer?.args || []\n    });\n\n    // Return updated options\n    return options;\n  }\n};\n"],"names":["dotenv","config","defaultConfig","puppeteer","args","value","type","description","highcharts","version","envLink","cdnURL","coreScripts","modules","indicators","scripts","forceFetch","export","infile","instr","options","outfile","constr","defaultHeight","defaultWidth","defaultScale","height","width","scale","globalOptions","themeOptions","batch","customCode","allowCodeExecution","allowFileResources","callback","resources","loadConfig","createConfig","server","enable","cliName","host","port","ssl","force","certPath","rateLimiting","maxRequests","window","delay","trustProxy","skipKey","skipToken","pool","initialWorkers","maxWorkers","workLimit","queueSize","timeoutThreshold","acquireTimeout","reaper","benchmarking","listenToProcessExits","logging","level","file","dest","ui","route","other","noLogo","payload","join","absoluteProps","nestedArgs","createNestedArgs","obj","propChain","Object","keys","forEach","k","includes","entry","substring","toConsole","toFile","pathCreated","levelsDesc","title","color","listeners","key","option","entries","log","newLevel","texts","length","prefix","Date","toString","split","trim","fn","existsSync","mkdirSync","appendFile","concat","error","console","apply","undefined","__dirname","fileURLToPath","URL","document","require","pathToFileURL","__filename","href","_documentCurrentScript","src","baseURI","clearText","text","rule","replacer","replaceAll","fixType","formats","outType","pop","find","t","handleResources","allowedProps","handledResources","correctResources","endsWith","isCorrectJSON","readFileSync","notice","files","propName","map","item","data","parsedData","JSON","parse","stringify","deepCopy","copy","Array","isArray","prototype","hasOwnProperty","call","optionsStringify","allowFunctions","name","startsWith","printUsage","bold","yellow","cycleCategories","categories","descName","green","i","blue","category","toUpperCase","red","toBoolean","wrapAround","replace","rateLimit","app","limitConfig","msg","rateOptions","max","limiter","windowMs","delayMs","handler","request","response","format","json","status","send","message","default","skip","query","access_token","use","async","fetch","url","requestOptions","Promise","resolve","reject","protocol","https","http","getProtocol","get","res","on","chunk","cachePath","cache","activeManifest","sources","hcVersion","appliedConfig","extractVersion","substr","indexOf","fetchScript","script","proxyAgent","agent","timeout","process","env","statusCode","updateCache","sourcePath","customScripts","allScripts","c","m","proxyHost","proxyPort","HttpsProxyAgent","fetchedModules","all","writeFileSync","checkCache","manifestPath","requestUpdate","manifest","moduleMap","numberOfModules","some","moduleName","newManifest","saveConfigToManifest","cache$1","newVersion","assign","RANDOM_PID","randomBytes","PUPPETEER_DIR","path","minimalArgs","template","fs","browser","newPage","p","setContent","addScriptTag","evaluate","setupHighcharts","err","$eval","element","errorMessage","_displayErrors","innerHTML","close","connected","__basedir","setAsConfig","page","chart","triggerExport","puppeteerExport","injectedResources","clearInjected","dispose","scriptsToRemove","getElementsByTagName","stylesToRemove","linksToRemove","remove","exportBench","exportOptions","requestAnimationFrame","displayErrors","debugger","d","svgBench","isSVG","setPageBench","svgTemplate","strInj","setContentBench","resBench","js","push","content","isLocal","cssBench","css","cssImports","match","cssImportPath","addStyleTag","size","chartHeight","baseVal","chartWidth","parseFloat","Highcharts","charts","vpBench","viewportHeight","Math","ceil","viewportWidth","setViewport","deviceScaleFactor","zoomCallback","body","style","zoom","margin","x","y","getBoundingClientRect","trunc","getClipRegion","round","expBenchmark","outerHTML","createSVG","encoding","clip","race","screenshot","omitBackground","setTimeout","Error","createImage","pdf","createPDF","oldCharts","oldChart","destroy","shift","puppeteerArgs","performedExports","exportAttempts","timeSpent","droppedExports","spentAverage","poolConfig","factory","create","id","uuid","s","getTime","browserNewPage","isClosed","workCount","random","validate","workerHandle","logLevel","init","allArgs","tryCount","open","launch","headless","userDataDir","e","createBrowser","killPool","code","exit","Pool","min","createRetryIntervalMillis","createTimeoutMillis","acquireTimeoutMillis","destroyTimeoutMillis","idleTimeoutMillis","reapIntervalMillis","propagateCreateError","eventId","resource","initialResources","acquire","promise","release","destroyed","postWork","fail","getPoolInfo","workStart","result","exportTime","available","borrowed","pending","spareResourceCapacity","pool$1","packageVersion","npm_package_version","serverStartTime","generalOptions","getOptions","mergeConfigOptions","newOptions","mergedOptions","updateDefaultConfig","configObj","customObj","customValue","numEnvVal","el","initOptions","items","startExport","settings","endCallback","svg","initExportSettings","exportAsString","readFile","doStraightInject","doExport","findChartSize","exporting","precision","multiplier","pow","roundNumber","sourceHeight","sourceWidth","chartJson","customCodeOptions","allowCodeExecutionScoped","enabled","optionsName","then","catch","requestId","stringToExport","chartJSON","reversedMime","png","jpeg","gif","requestsCounter","beforeRequest","afterRequest","doCallbacks","callbacks","uniqueId","callResponse","exportHandler","start","hrtime","bigint","measureTime","defaultOptions","headers","connection","remoteAddress","connectionAborted","socket","toLowerCase","b64","dataOptions","noDownload","ipRegEx","info","removeAllListeners","Buffer","from","header","attachment","params","filename","express","disable","cors","storage","multer","memoryStorage","upload","limits","fieldsSize","any","bodyParser","limit","urlencoded","extended","errorHandler","attachErrorHandlers","startServer","serverConfig","httpServer","createServer","listen","cert","fsPromises","posix","httpsServer","NaN","static","bootTime","uptime","floor","highchartsVersion","averageProcessingTime","failedExports","sucessRatio","healthRoute","post","exportRoutes","sendFile","uiRoute","ctoken","HIGHCHARTS_ADMIN_TOKEN","token","vswitchRoute","getExpress","getApp","middlewares","enableRateLimiting","index","mapToNewConfig","oldOptions","propertiesChain","reduce","prop","setOptions","userOptions","configIndex","findIndex","arg","fileName","loadConfigFile","pairArgumentValue","singleExport","batchExport","batchFunctions","pair","initPool","parseInt","logDest","logFile","enableFileLogging"],"mappings":"6uBAiBAA,EAAOC,SAIA,MAAMC,EAAgB,CAC3BC,UAAW,CACTC,KAAM,CACJC,MAAO,GACPC,KAAM,WACNC,YAAa,6CAGjBC,WAAY,CACVC,QAAS,CACPJ,MAAO,SACPK,QAAS,qBACTJ,KAAM,SACNC,YAAa,8BAEfI,OAAQ,CACNN,MAAO,+BACPK,QAAS,iBACTJ,KAAM,SACNC,YAAa,6CAEfK,YAAa,CACXF,QAAS,0BACTL,MAAO,CAAC,aAAc,kBAAmB,iBACzCC,KAAM,WACNC,YAAa,qCAEfM,QAAS,CACPH,QAAS,qBACTL,MAAO,CACL,QACA,MACA,QACA,YACA,cACA,uBACA,gBACA,uBACA,eACA,QACA,OACA,mBACA,eACA,cACA,UACA,UACA,WACA,UACA,YACA,cACA,YACA,sBACA,SACA,SACA,WACA,YACA,eACA,SACA,eACA,YACA,kBACA,SACA,cACA,mBACA,eACA,cACA,eACA,cACA,cACA,WACA,eACA,WACA,SACA,OACA,WACA,YACA,SACA,qBACA,aACA,WACA,WACA,WACA,WACA,eACA,UACA,kBACA,oBACA,cAEFC,KAAM,WACNC,YAAa,gCAEfO,WAAY,CACVJ,QAAS,wBACTL,MAAO,CAAC,kBACRC,KAAM,WACNC,YAAa,mCAEfQ,QAAS,CACPV,MAAO,CACL,wEACA,kGAEFC,KAAM,WACNC,YACE,qEAEJS,WAAY,CACVN,QAAS,yBACTL,OAAO,EACPC,KAAM,UACNC,YACE,oEAGNU,OAAQ,CACNC,OAAQ,CACNb,OAAO,EACPC,KAAM,SACNC,YACE,8FAEJY,MAAO,CACLd,OAAO,EACPC,KAAM,SACNC,YACE,iFAEJa,QAAS,CACPf,OAAO,EACPC,KAAM,SACNC,YAAa,oCAEfc,QAAS,CACPhB,OAAO,EACPC,KAAM,SACNC,YACE,2FAEJD,KAAM,CACJI,QAAS,sBACTL,MAAO,MACPC,KAAM,SACNC,YACE,sEAEJe,OAAQ,CACNZ,QAAS,wBACTL,MAAO,QACPC,KAAM,SACNC,YACE,6EAEJgB,cAAe,CACbb,QAAS,wBACTL,MAAO,IACPC,KAAM,SACNC,YACE,gFAEJiB,aAA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"} diff --git a/dist/index.esm.js b/dist/index.esm.js index 5687f6d0..d17d1a14 100644 --- a/dist/index.esm.js +++ b/dist/index.esm.js @@ -1,2 +1,2 @@ -import"colors";import e,{existsSync as t,mkdirSync as o,appendFile as r,readFileSync as i,writeFileSync as n,readFile as s,promises as a}from"fs";import l,{join as c,posix as p}from"path";import u from"body-parser";import d from"cors";import h from"express";import g from"multer";import m from"http";import f from"https";import v from"dotenv";import y from"express-rate-limit";import*as b from"url";import{fileURLToPath as w}from"url";import T from"https-proxy-agent";import{v4 as x}from"uuid";import{Pool as k}from"tarn";import S from"puppeteer";import H from"node:path";import{randomBytes as E}from"node:crypto";import"prompts";v.config();const R={puppeteer:{args:{value:[],type:"string[]",description:"Array of arguments to send to puppeteer."}},highcharts:{version:{value:"latest",envLink:"HIGHCHARTS_VERSION",type:"string",description:"Highcharts version to use."},cdnURL:{value:"https://code.highcharts.com/",envLink:"HIGHCHARTS_CDN",type:"string",description:"The CDN URL of Highcharts scripts to use."},coreScripts:{envLink:"HIGHCHARTS_CORE_SCRIPTS",value:["highcharts","highcharts-more","highcharts-3d"],type:"string[]",description:"Highcharts core scripts to fetch."},modules:{envLink:"HIGHCHARTS_MODULES",value:["stock","map","gantt","exporting","export-data","parallel-coordinates","accessibility","annotations-advanced","boost-canvas","boost","data","draggable-points","static-scale","broken-axis","heatmap","tilemap","timeline","treemap","item-series","drilldown","histogram-bellcurve","bullet","funnel","funnel3d","pyramid3d","networkgraph","pareto","pattern-fill","pictorial","price-indicator","sankey","arc-diagram","dependency-wheel","series-label","solid-gauge","sonification","stock-tools","streamgraph","sunburst","variable-pie","variwide","vector","venn","windbarb","wordcloud","xrange","no-data-to-display","drag-panes","debugger","dumbbell","lollipop","cylinder","organization","dotplot","marker-clusters","hollowcandlestick","heikinashi"],type:"string[]",description:"Highcharts modules to fetch."},indicators:{envLink:"HIGHCHARTS_INDICATORS",value:["indicators-all"],type:"string[]",description:"Highcharts indicators to fetch."},scripts:{value:["https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.29.4/moment.min.js","https://cdnjs.cloudflare.com/ajax/libs/moment-timezone/0.5.34/moment-timezone-with-data.min.js"],type:"string[]",description:"Additional direct scripts/optional dependencies (e.g. moment.js)."},forceFetch:{envLink:"HIGHCHARTS_FORCE_FETCH",value:!1,type:"boolean",description:"Should all the scripts be refetched after rerunning the server."}},export:{infile:{value:!1,type:"string",description:"The input file name along with a type (json or svg). It can be a correct JSON or SVG file."},instr:{value:!1,type:"string",description:"An input in a form of a stringified JSON or SVG file. Overrides the --infile."},options:{value:!1,type:"string",description:"An alias for the --instr option."},outfile:{value:!1,type:"string",description:"The output filename along with a type (jpeg, png, pdf or svg). Ignores the --type flag."},type:{envLink:"EXPORT_DEFAULT_TYPE",value:"png",type:"string",description:"The format of the file to export to. Can be jpeg, png, pdf or svg."},constr:{envLink:"EXPORT_DEFAULT_CONSTR",value:"chart",type:"string",description:"The constructor to use. Can be chart, stockChart, mapChart or ganttChart."},defaultHeight:{envLink:"EXPORT_DEFAULT_HEIGHT",value:400,type:"number",description:"The default height of the exported chart. Used when not found any value set."},defaultWidth:{envLink:"EXPORT_DEFAULT_WIDTH",value:600,type:"number",description:"The default width of the exported chart. Used when not found any value set."},defaultScale:{envLink:"EXPORT_DEFAULT_SCALE",value:1,type:"number",description:"The default scale of the exported chart. Ranges between 1 and 5."},height:{type:"number",value:!1,description:"The default height of the exported chart. Overrides the option in the chart settings."},width:{type:"number",value:!1,description:"The width of the exported chart. Overrides the option in the chart settings."},scale:{value:!1,type:"number",description:"The scale of the exported chart. Ranges between 1 and 5."},globalOptions:{value:!1,type:"string",description:"A stringified JSON or a filename with options to be passed into the Highcharts.setOptions."},themeOptions:{value:!1,type:"string",description:"A stringified JSON or a filename with theme options to be passed into the Highcharts.setOptions."},batch:{value:!1,type:"string",description:'Starts a batch job. A string that contains input/output pairs: "in=out;in=out;..".'}},customCode:{allowCodeExecution:{envLink:"HIGHCHARTS_ALLOW_CODE_EXECUTION",value:!1,type:"boolean",description:"If set to true, allow for the execution of arbitrary code when exporting."},allowFileResources:{envLink:"HIGHCHARTS_ALLOW_FILE_RESOURCES",value:!0,type:"boolean",description:"Allow injecting resources from the filesystem. Has no effect when running as a server."},customCode:{value:!1,type:"string",description:"A function to be called before chart initialization. Can be a filename with the js extension."},callback:{value:!1,type:"string",description:"A JavaScript file with a function to run on construction."},resources:{value:!1,type:"string",description:"An additional resource in a form of stringified JSON. It can contain files, js and css sections."},loadConfig:{value:!1,type:"string",description:"A file that contains a pre-defined config to use."},createConfig:{value:!1,type:"string",description:"Allows to set options through a prompt and save in a provided config file."}},server:{enable:{envLink:"HIGHCHARTS_SERVER_ENABLE",value:!1,type:"boolean",cliName:"enableServer",description:"If set to true, starts a server on 0.0.0.0."},host:{envLink:"HIGHCHARTS_SERVER_HOST",value:"0.0.0.0",type:"string",description:"The hostname of the server. Also starts a server listening on the supplied hostname."},port:{envLink:"HIGHCHARTS_SERVER_PORT",value:7801,type:"number",description:"The port to use for the server. Defaults to 7801."},ssl:{enable:{envLink:"HIGHCHARTS_SERVER_SSL_ENABLE",value:!1,type:"boolean",cliName:"enableSsl",description:"Enables the SSL protocol."},force:{envLink:"HIGHCHARTS_SERVER_SSL_FORCE",value:!1,type:"boolean",cliName:"sslForced",description:"If set to true, forces the server to only serve over HTTPS."},port:{envLink:"HIGHCHARTS_SERVER_SSL_PORT",value:443,type:"number",cliName:"sslPort",description:"The port on which to run the SSL server."},certPath:{envLink:"HIGHCHARTS_SSL_CERT_PATH",value:"",type:"string",description:"The path to the SSL certificate/key."}},rateLimiting:{enable:{envLink:"HIGHCHARTS_RATE_LIMIT_ENABLE",value:!1,type:"boolean",cliName:"enableRateLimiting",description:"Enables rate limiting."},maxRequests:{envLink:"HIGHCHARTS_RATE_LIMIT_MAX",value:10,type:"number",description:"Max requests allowed in a one minute."},window:{envLink:"HIGHCHARTS_RATE_LIMIT_WINDOW",value:1,type:"number",description:"The time window in minutes for rate limiting."},delay:{envLink:"HIGHCHARTS_RATE_LIMIT_DELAY",value:0,type:"number",description:"The amount to delay each successive request before hitting the max."},trustProxy:{envLink:"HIGHCHARTS_RATE_LIMIT_TRUST_PROXY",value:!1,type:"boolean",description:"Set this to true if behind a load balancer."},skipKey:{envLink:"HIGHCHARTS_RATE_LIMIT_SKIP_KEY",value:"",type:"number|string",description:"Allows bypassing the rate limiter and should be provided with skipToken argument."},skipToken:{envLink:"HIGHCHARTS_RATE_LIMIT_SKIP_TOKEN",value:"",type:"number|string",description:"Allows bypassing the rate limiter and should be provided with skipKey argument."}}},pool:{initialWorkers:{envLink:"HIGHCHARTS_POOL_MIN_WORKERS",value:4,type:"number",description:"The number of initial workers to spawn."},maxWorkers:{envLink:"HIGHCHARTS_POOL_MAX_WORKERS",value:8,type:"number",description:"The number of max workers to spawn."},workLimit:{envLink:"HIGHCHARTS_POOL_WORK_LIMIT",value:40,type:"number",description:"The pieces of work that can be performed before restarting process."},queueSize:{envLink:"HIGHCHARTS_POOL_QUEUE_SIZE",value:5,type:"number",description:"The size of the request overflow queue."},timeoutThreshold:{envLink:"HIGHCHARTS_POOL_TIMEOUT",value:5e3,type:"number",description:"The number of milliseconds before timing out."},acquireTimeout:{envLink:"HIGHCHARTS_POOL_ACQUIRE_TIMEOUT",value:5e3,type:"number",description:"The number of milliseconds to wait for acquiring a resource."},reaper:{envLink:"HIGHCHARTS_POOL_ENABLE_REAPER",value:!0,type:"boolean",description:"Whether or not to evict workers after a certain time period."},benchmarking:{envLink:"HIGHCHARTS_POOL_BENCHMARKING",value:!1,type:"boolean",description:"Enable benchmarking."},listenToProcessExits:{envLink:"HIGHCHARTS_POOL_LISTEN_TO_PROCESS_EXITS",value:!0,type:"boolean",description:"Set to false in order to skip attaching process.exit handlers."}},logging:{level:{envLink:"HIGHCHARTS_LOG_LEVEL",value:4,type:"number",cliName:"logLevel",description:"The log level (0: silent, 1: error, 2: warning, 3: notice, 4: verbose)."},file:{envLink:"HIGHCHARTS_LOG_FILE",value:"highcharts-export-server.log",type:"string",cliName:"logFile",description:"A name of a log file. The --logDest also needs to be set to enable file logging."},dest:{envLink:"HIGHCHARTS_LOG_DEST",value:"log/",type:"string",cliName:"logDest",description:"The path to store log files. Also enables file logging."}},ui:{enable:{envLink:"HIGHCHARTS_UI_ENABLE",value:!1,type:"boolean",cliName:"enableUi",description:"Enables the UI for the export server."},route:{envLink:"HIGHCHARTS_UI_ROUTE",value:"/",type:"string",cliName:"uiRoute",description:"The route to attach the UI to."}},other:{noLogo:{envLink:"HIGHCHARTS_NO_LOGO",value:!1,type:"boolean",description:"Skip printing the logo on a startup. Will be replaced by a simple text."}},payload:{}};R.puppeteer.args.value.join(","),R.highcharts.version.value,R.highcharts.cdnURL.value,R.highcharts.modules.value,R.highcharts.scripts.value.join(","),R.highcharts.forceFetch.value,R.export.type.value,R.export.constr.value,R.export.defaultHeight.value,R.export.defaultWidth.value,R.export.defaultScale.value,R.customCode.allowCodeExecution.value,R.customCode.allowFileResources.value,R.server.enable.value,R.server.host.value,R.server.port.value,R.server.ssl.enable.value,R.server.ssl.force.value,R.server.ssl.port.value,R.server.ssl.certPath.value,R.server.rateLimiting.enable.value,R.server.rateLimiting.maxRequests.value,R.server.rateLimiting.window.value,R.server.rateLimiting.delay.value,R.server.rateLimiting.trustProxy.value,R.server.rateLimiting.skipKey.value,R.server.rateLimiting.skipToken.value,R.pool.initialWorkers.value,R.pool.maxWorkers.value,R.pool.workLimit.value,R.pool.queueSize.value,R.pool.timeoutThreshold.value,R.pool.acquireTimeout.value,R.pool.reaper.value,R.pool.benchmarking.value,R.pool.listenToProcessExits.value,R.logging.level.value,R.logging.file.value,R.logging.dest.value,R.ui.enable.value,R.ui.route.value,R.other.noLogo.value;const L=["options","globalOptions","themeOptions","resources","payload"],C={},O=(e,t="")=>{Object.keys(e).forEach((o=>{if(!["puppeteer","highcharts"].includes(o)){const r=e[o];void 0===r.value?O(r,`${t}.${o}`):C[r.cliName||o]=`${t}.${o}`.substring(1)}}))};O(R);let _={toConsole:!0,toFile:!1,pathCreated:!1,levelsDesc:[{title:"error",color:"red"},{title:"warning",color:"yellow"},{title:"notice",color:"blue"},{title:"verbose",color:"gray"}],listeners:[]};for(const[e,t]of Object.entries(R.logging))_[e]=t.value;const A=(...e)=>{const[i,...n]=e,{level:s,levelsDesc:a}=_;if(0===i||i>s||s>a.length)return;const l=`${(new Date).toString().split("(")[0].trim()} [${a[i-1].title}] -`;_.listeners.forEach((e=>{e(l,n.join(" "))})),_.toFile&&(_.pathCreated||(!t(_.dest)&&o(_.dest),_.pathCreated=!0),r(`${_.dest}${_.file}`,[l].concat(n).join(" ")+"\n",(e=>{e&&(console.log(`[logger] Unable to write to log file: ${e}`),_.toFile=!1)}))),_.toConsole&&console.log.apply(void 0,[l.toString()[_.levelsDesc[i-1].color]].concat(n))},$=w(new URL("../.",import.meta.url)),I=(e,t=/\s\s+/g,o=" ")=>e.replaceAll(t,o).trim(),P=(e,t)=>{const o=["png","jpeg","pdf","svg"];if(t){const r=t.split(".").pop();o.includes(r)&&e!==r&&(e=r)}return{"image/png":"png","image/jpeg":"jpeg","application/pdf":"pdf","image/svg+xml":"svg"}[e]||o.find((t=>t===e))||"png"},j=(e=!1,t)=>{const o=["js","css","files"];let r=e,n=!1;if(t&&e.endsWith(".json"))try{e?e&&e.endsWith(".json")?r=N(i(e,"utf8")):(r=N(e),!0===r&&(r=N(i("resources.json","utf8")))):r=N(i("resources.json","utf8"))}catch(e){return A(3,"[cli] No resources found.")}else r=N(e),t||delete r.files;for(const e in r)o.includes(e)?n||(n=!0):delete r[e];return n?(r.files&&(r.files=r.files.map((e=>e.trim())),(!r.files||r.files.length<=0)&&delete r.files),r):A(3,"[cli] No resources found.")};function N(e,t){try{const o=JSON.parse("string"!=typeof e?JSON.stringify(e):e);return"string"!=typeof o&&t?JSON.stringify(o):o}catch(e){return!1}}const G=e=>{if(null===e||"object"!=typeof e)return e;const t=Array.isArray(e)?[]:{};for(const o in e)Object.prototype.hasOwnProperty.call(e,o)&&(t[o]=G(e[o]));return t},U=(e,t)=>JSON.stringify(e,((e,o)=>("string"==typeof o&&((o=o.trim()).startsWith("function(")||o.startsWith("function ("))&&o.endsWith("}")&&(o=t?`EXP_FUN${(o+"").replaceAll(/\n|\t|\r/g," ")}EXP_FUN`:void 0),"function"==typeof o?`EXP_FUN${(o+"").replaceAll(/\n|\t|\r/g," ")}EXP_FUN`:o))).replaceAll(/"EXP_FUN|EXP_FUN"/g,"");function W(){console.log("Usage of CLI arguments:".bold,"\n------",`\nFor more detailed information visit readme at: ${"https://github.com/highcharts/node-export-server#readme".bold.yellow}.`);const e=t=>{for(const[o,r]of Object.entries(t))if(Object.prototype.hasOwnProperty.call(r,"value")){let e=` --${r.cliName||o} ${("<"+r.type+">").green} `;if(e.length<48)for(let t=e.length;t<48;t++)e+=".";console.log(e,r.description,`[Default: ${r.value.toString().bold}]`.blue)}else e(r)};Object.keys(R).forEach((t=>{["puppeteer","highcharts"].includes(t)||(console.log(`\n${t.toUpperCase()}`.red),e(R[t]))})),console.log("\n")}const F=e=>!["false","undefined","null","NaN","0",""].includes(e)&&!!e,M=(e,t)=>{if(e&&"string"==typeof e)return(e=e.trim()).endsWith(".js")?!!t&&M(i(e,"utf8")):e.startsWith("function()")||e.startsWith("function ()")||e.startsWith("()=>")||e.startsWith("() =>")?`(${e})()`:e.replace(/;$/,"")};var q=(e,t)=>{const o="Too many requests, you have been rate limited. Please try again later.",r={max:t.maxRequests||30,window:t.window||1,delay:t.delay||0,trustProxy:t.trustProxy||!1,skipKey:t.skipKey||!1,skipToken:t.skipToken||!1};r.trustProxy&&e.enable("trust proxy");const i=y({windowMs:60*r.window*1e3,max:r.max,delayMs:r.delay,handler:(e,t)=>{t.format({json:()=>{t.status(429).send({message:o})},default:()=>{t.status(429).send(o)}})},skip:e=>!1!==r.skipKey&&!1!==r.skipToken&&e.query.key===r.skipKey&&e.query.access_token===r.skipToken&&(A(4,"[rate-limiting] Skipping rate limiter."),!0)});e.use(i),A(3,I(`[rate-limiting] Enabled rate limiting: ${r.max} requests\n per ${r.window} minute per IP, trusting proxy:\n ${r.trustProxy}.`))};async function D(e,t={}){return new Promise(((o,r)=>{const i=(e=>e.startsWith("https")?f:m)(e);i.get(e,t,(e=>{let t="";e.on("data",(e=>{t+=e})),e.on("end",(()=>{t||r("Nothing was fetched from the URL."),e.text=t,o(e)}))})).on("error",(e=>{r(e)}))}))}v.config();const V=c($,".cache"),J={cdnURL:"https://code.highcharts.com/",activeManifest:{},sources:"",hcVersion:""};let z=!1;const K=()=>J.hcVersion=J.sources.substr(0,J.sources.indexOf("*/")).replace("/*","").replace("*/","").replace(/\n/g,"").trim(),X=async(e,t)=>{try{e.endsWith(".js")&&(e=e.substring(0,e.length-3)),A(4,`[cache] Fetching script - ${e}.js`);const o=t?{agent:t,timeout:+process.env.PROXY_SERVER_TIMEOUT||5e3}:{},r=await D(`${e}.js`,o);if(200===r.statusCode)return r.text;throw`${r.statusCode}`}catch(t){throw A(1,`[cache] Error fetching script ${e}.js: ${t}.`),t}},B=async(e,t)=>{const{coreScripts:o,modules:r,indicators:i,scripts:s}=e,a="latest"!==e.version&&e.version?`${e.version}/`:"";A(3,"[cache] Updating cache to Highcharts ",a);const l=[...o.map((e=>`${a}${e}`)),...r.map((e=>"map"===e?`maps/${a}modules/${e}`:`${a}modules/${e}`)),...i.map((e=>`stock/${a}indicators/${e}`))];let c;const p=process.env.PROXY_SERVER_HOST,u=process.env.PROXY_SERVER_PORT;p&&u&&(c=new T({host:p,port:+u}));const d={};try{return J.sources=(await Promise.all([...l.map((async t=>{const o=await X(`${e.cdnURL||J.cdnURL}${t}`,c);return"string"==typeof o&&(d[t.replace(/(.*)\/|(.*)modules\/|stock\/(.*)indicators\/|maps\/(.*)modules\//gi,"")]=1),o})),...s.map((e=>X(e,c)))])).join(";\n"),K(),n(t,J.sources),d}catch(e){A(1,"[cache] Unable to update local Highcharts cache.")}},Y=async e=>{let r;const s=c(V,"manifest.json"),a=c(V,"sources.js");if(z=e,!t(V)&&o(V),!t(s)||e.forceFetch)A(3,"[cache] Fetching and caching Highcharts dependencies."),r=await B(e,a);else{let t=!1;const o=JSON.parse(i(s));if(o.modules&&Array.isArray(o.modules)){const e={};o.modules.forEach((t=>e[t]=1)),o.modules=e}const{modules:n,coreScripts:l,indicators:c}=e,p=n.length+l.length+c.length;o.version!==e.version?(A(3,"[cache] Highcharts version mismatch in cache, need to re-fetch."),t=!0):Object.keys(o.modules||{}).length!==p?(A(3,"[cache] Cache and requested modules does not match, need to re-fetch."),t=!0):t=(e.modules||[]).some((e=>{if(!o.modules[e])return A(3,`[cache] The ${e} missing in cache, need to re-fetch.`),!0})),t?r=await B(e,a):(A(3,"[cache] Dependency cache is up to date, proceeding."),J.sources=i(a,"utf8"),r=o.modules,K())}await(async(e,t)=>{const o={version:e.version,modules:t||{}};J.activeManifest=o,A(4,"[cache] writing new manifest");try{n(c(V,"manifest.json"),JSON.stringify(o),"utf8")}catch(e){A(1,`[cache] Error writing cache manifest: ${e}.`)}})(e,r)};var Q=async e=>!!z&&await Y(Object.assign(z,{version:e})),Z=()=>J,ee=()=>J.hcVersion;const te=E(64).toString("base64url"),oe=H.join("tmp",`puppeteer-${te}`),re=[`--user-data-dir=${H.join(oe,"profile")}`,"--autoplay-policy=user-gesture-required","--disable-background-networking","--disable-background-timer-throttling","--disable-backgrounding-occluded-windows","--disable-breakpad","--disable-client-side-phishing-detection","--disable-component-update","--disable-default-apps","--disable-dev-shm-usage","--disable-domain-reliability","--disable-extensions","--disable-features=AudioServiceOutOfProcess","--disable-hang-monitor","--disable-ipc-flooding-protection","--disable-notifications","--disable-offer-store-unmasked-wallet-cards","--disable-popup-blocking","--disable-print-preview","--disable-prompt-on-repost","--disable-renderer-backgrounding","--disable-session-crashed-bubble","--disable-setuid-sandbox","--disable-speech-api","--disable-sync","--hide-crash-restore-bubble","--hide-scrollbars","--ignore-gpu-blacklist","--metrics-recording-only","--mute-audio","--no-default-browser-check","--no-first-run","--no-pings","--no-sandbox","--no-zygote","--password-store=basic","--use-mock-keychain"],ie=b.fileURLToPath(new URL(".",import.meta.url)),ne=e.readFileSync(ie+"/../templates/template.html","utf8");let se;const ae=async()=>{if(!se)return!1;const e=await se.newPage();return await e.setContent(ne),await e.addScriptTag({path:ie+"/../.cache/sources.js"}),await e.evaluate((()=>window.setupHighcharts())),e.on("pageerror",(async t=>{A(1,"[page error]",t),await e.$eval("#container",((e,t)=>{window._displayErrors&&(e.innerHTML=t)}),`

Chart input data error

${t.toString()}`)})),e},le=async()=>{se.connected&&await se.close()};const ce=b.fileURLToPath(new URL(".",import.meta.url)),pe=async(e,t,o)=>await e.evaluate(((e,t)=>window.triggerExport(e,t)),t,o);var ue=async(e,t,o)=>{const r=[],n=async e=>{for(const e of r)await e.dispose();await e.evaluate((()=>{const[,...e]=document.getElementsByTagName("script"),[,...t]=document.getElementsByTagName("style"),[...o]=document.getElementsByTagName("link");for(const r of[...e,...t,...o])r.remove()}))};try{const s=()=>{};A(4,"[export] Determining export path.");const a=o.export;await e.evaluate((()=>requestAnimationFrame((()=>{}))));const c=a?.options?.chart?.displayErrors&&Z().activeManifest.modules.debugger;await e.evaluate((e=>window._displayErrors=e),c);const p=()=>{};let u;if(t.indexOf&&(t.indexOf("=0||t.indexOf("=0)){if(A(4,"[export] Treating as SVG."),"svg"===a.type)return t;u=!0;const o=()=>{};await e.setContent((e=>`\n\n\n \n \n Highcarts Export\n \n \n \n
\n ${e}\n
\n \n\n\n`)(t)),o()}else if(A(4,"[export] Treating as config."),a.strInj){const t=()=>{};await pe(e,{chart:{height:a.height,width:a.width}},o),t()}else{t.chart.height=a.height,t.chart.width=a.width;const r=()=>{};await pe(e,t,o),r()}p();const d=()=>{},h=o.customCode.resources;if(h){if(h.js&&r.push(await e.addScriptTag({content:h.js})),h.files)for(const t of h.files)try{const o=!t.startsWith("http");r.push(await e.addScriptTag(o?{content:i(t,"utf8")}:{url:t}))}catch(e){A(4,"[export] JS file not found.")}const t=()=>{};if(h.css){let t=h.css.match(/@import\s*([^;]*);/g);if(t)for(let i of t)i&&(i=i.replace("url(","").replace("@import","").replace(/"/g,"").replace(/'/g,"").replace(/;/,"").replace(/\)/g,"").trim(),i.startsWith("http")?r.push(await e.addStyleTag({url:i})):o.customCode.allowFileResources&&r.push(await e.addStyleTag({path:l.join(ce,i)})));r.push(await e.addStyleTag({content:h.css.replace(/@import\s*([^;]*);/g,"")||" "}))}t()}d();const g=u?await e.$eval("#chart-container svg:first-of-type",(async(e,t)=>({chartHeight:e.height.baseVal.value*t,chartWidth:e.width.baseVal.value*t})),parseFloat(a.scale)):await e.evaluate((async()=>{const{chartHeight:e,chartWidth:t}=window.Highcharts.charts[0];return{chartHeight:e,chartWidth:t}})),m=()=>{},f=Math.ceil(g?.chartHeight||a.height),v=Math.ceil(g?.chartWidth||a.width);await e.setViewport({height:f,width:v,deviceScaleFactor:u?1:parseFloat(a.scale)});const y=u?e=>{document.body.style.zoom=e,document.body.style.margin="0px"}:()=>{document.body.style.zoom=1};await e.evaluate(y,parseFloat(a.scale));const{height:b,width:w,x:T,y:x}=await(e=>e.$eval("#chart-container",(e=>{const{x:t,y:o,width:r,height:i}=e.getBoundingClientRect();return{x:t,y:o,width:r,height:Math.trunc(i>1?i:500)}})))(e);let k;u||await e.setViewport({width:Math.round(w),height:Math.round(b),deviceScaleFactor:parseFloat(a.scale)}),m();const S=()=>{};if("svg"===a.type)k=await(async e=>await e.$eval("#container svg:first-of-type",(e=>e.outerHTML)))(e);else if("png"===a.type||"jpeg"===a.type)k=await(async(e,t,o,r)=>await Promise.race([e.screenshot({type:t,encoding:o,clip:r,omitBackground:!0}),new Promise(((e,t)=>setTimeout((()=>t(new Error("Rasterization timeout"))),1500)))]))(e,a.type,"base64",{width:v,height:f,x:T,y:x});else{if("pdf"!==a.type)throw`Unsupported output format ${a.type}`;k=await(async(e,t,o,r)=>await e.pdf({height:t+1,width:o,encoding:r}))(e,f,v,"base64")}return await e.evaluate((()=>{const e=Highcharts.charts;if(e.length)for(const t of e)t&&t.destroy(),Highcharts.charts.shift()})),S(),s(),await n(e),k}catch(t){return await n(e),A(1,`[export] Error encountered during export: ${t}`),t}};let de,he=0,ge=0,me=0,fe=0,ve=0,ye={},be=!1;const we={create:async()=>{const e=x();let t=!1;const o=(new Date).getTime();try{if(t=await ae(),!t||t.isClosed())throw"invalid page";A(3,`[pool] Successfully created a worker ${e} - took ${(new Date).getTime()-o} ms.`)}catch(e){throw A(1,`[pool] Error creating a new page in pool entry creation! ${e}`),"Error creating page"}return{id:e,page:t,workCount:Math.round(Math.random()*(ye.workLimit/2))}},validate:e=>!(ye.workLimit&&++e.workCount>ye.workLimit)||(A(3,"[pool] Worker failed validation:",`exceeded work limit (limit is ${ye.workLimit})`),!1),destroy:e=>{A(3,`[pool] Destroying pool entry ${e.id}.`),e.page&&e.page.close()},log:(e,t)=>console.log(`${t}: ${e}`)},Te=async e=>{de=e.puppeteerArgs;try{await(async e=>{const t=[...re,...e||[]];if(!se){let e=0;const o=async()=>{try{A(3,"[browser] attempting to get a browser instance (try",e+")"),se=await S.launch({headless:"new",args:t,userDataDir:"./tmp/"})}catch(t){A(0,"[browser]",t),++e<25?(A(3,"[browser] failed:",t),await new Promise((e=>setTimeout(e,4e3))),await o()):A(0,"Max retries reached")}};try{await o()}catch(e){return A(0,"[browser] Unable to open browser"),!1}if(!se)return A(0,"[browser] Unable to open browser"),!1}return se})(de)}catch(e){A(0,"[pool|browser]",e)}if(ye=e&&e.pool?{...e.pool}:{},A(3,"[pool] Initializing pool:",`min ${ye.initialWorkers}, max ${ye.maxWorkers}.`),be)return A(4,"[pool] Already initialized, please kill it before creating a new one.");ye.listenToProcessExits&&(A(4,"[pool] Attaching exit listeners to the process."),process.on("exit",(async()=>{await xe()})),process.on("SIGINT",((e,t)=>{A(4,`The ${e} event with code: ${t}.`),process.exit(1)})),process.on("SIGTERM",((e,t)=>{A(4,`The ${e} event with code: ${t}.`),process.exit(1)})),process.on("uncaughtException",(async(e,t)=>{A(4,`The ${t} error, message: ${e.message}.`)})));try{be=new k({...we,min:ye.initialWorkers,max:ye.maxWorkers,createRetryIntervalMillis:200,createTimeoutMillis:ye.acquireTimeout,acquireTimeoutMillis:ye.acquireTimeout,destroyTimeoutMillis:ye.acquireTimeout,idleTimeoutMillis:ye.timeoutThreshold,reapIntervalMillis:1e3,propagateCreateError:!1}),be.on("createFail",((e,t)=>{A(1,`[pool] Error when creating worker of an event id ${e}:`,t)})),be.on("acquireFail",((e,t)=>{A(1,`[pool] Error when acquiring worker of an event id ${e}:`,t)})),be.on("destroyFail",((e,t,o)=>{A(1,`[pool] Error when destroying worker of an id ${t.id}, event id ${e}:`,o)})),be.on("release",(e=>{A(4,`[pool] Releasing a worker of an id ${e.id}`)})),be.on("destroySuccess",((e,t)=>{A(4,`[pool] Destroyed a worker of an id ${t.id}`)}));const e=[];for(let t=0;t{be.release(e)})),A(3,`[pool] The pool is ready with ${ye.initialWorkers} initial resources waiting.`)}catch(e){throw A(1,`[pool] Couldn't create the worker pool ${e}`),e}};async function xe(){return A(3,"[pool] Killing all workers."),be.destroyed?(await le(),!0):(await be.destroy(),await le(),!0)}const ke=async(e,t)=>{let o;const r=e=>{throw++fe,o&&be.release(o),"In pool.postWork: "+e};if(A(4,"[pool] Work received, starting to process."),ye.benchmarking&&Se(),++ge,!be)return A(1,"[pool] Work received, but pool has not been started."),r("Pool is not inited but work was posted to it!");try{A(4,"[pool] Acquiring worker"),o=await be.acquire().promise}catch(e){return r(`[pool] Error when acquiring available entry: ${e}`)}if(A(4,"[pool] Acquired worker handle"),!o.page)return r("Resolved worker page is invalid: pool setup is wonky");try{let i=(new Date).getTime();A(4,`[pool] Starting work on pool entry ${o.id}.`);const n=await ue(o.page,e,t);if(n instanceof Error)return"Rasterization timeout"===n.message&&(o.page.close(),o.page=await ae()),r(n);be.release(o);const s=(new Date).getTime()-i;return me+=s,ve=me/++he,A(4,`[pool] Work completed in ${s} ms.`),{data:n,options:t}}catch(e){r(`Error trying to perform puppeteer export: ${e}.`)}};function Se(){const{min:e,max:t,size:o,available:r,borrowed:i,pending:n,spareResourceCapacity:s}=be;A(4,`[pool] The minimum number of resources allowed by pool: ${e}.`),A(4,`[pool] The maximum number of resources allowed by pool: ${t}.`),A(4,`[pool] The number of all resources in pool (free or in use): ${o}.`),A(4,`[pool] The number of resources that are currently available: ${r}.`),A(4,`[pool] The number of resources that are currently acquired: ${i}.`),A(4,`[pool] The number of callers waiting to acquire a resource: ${n}.`),A(4,`[pool] The number of how many more resources can the pool manage/create: ${s}.`)}var He=()=>({min:be.min,max:be.max,size:be.size,available:be.available,borrowed:be.borrowed,pending:be.pending,spareResourceCapacity:be.spareResourceCapacity}),Ee=()=>ge,Re=()=>fe,Le=()=>ve,Ce=()=>he;const Oe=process.env.npm_package_version,_e=new Date;let Ae={};const $e=()=>Ae,Ie=(e,t,o=[])=>{const r=G(e);for(const[e,n]of Object.entries(t))r[e]="object"!=typeof(i=n)||Array.isArray(i)||null===i||o.includes(e)||void 0===r[e]?void 0!==n?n:r[e]:Ie(r[e],n,o);var i;return r};function Pe(e,t={},o=""){Object.keys(e).forEach((r=>{if(!["puppeteer","highcharts"].includes(r)){const i=e[r],n=t&&t[r];let s;void 0===i.value?Pe(i,n,`${o}.${r}`):(void 0!==n&&(i.value=n),i.envLink&&("boolean"===i.type?i.value=F([process.env[i.envLink],i.value].find((e=>e||"false"===e))):"number"===i.type?(s=+process.env[i.envLink],i.value=s>=0?s:i.value):i.type.indexOf("]")>=0&&process.env[i.envLink]?i.value=process.env[i.envLink].split(","):i.value=process.env[i.envLink]||i.value))}}))}function je(e){let t={};for(const[o,r]of Object.entries(e))t[o]=Object.prototype.hasOwnProperty.call(r,"value")?r.value:je(r);return t}let Ne=!1;const Ge=async(e,t)=>{A(4,"[chart] Starting exporting process.");const o=((e,t={})=>{let o={};return e.svg?(o=G(t),o.export.type=e.type||e.export.type,o.export.scale=e.scale||e.export.scale,o.export.outfile=e.outfile||e.export.outfile,o.payload={svg:e.svg}):o=Ie(t,e,L),o.export.outfile=o.export?.outfile||`chart.${o.export?.type||"png"}`,o})(e,$e()),r=o.export;return o.payload?.svg&&""!==o.payload.svg?Me(o.payload.svg.trim(),o,t):r.infile&&r.infile.length?(A(4,"[chart] Attempting to export from an input file."),s(r.infile,"utf8",((e,r)=>e?A(1,`[chart] Error loading input file: ${e}.`):(o.export.instr=r,Me(o.export.instr.trim(),o,t))))):r.instr&&""!==r.instr||r.options&&""!==r.options?(A(4,"[chart] Attempting to export from a raw input."),F(o.customCode?.allowCodeExecution)?Fe(o,t):"string"==typeof r.instr?Me(r.instr.trim(),o,t):We(o,r.instr||r.options,t)):(A(1,I(`[chart] No input specified.\n ${JSON.stringify(r,void 0," ")}.`)),t&&t(!1,{error:!0,message:"No input specified."}))},Ue=e=>{const{chart:t,exporting:o}=e.export?.options||N(e.export?.instr),r=N(e.export?.globalOptions);let i=e.export?.scale||o?.scale||r?.exporting?.scale||e.export?.defaultScale||1;return i=Math.max(.1,Math.min(i,5)),i=((e,t=1)=>{const o=Math.pow(10,t||0);return Math.round(+e*o)/o})(i,2),{height:e.export?.height||o?.sourceHeight||t?.height||r?.exporting?.sourceHeight||r?.chart?.height||e.export?.defaultHeight||400,width:e.export?.width||o?.sourceWidth||t?.width||r?.exporting?.sourceWidth||r?.chart?.width||e.export?.defaultWidth||600,scale:i}},We=(e,t,o,r)=>{let{export:n,customCode:s}=e;const a="boolean"==typeof s.allowCodeExecution?s.allowCodeExecution:Ne;if(s){if("string"==typeof e.customCode.resources)e.customCode.resources=j(e.customCode.resources,F(e.customCode.allowFileResources));else if(!e.customCode.resources)try{const t=i("resources.json","utf8");e.customCode.resources=j(t,F(e.customCode.allowFileResources))}catch(e){A(3,"[chart] The default resources.json file not found.")}}else s=e.customCode={};if(!a&&s){if(s.callback||s.resources||s.customCode)return o&&o(!1,{error:!0,message:I("The callback, resources and customCode have been disabled for this\n server.")});s.callback=!1,s.resources=!1,s.customCode=!1}if(t&&(t.chart=t.chart||{},t.exporting=t.exporting||{},t.exporting.enabled=!1),n.constr=n.constr||"chart",n.type=P(n.type,n.outfile),"svg"===n.type&&(n.width=!1),["globalOptions","themeOptions"].forEach((e=>{try{n&&n[e]&&("string"==typeof n[e]&&n[e].endsWith(".json")?n[e]=N(i(n[e],"utf8"),!0):n[e]=N(n[e],!0))}catch(t){n[e]={},A(1,`[chart] The ${e} not found.`)}})),s.allowCodeExecution&&(s.customCode=M(s.customCode,s.allowFileResources)),s&&s.callback&&s.callback?.indexOf("{")<0)if(s.allowFileResources)try{s.callback=i(s.callback,"utf8")}catch(e){A(2,`[chart] Error loading callback: ${e}.`),s.callback=!1}else s.callback=!1;e.export={...e.export,...Ue(e)},ke(n.strInj||t||r,e).then((e=>o(e))).catch((e=>(A(0,"[chart] When posting work:",e),o(!1,e))))},Fe=(e,t)=>{try{let o,r=e.export.instr||e.export.options;return"string"!=typeof r&&(o=r=U(r,e.customCode?.allowCodeExecution)),o=r.replaceAll(/\t|\n|\r/g,"").trim(),";"===o[o.length-1]&&(o=o.substring(0,o.length-1)),e.export.strInj=o,We(e,!1,t)}catch(o){const r=I(`Malformed input detected for ${e.export?.requestId||"?"}:\n Please make sure that your JSON/JavaScript options\n are sent using the "options" attribute, and that if you're using\n SVG, it is unescaped.`);return A(1,r),t&&t(!1,JSON.stringify({error:!0,message:r}))}},Me=(e,t,o)=>{const{allowCodeExecution:r}=t.customCode;if(e.indexOf("=0||e.indexOf("=0)return A(4,"[chart] Parsing input as SVG."),We(t,!1,o,e);try{const r=JSON.parse(e.replaceAll(/\t|\n|\r/g," "));return We(t,r,o)}catch(e){return F(r)?Fe(t,o):o&&o(!1,{error:!0,message:I("Only JSON configurations and SVG is allowed for this server. If\n this is your server, JavaScript exporting can be enabled by starting\n the server with the --allowCodeExecution flag.")})}},qe={png:"image/png",jpeg:"image/jpeg",gif:"image/gif",pdf:"application/pdf",svg:"image/svg+xml"};let De=0;const Ve=[],Je=[],ze=(e,t,o,r)=>{let i=!0;const{id:n,uniqueId:s,type:a,body:l}=r;return e.some((e=>{if(e){let r=e(t,o,n,s,a,l);return void 0!==r&&!0!==r&&(i=r),!0}})),i},Ke=(e,t)=>{(()=>{const e=process.hrtime.bigint()})();const o=$e(),r=e.body,i=++De,n=x().replace(/-/g,"");let s=P(r.type);if(!r)return t.status(400).send(I("Body is required. Sending a body? Make sure your Content-type header\n is correct. Accepted is application/json and multipart/form-data."));let a=N(r.infile||r.options||r.data);if(!a&&!r.svg)return A(2,I(`Request ${n} from ${e.headers["x-forwarded-for"]||e.connection.remoteAddress} was incorrect. Check your payload.`)),t.status(400).send(I("No correct chart data found. Please make sure you are using\n application/json or multipart/form-data headers, and that the chart\n data is in the 'infile', 'options' or 'data' attribute if sending\n JSON or in the 'svg' if sending SVG."));let l=!1;if(l=ze(Ve,e,t,{id:i,uniqueId:n,type:s,body:r}),!0!==l)return t.send(l);let c=!1;e.socket.on("close",(()=>{c=!0})),A(4,`[export] Got an incoming HTTP request ${n}.`),r.constr="string"==typeof r.constr&&r.constr||"chart";const p={export:{instr:a,type:s,constr:r.constr[0].toLowerCase()+r.constr.substr(1),height:r.height,width:r.width,scale:r.scale||o.export.scale,globalOptions:N(r.globalOptions,!0),themeOptions:N(r.themeOptions,!0)},customCode:{allowCodeExecution:Ne,allowFileResources:!1,resources:N(r.resources,!0),callback:r.callback,customCode:r.customCode}};a&&(p.export.instr=U(a,p.customCode.allowCodeExecution));const u=Ie(o,p);if(u.export.options=a,u.payload={svg:r.svg||!1,b64:r.b64||!1,dataOptions:N(r.dataOptions,!0),noDownload:r.noDownload||!1,requestId:n},r.svg&&(d=u.payload.svg,["localhost","(10).(.*).(.*).(.*)","(127).(.*).(.*).(.*)","(172).(1[6-9]|2[0-9]|3[0-1]).(.*).(.*)","(192).(168).(.*).(.*)"].some((e=>d.match(`xlink:href="(?:(http://|https://))?${e}`)))))return t.status(400).send("SVG potentially contain at least one forbidden URL in xlink:href element.");var d;Ge(u,((o,a)=>(e.socket.removeAllListeners("close"),c?A(3,I("[export] The client closed the connection before the chart was done\n processing.")):a?(A(1,I(`[export] Work: ${n} could not be completed, sending:\n ${a}`)),t.status(400).send(a.message)):o&&o.data?(s=o.options.export.type,ze(Je,e,t,{id:i,body:o.data}),o.data?r.b64?"pdf"===s?t.send(Buffer.from(o.data,"utf8").toString("base64")):t.send(o.data):(t.header("Content-Type",qe[s]||"image/png"),r.noDownload||t.attachment(`${e.params.filename||"chart"}.${s||"png"}`),"svg"===s?t.send(o.data):t.send(Buffer.from(o.data,"base64"))):void 0):(A(1,I(`[export] Unexpected return from chart generation, please check your\n data Request: ${n} is ${o.data}.`)),t.status(400).send("Unexpected return from chart generation, please check your data.")))))};const Xe=h();Xe.disable("x-powered-by"),Xe.use(d());const Be=g.memoryStorage(),Ye=g({storage:Be,limits:{fieldsSize:"50MB"}});Xe.use(Ye.any()),Xe.use(u.json({limit:"50mb"})),Xe.use(u.urlencoded({extended:!0,limit:"50mb"})),Xe.use(u.urlencoded({extended:!1,limit:"50mb"}));const Qe=e=>A(1,`[server] Socket error: ${e}`),Ze=e=>{e.on("clientError",Qe),e.on("error",Qe),e.on("connection",(e=>e.on("error",(e=>Qe(e)))))},et=async e=>{if(!e.enable)return!1;if(!e.ssl.enable&&!e.ssl.force){const t=m.createServer(Xe);Ze(t),t.listen(e.port,e.host),A(3,`[server] Started HTTP server on ${e.host}:${e.port}.`)}if(e.ssl.enable){let t,o;try{t=await a.readFile(p.join(e.ssl.certPath,"server.key"),"utf8"),o=await a.readFile(p.join(e.ssl.certPath,"server.crt"),"utf8")}catch(t){A(1,`[server] Unable to load key/certificate from ${e.ssl.certPath}.`)}if(t&&o){const t=f.createServer(Xe);Ze(t),t.listen(e.ssl.port,e.host),A(3,`[server] Started HTTPS server on ${e.host}:${e.ssl.port}.`)}}e.rateLimiting&&e.rateLimiting.enable&&![0,NaN].includes(e.rateLimiting.maxRequests)&&q(Xe,e.rateLimiting),Xe.use(h.static(p.join($,"public"))),(e=>{!!e&&e.get("/health",((e,t)=>{t.send({status:"OK",bootTime:_e,uptime:Math.floor(((new Date).getTime()-_e.getTime())/1e3/60)+" minutes",version:Oe,highchartsVersion:ee(),averageProcessingTime:Le(),performedExports:Ce(),failedExports:Re(),exportAttempts:Ee(),sucessRatio:Ce()/Ee()*100,pool:He()})}))})(Xe),(e=>{e.post("/",Ke),e.post("/:filename",Ke)})(Xe),(e=>{!!e&&e.get("/",((e,t)=>{t.sendFile(c($,"public","index.html"))}))})(Xe),(e=>{!!e&&e.post("/change_hc_version/:newVersion",(async(e,t)=>{const o=process.env.HIGHCHARTS_ADMIN_TOKEN;if(!o||!o.length)return t.send({error:!0,message:"Server not configured to do run-time version changes: HIGHCHARTS_ADMIN_TOKEN not set"});const r=e.get("hc-auth");if(!r||r!==o)return t.send({error:!0,message:"Invalid or missing token: set token in the hc-auth header"});const i=e.params.newVersion;if(i){try{await Q(i)}catch(e){t.send({error:!0,message:e})}t.send({version:ee()})}else t.send({error:!0,message:"No new version supplied"})}))})(Xe)};var tt={startServer:et,getExpress:()=>h,getApp:()=>Xe,use:(e,...t)=>{Xe.use(e,...t)},get:(e,...t)=>{Xe.get(e,...t)},post:(e,...t)=>{Xe.post(e,...t)},enableRateLimiting:e=>q(Xe,e)},ot={log:A,mapToNewConfig:e=>{const t={};for(const[o,r]of Object.entries(e)){const e=C[o]?C[o].split("."):[];e.reduce(((t,o,i)=>t[o]=e.length-1===i?r:t[o]||{}),t)}return t},setOptions:(e,t)=>(t?.length&&(Ae=function(e){const t=e.findIndex((e=>"loadConfig"===e.replace(/-/g,"")));if(t>-1&&e[t+1]){const o=e[t+1];try{if(o&&o.endsWith(".json"))return JSON.parse(i(o))}catch(e){A(1,`[config] Unable to load config from the ${o}: ${e}`)}}return{}}(t)),Pe(R,Ae),Ae=je(R),e&&(Ae=Ie(Ae,e,L)),t?.length&&(Ae=function(e,t,o){for(let o=0;o(i.length-1===a&&void 0!==n[s]&&(t[++o]?n[s]=t[o]||n[s]:(console.log(`Missing argument value for ${r}!`.red,"\n"),e=W())),n[s])),e)}return e}(Ae,t)),Ae),singleExport:e=>{e.export.instr=e.export.instr||e.export.options,Ge(e,((e,t)=>{t&&(A(1,`[cli] ${t.message}`),process.exit(1));const{outfile:o,type:r}=e.options.export;n(o||`chart.${r}`,"svg"!==r?Buffer.from(e.data,"base64"):e.data),xe()}))},startExport:Ge,batchExport:e=>{const t=[];for(let o of e.export.batch.split(";"))o=o.split("="),2===o.length&&t.push(new Promise(((t,r)=>{Ge({...e,export:{...e.export,infile:o[0],outfile:o[1]}},((e,o)=>{if(o)return r(o);n(e.options.export.outfile,Buffer.from(e.data,"base64")),t()}))})));Promise.all(t).then((()=>{xe()})).catch((e=>{A(1,`[chart] Error encountered during batch export: ${e}`),xe()}))},server:tt,startServer:et,killPool:xe,initPool:async(e={})=>{var t,o;return t=e.customCode&&e.customCode.allowCodeExecution,Ne=F(t),(o=e.logging&&parseInt(e.logging.level))>=0&&o<=_.levelsDesc.length&&(_.level=o),e.logging&&e.logging.dest&&((e,t)=>{if(_={..._,dest:e||_.dest,file:t||_.file,toFile:!0},0===_.dest.length)return A(1,"[logger] File logging init: no path supplied.");_.dest.endsWith("/")||(_.dest+="/")})(e.logging.dest,e.logging.file||"highcharts-export-server.log"),await Y(e.highcharts||{version:"latest"}),await Te({pool:e.pool||{initialWorkers:1,maxWorkers:1},puppeteerArgs:e.puppeteer?.args||[]}),e}};export{ot as default}; +import"colors";import e,{existsSync as t,mkdirSync as o,appendFile as r,readFileSync as i,writeFileSync as n,readFile as s,promises as a}from"fs";import l,{join as c,posix as p}from"path";import u from"body-parser";import d from"cors";import h from"express";import g from"multer";import m from"http";import f from"https";import v from"dotenv";import y from"express-rate-limit";import*as b from"url";import{fileURLToPath as w}from"url";import T from"https-proxy-agent";import{v4 as x}from"uuid";import{Pool as k}from"tarn";import S from"puppeteer";import H from"node:path";import{randomBytes as E}from"node:crypto";import"prompts";v.config();const R={puppeteer:{args:{value:[],type:"string[]",description:"Array of arguments to send to puppeteer."}},highcharts:{version:{value:"latest",envLink:"HIGHCHARTS_VERSION",type:"string",description:"Highcharts version to use."},cdnURL:{value:"https://code.highcharts.com/",envLink:"HIGHCHARTS_CDN",type:"string",description:"The CDN URL of Highcharts scripts to use."},coreScripts:{envLink:"HIGHCHARTS_CORE_SCRIPTS",value:["highcharts","highcharts-more","highcharts-3d"],type:"string[]",description:"Highcharts core scripts to fetch."},modules:{envLink:"HIGHCHARTS_MODULES",value:["stock","map","gantt","exporting","export-data","parallel-coordinates","accessibility","annotations-advanced","boost-canvas","boost","data","draggable-points","static-scale","broken-axis","heatmap","tilemap","timeline","treemap","treegraph","item-series","drilldown","histogram-bellcurve","bullet","funnel","funnel3d","pyramid3d","networkgraph","pareto","pattern-fill","pictorial","price-indicator","sankey","arc-diagram","dependency-wheel","series-label","solid-gauge","sonification","stock-tools","streamgraph","sunburst","variable-pie","variwide","vector","venn","windbarb","wordcloud","xrange","no-data-to-display","drag-panes","debugger","dumbbell","lollipop","cylinder","organization","dotplot","marker-clusters","hollowcandlestick","heikinashi"],type:"string[]",description:"Highcharts modules to fetch."},indicators:{envLink:"HIGHCHARTS_INDICATORS",value:["indicators-all"],type:"string[]",description:"Highcharts indicators to fetch."},scripts:{value:["https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.29.4/moment.min.js","https://cdnjs.cloudflare.com/ajax/libs/moment-timezone/0.5.34/moment-timezone-with-data.min.js"],type:"string[]",description:"Additional direct scripts/optional dependencies (e.g. moment.js)."},forceFetch:{envLink:"HIGHCHARTS_FORCE_FETCH",value:!1,type:"boolean",description:"Should all the scripts be refetched after rerunning the server."}},export:{infile:{value:!1,type:"string",description:"The input file name along with a type (json or svg). It can be a correct JSON or SVG file."},instr:{value:!1,type:"string",description:"An input in a form of a stringified JSON or SVG file. Overrides the --infile."},options:{value:!1,type:"string",description:"An alias for the --instr option."},outfile:{value:!1,type:"string",description:"The output filename along with a type (jpeg, png, pdf or svg). Ignores the --type flag."},type:{envLink:"EXPORT_DEFAULT_TYPE",value:"png",type:"string",description:"The format of the file to export to. Can be jpeg, png, pdf or svg."},constr:{envLink:"EXPORT_DEFAULT_CONSTR",value:"chart",type:"string",description:"The constructor to use. Can be chart, stockChart, mapChart or ganttChart."},defaultHeight:{envLink:"EXPORT_DEFAULT_HEIGHT",value:400,type:"number",description:"The default height of the exported chart. Used when not found any value set."},defaultWidth:{envLink:"EXPORT_DEFAULT_WIDTH",value:600,type:"number",description:"The default width of the exported chart. Used when not found any value set."},defaultScale:{envLink:"EXPORT_DEFAULT_SCALE",value:1,type:"number",description:"The default scale of the exported chart. Ranges between 1 and 5."},height:{type:"number",value:!1,description:"The default height of the exported chart. Overrides the option in the chart settings."},width:{type:"number",value:!1,description:"The width of the exported chart. Overrides the option in the chart settings."},scale:{value:!1,type:"number",description:"The scale of the exported chart. Ranges between 1 and 5."},globalOptions:{value:!1,type:"string",description:"A stringified JSON or a filename with options to be passed into the Highcharts.setOptions."},themeOptions:{value:!1,type:"string",description:"A stringified JSON or a filename with theme options to be passed into the Highcharts.setOptions."},batch:{value:!1,type:"string",description:'Starts a batch job. A string that contains input/output pairs: "in=out;in=out;..".'}},customCode:{allowCodeExecution:{envLink:"HIGHCHARTS_ALLOW_CODE_EXECUTION",value:!1,type:"boolean",description:"If set to true, allow for the execution of arbitrary code when exporting."},allowFileResources:{envLink:"HIGHCHARTS_ALLOW_FILE_RESOURCES",value:!0,type:"boolean",description:"Allow injecting resources from the filesystem. Has no effect when running as a server."},customCode:{value:!1,type:"string",description:"A function to be called before chart initialization. Can be a filename with the js extension."},callback:{value:!1,type:"string",description:"A JavaScript file with a function to run on construction."},resources:{value:!1,type:"string",description:"An additional resource in a form of stringified JSON. It can contain files, js and css sections."},loadConfig:{value:!1,type:"string",description:"A file that contains a pre-defined config to use."},createConfig:{value:!1,type:"string",description:"Allows to set options through a prompt and save in a provided config file."}},server:{enable:{envLink:"HIGHCHARTS_SERVER_ENABLE",value:!1,type:"boolean",cliName:"enableServer",description:"If set to true, starts a server on 0.0.0.0."},host:{envLink:"HIGHCHARTS_SERVER_HOST",value:"0.0.0.0",type:"string",description:"The hostname of the server. Also starts a server listening on the supplied hostname."},port:{envLink:"HIGHCHARTS_SERVER_PORT",value:7801,type:"number",description:"The port to use for the server. Defaults to 7801."},ssl:{enable:{envLink:"HIGHCHARTS_SERVER_SSL_ENABLE",value:!1,type:"boolean",cliName:"enableSsl",description:"Enables the SSL protocol."},force:{envLink:"HIGHCHARTS_SERVER_SSL_FORCE",value:!1,type:"boolean",cliName:"sslForced",description:"If set to true, forces the server to only serve over HTTPS."},port:{envLink:"HIGHCHARTS_SERVER_SSL_PORT",value:443,type:"number",cliName:"sslPort",description:"The port on which to run the SSL server."},certPath:{envLink:"HIGHCHARTS_SSL_CERT_PATH",value:"",type:"string",description:"The path to the SSL certificate/key."}},rateLimiting:{enable:{envLink:"HIGHCHARTS_RATE_LIMIT_ENABLE",value:!1,type:"boolean",cliName:"enableRateLimiting",description:"Enables rate limiting."},maxRequests:{envLink:"HIGHCHARTS_RATE_LIMIT_MAX",value:10,type:"number",description:"Max requests allowed in a one minute."},window:{envLink:"HIGHCHARTS_RATE_LIMIT_WINDOW",value:1,type:"number",description:"The time window in minutes for rate limiting."},delay:{envLink:"HIGHCHARTS_RATE_LIMIT_DELAY",value:0,type:"number",description:"The amount to delay each successive request before hitting the max."},trustProxy:{envLink:"HIGHCHARTS_RATE_LIMIT_TRUST_PROXY",value:!1,type:"boolean",description:"Set this to true if behind a load balancer."},skipKey:{envLink:"HIGHCHARTS_RATE_LIMIT_SKIP_KEY",value:"",type:"number|string",description:"Allows bypassing the rate limiter and should be provided with skipToken argument."},skipToken:{envLink:"HIGHCHARTS_RATE_LIMIT_SKIP_TOKEN",value:"",type:"number|string",description:"Allows bypassing the rate limiter and should be provided with skipKey argument."}}},pool:{initialWorkers:{envLink:"HIGHCHARTS_POOL_MIN_WORKERS",value:4,type:"number",description:"The number of initial workers to spawn."},maxWorkers:{envLink:"HIGHCHARTS_POOL_MAX_WORKERS",value:8,type:"number",description:"The number of max workers to spawn."},workLimit:{envLink:"HIGHCHARTS_POOL_WORK_LIMIT",value:40,type:"number",description:"The pieces of work that can be performed before restarting process."},queueSize:{envLink:"HIGHCHARTS_POOL_QUEUE_SIZE",value:5,type:"number",description:"The size of the request overflow queue."},timeoutThreshold:{envLink:"HIGHCHARTS_POOL_TIMEOUT",value:5e3,type:"number",description:"The number of milliseconds before timing out."},acquireTimeout:{envLink:"HIGHCHARTS_POOL_ACQUIRE_TIMEOUT",value:5e3,type:"number",description:"The number of milliseconds to wait for acquiring a resource."},reaper:{envLink:"HIGHCHARTS_POOL_ENABLE_REAPER",value:!0,type:"boolean",description:"Whether or not to evict workers after a certain time period."},benchmarking:{envLink:"HIGHCHARTS_POOL_BENCHMARKING",value:!1,type:"boolean",description:"Enable benchmarking."},listenToProcessExits:{envLink:"HIGHCHARTS_POOL_LISTEN_TO_PROCESS_EXITS",value:!0,type:"boolean",description:"Set to false in order to skip attaching process.exit handlers."}},logging:{level:{envLink:"HIGHCHARTS_LOG_LEVEL",value:4,type:"number",cliName:"logLevel",description:"The log level (0: silent, 1: error, 2: warning, 3: notice, 4: verbose)."},file:{envLink:"HIGHCHARTS_LOG_FILE",value:"highcharts-export-server.log",type:"string",cliName:"logFile",description:"A name of a log file. The --logDest also needs to be set to enable file logging."},dest:{envLink:"HIGHCHARTS_LOG_DEST",value:"log/",type:"string",cliName:"logDest",description:"The path to store log files. Also enables file logging."}},ui:{enable:{envLink:"HIGHCHARTS_UI_ENABLE",value:!1,type:"boolean",cliName:"enableUi",description:"Enables the UI for the export server."},route:{envLink:"HIGHCHARTS_UI_ROUTE",value:"/",type:"string",cliName:"uiRoute",description:"The route to attach the UI to."}},other:{noLogo:{envLink:"HIGHCHARTS_NO_LOGO",value:!1,type:"boolean",description:"Skip printing the logo on a startup. Will be replaced by a simple text."}},payload:{}};R.puppeteer.args.value.join(","),R.highcharts.version.value,R.highcharts.cdnURL.value,R.highcharts.modules.value,R.highcharts.scripts.value.join(","),R.highcharts.forceFetch.value,R.export.type.value,R.export.constr.value,R.export.defaultHeight.value,R.export.defaultWidth.value,R.export.defaultScale.value,R.customCode.allowCodeExecution.value,R.customCode.allowFileResources.value,R.server.enable.value,R.server.host.value,R.server.port.value,R.server.ssl.enable.value,R.server.ssl.force.value,R.server.ssl.port.value,R.server.ssl.certPath.value,R.server.rateLimiting.enable.value,R.server.rateLimiting.maxRequests.value,R.server.rateLimiting.window.value,R.server.rateLimiting.delay.value,R.server.rateLimiting.trustProxy.value,R.server.rateLimiting.skipKey.value,R.server.rateLimiting.skipToken.value,R.pool.initialWorkers.value,R.pool.maxWorkers.value,R.pool.workLimit.value,R.pool.queueSize.value,R.pool.timeoutThreshold.value,R.pool.acquireTimeout.value,R.pool.reaper.value,R.pool.benchmarking.value,R.pool.listenToProcessExits.value,R.logging.level.value,R.logging.file.value,R.logging.dest.value,R.ui.enable.value,R.ui.route.value,R.other.noLogo.value;const L=["options","globalOptions","themeOptions","resources","payload"],C={},O=(e,t="")=>{Object.keys(e).forEach((o=>{if(!["puppeteer","highcharts"].includes(o)){const r=e[o];void 0===r.value?O(r,`${t}.${o}`):C[r.cliName||o]=`${t}.${o}`.substring(1)}}))};O(R);let _={toConsole:!0,toFile:!1,pathCreated:!1,levelsDesc:[{title:"error",color:"red"},{title:"warning",color:"yellow"},{title:"notice",color:"blue"},{title:"verbose",color:"gray"}],listeners:[]};for(const[e,t]of Object.entries(R.logging))_[e]=t.value;const A=(...e)=>{const[i,...n]=e,{level:s,levelsDesc:a}=_;if(0===i||i>s||s>a.length)return;const l=`${(new Date).toString().split("(")[0].trim()} [${a[i-1].title}] -`;_.listeners.forEach((e=>{e(l,n.join(" "))})),_.toFile&&(_.pathCreated||(!t(_.dest)&&o(_.dest),_.pathCreated=!0),r(`${_.dest}${_.file}`,[l].concat(n).join(" ")+"\n",(e=>{e&&(console.log(`[logger] Unable to write to log file: ${e}`),_.toFile=!1)}))),_.toConsole&&console.log.apply(void 0,[l.toString()[_.levelsDesc[i-1].color]].concat(n))},$=w(new URL("../.",import.meta.url)),I=(e,t=/\s\s+/g,o=" ")=>e.replaceAll(t,o).trim(),P=(e,t)=>{const o=["png","jpeg","pdf","svg"];if(t){const r=t.split(".").pop();o.includes(r)&&e!==r&&(e=r)}return{"image/png":"png","image/jpeg":"jpeg","application/pdf":"pdf","image/svg+xml":"svg"}[e]||o.find((t=>t===e))||"png"},j=(e=!1,t)=>{const o=["js","css","files"];let r=e,n=!1;if(t&&e.endsWith(".json"))try{e?e&&e.endsWith(".json")?r=N(i(e,"utf8")):(r=N(e),!0===r&&(r=N(i("resources.json","utf8")))):r=N(i("resources.json","utf8"))}catch(e){return A(3,"[cli] No resources found.")}else r=N(e),t||delete r.files;for(const e in r)o.includes(e)?n||(n=!0):delete r[e];return n?(r.files&&(r.files=r.files.map((e=>e.trim())),(!r.files||r.files.length<=0)&&delete r.files),r):A(3,"[cli] No resources found.")};function N(e,t){try{const o=JSON.parse("string"!=typeof e?JSON.stringify(e):e);return"string"!=typeof o&&t?JSON.stringify(o):o}catch(e){return!1}}const G=e=>{if(null===e||"object"!=typeof e)return e;const t=Array.isArray(e)?[]:{};for(const o in e)Object.prototype.hasOwnProperty.call(e,o)&&(t[o]=G(e[o]));return t},U=(e,t)=>JSON.stringify(e,((e,o)=>("string"==typeof o&&((o=o.trim()).startsWith("function(")||o.startsWith("function ("))&&o.endsWith("}")&&(o=t?`EXP_FUN${(o+"").replaceAll(/\n|\t|\r/g," ")}EXP_FUN`:void 0),"function"==typeof o?`EXP_FUN${(o+"").replaceAll(/\n|\t|\r/g," ")}EXP_FUN`:o))).replaceAll(/"EXP_FUN|EXP_FUN"/g,"");function W(){console.log("Usage of CLI arguments:".bold,"\n------",`\nFor more detailed information visit readme at: ${"https://github.com/highcharts/node-export-server#readme".bold.yellow}.`);const e=t=>{for(const[o,r]of Object.entries(t))if(Object.prototype.hasOwnProperty.call(r,"value")){let e=` --${r.cliName||o} ${("<"+r.type+">").green} `;if(e.length<48)for(let t=e.length;t<48;t++)e+=".";console.log(e,r.description,`[Default: ${r.value.toString().bold}]`.blue)}else e(r)};Object.keys(R).forEach((t=>{["puppeteer","highcharts"].includes(t)||(console.log(`\n${t.toUpperCase()}`.red),e(R[t]))})),console.log("\n")}const F=e=>!["false","undefined","null","NaN","0",""].includes(e)&&!!e,M=(e,t)=>{if(e&&"string"==typeof e)return(e=e.trim()).endsWith(".js")?!!t&&M(i(e,"utf8")):e.startsWith("function()")||e.startsWith("function ()")||e.startsWith("()=>")||e.startsWith("() =>")?`(${e})()`:e.replace(/;$/,"")};var q=(e,t)=>{const o="Too many requests, you have been rate limited. Please try again later.",r={max:t.maxRequests||30,window:t.window||1,delay:t.delay||0,trustProxy:t.trustProxy||!1,skipKey:t.skipKey||!1,skipToken:t.skipToken||!1};r.trustProxy&&e.enable("trust proxy");const i=y({windowMs:60*r.window*1e3,max:r.max,delayMs:r.delay,handler:(e,t)=>{t.format({json:()=>{t.status(429).send({message:o})},default:()=>{t.status(429).send(o)}})},skip:e=>!1!==r.skipKey&&!1!==r.skipToken&&e.query.key===r.skipKey&&e.query.access_token===r.skipToken&&(A(4,"[rate-limiting] Skipping rate limiter."),!0)});e.use(i),A(3,I(`[rate-limiting] Enabled rate limiting: ${r.max} requests\n per ${r.window} minute per IP, trusting proxy:\n ${r.trustProxy}.`))};async function D(e,t={}){return new Promise(((o,r)=>{const i=(e=>e.startsWith("https")?f:m)(e);i.get(e,t,(e=>{let t="";e.on("data",(e=>{t+=e})),e.on("end",(()=>{t||r("Nothing was fetched from the URL."),e.text=t,o(e)}))})).on("error",(e=>{r(e)}))}))}v.config();const V=c($,".cache"),J={cdnURL:"https://code.highcharts.com/",activeManifest:{},sources:"",hcVersion:""};let z=!1;const K=()=>J.hcVersion=J.sources.substr(0,J.sources.indexOf("*/")).replace("/*","").replace("*/","").replace(/\n/g,"").trim(),X=async(e,t)=>{try{e.endsWith(".js")&&(e=e.substring(0,e.length-3)),A(4,`[cache] Fetching script - ${e}.js`);const o=t?{agent:t,timeout:+process.env.PROXY_SERVER_TIMEOUT||5e3}:{},r=await D(`${e}.js`,o);if(200===r.statusCode)return r.text;throw`${r.statusCode}`}catch(t){throw A(1,`[cache] Error fetching script ${e}.js: ${t}.`),t}},B=async(e,t)=>{const{coreScripts:o,modules:r,indicators:i,scripts:s}=e,a="latest"!==e.version&&e.version?`${e.version}/`:"";A(3,"[cache] Updating cache to Highcharts ",a);const l=[...o.map((e=>`${a}${e}`)),...r.map((e=>"map"===e?`maps/${a}modules/${e}`:`${a}modules/${e}`)),...i.map((e=>`stock/${a}indicators/${e}`))];let c;const p=process.env.PROXY_SERVER_HOST,u=process.env.PROXY_SERVER_PORT;p&&u&&(c=new T({host:p,port:+u}));const d={};try{return J.sources=(await Promise.all([...l.map((async t=>{const o=await X(`${e.cdnURL||J.cdnURL}${t}`,c);return"string"==typeof o&&(d[t.replace(/(.*)\/|(.*)modules\/|stock\/(.*)indicators\/|maps\/(.*)modules\//gi,"")]=1),o})),...s.map((e=>X(e,c)))])).join(";\n"),K(),n(t,J.sources),d}catch(e){A(1,"[cache] Unable to update local Highcharts cache.")}},Y=async e=>{let r;const s=c(V,"manifest.json"),a=c(V,"sources.js");if(z=e,!t(V)&&o(V),!t(s)||e.forceFetch)A(3,"[cache] Fetching and caching Highcharts dependencies."),r=await B(e,a);else{let t=!1;const o=JSON.parse(i(s));if(o.modules&&Array.isArray(o.modules)){const e={};o.modules.forEach((t=>e[t]=1)),o.modules=e}const{modules:n,coreScripts:l,indicators:c}=e,p=n.length+l.length+c.length;o.version!==e.version?(A(3,"[cache] Highcharts version mismatch in cache, need to re-fetch."),t=!0):Object.keys(o.modules||{}).length!==p?(A(3,"[cache] Cache and requested modules does not match, need to re-fetch."),t=!0):t=(e.modules||[]).some((e=>{if(!o.modules[e])return A(3,`[cache] The ${e} missing in cache, need to re-fetch.`),!0})),t?r=await B(e,a):(A(3,"[cache] Dependency cache is up to date, proceeding."),J.sources=i(a,"utf8"),r=o.modules,K())}await(async(e,t)=>{const o={version:e.version,modules:t||{}};J.activeManifest=o,A(4,"[cache] writing new manifest");try{n(c(V,"manifest.json"),JSON.stringify(o),"utf8")}catch(e){A(1,`[cache] Error writing cache manifest: ${e}.`)}})(e,r)};var Q=async e=>!!z&&await Y(Object.assign(z,{version:e})),Z=()=>J,ee=()=>J.hcVersion;const te=E(64).toString("base64url"),oe=H.join("tmp",`puppeteer-${te}`),re=[`--user-data-dir=${H.join(oe,"profile")}`,"--autoplay-policy=user-gesture-required","--disable-background-networking","--disable-background-timer-throttling","--disable-backgrounding-occluded-windows","--disable-breakpad","--disable-client-side-phishing-detection","--disable-component-update","--disable-default-apps","--disable-dev-shm-usage","--disable-domain-reliability","--disable-extensions","--disable-features=AudioServiceOutOfProcess","--disable-hang-monitor","--disable-ipc-flooding-protection","--disable-notifications","--disable-offer-store-unmasked-wallet-cards","--disable-popup-blocking","--disable-print-preview","--disable-prompt-on-repost","--disable-renderer-backgrounding","--disable-session-crashed-bubble","--disable-setuid-sandbox","--disable-speech-api","--disable-sync","--hide-crash-restore-bubble","--hide-scrollbars","--ignore-gpu-blacklist","--metrics-recording-only","--mute-audio","--no-default-browser-check","--no-first-run","--no-pings","--no-sandbox","--no-zygote","--password-store=basic","--use-mock-keychain"],ie=b.fileURLToPath(new URL(".",import.meta.url)),ne=e.readFileSync(ie+"/../templates/template.html","utf8");let se;const ae=async()=>{if(!se)return!1;const e=await se.newPage();return await e.setContent(ne),await e.addScriptTag({path:ie+"/../.cache/sources.js"}),await e.evaluate((()=>window.setupHighcharts())),e.on("pageerror",(async t=>{A(1,"[page error]",t),await e.$eval("#container",((e,t)=>{window._displayErrors&&(e.innerHTML=t)}),`

Chart input data error

${t.toString()}`)})),e},le=async()=>{se.connected&&await se.close()};const ce=b.fileURLToPath(new URL(".",import.meta.url)),pe=async(e,t,o)=>await e.evaluate(((e,t)=>window.triggerExport(e,t)),t,o);var ue=async(e,t,o)=>{const r=[],n=async e=>{for(const e of r)await e.dispose();await e.evaluate((()=>{const[,...e]=document.getElementsByTagName("script"),[,...t]=document.getElementsByTagName("style"),[...o]=document.getElementsByTagName("link");for(const r of[...e,...t,...o])r.remove()}))};try{const s=()=>{};A(4,"[export] Determining export path.");const a=o.export;await e.evaluate((()=>requestAnimationFrame((()=>{}))));const c=a?.options?.chart?.displayErrors&&Z().activeManifest.modules.debugger;await e.evaluate((e=>window._displayErrors=e),c);const p=()=>{};let u;if(t.indexOf&&(t.indexOf("=0||t.indexOf("=0)){if(A(4,"[export] Treating as SVG."),"svg"===a.type)return t;u=!0;const o=()=>{};await e.setContent((e=>`\n\n\n \n \n Highcarts Export\n \n \n \n
\n ${e}\n
\n \n\n\n`)(t)),o()}else if(A(4,"[export] Treating as config."),a.strInj){const t=()=>{};await pe(e,{chart:{height:a.height,width:a.width}},o),t()}else{t.chart.height=a.height,t.chart.width=a.width;const r=()=>{};await pe(e,t,o),r()}p();const d=()=>{},h=o.customCode.resources;if(h){if(h.js&&r.push(await e.addScriptTag({content:h.js})),h.files)for(const t of h.files)try{const o=!t.startsWith("http");r.push(await e.addScriptTag(o?{content:i(t,"utf8")}:{url:t}))}catch(e){A(4,"[export] JS file not found.")}const t=()=>{};if(h.css){let t=h.css.match(/@import\s*([^;]*);/g);if(t)for(let i of t)i&&(i=i.replace("url(","").replace("@import","").replace(/"/g,"").replace(/'/g,"").replace(/;/,"").replace(/\)/g,"").trim(),i.startsWith("http")?r.push(await e.addStyleTag({url:i})):o.customCode.allowFileResources&&r.push(await e.addStyleTag({path:l.join(ce,i)})));r.push(await e.addStyleTag({content:h.css.replace(/@import\s*([^;]*);/g,"")||" "}))}t()}d();const g=u?await e.$eval("#chart-container svg:first-of-type",(async(e,t)=>({chartHeight:e.height.baseVal.value*t,chartWidth:e.width.baseVal.value*t})),parseFloat(a.scale)):await e.evaluate((async()=>{const{chartHeight:e,chartWidth:t}=window.Highcharts.charts[0];return{chartHeight:e,chartWidth:t}})),m=()=>{},f=Math.ceil(g?.chartHeight||a.height),v=Math.ceil(g?.chartWidth||a.width);await e.setViewport({height:f,width:v,deviceScaleFactor:u?1:parseFloat(a.scale)});const y=u?e=>{document.body.style.zoom=e,document.body.style.margin="0px"}:()=>{document.body.style.zoom=1};await e.evaluate(y,parseFloat(a.scale));const{height:b,width:w,x:T,y:x}=await(e=>e.$eval("#chart-container",(e=>{const{x:t,y:o,width:r,height:i}=e.getBoundingClientRect();return{x:t,y:o,width:r,height:Math.trunc(i>1?i:500)}})))(e);let k;u||await e.setViewport({width:Math.round(w),height:Math.round(b),deviceScaleFactor:parseFloat(a.scale)}),m();const S=()=>{};if("svg"===a.type)k=await(async e=>await e.$eval("#container svg:first-of-type",(e=>e.outerHTML)))(e);else if("png"===a.type||"jpeg"===a.type)k=await(async(e,t,o,r)=>await Promise.race([e.screenshot({type:t,encoding:o,clip:r,omitBackground:!0}),new Promise(((e,t)=>setTimeout((()=>t(new Error("Rasterization timeout"))),1500)))]))(e,a.type,"base64",{width:v,height:f,x:T,y:x});else{if("pdf"!==a.type)throw`Unsupported output format ${a.type}`;k=await(async(e,t,o,r)=>await e.pdf({height:t+1,width:o,encoding:r}))(e,f,v,"base64")}return await e.evaluate((()=>{const e=Highcharts.charts;if(e.length)for(const t of e)t&&t.destroy(),Highcharts.charts.shift()})),S(),s(),await n(e),k}catch(t){return await n(e),A(1,`[export] Error encountered during export: ${t}`),t}};let de,he=0,ge=0,me=0,fe=0,ve=0,ye={},be=!1;const we={create:async()=>{const e=x();let t=!1;const o=(new Date).getTime();try{if(t=await ae(),!t||t.isClosed())throw"invalid page";A(3,`[pool] Successfully created a worker ${e} - took ${(new Date).getTime()-o} ms.`)}catch(e){throw A(1,`[pool] Error creating a new page in pool entry creation! ${e}`),"Error creating page"}return{id:e,page:t,workCount:Math.round(Math.random()*(ye.workLimit/2))}},validate:e=>!(ye.workLimit&&++e.workCount>ye.workLimit)||(A(3,"[pool] Worker failed validation:",`exceeded work limit (limit is ${ye.workLimit})`),!1),destroy:e=>{A(3,`[pool] Destroying pool entry ${e.id}.`),e.page&&e.page.close()},log:(e,t)=>console.log(`${t}: ${e}`)},Te=async e=>{de=e.puppeteerArgs;try{await(async e=>{const t=[...re,...e||[]];if(!se){let e=0;const o=async()=>{try{A(3,"[browser] attempting to get a browser instance (try",e+")"),se=await S.launch({headless:"new",args:t,userDataDir:"./tmp/"})}catch(t){A(0,"[browser]",t),++e<25?(A(3,"[browser] failed:",t),await new Promise((e=>setTimeout(e,4e3))),await o()):A(0,"Max retries reached")}};try{await o()}catch(e){return A(0,"[browser] Unable to open browser"),!1}if(!se)return A(0,"[browser] Unable to open browser"),!1}return se})(de)}catch(e){A(0,"[pool|browser]",e)}if(ye=e&&e.pool?{...e.pool}:{},A(3,"[pool] Initializing pool:",`min ${ye.initialWorkers}, max ${ye.maxWorkers}.`),be)return A(4,"[pool] Already initialized, please kill it before creating a new one.");ye.listenToProcessExits&&(A(4,"[pool] Attaching exit listeners to the process."),process.on("exit",(async()=>{await xe()})),process.on("SIGINT",((e,t)=>{A(4,`The ${e} event with code: ${t}.`),process.exit(1)})),process.on("SIGTERM",((e,t)=>{A(4,`The ${e} event with code: ${t}.`),process.exit(1)})),process.on("uncaughtException",(async(e,t)=>{A(4,`The ${t} error, message: ${e.message}.`)})));try{be=new k({...we,min:ye.initialWorkers,max:ye.maxWorkers,createRetryIntervalMillis:200,createTimeoutMillis:ye.acquireTimeout,acquireTimeoutMillis:ye.acquireTimeout,destroyTimeoutMillis:ye.acquireTimeout,idleTimeoutMillis:ye.timeoutThreshold,reapIntervalMillis:1e3,propagateCreateError:!1}),be.on("createFail",((e,t)=>{A(1,`[pool] Error when creating worker of an event id ${e}:`,t)})),be.on("acquireFail",((e,t)=>{A(1,`[pool] Error when acquiring worker of an event id ${e}:`,t)})),be.on("destroyFail",((e,t,o)=>{A(1,`[pool] Error when destroying worker of an id ${t.id}, event id ${e}:`,o)})),be.on("release",(e=>{A(4,`[pool] Releasing a worker of an id ${e.id}`)})),be.on("destroySuccess",((e,t)=>{A(4,`[pool] Destroyed a worker of an id ${t.id}`)}));const e=[];for(let t=0;t{be.release(e)})),A(3,`[pool] The pool is ready with ${ye.initialWorkers} initial resources waiting.`)}catch(e){throw A(1,`[pool] Couldn't create the worker pool ${e}`),e}};async function xe(){return A(3,"[pool] Killing all workers."),be.destroyed?(await le(),!0):(await be.destroy(),await le(),!0)}const ke=async(e,t)=>{let o;const r=e=>{throw++fe,o&&be.release(o),"In pool.postWork: "+e};if(A(4,"[pool] Work received, starting to process."),ye.benchmarking&&Se(),++ge,!be)return A(1,"[pool] Work received, but pool has not been started."),r("Pool is not inited but work was posted to it!");try{A(4,"[pool] Acquiring worker"),o=await be.acquire().promise}catch(e){return r(`[pool] Error when acquiring available entry: ${e}`)}if(A(4,"[pool] Acquired worker handle"),!o.page)return r("Resolved worker page is invalid: pool setup is wonky");try{let i=(new Date).getTime();A(4,`[pool] Starting work on pool entry ${o.id}.`);const n=await ue(o.page,e,t);if(n instanceof Error)return"Rasterization timeout"===n.message&&(o.page.close(),o.page=await ae()),r(n);be.release(o);const s=(new Date).getTime()-i;return me+=s,ve=me/++he,A(4,`[pool] Work completed in ${s} ms.`),{data:n,options:t}}catch(e){r(`Error trying to perform puppeteer export: ${e}.`)}};function Se(){const{min:e,max:t,size:o,available:r,borrowed:i,pending:n,spareResourceCapacity:s}=be;A(4,`[pool] The minimum number of resources allowed by pool: ${e}.`),A(4,`[pool] The maximum number of resources allowed by pool: ${t}.`),A(4,`[pool] The number of all resources in pool (free or in use): ${o}.`),A(4,`[pool] The number of resources that are currently available: ${r}.`),A(4,`[pool] The number of resources that are currently acquired: ${i}.`),A(4,`[pool] The number of callers waiting to acquire a resource: ${n}.`),A(4,`[pool] The number of how many more resources can the pool manage/create: ${s}.`)}var He=()=>({min:be.min,max:be.max,size:be.size,available:be.available,borrowed:be.borrowed,pending:be.pending,spareResourceCapacity:be.spareResourceCapacity}),Ee=()=>ge,Re=()=>fe,Le=()=>ve,Ce=()=>he;const Oe=process.env.npm_package_version,_e=new Date;let Ae={};const $e=()=>Ae,Ie=(e,t,o=[])=>{const r=G(e);for(const[e,n]of Object.entries(t))r[e]="object"!=typeof(i=n)||Array.isArray(i)||null===i||o.includes(e)||void 0===r[e]?void 0!==n?n:r[e]:Ie(r[e],n,o);var i;return r};function Pe(e,t={},o=""){Object.keys(e).forEach((r=>{if(!["puppeteer","highcharts"].includes(r)){const i=e[r],n=t&&t[r];let s;void 0===i.value?Pe(i,n,`${o}.${r}`):(void 0!==n&&(i.value=n),i.envLink&&("boolean"===i.type?i.value=F([process.env[i.envLink],i.value].find((e=>e||"false"===e))):"number"===i.type?(s=+process.env[i.envLink],i.value=s>=0?s:i.value):i.type.indexOf("]")>=0&&process.env[i.envLink]?i.value=process.env[i.envLink].split(","):i.value=process.env[i.envLink]||i.value))}}))}function je(e){let t={};for(const[o,r]of Object.entries(e))t[o]=Object.prototype.hasOwnProperty.call(r,"value")?r.value:je(r);return t}let Ne=!1;const Ge=async(e,t)=>{A(4,"[chart] Starting exporting process.");const o=((e,t={})=>{let o={};return e.svg?(o=G(t),o.export.type=e.type||e.export.type,o.export.scale=e.scale||e.export.scale,o.export.outfile=e.outfile||e.export.outfile,o.payload={svg:e.svg}):o=Ie(t,e,L),o.export.outfile=o.export?.outfile||`chart.${o.export?.type||"png"}`,o})(e,$e()),r=o.export;return o.payload?.svg&&""!==o.payload.svg?Me(o.payload.svg.trim(),o,t):r.infile&&r.infile.length?(A(4,"[chart] Attempting to export from an input file."),s(r.infile,"utf8",((e,r)=>e?A(1,`[chart] Error loading input file: ${e}.`):(o.export.instr=r,Me(o.export.instr.trim(),o,t))))):r.instr&&""!==r.instr||r.options&&""!==r.options?(A(4,"[chart] Attempting to export from a raw input."),F(o.customCode?.allowCodeExecution)?Fe(o,t):"string"==typeof r.instr?Me(r.instr.trim(),o,t):We(o,r.instr||r.options,t)):(A(1,I(`[chart] No input specified.\n ${JSON.stringify(r,void 0," ")}.`)),t&&t(!1,{error:!0,message:"No input specified."}))},Ue=e=>{const{chart:t,exporting:o}=e.export?.options||N(e.export?.instr),r=N(e.export?.globalOptions);let i=e.export?.scale||o?.scale||r?.exporting?.scale||e.export?.defaultScale||1;return i=Math.max(.1,Math.min(i,5)),i=((e,t=1)=>{const o=Math.pow(10,t||0);return Math.round(+e*o)/o})(i,2),{height:e.export?.height||o?.sourceHeight||t?.height||r?.exporting?.sourceHeight||r?.chart?.height||e.export?.defaultHeight||400,width:e.export?.width||o?.sourceWidth||t?.width||r?.exporting?.sourceWidth||r?.chart?.width||e.export?.defaultWidth||600,scale:i}},We=(e,t,o,r)=>{let{export:n,customCode:s}=e;const a="boolean"==typeof s.allowCodeExecution?s.allowCodeExecution:Ne;if(s){if("string"==typeof e.customCode.resources)e.customCode.resources=j(e.customCode.resources,F(e.customCode.allowFileResources));else if(!e.customCode.resources)try{const t=i("resources.json","utf8");e.customCode.resources=j(t,F(e.customCode.allowFileResources))}catch(e){A(3,"[chart] The default resources.json file not found.")}}else s=e.customCode={};if(!a&&s){if(s.callback||s.resources||s.customCode)return o&&o(!1,{error:!0,message:I("The callback, resources and customCode have been disabled for this\n server.")});s.callback=!1,s.resources=!1,s.customCode=!1}if(t&&(t.chart=t.chart||{},t.exporting=t.exporting||{},t.exporting.enabled=!1),n.constr=n.constr||"chart",n.type=P(n.type,n.outfile),"svg"===n.type&&(n.width=!1),["globalOptions","themeOptions"].forEach((e=>{try{n&&n[e]&&("string"==typeof n[e]&&n[e].endsWith(".json")?n[e]=N(i(n[e],"utf8"),!0):n[e]=N(n[e],!0))}catch(t){n[e]={},A(1,`[chart] The ${e} not found.`)}})),s.allowCodeExecution&&(s.customCode=M(s.customCode,s.allowFileResources)),s&&s.callback&&s.callback?.indexOf("{")<0)if(s.allowFileResources)try{s.callback=i(s.callback,"utf8")}catch(e){A(2,`[chart] Error loading callback: ${e}.`),s.callback=!1}else s.callback=!1;e.export={...e.export,...Ue(e)},ke(n.strInj||t||r,e).then((e=>o(e))).catch((e=>(A(0,"[chart] When posting work:",e),o(!1,e))))},Fe=(e,t)=>{try{let o,r=e.export.instr||e.export.options;return"string"!=typeof r&&(o=r=U(r,e.customCode?.allowCodeExecution)),o=r.replaceAll(/\t|\n|\r/g,"").trim(),";"===o[o.length-1]&&(o=o.substring(0,o.length-1)),e.export.strInj=o,We(e,!1,t)}catch(o){const r=I(`Malformed input detected for ${e.export?.requestId||"?"}:\n Please make sure that your JSON/JavaScript options\n are sent using the "options" attribute, and that if you're using\n SVG, it is unescaped.`);return A(1,r),t&&t(!1,JSON.stringify({error:!0,message:r}))}},Me=(e,t,o)=>{const{allowCodeExecution:r}=t.customCode;if(e.indexOf("=0||e.indexOf("=0)return A(4,"[chart] Parsing input as SVG."),We(t,!1,o,e);try{const r=JSON.parse(e.replaceAll(/\t|\n|\r/g," "));return We(t,r,o)}catch(e){return F(r)?Fe(t,o):o&&o(!1,{error:!0,message:I("Only JSON configurations and SVG is allowed for this server. If\n this is your server, JavaScript exporting can be enabled by starting\n the server with the --allowCodeExecution flag.")})}},qe={png:"image/png",jpeg:"image/jpeg",gif:"image/gif",pdf:"application/pdf",svg:"image/svg+xml"};let De=0;const Ve=[],Je=[],ze=(e,t,o,r)=>{let i=!0;const{id:n,uniqueId:s,type:a,body:l}=r;return e.some((e=>{if(e){let r=e(t,o,n,s,a,l);return void 0!==r&&!0!==r&&(i=r),!0}})),i},Ke=(e,t)=>{(()=>{const e=process.hrtime.bigint()})();const o=$e(),r=e.body,i=++De,n=x().replace(/-/g,"");let s=P(r.type);if(!r)return t.status(400).send(I("Body is required. Sending a body? Make sure your Content-type header\n is correct. Accepted is application/json and multipart/form-data."));let a=N(r.infile||r.options||r.data);if(!a&&!r.svg)return A(2,I(`Request ${n} from ${e.headers["x-forwarded-for"]||e.connection.remoteAddress} was incorrect. Check your payload.`)),t.status(400).send(I("No correct chart data found. Please make sure you are using\n application/json or multipart/form-data headers, and that the chart\n data is in the 'infile', 'options' or 'data' attribute if sending\n JSON or in the 'svg' if sending SVG."));let l=!1;if(l=ze(Ve,e,t,{id:i,uniqueId:n,type:s,body:r}),!0!==l)return t.send(l);let c=!1;e.socket.on("close",(()=>{c=!0})),A(4,`[export] Got an incoming HTTP request ${n}.`),r.constr="string"==typeof r.constr&&r.constr||"chart";const p={export:{instr:a,type:s,constr:r.constr[0].toLowerCase()+r.constr.substr(1),height:r.height,width:r.width,scale:r.scale||o.export.scale,globalOptions:N(r.globalOptions,!0),themeOptions:N(r.themeOptions,!0)},customCode:{allowCodeExecution:Ne,allowFileResources:!1,resources:N(r.resources,!0),callback:r.callback,customCode:r.customCode}};a&&(p.export.instr=U(a,p.customCode.allowCodeExecution));const u=Ie(o,p);if(u.export.options=a,u.payload={svg:r.svg||!1,b64:r.b64||!1,dataOptions:N(r.dataOptions,!0),noDownload:r.noDownload||!1,requestId:n},r.svg&&(d=u.payload.svg,["localhost","(10).(.*).(.*).(.*)","(127).(.*).(.*).(.*)","(172).(1[6-9]|2[0-9]|3[0-1]).(.*).(.*)","(192).(168).(.*).(.*)"].some((e=>d.match(`xlink:href="(?:(http://|https://))?${e}`)))))return t.status(400).send("SVG potentially contain at least one forbidden URL in xlink:href element.");var d;Ge(u,((o,a)=>(e.socket.removeAllListeners("close"),c?A(3,I("[export] The client closed the connection before the chart was done\n processing.")):a?(A(1,I(`[export] Work: ${n} could not be completed, sending:\n ${a}`)),t.status(400).send(a.message)):o&&o.data?(s=o.options.export.type,ze(Je,e,t,{id:i,body:o.data}),o.data?r.b64?"pdf"===s?t.send(Buffer.from(o.data,"utf8").toString("base64")):t.send(o.data):(t.header("Content-Type",qe[s]||"image/png"),r.noDownload||t.attachment(`${e.params.filename||"chart"}.${s||"png"}`),"svg"===s?t.send(o.data):t.send(Buffer.from(o.data,"base64"))):void 0):(A(1,I(`[export] Unexpected return from chart generation, please check your\n data Request: ${n} is ${o.data}.`)),t.status(400).send("Unexpected return from chart generation, please check your data.")))))};const Xe=h();Xe.disable("x-powered-by"),Xe.use(d());const Be=g.memoryStorage(),Ye=g({storage:Be,limits:{fieldsSize:"50MB"}});Xe.use(Ye.any()),Xe.use(u.json({limit:"50mb"})),Xe.use(u.urlencoded({extended:!0,limit:"50mb"})),Xe.use(u.urlencoded({extended:!1,limit:"50mb"}));const Qe=e=>A(1,`[server] Socket error: ${e}`),Ze=e=>{e.on("clientError",Qe),e.on("error",Qe),e.on("connection",(e=>e.on("error",(e=>Qe(e)))))},et=async e=>{if(!e.enable)return!1;if(!e.ssl.enable&&!e.ssl.force){const t=m.createServer(Xe);Ze(t),t.listen(e.port,e.host),A(3,`[server] Started HTTP server on ${e.host}:${e.port}.`)}if(e.ssl.enable){let t,o;try{t=await a.readFile(p.join(e.ssl.certPath,"server.key"),"utf8"),o=await a.readFile(p.join(e.ssl.certPath,"server.crt"),"utf8")}catch(t){A(1,`[server] Unable to load key/certificate from ${e.ssl.certPath}.`)}if(t&&o){const t=f.createServer(Xe);Ze(t),t.listen(e.ssl.port,e.host),A(3,`[server] Started HTTPS server on ${e.host}:${e.ssl.port}.`)}}e.rateLimiting&&e.rateLimiting.enable&&![0,NaN].includes(e.rateLimiting.maxRequests)&&q(Xe,e.rateLimiting),Xe.use(h.static(p.join($,"public"))),(e=>{!!e&&e.get("/health",((e,t)=>{t.send({status:"OK",bootTime:_e,uptime:Math.floor(((new Date).getTime()-_e.getTime())/1e3/60)+" minutes",version:Oe,highchartsVersion:ee(),averageProcessingTime:Le(),performedExports:Ce(),failedExports:Re(),exportAttempts:Ee(),sucessRatio:Ce()/Ee()*100,pool:He()})}))})(Xe),(e=>{e.post("/",Ke),e.post("/:filename",Ke)})(Xe),(e=>{!!e&&e.get("/",((e,t)=>{t.sendFile(c($,"public","index.html"))}))})(Xe),(e=>{!!e&&e.post("/change_hc_version/:newVersion",(async(e,t)=>{const o=process.env.HIGHCHARTS_ADMIN_TOKEN;if(!o||!o.length)return t.send({error:!0,message:"Server not configured to do run-time version changes: HIGHCHARTS_ADMIN_TOKEN not set"});const r=e.get("hc-auth");if(!r||r!==o)return t.send({error:!0,message:"Invalid or missing token: set token in the hc-auth header"});const i=e.params.newVersion;if(i){try{await Q(i)}catch(e){t.send({error:!0,message:e})}t.send({version:ee()})}else t.send({error:!0,message:"No new version supplied"})}))})(Xe)};var tt={startServer:et,getExpress:()=>h,getApp:()=>Xe,use:(e,...t)=>{Xe.use(e,...t)},get:(e,...t)=>{Xe.get(e,...t)},post:(e,...t)=>{Xe.post(e,...t)},enableRateLimiting:e=>q(Xe,e)},ot={log:A,mapToNewConfig:e=>{const t={};for(const[o,r]of Object.entries(e)){const e=C[o]?C[o].split("."):[];e.reduce(((t,o,i)=>t[o]=e.length-1===i?r:t[o]||{}),t)}return t},setOptions:(e,t)=>(t?.length&&(Ae=function(e){const t=e.findIndex((e=>"loadConfig"===e.replace(/-/g,"")));if(t>-1&&e[t+1]){const o=e[t+1];try{if(o&&o.endsWith(".json"))return JSON.parse(i(o))}catch(e){A(1,`[config] Unable to load config from the ${o}: ${e}`)}}return{}}(t)),Pe(R,Ae),Ae=je(R),e&&(Ae=Ie(Ae,e,L)),t?.length&&(Ae=function(e,t,o){for(let o=0;o(i.length-1===a&&void 0!==n[s]&&(t[++o]?n[s]=t[o]||n[s]:(console.log(`Missing argument value for ${r}!`.red,"\n"),e=W())),n[s])),e)}return e}(Ae,t)),Ae),singleExport:e=>{e.export.instr=e.export.instr||e.export.options,Ge(e,((e,t)=>{t&&(A(1,`[cli] ${t.message}`),process.exit(1));const{outfile:o,type:r}=e.options.export;n(o||`chart.${r}`,"svg"!==r?Buffer.from(e.data,"base64"):e.data),xe()}))},startExport:Ge,batchExport:e=>{const t=[];for(let o of e.export.batch.split(";"))o=o.split("="),2===o.length&&t.push(new Promise(((t,r)=>{Ge({...e,export:{...e.export,infile:o[0],outfile:o[1]}},((e,o)=>{if(o)return r(o);n(e.options.export.outfile,Buffer.from(e.data,"base64")),t()}))})));Promise.all(t).then((()=>{xe()})).catch((e=>{A(1,`[chart] Error encountered during batch export: ${e}`),xe()}))},server:tt,startServer:et,killPool:xe,initPool:async(e={})=>{var t,o;return t=e.customCode&&e.customCode.allowCodeExecution,Ne=F(t),(o=e.logging&&parseInt(e.logging.level))>=0&&o<=_.levelsDesc.length&&(_.level=o),e.logging&&e.logging.dest&&((e,t)=>{if(_={..._,dest:e||_.dest,file:t||_.file,toFile:!0},0===_.dest.length)return A(1,"[logger] File logging init: no path supplied.");_.dest.endsWith("/")||(_.dest+="/")})(e.logging.dest,e.logging.file||"highcharts-export-server.log"),await Y(e.highcharts||{version:"latest"}),await Te({pool:e.pool||{initialWorkers:1,maxWorkers:1},puppeteerArgs:e.puppeteer?.args||[]}),e}};export{ot as default}; //# sourceMappingURL=index.esm.js.map diff --git a/dist/index.esm.js.map b/dist/index.esm.js.map index 6aadfc70..54adbb3a 100644 --- a/dist/index.esm.js.map +++ b/dist/index.esm.js.map @@ -1 +1 @@ -{"version":3,"file":"index.esm.js","sources":["../lib/schemas/config.js","../lib/logger.js","../lib/utils.js","../lib/server/rate_limit.js","../lib/fetch.js","../lib/cache.js","../lib/browser.js","../lib/export.js","../lib/benchmark.js","../templates/svg_export/svg_export.js","../lib/pool.js","../lib/server/routes/health.js","../lib/config.js","../lib/chart.js","../lib/server/routes/export.js","../lib/server/server.js","../lib/server/routes/ui.js","../lib/server/routes/change_hc_version.js","../lib/index.js"],"sourcesContent":["/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\n// Load .env into environment variables\nimport dotenv from 'dotenv';\n\ndotenv.config();\n\n// This is the configuration object with all options and their default values,\n// also from the .env file if one exists\nexport const defaultConfig = {\n puppeteer: {\n args: {\n value: [],\n type: 'string[]',\n description: 'Array of arguments to send to puppeteer.'\n }\n },\n highcharts: {\n version: {\n value: 'latest',\n envLink: 'HIGHCHARTS_VERSION',\n type: 'string',\n description: 'Highcharts version to use.'\n },\n cdnURL: {\n value: 'https://code.highcharts.com/',\n envLink: 'HIGHCHARTS_CDN',\n type: 'string',\n description: 'The CDN URL of Highcharts scripts to use.'\n },\n coreScripts: {\n envLink: 'HIGHCHARTS_CORE_SCRIPTS',\n value: ['highcharts', 'highcharts-more', 'highcharts-3d'],\n type: 'string[]',\n description: 'Highcharts core scripts to fetch.'\n },\n modules: {\n envLink: 'HIGHCHARTS_MODULES',\n value: [\n 'stock',\n 'map',\n 'gantt',\n 'exporting',\n 'export-data',\n 'parallel-coordinates',\n 'accessibility',\n 'annotations-advanced',\n 'boost-canvas',\n 'boost',\n 'data',\n 'draggable-points',\n 'static-scale',\n 'broken-axis',\n 'heatmap',\n 'tilemap',\n 'timeline',\n 'treemap',\n 'item-series',\n 'drilldown',\n 'histogram-bellcurve',\n 'bullet',\n 'funnel',\n 'funnel3d',\n 'pyramid3d',\n 'networkgraph',\n 'pareto',\n 'pattern-fill',\n 'pictorial',\n 'price-indicator',\n 'sankey',\n 'arc-diagram',\n 'dependency-wheel',\n 'series-label',\n 'solid-gauge',\n 'sonification',\n 'stock-tools',\n 'streamgraph',\n 'sunburst',\n 'variable-pie',\n 'variwide',\n 'vector',\n 'venn',\n 'windbarb',\n 'wordcloud',\n 'xrange',\n 'no-data-to-display',\n 'drag-panes',\n 'debugger',\n 'dumbbell',\n 'lollipop',\n 'cylinder',\n 'organization',\n 'dotplot',\n 'marker-clusters',\n 'hollowcandlestick',\n 'heikinashi'\n ],\n type: 'string[]',\n description: 'Highcharts modules to fetch.'\n },\n indicators: {\n envLink: 'HIGHCHARTS_INDICATORS',\n value: ['indicators-all'],\n type: 'string[]',\n description: 'Highcharts indicators to fetch.'\n },\n scripts: {\n value: [\n 'https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.29.4/moment.min.js',\n 'https://cdnjs.cloudflare.com/ajax/libs/moment-timezone/0.5.34/moment-timezone-with-data.min.js'\n ],\n type: 'string[]',\n description:\n 'Additional direct scripts/optional dependencies (e.g. moment.js).'\n },\n forceFetch: {\n envLink: 'HIGHCHARTS_FORCE_FETCH',\n value: false,\n type: 'boolean',\n description:\n 'Should all the scripts be refetched after rerunning the server.'\n }\n },\n export: {\n infile: {\n value: false,\n type: 'string',\n description:\n 'The input file name along with a type (json or svg). It can be a correct JSON or SVG file.'\n },\n instr: {\n value: false,\n type: 'string',\n description:\n 'An input in a form of a stringified JSON or SVG file. Overrides the --infile.'\n },\n options: {\n value: false,\n type: 'string',\n description: 'An alias for the --instr option.'\n },\n outfile: {\n value: false,\n type: 'string',\n description:\n 'The output filename along with a type (jpeg, png, pdf or svg). Ignores the --type flag.'\n },\n type: {\n envLink: 'EXPORT_DEFAULT_TYPE',\n value: 'png',\n type: 'string',\n description:\n 'The format of the file to export to. Can be jpeg, png, pdf or svg.'\n },\n constr: {\n envLink: 'EXPORT_DEFAULT_CONSTR',\n value: 'chart',\n type: 'string',\n description:\n 'The constructor to use. Can be chart, stockChart, mapChart or ganttChart.'\n },\n defaultHeight: {\n envLink: 'EXPORT_DEFAULT_HEIGHT',\n value: 400,\n type: 'number',\n description:\n 'The default height of the exported chart. Used when not found any value set.'\n },\n defaultWidth: {\n envLink: 'EXPORT_DEFAULT_WIDTH',\n value: 600,\n type: 'number',\n description:\n 'The default width of the exported chart. Used when not found any value set.'\n },\n defaultScale: {\n envLink: 'EXPORT_DEFAULT_SCALE',\n value: 1,\n type: 'number',\n description:\n 'The default scale of the exported chart. Ranges between 1 and 5.'\n },\n height: {\n type: 'number',\n value: false,\n description:\n 'The default height of the exported chart. Overrides the option in the chart settings.'\n },\n width: {\n type: 'number',\n value: false,\n description:\n 'The width of the exported chart. Overrides the option in the chart settings.'\n },\n scale: {\n value: false,\n type: 'number',\n description: 'The scale of the exported chart. Ranges between 1 and 5.'\n },\n globalOptions: {\n value: false,\n type: 'string',\n description:\n 'A stringified JSON or a filename with options to be passed into the Highcharts.setOptions.'\n },\n themeOptions: {\n value: false,\n type: 'string',\n description:\n 'A stringified JSON or a filename with theme options to be passed into the Highcharts.setOptions.'\n },\n batch: {\n value: false,\n type: 'string',\n description:\n 'Starts a batch job. A string that contains input/output pairs: \"in=out;in=out;..\".'\n }\n },\n customCode: {\n allowCodeExecution: {\n envLink: 'HIGHCHARTS_ALLOW_CODE_EXECUTION',\n value: false,\n type: 'boolean',\n description:\n 'If set to true, allow for the execution of arbitrary code when exporting.'\n },\n allowFileResources: {\n envLink: 'HIGHCHARTS_ALLOW_FILE_RESOURCES',\n value: true,\n type: 'boolean',\n description:\n 'Allow injecting resources from the filesystem. Has no effect when running as a server.'\n },\n customCode: {\n value: false,\n type: 'string',\n description:\n 'A function to be called before chart initialization. Can be a filename with the js extension.'\n },\n callback: {\n value: false,\n type: 'string',\n description: 'A JavaScript file with a function to run on construction.'\n },\n resources: {\n value: false,\n type: 'string',\n description:\n 'An additional resource in a form of stringified JSON. It can contain files, js and css sections.'\n },\n loadConfig: {\n value: false,\n type: 'string',\n description: 'A file that contains a pre-defined config to use.'\n },\n createConfig: {\n value: false,\n type: 'string',\n description:\n 'Allows to set options through a prompt and save in a provided config file.'\n }\n },\n server: {\n enable: {\n envLink: 'HIGHCHARTS_SERVER_ENABLE',\n value: false,\n type: 'boolean',\n cliName: 'enableServer',\n description: 'If set to true, starts a server on 0.0.0.0.'\n },\n host: {\n envLink: 'HIGHCHARTS_SERVER_HOST',\n value: '0.0.0.0',\n type: 'string',\n description:\n 'The hostname of the server. Also starts a server listening on the supplied hostname.'\n },\n port: {\n envLink: 'HIGHCHARTS_SERVER_PORT',\n value: 7801,\n type: 'number',\n description: 'The port to use for the server. Defaults to 7801.'\n },\n ssl: {\n enable: {\n envLink: 'HIGHCHARTS_SERVER_SSL_ENABLE',\n value: false,\n type: 'boolean',\n cliName: 'enableSsl',\n description: 'Enables the SSL protocol.'\n },\n force: {\n envLink: 'HIGHCHARTS_SERVER_SSL_FORCE',\n value: false,\n type: 'boolean',\n cliName: 'sslForced',\n description:\n 'If set to true, forces the server to only serve over HTTPS.'\n },\n port: {\n envLink: 'HIGHCHARTS_SERVER_SSL_PORT',\n value: 443,\n type: 'number',\n cliName: 'sslPort',\n description: 'The port on which to run the SSL server.'\n },\n certPath: {\n envLink: 'HIGHCHARTS_SSL_CERT_PATH',\n value: '',\n type: 'string',\n description: 'The path to the SSL certificate/key.'\n }\n },\n rateLimiting: {\n enable: {\n envLink: 'HIGHCHARTS_RATE_LIMIT_ENABLE',\n value: false,\n type: 'boolean',\n cliName: 'enableRateLimiting',\n description: 'Enables rate limiting.'\n },\n maxRequests: {\n envLink: 'HIGHCHARTS_RATE_LIMIT_MAX',\n value: 10,\n type: 'number',\n description: 'Max requests allowed in a one minute.'\n },\n window: {\n envLink: 'HIGHCHARTS_RATE_LIMIT_WINDOW',\n value: 1,\n type: 'number',\n description: 'The time window in minutes for rate limiting.'\n },\n delay: {\n envLink: 'HIGHCHARTS_RATE_LIMIT_DELAY',\n value: 0,\n type: 'number',\n description:\n 'The amount to delay each successive request before hitting the max.'\n },\n trustProxy: {\n envLink: 'HIGHCHARTS_RATE_LIMIT_TRUST_PROXY',\n value: false,\n type: 'boolean',\n description: 'Set this to true if behind a load balancer.'\n },\n skipKey: {\n envLink: 'HIGHCHARTS_RATE_LIMIT_SKIP_KEY',\n value: '',\n type: 'number|string',\n description:\n 'Allows bypassing the rate limiter and should be provided with skipToken argument.'\n },\n skipToken: {\n envLink: 'HIGHCHARTS_RATE_LIMIT_SKIP_TOKEN',\n value: '',\n type: 'number|string',\n description:\n 'Allows bypassing the rate limiter and should be provided with skipKey argument.'\n }\n }\n },\n pool: {\n initialWorkers: {\n envLink: 'HIGHCHARTS_POOL_MIN_WORKERS',\n value: 4,\n type: 'number',\n description: 'The number of initial workers to spawn.'\n },\n maxWorkers: {\n envLink: 'HIGHCHARTS_POOL_MAX_WORKERS',\n value: 8,\n type: 'number',\n description: 'The number of max workers to spawn.'\n },\n workLimit: {\n envLink: 'HIGHCHARTS_POOL_WORK_LIMIT',\n value: 40,\n type: 'number',\n description:\n 'The pieces of work that can be performed before restarting process.'\n },\n queueSize: {\n envLink: 'HIGHCHARTS_POOL_QUEUE_SIZE',\n value: 5,\n type: 'number',\n description: 'The size of the request overflow queue.'\n },\n timeoutThreshold: {\n envLink: 'HIGHCHARTS_POOL_TIMEOUT',\n value: 5000,\n type: 'number',\n description: 'The number of milliseconds before timing out.'\n },\n acquireTimeout: {\n envLink: 'HIGHCHARTS_POOL_ACQUIRE_TIMEOUT',\n value: 5000,\n type: 'number',\n description:\n 'The number of milliseconds to wait for acquiring a resource.'\n },\n reaper: {\n envLink: 'HIGHCHARTS_POOL_ENABLE_REAPER',\n value: true,\n type: 'boolean',\n description:\n 'Whether or not to evict workers after a certain time period.'\n },\n benchmarking: {\n envLink: 'HIGHCHARTS_POOL_BENCHMARKING',\n value: false,\n type: 'boolean',\n description: 'Enable benchmarking.'\n },\n listenToProcessExits: {\n envLink: 'HIGHCHARTS_POOL_LISTEN_TO_PROCESS_EXITS',\n value: true,\n type: 'boolean',\n description:\n 'Set to false in order to skip attaching process.exit handlers.'\n }\n },\n logging: {\n level: {\n envLink: 'HIGHCHARTS_LOG_LEVEL',\n value: 4,\n type: 'number',\n cliName: 'logLevel',\n description:\n 'The log level (0: silent, 1: error, 2: warning, 3: notice, 4: verbose).'\n },\n file: {\n envLink: 'HIGHCHARTS_LOG_FILE',\n value: 'highcharts-export-server.log',\n type: 'string',\n cliName: 'logFile',\n description:\n 'A name of a log file. The --logDest also needs to be set to enable file logging.'\n },\n dest: {\n envLink: 'HIGHCHARTS_LOG_DEST',\n value: 'log/',\n type: 'string',\n cliName: 'logDest',\n description: 'The path to store log files. Also enables file logging.'\n }\n },\n ui: {\n enable: {\n envLink: 'HIGHCHARTS_UI_ENABLE',\n value: false,\n type: 'boolean',\n cliName: 'enableUi',\n description: 'Enables the UI for the export server.'\n },\n route: {\n envLink: 'HIGHCHARTS_UI_ROUTE',\n value: '/',\n type: 'string',\n cliName: 'uiRoute',\n description: 'The route to attach the UI to.'\n }\n },\n other: {\n noLogo: {\n envLink: 'HIGHCHARTS_NO_LOGO',\n value: false,\n type: 'boolean',\n description:\n 'Skip printing the logo on a startup. Will be replaced by a simple text.'\n }\n },\n payload: {}\n};\n\n// The config descriptions object for the prompts functionality. It contains\n// information like:\n// * Type of a prompt\n// * Name of an option\n// * Short description of a chosen option\n// * Initial value\nexport const promptsConfig = {\n puppeteer: [\n {\n type: 'list',\n name: 'args',\n message: 'Puppeteer arguments',\n initial: defaultConfig.puppeteer.args.value.join(','),\n separator: ','\n }\n ],\n highcharts: [\n {\n type: 'text',\n name: 'version',\n message: 'Highcharts version',\n initial: defaultConfig.highcharts.version.value\n },\n {\n type: 'text',\n name: 'cdnURL',\n message: 'The url of CDN',\n initial: defaultConfig.highcharts.cdnURL.value\n },\n {\n type: 'multiselect',\n name: 'modules',\n message: 'Available modules',\n instructions: 'Space: Select specific, A: Select all, Enter: Confirm.',\n choices: defaultConfig.highcharts.modules.value\n },\n {\n type: 'list',\n name: 'scripts',\n message: 'Custom scripts',\n initial: defaultConfig.highcharts.scripts.value.join(','),\n separator: ','\n },\n {\n type: 'toggle',\n name: 'forceFetch',\n message: 'Should refetch all the scripts after each server rerun',\n initial: defaultConfig.highcharts.forceFetch.value\n }\n ],\n export: [\n {\n type: 'select',\n name: 'type',\n message: 'The default type of a file to export to',\n hint: `Default: ${defaultConfig.export.type.value}`,\n initial: 0,\n choices: ['png', 'jpeg', 'pdf', 'svg']\n },\n {\n type: 'select',\n name: 'constr',\n message: 'The default constructor for Highcharts to use',\n hint: `Default: ${defaultConfig.export.constr.value}`,\n initial: 0,\n choices: ['chart', 'stockChart', 'mapChart', 'ganttChart']\n },\n {\n type: 'number',\n name: 'defaultHeight',\n message: 'The default fallback height of the exported chart',\n initial: defaultConfig.export.defaultHeight.value\n },\n {\n type: 'number',\n name: 'defaultWidth',\n message: 'The default fallback width of the exported chart',\n initial: defaultConfig.export.defaultWidth.value\n },\n {\n type: 'number',\n name: 'defaultScale',\n message: 'The default fallback scale of the exported chart',\n initial: defaultConfig.export.defaultScale.value,\n min: 0.1,\n max: 5\n }\n ],\n customCode: [\n {\n type: 'toggle',\n name: 'allowCodeExecution',\n message: 'Allow to execute custom code',\n initial: defaultConfig.customCode.allowCodeExecution.value\n },\n {\n type: 'toggle',\n name: 'allowFileResources',\n message: 'Allow file resources',\n initial: defaultConfig.customCode.allowFileResources.value\n }\n ],\n server: [\n {\n type: 'toggle',\n name: 'enable',\n message: 'Starts a server on 0.0.0.0',\n initial: defaultConfig.server.enable.value\n },\n {\n type: 'text',\n name: 'host',\n message: 'A hostname of a server',\n initial: defaultConfig.server.host.value\n },\n {\n type: 'number',\n name: 'port',\n message: 'A port of a server',\n initial: defaultConfig.server.port.value\n },\n {\n type: 'toggle',\n name: 'ssl.enable',\n message: 'Enable SSL protocol',\n initial: defaultConfig.server.ssl.enable.value\n },\n {\n type: 'toggle',\n name: 'ssl.force',\n message: 'Force to only serve over HTTPS',\n initial: defaultConfig.server.ssl.force.value\n },\n {\n type: 'number',\n name: 'ssl.port',\n message: 'Port on which to run the SSL server',\n initial: defaultConfig.server.ssl.port.value\n },\n {\n type: 'text',\n name: 'ssl.certPath',\n message: 'A path where to find the SSL certificate/key',\n initial: defaultConfig.server.ssl.certPath.value\n },\n {\n type: 'toggle',\n name: 'rateLimiting.enable',\n message: 'Enable rate limiting',\n initial: defaultConfig.server.rateLimiting.enable.value\n },\n {\n type: 'number',\n name: 'rateLimiting.maxRequests',\n message: 'Max requests allowed in a one minute',\n initial: defaultConfig.server.rateLimiting.maxRequests.value\n },\n {\n type: 'number',\n name: 'rateLimiting.window',\n message: 'The time window in minutes for rate limiting',\n initial: defaultConfig.server.rateLimiting.window.value\n },\n {\n type: 'number',\n name: 'rateLimiting.delay',\n message:\n 'The amount to delay each successive request before hitting the max',\n initial: defaultConfig.server.rateLimiting.delay.value\n },\n {\n type: 'toggle',\n name: 'rateLimiting.trustProxy',\n message: 'Set this to true if behind a load balancer',\n initial: defaultConfig.server.rateLimiting.trustProxy.value\n },\n {\n type: 'text',\n name: 'rateLimiting.skipKey',\n message:\n 'Allows bypassing the rate limiter and should be provided with skipToken argument',\n initial: defaultConfig.server.rateLimiting.skipKey.value\n },\n {\n type: 'text',\n name: 'rateLimiting.skipToken',\n message:\n 'Allows bypassing the rate limiter and should be provided with skipKey argument',\n initial: defaultConfig.server.rateLimiting.skipToken.value\n }\n ],\n pool: [\n {\n type: 'number',\n name: 'initialWorkers',\n message: 'The number of initial workers to spawn',\n initial: defaultConfig.pool.initialWorkers.value\n },\n {\n type: 'number',\n name: 'maxWorkers',\n message: 'The number of max workers to spawn',\n initial: defaultConfig.pool.maxWorkers.value\n },\n {\n type: 'number',\n name: 'workLimit',\n message:\n 'The pieces of work that can be performed before restarting a puppeteer process',\n initial: defaultConfig.pool.workLimit.value\n },\n {\n type: 'number',\n name: 'queueSize',\n message: 'The size of the request overflow queue',\n initial: defaultConfig.pool.queueSize.value\n },\n {\n type: 'number',\n name: 'timeoutThreshold',\n message: 'The number of seconds before timing out',\n initial: defaultConfig.pool.timeoutThreshold.value\n },\n {\n type: 'number',\n name: 'acquireTimeout',\n message: 'The number of milliseconds to wait for acquiring a resource',\n initial: defaultConfig.pool.acquireTimeout.value\n },\n {\n type: 'toggle',\n name: 'reaper',\n message: 'The reaper to remove hanging processes',\n initial: defaultConfig.pool.reaper.value\n },\n {\n type: 'toggle',\n name: 'benchmarking',\n message: 'Set benchmarking',\n initial: defaultConfig.pool.benchmarking.value\n },\n {\n type: 'toggle',\n name: 'listenToProcessExits',\n message: 'Set to false in order to skip attaching process.exit handlers',\n initial: defaultConfig.pool.listenToProcessExits.value\n }\n ],\n logging: [\n {\n type: 'number',\n name: 'level',\n message:\n 'The log level (0: silent, 1: error, 2: warning, 3: notice, 4: verbose)',\n initial: defaultConfig.logging.level.value,\n round: 0,\n min: 0,\n max: 4\n },\n {\n type: 'text',\n name: 'file',\n message:\n 'A name of a log file. The --logDest also needs to be set to enable file logging',\n initial: defaultConfig.logging.file.value\n },\n {\n type: 'text',\n name: 'dest',\n message: 'A path to log files. It enables file logging',\n initial: defaultConfig.logging.dest.value\n }\n ],\n ui: [\n {\n type: 'toggle',\n name: 'enable',\n message: 'Enable UI for the export server',\n initial: defaultConfig.ui.enable.value\n },\n {\n type: 'text',\n name: 'route',\n message: 'A route to attach the UI to',\n initial: defaultConfig.ui.route.value\n }\n ],\n other: [\n {\n type: 'toggle',\n name: 'noLogo',\n message:\n 'Skip printing the logo on a startup. Will be replaced by a simple text',\n initial: defaultConfig.other.noLogo.value\n }\n ]\n};\n\n// Absolute props that, in case of merging recursively, need to be force merged\nexport const absoluteProps = [\n 'options',\n 'globalOptions',\n 'themeOptions',\n 'resources',\n 'payload'\n];\n\n// Argument nesting level of all export server options\nexport const nestedArgs = {};\n\n/**\n * Creates nested arguments chain for all options\n *\n * @param {object} obj - The object based on which the initial configuration be\n * made.\n * @param {string } propChain - Required for creating a string chain of\n * properties for nested arguments.\n */\nconst createNestedArgs = (obj, propChain = '') => {\n Object.keys(obj).forEach((k) => {\n if (!['puppeteer', 'highcharts'].includes(k)) {\n const entry = obj[k];\n if (typeof entry.value === 'undefined') {\n // Go deeper in the nested arguments\n createNestedArgs(entry, `${propChain}.${k}`);\n } else {\n // Create the chain of nested arguments\n nestedArgs[entry.cliName || k] = `${propChain}.${k}`.substring(1);\n }\n }\n });\n};\n\ncreateNestedArgs(defaultConfig);\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { appendFile, existsSync, mkdirSync } from 'fs';\n\nimport { defaultConfig } from './schemas/config.js';\n\n// The default logging config\nlet logging = {\n // Flags for logging status\n toConsole: true,\n toFile: false,\n pathCreated: false,\n // Log levels\n levelsDesc: [\n {\n title: 'error',\n color: 'red'\n },\n {\n title: 'warning',\n color: 'yellow'\n },\n {\n title: 'notice',\n color: 'blue'\n },\n {\n title: 'verbose',\n color: 'gray'\n }\n ],\n // Log listeners\n listeners: []\n};\n\n// Gather init logging options\nfor (const [key, option] of Object.entries(defaultConfig.logging)) {\n logging[key] = option.value;\n}\n\n/**\n * Logs a message. Accepts a variable amount of arguments. Arguments after\n * `level` will be passed directly to console.log, and/or will be joined\n * and appended to the log file.\n *\n * @param {any} args - An array of arguments where the first is the log level\n * and the rest are strings to build a message with.\n */\nexport const log = (...args) => {\n const [newLevel, ...texts] = args;\n\n // Current logging options\n const { level, levelsDesc } = logging;\n\n // Check if log level is within a correct range\n if (newLevel === 0 || newLevel > level || level > levelsDesc.length) {\n return;\n }\n\n // Get rid of the GMT text information\n const newDate = new Date().toString().split('(')[0].trim();\n\n // Create a message's prefix\n const prefix = `${newDate} [${levelsDesc[newLevel - 1].title}] -`;\n\n // Call available log listeners\n logging.listeners.forEach((fn) => {\n fn(prefix, texts.join(' '));\n });\n\n // Log to file\n if (logging.toFile) {\n if (!logging.pathCreated) {\n // Create if does not exist\n !existsSync(logging.dest) && mkdirSync(logging.dest);\n\n // We now assume the path is available, e.g. it's the responsibility\n // of the user to create the path with the correct access rights.\n logging.pathCreated = true;\n }\n\n // Add the content to a file\n appendFile(\n `${logging.dest}${logging.file}`,\n [prefix].concat(texts).join(' ') + '\\n',\n (error) => {\n if (error) {\n console.log(`[logger] Unable to write to log file: ${error}`);\n logging.toFile = false;\n }\n }\n );\n }\n\n // Log to console\n if (logging.toConsole) {\n console.log.apply(\n undefined,\n [prefix.toString()[logging.levelsDesc[newLevel - 1].color]].concat(texts)\n );\n }\n};\n\n/**\n * Sets the file logging configuration.\n *\n * @param {string} logDest - A path to log to.\n * @param {string} logFile - The name of the log file.\n */\nexport const enableFileLogging = (logDest, logFile) => {\n // Update logging options\n logging = {\n ...logging,\n dest: logDest || logging.dest,\n file: logFile || logging.file,\n toFile: true\n };\n\n if (logging.dest.length === 0) {\n return log(1, '[logger] File logging init: no path supplied.');\n }\n\n if (!logging.dest.endsWith('/')) {\n logging.dest += '/';\n }\n};\n\n/**\n * Adds a log listener.\n *\n * @param {function} fn - The function to call when getting a log event.\n */\nexport const listen = (fn) => {\n logging.listeners.push(fn);\n};\n\n/**\n * Sets the current log level. Log levels are:\n * - 0 = no logging\n * - 1 = error\n * - 2 = warning\n * - 3 = notice\n * - 4 = verbose\n *\n * @param {number} newLevel - The new log level (0 - 4).\n */\nexport const setLogLevel = (newLevel) => {\n if (newLevel >= 0 && newLevel <= logging.levelsDesc.length) {\n logging.level = newLevel;\n }\n};\n\n/**\n * Enables or disables logging to the stdout.\n *\n * @param {boolean} enabled - Whether log to console or not.\n */\nexport const toggleSTDOut = (enabled) => {\n logging.toConsole = enabled;\n};\n\nexport default {\n log,\n enableFileLogging,\n listen,\n setLogLevel,\n toggleSTDOut\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { readFileSync } from 'fs';\nimport { fileURLToPath } from 'url';\n\nimport { defaultConfig } from '../lib/schemas/config.js';\nimport { log } from './logger.js';\n\nconst MAX_BACKOFF_ATTEMPTS = 6;\n\nexport const __dirname = fileURLToPath(new URL('../.', import.meta.url));\n\n/**\n * Clears text from whitespaces with a regex rule.\n *\n * @param {string} rule - The rule for clearing a string, default to /\\s\\s+/g.\n * @return {string} - Cleared text.\n */\nexport const clearText = (text, rule = /\\s\\s+/g, replacer = ' ') =>\n text.replaceAll(rule, replacer).trim();\n\n/**\n * Delays calling the function by time calculated based on the backoff\n * algorithm.\n *\n * @param {function} fn - A function to try to call with the backoff algorithm\n * on.\n * @param {number} attempt - The number of an attempt, where the first one is 0.\n */\nexport const expBackoff = async (fn, attempt = 0, ...args) => {\n try {\n // Try to call the function\n return await fn(...args);\n } catch (error) {\n // Calculate delay in ms\n const delayInMs = 2 ** attempt * 1000;\n\n // If the attempt exceeds the maximum attempts of reapeat, throw an error\n if (++attempt >= MAX_BACKOFF_ATTEMPTS) {\n throw error;\n }\n\n // Wait given amount of time\n await new Promise((response) => setTimeout(response, delayInMs));\n log(\n 3,\n `[pool] Waited ${delayInMs}ms until next call for the resource id: ${args[0]}.`\n );\n\n // Try again\n return expBackoff(fn, attempt, ...args);\n }\n};\n\n/**\n * Fixes to supported type format if MIME.\n *\n * @param {string} type - Type to be corrected.\n * @param {string} outfile - Name of the outfile.\n */\nexport const fixType = (type, outfile) => {\n // MIME types\n const mimeTypes = {\n 'image/png': 'png',\n 'image/jpeg': 'jpeg',\n 'application/pdf': 'pdf',\n 'image/svg+xml': 'svg'\n };\n\n // Formats\n const formats = ['png', 'jpeg', 'pdf', 'svg'];\n\n // Check if type and outfile's extensions are the same\n if (outfile) {\n const outType = outfile.split('.').pop();\n\n // Check if extension has a correct type\n if (formats.includes(outType) && type !== outType) {\n type = outType;\n }\n }\n\n // Return a correct type\n return mimeTypes[type] || formats.find((t) => t === type) || 'png';\n};\n\n/**\n * Handles the provided resources.\n *\n * @param {string} resources - The stringified resources.\n * @param {string} allowFileResources - Decide if resources from file are\n * allowed.\n */\nexport const handleResources = (resources = false, allowFileResources) => {\n const allowedProps = ['js', 'css', 'files'];\n\n let handledResources = resources;\n let correctResources = false;\n\n // Try to load resources from a file\n if (allowFileResources && resources.endsWith('.json')) {\n try {\n if (!resources) {\n handledResources = isCorrectJSON(\n readFileSync('resources.json', 'utf8')\n );\n } else if (resources && resources.endsWith('.json')) {\n handledResources = isCorrectJSON(readFileSync(resources, 'utf8'));\n } else {\n handledResources = isCorrectJSON(resources);\n if (handledResources === true) {\n handledResources = isCorrectJSON(\n readFileSync('resources.json', 'utf8')\n );\n }\n }\n } catch (notice) {\n return log(3, `[cli] No resources found.`);\n }\n } else {\n // Try to get JSON\n handledResources = isCorrectJSON(resources);\n\n // Get rid of the files section\n if (!allowFileResources) {\n delete handledResources.files;\n }\n }\n\n // Filter from unnecessary properties\n for (const propName in handledResources) {\n if (!allowedProps.includes(propName)) {\n delete handledResources[propName];\n } else if (!correctResources) {\n correctResources = true;\n }\n }\n\n // Check if at least one of allowed properties is present\n if (!correctResources) {\n return log(3, `[cli] No resources found.`);\n }\n\n // Handle files section\n if (handledResources.files) {\n handledResources.files = handledResources.files.map((item) => item.trim());\n if (!handledResources.files || handledResources.files.length <= 0) {\n delete handledResources.files;\n }\n }\n\n // Return resources\n return handledResources;\n};\n\n/**\n * Checks if provided data is or can be a correct JSON.\n *\n * @param {any} data - Data to be checked.\n * @param {boolean} toString - If true, return stringified representation.\n */\nexport function isCorrectJSON(data, toString) {\n try {\n // Get the string representation if not already before parsing\n const parsedData = JSON.parse(\n typeof data !== 'string' ? JSON.stringify(data) : data\n );\n\n // Return a stringified representation of a JSON if required\n if (typeof parsedData !== 'string' && toString) {\n return JSON.stringify(parsedData);\n }\n\n // Return a JSON\n return parsedData;\n } catch (error) {\n return false;\n }\n}\n\n/**\n * Checks if item is an object.\n *\n * @param {any} item - Item to be checked.\n */\nexport const isObject = (item) =>\n typeof item === 'object' && !Array.isArray(item) && item !== null;\n\n/**\n * Checks if string contains private range urls.\n *\n * @export utils\n * @param item {string} item to be checked\n */\nexport const isPrivateRangeUrlFound = (item) => {\n return [\n 'localhost',\n '(10).(.*).(.*).(.*)',\n '(127).(.*).(.*).(.*)',\n '(172).(1[6-9]|2[0-9]|3[0-1]).(.*).(.*)',\n '(192).(168).(.*).(.*)'\n ].some((ipRegEx) =>\n item.match(`xlink:href=\"(?:(http://|https://))?${ipRegEx}`)\n );\n};\n\n/**\n * Creates and returns a deep copy of the given object.\n *\n * @param {object} object - Object to copy.\n * @return {object} - Deep copy of the object.\n */\nexport const deepCopy = (obj) => {\n if (obj === null || typeof obj !== 'object') {\n return obj;\n }\n\n const copy = Array.isArray(obj) ? [] : {};\n\n for (const key in obj) {\n if (Object.prototype.hasOwnProperty.call(obj, key)) {\n copy[key] = deepCopy(obj[key]);\n }\n }\n\n return copy;\n};\n\n/**\n * Stringifies object with options. Possible to preserve functions with\n * allowFunctions flag.\n *\n * @param {object} options - Options to stringify.\n * @param {boolean} allowFunctions - Flag for keeping functions.\n */\nexport const optionsStringify = (options, allowFunctions) => {\n const replacerCallback = (name, value) => {\n if (typeof value === 'string') {\n value = value.trim();\n\n // If allowFunctions is set to true, preserve functions\n if (\n (value.startsWith('function(') || value.startsWith('function (')) &&\n value.endsWith('}')\n ) {\n value = allowFunctions\n ? `EXP_FUN${(value + '').replaceAll(/\\n|\\t|\\r/g, ' ')}EXP_FUN`\n : undefined;\n }\n }\n\n return typeof value === 'function'\n ? `EXP_FUN${(value + '').replaceAll(/\\n|\\t|\\r/g, ' ')}EXP_FUN`\n : value;\n };\n\n // Stringify options and if required, replace special functions marks\n return JSON.stringify(options, replacerCallback).replaceAll(\n /\"EXP_FUN|EXP_FUN\"/g,\n ''\n );\n};\n\n/**\n * Prints the export server logo.\n *\n * @param {boolean} noLogo - Whether to display logo or text.\n */\nexport const printLogo = (noLogo) => {\n // Get package version either from env or from package.json\n const packageVersion =\n process.env.npm_package_version ||\n JSON.parse(readFileSync(new URL('../package.json', import.meta.url)))\n .version;\n\n // Print text only\n if (noLogo) {\n console.log(`Starting highcharts export server v${packageVersion}...`);\n return;\n }\n\n // Print the logo\n console.log(\n readFileSync(__dirname + '/msg/startup.msg').toString().bold.yellow,\n `v${packageVersion}`\n );\n};\n\n/**\n * Prints the CLI usage. If required, it can list properties recursively\n */\nexport function printUsage() {\n const pad = 48;\n const readme = 'https://github.com/highcharts/node-export-server#readme';\n\n // Display readme information\n console.log(\n 'Usage of CLI arguments:'.bold,\n '\\n------',\n `\\nFor more detailed information visit readme at: ${readme.bold.yellow}.`\n );\n\n const cycleCategories = (categories) => {\n for (const [name, option] of Object.entries(categories)) {\n // If category has more levels, go further\n if (!Object.prototype.hasOwnProperty.call(option, 'value')) {\n cycleCategories(option);\n } else {\n let descName = ` --${option.cliName || name} ${\n ('<' + option.type + '>').green\n } `;\n if (descName.length < pad) {\n for (let i = descName.length; i < pad; i++) {\n descName += '.';\n }\n }\n\n // Display correctly aligned messages\n console.log(\n descName,\n option.description,\n `[Default: ${option.value.toString().bold}]`.blue\n );\n }\n }\n };\n\n // Cycle through options of each categories and display the usage info\n Object.keys(defaultConfig).forEach((category) => {\n // Only puppeteer and highcharts categories cannot be configured through CLI\n if (!['puppeteer', 'highcharts'].includes(category)) {\n console.log(`\\n${category.toUpperCase()}`.red);\n cycleCategories(defaultConfig[category]);\n }\n });\n console.log('\\n');\n}\n\n/**\n * Rounds number to passed precision.\n *\n * @param {number} value - Number to round.\n * @param {number} precision - A precision of rounding.\n */\nexport const roundNumber = (value, precision = 1) => {\n const multiplier = Math.pow(10, precision || 0);\n return Math.round(+value * multiplier) / multiplier;\n};\n\n/**\n * Casts the item to boolean.\n *\n * @param {any} item - Item to be cast.\n */\nexport const toBoolean = (item) =>\n ['false', 'undefined', 'null', 'NaN', '0', ''].includes(item)\n ? false\n : !!item;\n\n/**\n * If necessary, places a custom code inside a function.\n *\n * @param {any} customCode - The customCode.\n */\nexport const wrapAround = (customCode, allowFileResources) => {\n if (customCode && typeof customCode === 'string') {\n customCode = customCode.trim();\n\n if (customCode.endsWith('.js')) {\n return allowFileResources\n ? wrapAround(readFileSync(customCode, 'utf8'))\n : false;\n } else if (\n customCode.startsWith('function()') ||\n customCode.startsWith('function ()') ||\n customCode.startsWith('()=>') ||\n customCode.startsWith('() =>')\n ) {\n return `(${customCode})()`;\n }\n return customCode.replace(/;$/, '');\n }\n};\n\n/**\n * Utility to measure time.\n */\nexport const measureTime = () => {\n const start = process.hrtime.bigint();\n return () => Number(process.hrtime.bigint() - start) / 1000000;\n};\n\nexport default {\n __dirname,\n clearText,\n expBackoff,\n fixType,\n handleResources,\n isCorrectJSON,\n isObject,\n isPrivateRangeUrlFound,\n optionsStringify,\n printLogo,\n printUsage,\n roundNumber,\n toBoolean,\n wrapAround,\n measureTime\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport rateLimit from 'express-rate-limit';\n\nimport { clearText } from '../utils.js';\nimport { log } from '../logger.js';\n\n/**\n * Enables rate limiting for a given app.\n *\n * @param {object} app - The express app.\n * @param {object} limitConfig - The options for the rate limiting.\n */\nexport default (app, limitConfig) => {\n const msg =\n 'Too many requests, you have been rate limited. Please try again later.';\n\n // Options for the rate limiter\n const rateOptions = {\n max: limitConfig.maxRequests || 30,\n window: limitConfig.window || 1,\n delay: limitConfig.delay || 0,\n trustProxy: limitConfig.trustProxy || false,\n skipKey: limitConfig.skipKey || false,\n skipToken: limitConfig.skipToken || false\n };\n\n // Set if behind a proxy\n if (rateOptions.trustProxy) {\n app.enable('trust proxy');\n }\n\n // Create a limiter\n const limiter = rateLimit({\n windowMs: rateOptions.window * 60 * 1000,\n // Limit each IP to 100 requests per windowMs\n max: rateOptions.max,\n // Disable delaying, full speed until the max limit is reached\n delayMs: rateOptions.delay,\n handler: (request, response) => {\n response.format({\n json: () => {\n response.status(429).send({ message: msg });\n },\n default: () => {\n response.status(429).send(msg);\n }\n });\n },\n skip: (request) => {\n // Allow bypassing the limiter if a valid key/token has been sent\n if (\n rateOptions.skipKey !== false &&\n rateOptions.skipToken !== false &&\n request.query.key === rateOptions.skipKey &&\n request.query.access_token === rateOptions.skipToken\n ) {\n log(4, '[rate-limiting] Skipping rate limiter.');\n return true;\n }\n return false;\n }\n });\n\n // Use a limiter as a middleware\n app.use(limiter);\n\n log(\n 3,\n clearText(\n `[rate-limiting] Enabled rate limiting: ${rateOptions.max} requests\n per ${rateOptions.window} minute per IP, trusting proxy:\n ${rateOptions.trustProxy}.`\n )\n );\n};\n","/**\n * This module exports two functions: fetch (for GET requests) and post (for POST requests).\n */\n\nimport http from 'http';\nimport https from 'https';\n\n/**\n * Determines the protocol of the given URL (either `http` or `https`).\n *\n * @function\n * @param {string} url - The URL whose protocol needs to be determined.\n * @returns {Object} Returns the `https` module if the URL starts with 'https',\n * otherwise returns the `http` module.\n * @private\n *\n * @example\n *\n * const protocol = getProtocol('https://example.com');\n * console.log(protocol); // Outputs the 'https' module\n */\nconst getProtocol = (url) => {\n return url.startsWith('https') ? https : http;\n};\n\n/**\n * Sends a GET request to the specified URL with optional request options.\n *\n * @function\n * @async\n * @param {string} url - The URL to fetch.\n * @param {Object} [requestOptions={}] - Optional request options and headers.\n * @returns {Promise} Returns a promise that resolves with the response object.\n * The response object contains a `.text` property with the raw response data.\n * @throws {Error} Throws an error if the request fails or if no data is fetched from the URL.\n *\n * @example\n *\n * async function getData() {\n * try {\n * const response = await fetch('https://api.example.com/data');\n * console.log(response.text);\n * } catch (error) {\n * console.error('Error fetching data:', error);\n * }\n * }\n *\n * getData();\n */\nasync function fetch(url, requestOptions = {}) {\n return new Promise((resolve, reject) => {\n const protocol = getProtocol(url);\n\n protocol\n .get(url, requestOptions, (res) => {\n let data = '';\n\n // A chunk of data has been received.\n res.on('data', (chunk) => {\n data += chunk;\n });\n\n // The whole response has been received.\n res.on('end', () => {\n if (!data) {\n reject('Nothing was fetched from the URL.');\n }\n\n res.text = data;\n resolve(res);\n });\n })\n .on('error', (error) => {\n reject(error);\n });\n });\n}\n\n/**\n * Sends a POST request to the specified URL with the given body and request options.\n *\n * @function\n * @async\n * @param {string} url - The URL to which the request should be sent.\n * @param {Object} [body={}] - The data to be sent as the request body, in JSON format.\n * @param {Object} [requestOptions={}] - Optional request options and headers.\n * @returns {Promise} - Returns a promise that resolves with the parsed JSON response.\n * @throws {Error} Throws an error if the request fails or if the response cannot be parsed.\n *\n * @example\n *\n * async function sendData() {\n * const dataToSend = {\n * key1: 'value1',\n * key2: 'value2',\n * };\n * try {\n * const response = await post('https://api.example.com/data', dataToSend);\n * console.log(response);\n * } catch (error) {\n * console.error('Error sending data:', error);\n * }\n * }\n *\n * sendData();\n */\nasync function post(url, body = {}, requestOptions = {}) {\n return new Promise((resolve, reject) => {\n const protocol = getProtocol(url);\n const data = JSON.stringify(body);\n\n // Set default headers and merge with requestOptions\n const options = Object.assign(\n {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'Content-Length': data.length\n }\n },\n requestOptions\n );\n\n const req = protocol\n .request(url, options, (res) => {\n let responseData = '';\n\n // A chunk of data has been received.\n res.on('data', (chunk) => {\n responseData += chunk;\n });\n\n // The whole response has been received.\n res.on('end', () => {\n try {\n res.text = responseData;\n resolve(res);\n } catch (error) {\n reject(error);\n }\n });\n })\n .on('error', (error) => {\n reject(error);\n });\n\n // Write the request body and end the request.\n req.write(data);\n req.end();\n });\n}\n\nexport default fetch;\nexport { fetch, post };\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\n// The cache manager manages the Highcharts library and its dependencies.\n// The cache itself is stored in .cache, and is checked by the config system\n// before starting the service\n\nimport { existsSync, mkdirSync, readFileSync, writeFileSync } from 'fs';\nimport { join } from 'path';\n\nimport dotenv from 'dotenv';\nimport HttpsProxyAgent from 'https-proxy-agent';\nimport { fetch } from './fetch.js';\n\nimport { log } from './logger.js';\nimport { __dirname } from '../lib/utils.js';\n\ndotenv.config();\n\nconst cachePath = join(__dirname, '.cache');\n\nconst cache = {\n cdnURL: 'https://code.highcharts.com/',\n activeManifest: {},\n sources: '',\n hcVersion: ''\n};\n\n// TODO: The config should be accesssible globally so we don't have to do this sort of thing..\nlet appliedConfig = false;\n\n/**\n * Extracts the Highcharts version from the cache\n */\nconst extractVersion = () =>\n (cache.hcVersion = cache.sources\n .substr(0, cache.sources.indexOf('*/'))\n .replace('/*', '')\n .replace('*/', '')\n .replace(/\\n/g, '')\n .trim());\n\n/**\n * Saves the Highcharts part of a config to a manifest file in the cache\n *\n * @param {object} config - Highcharts related configuration object.\n * @param {object} fetchedModules - An object that contains mapped names of\n * fetched Highcharts modules to use.\n */\nconst saveConfigToManifest = async (config, fetchedModules) => {\n const newManifest = {\n version: config.version,\n modules: fetchedModules || {}\n };\n\n // Update cache object with the current modules\n cache.activeManifest = newManifest;\n\n log(4, '[cache] writing new manifest');\n\n try {\n writeFileSync(\n join(cachePath, 'manifest.json'),\n JSON.stringify(newManifest),\n 'utf8'\n );\n } catch (error) {\n log(1, `[cache] Error writing cache manifest: ${error}.`);\n }\n};\n\n/**\n * Fetches a single script.\n *\n * @param {string} script - A path to script to get.\n * @param {object} proxyAgent - The proxy agent to use for a request.\n */\nconst fetchScript = async (script, proxyAgent) => {\n try {\n // Get rid of the .js from the custom strings\n if (script.endsWith('.js')) {\n script = script.substring(0, script.length - 3);\n }\n\n log(4, `[cache] Fetching script - ${script}.js`);\n\n // If exists, add proxy agent to request options\n const requestOptions = proxyAgent\n ? {\n agent: proxyAgent,\n timeout: +process.env['PROXY_SERVER_TIMEOUT'] || 5000\n }\n : {};\n\n // Fetch the script\n const response = await fetch(`${script}.js`, requestOptions);\n\n // If OK, return its text representation\n if (response.statusCode === 200) {\n return response.text;\n }\n\n throw `${response.statusCode}`;\n } catch (error) {\n log(1, `[cache] Error fetching script ${script}.js: ${error}.`);\n throw error;\n }\n};\n\n/**\n * Updates the Highcharts cache.\n *\n * @param {object} config - Highcharts related configuration object.\n * @param {string} sourcePath - A path to the file where save updated sources.\n * @return {object} An object that contains mapped names of fetched Highcharts\n * modules to use.\n */\nconst updateCache = async (config, sourcePath) => {\n const { coreScripts, modules, indicators, scripts: customScripts } = config;\n const hcVersion =\n config.version === 'latest' || !config.version ? '' : `${config.version}/`;\n\n log(3, '[cache] Updating cache to Highcharts ', hcVersion);\n\n // Gather all scripts to fetch\n const allScripts = [\n ...coreScripts.map((c) => `${hcVersion}${c}`),\n ...modules.map((m) =>\n m === 'map' ? `maps/${hcVersion}modules/${m}` : `${hcVersion}modules/${m}`\n ),\n ...indicators.map((i) => `stock/${hcVersion}indicators/${i}`)\n ];\n\n // Configure proxy if exists\n let proxyAgent;\n const proxyHost = process.env['PROXY_SERVER_HOST'];\n const proxyPort = process.env['PROXY_SERVER_PORT'];\n\n if (proxyHost && proxyPort) {\n proxyAgent = new HttpsProxyAgent({\n host: proxyHost,\n port: +proxyPort\n });\n }\n\n const fetchedModules = {};\n try {\n cache.sources = // TODO: convert to for loop\n (\n await Promise.all([\n ...allScripts.map(async (script) => {\n const text = await fetchScript(\n `${config.cdnURL || cache.cdnURL}${script}`,\n proxyAgent\n );\n\n // If fetched correctly, set it\n if (typeof text === 'string') {\n fetchedModules[\n script.replace(\n /(.*)\\/|(.*)modules\\/|stock\\/(.*)indicators\\/|maps\\/(.*)modules\\//gi,\n ''\n )\n ] = 1;\n }\n\n return text;\n }),\n ...customScripts.map((script) => fetchScript(script, proxyAgent))\n ])\n ).join(';\\n');\n extractVersion();\n\n // Save the fetched modules into caches' source JSON\n writeFileSync(sourcePath, cache.sources);\n return fetchedModules;\n } catch (error) {\n log(1, '[cache] Unable to update local Highcharts cache.');\n }\n};\n\nexport const updateVersion = async (newVersion) =>\n appliedConfig\n ? await checkCache(\n Object.assign(appliedConfig, {\n version: newVersion\n })\n )\n : false;\n\n/**\n * Fetches any missing Highcharts and dependencies\n *\n * @param {object} config - Highcharts related configuration object.\n */\nexport const checkCache = async (config) => {\n let fetchedModules;\n // Prepare paths to manifest and sources from the .cache folder\n const manifestPath = join(cachePath, 'manifest.json');\n const sourcePath = join(cachePath, 'sources.js');\n\n // TODO: deal with trying to switch to the running version\n // const activeVersion = appliedConfig ? appliedConfig.version : false;\n\n appliedConfig = config;\n\n // Create the .cache destination if it doesn't exist already\n !existsSync(cachePath) && mkdirSync(cachePath);\n\n // Fetch all the scripts either if manifest.json does not exist\n // or if the forceFetch option is enabled\n if (!existsSync(manifestPath) || config.forceFetch) {\n log(3, '[cache] Fetching and caching Highcharts dependencies.');\n fetchedModules = await updateCache(config, sourcePath);\n } else {\n let requestUpdate = false;\n\n // Read the manifest JSON\n const manifest = JSON.parse(readFileSync(manifestPath));\n\n // Check if the modules is an array, if so, we rewrite it to a map to make\n // it easier to resolve modules.\n if (manifest.modules && Array.isArray(manifest.modules)) {\n const moduleMap = {};\n manifest.modules.forEach((m) => (moduleMap[m] = 1));\n manifest.modules = moduleMap;\n }\n\n const { modules, coreScripts, indicators } = config;\n const numberOfModules =\n modules.length + coreScripts.length + indicators.length;\n\n // Compare the loaded config with the contents in .cache.\n // If there are changes, fetch requested modules and products,\n // and bake them into a giant blob. Save the blob.\n if (manifest.version !== config.version) {\n log(3, '[cache] Highcharts version mismatch in cache, need to re-fetch.');\n requestUpdate = true;\n } else if (Object.keys(manifest.modules || {}).length !== numberOfModules) {\n log(\n 3,\n '[cache] Cache and requested modules does not match, need to re-fetch.'\n );\n requestUpdate = true;\n } else {\n // Check each module, if anything is missing refetch everything\n requestUpdate = (config.modules || []).some((moduleName) => {\n if (!manifest.modules[moduleName]) {\n log(\n 3,\n `[cache] The ${moduleName} missing in cache, need to re-fetch.`\n );\n return true;\n }\n });\n }\n\n if (requestUpdate) {\n fetchedModules = await updateCache(config, sourcePath);\n } else {\n log(3, '[cache] Dependency cache is up to date, proceeding.');\n\n // Load the sources\n cache.sources = readFileSync(sourcePath, 'utf8');\n\n // Get current modules map\n fetchedModules = manifest.modules;\n extractVersion();\n }\n }\n\n // Finally, save the new manifest, which is basically our current config\n // in a slightly different format\n await saveConfigToManifest(config, fetchedModules);\n};\n\nexport default {\n checkCache,\n updateVersion,\n getCache: () => cache,\n highcharts: () => cache.sources,\n version: () => cache.hcVersion\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport puppeteer from 'puppeteer';\nimport fs from 'fs';\nimport * as url from 'url';\nimport { log } from './logger.js';\nimport path from 'node:path';\n\n// Workaround for https://bugs.chromium.org/p/chromium/issues/detail?id=1463328\n// Not ideal - leaves trash in the FS\nimport { randomBytes } from 'node:crypto';\nconst RANDOM_PID = randomBytes(64).toString('base64url');\nconst PUPPETEER_DIR = path.join('tmp', `puppeteer-${RANDOM_PID}`);\nconst DATA_DIR = path.join(PUPPETEER_DIR, 'profile');\n\n// The minimal args to speed up the browser\nconst minimalArgs = [\n `--user-data-dir=${DATA_DIR}`,\n '--autoplay-policy=user-gesture-required',\n '--disable-background-networking',\n '--disable-background-timer-throttling',\n '--disable-backgrounding-occluded-windows',\n '--disable-breakpad',\n '--disable-client-side-phishing-detection',\n '--disable-component-update',\n '--disable-default-apps',\n '--disable-dev-shm-usage',\n '--disable-domain-reliability',\n '--disable-extensions',\n '--disable-features=AudioServiceOutOfProcess',\n '--disable-hang-monitor',\n '--disable-ipc-flooding-protection',\n '--disable-notifications',\n '--disable-offer-store-unmasked-wallet-cards',\n '--disable-popup-blocking',\n '--disable-print-preview',\n '--disable-prompt-on-repost',\n '--disable-renderer-backgrounding',\n '--disable-session-crashed-bubble',\n '--disable-setuid-sandbox',\n '--disable-speech-api',\n '--disable-sync',\n '--hide-crash-restore-bubble',\n '--hide-scrollbars',\n '--ignore-gpu-blacklist',\n '--metrics-recording-only',\n '--mute-audio',\n '--no-default-browser-check',\n '--no-first-run',\n '--no-pings',\n '--no-sandbox',\n '--no-zygote',\n '--password-store=basic',\n '--use-mock-keychain'\n];\n\nconst __dirname = url.fileURLToPath(new URL('.', import.meta.url));\n\nconst template = fs.readFileSync(\n __dirname + '/../templates/template.html',\n 'utf8'\n);\n\nlet browser;\n\nexport const newPage = async () => {\n if (!browser) return false;\n\n const p = await browser.newPage();\n\n await p.setContent(template);\n await p.addScriptTag({ path: __dirname + '/../.cache/sources.js' });\n // eslint-disable-next-line no-undef\n await p.evaluate(() => window.setupHighcharts());\n\n p.on('pageerror', async (err) => {\n // TODO: Consider adding a switch here that turns on log(0) logging\n // on page errors.\n log(1, '[page error]', err);\n await p.$eval(\n '#container',\n (element, errorMessage) => {\n // eslint-disable-next-line no-undef\n if (window._displayErrors) {\n element.innerHTML = errorMessage;\n }\n },\n `

Chart input data error

${err.toString()}`\n );\n });\n\n return p;\n};\n\nexport const create = async (puppeteerArgs) => {\n const allArgs = [...minimalArgs, ...(puppeteerArgs || [])];\n\n // Create a browser\n if (!browser) {\n let tryCount = 0;\n\n const open = async () => {\n try {\n log(\n 3,\n '[browser] attempting to get a browser instance (try',\n tryCount + ')'\n );\n\n browser = await puppeteer.launch({\n headless: 'new',\n args: allArgs,\n userDataDir: './tmp/'\n });\n } catch (e) {\n log(0, '[browser]', e);\n if (++tryCount < 25) {\n log(3, '[browser] failed:', e);\n await new Promise((response) => setTimeout(response, 4000));\n await open();\n } else {\n log(0, 'Max retries reached');\n }\n }\n };\n\n try {\n await open();\n } catch (e) {\n log(0, '[browser] Unable to open browser');\n return false;\n }\n\n if (!browser) {\n log(0, '[browser] Unable to open browser');\n return false;\n }\n }\n\n // Return a browser promise\n return browser;\n};\n\nexport const get = async () => {\n if (!browser) {\n throw 'No valid browser has been created';\n }\n\n return browser;\n};\n\nexport const close = async () => {\n // Close the browser when connnected\n if (browser.connected) {\n await browser.close();\n }\n};\n\nexport default {\n get,\n close,\n newPage\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\n// TODO: remove this temp benchmark stuff. I had this idea of doing a general benchmarking\n// system, but it adds so much bloat in the code that it shouldn't be there.\n\nimport benchmark from './benchmark.js';\nimport cache from './cache.js';\nimport { log } from './logger.js';\nimport svgTemplate from './../templates/svg_export/svg_export.js';\n\nimport { readFileSync } from 'fs';\nimport path from 'path';\nimport * as url from 'url';\n\nconst __basedir = url.fileURLToPath(new URL('.', import.meta.url));\n\n// const jsonTemplate = require('./../templates/json_export/json_export.js');\n\n/**\n * Gets the clip region for the chart DOM node.\n *\n * @param {object} page - A page of a browser instance.\n * @return {object} - A clipped region.\n */\nconst getClipRegion = (page) =>\n page.$eval('#chart-container', (element) => {\n const { x, y, width, height } = element.getBoundingClientRect();\n return {\n x,\n y,\n width,\n height: Math.trunc(height > 1 ? height : 500)\n };\n });\n\n/**\n * Rasterizes the page to an image (PNG or JPEG)\n *\n * @param {object} page - A page of a browser instance.\n * @param {string} type - The type of a result image.\n * @param {string} encoding - The type of encoding used.\n * @param {string} clip - The clip region.\n * @returns {string} - A string representation of a screenshot.\n */\nconst createImage = async (page, type, encoding, clip) =>\n await Promise.race([\n page.screenshot({\n type,\n encoding,\n clip,\n\n // #447 - always render on a transparent page\n // this will not affect users who do not explicitly set\n // chart.backgroundColor to a color with opacity lower than 1\n omitBackground: true\n }),\n new Promise((resolve, reject) =>\n setTimeout(() => reject(new Error('Rasterization timeout')), 1500)\n )\n ]);\n\n/**\n * Turns page into a PDF.\n *\n * @param {object} page - A page of a browser instance.\n * @param {number} height - The height of a chart.\n * @param {number} width - The width of a chart.\n * @param {string} encoding - The type of encoding used.\n * @return {object} - A buffer with PDF representation.\n */\nconst createPDF = async (page, height, width, encoding) =>\n await page.pdf({\n // This will remove an extra empty page in PDF exports\n height: height + 1,\n width,\n encoding\n });\n\n/**\n * Exports as a SVG.\n *\n * @param {object} page - A page of a browser instance.\n * @return {object} - The outerHTML element with the SVG representation.\n */\nconst createSVG = async (page) =>\n await page.$eval(\n '#container svg:first-of-type',\n (element) => element.outerHTML\n );\n\n/** Load config into a page and render a chart */\nconst setAsConfig = async (page, chart, options) =>\n await page.evaluate(\n // eslint-disable-next-line no-undef\n (chart, options) => window.triggerExport(chart, options),\n chart,\n options\n );\n\n/** Load SVG into a page */\n// const setAsSVG = async (page, svgStr) => true;\n\n/**\n * Does an export for a given browser.\n *\n * @param {object} browser - A browser instance.\n * @param {object} chart - Chart's options.\n * @param {object} options - All options object.\n * @return {object} - The data returned from one of the methods for exporting\n * a specific type of an image.\n */\nexport default async (page, chart, options) => {\n /**\n * Keeps track of all resources added on the page with addXXXTag. etc\n * It's VITAL that all added resources ends up here so we can clear things\n * out when doing a new export in the same page!\n */\n const injectedResources = [];\n\n /** Clear out all state set on the page with addScriptTag/addStyleTag. */\n const clearInjected = async (page) => {\n for (const res of injectedResources) {\n await res.dispose();\n }\n\n // Reset all CSS and script tags\n await page.evaluate(() => {\n // eslint-disable-next-line no-undef\n const [, ...scriptsToRemove] = document.getElementsByTagName('script');\n // eslint-disable-next-line no-undef\n const [, ...stylesToRemove] = document.getElementsByTagName('style');\n // eslint-disable-next-line no-undef\n const [...linksToRemove] = document.getElementsByTagName('link');\n\n // Remove tags\n for (const element of [\n ...scriptsToRemove,\n ...stylesToRemove,\n ...linksToRemove\n ]) {\n element.remove();\n }\n });\n };\n\n try {\n const exportBench = benchmark('Puppeteer');\n\n log(4, '[export] Determining export path.');\n\n const exportOptions = options.export;\n\n // Force a rAF\n // See https://github.com/puppeteer/puppeteer/issues/7507\n // eslint-disable-next-line no-undef\n await page.evaluate(() => requestAnimationFrame(() => {}));\n\n // Decide whether display error or debbuger wrapper around it\n const displayErrors =\n exportOptions?.options?.chart?.displayErrors &&\n cache.getCache().activeManifest.modules.debugger;\n\n // eslint-disable-next-line no-undef\n await page.evaluate((d) => (window._displayErrors = d), displayErrors);\n\n const svgBench = benchmark('SVG handling');\n\n let isSVG;\n\n if (\n chart.indexOf &&\n (chart.indexOf('= 0 || chart.indexOf('= 0)\n ) {\n // SVG INPUT HANDLING\n\n log(4, '[export] Treating as SVG.');\n\n // If input is also svg, just return it\n if (exportOptions.type === 'svg') {\n return chart;\n }\n\n isSVG = true;\n const setPageBench = benchmark('Setting content');\n await page.setContent(svgTemplate(chart));\n setPageBench();\n } else {\n // JSON Config handling\n\n log(4, '[export] Treating as config.');\n\n // Need to perform straight inject\n if (exportOptions.strInj) {\n // Injection based configuration export\n const setPageBench = benchmark('Setting page content (inject)');\n\n await setAsConfig(\n page,\n {\n chart: {\n height: exportOptions.height,\n width: exportOptions.width\n }\n },\n options\n );\n\n setPageBench();\n } else {\n // Basic configuration export\n\n chart.chart.height = exportOptions.height;\n chart.chart.width = exportOptions.width;\n\n const setContentBench = benchmark('Setting page content (config)');\n await setAsConfig(page, chart, options);\n setContentBench();\n }\n }\n\n svgBench();\n const resBench = benchmark('Applying resources');\n\n // Use resources\n const resources = options.customCode.resources;\n if (resources) {\n // Load custom JS code\n if (resources.js) {\n injectedResources.push(\n await page.addScriptTag({\n content: resources.js\n })\n );\n }\n\n // Load scripts from all custom files\n if (resources.files) {\n for (const file of resources.files) {\n try {\n const isLocal = !file.startsWith('http') ? true : false;\n\n // Add each custom script from resources' files\n injectedResources.push(\n await page.addScriptTag(\n isLocal\n ? {\n content: readFileSync(file, 'utf8')\n }\n : {\n url: file\n }\n )\n );\n } catch (notice) {\n log(4, '[export] JS file not found.');\n }\n }\n }\n\n const cssBench = benchmark('Loading css');\n\n // Load CSS\n if (resources.css) {\n let cssImports = resources.css.match(/@import\\s*([^;]*);/g);\n if (cssImports) {\n // Handle css section\n for (let cssImportPath of cssImports) {\n if (cssImportPath) {\n cssImportPath = cssImportPath\n .replace('url(', '')\n .replace('@import', '')\n .replace(/\"/g, '')\n .replace(/'/g, '')\n .replace(/;/, '')\n .replace(/\\)/g, '')\n .trim();\n\n // Add each custom css from resources\n if (cssImportPath.startsWith('http')) {\n injectedResources.push(\n await page.addStyleTag({\n url: cssImportPath\n })\n );\n } else if (options.customCode.allowFileResources) {\n injectedResources.push(\n await page.addStyleTag({\n path: path.join(__basedir, cssImportPath)\n })\n );\n }\n }\n }\n }\n\n // The rest of the CSS section will be content by now\n injectedResources.push(\n await page.addStyleTag({\n content: resources.css.replace(/@import\\s*([^;]*);/g, '') || ' '\n })\n );\n }\n\n cssBench();\n }\n\n resBench();\n\n // Get the real chart size\n const size = isSVG\n ? await page.$eval(\n '#chart-container svg:first-of-type',\n async (element, scale) => {\n return {\n chartHeight: element.height.baseVal.value * scale,\n chartWidth: element.width.baseVal.value * scale\n };\n },\n parseFloat(exportOptions.scale)\n )\n : await page.evaluate(async () => {\n // eslint-disable-next-line no-undef\n const { chartHeight, chartWidth } = window.Highcharts.charts[0];\n return {\n chartHeight,\n chartWidth\n };\n });\n\n const vpBench = benchmark('Setting viewport');\n\n // Set final height and width for viewport\n const viewportHeight = Math.ceil(size?.chartHeight || exportOptions.height);\n const viewportWidth = Math.ceil(size?.chartWidth || exportOptions.width);\n\n // Set the viewport for the first time\n // NOTE: the call to setViewport is expensive - can we get away with only\n // calling it once, e.g. moving this one into the isSVG condition below?\n await page.setViewport({\n height: viewportHeight,\n width: viewportWidth,\n deviceScaleFactor: isSVG ? 1 : parseFloat(exportOptions.scale)\n });\n\n // Prepare a zoom callback for the next evaluate call\n const zoomCallback = isSVG\n ? // In case of SVG the zoom must be set directly for body\n (scale) => {\n // Set the zoom as scale\n // eslint-disable-next-line no-undef\n document.body.style.zoom = scale;\n\n // Set the margin to 0px\n // eslint-disable-next-line no-undef\n document.body.style.margin = '0px';\n }\n : // No need for such scale manipulation in case of other types of exports\n () => {\n // Reset the zoom for other exports than to SVGs\n // eslint-disable-next-line no-undef\n document.body.style.zoom = 1;\n };\n\n // Set the zoom accordingly\n await page.evaluate(zoomCallback, parseFloat(exportOptions.scale));\n\n // Get the clip region for the page\n const { height, width, x, y } = await getClipRegion(page);\n\n if (!isSVG) {\n // Set the final viewport now that we have the real height\n await page.setViewport({\n width: Math.round(width),\n height: Math.round(height),\n deviceScaleFactor: parseFloat(exportOptions.scale)\n });\n }\n\n vpBench();\n\n let data;\n\n const expBenchmark = benchmark('Rasterizing chart');\n\n // RASTERIZATION\n if (exportOptions.type === 'svg') {\n // SVG\n data = await createSVG(page);\n } else if (exportOptions.type === 'png' || exportOptions.type === 'jpeg') {\n // PNG or JPEG\n data = await createImage(page, exportOptions.type, 'base64', {\n width: viewportWidth,\n height: viewportHeight,\n x,\n y\n });\n } else if (exportOptions.type === 'pdf') {\n // PDF\n data = await createPDF(page, viewportHeight, viewportWidth, 'base64');\n } else {\n throw `Unsupported output format ${exportOptions.type}`;\n }\n\n // Destroy old charts after the export is done\n await page.evaluate(() => {\n // eslint-disable-next-line no-undef\n const oldCharts = Highcharts.charts;\n\n // Check in any already existing charts\n if (oldCharts.length) {\n // Destroy old charts\n for (const oldChart of oldCharts) {\n oldChart && oldChart.destroy();\n // eslint-disable-next-line no-undef\n Highcharts.charts.shift();\n }\n }\n });\n\n expBenchmark();\n exportBench();\n\n await clearInjected(page);\n\n return data;\n } catch (error) {\n await clearInjected(page);\n log(1, `[export] Error encountered during export: ${error}`);\n\n return error;\n }\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2022, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { log } from './logger.js';\nconst timers = {};\n\n// TODO: Read from config\nlet enabled = false;\n\nexport default (id) => {\n if (!enabled) {\n return () => {};\n }\n\n timers[id] = new Date();\n return () => {\n log(\n 3,\n `[benchmark] - ${id}: ${new Date().getTime() - timers[id].getTime()}ms`\n );\n };\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport cssTemplate from './css.js';\n\nexport default (chart) => `\n\n\n \n \n Highcarts Export\n \n \n \n
\n ${chart}\n
\n \n\n\n`;\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { v4 as uuid } from 'uuid';\nimport { Pool } from 'tarn';\nimport {\n close,\n newPage as browserNewPage,\n create as createBrowser\n} from './browser.js';\nimport { log } from './logger.js';\n\nimport puppeteerExport from './export.js';\n\nlet performedExports = 0;\nlet exportAttempts = 0;\nlet timeSpent = 0;\nlet droppedExports = 0;\nlet spentAverage = 0;\nlet poolConfig = {};\n\n// The pool instance\nlet pool = false;\n\n// Custom puppeteer arguments\nlet puppeteerArgs;\n\nconst factory = {\n /**\n * Creates a new worker.\n *\n * @return {object} - An object with the id of a resource, the work count and\n * a reference to the browser page.\n */\n create: async () => {\n const id = uuid();\n let page = false;\n\n const s = new Date().getTime();\n\n try {\n page = await browserNewPage();\n\n if (!page || page.isClosed()) {\n throw 'invalid page';\n }\n\n log(\n 3,\n `[pool] Successfully created a worker ${id} - took ${\n new Date().getTime() - s\n } ms.`\n );\n } catch (error) {\n log(\n 1,\n `[pool] Error creating a new page in pool entry creation! ${error}`\n );\n\n throw 'Error creating page';\n }\n\n return {\n id,\n page,\n // Try to distribute the initial work count\n workCount: Math.round(Math.random() * (poolConfig.workLimit / 2))\n };\n },\n\n /**\n * Validates a worker.\n *\n * @param {object} workerHandle - A browser's instance.\n *\n * @return {boolean} - Bool that indicates if a resource is valid or not.\n */\n validate: (workerHandle) => {\n if (\n poolConfig.workLimit &&\n ++workerHandle.workCount > poolConfig.workLimit\n ) {\n log(\n 3,\n `[pool] Worker failed validation:`,\n `exceeded work limit (limit is ${poolConfig.workLimit})`\n );\n return false;\n }\n return true;\n },\n\n /**\n * Destroys a worker.\n *\n * @param {object} workerHandle - A browser's instance.\n */\n destroy: (workerHandle) => {\n log(3, `[pool] Destroying pool entry ${workerHandle.id}.`);\n\n if (workerHandle.page) {\n // We don't really need to wait around for this.\n workerHandle.page.close();\n }\n },\n\n // Logger function\n log: (message, logLevel) => console.log(`${logLevel}: ${message}`)\n};\n\n/**\n * Inits the pool of resources.\n *\n * @param {object} config - Pool configuration along with custom puppeteer\n * arguments for the puppeteer.launch function.\n */\nexport const init = async (config) => {\n // The newest puppeteer arguments for the browser creation\n puppeteerArgs = config.puppeteerArgs;\n\n // Wait until we've sucessfully created a browser instance.\n try {\n await createBrowser(puppeteerArgs);\n } catch (e) {\n log(0, '[pool|browser]', e);\n }\n\n // For the module scope usage\n poolConfig = config && config.pool ? { ...config.pool } : {};\n\n log(\n 3,\n '[pool] Initializing pool:',\n `min ${poolConfig.initialWorkers}, max ${poolConfig.maxWorkers}.`\n );\n\n if (pool) {\n return log(\n 4,\n '[pool] Already initialized, please kill it before creating a new one.'\n );\n }\n\n // Attach process' exit listeners\n if (poolConfig.listenToProcessExits) {\n attachProcessExitListeners();\n }\n\n try {\n // Create a pool along with a minimal number of resources\n pool = new Pool({\n // Get the create/validate/destroy/log functions\n ...factory,\n min: poolConfig.initialWorkers,\n max: poolConfig.maxWorkers,\n createRetryIntervalMillis: 200,\n createTimeoutMillis: poolConfig.acquireTimeout,\n acquireTimeoutMillis: poolConfig.acquireTimeout,\n destroyTimeoutMillis: poolConfig.acquireTimeout,\n idleTimeoutMillis: poolConfig.timeoutThreshold,\n reapIntervalMillis: 1000, // poolConfig.reaper ? 120000 : 0, for now\n propagateCreateError: false\n });\n\n // Set events\n pool.on('createFail', (eventId, err) => {\n log(\n 1,\n `[pool] Error when creating worker of an event id ${eventId}:`,\n err\n );\n });\n\n pool.on('acquireFail', (eventId, err) => {\n log(\n 1,\n `[pool] Error when acquiring worker of an event id ${eventId}:`,\n err\n );\n });\n\n pool.on('destroyFail', (eventId, resource, err) => {\n log(\n 1,\n `[pool] Error when destroying worker of an id ${resource.id}, event id ${eventId}:`,\n err\n );\n });\n\n pool.on('release', (resource) => {\n log(4, `[pool] Releasing a worker of an id ${resource.id}`);\n });\n\n pool.on('destroySuccess', (eventId, resource) => {\n log(4, `[pool] Destroyed a worker of an id ${resource.id}`);\n });\n\n const initialResources = [];\n // Create an initial number of resources\n for (let i = 0; i < poolConfig.initialWorkers; i++) {\n initialResources.push(await pool.acquire().promise);\n }\n\n // Release the initial number of resources back to the pool\n initialResources.forEach((resource) => {\n pool.release(resource);\n });\n\n log(\n 3,\n `[pool] The pool is ready with ${poolConfig.initialWorkers} initial resources waiting.`\n );\n } catch (error) {\n log(1, `[pool] Couldn't create the worker pool ${error}`);\n throw error;\n }\n};\n\n/**\n * Attaches process' exit listeners.\n */\nexport function attachProcessExitListeners() {\n log(4, '[pool] Attaching exit listeners to the process.');\n\n // Kill all pool resources on exit\n process.on('exit', async () => {\n await killPool();\n });\n\n // Handler for the SIGINT\n process.on('SIGINT', (name, code) => {\n log(4, `The ${name} event with code: ${code}.`);\n process.exit(1);\n });\n\n // Handler for the SIGTERM\n process.on('SIGTERM', (name, code) => {\n log(4, `The ${name} event with code: ${code}.`);\n process.exit(1);\n });\n\n // Handler for the uncaughtException\n process.on('uncaughtException', async (error, name) => {\n log(4, `The ${name} error, message: ${error.message}.`);\n });\n}\n\n/**\n * Kills the pool and flush the browser instance.\n */\nexport async function killPool() {\n log(3, '[pool] Killing all workers.');\n\n // Return true when the pool is already destroyed\n if (pool.destroyed) {\n // Close the browser instance if still connected\n await close();\n return true;\n }\n\n // If still alive, destroy the pool of pages before closing a browser\n await pool.destroy();\n\n // Close the browser instance\n await close();\n return true;\n}\n\n/**\n * Posts work to the pool.\n *\n * @param {object} chart - Chart's options.\n * @param {object} options - All options object.\n */\nexport const postWork = async (chart, options) => {\n let workerHandle;\n\n // Handle fail conditions\n const fail = (msg) => {\n ++droppedExports;\n\n if (workerHandle) {\n pool.release(workerHandle);\n }\n\n throw 'In pool.postWork: ' + msg;\n };\n\n log(4, '[pool] Work received, starting to process.');\n\n if (poolConfig.benchmarking) {\n getPoolInfo();\n }\n\n ++exportAttempts;\n\n if (!pool) {\n log(1, '[pool] Work received, but pool has not been started.');\n return fail('Pool is not inited but work was posted to it!');\n }\n\n // Acquire the worker along with the id of resource and work count\n try {\n log(4, '[pool] Acquiring worker');\n workerHandle = await pool.acquire().promise;\n } catch (error) {\n return fail(`[pool] Error when acquiring available entry: ${error}`);\n }\n\n log(4, '[pool] Acquired worker handle');\n\n if (!workerHandle.page) {\n return fail('Resolved worker page is invalid: pool setup is wonky');\n }\n\n try {\n // Save the start time\n let workStart = new Date().getTime();\n\n log(4, `[pool] Starting work on pool entry ${workerHandle.id}.`);\n\n // Perform an export on a puppeteer level\n const result = await puppeteerExport(workerHandle.page, chart, options);\n\n // Check if it's an error\n if (result instanceof Error) {\n // TODO: If the export failed because puppeteer timed out, we need to force kill the worker so we get a new page. That needs to be handled better than this hack.\n if (result.message === 'Rasterization timeout') {\n workerHandle.page.close();\n workerHandle.page = await browserNewPage();\n }\n\n return fail(result);\n }\n\n // Release the resource back to the pool\n pool.release(workerHandle);\n\n // Used for statistics in averageTime and processedWorkCount, which\n // in turn is used by the /health route.\n const workEnd = new Date().getTime();\n const exportTime = workEnd - workStart;\n timeSpent += exportTime;\n spentAverage = timeSpent / ++performedExports;\n\n log(4, `[pool] Work completed in ${exportTime} ms.`);\n\n // Otherwise return the result\n return {\n data: result,\n options\n };\n } catch (error) {\n fail(`Error trying to perform puppeteer export: ${error}.`);\n }\n};\n\n/**\n * Gets the pool.\n */\nexport function getPool() {\n return pool;\n}\n\nexport const getPoolInfoJSON = () => ({\n min: pool.min,\n max: pool.max,\n size: pool.size,\n available: pool.available,\n borrowed: pool.borrowed,\n pending: pool.pending,\n spareResourceCapacity: pool.spareResourceCapacity\n});\n\n/**\n * Gets the pool's information.\n */\nexport function getPoolInfo() {\n const {\n min,\n max,\n size,\n available,\n borrowed,\n pending,\n spareResourceCapacity\n } = pool;\n\n log(4, `[pool] The minimum number of resources allowed by pool: ${min}.`);\n log(4, `[pool] The maximum number of resources allowed by pool: ${max}.`);\n log(\n 4,\n `[pool] The number of all resources in pool (free or in use): ${size}.`\n );\n log(\n 4,\n `[pool] The number of resources that are currently available: ${available}.`\n );\n log(\n 4,\n `[pool] The number of resources that are currently acquired: ${borrowed}.`\n );\n log(\n 4,\n `[pool] The number of callers waiting to acquire a resource: ${pending}.`\n );\n log(\n 4,\n `[pool] The number of how many more resources can the pool manage/create: ${spareResourceCapacity}.`\n );\n}\n\nexport default {\n init,\n killPool,\n postWork,\n getPool,\n getPoolInfo,\n getPoolInfoJSON,\n workAttempts: () => exportAttempts,\n droppedWork: () => droppedExports,\n averageTime: () => spentAverage,\n processedWorkCount: () => performedExports\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport cache from '../../cache.js';\nimport pool from '../../pool.js';\n\nconst packageVersion = process.env.npm_package_version;\nconst serverStartTime = new Date();\n\n/**\n * Adds the /health route which outputs basic stats for the server\n */\nexport default (app) =>\n !app\n ? false\n : app.get('/health', (request, response) => {\n response.send({\n status: 'OK',\n bootTime: serverStartTime,\n uptime:\n Math.floor(\n (new Date().getTime() - serverStartTime.getTime()) / 1000 / 60\n ) + ' minutes',\n version: packageVersion,\n highchartsVersion: cache.version(),\n averageProcessingTime: pool.averageTime(),\n performedExports: pool.processedWorkCount(),\n failedExports: pool.droppedWork(),\n exportAttempts: pool.workAttempts(),\n sucessRatio: (pool.processedWorkCount() / pool.workAttempts()) * 100,\n // eslint-disable-next-line import/no-named-as-default-member\n pool: pool.getPoolInfoJSON()\n });\n });\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { existsSync, readFileSync, promises as fsPromises } from 'fs';\n\nimport prompts from 'prompts';\n\nimport { log } from './logger.js';\nimport { deepCopy, isObject, printUsage, toBoolean } from './utils.js';\nimport {\n absoluteProps,\n defaultConfig,\n nestedArgs,\n promptsConfig\n} from './schemas/config.js';\n\nlet generalOptions = {};\n\n/**\n * Getter for the general options.\n *\n * @return {object} - General options object.\n */\nexport const getOptions = () => generalOptions;\n\n/**\n * Initializes and sets the general options for the server instace.\n *\n * @param {object} userOptions - Additional user options (e.g. from the node\n * module usage).\n * @param {string[]} args - CLI arguments.\n * @return {object} - General options object.\n */\nexport const setOptions = (userOptions, args) => {\n // Only for the CLI usage\n if (args?.length) {\n // Get the additional options from the custom JSON file\n generalOptions = loadConfigFile(args);\n }\n\n // Update the default config with a correct option values\n updateDefaultConfig(defaultConfig, generalOptions);\n\n // Set values for server's options and returns them\n generalOptions = initOptions(defaultConfig);\n\n // Apply user options if there are any\n if (userOptions) {\n // Merge user options\n generalOptions = mergeConfigOptions(\n generalOptions,\n userOptions,\n absoluteProps\n );\n }\n\n // Only for the CLI usage\n if (args?.length) {\n // Pair provided arguments\n generalOptions = pairArgumentValue(generalOptions, args, defaultConfig);\n }\n\n // Return final general options\n return generalOptions;\n};\n\n/**\n * Displays a prompt for the manual configuration.\n *\n * @param {string} configFileName - The name of a configuration file.\n */\nexport const manualConfig = async (configFileName) => {\n // Prepare a config object\n let configFile = {};\n\n // Check if provided config file exists\n if (existsSync(configFileName)) {\n configFile = JSON.parse(readFileSync(configFileName, 'utf8'));\n }\n\n // Question about a configuration category\n const onSubmit = async (p, categories) => {\n let questionsCounter = 0;\n let allQuestions = [];\n\n // Create a corresponding property in the manualConfig object\n for (const section of categories) {\n // Mark each option with a section\n promptsConfig[section] = promptsConfig[section].map((option) => ({\n ...option,\n section\n }));\n\n // Collect the questions\n allQuestions = [...allQuestions, ...promptsConfig[section]];\n }\n\n await prompts(allQuestions, {\n onSubmit: async (prompt, answer) => {\n // Get the default modules\n if (prompt.name === 'modules') {\n answer = answer.length\n ? answer.map((module) => prompt.choices[module])\n : prompt.choices;\n\n configFile[prompt.section][prompt.name] = answer;\n } else {\n configFile[prompt.section] = recursiveProps(\n Object.assign({}, configFile[prompt.section] || {}),\n prompt.name.split('.'),\n answer\n );\n }\n\n if (++questionsCounter === allQuestions.length) {\n try {\n await fsPromises.writeFile(\n configFileName,\n JSON.stringify(configFile, null, 2),\n 'utf8'\n );\n } catch (error) {\n log(1, `[config] Error while creating config.json: ${error}`);\n }\n return true;\n }\n }\n });\n\n return true;\n };\n\n // Find the categories\n const choices = Object.keys(promptsConfig).map((choice) => ({\n title: `${choice} options`,\n value: choice\n }));\n\n // Category prompt\n return prompts(\n {\n type: 'multiselect',\n name: 'category',\n message: 'Which category do you want to configure?',\n hint: 'Space: Select specific, A: Select all, Enter: Confirm.',\n instructions: '',\n choices\n },\n { onSubmit }\n );\n};\n\n/**\n * Maps the old options to the new config structure.\n *\n * @param {object} oldOptions - Options to be mapped.\n */\nexport const mapToNewConfig = (oldOptions) => {\n const newOptions = {};\n // Cycle through old-structured options\n for (const [key, value] of Object.entries(oldOptions)) {\n const propertiesChain = nestedArgs[key] ? nestedArgs[key].split('.') : [];\n\n // Populate object in correct properties levels\n propertiesChain.reduce(\n (obj, prop, index) =>\n (obj[prop] =\n propertiesChain.length - 1 === index ? value : obj[prop] || {}),\n newOptions\n );\n }\n return newOptions;\n};\n\n/**\n * Merges the new options to the options object. It omits undefined values.\n *\n * @param {object} options - Old options.\n * @param {object} newOptions - New options.\n * @param {string[]} absoluteProps - Array of object names that should be force\n * merged.\n */\nexport const mergeConfigOptions = (options, newOptions, absoluteProps = []) => {\n const mergedOptions = deepCopy(options);\n\n for (const [key, value] of Object.entries(newOptions)) {\n mergedOptions[key] =\n isObject(value) &&\n !absoluteProps.includes(key) &&\n mergedOptions[key] !== undefined\n ? mergeConfigOptions(mergedOptions[key], value, absoluteProps)\n : value !== undefined\n ? value\n : mergedOptions[key];\n }\n\n return mergedOptions;\n};\n\n/**\n * Initializes options for the `startExport` method by merging user options\n * with the general options.\n *\n * @param {any} exportOptions - User options for exporting.\n * @param {any} generalOptions - General options are used for the export server.\n * @return {object} - User options merged with default options.\n */\nexport const initExportSettings = (exportOptions, generalOptions = {}) => {\n let options = {};\n\n if (exportOptions.svg) {\n options = deepCopy(generalOptions);\n options.export.type = exportOptions.type || exportOptions.export.type;\n options.export.scale = exportOptions.scale || exportOptions.export.scale;\n options.export.outfile =\n exportOptions.outfile || exportOptions.export.outfile;\n options.payload = {\n svg: exportOptions.svg\n };\n } else {\n options = mergeConfigOptions(\n generalOptions,\n exportOptions,\n // Omit going down recursively with the belows\n absoluteProps\n );\n }\n\n options.export.outfile =\n options.export?.outfile || `chart.${options.export?.type || 'png'}`;\n return options;\n};\n\n/**\n * Loads the configuration from a custom JSON file.\n *\n * @param {string[]} args - CLI arguments.\n * @return {object} - Options object from the JSON file.\n */\nfunction loadConfigFile(args) {\n // Check if the --loadConfig option was used\n const configIndex = args.findIndex(\n (arg) => arg.replace(/-/g, '') === 'loadConfig'\n );\n\n // Check if the --loadConfig has a value\n if (configIndex > -1 && args[configIndex + 1]) {\n const fileName = args[configIndex + 1];\n try {\n // Check if an additional config file is a correct JSON file\n if (fileName && fileName.endsWith('.json')) {\n // Load an optional custom JSON config file\n return JSON.parse(readFileSync(fileName));\n }\n } catch (error) {\n log(1, `[config] Unable to load config from the ${fileName}: ${error}`);\n }\n }\n\n // No additional options to return\n return {};\n}\n\n/**\n * Setting correct values of the options from the default config.\n *\n * @param {object} configObj - The config object based on which the initial\n * configuration be made.\n * @param {object} customObj - The custom object which can contain additional\n * option values to set.\n * @param {string} propChain - Required for creating a string chain of\n * properties for nested arguments.\n */\nfunction updateDefaultConfig(configObj, customObj = {}, propChain = '') {\n Object.keys(configObj).forEach((key) => {\n if (!['puppeteer', 'highcharts'].includes(key)) {\n const entry = configObj[key];\n const customValue = customObj && customObj[key];\n let numEnvVal;\n\n if (typeof entry.value === 'undefined') {\n updateDefaultConfig(entry, customValue, `${propChain}.${key}`);\n } else {\n // If a value from a custom JSON exists, it take precedence\n if (customValue !== undefined) {\n entry.value = customValue;\n }\n\n // If a value from an env variable exists, it take precedence\n if (entry.envLink) {\n // Load the env var\n if (entry.type === 'boolean') {\n entry.value = toBoolean(\n [process.env[entry.envLink], entry.value].find(\n (el) => el || el === 'false'\n )\n );\n } else if (entry.type === 'number') {\n numEnvVal = +process.env[entry.envLink];\n entry.value = numEnvVal >= 0 ? numEnvVal : entry.value;\n } else if (\n entry.type.indexOf(']') >= 0 &&\n process.env[entry.envLink]\n ) {\n entry.value = process.env[entry.envLink].split(',');\n } else {\n entry.value = process.env[entry.envLink] || entry.value;\n }\n }\n }\n }\n });\n}\n\n/**\n * Inits options recursively.\n *\n * @param {any} items - Items to update options from.\n * @return {object} - Updated options object.\n */\nfunction initOptions(items) {\n let options = {};\n for (const [name, item] of Object.entries(items)) {\n options[name] = Object.prototype.hasOwnProperty.call(item, 'value')\n ? item.value\n : initOptions(item);\n }\n return options;\n}\n\n/**\n * Pairs argument with a corresponding value.\n *\n * @param {object} options - All server options.\n * @param {string[]} args - Array of arguments from a user.\n * @param {object} defaultConfig - The default config object.\n */\nfunction pairArgumentValue(options, args, defaultConfig) {\n for (let i = 0; i < args.length; i++) {\n let option = args[i].replace(/-/g, '');\n\n // Find the right place for property's value\n const propertiesChain = nestedArgs[option]\n ? nestedArgs[option].split('.')\n : [];\n\n propertiesChain.reduce((obj, prop, index) => {\n if (propertiesChain.length - 1 === index) {\n // Finds an option and set a corresponding value\n if (typeof obj[prop] !== 'undefined') {\n if (args[++i]) {\n obj[prop] = args[i] || obj[prop];\n } else {\n console.log(`Missing argument value for ${option}!`.red, '\\n');\n options = printUsage(defaultConfig);\n }\n }\n }\n return obj[prop];\n }, options);\n }\n\n return options;\n}\n\n/**\n * Recursively sets a property in a correct indentation level based on the\n * array of nested properties names.\n *\n * @param {object} objectToUpdate - Object where a property must be set on a\n * correct level.\n * @param {string[]}nestedNames - Array of nasted names that indicates\n * indentation level.\n * @param {any} value - A value to assign to the property.\n * @return {object} - Updated options object.\n */\nfunction recursiveProps(objectToUpdate, nestedNames, value) {\n while (nestedNames.length > 1) {\n const propName = nestedNames.shift();\n\n // Create a property in object if it doesn't exist\n if (!Object.prototype.hasOwnProperty.call(objectToUpdate, propName)) {\n objectToUpdate[propName] = {};\n }\n\n // Call function again if there still names to go\n objectToUpdate[propName] = recursiveProps(\n Object.assign({}, objectToUpdate[propName]),\n nestedNames,\n value\n );\n\n return objectToUpdate;\n }\n\n // Assign the final value\n objectToUpdate[nestedNames[0]] = value;\n return objectToUpdate;\n}\n\nexport default {\n getOptions,\n setOptions,\n manualConfig,\n mapToNewConfig,\n mergeConfigOptions,\n initExportSettings\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { readFile, readFileSync, writeFileSync } from 'fs';\n\nimport { log } from './logger.js';\nimport { killPool, postWork } from './pool.js';\nimport {\n clearText,\n fixType,\n handleResources,\n isCorrectJSON,\n optionsStringify,\n roundNumber,\n toBoolean,\n wrapAround\n} from './utils.js';\nimport { initExportSettings, getOptions } from './config.js';\n\nlet allowCodeExecution = false;\n\nexport const startExport = async (settings, endCallback) => {\n // Starting exporting process message\n log(4, '[chart] Starting exporting process.');\n\n // Initialize options\n const options = initExportSettings(settings, getOptions());\n\n // Get the export options\n const exportOptions = options.export;\n\n // If SVG is an input (argument can be sent only by the request)\n if (options.payload?.svg && options.payload.svg !== '') {\n return exportAsString(options.payload.svg.trim(), options, endCallback);\n }\n\n // Export using options from the file\n if (exportOptions.infile && exportOptions.infile.length) {\n log(4, '[chart] Attempting to export from an input file.');\n\n // Try to read the file\n return readFile(exportOptions.infile, 'utf8', (error, infile) => {\n if (error) {\n return log(1, `[chart] Error loading input file: ${error}.`);\n }\n\n // Get the string representation\n options.export.instr = infile;\n return exportAsString(options.export.instr.trim(), options, endCallback);\n });\n }\n\n // Export with options from the raw representation\n if (\n (exportOptions.instr && exportOptions.instr !== '') ||\n (exportOptions.options && exportOptions.options !== '')\n ) {\n log(4, '[chart] Attempting to export from a raw input.');\n\n // Perform a direct inject when forced\n if (toBoolean(options.customCode?.allowCodeExecution)) {\n return doStraightInject(options, endCallback);\n }\n\n // Either try to parse to JSON first or do the direct export\n return typeof exportOptions.instr === 'string'\n ? exportAsString(exportOptions.instr.trim(), options, endCallback)\n : doExport(\n options,\n exportOptions.instr || exportOptions.options,\n endCallback\n );\n }\n\n // No input specified, pass an error message to the callback\n log(\n 1,\n clearText(\n `[chart] No input specified.\n ${JSON.stringify(exportOptions, undefined, ' ')}.`\n )\n );\n\n return (\n endCallback &&\n endCallback(false, {\n error: true,\n message: 'No input specified.'\n })\n );\n};\n\nexport const batchExport = (options) => {\n const batchFunctions = [];\n\n // Split and pair the --batch arguments\n for (let pair of options.export.batch.split(';')) {\n pair = pair.split('=');\n if (pair.length === 2) {\n batchFunctions.push(\n new Promise((resolve, reject) => {\n startExport(\n {\n ...options,\n export: {\n ...options.export,\n infile: pair[0],\n outfile: pair[1]\n }\n },\n (info, error) => {\n // Throw an error\n if (error) {\n return reject(error);\n }\n\n // Save the base64 from a buffer to a correct image file\n writeFileSync(\n info.options.export.outfile,\n Buffer.from(info.data, 'base64')\n );\n\n resolve();\n }\n );\n })\n );\n }\n }\n\n // Kill the pool after all exports are done\n Promise.all(batchFunctions)\n .then(() => {\n killPool();\n })\n .catch((error) => {\n log(1, `[chart] Error encountered during batch export: ${error}`);\n killPool();\n });\n};\n\nexport const singleExport = (options) => {\n // Use instr or its alias, options\n options.export.instr = options.export.instr || options.export.options;\n\n // Perform an export\n startExport(options, (info, error) => {\n // Exit process when error\n if (error) {\n log(1, `[cli] ${error.message}`);\n process.exit(1);\n }\n\n const { outfile, type } = info.options.export;\n\n // Save the base64 from a buffer to a correct image file\n writeFileSync(\n outfile || `chart.${type}`,\n type !== 'svg' ? Buffer.from(info.data, 'base64') : info.data\n );\n\n // Kill the pool\n killPool();\n });\n};\n\n/**\n * Function for choosing chart size and scale based on options prioritization.\n *\n * @param {object} options - All options object.\n * @return {object} - An object with updated size and scale for a chart.\n */\nexport const findChartSize = (options) => {\n const { chart, exporting } =\n options.export?.options || isCorrectJSON(options.export?.instr);\n\n // See if globalOptions holds chart or exporting size\n const globalOptions = isCorrectJSON(options.export?.globalOptions);\n\n // Secure scale value\n let scale =\n options.export?.scale ||\n exporting?.scale ||\n globalOptions?.exporting?.scale ||\n options.export?.defaultScale ||\n 1;\n\n // the scale cannot be lower than 0.1 and cannot be higher than 5.0\n scale = Math.max(0.1, Math.min(scale, 5.0));\n\n // we want to round the numbers like 0.23234 -> 0.23\n scale = roundNumber(scale, 2);\n\n // Find chart size and scale\n return {\n height:\n options.export?.height ||\n exporting?.sourceHeight ||\n chart?.height ||\n globalOptions?.exporting?.sourceHeight ||\n globalOptions?.chart?.height ||\n options.export?.defaultHeight ||\n 400,\n width:\n options.export?.width ||\n exporting?.sourceWidth ||\n chart?.width ||\n globalOptions?.exporting?.sourceWidth ||\n globalOptions?.chart?.width ||\n options.export?.defaultWidth ||\n 600,\n scale\n };\n};\n\n/**\n * Function for final options preparation before export.\n *\n * @param {object} options - All options object.\n * @param {object} chartJson - Chart JSON.\n * @param {function} endCallback - The end callback.\n * @param {string} svg - The SVG representation.\n */\nconst doExport = (options, chartJson, endCallback, svg) => {\n let { export: exportOptions, customCode: customCodeOptions } = options;\n\n const allowCodeExecutionScoped =\n typeof customCodeOptions.allowCodeExecution === 'boolean'\n ? customCodeOptions.allowCodeExecution\n : allowCodeExecution;\n\n if (!customCodeOptions) {\n customCodeOptions = options.customCode = {};\n } else if (typeof options.customCode.resources === 'string') {\n // Process resources\n options.customCode.resources = handleResources(\n options.customCode.resources,\n toBoolean(options.customCode.allowFileResources)\n );\n } else if (!options.customCode.resources) {\n try {\n const resources = readFileSync('resources.json', 'utf8');\n options.customCode.resources = handleResources(\n resources,\n toBoolean(options.customCode.allowFileResources)\n );\n } catch (err) {\n log(3, `[chart] The default resources.json file not found.`);\n }\n }\n\n // If the allowCodeExecution flag isn't set, we should refuse the usage\n // of callback, resources, and custom code. Additionally, the worker will\n // refuse to run arbitrary JavaScript. Prioritized should be the scoped\n // option, then we should take a look at the overall pool option.\n if (!allowCodeExecutionScoped && customCodeOptions) {\n if (\n customCodeOptions.callback ||\n customCodeOptions.resources ||\n customCodeOptions.customCode\n ) {\n // Send back a friendly message saying that the exporter does not support\n // these settings.\n return (\n endCallback &&\n endCallback(false, {\n error: true,\n message: clearText(\n `The callback, resources and customCode have been disabled for this\n server.`\n )\n })\n );\n }\n\n // Reset all additional custom code\n customCodeOptions.callback = false;\n customCodeOptions.resources = false;\n customCodeOptions.customCode = false;\n }\n\n // Clean properties to keep it lean and mean\n if (chartJson) {\n chartJson.chart = chartJson.chart || {};\n chartJson.exporting = chartJson.exporting || {};\n chartJson.exporting.enabled = false;\n }\n\n exportOptions.constr = exportOptions.constr || 'chart';\n exportOptions.type = fixType(exportOptions.type, exportOptions.outfile);\n if (exportOptions.type === 'svg') {\n exportOptions.width = false;\n }\n\n // Prepare global and theme options\n ['globalOptions', 'themeOptions'].forEach((optionsName) => {\n try {\n if (exportOptions && exportOptions[optionsName]) {\n if (\n typeof exportOptions[optionsName] === 'string' &&\n exportOptions[optionsName].endsWith('.json')\n ) {\n exportOptions[optionsName] = isCorrectJSON(\n readFileSync(exportOptions[optionsName], 'utf8'),\n true\n );\n } else {\n exportOptions[optionsName] = isCorrectJSON(\n exportOptions[optionsName],\n true\n );\n }\n }\n } catch (error) {\n exportOptions[optionsName] = {};\n log(1, `[chart] The ${optionsName} not found.`);\n }\n });\n\n // Prepare customCode\n if (customCodeOptions.allowCodeExecution) {\n customCodeOptions.customCode = wrapAround(\n customCodeOptions.customCode,\n customCodeOptions.allowFileResources\n );\n }\n\n // Get the callback\n if (\n customCodeOptions &&\n customCodeOptions.callback &&\n customCodeOptions.callback?.indexOf('{') < 0\n ) {\n // The allowFileResources is always set to false for HTTP requests to avoid\n // injecting arbitrary files from the fs\n if (customCodeOptions.allowFileResources) {\n try {\n customCodeOptions.callback = readFileSync(\n customCodeOptions.callback,\n 'utf8'\n );\n } catch (error) {\n log(2, `[chart] Error loading callback: ${error}.`);\n customCodeOptions.callback = false;\n }\n } else {\n customCodeOptions.callback = false;\n }\n }\n\n // Size search\n options.export = {\n ...options.export,\n ...findChartSize(options)\n };\n\n // Post the work to the pool\n postWork(exportOptions.strInj || chartJson || svg, options)\n .then((result) => endCallback(result))\n .catch((error) => {\n log(0, '[chart] When posting work:', error);\n return endCallback(false, error);\n });\n};\n\n/**\n * Function for straight injecting the code.\n * Dangerous and must be used deliberately by someone who sets up a server\n * (see --allowCodeExecution).\n *\n * @param {object} options - All options object.\n * @param {function} endCallback - The function to call when exporting is done.\n */\nconst doStraightInject = (options, endCallback) => {\n try {\n let strInj;\n let instr = options.export.instr || options.export.options;\n\n if (typeof instr !== 'string') {\n // Try to stringify options\n strInj = instr = optionsStringify(\n instr,\n options.customCode?.allowCodeExecution\n );\n }\n strInj = instr.replaceAll(/\\t|\\n|\\r/g, '').trim();\n\n // Get rid of the ;\n if (strInj[strInj.length - 1] === ';') {\n strInj = strInj.substring(0, strInj.length - 1);\n }\n\n // Save as stright inject string\n options.export.strInj = strInj;\n return doExport(options, false, endCallback);\n } catch (error) {\n const message = clearText(\n `Malformed input detected for ${options.export?.requestId || '?'}:\n Please make sure that your JSON/JavaScript options\n are sent using the \"options\" attribute, and that if you're using\n SVG, it is unescaped.`\n );\n\n log(1, message);\n return (\n endCallback &&\n endCallback(\n false,\n JSON.stringify({\n error: true,\n message\n })\n )\n );\n }\n};\n\n/**\n * Prepares an input before exporting.\n *\n * @param {string} stringToExport - String representation of SVG/export options.\n * @param {object} options - All options object.\n * @param {function} endCallback - The function to call when exporting is done.\n */\nconst exportAsString = (stringToExport, options, endCallback) => {\n const { allowCodeExecution } = options.customCode;\n\n // Check if it is SVG\n if (\n stringToExport.indexOf('= 0 ||\n stringToExport.indexOf('= 0\n ) {\n log(4, '[chart] Parsing input as SVG.');\n return doExport(options, false, endCallback, stringToExport);\n }\n\n try {\n // Try to parse to JSON and call the doExport function\n const chartJSON = JSON.parse(stringToExport.replaceAll(/\\t|\\n|\\r/g, ' '));\n\n // If a correct JSON, do the export\n return doExport(options, chartJSON, endCallback);\n } catch (error) {\n // Not a valid JSON\n if (toBoolean(allowCodeExecution)) {\n return doStraightInject(options, endCallback);\n } else {\n // Do not allow straight injection without the allowCodeExecution flag\n return (\n endCallback &&\n endCallback(false, {\n error: true,\n message: clearText(\n `Only JSON configurations and SVG is allowed for this server. If\n this is your server, JavaScript exporting can be enabled by starting\n the server with the --allowCodeExecution flag.`\n )\n })\n );\n }\n }\n};\n\nexport const getAllowCodeExecution = () => allowCodeExecution;\n\nexport const setAllowCodeExecution = (value) => {\n allowCodeExecution = toBoolean(value);\n};\n\n/**\n * Starts an exporting process\n *\n * @param {object} settings - Settings for export.\n * @param {function} endCallback - The function to call when exporting is done.\n */\nexport default {\n batchExport,\n singleExport,\n getAllowCodeExecution,\n setAllowCodeExecution,\n startExport,\n findChartSize\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { v4 as uuid } from 'uuid';\n\nimport { getAllowCodeExecution, startExport } from '../../chart.js';\nimport { getOptions, mergeConfigOptions } from '../../config.js';\nimport { log } from '../../logger.js';\nimport {\n clearText,\n fixType,\n isCorrectJSON,\n isPrivateRangeUrlFound,\n optionsStringify,\n measureTime\n} from '../../utils.js';\n\n// Reversed MIME types\nconst reversedMime = {\n png: 'image/png',\n jpeg: 'image/jpeg',\n gif: 'image/gif',\n pdf: 'application/pdf',\n svg: 'image/svg+xml'\n};\n\n// The requests counter\nlet requestsCounter = 0;\n\nconst benchmark = false;\n\n// The array of callbacks to call before a request\nconst beforeRequest = [];\n\n// The array of callbacks to call after a request\nconst afterRequest = [];\n\n/**\n * Calls callbacks.\n *\n * @param {Array} callbacks - An array of callbacks.\n * @param {object} request - The request.\n * @param {object} response - The response.\n * @param {object} data - The data to send to callbacks.\n * @return {object} - The result from a callback.\n */\nconst doCallbacks = (callbacks, request, response, data) => {\n let result = true;\n const { id, uniqueId, type, body } = data;\n\n callbacks.some((callback) => {\n if (callback) {\n let callResponse = callback(request, response, id, uniqueId, type, body);\n\n if (callResponse !== undefined && callResponse !== true) {\n result = callResponse;\n }\n\n return true;\n }\n });\n\n return result;\n};\n\n/**\n * Handles an export.\n *\n * @param {object} request - The request.\n * @param {object} response - The response.\n */\nconst exportHandler = (request, response) => {\n // Start counting time\n const stopCounter = measureTime();\n\n // Get the current server's general options\n const defaultOptions = getOptions();\n\n // Init default options\n if (benchmark) {\n console.log('Init default options:', stopCounter(), 'ms.');\n }\n\n const body = request.body;\n const id = ++requestsCounter;\n const uniqueId = uuid().replace(/-/g, '');\n let type = fixType(body.type);\n\n // Fix type\n if (benchmark) {\n console.log('Fix type:', stopCounter(), 'ms.');\n }\n\n // Throw 'Bad Request' if there's no body\n if (!body) {\n return response.status(400).send(\n clearText(\n `Body is required. Sending a body? Make sure your Content-type header\n is correct. Accepted is application/json and multipart/form-data.`\n )\n );\n }\n\n // All of the below can be used\n let instr = isCorrectJSON(body.infile || body.options || body.data);\n\n // Is correct JSON\n if (benchmark) {\n console.log('Is correct JSON:', stopCounter(), 'ms.');\n }\n\n // Throw 'Bad Request' if there's no JSON or SVG to export\n if (!instr && !body.svg) {\n log(\n 2,\n clearText(\n `Request ${uniqueId} from ${\n request.headers['x-forwarded-for'] || request.connection.remoteAddress\n } was incorrect. Check your payload.`\n )\n );\n\n return response.status(400).send(\n clearText(\n `No correct chart data found. Please make sure you are using\n application/json or multipart/form-data headers, and that the chart\n data is in the 'infile', 'options' or 'data' attribute if sending\n JSON or in the 'svg' if sending SVG.`\n )\n );\n }\n\n let callResponse = false;\n\n // Call the before request functions\n callResponse = doCallbacks(beforeRequest, request, response, {\n id,\n uniqueId,\n type,\n body\n });\n\n // Do callbacks\n if (benchmark) {\n console.log('Do callbacks:', stopCounter(), 'ms.');\n }\n\n // Block the request if one of a callbacks failed\n if (callResponse !== true) {\n return response.send(callResponse);\n }\n\n let connectionAborted = false;\n\n // In case the connection is closed, force to abort further actions\n request.socket.on('close', () => {\n connectionAborted = true;\n });\n\n log(4, `[export] Got an incoming HTTP request ${uniqueId}.`);\n\n body.constr = (typeof body.constr === 'string' && body.constr) || 'chart';\n\n // Gather and organize options from the payload\n const requestOptions = {\n export: {\n instr,\n type,\n constr: body.constr[0].toLowerCase() + body.constr.substr(1),\n height: body.height,\n width: body.width,\n scale: body.scale || defaultOptions.export.scale,\n globalOptions: isCorrectJSON(body.globalOptions, true),\n themeOptions: isCorrectJSON(body.themeOptions, true)\n },\n customCode: {\n allowCodeExecution: getAllowCodeExecution(),\n allowFileResources: false,\n resources: isCorrectJSON(body.resources, true),\n callback: body.callback,\n customCode: body.customCode\n }\n };\n\n // Organize options\n if (benchmark) {\n console.log('Organize options:', stopCounter(), 'ms.');\n }\n\n if (instr) {\n // Stringify JSON with options\n requestOptions.export.instr = optionsStringify(\n instr,\n requestOptions.customCode.allowCodeExecution\n );\n\n // Stringify JSON with options\n if (benchmark) {\n console.log('Stringify JSON with options:', stopCounter(), 'ms.');\n }\n }\n\n // Merge the request options into default ones\n const options = mergeConfigOptions(defaultOptions, requestOptions);\n\n // Merge config options\n if (benchmark) {\n console.log('Merge config options:', stopCounter(), 'ms.');\n }\n\n // Save the JSON if exists\n options.export.options = instr;\n\n // Lastly, add the server specific arguments into options as payload\n options.payload = {\n svg: body.svg || false,\n b64: body.b64 || false,\n dataOptions: isCorrectJSON(body.dataOptions, true),\n noDownload: body.noDownload || false,\n requestId: uniqueId\n };\n\n // Setting payload\n if (benchmark) {\n console.log('Setting payload:', stopCounter(), 'ms.');\n }\n\n // Test xlink:href elements from payload's SVG\n if (body.svg && isPrivateRangeUrlFound(options.payload.svg)) {\n return response\n .status(400)\n .send(\n 'SVG potentially contain at least one forbidden URL in xlink:href element.'\n );\n }\n\n // Check URL range\n if (benchmark) {\n console.log('Check URL range:', stopCounter(), 'ms.');\n }\n\n // Start the export process\n startExport(options, (info, error) => {\n // Remove the close event from the socket\n request.socket.removeAllListeners('close');\n\n // After Puppeteer exporting\n if (benchmark) {\n console.log('After Puppeteer exporting:', stopCounter(), 'ms.', '\\n');\n }\n\n // If the connection was closed, do nothing\n if (connectionAborted) {\n return log(\n 3,\n clearText(\n `[export] The client closed the connection before the chart was done\n processing.`\n )\n );\n }\n\n // If error, return it\n if (error) {\n log(\n 1,\n clearText(\n `[export] Work: ${uniqueId} could not be completed, sending:\n ${error}`\n )\n );\n return response.status(400).send(error.message);\n }\n\n // If data is missing, return the error\n if (!info || !info.data) {\n log(\n 1,\n clearText(\n `[export] Unexpected return from chart generation, please check your\n data Request: ${uniqueId} is ${info.data}.`\n )\n );\n return response\n .status(400)\n .send(\n 'Unexpected return from chart generation, please check your data.'\n );\n }\n\n // Get the type from options\n type = info.options.export.type;\n\n // The after request callbacks\n doCallbacks(afterRequest, request, response, { id, body: info.data });\n\n if (info.data) {\n // If only base64 is required, return it\n if (body.b64) {\n // Check if it is already base64 or a raw SVG\n if (type === 'pdf') {\n return response.send(\n Buffer.from(info.data, 'utf8').toString('base64')\n );\n }\n return response.send(info.data);\n }\n\n // Set correct content type\n response.header('Content-Type', reversedMime[type] || 'image/png');\n\n // Decide whether to download or not chart file\n if (!body.noDownload) {\n response.attachment(\n `${request.params.filename || 'chart'}.${type || 'png'}`\n );\n }\n\n // If SVG, return plain content\n return type === 'svg'\n ? response.send(info.data)\n : response.send(Buffer.from(info.data, 'base64'));\n }\n });\n};\n\nexport default (app) => {\n app.post('/', exportHandler);\n app.post('/:filename', exportHandler);\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { promises as fsPromises } from 'fs';\nimport { posix } from 'path';\n\nimport bodyParser from 'body-parser';\nimport cors from 'cors';\nimport express from 'express';\nimport multer from 'multer';\nimport http from 'http';\nimport https from 'https';\n\nimport { log } from '../logger.js';\nimport rateLimit from './rate_limit.js';\nimport { __dirname } from '../utils.js';\n\nimport healthRoute from './routes/health.js';\nimport exportRoutes from './routes/export.js';\nimport vswitchRoute from './routes/change_hc_version.js';\nimport uiRoute from './routes/ui.js';\n\n// Create express app\nconst app = express();\n\n// Disable the X-Powered-By header\napp.disable('x-powered-by');\n\n// Enable CORS support\napp.use(cors());\n\n// Enable parsing of form data (files) with Multer package\nconst storage = multer.memoryStorage();\nconst upload = multer({\n storage,\n limits: {\n fieldsSize: '50MB'\n }\n});\n\napp.use(upload.any());\n\n// Enable body parser\napp.use(bodyParser.json({ limit: '50mb' }));\napp.use(bodyParser.urlencoded({ extended: true, limit: '50mb' }));\napp.use(bodyParser.urlencoded({ extended: false, limit: '50mb' }));\n\n/**\n * Error handler function.\n *\n * @param {object} error - An error object.\n * @return {string} - An error message.\n */\nconst errorHandler = (error) => log(1, `[server] Socket error: ${error}`);\n\n/**\n * Attaches error handlers for a server.\n *\n * @param {object} server - The http/https server.\n */\nconst attachErrorHandlers = (server) => {\n server.on('clientError', errorHandler);\n server.on('error', errorHandler);\n server.on('connection', (socket) =>\n socket.on('error', (error) => errorHandler(error, socket))\n );\n};\n\nexport const startServer = async (serverConfig) => {\n // Stop if not enabled\n if (!serverConfig.enable) {\n return false;\n }\n\n // // Get the pool\n // const pool = getPool();\n\n // // Try to create browser instance before starting the server\n // const resource = await pool.acquire();\n\n // // If not found, throw an error\n // if (!resource.browser) {\n // log(1, `[server] Could not acquire browser instance.`);\n // process.exit(1);\n // }\n\n // // Release the resource\n // pool.release(resource);\n\n // Listen HTTP server\n if (!serverConfig.ssl.enable && !serverConfig.ssl.force) {\n // Main server instance (HTTP)\n const httpServer = http.createServer(app);\n // Attach error handlers and listen to the server\n attachErrorHandlers(httpServer);\n // Listen\n httpServer.listen(serverConfig.port, serverConfig.host);\n\n log(\n 3,\n `[server] Started HTTP server on ${serverConfig.host}:${serverConfig.port}.`\n );\n }\n\n // Listen HTTPS server\n if (serverConfig.ssl.enable) {\n // Set up an SSL server also\n let key, cert;\n\n try {\n // Get the SSL key\n key = await fsPromises.readFile(\n posix.join(serverConfig.ssl.certPath, 'server.key'),\n 'utf8'\n );\n\n // Get the SSL certificate\n cert = await fsPromises.readFile(\n posix.join(serverConfig.ssl.certPath, 'server.crt'),\n 'utf8'\n );\n } catch (error) {\n log(\n 1,\n `[server] Unable to load key/certificate from ${serverConfig.ssl.certPath}.`\n );\n }\n\n if (key && cert) {\n // Main server instance (HTTPS)\n const httpsServer = https.createServer(app);\n // Attach error handlers and listen to the server\n attachErrorHandlers(httpsServer);\n // Listen\n httpsServer.listen(serverConfig.ssl.port, serverConfig.host);\n\n log(\n 3,\n `[server] Started HTTPS server on ${serverConfig.host}:${serverConfig.ssl.port}.`\n );\n }\n }\n\n // Enable the rate limiter if config says so\n if (\n serverConfig.rateLimiting &&\n serverConfig.rateLimiting.enable &&\n ![0, NaN].includes(serverConfig.rateLimiting.maxRequests)\n ) {\n rateLimit(app, serverConfig.rateLimiting);\n }\n\n // Set up static folder's route\n app.use(express.static(posix.join(__dirname, 'public')));\n\n // Set up routes\n healthRoute(app);\n exportRoutes(app);\n uiRoute(app);\n vswitchRoute(app);\n};\n\n/**\n * Returns the express instance.\n */\nexport const getExpress = () => {\n return express;\n};\n\n/**\n * Returns the app instance.\n */\nexport const getApp = () => {\n return app;\n};\n\n/**\n * Adds a middleware to the server.\n *\n * @param {object} path - An endpoint path to add middlewares to.\n * @param {Array} middlewares - An unlimited number of middlewares to use\n * against the specific endpoint.\n */\nexport const use = (path, ...middlewares) => {\n app.use(path, ...middlewares);\n};\n\n/**\n * Adds a get route to the server.\n *\n * @param {object} path - An endpoint path to add middlewares to.\n * @param {Array} middlewares - An unlimited number of middlewares to use\n * against the specific endpoint for GET method.\n */\nexport const get = (path, ...middlewares) => {\n app.get(path, ...middlewares);\n};\n\n/**\n * Adds a post route to the server.\n *\n * @param {object} path - An endpoint path to add middlewares to.\n * @param {Array} middlewares - An unlimited number of middlewares to use\n * against the specific endpoint for POST method.\n */\nexport const post = (path, ...middlewares) => {\n app.post(path, ...middlewares);\n};\n\n/**\n * Forcefully enables rate limiting.\n *\n * @param {object} limitConfig - The options object for the rate limiter\n * configuration.\n */\nexport const enableRateLimiting = (limitConfig) => {\n return rateLimit(app, limitConfig);\n};\n\nexport default {\n startServer,\n getExpress,\n getApp,\n use,\n get,\n post,\n enableRateLimiting\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { join } from 'path';\n\nimport { __dirname } from '../../utils.js';\n/**\n * Adds the / route for a UI when enabled for the export server\n */\nexport default (app) =>\n !app\n ? false\n : app.get('/', (request, response) => {\n response.sendFile(join(__dirname, 'public', 'index.html'));\n });\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport cache from '../../cache.js';\n\n/**\n * Adds a route that can be used to change the HC version on the server\n * TODO: Add auth token and connect to API\n */\nexport default (app) =>\n !app\n ? false\n : app.post('/change_hc_version/:newVersion', async (request, response) => {\n const ctoken = process.env.HIGHCHARTS_ADMIN_TOKEN;\n\n if (!ctoken || !ctoken.length) {\n return response.send({\n error: true,\n message:\n 'Server not configured to do run-time version changes: HIGHCHARTS_ADMIN_TOKEN not set'\n });\n }\n\n const token = request.get('hc-auth');\n\n if (!token || token !== ctoken) {\n return response.send({\n error: true,\n message: 'Invalid or missing token: set token in the hc-auth header'\n });\n }\n\n const newVersion = request.params.newVersion;\n\n if (newVersion) {\n try {\n // eslint-disable-next-line import/no-named-as-default-member\n await cache.updateVersion(newVersion);\n } catch (e) {\n response.send({\n error: true,\n message: e\n });\n }\n\n response.send({\n version: cache.version()\n });\n } else {\n response.send({\n error: true,\n message: 'No new version supplied'\n });\n }\n });\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\n// Add the main directory in the global object\nimport 'colors';\n\nimport server, { startServer } from './server/server.js';\nimport {\n setAllowCodeExecution,\n batchExport,\n singleExport,\n startExport\n} from './chart.js';\nimport { mapToNewConfig, setOptions } from './config.js';\nimport { log, setLogLevel, enableFileLogging } from './logger.js';\nimport { killPool, init } from './pool.js';\nimport { checkCache } from './cache.js';\n\nexport default {\n log,\n mapToNewConfig,\n setOptions,\n singleExport,\n startExport,\n batchExport,\n server,\n startServer,\n killPool,\n initPool: async (options = {}) => {\n // Set the allowCodeExecution per export module scope\n setAllowCodeExecution(\n options.customCode && options.customCode.allowCodeExecution\n );\n\n // Set the log level\n setLogLevel(options.logging && parseInt(options.logging.level));\n\n // Set the log file path and name\n if (options.logging && options.logging.dest) {\n enableFileLogging(\n options.logging.dest,\n options.logging.file || 'highcharts-export-server.log'\n );\n }\n\n // Check if cache needs to be updated\n await checkCache(options.highcharts || { version: 'latest' });\n\n // Init the pool\n await init({\n pool: options.pool || {\n initialWorkers: 1,\n maxWorkers: 1\n },\n puppeteerArgs: options.puppeteer?.args || []\n });\n\n // Return updated options\n return options;\n }\n};\n"],"names":["dotenv","config","defaultConfig","puppeteer","args","value","type","description","highcharts","version","envLink","cdnURL","coreScripts","modules","indicators","scripts","forceFetch","export","infile","instr","options","outfile","constr","defaultHeight","defaultWidth","defaultScale","height","width","scale","globalOptions","themeOptions","batch","customCode","allowCodeExecution","allowFileResources","callback","resources","loadConfig","createConfig","server","enable","cliName","host","port","ssl","force","certPath","rateLimiting","maxRequests","window","delay","trustProxy","skipKey","skipToken","pool","initialWorkers","maxWorkers","workLimit","queueSize","timeoutThreshold","acquireTimeout","reaper","benchmarking","listenToProcessExits","logging","level","file","dest","ui","route","other","noLogo","payload","join","absoluteProps","nestedArgs","createNestedArgs","obj","propChain","Object","keys","forEach","k","includes","entry","substring","toConsole","toFile","pathCreated","levelsDesc","title","color","listeners","key","option","entries","log","newLevel","texts","length","prefix","Date","toString","split","trim","fn","existsSync","mkdirSync","appendFile","concat","error","console","apply","undefined","__dirname","fileURLToPath","URL","url","clearText","text","rule","replacer","replaceAll","fixType","formats","outType","pop","find","t","handleResources","allowedProps","handledResources","correctResources","endsWith","isCorrectJSON","readFileSync","notice","files","propName","map","item","data","parsedData","JSON","parse","stringify","deepCopy","copy","Array","isArray","prototype","hasOwnProperty","call","optionsStringify","allowFunctions","name","startsWith","printUsage","bold","yellow","cycleCategories","categories","descName","green","i","blue","category","toUpperCase","red","toBoolean","wrapAround","replace","rateLimit","app","limitConfig","msg","rateOptions","max","limiter","windowMs","delayMs","handler","request","response","format","json","status","send","message","default","skip","query","access_token","use","async","fetch","requestOptions","Promise","resolve","reject","protocol","https","http","getProtocol","get","res","on","chunk","cachePath","cache","activeManifest","sources","hcVersion","appliedConfig","extractVersion","substr","indexOf","fetchScript","script","proxyAgent","agent","timeout","process","env","statusCode","updateCache","sourcePath","customScripts","allScripts","c","m","proxyHost","proxyPort","HttpsProxyAgent","fetchedModules","all","writeFileSync","checkCache","manifestPath","requestUpdate","manifest","moduleMap","numberOfModules","some","moduleName","newManifest","saveConfigToManifest","cache$1","newVersion","assign","RANDOM_PID","randomBytes","PUPPETEER_DIR","path","minimalArgs","template","fs","browser","newPage","p","setContent","addScriptTag","evaluate","setupHighcharts","err","$eval","element","errorMessage","_displayErrors","innerHTML","close","connected","__basedir","setAsConfig","page","chart","triggerExport","puppeteerExport","injectedResources","clearInjected","dispose","scriptsToRemove","document","getElementsByTagName","stylesToRemove","linksToRemove","remove","exportBench","exportOptions","requestAnimationFrame","displayErrors","debugger","d","svgBench","isSVG","setPageBench","svgTemplate","strInj","setContentBench","resBench","js","push","content","isLocal","cssBench","css","cssImports","match","cssImportPath","addStyleTag","size","chartHeight","baseVal","chartWidth","parseFloat","Highcharts","charts","vpBench","viewportHeight","Math","ceil","viewportWidth","setViewport","deviceScaleFactor","zoomCallback","body","style","zoom","margin","x","y","getBoundingClientRect","trunc","getClipRegion","round","expBenchmark","outerHTML","createSVG","encoding","clip","race","screenshot","omitBackground","setTimeout","Error","createImage","pdf","createPDF","oldCharts","oldChart","destroy","shift","puppeteerArgs","performedExports","exportAttempts","timeSpent","droppedExports","spentAverage","poolConfig","factory","create","id","uuid","s","getTime","browserNewPage","isClosed","workCount","random","validate","workerHandle","logLevel","init","allArgs","tryCount","open","launch","headless","userDataDir","e","createBrowser","killPool","code","exit","Pool","min","createRetryIntervalMillis","createTimeoutMillis","acquireTimeoutMillis","destroyTimeoutMillis","idleTimeoutMillis","reapIntervalMillis","propagateCreateError","eventId","resource","initialResources","acquire","promise","release","destroyed","postWork","fail","getPoolInfo","workStart","result","exportTime","available","borrowed","pending","spareResourceCapacity","pool$1","packageVersion","npm_package_version","serverStartTime","generalOptions","getOptions","mergeConfigOptions","newOptions","mergedOptions","updateDefaultConfig","configObj","customObj","customValue","numEnvVal","el","initOptions","items","startExport","settings","endCallback","svg","initExportSettings","exportAsString","readFile","doStraightInject","doExport","findChartSize","exporting","precision","multiplier","pow","roundNumber","sourceHeight","sourceWidth","chartJson","customCodeOptions","allowCodeExecutionScoped","enabled","optionsName","then","catch","requestId","stringToExport","chartJSON","reversedMime","png","jpeg","gif","requestsCounter","beforeRequest","afterRequest","doCallbacks","callbacks","uniqueId","callResponse","exportHandler","start","hrtime","bigint","measureTime","defaultOptions","headers","connection","remoteAddress","connectionAborted","socket","toLowerCase","b64","dataOptions","noDownload","ipRegEx","info","removeAllListeners","Buffer","from","header","attachment","params","filename","express","disable","cors","storage","multer","memoryStorage","upload","limits","fieldsSize","any","bodyParser","limit","urlencoded","extended","errorHandler","attachErrorHandlers","startServer","serverConfig","httpServer","createServer","listen","cert","fsPromises","posix","httpsServer","NaN","static","bootTime","uptime","floor","highchartsVersion","averageProcessingTime","failedExports","sucessRatio","healthRoute","post","exportRoutes","sendFile","uiRoute","ctoken","HIGHCHARTS_ADMIN_TOKEN","token","vswitchRoute","getExpress","getApp","middlewares","enableRateLimiting","index","mapToNewConfig","oldOptions","propertiesChain","reduce","prop","setOptions","userOptions","configIndex","findIndex","arg","fileName","loadConfigFile","pairArgumentValue","singleExport","batchExport","batchFunctions","pair","initPool","parseInt","logDest","logFile","enableFileLogging"],"mappings":"snBAiBAA,EAAOC,SAIA,MAAMC,EAAgB,CAC3BC,UAAW,CACTC,KAAM,CACJC,MAAO,GACPC,KAAM,WACNC,YAAa,6CAGjBC,WAAY,CACVC,QAAS,CACPJ,MAAO,SACPK,QAAS,qBACTJ,KAAM,SACNC,YAAa,8BAEfI,OAAQ,CACNN,MAAO,+BACPK,QAAS,iBACTJ,KAAM,SACNC,YAAa,6CAEfK,YAAa,CACXF,QAAS,0BACTL,MAAO,CAAC,aAAc,kBAAmB,iBACzCC,KAAM,WACNC,YAAa,qCAEfM,QAAS,CACPH,QAAS,qBACTL,MAAO,CACL,QACA,MACA,QACA,YACA,cACA,uBACA,gBACA,uBACA,eACA,QACA,OACA,mBACA,eACA,cACA,UACA,UACA,WACA,UACA,cACA,YACA,sBACA,SACA,SACA,WACA,YACA,eACA,SACA,eACA,YACA,kBACA,SACA,cACA,mBACA,eACA,cACA,eACA,cACA,cACA,WACA,eACA,WACA,SACA,OACA,WACA,YACA,SACA,qBACA,aACA,WACA,WACA,WACA,WACA,eACA,UACA,kBACA,oBACA,cAEFC,KAAM,WACNC,YAAa,gCAEfO,WAAY,CACVJ,QAAS,wBACTL,MAAO,CAAC,kBACRC,KAAM,WACNC,YAAa,mCAEfQ,QAAS,CACPV,MAAO,CACL,wEACA,kGAEFC,KAAM,WACNC,YACE,qEAEJS,WAAY,CACVN,QAAS,yBACTL,OAAO,EACPC,KAAM,UACNC,YACE,oEAGNU,OAAQ,CACNC,OAAQ,CACNb,OAAO,EACPC,KAAM,SACNC,YACE,8FAEJY,MAAO,CACLd,OAAO,EACPC,KAAM,SACNC,YACE,iFAEJa,QAAS,CACPf,OAAO,EACPC,KAAM,SACNC,YAAa,oCAEfc,QAAS,CACPhB,OAAO,EACPC,KAAM,SACNC,YACE,2FAEJD,KAAM,CACJI,QAAS,sBACTL,MAAO,MACPC,KAAM,SACNC,YACE,sEAEJe,OAAQ,CACNZ,QAAS,wBACTL,MAAO,QACPC,KAAM,SACNC,YACE,6EAEJgB,cAAe,CACbb,QAAS,wBACTL,MAAO,IACPC,KAAM,SACNC,YACE,gFAEJiB,aAAc,CAC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"} \ No newline at end of file +{"version":3,"file":"index.esm.js","sources":["../lib/schemas/config.js","../lib/logger.js","../lib/utils.js","../lib/server/rate_limit.js","../lib/fetch.js","../lib/cache.js","../lib/browser.js","../lib/export.js","../lib/benchmark.js","../templates/svg_export/svg_export.js","../lib/pool.js","../lib/server/routes/health.js","../lib/config.js","../lib/chart.js","../lib/server/routes/export.js","../lib/server/server.js","../lib/server/routes/ui.js","../lib/server/routes/change_hc_version.js","../lib/index.js"],"sourcesContent":["/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\n// Load .env into environment variables\nimport dotenv from 'dotenv';\n\ndotenv.config();\n\n// This is the configuration object with all options and their default values,\n// also from the .env file if one exists\nexport const defaultConfig = {\n puppeteer: {\n args: {\n value: [],\n type: 'string[]',\n description: 'Array of arguments to send to puppeteer.'\n }\n },\n highcharts: {\n version: {\n value: 'latest',\n envLink: 'HIGHCHARTS_VERSION',\n type: 'string',\n description: 'Highcharts version to use.'\n },\n cdnURL: {\n value: 'https://code.highcharts.com/',\n envLink: 'HIGHCHARTS_CDN',\n type: 'string',\n description: 'The CDN URL of Highcharts scripts to use.'\n },\n coreScripts: {\n envLink: 'HIGHCHARTS_CORE_SCRIPTS',\n value: ['highcharts', 'highcharts-more', 'highcharts-3d'],\n type: 'string[]',\n description: 'Highcharts core scripts to fetch.'\n },\n modules: {\n envLink: 'HIGHCHARTS_MODULES',\n value: [\n 'stock',\n 'map',\n 'gantt',\n 'exporting',\n 'export-data',\n 'parallel-coordinates',\n 'accessibility',\n 'annotations-advanced',\n 'boost-canvas',\n 'boost',\n 'data',\n 'draggable-points',\n 'static-scale',\n 'broken-axis',\n 'heatmap',\n 'tilemap',\n 'timeline',\n 'treemap',\n 'treegraph',\n 'item-series',\n 'drilldown',\n 'histogram-bellcurve',\n 'bullet',\n 'funnel',\n 'funnel3d',\n 'pyramid3d',\n 'networkgraph',\n 'pareto',\n 'pattern-fill',\n 'pictorial',\n 'price-indicator',\n 'sankey',\n 'arc-diagram',\n 'dependency-wheel',\n 'series-label',\n 'solid-gauge',\n 'sonification',\n 'stock-tools',\n 'streamgraph',\n 'sunburst',\n 'variable-pie',\n 'variwide',\n 'vector',\n 'venn',\n 'windbarb',\n 'wordcloud',\n 'xrange',\n 'no-data-to-display',\n 'drag-panes',\n 'debugger',\n 'dumbbell',\n 'lollipop',\n 'cylinder',\n 'organization',\n 'dotplot',\n 'marker-clusters',\n 'hollowcandlestick',\n 'heikinashi'\n ],\n type: 'string[]',\n description: 'Highcharts modules to fetch.'\n },\n indicators: {\n envLink: 'HIGHCHARTS_INDICATORS',\n value: ['indicators-all'],\n type: 'string[]',\n description: 'Highcharts indicators to fetch.'\n },\n scripts: {\n value: [\n 'https://cdnjs.cloudflare.com/ajax/libs/moment.js/2.29.4/moment.min.js',\n 'https://cdnjs.cloudflare.com/ajax/libs/moment-timezone/0.5.34/moment-timezone-with-data.min.js'\n ],\n type: 'string[]',\n description:\n 'Additional direct scripts/optional dependencies (e.g. moment.js).'\n },\n forceFetch: {\n envLink: 'HIGHCHARTS_FORCE_FETCH',\n value: false,\n type: 'boolean',\n description:\n 'Should all the scripts be refetched after rerunning the server.'\n }\n },\n export: {\n infile: {\n value: false,\n type: 'string',\n description:\n 'The input file name along with a type (json or svg). It can be a correct JSON or SVG file.'\n },\n instr: {\n value: false,\n type: 'string',\n description:\n 'An input in a form of a stringified JSON or SVG file. Overrides the --infile.'\n },\n options: {\n value: false,\n type: 'string',\n description: 'An alias for the --instr option.'\n },\n outfile: {\n value: false,\n type: 'string',\n description:\n 'The output filename along with a type (jpeg, png, pdf or svg). Ignores the --type flag.'\n },\n type: {\n envLink: 'EXPORT_DEFAULT_TYPE',\n value: 'png',\n type: 'string',\n description:\n 'The format of the file to export to. Can be jpeg, png, pdf or svg.'\n },\n constr: {\n envLink: 'EXPORT_DEFAULT_CONSTR',\n value: 'chart',\n type: 'string',\n description:\n 'The constructor to use. Can be chart, stockChart, mapChart or ganttChart.'\n },\n defaultHeight: {\n envLink: 'EXPORT_DEFAULT_HEIGHT',\n value: 400,\n type: 'number',\n description:\n 'The default height of the exported chart. Used when not found any value set.'\n },\n defaultWidth: {\n envLink: 'EXPORT_DEFAULT_WIDTH',\n value: 600,\n type: 'number',\n description:\n 'The default width of the exported chart. Used when not found any value set.'\n },\n defaultScale: {\n envLink: 'EXPORT_DEFAULT_SCALE',\n value: 1,\n type: 'number',\n description:\n 'The default scale of the exported chart. Ranges between 1 and 5.'\n },\n height: {\n type: 'number',\n value: false,\n description:\n 'The default height of the exported chart. Overrides the option in the chart settings.'\n },\n width: {\n type: 'number',\n value: false,\n description:\n 'The width of the exported chart. Overrides the option in the chart settings.'\n },\n scale: {\n value: false,\n type: 'number',\n description: 'The scale of the exported chart. Ranges between 1 and 5.'\n },\n globalOptions: {\n value: false,\n type: 'string',\n description:\n 'A stringified JSON or a filename with options to be passed into the Highcharts.setOptions.'\n },\n themeOptions: {\n value: false,\n type: 'string',\n description:\n 'A stringified JSON or a filename with theme options to be passed into the Highcharts.setOptions.'\n },\n batch: {\n value: false,\n type: 'string',\n description:\n 'Starts a batch job. A string that contains input/output pairs: \"in=out;in=out;..\".'\n }\n },\n customCode: {\n allowCodeExecution: {\n envLink: 'HIGHCHARTS_ALLOW_CODE_EXECUTION',\n value: false,\n type: 'boolean',\n description:\n 'If set to true, allow for the execution of arbitrary code when exporting.'\n },\n allowFileResources: {\n envLink: 'HIGHCHARTS_ALLOW_FILE_RESOURCES',\n value: true,\n type: 'boolean',\n description:\n 'Allow injecting resources from the filesystem. Has no effect when running as a server.'\n },\n customCode: {\n value: false,\n type: 'string',\n description:\n 'A function to be called before chart initialization. Can be a filename with the js extension.'\n },\n callback: {\n value: false,\n type: 'string',\n description: 'A JavaScript file with a function to run on construction.'\n },\n resources: {\n value: false,\n type: 'string',\n description:\n 'An additional resource in a form of stringified JSON. It can contain files, js and css sections.'\n },\n loadConfig: {\n value: false,\n type: 'string',\n description: 'A file that contains a pre-defined config to use.'\n },\n createConfig: {\n value: false,\n type: 'string',\n description:\n 'Allows to set options through a prompt and save in a provided config file.'\n }\n },\n server: {\n enable: {\n envLink: 'HIGHCHARTS_SERVER_ENABLE',\n value: false,\n type: 'boolean',\n cliName: 'enableServer',\n description: 'If set to true, starts a server on 0.0.0.0.'\n },\n host: {\n envLink: 'HIGHCHARTS_SERVER_HOST',\n value: '0.0.0.0',\n type: 'string',\n description:\n 'The hostname of the server. Also starts a server listening on the supplied hostname.'\n },\n port: {\n envLink: 'HIGHCHARTS_SERVER_PORT',\n value: 7801,\n type: 'number',\n description: 'The port to use for the server. Defaults to 7801.'\n },\n ssl: {\n enable: {\n envLink: 'HIGHCHARTS_SERVER_SSL_ENABLE',\n value: false,\n type: 'boolean',\n cliName: 'enableSsl',\n description: 'Enables the SSL protocol.'\n },\n force: {\n envLink: 'HIGHCHARTS_SERVER_SSL_FORCE',\n value: false,\n type: 'boolean',\n cliName: 'sslForced',\n description:\n 'If set to true, forces the server to only serve over HTTPS.'\n },\n port: {\n envLink: 'HIGHCHARTS_SERVER_SSL_PORT',\n value: 443,\n type: 'number',\n cliName: 'sslPort',\n description: 'The port on which to run the SSL server.'\n },\n certPath: {\n envLink: 'HIGHCHARTS_SSL_CERT_PATH',\n value: '',\n type: 'string',\n description: 'The path to the SSL certificate/key.'\n }\n },\n rateLimiting: {\n enable: {\n envLink: 'HIGHCHARTS_RATE_LIMIT_ENABLE',\n value: false,\n type: 'boolean',\n cliName: 'enableRateLimiting',\n description: 'Enables rate limiting.'\n },\n maxRequests: {\n envLink: 'HIGHCHARTS_RATE_LIMIT_MAX',\n value: 10,\n type: 'number',\n description: 'Max requests allowed in a one minute.'\n },\n window: {\n envLink: 'HIGHCHARTS_RATE_LIMIT_WINDOW',\n value: 1,\n type: 'number',\n description: 'The time window in minutes for rate limiting.'\n },\n delay: {\n envLink: 'HIGHCHARTS_RATE_LIMIT_DELAY',\n value: 0,\n type: 'number',\n description:\n 'The amount to delay each successive request before hitting the max.'\n },\n trustProxy: {\n envLink: 'HIGHCHARTS_RATE_LIMIT_TRUST_PROXY',\n value: false,\n type: 'boolean',\n description: 'Set this to true if behind a load balancer.'\n },\n skipKey: {\n envLink: 'HIGHCHARTS_RATE_LIMIT_SKIP_KEY',\n value: '',\n type: 'number|string',\n description:\n 'Allows bypassing the rate limiter and should be provided with skipToken argument.'\n },\n skipToken: {\n envLink: 'HIGHCHARTS_RATE_LIMIT_SKIP_TOKEN',\n value: '',\n type: 'number|string',\n description:\n 'Allows bypassing the rate limiter and should be provided with skipKey argument.'\n }\n }\n },\n pool: {\n initialWorkers: {\n envLink: 'HIGHCHARTS_POOL_MIN_WORKERS',\n value: 4,\n type: 'number',\n description: 'The number of initial workers to spawn.'\n },\n maxWorkers: {\n envLink: 'HIGHCHARTS_POOL_MAX_WORKERS',\n value: 8,\n type: 'number',\n description: 'The number of max workers to spawn.'\n },\n workLimit: {\n envLink: 'HIGHCHARTS_POOL_WORK_LIMIT',\n value: 40,\n type: 'number',\n description:\n 'The pieces of work that can be performed before restarting process.'\n },\n queueSize: {\n envLink: 'HIGHCHARTS_POOL_QUEUE_SIZE',\n value: 5,\n type: 'number',\n description: 'The size of the request overflow queue.'\n },\n timeoutThreshold: {\n envLink: 'HIGHCHARTS_POOL_TIMEOUT',\n value: 5000,\n type: 'number',\n description: 'The number of milliseconds before timing out.'\n },\n acquireTimeout: {\n envLink: 'HIGHCHARTS_POOL_ACQUIRE_TIMEOUT',\n value: 5000,\n type: 'number',\n description:\n 'The number of milliseconds to wait for acquiring a resource.'\n },\n reaper: {\n envLink: 'HIGHCHARTS_POOL_ENABLE_REAPER',\n value: true,\n type: 'boolean',\n description:\n 'Whether or not to evict workers after a certain time period.'\n },\n benchmarking: {\n envLink: 'HIGHCHARTS_POOL_BENCHMARKING',\n value: false,\n type: 'boolean',\n description: 'Enable benchmarking.'\n },\n listenToProcessExits: {\n envLink: 'HIGHCHARTS_POOL_LISTEN_TO_PROCESS_EXITS',\n value: true,\n type: 'boolean',\n description:\n 'Set to false in order to skip attaching process.exit handlers.'\n }\n },\n logging: {\n level: {\n envLink: 'HIGHCHARTS_LOG_LEVEL',\n value: 4,\n type: 'number',\n cliName: 'logLevel',\n description:\n 'The log level (0: silent, 1: error, 2: warning, 3: notice, 4: verbose).'\n },\n file: {\n envLink: 'HIGHCHARTS_LOG_FILE',\n value: 'highcharts-export-server.log',\n type: 'string',\n cliName: 'logFile',\n description:\n 'A name of a log file. The --logDest also needs to be set to enable file logging.'\n },\n dest: {\n envLink: 'HIGHCHARTS_LOG_DEST',\n value: 'log/',\n type: 'string',\n cliName: 'logDest',\n description: 'The path to store log files. Also enables file logging.'\n }\n },\n ui: {\n enable: {\n envLink: 'HIGHCHARTS_UI_ENABLE',\n value: false,\n type: 'boolean',\n cliName: 'enableUi',\n description: 'Enables the UI for the export server.'\n },\n route: {\n envLink: 'HIGHCHARTS_UI_ROUTE',\n value: '/',\n type: 'string',\n cliName: 'uiRoute',\n description: 'The route to attach the UI to.'\n }\n },\n other: {\n noLogo: {\n envLink: 'HIGHCHARTS_NO_LOGO',\n value: false,\n type: 'boolean',\n description:\n 'Skip printing the logo on a startup. Will be replaced by a simple text.'\n }\n },\n payload: {}\n};\n\n// The config descriptions object for the prompts functionality. It contains\n// information like:\n// * Type of a prompt\n// * Name of an option\n// * Short description of a chosen option\n// * Initial value\nexport const promptsConfig = {\n puppeteer: [\n {\n type: 'list',\n name: 'args',\n message: 'Puppeteer arguments',\n initial: defaultConfig.puppeteer.args.value.join(','),\n separator: ','\n }\n ],\n highcharts: [\n {\n type: 'text',\n name: 'version',\n message: 'Highcharts version',\n initial: defaultConfig.highcharts.version.value\n },\n {\n type: 'text',\n name: 'cdnURL',\n message: 'The url of CDN',\n initial: defaultConfig.highcharts.cdnURL.value\n },\n {\n type: 'multiselect',\n name: 'modules',\n message: 'Available modules',\n instructions: 'Space: Select specific, A: Select all, Enter: Confirm.',\n choices: defaultConfig.highcharts.modules.value\n },\n {\n type: 'list',\n name: 'scripts',\n message: 'Custom scripts',\n initial: defaultConfig.highcharts.scripts.value.join(','),\n separator: ','\n },\n {\n type: 'toggle',\n name: 'forceFetch',\n message: 'Should refetch all the scripts after each server rerun',\n initial: defaultConfig.highcharts.forceFetch.value\n }\n ],\n export: [\n {\n type: 'select',\n name: 'type',\n message: 'The default type of a file to export to',\n hint: `Default: ${defaultConfig.export.type.value}`,\n initial: 0,\n choices: ['png', 'jpeg', 'pdf', 'svg']\n },\n {\n type: 'select',\n name: 'constr',\n message: 'The default constructor for Highcharts to use',\n hint: `Default: ${defaultConfig.export.constr.value}`,\n initial: 0,\n choices: ['chart', 'stockChart', 'mapChart', 'ganttChart']\n },\n {\n type: 'number',\n name: 'defaultHeight',\n message: 'The default fallback height of the exported chart',\n initial: defaultConfig.export.defaultHeight.value\n },\n {\n type: 'number',\n name: 'defaultWidth',\n message: 'The default fallback width of the exported chart',\n initial: defaultConfig.export.defaultWidth.value\n },\n {\n type: 'number',\n name: 'defaultScale',\n message: 'The default fallback scale of the exported chart',\n initial: defaultConfig.export.defaultScale.value,\n min: 0.1,\n max: 5\n }\n ],\n customCode: [\n {\n type: 'toggle',\n name: 'allowCodeExecution',\n message: 'Allow to execute custom code',\n initial: defaultConfig.customCode.allowCodeExecution.value\n },\n {\n type: 'toggle',\n name: 'allowFileResources',\n message: 'Allow file resources',\n initial: defaultConfig.customCode.allowFileResources.value\n }\n ],\n server: [\n {\n type: 'toggle',\n name: 'enable',\n message: 'Starts a server on 0.0.0.0',\n initial: defaultConfig.server.enable.value\n },\n {\n type: 'text',\n name: 'host',\n message: 'A hostname of a server',\n initial: defaultConfig.server.host.value\n },\n {\n type: 'number',\n name: 'port',\n message: 'A port of a server',\n initial: defaultConfig.server.port.value\n },\n {\n type: 'toggle',\n name: 'ssl.enable',\n message: 'Enable SSL protocol',\n initial: defaultConfig.server.ssl.enable.value\n },\n {\n type: 'toggle',\n name: 'ssl.force',\n message: 'Force to only serve over HTTPS',\n initial: defaultConfig.server.ssl.force.value\n },\n {\n type: 'number',\n name: 'ssl.port',\n message: 'Port on which to run the SSL server',\n initial: defaultConfig.server.ssl.port.value\n },\n {\n type: 'text',\n name: 'ssl.certPath',\n message: 'A path where to find the SSL certificate/key',\n initial: defaultConfig.server.ssl.certPath.value\n },\n {\n type: 'toggle',\n name: 'rateLimiting.enable',\n message: 'Enable rate limiting',\n initial: defaultConfig.server.rateLimiting.enable.value\n },\n {\n type: 'number',\n name: 'rateLimiting.maxRequests',\n message: 'Max requests allowed in a one minute',\n initial: defaultConfig.server.rateLimiting.maxRequests.value\n },\n {\n type: 'number',\n name: 'rateLimiting.window',\n message: 'The time window in minutes for rate limiting',\n initial: defaultConfig.server.rateLimiting.window.value\n },\n {\n type: 'number',\n name: 'rateLimiting.delay',\n message:\n 'The amount to delay each successive request before hitting the max',\n initial: defaultConfig.server.rateLimiting.delay.value\n },\n {\n type: 'toggle',\n name: 'rateLimiting.trustProxy',\n message: 'Set this to true if behind a load balancer',\n initial: defaultConfig.server.rateLimiting.trustProxy.value\n },\n {\n type: 'text',\n name: 'rateLimiting.skipKey',\n message:\n 'Allows bypassing the rate limiter and should be provided with skipToken argument',\n initial: defaultConfig.server.rateLimiting.skipKey.value\n },\n {\n type: 'text',\n name: 'rateLimiting.skipToken',\n message:\n 'Allows bypassing the rate limiter and should be provided with skipKey argument',\n initial: defaultConfig.server.rateLimiting.skipToken.value\n }\n ],\n pool: [\n {\n type: 'number',\n name: 'initialWorkers',\n message: 'The number of initial workers to spawn',\n initial: defaultConfig.pool.initialWorkers.value\n },\n {\n type: 'number',\n name: 'maxWorkers',\n message: 'The number of max workers to spawn',\n initial: defaultConfig.pool.maxWorkers.value\n },\n {\n type: 'number',\n name: 'workLimit',\n message:\n 'The pieces of work that can be performed before restarting a puppeteer process',\n initial: defaultConfig.pool.workLimit.value\n },\n {\n type: 'number',\n name: 'queueSize',\n message: 'The size of the request overflow queue',\n initial: defaultConfig.pool.queueSize.value\n },\n {\n type: 'number',\n name: 'timeoutThreshold',\n message: 'The number of seconds before timing out',\n initial: defaultConfig.pool.timeoutThreshold.value\n },\n {\n type: 'number',\n name: 'acquireTimeout',\n message: 'The number of milliseconds to wait for acquiring a resource',\n initial: defaultConfig.pool.acquireTimeout.value\n },\n {\n type: 'toggle',\n name: 'reaper',\n message: 'The reaper to remove hanging processes',\n initial: defaultConfig.pool.reaper.value\n },\n {\n type: 'toggle',\n name: 'benchmarking',\n message: 'Set benchmarking',\n initial: defaultConfig.pool.benchmarking.value\n },\n {\n type: 'toggle',\n name: 'listenToProcessExits',\n message: 'Set to false in order to skip attaching process.exit handlers',\n initial: defaultConfig.pool.listenToProcessExits.value\n }\n ],\n logging: [\n {\n type: 'number',\n name: 'level',\n message:\n 'The log level (0: silent, 1: error, 2: warning, 3: notice, 4: verbose)',\n initial: defaultConfig.logging.level.value,\n round: 0,\n min: 0,\n max: 4\n },\n {\n type: 'text',\n name: 'file',\n message:\n 'A name of a log file. The --logDest also needs to be set to enable file logging',\n initial: defaultConfig.logging.file.value\n },\n {\n type: 'text',\n name: 'dest',\n message: 'A path to log files. It enables file logging',\n initial: defaultConfig.logging.dest.value\n }\n ],\n ui: [\n {\n type: 'toggle',\n name: 'enable',\n message: 'Enable UI for the export server',\n initial: defaultConfig.ui.enable.value\n },\n {\n type: 'text',\n name: 'route',\n message: 'A route to attach the UI to',\n initial: defaultConfig.ui.route.value\n }\n ],\n other: [\n {\n type: 'toggle',\n name: 'noLogo',\n message:\n 'Skip printing the logo on a startup. Will be replaced by a simple text',\n initial: defaultConfig.other.noLogo.value\n }\n ]\n};\n\n// Absolute props that, in case of merging recursively, need to be force merged\nexport const absoluteProps = [\n 'options',\n 'globalOptions',\n 'themeOptions',\n 'resources',\n 'payload'\n];\n\n// Argument nesting level of all export server options\nexport const nestedArgs = {};\n\n/**\n * Creates nested arguments chain for all options\n *\n * @param {object} obj - The object based on which the initial configuration be\n * made.\n * @param {string } propChain - Required for creating a string chain of\n * properties for nested arguments.\n */\nconst createNestedArgs = (obj, propChain = '') => {\n Object.keys(obj).forEach((k) => {\n if (!['puppeteer', 'highcharts'].includes(k)) {\n const entry = obj[k];\n if (typeof entry.value === 'undefined') {\n // Go deeper in the nested arguments\n createNestedArgs(entry, `${propChain}.${k}`);\n } else {\n // Create the chain of nested arguments\n nestedArgs[entry.cliName || k] = `${propChain}.${k}`.substring(1);\n }\n }\n });\n};\n\ncreateNestedArgs(defaultConfig);\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { appendFile, existsSync, mkdirSync } from 'fs';\n\nimport { defaultConfig } from './schemas/config.js';\n\n// The default logging config\nlet logging = {\n // Flags for logging status\n toConsole: true,\n toFile: false,\n pathCreated: false,\n // Log levels\n levelsDesc: [\n {\n title: 'error',\n color: 'red'\n },\n {\n title: 'warning',\n color: 'yellow'\n },\n {\n title: 'notice',\n color: 'blue'\n },\n {\n title: 'verbose',\n color: 'gray'\n }\n ],\n // Log listeners\n listeners: []\n};\n\n// Gather init logging options\nfor (const [key, option] of Object.entries(defaultConfig.logging)) {\n logging[key] = option.value;\n}\n\n/**\n * Logs a message. Accepts a variable amount of arguments. Arguments after\n * `level` will be passed directly to console.log, and/or will be joined\n * and appended to the log file.\n *\n * @param {any} args - An array of arguments where the first is the log level\n * and the rest are strings to build a message with.\n */\nexport const log = (...args) => {\n const [newLevel, ...texts] = args;\n\n // Current logging options\n const { level, levelsDesc } = logging;\n\n // Check if log level is within a correct range\n if (newLevel === 0 || newLevel > level || level > levelsDesc.length) {\n return;\n }\n\n // Get rid of the GMT text information\n const newDate = new Date().toString().split('(')[0].trim();\n\n // Create a message's prefix\n const prefix = `${newDate} [${levelsDesc[newLevel - 1].title}] -`;\n\n // Call available log listeners\n logging.listeners.forEach((fn) => {\n fn(prefix, texts.join(' '));\n });\n\n // Log to file\n if (logging.toFile) {\n if (!logging.pathCreated) {\n // Create if does not exist\n !existsSync(logging.dest) && mkdirSync(logging.dest);\n\n // We now assume the path is available, e.g. it's the responsibility\n // of the user to create the path with the correct access rights.\n logging.pathCreated = true;\n }\n\n // Add the content to a file\n appendFile(\n `${logging.dest}${logging.file}`,\n [prefix].concat(texts).join(' ') + '\\n',\n (error) => {\n if (error) {\n console.log(`[logger] Unable to write to log file: ${error}`);\n logging.toFile = false;\n }\n }\n );\n }\n\n // Log to console\n if (logging.toConsole) {\n console.log.apply(\n undefined,\n [prefix.toString()[logging.levelsDesc[newLevel - 1].color]].concat(texts)\n );\n }\n};\n\n/**\n * Sets the file logging configuration.\n *\n * @param {string} logDest - A path to log to.\n * @param {string} logFile - The name of the log file.\n */\nexport const enableFileLogging = (logDest, logFile) => {\n // Update logging options\n logging = {\n ...logging,\n dest: logDest || logging.dest,\n file: logFile || logging.file,\n toFile: true\n };\n\n if (logging.dest.length === 0) {\n return log(1, '[logger] File logging init: no path supplied.');\n }\n\n if (!logging.dest.endsWith('/')) {\n logging.dest += '/';\n }\n};\n\n/**\n * Adds a log listener.\n *\n * @param {function} fn - The function to call when getting a log event.\n */\nexport const listen = (fn) => {\n logging.listeners.push(fn);\n};\n\n/**\n * Sets the current log level. Log levels are:\n * - 0 = no logging\n * - 1 = error\n * - 2 = warning\n * - 3 = notice\n * - 4 = verbose\n *\n * @param {number} newLevel - The new log level (0 - 4).\n */\nexport const setLogLevel = (newLevel) => {\n if (newLevel >= 0 && newLevel <= logging.levelsDesc.length) {\n logging.level = newLevel;\n }\n};\n\n/**\n * Enables or disables logging to the stdout.\n *\n * @param {boolean} enabled - Whether log to console or not.\n */\nexport const toggleSTDOut = (enabled) => {\n logging.toConsole = enabled;\n};\n\nexport default {\n log,\n enableFileLogging,\n listen,\n setLogLevel,\n toggleSTDOut\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { readFileSync } from 'fs';\nimport { fileURLToPath } from 'url';\n\nimport { defaultConfig } from '../lib/schemas/config.js';\nimport { log } from './logger.js';\n\nconst MAX_BACKOFF_ATTEMPTS = 6;\n\nexport const __dirname = fileURLToPath(new URL('../.', import.meta.url));\n\n/**\n * Clears text from whitespaces with a regex rule.\n *\n * @param {string} rule - The rule for clearing a string, default to /\\s\\s+/g.\n * @return {string} - Cleared text.\n */\nexport const clearText = (text, rule = /\\s\\s+/g, replacer = ' ') =>\n text.replaceAll(rule, replacer).trim();\n\n/**\n * Delays calling the function by time calculated based on the backoff\n * algorithm.\n *\n * @param {function} fn - A function to try to call with the backoff algorithm\n * on.\n * @param {number} attempt - The number of an attempt, where the first one is 0.\n */\nexport const expBackoff = async (fn, attempt = 0, ...args) => {\n try {\n // Try to call the function\n return await fn(...args);\n } catch (error) {\n // Calculate delay in ms\n const delayInMs = 2 ** attempt * 1000;\n\n // If the attempt exceeds the maximum attempts of reapeat, throw an error\n if (++attempt >= MAX_BACKOFF_ATTEMPTS) {\n throw error;\n }\n\n // Wait given amount of time\n await new Promise((response) => setTimeout(response, delayInMs));\n log(\n 3,\n `[pool] Waited ${delayInMs}ms until next call for the resource id: ${args[0]}.`\n );\n\n // Try again\n return expBackoff(fn, attempt, ...args);\n }\n};\n\n/**\n * Fixes to supported type format if MIME.\n *\n * @param {string} type - Type to be corrected.\n * @param {string} outfile - Name of the outfile.\n */\nexport const fixType = (type, outfile) => {\n // MIME types\n const mimeTypes = {\n 'image/png': 'png',\n 'image/jpeg': 'jpeg',\n 'application/pdf': 'pdf',\n 'image/svg+xml': 'svg'\n };\n\n // Formats\n const formats = ['png', 'jpeg', 'pdf', 'svg'];\n\n // Check if type and outfile's extensions are the same\n if (outfile) {\n const outType = outfile.split('.').pop();\n\n // Check if extension has a correct type\n if (formats.includes(outType) && type !== outType) {\n type = outType;\n }\n }\n\n // Return a correct type\n return mimeTypes[type] || formats.find((t) => t === type) || 'png';\n};\n\n/**\n * Handles the provided resources.\n *\n * @param {string} resources - The stringified resources.\n * @param {string} allowFileResources - Decide if resources from file are\n * allowed.\n */\nexport const handleResources = (resources = false, allowFileResources) => {\n const allowedProps = ['js', 'css', 'files'];\n\n let handledResources = resources;\n let correctResources = false;\n\n // Try to load resources from a file\n if (allowFileResources && resources.endsWith('.json')) {\n try {\n if (!resources) {\n handledResources = isCorrectJSON(\n readFileSync('resources.json', 'utf8')\n );\n } else if (resources && resources.endsWith('.json')) {\n handledResources = isCorrectJSON(readFileSync(resources, 'utf8'));\n } else {\n handledResources = isCorrectJSON(resources);\n if (handledResources === true) {\n handledResources = isCorrectJSON(\n readFileSync('resources.json', 'utf8')\n );\n }\n }\n } catch (notice) {\n return log(3, `[cli] No resources found.`);\n }\n } else {\n // Try to get JSON\n handledResources = isCorrectJSON(resources);\n\n // Get rid of the files section\n if (!allowFileResources) {\n delete handledResources.files;\n }\n }\n\n // Filter from unnecessary properties\n for (const propName in handledResources) {\n if (!allowedProps.includes(propName)) {\n delete handledResources[propName];\n } else if (!correctResources) {\n correctResources = true;\n }\n }\n\n // Check if at least one of allowed properties is present\n if (!correctResources) {\n return log(3, `[cli] No resources found.`);\n }\n\n // Handle files section\n if (handledResources.files) {\n handledResources.files = handledResources.files.map((item) => item.trim());\n if (!handledResources.files || handledResources.files.length <= 0) {\n delete handledResources.files;\n }\n }\n\n // Return resources\n return handledResources;\n};\n\n/**\n * Checks if provided data is or can be a correct JSON.\n *\n * @param {any} data - Data to be checked.\n * @param {boolean} toString - If true, return stringified representation.\n */\nexport function isCorrectJSON(data, toString) {\n try {\n // Get the string representation if not already before parsing\n const parsedData = JSON.parse(\n typeof data !== 'string' ? JSON.stringify(data) : data\n );\n\n // Return a stringified representation of a JSON if required\n if (typeof parsedData !== 'string' && toString) {\n return JSON.stringify(parsedData);\n }\n\n // Return a JSON\n return parsedData;\n } catch (error) {\n return false;\n }\n}\n\n/**\n * Checks if item is an object.\n *\n * @param {any} item - Item to be checked.\n */\nexport const isObject = (item) =>\n typeof item === 'object' && !Array.isArray(item) && item !== null;\n\n/**\n * Checks if string contains private range urls.\n *\n * @export utils\n * @param item {string} item to be checked\n */\nexport const isPrivateRangeUrlFound = (item) => {\n return [\n 'localhost',\n '(10).(.*).(.*).(.*)',\n '(127).(.*).(.*).(.*)',\n '(172).(1[6-9]|2[0-9]|3[0-1]).(.*).(.*)',\n '(192).(168).(.*).(.*)'\n ].some((ipRegEx) =>\n item.match(`xlink:href=\"(?:(http://|https://))?${ipRegEx}`)\n );\n};\n\n/**\n * Creates and returns a deep copy of the given object.\n *\n * @param {object} object - Object to copy.\n * @return {object} - Deep copy of the object.\n */\nexport const deepCopy = (obj) => {\n if (obj === null || typeof obj !== 'object') {\n return obj;\n }\n\n const copy = Array.isArray(obj) ? [] : {};\n\n for (const key in obj) {\n if (Object.prototype.hasOwnProperty.call(obj, key)) {\n copy[key] = deepCopy(obj[key]);\n }\n }\n\n return copy;\n};\n\n/**\n * Stringifies object with options. Possible to preserve functions with\n * allowFunctions flag.\n *\n * @param {object} options - Options to stringify.\n * @param {boolean} allowFunctions - Flag for keeping functions.\n */\nexport const optionsStringify = (options, allowFunctions) => {\n const replacerCallback = (name, value) => {\n if (typeof value === 'string') {\n value = value.trim();\n\n // If allowFunctions is set to true, preserve functions\n if (\n (value.startsWith('function(') || value.startsWith('function (')) &&\n value.endsWith('}')\n ) {\n value = allowFunctions\n ? `EXP_FUN${(value + '').replaceAll(/\\n|\\t|\\r/g, ' ')}EXP_FUN`\n : undefined;\n }\n }\n\n return typeof value === 'function'\n ? `EXP_FUN${(value + '').replaceAll(/\\n|\\t|\\r/g, ' ')}EXP_FUN`\n : value;\n };\n\n // Stringify options and if required, replace special functions marks\n return JSON.stringify(options, replacerCallback).replaceAll(\n /\"EXP_FUN|EXP_FUN\"/g,\n ''\n );\n};\n\n/**\n * Prints the export server logo.\n *\n * @param {boolean} noLogo - Whether to display logo or text.\n */\nexport const printLogo = (noLogo) => {\n // Get package version either from env or from package.json\n const packageVersion =\n process.env.npm_package_version ||\n JSON.parse(readFileSync(new URL('../package.json', import.meta.url)))\n .version;\n\n // Print text only\n if (noLogo) {\n console.log(`Starting highcharts export server v${packageVersion}...`);\n return;\n }\n\n // Print the logo\n console.log(\n readFileSync(__dirname + '/msg/startup.msg').toString().bold.yellow,\n `v${packageVersion}`\n );\n};\n\n/**\n * Prints the CLI usage. If required, it can list properties recursively\n */\nexport function printUsage() {\n const pad = 48;\n const readme = 'https://github.com/highcharts/node-export-server#readme';\n\n // Display readme information\n console.log(\n 'Usage of CLI arguments:'.bold,\n '\\n------',\n `\\nFor more detailed information visit readme at: ${readme.bold.yellow}.`\n );\n\n const cycleCategories = (categories) => {\n for (const [name, option] of Object.entries(categories)) {\n // If category has more levels, go further\n if (!Object.prototype.hasOwnProperty.call(option, 'value')) {\n cycleCategories(option);\n } else {\n let descName = ` --${option.cliName || name} ${\n ('<' + option.type + '>').green\n } `;\n if (descName.length < pad) {\n for (let i = descName.length; i < pad; i++) {\n descName += '.';\n }\n }\n\n // Display correctly aligned messages\n console.log(\n descName,\n option.description,\n `[Default: ${option.value.toString().bold}]`.blue\n );\n }\n }\n };\n\n // Cycle through options of each categories and display the usage info\n Object.keys(defaultConfig).forEach((category) => {\n // Only puppeteer and highcharts categories cannot be configured through CLI\n if (!['puppeteer', 'highcharts'].includes(category)) {\n console.log(`\\n${category.toUpperCase()}`.red);\n cycleCategories(defaultConfig[category]);\n }\n });\n console.log('\\n');\n}\n\n/**\n * Rounds number to passed precision.\n *\n * @param {number} value - Number to round.\n * @param {number} precision - A precision of rounding.\n */\nexport const roundNumber = (value, precision = 1) => {\n const multiplier = Math.pow(10, precision || 0);\n return Math.round(+value * multiplier) / multiplier;\n};\n\n/**\n * Casts the item to boolean.\n *\n * @param {any} item - Item to be cast.\n */\nexport const toBoolean = (item) =>\n ['false', 'undefined', 'null', 'NaN', '0', ''].includes(item)\n ? false\n : !!item;\n\n/**\n * If necessary, places a custom code inside a function.\n *\n * @param {any} customCode - The customCode.\n */\nexport const wrapAround = (customCode, allowFileResources) => {\n if (customCode && typeof customCode === 'string') {\n customCode = customCode.trim();\n\n if (customCode.endsWith('.js')) {\n return allowFileResources\n ? wrapAround(readFileSync(customCode, 'utf8'))\n : false;\n } else if (\n customCode.startsWith('function()') ||\n customCode.startsWith('function ()') ||\n customCode.startsWith('()=>') ||\n customCode.startsWith('() =>')\n ) {\n return `(${customCode})()`;\n }\n return customCode.replace(/;$/, '');\n }\n};\n\n/**\n * Utility to measure time.\n */\nexport const measureTime = () => {\n const start = process.hrtime.bigint();\n return () => Number(process.hrtime.bigint() - start) / 1000000;\n};\n\nexport default {\n __dirname,\n clearText,\n expBackoff,\n fixType,\n handleResources,\n isCorrectJSON,\n isObject,\n isPrivateRangeUrlFound,\n optionsStringify,\n printLogo,\n printUsage,\n roundNumber,\n toBoolean,\n wrapAround,\n measureTime\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport rateLimit from 'express-rate-limit';\n\nimport { clearText } from '../utils.js';\nimport { log } from '../logger.js';\n\n/**\n * Enables rate limiting for a given app.\n *\n * @param {object} app - The express app.\n * @param {object} limitConfig - The options for the rate limiting.\n */\nexport default (app, limitConfig) => {\n const msg =\n 'Too many requests, you have been rate limited. Please try again later.';\n\n // Options for the rate limiter\n const rateOptions = {\n max: limitConfig.maxRequests || 30,\n window: limitConfig.window || 1,\n delay: limitConfig.delay || 0,\n trustProxy: limitConfig.trustProxy || false,\n skipKey: limitConfig.skipKey || false,\n skipToken: limitConfig.skipToken || false\n };\n\n // Set if behind a proxy\n if (rateOptions.trustProxy) {\n app.enable('trust proxy');\n }\n\n // Create a limiter\n const limiter = rateLimit({\n windowMs: rateOptions.window * 60 * 1000,\n // Limit each IP to 100 requests per windowMs\n max: rateOptions.max,\n // Disable delaying, full speed until the max limit is reached\n delayMs: rateOptions.delay,\n handler: (request, response) => {\n response.format({\n json: () => {\n response.status(429).send({ message: msg });\n },\n default: () => {\n response.status(429).send(msg);\n }\n });\n },\n skip: (request) => {\n // Allow bypassing the limiter if a valid key/token has been sent\n if (\n rateOptions.skipKey !== false &&\n rateOptions.skipToken !== false &&\n request.query.key === rateOptions.skipKey &&\n request.query.access_token === rateOptions.skipToken\n ) {\n log(4, '[rate-limiting] Skipping rate limiter.');\n return true;\n }\n return false;\n }\n });\n\n // Use a limiter as a middleware\n app.use(limiter);\n\n log(\n 3,\n clearText(\n `[rate-limiting] Enabled rate limiting: ${rateOptions.max} requests\n per ${rateOptions.window} minute per IP, trusting proxy:\n ${rateOptions.trustProxy}.`\n )\n );\n};\n","/**\n * This module exports two functions: fetch (for GET requests) and post (for POST requests).\n */\n\nimport http from 'http';\nimport https from 'https';\n\n/**\n * Determines the protocol of the given URL (either `http` or `https`).\n *\n * @function\n * @param {string} url - The URL whose protocol needs to be determined.\n * @returns {Object} Returns the `https` module if the URL starts with 'https',\n * otherwise returns the `http` module.\n * @private\n *\n * @example\n *\n * const protocol = getProtocol('https://example.com');\n * console.log(protocol); // Outputs the 'https' module\n */\nconst getProtocol = (url) => {\n return url.startsWith('https') ? https : http;\n};\n\n/**\n * Sends a GET request to the specified URL with optional request options.\n *\n * @function\n * @async\n * @param {string} url - The URL to fetch.\n * @param {Object} [requestOptions={}] - Optional request options and headers.\n * @returns {Promise} Returns a promise that resolves with the response object.\n * The response object contains a `.text` property with the raw response data.\n * @throws {Error} Throws an error if the request fails or if no data is fetched from the URL.\n *\n * @example\n *\n * async function getData() {\n * try {\n * const response = await fetch('https://api.example.com/data');\n * console.log(response.text);\n * } catch (error) {\n * console.error('Error fetching data:', error);\n * }\n * }\n *\n * getData();\n */\nasync function fetch(url, requestOptions = {}) {\n return new Promise((resolve, reject) => {\n const protocol = getProtocol(url);\n\n protocol\n .get(url, requestOptions, (res) => {\n let data = '';\n\n // A chunk of data has been received.\n res.on('data', (chunk) => {\n data += chunk;\n });\n\n // The whole response has been received.\n res.on('end', () => {\n if (!data) {\n reject('Nothing was fetched from the URL.');\n }\n\n res.text = data;\n resolve(res);\n });\n })\n .on('error', (error) => {\n reject(error);\n });\n });\n}\n\n/**\n * Sends a POST request to the specified URL with the given body and request options.\n *\n * @function\n * @async\n * @param {string} url - The URL to which the request should be sent.\n * @param {Object} [body={}] - The data to be sent as the request body, in JSON format.\n * @param {Object} [requestOptions={}] - Optional request options and headers.\n * @returns {Promise} - Returns a promise that resolves with the parsed JSON response.\n * @throws {Error} Throws an error if the request fails or if the response cannot be parsed.\n *\n * @example\n *\n * async function sendData() {\n * const dataToSend = {\n * key1: 'value1',\n * key2: 'value2',\n * };\n * try {\n * const response = await post('https://api.example.com/data', dataToSend);\n * console.log(response);\n * } catch (error) {\n * console.error('Error sending data:', error);\n * }\n * }\n *\n * sendData();\n */\nasync function post(url, body = {}, requestOptions = {}) {\n return new Promise((resolve, reject) => {\n const protocol = getProtocol(url);\n const data = JSON.stringify(body);\n\n // Set default headers and merge with requestOptions\n const options = Object.assign(\n {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n 'Content-Length': data.length\n }\n },\n requestOptions\n );\n\n const req = protocol\n .request(url, options, (res) => {\n let responseData = '';\n\n // A chunk of data has been received.\n res.on('data', (chunk) => {\n responseData += chunk;\n });\n\n // The whole response has been received.\n res.on('end', () => {\n try {\n res.text = responseData;\n resolve(res);\n } catch (error) {\n reject(error);\n }\n });\n })\n .on('error', (error) => {\n reject(error);\n });\n\n // Write the request body and end the request.\n req.write(data);\n req.end();\n });\n}\n\nexport default fetch;\nexport { fetch, post };\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\n// The cache manager manages the Highcharts library and its dependencies.\n// The cache itself is stored in .cache, and is checked by the config system\n// before starting the service\n\nimport { existsSync, mkdirSync, readFileSync, writeFileSync } from 'fs';\nimport { join } from 'path';\n\nimport dotenv from 'dotenv';\nimport HttpsProxyAgent from 'https-proxy-agent';\nimport { fetch } from './fetch.js';\n\nimport { log } from './logger.js';\nimport { __dirname } from '../lib/utils.js';\n\ndotenv.config();\n\nconst cachePath = join(__dirname, '.cache');\n\nconst cache = {\n cdnURL: 'https://code.highcharts.com/',\n activeManifest: {},\n sources: '',\n hcVersion: ''\n};\n\n// TODO: The config should be accesssible globally so we don't have to do this sort of thing..\nlet appliedConfig = false;\n\n/**\n * Extracts the Highcharts version from the cache\n */\nconst extractVersion = () =>\n (cache.hcVersion = cache.sources\n .substr(0, cache.sources.indexOf('*/'))\n .replace('/*', '')\n .replace('*/', '')\n .replace(/\\n/g, '')\n .trim());\n\n/**\n * Saves the Highcharts part of a config to a manifest file in the cache\n *\n * @param {object} config - Highcharts related configuration object.\n * @param {object} fetchedModules - An object that contains mapped names of\n * fetched Highcharts modules to use.\n */\nconst saveConfigToManifest = async (config, fetchedModules) => {\n const newManifest = {\n version: config.version,\n modules: fetchedModules || {}\n };\n\n // Update cache object with the current modules\n cache.activeManifest = newManifest;\n\n log(4, '[cache] writing new manifest');\n\n try {\n writeFileSync(\n join(cachePath, 'manifest.json'),\n JSON.stringify(newManifest),\n 'utf8'\n );\n } catch (error) {\n log(1, `[cache] Error writing cache manifest: ${error}.`);\n }\n};\n\n/**\n * Fetches a single script.\n *\n * @param {string} script - A path to script to get.\n * @param {object} proxyAgent - The proxy agent to use for a request.\n */\nconst fetchScript = async (script, proxyAgent) => {\n try {\n // Get rid of the .js from the custom strings\n if (script.endsWith('.js')) {\n script = script.substring(0, script.length - 3);\n }\n\n log(4, `[cache] Fetching script - ${script}.js`);\n\n // If exists, add proxy agent to request options\n const requestOptions = proxyAgent\n ? {\n agent: proxyAgent,\n timeout: +process.env['PROXY_SERVER_TIMEOUT'] || 5000\n }\n : {};\n\n // Fetch the script\n const response = await fetch(`${script}.js`, requestOptions);\n\n // If OK, return its text representation\n if (response.statusCode === 200) {\n return response.text;\n }\n\n throw `${response.statusCode}`;\n } catch (error) {\n log(1, `[cache] Error fetching script ${script}.js: ${error}.`);\n throw error;\n }\n};\n\n/**\n * Updates the Highcharts cache.\n *\n * @param {object} config - Highcharts related configuration object.\n * @param {string} sourcePath - A path to the file where save updated sources.\n * @return {object} An object that contains mapped names of fetched Highcharts\n * modules to use.\n */\nconst updateCache = async (config, sourcePath) => {\n const { coreScripts, modules, indicators, scripts: customScripts } = config;\n const hcVersion =\n config.version === 'latest' || !config.version ? '' : `${config.version}/`;\n\n log(3, '[cache] Updating cache to Highcharts ', hcVersion);\n\n // Gather all scripts to fetch\n const allScripts = [\n ...coreScripts.map((c) => `${hcVersion}${c}`),\n ...modules.map((m) =>\n m === 'map' ? `maps/${hcVersion}modules/${m}` : `${hcVersion}modules/${m}`\n ),\n ...indicators.map((i) => `stock/${hcVersion}indicators/${i}`)\n ];\n\n // Configure proxy if exists\n let proxyAgent;\n const proxyHost = process.env['PROXY_SERVER_HOST'];\n const proxyPort = process.env['PROXY_SERVER_PORT'];\n\n if (proxyHost && proxyPort) {\n proxyAgent = new HttpsProxyAgent({\n host: proxyHost,\n port: +proxyPort\n });\n }\n\n const fetchedModules = {};\n try {\n cache.sources = // TODO: convert to for loop\n (\n await Promise.all([\n ...allScripts.map(async (script) => {\n const text = await fetchScript(\n `${config.cdnURL || cache.cdnURL}${script}`,\n proxyAgent\n );\n\n // If fetched correctly, set it\n if (typeof text === 'string') {\n fetchedModules[\n script.replace(\n /(.*)\\/|(.*)modules\\/|stock\\/(.*)indicators\\/|maps\\/(.*)modules\\//gi,\n ''\n )\n ] = 1;\n }\n\n return text;\n }),\n ...customScripts.map((script) => fetchScript(script, proxyAgent))\n ])\n ).join(';\\n');\n extractVersion();\n\n // Save the fetched modules into caches' source JSON\n writeFileSync(sourcePath, cache.sources);\n return fetchedModules;\n } catch (error) {\n log(1, '[cache] Unable to update local Highcharts cache.');\n }\n};\n\nexport const updateVersion = async (newVersion) =>\n appliedConfig\n ? await checkCache(\n Object.assign(appliedConfig, {\n version: newVersion\n })\n )\n : false;\n\n/**\n * Fetches any missing Highcharts and dependencies\n *\n * @param {object} config - Highcharts related configuration object.\n */\nexport const checkCache = async (config) => {\n let fetchedModules;\n // Prepare paths to manifest and sources from the .cache folder\n const manifestPath = join(cachePath, 'manifest.json');\n const sourcePath = join(cachePath, 'sources.js');\n\n // TODO: deal with trying to switch to the running version\n // const activeVersion = appliedConfig ? appliedConfig.version : false;\n\n appliedConfig = config;\n\n // Create the .cache destination if it doesn't exist already\n !existsSync(cachePath) && mkdirSync(cachePath);\n\n // Fetch all the scripts either if manifest.json does not exist\n // or if the forceFetch option is enabled\n if (!existsSync(manifestPath) || config.forceFetch) {\n log(3, '[cache] Fetching and caching Highcharts dependencies.');\n fetchedModules = await updateCache(config, sourcePath);\n } else {\n let requestUpdate = false;\n\n // Read the manifest JSON\n const manifest = JSON.parse(readFileSync(manifestPath));\n\n // Check if the modules is an array, if so, we rewrite it to a map to make\n // it easier to resolve modules.\n if (manifest.modules && Array.isArray(manifest.modules)) {\n const moduleMap = {};\n manifest.modules.forEach((m) => (moduleMap[m] = 1));\n manifest.modules = moduleMap;\n }\n\n const { modules, coreScripts, indicators } = config;\n const numberOfModules =\n modules.length + coreScripts.length + indicators.length;\n\n // Compare the loaded config with the contents in .cache.\n // If there are changes, fetch requested modules and products,\n // and bake them into a giant blob. Save the blob.\n if (manifest.version !== config.version) {\n log(3, '[cache] Highcharts version mismatch in cache, need to re-fetch.');\n requestUpdate = true;\n } else if (Object.keys(manifest.modules || {}).length !== numberOfModules) {\n log(\n 3,\n '[cache] Cache and requested modules does not match, need to re-fetch.'\n );\n requestUpdate = true;\n } else {\n // Check each module, if anything is missing refetch everything\n requestUpdate = (config.modules || []).some((moduleName) => {\n if (!manifest.modules[moduleName]) {\n log(\n 3,\n `[cache] The ${moduleName} missing in cache, need to re-fetch.`\n );\n return true;\n }\n });\n }\n\n if (requestUpdate) {\n fetchedModules = await updateCache(config, sourcePath);\n } else {\n log(3, '[cache] Dependency cache is up to date, proceeding.');\n\n // Load the sources\n cache.sources = readFileSync(sourcePath, 'utf8');\n\n // Get current modules map\n fetchedModules = manifest.modules;\n extractVersion();\n }\n }\n\n // Finally, save the new manifest, which is basically our current config\n // in a slightly different format\n await saveConfigToManifest(config, fetchedModules);\n};\n\nexport default {\n checkCache,\n updateVersion,\n getCache: () => cache,\n highcharts: () => cache.sources,\n version: () => cache.hcVersion\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport puppeteer from 'puppeteer';\nimport fs from 'fs';\nimport * as url from 'url';\nimport { log } from './logger.js';\nimport path from 'node:path';\n\n// Workaround for https://bugs.chromium.org/p/chromium/issues/detail?id=1463328\n// Not ideal - leaves trash in the FS\nimport { randomBytes } from 'node:crypto';\nconst RANDOM_PID = randomBytes(64).toString('base64url');\nconst PUPPETEER_DIR = path.join('tmp', `puppeteer-${RANDOM_PID}`);\nconst DATA_DIR = path.join(PUPPETEER_DIR, 'profile');\n\n// The minimal args to speed up the browser\nconst minimalArgs = [\n `--user-data-dir=${DATA_DIR}`,\n '--autoplay-policy=user-gesture-required',\n '--disable-background-networking',\n '--disable-background-timer-throttling',\n '--disable-backgrounding-occluded-windows',\n '--disable-breakpad',\n '--disable-client-side-phishing-detection',\n '--disable-component-update',\n '--disable-default-apps',\n '--disable-dev-shm-usage',\n '--disable-domain-reliability',\n '--disable-extensions',\n '--disable-features=AudioServiceOutOfProcess',\n '--disable-hang-monitor',\n '--disable-ipc-flooding-protection',\n '--disable-notifications',\n '--disable-offer-store-unmasked-wallet-cards',\n '--disable-popup-blocking',\n '--disable-print-preview',\n '--disable-prompt-on-repost',\n '--disable-renderer-backgrounding',\n '--disable-session-crashed-bubble',\n '--disable-setuid-sandbox',\n '--disable-speech-api',\n '--disable-sync',\n '--hide-crash-restore-bubble',\n '--hide-scrollbars',\n '--ignore-gpu-blacklist',\n '--metrics-recording-only',\n '--mute-audio',\n '--no-default-browser-check',\n '--no-first-run',\n '--no-pings',\n '--no-sandbox',\n '--no-zygote',\n '--password-store=basic',\n '--use-mock-keychain'\n];\n\nconst __dirname = url.fileURLToPath(new URL('.', import.meta.url));\n\nconst template = fs.readFileSync(\n __dirname + '/../templates/template.html',\n 'utf8'\n);\n\nlet browser;\n\nexport const newPage = async () => {\n if (!browser) return false;\n\n const p = await browser.newPage();\n\n await p.setContent(template);\n await p.addScriptTag({ path: __dirname + '/../.cache/sources.js' });\n // eslint-disable-next-line no-undef\n await p.evaluate(() => window.setupHighcharts());\n\n p.on('pageerror', async (err) => {\n // TODO: Consider adding a switch here that turns on log(0) logging\n // on page errors.\n log(1, '[page error]', err);\n await p.$eval(\n '#container',\n (element, errorMessage) => {\n // eslint-disable-next-line no-undef\n if (window._displayErrors) {\n element.innerHTML = errorMessage;\n }\n },\n `

Chart input data error

${err.toString()}`\n );\n });\n\n return p;\n};\n\nexport const create = async (puppeteerArgs) => {\n const allArgs = [...minimalArgs, ...(puppeteerArgs || [])];\n\n // Create a browser\n if (!browser) {\n let tryCount = 0;\n\n const open = async () => {\n try {\n log(\n 3,\n '[browser] attempting to get a browser instance (try',\n tryCount + ')'\n );\n\n browser = await puppeteer.launch({\n headless: 'new',\n args: allArgs,\n userDataDir: './tmp/'\n });\n } catch (e) {\n log(0, '[browser]', e);\n if (++tryCount < 25) {\n log(3, '[browser] failed:', e);\n await new Promise((response) => setTimeout(response, 4000));\n await open();\n } else {\n log(0, 'Max retries reached');\n }\n }\n };\n\n try {\n await open();\n } catch (e) {\n log(0, '[browser] Unable to open browser');\n return false;\n }\n\n if (!browser) {\n log(0, '[browser] Unable to open browser');\n return false;\n }\n }\n\n // Return a browser promise\n return browser;\n};\n\nexport const get = async () => {\n if (!browser) {\n throw 'No valid browser has been created';\n }\n\n return browser;\n};\n\nexport const close = async () => {\n // Close the browser when connnected\n if (browser.connected) {\n await browser.close();\n }\n};\n\nexport default {\n get,\n close,\n newPage\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\n// TODO: remove this temp benchmark stuff. I had this idea of doing a general benchmarking\n// system, but it adds so much bloat in the code that it shouldn't be there.\n\nimport benchmark from './benchmark.js';\nimport cache from './cache.js';\nimport { log } from './logger.js';\nimport svgTemplate from './../templates/svg_export/svg_export.js';\n\nimport { readFileSync } from 'fs';\nimport path from 'path';\nimport * as url from 'url';\n\nconst __basedir = url.fileURLToPath(new URL('.', import.meta.url));\n\n// const jsonTemplate = require('./../templates/json_export/json_export.js');\n\n/**\n * Gets the clip region for the chart DOM node.\n *\n * @param {object} page - A page of a browser instance.\n * @return {object} - A clipped region.\n */\nconst getClipRegion = (page) =>\n page.$eval('#chart-container', (element) => {\n const { x, y, width, height } = element.getBoundingClientRect();\n return {\n x,\n y,\n width,\n height: Math.trunc(height > 1 ? height : 500)\n };\n });\n\n/**\n * Rasterizes the page to an image (PNG or JPEG)\n *\n * @param {object} page - A page of a browser instance.\n * @param {string} type - The type of a result image.\n * @param {string} encoding - The type of encoding used.\n * @param {string} clip - The clip region.\n * @returns {string} - A string representation of a screenshot.\n */\nconst createImage = async (page, type, encoding, clip) =>\n await Promise.race([\n page.screenshot({\n type,\n encoding,\n clip,\n\n // #447 - always render on a transparent page\n // this will not affect users who do not explicitly set\n // chart.backgroundColor to a color with opacity lower than 1\n omitBackground: true\n }),\n new Promise((resolve, reject) =>\n setTimeout(() => reject(new Error('Rasterization timeout')), 1500)\n )\n ]);\n\n/**\n * Turns page into a PDF.\n *\n * @param {object} page - A page of a browser instance.\n * @param {number} height - The height of a chart.\n * @param {number} width - The width of a chart.\n * @param {string} encoding - The type of encoding used.\n * @return {object} - A buffer with PDF representation.\n */\nconst createPDF = async (page, height, width, encoding) =>\n await page.pdf({\n // This will remove an extra empty page in PDF exports\n height: height + 1,\n width,\n encoding\n });\n\n/**\n * Exports as a SVG.\n *\n * @param {object} page - A page of a browser instance.\n * @return {object} - The outerHTML element with the SVG representation.\n */\nconst createSVG = async (page) =>\n await page.$eval(\n '#container svg:first-of-type',\n (element) => element.outerHTML\n );\n\n/** Load config into a page and render a chart */\nconst setAsConfig = async (page, chart, options) =>\n await page.evaluate(\n // eslint-disable-next-line no-undef\n (chart, options) => window.triggerExport(chart, options),\n chart,\n options\n );\n\n/** Load SVG into a page */\n// const setAsSVG = async (page, svgStr) => true;\n\n/**\n * Does an export for a given browser.\n *\n * @param {object} browser - A browser instance.\n * @param {object} chart - Chart's options.\n * @param {object} options - All options object.\n * @return {object} - The data returned from one of the methods for exporting\n * a specific type of an image.\n */\nexport default async (page, chart, options) => {\n /**\n * Keeps track of all resources added on the page with addXXXTag. etc\n * It's VITAL that all added resources ends up here so we can clear things\n * out when doing a new export in the same page!\n */\n const injectedResources = [];\n\n /** Clear out all state set on the page with addScriptTag/addStyleTag. */\n const clearInjected = async (page) => {\n for (const res of injectedResources) {\n await res.dispose();\n }\n\n // Reset all CSS and script tags\n await page.evaluate(() => {\n // eslint-disable-next-line no-undef\n const [, ...scriptsToRemove] = document.getElementsByTagName('script');\n // eslint-disable-next-line no-undef\n const [, ...stylesToRemove] = document.getElementsByTagName('style');\n // eslint-disable-next-line no-undef\n const [...linksToRemove] = document.getElementsByTagName('link');\n\n // Remove tags\n for (const element of [\n ...scriptsToRemove,\n ...stylesToRemove,\n ...linksToRemove\n ]) {\n element.remove();\n }\n });\n };\n\n try {\n const exportBench = benchmark('Puppeteer');\n\n log(4, '[export] Determining export path.');\n\n const exportOptions = options.export;\n\n // Force a rAF\n // See https://github.com/puppeteer/puppeteer/issues/7507\n // eslint-disable-next-line no-undef\n await page.evaluate(() => requestAnimationFrame(() => {}));\n\n // Decide whether display error or debbuger wrapper around it\n const displayErrors =\n exportOptions?.options?.chart?.displayErrors &&\n cache.getCache().activeManifest.modules.debugger;\n\n // eslint-disable-next-line no-undef\n await page.evaluate((d) => (window._displayErrors = d), displayErrors);\n\n const svgBench = benchmark('SVG handling');\n\n let isSVG;\n\n if (\n chart.indexOf &&\n (chart.indexOf('= 0 || chart.indexOf('= 0)\n ) {\n // SVG INPUT HANDLING\n\n log(4, '[export] Treating as SVG.');\n\n // If input is also svg, just return it\n if (exportOptions.type === 'svg') {\n return chart;\n }\n\n isSVG = true;\n const setPageBench = benchmark('Setting content');\n await page.setContent(svgTemplate(chart));\n setPageBench();\n } else {\n // JSON Config handling\n\n log(4, '[export] Treating as config.');\n\n // Need to perform straight inject\n if (exportOptions.strInj) {\n // Injection based configuration export\n const setPageBench = benchmark('Setting page content (inject)');\n\n await setAsConfig(\n page,\n {\n chart: {\n height: exportOptions.height,\n width: exportOptions.width\n }\n },\n options\n );\n\n setPageBench();\n } else {\n // Basic configuration export\n\n chart.chart.height = exportOptions.height;\n chart.chart.width = exportOptions.width;\n\n const setContentBench = benchmark('Setting page content (config)');\n await setAsConfig(page, chart, options);\n setContentBench();\n }\n }\n\n svgBench();\n const resBench = benchmark('Applying resources');\n\n // Use resources\n const resources = options.customCode.resources;\n if (resources) {\n // Load custom JS code\n if (resources.js) {\n injectedResources.push(\n await page.addScriptTag({\n content: resources.js\n })\n );\n }\n\n // Load scripts from all custom files\n if (resources.files) {\n for (const file of resources.files) {\n try {\n const isLocal = !file.startsWith('http') ? true : false;\n\n // Add each custom script from resources' files\n injectedResources.push(\n await page.addScriptTag(\n isLocal\n ? {\n content: readFileSync(file, 'utf8')\n }\n : {\n url: file\n }\n )\n );\n } catch (notice) {\n log(4, '[export] JS file not found.');\n }\n }\n }\n\n const cssBench = benchmark('Loading css');\n\n // Load CSS\n if (resources.css) {\n let cssImports = resources.css.match(/@import\\s*([^;]*);/g);\n if (cssImports) {\n // Handle css section\n for (let cssImportPath of cssImports) {\n if (cssImportPath) {\n cssImportPath = cssImportPath\n .replace('url(', '')\n .replace('@import', '')\n .replace(/\"/g, '')\n .replace(/'/g, '')\n .replace(/;/, '')\n .replace(/\\)/g, '')\n .trim();\n\n // Add each custom css from resources\n if (cssImportPath.startsWith('http')) {\n injectedResources.push(\n await page.addStyleTag({\n url: cssImportPath\n })\n );\n } else if (options.customCode.allowFileResources) {\n injectedResources.push(\n await page.addStyleTag({\n path: path.join(__basedir, cssImportPath)\n })\n );\n }\n }\n }\n }\n\n // The rest of the CSS section will be content by now\n injectedResources.push(\n await page.addStyleTag({\n content: resources.css.replace(/@import\\s*([^;]*);/g, '') || ' '\n })\n );\n }\n\n cssBench();\n }\n\n resBench();\n\n // Get the real chart size\n const size = isSVG\n ? await page.$eval(\n '#chart-container svg:first-of-type',\n async (element, scale) => {\n return {\n chartHeight: element.height.baseVal.value * scale,\n chartWidth: element.width.baseVal.value * scale\n };\n },\n parseFloat(exportOptions.scale)\n )\n : await page.evaluate(async () => {\n // eslint-disable-next-line no-undef\n const { chartHeight, chartWidth } = window.Highcharts.charts[0];\n return {\n chartHeight,\n chartWidth\n };\n });\n\n const vpBench = benchmark('Setting viewport');\n\n // Set final height and width for viewport\n const viewportHeight = Math.ceil(size?.chartHeight || exportOptions.height);\n const viewportWidth = Math.ceil(size?.chartWidth || exportOptions.width);\n\n // Set the viewport for the first time\n // NOTE: the call to setViewport is expensive - can we get away with only\n // calling it once, e.g. moving this one into the isSVG condition below?\n await page.setViewport({\n height: viewportHeight,\n width: viewportWidth,\n deviceScaleFactor: isSVG ? 1 : parseFloat(exportOptions.scale)\n });\n\n // Prepare a zoom callback for the next evaluate call\n const zoomCallback = isSVG\n ? // In case of SVG the zoom must be set directly for body\n (scale) => {\n // Set the zoom as scale\n // eslint-disable-next-line no-undef\n document.body.style.zoom = scale;\n\n // Set the margin to 0px\n // eslint-disable-next-line no-undef\n document.body.style.margin = '0px';\n }\n : // No need for such scale manipulation in case of other types of exports\n () => {\n // Reset the zoom for other exports than to SVGs\n // eslint-disable-next-line no-undef\n document.body.style.zoom = 1;\n };\n\n // Set the zoom accordingly\n await page.evaluate(zoomCallback, parseFloat(exportOptions.scale));\n\n // Get the clip region for the page\n const { height, width, x, y } = await getClipRegion(page);\n\n if (!isSVG) {\n // Set the final viewport now that we have the real height\n await page.setViewport({\n width: Math.round(width),\n height: Math.round(height),\n deviceScaleFactor: parseFloat(exportOptions.scale)\n });\n }\n\n vpBench();\n\n let data;\n\n const expBenchmark = benchmark('Rasterizing chart');\n\n // RASTERIZATION\n if (exportOptions.type === 'svg') {\n // SVG\n data = await createSVG(page);\n } else if (exportOptions.type === 'png' || exportOptions.type === 'jpeg') {\n // PNG or JPEG\n data = await createImage(page, exportOptions.type, 'base64', {\n width: viewportWidth,\n height: viewportHeight,\n x,\n y\n });\n } else if (exportOptions.type === 'pdf') {\n // PDF\n data = await createPDF(page, viewportHeight, viewportWidth, 'base64');\n } else {\n throw `Unsupported output format ${exportOptions.type}`;\n }\n\n // Destroy old charts after the export is done\n await page.evaluate(() => {\n // eslint-disable-next-line no-undef\n const oldCharts = Highcharts.charts;\n\n // Check in any already existing charts\n if (oldCharts.length) {\n // Destroy old charts\n for (const oldChart of oldCharts) {\n oldChart && oldChart.destroy();\n // eslint-disable-next-line no-undef\n Highcharts.charts.shift();\n }\n }\n });\n\n expBenchmark();\n exportBench();\n\n await clearInjected(page);\n\n return data;\n } catch (error) {\n await clearInjected(page);\n log(1, `[export] Error encountered during export: ${error}`);\n\n return error;\n }\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2022, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { log } from './logger.js';\nconst timers = {};\n\n// TODO: Read from config\nlet enabled = false;\n\nexport default (id) => {\n if (!enabled) {\n return () => {};\n }\n\n timers[id] = new Date();\n return () => {\n log(\n 3,\n `[benchmark] - ${id}: ${new Date().getTime() - timers[id].getTime()}ms`\n );\n };\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport cssTemplate from './css.js';\n\nexport default (chart) => `\n\n\n \n \n Highcarts Export\n \n \n \n
\n ${chart}\n
\n \n\n\n`;\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { v4 as uuid } from 'uuid';\nimport { Pool } from 'tarn';\nimport {\n close,\n newPage as browserNewPage,\n create as createBrowser\n} from './browser.js';\nimport { log } from './logger.js';\n\nimport puppeteerExport from './export.js';\n\nlet performedExports = 0;\nlet exportAttempts = 0;\nlet timeSpent = 0;\nlet droppedExports = 0;\nlet spentAverage = 0;\nlet poolConfig = {};\n\n// The pool instance\nlet pool = false;\n\n// Custom puppeteer arguments\nlet puppeteerArgs;\n\nconst factory = {\n /**\n * Creates a new worker.\n *\n * @return {object} - An object with the id of a resource, the work count and\n * a reference to the browser page.\n */\n create: async () => {\n const id = uuid();\n let page = false;\n\n const s = new Date().getTime();\n\n try {\n page = await browserNewPage();\n\n if (!page || page.isClosed()) {\n throw 'invalid page';\n }\n\n log(\n 3,\n `[pool] Successfully created a worker ${id} - took ${\n new Date().getTime() - s\n } ms.`\n );\n } catch (error) {\n log(\n 1,\n `[pool] Error creating a new page in pool entry creation! ${error}`\n );\n\n throw 'Error creating page';\n }\n\n return {\n id,\n page,\n // Try to distribute the initial work count\n workCount: Math.round(Math.random() * (poolConfig.workLimit / 2))\n };\n },\n\n /**\n * Validates a worker.\n *\n * @param {object} workerHandle - A browser's instance.\n *\n * @return {boolean} - Bool that indicates if a resource is valid or not.\n */\n validate: (workerHandle) => {\n if (\n poolConfig.workLimit &&\n ++workerHandle.workCount > poolConfig.workLimit\n ) {\n log(\n 3,\n `[pool] Worker failed validation:`,\n `exceeded work limit (limit is ${poolConfig.workLimit})`\n );\n return false;\n }\n return true;\n },\n\n /**\n * Destroys a worker.\n *\n * @param {object} workerHandle - A browser's instance.\n */\n destroy: (workerHandle) => {\n log(3, `[pool] Destroying pool entry ${workerHandle.id}.`);\n\n if (workerHandle.page) {\n // We don't really need to wait around for this.\n workerHandle.page.close();\n }\n },\n\n // Logger function\n log: (message, logLevel) => console.log(`${logLevel}: ${message}`)\n};\n\n/**\n * Inits the pool of resources.\n *\n * @param {object} config - Pool configuration along with custom puppeteer\n * arguments for the puppeteer.launch function.\n */\nexport const init = async (config) => {\n // The newest puppeteer arguments for the browser creation\n puppeteerArgs = config.puppeteerArgs;\n\n // Wait until we've sucessfully created a browser instance.\n try {\n await createBrowser(puppeteerArgs);\n } catch (e) {\n log(0, '[pool|browser]', e);\n }\n\n // For the module scope usage\n poolConfig = config && config.pool ? { ...config.pool } : {};\n\n log(\n 3,\n '[pool] Initializing pool:',\n `min ${poolConfig.initialWorkers}, max ${poolConfig.maxWorkers}.`\n );\n\n if (pool) {\n return log(\n 4,\n '[pool] Already initialized, please kill it before creating a new one.'\n );\n }\n\n // Attach process' exit listeners\n if (poolConfig.listenToProcessExits) {\n attachProcessExitListeners();\n }\n\n try {\n // Create a pool along with a minimal number of resources\n pool = new Pool({\n // Get the create/validate/destroy/log functions\n ...factory,\n min: poolConfig.initialWorkers,\n max: poolConfig.maxWorkers,\n createRetryIntervalMillis: 200,\n createTimeoutMillis: poolConfig.acquireTimeout,\n acquireTimeoutMillis: poolConfig.acquireTimeout,\n destroyTimeoutMillis: poolConfig.acquireTimeout,\n idleTimeoutMillis: poolConfig.timeoutThreshold,\n reapIntervalMillis: 1000, // poolConfig.reaper ? 120000 : 0, for now\n propagateCreateError: false\n });\n\n // Set events\n pool.on('createFail', (eventId, err) => {\n log(\n 1,\n `[pool] Error when creating worker of an event id ${eventId}:`,\n err\n );\n });\n\n pool.on('acquireFail', (eventId, err) => {\n log(\n 1,\n `[pool] Error when acquiring worker of an event id ${eventId}:`,\n err\n );\n });\n\n pool.on('destroyFail', (eventId, resource, err) => {\n log(\n 1,\n `[pool] Error when destroying worker of an id ${resource.id}, event id ${eventId}:`,\n err\n );\n });\n\n pool.on('release', (resource) => {\n log(4, `[pool] Releasing a worker of an id ${resource.id}`);\n });\n\n pool.on('destroySuccess', (eventId, resource) => {\n log(4, `[pool] Destroyed a worker of an id ${resource.id}`);\n });\n\n const initialResources = [];\n // Create an initial number of resources\n for (let i = 0; i < poolConfig.initialWorkers; i++) {\n initialResources.push(await pool.acquire().promise);\n }\n\n // Release the initial number of resources back to the pool\n initialResources.forEach((resource) => {\n pool.release(resource);\n });\n\n log(\n 3,\n `[pool] The pool is ready with ${poolConfig.initialWorkers} initial resources waiting.`\n );\n } catch (error) {\n log(1, `[pool] Couldn't create the worker pool ${error}`);\n throw error;\n }\n};\n\n/**\n * Attaches process' exit listeners.\n */\nexport function attachProcessExitListeners() {\n log(4, '[pool] Attaching exit listeners to the process.');\n\n // Kill all pool resources on exit\n process.on('exit', async () => {\n await killPool();\n });\n\n // Handler for the SIGINT\n process.on('SIGINT', (name, code) => {\n log(4, `The ${name} event with code: ${code}.`);\n process.exit(1);\n });\n\n // Handler for the SIGTERM\n process.on('SIGTERM', (name, code) => {\n log(4, `The ${name} event with code: ${code}.`);\n process.exit(1);\n });\n\n // Handler for the uncaughtException\n process.on('uncaughtException', async (error, name) => {\n log(4, `The ${name} error, message: ${error.message}.`);\n });\n}\n\n/**\n * Kills the pool and flush the browser instance.\n */\nexport async function killPool() {\n log(3, '[pool] Killing all workers.');\n\n // Return true when the pool is already destroyed\n if (pool.destroyed) {\n // Close the browser instance if still connected\n await close();\n return true;\n }\n\n // If still alive, destroy the pool of pages before closing a browser\n await pool.destroy();\n\n // Close the browser instance\n await close();\n return true;\n}\n\n/**\n * Posts work to the pool.\n *\n * @param {object} chart - Chart's options.\n * @param {object} options - All options object.\n */\nexport const postWork = async (chart, options) => {\n let workerHandle;\n\n // Handle fail conditions\n const fail = (msg) => {\n ++droppedExports;\n\n if (workerHandle) {\n pool.release(workerHandle);\n }\n\n throw 'In pool.postWork: ' + msg;\n };\n\n log(4, '[pool] Work received, starting to process.');\n\n if (poolConfig.benchmarking) {\n getPoolInfo();\n }\n\n ++exportAttempts;\n\n if (!pool) {\n log(1, '[pool] Work received, but pool has not been started.');\n return fail('Pool is not inited but work was posted to it!');\n }\n\n // Acquire the worker along with the id of resource and work count\n try {\n log(4, '[pool] Acquiring worker');\n workerHandle = await pool.acquire().promise;\n } catch (error) {\n return fail(`[pool] Error when acquiring available entry: ${error}`);\n }\n\n log(4, '[pool] Acquired worker handle');\n\n if (!workerHandle.page) {\n return fail('Resolved worker page is invalid: pool setup is wonky');\n }\n\n try {\n // Save the start time\n let workStart = new Date().getTime();\n\n log(4, `[pool] Starting work on pool entry ${workerHandle.id}.`);\n\n // Perform an export on a puppeteer level\n const result = await puppeteerExport(workerHandle.page, chart, options);\n\n // Check if it's an error\n if (result instanceof Error) {\n // TODO: If the export failed because puppeteer timed out, we need to force kill the worker so we get a new page. That needs to be handled better than this hack.\n if (result.message === 'Rasterization timeout') {\n workerHandle.page.close();\n workerHandle.page = await browserNewPage();\n }\n\n return fail(result);\n }\n\n // Release the resource back to the pool\n pool.release(workerHandle);\n\n // Used for statistics in averageTime and processedWorkCount, which\n // in turn is used by the /health route.\n const workEnd = new Date().getTime();\n const exportTime = workEnd - workStart;\n timeSpent += exportTime;\n spentAverage = timeSpent / ++performedExports;\n\n log(4, `[pool] Work completed in ${exportTime} ms.`);\n\n // Otherwise return the result\n return {\n data: result,\n options\n };\n } catch (error) {\n fail(`Error trying to perform puppeteer export: ${error}.`);\n }\n};\n\n/**\n * Gets the pool.\n */\nexport function getPool() {\n return pool;\n}\n\nexport const getPoolInfoJSON = () => ({\n min: pool.min,\n max: pool.max,\n size: pool.size,\n available: pool.available,\n borrowed: pool.borrowed,\n pending: pool.pending,\n spareResourceCapacity: pool.spareResourceCapacity\n});\n\n/**\n * Gets the pool's information.\n */\nexport function getPoolInfo() {\n const {\n min,\n max,\n size,\n available,\n borrowed,\n pending,\n spareResourceCapacity\n } = pool;\n\n log(4, `[pool] The minimum number of resources allowed by pool: ${min}.`);\n log(4, `[pool] The maximum number of resources allowed by pool: ${max}.`);\n log(\n 4,\n `[pool] The number of all resources in pool (free or in use): ${size}.`\n );\n log(\n 4,\n `[pool] The number of resources that are currently available: ${available}.`\n );\n log(\n 4,\n `[pool] The number of resources that are currently acquired: ${borrowed}.`\n );\n log(\n 4,\n `[pool] The number of callers waiting to acquire a resource: ${pending}.`\n );\n log(\n 4,\n `[pool] The number of how many more resources can the pool manage/create: ${spareResourceCapacity}.`\n );\n}\n\nexport default {\n init,\n killPool,\n postWork,\n getPool,\n getPoolInfo,\n getPoolInfoJSON,\n workAttempts: () => exportAttempts,\n droppedWork: () => droppedExports,\n averageTime: () => spentAverage,\n processedWorkCount: () => performedExports\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport cache from '../../cache.js';\nimport pool from '../../pool.js';\n\nconst packageVersion = process.env.npm_package_version;\nconst serverStartTime = new Date();\n\n/**\n * Adds the /health route which outputs basic stats for the server\n */\nexport default (app) =>\n !app\n ? false\n : app.get('/health', (request, response) => {\n response.send({\n status: 'OK',\n bootTime: serverStartTime,\n uptime:\n Math.floor(\n (new Date().getTime() - serverStartTime.getTime()) / 1000 / 60\n ) + ' minutes',\n version: packageVersion,\n highchartsVersion: cache.version(),\n averageProcessingTime: pool.averageTime(),\n performedExports: pool.processedWorkCount(),\n failedExports: pool.droppedWork(),\n exportAttempts: pool.workAttempts(),\n sucessRatio: (pool.processedWorkCount() / pool.workAttempts()) * 100,\n // eslint-disable-next-line import/no-named-as-default-member\n pool: pool.getPoolInfoJSON()\n });\n });\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { existsSync, readFileSync, promises as fsPromises } from 'fs';\n\nimport prompts from 'prompts';\n\nimport { log } from './logger.js';\nimport { deepCopy, isObject, printUsage, toBoolean } from './utils.js';\nimport {\n absoluteProps,\n defaultConfig,\n nestedArgs,\n promptsConfig\n} from './schemas/config.js';\n\nlet generalOptions = {};\n\n/**\n * Getter for the general options.\n *\n * @return {object} - General options object.\n */\nexport const getOptions = () => generalOptions;\n\n/**\n * Initializes and sets the general options for the server instace.\n *\n * @param {object} userOptions - Additional user options (e.g. from the node\n * module usage).\n * @param {string[]} args - CLI arguments.\n * @return {object} - General options object.\n */\nexport const setOptions = (userOptions, args) => {\n // Only for the CLI usage\n if (args?.length) {\n // Get the additional options from the custom JSON file\n generalOptions = loadConfigFile(args);\n }\n\n // Update the default config with a correct option values\n updateDefaultConfig(defaultConfig, generalOptions);\n\n // Set values for server's options and returns them\n generalOptions = initOptions(defaultConfig);\n\n // Apply user options if there are any\n if (userOptions) {\n // Merge user options\n generalOptions = mergeConfigOptions(\n generalOptions,\n userOptions,\n absoluteProps\n );\n }\n\n // Only for the CLI usage\n if (args?.length) {\n // Pair provided arguments\n generalOptions = pairArgumentValue(generalOptions, args, defaultConfig);\n }\n\n // Return final general options\n return generalOptions;\n};\n\n/**\n * Displays a prompt for the manual configuration.\n *\n * @param {string} configFileName - The name of a configuration file.\n */\nexport const manualConfig = async (configFileName) => {\n // Prepare a config object\n let configFile = {};\n\n // Check if provided config file exists\n if (existsSync(configFileName)) {\n configFile = JSON.parse(readFileSync(configFileName, 'utf8'));\n }\n\n // Question about a configuration category\n const onSubmit = async (p, categories) => {\n let questionsCounter = 0;\n let allQuestions = [];\n\n // Create a corresponding property in the manualConfig object\n for (const section of categories) {\n // Mark each option with a section\n promptsConfig[section] = promptsConfig[section].map((option) => ({\n ...option,\n section\n }));\n\n // Collect the questions\n allQuestions = [...allQuestions, ...promptsConfig[section]];\n }\n\n await prompts(allQuestions, {\n onSubmit: async (prompt, answer) => {\n // Get the default modules\n if (prompt.name === 'modules') {\n answer = answer.length\n ? answer.map((module) => prompt.choices[module])\n : prompt.choices;\n\n configFile[prompt.section][prompt.name] = answer;\n } else {\n configFile[prompt.section] = recursiveProps(\n Object.assign({}, configFile[prompt.section] || {}),\n prompt.name.split('.'),\n answer\n );\n }\n\n if (++questionsCounter === allQuestions.length) {\n try {\n await fsPromises.writeFile(\n configFileName,\n JSON.stringify(configFile, null, 2),\n 'utf8'\n );\n } catch (error) {\n log(1, `[config] Error while creating config.json: ${error}`);\n }\n return true;\n }\n }\n });\n\n return true;\n };\n\n // Find the categories\n const choices = Object.keys(promptsConfig).map((choice) => ({\n title: `${choice} options`,\n value: choice\n }));\n\n // Category prompt\n return prompts(\n {\n type: 'multiselect',\n name: 'category',\n message: 'Which category do you want to configure?',\n hint: 'Space: Select specific, A: Select all, Enter: Confirm.',\n instructions: '',\n choices\n },\n { onSubmit }\n );\n};\n\n/**\n * Maps the old options to the new config structure.\n *\n * @param {object} oldOptions - Options to be mapped.\n */\nexport const mapToNewConfig = (oldOptions) => {\n const newOptions = {};\n // Cycle through old-structured options\n for (const [key, value] of Object.entries(oldOptions)) {\n const propertiesChain = nestedArgs[key] ? nestedArgs[key].split('.') : [];\n\n // Populate object in correct properties levels\n propertiesChain.reduce(\n (obj, prop, index) =>\n (obj[prop] =\n propertiesChain.length - 1 === index ? value : obj[prop] || {}),\n newOptions\n );\n }\n return newOptions;\n};\n\n/**\n * Merges the new options to the options object. It omits undefined values.\n *\n * @param {object} options - Old options.\n * @param {object} newOptions - New options.\n * @param {string[]} absoluteProps - Array of object names that should be force\n * merged.\n */\nexport const mergeConfigOptions = (options, newOptions, absoluteProps = []) => {\n const mergedOptions = deepCopy(options);\n\n for (const [key, value] of Object.entries(newOptions)) {\n mergedOptions[key] =\n isObject(value) &&\n !absoluteProps.includes(key) &&\n mergedOptions[key] !== undefined\n ? mergeConfigOptions(mergedOptions[key], value, absoluteProps)\n : value !== undefined\n ? value\n : mergedOptions[key];\n }\n\n return mergedOptions;\n};\n\n/**\n * Initializes options for the `startExport` method by merging user options\n * with the general options.\n *\n * @param {any} exportOptions - User options for exporting.\n * @param {any} generalOptions - General options are used for the export server.\n * @return {object} - User options merged with default options.\n */\nexport const initExportSettings = (exportOptions, generalOptions = {}) => {\n let options = {};\n\n if (exportOptions.svg) {\n options = deepCopy(generalOptions);\n options.export.type = exportOptions.type || exportOptions.export.type;\n options.export.scale = exportOptions.scale || exportOptions.export.scale;\n options.export.outfile =\n exportOptions.outfile || exportOptions.export.outfile;\n options.payload = {\n svg: exportOptions.svg\n };\n } else {\n options = mergeConfigOptions(\n generalOptions,\n exportOptions,\n // Omit going down recursively with the belows\n absoluteProps\n );\n }\n\n options.export.outfile =\n options.export?.outfile || `chart.${options.export?.type || 'png'}`;\n return options;\n};\n\n/**\n * Loads the configuration from a custom JSON file.\n *\n * @param {string[]} args - CLI arguments.\n * @return {object} - Options object from the JSON file.\n */\nfunction loadConfigFile(args) {\n // Check if the --loadConfig option was used\n const configIndex = args.findIndex(\n (arg) => arg.replace(/-/g, '') === 'loadConfig'\n );\n\n // Check if the --loadConfig has a value\n if (configIndex > -1 && args[configIndex + 1]) {\n const fileName = args[configIndex + 1];\n try {\n // Check if an additional config file is a correct JSON file\n if (fileName && fileName.endsWith('.json')) {\n // Load an optional custom JSON config file\n return JSON.parse(readFileSync(fileName));\n }\n } catch (error) {\n log(1, `[config] Unable to load config from the ${fileName}: ${error}`);\n }\n }\n\n // No additional options to return\n return {};\n}\n\n/**\n * Setting correct values of the options from the default config.\n *\n * @param {object} configObj - The config object based on which the initial\n * configuration be made.\n * @param {object} customObj - The custom object which can contain additional\n * option values to set.\n * @param {string} propChain - Required for creating a string chain of\n * properties for nested arguments.\n */\nfunction updateDefaultConfig(configObj, customObj = {}, propChain = '') {\n Object.keys(configObj).forEach((key) => {\n if (!['puppeteer', 'highcharts'].includes(key)) {\n const entry = configObj[key];\n const customValue = customObj && customObj[key];\n let numEnvVal;\n\n if (typeof entry.value === 'undefined') {\n updateDefaultConfig(entry, customValue, `${propChain}.${key}`);\n } else {\n // If a value from a custom JSON exists, it take precedence\n if (customValue !== undefined) {\n entry.value = customValue;\n }\n\n // If a value from an env variable exists, it take precedence\n if (entry.envLink) {\n // Load the env var\n if (entry.type === 'boolean') {\n entry.value = toBoolean(\n [process.env[entry.envLink], entry.value].find(\n (el) => el || el === 'false'\n )\n );\n } else if (entry.type === 'number') {\n numEnvVal = +process.env[entry.envLink];\n entry.value = numEnvVal >= 0 ? numEnvVal : entry.value;\n } else if (\n entry.type.indexOf(']') >= 0 &&\n process.env[entry.envLink]\n ) {\n entry.value = process.env[entry.envLink].split(',');\n } else {\n entry.value = process.env[entry.envLink] || entry.value;\n }\n }\n }\n }\n });\n}\n\n/**\n * Inits options recursively.\n *\n * @param {any} items - Items to update options from.\n * @return {object} - Updated options object.\n */\nfunction initOptions(items) {\n let options = {};\n for (const [name, item] of Object.entries(items)) {\n options[name] = Object.prototype.hasOwnProperty.call(item, 'value')\n ? item.value\n : initOptions(item);\n }\n return options;\n}\n\n/**\n * Pairs argument with a corresponding value.\n *\n * @param {object} options - All server options.\n * @param {string[]} args - Array of arguments from a user.\n * @param {object} defaultConfig - The default config object.\n */\nfunction pairArgumentValue(options, args, defaultConfig) {\n for (let i = 0; i < args.length; i++) {\n let option = args[i].replace(/-/g, '');\n\n // Find the right place for property's value\n const propertiesChain = nestedArgs[option]\n ? nestedArgs[option].split('.')\n : [];\n\n propertiesChain.reduce((obj, prop, index) => {\n if (propertiesChain.length - 1 === index) {\n // Finds an option and set a corresponding value\n if (typeof obj[prop] !== 'undefined') {\n if (args[++i]) {\n obj[prop] = args[i] || obj[prop];\n } else {\n console.log(`Missing argument value for ${option}!`.red, '\\n');\n options = printUsage(defaultConfig);\n }\n }\n }\n return obj[prop];\n }, options);\n }\n\n return options;\n}\n\n/**\n * Recursively sets a property in a correct indentation level based on the\n * array of nested properties names.\n *\n * @param {object} objectToUpdate - Object where a property must be set on a\n * correct level.\n * @param {string[]}nestedNames - Array of nasted names that indicates\n * indentation level.\n * @param {any} value - A value to assign to the property.\n * @return {object} - Updated options object.\n */\nfunction recursiveProps(objectToUpdate, nestedNames, value) {\n while (nestedNames.length > 1) {\n const propName = nestedNames.shift();\n\n // Create a property in object if it doesn't exist\n if (!Object.prototype.hasOwnProperty.call(objectToUpdate, propName)) {\n objectToUpdate[propName] = {};\n }\n\n // Call function again if there still names to go\n objectToUpdate[propName] = recursiveProps(\n Object.assign({}, objectToUpdate[propName]),\n nestedNames,\n value\n );\n\n return objectToUpdate;\n }\n\n // Assign the final value\n objectToUpdate[nestedNames[0]] = value;\n return objectToUpdate;\n}\n\nexport default {\n getOptions,\n setOptions,\n manualConfig,\n mapToNewConfig,\n mergeConfigOptions,\n initExportSettings\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { readFile, readFileSync, writeFileSync } from 'fs';\n\nimport { log } from './logger.js';\nimport { killPool, postWork } from './pool.js';\nimport {\n clearText,\n fixType,\n handleResources,\n isCorrectJSON,\n optionsStringify,\n roundNumber,\n toBoolean,\n wrapAround\n} from './utils.js';\nimport { initExportSettings, getOptions } from './config.js';\n\nlet allowCodeExecution = false;\n\nexport const startExport = async (settings, endCallback) => {\n // Starting exporting process message\n log(4, '[chart] Starting exporting process.');\n\n // Initialize options\n const options = initExportSettings(settings, getOptions());\n\n // Get the export options\n const exportOptions = options.export;\n\n // If SVG is an input (argument can be sent only by the request)\n if (options.payload?.svg && options.payload.svg !== '') {\n return exportAsString(options.payload.svg.trim(), options, endCallback);\n }\n\n // Export using options from the file\n if (exportOptions.infile && exportOptions.infile.length) {\n log(4, '[chart] Attempting to export from an input file.');\n\n // Try to read the file\n return readFile(exportOptions.infile, 'utf8', (error, infile) => {\n if (error) {\n return log(1, `[chart] Error loading input file: ${error}.`);\n }\n\n // Get the string representation\n options.export.instr = infile;\n return exportAsString(options.export.instr.trim(), options, endCallback);\n });\n }\n\n // Export with options from the raw representation\n if (\n (exportOptions.instr && exportOptions.instr !== '') ||\n (exportOptions.options && exportOptions.options !== '')\n ) {\n log(4, '[chart] Attempting to export from a raw input.');\n\n // Perform a direct inject when forced\n if (toBoolean(options.customCode?.allowCodeExecution)) {\n return doStraightInject(options, endCallback);\n }\n\n // Either try to parse to JSON first or do the direct export\n return typeof exportOptions.instr === 'string'\n ? exportAsString(exportOptions.instr.trim(), options, endCallback)\n : doExport(\n options,\n exportOptions.instr || exportOptions.options,\n endCallback\n );\n }\n\n // No input specified, pass an error message to the callback\n log(\n 1,\n clearText(\n `[chart] No input specified.\n ${JSON.stringify(exportOptions, undefined, ' ')}.`\n )\n );\n\n return (\n endCallback &&\n endCallback(false, {\n error: true,\n message: 'No input specified.'\n })\n );\n};\n\nexport const batchExport = (options) => {\n const batchFunctions = [];\n\n // Split and pair the --batch arguments\n for (let pair of options.export.batch.split(';')) {\n pair = pair.split('=');\n if (pair.length === 2) {\n batchFunctions.push(\n new Promise((resolve, reject) => {\n startExport(\n {\n ...options,\n export: {\n ...options.export,\n infile: pair[0],\n outfile: pair[1]\n }\n },\n (info, error) => {\n // Throw an error\n if (error) {\n return reject(error);\n }\n\n // Save the base64 from a buffer to a correct image file\n writeFileSync(\n info.options.export.outfile,\n Buffer.from(info.data, 'base64')\n );\n\n resolve();\n }\n );\n })\n );\n }\n }\n\n // Kill the pool after all exports are done\n Promise.all(batchFunctions)\n .then(() => {\n killPool();\n })\n .catch((error) => {\n log(1, `[chart] Error encountered during batch export: ${error}`);\n killPool();\n });\n};\n\nexport const singleExport = (options) => {\n // Use instr or its alias, options\n options.export.instr = options.export.instr || options.export.options;\n\n // Perform an export\n startExport(options, (info, error) => {\n // Exit process when error\n if (error) {\n log(1, `[cli] ${error.message}`);\n process.exit(1);\n }\n\n const { outfile, type } = info.options.export;\n\n // Save the base64 from a buffer to a correct image file\n writeFileSync(\n outfile || `chart.${type}`,\n type !== 'svg' ? Buffer.from(info.data, 'base64') : info.data\n );\n\n // Kill the pool\n killPool();\n });\n};\n\n/**\n * Function for choosing chart size and scale based on options prioritization.\n *\n * @param {object} options - All options object.\n * @return {object} - An object with updated size and scale for a chart.\n */\nexport const findChartSize = (options) => {\n const { chart, exporting } =\n options.export?.options || isCorrectJSON(options.export?.instr);\n\n // See if globalOptions holds chart or exporting size\n const globalOptions = isCorrectJSON(options.export?.globalOptions);\n\n // Secure scale value\n let scale =\n options.export?.scale ||\n exporting?.scale ||\n globalOptions?.exporting?.scale ||\n options.export?.defaultScale ||\n 1;\n\n // the scale cannot be lower than 0.1 and cannot be higher than 5.0\n scale = Math.max(0.1, Math.min(scale, 5.0));\n\n // we want to round the numbers like 0.23234 -> 0.23\n scale = roundNumber(scale, 2);\n\n // Find chart size and scale\n return {\n height:\n options.export?.height ||\n exporting?.sourceHeight ||\n chart?.height ||\n globalOptions?.exporting?.sourceHeight ||\n globalOptions?.chart?.height ||\n options.export?.defaultHeight ||\n 400,\n width:\n options.export?.width ||\n exporting?.sourceWidth ||\n chart?.width ||\n globalOptions?.exporting?.sourceWidth ||\n globalOptions?.chart?.width ||\n options.export?.defaultWidth ||\n 600,\n scale\n };\n};\n\n/**\n * Function for final options preparation before export.\n *\n * @param {object} options - All options object.\n * @param {object} chartJson - Chart JSON.\n * @param {function} endCallback - The end callback.\n * @param {string} svg - The SVG representation.\n */\nconst doExport = (options, chartJson, endCallback, svg) => {\n let { export: exportOptions, customCode: customCodeOptions } = options;\n\n const allowCodeExecutionScoped =\n typeof customCodeOptions.allowCodeExecution === 'boolean'\n ? customCodeOptions.allowCodeExecution\n : allowCodeExecution;\n\n if (!customCodeOptions) {\n customCodeOptions = options.customCode = {};\n } else if (typeof options.customCode.resources === 'string') {\n // Process resources\n options.customCode.resources = handleResources(\n options.customCode.resources,\n toBoolean(options.customCode.allowFileResources)\n );\n } else if (!options.customCode.resources) {\n try {\n const resources = readFileSync('resources.json', 'utf8');\n options.customCode.resources = handleResources(\n resources,\n toBoolean(options.customCode.allowFileResources)\n );\n } catch (err) {\n log(3, `[chart] The default resources.json file not found.`);\n }\n }\n\n // If the allowCodeExecution flag isn't set, we should refuse the usage\n // of callback, resources, and custom code. Additionally, the worker will\n // refuse to run arbitrary JavaScript. Prioritized should be the scoped\n // option, then we should take a look at the overall pool option.\n if (!allowCodeExecutionScoped && customCodeOptions) {\n if (\n customCodeOptions.callback ||\n customCodeOptions.resources ||\n customCodeOptions.customCode\n ) {\n // Send back a friendly message saying that the exporter does not support\n // these settings.\n return (\n endCallback &&\n endCallback(false, {\n error: true,\n message: clearText(\n `The callback, resources and customCode have been disabled for this\n server.`\n )\n })\n );\n }\n\n // Reset all additional custom code\n customCodeOptions.callback = false;\n customCodeOptions.resources = false;\n customCodeOptions.customCode = false;\n }\n\n // Clean properties to keep it lean and mean\n if (chartJson) {\n chartJson.chart = chartJson.chart || {};\n chartJson.exporting = chartJson.exporting || {};\n chartJson.exporting.enabled = false;\n }\n\n exportOptions.constr = exportOptions.constr || 'chart';\n exportOptions.type = fixType(exportOptions.type, exportOptions.outfile);\n if (exportOptions.type === 'svg') {\n exportOptions.width = false;\n }\n\n // Prepare global and theme options\n ['globalOptions', 'themeOptions'].forEach((optionsName) => {\n try {\n if (exportOptions && exportOptions[optionsName]) {\n if (\n typeof exportOptions[optionsName] === 'string' &&\n exportOptions[optionsName].endsWith('.json')\n ) {\n exportOptions[optionsName] = isCorrectJSON(\n readFileSync(exportOptions[optionsName], 'utf8'),\n true\n );\n } else {\n exportOptions[optionsName] = isCorrectJSON(\n exportOptions[optionsName],\n true\n );\n }\n }\n } catch (error) {\n exportOptions[optionsName] = {};\n log(1, `[chart] The ${optionsName} not found.`);\n }\n });\n\n // Prepare customCode\n if (customCodeOptions.allowCodeExecution) {\n customCodeOptions.customCode = wrapAround(\n customCodeOptions.customCode,\n customCodeOptions.allowFileResources\n );\n }\n\n // Get the callback\n if (\n customCodeOptions &&\n customCodeOptions.callback &&\n customCodeOptions.callback?.indexOf('{') < 0\n ) {\n // The allowFileResources is always set to false for HTTP requests to avoid\n // injecting arbitrary files from the fs\n if (customCodeOptions.allowFileResources) {\n try {\n customCodeOptions.callback = readFileSync(\n customCodeOptions.callback,\n 'utf8'\n );\n } catch (error) {\n log(2, `[chart] Error loading callback: ${error}.`);\n customCodeOptions.callback = false;\n }\n } else {\n customCodeOptions.callback = false;\n }\n }\n\n // Size search\n options.export = {\n ...options.export,\n ...findChartSize(options)\n };\n\n // Post the work to the pool\n postWork(exportOptions.strInj || chartJson || svg, options)\n .then((result) => endCallback(result))\n .catch((error) => {\n log(0, '[chart] When posting work:', error);\n return endCallback(false, error);\n });\n};\n\n/**\n * Function for straight injecting the code.\n * Dangerous and must be used deliberately by someone who sets up a server\n * (see --allowCodeExecution).\n *\n * @param {object} options - All options object.\n * @param {function} endCallback - The function to call when exporting is done.\n */\nconst doStraightInject = (options, endCallback) => {\n try {\n let strInj;\n let instr = options.export.instr || options.export.options;\n\n if (typeof instr !== 'string') {\n // Try to stringify options\n strInj = instr = optionsStringify(\n instr,\n options.customCode?.allowCodeExecution\n );\n }\n strInj = instr.replaceAll(/\\t|\\n|\\r/g, '').trim();\n\n // Get rid of the ;\n if (strInj[strInj.length - 1] === ';') {\n strInj = strInj.substring(0, strInj.length - 1);\n }\n\n // Save as stright inject string\n options.export.strInj = strInj;\n return doExport(options, false, endCallback);\n } catch (error) {\n const message = clearText(\n `Malformed input detected for ${options.export?.requestId || '?'}:\n Please make sure that your JSON/JavaScript options\n are sent using the \"options\" attribute, and that if you're using\n SVG, it is unescaped.`\n );\n\n log(1, message);\n return (\n endCallback &&\n endCallback(\n false,\n JSON.stringify({\n error: true,\n message\n })\n )\n );\n }\n};\n\n/**\n * Prepares an input before exporting.\n *\n * @param {string} stringToExport - String representation of SVG/export options.\n * @param {object} options - All options object.\n * @param {function} endCallback - The function to call when exporting is done.\n */\nconst exportAsString = (stringToExport, options, endCallback) => {\n const { allowCodeExecution } = options.customCode;\n\n // Check if it is SVG\n if (\n stringToExport.indexOf('= 0 ||\n stringToExport.indexOf('= 0\n ) {\n log(4, '[chart] Parsing input as SVG.');\n return doExport(options, false, endCallback, stringToExport);\n }\n\n try {\n // Try to parse to JSON and call the doExport function\n const chartJSON = JSON.parse(stringToExport.replaceAll(/\\t|\\n|\\r/g, ' '));\n\n // If a correct JSON, do the export\n return doExport(options, chartJSON, endCallback);\n } catch (error) {\n // Not a valid JSON\n if (toBoolean(allowCodeExecution)) {\n return doStraightInject(options, endCallback);\n } else {\n // Do not allow straight injection without the allowCodeExecution flag\n return (\n endCallback &&\n endCallback(false, {\n error: true,\n message: clearText(\n `Only JSON configurations and SVG is allowed for this server. If\n this is your server, JavaScript exporting can be enabled by starting\n the server with the --allowCodeExecution flag.`\n )\n })\n );\n }\n }\n};\n\nexport const getAllowCodeExecution = () => allowCodeExecution;\n\nexport const setAllowCodeExecution = (value) => {\n allowCodeExecution = toBoolean(value);\n};\n\n/**\n * Starts an exporting process\n *\n * @param {object} settings - Settings for export.\n * @param {function} endCallback - The function to call when exporting is done.\n */\nexport default {\n batchExport,\n singleExport,\n getAllowCodeExecution,\n setAllowCodeExecution,\n startExport,\n findChartSize\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { v4 as uuid } from 'uuid';\n\nimport { getAllowCodeExecution, startExport } from '../../chart.js';\nimport { getOptions, mergeConfigOptions } from '../../config.js';\nimport { log } from '../../logger.js';\nimport {\n clearText,\n fixType,\n isCorrectJSON,\n isPrivateRangeUrlFound,\n optionsStringify,\n measureTime\n} from '../../utils.js';\n\n// Reversed MIME types\nconst reversedMime = {\n png: 'image/png',\n jpeg: 'image/jpeg',\n gif: 'image/gif',\n pdf: 'application/pdf',\n svg: 'image/svg+xml'\n};\n\n// The requests counter\nlet requestsCounter = 0;\n\nconst benchmark = false;\n\n// The array of callbacks to call before a request\nconst beforeRequest = [];\n\n// The array of callbacks to call after a request\nconst afterRequest = [];\n\n/**\n * Calls callbacks.\n *\n * @param {Array} callbacks - An array of callbacks.\n * @param {object} request - The request.\n * @param {object} response - The response.\n * @param {object} data - The data to send to callbacks.\n * @return {object} - The result from a callback.\n */\nconst doCallbacks = (callbacks, request, response, data) => {\n let result = true;\n const { id, uniqueId, type, body } = data;\n\n callbacks.some((callback) => {\n if (callback) {\n let callResponse = callback(request, response, id, uniqueId, type, body);\n\n if (callResponse !== undefined && callResponse !== true) {\n result = callResponse;\n }\n\n return true;\n }\n });\n\n return result;\n};\n\n/**\n * Handles an export.\n *\n * @param {object} request - The request.\n * @param {object} response - The response.\n */\nconst exportHandler = (request, response) => {\n // Start counting time\n const stopCounter = measureTime();\n\n // Get the current server's general options\n const defaultOptions = getOptions();\n\n // Init default options\n if (benchmark) {\n console.log('Init default options:', stopCounter(), 'ms.');\n }\n\n const body = request.body;\n const id = ++requestsCounter;\n const uniqueId = uuid().replace(/-/g, '');\n let type = fixType(body.type);\n\n // Fix type\n if (benchmark) {\n console.log('Fix type:', stopCounter(), 'ms.');\n }\n\n // Throw 'Bad Request' if there's no body\n if (!body) {\n return response.status(400).send(\n clearText(\n `Body is required. Sending a body? Make sure your Content-type header\n is correct. Accepted is application/json and multipart/form-data.`\n )\n );\n }\n\n // All of the below can be used\n let instr = isCorrectJSON(body.infile || body.options || body.data);\n\n // Is correct JSON\n if (benchmark) {\n console.log('Is correct JSON:', stopCounter(), 'ms.');\n }\n\n // Throw 'Bad Request' if there's no JSON or SVG to export\n if (!instr && !body.svg) {\n log(\n 2,\n clearText(\n `Request ${uniqueId} from ${\n request.headers['x-forwarded-for'] || request.connection.remoteAddress\n } was incorrect. Check your payload.`\n )\n );\n\n return response.status(400).send(\n clearText(\n `No correct chart data found. Please make sure you are using\n application/json or multipart/form-data headers, and that the chart\n data is in the 'infile', 'options' or 'data' attribute if sending\n JSON or in the 'svg' if sending SVG.`\n )\n );\n }\n\n let callResponse = false;\n\n // Call the before request functions\n callResponse = doCallbacks(beforeRequest, request, response, {\n id,\n uniqueId,\n type,\n body\n });\n\n // Do callbacks\n if (benchmark) {\n console.log('Do callbacks:', stopCounter(), 'ms.');\n }\n\n // Block the request if one of a callbacks failed\n if (callResponse !== true) {\n return response.send(callResponse);\n }\n\n let connectionAborted = false;\n\n // In case the connection is closed, force to abort further actions\n request.socket.on('close', () => {\n connectionAborted = true;\n });\n\n log(4, `[export] Got an incoming HTTP request ${uniqueId}.`);\n\n body.constr = (typeof body.constr === 'string' && body.constr) || 'chart';\n\n // Gather and organize options from the payload\n const requestOptions = {\n export: {\n instr,\n type,\n constr: body.constr[0].toLowerCase() + body.constr.substr(1),\n height: body.height,\n width: body.width,\n scale: body.scale || defaultOptions.export.scale,\n globalOptions: isCorrectJSON(body.globalOptions, true),\n themeOptions: isCorrectJSON(body.themeOptions, true)\n },\n customCode: {\n allowCodeExecution: getAllowCodeExecution(),\n allowFileResources: false,\n resources: isCorrectJSON(body.resources, true),\n callback: body.callback,\n customCode: body.customCode\n }\n };\n\n // Organize options\n if (benchmark) {\n console.log('Organize options:', stopCounter(), 'ms.');\n }\n\n if (instr) {\n // Stringify JSON with options\n requestOptions.export.instr = optionsStringify(\n instr,\n requestOptions.customCode.allowCodeExecution\n );\n\n // Stringify JSON with options\n if (benchmark) {\n console.log('Stringify JSON with options:', stopCounter(), 'ms.');\n }\n }\n\n // Merge the request options into default ones\n const options = mergeConfigOptions(defaultOptions, requestOptions);\n\n // Merge config options\n if (benchmark) {\n console.log('Merge config options:', stopCounter(), 'ms.');\n }\n\n // Save the JSON if exists\n options.export.options = instr;\n\n // Lastly, add the server specific arguments into options as payload\n options.payload = {\n svg: body.svg || false,\n b64: body.b64 || false,\n dataOptions: isCorrectJSON(body.dataOptions, true),\n noDownload: body.noDownload || false,\n requestId: uniqueId\n };\n\n // Setting payload\n if (benchmark) {\n console.log('Setting payload:', stopCounter(), 'ms.');\n }\n\n // Test xlink:href elements from payload's SVG\n if (body.svg && isPrivateRangeUrlFound(options.payload.svg)) {\n return response\n .status(400)\n .send(\n 'SVG potentially contain at least one forbidden URL in xlink:href element.'\n );\n }\n\n // Check URL range\n if (benchmark) {\n console.log('Check URL range:', stopCounter(), 'ms.');\n }\n\n // Start the export process\n startExport(options, (info, error) => {\n // Remove the close event from the socket\n request.socket.removeAllListeners('close');\n\n // After Puppeteer exporting\n if (benchmark) {\n console.log('After Puppeteer exporting:', stopCounter(), 'ms.', '\\n');\n }\n\n // If the connection was closed, do nothing\n if (connectionAborted) {\n return log(\n 3,\n clearText(\n `[export] The client closed the connection before the chart was done\n processing.`\n )\n );\n }\n\n // If error, return it\n if (error) {\n log(\n 1,\n clearText(\n `[export] Work: ${uniqueId} could not be completed, sending:\n ${error}`\n )\n );\n return response.status(400).send(error.message);\n }\n\n // If data is missing, return the error\n if (!info || !info.data) {\n log(\n 1,\n clearText(\n `[export] Unexpected return from chart generation, please check your\n data Request: ${uniqueId} is ${info.data}.`\n )\n );\n return response\n .status(400)\n .send(\n 'Unexpected return from chart generation, please check your data.'\n );\n }\n\n // Get the type from options\n type = info.options.export.type;\n\n // The after request callbacks\n doCallbacks(afterRequest, request, response, { id, body: info.data });\n\n if (info.data) {\n // If only base64 is required, return it\n if (body.b64) {\n // Check if it is already base64 or a raw SVG\n if (type === 'pdf') {\n return response.send(\n Buffer.from(info.data, 'utf8').toString('base64')\n );\n }\n return response.send(info.data);\n }\n\n // Set correct content type\n response.header('Content-Type', reversedMime[type] || 'image/png');\n\n // Decide whether to download or not chart file\n if (!body.noDownload) {\n response.attachment(\n `${request.params.filename || 'chart'}.${type || 'png'}`\n );\n }\n\n // If SVG, return plain content\n return type === 'svg'\n ? response.send(info.data)\n : response.send(Buffer.from(info.data, 'base64'));\n }\n });\n};\n\nexport default (app) => {\n app.post('/', exportHandler);\n app.post('/:filename', exportHandler);\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { promises as fsPromises } from 'fs';\nimport { posix } from 'path';\n\nimport bodyParser from 'body-parser';\nimport cors from 'cors';\nimport express from 'express';\nimport multer from 'multer';\nimport http from 'http';\nimport https from 'https';\n\nimport { log } from '../logger.js';\nimport rateLimit from './rate_limit.js';\nimport { __dirname } from '../utils.js';\n\nimport healthRoute from './routes/health.js';\nimport exportRoutes from './routes/export.js';\nimport vswitchRoute from './routes/change_hc_version.js';\nimport uiRoute from './routes/ui.js';\n\n// Create express app\nconst app = express();\n\n// Disable the X-Powered-By header\napp.disable('x-powered-by');\n\n// Enable CORS support\napp.use(cors());\n\n// Enable parsing of form data (files) with Multer package\nconst storage = multer.memoryStorage();\nconst upload = multer({\n storage,\n limits: {\n fieldsSize: '50MB'\n }\n});\n\napp.use(upload.any());\n\n// Enable body parser\napp.use(bodyParser.json({ limit: '50mb' }));\napp.use(bodyParser.urlencoded({ extended: true, limit: '50mb' }));\napp.use(bodyParser.urlencoded({ extended: false, limit: '50mb' }));\n\n/**\n * Error handler function.\n *\n * @param {object} error - An error object.\n * @return {string} - An error message.\n */\nconst errorHandler = (error) => log(1, `[server] Socket error: ${error}`);\n\n/**\n * Attaches error handlers for a server.\n *\n * @param {object} server - The http/https server.\n */\nconst attachErrorHandlers = (server) => {\n server.on('clientError', errorHandler);\n server.on('error', errorHandler);\n server.on('connection', (socket) =>\n socket.on('error', (error) => errorHandler(error, socket))\n );\n};\n\nexport const startServer = async (serverConfig) => {\n // Stop if not enabled\n if (!serverConfig.enable) {\n return false;\n }\n\n // // Get the pool\n // const pool = getPool();\n\n // // Try to create browser instance before starting the server\n // const resource = await pool.acquire();\n\n // // If not found, throw an error\n // if (!resource.browser) {\n // log(1, `[server] Could not acquire browser instance.`);\n // process.exit(1);\n // }\n\n // // Release the resource\n // pool.release(resource);\n\n // Listen HTTP server\n if (!serverConfig.ssl.enable && !serverConfig.ssl.force) {\n // Main server instance (HTTP)\n const httpServer = http.createServer(app);\n // Attach error handlers and listen to the server\n attachErrorHandlers(httpServer);\n // Listen\n httpServer.listen(serverConfig.port, serverConfig.host);\n\n log(\n 3,\n `[server] Started HTTP server on ${serverConfig.host}:${serverConfig.port}.`\n );\n }\n\n // Listen HTTPS server\n if (serverConfig.ssl.enable) {\n // Set up an SSL server also\n let key, cert;\n\n try {\n // Get the SSL key\n key = await fsPromises.readFile(\n posix.join(serverConfig.ssl.certPath, 'server.key'),\n 'utf8'\n );\n\n // Get the SSL certificate\n cert = await fsPromises.readFile(\n posix.join(serverConfig.ssl.certPath, 'server.crt'),\n 'utf8'\n );\n } catch (error) {\n log(\n 1,\n `[server] Unable to load key/certificate from ${serverConfig.ssl.certPath}.`\n );\n }\n\n if (key && cert) {\n // Main server instance (HTTPS)\n const httpsServer = https.createServer(app);\n // Attach error handlers and listen to the server\n attachErrorHandlers(httpsServer);\n // Listen\n httpsServer.listen(serverConfig.ssl.port, serverConfig.host);\n\n log(\n 3,\n `[server] Started HTTPS server on ${serverConfig.host}:${serverConfig.ssl.port}.`\n );\n }\n }\n\n // Enable the rate limiter if config says so\n if (\n serverConfig.rateLimiting &&\n serverConfig.rateLimiting.enable &&\n ![0, NaN].includes(serverConfig.rateLimiting.maxRequests)\n ) {\n rateLimit(app, serverConfig.rateLimiting);\n }\n\n // Set up static folder's route\n app.use(express.static(posix.join(__dirname, 'public')));\n\n // Set up routes\n healthRoute(app);\n exportRoutes(app);\n uiRoute(app);\n vswitchRoute(app);\n};\n\n/**\n * Returns the express instance.\n */\nexport const getExpress = () => {\n return express;\n};\n\n/**\n * Returns the app instance.\n */\nexport const getApp = () => {\n return app;\n};\n\n/**\n * Adds a middleware to the server.\n *\n * @param {object} path - An endpoint path to add middlewares to.\n * @param {Array} middlewares - An unlimited number of middlewares to use\n * against the specific endpoint.\n */\nexport const use = (path, ...middlewares) => {\n app.use(path, ...middlewares);\n};\n\n/**\n * Adds a get route to the server.\n *\n * @param {object} path - An endpoint path to add middlewares to.\n * @param {Array} middlewares - An unlimited number of middlewares to use\n * against the specific endpoint for GET method.\n */\nexport const get = (path, ...middlewares) => {\n app.get(path, ...middlewares);\n};\n\n/**\n * Adds a post route to the server.\n *\n * @param {object} path - An endpoint path to add middlewares to.\n * @param {Array} middlewares - An unlimited number of middlewares to use\n * against the specific endpoint for POST method.\n */\nexport const post = (path, ...middlewares) => {\n app.post(path, ...middlewares);\n};\n\n/**\n * Forcefully enables rate limiting.\n *\n * @param {object} limitConfig - The options object for the rate limiter\n * configuration.\n */\nexport const enableRateLimiting = (limitConfig) => {\n return rateLimit(app, limitConfig);\n};\n\nexport default {\n startServer,\n getExpress,\n getApp,\n use,\n get,\n post,\n enableRateLimiting\n};\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport { join } from 'path';\n\nimport { __dirname } from '../../utils.js';\n/**\n * Adds the / route for a UI when enabled for the export server\n */\nexport default (app) =>\n !app\n ? false\n : app.get('/', (request, response) => {\n response.sendFile(join(__dirname, 'public', 'index.html'));\n });\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\nimport cache from '../../cache.js';\n\n/**\n * Adds a route that can be used to change the HC version on the server\n * TODO: Add auth token and connect to API\n */\nexport default (app) =>\n !app\n ? false\n : app.post('/change_hc_version/:newVersion', async (request, response) => {\n const ctoken = process.env.HIGHCHARTS_ADMIN_TOKEN;\n\n if (!ctoken || !ctoken.length) {\n return response.send({\n error: true,\n message:\n 'Server not configured to do run-time version changes: HIGHCHARTS_ADMIN_TOKEN not set'\n });\n }\n\n const token = request.get('hc-auth');\n\n if (!token || token !== ctoken) {\n return response.send({\n error: true,\n message: 'Invalid or missing token: set token in the hc-auth header'\n });\n }\n\n const newVersion = request.params.newVersion;\n\n if (newVersion) {\n try {\n // eslint-disable-next-line import/no-named-as-default-member\n await cache.updateVersion(newVersion);\n } catch (e) {\n response.send({\n error: true,\n message: e\n });\n }\n\n response.send({\n version: cache.version()\n });\n } else {\n response.send({\n error: true,\n message: 'No new version supplied'\n });\n }\n });\n","/*******************************************************************************\n\nHighcharts Export Server\n\nCopyright (c) 2016-2023, Highsoft\n\nLicenced under the MIT licence.\n\nAdditionally a valid Highcharts license is required for use.\n\nSee LICENSE file in root for details.\n\n*******************************************************************************/\n\n// Add the main directory in the global object\nimport 'colors';\n\nimport server, { startServer } from './server/server.js';\nimport {\n setAllowCodeExecution,\n batchExport,\n singleExport,\n startExport\n} from './chart.js';\nimport { mapToNewConfig, setOptions } from './config.js';\nimport { log, setLogLevel, enableFileLogging } from './logger.js';\nimport { killPool, init } from './pool.js';\nimport { checkCache } from './cache.js';\n\nexport default {\n log,\n mapToNewConfig,\n setOptions,\n singleExport,\n startExport,\n batchExport,\n server,\n startServer,\n killPool,\n initPool: async (options = {}) => {\n // Set the allowCodeExecution per export module scope\n setAllowCodeExecution(\n options.customCode && options.customCode.allowCodeExecution\n );\n\n // Set the log level\n setLogLevel(options.logging && parseInt(options.logging.level));\n\n // Set the log file path and name\n if (options.logging && options.logging.dest) {\n enableFileLogging(\n options.logging.dest,\n options.logging.file || 'highcharts-export-server.log'\n );\n }\n\n // Check if cache needs to be updated\n await checkCache(options.highcharts || { version: 'latest' });\n\n // Init the pool\n await init({\n pool: options.pool || {\n initialWorkers: 1,\n maxWorkers: 1\n },\n puppeteerArgs: options.puppeteer?.args || []\n });\n\n // Return updated options\n return options;\n }\n};\n"],"names":["dotenv","config","defaultConfig","puppeteer","args","value","type","description","highcharts","version","envLink","cdnURL","coreScripts","modules","indicators","scripts","forceFetch","export","infile","instr","options","outfile","constr","defaultHeight","defaultWidth","defaultScale","height","width","scale","globalOptions","themeOptions","batch","customCode","allowCodeExecution","allowFileResources","callback","resources","loadConfig","createConfig","server","enable","cliName","host","port","ssl","force","certPath","rateLimiting","maxRequests","window","delay","trustProxy","skipKey","skipToken","pool","initialWorkers","maxWorkers","workLimit","queueSize","timeoutThreshold","acquireTimeout","reaper","benchmarking","listenToProcessExits","logging","level","file","dest","ui","route","other","noLogo","payload","join","absoluteProps","nestedArgs","createNestedArgs","obj","propChain","Object","keys","forEach","k","includes","entry","substring","toConsole","toFile","pathCreated","levelsDesc","title","color","listeners","key","option","entries","log","newLevel","texts","length","prefix","Date","toString","split","trim","fn","existsSync","mkdirSync","appendFile","concat","error","console","apply","undefined","__dirname","fileURLToPath","URL","url","clearText","text","rule","replacer","replaceAll","fixType","formats","outType","pop","find","t","handleResources","allowedProps","handledResources","correctResources","endsWith","isCorrectJSON","readFileSync","notice","files","propName","map","item","data","parsedData","JSON","parse","stringify","deepCopy","copy","Array","isArray","prototype","hasOwnProperty","call","optionsStringify","allowFunctions","name","startsWith","printUsage","bold","yellow","cycleCategories","categories","descName","green","i","blue","category","toUpperCase","red","toBoolean","wrapAround","replace","rateLimit","app","limitConfig","msg","rateOptions","max","limiter","windowMs","delayMs","handler","request","response","format","json","status","send","message","default","skip","query","access_token","use","async","fetch","requestOptions","Promise","resolve","reject","protocol","https","http","getProtocol","get","res","on","chunk","cachePath","cache","activeManifest","sources","hcVersion","appliedConfig","extractVersion","substr","indexOf","fetchScript","script","proxyAgent","agent","timeout","process","env","statusCode","updateCache","sourcePath","customScripts","allScripts","c","m","proxyHost","proxyPort","HttpsProxyAgent","fetchedModules","all","writeFileSync","checkCache","manifestPath","requestUpdate","manifest","moduleMap","numberOfModules","some","moduleName","newManifest","saveConfigToManifest","cache$1","newVersion","assign","RANDOM_PID","randomBytes","PUPPETEER_DIR","path","minimalArgs","template","fs","browser","newPage","p","setContent","addScriptTag","evaluate","setupHighcharts","err","$eval","element","errorMessage","_displayErrors","innerHTML","close","connected","__basedir","setAsConfig","page","chart","triggerExport","puppeteerExport","injectedResources","clearInjected","dispose","scriptsToRemove","document","getElementsByTagName","stylesToRemove","linksToRemove","remove","exportBench","exportOptions","requestAnimationFrame","displayErrors","debugger","d","svgBench","isSVG","setPageBench","svgTemplate","strInj","setContentBench","resBench","js","push","content","isLocal","cssBench","css","cssImports","match","cssImportPath","addStyleTag","size","chartHeight","baseVal","chartWidth","parseFloat","Highcharts","charts","vpBench","viewportHeight","Math","ceil","viewportWidth","setViewport","deviceScaleFactor","zoomCallback","body","style","zoom","margin","x","y","getBoundingClientRect","trunc","getClipRegion","round","expBenchmark","outerHTML","createSVG","encoding","clip","race","screenshot","omitBackground","setTimeout","Error","createImage","pdf","createPDF","oldCharts","oldChart","destroy","shift","puppeteerArgs","performedExports","exportAttempts","timeSpent","droppedExports","spentAverage","poolConfig","factory","create","id","uuid","s","getTime","browserNewPage","isClosed","workCount","random","validate","workerHandle","logLevel","init","allArgs","tryCount","open","launch","headless","userDataDir","e","createBrowser","killPool","code","exit","Pool","min","createRetryIntervalMillis","createTimeoutMillis","acquireTimeoutMillis","destroyTimeoutMillis","idleTimeoutMillis","reapIntervalMillis","propagateCreateError","eventId","resource","initialResources","acquire","promise","release","destroyed","postWork","fail","getPoolInfo","workStart","result","exportTime","available","borrowed","pending","spareResourceCapacity","pool$1","packageVersion","npm_package_version","serverStartTime","generalOptions","getOptions","mergeConfigOptions","newOptions","mergedOptions","updateDefaultConfig","configObj","customObj","customValue","numEnvVal","el","initOptions","items","startExport","settings","endCallback","svg","initExportSettings","exportAsString","readFile","doStraightInject","doExport","findChartSize","exporting","precision","multiplier","pow","roundNumber","sourceHeight","sourceWidth","chartJson","customCodeOptions","allowCodeExecutionScoped","enabled","optionsName","then","catch","requestId","stringToExport","chartJSON","reversedMime","png","jpeg","gif","requestsCounter","beforeRequest","afterRequest","doCallbacks","callbacks","uniqueId","callResponse","exportHandler","start","hrtime","bigint","measureTime","defaultOptions","headers","connection","remoteAddress","connectionAborted","socket","toLowerCase","b64","dataOptions","noDownload","ipRegEx","info","removeAllListeners","Buffer","from","header","attachment","params","filename","express","disable","cors","storage","multer","memoryStorage","upload","limits","fieldsSize","any","bodyParser","limit","urlencoded","extended","errorHandler","attachErrorHandlers","startServer","serverConfig","httpServer","createServer","listen","cert","fsPromises","posix","httpsServer","NaN","static","bootTime","uptime","floor","highchartsVersion","averageProcessingTime","failedExports","sucessRatio","healthRoute","post","exportRoutes","sendFile","uiRoute","ctoken","HIGHCHARTS_ADMIN_TOKEN","token","vswitchRoute","getExpress","getApp","middlewares","enableRateLimiting","index","mapToNewConfig","oldOptions","propertiesChain","reduce","prop","setOptions","userOptions","configIndex","findIndex","arg","fileName","loadConfigFile","pairArgumentValue","singleExport","batchExport","batchFunctions","pair","initPool","parseInt","logDest","logFile","enableFileLogging"],"mappings":"snBAiBAA,EAAOC,SAIA,MAAMC,EAAgB,CAC3BC,UAAW,CACTC,KAAM,CACJC,MAAO,GACPC,KAAM,WACNC,YAAa,6CAGjBC,WAAY,CACVC,QAAS,CACPJ,MAAO,SACPK,QAAS,qBACTJ,KAAM,SACNC,YAAa,8BAEfI,OAAQ,CACNN,MAAO,+BACPK,QAAS,iBACTJ,KAAM,SACNC,YAAa,6CAEfK,YAAa,CACXF,QAAS,0BACTL,MAAO,CAAC,aAAc,kBAAmB,iBACzCC,KAAM,WACNC,YAAa,qCAEfM,QAAS,CACPH,QAAS,qBACTL,MAAO,CACL,QACA,MACA,QACA,YACA,cACA,uBACA,gBACA,uBACA,eACA,QACA,OACA,mBACA,eACA,cACA,UACA,UACA,WACA,UACA,YACA,cACA,YACA,sBACA,SACA,SACA,WACA,YACA,eACA,SACA,eACA,YACA,kBACA,SACA,cACA,mBACA,eACA,cACA,eACA,cACA,cACA,WACA,eACA,WACA,SACA,OACA,WACA,YACA,SACA,qBACA,aACA,WACA,WACA,WACA,WACA,eACA,UACA,kBACA,oBACA,cAEFC,KAAM,WACNC,YAAa,gCAEfO,WAAY,CACVJ,QAAS,wBACTL,MAAO,CAAC,kBACRC,KAAM,WACNC,YAAa,mCAEfQ,QAAS,CACPV,MAAO,CACL,wEACA,kGAEFC,KAAM,WACNC,YACE,qEAEJS,WAAY,CACVN,QAAS,yBACTL,OAAO,EACPC,KAAM,UACNC,YACE,oEAGNU,OAAQ,CACNC,OAAQ,CACNb,OAAO,EACPC,KAAM,SACNC,YACE,8FAEJY,MAAO,CACLd,OAAO,EACPC,KAAM,SACNC,YACE,iFAEJa,QAAS,CACPf,OAAO,EACPC,KAAM,SACNC,YAAa,oCAEfc,QAAS,CACPhB,OAAO,EACPC,KAAM,SACNC,YACE,2FAEJD,KAAM,CACJI,QAAS,sBACTL,MAAO,MACPC,KAAM,SACNC,YACE,sEAEJe,OAAQ,CACNZ,QAAS,wBACTL,MAAO,QACPC,KAAM,SACNC,YACE,6EAEJgB,cAAe,CACbb,QAAS,wBACTL,MAAO,IACPC,KAAM,SACNC,YACE,gFAEJiB,aAA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"} \ No newline at end of file diff --git a/lib/schemas/config.js b/lib/schemas/config.js index d648323e..4380b8f5 100644 --- a/lib/schemas/config.js +++ b/lib/schemas/config.js @@ -67,6 +67,7 @@ export const defaultConfig = { 'tilemap', 'timeline', 'treemap', + 'treegraph', 'item-series', 'drilldown', 'histogram-bellcurve',