hotstaq 0.6.21 → 0.6.23

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,2 +1,2 @@
1
- var HotStaqWeb;(()=>{var __webpack_modules__={230:t=>{t.exports="object"==typeof self?self.FormData:window.FormData},646:function(t){t.exports=function(){"use strict";function t(t){for(var e=1;e<arguments.length;e++){var n=arguments[e];for(var o in n)t[o]=n[o]}return t}return function e(n,o){function r(e,r,s){if("undefined"!=typeof document){"number"==typeof(s=t({},o,s)).expires&&(s.expires=new Date(Date.now()+864e5*s.expires)),s.expires&&(s.expires=s.expires.toUTCString()),e=encodeURIComponent(e).replace(/%(2[346B]|5E|60|7C)/g,decodeURIComponent).replace(/[()]/g,escape);var i="";for(var l in s)s[l]&&(i+="; "+l,!0!==s[l]&&(i+="="+s[l].split(";")[0]));return document.cookie=e+"="+n.write(r,e)+i}}return Object.create({set:r,get:function(t){if("undefined"!=typeof document&&(!arguments.length||t)){for(var e=document.cookie?document.cookie.split("; "):[],o={},r=0;r<e.length;r++){var s=e[r].split("="),i=s.slice(1).join("=");try{var l=decodeURIComponent(s[0]);if(o[l]=n.read(i,l),t===l)break}catch(t){}}return t?o[t]:o}},remove:function(e,n){r(e,"",t({},n,{expires:-1}))},withAttributes:function(n){return e(this.converter,t({},this.attributes,n))},withConverter:function(n){return e(t({},this.converter,n),this.attributes)}},{attributes:{value:Object.freeze(o)},converter:{value:Object.freeze(n)}})}({read:function(t){return'"'===t[0]&&(t=t.slice(1,-1)),t.replace(/(%[\dA-F]{2})+/gi,decodeURIComponent)},write:function(t){return encodeURIComponent(t).replace(/%(2[346BF]|3[AC-F]|40|5[BDE]|60|7[BCD])/g,decodeURIComponent)}},{path:"/"})}()},300:(t,e)=>{"use strict";var n=function(){if("undefined"!=typeof self)return self;if("undefined"!=typeof window)return window;if(void 0!==n)return n;throw new Error("unable to locate global object")}();t.exports=e=n.fetch,n.fetch&&(e.default=n.fetch.bind(n)),e.Headers=n.Headers,e.Request=n.Request,e.Response=n.Response},507:function(__unused_webpack_module,exports,__webpack_require__){"use strict";var __awaiter=this&&this.__awaiter||function(t,e,n,o){return new(n||(n=Promise))((function(r,s){function i(t){try{a(o.next(t))}catch(t){s(t)}}function l(t){try{a(o.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?r(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((o=o.apply(t,e||[])).next())}))},__importDefault=this&&this.__importDefault||function(t){return t&&t.__esModule?t:{default:t}};Object.defineProperty(exports,"__esModule",{value:!0}),exports.Hot=exports.DeveloperMode=void 0;const HotFile_1=__webpack_require__(732),HotStaq_1=__webpack_require__(243),HotTestElement_1=__webpack_require__(967),js_cookie_1=__importDefault(__webpack_require__(646)),node_fetch_1=__importDefault(__webpack_require__(300));var DeveloperMode;!function(t){t[t.Production=0]="Production",t[t.Development=1]="Development"}(DeveloperMode=exports.DeveloperMode||(exports.DeveloperMode={}));class Hot{static include(t,e=null){return __awaiter(this,void 0,void 0,(function*(){if(!0===HotStaq_1.HotStaq.isWeb&&"string"==typeof t){const e=t.toLowerCase();e.indexOf(".hott")>-1&&e.indexOf("nahfam")<0&&(t+="?hstqserve=nahfam")}Hot.echo(yield Hot.getFile(t,e))}))}static includeJS(file){return __awaiter(this,void 0,void 0,(function*(){const res=yield Hot.httpRequest(file),output=yield res.text();!0===HotStaq_1.HotStaq.isWeb?eval.apply(window,[output]):eval(output)}))}static runFile(t,e=null){return __awaiter(this,void 0,void 0,(function*(){let n=Hot.CurrentPage.processor.getFile(t);n.page=this.CurrentPage;let o=yield n.process(e);Hot.echo(o)}))}static getFile(t,e=null){return __awaiter(this,void 0,void 0,(function*(){let n=null;return"string"==typeof t?(n=new HotFile_1.HotFile,!0===HotStaq_1.HotStaq.isWeb?n.url=t:n.localFile=t):n=t,yield n.load(),n.page=this.CurrentPage,yield n.process(e)}))}static apiCall(t,e=null,n="POST",o={}){return __awaiter(this,void 0,void 0,(function*(){let r=null;if(null==Hot.CurrentPage)throw new Error("Current page is null!");if(null==Hot.CurrentPage.processor)throw new Error("Current page's processor is null!");if(null==Hot.CurrentPage.processor.api)throw new Error("Current page's processor api is null! Did you forget to set the API name or URL?");return null!=Hot.CurrentPage.processor.api&&(r=yield Hot.CurrentPage.processor.api.makeCall(t,e,n,o)),r}))}static jsonRequest(t,e=null,n="POST"){return __awaiter(this,void 0,void 0,(function*(){try{let o={method:n,headers:{Accept:"application/json","Content-Type":"application/json"}};"POST"===n&&(o.body=JSON.stringify(e));let r=yield(0,node_fetch_1.default)(t,o);if(!1===r.ok)throw new Error(`${r.status}: ${r.statusText}`);return yield r.json()}catch(e){return JSON.stringify({error:`${e.message} - Could not fetch ${t}`})}}))}static httpRequest(t,e){return __awaiter(this,void 0,void 0,(function*(){return yield(0,node_fetch_1.default)(t,e)}))}static echo(t){Hot.Output+=t}static displayCSS(){for(let t=0;t<Hot.CSS.length;t++){let e=Hot.CSS[t],n=Hot.cssStr;n=n.replace(/\%CSS_FILE\%/g,e),Hot.echo(n)}}static displayJSFiles(){for(let t=0;t<Hot.JSFiles.length;t++){let e=Hot.JSFiles[t],n=Hot.jsFileStr;n=n.replace(/\%JS_FILE\%/g,e),Hot.echo(n)}}static displayJSScripts(){for(let t=0;t<Hot.JSScripts.length;t++){let e=Hot.JSScripts[t],n=Hot.jsScriptsStr;n=n.replace(/\%JS_CODE\%/g,e),Hot.echo(n)}}}exports.Hot=Hot,Hot.CurrentPage=null,Hot.Arguments=null,Hot.DeveloperMode=DeveloperMode,Hot.HotTestElement=HotTestElement_1.HotTestElement,Hot.Mode=DeveloperMode.Production,Hot.API=null,Hot.TesterAPI=null,Hot.Output="",Hot.Data={},Hot.Cookies=js_cookie_1.default,Hot.PublicKeys={},Hot.cssStr='<link rel = "stylesheet" href = "%CSS_FILE%" />',Hot.CSS=[],Hot.JSFiles=[],Hot.JSScripts=[],Hot.jsFileStr='<script type = "text/javascript" src = "%JS_FILE%"><\/script>',Hot.jsScriptsStr='<script type = "text/javascript">%JS_CODE%<\/script>'},233:function(t,e,n){"use strict";var o=this&&this.__awaiter||function(t,e,n,o){return new(n||(n=Promise))((function(r,s){function i(t){try{a(o.next(t))}catch(t){s(t)}}function l(t){try{a(o.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?r(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((o=o.apply(t,e||[])).next())}))},r=this&&this.__importDefault||function(t){return t&&t.__esModule?t:{default:t}};Object.defineProperty(e,"__esModule",{value:!0}),e.HotAPI=e.EventExecutionType=void 0;const s=r(n(300)),i=r(n(230)),l=n(810),a=n(761),u=n(70);var h;!function(t){t[t.HotRoute=0]="HotRoute",t[t.HotMethod=1]="HotMethod",t[t.HotAPI=2]="HotAPI"}(h=e.EventExecutionType||(e.EventExecutionType={})),e.HotAPI=class{constructor(t,e=null,n=null){this.connection=e,this.description="",this.baseUrl=t,this.createFunctions=!0,this.executeEventsUsing=h.HotRoute,this.db=n,this.authCredentials=null,this.userAuth=null,this.routes={},this.onPreRegister=null,this.onPostRegister=null}setDBSchema(t){if(null==this.connection.api)throw new Error("No API has been set!");if(null==this.connection.api.db)throw new Error(`No database has been set for API base url ${this.connection.api.baseUrl}`);this.connection.api.db.schema=t}getDB(){if(null==this.connection.api.db)throw new Error(`No database has been set for API base url ${this.connection.api.baseUrl}`);return this.connection.api.db}getDBSchema(){if(null==this.connection.api.db)throw new Error(`No database has been set for API base url ${this.connection.api.baseUrl}`);return this.connection.api.db.schema}addRoute(t,e=null,n=null){let o="";if(t instanceof a.HotRoute?(o=t.route,this.routes[t.route]=t):(o=t,null==this.routes[o]&&(this.routes[o]=new a.HotRoute(this.connection,o)),e instanceof u.HotRouteMethod?this.routes[o].addMethod(e):this.routes[o].addMethod(new u.HotRouteMethod(this.routes[o],e,n))),this.routes[o].connection=this.connection,!0===this.createFunctions){let t=this[o];null==t&&(t={});for(let e=0;e<this.routes[o].methods.length;e++){let n=this.routes[o],r=this.routes[o].methods[e];t[r.name]=(t,e)=>{let o=r.type,s="";""!==n.version&&(s+=`/${n.version}`),""!==n.route&&(s+=`/${n.route}`),""!==r.name&&(s+=`/${r.name}`);let i=null;if(null!=this.authCredentials&&(i=this.authCredentials),null!=r.authCredentials?i=r.authCredentials:null!=r.route.authCredentials&&(i=r.route.authCredentials),null==i&&"undefined"!=typeof Hot&&null!=Hot&&null!=Hot.API&&null!=Hot.API[n.route]&&null!=Hot.API[n.route].authCredentials&&(i=Hot.API[n.route].authCredentials),null!=i)for(let e in i){let n=i[e];null==t[e]&&(t[e]=n)}let l=[s,t,o,e];return this.makeCall.apply(this,l)}}this[o]=t}}registerRoute(t){return o(this,void 0,void 0,(function*(){this.connection instanceof l.HotServer&&(yield this.connection.registerRoute(t))}))}registerRoutes(){return o(this,void 0,void 0,(function*(){for(let t in this.routes){let e=this.routes[t];yield this.registerRoute(e)}}))}makeCall(t,e,n="POST",r={}){return o(this,void 0,void 0,(function*(){let l=this.baseUrl;if(n=n.toUpperCase(),"/"===l[l.length-1]&&(l=l.substr(0,l.length-1)),"/"!==t[0]&&(l+="/"),l+=t,Object.keys(r).length>0){if("POST"!==n)throw new Error("To upload files, you must set the httpMethod to POST.");const o=new i.default;for(let t in r)o.append(t,r[t]);let a=yield(0,s.default)(l,{method:"POST",body:o}),u=yield a.json();return null==e.hotstaq&&(e.hotstaq={}),null==e.hotstaq.uploads&&(e.hotstaq.uploads={}),e.hotstaq.uploads.uploadId=u.hotstaq.uploads.uploadId,yield this.makeCall(t,e,n)}let a={method:n,headers:{Accept:"application/json","Content-Type":"application/json"}};return"GET"!==n&&"HEAD"!==n&&(a.body=JSON.stringify(e)),new Promise(((t,e)=>{(0,s.default)(l,a).then((e=>o(this,void 0,void 0,(function*(){let n=yield e.json();t(n)})))).catch((t=>{throw new Error(`${l}: ${t.message}`)}))}))}))}}},957:(t,e,n)=>{"use strict";Object.defineProperty(e,"__esModule",{value:!0}),e.HotClient=void 0;const o=n(810);e.HotClient=class{constructor(t){this.processor=t,this.api=null,this.testerAPI=null,this.type=o.HotServerType.HTTP,this.logger=t.logger}}},260:function(t,e,n){"use strict";var o=this&&this.__awaiter||function(t,e,n,o){return new(n||(n=Promise))((function(r,s){function i(t){try{a(o.next(t))}catch(t){s(t)}}function l(t){try{a(o.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?r(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((o=o.apply(t,e||[])).next())}))};Object.defineProperty(e,"__esModule",{value:!0}),e.HotComponent=void 0;const r=n(243);e.HotComponent=class{constructor(t,e=null){t instanceof r.HotStaq||null==t?(this.processor=t,this.htmlElements=[],this.name="",this.tag="",this.api=null,this.elementOptions=void 0,this.observedAttributes=[],this.type="",this.value=null,this.inner=null,this.events={}):(this.processor=t.processor,this.htmlElements=t.htmlElements||[],this.name=t.name||"",this.tag=t.tag||this.name,this.api=t.api||null,this.elementOptions=t.elementOptions||void 0,this.observedAttributes=t.observedAttributes||[],this.type=t.type||"",this.value=t.value||null,this.inner=t.inner||null,this.events={}),null!=e&&(this.api=e)}onCreated(t){return o(this,void 0,void 0,(function*(){return t}))}}},732:function(t,e,n){"use strict";var o=this&&this.__createBinding||(Object.create?function(t,e,n,o){void 0===o&&(o=n),Object.defineProperty(t,o,{enumerable:!0,get:function(){return e[n]}})}:function(t,e,n,o){void 0===o&&(o=n),t[o]=e[n]}),r=this&&this.__setModuleDefault||(Object.create?function(t,e){Object.defineProperty(t,"default",{enumerable:!0,value:e})}:function(t,e){t.default=e}),s=this&&this.__importStar||function(t){if(t&&t.__esModule)return t;var e={};if(null!=t)for(var n in t)"default"!==n&&Object.prototype.hasOwnProperty.call(t,n)&&o(e,t,n);return r(e,t),e},i=this&&this.__awaiter||function(t,e,n,o){return new(n||(n=Promise))((function(r,s){function i(t){try{a(o.next(t))}catch(t){s(t)}}function l(t){try{a(o.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?r(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((o=o.apply(t,e||[])).next())}))},l=this&&this.__importDefault||function(t){return t&&t.__esModule?t:{default:t}};Object.defineProperty(e,"__esModule",{value:!0}),e.HotFile=void 0;const a=s(n(351)),u=l(n(300)),h=n(507);class c{constructor(t={}){this.page=t.page||null,this.name=t.name||"",this.url=t.url||"",this.localFile=t.localFile||"",this.content=t.content||"",this.throwAllErrors=t.throwAllErrors||!1}setContent(t){this.content=t}getContent(){return this.content}static httpGet(t){return i(this,void 0,void 0,(function*(){try{let e=yield(0,u.default)(t);if(!1===e.ok)throw new Error(`${e.status}: ${e.statusText}`);return yield e.text()}catch(e){return JSON.stringify({error:`${e.message} - Could not fetch ${t}`})}}))}loadUrl(){return i(this,void 0,void 0,(function*(){return this.content=yield c.httpGet(this.url),this.content}))}loadLocalFile(){return i(this,void 0,void 0,(function*(){return new Promise(((t,e)=>{a.readFile(this.localFile,((e,n)=>{if(null!=e)throw e;let o=n.toString();this.content=o,t(this.content)}))}))}))}load(){return i(this,void 0,void 0,(function*(){let t="";return""!==this.url&&(t=yield this.loadUrl()),""!==this.localFile&&(t=yield this.loadLocalFile()),t}))}static processContent(t,e,n,o,r=2,s=2){let i=e.exec(t),l=0,a="";for(;null!=i;){let u=i.index-r,h=e.lastIndex+s,c=t.substr(l,u-l);l=h,a+=o(c),a+=n(i[0]),i=e.exec(t)}return a+=o(t.substr(l)),a}static processNestedContent(t,e,n,o,r,s,i=2,l=1){let a=t.indexOf(e),u=0,h=t.indexOf(o,a),c="";for(;a>-1;){let p=t.indexOf(n,a),d=0;if(""!==o){let e=t.lastIndexOf(o,p-l);for(;e>-1&&e!==h;)e=t.lastIndexOf(o,e-l),d++}if(d>0){let o=t.indexOf(n,p+l),r=o;for(;o>-1&&d>0&&!(r<0)&&!(t.lastIndexOf(e,r-l)>o);)o=r,r=t.indexOf(n,o+l),d--;p=o}c+=s(t.substr(u,a-u)),c+=r(t.substr(a+i,p-(a+i))),a=t.indexOf(e,p+l),h=t.indexOf(o,a),u=p+l}return c+=s(t.substr(u)),c}static parseContent(t,e,n=null){null==n&&(n={outputCommands:!0,allowStringify:!0});let o="JSON.stringify (",r=")";return!1===n.allowStringify&&(o="",r=""),c.processContent(t,new RegExp("(?=\\<\\*)([\\s\\S]*?)(?=\\*\\>)","g"),(t=>`${t=t.substr(2)}`),(t=>{if(""===t)return"";let s=c.processNestedContent(t,"!{","}","{",(t=>`*&&%*%@#@!${t}*&!#%@!@*!`),(t=>t)),i=c.processNestedContent(s,"STR{","}","{",(t=>{let s="";return s=!0===n.outputCommands?`*&&%*%@#@!echoOutput (JSON.stringify(${t}), ${e});*&!#%@!@*!`:`*&&%*%@#@!${o}${t}${r}, ${e});*&!#%@!@*!`,s}),(t=>t),4,1),l=c.processNestedContent(i,"${","}","{",(t=>{let o="";return!0===n.outputCommands?(o=`*&&%*%@#@!try { Hot.echo (${t}); }catch (ex){Hot.echo ("");}*&!#%@!@*!`,!0===e&&(o=`*&&%*%@#@!Hot.echo (${t});*&!#%@!@*!`)):o=`*&&%*%@#@!${t}*&!#%@!@*!`,o}),(t=>t)),a="";h.Hot.Mode===h.DeveloperMode.Production&&(a=c.processNestedContent(l,"?(",")","(",(t=>""),(t=>t))),h.Hot.Mode===h.DeveloperMode.Development&&(a=c.processNestedContent(l,"?(",")","(",(t=>{let e="";try{JSON.parse(t),e=!0===n.allowStringify?JSON.stringify(t):`${t}`}catch(n){e=`${t}`}let o="";if(!0===n.outputCommands)o=`*&&%*%@#@!{\nconst testElm = createTestElement (${e});\nHot.echo (\`data-test-object-name = "\${testElm.name}" data-test-object-func = "\${testElm.func}" data-test-object-value = "\${testElm.value}"\`);\n}*&!#%@!@*!\n`;else{let t=t=>{let e=null;try{let n=t;if("string"==typeof t&&(n=JSON.parse(t)),"string"==typeof n&&(e=new h.Hot.HotTestElement(n)),n instanceof Array&&(e=new h.Hot.HotTestElement(n[0],n[1],n[2])),null!=n.name&&(e=new h.Hot.HotTestElement(n)),null!=h.Hot.CurrentPage.testElements[e.name])throw new Error(`Test element ${e.name} already exists!`)}catch(e){throw new Error(`Error processing test element ${t} in ${h.Hot.CurrentPage.name}. Error: ${e.message}`)}return e};const n=t(e);o=`*&&%*%@#@!data-test-object-name = "${n.name}" data-test-object-func = "${n.func}" data-test-object-value = "${n.value}"*&!#%@!@*!`}return o}),(t=>t)));let u=c.processNestedContent(a,"*&&%*%@#@!","*&!#%@!@*!","*&&%*%@#@!",(t=>t),(t=>{let o="";o=!0===n.allowStringify?JSON.stringify(t):t;let r="";return r=!0===n.outputCommands?`echoOutput (${o}, ${e});\n`:o,r}),"*&&%*%@#@!".length,"*&!#%@!@*!".length);return u=u.replace(/\*\&\&\%\*\%\@\#\@\!/g,""),u=u.replace(/\*\&\!\#\%\@\!\@\*\!/g,""),u}),0)}process(t=null){return i(this,void 0,void 0,(function*(){let e=this.content;h.Hot.Mode=this.page.processor.mode,h.Hot.Arguments=t,h.Hot.CurrentPage=this.page,h.Hot.PublicKeys=this.page.processor.publicKeys,h.Hot.API=this.page.getAPI(),h.Hot.TesterAPI=this.page.getTesterAPI();let n=c.parseContent(e,this.throwAllErrors),o=null;try{let e="\n\t\t\tvar Hot = arguments[0];\n\t\t\tvar PassedHotFile = arguments[1];\n\n\t\t\t";if("string"==typeof t)throw new Error("The passing arguments cannot be a string!");for(let n in t){let o="",r=t[n];o=`var ${n} = ${JSON.stringify(r)};\n`,e+=o}let r=this.name;""===r&&(r=this.localFile),""===r&&(r=this.url),e+='\n\n\t\t\tfunction echoOutput (content, throwErrors)\n\t\t\t{\n\t\t\t\tif (throwErrors == null)\n\t\t\t\t\tthrowErrors = true;\n\n\t\t\t\tif (throwErrors === true)\n\t\t\t\t{\n\t\t\t\t\tHot.echo (content);\n\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\ttry\n\t\t\t\t{\n\t\t\t\t\tHot.echo (content);\n\t\t\t\t}\n\t\t\t\tcatch (ex)\n\t\t\t\t{\n\t\t\t\t\tHot.echo ("");\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tfunction createTestElement (foundStr)\n\t\t\t{\n\t\t\t\tlet testElm = null;\n\n\t\t\t\ttry\n\t\t\t\t{\n\t\t\t\t\tlet obj = foundStr;\n\n\t\t\t\t\tif (typeof (foundStr) === "string")\n\t\t\t\t\t\tobj = JSON.parse (foundStr);\n\n\t\t\t\t\tif (typeof (obj) === "string")\n\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj);\n\n\t\t\t\t\tif (obj instanceof Array)\n\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj[0], obj[1], obj[2]);\n\n\t\t\t\t\tif (obj["name"] != null)\n\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj);\n\n\t\t\t\t\tif (Hot.CurrentPage.testElements[testElm.name] != null)\n\t\t\t\t\t\tthrow new Error (`Test element ${testElm.name} already exists!`);\n\n\t\t\t\t\tHot.CurrentPage.addTestElement (testElm);\n\t\t\t\t}\n\t\t\t\tcatch (ex)\n\t\t\t\t{\n\t\t\t\t\tthrow new Error (\n\t\t\t`Error processing test element ${foundStr} in ${Hot.CurrentPage.name}. Error: ${ex.message}`\n\t\t\t\t\t\t);\n\t\t\t\t}\n\n\t\t\t\treturn (testElm);\n\t\t\t}\n\n\t\t\tasync function runContent (CurrentHotFile)\n\t\t\t{\n',e+=n,e+="\n\t\t\t}\n\n\t\t\treturn (runContent (PassedHotFile).then (() =>\n\t\t\t{\n\t\t\t\treturn ({\n\t\t\t\t\t\thot: Hot,\n\t\t\t\t\t\toutput: Hot.Output,\n\t\t\t\t\t\tdata: JSON.stringify (Hot.Data)\n\t\t\t\t\t});\n\t\t\t}));";let s=new Function(e);o=yield s.apply(this,[h.Hot,this])}catch(t){throw SyntaxError,t}h.Hot.Data=o.hot.Data;let r=o.output;return h.Hot.Output="",r}))}}e.HotFile=c},802:(t,e)=>{"use strict";var n;Object.defineProperty(e,"__esModule",{value:!0}),e.HotLog=e.HotLogLevel=void 0,function(t){t[t.Info=0]="Info",t[t.Warning=1]="Warning",t[t.Error=2]="Error",t[t.Verbose=3]="Verbose",t[t.All=4]="All",t[t.None=5]="None"}(n=e.HotLogLevel||(e.HotLogLevel={})),e.HotLog=class{constructor(t=n.All){this.logLevel=t}log(t,e){this.logLevel===n.Verbose&&(t===n.Error&&this.error(e),t===n.Warning&&this.warning(e),t!==n.Info&&t!==n.Verbose||this.info(e)),this.logLevel===n.All&&(t===n.Error&&this.error(e),t===n.Warning&&this.warning(e),t===n.Info&&this.info(e)),this.logLevel===n.Error&&t===n.Error&&this.error(e),this.logLevel===n.Warning&&t===n.Warning&&this.warning(e),this.logLevel===n.Info&&t===n.Info&&this.info(e)}verbose(t){this.logLevel===n.Verbose&&console.info(t)}info(t){this.logLevel!==n.All&&this.logLevel!==n.Verbose&&this.logLevel!==n.Info||console.info(t)}warning(t){this.logLevel!==n.All&&this.logLevel!==n.Verbose&&this.logLevel!==n.Warning||console.warn(t)}error(t){if(this.logLevel===n.All||this.logLevel===n.Verbose||this.logLevel===n.Error){let e="";"string"==typeof t?e=t:(null!=t.message&&(e=t.message),null!=t.stack&&(e=t.stack)),console.error(e)}}}},136:function(t,e,n){"use strict";var o=this&&this.__awaiter||function(t,e,n,o){return new(n||(n=Promise))((function(r,s){function i(t){try{a(o.next(t))}catch(t){s(t)}}function l(t){try{a(o.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?r(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((o=o.apply(t,e||[])).next())}))};Object.defineProperty(e,"__esModule",{value:!0}),e.HotPage=void 0;const r=n(507),s=n(243);e.HotPage=class{constructor(t){t instanceof s.HotStaq?(this.processor=t,this.name="",this.testerName="",this.testerMap="",this.route="",this.files=[],this.testElements={},this.testPaths={}):(this.processor=t.processor,this.name=t.name||"",this.testerName=t.testerName||"",this.testerMap=t.testerMap||"",this.route=t.route||"",this.files=t.files||[],this.testElements=t.testElements||{},this.testPaths=t.testPaths||{})}addFile(t){return o(this,void 0,void 0,(function*(){t.page=this,this.files.push(t)}))}getAPI(){return this.processor.api}getTesterAPI(){return this.processor.testerAPI}load(t){return o(this,void 0,void 0,(function*(){for(let t=0;t<this.files.length;t++){let e=this.files[t];yield e.load()}}))}process(t=null){return o(this,void 0,void 0,(function*(){let e="";for(let n=0;n<this.files.length;n++){let o=this.files[n];r.Hot.Output="",o.page=this,e+=(yield o.process(t))}return e}))}addTestElement(t){if(null!=this.testElements[t.name])throw new Error(`Test element ${t.name} already exists!`);this.testElements[t.name]=t}getTestElement(t){if(null==this.testElements[t])throw new Error(`Test element ${t} doest not exist!`);return this.testElements[t]}createTestPath(t,e){if(null!=this.testPaths[t])throw new Error(`Test path ${t} already exists!`);this.testPaths[t]=e}}},761:(t,e,n)=>{"use strict";Object.defineProperty(e,"__esModule",{value:!0}),e.HotRoute=void 0;const o=n(70);class r{constructor(t,e,n=[]){this.onPreRegister=null,this.onRegister=null,this.onPostRegister=null,this.onAuthorizeUser=null,this.connection=t,this.logger=null,null!=this.connection&&null!=this.connection.processor&&(this.logger=this.connection.processor.logger),this.route=e,this.description="",this.version="v1",this.prefix="",this.authCredentials=null,this.methods=n,this.errors={not_authorized:r.createError("Not authorized."),no_server_execute_function:r.createError("Missing server execute function.")}}static createError(t){return{error:t}}addMethod(t,e=null,n=o.HTTPMethod.POST,r=null){"string"==typeof t&&(t=new o.HotRouteMethod(this,t,e,n,null,null,null,r)),t instanceof o.HotRouteMethod||(null==t.route&&(t.route=this),t=new o.HotRouteMethod(t)),this.methods.push(t)}getMethod(t){let e=null;for(let n=0;n<this.methods.length;n++){let o=this.methods[n];if(o.name===t){e=o;break}}return e}}e.HotRoute=r},70:(t,e,n)=>{"use strict";Object.defineProperty(e,"__esModule",{value:!0}),e.HotRouteMethod=e.HTTPMethod=void 0;const o=n(507),r=n(761),s=n(810);var i;!function(t){t.GET="get",t.POST="post",t.FILE_UPLOAD="file_upload_then_post_json"}(i=e.HTTPMethod||(e.HTTPMethod={})),e.HotRouteMethod=class{constructor(t,e="",n=null,l=i.POST,a=null,u=null,h=null,c=null){let p=null;if(t instanceof r.HotRoute)p=t;else{if(p=t.route,null!=t.type&&(l=t.type),null!=t.name&&(e=t.name),null!=t.description&&(this.description=t.description),null!=t.returns&&("string"==typeof t.returns?this.returns={type:"string",required:!0,description:t.returns}:this.returns=t.returns),null!=t.parameters){this.parameters={};for(let e in t.parameters){let n=t.parameters[e];"string"==typeof n?this.parameters[e]={type:n,required:!1,description:""}:(null==n.type&&(n.type="string"),this.parameters[e]=n)}}null!=t.authCredentials&&(h=t.authCredentials),null!=t.onServerExecute&&(n=t.onServerExecute),null!=t.onServerAuthorize&&(a=t.onServerAuthorize),null!=t.onRegister&&(u=t.onRegister),null!=t.onPostRegister&&(this.onPostRegister=t.onPostRegister),null!=t.onServerExecute&&(this.onServerExecute=t.onServerExecute),null!=t.onClientExecute&&(this.onClientExecute=t.onClientExecute),null!=t.testCases&&(c=t.testCases)}if(""===e)throw new Error("All route methods must have a name!");if(this.route=p,this.name=e,this.type=l,this.isRegistered=!1,this.executeSetup=!1,this.authCredentials=h,this.onServerAuthorize=a,this.onRegister=u,this.testCases={},this.route.connection.processor.mode===o.DeveloperMode.Development&&null!=c)if(c instanceof Array)for(let t=0;t<c.length;t++){let e=c[t];if("string"==typeof e){const n=e,o=c[t+1];this.addTestCase(n,o),t++}else this.addTestCase(e)}else for(let t in c){let e=c[t];this.addTestCase(e)}this.route.connection instanceof s.HotServer&&(this.onServerExecute=n)}addTestCase(t,e=null){if("string"==typeof t){const n=t,o=e;return void(this.testCases[n]={name:n,func:o})}if("function"==typeof t){const e=Object.keys(this.testCases).length,n=`${this.route.route}/${this.name} test case ${e}`,o=t;return void(this.testCases[n]={name:n,func:o})}const n=t;this.testCases[n.name]=n}}},810:function(t,e,n){"use strict";var o=this&&this.__awaiter||function(t,e,n,o){return new(n||(n=Promise))((function(r,s){function i(t){try{a(o.next(t))}catch(t){s(t)}}function l(t){try{a(o.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?r(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((o=o.apply(t,e||[])).next())}))};Object.defineProperty(e,"__esModule",{value:!0}),e.HotServer=e.HotServerType=void 0;const r=n(243);var s;!function(t){t[t.HTTP=0]="HTTP",t[t.WebSockets=1]="WebSockets",t[t.Generate=2]="Generate"}(s=e.HotServerType||(e.HotServerType={})),e.HotServer=class{constructor(t){t instanceof r.HotStaq?(this.processor=t,this.serverType="Server",this.api=null,this.listenAddress="0.0.0.0",this.ports={http:80,https:443},this.ssl={cert:"",key:"",ca:""},this.redirectHTTPtoHTTPS=!0,this.type=s.HTTP,this.logger=t.logger,this.secrets={}):(this.processor=t.processor,this.serverType=t.serverType||"Server",this.api=t.api||null,this.listenAddress=t.listenAddress||"0.0.0.0",this.ports=t.ports||{http:80,https:443},this.ssl=t.ssl||{cert:"",key:"",ca:""},this.redirectHTTPtoHTTPS=null==t.redirectHTTPtoHTTPS||t.redirectHTTPtoHTTPS,this.type=t.type||s.HTTP,this.logger=t.logger,this.secrets=t.secrets||{})}setAPI(t){return o(this,void 0,void 0,(function*(){this.processor.api=t,this.api=t}))}}},243:function(__unused_webpack_module,exports,__webpack_require__){"use strict";var __createBinding=this&&this.__createBinding||(Object.create?function(t,e,n,o){void 0===o&&(o=n),Object.defineProperty(t,o,{enumerable:!0,get:function(){return e[n]}})}:function(t,e,n,o){void 0===o&&(o=n),t[o]=e[n]}),__setModuleDefault=this&&this.__setModuleDefault||(Object.create?function(t,e){Object.defineProperty(t,"default",{enumerable:!0,value:e})}:function(t,e){t.default=e}),__importStar=this&&this.__importStar||function(t){if(t&&t.__esModule)return t;var e={};if(null!=t)for(var n in t)"default"!==n&&Object.prototype.hasOwnProperty.call(t,n)&&__createBinding(e,t,n);return __setModuleDefault(e,t),e},__awaiter=this&&this.__awaiter||function(t,e,n,o){return new(n||(n=Promise))((function(r,s){function i(t){try{a(o.next(t))}catch(t){s(t)}}function l(t){try{a(o.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?r(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((o=o.apply(t,e||[])).next())}))},__importDefault=this&&this.__importDefault||function(t){return t&&t.__esModule?t:{default:t}};Object.defineProperty(exports,"__esModule",{value:!0}),exports.HotStaq=void 0;const fs=__importStar(__webpack_require__(351)),ppath=__importStar(__webpack_require__(606)),node_fetch_1=__importDefault(__webpack_require__(300)),validate_npm_package_name_1=__importDefault(__webpack_require__(651)),HotPage_1=__webpack_require__(136),HotFile_1=__webpack_require__(732),HotComponent_1=__webpack_require__(260),HotLog_1=__webpack_require__(802),Hot_1=__webpack_require__(507),HotClient_1=__webpack_require__(957),HotTesterAPI_1=__webpack_require__(414),HotTestMap_1=__webpack_require__(985);var HotTesterMocha=null,HotTesterMochaSelenium=null,HotTestSeleniumDriver=null;class HotStaq{constructor(t={}){this.api=t.api||null,this.testerAPI=t.testerAPI||null,this.mode=t.mode||Hot_1.DeveloperMode.Production,this.pages=t.pages||{},this.components=t.components||{},this.files=t.files||{},this.hotSite=t.hotSite||null,this.apiContent="\n\t\t\t{\n\t\t\t\tvar %api_name% = %api_exported_name%.%api_name%;\n\t\t\t\tvar newHotClient = new HotClient (processor);\n\t\t\t\tvar newapi = new %api_name% (%base_url%, newHotClient);\n\t\t\t\tnewHotClient.api = newapi;\n\t\t\t\tprocessor.api = newapi;\n\t\t\t}",this.testerApiContent="\n\t\t\tvar HotTesterAPI = HotStaqWeb.HotTesterAPI;\n\t\t\tvar newHotTesterClient = new HotClient (processor);\n\t\t\tvar newtesterapi = new HotTesterAPI (%base_tester_url%, newHotTesterClient);\n\t\t\tnewHotTesterClient.testerAPI = newtesterapi;\n\t\t\tprocessor.testerAPI = newtesterapi;",this.pageContent='<!DOCTYPE html>\n<html>\n\n<head>\n\t<title>%title%</title>\n\n\t<script type = "text/javascript" src = "%hotstaq_js_src%"><\/script>\n\t<script type = "text/javascript">\n\t\twindow.HotStaq = HotStaqWeb.HotStaq;\n\t\twindow.HotClient = HotStaqWeb.HotClient;\n\t\twindow.HotAPI = HotStaqWeb.HotAPI;\n\t\twindow.Hot = HotStaqWeb.Hot;\n\t<\/script>\n\n%apis_to_load%\n\n\t<script type = "text/javascript">\n\t\tfunction hotstaq_startApp ()\n\t\t{\n\t\t\tlet tempMode = 0;\n\n\t\t\tif (window["Hot"] != null)\n\t\t\t\ttempMode = Hot.Mode;\n\n\t\t\t%load_hot_site%\n\n\t\t\tvar processor = new HotStaq ();\n\t\t\tvar promises = [];\n\t\t\t%developer_mode%\n\n\t\t\t%api_code%\n\n\t\t\t%public_secrets%\n\t\t\t%tester_api%\n\t\t\t%load_files%\n\n\t\t\tprocessor.mode = tempMode;\n\n\t\t\tPromise.all (promises).then (function ()\n\t\t\t\t{\n\t\t\t\t\tHotStaq.displayUrl ({\n\t\t\t\t\t\t\turl: "%url%",\n\t\t\t\t\t\t\tname: "%title%",\n\t\t\t\t\t\t\tprocessor: processor,\n\t\t\t\t\t\t\targs: %args%,\n\t\t\t\t\t\t\ttesterName: %tester_name%,\n\t\t\t\t\t\t\ttesterMap: %tester_map%,\n\t\t\t\t\t\t\ttesterAPIBaseUrl: %tester_api_base_url%,\n\t\t\t\t\t\t\ttesterLaunchpadUrl: %tester_launchpad_url%\n\t\t\t\t\t\t});\n\t\t\t\t});\n\t\t}\n\n\t\thotstaq_startApp ();\n\t<\/script>\n</head>\n\n<body>\n</body>\n\n</html>',this.logger=new HotLog_1.HotLog(HotLog_1.HotLogLevel.None),this.publicKeys={},this.testers={}}static parseBoolean(t){if("true"===(t=t.toLowerCase()))return!0;if("false"===t)return!1;if("yes"===t)return!0;if("no"===t)return!1;if("yep"===t)return!0;if("nah"===t)return!1;try{if(0!=parseInt(t))return!0}catch(t){}return!1}static getParam(t,e,n=!0,o=!0){let r=e[t];if(null==r&&!0===n&&!0===o)throw new Error(`Missing required parameter ${t}.`);if("string"==typeof r&&!0===n&&""===r&&!0===o)throw new Error(`Missing required parameter ${t}.`);return r}static getParamDefault(t,e,n){let o=e[t];return null==o||"string"==typeof o&&""===o?n:o}static executeError(t){if(null!=HotStaq.errors[t]){let e=HotStaq.errors[t].redirectToUrl;if(null!=e&&""!==e)return void(window.location.href=e);let n=HotStaq.errors[t].func;null!=n&&n(t)}}static wait(t){return __awaiter(this,void 0,void 0,(function*(){return yield new Promise(((e,n)=>{setTimeout((()=>{e()}),t)}))}))}addPage(t){this.pages[t.name]=t}getPage(t){return this.pages[t]}addFile(t){let e=t.name;""===e&&(e=t.localFile),""===e&&(e=t.url),this.files[e]=t}getFile(t){if(null==this.files[t])throw new Error(`Unable to find file ${t}`);return this.files[t]}static keepContext(t,e,n){return function(){var o=Array.prototype.slice.call(arguments);return null!=n&&o.push(n),null==e?t.apply(this,o):t.apply(e,o)}}addComponent(t,e=null,n){let o=this.api;null!=e&&(o=e);let r=new t(this,o);if(null!=this.components[r.tag])throw new Error(`Component ${r.tag} already exists!`);this.components[r.tag]={componentType:t,processor:this,api:o},this.registerComponent(r.tag,n)}static fixHTML(t){const e=t.replace(/ \/>/g,">").replace(/(<(area|base|br|col|command|embed|hr|img|input|keygen|link|meta|param|source|track|wbr).*?>)/g,"$1</$2>"),n=(new DOMParser).parseFromString(`<xml>${e}</xml>`,"text/xml");let o="";if(n.documentElement.children.length>0){const e=n.documentElement.children[0].tagName.toLowerCase();"tr"===e&&(t=`<table>${t}</table>`,o="tbody"),"th"===e&&(t=`<table>${t}</table>`,o=e)}return{fixedStr:t,querySelector:o}}registerComponent(t,e){if(null==t||""===t)throw new Error("All components must have a tag!");if(void 0!==customElements.get(t))return;let n=this.components;customElements.define(t,class extends HTMLElement{constructor(){super();let e=n[t];this.component=new e.componentType(e.processor,e.api)}looseParseFromString(t,e){e=e.replace(/ \/>/g,">").replace(/(<(area|base|br|col|command|embed|hr|img|input|keygen|link|meta|param|source|track|wbr).*?>)/g,"$1</$2>");const n=t.parseFromString("<xml>"+e+"</xml>","text/xml"),o=t.parseFromString("","text/html");for(let t of Array.from(n.documentElement.children))o.body.appendChild(t);for(let t of Array.from(o.querySelectorAll("area,base,br,col,command,embed,hr,img,input,keygen,link,meta,param,source,track,wbr")))t.outerHTML="<"+t.outerHTML.slice(1).split("<")[0];return o}get observedAttributes(){return this.component.observedAttributes}connectedCallback(){return __awaiter(this,void 0,void 0,(function*(){if(this.hotComponent=this.component,this.component.htmlElements=[this],this.component.inner=this.innerHTML,null!=this.component.handleAttributes)yield this.component.handleAttributes(this.attributes);else for(let t=0;t<this.attributes.length;t++){const e=this.attributes[t],n=e.name.toLowerCase(),o=e.value;if("id"===n&&(this.component.name=o),"name"===n&&(this.component.name=o),"value"===n&&(this.component.value=o),n.indexOf("hot-")>-1){const t=n.substring(4);this.component[t]=o}}if(null!=this.component.onPreOutput&&!1===(yield this.component.onPreOutput()))return;let t=yield this.component.output();null!=this.component.onPostOutput&&(t=yield this.component.onPostOutput(t));let e=[];"string"==typeof t?e.push({html:t}):e=t instanceof Array?t:[t];for(let t=0;t<e.length;t++){let n=e[t],o=n.html,r="";null!=n.addFunctionsTo&&(r=n.addFunctionsTo);let s=HotFile_1.HotFile.parseContent(o,!0,{outputCommands:!1});null!=this.component.onParsed&&(s=yield this.component.onParsed(s));let i={fixedStr:"",querySelector:""};i=null!=this.component.onFixHTML?yield this.component.onFixHTML(s):HotStaq.fixHTML(s);let l=null,a=null;if(l=null!=this.component.onParseDOM?yield this.component.onParseDOM(i.fixedStr):(new DOMParser).parseFromString(i.fixedStr,"text/html"),l.body.children.length<1)throw new Error(`No component output from ${this.component.name}`);if(l.body.children.length>1){let t=!0;for(let e=0;e<l.body.children.length;e++){let n=l.body.children[e];if(n instanceof HTMLElement&&"parsererror"===n.tagName.toLowerCase()){a=n,t=!1;break}}if(!0===t)throw new Error(`Only a single html element can come from component ${this.component.name}, multiple elements were detected.`)}a=""===i.querySelector?l.body.children[0]:l.querySelector(i.querySelector);let u=[];for(let t=this.children.length-1;t>-1;t--){let e=this.children[t];u.push(this.removeChild(e))}this.replaceWith(a),null!=this.component.click&&(a.onclick=this.component.click.bind(this.component));for(let t in this.component.events){let e=this.component.events[t];a.addEventListener(e.type,e.func,e.options)}let h=Object.getOwnPropertyNames(this.component.constructor.prototype);for(let t=0;t<h.length;t++){let e=h[t];if("constructor"!==e&&"function"==typeof this.component[e]){let t=!0;for(let n in HotComponent_1.HotComponent.prototype)if(e===n){t=!1;break}!0===t&&(a[e]=HotStaq.keepContext(this.component[e],this.component),""!==r)&&(document.querySelector(r)[e]=HotStaq.keepContext(this.component[e],this.component))}}null!=this.component.onPrePlace&&(a=yield this.component.onPrePlace(a));let c=yield this.component.onCreated(a);if(null!=this.component.onParentPlace&&(c.onParentPlace=this.component.onParentPlace),c.hotComponent=this.component,this.component.htmlElements.push(c),null!=n.parentSelector){let t=document.querySelector(n.parentSelector);c.parentElement.removeChild(c),t.appendChild(c),null!=c.onParentPlace&&(yield c.hotComponent.onParentPlace(t,c))}if(null!=n.placeHereParent){let t=c.parentNode,e=0;for(;e<10&&null!=t&&!(t instanceof HTMLHtmlElement);){let o=t.querySelectorAll(`hot-place-here[name="${n.placeHereParent}"]`);if(o.length>0){let t=o[0];c.parentNode.removeChild(c),t.appendChild(c),null!=c.onParentPlace&&(yield c.hotComponent.onParentPlace(t,c));break}if(o.length<1){let e=t.querySelectorAll(`[hot-place-here="${n.placeHereParent}"]`);if(e.length>0){let t=e[0];c.parentNode.removeChild(c),t.appendChild(c),null!=c.onParentPlace&&(yield c.hotComponent.onParentPlace(t,c));break}}t=t.parentNode,e++}}for(let t=0;t<u.length;t++){const e=u[t];c.appendChild(e),null!=e.onParentPlace&&(yield e.hotComponent.onParentPlace(c,e))}null!=this.component.onPostPlace&&(c=yield this.component.onPostPlace(c.parentNode,c))}}))}},e)}static addHtml(t,e){let n=null;if(n="string"==typeof t?document.querySelector(t):t,null==n)throw new Error(`Unable to find parent ${t}!`);let o=null;if("string"==typeof e){let t=HotStaq.fixHTML(e),o=(new DOMParser).parseFromString(t.fixedStr,"text/html"),r=null;r=""===t.querySelector?o.body.children:o.querySelector(t.querySelector).children;let s=[];for(let t=0;t<r.length;t++){let e=r[t];s.push(n.appendChild(e))}return s}return o=n.appendChild(e),o}static checkHotSiteName(t,e=!1){let n=(0,validate_npm_package_name_1.default)(t);return null!=n.errors&&n.errors.length>0&&(()=>{if(!0===e)throw new Error(`HotSite ${t} has an invalid name! The name cannot be empty and must have a valid NPM module name.`)})(),!0}static replaceKey(t,e,n){return t.replace(new RegExp(`\\$\\{${e}\\}`,"g"),n)}static getValueFromHotSiteObj(t,e){let n=null;if(null!=t){let o=t;for(let t=0;t<e.length;t++){let n=e[t];if(null==o[n]){o=null;break}o=o[n]}null!=o&&(n=o)}return n}processHotSite(){return __awaiter(this,void 0,void 0,(function*(){HotStaq.checkHotSiteName(this.hotSite.name,!0);let routes=this.hotSite.routes,testerUrl="http://127.0.0.1:8182",tester=null,driver=null;if(!1===HotStaq.isWeb&&this.mode===Hot_1.DeveloperMode.Development&&null!=this.hotSite.testing){let t=t=>{let e=!0;null!=t.createNewTester&&(e=t.createNewTester);let n="Tester";if(null!=t.tester&&(n=t.tester),null!=t.testerName&&(n=t.testerName),!0===e?(HotTesterMocha=Object(function(){var t=new Error("Cannot find module './HotTesterMocha'");throw t.code="MODULE_NOT_FOUND",t}()),HotTesterMochaSelenium=Object(function(){var t=new Error("Cannot find module './HotTesterMochaSelenium'");throw t.code="MODULE_NOT_FOUND",t}()),HotTestSeleniumDriver=Object(function(){var t=new Error("Cannot find module './HotTestSeleniumDriver'");throw t.code="MODULE_NOT_FOUND",t}()),""===t.testerAPIUrl&&(testerUrl=t.testerAPIUrl),"HotTestSeleniumDriver"===t.driver&&(driver=new HotTestSeleniumDriver),"HotTesterMocha"===t.tester&&(tester=new HotTesterMocha(this,n,testerUrl,driver)),"HotTesterMochaSelenium"===t.tester&&(tester=new HotTesterMochaSelenium(this,n,testerUrl))):tester=this.testers[n],null==tester.driver)throw new Error(`Tester ${n} does not have a driver set!`);null!=t.commandDelay&&(tester.driver.commandDelay=t.commandDelay)};null!=this.hotSite.testing.web&&t(this.hotSite.testing.web),null!=this.hotSite.testing.api&&t(this.hotSite.testing.api)}if(null!=routes)for(let t in routes){let e=routes[t],n=new HotFile_1.HotFile(e),o=new HotPage_1.HotPage({processor:this,name:e.name||"",route:t,files:[n]});if(null!=tester&&this.mode===Hot_1.DeveloperMode.Development){let t=e.name,n=null;if(null!=e.map){if("string"==typeof e.map){if(null==tester.testMaps[e.map])throw new Error(`Test map ${e.map} does not exist!`);tester.testMaps[t]=tester.testMaps[e.map]}else{n=new HotTestMap_1.HotTestMap;let t=null;if(e.map instanceof Array){t=[];for(let n=0;n<e.map.length;n++){let o=e.map[n];t.push(new HotTestMap_1.HotTestDestination(o))}}else{t={};for(let n in e.map){let o=e.map[n];t[n]=new HotTestMap_1.HotTestDestination(o)}}n.destinations=t}tester.testMaps[t]=n}null!=e.destinationOrder&&(tester.testMaps[t].destinationOrder=e.destinationOrder)}this.addPage(o)}if(null!=this.hotSite.apis)for(let t in this.hotSite.apis){let e=this.hotSite.apis[t];if(null!=e.map&&!1===HotStaq.isWeb&&this.mode===Hot_1.DeveloperMode.Development){let n=t,o=new HotTestMap_1.HotTestMap;o.destinations=[];for(let t=0;t<e.map.length;t++){let n=e.map[t];o.destinations.push(new HotTestMap_1.HotTestDestination(n))}if(null==tester)throw new Error("A tester was not created first! You must specify one in the CLI or in HotSite.json.");tester.testMaps[n]=o}}if(!0===HotStaq.isWeb)for(let key in this.hotSite.components){let component=this.hotSite.components[key],componentUrl=component.url,res=yield(0,node_fetch_1.default)(componentUrl),ComponentClass=eval(res);this.addComponent(ComponentClass)}null==this.hotSite.routes&&(this.hotSite.routes={});let disableFileLoading=!1;null!=this.hotSite.disableFileLoading&&(disableFileLoading=this.hotSite.disableFileLoading),!1===disableFileLoading?yield this.loadHotFiles(this.hotSite.files):this.logger.verbose("Hotsite has file loading disabled..."),null!=tester&&this.addTester(tester)}))}loadHotSite(t){return __awaiter(this,void 0,void 0,(function*(){let e="";if(!0===HotStaq.isWeb){this.logger.info(`Downloading HotSite ${t}`);let n=yield(0,node_fetch_1.default)(t);this.logger.info(`Downloaded site ${t}`),e=n.text()}else t=ppath.normalize(t),this.logger.info(`Accessing HotSite ${t}`),e=yield new Promise(((e,n)=>{fs.readFile(t,((n,o)=>{if(null!=n)throw n;let r=o.toString();this.logger.info(`Accessed site ${t}`),e(r)}))}));this.hotSite=JSON.parse(e),this.hotSite.hotsitePath=t}))}loadHotFiles(t,e=!1){return __awaiter(this,void 0,void 0,(function*(){this.logger.verbose("Loading Hott files...");for(let n in t){let o=t[n],r=null;HotStaq.isWeb,r=new HotFile_1.HotFile({name:n}),null!=o.url&&(r.url=o.url),!1===HotStaq.isWeb&&null!=o.localFile&&(r.localFile=o.localFile);let s=!0;if(null!=o.content&&(r.content=o.content,s=!1),!0===e&&(s=!0),!0===s){let t="";""!==r.url&&(t=r.url),""!==r.localFile&&(t=r.localFile),this.logger.verbose(`Loading Hott file: ${t}`),yield r.load(),this.logger.verbose(`Finished loading Hott file: ${t}`)}this.addFile(r)}this.logger.verbose("Finished loading Hott files...")}))}generateContent(t,e="",n="./",o="./js/HotStaq.min.js",r=!0,s=null){let i="",l="",a="";if(null!=this.hotSite){if(null!=this.hotSite.server.globalApi&&""!==this.hotSite.server.globalApi){const t=this.hotSite.apis[this.hotSite.server.globalApi];if(null==t)this.logger.warning(`API with name ${this.hotSite.server.globalApi} doesn't exist!`);else{let e=!0;if(null==t.jsapi&&(e=!1,this.logger.warning(`API with name ${this.hotSite.server.globalApi} doesn't have a jsapi set. Will not send js content to client.`)),null==t.libraryName&&(e=!1,this.logger.warning(`API with name ${this.hotSite.server.globalApi} doesn't have a libraryName set. Will not send js content to client.`)),null==t.apiName&&(e=!1,this.logger.warning(`API with name ${this.hotSite.server.globalApi} doesn't have a apiName set. Will not send js content to client.`)),!0===e){i+=`\t<script type = "text/javascript" src = "${t.jsapi}"><\/script>\n`;let e='""';null!=t.url&&(e=`"${t.url}"`),null!=this.api&&(e=`"${this.api.baseUrl}"`);let n=this.apiContent;n=n.replace(/\%api\_name\%/g,t.apiName),n=n.replace(/\%api\_exported\_name\%/g,t.libraryName),n=n.replace(/\%base\_url\%/g,e),l+=n}}}if(null!=this.hotSite.apis){let e=this.hotSite.routes[t];if(null!=e&&null!=e.api){let t=this.hotSite.apis[e.api];if(null==t)throw new Error(`Unable to find API ${e.api}`);let n=!0;if(null==t.jsapi&&(n=!1,this.logger.warning(`API with name ${e.api} doesn't have a jsapi set. Will not send js content to client.`)),null==t.libraryName&&(n=!1,this.logger.warning(`API with name ${e.api} doesn't have a libraryName set. Will not send js content to client.`)),null==t.apiName&&(n=!1,this.logger.warning(`API with name ${e.api} doesn't have a apiName set. Will not send js content to client.`)),!0===n){let e=t.jsapi;i+=`\t<script type = "text/javascript" src = "${e}"><\/script>\n`;let n='""';null!=t.url&&(n=`"${t.url}"`),null!=this.api&&(n=`"${this.api.baseUrl}"`);let o=this.apiContent;o=o.replace(/\%api\_name\%/g,t.apiName),o=o.replace(/\%api\_exported\_name\%/g,t.libraryName),o=o.replace(/\%base\_url\%/g,n),l+=o}}}if(null!=this.hotSite.server&&null!=this.hotSite.server.jsSrcPath&&(o=this.hotSite.server.jsSrcPath),null!=this.hotSite.publicKeys)for(let t in this.hotSite.publicKeys){let e,n=this.hotSite.publicKeys[t];if("string"==typeof n)e=JSON.stringify(n);else if(!1===HotStaq.isWeb){if(null!=this.api){if(null==this.api.connection)throw new Error("Cannot pass secrets from the API if there's no connection!");let o=this.api.connection;null!=n.passSecretFromAPI&&(e=JSON.stringify(o.secrets[t]))}if(null!=n.env){const t=n.env;e=JSON.stringify(process.env[t])}}a+=`processor.publicKeys["${t}"] = ${e};\n`}}let u=this.pageContent;return u=(u=>{let h="",c="";this.mode===Hot_1.DeveloperMode.Development&&(h="tempMode = HotStaqWeb.DeveloperMode.Development;",null!=this.hotSite&&null!=this.hotSite.testing&&null!=this.hotSite.testing.web&&(c=this.testerApiContent,null==this.hotSite.testing.web.testerAPIUrl&&(this.hotSite.testing.web.testerAPIUrl="http://127.0.0.1:8182"),c=c.replace(/\%base\_tester\_url\%/g,`"${this.hotSite.testing.web.testerAPIUrl}"`)));let p="";if(Object.keys(this.files).length>0){p+="var files = {};\n\n";for(let t in this.files){let e=this.files[t],n=`"${e.url}"`,o="";if(""!==e.content){let t=JSON.stringify(e.content);t=t.replace(new RegExp("\\<script","gmi"),'<scr" + "ipt'),t=t.replace(new RegExp("\\<\\/script","gmi"),'</scr" + "ipt'),o=`, "content": ${t}`}p+=`\t\t\tfiles["${t}"] = { "url": ${n}${o} };\n`}p+="\t\t\tpromises.push (processor.loadHotFiles (files));\n"}u=u.replace(/\%title\%/g,e),!0===r&&(u=u.replace(/\%args\%/g,"Hot.Arguments")),null!=s&&(u=u.replace(/\%args\%/g,JSON.stringify(s)));let d=t,m="",f="",g="Tester";return null!=this.hotSite&&(null!=this.hotSite.testing&&null!=this.hotSite.testing.web&&(null!=this.hotSite.testing.web.tester&&(g=this.hotSite.testing.web.tester),null!=this.hotSite.testing.web.testerName&&(g=this.hotSite.testing.web.testerName),null!=this.hotSite.testing.web.testerAPIUrl&&(m=this.hotSite.testing.web.testerAPIUrl),null!=this.hotSite.testing.web.launchpadUrl&&(f=this.hotSite.testing.web.launchpadUrl)),null!=this.hotSite.routes&&null!=this.hotSite.routes[t])&&(d=this.hotSite.routes[t].name),(u=(u=(u=(u=(u=(u=(u=(u=(u=(u=(u=(u=u.replace(/\%hotstaq\_js\_src\%/g,o)).replace(/\%developer\_mode\%/g,h)).replace(/\%tester\_api\%/g,c)).replace(/\%apis\_to\_load\%/g,i)).replace(/\%load\_hot\_site\%/g,"")).replace(/\%load\_files\%/g,p)).replace(/\%api\_code\%/g,l)).replace(/\%public\_secrets\%/g,a)).replace(/\%url\%/g,n)).replace(/\%tester\_name\%/g,`"${g}"`)).replace(/\%tester\_map\%/g,`"${d}"`)).replace(/\%tester\_api\_base\_url\%/g,`"${m}"`)).replace(/\%tester\_launchpad\_url\%/g,`"${f}"`)})(u),u}createExpressRoutes(t,e="./js/HotStaq.min.js"){for(let n in this.pages){let o=this.pages[n];const r=this.generateContent(o.route,o.name,o.files[0].url,e);t.get(o.route,((t,e)=>{e.send(r)}))}}addTester(t){this.testers[t.name]=t}getWebTestingMaps(){if(null==this.hotSite)throw new Error("No HotSite was loaded!");if(null==this.hotSite.testing)throw new Error("The HotSite does not have a testing object!");if(null==this.hotSite.testing.web)throw new Error("The HotSite does not have a testing web object!");if(null==this.hotSite.testing.web.maps)throw new Error("The HotSite testing object does not have any maps!");return this.hotSite.testing.web.maps}getAPITestingMaps(){if(null==this.hotSite)throw new Error("No HotSite was loaded!");if(null==this.hotSite.testing)throw new Error("The HotSite does not have a testing object!");if(null==this.hotSite.testing.api)throw new Error("The HotSite does not have a testing api object!");if(null==this.hotSite.testing.api.maps)throw new Error("The HotSite testing object does not have any maps!");return this.hotSite.testing.api.maps}getRouteKeyFromName(t){let e="";if(null!=this.hotSite&&null!=this.hotSite.routes)for(let n in this.hotSite.routes)if(this.hotSite.routes[n].name===t){e=n;break}return e}getRouteFromName(t){let e=null,n=this.getRouteKeyFromName(t);return""!==n&&(e=this.hotSite.routes[n]),e}executeTests(t,e){return __awaiter(this,void 0,void 0,(function*(){let n=this.testers[t];if(null==n)throw new Error(`Unable to execute tests. Tester ${t} does not exist!`);return n.execute(e)}))}executeAllWebTests(t){return __awaiter(this,void 0,void 0,(function*(){let e=this.getWebTestingMaps();if(null==this.testers[t])throw new Error(`Unable to execute tests. Tester ${t} does not exist!`);for(let n=0;n<e.length;n++){let o=e[n];yield this.executeTests(t,o)}}))}executeAllAPITests(t){return __awaiter(this,void 0,void 0,(function*(){let e=this.getAPITestingMaps();if(null==this.testers[t])throw new Error(`Unable to execute tests. Tester ${t} does not exist!`);for(let n=0;n<e.length;n++){let o=e[n];yield this.executeTests(t,o)}}))}process(t,e=null){return __awaiter(this,void 0,void 0,(function*(){let n=this.getPage(t);return yield n.process(e)}))}static processLocalFile(t,e=t,n=null){return __awaiter(this,void 0,void 0,(function*(){let o=new HotStaq,r=new HotFile_1.HotFile({localFile:t});yield r.load();let s=new HotPage_1.HotPage({processor:o,name:e,files:[r]});return o.addPage(s),yield o.process(e,n)}))}static processUrl(t){return __awaiter(this,void 0,void 0,(function*(){let e=new HotFile_1.HotFile({url:t.url});yield e.load();let n=new HotPage_1.HotPage({processor:t.processor,name:t.name,files:[e],testerName:t.testerName,testerMap:t.testerMap});return t.processor.addPage(n),yield t.processor.process(t.name,t.args)}))}static processContent(t){return __awaiter(this,void 0,void 0,(function*(){let e=new HotFile_1.HotFile({content:t.content});yield e.load();let n=new HotPage_1.HotPage({processor:t.processor,name:t.name,files:[e]});return t.processor.addPage(n),yield t.processor.process(t.name,t.args)}))}static onReady(t){"complete"===document.readyState||"interactive"===document.readyState?t():window.addEventListener("load",t)}static useOutput(t){return __awaiter(this,void 0,void 0,(function*(){let e=(new DOMParser).parseFromString(t,"text/html");document.getElementsByTagName("html")[0].innerHTML=e.getElementsByTagName("html")[0].innerHTML;let n=document.getElementsByTagName("script");if(n.length>0){let t=[];for(let e=0;e<n.length;e++)t.push(n[e]);for(let e=0;e<t.length;e++){let n=document.createElement("script");t[e].parentNode.appendChild(n),t[e].parentNode.removeChild(t[e]),yield new Promise(((o,r)=>{n.onload=()=>{o()};let s=!1;null!=t[e].getAttribute("src")&&""!==t[e].getAttribute("src")&&(n.setAttribute("src",t[e].getAttribute("src")),s=!0),null!=t[e].getAttribute("type")&&""!==t[e].getAttribute("type")&&n.setAttribute("type",t[e].getAttribute("type")),n.innerHTML=t[e].innerHTML,!1===s&&o()}))}}}))}static waitForTesters(){return __awaiter(this,void 0,void 0,(function*(){for(;!1===HotStaq.isReadyForTesting;)yield HotStaq.wait(10);null!=HotStaq.onReadyForTesting&&(yield HotStaq.onReadyForTesting())}))}static setupTesters(t,e){if(t.mode===Hot_1.DeveloperMode.Development&&null==t.testerAPI){null==e.testerAPIBaseUrl&&(e.testerAPIBaseUrl=""),""===e.testerAPIBaseUrl&&(e.testerAPIBaseUrl="http://127.0.0.1:8182");let n=new HotClient_1.HotClient(t),o=new HotTesterAPI_1.HotTesterAPI(e.testerAPIBaseUrl,n);o.connection.api=o,t.testerAPI=o}}static setupClientTesters(t){let e="";return t.mode===Hot_1.DeveloperMode.Development&&(e+='<script type = "text/javascript">\nfunction hotstaq_isDocumentReady ()\n{\nif (window["Hot"] != null)\n{\nif (Hot.Mode === HotStaqWeb.DeveloperMode.Development)\n{\nlet func = function ()\n\t{\n\t\tif (Hot.TesterAPI != null)\n\t\t{\n\t\t\tlet testPaths = {};\n\t\t\tlet testElements = JSON.stringify (Hot.CurrentPage.testElements);\n\t\t\tlet testMaps = JSON.stringify (Hot.CurrentPage.testMaps);\n\n\t\t\tfor (let key in Hot.CurrentPage.testPaths)\n\t\t\t{\n\t\t\t\tlet testPath = Hot.CurrentPage.testPaths[key];\n\n\t\t\t\ttestPaths[key] = testPath.toString ();\n\t\t\t}\n\n\t\t\tlet testPathsStr = JSON.stringify (testPaths);\n\n\t\t\tHot.TesterAPI.tester.pageLoaded ({\n\t\t\t\t\ttesterName: Hot.CurrentPage.testerName,\n\t\t\t\t\ttesterMap: Hot.CurrentPage.testerMap,\n\t\t\t\t\tpageName: Hot.CurrentPage.name,\n\t\t\t\t\ttestElements: testElements,\n\t\t\t\t\ttestPaths: testPathsStr\n\t\t\t\t}).then (function (resp)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (resp.error != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (resp.error !== "")\n\t\t\t\t\t\t\t\tthrow new Error (resp.error);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tHotStaqWeb.HotStaq.isReadyForTesting = true;\n\t\t\t\t\t});\n\t\t}\n\t};\n\nif ((document.readyState === "complete") || (document.readyState === "interactive"))\n\tfunc ();\nelse\n\tdocument.addEventListener ("DOMContentLoaded", func);\n}\n}\n}\n\nhotstaq_isDocumentReady ();\n<\/script>'),e}static displayUrl(t,e=null,n=null,o=null){return __awaiter(this,void 0,void 0,(function*(){return new Promise(((r,s)=>{HotStaq.onReady((()=>__awaiter(this,void 0,void 0,(function*(){let s={url:""};s.name=null==e?"string"==typeof t?t:t.name:e,""===s.name&&(s.name="string"==typeof t?t:t.name),"string"==typeof t?s.url=t:(s.url=t.url,null==n&&null!=t.processor&&(n=t.processor),null==o&&null!=t.args&&(o=t.args),null!=t.testerMap&&(s.testerMap=t.testerMap),null!=t.testerName&&(s.testerName=t.testerName),null!=t.testerAPIBaseUrl&&(s.testerAPIBaseUrl=t.testerAPIBaseUrl)),null==n&&(n=new HotStaq),HotStaq.setupTesters(n,s),s.processor=n,s.args=o,s.url.indexOf("hstqserve")<0&&(s.url+="?hstqserve=nahfam");let i=yield HotStaq.processUrl(s);i+=HotStaq.setupClientTesters(n),yield HotStaq.useOutput(i),r(n)}))))}))}))}static displayContent(t,e=null,n=null,o=null){return __awaiter(this,void 0,void 0,(function*(){return new Promise(((r,s)=>{HotStaq.onReady((()=>__awaiter(this,void 0,void 0,(function*(){let s={content:""};s.name=null==e?"string"==typeof t?"":t.name:e,""===s.name&&(s.name="string"==typeof t?"":t.name),"string"==typeof t?s.content=t:(s.content=t.content,null==n&&null!=t.processor&&(n=t.processor),null==o&&null!=t.args&&(o=t.args),null!=t.testerMap&&(s.testerMap=t.testerMap),null!=t.testerName&&(s.testerName=t.testerName),null!=t.testerAPIBaseUrl&&(s.testerAPIBaseUrl=t.testerAPIBaseUrl)),null==n&&(n=new HotStaq),HotStaq.setupTesters(n,s),s.processor=n,s.args=o;let i=yield HotStaq.processContent(s);yield HotStaq.useOutput(i),r(n)}))))}))}))}}if(exports.HotStaq=HotStaq,HotStaq.version="0.6.21",HotStaq.isWeb=!1,HotStaq.isReadyForTesting=!1,HotStaq.onReadyForTesting=null,HotStaq.errors={},"undefined"!=typeof document){let loadHotStaqSite=function(){let hotstaqElms=document.getElementsByTagName("hotstaq");if(HotStaq.isWeb=!0,void 0!==HotStaqWeb&&(window.HotStaq=HotStaqWeb.HotStaq,window.HotClient=HotStaqWeb.HotClient,window.HotAPI=HotStaqWeb.HotAPI,window.Hot=HotStaqWeb.Hot,window.HotComponent=HotStaqWeb.HotComponent),hotstaqElms.length>0){let hotstaqElm=hotstaqElms[0];setTimeout((function(){return __awaiter(this,void 0,void 0,(function*(){let getAttr=(t,e)=>{for(let n=0;n<e.length;n++){let o=e[n];if(null!=t.getAttribute(o))return t.getAttribute(o);if(null!=t.getAttribute(`data-${o}`))return t.getAttribute(`data-${o}`)}},loadPage=getAttr(hotstaqElm,["load-page","loadPage","src"])||"",router=getAttr(hotstaqElm,["router"])||"",name=getAttr(hotstaqElm,["name"])||"default",args=getAttr(hotstaqElm,["args"])||null,apiLibrary=getAttr(hotstaqElm,["api-library","apiLibrary"])||null,apiName=getAttr(hotstaqElm,["api-name","apiName"])||null,apiUrl=getAttr(hotstaqElm,["api-url","apiUrl"])||null,testerName=getAttr(hotstaqElm,["tester-name","testerName"])||"HotTesterMochaSelenium",testerMap=getAttr(hotstaqElm,["tester-map","testerMap"])||null,testerApiBaseUrl=getAttr(hotstaqElm,["tester-api-base-url","testerApiBaseUrl"])||null,testerLaunchpadUrl=getAttr(hotstaqElm,["tester-launchpad-url","testerLaunchpadUrl"])||null,dontReuseProcessor=!1,passRawUrl=!1,htmlSource=hotstaqElm.innerHTML||"",routerManager={},routerWildcards=[];null!=getAttr(hotstaqElm,["src"])&&(loadPage=getAttr(hotstaqElm,["src"])),null!=getAttr(hotstaqElm,["passRawUrl"])&&(passRawUrl=!0),null!=getAttr(hotstaqElm,["dont-reuse-processor","dontReuseProcessor"])&&(dontReuseProcessor=!0);let hotstaqErrors=document.getElementsByTagName("hotstaq-error");for(let t=0;t<hotstaqErrors.length;t++){let e=hotstaqErrors[t],n=getAttr(e,["status"]),o=getAttr(e,["unsupported-browser-redirect"]);null!=o?HotStaq.errors.unsupportedBrowser={redirectToUrl:o}:HotStaq.errors[`${n}`]={redirectToUrl:o}}try{eval("async () => {}")}catch(t){HotStaq.executeError("unsupportedBrowser")}if(""!==router){let t=document.getElementsByTagName("hotstaq-router");for(let e=0;e<t.length;e++){let n=t[e],o=getAttr(n,["name"]),r=getAttr(n,["serve-local","serveLocally"]);if(o===router){for(let t=0;t<n.childNodes.length;t++){let e=n.childNodes[t];if(e instanceof HTMLElement&&"ROUTE"===e.tagName.toUpperCase()){let t=getAttr(e,["path"]),n=getAttr(e,["redirect"]),o=getAttr(e,["base"]),r=getAttr(e,["src"]);t.indexOf("*")>-1&&routerWildcards.push(t),routerManager[t]={redirect:n||void 0,base:o||void 0,src:r||void 0}}}let t=window.location.pathname;if(null!=r){const e=r.toLowerCase();if("true"===e||"yes"===e||"1"===e){const e=t.lastIndexOf("/");e>-1&&(t=t.substring(e))}}if(routerWildcards.length>0)for(let e=0;e<routerWildcards.length;e++){let n=routerWildcards[e],o=n.replace("*","");if(t.indexOf(o)>-1){t=n;break}}if(null!=routerManager[t]){if(null!=routerManager[t].redirect)return void(window.location.href=routerManager[t].redirect);null!=routerManager[t].src&&(loadPage=routerManager[t].src)}break}}}args=null!=args?JSON.parse(args):Hot_1.Hot.Arguments;let hasHtmlSource=!1;if(""!==htmlSource){const t=htmlSource.replace(/\s/g,"");""!==t&&(hasHtmlSource=!0)}let tempMode=0;null!=window.Hot&&(tempMode=Hot_1.Hot.Mode);let processor=null;!1===dontReuseProcessor&&void 0!==Hot_1.Hot&&null!=Hot_1.Hot.CurrentPage&&null!=Hot_1.Hot.CurrentPage.processor&&(processor=Hot_1.Hot.CurrentPage.processor),null==processor&&(processor=new HotStaq),processor.mode=tempMode;let options={name,processor,args};if(""!==loadPage&&(!1===passRawUrl&&loadPage.indexOf("hstqserve")<0&&(loadPage+="?hstqserve=nahfam"),options.url=loadPage),null!=testerMap&&(options.testerMap=testerMap,options.testerName=testerName),null!=testerName&&(options.testerName=testerName),null!=testerApiBaseUrl&&(options.testerAPIBaseUrl=testerApiBaseUrl),null!=testerLaunchpadUrl&&(options.testerLaunchpadUrl=testerLaunchpadUrl),null!=apiName){let t=new HotClient_1.HotClient(processor);if(""===apiUrl)throw new Error("api-url was not set!");let e=window;null!=apiLibrary&&(e=window[apiLibrary]);let n=new e[apiName](apiUrl,t);n.connection.api=n,processor.api=n}if(!1===hasHtmlSource){if(""===loadPage)throw new Error("The hotstaq tag must have a src, HTML contents inside it, or a router set.");HotStaq.displayUrl(options)}else HotStaq.displayContent(options)}))}),50)}};window.ethereum22=window.ethereum,window.addEventListener("load",loadHotStaqSite)}},628:function(t,e,n){"use strict";var o=this&&this.__awaiter||function(t,e,n,o){return new(n||(n=Promise))((function(r,s){function i(t){try{a(o.next(t))}catch(t){s(t)}}function l(t){try{a(o.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?r(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((o=o.apply(t,e||[])).next())}))};Object.defineProperty(e,"__esModule",{value:!0}),e.HotTestDriver=void 0;const r=n(243),s=n(967);e.HotTestDriver=class{constructor(t,e=null){this.processor=t,this.page=e,this.commandDelay=20,this.persistentData={}}parseTestObject(t){let e=t.indexOf("*"),n="";e>-1&&(t=t.replace(/\*/,""),n="*");let o=`[data-test-object-name${n}='${t}']`;return e=t.indexOf(">"),e>-1&&(o=t=t.replace(/\>/,"")),o}wait(t){return o(this,void 0,void 0,(function*(){return yield new Promise(((e,n)=>{setTimeout((()=>{e()}),t)}))}))}print(t){return o(this,void 0,void 0,(function*(){process.stdout.write(t)}))}println(t){return o(this,void 0,void 0,(function*(){yield this.print(`${t}\n`)}))}assert(t,e=""){return o(this,void 0,void 0,(function*(){if(!t)throw new Error(e)}))}run(t){return o(this,void 0,void 0,(function*(){let e=[];for(let n=0;n<t.length;n++){let o=t[n],i=null,l="",a="";if("string"==typeof o){if(i=this.page.testElements[o],null==i)throw new Error(`HotTestDriver: Unable to find test element ${o}`);l=i.func,a=i.value}if(o instanceof Array){let t=o[0];i=this.page.testElements[t],null==i?(i=new s.HotTestElement(t),l=o[1],a=o[2]):(l=i.func,a=i.value,o.length>1&&(l=o[1]),o.length>2&&(a=o[2]))}i.func=l,i.value=a;let u=yield this.runCommand(i);yield r.HotStaq.wait(this.commandDelay),e.push(u)}return e}))}}},967:(t,e)=>{"use strict";Object.defineProperty(e,"__esModule",{value:!0}),e.HotTestElement=e.HotTestElementOptions=void 0,e.HotTestElementOptions=class{constructor(t={}){this.mustBeVisible=t.mustBeVisible||!0,this.ignoreMissingElementError=t.ignoreMissingElementError||!1}},e.HotTestElement=class{constructor(t,e="",n=null){"string"==typeof t?(this.name=t,this.func=e,this.value=n):(this.name=t.name,this.func=t.func||e,this.value=t.value||n)}}},985:(t,e)=>{"use strict";Object.defineProperty(e,"__esModule",{value:!0}),e.HotTestMap=e.HotTestDestination=void 0;class n{constructor(t="",e=!0){"string"==typeof t?(this.destination=t,this.autoStart=e):t instanceof n?(this.destination=t.destination,this.autoStart=t.autoStart):(this.destination=t.path,this.autoStart=t.autoStart)}}e.HotTestDestination=n,e.HotTestMap=class{constructor(t=[],e={},o=[]){if(t instanceof Array){this.destinations=[];for(let e=0;e<t.length;e++){let o=t[e];this.destinations.push(new n(o))}}else{this.destinations={};for(let e in t){let o=t[e];this.destinations[e]=new n(o)}}this.destinationOrder=o,this.pages=e}}},677:function(t,e,n){"use strict";var o=this&&this.__awaiter||function(t,e,n,o){return new(n||(n=Promise))((function(r,s){function i(t){try{a(o.next(t))}catch(t){s(t)}}function l(t){try{a(o.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?r(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((o=o.apply(t,e||[])).next())}))};Object.defineProperty(e,"__esModule",{value:!0}),e.HotTester=void 0;const r=n(243);class s{constructor(t,e,n,o,r={}){this.processor=t,this.name=e,this.baseUrl=n,this.testMaps=r,this.driver=o,this.finishedLoading=!1,this.hasBeenSetup=!1,this.hasBeenDestroyed=!1}waitForData(){return o(this,void 0,void 0,(function*(){for(;!1===this.finishedLoading;)yield r.HotStaq.wait(10)}))}getTestPage(t){return this.testMaps[t.mapName].pages[t.page]}getTestPath(t,e){return this.testMaps[t.mapName].pages[t.page].testPaths[e]}static interpretDestination(t,e){let n={mapName:t,page:"",api:"",paths:[]},o=e.destination.split(/\-\>/g),r=o[0];if(r.length<2)return null;if("/"===r[0]&&"/"===r[1])return null;let s=(t,e)=>{let n=t.indexOf(e),o="";return n>-1&&(o=t.substr(n+e.length),o=o.trim()),o};n.page=s(r,"page:"),n.api=s(r,"api:");for(let t=1;t<o.length;t++){let e=o[t],r={cmd:"",dest:"",path:""};e=e.trim(),r.dest=s(e,"dest:"),r.cmd=s(e,"cmd:"),r.path=s(e,"path:"),""==r.dest&&""==r.cmd&&""==r.path&&(r.path=e),n.paths.push(r)}return n}executeTestAPIPath(t,e,n,r=!1,s=!1){return o(this,void 0,void 0,(function*(){let o=!0;if(null==e)throw new Error(`Trying to access null method on destination map ${t.mapName}.`);!1===r&&null!=this.onTestAPIPathStart&&(o=yield this.onTestAPIPathStart(t,e,n,s));let i=null;if(!0===o){let t=e.testCases[n];if(null==t)throw new Error(`HotTester: Test case object ${n} does not exist!`);i=yield t.func(this.driver)}return!1===r&&null!=this.onTestAPIPathEnd&&(yield this.onTestAPIPathEnd(t,e,n,i,s)),i}))}executeTestAPIPaths(t){return o(this,void 0,void 0,(function*(){let e=[];if(null==this.testMaps[t.mapName])throw new Error(`HotTester: Map ${t.mapName} does not exist!`);if(null==this.processor.api)throw new Error("HotTester: Associated processor does not have an API!");let n=this.processor.api.routes[t.api];if(null==n)throw new Error(`HotTester: API does not have route ${t.api}!`);for(let o=0;o<t.paths.length;o+=2){let r=t.paths[o].path,s=n.getMethod(r),i=t.paths[o+1].path;if(null==s)throw new Error(`Unable to find method related to path ${r} in map ${t.mapName}`);let l=yield this.executeTestAPIPath(t,s,i);e.push(l)}return e}))}executeTestPagePath(t,e,n=!1,r=!1){return o(this,void 0,void 0,(function*(){let o=!0,s=this.testMaps[t.mapName];if(null==s)throw new Error(`HotTester: Map ${t.mapName} does not exist!`);let i=s.pages[t.page];if(null==i)throw new Error(`HotTester: Page ${t.page} does not exist!`);this.driver.page=i;let l=e.path,a=i.testPaths[l];!1===n&&null!=this.onTestPagePathStart&&(o=yield this.onTestPagePathStart(t,i,e,r));let u=null;if(!0===o){if(null==a)throw new Error(`HotTester: Test path ${l} does not have a function!`);u=yield a(this.driver)}return!1===n&&null!=this.onTestPagePathEnd&&(yield this.onTestPagePathEnd(t,a,u,r)),u}))}executeCommand(t,e,n,s){return o(this,void 0,void 0,(function*(){let i=(t,e,o)=>{let r=!1;n.cmd===e&&(r=!0);const s=n.cmd.indexOf("(");return s>-1&&n.cmd.substr(0,s)===e&&(r=!0),r},l=t=>{let e=[],n=t.match(/(?=\()(.*?)(?=\))/g);if(null!=n){let t=n[0];t=t.substr(2,t.length),e.push(t)}if(e.length<1)throw new Error(`HotTester: Command ${t} requires arguments, but none were supplied.`);return e},a=null,u=[];if(!0===i(n.cmd,"waitForTesterAPIData")&&(a=t=>o(this,void 0,void 0,(function*(){this.finishedLoading=!1,yield this.waitForData()}))),!0===i(n.cmd,"wait")&&(u=l(n.cmd),a=t=>o(this,void 0,void 0,(function*(){let e=parseInt(t[0]);yield r.HotStaq.wait(e)}))),!0===i(n.cmd,"url")&&(u=l(n.cmd),a=t=>o(this,void 0,void 0,(function*(){let e=t[0];yield this.driver.navigateToUrl(e)}))),!0===i(n.cmd,"print")&&(u=l(n.cmd),a=t=>o(this,void 0,void 0,(function*(){let e=t[0];yield this.driver.print(e)}))),!0===i(n.cmd,"println")&&(u=l(n.cmd),a=t=>o(this,void 0,void 0,(function*(){let e=t[0];yield this.driver.println(e)}))),!0===i(n.cmd,"waitForTestObject")&&(u=l(n.cmd),a=t=>o(this,void 0,void 0,(function*(){let e=JSON.parse(t[0]);yield this.driver.waitForTestElement(e)}))),null==a)throw new Error(`HotTester: Command ${n.cmd} does not exist!`);yield this.onCommand(t,e,n,s,u,a)}))}executeTestPagePaths(t,e=!1){return o(this,void 0,void 0,(function*(){let n=[],o=this.testMaps[t.mapName];if(null==o)throw new Error(`HotTester: Map ${t.mapName} does not exist!`);for(let r=0;r<t.paths.length;r++){let i=t.paths[r],l=null,a=o.pages[t.page];if(null==a)throw new Error(`HotTester: Page ${t.page} does not exist!`);if(""!==i.dest){if(o.destinations instanceof Array)throw new Error(`HotTester: When using type 'dest' in a destination string, all destinations in map ${t.mapName} must be named.`);let e=o.destinations[i.dest],n=s.interpretDestination(t.mapName,e);null!=n&&(l=yield this.executeTestPagePaths(n))}""!==i.cmd&&(yield this.executeCommand(t,a,i,i.cmd)),""!==i.path&&(l=yield this.executeTestPagePath(t,i,!1,e)),n.push(l)}return n}))}execute(t){return o(this,void 0,void 0,(function*(){let e=this.testMaps[t];if(null==e)throw new Error(`HotTester: Map ${t} does not exist!`);let n=this.processor.getRouteKeyFromName(t),r="";""!==n&&(r=`${this.baseUrl}${n}`);let i=(e,n="")=>o(this,void 0,void 0,(function*(){if(!1===e.autoStart)return;let o=s.interpretDestination(t,e),i=!1,l=!0;""!==o.page&&(i=!0),null!=this.setup&&!1===this.hasBeenSetup&&(yield this.setup(i,r,n),this.hasBeenSetup=!0,this.hasBeenDestroyed=!1),null!=this.onTestStart&&(l=yield this.onTestStart(o,r,n)),!0===l&&(""!==o.page&&(yield this.executeTestPagePaths(o)),""!==o.api&&(yield this.executeTestAPIPaths(o))),null!=this.onTestEnd&&(yield this.onTestEnd(o)),null!=this.destroy&&!1===this.hasBeenDestroyed&&(yield this.destroy(),this.hasBeenDestroyed=!0,this.hasBeenSetup=!1)}));if(e.destinations instanceof Array)for(let t=0;t<e.destinations.length;t++){let n=e.destinations[t];yield i(n)}else if(e.destinationOrder.length>0){let t=[];for(let n=0;n<e.destinationOrder.length;n++){let o=e.destinationOrder[n],r=e.destinations[o];if(null==r)throw new Error(`HotTester: Destination ${o} does not exist!`);t.push(o),yield i(r,o)}for(let n in e.destinations){let o=!0;for(let e=0;e<t.length;e++)if(t[e]===n){o=!1;break}if(!0===o){let t=e.destinations[n];yield i(t,n)}}}else for(let t in e.destinations){let n=e.destinations[t];yield i(n,t)}}))}}e.HotTester=s},414:function(__unused_webpack_module,exports,__webpack_require__){"use strict";var __awaiter=this&&this.__awaiter||function(t,e,n,o){return new(n||(n=Promise))((function(r,s){function i(t){try{a(o.next(t))}catch(t){s(t)}}function l(t){try{a(o.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?r(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((o=o.apply(t,e||[])).next())}))};Object.defineProperty(exports,"__esModule",{value:!0}),exports.HotTesterAPI=void 0;const HotAPI_1=__webpack_require__(233),HotRoute_1=__webpack_require__(761),HotRouteMethod_1=__webpack_require__(70);class HotTesterAPI extends HotAPI_1.HotAPI{constructor(t,e=null,n=null){super(t,e,n),this.executeEventsUsing=HotAPI_1.EventExecutionType.HotAPI;let o=new HotRoute_1.HotRoute(e,"tester");o.addMethod({name:"pageLoaded",onServerExecute:this.pageLoaded,parameters:{testerName:{required:!0,type:"string",description:"The name of the tester executing the test."},testerMap:{required:!0,type:"string",description:"The tester map executing the test."},pageName:{required:!0,type:"string",description:"The name of the page executing the test."},testElements:{required:!0,type:"array",description:"The test elements on the page."},testPaths:{required:!0,type:"array",description:"The test paths on the page."}},returns:"Returns true as an acknowledgement."}),o.addMethod({name:"executeTests",onServerExecute:this.executeTests,parameters:{testerName:{required:!0,type:"string",description:"The name of the tester executing the test."},testerMap:{required:!0,type:"object",description:"The tester map to execute."}},returns:"Returns true when tests are complete."}),o.addMethod({name:"heartbeat",type:HotRouteMethod_1.HTTPMethod.GET,onServerExecute:this.heartbeat,returns:"Returns true as an acknowledgement."}),this.addRoute(o)}pageLoaded(req,res,authorizedValue,jsonObj,queryObj){return __awaiter(this,void 0,void 0,(function*(){let testerObj={testerName:jsonObj.testerName,testerMap:jsonObj.testerMap,pageName:jsonObj.pageName,testElements:jsonObj.testElements,testPathsStrs:jsonObj.testPaths};for(let t in testerObj){let e=testerObj[t],n=!1;if(null==e&&(n=!0),""!=testerObj.testerName&&""!==testerObj.testerMap&&""!==testerObj.testElements&&""!==testerObj.testPathsStrs||(n=!0),!0===n)throw new Error(`TesterAPI: Object ${t} was not passed.`)}testerObj.testElements=JSON.parse(testerObj.testElements),testerObj.testPathsStrs=JSON.parse(testerObj.testPathsStrs);let testPaths={};for(let key in testerObj.testPathsStrs){let testPath=eval(testerObj.testPathsStrs[key]);testPaths[key]=testPath}let tester=this.connection.processor.testers[testerObj.testerName];if(null==tester)throw new Error(`TesterAPI: Tester ${testerObj.testerMap} does not exist!`);let testMap=tester.testMaps[testerObj.testerMap];if(null==testMap)throw new Error(`TesterAPI: Tester map ${testerObj.testerMap} does not exist!`);return testMap.pages[testerObj.pageName]={testElements:{},testPaths:{}},testMap.pages[testerObj.pageName].testElements=testerObj.testElements,testMap.pages[testerObj.pageName].testPaths=testPaths,tester.finishedLoading=!0,null!=tester.onFinishedLoading&&(yield tester.onFinishedLoading()),!0}))}executeTests(t,e,n,o,r){return __awaiter(this,void 0,void 0,(function*(){let t=o.testerName,e=o.testerMap;if(null==t||null==e)throw new Error("TesterAPI: Not all required json objects were passed.");if(""===t||""===e)throw new Error("TesterAPI: Not all required json objects were passed.");let n=this.connection;return null!=n.executeTests&&(yield n.executeTests(t,e)),!0}))}heartbeat(t,e,n,o,r){return __awaiter(this,void 0,void 0,(function*(){return!0}))}}exports.HotTesterAPI=HotTesterAPI},859:(t,e,n)=>{"use strict";Object.defineProperty(e,"__esModule",{value:!0});const o=n(243),r=n(507),s=n(260),i=n(732),l=n(802),a=n(136),u=n(233),h=n(761),c=n(70),p=n(810),d=n(957),m=n(628),f=n(967),g=n(677),_=n(414),H=n(985);o.HotStaq.isWeb=!0,t.exports.HotStaq=o.HotStaq,t.exports.Hot=r.Hot,t.exports.DeveloperMode=r.DeveloperMode,t.exports.HotComponent=s.HotComponent,t.exports.HotAPI=u.HotAPI,t.exports.EventExecutionType=u.EventExecutionType,t.exports.HotFile=i.HotFile,t.exports.HotLog=l.HotLog,t.exports.HotLogLevel=l.HotLogLevel,t.exports.HotPage=a.HotPage,t.exports.HotRoute=h.HotRoute,t.exports.HotRouteMethod=c.HotRouteMethod,t.exports.HTTPMethod=c.HTTPMethod,t.exports.HotServer=p.HotServer,t.exports.HotServerType=p.HotServerType,t.exports.HotClient=d.HotClient,t.exports.HotTester=g.HotTester,t.exports.HotTesterAPI=_.HotTesterAPI,t.exports.HotTestMap=H.HotTestMap,t.exports.HotTestDestination=H.HotTestDestination,t.exports.HotTestElement=f.HotTestElement,t.exports.HotTestElementOptions=f.HotTestElementOptions,t.exports.HotTestDriver=m.HotTestDriver},651:t=>{"use strict";t.exports={}},351:()=>{},606:()=>{}},__webpack_module_cache__={};function __webpack_require__(t){var e=__webpack_module_cache__[t];if(void 0!==e)return e.exports;var n=__webpack_module_cache__[t]={exports:{}};return __webpack_modules__[t].call(n.exports,n,n.exports,__webpack_require__),n.exports}var __webpack_exports__=__webpack_require__(859);HotStaqWeb=__webpack_exports__})();
2
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"HotStaq.js","mappings":"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","sources":["webpack://HotStaqWeb/./node_modules/form-data/lib/browser.js","webpack://HotStaqWeb/./node_modules/js-cookie/dist/js.cookie.js","webpack://HotStaqWeb/./node_modules/node-fetch/browser.js","webpack://HotStaqWeb/./src/Hot.ts","webpack://HotStaqWeb/./src/HotAPI.ts","webpack://HotStaqWeb/./src/HotClient.ts","webpack://HotStaqWeb/./src/HotComponent.ts","webpack://HotStaqWeb/./src/HotFile.ts","webpack://HotStaqWeb/./src/HotLog.ts","webpack://HotStaqWeb/./src/HotPage.ts","webpack://HotStaqWeb/./src/HotRoute.ts","webpack://HotStaqWeb/./src/HotRouteMethod.ts","webpack://HotStaqWeb/./src/HotServer.ts","webpack://HotStaqWeb/./src/HotStaq.ts","webpack://HotStaqWeb/./src/HotTestDriver.ts","webpack://HotStaqWeb/./src/HotTestElement.ts","webpack://HotStaqWeb/./src/HotTestMap.ts","webpack://HotStaqWeb/./src/HotTester.ts","webpack://HotStaqWeb/./src/HotTesterAPI.ts","webpack://HotStaqWeb/./src/api-web.ts","webpack://HotStaqWeb/external var \"{}\"","webpack://HotStaqWeb/webpack/bootstrap","webpack://HotStaqWeb/webpack/startup"],"sourcesContent":["/* eslint-env browser */\nmodule.exports = typeof self == 'object' ? self.FormData : window.FormData;\n","/*! js-cookie v3.0.1 | MIT */\n;\n(function (global, factory) {\n  typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :\n  typeof define === 'function' && define.amd ? define(factory) :\n  (global = global || self, (function () {\n    var current = global.Cookies;\n    var exports = global.Cookies = factory();\n    exports.noConflict = function () { global.Cookies = current; return exports; };\n  }()));\n}(this, (function () { 'use strict';\n\n  /* eslint-disable no-var */\n  function assign (target) {\n    for (var i = 1; i < arguments.length; i++) {\n      var source = arguments[i];\n      for (var key in source) {\n        target[key] = source[key];\n      }\n    }\n    return target\n  }\n  /* eslint-enable no-var */\n\n  /* eslint-disable no-var */\n  var defaultConverter = {\n    read: function (value) {\n      if (value[0] === '\"') {\n        value = value.slice(1, -1);\n      }\n      return value.replace(/(%[\\dA-F]{2})+/gi, decodeURIComponent)\n    },\n    write: function (value) {\n      return encodeURIComponent(value).replace(\n        /%(2[346BF]|3[AC-F]|40|5[BDE]|60|7[BCD])/g,\n        decodeURIComponent\n      )\n    }\n  };\n  /* eslint-enable no-var */\n\n  /* eslint-disable no-var */\n\n  function init (converter, defaultAttributes) {\n    function set (key, value, attributes) {\n      if (typeof document === 'undefined') {\n        return\n      }\n\n      attributes = assign({}, defaultAttributes, attributes);\n\n      if (typeof attributes.expires === 'number') {\n        attributes.expires = new Date(Date.now() + attributes.expires * 864e5);\n      }\n      if (attributes.expires) {\n        attributes.expires = attributes.expires.toUTCString();\n      }\n\n      key = encodeURIComponent(key)\n        .replace(/%(2[346B]|5E|60|7C)/g, decodeURIComponent)\n        .replace(/[()]/g, escape);\n\n      var stringifiedAttributes = '';\n      for (var attributeName in attributes) {\n        if (!attributes[attributeName]) {\n          continue\n        }\n\n        stringifiedAttributes += '; ' + attributeName;\n\n        if (attributes[attributeName] === true) {\n          continue\n        }\n\n        // Considers RFC 6265 section 5.2:\n        // ...\n        // 3.  If the remaining unparsed-attributes contains a %x3B (\";\")\n        //     character:\n        // Consume the characters of the unparsed-attributes up to,\n        // not including, the first %x3B (\";\") character.\n        // ...\n        stringifiedAttributes += '=' + attributes[attributeName].split(';')[0];\n      }\n\n      return (document.cookie =\n        key + '=' + converter.write(value, key) + stringifiedAttributes)\n    }\n\n    function get (key) {\n      if (typeof document === 'undefined' || (arguments.length && !key)) {\n        return\n      }\n\n      // To prevent the for loop in the first place assign an empty array\n      // in case there are no cookies at all.\n      var cookies = document.cookie ? document.cookie.split('; ') : [];\n      var jar = {};\n      for (var i = 0; i < cookies.length; i++) {\n        var parts = cookies[i].split('=');\n        var value = parts.slice(1).join('=');\n\n        try {\n          var foundKey = decodeURIComponent(parts[0]);\n          jar[foundKey] = converter.read(value, foundKey);\n\n          if (key === foundKey) {\n            break\n          }\n        } catch (e) {}\n      }\n\n      return key ? jar[key] : jar\n    }\n\n    return Object.create(\n      {\n        set: set,\n        get: get,\n        remove: function (key, attributes) {\n          set(\n            key,\n            '',\n            assign({}, attributes, {\n              expires: -1\n            })\n          );\n        },\n        withAttributes: function (attributes) {\n          return init(this.converter, assign({}, this.attributes, attributes))\n        },\n        withConverter: function (converter) {\n          return init(assign({}, this.converter, converter), this.attributes)\n        }\n      },\n      {\n        attributes: { value: Object.freeze(defaultAttributes) },\n        converter: { value: Object.freeze(converter) }\n      }\n    )\n  }\n\n  var api = init(defaultConverter, { path: '/' });\n  /* eslint-enable no-var */\n\n  return api;\n\n})));\n","\"use strict\";\n\n// ref: https://github.com/tc39/proposal-global\nvar getGlobal = function () {\n\t// the only reliable means to get the global object is\n\t// `Function('return this')()`\n\t// However, this causes CSP violations in Chrome apps.\n\tif (typeof self !== 'undefined') { return self; }\n\tif (typeof window !== 'undefined') { return window; }\n\tif (typeof global !== 'undefined') { return global; }\n\tthrow new Error('unable to locate global object');\n}\n\nvar global = getGlobal();\n\nmodule.exports = exports = global.fetch;\n\n// Needed for TypeScript and Webpack.\nif (global.fetch) {\n\texports.default = global.fetch.bind(global);\n}\n\nexports.Headers = global.Headers;\nexports.Request = global.Request;\nexports.Response = global.Response;","import { HotFile } from \"./HotFile\";\nimport { HotPage } from \"./HotPage\";\nimport { HotStaq } from \"./HotStaq\";\nimport { HotAPI } from \"./HotAPI\";\nimport { HotTestElement } from \"./HotTestElement\";\n\nimport Cookies from \"js-cookie\";\nimport fetch from \"node-fetch\";\n\n/**\n * The available developer modes.\n */\nexport enum DeveloperMode\n{\n\t/**\n\t * The default developer mode. No tests will be executed and \n\t * any test related data will be ignored.\n\t */\n\tProduction,\n\t/**\n\t * For use during development/debugging. All test data will \n\t * be collected and executed if necessary.\n\t */\n\tDevelopment\n}\n\n/**\n * A CSS object to embed.\n */\nexport interface CSSObject\n{\n\t/**\n\t * The url to the CSS file to embed.\n\t */\n\turl: string;\n\t/**\n\t * The integrity hash to generate during initial compilation.\n\t */\n\tintegrityHash: string;\n}\n\n/**\n * The api used during processing.\n */\nexport class Hot\n{\n\t/**\n\t * The currently generated page being displayed. This is cleared between every file processed.\n\t */\n\tstatic CurrentPage: HotPage = null;\n\t/**\n\t * The arguments passed.\n\t */\n\tstatic Arguments: any = null;\n\t/**\n\t * The mode in which this application is running. If it's set to development mode, all testing\n\t * related data will be collected, parsed, and executed if necessary.\n\t */\n\tstatic DeveloperMode = DeveloperMode;\n\t/**\n\t * The mode in which this application is running. If it's set to development mode, all testing\n\t * related data will be collected, parsed, and executed if necessary.\n\t */\n\tstatic HotTestElement = HotTestElement;\n\t/**\n\t * The mode in which this application is running. If it's set to development mode, all testing\n\t * related data will be collected, parsed, and executed if necessary.\n\t */\n\tstatic Mode: DeveloperMode = DeveloperMode.Production;\n\t/**\n\t *The current API used on this page. This is cleared between every file processed.\n\t */\n\tstatic API: HotAPI = null;\n\t/**\n\t * The API being used by the tester.\n\t */\n\tstatic TesterAPI: HotAPI = null;\n\t/**\n\t * Contains the buffer to output. This is cleared between every file processed.\n\t */\n\tstatic Output: string = \"\";\n\t/**\n\t * The data to share across all the different files and pages. This data will be public.\n\t */\n\tstatic Data: any = {};\n\t/**\n\t * The cookies to use between pages.\n\t */\n\tstatic Cookies: Cookies.CookiesStatic = Cookies;\n\t/**\n\t * Any public keys that need to be shown. These can be passed from HotSite.json.\n\t */\n\tstatic PublicKeys: any = {};\n\t/**\n\t * The CSS string to use when echoing out the CSS files.\n\t */\n\tstatic cssStr: string = `<link rel = \"stylesheet\" href = \"%CSS_FILE%\" />`;\n\t/**\n\t * The CSS files to use in the current page being generated.\n\t * \n\t * @todo Make this a \"string | CSSObject\" data type so it can also include \n\t * the integrity hashes as well.\n\t */\n\tstatic CSS: string[] = [];\n\t/**\n\t * The JavaScript files to use in the current page being generated.\n\t * \n\t * @todo Make this a \"string | JSFileObject\" data type so it can also include \n\t * the integrity hashes as well.\n\t */\n\tstatic JSFiles: any[] = [];\n\t/**\n\t * The JavaScript inline code to use in the current page being generated.\n\t */\n\tstatic JSScripts: any[] = [];\n\t/**\n\t * The JavaScript string to use when echoing out the Scripts files.\n\t */\n\tstatic jsFileStr: string = `<script type = \"text/javascript\" src = \"%JS_FILE%\"></script>`;\n\t/**\n\t * The JavaScript string to use when echoing out the Scripts files.\n\t */\n\tstatic jsScriptsStr: string = `<script type = \"text/javascript\">%JS_CODE%</script>`;\n\n\t/**\n\t * Retrieve a file and echo out it's contents.\n\t */\n\tstatic async include (file: HotFile | string, args: any[] = null): Promise<void>\n\t{\n\t\tif (HotStaq.isWeb === true)\n\t\t{\n\t\t\tif (typeof (file) === \"string\")\n\t\t\t{\n\t\t\t\tconst lowerFile: string = file.toLowerCase ();\n\n\t\t\t\t// If the file to be included does not have a nahfam, add it. This \n\t\t\t\t// will ensure the server sends only the file content.\n\t\t\t\tif (lowerFile.indexOf (\".hott\") > -1)\n\t\t\t\t{\n\t\t\t\t\tif (lowerFile.indexOf (\"nahfam\") < 0)\n\t\t\t\t\t\tfile += \"?hstqserve=nahfam\";\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tHot.echo (await Hot.getFile (file, args));\n\t}\n\n\t/**\n\t * Include and execute JavaScript for use when running the preprocessor.\n\t */\n\tstatic async includeJS (file: string): Promise<any>\n\t{\n\t\tconst res = await Hot.httpRequest (file);\n\t\tconst output: string = await res.text ();\n\n\t\tif (HotStaq.isWeb === true)\n\t\t\teval.apply (window, [output]);\n\t\telse\n\t\t\teval (output);\n\t}\n\n\t/**\n\t * Run an already loaded file and echo out it's contents.\n\t */\n\tstatic async runFile (fileName: string, args: any[] = null): Promise<void>\n\t{\n\t\tlet file: HotFile = Hot.CurrentPage.processor.getFile (fileName);\n\t\t/// @fixme Does the file need to be deep cloned first?\n\t\t//let clonedFile: HotFile = new HotFile (Object.assign ({}, file));\n\t\tlet tempFile: HotFile = file;\n\n\t\ttempFile.page = this.CurrentPage;\n\t\tlet content: string = await tempFile.process (args);\n\n\t\tHot.echo (content);\n\t}\n\n\t/**\n\t * Get the content of a file.\n\t */\n\tstatic async getFile (path: HotFile | string, args: any[] = null): Promise<string>\n\t{\n\t\tlet tempFile: HotFile = null;\n\n\t\tif (typeof (path) === \"string\")\n\t\t{\n\t\t\ttempFile = new HotFile ();\n\n\t\t\tif (HotStaq.isWeb === true)\n\t\t\t\ttempFile.url = path;\n\t\t\telse\n\t\t\t\ttempFile.localFile = path;\n\t\t}\n\t\telse\n\t\t\ttempFile = path;\n\n\t\tawait tempFile.load ();\n\n\t\ttempFile.page = this.CurrentPage;\n\t\tlet content: string = await tempFile.process (args);\n\n\t\treturn (content);\n\t}\n\n\t/**\n\t * Make an api call.\n\t */\n\tstatic async apiCall (route: string, data: any = null, \n\t\thttpMethod: string = \"POST\", \n\t\tfiles: { [name: string]: any } = {}): Promise<any>\n\t{\n\t\tlet result: any = null;\n\n\t\tif (Hot.CurrentPage == null)\n\t\t\tthrow new Error (\"Current page is null!\");\n\n\t\tif (Hot.CurrentPage.processor == null)\n\t\t\tthrow new Error (\"Current page's processor is null!\");\n\n\t\tif (Hot.CurrentPage.processor.api == null)\n\t\t\tthrow new Error (\"Current page's processor api is null! Did you forget to set the API name or URL?\");\n\n\t\tif (Hot.CurrentPage.processor.api != null)\n\t\t{\n\t\t\tresult = await Hot.CurrentPage.processor.api.makeCall (route, \n\t\t\t\t\t\t\tdata, httpMethod, files);\n\t\t}\n\n\t\treturn (result);\n\t}\n\n\t/**\n\t * Make a HTTP JSON request.\n\t * \n\t * @param url The full url to make the HTTP call.\n\t * @param data The data to JSON.stringify and send.\n\t * @param httpMethod The HTTP method to use to send the data.\n\t * \n\t * @returns The parsed JSON object.\n\t */\n\tstatic async jsonRequest (url: string, data: any = null, httpMethod: string = \"POST\"): Promise<any>\n\t{\n\t\ttry\n\t\t{\n\t\t\tlet fetchObj = {\n\t\t\t\t\"method\": httpMethod,\n\t\t\t\t\"headers\": {\n\t\t\t\t\t\t\"Accept\": \"application/json\",\n\t\t\t\t\t\t\"Content-Type\": \"application/json\"\n\t\t\t\t\t}\n\t\t\t}\n\n\t\t\tif (httpMethod === \"POST\")\n\t\t\t{\n\t\t\t\t/// @ts-ignore\n\t\t\t\tfetchObj[\"body\"] = JSON.stringify (data);\n\t\t\t}\n\n\t\t\tlet res = await fetch (url, fetchObj);\n\n\t\t\tif (res.ok === false)\n\t\t\t\tthrow new Error (`${res.status}: ${res.statusText}`);\n\n\t\t\tlet result: any = await res.json ();\n\n\t\t\treturn (result);\n\t\t}\n\t\tcatch (ex)\n\t\t{\n\t\t\treturn (JSON.stringify ({ \"error\": `${ex.message} - Could not fetch ${url}` }));\n\t\t}\n\t}\n\n\t/**\n\t * Make a HTTP request. This is basically just a wrapper for fetch.\n\t * \n\t * @param {string} url The full url to make the HTTP call.\n\t * @param {RequestInit} requestInit The request parameters to send.\n\t * \n\t * @returns The HTTP response.\n\t */\n\tstatic async httpRequest (url: string, requestInit: any = undefined): Promise<any>\n\t{\n\t\tlet res = await fetch (url, requestInit);\n\n\t\treturn (res);\n\t}\n\n\t/**\n\t * Echo out some output.\n\t */\n\tstatic echo (message: string): void\n\t{\n\t\tHot.Output += message;\n\t}\n\n\t/**\n\t * Echo out the CSS for the current page being generated.\n\t */\n\tstatic displayCSS (): void\n\t{\n\t\tfor (let iIdx = 0; iIdx < Hot.CSS.length; iIdx++)\n\t\t{\n\t\t\tlet cssFile: string = Hot.CSS[iIdx];\n\t\t\tlet cssOut: string = Hot.cssStr;\n\n\t\t\tcssOut = cssOut.replace (/\\%CSS_FILE\\%/g, cssFile);\n\n\t\t\tHot.echo (cssOut);\n\t\t}\n\t}\n\n\t/**\n\t * Echo out the JS files for the current page being generated.\n\t */\n\tstatic displayJSFiles (): void\n\t{\n\t\tfor (let iIdx = 0; iIdx < Hot.JSFiles.length; iIdx++)\n\t\t{\n\t\t\tlet jsFile: string = Hot.JSFiles[iIdx];\n\t\t\tlet jsFileOut: string = Hot.jsFileStr;\n\n\t\t\tjsFileOut = jsFileOut.replace (/\\%JS_FILE\\%/g, jsFile);\n\n\t\t\tHot.echo (jsFileOut);\n\t\t}\n\t}\n\n\t/**\n\t * Echo out the JS scripts for the current page being generated.\n\t */\n\tstatic displayJSScripts (): void\n\t{\n\t\tfor (let iIdx = 0; iIdx < Hot.JSScripts.length; iIdx++)\n\t\t{\n\t\t\tlet jsScript: string = Hot.JSScripts[iIdx];\n\t\t\tlet jsScriptOut: string = Hot.jsScriptsStr;\n\n\t\t\tjsScriptOut = jsScriptOut.replace (/\\%JS_CODE\\%/g, jsScript);\n\n\t\t\tHot.echo (jsScriptOut);\n\t\t}\n\t}\n}","import fetch from \"node-fetch\";\nimport FormData from \"form-data\";\n\nimport { HotServer } from \"./HotServer\";\nimport { HotRoute } from \"./HotRoute\";\nimport { HotClient } from \"./HotClient\";\nimport { HotRouteMethod, ServerAuthorizationFunction } from \"./HotRouteMethod\";\nimport { HotDB } from \"./HotDB\";\n\nimport { HotDBSchema } from \"./schemas/HotDBSchema\";\n\n/**\n * The API to load.\n */\nexport type APItoLoad = {\n\texportedClassName: string;\n\tpath: string;\n };\n\n/**\n * The type of object to use during event executions.\n */\nexport enum EventExecutionType\n{\n\tHotRoute,\n\tHotMethod,\n\tHotAPI\n}\n\n/**\n * The API to use.\n */\nexport abstract class HotAPI\n{\n\t/**\n\t * The server connection.\n\t */\n\tconnection: HotServer | HotClient;\n\t/**\n\t * The description of the API.\n\t */\n\tdescription: string;\n\t/**\n\t * The base url for the server.\n\t */\n\tbaseUrl: string;\n\t/**\n\t * If set, this will create the route variables and functions for \n\t * easy client/server calling.\n\t */\n\tcreateFunctions: boolean;\n\t/**\n\t * The database connection.\n\t */\n\texecuteEventsUsing: EventExecutionType;\n\t/**\n\t * The database connection.\n\t */\n\tdb: HotDB;\n\t/**\n\t * The authorization credentials to use throughout the application. If using \n\t * basic authentication for HTTP requests, be sure to override the \n\t * `toAuthorizationHeaderString` in this object.\n\t * \n\t * Example of this object:\n\t * ```ts\n\t * this.authCredentials = {\n\t * \t\tuser: \"username\",\n\t * \t\tpassword: \"password\",\n\t * \t\ttoAuthorizationHeaderString: function ()\n\t * \t\t\t{\n\t * \t\t\t\treturn (btoa (this.user + \":\" + this.password));\n\t * \t\t\t}\n\t * \t};\n\t * ```\n\t */\n\tauthCredentials: any;\n\t/**\n\t * The function used for user authentication.\n\t */\n\tuserAuth: ServerAuthorizationFunction;\n\t/**\n\t * The database connection.\n\t */\n\troutes: { [name: string]: HotRoute };\n\t/**\n\t * Executed when the API is about to start registering routes. If \n\t * this function returns false, the server will not start.\n\t */\n\tonPreRegister: () => Promise<boolean>;\n\t/**\n\t * Executed when the API has finished registering routes. If \n\t * this function returns false, the server will not start.\n\t */\n\tonPostRegister: () => Promise<boolean>;\n\n\tconstructor (baseUrl: string, connection: HotServer | HotClient = null, db: HotDB = null)\n\t{\n\t\tthis.connection = connection;\n\t\tthis.description = \"\";\n\t\tthis.baseUrl = baseUrl;\n\t\tthis.createFunctions = true;\n\t\tthis.executeEventsUsing = EventExecutionType.HotRoute;\n\t\tthis.db = db;\n\t\tthis.authCredentials = null;\n\t\tthis.userAuth = null;\n\t\tthis.routes = {};\n\t\tthis.onPreRegister = null;\n\t\tthis.onPostRegister = null;\n\t}\n\n\t/**\n\t * Set the database schema for use.\n\t */\n\tsetDBSchema (schema: HotDBSchema): void\n\t{\n\t\tif (this.connection.api == null)\n\t\t\tthrow new Error (`No API has been set!`);\n\n\t\tif (this.connection.api.db == null)\n\t\t\tthrow new Error (`No database has been set for API base url ${this.connection.api.baseUrl}`);\n\n\t\tthis.connection.api.db.schema = schema;\n\t}\n\n\t/**\n\t * Get the database being used.\n\t */\n\tgetDB (): HotDB\n\t{\n\t\tif (this.connection.api.db == null)\n\t\t\tthrow new Error (`No database has been set for API base url ${this.connection.api.baseUrl}`);\n\n\t\treturn (this.connection.api.db);\n\t}\n\n\t/**\n\t * Get the database schema being used.\n\t */\n\tgetDBSchema (): HotDBSchema\n\t{\n\t\tif (this.connection.api.db == null)\n\t\t\tthrow new Error (`No database has been set for API base url ${this.connection.api.baseUrl}`);\n\n\t\treturn (this.connection.api.db.schema);\n\t}\n\n\t/**\n\t * Add a route. If this.createFunctions is set to true, this will take the incoming \n\t * route and create an object in this HotAPI object using the name of the route. If there's \n\t * any HotRouteMethods inside of the incoming HotRoute, it will create the methods \n\t * and attach them to the newly created HotRoute object.\n\t * \n\t * Example:\n\t * ```\n\t * export class Users extends HotRoute\n\t * {\n\t * \t\tconstructor (api: FreeLightAPI)\n\t * \t\t{\n\t * \t\t\tsuper (api.connection, \"user\");\n\t * \n\t * \t\t\tthis.addMethod (\"create\", this._create, HTTPMethod.POST);\n\t * \t\t}\n\t * \n\t * \t\tprotected async _create (req: any, res: any, authorizedValue: any, jsonObj: any, queryObj: any): Promise<any>\n\t * \t\t{\n\t * \t\t\treturn (true);\n\t * \t\t}\n\t * }\n\t * ```\n\t * \n\t * This in turn could be used like so:\n\t * ```\n\t * Hot.API.user.create ({});\n\t * ```\n\t * \n\t * Additionally it would create the endpoint: ```http://127.0.0.1:8080/v1/user/create```\n\t * \n\t * @param route The route to add. Can be either a full HotRoute object, or just \n\t * the route's name. If a HotRoute object is supplied, the rest of the parameters \n\t * will be ignored.\n\t * @param routeMethod The route's method to add. If the route parameter is a string, \n\t * it will be interpreted as the route's name, and this will be the method added to \n\t * the new route.\n\t * @param executeFunction The function to execute when routeMethod is called by the API.\n\t */\n\taddRoute (\n\t\troute: HotRoute | string,\n\t\trouteMethod: HotRouteMethod | string = null,\n\t\texecuteFunction: (req: any, res: any, authorizedValue: any, jsonObj: any, queryObj: any) => Promise<any> = null\n\t\t): void\n\t{\n\t\tlet routeName: string = \"\";\n\n\t\tif (route instanceof HotRoute)\n\t\t{\n\t\t\trouteName = route.route;\n\t\t\tthis.routes[route.route] = route;\n\t\t}\n\t\telse\n\t\t{\n\t\t\trouteName = route;\n\n\t\t\tif (this.routes[routeName] == null)\n\t\t\t\tthis.routes[routeName] = new HotRoute (this.connection, routeName);\n\n\t\t\tif (routeMethod instanceof HotRouteMethod)\n\t\t\t\tthis.routes[routeName].addMethod (routeMethod);\n\t\t\telse\n\t\t\t{\n\t\t\t\tthis.routes[routeName].addMethod (new HotRouteMethod (\n\t\t\t\t\tthis.routes[routeName], routeMethod, executeFunction));\n\t\t\t}\n\t\t}\n\n\t\tthis.routes[routeName].connection = this.connection;\n\n\t\t// Create the route functions for the server/client.\n\t\tif (this.createFunctions === true)\n\t\t{\n\t\t\t// @ts-ignore\n\t\t\tlet newRoute: { [name: string]: Function } = this[routeName];\n\n\t\t\tif (newRoute == null)\n\t\t\t\tnewRoute = {};\n\n\t\t\tfor (let iIdx = 0; iIdx < this.routes[routeName].methods.length; iIdx++)\n\t\t\t{\n\t\t\t\tlet currentRoute: HotRoute = this.routes[routeName];\n\t\t\t\tlet newRouteMethod: HotRouteMethod = this.routes[routeName].methods[iIdx];\n\n\t\t\t\t/*\n\t\t\t\t/// @fixme Is this really necessary? A HTTP call is much more preferable, \n\t\t\t\t/// especially for accruate testing.\n\t\t\t\tif (this.connection instanceof HotServer)\n\t\t\t\t{\n\t\t\t\t\tif (newRouteMethod.onServerExecute != null)\n\t\t\t\t\t\tnewRoute[newRouteMethod.name] = newRouteMethod.onServerExecute;\n\t\t\t\t}\n\t\t\t\telse*/\n\t\t\t\t{\n\t\t\t\t\t/*\n\t\t\t\t\t/// @fixme Is onClientExecute necessary? I'm thinking the dev can just simply create \n\t\t\t\t\t/// their own function to call.\n\t\t\t\t\tif (newRouteMethod.onClientExecute != null)\n\t\t\t\t\t\tnewRoute[newRouteMethod.name] = newRouteMethod.onClientExecute;\n\t\t\t\t\telse\n\t\t\t\t\t{*/\n\t\t\t\t\t\tnewRoute[newRouteMethod.name] = (data: any, files: any): any =>\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tlet httpMethod: string = newRouteMethod.type;\n\t\t\t\t\t\t\t\t// Construct the url here. Base + route + route method\n\t\t\t\t\t\t\t\tlet routeStr: string = \"\";\n\n\t\t\t\t\t\t\t\tif (currentRoute.version !== \"\")\n\t\t\t\t\t\t\t\t\trouteStr += `/${currentRoute.version}`;\n\n\t\t\t\t\t\t\t\tif (currentRoute.route !== \"\")\n\t\t\t\t\t\t\t\t\trouteStr += `/${currentRoute.route}`;\n\n\t\t\t\t\t\t\t\tif (newRouteMethod.name !== \"\")\n\t\t\t\t\t\t\t\t\trouteStr += `/${newRouteMethod.name}`;\n\n\t\t\t\t\t\t\t\tlet authCredentials: any = null;\n\n\t\t\t\t\t\t\t\t// Getting the authorization credentials from the API is the lowest \n\t\t\t\t\t\t\t\t// priority for getting credentials. The priorities are in this order: \n\t\t\t\t\t\t\t\t// 1. HotRouteMethod\n\t\t\t\t\t\t\t\t// 2. HotRoute\n\t\t\t\t\t\t\t\t// 3. HotAPI\n\t\t\t\t\t\t\t\tif (this.authCredentials != null)\n\t\t\t\t\t\t\t\t\tauthCredentials = this.authCredentials;\n\n\t\t\t\t\t\t\t\t// Find the authorization credentials. Prioritize them when they're \n\t\t\t\t\t\t\t\t// in the method. Only add the ones from the route if the ones from \n\t\t\t\t\t\t\t\t// the method are missing.\n\t\t\t\t\t\t\t\tif (newRouteMethod.authCredentials != null)\n\t\t\t\t\t\t\t\t\tauthCredentials = newRouteMethod.authCredentials;\n\t\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tif (newRouteMethod.route.authCredentials != null)\n\t\t\t\t\t\t\t\t\t\tauthCredentials = newRouteMethod.route.authCredentials;\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tif (authCredentials == null)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\tif (typeof (Hot) !== \"undefined\")\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\tif (Hot != null)\n\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\t\tif (Hot.API != null)\n\t\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\t\t\tif (Hot.API[currentRoute.route] != null)\n\t\t\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\t\t\t\tif (Hot.API[currentRoute.route].authCredentials != null)\n\t\t\t\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\t\t\t\t\tauthCredentials = Hot.API[currentRoute.route].authCredentials;\n\t\t\t\t\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tif (authCredentials != null)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t// Add the authorization credentials to the data being sent.\n\t\t\t\t\t\t\t\t\tfor (let key in authCredentials)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tlet authCredential: any = authCredentials[key];\n\n\t\t\t\t\t\t\t\t\t\t// Do not overwrite any existing keys in the data about \n\t\t\t\t\t\t\t\t\t\t// to be sent.\n\t\t\t\t\t\t\t\t\t\tif (data[key] == null)\n\t\t\t\t\t\t\t\t\t\t\tdata[key] = authCredential;\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tlet args: any[] = [routeStr, data, httpMethod, files];\n\n\t\t\t\t\t\t\t\treturn (this.makeCall.apply (this, args));\n\t\t\t\t\t\t\t};\n\t\t\t\t\t//}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// @ts-ignore\n\t\t\tthis[routeName] = newRoute;\n\t\t}\n\t}\n\n\t/**\n\t * Register a route with the server.\n\t */\n\tasync registerRoute (route: HotRoute): Promise<void>\n\t{\n\t\tif (this.connection instanceof HotServer)\n\t\t\tawait this.connection.registerRoute (route);\n\t}\n\n\t/**\n\t * Register all routes with the server.\n\t */\n\tasync registerRoutes (): Promise<void>\n\t{\n\t\tfor (let key in this.routes)\n\t\t{\n\t\t\tlet route: HotRoute = this.routes[key];\n\n\t\t\tawait this.registerRoute (route);\n\t\t}\n\t}\n\n\t/**\n\t * Make a call to the API.\n\t */\n\tasync makeCall (route: string, data: any, httpMethod: string = \"POST\", \n\t\tfiles: { [name: string]: any } = {}): Promise<any>\n\t{\n\t\tlet url: string = this.baseUrl;\n\n\t\thttpMethod = httpMethod.toUpperCase ();\n\n\t\tif (url[(url.length - 1)] === \"/\")\n\t\t\turl = url.substr (0, (url.length - 1));\n\n\t\tif (route[0] !== \"/\")\n\t\t\turl += \"/\";\n\n\t\turl += route;\n\n\t\tconst numFiles: number = Object.keys (files).length;\n\n\t\tif (numFiles > 0)\n\t\t{\n\t\t\tif (httpMethod !== \"POST\")\n\t\t\t\tthrow new Error (`To upload files, you must set the httpMethod to POST.`);\n\n\t\t\tconst formData: FormData = new FormData ();\n\n\t\t\tfor (let key in files)\n\t\t\t\tformData.append (key, files[key]);\n\n\t\t\tlet res = await fetch (url, {\n\t\t\t\t\tmethod: \"POST\",\n\t\t\t\t\t// @ts-ignore\n\t\t\t\t\tbody: formData\n\t\t\t\t});\n\t\t\tlet jsonRes: any = await res.json ();\n\n\t\t\tif (data[\"hotstaq\"] == null)\n\t\t\t\tdata[\"hotstaq\"] = {};\n\n\t\t\tif (data[\"hotstaq\"][\"uploads\"] == null)\n\t\t\t\tdata[\"hotstaq\"][\"uploads\"] = {};\n\n\t\t\tdata[\"hotstaq\"][\"uploads\"][\"uploadId\"] = \n\t\t\t\t\tjsonRes[\"hotstaq\"][\"uploads\"][\"uploadId\"];\n\n\t\t\t// After the upload, make the actual JSON call. Do not pass files again.\n\t\t\tconst result: any = await this.makeCall (route, data, httpMethod);\n\n\t\t\treturn (result);\n\t\t}\n\n\t\tlet fetchObj: any = {\n\t\t\t\tmethod: httpMethod,\n\t\t\t\theaders: {\n\t\t\t\t\t\t\"Accept\": \"application/json\",\n\t\t\t\t\t\t\"Content-Type\": \"application/json\"\n\t\t\t\t\t}\n\t\t\t};\n\n\t\tif ((httpMethod !== \"GET\") && \n\t\t\t(httpMethod !== \"HEAD\"))\n\t\t{\n\t\t\tfetchObj[\"body\"] = JSON.stringify (data);\n\t\t}\n\n\t\tlet promise = new Promise ((resolve, reject) => \n\t\t\t{\n\t\t\t\tfetch (url, fetchObj).then (async (res) =>\n\t\t\t\t\t{\n\t\t\t\t\t\tlet jsonObj: any = await res.json ();\n\t\t\n\t\t\t\t\t\tresolve (jsonObj);\n\t\t\t\t\t})\n\t\t\t\t\t.catch ((reason: any) =>\n\t\t\t\t\t{\n\t\t\t\t\t\tthrow new Error (`${url}: ${reason.message}`);\n\t\t\t\t\t});\n\t\t\t});\n\n\t\treturn (promise);\n\t}\n}\n","import { HotStaq } from \"./HotStaq\";\nimport { HotAPI } from \"./HotAPI\";\nimport { HotServerType } from \"./HotServer\";\nimport { HotLog } from \"./HotLog\";\n\n/**\n * A client connected to a server.\n */\nexport class HotClient\n{\n\t/**\n\t * The processor to use.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The API to use.\n\t */\n\tapi: HotAPI;\n\t/**\n\t * The tester API to use.\n\t */\n\ttesterAPI: HotAPI;\n\t/**\n\t * The type of server.\n\t */\n\ttype: HotServerType;\n\t/**\n\t * The logger.\n\t */\n\tlogger: HotLog;\n\n\tconstructor (processor: HotStaq)\n\t{\n\t\tthis.processor = processor;\n\t\tthis.api = null;\n\t\tthis.testerAPI = null;\n\t\tthis.type = HotServerType.HTTP;\n\t\tthis.logger = processor.logger;\n\t}\n}","import { HotAPI } from \"./HotAPI\";\nimport { HotStaq } from \"./HotStaq\";\n\nexport interface HotComponentOutput\n{\n\t/**\n\t * The HTML to output.\n\t */\n\thtml: string;\n\t/**\n\t * The query selector to add this component's functions to.\n\t * \n\t * @example #objectId\n\t */\n\taddFunctionsTo?: string;\n\t/**\n\t * The place here parent name to attach this html to.\n\t * \n\t * @example top\n\t */\n\tplaceHereParent?: string;\n\t/**\n\t * Append the output to an existing element.\n\t * \n\t * @example div .testClass\n\t */\n\tparentSelector?: string;\n}\n\n/**\n * A component to preprocess.\n */\nexport interface IHotComponent\n{\n\t/**\n\t * The processor to use.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The associated HTMLElements.\n\t */\n\thtmlElements?: HTMLElement[];\n\t/**\n\t * The name of the page.\n\t */\n\tname?: string;\n\t/**\n\t * The name of the tag.\n\t */\n\ttag?: string;\n\t/**\n\t * The connected API.\n\t */\n\tapi?: HotAPI;\n\t/**\n\t * The options to include with registering this component.\n\t */\n\telementOptions?: ElementDefinitionOptions;\n\t/**\n\t * Any extra attributes to register.\n\t */\n\tobservedAttributes?: string[];\n\t/**\n\t * The type of component.\n\t */\n\ttype?: string;\n\t/**\n\t * The value of the component.\n\t */\n\tvalue?: any;\n\t/**\n\t * The inner HTML or value of the component.\n\t */\n\tinner: any;\n\t/**\n\t * The events to trigger.\n\t */\n\tevents?: {\n\t\t\t[name: string]: {\n\t\t\t\ttype: string;\n\t\t\t\tfunc: Function;\n\t\t\t\toptions?: any;\n\t\t\t}\n\t\t};\n\t/**\n\t * Execute prior to output.\n\t * \n\t * @returns If set to false, the component will not be registered.\n\t */\n\tonPreOutput?: () => Promise<boolean>;\n\t/**\n\t * Execute after getting the output, but before the DOM parsing.\n\t * \n\t * @param output The output from the component to register. Can be manipulated one last time prior to \n\t * being parsed into a DOM element.\n\t * \n\t * @returns The final output to be parsed as a DOM element.\n\t */\n\tonPostOutput?: (output: (string | HotComponentOutput[])) => Promise<(string | HotComponentOutput[])>;\n\t/**\n\t * Execute when its time to fix the HTML prior to DOM parsing. This will skip the HotStaq default fixing.\n\t */\n\tonFixHTML?: (output: string) => Promise<{ fixedStr: string, querySelector: string; }>;\n\t/**\n\t * Execute a custom DOM parser.\n\t */\n\tonParseDOM?: (output: string) => Promise<Document>;\n\t/**\n\t * Execute after the output has been parsed and is ready to be placed into the DOM.\n\t */\n\tonParsed?: (output: string) => Promise<string>;\n\t/**\n\t * Execute prior to placing the new DOM element.\n\t */\n\tonPrePlace?: (htmlElement: HTMLElement) => Promise<HTMLElement>;\n\t/**\n\t * Execute after placing the new DOM element. Can be manipulated one final time prior to being rendered.\n\t */\n\tonPostPlace?: (parentHtmlElement: HTMLElement, htmlElement: HTMLElement) => Promise<HTMLElement>;\n\t/**\n\t * Execute after placing the DOM element onto the newly created parent.\n\t */\n\tonParentPlace?: (parentHtmlElement: HTMLElement, htmlElement: HTMLElement) => Promise<void>;\n}\n\n/**\n * A component to preprocess.\n */\nexport abstract class HotComponent implements IHotComponent\n{\n\t/**\n\t * The processor to use.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The associated HTMLElements.\n\t */\n\thtmlElements: HTMLElement[];\n\t/**\n\t * The name of the component.\n\t */\n\tname: string;\n\t/**\n\t * The name of the tag.\n\t */\n\ttag: string;\n\t/**\n\t * The connected API.\n\t */\n\tapi: HotAPI;\n\t/**\n\t * The options to include with registering this component.\n\t */\n\telementOptions: ElementDefinitionOptions;\n\t/**\n\t * Any extra attributes to register.\n\t */\n\tobservedAttributes: string[];\n\t/**\n\t * The type of component.\n\t */\n\ttype: string;\n\t/**\n\t * The value of the component.\n\t */\n\tvalue: any;\n\t/**\n\t * The inner HTML or value of the component.\n\t */\n\tinner: any;\n\t/**\n\t * The events to trigger.\n\t */\n\tevents: {\n\t\t[name: string]: {\n\t\t\t\ttype: string;\n\t\t\t\tfunc: Function;\n\t\t\t\toptions: any;\n\t\t\t}\n\t\t};\n\t/**\n\t * Execute prior to output.\n\t * \n\t * @returns If set to false, the component will not be registered.\n\t */\n\tonPreOutput?: () => Promise<boolean>;\n\t/**\n\t * Execute after getting the output, but before the DOM parsing.\n\t * \n\t * @param output The output from the component to register. Can be manipulated one last time prior to \n\t * being parsed into a DOM element.\n\t * \n\t * @returns The final output to be parsed as a DOM element.\n\t */\n\tonPostOutput?: (output: (string | HotComponentOutput[])) => Promise<(string | HotComponentOutput[])>;\n\t/**\n\t * Execute when its time to fix the HTML prior to DOM parsing. This will skip the HotStaq default fixing.\n\t */\n\tonFixHTML?: (output: string) => Promise<{ fixedStr: string, querySelector: string; }>;\n\t/**\n\t * Execute a custom DOM parser.\n\t */\n\tonParseDOM?: (output: string) => Promise<Document>;\n\t/**\n\t * Execute after the output has been parsed and is ready to be placed into the DOM.\n\t */\n\tonParsed?: (output: string) => Promise<string>;\n\t/**\n\t * Execute prior to placing the new DOM element.\n\t */\n\tonPrePlace?: (htmlElement: HTMLElement) => Promise<HTMLElement>;\n\t/**\n\t * Execute after placing the new DOM element. Can be manipulated one final time prior to being rendered.\n\t */\n\tonPostPlace?: (parentHtmlElement: HTMLElement, htmlElement: HTMLElement) => Promise<HTMLElement>;\n\t/**\n\t * Execute after placing the DOM element onto the newly created parent.\n\t */\n\tonParentPlace?: (parentHtmlElement: HTMLElement, htmlElement: HTMLElement) => Promise<void>;\n\n\tconstructor (copy: IHotComponent | HotStaq, api: HotAPI = null)\n\t{\n\t\tif ((copy instanceof HotStaq) || (copy == null))\n\t\t{\n\t\t\t// @ts-ignore\n\t\t\tthis.processor = copy;\n\t\t\tthis.htmlElements = [];\n\t\t\tthis.name = \"\";\n\t\t\tthis.tag = \"\";\n\t\t\tthis.api = null;\n\t\t\tthis.elementOptions = undefined;\n\t\t\tthis.observedAttributes = [];\n\t\t\tthis.type = \"\";\n\t\t\tthis.value = null;\n\t\t\tthis.inner = null;\n\t\t\tthis.events = {};\n\t\t}\n\t\telse\n\t\t{\n\t\t\tthis.processor = copy.processor;\n\t\t\tthis.htmlElements = copy.htmlElements || [];\n\t\t\tthis.name = copy.name || \"\";\n\t\t\tthis.tag = copy.tag || this.name;\n\t\t\tthis.api = copy.api || null;\n\t\t\tthis.elementOptions = copy.elementOptions || undefined;\n\t\t\tthis.observedAttributes = copy.observedAttributes || [];\n\t\t\tthis.type = copy.type || \"\";\n\t\t\tthis.value = copy.value || null;\n\t\t\tthis.inner = copy.inner || null;\n\t\t\tthis.events = {};\n\t\t}\n\n\t\tif (api != null)\n\t\t\tthis.api = api;\n\t}\n\n\t/**\n\t * Event that's called when this component's DOM element has been created.\n\t * \n\t * @returns The modified DOM element.\n\t */\n\tasync onCreated (element: HTMLElement): Promise<any>\n\t{\n\t\treturn (element);\n\t}\n\n\t/**\n\t * Handle the attributes manually.\n\t */\n\tasync handleAttributes? (attributes: NamedNodeMap): Promise<void>;\n\n\t/**\n\t * Handle a click event.\n\t */\n\tabstract click? (): Promise<void>;\n\n\t/**\n\t * Output the component.\n\t */\n\tabstract output (): Promise<string | HotComponentOutput[]>;\n}","import * as fs from \"fs\";\n\nimport fetch from \"node-fetch\";\n\nimport { DeveloperMode, Hot } from \"./Hot\";\nimport { HotPage } from \"./HotPage\";\nimport { HotTestElement } from \"./HotTestElement\";\n\n/**\n * A file to process.\n */\nexport interface IHotFile\n{\n\t/**\n\t * The parent page.\n\t */\n\tpage?: HotPage;\n\t/**\n\t * The name of the file.\n\t */\n\tname?: string;\n\t/**\n\t * The url to the file to get.\n\t */\n\turl?: string;\n\t/**\n\t * The path to the local file to get.\n\t */\n\tlocalFile?: string;\n\t/**\n\t * The content of the file to process.\n\t */\n\tcontent?: string;\n\t/**\n\t * Force all errors to be thrown.\n\t */\n\tthrowAllErrors?: boolean;\n}\n\n/**\n * Parser options for when processing a string or file.\n */\nexport interface ParserOptions\n{\n\t/**\n\t * Output the commands generated from processing. Default: true\n\t */\n\toutputCommands?: boolean;\n\t/**\n\t * Allow JSON.stringify to be used during processing. Default: true\n\t */\n\tallowStringify?: boolean;\n}\n\n/**\n * A file to process.\n */\nexport class HotFile implements IHotFile\n{\n\t/**\n\t * The parent page.\n\t */\n\tpage: HotPage;\n\t/**\n\t * The name of the file.\n\t */\n\tname: string;\n\t/**\n\t * The url to the file to get.\n\t */\n\turl: string;\n\t/**\n\t * The path to the local file to get.\n\t */\n\tlocalFile: string;\n\t/**\n\t * The content of the file to process.\n\t */\n\tcontent: string;\n\t/**\n\t * Force all errors to be thrown.\n\t */\n\tthrowAllErrors: boolean;\n\n\tconstructor (copy: IHotFile = {})\n\t{\n\t\tthis.page = copy.page || null;\n\t\tthis.name = copy.name || \"\";\n\t\tthis.url = copy.url || \"\";\n\t\tthis.localFile = copy.localFile || \"\";\n\t\tthis.content = copy.content || \"\";\n\t\tthis.throwAllErrors = copy.throwAllErrors || false;\n\t}\n\n\t/**\n\t * Set the content of this file.\n\t */\n\tsetContent (content: string): void\n\t{\n\t\tthis.content = content;\n\t}\n\n\t/**\n\t * Get the content of this file.\n\t */\n\tgetContent (): string\n\t{\n\t\treturn (this.content);\n\t}\n\n\t/**\n\t * Make a HTTP get request.\n\t */\n\tstatic async httpGet (url: string): Promise<string>\n\t{\n\t\ttry\n\t\t{\n\t\t\tlet res = await fetch (url);\n\n\t\t\tif (res.ok === false)\n\t\t\t\tthrow new Error (`${res.status}: ${res.statusText}`);\n\n\t\t\tlet content: string = await res.text ();\n\n\t\t\treturn (content);\n\t\t}\n\t\tcatch (ex)\n\t\t{\n\t\t\treturn (JSON.stringify ({ \"error\": `${ex.message} - Could not fetch ${url}` }));\n\t\t}\n\t}\n\n\t/**\n\t * Load content from a url.\n\t */\n\tasync loadUrl (): Promise<string>\n\t{\n\t\tthis.content = await HotFile.httpGet (this.url);\n\n\t\treturn (this.content);\n\t}\n\n\t/**\n\t * Load content from a local file.\n\t */\n\tasync loadLocalFile (): Promise<string>\n\t{\n\t\tlet promise: Promise<string> = new Promise (\n\t\t\t(resolve: any, reject: any): void =>\n\t\t\t{\n\t\t\t\tfs.readFile (this.localFile, (err: NodeJS.ErrnoException, data: Buffer): void =>\n\t\t\t\t\t{\n\t\t\t\t\t\tif (err != null)\n\t\t\t\t\t\t\tthrow err;\n\n\t\t\t\t\t\tlet content: string = data.toString ();\n\t\t\t\t\t\tthis.content = content;\n\n\t\t\t\t\t\tresolve (this.content);\n\t\t\t\t\t});\n\t\t\t});\n\n\t\treturn (promise);\n\t}\n\n\t/**\n\t * Load the contents of the file.\n\t */\n\tasync load (): Promise<string>\n\t{\n\t\tlet content: string = \"\";\n\n\t\tif (this.url !== \"\")\n\t\t\tcontent = await this.loadUrl ();\n\n\t\tif (this.localFile !== \"\")\n\t\t\tcontent = await this.loadLocalFile ();\n\n\t\treturn (content);\n\t}\n\n\t/**\n\t * Process string content. This will take in a regular expression and \n\t * parse the content based on the regex. When the regex content is found \n\t * contentProcessor will be executed with the regex content found. When \n\t * the regex content is not found, offContentProcessor will be called with \n\t * the content outside of the regex.\n\t * \n\t * @param content The content to parse.\n\t * @param contentRegex The regex to use to parse the content.\n\t * @param contentProcessor The content found inside the regex.\n\t * @param offContentProcessor The content found outside of the regex.\n\t * @param numRemoveFromBeginning The number of characters to remove from the \n\t * beginning of the found content.\n\t * @param numRemoveFromEnd The number of characters to remove from the end of \n\t * the found content.\n\t */\n\tstatic processContent (content: string, contentRegex: RegExp,\n\t\tcontentProcessor: (regexFound: string) => string,\n\t\toffContentProcessor: (offContent: string) => string,\n\t\tnumRemoveFromBeginning: number = 2,\n\t\tnumRemoveFromEnd: number = 2): string\n\t{\n\t\tlet result: RegExpExecArray = contentRegex.exec (content);\n\t\tlet previousIndex: number = 0;\n\t\tlet output: string = \"\";\n\n\t\twhile (result != null)\n\t\t{\n\t\t\tlet start: number = result.index - numRemoveFromBeginning;\n\t\t\tlet end: number = contentRegex.lastIndex + numRemoveFromEnd;\n\n\t\t\t// Get the previous section.\n\t\t\tlet prevContent: string = content.substr (previousIndex, (start - previousIndex));\n\t\t\tpreviousIndex = end;\n\n\t\t\toutput += offContentProcessor (prevContent);\n\n\t\t\t// Process the content found from the regex\n\t\t\tlet contentFound: string = result[0];\n\t\t\toutput += contentProcessor (contentFound);\n\n\t\t\t// Move on to the next section to parse.\n\t\t\tresult = contentRegex.exec (content);\n\t\t}\n\n\t\t// Append whatever else is after the last parsed section.\n\t\tlet lastContent: string = content.substr (previousIndex);\n\n\t\toutput += offContentProcessor (lastContent);\n\n\t\treturn (output);\n\t}\n\n\t/**\n\t * Process any content that could have nested values. This will \n\t * take in a regular expression and \n\t * parse the content based on the regex. When the regex content is found \n\t * contentProcessor will be executed with the regex content found. When \n\t * the regex content is not found, offContentProcessor will be called with \n\t * the content outside of the regex.\n\t * \n\t * @fixme Needs to be able to ignore any characters found inside comments \n\t * or a string. For example, if the following is used ```${\"Test }\"}``` It \n\t * will error out.\n\t * \n\t * @param content The content to parse.\n\t * @param contentRegex The regex to use to parse the content.\n\t * @param contentProcessor The content found inside the regex.\n\t * @param offContentProcessor The content found outside of the regex.\n\t * @param numRemoveFromBeginning The number of characters to remove from the \n\t * beginning of the found content.\n\t * @param numRemoveFromEnd The number of characters to remove from the end of \n\t * the found content.\n\t */\n\tstatic processNestedContent (content: string, startChars: string, endChars: string, \n\t\ttriggerChar: string, contentProcessor: (regexFound: string) => string,\n\t\toffContentProcessor: (offContent: string) => string,\n\t\tnumRemoveFromBeginning: number = 2,\n\t\tnumRemoveFromEnd: number = 1): string\n\t{\n\t\tlet pos: number = content.indexOf (startChars);\n\t\tlet previousIndex: number = 0;\n\t\tlet startTriggerPos: number = content.indexOf (triggerChar, pos);\n\t\tlet output: string = \"\";\n\n\t\twhile (pos > -1)\n\t\t{\n\t\t\tlet end: number = content.indexOf (endChars, pos);\n\t\t\tlet nestedCounter: number = 0;\n\n\t\t\tif (triggerChar !== \"\")\n\t\t\t{\n\t\t\t\t// Reverse search the trigger characters and count the number of \n\t\t\t\t// occurrences.\n\t\t\t\tlet rpos: number = content.lastIndexOf (triggerChar, end - numRemoveFromEnd);\n\n\t\t\t\twhile (rpos > -1)\n\t\t\t\t{\n\t\t\t\t\tif (rpos === startTriggerPos)\n\t\t\t\t\t\tbreak;\n\n\t\t\t\t\trpos = content.lastIndexOf (triggerChar, rpos - numRemoveFromEnd);\n\t\t\t\t\tnestedCounter++;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// If there's nested trigger characters, get the last occurrence of \n\t\t\t// the end character.\n\t\t\tif (nestedCounter > 0)\n\t\t\t{\n\t\t\t\tlet epos: number = content.indexOf (endChars, end + numRemoveFromEnd);\n\t\t\t\tlet tempepos: number = epos;\n\n\t\t\t\twhile ((epos > -1) && (nestedCounter > 0))\n\t\t\t\t{\n\t\t\t\t\tif (tempepos < 0)\n\t\t\t\t\t\tbreak;\n\n\t\t\t\t\t// Make sure we aren't discovering endChars that we shouldn't be.\n\t\t\t\t\tlet posOutsideOfContent: number = content.lastIndexOf (startChars, tempepos - numRemoveFromEnd);\n\n\t\t\t\t\tif (posOutsideOfContent > epos)\n\t\t\t\t\t\tbreak;\n\n\t\t\t\t\tepos = tempepos;\n\n\t\t\t\t\ttempepos = content.indexOf (endChars, epos + numRemoveFromEnd);\n\t\t\t\t\tnestedCounter--;\n\t\t\t\t}\n\n\t\t\t\tend = epos;\n\t\t\t}\n\n\t\t\tlet offContentStr: string = content.substr (previousIndex, (pos - previousIndex));\n\t\t\toutput += offContentProcessor (offContentStr);\n\n\t\t\tlet foundContent: string = content.substr (\n\t\t\t\tpos + numRemoveFromBeginning, (end - (pos + numRemoveFromBeginning)));\n\t\t\toutput += contentProcessor (foundContent);\n\n\t\t\t// Get the next content\n\t\t\tpos = content.indexOf (startChars, end + numRemoveFromEnd);\n\t\t\tstartTriggerPos = content.indexOf (triggerChar, pos);\n\t\t\tpreviousIndex = end + numRemoveFromEnd;\n\t\t}\n\n\t\t// Append whatever else is after the last parsed section.\n\t\tlet lastContent: string = content.substr (previousIndex);\n\n\t\toutput += offContentProcessor (lastContent);\n\n\t\treturn (output);\n\t}\n\n\t/**\n\t * Parse content.\n\t * \n\t * @param thisContent The content to parse.\n\t * @param throwAllErrors If set to true, any error that occurs will be thrown as an exception.\n\t * @param options The options to use when parsing the content.\n\t */\n\tstatic parseContent (thisContent: string, throwAllErrors: boolean, parserOptions: ParserOptions = null): string\n\t{\n\t\tif (parserOptions == null)\n\t\t{\n\t\t\tparserOptions = {\n\t\t\t\t\toutputCommands: true,\n\t\t\t\t\tallowStringify: true\n\t\t\t\t};\n\t\t}\n\n\t\tlet STRINGIFY_START: string = \"JSON.stringify (\";\n\t\tlet STRINGIFY_END: string = \")\";\n\n\t\tif (parserOptions.allowStringify === false)\n\t\t{\n\t\t\tSTRINGIFY_START = \"\";\n\t\t\tSTRINGIFY_END = \"\";\n\t\t}\n\n\t\t// Assemble the JS to evaluate. This will take all content outside of \n\t\t// <* and *> and wrap a Hot.echo around it. Any JS found inside of the \n\t\t// <* and *> will be executed as is.\n\t\tlet output: string = HotFile.processContent (thisContent, \n\t\t\tnew RegExp (\"(?=\\\\<\\\\*)([\\\\s\\\\S]*?)(?=\\\\*\\\\>)\", \"g\"), \n\t\t\t(regexFound: string): string =>\n\t\t\t{\n\t\t\t\t// A little hack, since I suck at Regex :(\n\t\t\t\tregexFound = regexFound.substr (2);\n\n\t\t\t\treturn (`${regexFound}`);\n\t\t\t}, \n\t\t\t(offContent: string): string =>\n\t\t\t{\n\t\t\t\tif (offContent === \"\")\n\t\t\t\t\treturn (\"\");\n\n\t\t\t\tlet tempOutput: string = HotFile.processNestedContent (\n\t\t\t\t\toffContent, \"!{\", \"}\", \"{\", \n\t\t\t\t\t(regexFound2: string): string =>\n\t\t\t\t\t{\n\t\t\t\t\t\tlet out: string = `*&&%*%@#@!${regexFound2}*&!#%@!@*!`;\n\n\t\t\t\t\t\treturn (out);\n\t\t\t\t\t}, \n\t\t\t\t\t(offContent3: string): string =>\n\t\t\t\t\t{\n\t\t\t\t\t\treturn (offContent3);\n\t\t\t\t\t});\n\t\t\t\tlet tempOutput2: string = HotFile.processNestedContent (\n\t\t\t\t\ttempOutput, \"STR{\", \"}\", \"{\", \n\t\t\t\t\t(regexFound2: string): string =>\n\t\t\t\t\t{\n\t\t\t\t\t\tlet out: string = \"\";\n\n\t\t\t\t\t\tif (parserOptions.outputCommands === true)\n\t\t\t\t\t\t\tout = `*&&%*%@#@!echoOutput (JSON.stringify(${regexFound2}), ${throwAllErrors});*&!#%@!@*!`;\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tout = `*&&%*%@#@!${STRINGIFY_START}${regexFound2}${STRINGIFY_END}, ${throwAllErrors});*&!#%@!@*!`;\n\n\t\t\t\t\t\treturn (out);\n\t\t\t\t\t}, \n\t\t\t\t\t(offContent3: string): string =>\n\t\t\t\t\t{\n\t\t\t\t\t\treturn (offContent3);\n\t\t\t\t\t}, 4, 1);\n\t\t\t\tlet tempOutput3: string = HotFile.processNestedContent (\n\t\t\t\t\ttempOutput2, \"${\", \"}\", \"{\", \n\t\t\t\t\t(regexFound2: string): string =>\n\t\t\t\t\t{\n\t\t\t\t\t\tlet out: string = \"\";\n\n\t\t\t\t\t\tif (parserOptions.outputCommands === true)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tout = `*&&%*%@#@!try { Hot.echo (${regexFound2}); }catch (ex){Hot.echo (\"\");}*&!#%@!@*!`;\n\n\t\t\t\t\t\t\tif (throwAllErrors === true)\n\t\t\t\t\t\t\t\tout = `*&&%*%@#@!Hot.echo (${regexFound2});*&!#%@!@*!`;\n\t\t\t\t\t\t}\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tout = `*&&%*%@#@!${regexFound2}*&!#%@!@*!`;\n\n\t\t\t\t\t\treturn (out);\n\t\t\t\t\t}, \n\t\t\t\t\t(offContent3: string): string =>\n\t\t\t\t\t{\n\t\t\t\t\t\treturn (offContent3);\n\t\t\t\t\t\t/*let escapedContent: string = JSON.stringify (offContent3);\n\t\t\t\t\t\tlet out: string = `echoOutput (${escapedContent}, ${throwAllErrors});\\n`;\n\n\t\t\t\t\t\treturn (out);*/\n\t\t\t\t\t});\n\n\t\t\t\tlet tempOutput4: string = \"\";\n\n\t\t\t\tif (Hot.Mode === DeveloperMode.Production)\n\t\t\t\t{\n\t\t\t\t\ttempOutput4 = HotFile.processNestedContent (\n\t\t\t\t\t\ttempOutput3, \"?(\", \")\", \"(\", \n\t\t\t\t\t\t(regexFound2: string): string =>\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\treturn (\"\");\n\t\t\t\t\t\t}, \n\t\t\t\t\t\t(offContent3: string): string =>\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\treturn (offContent3);\n\t\t\t\t\t\t\t/*let out: string = `echoOutput (${offContent3}, ${throwAllErrors});\\n`;\n\n\t\t\t\t\t\t\treturn (out);*/\n\t\t\t\t\t\t});\n\t\t\t\t}\n\n\t\t\t\tif (Hot.Mode === DeveloperMode.Development)\n\t\t\t\t{\n\t\t\t\t\ttempOutput4 = HotFile.processNestedContent (\n\t\t\t\t\t\ttempOutput3, \"?(\", \")\", \"(\", \n\t\t\t\t\t\t(regexFound2: string): string =>\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet foundStr: string = \"\";\n\n\t\t\t\t\t\t\ttry\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t// Check to see if it be parsed. If so, stringify it.\n\t\t\t\t\t\t\t\tJSON.parse (regexFound2);\n\n\t\t\t\t\t\t\t\tif (parserOptions.allowStringify === true)\n\t\t\t\t\t\t\t\t\tfoundStr = JSON.stringify (regexFound2);\n\t\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\t\tfoundStr = `${regexFound2}`;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tcatch (ex)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t// If valid JSON is not received, don't worry about it, pass it \n\t\t\t\t\t\t\t\t// along to the function below for it to be parsed in the page.\n\t\t\t\t\t\t\t\t// The exception should be thrown there instead.\n\t\t\t\t\t\t\t\tfoundStr = `${regexFound2}`;\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t/// @fixme Make this a callable function and pass foundStr, etc.\n\t\t\t\t\t\t\tlet out: string = \"\";\n\n\t\t\t\t\t\t\tif (parserOptions.outputCommands === true)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tout = `*&&%*%@#@!{\nconst testElm = createTestElement (${foundStr});\nHot.echo (\\`data-test-object-name = \"\\${testElm.name}\" data-test-object-func = \"\\${testElm.func}\" data-test-object-value = \"\\${testElm.value}\"\\`);\n}*&!#%@!@*!\\n`;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tlet createTestElement = (foundStr2: any) =>\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tlet testElm = null;\n\n\t\t\t\t\t\t\t\t\ttry\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tlet obj = foundStr2;\n\n\t\t\t\t\t\t\t\t\t\tif (typeof (foundStr2) === \"string\")\n\t\t\t\t\t\t\t\t\t\t\tobj = JSON.parse (foundStr2);\n\n\t\t\t\t\t\t\t\t\t\tif (typeof (obj) === \"string\")\n\t\t\t\t\t\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj);\n\n\t\t\t\t\t\t\t\t\t\tif (obj instanceof Array)\n\t\t\t\t\t\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj[0], obj[1], obj[2]);\n\n\t\t\t\t\t\t\t\t\t\tif (obj[\"name\"] != null)\n\t\t\t\t\t\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj);\n\n\t\t\t\t\t\t\t\t\t\tif (Hot.CurrentPage.testElements[testElm.name] != null)\n\t\t\t\t\t\t\t\t\t\t\tthrow new Error (`Test element ${testElm.name} already exists!`);\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t\tcatch (ex)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tthrow new Error (\n\t\t\t\t\t\t\t\t`Error processing test element ${foundStr2} in ${Hot.CurrentPage.name}. Error: ${ex.message}`\n\t\t\t\t\t\t\t\t\t\t\t);\n\t\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\t\treturn (testElm);\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tconst testElm = createTestElement (foundStr);\n\t\t\t\t\t\t\t\tout = `*&&%*%@#@!data-test-object-name = \"${testElm.name}\" data-test-object-func = \"${testElm.func}\" data-test-object-value = \"${testElm.value}\"*&!#%@!@*!`;\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\treturn (out);\n\t\t\t\t\t\t}, \n\t\t\t\t\t\t(offContent3: string): string =>\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\treturn (offContent3);\n\t\t\t\t\t\t\t/*let out: string = `echoOutput (${offContent3}, ${throwAllErrors});\\n`;\n\n\t\t\t\t\t\t\treturn (out);*/\n\t\t\t\t\t\t});\n\t\t\t\t}\n\n\t\t\t\tlet tempOutput5: string = HotFile.processNestedContent (\n\t\t\t\t\ttempOutput4, \"*&&%*%@#@!\", \"*&!#%@!@*!\", \"*&&%*%@#@!\", \n\t\t\t\t\t(regexFound: string): string =>\n\t\t\t\t\t{\n\t\t\t\t\t\treturn (regexFound);\n\t\t\t\t\t}, \n\t\t\t\t\t(offContent: string): string =>\n\t\t\t\t\t{\n\t\t\t\t\t\tlet escapedContent: string = \"\";\n\n\t\t\t\t\t\tif (parserOptions.allowStringify === true)\n\t\t\t\t\t\t\tescapedContent = JSON.stringify (offContent);\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tescapedContent = offContent;\n\n\t\t\t\t\t\tlet out: string = \"\";\n\t\t\t\t\t\t\n\t\t\t\t\t\tif (parserOptions.outputCommands === true)\n\t\t\t\t\t\t\tout = `echoOutput (${escapedContent}, ${throwAllErrors});\\n`;\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tout = escapedContent;\n\n\t\t\t\t\t\treturn (out);\n\t\t\t\t\t}, \n\t\t\t\t\t\"*&&%*%@#@!\".length, \"*&!#%@!@*!\".length);\n\n\t\t\t\t/// @fixme Temporary hack. These delimiters should be removed from tempOutput when \n\t\t\t\t/// executing processNestedContent.\n\t\t\t\ttempOutput5 = tempOutput5.replace (/\\*\\&\\&\\%\\*\\%\\@\\#\\@\\!/g, \"\");\n\t\t\t\ttempOutput5 = tempOutput5.replace (/\\*\\&\\!\\#\\%\\@\\!\\@\\*\\!/g, \"\");\n\n\t\t\t\treturn (tempOutput5);\n\t\t\t}, 0);\n\n\t\treturn (output);\n\t}\n\n\t/**\n\t * Process the content in this file. This treats each file as one large JavaScript\n\t * file. Any text outside of the <* *> areas will be treated as:\n\t * \n\t * \t\tHot.echo (\"text\");\n\t * \n\t * @fixme The regex's in the offContent functions need to be fixed. There's several \n\t * test cases where they will fail.\n\t */\n\tasync process (args: any = null): Promise<string>\n\t{\n\t\tlet thisContent: string = this.content;\n\n\t\tHot.Mode = this.page.processor.mode;\n\t\tHot.Arguments = args;\n\t\tHot.CurrentPage = this.page;\n\t\tHot.PublicKeys = this.page.processor.publicKeys;\n\t\tHot.API = this.page.getAPI ();\n\t\tHot.TesterAPI = this.page.getTesterAPI ();\n\n\t\tlet output: string = HotFile.parseContent (thisContent, this.throwAllErrors);\n\n\t\t// Execute the assembled JS file.\n\t\tlet returnedOutput: any = null;\n\n\t\ttry\n\t\t{\n\t\t\tlet executionContent: string = `\n\t\t\tvar Hot = arguments[0];\n\t\t\tvar PassedHotFile = arguments[1];\n\n\t\t\t`;\n\n\t\t\t// Output the arguments so it's usable in the entire document.\n\t\t\tif (typeof (args) === \"string\")\n\t\t\t\tthrow new Error (`The passing arguments cannot be a string!`);\n\n\t\t\tfor (let key in args)\n\t\t\t{\n\t\t\t\tlet newVar: string = \"\";\n\t\t\t\tlet newVarValue: any = args[key];\n\t\t\t\tlet newVarValueStr: string = JSON.stringify (newVarValue);\n\n\t\t\t\tnewVar = `var ${key} = ${newVarValueStr};\\n`;\n\n\t\t\t\texecutionContent += newVar;\n\t\t\t}\n\n\t\t\tlet contentName: string = this.name;\n\n\t\t\tif (contentName === \"\")\n\t\t\t\tcontentName = this.localFile;\n\n\t\t\tif (contentName === \"\")\n\t\t\t\tcontentName = this.url;\n\n\t\t\texecutionContent += `\n\n\t\t\tfunction echoOutput (content, throwErrors)\n\t\t\t{\n\t\t\t\tif (throwErrors == null)\n\t\t\t\t\tthrowErrors = true;\n\n\t\t\t\tif (throwErrors === true)\n\t\t\t\t{\n\t\t\t\t\tHot.echo (content);\n\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\ttry\n\t\t\t\t{\n\t\t\t\t\tHot.echo (content);\n\t\t\t\t}\n\t\t\t\tcatch (ex)\n\t\t\t\t{\n\t\t\t\t\tHot.echo (\"\");\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tfunction createTestElement (foundStr)\n\t\t\t{\n\t\t\t\tlet testElm = null;\n\n\t\t\t\ttry\n\t\t\t\t{\n\t\t\t\t\tlet obj = foundStr;\n\n\t\t\t\t\tif (typeof (foundStr) === \"string\")\n\t\t\t\t\t\tobj = JSON.parse (foundStr);\n\n\t\t\t\t\tif (typeof (obj) === \"string\")\n\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj);\n\n\t\t\t\t\tif (obj instanceof Array)\n\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj[0], obj[1], obj[2]);\n\n\t\t\t\t\tif (obj[\"name\"] != null)\n\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj);\n\n\t\t\t\t\tif (Hot.CurrentPage.testElements[testElm.name] != null)\n\t\t\t\t\t\tthrow new Error (\\`Test element \\${testElm.name} already exists!\\`);\n\n\t\t\t\t\tHot.CurrentPage.addTestElement (testElm);\n\t\t\t\t}\n\t\t\t\tcatch (ex)\n\t\t\t\t{\n\t\t\t\t\tthrow new Error (\n\t\t\t\\`Error processing test element \\${foundStr} in \\${Hot.CurrentPage.name}. Error: \\${ex.message}\\`\n\t\t\t\t\t\t);\n\t\t\t\t}\n\n\t\t\t\treturn (testElm);\n\t\t\t}\n\n\t\t\tasync function runContent (CurrentHotFile)\n\t\t\t{\\n`;\n\t\t\texecutionContent += output;\n\t\t\texecutionContent += `\n\t\t\t}\n\n\t\t\treturn (runContent (PassedHotFile).then (() =>\n\t\t\t{\n\t\t\t\treturn ({\n\t\t\t\t\t\thot: Hot,\n\t\t\t\t\t\toutput: Hot.Output,\n\t\t\t\t\t\tdata: JSON.stringify (Hot.Data)\n\t\t\t\t\t});\n\t\t\t}));`;\n\n\t\t\t/// @fixme Prior to execution compile any TypeScript and make it ES5 compatible.\n\t\t\tlet func: Function = new Function (executionContent);\n\t\t\treturnedOutput = await func.apply (this, [Hot, this]);\n\t\t}\n\t\tcatch (ex)\n\t\t{\n\t\t\tif (ex instanceof SyntaxError)\n\t\t\t{\n\t\t\t\t/// @fixme Put what's in the content variable into a prev content variable?\n\t\t\t\t/// Then once there's no longer any syntax errors being thrown, execute the \n\t\t\t\t/// code? This would also require saving any HTML outside of the *> and <* \n\t\t\t\t/// then echoing it out. The throw below would have to be removed as well.\n\t\t\t\tthrow ex;\n\t\t\t}\n\t\t\telse\n\t\t\t\tthrow ex;\n\t\t}\n\n\t\tHot.Data = returnedOutput.hot.Data;\n\t\tlet finalOutput: string = returnedOutput.output;\n\t\tHot.Output = \"\";\n\n\t\treturn (finalOutput);\n\t}\n}","/**\n * The logging level.\n */\nexport enum HotLogLevel\n{\n\t/**\n\t * Prints only info messages.\n\t */\n\tInfo,\n\t/**\n\t * Prints only warning messages.\n\t */\n\tWarning,\n\t/**\n\t * Prints only error messages.\n\t */\n\tError,\n\t/**\n\t * Prints all messages.\n\t */\n\tVerbose,\n\t/**\n\t * Prints all messages, except verbose.\n\t */\n\tAll,\n\t/**\n\t * Doesn't print any message.\n\t */\n\tNone\n}\n\n/**\n * The logger.\n */\nexport class HotLog\n{\n\t/**\n\t * The logging level.\n\t */\n\tlogLevel: HotLogLevel;\n\n\tconstructor (logLevel: HotLogLevel = HotLogLevel.All)\n\t{\n\t\tthis.logLevel = logLevel;\n\t}\n\n\t/**\n\t * Log a message.\n\t */\n\tlog (level: HotLogLevel, message: string)\n\t{\n\t\tif (this.logLevel === HotLogLevel.Verbose)\n\t\t{\n\t\t\tif (level === HotLogLevel.Error)\n\t\t\t\tthis.error (message);\n\n\t\t\tif (level === HotLogLevel.Warning)\n\t\t\t\tthis.warning (message);\n\n\t\t\tif ((level === HotLogLevel.Info) || \n\t\t\t\t(level === HotLogLevel.Verbose))\n\t\t\t{\n\t\t\t\tthis.info (message);\n\t\t\t}\n\t\t}\n\n\t\tif (this.logLevel === HotLogLevel.All)\n\t\t{\n\t\t\tif (level === HotLogLevel.Error)\n\t\t\t\tthis.error (message);\n\n\t\t\tif (level === HotLogLevel.Warning)\n\t\t\t\tthis.warning (message);\n\n\t\t\tif (level === HotLogLevel.Info)\n\t\t\t\tthis.info (message);\n\t\t}\n\n\t\tif (this.logLevel === HotLogLevel.Error)\n\t\t{\n\t\t\tif (level === HotLogLevel.Error)\n\t\t\t\tthis.error (message);\n\t\t}\n\n\t\tif (this.logLevel === HotLogLevel.Warning)\n\t\t{\n\t\t\tif (level === HotLogLevel.Warning)\n\t\t\t\tthis.warning (message);\n\t\t}\n\n\t\tif (this.logLevel === HotLogLevel.Info)\n\t\t{\n\t\t\tif (level === HotLogLevel.Info)\n\t\t\t\tthis.info (message);\n\t\t}\n\t}\n\n\t/**\n\t * Log a verbose message.\n\t */\n\tverbose (message: string)\n\t{\n\t\tif (this.logLevel === HotLogLevel.Verbose)\n\t\t\tconsole.info (message);\n\t}\n\n\t/**\n\t * Log a message.\n\t */\n\tinfo (message: string)\n\t{\n\t\tif ((this.logLevel === HotLogLevel.All) || \n\t\t\t(this.logLevel === HotLogLevel.Verbose) || \n\t\t\t(this.logLevel === HotLogLevel.Info))\n\t\t{\n\t\t\tconsole.info (message);\n\t\t}\n\t}\n\n\t/**\n\t * Log a warning.\n\t */\n\twarning (message: string)\n\t{\n\t\tif ((this.logLevel === HotLogLevel.All) || \n\t\t\t(this.logLevel === HotLogLevel.Verbose) || \n\t\t\t(this.logLevel === HotLogLevel.Warning))\n\t\t{\n\t\t\tconsole.warn (message);\n\t\t}\n\t}\n\n\t/**\n\t * Log an error message.\n\t */\n\terror (message: string | Error)\n\t{\n\t\tif ((this.logLevel === HotLogLevel.All) || \n\t\t\t(this.logLevel === HotLogLevel.Verbose) || \n\t\t\t(this.logLevel === HotLogLevel.Error))\n\t\t{\n\t\t\tlet msg: string = \"\";\n\n\t\t\tif (typeof (message) === \"string\")\n\t\t\t\tmsg = message;\n\t\t\telse\n\t\t\t{\n\t\t\t\tif (message.message != null)\n\t\t\t\t\tmsg = message.message;\n\n\t\t\t\tif (message.stack != null)\n\t\t\t\t\tmsg = message.stack;\n\t\t\t}\n\n\t\t\tconsole.error (msg);\n\t\t}\n\t}\n}","import { Hot } from \"./Hot\";\nimport { HotFile } from \"./HotFile\";\nimport { HotStaq } from \"./HotStaq\";\nimport { HotAPI } from \"./HotAPI\";\nimport { HotTestElement } from \"./HotTestElement\";\nimport { HotTestMap, HotTestPath } from \"./HotTestMap\";\n\n/**\n * A page to preprocess.\n */\nexport interface IHotPage\n{\n\t/**\n\t * The processor to use.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The name of the page.\n\t */\n\tname?: string;\n\t/**\n\t * The route used to get to this page.\n\t */\n\troute?: string;\n\t/**\n\t * The name of the page. File ordering matters here.\n\t * Every file is processed incrementally.\n\t */\n\tfiles?: HotFile[];\n\t/**\n\t * The associated tester name.\n\t */\n\ttesterName?: string;\n\t/**\n\t * The associated tester map.\n\t */\n\ttesterMap?: string;\n\t/**\n\t * The elements to test on this page.\n\t */\n\ttestElements?: { [name: string]: HotTestElement; };\n\t/**\n\t * The test paths to test on this page.\n\t */\n\ttestPaths?: { [name: string]: HotTestPath; };\n}\n\n/**\n * A page to preprocess.\n */\nexport class HotPage implements IHotPage\n{\n\t/**\n\t * The processor to use.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The name of the page.\n\t */\n\tname: string;\n\t/**\n\t * The route used to get to this page.\n\t */\n\troute: string;\n\t/**\n\t * The name of the page. File ordering matters here.\n\t * Every file is processed incrementally.\n\t */\n\tfiles: HotFile[];\n\t/**\n\t * The associated tester name.\n\t */\n\ttesterName: string;\n\t/**\n\t * The associated tester map.\n\t */\n\ttesterMap: string;\n\t/**\n\t * The elements to test on this page.\n\t */\n\ttestElements: { [name: string]: HotTestElement; };\n\t/**\n\t * The test paths to test on this page.\n\t */\n\ttestPaths: { [name: string]: HotTestPath; };\n\n\tconstructor (copy: IHotPage | HotStaq)\n\t{\n\t\tif (copy instanceof HotStaq)\n\t\t{\n\t\t\tthis.processor = copy;\n\t\t\tthis.name = \"\";\n\t\t\tthis.testerName = \"\";\n\t\t\tthis.testerMap = \"\";\n\t\t\tthis.route = \"\";\n\t\t\tthis.files = [];\n\t\t\tthis.testElements = {};\n\t\t\tthis.testPaths = {};\n\t\t}\n\t\telse\n\t\t{\n\t\t\tthis.processor = copy.processor;\n\t\t\tthis.name = copy.name || \"\";\n\t\t\tthis.testerName = copy.testerName || \"\";\n\t\t\tthis.testerMap = copy.testerMap || \"\";\n\t\t\tthis.route = copy.route || \"\";\n\t\t\tthis.files = copy.files || [];\n\t\t\tthis.testElements = copy.testElements || {};\n\t\t\tthis.testPaths = copy.testPaths || {};\n\t\t}\n\t}\n\n\t/**\n\t * Add a file to process. It's recommend to load the file prior to \n\t * adding it to a page if it's about to be used.\n\t */\n\tasync addFile (file: HotFile): Promise<void>\n\t{\n\t\tfile.page = this;\n\n\t\tthis.files.push (file);\n\t}\n\n\t/**\n\t * Get the API associated with this page.\n\t */\n\tgetAPI (): HotAPI\n\t{\n\t\treturn (this.processor.api);\n\t}\n\n\t/**\n\t * Get the tester API associated with this page.\n\t */\n\tgetTesterAPI (): HotAPI\n\t{\n\t\treturn (this.processor.testerAPI);\n\t}\n\n\t/**\n\t * Add all files in the page. Could decrease page loading performance.\n\t * It's recommend to load the file prior to adding it to a page.\n\t */\n\tasync load (file: HotFile): Promise<void>\n\t{\n\t\tfor (let iIdx = 0; iIdx < this.files.length; iIdx++)\n\t\t{\n\t\t\tlet file: HotFile = this.files[iIdx];\n\n\t\t\tawait file.load ();\n\t\t}\n\t}\n\n\t/**\n\t * Process a page and get the result.\n\t */\n\tasync process (args: any = null): Promise<string>\n\t{\n\t\tlet output: string = \"\";\n\n\t\tfor (let iIdx = 0; iIdx < this.files.length; iIdx++)\n\t\t{\n\t\t\tlet file: HotFile = this.files[iIdx];\n\n\t\t\tHot.Output = \"\";\n\t\t\tfile.page = this;\n\n\t\t\toutput += await file.process (args);\n\t\t}\n\n\t\treturn (output);\n\t}\n\n\t/**\n\t * Add a test element.\n\t */\n\taddTestElement (elm: HotTestElement): void\n\t{\n\t\tif (this.testElements[elm.name] != null)\n\t\t\tthrow new Error (`Test element ${elm.name} already exists!`);\n\n\t\tthis.testElements[elm.name] = elm;\n\t}\n\n\t/**\n\t * Get a test element.\n\t */\n\tgetTestElement (name: string): HotTestElement\n\t{\n\t\tif (this.testElements[name] == null)\n\t\t\tthrow new Error (`Test element ${name} doest not exist!`);\n\n\t\treturn (this.testElements[name]);\n\t}\n\n\t/**\n\t * Create a test path.\n\t */\n\tcreateTestPath (pathName: string, driverFunc: HotTestPath): void\n\t{\n\t\tif (this.testPaths[pathName] != null)\n\t\t\tthrow new Error (`Test path ${pathName} already exists!`);\n\n\t\tthis.testPaths[pathName] = driverFunc;\n\t}\n}","import { HotServer } from \"./HotServer\";\nimport { HotRouteMethod, HTTPMethod, IHotRouteMethod, ServerExecutionFunction, TestCaseFunction, TestCaseObject } from \"./HotRouteMethod\";\nimport { HotClient } from \"./HotClient\";\nimport { HotLog } from \"./HotLog\";\n\n/**\n * The route to use.\n */\nexport class HotRoute\n{\n\t/**\n\t * The server that maintains the connections.\n\t */\n\tconnection: HotServer | HotClient;\n\t/**\n\t * The associated logger.\n\t */\n\tlogger: HotLog;\n\t/**\n\t * The route.\n\t */\n\troute: string;\n\t/**\n\t * The description of the route.\n\t */\n\tdescription: string;\n\t/**\n\t * The version.\n\t */\n\tversion: string;\n\t/**\n\t * The prefix to add to the beginning of each route method.\n\t */\n\tprefix: string;\n\t/**\n\t * The authorization credentials to be used by the client \n\t * when connecting to the server.\n\t */\n\tauthCredentials: any;\n\t/**\n\t * The calls that can be made.\n\t */\n\tmethods: HotRouteMethod[];\n\t/**\n\t * The errors and their JSON that can be thrown. Can be:\n\t * * not_authorized\n\t * * no_server_execute_function\n\t */\n\terrors: { [error: string]: any };\n\n\tconstructor (connection: HotServer | HotClient, route: string, methods: HotRouteMethod[] = [])\n\t{\n\t\tthis.connection = connection;\n\t\tthis.logger = null;\n\n\t\tif (this.connection != null)\n\t\t{\n\t\t\tif (this.connection.processor != null)\n\t\t\t\tthis.logger = this.connection.processor.logger;\n\t\t}\n\n\t\tthis.route = route;\n\t\tthis.description = \"\";\n\t\tthis.version = \"v1\";\n\t\tthis.prefix = \"\";\n\t\tthis.authCredentials = null;\n\t\tthis.methods = methods;\n\t\tthis.errors = {\n\t\t\t\t\"not_authorized\": HotRoute.createError (\"Not authorized.\"),\n\t\t\t\t\"no_server_execute_function\": HotRoute.createError (\"Missing server execute function.\"),\n\t\t\t};\n\t}\n\n\t/**\n\t * Create an error JSON object.\n\t */\n\tstatic createError (message: string): any\n\t{\n\t\treturn ({ error: message });\n\t}\n\n\t/**\n\t * Add an API method to this route.\n\t * \n\t * @param method The name of the method to add. If a HotRouteMethod is supplied, the \n\t * rest of the arguments supplied will be ignored.\n\t */\n\taddMethod (\n\t\tmethod: HotRouteMethod | IHotRouteMethod | string,\n\t\texecuteFunction: ServerExecutionFunction = null,\n\t\ttype: HTTPMethod = HTTPMethod.POST,\n\t\ttestCases: (string | TestCaseFunction)[] | TestCaseFunction[] | TestCaseObject[] = null\n\t\t): void\n\t{\n\t\tif (typeof (method) === \"string\")\n\t\t\tmethod = new HotRouteMethod (this, method, executeFunction, type, null, null, null, testCases);\n\n\t\tif (method instanceof HotRouteMethod)\n\t\t\tthis.methods.push (method);\n\t\telse\n\t\t{\n\t\t\tif (method.route == null)\n\t\t\t\tmethod.route = this;\n\n\t\t\tmethod = new HotRouteMethod (method);\n\t\t\tthis.methods.push ((<HotRouteMethod>method));\n\t\t}\n\t}\n\n\t/**\n\t * Get a method by it's name.\n\t */\n\tgetMethod (name: string): HotRouteMethod\n\t{\n\t\tlet foundMethod: HotRouteMethod = null;\n\n\t\tfor (let iIdx = 0; iIdx < this.methods.length; iIdx++)\n\t\t{\n\t\t\tlet method: HotRouteMethod = this.methods[iIdx];\n\n\t\t\tif (method.name === name)\n\t\t\t{\n\t\t\t\tfoundMethod = method;\n\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\n\t\treturn (foundMethod);\n\t}\n\n\t/**\n\t * Executes before all routes have been registered.\n\t */\n\tonPreRegister: () => Promise<void> = null;\n\t/**\n\t * Executes when first registering this route with Express. If \n\t * this returns false, the route will not be registered.\n\t */\n\tonRegister: () => Promise<boolean> = null;\n\t/**\n\t * Executes after all routes have been registered.\n\t */\n\tonPostRegister: () => Promise<void> = null;\n\n\t/**\n\t * Executes when authorizing a called method.\n\t * The value returned from here will be passed to onExecute in the \n\t * called HotRouteMethod. Undefined returning from here will mean \n\t * the authorization failed.\n\t */\n\tonAuthorizeUser: (req: any, res: any) => Promise<any> = null;\n}","import { DeveloperMode } from \"./Hot\";\nimport { HotTestDriver } from \"./HotTestDriver\";\nimport { HotRoute } from \"./HotRoute\";\nimport { HotServer } from \"./HotServer\";\n\n/**\n * Available HTTP methods.\n */\nexport enum HTTPMethod\n{\n\t/**\n\t * A HTTP GET request.\n\t */\n\tGET = \"get\",\n\t/**\n\t * A HTTP POST request.\n\t */\n\tPOST = \"post\",\n\t/**\n\t * This will upload a file, then post the json request afterwards.\n\t */\n\tFILE_UPLOAD = \"file_upload_then_post_json\"\n}\n\n/**\n * A function that will be executed by the server when first registering with Express.\n * If this returns false, this route method will not be registered.\n */\nexport type ServerRegistrationFunction = () => Promise<boolean>;\n/**\n * A function that will be executed by the server.\n */\nexport type ServerExecutionFunction = \n\t(req: any, res: any, authorizedValue: any, jsonObj: any, queryObj: any, files?: any) => Promise<any>;\n/**\n * A function that will be executed by the client.\n */\nexport type ClientExecutionFunction = (...args: any[]) => Promise<any>;\n/**\n * A function that will be executed by the server for authorization. Any value \n * returned from this function will be passed to the ServerExecutionFunction.\n * If an undefined value is returned, this indicates the server was not able \n * to authenticate the user, so the ServerExecutionFunction will not be \n * executed.\n */\nexport type ServerAuthorizationFunction = (req: any, res: any, jsonObj: any, queryObj: any) => Promise<any>;\n/**\n * The test case function to execute.\n */\nexport type TestCaseFunction = ((driver: HotTestDriver) => Promise<any>) | ((driver: HotTestDriver) => any);\n/**\n * The test case object to pass.\n */\nexport interface TestCaseObject\n{\n\t/**\n\t * The name of the test case.\n\t */\n\tname: string;\n\t/**\n\t * The function to execute.\n\t */\n\tfunc: TestCaseFunction;\n}\n\n/**\n * A method parameter.\n */\nexport interface HotRouteMethodParameter\n{\n\t/**\n\t * The type of parameter. Default: string\n\t * Can be:\n\t * * string\n\t * * integer\n\t * * number\n\t * * boolean\n\t * * array\n\t * * object\n\t */\n\ttype?: string;\n\t/**\n\t * The description of the parameter. Default: \"\"\n\t */\n\tdescription?: string;\n\t/**\n\t * Is this parameter required? Default: false\n\t */\n\trequired?: boolean;\n\t/**\n\t * The parameters in the object.\n\t */\n\tparameters?: { [name: string]: string | HotRouteMethodParameter; };\n}\n\n/**\n * An API method to make.\n */\nexport interface IHotRouteMethod\n{\n\t/**\n\t * The parent route.\n\t */\n\troute?: HotRoute;\n\t/**\n\t * The api call name.\n\t */\n\tname: string;\n\t/**\n\t * The description of the api method.\n\t */\n\tdescription?: string;\n\t/**\n\t * The description of what returns from the api method.\n\t */\n\treturns?: string | HotRouteMethodParameter;\n\t/**\n\t * The parameters in the api method.\n\t */\n\tparameters?: { [name: string]: string | HotRouteMethodParameter; };\n\t/**\n\t * The api call name.\n\t */\n\ttype?: HTTPMethod;\n\t/**\n\t * The authorization credentials to be used by the client \n\t * when connecting to the server.\n\t */\n\tauthCredentials?: any;\n\t/**\n\t * The test case objects to execute during tests.\n\t */\n\ttestCases?: {\n\t\t\t[name: string]: TestCaseObject;\n\t\t} | (string | TestCaseFunction)[] | TestCaseFunction[] | TestCaseObject[];\n\t/**\n\t * Executes before all routes have been registered.\n\t */\n\tonPreRegister?: () => Promise<void>;\n\t/**\n\t * Executes when first registering this method with Express. If \n\t * this returns false, the method will not be registered.\n\t */\n\tonRegister?: ServerRegistrationFunction;\n\t/**\n\t * Executes after all routes have been registered.\n\t */\n\tonPostRegister?: () => Promise<void>;\n\n\t/**\n\t * Executes when authorizing a called method. If this method \n\t * is set, this will not call onAuthorize for the parent HotRoute.\n\t * The value returned from here will be passed to onExecute. \n\t * Undefined returning from here will mean the authorization failed.\n\t * If any exceptions are thrown from this function, they will be sent \n\t * to the server as an { error: string; } object with the exception \n\t * message as the error.\n\t */\n\tonServerAuthorize?: ServerAuthorizationFunction;\n\n\t/**\n\t * Executes when executing a called method from the server side. \n\t * This will stringify any JSON object and send it as a JSON response. \n\t * If undefined is returned no response will be sent to the server. \n\t * So the developer would have to send a response using \"res\".\n\t * If any exceptions are thrown from this function, they will be sent \n\t * to the server as an { error: string; } object with the exception \n\t * message as the error.\n\t */\n\tonServerExecute?: ServerExecutionFunction;\n\t/**\n\t * Executes when executing a called method from the client side.\n\t * @fixme Is this necessary?\n\t */\n\tonClientExecute?: ClientExecutionFunction;\n}\n\n/**\n * An API method to make.\n */\nexport class HotRouteMethod implements IHotRouteMethod\n{\n\t/**\n\t * The parent route.\n\t */\n\troute: HotRoute;\n\t/**\n\t * The api call name.\n\t */\n\tname: string;\n\t/**\n\t * The description of the api method.\n\t */\n\tdescription: string;\n\t/**\n\t * The description of what returns from the api method.\n\t */\n\treturns: HotRouteMethodParameter;\n\t/**\n\t * The parameters in the api method.\n\t */\n\tparameters: { [name: string]: HotRouteMethodParameter; };\n\t/**\n\t * The api call name.\n\t */\n\ttype: HTTPMethod;\n\t/**\n\t * Has this method been registered with the server? This \n\t * prevents the method from being reregistered.\n\t */\n\tisRegistered: boolean;\n\t/**\n\t * Has this method been registered with the server? This \n\t * prevents the method from being reregistered.\n\t */\n\texecuteSetup: boolean;\n\t/**\n\t * The authorization credentials to be used by the client \n\t * when connecting to the server.\n\t */\n\tauthCredentials: any;\n\t/**\n\t * The test case objects to execute during tests.\n\t */\n\ttestCases: {\n\t\t\t[name: string]: TestCaseObject;\n\t\t};\n\t/**\n\t * Executes before all routes have been registered.\n\t */\n\tonPreRegister?: () => Promise<void>;\n\t/**\n\t * Executes when first registering this method with Express. If \n\t * this returns false, the method will not be registered.\n\t */\n\tonRegister?: ServerRegistrationFunction;\n\t/**\n\t * Executes after all routes have been registered.\n\t */\n\tonPostRegister?: () => Promise<void>;\n\n\t/**\n\t * Executes when authorizing a called method. If this method \n\t * is set, this will not call onAuthorize for the parent HotRoute.\n\t * The value returned from here will be passed to onExecute. \n\t * Undefined returning from here will mean the authorization failed.\n\t * If any exceptions are thrown from this function, they will be sent \n\t * to the server as an { error: string; } object with the exception \n\t * message as the error.\n\t */\n\tonServerAuthorize?: ServerAuthorizationFunction;\n\n\t/**\n\t * Executes when executing a called method from the server side. \n\t * This will stringify any JSON object and send it as a JSON response. \n\t * If undefined is returned no response will be sent to the server. \n\t * So the developer would have to send a response using \"res\".\n\t * If any exceptions are thrown from this function, they will be sent \n\t * to the server as an { error: string; } object with the exception \n\t * message as the error.\n\t */\n\tonServerExecute?: ServerExecutionFunction;\n\t/**\n\t * Executes when executing a called method from the client side.\n\t * @fixme Is this necessary?\n\t */\n\tonClientExecute?: ClientExecutionFunction;\n\n\tconstructor (route: HotRoute | IHotRouteMethod, name: string = \"\", \n\t\tonExecute: ServerExecutionFunction | ClientExecutionFunction = null, \n\t\ttype: HTTPMethod = HTTPMethod.POST, onServerAuthorize: ServerAuthorizationFunction = null, \n\t\tonRegister: ServerRegistrationFunction = null, authCredentials: any = null, \n\t\ttestCases: { [name: string]: TestCaseObject; } | (string | TestCaseFunction)[] | TestCaseFunction[] | TestCaseObject[] = null)\n\t{\n\t\tlet newRoute: HotRoute = null;\n\n\t\tif (route instanceof HotRoute)\n\t\t\tnewRoute = route;\n\t\telse\n\t\t{\n\t\t\tnewRoute = route.route;\n\n\t\t\tif (route.type != null)\n\t\t\t\ttype = route.type;\n\n\t\t\tif (route.name != null)\n\t\t\t\tname = route.name;\n\n\t\t\tif (route.description != null)\n\t\t\t\tthis.description = route.description;\n\n\t\t\tif (route.returns != null)\n\t\t\t{\n\t\t\t\tif (typeof (route.returns) === \"string\")\n\t\t\t\t{\n\t\t\t\t\tthis.returns = {\n\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\"required\": true,\n\t\t\t\t\t\t\t\"description\": route.returns\n\t\t\t\t\t\t};\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t\tthis.returns = route.returns;\n\t\t\t}\n\n\t\t\tif (route.parameters != null)\n\t\t\t{\n\t\t\t\tthis.parameters = {};\n\n\t\t\t\tfor (let key in route.parameters)\n\t\t\t\t{\n\t\t\t\t\tlet param = route.parameters[key];\n\n\t\t\t\t\tif (typeof (param) === \"string\")\n\t\t\t\t\t{\n\t\t\t\t\t\tthis.parameters[key] = {\n\t\t\t\t\t\t\t\t\"type\": param,\n\t\t\t\t\t\t\t\t\"required\": false,\n\t\t\t\t\t\t\t\t\"description\": \"\"\n\t\t\t\t\t\t\t};\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tif (param.type == null)\n\t\t\t\t\t\t\tparam.type = \"string\";\n\n\t\t\t\t\t\tthis.parameters[key] = param;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif (route.authCredentials != null)\n\t\t\t\tauthCredentials = route.authCredentials;\n\n\t\t\tif (route.onServerExecute != null)\n\t\t\t\tonExecute = route.onServerExecute;\n\n\t\t\tif (route.onServerAuthorize != null)\n\t\t\t\tonServerAuthorize = route.onServerAuthorize;\n\n\t\t\tif (route.onRegister != null)\n\t\t\t\tonRegister = route.onRegister;\n\n\t\t\tif (route.onPostRegister != null)\n\t\t\t\tthis.onPostRegister = route.onPostRegister;\n\n\t\t\tif (route.onServerExecute != null)\n\t\t\t\tthis.onServerExecute = route.onServerExecute;\n\n\t\t\tif (route.onClientExecute != null)\n\t\t\t\tthis.onClientExecute = route.onClientExecute;\n\n\t\t\tif (route.testCases != null)\n\t\t\t\ttestCases = route.testCases;\n\t\t}\n\n\t\tif (name === \"\")\n\t\t\tthrow new Error (`All route methods must have a name!`);\n\n\t\tthis.route = newRoute;\n\t\tthis.name = name;\n\t\tthis.type = type;\n\t\tthis.isRegistered = false;\n\t\tthis.executeSetup = false;\n\t\tthis.authCredentials = authCredentials;\n\t\tthis.onServerAuthorize = onServerAuthorize;\n\t\tthis.onRegister = onRegister;\n\t\tthis.testCases = {};\n\n\t\tif (this.route.connection.processor.mode === DeveloperMode.Development)\n\t\t{\n\t\t\tif (testCases != null)\n\t\t\t{\n\t\t\t\tif (testCases instanceof Array)\n\t\t\t\t{\n\t\t\t\t\tfor (let iIdx = 0; iIdx < testCases.length; iIdx++)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet obj = testCases[iIdx];\n\n\t\t\t\t\t\tif (typeof (obj) === \"string\")\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tconst testCaseName: string = obj;\n\t\t\t\t\t\t\tconst func: TestCaseFunction = (<TestCaseFunction>testCases[iIdx + 1]);\n\n\t\t\t\t\t\t\tthis.addTestCase (testCaseName, func);\n\t\t\t\t\t\t\tiIdx++;\n\t\t\t\t\t\t}\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tthis.addTestCase (obj);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\tfor (let key in testCases)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet obj = testCases[key];\n\n\t\t\t\t\t\tthis.addTestCase (obj);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tif (this.route.connection instanceof HotServer)\n\t\t\tthis.onServerExecute = onExecute;\n\t\t//else\n\t\t\t//this.onClientExecute = onExecute;\n\t}\n\n\t/**\n\t * Add a new test case.\n\t */\n\taddTestCase (newTestCase: TestCaseObject | string | TestCaseFunction, \n\t\t\ttestCaseFunction: TestCaseFunction = null): void\n\t{\n\t\tif (typeof (newTestCase) === \"string\")\n\t\t{\n\t\t\tconst name: string = newTestCase;\n\t\t\tconst func: TestCaseFunction = testCaseFunction;\n\n\t\t\tthis.testCases[name] = {\n\t\t\t\t\tname: name,\n\t\t\t\t\tfunc: func\n\t\t\t\t};\n\n\t\t\treturn;\n\t\t}\n\n\t\tif (typeof (newTestCase) === \"function\")\n\t\t{\n\t\t\tconst testCaseId: number = Object.keys (this.testCases).length;\n\t\t\tconst name: string = `${this.route.route}/${this.name} test case ${testCaseId}`;\n\t\t\tconst func: TestCaseFunction = (<TestCaseFunction>newTestCase);\n\n\t\t\tthis.testCases[name] = {\n\t\t\t\t\tname: name,\n\t\t\t\t\tfunc: func\n\t\t\t\t};\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst testCase: TestCaseObject = (<TestCaseObject>newTestCase);\n\t\tthis.testCases[testCase.name] = testCase;\n\t}\n}","import { HotStaq } from \"./HotStaq\";\nimport { HotLog } from \"./HotLog\";\nimport { HotAPI } from \"./HotAPI\";\nimport { HotRoute } from \"./HotRoute\";\n\n/**\n * The type of server.\n */\nexport enum HotServerType\n{\n\tHTTP,\n\tWebSockets,\n\tGenerate\n}\n\n/**\n * The server.\n */\nexport interface IHotServer\n{\n\t/**\n\t * The processor to use.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The server type.\n\t */\n\tserverType: string;\n\t/**\n\t * The API to use.\n\t */\n\tapi: HotAPI;\n\t/**\n\t * The network address to listen on.\n\t */\n\tlistenAddress: string;\n\t/**\n\t * The ports to use.\n\t */\n\tports: {\n\t\t\thttp: number;\n\t\t\thttps: number;\n\t\t};\n\t/**\n\t * SSL settings.\n\t */\n\tssl: {\n\t\t\t/**\n\t\t\t * The SSL certificate to use.\n\t\t\t */\n\t\t\tcert: string;\n\t\t\t/**\n\t\t\t * The SSL certificate key to use.\n\t\t\t */\n\t\t\tkey: string;\n\t\t\t/**\n\t\t\t * The SSL certificate CA to use.\n\t\t\t */\n\t\t\tca: string;\n\t\t};\n\t/**\n\t * Redirect HTTP traffic to HTTPS.\n\t */\n\tredirectHTTPtoHTTPS: boolean;\n\t/**\n\t * The type of server.\n\t */\n\ttype: HotServerType;\n\t/**\n\t * The logger.\n\t */\n\tlogger: HotLog;\n\t/**\n\t * Any secrets associated with this server.\n\t */\n\tsecrets: any;\n}\n\n/**\n * The server.\n */\nexport class HotServer implements IHotServer\n{\n\t/**\n\t * The processor to use.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The server type.\n\t */\n\tserverType: string;\n\t/**\n\t * The API to use.\n\t */\n\tapi: HotAPI;\n\t/**\n\t * The network address to listen on.\n\t */\n\tlistenAddress: string;\n\t/**\n\t * The ports to use.\n\t */\n\tports: {\n\t\t\thttp: number;\n\t\t\thttps: number;\n\t\t};\n\t/**\n\t * SSL settings.\n\t */\n\tssl: {\n\t\t\t/**\n\t\t\t * The SSL certificate to use.\n\t\t\t */\n\t\t\tcert: string;\n\t\t\t/**\n\t\t\t * The SSL certificate key to use.\n\t\t\t */\n\t\t\tkey: string;\n\t\t\t/**\n\t\t\t * The SSL certificate CA to use.\n\t\t\t */\n\t\t\tca: string;\n\t\t};\n\t/**\n\t * Redirect HTTP traffic to HTTPS.\n\t */\n\tredirectHTTPtoHTTPS: boolean;\n\t/**\n\t * The type of server.\n\t */\n\ttype: HotServerType;\n\t/**\n\t * The logger.\n\t */\n\tlogger: HotLog;\n\t/**\n\t * Any secrets associated with this server.\n\t */\n\tsecrets: any;\n\n\tconstructor (processor: HotStaq | HotServer)\n\t{\n\t\tif (processor instanceof HotStaq)\n\t\t{\n\t\t\tthis.processor = processor;\n\t\t\tthis.serverType = \"Server\";\n\t\t\tthis.api = null;\n\t\t\tthis.listenAddress = \"0.0.0.0\";\n\t\t\tthis.ports = {\n\t\t\t\t\thttp: 80,\n\t\t\t\t\thttps: 443\n\t\t\t\t};\n\t\t\tthis.ssl = {\n\t\t\t\t\tcert: \"\",\n\t\t\t\t\tkey: \"\",\n\t\t\t\t\tca: \"\"\n\t\t\t\t};\n\t\t\tthis.redirectHTTPtoHTTPS = true;\n\t\t\tthis.type = HotServerType.HTTP;\n\t\t\tthis.logger = processor.logger;\n\t\t\tthis.secrets = {};\n\t\t}\n\t\telse\n\t\t{\n\t\t\tthis.processor = processor.processor;\n\t\t\tthis.serverType = processor.serverType || \"Server\";\n\t\t\tthis.api = processor.api || null;\n\t\t\tthis.listenAddress = processor.listenAddress || \"0.0.0.0\";\n\t\t\tthis.ports = processor.ports || {\n\t\t\t\t\thttp: 80,\n\t\t\t\t\thttps: 443\n\t\t\t\t};\n\t\t\tthis.ssl = processor.ssl || {\n\t\t\t\t\tcert: \"\",\n\t\t\t\t\tkey: \"\",\n\t\t\t\t\tca: \"\"\n\t\t\t\t};\n\t\t\tthis.redirectHTTPtoHTTPS = processor.redirectHTTPtoHTTPS != null ? processor.redirectHTTPtoHTTPS : true;\n\t\t\tthis.type = processor.type || HotServerType.HTTP;\n\t\t\tthis.logger = processor.logger;\n\t\t\tthis.secrets = processor.secrets || {};\n\t\t}\n\t}\n\n\t/**\n\t * Set an API to this server. This will also set the associated \n\t * processor to this API as well.\n\t */\n\tasync setAPI (api: HotAPI): Promise<void>\n\t{\n\t\tthis.processor.api = api;\n\t\tthis.api = api;\n\n\t\t//if (registerRoutes === true)\n\t\t\t//await this.api.registerRoutes ();\n\t}\n\n\t/**\n\t * Register a route with the server.\n\t */\n\tasync registerRoute? (route: HotRoute): Promise<void>;\n\n\t/**\n\t * Start listening for requests.\n\t */\n\tasync listen? (): Promise<void>;\n\n\t/**\n\t * Shutdown the server.\n\t */\n\tasync shutdown? (): Promise<void>;\n}","import * as fs from \"fs\";\nimport * as ppath from \"path\";\n\nimport fetch from \"node-fetch\";\nimport validateModuleName from \"validate-npm-package-name\";\n\nimport { HotPage } from \"./HotPage\";\nimport { HotFile } from \"./HotFile\";\n\nimport { HotComponent, HotComponentOutput, IHotComponent } from \"./HotComponent\";\nimport { HotLog, HotLogLevel } from \"./HotLog\";\nimport { HotAPI } from \"./HotAPI\";\nimport { HotServer } from \"./HotServer\";\nimport { DeveloperMode, Hot } from \"./Hot\";\nimport { HotClient } from \"./HotClient\";\n\nimport { HotTester } from \"./HotTester\";\nimport { HotTesterAPI } from \"./HotTesterAPI\";\nimport { HotTestDriver } from \"./HotTestDriver\";\nimport { HotTestDestination, HotTestMap } from \"./HotTestMap\";\nimport { ServableFileExtension } from \"./HotHTTPServer\";\n\nvar HotTesterMocha: any = null;\nvar HotTesterMochaSelenium: any = null;\nvar HotTestSeleniumDriver: any = null;\n\n/**\n * A map path for testing.\n */\nexport interface HotSiteMapPath\n{\n\t/**\n\t * If set to true, this will start automatically when tests start.\n\t * The default is true.\n\t */\n\tautoStart?: boolean;\n\t/**\n\t * The path to the \n\t */\n\tpath?: string;\n}\n\n/**\n * A route used in a HotSite.\n */\nexport interface HotSiteRoute\n{\n\t/**\n\t * The name of the route. Will appear in the title.\n\t */\n\tname: string;\n\t/**\n\t * The url to the file to load.\n\t */\n\turl: string;\n\t/**\n\t * The name of the API to interface with.\n\t */\n\tapi?: string;\n\t/**\n\t * The order in which destinations are supposed to execute. This is \n\t * ignored if the destinations are an array.\n\t */\n\tdestinationOrder?: string[];\n\t/**\n\t * The HotTesterMap to use. This can be the name of an \n\t * existing one attached to the selected tester, or \n\t * can be an array of destinations that will be used to \n\t * create a new map.\n\t */\n\tmap?: string | string[] | { [name: string]: string | HotSiteMapPath; } | HotSiteMapPath[];\n}\n\n/**\n * A HotSite to load. This SHOULD NOT contain any private secret keys, passwords, \n * or database connection information related to the server. As such, future \n * versions of the HotSite interface should not contain any database related \n * connection info.\n */\nexport interface HotSite\n{\n\t/**\n\t * The name of this HotSite.\n\t */\n\tname: string;\n\t/**\n\t * The version of this HotSite.\n\t */\n\tversion?: string;\n\t/**\n\t * The description of this HotSite.\n\t */\n\tdescription?: string;\n\t/**\n\t * The path to the current HotSite. This is filled in during parsing.\n\t */\n\thotsitePath?: string;\n\t/**\n\t * Additional web server configuration.\n\t */\n\tserver?: {\n\t\t\t/**\n\t\t\t * The default name for a served Hott file.\n\t\t\t */\n\t\t\tname?: string;\n\t\t\t/**\n\t\t\t * If set to true, this will serve ALL files, including potentially secret files.\n\t\t\t */\n\t\t\tserveSecretFiles?: boolean;\n\t\t\t/**\n\t\t\t * Serve the following file extensions when requested.\n\t\t\t */\n\t\t\tserveFileExtensions?: (string | ServableFileExtension)[];\n\t\t\t/**\n\t\t\t * The name of the API to interface with across all pages.\n\t\t\t */\n\t\t\tglobalApi?: string;\n\t\t\t/**\n\t\t\t * The base url for the server.\n\t\t\t */\n\t\t\turl?: string;\n\t\t\t/**\n\t\t\t * The JavaScript source path.\n\t\t\t */\n\t\t\tjsSrcPath?: string;\n\t\t\t/**\n\t\t\t * The ports to use.\n\t\t\t */\n\t\t\tports?: {\n\t\t\t\t\t/**\n\t\t\t\t\t * The web HTTP port to serve on.\n\t\t\t\t\t */\n\t\t\t\t\thttp?: number;\n\t\t\t\t\t/**\n\t\t\t\t\t * The web HTTPS port to serve on.\n\t\t\t\t\t */\n\t\t\t\t\thttps?: number;\n\t\t\t\t\t/**\n\t\t\t\t\t * If set to true, this will redirect from HTTP to HTTPS for a web and web-api server.\n\t\t\t\t\t */\n\t\t\t\t\tredirectHTTPtoHTTPS?: boolean;\n\t\t\t\t\t/**\n\t\t\t\t\t * The api HTTP port to serve on.\n\t\t\t\t\t */\n\t\t\t\t\tapiHttp?: number;\n\t\t\t\t\t/**\n\t\t\t\t\t * The api HTTPS port to serve on.\n\t\t\t\t\t */\n\t\t\t\t\tapiHttps?: number;\n\t\t\t\t\t/**\n\t\t\t\t\t * If set to true, this will redirect from HTTP to HTTPS for an api server.\n\t\t\t\t\t */\n\t\t\t\t\tapiRedirectHTTPtoHTTPS?: boolean;\n\t\t\t\t};\n\t\t\t/**\n\t\t\t * The list of directory to serve to the client from the server.\n\t\t\t */\n\t\t\tserveDirectories?: {\n\t\t\t\t\t/**\n\t\t\t\t\t * The web route to take.\n\t\t\t\t\t */\n\t\t\t\t\troute: string;\n\t\t\t\t\t/**\n\t\t\t\t\t * The local filesystem path to serve pages from.\n\t\t\t\t\t */\n\t\t\t\t\tlocalPath: string;\n\t\t\t\t}[];\n\t\t\t/**\n\t\t\t * How to handle errors.\n\t\t\t */\n\t\t\terrors?: {\n\t\t\t\t/**\n\t\t\t\t * On a 404, serve a local file.\n\t\t\t\t */\n\t\t\t\ton404?: string;\n\t\t\t\t/**\n\t\t\t\t * On an error other than a 404, serve a local file.\n\t\t\t\t */\n\t\t\t\tonOther?: string;\n\t\t\t};\n\t\t};\n\t/**\n\t * Testing related functionality.\n\t */\n\ttesting?: {\n\t\t\tweb?: {\n\t\t\t\t/**\n\t\t\t\t * The tester class to use. EX: HotTesterMochaSelenium\n\t\t\t\t */\n\t\t\t\ttester?: string;\n\t\t\t\t/**\n\t\t\t\t * The name of the tester to use.\n\t\t\t\t */\n\t\t\t\ttesterName?: string;\n\t\t\t\t/**\n\t\t\t\t * If set to true, this will create a new tester.\n\t\t\t\t * Default Value: true\n\t\t\t\t */\n\t\t\t\tcreateNewTester?: boolean;\n\t\t\t\t/**\n\t\t\t\t * The url that connects to the tester api server.\n\t\t\t\t */\n\t\t\t\ttesterAPIUrl?: string;\n\t\t\t\t/**\n\t\t\t\t * The name of the test driver to use.\n\t\t\t\t */\n\t\t\t\tdriver?: string;\n\t\t\t\t/**\n\t\t\t\t * The number of milliseconds to wait before executing the next command.\n\t\t\t\t * Default is set to 20.\n\t\t\t\t */\n\t\t\t\tcommandDelay?: number;\n\t\t\t\t/**\n\t\t\t\t * The url to the html that loads the hott files.\n\t\t\t\t */\n\t\t\t\tlaunchpadUrl?: string;\n\t\t\t\t/**\n\t\t\t\t * The maps to test in order.\n\t\t\t\t */\n\t\t\t\tmaps?: string[];\n\t\t\t},\n\t\t\tapi?: {\n\t\t\t\t/**\n\t\t\t\t * The tester class to use. EX: HotTesterMocha\n\t\t\t\t */\n\t\t\t\ttester?: string;\n\t\t\t\t/**\n\t\t\t\t * The name of the tester to use.\n\t\t\t\t */\n\t\t\t\ttesterName?: string;\n\t\t\t\t/**\n\t\t\t\t * If set to true, this will create a new tester.\n\t\t\t\t * Default Value: true\n\t\t\t\t */\n\t\t\t\tcreateNewTester?: boolean;\n\t\t\t\t/**\n\t\t\t\t * The url that connects to the tester api server.\n\t\t\t\t */\n\t\t\t\ttesterAPIUrl?: string;\n\t\t\t\t/**\n\t\t\t\t * The name of the test driver to use.\n\t\t\t\t */\n\t\t\t\tdriver?: string;\n\t\t\t\t/**\n\t\t\t\t * The url to the html that loads the hott files.\n\t\t\t\t */\n\t\t\t\tlaunchpadUrl?: string;\n\t\t\t\t/**\n\t\t\t\t * The maps to test in order.\n\t\t\t\t */\n\t\t\t\tmaps?: string[];\n\t\t\t}\n\t\t};\n\t/**\n\t * The routes to load.\n\t */\n\troutes?: {\n\t\t\t[routeName: string]: HotSiteRoute;\n\t\t};\n\t/**\n\t * The available APIs on the server. The server must already have these \n\t * loaded.\n\t */\n\tapis?: {\n\t\t\t[name: string]: {\n\t\t\t\t\t/**\n\t\t\t\t\t * The JS API file to load.\n\t\t\t\t\t */\n\t\t\t\t\tjsapi?: string;\n\t\t\t\t\t/**\n\t\t\t\t\t * The exported JS library name to use.\n\t\t\t\t\t */\n\t\t\t\t\tlibraryName?: string;\n\t\t\t\t\t/**\n\t\t\t\t\t * The name of the api to use.\n\t\t\t\t\t */\n\t\t\t\t\tapiName?: string;\n\t\t\t\t\t/**\n\t\t\t\t\t * The port to use.\n\t\t\t\t\t */\n\t\t\t\t\tport?: number;\n\t\t\t\t\t/**\n\t\t\t\t\t * The public base url for the api.\n\t\t\t\t\t */\n\t\t\t\t\turl?: string;\n\t\t\t\t\t/**\n\t\t\t\t\t * The server-side filepath for the api.\n\t\t\t\t\t */\n\t\t\t\t\tfilepath?: string;\n\t\t\t\t\t/**\n\t\t\t\t\t * The maps to test in order.\n\t\t\t\t\t */\n\t\t\t\t\tmap?: string[];\n\t\t\t\t};\n\t\t};\n\t/**\n\t * Public keys that are embedded into the page.\n\t */\n\tpublicKeys?: {\n\t\t\t[name: string]: string | {\n\t\t\t\t\t/**\n\t\t\t\t\t * The key of an API secret to pass to the site to \n\t\t\t\t\t * be used publicly.\n\t\t\t\t\t */\n\t\t\t\t\tpassSecretFromAPI?: string;\n\t\t\t\t\t/**\n\t\t\t\t\t * Get the public secret from an environment variable.\n\t\t\t\t\t */\n\t\t\t\t\tenv?: string;\n\t\t\t\t};\n\t\t};\n\t/**\n\t * The components to load and register.\n\t */\n\tcomponents?: {\n\t\t\t[name: string]: {\n\t\t\t\t\t/**\n\t\t\t\t\t * The url to the component to load and register.\n\t\t\t\t\t */\n\t\t\t\t\turl: string;\n\t\t\t\t};\n\t\t};\n\t/**\n\t * The files to load and save in memory.\n\t */\n\tfiles?: {\n\t\t\t[name: string]: {\n\t\t\t\t\t/**\n\t\t\t\t\t * The url to the file to load.\n\t\t\t\t\t */\n\t\t\t\t\turl: string;\n\t\t\t\t};\n\t\t};\n\t/**\n\t * If set to true, this will disable loading files into memory.\n\t */\n\tdisableFileLoading?: boolean;\n}\n\n/**\n * The options to use when starting a page.\n */\nexport interface HotStartOptions\n{\n\t/**\n\t * The Hott site to load.\n\t */\n\turl?: string;\n\t/**\n\t * The content to display.\n\t */\n\tcontent?: string;\n\t/**\n\t * The name of the page to load.\n\t */\n\tname?: string;\n\t/**\n\t * The processor to use to load the page.\n\t */\n\tprocessor?: HotStaq;\n\t/**\n\t * Any arguments to pass to the new page.\n\t */\n\targs?: any;\n\t/**\n\t * The name of the tester to use.\n\t */\n\ttesterName?: string;\n\t/**\n\t * The name of the tester map to use.\n\t */\n\ttesterMap?: string;\n\t/**\n\t * The base url for the tester api.\n\t */\n\ttesterAPIBaseUrl?: string;\n\t/**\n\t * The url to the html that loads the hott file that's \n\t * pointed at the url above.\n\t */\n\ttesterLaunchpadUrl?: string;\n}\n\n/**\n * The main class that handles all HTML preprocessing, then outputs the \n * results.\n */\nexport interface IHotStaq\n{\n\t/**\n\t * The api that's used to communicate with.\n\t */\n\tapi?: HotAPI;\n\t/**\n\t * The tester api that's used to communicate with.\n\t */\n\ttesterAPI?: HotAPI;\n\t/**\n\t * Indicates what type of execution this is.\n\t */\n\tmode?: DeveloperMode;\n\t/**\n\t * The pages that can be constructed.\n\t */\n\tpages?: { [name: string]: HotPage };\n\t/**\n\t * The components that can be constructed.\n\t */\n\tcomponents?: {\n\t\t[tagName: string]: {\n\t\t\t\tcomponentType: (new  (copy: IHotComponent | HotStaq, api?: HotAPI) => HotComponent), \n\t\t\t\tprocessor: HotStaq, \n\t\t\t\tapi: HotAPI\n\t\t\t}\n\t\t};\n\t/**\n\t * The files that can be stored for later use.\n\t */\n\tfiles?: { [name: string]: HotFile };\n\t/**\n\t * The loaded hotsite.\n\t */\n\thotSite?: HotSite;\n}\n\n/**\n * The main class that handles all HTML preprocessing, then outputs the \n * results.\n */\nexport class HotStaq implements IHotStaq\n{\n\t/**\n\t * The current version of HotStaq.\n\t */\n\tstatic version: string = \"0.6.21\";\n\t/**\n\t * Indicates if this is a web build.\n\t */\n\tstatic isWeb: boolean = false;\n\t/**\n\t * Indicates if this is ready for testing.\n\t */\n\tstatic isReadyForTesting: boolean = false;\n\t/**\n\t * Executes this event when this page is ready for testing.\n\t */\n\tstatic onReadyForTesting: () => Promise<void> = null;\n\t/**\n\t * Errors to execute when something goes wrong.\n\t */\n\tstatic errors: { [name: string]: { redirectToUrl?: string; func?: (errType: string) => void; }; } = {};\n\t/**\n\t * Indicates what type of execution this is.\n\t */\n\tmode: DeveloperMode;\n\t/**\n\t * The api that's used to communicate with.\n\t */\n\tapi: HotAPI;\n\t/**\n\t * The tester api that's used to communicate with.\n\t */\n\ttesterAPI: HotAPI;\n\t/**\n\t * The pages that can be constructed.\n\t */\n\tpages: { [name: string]: HotPage };\n\t/**\n\t * The components that can be constructed.\n\t */\n\tcomponents: {\n\t\t[tagName: string]: {\n\t\t\t\tcomponentType: (new  (copy: IHotComponent | HotStaq, api?: HotAPI) => HotComponent), \n\t\t\t\tprocessor: HotStaq, \n\t\t\t\tapi: HotAPI\n\t\t\t}\n\t\t};\n\t/**\n\t * The files that can be stored for later use.\n\t */\n\tfiles: { [name: string]: HotFile };\n\t/**\n\t * The loaded hotsite.\n\t */\n\thotSite: HotSite;\n\t/**\n\t * The api content to use when about to load HotStaq.\n\t */\n\tapiContent: string;\n\t/**\n\t * The tester api content to use when about to load HotStaq.\n\t */\n\ttesterApiContent: string;\n\t/**\n\t * The page content to use when about to load HotStaq.\n\t */\n\tpageContent: string;\n\t/**\n\t * The logger.\n\t */\n\tlogger: HotLog;\n\t/**\n\t * The public keys to be exposed.\n\t */\n\tpublicKeys: any;\n\t/**\n\t * The testers that will be used to execute tests.\n\t */\n\ttesters: { [name: string]: HotTester };\n\n\tconstructor (copy: IHotStaq = {})\n\t{\n\t\tthis.api = copy.api || null;\n\t\tthis.testerAPI = copy.testerAPI || null;\n\t\tthis.mode = copy.mode || DeveloperMode.Production;\n\t\tthis.pages = copy.pages || {};\n\t\tthis.components = copy.components || {};\n\t\tthis.files = copy.files || {};\n\t\tthis.hotSite = copy.hotSite || null;\n\t\tthis.apiContent = `\n\t\t\t{\n\t\t\t\tvar %api_name% = %api_exported_name%.%api_name%;\n\t\t\t\tvar newHotClient = new HotClient (processor);\n\t\t\t\tvar newapi = new %api_name% (%base_url%, newHotClient);\n\t\t\t\tnewHotClient.api = newapi;\n\t\t\t\tprocessor.api = newapi;\n\t\t\t}`;\n\t\tthis.testerApiContent = `\n\t\t\tvar HotTesterAPI = HotStaqWeb.HotTesterAPI;\n\t\t\tvar newHotTesterClient = new HotClient (processor);\n\t\t\tvar newtesterapi = new HotTesterAPI (%base_tester_url%, newHotTesterClient);\n\t\t\tnewHotTesterClient.testerAPI = newtesterapi;\n\t\t\tprocessor.testerAPI = newtesterapi;`;\n\t\tthis.pageContent = \n`<!DOCTYPE html>\n<html>\n\n<head>\n\t<title>%title%</title>\n\n\t<script type = \"text/javascript\" src = \"%hotstaq_js_src%\"></script>\n\t<script type = \"text/javascript\">\n\t\twindow.HotStaq = HotStaqWeb.HotStaq;\n\t\twindow.HotClient = HotStaqWeb.HotClient;\n\t\twindow.HotAPI = HotStaqWeb.HotAPI;\n\t\twindow.Hot = HotStaqWeb.Hot;\n\t</script>\n\n%apis_to_load%\n\n\t<script type = \"text/javascript\">\n\t\tfunction hotstaq_startApp ()\n\t\t{\n\t\t\tlet tempMode = 0;\n\n\t\t\tif (window[\"Hot\"] != null)\n\t\t\t\ttempMode = Hot.Mode;\n\n\t\t\t%load_hot_site%\n\n\t\t\tvar processor = new HotStaq ();\n\t\t\tvar promises = [];\n\t\t\t%developer_mode%\n\n\t\t\t%api_code%\n\n\t\t\t%public_secrets%\n\t\t\t%tester_api%\n\t\t\t%load_files%\n\n\t\t\tprocessor.mode = tempMode;\n\n\t\t\tPromise.all (promises).then (function ()\n\t\t\t\t{\n\t\t\t\t\tHotStaq.displayUrl ({\n\t\t\t\t\t\t\turl: \"%url%\",\n\t\t\t\t\t\t\tname: \"%title%\",\n\t\t\t\t\t\t\tprocessor: processor,\n\t\t\t\t\t\t\targs: %args%,\n\t\t\t\t\t\t\ttesterName: %tester_name%,\n\t\t\t\t\t\t\ttesterMap: %tester_map%,\n\t\t\t\t\t\t\ttesterAPIBaseUrl: %tester_api_base_url%,\n\t\t\t\t\t\t\ttesterLaunchpadUrl: %tester_launchpad_url%\n\t\t\t\t\t\t});\n\t\t\t\t});\n\t\t}\n\n\t\thotstaq_startApp ();\n\t</script>\n</head>\n\n<body>\n</body>\n\n</html>`;\n\t\tthis.logger = new HotLog (HotLogLevel.None);\n\t\tthis.publicKeys = {};\n\t\tthis.testers = {};\n\t}\n\n\t/**\n\t * Parse a boolean value.\n\t */\n\tstatic parseBoolean (value: string): boolean\n\t{\n\t\tvalue = value.toLowerCase ();\n\n\t\tif (value === \"true\")\n\t\t\treturn (true);\n\n\t\tif (value === \"false\")\n\t\t\treturn (false);\n\n\t\tif (value === \"yes\")\n\t\t\treturn (true);\n\n\t\tif (value === \"no\")\n\t\t\treturn (false);\n\n\t\tif (value === \"yep\")\n\t\t\treturn (true);\n\n\t\tif (value === \"nah\")\n\t\t\treturn (false);\n\n\t\ttry\n\t\t{\n\t\t\tif (parseInt (value) != 0)\n\t\t\t\treturn (true);\n\t\t}\n\t\tcatch (ex)\n\t\t{\n\t\t}\n\n\t\treturn (false);\n\t}\n\n\t/**\n\t * Check if a required parameter exists inside an object. If it exists, return the value.\n\t */\n\tstatic getParam (name: string, objWithParam: any, required: boolean = true, throwException: boolean = true): any\n\t{\n\t\tlet value: any = objWithParam[name];\n\n\t\tif (value == null)\n\t\t{\n\t\t\tif (required === true)\n\t\t\t{\n\t\t\t\tif (throwException === true)\n\t\t\t\t\tthrow new Error (`Missing required parameter ${name}.`);\n\t\t\t}\n\t\t}\n\n\t\tif (typeof (value) === \"string\")\n\t\t{\n\t\t\tif (required === true)\n\t\t\t{\n\t\t\t\tif (value === \"\")\n\t\t\t\t{\n\t\t\t\t\tif (throwException === true)\n\t\t\t\t\t\tthrow new Error (`Missing required parameter ${name}.`);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn (value);\n\t}\n\n\t/**\n\t * Check if a required parameter exists inside an object. If it exists, return the value.\n\t * If it does not exist, return a default value instead.\n\t */\n\tstatic getParamDefault (name: string, objWithParam: any, defaultValue: any): any\n\t{\n\t\tlet value: any = objWithParam[name];\n\n\t\tif (value == null)\n\t\t\treturn (defaultValue);\n\n\t\tif (typeof (value) === \"string\")\n\t\t{\n\t\t\tif (value === \"\")\n\t\t\t\treturn (defaultValue);\n\t\t}\n\n\t\treturn (value);\n\t}\n\n\t/**\n\t * Execute an error. This cannot be an async function due to the nature of how this works.\n\t */\n\tstatic executeError (errType: string)\n\t{\n\t\tif (HotStaq.errors[errType] != null)\n\t\t{\n\t\t\tlet url: string = HotStaq.errors[errType].redirectToUrl;\n\n\t\t\tif (url != null)\n\t\t\t{\n\t\t\t\tif (url !== \"\")\n\t\t\t\t{\n\t\t\t\t\twindow.location.href = url;\n\n\t\t\t\t\treturn;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tlet func = HotStaq.errors[errType].func;\n\n\t\t\tif (func != null)\n\t\t\t\tfunc (errType);\n\t\t}\n\t}\n\n\t/**\n\t * Wait for a number of milliseconds.\n\t */\n\tstatic async wait (numMilliseconds: number): Promise<void>\n\t{\n\t\treturn (await new Promise ((resolve, reject) =>\n\t\t\t{\n\t\t\t\tsetTimeout (() =>\n\t\t\t\t\t{\n\t\t\t\t\t\tresolve ();\n\t\t\t\t\t}, numMilliseconds);\n\t\t\t}));\n\t}\n\n\t/**\n\t * Add a page.\n\t */\n\taddPage (page: HotPage): void\n\t{\n\t\tthis.pages[page.name] = page;\n\t}\n\n\t/**\n\t * Get a page to process.\n\t */\n\tgetPage (pageName: string): HotPage\n\t{\n\t\treturn (this.pages[pageName]);\n\t}\n\n\t/**\n\t * Add a file.\n\t */\n\taddFile (file: HotFile): void\n\t{\n\t\tlet name: string = file.name;\n\n\t\tif (name === \"\")\n\t\t\tname = file.localFile;\n\n\t\tif (name === \"\")\n\t\t\tname = file.url;\n\n\t\tthis.files[name] = file;\n\t}\n\n\t/**\n\t * Get a file.\n\t */\n\tgetFile (name: string): HotFile\n\t{\n\t\tif (this.files[name] == null)\n\t\t\tthrow new Error (`Unable to find file ${name}`);\n\n\t\treturn (this.files[name]);\n\t}\n\n\t/** \n\t * Keep the context the object is currently in.\n\t * \n\t * @param func The document element's id.\n\t * @param context The object to remain in context.\n\t * @param [val=undefined] An additional value to pass to the context.\n\t * @return The returned result from the function func.\n\t */\n\tstatic keepContext(func: Function, context: any, val?: any): any\n\t{\n\t\tvar objReturn = function()\n\t\t\t{\n\t\t\t\tvar aryArgs = Array.prototype.slice.call(arguments);\n\n\t\t\t\tif (val != undefined)\n\t\t\t\t\taryArgs.push(val);\n\n\t\t\t\tif (context == null)\n\t\t\t\t\treturn func.apply(this, aryArgs);\n\t\t\t\telse\n\t\t\t\t\treturn func.apply(context, aryArgs);\n\t\t\t};\n\n\t\treturn objReturn;\n\t}\n\n\t/**\n\t * Add and register a component.\n\t */\n\taddComponent (ComponentType: (new  (copy: IHotComponent | HotStaq, api?: HotAPI) => HotComponent), api: HotAPI = null, \n\t\telementOptions: ElementDefinitionOptions = undefined): void\n\t{\n\t\tlet tempApi = this.api\n\t\t\n\t\tif (api != null)\n\t\t\ttempApi = api;\n\n\t\tlet tempComponentObj = new ComponentType (this, tempApi);\n\n\t\tif (this.components[tempComponentObj.tag] != null)\n\t\t\tthrow new Error (`Component ${tempComponentObj.tag} already exists!`);\n\n\t\tthis.components[tempComponentObj.tag] = { componentType: ComponentType, processor: this, api: tempApi };\n\t\tthis.registerComponent (tempComponentObj.tag, elementOptions);\n\t}\n\n\t/**\n\t * Correct any HTML prior to DOM parsing. This only accounts for <tr> currently.\n\t */\n\tprotected static fixHTML (str: string): { fixedStr: string, querySelector: string; }\n\t{\n\t\t// Take into account the difference between XML and HTML.\n\t\tconst tempStr: string = str.replace(/ \\/>/g, '>').replace(\n\t\t\t/(<(area|base|br|col|command|embed|hr|img|input|keygen|link|meta|param|source|track|wbr).*?>)/g, '$1</$2>');\n\t\tconst parsedXML = new DOMParser ().parseFromString (`<xml>${tempStr}</xml>`, \"text/xml\");\n\t\tlet querySelector: string = \"\";\n\n\t\tif (parsedXML.documentElement.children.length > 0)\n\t\t{\n\t\t\tconst tagName: string = parsedXML.documentElement.children[0].tagName.toLowerCase ();\n\n\t\t\tif (tagName === \"tr\")\n\t\t\t{\n\t\t\t\tstr = `<table>${str}</table>`;\n\t\t\t\tquerySelector = \"tbody\";\n\t\t\t}\n\n\t\t\tif (tagName === \"th\")\n\t\t\t{\n\t\t\t\tstr = `<table>${str}</table>`;\n\t\t\t\tquerySelector = tagName;\n\t\t\t}\n\t\t}\n\n\t\treturn ({ fixedStr: str, querySelector: querySelector });\n\t}\n\n\t/**\n\t * Register a component for use as a HTML tag.\n\t */\n\tprotected registerComponent (tag: string, elementOptions: ElementDefinitionOptions = undefined): void\n\t{\n\t\tif ((tag == null) || (tag === \"\"))\n\t\t\tthrow new Error (`All components must have a tag!`);\n\n\t\tif (customElements.get (tag) !== undefined)\n\t\t{\n\t\t\t/// @fixme This element has already been defined. Should this throw an error or warning? I don't think it should...\n\n\t\t\treturn;\n\t\t}\n\n\t\tlet processorComponents = this.components;\n\n\t\tcustomElements.define (tag, class extends HTMLElement\n\t\t\t{\n\t\t\t\t/**\n\t\t\t\t * The connected HotComponent.\n\t\t\t\t */\n\t\t\t\tcomponent: HotComponent;\n\t\t\t\n\t\t\t\tconstructor ()\n\t\t\t\t{\n\t\t\t\t\tsuper ();\n\n\t\t\t\t\tlet componentInfo = processorComponents[tag];\n\t\t\t\t\tthis.component = new componentInfo.componentType (componentInfo.processor, componentInfo.api);\n\t\t\t\t}\n\t\t\t\n\t\t\t\t/**\n\t\t\t\t * This helps parse <tr> and other tags that do not have a parent.\n\t\t\t\t * \n\t\t\t\t * Thanks Brandon McConnell!\n\t\t\t\t * \n\t\t\t\t * From: https://stackoverflow.com/questions/67313479/make-parsefromstring-parse-without-validation\n\t\t\t\t * \n\t\t\t\t * @todo May remove this as it does not seem to work well in a lot of edge cases.\n\t\t\t\t */\n\t\t\t\tprotected looseParseFromString (parser: DOMParser, str: string) {\n\t\t\t\t\tstr = str.replace(/ \\/>/g, '>').replace(/(<(area|base|br|col|command|embed|hr|img|input|keygen|link|meta|param|source|track|wbr).*?>)/g, '$1</$2>');\n\t\t\t\t\tconst xdom = parser.parseFromString('<xml>'+str+'</xml>', 'text/xml');\n\t\t\t\t\tconst hdom = parser.parseFromString('', 'text/html');\n\t\t\t\t\tfor (let elem of Array.from(xdom.documentElement.children)) {\n\t\t\t\t\t\t/// @ts-ignore\n\t\t\t\t\t\thdom.body.appendChild(elem);\n\t\t\t\t\t}\n\t\t\t\t\tfor (let elem of Array.from(hdom.querySelectorAll('area,base,br,col,command,embed,hr,img,input,keygen,link,meta,param,source,track,wbr'))) {\n\t\t\t\t\t\t/// @ts-ignore\n\t\t\t\t\t\telem.outerHTML = '<'+elem.outerHTML.slice(1).split('<')[0];\n\t\t\t\t\t}\n\t\t\t\t\treturn hdom;\n\t\t\t\t}\n\n\t\t\t\tget observedAttributes(): string[] /// @fixme Does this REALLY have to be static? Awful if it does...\n\t\t\t\t{\n\t\t\t\t\treturn (this.component.observedAttributes);\n\t\t\t\t}\n\t\t\t\n\t\t\t\tasync connectedCallback ()\n\t\t\t\t{\n\t\t\t\t\tlet compHtmlElement = this;\n\t\t\t\n\t\t\t\t\t// @ts-ignore\n\t\t\t\t\tcompHtmlElement.hotComponent = this.component;\n\t\t\t\n\t\t\t\t\tthis.component.htmlElements = [compHtmlElement];\n\t\t\t\t\tthis.component.inner = this.innerHTML;\n\t\t\t\n\t\t\t\t\tif (this.component.handleAttributes != null)\n\t\t\t\t\t\tawait this.component.handleAttributes (this.attributes);\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tfor (let iIdx = 0; iIdx < this.attributes.length; iIdx++)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tconst attr: Attr = this.attributes[iIdx];\n\t\t\t\t\t\t\tconst attrName: string = attr.name.toLowerCase ();\n\t\t\t\t\t\t\tconst attrValue: string = attr.value;\n\t\t\t\n\t\t\t\t\t\t\tif (attrName === \"id\")\n\t\t\t\t\t\t\t\tthis.component.name = attrValue;\n\t\t\t\n\t\t\t\t\t\t\tif (attrName === \"name\")\n\t\t\t\t\t\t\t\tthis.component.name = attrValue;\n\t\t\t\n\t\t\t\t\t\t\tif (attrName === \"value\")\n\t\t\t\t\t\t\t\tthis.component.value = attrValue;\n\t\t\t\n\t\t\t\t\t\t\tif (attrName.indexOf (\"hot-\") > -1)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tconst attrTempName: string = attrName.substring (4);\n\t\t\t\n\t\t\t\t\t\t\t\t/// @ts-ignore\n\t\t\t\t\t\t\t\tthis.component[attrTempName] = attrValue;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\tif (this.component.onPreOutput != null)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (await this.component.onPreOutput () === false)\n\t\t\t\t\t\t\treturn;\n\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\tlet outputs = await this.component.output ();\n\t\t\t\n\t\t\t\t\tif (this.component.onPostOutput != null)\n\t\t\t\t\t\toutputs = await this.component.onPostOutput (outputs);\n\t\t\t\n\t\t\t\t\tlet componentOutputs: HotComponentOutput[] = [];\n\t\t\t\n\t\t\t\t\tif (typeof (outputs) === \"string\")\n\t\t\t\t\t\tcomponentOutputs.push ({ html: outputs });\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tif (outputs instanceof Array)\n\t\t\t\t\t\t\tcomponentOutputs = outputs;\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tcomponentOutputs = [outputs];\n\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\tfor (let iKdx = 0; iKdx < componentOutputs.length; iKdx++)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet output = componentOutputs[iKdx];\n\t\t\t\t\t\tlet htmlStr: string = output.html;\n\t\t\t\t\t\tlet addFunctionsTo: string = \"\";\n\t\t\t\n\t\t\t\t\t\tif (output.addFunctionsTo != null)\n\t\t\t\t\t\t\taddFunctionsTo = output.addFunctionsTo;\n\t\t\t\n\t\t\t\t\t\tlet str: string = HotFile.parseContent (htmlStr, true, { \"outputCommands\": false });\n\t\t\t\n\t\t\t\t\t\tif (this.component.onParsed != null)\n\t\t\t\t\t\t\tstr = await this.component.onParsed (str);\n\t\t\t\n\t\t\t\t\t\tlet htmlHandler: { fixedStr: string, querySelector: string; } = { fixedStr: \"\", querySelector: \"\" };\n\t\t\t\n\t\t\t\t\t\tif (this.component.onFixHTML != null)\n\t\t\t\t\t\t\thtmlHandler = await this.component.onFixHTML (str);\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\thtmlHandler = HotStaq.fixHTML (str);\n\t\t\t\n\t\t\t\t\t\tlet newDOM: Document = null;\n\t\t\t\t\t\tlet newObj: HTMLElement = null;\n\t\t\t\n\t\t\t\t\t\tif (this.component.onParseDOM != null)\n\t\t\t\t\t\t\tnewDOM = await this.component.onParseDOM (htmlHandler.fixedStr);\n\t\t\t\t\t\telse\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\t/// @ts-ignore\n\t\t\t\t\t\t\t//newDOM = this.looseParseFromString (new DOMParser (), str);\n\t\t\t\t\t\t\tnewDOM = new DOMParser ().parseFromString (htmlHandler.fixedStr, \"text/html\");\n\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\tif (newDOM.body.children.length < 1)\n\t\t\t\t\t\t\tthrow new Error (`No component output from ${this.component.name}`);\n\t\t\t\n\t\t\t\t\t\tif (newDOM.body.children.length > 1)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet throwErr: boolean = true;\n\t\t\t\n\t\t\t\t\t\t\tfor (let iIdx = 0; iIdx < newDOM.body.children.length; iIdx++)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tlet child = newDOM.body.children[iIdx];\n\t\t\t\n\t\t\t\t\t\t\t\tif (child instanceof HTMLElement)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tif (child.tagName.toLowerCase () === \"parsererror\")\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tnewObj = child;\n\t\t\t\t\t\t\t\t\t\tthrowErr = false;\n\t\t\t\n\t\t\t\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\t\tif (throwErr === true)\n\t\t\t\t\t\t\t\tthrow new Error (`Only a single html element can come from component ${this.component.name}, multiple elements were detected.`);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (htmlHandler.querySelector === \"\")\n\t\t\t\t\t\t\tnewObj = (<HTMLElement>newDOM.body.children[0]);\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tnewObj = newDOM.querySelector (htmlHandler.querySelector);\n\n\t\t\t\t\t\tlet childrenToReadd: Node[] = [];\n\t\t\t\n\t\t\t\t\t\t// Save the children from being replaced.\n\t\t\t\t\t\tfor (let iIdx = (this.children.length - 1); iIdx > -1; iIdx--)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet child: Node = this.children[iIdx];\n\t\t\t\n\t\t\t\t\t\t\tchildrenToReadd.push (this.removeChild (child));\n\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\tthis.replaceWith (newObj);\n\t\t\t\n\t\t\t\t\t\tif (this.component.click != null)\n\t\t\t\t\t\t\tnewObj.onclick = this.component.click.bind (this.component);\n\t\t\t\n\t\t\t\t\t\tfor (let key in this.component.events)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet event = this.component.events[key];\n\t\t\t\n\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\tnewObj.addEventListener (event.type, event.func, event.options);\n\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\tlet objectFunctions: string[] = Object.getOwnPropertyNames (this.component.constructor.prototype);\n\t\t\t\n\t\t\t\t\t\t// Associate any functions to the newly created element.\n\t\t\t\t\t\tfor (let iIdx = 0; iIdx < objectFunctions.length; iIdx++)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet objFunc: string = objectFunctions[iIdx];\n\t\t\t\n\t\t\t\t\t\t\tif (objFunc === \"constructor\")\n\t\t\t\t\t\t\t\tcontinue;\n\t\t\t\n\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\tlet prop = this.component[objFunc];\n\t\t\t\n\t\t\t\t\t\t\tif (typeof (prop) === \"function\")\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tlet isNewFunction: boolean = true;\n\t\t\t\n\t\t\t\t\t\t\t\t// Go through each function in the base HotComponent and see \n\t\t\t\t\t\t\t\t// if there's any matches. If there's a match, that means \n\t\t\t\t\t\t\t\t// we're trying to add an existing function, and we don't\n\t\t\t\t\t\t\t\t// wanna do that. Skip it.\n\t\t\t\t\t\t\t\tfor (let key2 in HotComponent.prototype)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tif (objFunc === key2)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tisNewFunction = false;\n\t\t\t\n\t\t\t\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\t\t\tif (isNewFunction === true)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\tnewObj[objFunc] = HotStaq.keepContext (this.component[objFunc], this.component);\n\t\t\t\n\t\t\t\t\t\t\t\t\tif (addFunctionsTo !== \"\")\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tlet query: HTMLElement = document.querySelector (addFunctionsTo);\n\t\t\t\n\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\tquery[objFunc] = HotStaq.keepContext (this.component[objFunc], this.component);\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\tif (this.component.onPrePlace != null)\n\t\t\t\t\t\t\tnewObj = await this.component.onPrePlace (newObj);\n\t\t\t\n\t\t\t\t\t\tlet compHtmlElement2: HTMLElement = await this.component.onCreated (newObj);\n\t\t\t\n\t\t\t\t\t\tif (this.component.onParentPlace != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\tcompHtmlElement2.onParentPlace = this.component.onParentPlace;\n\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\tcompHtmlElement2.hotComponent = this.component;\n\t\t\t\t\t\tthis.component.htmlElements.push (compHtmlElement2);\n\t\t\t\n\t\t\t\t\t\tif (output.parentSelector != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet parentNode: Node = document.querySelector (output.parentSelector);\n\t\t\t\n\t\t\t\t\t\t\tcompHtmlElement2.parentElement.removeChild (compHtmlElement2);\n\t\t\t\t\t\t\tparentNode.appendChild (compHtmlElement2);\n\t\t\t\n\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\tif (compHtmlElement2.onParentPlace != null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\tawait compHtmlElement2.hotComponent.onParentPlace (parentNode, compHtmlElement2);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\tif (output.placeHereParent != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet parentNodeToCheck = compHtmlElement2.parentNode;\n\t\t\t\t\t\t\tlet parentNodeCheckCounter: number = 0;\n\t\t\t\n\t\t\t\t\t\t\twhile (parentNodeCheckCounter < 10) /// @todo Make this controllable with a variable from the component.\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (parentNodeToCheck == null)\n\t\t\t\t\t\t\t\t\tbreak;\n\t\t\t\n\t\t\t\t\t\t\t\tif (parentNodeToCheck instanceof HTMLHtmlElement)\n\t\t\t\t\t\t\t\t\tbreak;\n\t\t\t\n\t\t\t\t\t\t\t\t// If the hot-place-here exists, place the children there. If not, place it under the \n\t\t\t\t\t\t\t\t// new element.\n\t\t\t\t\t\t\t\tlet placeHereArray = parentNodeToCheck.querySelectorAll (`hot-place-here[name=\"${output.placeHereParent}\"]`);\n\t\t\t\n\t\t\t\t\t\t\t\tif (placeHereArray.length > 0)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tlet placeHere = placeHereArray[0];\n\t\t\t\n\t\t\t\t\t\t\t\t\tcompHtmlElement2.parentNode.removeChild (compHtmlElement2);\n\t\t\t\t\t\t\t\t\tplaceHere.appendChild (compHtmlElement2);\n\t\t\t\n\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\tif (compHtmlElement2.onParentPlace != null)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\tawait compHtmlElement2.hotComponent.onParentPlace (placeHere, compHtmlElement2);\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\t\t\tif (placeHereArray.length < 1)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tlet placeHereAttrArray = parentNodeToCheck.querySelectorAll (`[hot-place-here=\"${output.placeHereParent}\"]`);\n\t\t\t\n\t\t\t\t\t\t\t\t\tif (placeHereAttrArray.length > 0)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tlet placeHere = placeHereAttrArray[0];\n\t\t\t\t\t\t\t\t\t\tcompHtmlElement2.parentNode.removeChild (compHtmlElement2);\n\t\t\t\t\t\t\t\t\t\tplaceHere.appendChild (compHtmlElement2);\n\t\t\t\n\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\tif (compHtmlElement2.onParentPlace != null)\n\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\t\tawait compHtmlElement2.hotComponent.onParentPlace (placeHere, compHtmlElement2);\n\t\t\t\t\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\t\t\tparentNodeToCheck = parentNodeToCheck.parentNode;\n\t\t\t\t\t\t\t\tparentNodeCheckCounter++;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\t// Append the children to the newly created HTML element.\n\t\t\t\t\t\tfor (let iIdx = 0; iIdx < childrenToReadd.length; iIdx++)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tconst child: Node = childrenToReadd[iIdx];\n\t\t\t\n\t\t\t\t\t\t\tcompHtmlElement2.appendChild (child);\n\t\t\t\n\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\tif (child.onParentPlace != null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\tawait child.hotComponent.onParentPlace (compHtmlElement2, child);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\tif (this.component.onPostPlace != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\t/// @ts-ignore\n\t\t\t\t\t\t\tcompHtmlElement2 = await this.component.onPostPlace (compHtmlElement2.parentNode, compHtmlElement2);\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}, elementOptions);\n\t}\n\n\t/**\n\t * Add a new HTML element(s) to the current document.\n\t */\n\tstatic addHtml (parent: string | HTMLElement, html: string | HTMLElement): HTMLElement | HTMLElement[]\n\t{\n\t\tlet foundParent: HTMLElement = null;\n\n\t\tif (typeof (parent) === \"string\")\n\t\t\tfoundParent = document.querySelector (parent);\n\t\telse\n\t\t\tfoundParent = parent;\n\n\t\tif (foundParent == null)\n\t\t\tthrow new Error (`Unable to find parent ${parent}!`);\n\n\t\tlet result: HTMLElement = null;\n\n\t\tif (typeof (html) === \"string\")\n\t\t{\n\t\t\tlet htmlHandler = HotStaq.fixHTML (html);\n\t\t\tlet newDOM: Document = new DOMParser ().parseFromString (htmlHandler.fixedStr, \"text/html\");\n\t\t\tlet children: any = null;\n\n\t\t\tif (htmlHandler.querySelector === \"\")\n\t\t\t\tchildren = newDOM.body.children;\n\t\t\telse\n\t\t\t\tchildren = newDOM.querySelector (htmlHandler.querySelector).children;\n\n\t\t\tlet results: HTMLElement[] = [];\n\n\t\t\tfor (let iIdx = 0; iIdx < children.length; iIdx++)\n\t\t\t{\n\t\t\t\tlet child: HTMLElement = (<HTMLElement>children[iIdx]);\n\n\t\t\t\tresults.push (foundParent.appendChild (child));\n\t\t\t}\n\n\t\t\treturn (results);\n\t\t}\n\t\telse\n\t\t\tresult = foundParent.appendChild (html);\n\n\t\treturn (result);\n\t}\n\n\t/**\n\t * Check if a HotSite's name is valid.\n\t */\n\tstatic checkHotSiteName (hotsiteName: string, throwException: boolean = false): boolean\n\t{\n\t\tlet throwTheException = () =>\n\t\t\t{\n\t\t\t\tif (throwException === true)\n\t\t\t\t\tthrow new Error (`HotSite ${hotsiteName} has an invalid name! The name cannot be empty and must have a valid NPM module name.`);\n\t\t\t};\n\n\t\tlet results = validateModuleName (hotsiteName);\n\n\t\tif (results.errors != null)\n\t\t{\n\t\t\tif (results.errors.length > 0)\n\t\t\t\tthrowTheException ();\n\t\t}\n\n\t\treturn (true);\n\t}\n\n\t/**\n\t * In the supplied content, replace a key in a ${KEY} with a value.\n\t * \n\t * @returns The content with the correct values.\n\t */\n\tstatic replaceKey (content: string, key: string, value: string): string\n\t{\n\t\tconst finalStr: string = content.replace (new RegExp (`\\\\$\\\\{${key}\\\\}`, \"g\"), value);\n\n\t\treturn (finalStr);\n\t}\n\n\t/**\n\t * Get a value from a HotSite object.\n\t * \n\t * @returns Returns the value from the hotsite object. Returns null if it doesn't exist.\n\t */\n\tstatic getValueFromHotSiteObj (hotsite: HotSite, params: string[]): any\n\t{\n\t\tlet value: any = null;\n\n\t\tif (hotsite != null)\n\t\t{\n\t\t\tlet prevValue: any = hotsite;\n\n\t\t\t// Go through each object in the list of parameters and \n\t\t\t// get the value of the final parameter.\n\t\t\tfor (let iIdx = 0; iIdx < params.length; iIdx++)\n\t\t\t{\n\t\t\t\tlet param: string = params[iIdx];\n\n\t\t\t\tif (prevValue[param] == null)\n\t\t\t\t{\n\t\t\t\t\tprevValue = null;\n\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\n\t\t\t\tprevValue = prevValue[param];\n\t\t\t}\n\n\t\t\tif (prevValue != null)\n\t\t\t\tvalue = prevValue;\n\t\t}\n\n\t\treturn (value);\n\t}\n\n\t/**\n\t * Process a HotSite.\n\t */\n\tasync processHotSite (): Promise<void>\n\t{\n\t\tHotStaq.checkHotSiteName (this.hotSite.name, true);\n\n\t\tlet routes = this.hotSite.routes;\n\t\tlet testerUrl: string = \"http://127.0.0.1:8182\";\n\t\tlet tester: HotTester = null;\n\t\tlet driver: HotTestDriver = null;\n\n\t\tif (HotStaq.isWeb === false)\n\t\t{\n\t\t\tif (this.mode === DeveloperMode.Development)\n\t\t\t{\n\t\t\t\tif (this.hotSite.testing != null)\n\t\t\t\t{\n\t\t\t\t\tlet setupTester = (parentObj: any) => \n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet createNewTester: boolean = true;\n\t\t\n\t\t\t\t\t\t\tif (parentObj.createNewTester != null)\n\t\t\t\t\t\t\t\tcreateNewTester = parentObj.createNewTester;\n\t\t\n\t\t\t\t\t\t\tlet testerName: string = \"Tester\";\n\t\t\n\t\t\t\t\t\t\tif (parentObj.tester != null)\n\t\t\t\t\t\t\t\ttesterName = parentObj.tester;\n\t\t\n\t\t\t\t\t\t\tif (parentObj.testerName != null)\n\t\t\t\t\t\t\t\ttesterName = parentObj.testerName;\n\t\t\n\t\t\t\t\t\t\tif (createNewTester === true)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t/// @fixme Find a way to securely allow devs to use their own drivers and testers...\n\t\t\t\t\t\t\t\t/// @fixme Hack for dealing with WebPack's bs.\n\t\t\t\t\t\t\t\tHotTesterMocha = require (\"./HotTesterMocha\").HotTesterMocha;\n\t\t\t\t\t\t\t\tHotTesterMochaSelenium = require (\"./HotTesterMochaSelenium\").HotTesterMochaSelenium;\n\t\t\t\t\t\t\t\tHotTestSeleniumDriver = require (\"./HotTestSeleniumDriver\").HotTestSeleniumDriver;\n\t\t\n\t\t\t\t\t\t\t\tif (parentObj.testerAPIUrl === \"\")\n\t\t\t\t\t\t\t\t\ttesterUrl = parentObj.testerAPIUrl;\n\t\t\n\t\t\t\t\t\t\t\tif (parentObj.driver === \"HotTestSeleniumDriver\")\n\t\t\t\t\t\t\t\t\tdriver = new HotTestSeleniumDriver ();\n\t\t\n\t\t\t\t\t\t\t\tif (parentObj.tester === \"HotTesterMocha\")\n\t\t\t\t\t\t\t\t\ttester = new HotTesterMocha (this, testerName, testerUrl, driver);\n\t\t\n\t\t\t\t\t\t\t\tif (parentObj.tester === \"HotTesterMochaSelenium\")\n\t\t\t\t\t\t\t\t\ttester = new HotTesterMochaSelenium (this, testerName, testerUrl);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\ttester = this.testers[testerName];\n\n\t\t\t\t\t\t\tif (tester.driver == null)\n\t\t\t\t\t\t\t\tthrow new Error (`Tester ${testerName} does not have a driver set!`);\n\n\t\t\t\t\t\t\tif (parentObj.commandDelay != null)\n\t\t\t\t\t\t\t\ttester.driver.commandDelay = parentObj.commandDelay;\n\t\t\t\t\t\t};\n\n\t\t\t\t\tif (this.hotSite.testing.web != null)\n\t\t\t\t\t\tsetupTester (this.hotSite.testing.web);\n\n\t\t\t\t\tif (this.hotSite.testing.api != null)\n\t\t\t\t\t\tsetupTester (this.hotSite.testing.api);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tif (routes != null)\n\t\t{\n\t\t\tfor (let key in routes)\n\t\t\t{\n\t\t\t\tlet route: HotSiteRoute = routes[key];\n\t\t\t\tlet file: HotFile = new HotFile (route);\n\t\t\t\tlet page: HotPage = new HotPage ({\n\t\t\t\t\t\tprocessor: this,\n\t\t\t\t\t\tname: route.name || \"\",\n\t\t\t\t\t\troute: key,\n\t\t\t\t\t\tfiles: [file]\n\t\t\t\t\t});\n\n\t\t\t\tif (tester != null)\n\t\t\t\t{\n\t\t\t\t\tif (this.mode === DeveloperMode.Development)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet mapName: string = route.name;\n\t\t\t\t\t\tlet testMap: HotTestMap = null;\n\n\t\t\t\t\t\tif (route.map != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (typeof (route.map) === \"string\")\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (tester.testMaps[route.map] == null)\n\t\t\t\t\t\t\t\t\tthrow new Error (`Test map ${route.map} does not exist!`);\n\n\t\t\t\t\t\t\t\ttester.testMaps[mapName] = tester.testMaps[route.map];\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\ttestMap = new HotTestMap ();\n\t\t\t\t\t\t\t\tlet destinations: HotTestDestination[] | { [name: string]: HotTestDestination } = null;\n\n\t\t\t\t\t\t\t\tif (route.map instanceof Array)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tdestinations = [];\n\n\t\t\t\t\t\t\t\t\tfor (let iIdx = 0; iIdx < route.map.length; iIdx++)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tlet dest = route.map[iIdx];\n\n\t\t\t\t\t\t\t\t\t\tdestinations.push (new HotTestDestination (dest));\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tdestinations = {};\n\n\t\t\t\t\t\t\t\t\tfor (let key2 in route.map)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tlet dest = route.map[key2];\n\n\t\t\t\t\t\t\t\t\t\tdestinations[key2] = new HotTestDestination (dest);\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\ttestMap.destinations = destinations;\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\ttester.testMaps[mapName] = testMap;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (route.destinationOrder != null)\n\t\t\t\t\t\t\ttester.testMaps[mapName].destinationOrder = route.destinationOrder;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tthis.addPage (page);\n\t\t\t}\n\t\t}\n\n\t\tif (this.hotSite.apis != null)\n\t\t{\n\t\t\tfor (let key in this.hotSite.apis)\n\t\t\t{\n\t\t\t\tlet api = this.hotSite.apis[key];\n\n\t\t\t\tif (api.map == null)\n\t\t\t\t\tcontinue;\n\n\t\t\t\tif (HotStaq.isWeb === false)\n\t\t\t\t{\n\t\t\t\t\tif (this.mode === DeveloperMode.Development)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet mapName: string = key;\n\t\t\t\t\t\tlet testMap: HotTestMap = new HotTestMap ();\n\n\t\t\t\t\t\ttestMap.destinations = [];\n\n\t\t\t\t\t\tfor (let iIdx = 0; iIdx < api.map.length; iIdx++)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet map: string = api.map[iIdx];\n\n\t\t\t\t\t\t\ttestMap.destinations.push (new HotTestDestination (map));\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (tester == null)\n\t\t\t\t\t\t\tthrow new Error (`A tester was not created first! You must specify one in the CLI or in HotSite.json.`);\n\n\t\t\t\t\t\ttester.testMaps[mapName] = testMap;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\t/// @fixme Allow this to work for server-side as well...\n\t\tif (HotStaq.isWeb === true)\n\t\t{\n\t\t\tfor (let key in this.hotSite.components)\n\t\t\t{\n\t\t\t\tlet component = this.hotSite.components[key];\n\t\t\t\tlet componentUrl: string = component.url;\n\n\t\t\t\t/// @fixme Create unit test for fetching, loading, and registering.\n\t\t\t\tlet res: any = await fetch (componentUrl);\n\t\t\t\tlet ComponentClass = eval (res);\n\n\t\t\t\tthis.addComponent (ComponentClass);\n\t\t\t}\n\t\t}\n\n\t\tif (this.hotSite.routes == null)\n\t\t\tthis.hotSite.routes = {};\n\n\t\tlet disableFileLoading: boolean = false;\n\n\t\tif (this.hotSite.disableFileLoading != null)\n\t\t\tdisableFileLoading = this.hotSite.disableFileLoading;\n\n\t\tif (disableFileLoading === false)\n\t\t\tawait this.loadHotFiles (this.hotSite.files);\n\t\telse\n\t\t\tthis.logger.verbose (`Hotsite has file loading disabled...`);\n\n\t\tif (tester != null)\n\t\t\tthis.addTester (tester);\n\t}\n\n\t/**\n\t * Load from a HotSite.json file. Be sure to load and attach any testers before \n\t * loading a HotSite.\n\t */\n\tasync loadHotSite (path: string): Promise<void>\n\t{\n\t\tlet jsonStr: string = \"\";\n\n\t\tif (HotStaq.isWeb === true)\n\t\t{\n\t\t\tthis.logger.info (`Downloading HotSite ${path}`);\n\n\t\t\tlet res: any = await fetch (path);\n\n\t\t\tthis.logger.info (`Downloaded site ${path}`);\n\n\t\t\tjsonStr = res.text ();\n\t\t}\n\t\telse\n\t\t{\n\t\t\tpath = ppath.normalize (path);\n\n\t\t\tthis.logger.info (`Accessing HotSite ${path}`);\n\n\t\t\tjsonStr = await new Promise (\n\t\t\t\t(resolve: any, reject: any): void =>\n\t\t\t\t{\n\t\t\t\t\tfs.readFile (path, (err: NodeJS.ErrnoException, data: Buffer): void =>\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (err != null)\n\t\t\t\t\t\t\t\tthrow err;\n\t\n\t\t\t\t\t\t\tlet content: string = data.toString ();\n\n\t\t\t\t\t\t\tthis.logger.info (`Accessed site ${path}`);\n\t\n\t\t\t\t\t\t\tresolve (content);\n\t\t\t\t\t\t});\n\t\t\t\t});\n\t\t}\n\n\t\tthis.hotSite = JSON.parse (jsonStr);\n\t\tthis.hotSite.hotsitePath = path;\n\t}\n\n\t/**\n\t * Load an array of files. If a file already has content, it will not be reloaded \n\t * unless forceContentLoading is set to true.\n\t */\n\tasync loadHotFiles (files: { [name: string]: { url?: string; localFile?: string; content?: string; } }, \n\t\t\tforceContentLoading: boolean = false): Promise<void>\n\t{\n\t\tthis.logger.verbose (`Loading Hott files...`);\n\n\t\tfor (let key in files)\n\t\t{\n\t\t\tlet file = files[key];\n\t\t\tlet newFile: HotFile = null;\n\n\t\t\tif (HotStaq.isWeb === true)\n\t\t\t{\n\t\t\t\tnewFile = new HotFile ({\n\t\t\t\t\t\t\"name\": key\n\t\t\t\t\t});\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\tnewFile = new HotFile ({\n\t\t\t\t\t\t\"name\": key\n\t\t\t\t\t});\n\t\t\t}\n\n\t\t\tif (file.url != null)\n\t\t\t\tnewFile.url = file.url;\n\n\t\t\tif (HotStaq.isWeb === false)\n\t\t\t{\n\t\t\t\tif (file.localFile != null)\n\t\t\t\t\tnewFile.localFile = file.localFile;\n\t\t\t}\n\n\t\t\tlet loadContent: boolean = true;\n\n\t\t\tif (file.content != null)\n\t\t\t{\n\t\t\t\tnewFile.content = file.content;\n\t\t\t\tloadContent = false;\n\t\t\t}\n\n\t\t\tif (forceContentLoading === true)\n\t\t\t\tloadContent = true;\n\n\t\t\tif (loadContent === true)\n\t\t\t{\n\t\t\t\tlet filepath: string = \"\";\n\n\t\t\t\tif (newFile.url !== \"\")\n\t\t\t\t\tfilepath = newFile.url;\n\n\t\t\t\tif (newFile.localFile !== \"\")\n\t\t\t\t\tfilepath = newFile.localFile;\n\n\t\t\t\tthis.logger.verbose (`Loading Hott file: ${filepath}`);\n\t\t\t\tawait newFile.load ();\n\t\t\t\tthis.logger.verbose (`Finished loading Hott file: ${filepath}`);\n\t\t\t}\n\n\t\t\tthis.addFile (newFile);\n\t\t}\n\n\t\tthis.logger.verbose (`Finished loading Hott files...`);\n\t}\n\n\t/**\n\t * Generate the content to send to a client.\n\t */\n\tgenerateContent (routeKey: string, name: string = \"\", url: string = \"./\",\n\t\t\tjsSrcPath: string = \"./js/HotStaq.min.js\", passArgs: boolean = true, \n\t\t\targs: any = null): string\n\t{\n\t\tlet apiScripts: string = \"\";\n\t\tlet apiCode: string = \"\";\n\t\tlet publicKeys: string = \"\";\n\n\t\t/// @todo Optimize this function as much as possible.\n\n\t\t// Load the API string.\n\t\tif (this.hotSite != null)\n\t\t{\n\t\t\tif (this.hotSite.server.globalApi != null)\n\t\t\t{\n\t\t\t\tif (this.hotSite.server.globalApi !== \"\")\n\t\t\t\t{\n\t\t\t\t\tconst globalApi = this.hotSite.apis[this.hotSite.server.globalApi];\n\n\t\t\t\t\tif (globalApi == null)\n\t\t\t\t\t\tthis.logger.warning (`API with name ${this.hotSite.server.globalApi} doesn't exist!`);\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tlet sendJSContent: boolean = true;\n\n\t\t\t\t\t\tif (globalApi.jsapi == null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tsendJSContent = false;\n\t\t\t\t\t\t\tthis.logger.warning (`API with name ${this.hotSite.server.globalApi} doesn't have a jsapi set. Will not send js content to client.`);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (globalApi.libraryName == null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tsendJSContent = false;\n\t\t\t\t\t\t\tthis.logger.warning (`API with name ${this.hotSite.server.globalApi} doesn't have a libraryName set. Will not send js content to client.`);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (globalApi.apiName == null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tsendJSContent = false;\n\t\t\t\t\t\t\tthis.logger.warning (`API with name ${this.hotSite.server.globalApi} doesn't have a apiName set. Will not send js content to client.`);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (sendJSContent === true)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tapiScripts += `\\t<script type = \"text/javascript\" src = \"${globalApi.jsapi}\"></script>\\n`;\n\n\t\t\t\t\t\t\tlet baseUrl: string = \"\\\"\\\"\";\n\n\t\t\t\t\t\t\tif (globalApi.url != null)\n\t\t\t\t\t\t\t\tbaseUrl = `\\\"${globalApi.url}\\\"`;\n\n\t\t\t\t\t\t\tif (this.api != null)\n\t\t\t\t\t\t\t\tbaseUrl = `\\\"${this.api.baseUrl}\\\"`;\n\n\t\t\t\t\t\t\tlet tempAPIContent: string = this.apiContent;\n\t\t\t\t\t\t\ttempAPIContent = tempAPIContent.replace (/\\%api\\_name\\%/g, globalApi.apiName);\n\t\t\t\t\t\t\ttempAPIContent = tempAPIContent.replace (/\\%api\\_exported\\_name\\%/g, globalApi.libraryName);\n\t\t\t\t\t\t\ttempAPIContent = tempAPIContent.replace (/\\%base\\_url\\%/g, baseUrl);\n\n\t\t\t\t\t\t\tapiCode += tempAPIContent;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif (this.hotSite.apis != null)\n\t\t\t{\n\t\t\t\tlet route = this.hotSite.routes[routeKey];\n\n\t\t\t\tif (route != null)\n\t\t\t\t{\n\t\t\t\t\tif (route.api != null)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet api = this.hotSite.apis[route.api];\n\n\t\t\t\t\t\tif (api == null)\n\t\t\t\t\t\t\tthrow new Error (`Unable to find API ${route.api}`);\n\n\t\t\t\t\t\tlet sendJSContent: boolean = true;\n\n\t\t\t\t\t\tif (api.jsapi == null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tsendJSContent = false;\n\t\t\t\t\t\t\tthis.logger.warning (`API with name ${route.api} doesn't have a jsapi set. Will not send js content to client.`);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (api.libraryName == null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tsendJSContent = false;\n\t\t\t\t\t\t\tthis.logger.warning (`API with name ${route.api} doesn't have a libraryName set. Will not send js content to client.`);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (api.apiName == null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tsendJSContent = false;\n\t\t\t\t\t\t\tthis.logger.warning (`API with name ${route.api} doesn't have a apiName set. Will not send js content to client.`);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (sendJSContent === true)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet jsapipath = api.jsapi;\n\t\t\t\t\t\t\tapiScripts += `\\t<script type = \"text/javascript\" src = \"${jsapipath}\"></script>\\n`;\n\n\t\t\t\t\t\t\tlet baseUrl: string = \"\\\"\\\"\";\n\n\t\t\t\t\t\t\tif (api.url != null)\n\t\t\t\t\t\t\t\tbaseUrl = `\\\"${api.url}\\\"`;\n\n\t\t\t\t\t\t\tif (this.api != null)\n\t\t\t\t\t\t\t\tbaseUrl = `\\\"${this.api.baseUrl}\\\"`;\n\n\t\t\t\t\t\t\tlet tempAPIContent: string = this.apiContent;\n\t\t\t\t\t\t\ttempAPIContent = tempAPIContent.replace (/\\%api\\_name\\%/g, api.apiName);\n\t\t\t\t\t\t\ttempAPIContent = tempAPIContent.replace (/\\%api\\_exported\\_name\\%/g, api.libraryName);\n\t\t\t\t\t\t\ttempAPIContent = tempAPIContent.replace (/\\%base\\_url\\%/g, baseUrl);\n\n\t\t\t\t\t\t\tapiCode += tempAPIContent;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif (this.hotSite.server != null)\n\t\t\t{\n\t\t\t\tif (this.hotSite.server.jsSrcPath != null)\n\t\t\t\t\tjsSrcPath = this.hotSite.server.jsSrcPath;\n\t\t\t}\n\n\t\t\tif (this.hotSite.publicKeys != null)\n\t\t\t{\n\t\t\t\tfor (let key in this.hotSite.publicKeys)\n\t\t\t\t{\n\t\t\t\t\tlet secret = this.hotSite.publicKeys[key];\n\t\t\t\t\tlet value: string = undefined;\n\n\t\t\t\t\tif (typeof (secret) === \"string\")\n\t\t\t\t\t\tvalue = JSON.stringify (secret);\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tif (HotStaq.isWeb === false)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (this.api != null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (this.api.connection == null)\n\t\t\t\t\t\t\t\t\tthrow new Error (`Cannot pass secrets from the API if there's no connection!`);\n\n\t\t\t\t\t\t\t\tlet serverConn: HotServer = (<HotServer>this.api.connection);\n\n\t\t\t\t\t\t\t\tif (secret.passSecretFromAPI != null)\n\t\t\t\t\t\t\t\t\tvalue = JSON.stringify (serverConn.secrets[key]);\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tif (secret.env != null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t/// @fixme @secvul Is this a security vulnerability? Need to verify that \n\t\t\t\t\t\t\t\t/// only the server has access to this. At this point, I think only the \n\t\t\t\t\t\t\t\t/// server has access.\n\t\t\t\t\t\t\t\tconst envKey: string = secret.env;\n\n\t\t\t\t\t\t\t\tvalue = JSON.stringify (process.env[envKey]);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tpublicKeys += `processor.publicKeys[\"${key}\"] = ${value};\\n`;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tlet content: string = this.pageContent;\n\t\tlet fixContent = (tempContent: string) =>\n\t\t\t{\n\t\t\t\tlet developerModeStr: string = \"\";\n\t\t\t\tlet testerAPIStr: string = \"\";\n\n\t\t\t\tif (this.mode === DeveloperMode.Development)\n\t\t\t\t{\n\t\t\t\t\tdeveloperModeStr = `tempMode = HotStaqWeb.DeveloperMode.Development;`;\n\n\t\t\t\t\tif (this.hotSite != null)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (this.hotSite.testing != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (this.hotSite.testing.web != null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\ttesterAPIStr = this.testerApiContent;\n\n\t\t\t\t\t\t\t\tif (this.hotSite.testing.web.testerAPIUrl == null)\n\t\t\t\t\t\t\t\t\tthis.hotSite.testing.web.testerAPIUrl = \"http://127.0.0.1:8182\";\n\n\t\t\t\t\t\t\t\ttesterAPIStr = testerAPIStr.replace (/\\%base\\_tester\\_url\\%/g, `\\\"${this.hotSite.testing.web.testerAPIUrl}\\\"`);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tlet loadFiles: string = \"\";\n\n\t\t\t\tif (Object.keys (this.files).length > 0)\n\t\t\t\t{\n\t\t\t\t\tloadFiles += `var files = {};\\n\\n`;\n\n\t\t\t\t\tfor (let key in this.files)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet file = this.files[key];\n\t\t\t\t\t\tlet fileUrl: string = `\"${file.url}\"`;\n\t\t\t\t\t\tlet fileContent: string = \"\";\n\n\t\t\t\t\t\tif (file.content !== \"\")\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet escapedContent: string = JSON.stringify (file.content);\n\n\t\t\t\t\t\t\t// Find any script tags and interrupt them so the HTML parsers \n\t\t\t\t\t\t\t// don't get confused.\n\t\t\t\t\t\t\tescapedContent = escapedContent.replace (new RegExp (\"\\\\<script\", \"gmi\"), \"<scr\\\" + \\\"ipt\");\n\t\t\t\t\t\t\tescapedContent = escapedContent.replace (new RegExp (\"\\\\<\\\\/script\", \"gmi\"), \"</scr\\\" + \\\"ipt\");\n\n\t\t\t\t\t\t\tfileContent = `, \"content\": ${escapedContent}`;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tloadFiles += `\\t\\t\\tfiles[\"${key}\"] = { \"url\": ${fileUrl}${fileContent} };\\n`;\n\t\t\t\t\t}\n\n\t\t\t\t\tloadFiles += `\\t\\t\\tpromises.push (processor.loadHotFiles (files));\\n`;\n\t\t\t\t}\n\n\t\t\t\ttempContent = tempContent.replace (/\\%title\\%/g, name);\n\n\t\t\t\tif (passArgs === true)\n\t\t\t\t\ttempContent = tempContent.replace (/\\%args\\%/g, \"Hot.Arguments\");\n\n\t\t\t\tif (args != null)\n\t\t\t\t\ttempContent = tempContent.replace (/\\%args\\%/g, JSON.stringify (args));\n\n\t\t\t\tlet testerMap: string = routeKey;\n\t\t\t\tlet testerUrl: string = \"\";\n\t\t\t\tlet testerLaunchpadUrl: string = \"\";\n\t\t\t\tlet testerName: string = \"Tester\";\n\n\t\t\t\tif (this.hotSite != null)\n\t\t\t\t{\n\t\t\t\t\tif (this.hotSite.testing != null)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (this.hotSite.testing.web != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (this.hotSite.testing.web.tester != null)\n\t\t\t\t\t\t\t\ttesterName = this.hotSite.testing.web.tester;\n\n\t\t\t\t\t\t\tif (this.hotSite.testing.web.testerName != null)\n\t\t\t\t\t\t\t\ttesterName = this.hotSite.testing.web.testerName;\n\n\t\t\t\t\t\t\tif (this.hotSite.testing.web.testerAPIUrl != null)\n\t\t\t\t\t\t\t\ttesterUrl = this.hotSite.testing.web.testerAPIUrl;\n\n\t\t\t\t\t\t\tif (this.hotSite.testing.web.launchpadUrl != null)\n\t\t\t\t\t\t\t\ttesterLaunchpadUrl = this.hotSite.testing.web.launchpadUrl;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tif (this.hotSite.routes != null)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (this.hotSite.routes[routeKey] != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet route = this.hotSite.routes[routeKey];\n\t\t\t\t\t\t\ttesterMap = route.name;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\ttempContent = tempContent.replace (/\\%hotstaq\\_js\\_src\\%/g, jsSrcPath);\n\t\t\t\ttempContent = tempContent.replace (/\\%developer\\_mode\\%/g, developerModeStr);\n\t\t\t\ttempContent = tempContent.replace (/\\%tester\\_api\\%/g, testerAPIStr);\n\t\t\t\ttempContent = tempContent.replace (/\\%apis\\_to\\_load\\%/g, apiScripts);\n\t\t\t\ttempContent = tempContent.replace (/\\%load\\_hot\\_site\\%/g, \"\"); /// @fixme Should this only be done server-side?\n\t\t\t\ttempContent = tempContent.replace (/\\%load\\_files\\%/g, loadFiles);\n\t\t\t\ttempContent = tempContent.replace (/\\%api\\_code\\%/g, apiCode);\n\t\t\t\ttempContent = tempContent.replace (/\\%public\\_secrets\\%/g, publicKeys);\n\t\t\t\ttempContent = tempContent.replace (/\\%url\\%/g, url);\n\t\t\t\ttempContent = tempContent.replace (/\\%tester\\_name\\%/g, `\"${testerName}\"`);\n\t\t\t\ttempContent = tempContent.replace (/\\%tester\\_map\\%/g, `\"${testerMap}\"`);\n\t\t\t\ttempContent = tempContent.replace (/\\%tester\\_api\\_base\\_url\\%/g, `\"${testerUrl}\"`);\n\t\t\t\ttempContent = tempContent.replace (/\\%tester\\_launchpad\\_url\\%/g, `\"${testerLaunchpadUrl}\"`);\n\n\t\t\t\treturn (tempContent);\n\t\t\t};\n\t\tcontent = fixContent (content);\n\n\t\treturn (content);\n\t}\n\n\t/**\n\t * Create the Express routes from the given pages. Be sure to load the \n\t * pages first before doing this. This method is meant to be used for \n\t * customized Express applications. If you wish to use the loaded routes \n\t * from this HotStaq object with HotHTTPServer, be sure to use \n\t * the loadHotSite method in HotHTTPServer.\n\t */\n\tcreateExpressRoutes (expressApp: any, jsSrcPath: string = \"./js/HotStaq.min.js\"): void\n\t{\n\t\tfor (let key in this.pages)\n\t\t{\n\t\t\tlet page: HotPage = this.pages[key];\n\t\t\tconst content: string = this.generateContent (page.route, page.name, page.files[0].url, jsSrcPath);\n\n\t\t\texpressApp.get (page.route, (req: any, res: any) =>\n\t\t\t\t{\n\t\t\t\t\tres.send (content);\n\t\t\t\t});\n\t\t}\n\t}\n\n\t/**\n\t * Add a tester for use later.\n\t */\n\taddTester (tester: HotTester): void\n\t{\n\t\tthis.testers[tester.name] = tester;\n\t}\n\n\t/**\n\t * Get the list of maps for testing from the HotSite.\n\t */\n\tgetWebTestingMaps (): string[]\n\t{\n\t\tif (this.hotSite == null)\n\t\t\tthrow new Error (\"No HotSite was loaded!\");\n\n\t\tif (this.hotSite.testing == null)\n\t\t\tthrow new Error (\"The HotSite does not have a testing object!\");\n\n\t\tif (this.hotSite.testing.web == null)\n\t\t\tthrow new Error (\"The HotSite does not have a testing web object!\");\n\n\t\tif (this.hotSite.testing.web.maps == null)\n\t\t\tthrow new Error (\"The HotSite testing object does not have any maps!\");\n\n\t\treturn (this.hotSite.testing.web.maps);\n\t}\n\n\t/**\n\t * Get the list of maps for testing from the HotSite.\n\t */\n\tgetAPITestingMaps (): string[]\n\t{\n\t\tif (this.hotSite == null)\n\t\t\tthrow new Error (\"No HotSite was loaded!\");\n\n\t\tif (this.hotSite.testing == null)\n\t\t\tthrow new Error (\"The HotSite does not have a testing object!\");\n\n\t\tif (this.hotSite.testing.api == null)\n\t\t\tthrow new Error (\"The HotSite does not have a testing api object!\");\n\n\t\tif (this.hotSite.testing.api.maps == null)\n\t\t\tthrow new Error (\"The HotSite testing object does not have any maps!\");\n\n\t\treturn (this.hotSite.testing.api.maps);\n\t}\n\n\t/**\n\t * Get a route's key from a route's name.\n\t */\n\tgetRouteKeyFromName (name: string): string\n\t{\n\t\tlet foundKey: string = \"\";\n\n\t\tif (this.hotSite != null)\n\t\t{\n\t\t\tif (this.hotSite.routes != null)\n\t\t\t{\n\t\t\t\tfor (let key in this.hotSite.routes)\n\t\t\t\t{\n\t\t\t\t\tlet route: HotSiteRoute = this.hotSite.routes[key];\n\n\t\t\t\t\tif (route.name === name)\n\t\t\t\t\t{\n\t\t\t\t\t\tfoundKey = key;\n\n\t\t\t\t\t\tbreak;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn (foundKey);\n\t}\n\n\t/**\n\t * Get a route from a route's name.\n\t */\n\tgetRouteFromName (name: string): HotSiteRoute\n\t{\n\t\tlet foundRoute: HotSiteRoute = null;\n\t\tlet foundKey: string = this.getRouteKeyFromName (name);\n\n\t\tif (foundKey !== \"\")\n\t\t\tfoundRoute = this.hotSite.routes[foundKey];\n\n\t\treturn (foundRoute);\n\t}\n\n\t/**\n\t * Execute tests.\n\t * \n\t * @param testerName The tester to use to execute tests.\n\t * @param mapName The map or maps to use to navigate through tests.\n\t */\n\tasync executeTests (testerName: string, mapName: string): Promise<void>\n\t{\n\t\tlet tester: HotTester = this.testers[testerName];\n\n\t\tif (tester == null)\n\t\t\tthrow new Error (`Unable to execute tests. Tester ${testerName} does not exist!`);\n\n\t\treturn (tester.execute (mapName));\n\t}\n\n\t/**\n\t * Execute all web tests from the HotSite testing web object.\n\t * \n\t * @param testerName The tester to use to execute tests.\n\t */\n\tasync executeAllWebTests (testerName: string): Promise<void>\n\t{\n\t\tlet maps: string[] = this.getWebTestingMaps ();\n\t\tlet tester: HotTester = this.testers[testerName];\n\n\t\tif (tester == null)\n\t\t\tthrow new Error (`Unable to execute tests. Tester ${testerName} does not exist!`);\n\n\t\tfor (let iIdx = 0; iIdx < maps.length; iIdx++)\n\t\t{\n\t\t\tlet mapName: string = maps[iIdx];\n\n\t\t\tawait this.executeTests (testerName, mapName);\n\t\t}\n\t}\n\n\t/**\n\t * Execute all api tests from the HotSite testing api object.\n\t * \n\t * @param testerName The tester to use to execute tests.\n\t */\n\tasync executeAllAPITests (testerName: string): Promise<void>\n\t{\n\t\tlet maps: string[] = this.getAPITestingMaps ();\n\t\tlet tester: HotTester = this.testers[testerName];\n\n\t\tif (tester == null)\n\t\t\tthrow new Error (`Unable to execute tests. Tester ${testerName} does not exist!`);\n\n\t\tfor (let iIdx = 0; iIdx < maps.length; iIdx++)\n\t\t{\n\t\t\tlet mapName: string = maps[iIdx];\n\n\t\t\tawait this.executeTests (testerName, mapName);\n\t\t}\n\t}\n\n\t/**\n\t * Process a page and get the result.\n\t */\n\tasync process (pageName: string, args: any = null): Promise<string>\n\t{\n\t\tlet page: HotPage = this.getPage (pageName);\n\t\tlet result: string = await page.process (args);\n\n\t\treturn (result);\n\t}\n\n\t/**\n\t * Process a local file and get the result.\n\t */\n\tstatic async processLocalFile (localFilepath: string, name: string = localFilepath, args: any = null): Promise<string>\n\t{\n\t\tlet processor: HotStaq = new HotStaq ();\n\t\tlet file: HotFile = new HotFile ({\n\t\t\t\"localFile\": localFilepath\n\t\t});\n\t\tawait file.load ();\n\t\tlet page: HotPage = new HotPage ({\n\t\t\t\t\"processor\": processor,\n\t\t\t\t\"name\": name,\n\t\t\t\t\"files\": [file]\n\t\t\t});\n\t\tprocessor.addPage (page);\n\t\tlet result: string = await processor.process (name, args);\n\n\t\treturn (result);\n\t}\n\n\t/**\n\t * Process a url and get the result.\n\t */\n\tstatic async processUrl (options: HotStartOptions): Promise<string>\n\t{\n\t\tlet file: HotFile = new HotFile ({\n\t\t\t\"url\": options.url\n\t\t});\n\n\t\tawait file.load ();\n\t\tlet page: HotPage = new HotPage ({\n\t\t\t\t\"processor\": options.processor,\n\t\t\t\t\"name\": options.name,\n\t\t\t\t\"files\": [file],\n\t\t\t\t\"testerName\": options.testerName,\n\t\t\t\t\"testerMap\": options.testerMap\n\t\t\t});\n\t\toptions.processor.addPage (page);\n\t\tlet result: string = await options.processor.process (options.name, options.args);\n\n\t\treturn (result);\n\t}\n\n\t/**\n\t * Process content and get the result.\n\t */\n\tstatic async processContent (options: HotStartOptions): Promise<string>\n\t{\n\t\tlet file: HotFile = new HotFile ({\n\t\t\t\"content\": options.content\n\t\t});\n\t\tawait file.load ();\n\t\tlet page: HotPage = new HotPage ({\n\t\t\t\t\"processor\": options.processor,\n\t\t\t\t\"name\": options.name,\n\t\t\t\t\"files\": [file]\n\t\t\t});\n\t\toptions.processor.addPage (page);\n\t\tlet result: string = await options.processor.process (options.name, options.args);\n\n\t\treturn (result);\n\t}\n\n\t/**\n\t * When the window has finished loading, execute the function.\n\t * This is meant for web browser use only.\n\t */\n\tstatic onReady (readyFunc: () => void): void\n\t{\n\t\tif ((document.readyState === \"complete\") || (document.readyState === \"interactive\"))\n\t\t\treadyFunc ();\n\t\telse\n\t\t\twindow.addEventListener (\"load\", readyFunc);\n\t}\n\n\t/**\n\t * Replace the current HTML page with the output.\n\t * This is meant for web browser use only.\n\t */\n\tstatic async useOutput (output: string): Promise<void>\n\t{\n\t\t\tlet parser = new DOMParser ();\n\t\t\tlet child = parser.parseFromString (output, \"text/html\");\n\t\t\tlet htmlObj: HTMLHtmlElement = document.getElementsByTagName('html')[0];\n\n\t\t\thtmlObj.innerHTML = child.getElementsByTagName('html')[0].innerHTML;\n\n\t\t\t// Thanks to newfurniturey at: \n\t\t\t// https://stackoverflow.com/questions/22945884/domparser-appending-script-tags-to-head-body-but-not-executing\n\t\t\tlet tmpScripts = document.getElementsByTagName('script');\n\t\t\tif (tmpScripts.length > 0) {\n\t\t\t\t// push all of the document's script tags into an array\n\t\t\t\t// (to prevent dom manipulation while iterating over dom nodes)\n\t\t\t\tlet scripts: HTMLScriptElement[] = [];\n\t\t\t\tfor (let i = 0; i < tmpScripts.length; i++) {\n\t\t\t\t\tscripts.push(tmpScripts[i]);\n\t\t\t\t}\n\n\t\t\t\t// iterate over all script tags and create duplicate tags for each\n\t\t\t\tfor (let i = 0; i < scripts.length; i++) {\n\t\t\t\t\tlet s: HTMLScriptElement = document.createElement('script');\n\n\t\t\t\t\t// add the new node to the page\n\t\t\t\t\tscripts[i].parentNode.appendChild(s);\n\n\t\t\t\t\t// remove the original (non-executing) node from the page\n\t\t\t\t\tscripts[i].parentNode.removeChild(scripts[i]);\n\n\t\t\t\t\tawait new Promise<void> ((resolve2, reject2) =>\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\ts.onload = () =>\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tresolve2 ();\n\t\t\t\t\t\t\t\t};\n\n\t\t\t\t\t\t\tlet hasSrc: boolean = false;\n\n\t\t\t\t\t\t\tif (scripts[i].getAttribute (\"src\") != null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (scripts[i].getAttribute (\"src\") !== \"\")\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\ts.setAttribute (\"src\", scripts[i].getAttribute (\"src\"));\n\t\t\t\t\t\t\t\t\thasSrc = true;\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tif (scripts[i].getAttribute (\"type\") != null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (scripts[i].getAttribute (\"type\") !== \"\")\n\t\t\t\t\t\t\t\t\ts.setAttribute (\"type\", scripts[i].getAttribute (\"type\"));\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\ts.innerHTML = scripts[i].innerHTML;\n\n\t\t\t\t\t\t\tif (hasSrc === false)\n\t\t\t\t\t\t\t\tresolve2 ();\n\t\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t}\n\t}\n\n\t/**\n\t * Wait for testers to load.\n\t * \n\t * @fixme This does not wait for ALL testers to finish loading. Only \n\t * the first one.\n\t */\n\tstatic async waitForTesters (): Promise<void>\n\t{\n\t\twhile (HotStaq.isReadyForTesting === false)\n\t\t\tawait HotStaq.wait (10);\n\n\t\tif (HotStaq.onReadyForTesting != null)\n\t\t\tawait HotStaq.onReadyForTesting ();\n\t}\n\n\t/**\n\t * Setup the testers api, if any.\n\t */\n\tstatic setupTesters (processor: HotStaq, options: HotStartOptions)\n\t{\n\t\tif (processor.mode === DeveloperMode.Development)\n\t\t{\n\t\t\tif (processor.testerAPI == null)\n\t\t\t{\n\t\t\t\tif (options.testerAPIBaseUrl == null)\n\t\t\t\t\toptions.testerAPIBaseUrl = \"\";\n\n\t\t\t\tif (options.testerAPIBaseUrl === \"\")\n\t\t\t\t\toptions.testerAPIBaseUrl = \"http://127.0.0.1:8182\";\n\n\t\t\t\tlet client: HotClient = new HotClient (processor);\n\t\t\t\tlet testerAPI: HotTesterAPI = new HotTesterAPI (options.testerAPIBaseUrl, client);\n\t\t\t\ttesterAPI.connection.api = testerAPI;\n\t\t\t\tprocessor.testerAPI = testerAPI;\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Setup the testers api on the client, if needed.\n\t */\n\tstatic setupClientTesters (processor: HotStaq): string\n\t{\n\t\tlet output: string = \"\";\n\n\t\tif (processor.mode === DeveloperMode.Development)\n\t\t{\n\t\t\toutput += \n`<script type = \"text/javascript\">\nfunction hotstaq_isDocumentReady ()\n{\nif (window[\"Hot\"] != null)\n{\nif (Hot.Mode === HotStaqWeb.DeveloperMode.Development)\n{\nlet func = function ()\n\t{\n\t\tif (Hot.TesterAPI != null)\n\t\t{\n\t\t\tlet testPaths = {};\n\t\t\tlet testElements = JSON.stringify (Hot.CurrentPage.testElements);\n\t\t\tlet testMaps = JSON.stringify (Hot.CurrentPage.testMaps);\n\n\t\t\tfor (let key in Hot.CurrentPage.testPaths)\n\t\t\t{\n\t\t\t\tlet testPath = Hot.CurrentPage.testPaths[key];\n\n\t\t\t\ttestPaths[key] = testPath.toString ();\n\t\t\t}\n\n\t\t\tlet testPathsStr = JSON.stringify (testPaths);\n\n\t\t\tHot.TesterAPI.tester.pageLoaded ({\n\t\t\t\t\ttesterName: Hot.CurrentPage.testerName,\n\t\t\t\t\ttesterMap: Hot.CurrentPage.testerMap,\n\t\t\t\t\tpageName: Hot.CurrentPage.name,\n\t\t\t\t\ttestElements: testElements,\n\t\t\t\t\ttestPaths: testPathsStr\n\t\t\t\t}).then (function (resp)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (resp.error != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (resp.error !== \"\")\n\t\t\t\t\t\t\t\tthrow new Error (resp.error);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tHotStaqWeb.HotStaq.isReadyForTesting = true;\n\t\t\t\t\t});\n\t\t}\n\t};\n\nif ((document.readyState === \"complete\") || (document.readyState === \"interactive\"))\n\tfunc ();\nelse\n\tdocument.addEventListener (\"DOMContentLoaded\", func);\n}\n}\n}\n\nhotstaq_isDocumentReady ();\n</script>`;\n\t\t}\n\n\t\treturn (output);\n\t}\n\n\t/**\n\t * Process and replace the current HTML page with the hott script from the given url.\n\t * This is meant for web browser use only.\n\t */\n\tstatic async displayUrl (url: string | HotStartOptions, name: string = null, \n\t\tprocessor: HotStaq = null, args: any = null): Promise<HotStaq>\n\t{\n\t\treturn (new Promise<HotStaq> ((resolve, reject) =>\n\t\t\t{\n\t\t\t\tHotStaq.onReady (async () =>\n\t\t\t\t\t{\n\t\t\t\t\t\tlet options: HotStartOptions = {\n\t\t\t\t\t\t\t\t\"url\": \"\"\n\t\t\t\t\t\t\t};\n\n\t\t\t\t\t\tif (name == null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (typeof (url) === \"string\")\n\t\t\t\t\t\t\t\toptions.name = url;\n\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\toptions.name = url.name;\n\t\t\t\t\t\t}\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\toptions.name = name;\n\n\t\t\t\t\t\tif (options.name === \"\")\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (typeof (url) === \"string\")\n\t\t\t\t\t\t\t\toptions.name = url;\n\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\toptions.name = url.name;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (typeof (url) === \"string\")\n\t\t\t\t\t\t\toptions.url = url;\n\t\t\t\t\t\telse\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\toptions.url = url.url;\n\n\t\t\t\t\t\t\tif (processor == null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (url.processor != null)\n\t\t\t\t\t\t\t\t\tprocessor = url.processor;\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tif (args == null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (url.args != null)\n\t\t\t\t\t\t\t\t\targs = url.args;\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tif (url.testerMap != null)\n\t\t\t\t\t\t\t\toptions.testerMap = url.testerMap;\n\n\t\t\t\t\t\t\tif (url.testerName != null)\n\t\t\t\t\t\t\t\toptions.testerName = url.testerName;\n\n\t\t\t\t\t\t\tif (url.testerAPIBaseUrl != null)\n\t\t\t\t\t\t\t\toptions.testerAPIBaseUrl = url.testerAPIBaseUrl;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (processor == null)\n\t\t\t\t\t\t\tprocessor = new HotStaq ();\n\n\t\t\t\t\t\tHotStaq.setupTesters (processor, options);\n\n\t\t\t\t\t\toptions.processor = processor;\n\t\t\t\t\t\toptions.args = args;\n\n\t\t\t\t\t\tif (options.url.indexOf (\"hstqserve\") < 0)\n\t\t\t\t\t\t\toptions.url += \"?hstqserve=nahfam\";\n\n\t\t\t\t\t\tlet output: string = await HotStaq.processUrl (options);\n\n\t\t\t\t\t\toutput += HotStaq.setupClientTesters (processor);\n\n\t\t\t\t\t\tawait HotStaq.useOutput (output);\n\t\t\t\t\t\tresolve (processor);\n\t\t\t\t\t});\n\t\t\t}));\n\t}\n\n\t/**\n\t * Process and replace the current HTML page with the hott script.\n\t * This is meant for web browser use only.\n\t */\n\tstatic async displayContent (content: string | HotStartOptions, name: string = null, \n\t\t\tprocessor: HotStaq = null, args: any = null): Promise<HotStaq>\n\t{\n\t\treturn (new Promise<HotStaq> ((resolve, reject) =>\n\t\t\t{\n\t\t\t\tHotStaq.onReady (async () =>\n\t\t\t\t\t{\n\t\t\t\t\t\tlet options: HotStartOptions = {\n\t\t\t\t\t\t\t\t\"content\": \"\"\n\t\t\t\t\t\t\t};\n\n\t\t\t\t\t\tif (name == null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (typeof (content) === \"string\")\n\t\t\t\t\t\t\t\toptions.name = \"\";\n\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\toptions.name = content.name;\n\t\t\t\t\t\t}\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\toptions.name = name;\n\n\t\t\t\t\t\tif (options.name === \"\")\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (typeof (content) === \"string\")\n\t\t\t\t\t\t\t\toptions.name = \"\"; /// @fixme Is this ok to do?\n\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\toptions.name = content.name;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (typeof (content) === \"string\")\n\t\t\t\t\t\t\toptions.content = content;\n\t\t\t\t\t\telse\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\toptions.content = content.content;\n\n\t\t\t\t\t\t\tif (processor == null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (content.processor != null)\n\t\t\t\t\t\t\t\t\tprocessor = content.processor;\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tif (args == null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (content.args != null)\n\t\t\t\t\t\t\t\t\targs = content.args;\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tif (content.testerMap != null)\n\t\t\t\t\t\t\t\toptions.testerMap = content.testerMap;\n\n\t\t\t\t\t\t\tif (content.testerName != null)\n\t\t\t\t\t\t\t\toptions.testerName = content.testerName;\n\n\t\t\t\t\t\t\tif (content.testerAPIBaseUrl != null)\n\t\t\t\t\t\t\t\toptions.testerAPIBaseUrl = content.testerAPIBaseUrl;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (processor == null)\n\t\t\t\t\t\t\tprocessor = new HotStaq ();\n\n\t\t\t\t\t\tHotStaq.setupTesters (processor, options);\n\n\t\t\t\t\t\toptions.processor = processor;\n\t\t\t\t\t\toptions.args = args;\n\n\t\t\t\t\t\tlet output: string = await HotStaq.processContent (options);\n\n\t\t\t\t\t\tawait HotStaq.useOutput (output);\n\t\t\t\t\t\tresolve (processor);\n\t\t\t\t\t});\n\t\t\t}));\n\t}\n}\n\nif (typeof (document) !== \"undefined\")\n{\n\tlet loadHotStaqSite = function ()\n\t{\n\t\tlet hotstaqElms = document.getElementsByTagName (\"hotstaq\");\n\n\t\t// Set this to true, just in case...\n\t\tHotStaq.isWeb = true;\n\n\t\t\t// @ts-ignore\n\t\tif (typeof (HotStaqWeb) !== \"undefined\")\n\t\t{\n\t\t\t// @ts-ignore\n\t\t\twindow.HotStaq = HotStaqWeb.HotStaq;\n\t\t\t// @ts-ignore\n\t\t\twindow.HotClient = HotStaqWeb.HotClient;\n\t\t\t// @ts-ignore\n\t\t\twindow.HotAPI = HotStaqWeb.HotAPI;\n\t\t\t// @ts-ignore\n\t\t\twindow.Hot = HotStaqWeb.Hot;\n\t\t\t// @ts-ignore\n\t\t\twindow.HotComponent = HotStaqWeb.HotComponent;\n\t\t}\n\n\t\tif (hotstaqElms.length > 0)\n\t\t{\n\t\t\t// @ts-ignore\n\t\t\tlet hotstaqElm: HTMLElement = hotstaqElms[0];\n\n\t\t\tsetTimeout (async function ()\n\t\t\t\t{\n\t\t\t\t\tlet getAttr = (elm: HTMLElement, attrNames: string[]) =>\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tfor (let iIdx = 0; iIdx < attrNames.length; iIdx++)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tlet attrName: string = attrNames[iIdx];\n\n\t\t\t\t\t\t\t\tif (elm.getAttribute (attrName) != null)\n\t\t\t\t\t\t\t\t\treturn (elm.getAttribute (attrName));\n\n\t\t\t\t\t\t\t\tif (elm.getAttribute (`data-${attrName}`) != null)\n\t\t\t\t\t\t\t\t\treturn (elm.getAttribute (`data-${attrName}`));\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\treturn (undefined);\n\t\t\t\t\t\t};\n\n\t\t\t\t\tlet loadPage: string = getAttr (hotstaqElm, [\"load-page\", \"loadPage\", \"src\"]) || \"\";\n\t\t\t\t\tlet router: string = getAttr (hotstaqElm, [\"router\"]) || \"\";\n\t\t\t\t\tlet name: string = getAttr (hotstaqElm, [\"name\"]) || \"default\";\n\t\t\t\t\tlet args: string = getAttr (hotstaqElm, [\"args\"]) || null;\n\t\t\t\t\tlet apiLibrary: string = getAttr (hotstaqElm, [\"api-library\", \"apiLibrary\"]) || null;\n\t\t\t\t\tlet apiName: string = getAttr (hotstaqElm, [\"api-name\", \"apiName\"]) || null;\n\t\t\t\t\tlet apiUrl: string = getAttr (hotstaqElm, [\"api-url\", \"apiUrl\"]) || null;\n\t\t\t\t\tlet testerName: string = getAttr (hotstaqElm, [\"tester-name\", \"testerName\"]) || \"HotTesterMochaSelenium\";\n\t\t\t\t\tlet testerMap: string = getAttr (hotstaqElm, [\"tester-map\", \"testerMap\"]) || null;\n\t\t\t\t\tlet testerApiBaseUrl: string = getAttr (hotstaqElm, [\"tester-api-base-url\", \"testerApiBaseUrl\"]) || null;\n\t\t\t\t\tlet testerLaunchpadUrl: string = getAttr (hotstaqElm, [\"tester-launchpad-url\", \"testerLaunchpadUrl\"]) || null;\n\t\t\t\t\tlet dontReuseProcessor: boolean = false;\n\t\t\t\t\tlet passRawUrl: boolean = false;\n\t\t\t\t\tlet htmlSource: string = hotstaqElm.innerHTML || \"\";\n\t\t\t\t\tlet routerManager: { [path: string]: { redirect: string; base: string; src: string; } } = {};\n\t\t\t\t\tlet routerWildcards: string[] = [];\n\n\t\t\t\t\tif (getAttr (hotstaqElm, [\"src\"]) != null)\n\t\t\t\t\t\tloadPage = getAttr (hotstaqElm, [\"src\"]);\n\n\t\t\t\t\tif (getAttr (hotstaqElm, [\"passRawUrl\"]) != null)\n\t\t\t\t\t\tpassRawUrl = true;\n\n\t\t\t\t\tif (getAttr (hotstaqElm, [\"dont-reuse-processor\", \"dontReuseProcessor\"]) != null)\n\t\t\t\t\t\tdontReuseProcessor = true;\n\n\t\t\t\t\tlet hotstaqErrors = document.getElementsByTagName (\"hotstaq-error\");\n\n\t\t\t\t\tfor (let iIdx = 0; iIdx < hotstaqErrors.length; iIdx++)\n\t\t\t\t\t{\n\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\tlet hotstaqErrorElm: HTMLElement = hotstaqErrors[iIdx];\n\t\t\t\t\t\tlet errorStatus: string = getAttr (hotstaqErrorElm, [\"status\"]);\n\t\t\t\t\t\tlet unsupportedBrowser: string = getAttr (hotstaqErrorElm, [\"unsupported-browser-redirect\"]);\n\n\t\t\t\t\t\tif (unsupportedBrowser != null)\n\t\t\t\t\t\t\tHotStaq.errors[\"unsupportedBrowser\"] = { redirectToUrl: unsupportedBrowser };\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tHotStaq.errors[`${errorStatus}`] = { redirectToUrl: unsupportedBrowser };\n\t\t\t\t\t}\n\n\t\t\t\t\t// Check if async/await is available.\n\t\t\t\t\ttry\n\t\t\t\t\t{\n\t\t\t\t\t\teval (\"async () => {}\");\n\t\t\t\t\t}\n\t\t\t\t\tcatch (ex)\n\t\t\t\t\t{\n\t\t\t\t\t\tHotStaq.executeError (\"unsupportedBrowser\");\n\t\t\t\t\t}\n\n\t\t\t\t\tif (router !== \"\")\n\t\t\t\t\t{\n\t\t\t\t\t\tlet hotstaqRouterElms = document.getElementsByTagName (\"hotstaq-router\");\n\n\t\t\t\t\t\tfor (let iIdx = 0; iIdx < hotstaqRouterElms.length; iIdx++)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\tlet hotstaqRouterElm: HTMLElement = hotstaqRouterElms[iIdx];\n\t\t\t\t\t\t\tlet routerName: string = getAttr (hotstaqRouterElm, [\"name\"]);\n\t\t\t\t\t\t\tlet serveLocally: string = getAttr (hotstaqRouterElm, [\"serve-local\", \"serveLocally\"]);\n\n\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\tif (routerName === router)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t// Load all routes from the router.\n\t\t\t\t\t\t\t\tfor (let iJdx = 0; iJdx < hotstaqRouterElm.childNodes.length; iJdx++)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\tlet routerElm: HTMLElement = hotstaqRouterElm.childNodes[iJdx];\n\n\t\t\t\t\t\t\t\t\tif (routerElm instanceof HTMLElement)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tif (routerElm.tagName.toUpperCase () === \"ROUTE\")\n\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\tlet routerPath: string = getAttr (routerElm, [\"path\"]);\n\t\t\t\t\t\t\t\t\t\t\tlet redirect: string = getAttr (routerElm, [\"redirect\"]);\n\t\t\t\t\t\t\t\t\t\t\tlet base: string = getAttr (routerElm, [\"base\"]);\n\t\t\t\t\t\t\t\t\t\t\tlet routerSrc: string = getAttr (routerElm, [\"src\"]);\n\n\t\t\t\t\t\t\t\t\t\t\tif (routerPath.indexOf (\"*\") > -1)\n\t\t\t\t\t\t\t\t\t\t\t\trouterWildcards.push (routerPath);\n\n\t\t\t\t\t\t\t\t\t\t\trouterManager[routerPath] = {\n\t\t\t\t\t\t\t\t\t\t\t\t\tredirect: redirect || undefined, \n\t\t\t\t\t\t\t\t\t\t\t\t\tbase: base || undefined, \n\t\t\t\t\t\t\t\t\t\t\t\t\tsrc: routerSrc || undefined\n\t\t\t\t\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tlet tempPathname: string = window.location.pathname;\n\n\t\t\t\t\t\t\t\tif (serveLocally != null)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tconst lowerServeLocally: string = serveLocally.toLowerCase ();\n\n\t\t\t\t\t\t\t\t\tif ((lowerServeLocally === \"true\") ||\n\t\t\t\t\t\t\t\t\t\t(lowerServeLocally === \"yes\") ||\n\t\t\t\t\t\t\t\t\t\t(lowerServeLocally === \"1\"))\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tconst lastSlashPos: number = tempPathname.lastIndexOf (\"/\");\n\n\t\t\t\t\t\t\t\t\t\tif (lastSlashPos > -1)\n\t\t\t\t\t\t\t\t\t\t\ttempPathname = tempPathname.substring (lastSlashPos);\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tif (routerWildcards.length > 0)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t// Serve locally doesn't really work with wildcards\n\t\t\t\t\t\t\t\t\t/// @fixme This isn't actually working like a wildcard should. This needs to be improved.\n\t\t\t\t\t\t\t\t\tfor (let iJdx = 0; iJdx < routerWildcards.length; iJdx++)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tlet routeWildcard: string = routerWildcards[iJdx];\n\t\t\t\t\t\t\t\t\t\tlet tempRouteWildcard: string = routeWildcard.replace (\"*\", \"\");\n\n\t\t\t\t\t\t\t\t\t\tif (tempPathname.indexOf (tempRouteWildcard) > -1)\n\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\ttempPathname = routeWildcard;\n\n\t\t\t\t\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\t// Find the correct route and load it.\n\t\t\t\t\t\t\t\tif (routerManager[tempPathname] != null)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tif (routerManager[tempPathname].redirect != null)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\twindow.location.href = routerManager[tempPathname].redirect;\n\n\t\t\t\t\t\t\t\t\t\treturn;\n\t\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\t\tif (routerManager[tempPathname].src != null)\n\t\t\t\t\t\t\t\t\t\tloadPage = routerManager[tempPathname].src;\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tif (args != null)\n\t\t\t\t\t\targs = JSON.parse (args);\n\t\t\t\t\telse\n\t\t\t\t\t\targs = Hot.Arguments;\n\n\t\t\t\t\tlet hasHtmlSource: boolean = false;\n\n\t\t\t\t\tif (htmlSource !== \"\")\n\t\t\t\t\t{\n\t\t\t\t\t\tconst htmlSourceCheck: string = htmlSource.replace (/\\s/g,'');\n\n\t\t\t\t\t\tif (htmlSourceCheck !== \"\")\n\t\t\t\t\t\t\thasHtmlSource = true;\n\t\t\t\t\t}\n\n\t\t\t\t\tlet tempMode = 0;\n\n\t\t\t\t\t// @ts-ignore\n\t\t\t\t\tif (window[\"Hot\"] != null)\n\t\t\t\t\t\ttempMode = Hot.Mode;\n\t\t\t\n\t\t\t\t\tlet processor: HotStaq = null;\n\n\t\t\t\t\tif (dontReuseProcessor === false)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (typeof (Hot) !== \"undefined\")\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (Hot.CurrentPage != null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (Hot.CurrentPage.processor != null)\n\t\t\t\t\t\t\t\t\tprocessor = Hot.CurrentPage.processor;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tif (processor == null)\n\t\t\t\t\t\tprocessor = new HotStaq ();\n\n\t\t\t\t\tprocessor.mode = tempMode;\n\n\t\t\t\t\tlet options: HotStartOptions = {\n\t\t\t\t\t\t\tname: name,\n\t\t\t\t\t\t\tprocessor: processor,\n\t\t\t\t\t\t\targs: args\n\t\t\t\t\t\t};\n\n\t\t\t\t\tif (loadPage !== \"\")\n\t\t\t\t\t{\n\t\t\t\t\t\tif (passRawUrl === false)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (loadPage.indexOf (\"hstqserve\") < 0)\n\t\t\t\t\t\t\t\tloadPage += \"?hstqserve=nahfam\";\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\toptions.url = loadPage;\n\t\t\t\t\t}\n\n\t\t\t\t\tif (testerMap != null)\n\t\t\t\t\t{\n\t\t\t\t\t\toptions.testerMap = testerMap;\n\t\t\t\t\t\toptions.testerName = testerName;\n\t\t\t\t\t}\n\n\t\t\t\t\tif (testerName != null)\n\t\t\t\t\t\toptions.testerName = testerName;\n\n\t\t\t\t\tif (testerApiBaseUrl != null)\n\t\t\t\t\t\toptions.testerAPIBaseUrl = testerApiBaseUrl;\n\n\t\t\t\t\tif (testerLaunchpadUrl != null)\n\t\t\t\t\t\toptions.testerLaunchpadUrl = testerLaunchpadUrl;\n\n\t\t\t\t\tif (apiName != null)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet client = new HotClient (processor);\n\n\t\t\t\t\t\tif (apiUrl === \"\")\n\t\t\t\t\t\t\tthrow new Error (`api-url was not set!`);\n\n\t\t\t\t\t\tlet parentLib: any = window;\n\n\t\t\t\t\t\tif (apiLibrary != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\tparentLib = window[apiLibrary];\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tlet newAPI = new parentLib[apiName] (apiUrl, client);\n\t\t\t\t\t\tnewAPI.connection.api = newAPI;\n\t\t\t\t\t\tprocessor.api = newAPI;\n\t\t\t\t\t}\n\n\t\t\t\t\tif (hasHtmlSource === false)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (loadPage === \"\")\n\t\t\t\t\t\t\tthrow new Error (`The hotstaq tag must have a src, HTML contents inside it, or a router set.`);\n\n\t\t\t\t\t\tHotStaq.displayUrl (options);\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tHotStaq.displayContent (options);\n\t\t\t\t\t}\n\t\t\t\t}, 50);\n\t\t}\n\t};\n\n\t/// @ts-ignore\n\twindow.ethereum22 = window.ethereum;\n\twindow.addEventListener (\"load\", loadHotStaqSite);\n}\n","import { HotStaq } from \"./HotStaq\";\nimport { HotTestElement, HotTestElementOptions } from \"./HotTestElement\";\nimport { HotTestPage } from \"./HotTestMap\";\n\n/**\n * This actually executes the tests.\n */\nexport abstract class HotTestDriver\n{\n\t/**\n\t * The current page.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The current page.\n\t */\n\tpage: HotTestPage;\n\t/**\n\t * The delay in milliseconds between each executed command.\n\t */\n\tcommandDelay: number;\n\t/**\n\t * Any data that needs to be saved between the different testing runs.\n\t */\n\tpersistentData: any;\n\n\tconstructor (processor: HotStaq, page: HotTestPage = null)\n\t{\n\t\tthis.processor = processor;\n\t\tthis.page = page;\n\t\tthis.commandDelay = 20;\n\t\tthis.persistentData = {};\n\t}\n\n\t/**\n\t * Get a test object by it's name. If a * is used, it will be used as a \n\t * wildcard for the object's name. If a > is used, then the name will \n\t * be treated as a CSS selector.\n\t */\n\tparseTestObject (name: string): string\n\t{\n\t\tlet pos: number = name.indexOf (\"*\");\n\t\tlet wildcard: string = \"\";\n\n\t\tif (pos > -1)\n\t\t{\n\t\t\tname = name.replace (/\\*/, \"\");\n\t\t\twildcard = \"*\";\n\t\t}\n\n\t\tlet selector: string = `[data-test-object-name${wildcard}='${name}']`;\n\t\tpos = name.indexOf (\">\");\n\n\t\tif (pos > -1)\n\t\t{\n\t\t\tname = name.replace (/\\>/, \"\");\n\t\t\tselector = name;\n\t\t}\n\n\t\treturn (selector);\n\t}\n\n\t/**\n\t * Wait for a number of milliseconds.\n\t */\n\tasync wait (numMilliseconds: number): Promise<void>\n\t{\n\t\treturn (await new Promise ((resolve, reject) =>\n\t\t\t{\n\t\t\t\tsetTimeout (() =>\n\t\t\t\t\t{\n\t\t\t\t\t\tresolve ();\n\t\t\t\t\t}, numMilliseconds);\n\t\t\t}));\n\t}\n\n\t/**\n\t * Print a message.\n\t */\n\tasync print (message: string): Promise<void>\n\t{\n\t\tprocess.stdout.write (message);\n\t}\n\n\t/**\n\t * Print a message line.\n\t */\n\tasync println (message: string): Promise<void>\n\t{\n\t\tawait this.print (`${message}\\n`);\n\t}\n\n\t/**\n\t * Disconnect this server or destroy anything associated with this HotTestDriver.\n\t */\n\tabstract destroy (): Promise<void>;\n\n\t/**\n\t * Navigate to a url.\n\t */\n\tabstract navigateToUrl (url: string): Promise<void>;\n\t/**\n\t * Wait for a HotTestElement to load.\n\t */\n\tabstract waitForTestElement (name: string | HotTestElement, options?: HotTestElementOptions): Promise<any>;\n\t/**\n\t * Find a HotTestElement to utilize.\n\t */\n\tabstract findTestElement (name: string | HotTestElement, options?: HotTestElementOptions): Promise<any>;\n\t/**\n\t * Run a HotTestElement command.\n\t */\n\tabstract runCommand (testElm: string | HotTestElement, funcName?: string, valueStr?: string): Promise<any>;\n\t/**\n\t * An expression to test.\n\t */\n\tabstract assertElementValue (name: string | HotTestElement, value: any, \n\t\terrorMessage?: string, options?: HotTestElementOptions): Promise<any>;\n\t/**\n\t * An expression to test.\n\t */\n\tasync assert (value: any, errorMessage: string = \"\"): Promise<any>\n\t{\n\t\tif (! (value))\n\t\t\tthrow new Error (errorMessage);\n\t}\n\n\t/**\n\t * Run a series of test elements.\n\t */\n\tasync run (executions: string[] | string[][]): Promise<any[]>\n\t{\n\t\tlet results: any[] = [];\n\n\t\tfor (let iIdx = 0; iIdx < executions.length; iIdx++)\n\t\t{\n\t\t\tlet execution: any = executions[iIdx];\n\t\t\tlet testElm: HotTestElement = null;\n\t\t\tlet func: string = \"\";\n\t\t\tlet value: string = \"\";\n\n\t\t\tif (typeof (execution) === \"string\")\n\t\t\t{\n\t\t\t\ttestElm = this.page.testElements[execution];\n\n\t\t\t\t/// @fixme This is going to wreck selecting test elements by wildcards.\n\t\t\t\tif (testElm == null)\n\t\t\t\t\tthrow new Error (`HotTestDriver: Unable to find test element ${execution}`);\n\n\t\t\t\tfunc = testElm.func;\n\t\t\t\tvalue = testElm.value;\n\t\t\t}\n\n\t\t\tif (execution instanceof Array)\n\t\t\t{\n\t\t\t\tlet name: string = execution[0];\n\t\t\t\ttestElm = this.page.testElements[name];\n\n\t\t\t\t// This null catch is specifically to help find wildcard test elements.\n\t\t\t\tif (testElm == null)\n\t\t\t\t{\n\t\t\t\t\ttestElm = new HotTestElement (name);\n\t\t\t\t\tfunc = execution[1];\n\t\t\t\t\tvalue = execution[2];\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\tfunc = testElm.func;\n\t\t\t\t\tvalue = testElm.value;\n\n\t\t\t\t\tif (execution.length > 1)\n\t\t\t\t\t\tfunc = execution[1];\n\n\t\t\t\t\tif (execution.length > 2)\n\t\t\t\t\t\tvalue = execution[2];\n\t\t\t\t}\n\t\t\t}\n\n\t\t\ttestElm.func = func;\n\t\t\ttestElm.value = value;\n\n\t\t\tlet result = await this.runCommand (testElm);\n\n\t\t\tawait HotStaq.wait (this.commandDelay);\n\n\t\t\tresults.push (result);\n\t\t}\n\n\t\treturn (results);\n\t}\n}","/**\n * Hot test element options.\n */\nexport interface IHotTestElementOptions\n{\n\t/**\n\t * Indicates that the test element must be visible in \n\t * order to select it.\n\t */\n\tmustBeVisible?: boolean;\n\t/**\n\t * If the test element is missing, ignore the error. This \n\t * will cause the rest of the function to return immediately \n\t * without any exceptions being thrown.\n\t */\n\tignoreMissingElementError?: boolean;\n}\n\n/**\n * Hot test element options.\n */\nexport class HotTestElementOptions implements IHotTestElementOptions\n{\n\t/**\n\t * Indicates that the test element must be visible in \n\t * order to select it.\n\t */\n\tmustBeVisible: boolean;\n\t/**\n\t * If the test element is missing, ignore the error. This \n\t * will cause the rest of the function to return immediately \n\t * without any exceptions being thrown.\n\t */\n\tignoreMissingElementError: boolean;\n\n\tconstructor (copy: IHotTestElementOptions = {})\n\t{\n\t\tthis.mustBeVisible = copy.mustBeVisible || true;\n\t\tthis.ignoreMissingElementError = copy.ignoreMissingElementError || false;\n\t}\n}\n\n/**\n * A test element.\n */\nexport interface IHotTestElement\n{\n\t/**\n\t * The name of the element.\n\t */\n\tname: string;\n\t/**\n\t * The name of the function to execute \n\t * while executing the test.\n\t */\n\tfunc?: string;\n\t/**\n\t * The value to use.\n\t */\n\tvalue?: any;\n}\n\n/**\n * A test element.\n */\nexport class HotTestElement implements IHotTestElement\n{\n\t/**\n\t * The name of the element.\n\t */\n\tname: string;\n\t/**\n\t * The name of the function to execute \n\t * while executing the test.\n\t */\n\tfunc: string;\n\t/**\n\t * The value to use.\n\t */\n\tvalue: any;\n\n\tconstructor (name: string | IHotTestElement, func: string = \"\", value: any = null)\n\t{\n\t\tif (typeof (name) === \"string\")\n\t\t{\n\t\t\tthis.name = name;\n\t\t\tthis.func = func;\n\t\t\tthis.value = value;\n\t\t}\n\t\telse\n\t\t{\n\t\t\tthis.name = name.name;\n\t\t\tthis.func = name.func || func;\n\t\t\tthis.value = name.value || value;\n\t\t}\n\t}\n}","import { HotTestElement } from \"./HotTestElement\";\nimport { HotTestDriver } from \"./HotTestDriver\";\nimport { HotSiteMapPath } from \"./HotStaq\";\n\n/**\n * Create a test path for later execution.\n */\nexport type HotTestPath = (driver: HotTestDriver, ...args: any) => Promise<any>;\n\n/**\n * The destination to take in a map.\n */\nexport class HotTestDestination\n{\n\t/**\n\t * The destination to take.\n\t */\n\tdestination: string;\n\t/**\n\t * If set to true, this will automatically start executing it's \n\t * tests when it's time.\n\t */\n\tautoStart: boolean;\n\n\tconstructor (destination: string | HotTestDestination | HotSiteMapPath = \"\", autoStart: boolean = true)\n\t{\n\t\tif (typeof (destination) === \"string\")\n\t\t{\n\t\t\tthis.destination = destination;\n\t\t\tthis.autoStart = autoStart;\n\t\t}\n\t\telse\n\t\t{\n\t\t\tif (destination instanceof HotTestDestination)\n\t\t\t{\n\t\t\t\tthis.destination = destination.destination;\n\t\t\t\tthis.autoStart = destination.autoStart;\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\tthis.destination = destination.path;\n\t\t\t\tthis.autoStart = destination.autoStart;\n\t\t\t}\n\t\t}\n\t}\n}\n\n/**\n * A page containing only test related info.\n */\nexport interface HotTestPage\n{\n\t/**\n\t * The elements to test on this map.\n\t */\n\ttestElements: { [name: string]: HotTestElement; };\n\t/**\n\t * The test paths to test on this map.\n\t */\n\ttestPaths: { [name: string]: HotTestPath; };\n}\n\n/**\n * Maps the paths that are taken to complete a test.\n */\nexport class HotTestMap\n{\n\t/**\n\t * The order in which paths are to be taken. Each destination is a string \n\t * in a type -> path order. The type could be either a page or api route. \n\t * For example:\n\t * ```\n\t * [\n\t *      \"page:signin_page -> signin_path\",\n\t *      \"page:account_page -> change_username_path\",\n\t *      \"page:account_page -> change_password_path\",\n\t *      \"page:account_page -> change_name_path -> change_address_path\",\n\t * \t\t\"page:account_page -> signout_path\",\n\t * \t\t\"api:account_api_route -> signout_route_method -> signout_test_path\"\n\t * ]\n\t * ```\n\t * \n\t * The first string to the left of the -> will always be the type, such as a \n\t * page or an api route. Any strings to the right of the -> will be a path, even \n\t * when chaining addtional ->'s.\n\t */\n\tdestinations: HotTestDestination[] | { [name: string]: HotTestDestination; };\n\t/**\n\t * The order in which destinations are supposed to execute. This is \n\t * ignored if the destinations are an array.\n\t */\n\tdestinationOrder: string[];\n\t/**\n\t * The test pages to execute.\n\t */\n\tpages: {\n\t\t\t[name: string]: HotTestPage\n\t\t};\n\n\tconstructor (destinations: string[] | HotTestDestination[] | { [name: string]: string | HotTestDestination; } = [], \n\t\tpages: { [name: string]: HotTestPage } = {}, destinationOrder: string[] = [])\n\t{\n\t\t// Go through and convert any strings into HotTestDestinations.\n\t\tif (destinations instanceof Array)\n\t\t{\n\t\t\tthis.destinations = [];\n\n\t\t\tfor (let iIdx = 0; iIdx < destinations.length; iIdx++)\n\t\t\t{\n\t\t\t\tlet dest = destinations[iIdx];\n\n\t\t\t\tthis.destinations.push (new HotTestDestination (dest));\n\t\t\t}\n\t\t}\n\t\telse\n\t\t{\n\t\t\tthis.destinations = {};\n\n\t\t\tfor (let key in destinations)\n\t\t\t{\n\t\t\t\tlet dest = destinations[key];\n\n\t\t\t\tthis.destinations[key] = new HotTestDestination (dest);\n\t\t\t}\n\t\t}\n\n\t\tthis.destinationOrder = destinationOrder;\n\t\tthis.pages = pages;\n\t}\n}","import { HotStaq } from \"./HotStaq\";\nimport { HotRoute } from \"./HotRoute\";\nimport { HotRouteMethod, TestCaseObject } from \"./HotRouteMethod\";\nimport { HotTestDriver } from \"./HotTestDriver\";\nimport { HotTestDestination, HotTestMap, HotTestPage, HotTestPath } from \"./HotTestMap\";\n\n/**\n * The test stop that is executed as either a destination or \n * a path.\n */\nexport interface HotTestStop\n{\n\t/**\n\t * A command to execute. Can be:\n\t * * print(x)\n\t *   * Print a message to the server's console.\n\t * * println(x)\n\t *   * Print a message with a new line to the server's console.\n\t * * url(x)\n\t *   * Open a url. Must be an absolute url.\n\t * * waitForTesterAPIData\n\t *   * This will wait for the tester API to receive data.\n\t * * wait(x)\n\t *   * This will wait for x number of milliseconds.\n\t * * waitForTestObject(x)\n\t *   * This will wait for a test object to be loaded.\n\t */\n\tcmd: string;\n\t/**\n\t * The destination to execute.\n\t */\n\tdest: string;\n\t/**\n\t * The path to execute.\n\t */\n\tpath: string;\n}\n\n/**\n * The destination for a test to take.\n */\nexport interface HotDestination\n{\n\t/**\n\t * The name of the map.\n\t */\n\tmapName: string;\n\t/**\n\t * The page to start at.\n\t */\n\tpage: string;\n\t/**\n\t * The API route to start using.\n\t */\n\tapi: string;\n\t/**\n\t * The paths to take on the page.\n\t */\n\tpaths: HotTestStop[];\n}\n\n/**\n * Executes tests.\n */\nexport abstract class HotTester\n{\n\t/**\n\t * The tester name.\n\t */\n\tname: string;\n\t/**\n\t * The base url that will construct future urls.\n\t */\n\tbaseUrl: string;\n\t/**\n\t * The associated processor.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The test maps to test.\n\t */\n\ttestMaps: { [name: string]: HotTestMap; };\n\t/**\n\t * The driver to use when running tests.\n\t */\n\tdriver: HotTestDriver;\n\t/**\n\t * Has this tester finished loading?\n\t */\n\tfinishedLoading: boolean;\n\t/**\n\t * Has this tester finished setting up?\n\t */\n\thasBeenSetup: boolean;\n\t/**\n\t * Has this tester finished setting up?\n\t */\n\thasBeenDestroyed: boolean;\n\n\tconstructor (processor: HotStaq, name: string, baseUrl: string, \n\t\tdriver: HotTestDriver, testMaps: { [name: string]: HotTestMap; } = {})\n\t{\n\t\tthis.processor = processor;\n\t\tthis.name = name;\n\t\tthis.baseUrl = baseUrl;\n\t\tthis.testMaps = testMaps;\n\t\tthis.driver = driver;\n\t\tthis.finishedLoading = false;\n\t\tthis.hasBeenSetup = false;\n\t\tthis.hasBeenDestroyed = false;\n\t}\n\n\t/**\n\t * Executed when setting up the tester.\n\t */\n\tabstract setup (isWebRoute: boolean, url: string, destinationKey?: string): Promise<void>;\n\t/**\n\t * Executed when destroying up the tester.\n\t */\n\tabstract destroy (): Promise<void>;\n\n\t/**\n\t * Executed when tests are started. If this returns true, it will \n\t * continue and execute all test paths. If this returns it will \n\t * skip all test paths and execute onTestEnd instead.\n\t */\n\tasync onTestStart? (destination: HotDestination, url: string, destinationKey?: string): Promise<boolean>;\n\t/**\n\t * Executed when an API test path has started. If this returns false, \n\t * the testPath will not be immediately executed afterwards.\n\t */\n\tasync onTestAPIPathStart? (destination: HotDestination, method: HotRouteMethod, \n\t\ttestName: string, continueWhenTestIsComplete?: boolean): Promise<boolean>;\n\t/**\n\t * Executed when an API test path has ended.\n\t */\n\tasync onTestAPIPathEnd? (destination: HotDestination, method: HotRouteMethod, \n\t\ttestName: string, result: any, continueWhenTestIsComplete?: boolean): Promise<void>;\n\t/**\n\t * Executed when page tests are started. If this returns false, the testPath will not be \n\t * immediately executed afterwards.\n\t */\n\tasync onTestPagePathStart? (destination: HotDestination, page: HotTestPage, \n\t\tstop: HotTestStop, continueWhenTestIsComplete?: boolean): Promise<boolean>;\n\t/**\n\t * Executed when a page test has ended.\n\t */\n\tasync onTestPagePathEnd? (destination: HotDestination, testPath: HotTestPath, \n\t\tresult: any, continueWhenTestIsComplete?: boolean): Promise<void>;\n\t/**\n\t * Executed when a command is executed.\n\t */\n\tasync onCommand? (destination: HotDestination, page: HotTestPage, stop: HotTestStop, \n\t\tcmd: string, args: string[], cmdFunc: ((cmdArgs: string[]) => Promise<void>)): Promise<void>;\n\t/**\n\t * Executed when tests are finished.\n\t */\n\tasync onTestEnd? (destination: HotDestination): Promise<void>;\n\n\t/**\n\t * Executed when this tester has been executed from the API.\n\t */\n\tasync onExecute? (): Promise<void>;\n\t/**\n\t * Executed when this tester has finished loading all data from the API.\n\t */\n\tasync onFinishedLoading? (): Promise<void>;\n\n\t/**\n\t * Waits for the API to finish loading all data.\n\t */\n\tasync waitForData (): Promise<void>\n\t{\n\t\twhile (this.finishedLoading === false)\n\t\t\tawait HotStaq.wait (10);\n\t}\n\n\t/**\n\t * Get a test page.\n\t */\n\tgetTestPage (destination: HotDestination): HotTestPage\n\t{\n\t\tlet page = this.testMaps[destination.mapName].pages[destination.page];\n\n\t\treturn (page);\n\t}\n\n\t/**\n\t * Get a test path.\n\t */\n\tgetTestPath (destination: HotDestination, pathName: string): HotTestPath\n\t{\n\t\tlet page = this.testMaps[destination.mapName].pages[destination.page];\n\n\t\treturn (page.testPaths[pathName]);\n\t}\n\n\t/**\n\t * Get a destination JSON object to use.\n\t */\n\tstatic interpretDestination (mapName: string, testDest: HotTestDestination): HotDestination\n\t{\n\t\tlet destination: string = testDest.destination;\n\t\tlet newDestination: HotDestination = {\n\t\t\t\tmapName: mapName,\n\t\t\t\tpage: \"\",\n\t\t\t\tapi: \"\",\n\t\t\t\tpaths: []\n\t\t\t};\n\t\tlet strs: string[] = destination.split (/\\-\\>/g);\n\t\tlet type: string = strs[0];\n\n\t\tif (type.length < 2)\n\t\t\treturn (null);\n\n\t\tif ((type[0] === \"/\") && (type[1] === \"/\"))\n\t\t\treturn (null);\n\n\t\tlet getType: (typeStr: string, typeDelimiter: string) => string = \n\t\t\t(typeStr: string, typeDelimiter: string): string =>\n\t\t\t{\n\t\t\t\tlet pos: number = typeStr.indexOf (typeDelimiter);\n\t\t\t\tlet typeValue: string = \"\";\n\t\t\n\t\t\t\tif (pos > -1)\n\t\t\t\t{\n\t\t\t\t\ttypeValue = typeStr.substr (pos + typeDelimiter.length);\n\t\t\t\t\ttypeValue = typeValue.trim ();\n\t\t\t\t}\n\n\t\t\t\treturn (typeValue);\n\t\t\t};\n\n\t\tnewDestination.page = getType (type, \"page:\");\n\t\tnewDestination.api = getType (type, \"api:\");\n\n\t\tfor (let iIdx = 1; iIdx < strs.length; iIdx++)\n\t\t{\n\t\t\tlet newPathStr: string = strs[iIdx];\n\t\t\tlet newPath: HotTestStop = {\n\t\t\t\t\tcmd: \"\",\n\t\t\t\t\tdest: \"\",\n\t\t\t\t\tpath: \"\"\n\t\t\t\t};\n\n\t\t\tnewPathStr = newPathStr.trim ();\n\t\t\tnewPath.dest = getType (newPathStr, \"dest:\");\n\t\t\tnewPath.cmd = getType (newPathStr, \"cmd:\");\n\t\t\tnewPath.path = getType (newPathStr, \"path:\");\n\n\t\t\tif ((newPath.dest == \"\") && (newPath.cmd == \"\") && (newPath.path == \"\"))\n\t\t\t\tnewPath.path = newPathStr;\n\n\t\t\tnewDestination.paths.push (newPath);\n\t\t}\n\n\t\treturn (newDestination);\n\t}\n\n\t/**\n\t * Execute an API's test path.\n\t */\n\tasync executeTestAPIPath (destination: HotDestination, method: HotRouteMethod, \n\t\ttestName: string, skipEventCalls: boolean = false, continueWhenTestIsComplete: boolean = false): Promise<any>\n\t{\n\t\tlet runTestPath: boolean = true;\n\n\t\tif (method == null)\n\t\t\tthrow new Error (`Trying to access null method on destination map ${destination.mapName}.`);\n\n\t\t// A dumb hack to prevent any recursion that could occur.\n\t\tif (skipEventCalls === false)\n\t\t{\n\t\t\tif (this.onTestAPIPathStart != null)\n\t\t\t\trunTestPath = await this.onTestAPIPathStart (destination, method, testName, continueWhenTestIsComplete);\n\t\t}\n\n\t\tlet result: any = null;\n\n\t\tif (runTestPath === true)\n\t\t{\n\t\t\tlet testCaseObject: TestCaseObject = method.testCases[testName];\n\n\t\t\tif (testCaseObject == null)\n\t\t\t\tthrow new Error (`HotTester: Test case object ${testName} does not exist!`);\n\n\t\t\tresult = await testCaseObject.func (this.driver);\n\t\t}\n\n\t\tif (skipEventCalls === false)\n\t\t{\n\t\t\tif (this.onTestAPIPathEnd != null)\n\t\t\t\tawait this.onTestAPIPathEnd (destination, method, testName, result, continueWhenTestIsComplete);\n\t\t}\n\n\t\treturn (result);\n\t}\n\n\t/**\n\t * Execute all test paths in an API route.\n\t * \n\t * @fixme This needs a better implementation...\n\t */\n\tasync executeTestAPIPaths (destination: HotDestination): Promise<any[]>\n\t{\n\t\tlet results: any[] = [];\n\t\tlet testMap: HotTestMap = this.testMaps[destination.mapName];\n\n\t\tif (testMap == null)\n\t\t\tthrow new Error (`HotTester: Map ${destination.mapName} does not exist!`);\n\n\t\tif (this.processor.api == null)\n\t\t\tthrow new Error (`HotTester: Associated processor does not have an API!`);\n\n\t\tlet route: HotRoute = this.processor.api.routes[destination.api];\n\n\t\tif (route == null)\n\t\t\tthrow new Error (`HotTester: API does not have route ${destination.api}!`);\n\n\t\t// Iterate through each path in the destination until complete.\n\t\tfor (let iIdx = 0; iIdx < destination.paths.length; iIdx += 2)\n\t\t{\n\t\t\tlet stop: HotTestStop = destination.paths[iIdx];\n\t\t\tlet pathName: string = stop.path;\n\t\t\tlet method: HotRouteMethod = route.getMethod (pathName);\n\t\t\tlet nextStop: HotTestStop = destination.paths[iIdx + 1];\n\t\t\tlet testName: string = nextStop.path;\n\n\t\t\tif (method == null)\n\t\t\t\tthrow new Error (`Unable to find method related to path ${pathName} in map ${destination.mapName}`);\n\n\t\t\tlet result: any = await this.executeTestAPIPath (destination, method, testName);\n\n\t\t\tresults.push (result);\n\t\t}\n\n\t\treturn (results);\n\t}\n\n\t/**\n\t * Execute a test page path.\n\t */\n\tasync executeTestPagePath (destination: HotDestination, stop: HotTestStop, \n\t\tskipEventCalls: boolean = false, continueWhenTestIsComplete: boolean = false): Promise<any>\n\t{\n\t\tlet runTestPath: boolean = true;\n\t\tlet testMap: HotTestMap = this.testMaps[destination.mapName];\n\n\t\t/// @fixme For some reason the errors being thrown here are not being thrown.\n\t\tif (testMap == null)\n\t\t\tthrow new Error (`HotTester: Map ${destination.mapName} does not exist!`);\n\n\t\tlet page: HotTestPage = testMap.pages[destination.page];\n\n\t\tif (page == null)\n\t\t\tthrow new Error (`HotTester: Page ${destination.page} does not exist!`);\n\n\t\tthis.driver.page = page;\n\n\t\tlet testPathName: string = stop.path;\n\t\tlet testPath: HotTestPath = page.testPaths[testPathName];\n\n\t\t// A dumb hack to prevent any recursion that could occur.\n\t\tif (skipEventCalls === false)\n\t\t{\n\t\t\tif (this.onTestPagePathStart != null)\n\t\t\t\trunTestPath = await this.onTestPagePathStart (destination, page, stop, continueWhenTestIsComplete);\n\t\t}\n\n\t\tlet result: any = null;\n\n\t\tif (runTestPath === true)\n\t\t{\n\t\t\tif (testPath == null)\n\t\t\t{\n\t\t\t\tthrow new Error (`HotTester: Test path ${testPathName} does not have a function!`);\n\t\t\t}\n\n\t\t\tresult = await testPath (this.driver);\n\t\t}\n\n\t\tif (skipEventCalls === false)\n\t\t{\n\t\t\tif (this.onTestPagePathEnd != null)\n\t\t\t\tawait this.onTestPagePathEnd (destination, testPath, result, continueWhenTestIsComplete);\n\t\t}\n\n\t\treturn (result);\n\t}\n\n\t/**\n\t * Execute a command.\n\t */\n\tasync executeCommand (destination: HotDestination, page: HotTestPage, stop: HotTestStop, cmd: string): Promise<void>\n\t{\n\t\t/**\n\t\t * Check if the input command matches.\n\t\t */\n\t\tlet hasCmd: (input: string, cmd: string, hasArguments: boolean) => boolean = \n\t\t\t(input: string, cmd: string, hasArguments: boolean): boolean =>\n\t\t\t{\n\t\t\t\tlet result: boolean = false;\n\n\t\t\t\tif (stop.cmd === cmd)\n\t\t\t\t\tresult = true;\n\n\t\t\t\tconst pos: number = stop.cmd.indexOf (\"(\");\n\n\t\t\t\t// If there's parenthesis, get the incoming command.\n\t\t\t\tif (pos > -1)\n\t\t\t\t{\n\t\t\t\t\tlet inputCmd: string = stop.cmd.substr (0, pos);\n\n\t\t\t\t\tif (inputCmd === cmd)\n\t\t\t\t\t\tresult = true;\n\t\t\t\t}\n\n\t\t\t\treturn (result);\n\t\t\t};\n\t\t/**\n\t\t * Get the arguments in a command. This will only return a \n\t\t * single argument for now.\n\t\t * \n\t\t * @fixme Add support for multiple arguments.\n\t\t */\n\t\tlet getCmdArgs: (input: string) => string[] = \n\t\t\t(input: string): string[] =>\n\t\t\t{\n\t\t\t\tlet results: string[] = [];\n\t\t\t\tlet matches = input.match (/(?=\\()(.*?)(?=\\))/g);\n\n\t\t\t\tif (matches != null)\n\t\t\t\t{\n\t\t\t\t\tlet tempMatch = matches[0];\n\n\t\t\t\t\t// A little hack, since I suck at Regex :(\n\t\t\t\t\ttempMatch = tempMatch.substr (2, tempMatch.length);\n\n\t\t\t\t\tresults.push (tempMatch);\n\t\t\t\t}\n\n\t\t\t\tif (results.length < 1)\n\t\t\t\t\tthrow new Error (`HotTester: Command ${input} requires arguments, but none were supplied.`);\n\n\t\t\t\treturn (results);\n\t\t\t};\n\n\t\tlet cmdFunc: ((cmdArgs: string[]) => Promise<void>) = null;\n\t\tlet args: string[] = [];\n\n\t\tif (hasCmd (stop.cmd, \"waitForTesterAPIData\", false) === true)\n\t\t{\n\t\t\tcmdFunc = async (cmdArgs: string[]): Promise<void> =>\n\t\t\t\t{\n\t\t\t\t\tthis.finishedLoading = false;\n\t\t\t\t\tawait this.waitForData ();\n\t\t\t\t};\n\t\t}\n\n\t\tif (hasCmd (stop.cmd, \"wait\", true) === true)\n\t\t{\n\t\t\targs = getCmdArgs (stop.cmd);\n\n\t\t\tcmdFunc = async (cmdArgs: string[]): Promise<void> =>\n\t\t\t\t{\n\t\t\t\t\tlet numMilliseconds: number = parseInt (cmdArgs[0]);\n\n\t\t\t\t\tawait HotStaq.wait (numMilliseconds);\n\t\t\t\t};\n\t\t}\n\n\t\tif (hasCmd (stop.cmd, \"url\", true) === true)\n\t\t{\n\t\t\targs = getCmdArgs (stop.cmd);\n\n\t\t\tcmdFunc = async (cmdArgs: string[]): Promise<void> =>\n\t\t\t\t{\n\t\t\t\t\tlet input: string = cmdArgs[0];\n\n\t\t\t\t\tawait this.driver.navigateToUrl (input);\n\t\t\t\t};\n\t\t}\n\n\t\tif (hasCmd (stop.cmd, \"print\", true) === true)\n\t\t{\n\t\t\targs = getCmdArgs (stop.cmd);\n\n\t\t\tcmdFunc = async (cmdArgs: string[]): Promise<void> =>\n\t\t\t\t{\n\t\t\t\t\tlet input: string = cmdArgs[0];\n\n\t\t\t\t\tawait this.driver.print (input);\n\t\t\t\t};\n\t\t}\n\n\t\tif (hasCmd (stop.cmd, \"println\", true) === true)\n\t\t{\n\t\t\targs = getCmdArgs (stop.cmd);\n\n\t\t\tcmdFunc = async (cmdArgs: string[]): Promise<void> =>\n\t\t\t\t{\n\t\t\t\t\tlet input: string = cmdArgs[0];\n\n\t\t\t\t\tawait this.driver.println (input);\n\t\t\t\t};\n\t\t}\n\n\t\tif (hasCmd (stop.cmd, \"waitForTestObject\", true) === true)\n\t\t{\n\t\t\targs = getCmdArgs (stop.cmd);\n\n\t\t\tcmdFunc = async (cmdArgs: string[]): Promise<void> =>\n\t\t\t\t{\n\t\t\t\t\tlet testObject: string = JSON.parse (cmdArgs[0]);\n\n\t\t\t\t\tawait this.driver.waitForTestElement (testObject);\n\t\t\t\t};\n\t\t}\n\n\t\tif (cmdFunc == null)\n\t\t\tthrow new Error (`HotTester: Command ${stop.cmd} does not exist!`);\n\n\t\tawait this.onCommand (destination, page, stop, cmd, args, cmdFunc);\n\t}\n\n\t/**\n\t * Execute all test paths in a page.\n\t */\n\tasync executeTestPagePaths (destination: HotDestination, continueWhenTestIsComplete: boolean = false): Promise<any[]>\n\t{\n\t\tlet results: any[] = [];\n\t\tlet testMap: HotTestMap = this.testMaps[destination.mapName];\n\n\t\t/// @fixme For some reason the errors being thrown here are not being thrown.\n\t\tif (testMap == null)\n\t\t\tthrow new Error (`HotTester: Map ${destination.mapName} does not exist!`);\n\n\t\t// Iterate through each path in the destination until complete.\n\t\tfor (let iIdx = 0; iIdx < destination.paths.length; iIdx++)\n\t\t{\n\t\t\tlet stop: HotTestStop = destination.paths[iIdx];\n\t\t\tlet result: any = null;\n\t\t\tlet page: HotTestPage = testMap.pages[destination.page];\n\t\n\t\t\tif (page == null)\n\t\t\t\tthrow new Error (`HotTester: Page ${destination.page} does not exist!`);\n\n\t\t\tif (stop.dest !== \"\")\n\t\t\t{\n\t\t\t\tif (testMap.destinations instanceof Array)\n\t\t\t\t\tthrow new Error (`HotTester: When using type 'dest' in a destination string, all destinations in map ${destination.mapName} must be named.`);\n\n\t\t\t\tlet testDest: HotTestDestination = testMap.destinations[stop.dest];\n\t\t\t\tlet newDestination: HotDestination = HotTester.interpretDestination (\n\t\t\t\t\t\t\t\t\t\t\t\t\tdestination.mapName, testDest);\n\n\t\t\t\tif (newDestination != null)\n\t\t\t\t\tresult = await this.executeTestPagePaths (newDestination);\n\t\t\t}\n\n\t\t\tif (stop.cmd !== \"\")\n\t\t\t\tawait this.executeCommand (destination, page, stop, stop.cmd);\n\n\t\t\tif (stop.path !== \"\")\n\t\t\t\tresult = await this.executeTestPagePath (destination, stop, false, continueWhenTestIsComplete);\n\n\t\t\tresults.push (result);\n\t\t}\n\n\t\treturn (results);\n\t}\n\n\t/**\n\t * Execute the tests.\n\t */\n\tasync execute (mapName: string): Promise<void>\n\t{\n\t\tlet map: HotTestMap = this.testMaps[mapName];\n\n\t\tif (map == null)\n\t\t\tthrow new Error (`HotTester: Map ${mapName} does not exist!`);\n\n\t\t// Process routes testing first.\n\t\tlet routeKey: string = this.processor.getRouteKeyFromName (mapName);\n\t\tlet url: string = \"\";\n\n\t\tif (routeKey !== \"\")\n\t\t\turl = `${this.baseUrl}${routeKey}`;\n\n\t\tlet executeDestination: (testDest: HotTestDestination, destinationKey?: string) => Promise<void> = \n\t\t\tasync (testDest: HotTestDestination, destinationKey: string = \"\") =>\n\t\t\t{\n\t\t\t\tif (testDest.autoStart === false)\n\t\t\t\t\treturn;\n\n\t\t\t\tlet destination: HotDestination = HotTester.interpretDestination (mapName, testDest);\n\t\t\t\tlet isWebRoute: boolean = false;\n\t\t\t\tlet runTestPaths: boolean = true;\n\n\t\t\t\tif (destination.page !== \"\")\n\t\t\t\t\tisWebRoute = true;\n\t\n\t\t\t\tif (this.setup != null)\n\t\t\t\t{\n\t\t\t\t\tif (this.hasBeenSetup === false)\n\t\t\t\t\t{\n\t\t\t\t\t\tawait this.setup (isWebRoute, url, destinationKey);\n\t\t\t\t\t\tthis.hasBeenSetup = true;\n\t\t\t\t\t\tthis.hasBeenDestroyed = false;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\n\t\t\t\tif (this.onTestStart != null)\n\t\t\t\t\trunTestPaths = await this.onTestStart (destination, url, destinationKey);\n\t\n\t\t\t\tif (runTestPaths === true)\n\t\t\t\t{\n\t\t\t\t\tif (destination.page !== \"\")\n\t\t\t\t\t\tawait this.executeTestPagePaths (destination);\n\t\n\t\t\t\t\tif (destination.api !== \"\")\n\t\t\t\t\t\tawait this.executeTestAPIPaths (destination);\n\t\t\t\t}\n\t\t\n\t\t\t\tif (this.onTestEnd != null)\n\t\t\t\t\tawait this.onTestEnd (destination);\n\t\n\t\t\t\tif (this.destroy != null)\n\t\t\t\t{\n\t\t\t\t\tif (this.hasBeenDestroyed === false)\n\t\t\t\t\t{\n\t\t\t\t\t\tawait this.destroy ();\n\t\t\t\t\t\tthis.hasBeenDestroyed = true;\n\t\t\t\t\t\tthis.hasBeenSetup = false;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t};\n\n\t\t// If the map destinations are in an array, just execute those in order.\n\t\tif (map.destinations instanceof Array)\n\t\t{\n\t\t\tfor (let iIdx = 0; iIdx < map.destinations.length; iIdx++)\n\t\t\t{\n\t\t\t\tlet testDest: HotTestDestination = map.destinations[iIdx];\n\n\t\t\t\tawait executeDestination (testDest);\n\t\t\t}\n\t\t}\n\t\telse\n\t\t{\n\t\t\t// If there's a destination order, use that.\n\t\t\tif (map.destinationOrder.length > 0)\n\t\t\t{\n\t\t\t\tlet hasExecutedKeys: string[] = [];\n\n\t\t\t\t// Go through the destination order and execute each one.\n\t\t\t\tfor (let iIdx = 0; iIdx < map.destinationOrder.length; iIdx++)\n\t\t\t\t{\n\t\t\t\t\tlet orderKey: string = map.destinationOrder[iIdx];\n\t\t\t\t\tlet testDest: HotTestDestination = map.destinations[orderKey];\n\n\t\t\t\t\tif (testDest == null)\n\t\t\t\t\t\tthrow new Error (`HotTester: Destination ${orderKey} does not exist!`);\n\n\t\t\t\t\thasExecutedKeys.push (orderKey);\n\t\t\t\t\tawait executeDestination (testDest, orderKey);\n\t\t\t\t}\n\n\t\t\t\t// Execute the rest of the destinations that have not been executed yet.\n\t\t\t\tfor (let key in map.destinations)\n\t\t\t\t{\n\t\t\t\t\tlet executeDest: boolean = true;\n\n\t\t\t\t\tfor (let iIdx = 0; iIdx < hasExecutedKeys.length; iIdx++)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet executedKey: string = hasExecutedKeys[iIdx];\n\n\t\t\t\t\t\tif (executedKey === key)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\texecuteDest = false;\n\n\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tif (executeDest === true)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet testDest: HotTestDestination = map.destinations[key];\n\n\t\t\t\t\t\tawait executeDestination (testDest, key);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Execute the destinations in any order.\n\t\t\t\tfor (let key in map.destinations)\n\t\t\t\t{\n\t\t\t\t\tlet testDest: HotTestDestination = map.destinations[key];\n\n\t\t\t\t\tawait executeDestination (testDest, key);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\t// End of routes testing\n\n\t\t// Start of API testing\n\n\t}\n}","import { EventExecutionType, HotAPI } from \"./HotAPI\";\nimport { HotRoute } from \"./HotRoute\";\nimport { HotClient } from \"./HotClient\";\nimport { HotServer } from \"./HotServer\";\nimport { HotTestDriver } from \"./HotTestDriver\";\nimport { HotTester } from \"./HotTester\";\nimport { HotTestMap, HotTestPath, HotTestPage } from \"./HotTestMap\";\nimport { HTTPMethod } from \"./HotRouteMethod\";\n\nexport class HotTesterAPI extends HotAPI\n{\n\tconstructor (baseUrl: string, connection: HotServer | HotClient = null, db: any = null)\n\t{\n\t\tsuper(baseUrl, connection, db);\n\n\t\tthis.executeEventsUsing = EventExecutionType.HotAPI;\n\n\t\tlet route: HotRoute = new HotRoute (connection, \"tester\");\n\t\troute.addMethod ({\n\t\t\t\t\"name\": \"pageLoaded\", \n\t\t\t\t\"onServerExecute\": this.pageLoaded,\n\t\t\t\t\"parameters\": {\n\t\t\t\t\t\t\"testerName\": {\n\t\t\t\t\t\t\t\"required\": true,\n\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\"description\": \"The name of the tester executing the test.\"\n\t\t\t\t\t\t},\n\t\t\t\t\t\t\"testerMap\": {\n\t\t\t\t\t\t\t\"required\": true,\n\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\"description\": \"The tester map executing the test.\"\n\t\t\t\t\t\t},\n\t\t\t\t\t\t\"pageName\": {\n\t\t\t\t\t\t\t\"required\": true,\n\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\"description\": \"The name of the page executing the test.\"\n\t\t\t\t\t\t},\n\t\t\t\t\t\t\"testElements\": {\n\t\t\t\t\t\t\t\"required\": true,\n\t\t\t\t\t\t\t\"type\": \"array\",\n\t\t\t\t\t\t\t\"description\": \"The test elements on the page.\"\n\t\t\t\t\t\t},\n\t\t\t\t\t\t\"testPaths\": {\n\t\t\t\t\t\t\t\"required\": true,\n\t\t\t\t\t\t\t\"type\": \"array\",\n\t\t\t\t\t\t\t\"description\": \"The test paths on the page.\"\n\t\t\t\t\t\t}\n\t\t\t\t\t},\n\t\t\t\t\"returns\": \"Returns true as an acknowledgement.\"\n\t\t\t});\n\t\troute.addMethod ({\n\t\t\t\t\"name\": \"executeTests\",\n\t\t\t\t\"onServerExecute\": this.executeTests,\n\t\t\t\t\"parameters\": {\n\t\t\t\t\t\"testerName\": {\n\t\t\t\t\t\t\"required\": true,\n\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\"description\": \"The name of the tester executing the test.\"\n\t\t\t\t\t},\n\t\t\t\t\t\"testerMap\": {\n\t\t\t\t\t\t\"required\": true,\n\t\t\t\t\t\t\"type\": \"object\",\n\t\t\t\t\t\t\"description\": \"The tester map to execute.\"\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\t\"returns\": \"Returns true when tests are complete.\"\n\t\t\t});\n\t\troute.addMethod ({\n\t\t\t\t\"name\": \"heartbeat\",\n\t\t\t\t\"type\": HTTPMethod.GET,\n\t\t\t\t\"onServerExecute\": this.heartbeat,\n\t\t\t\t\"returns\": \"Returns true as an acknowledgement.\"\n\t\t\t});\n\t\tthis.addRoute (route);\n\t}\n\n\t/**\n\t * This is called when the page has finished loading in development mode.\n\t */\n\tasync pageLoaded (req: any, res: any, authorizedValue: any, jsonObj: any, queryObj: any): Promise<any>\n\t{\n\t\tlet testerObj: {\n\t\t\t\ttesterName: string;\n\t\t\t\ttesterMap: string;\n\t\t\t\tpageName: string;\n\t\t\t\ttestElements: any;\n\t\t\t\ttestPathsStrs: any;\n\t\t\t} = {\n\t\t\t\ttesterName: jsonObj[\"testerName\"],\n\t\t\t\ttesterMap: jsonObj[\"testerMap\"],\n\t\t\t\tpageName: jsonObj[\"pageName\"],\n\t\t\t\ttestElements: jsonObj[\"testElements\"],\n\t\t\t\ttestPathsStrs: jsonObj[\"testPaths\"]\n\t\t\t};\n\n\t\tfor (let key in testerObj)\n\t\t{\n\t\t\t// @ts-ignore\n\t\t\tlet testObj: any = testerObj[key];\n\t\t\tlet throwError: boolean = false;\n\n\t\t\tif (testObj == null)\n\t\t\t\tthrowError = true;\n\n\t\t\tif ((testerObj.testerName == \"\") || \n\t\t\t\t(testerObj.testerMap === \"\") || \n\t\t\t\t(testerObj.testElements === \"\") || \n\t\t\t\t(testerObj.testPathsStrs === \"\"))\n\t\t\t{\n\t\t\t\tthrowError = true;\n\t\t\t}\n\n\t\t\tif (throwError === true)\n\t\t\t\tthrow new Error (`TesterAPI: Object ${key} was not passed.`);\n\t\t}\n\n\t\ttesterObj.testElements = JSON.parse (testerObj.testElements);\n\t\ttesterObj.testPathsStrs = JSON.parse (testerObj.testPathsStrs);\n\n\t\tlet testPaths: { [name: string]: HotTestPath; } = {};\n\n\t\tfor (let key in testerObj.testPathsStrs)\n\t\t{\n\t\t\tlet testPath: (driver: HotTestDriver, ...args: any) => Promise<any> = \n\t\t\t\teval (testerObj.testPathsStrs[key]);\n\n\t\t\ttestPaths[key] = testPath;\n\t\t}\n\n\t\tlet tester: HotTester = this.connection.processor.testers[testerObj.testerName];\n\n\t\tif (tester == null)\n\t\t\tthrow new Error (`TesterAPI: Tester ${testerObj.testerMap} does not exist!`);\n\n\t\tlet testMap: HotTestMap = tester.testMaps[testerObj.testerMap];\n\n\t\tif (testMap == null)\n\t\t\tthrow new Error (`TesterAPI: Tester map ${testerObj.testerMap} does not exist!`);\n\n\t\ttestMap.pages[testerObj.pageName] = {\n\t\t\t\t\"testElements\": {},\n\t\t\t\t\"testPaths\": {}\n\t\t\t};\n\t\ttestMap.pages[testerObj.pageName].testElements = testerObj.testElements;\n\t\ttestMap.pages[testerObj.pageName].testPaths = testPaths;\n\n\t\ttester.finishedLoading = true;\n\n\t\tif (tester.onFinishedLoading != null)\n\t\t\tawait tester.onFinishedLoading ();\n\n\t\treturn (true);\n\t}\n\n\t/**\n\t * Execute the tests for a page.\n\t */\n\tasync executeTests (req: any, res: any, authorizedValue: any, jsonObj: any, queryObj: any): Promise<any>\n\t{\n\t\tlet testerName: string = jsonObj[\"testerName\"];\n\t\tlet testerMap: string = jsonObj[\"testerMap\"];\n\n\t\tif ((testerName == null) || (testerMap == null))\n\t\t\tthrow new Error (\"TesterAPI: Not all required json objects were passed.\");\n\n\t\tif ((testerName === \"\") || (testerMap === \"\"))\n\t\t\tthrow new Error (\"TesterAPI: Not all required json objects were passed.\");\n\n\t\tlet server: HotServer = (<HotServer>this.connection);\n\n\t\t// @ts-ignore\n\t\tif (server.executeTests != null)\n\t\t{\n\t\t\t// @ts-ignore\n\t\t\tawait server.executeTests (testerName, testerMap);\n\t\t}\n\n\t\treturn (true);\n\t}\n\n\t/**\n\t * Responds with true to heartbeat requests.\n\t */\n\tasync heartbeat (req: any, res: any, authorizedValue: any, jsonObj: any, queryObj: any): Promise<any>\n\t{\n\t\treturn (true);\n\t}\n}","import { HotStaq, HotStartOptions, IHotStaq, \n\tHotSite, HotSiteRoute, HotSiteMapPath } from \"./HotStaq\";\nimport { Hot, DeveloperMode } from \"./Hot\";\nimport { HotComponent, IHotComponent } from \"./HotComponent\";\nimport { HotFile } from \"./HotFile\";\nimport { HotLog, HotLogLevel } from \"./HotLog\";\nimport { HotPage } from \"./HotPage\";\n\n// Server stuff\nimport { HotAPI, EventExecutionType, APItoLoad } from \"./HotAPI\";\nimport { HotRoute } from \"./HotRoute\";\nimport { HotRouteMethod, HTTPMethod, ServerAuthorizationFunction, ServerExecutionFunction } from \"./HotRouteMethod\";\nimport { HotServer, HotServerType } from \"./HotServer\";\nimport { HotClient } from \"./HotClient\";\n\n// Testing stuff\nimport { HotTestDriver } from \"./HotTestDriver\";\nimport { IHotTestElement, HotTestElement, HotTestElementOptions, IHotTestElementOptions } from \"./HotTestElement\";\nimport { HotTester, HotTestStop, HotDestination } from \"./HotTester\";\nimport { HotTesterAPI } from \"./HotTesterAPI\";\nimport { HotTestMap, HotTestDestination, HotTestPath, HotTestPage } from \"./HotTestMap\";\n\nHotStaq.isWeb = true;\n\n// Can't export interfaces from here :(\n\nmodule.exports[\"HotStaq\"] = HotStaq;\nmodule.exports[\"Hot\"] = Hot;\nmodule.exports[\"DeveloperMode\"] = DeveloperMode;\nmodule.exports[\"HotComponent\"] = HotComponent;\nmodule.exports[\"HotAPI\"] = HotAPI;\nmodule.exports[\"EventExecutionType\"] = EventExecutionType;\nmodule.exports[\"HotFile\"] = HotFile;\nmodule.exports[\"HotLog\"] = HotLog;\nmodule.exports[\"HotLogLevel\"] = HotLogLevel;\nmodule.exports[\"HotPage\"] = HotPage;\nmodule.exports[\"HotRoute\"] = HotRoute;\nmodule.exports[\"HotRouteMethod\"] = HotRouteMethod;\nmodule.exports[\"HTTPMethod\"] = HTTPMethod;\nmodule.exports[\"HotServer\"] = HotServer;\nmodule.exports[\"HotServerType\"] = HotServerType;\nmodule.exports[\"HotClient\"] = HotClient;\nmodule.exports[\"HotTester\"] = HotTester;\nmodule.exports[\"HotTesterAPI\"] = HotTesterAPI;\nmodule.exports[\"HotTestMap\"] = HotTestMap;\nmodule.exports[\"HotTestDestination\"] = HotTestDestination;\nmodule.exports[\"HotTestElement\"] = HotTestElement;\nmodule.exports[\"HotTestElementOptions\"] = HotTestElementOptions;\nmodule.exports[\"HotTestDriver\"] = HotTestDriver;","module.exports = {};","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","// startup\n// Load entry module and return exports\n// This entry module used 'module' so it can't be inlined\nvar __webpack_exports__ = __webpack_require__(859);\n"],"names":["module","exports","self","FormData","window","assign","target","i","arguments","length","source","key","init","converter","defaultAttributes","set","value","attributes","document","expires","Date","now","toUTCString","encodeURIComponent","replace","decodeURIComponent","escape","stringifiedAttributes","attributeName","split","cookie","write","Object","create","get","cookies","jar","parts","slice","join","foundKey","read","e","remove","withAttributes","this","withConverter","freeze","path","factory","global","Error","getGlobal","fetch","bind","Headers","Request","Response","DeveloperMode","Hot","static","file","args","HotStaq","isWeb","lowerFile","toLowerCase","indexOf","echo","getFile","res","httpRequest","output","text","eval","apply","fileName","tempFile","CurrentPage","processor","page","content","process","HotFile","url","localFile","load","route","data","httpMethod","files","result","api","makeCall","fetchObj","JSON","stringify","ok","status","statusText","json","ex","message","requestInit","Output","iIdx","CSS","cssFile","cssOut","cssStr","JSFiles","jsFile","jsFileOut","jsFileStr","JSScripts","jsScript","jsScriptOut","jsScriptsStr","Arguments","HotTestElement","Mode","Production","API","TesterAPI","Data","Cookies","PublicKeys","EventExecutionType","constructor","baseUrl","connection","db","description","createFunctions","executeEventsUsing","HotRoute","authCredentials","userAuth","routes","onPreRegister","onPostRegister","setDBSchema","schema","getDB","getDBSchema","addRoute","routeMethod","executeFunction","routeName","HotRouteMethod","addMethod","newRoute","methods","currentRoute","newRouteMethod","name","type","routeStr","version","authCredential","registerRoute","HotServer","registerRoutes","toUpperCase","substr","keys","formData","append","method","body","jsonRes","headers","Promise","resolve","reject","then","jsonObj","catch","reason","testerAPI","HotServerType","HTTP","logger","copy","htmlElements","tag","elementOptions","undefined","observedAttributes","inner","events","onCreated","element","throwAllErrors","setContent","getContent","loadUrl","httpGet","loadLocalFile","fs","readFile","err","toString","contentRegex","contentProcessor","offContentProcessor","numRemoveFromBeginning","numRemoveFromEnd","exec","previousIndex","start","index","end","lastIndex","prevContent","startChars","endChars","triggerChar","pos","startTriggerPos","nestedCounter","rpos","lastIndexOf","epos","tempepos","thisContent","parserOptions","outputCommands","allowStringify","STRINGIFY_START","STRINGIFY_END","processContent","RegExp","regexFound","offContent","tempOutput","processNestedContent","regexFound2","offContent3","tempOutput2","out","tempOutput3","tempOutput4","Development","foundStr","parse","createTestElement","foundStr2","testElm","obj","Array","testElements","func","tempOutput5","escapedContent","mode","publicKeys","getAPI","getTesterAPI","parseContent","returnedOutput","executionContent","newVar","newVarValue","contentName","Function","SyntaxError","hot","finalOutput","HotLogLevel","logLevel","All","log","level","Verbose","error","Warning","warning","Info","info","verbose","console","warn","msg","stack","testerName","testerMap","testPaths","addFile","push","addTestElement","elm","getTestElement","createTestPath","pathName","driverFunc","onRegister","onAuthorizeUser","prefix","errors","createError","HTTPMethod","POST","testCases","getMethod","foundMethod","onExecute","onServerAuthorize","returns","parameters","param","onServerExecute","onClientExecute","isRegistered","executeSetup","testCaseName","addTestCase","newTestCase","testCaseFunction","testCaseId","testCase","serverType","listenAddress","ports","http","https","ssl","cert","ca","redirectHTTPtoHTTPS","secrets","setAPI","HotTesterMocha","HotTesterMochaSelenium","HotTestSeleniumDriver","pages","components","hotSite","apiContent","testerApiContent","pageContent","HotLog","None","testers","parseInt","objWithParam","required","throwException","defaultValue","errType","redirectToUrl","location","href","numMilliseconds","setTimeout","addPage","getPage","pageName","context","val","aryArgs","prototype","call","addComponent","ComponentType","tempApi","tempComponentObj","componentType","registerComponent","str","tempStr","parsedXML","DOMParser","parseFromString","querySelector","documentElement","children","tagName","fixedStr","customElements","processorComponents","define","HTMLElement","super","componentInfo","component","looseParseFromString","parser","xdom","hdom","elem","from","appendChild","querySelectorAll","outerHTML","connectedCallback","hotComponent","innerHTML","handleAttributes","attr","attrName","attrValue","attrTempName","substring","onPreOutput","outputs","onPostOutput","componentOutputs","html","iKdx","htmlStr","addFunctionsTo","onParsed","htmlHandler","onFixHTML","fixHTML","newDOM","newObj","onParseDOM","throwErr","child","childrenToReadd","removeChild","replaceWith","click","onclick","event","addEventListener","options","objectFunctions","getOwnPropertyNames","objFunc","isNewFunction","key2","HotComponent","keepContext","onPrePlace","compHtmlElement2","onParentPlace","parentSelector","parentNode","parentElement","placeHereParent","parentNodeToCheck","parentNodeCheckCounter","HTMLHtmlElement","placeHereArray","placeHere","placeHereAttrArray","onPostPlace","parent","foundParent","results","hotsiteName","throwTheException","hotsite","params","prevValue","processHotSite","checkHotSiteName","testerUrl","tester","driver","testing","setupTester","parentObj","createNewTester","testerAPIUrl","commandDelay","web","HotPage","mapName","testMap","map","testMaps","HotTestMap","destinations","dest","HotTestDestination","destinationOrder","apis","componentUrl","ComponentClass","disableFileLoading","loadHotFiles","addTester","loadHotSite","jsonStr","ppath","normalize","hotsitePath","forceContentLoading","newFile","loadContent","filepath","generateContent","routeKey","jsSrcPath","passArgs","apiScripts","apiCode","server","globalApi","sendJSContent","jsapi","libraryName","apiName","tempAPIContent","jsapipath","secret","serverConn","passSecretFromAPI","env","envKey","tempContent","developerModeStr","testerAPIStr","loadFiles","fileUrl","fileContent","testerLaunchpadUrl","launchpadUrl","fixContent","createExpressRoutes","expressApp","req","send","getWebTestingMaps","maps","getAPITestingMaps","getRouteKeyFromName","getRouteFromName","foundRoute","executeTests","execute","executeAllWebTests","executeAllAPITests","localFilepath","readyFunc","readyState","getElementsByTagName","tmpScripts","scripts","s","createElement","resolve2","reject2","onload","hasSrc","getAttribute","setAttribute","isReadyForTesting","wait","onReadyForTesting","testerAPIBaseUrl","client","HotClient","HotTesterAPI","onReady","setupTesters","processUrl","setupClientTesters","useOutput","loadHotStaqSite","hotstaqElms","HotStaqWeb","HotAPI","hotstaqElm","getAttr","attrNames","loadPage","router","apiLibrary","apiUrl","testerApiBaseUrl","dontReuseProcessor","passRawUrl","htmlSource","routerManager","routerWildcards","hotstaqErrors","hotstaqErrorElm","errorStatus","unsupportedBrowser","executeError","hotstaqRouterElms","hotstaqRouterElm","routerName","serveLocally","iJdx","childNodes","routerElm","routerPath","redirect","base","routerSrc","src","tempPathname","pathname","lowerServeLocally","lastSlashPos","routeWildcard","tempRouteWildcard","hasHtmlSource","htmlSourceCheck","tempMode","parentLib","newAPI","displayUrl","displayContent","ethereum22","ethereum","persistentData","parseTestObject","wildcard","selector","print","stdout","println","assert","errorMessage","run","executions","execution","runCommand","mustBeVisible","ignoreMissingElementError","destination","autoStart","HotTester","finishedLoading","hasBeenSetup","hasBeenDestroyed","waitForData","getTestPage","getTestPath","testDest","newDestination","paths","strs","getType","typeStr","typeDelimiter","typeValue","trim","newPathStr","newPath","cmd","executeTestAPIPath","testName","skipEventCalls","continueWhenTestIsComplete","runTestPath","onTestAPIPathStart","testCaseObject","onTestAPIPathEnd","executeTestAPIPaths","executeTestPagePath","stop","testPathName","testPath","onTestPagePathStart","onTestPagePathEnd","executeCommand","hasCmd","input","hasArguments","getCmdArgs","matches","match","tempMatch","cmdFunc","cmdArgs","navigateToUrl","testObject","waitForTestElement","onCommand","executeTestPagePaths","interpretDestination","executeDestination","destinationKey","isWebRoute","runTestPaths","setup","onTestStart","onTestEnd","destroy","hasExecutedKeys","orderKey","executeDest","pageLoaded","GET","heartbeat","authorizedValue","queryObj","testerObj","testPathsStrs","testObj","throwError","onFinishedLoading","HotTestElementOptions","HotTestDriver","__webpack_module_cache__","__webpack_require__","moduleId","cachedModule","__webpack_modules__","__webpack_exports__"],"sourceRoot":""}
1
+ var HotStaqWeb;(()=>{var __webpack_modules__={230:t=>{t.exports="object"==typeof self?self.FormData:window.FormData},646:function(t){t.exports=function(){"use strict";function t(t){for(var e=1;e<arguments.length;e++){var n=arguments[e];for(var r in n)t[r]=n[r]}return t}return function e(n,r){function o(e,o,s){if("undefined"!=typeof document){"number"==typeof(s=t({},r,s)).expires&&(s.expires=new Date(Date.now()+864e5*s.expires)),s.expires&&(s.expires=s.expires.toUTCString()),e=encodeURIComponent(e).replace(/%(2[346B]|5E|60|7C)/g,decodeURIComponent).replace(/[()]/g,escape);var i="";for(var l in s)s[l]&&(i+="; "+l,!0!==s[l]&&(i+="="+s[l].split(";")[0]));return document.cookie=e+"="+n.write(o,e)+i}}return Object.create({set:o,get:function(t){if("undefined"!=typeof document&&(!arguments.length||t)){for(var e=document.cookie?document.cookie.split("; "):[],r={},o=0;o<e.length;o++){var s=e[o].split("="),i=s.slice(1).join("=");try{var l=decodeURIComponent(s[0]);if(r[l]=n.read(i,l),t===l)break}catch(t){}}return t?r[t]:r}},remove:function(e,n){o(e,"",t({},n,{expires:-1}))},withAttributes:function(n){return e(this.converter,t({},this.attributes,n))},withConverter:function(n){return e(t({},this.converter,n),this.attributes)}},{attributes:{value:Object.freeze(r)},converter:{value:Object.freeze(n)}})}({read:function(t){return'"'===t[0]&&(t=t.slice(1,-1)),t.replace(/(%[\dA-F]{2})+/gi,decodeURIComponent)},write:function(t){return encodeURIComponent(t).replace(/%(2[346BF]|3[AC-F]|40|5[BDE]|60|7[BCD])/g,decodeURIComponent)}},{path:"/"})}()},300:(t,e)=>{"use strict";var n=function(){if("undefined"!=typeof self)return self;if("undefined"!=typeof window)return window;if(void 0!==n)return n;throw new Error("unable to locate global object")}();t.exports=e=n.fetch,n.fetch&&(e.default=n.fetch.bind(n)),e.Headers=n.Headers,e.Request=n.Request,e.Response=n.Response},507:function(__unused_webpack_module,exports,__webpack_require__){"use strict";var __awaiter=this&&this.__awaiter||function(t,e,n,r){return new(n||(n=Promise))((function(o,s){function i(t){try{a(r.next(t))}catch(t){s(t)}}function l(t){try{a(r.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?o(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((r=r.apply(t,e||[])).next())}))},__importDefault=this&&this.__importDefault||function(t){return t&&t.__esModule?t:{default:t}};Object.defineProperty(exports,"__esModule",{value:!0}),exports.Hot=exports.DeveloperMode=void 0;const HotFile_1=__webpack_require__(732),HotStaq_1=__webpack_require__(243),HotTestElement_1=__webpack_require__(967),js_cookie_1=__importDefault(__webpack_require__(646)),node_fetch_1=__importDefault(__webpack_require__(300));var DeveloperMode;!function(t){t[t.Production=0]="Production",t[t.Development=1]="Development"}(DeveloperMode=exports.DeveloperMode||(exports.DeveloperMode={}));class Hot{static include(t,e=null){return __awaiter(this,void 0,void 0,(function*(){if(!0===HotStaq_1.HotStaq.isWeb&&"string"==typeof t){const e=t.toLowerCase();e.indexOf(".hott")>-1&&e.indexOf("nahfam")<0&&(t+="?hstqserve=nahfam")}Hot.echo(yield Hot.getFile(t,e))}))}static includeJS(file){return __awaiter(this,void 0,void 0,(function*(){const res=yield Hot.httpRequest(file),output=yield res.text();!0===HotStaq_1.HotStaq.isWeb?eval.apply(window,[output]):eval(output)}))}static runFile(t,e=null){return __awaiter(this,void 0,void 0,(function*(){let n=Hot.CurrentPage.processor.getFile(t);n.page=this.CurrentPage;let r=yield n.process(e);Hot.echo(r)}))}static getFile(t,e=null){return __awaiter(this,void 0,void 0,(function*(){let n=null;return"string"==typeof t?(n=new HotFile_1.HotFile,!0===HotStaq_1.HotStaq.isWeb?n.url=t:n.localFile=t):n=t,yield n.load(),n.page=this.CurrentPage,yield n.process(e)}))}static apiCall(t,e=null,n="POST",r={}){return __awaiter(this,void 0,void 0,(function*(){let o=null;if(null==Hot.CurrentPage)throw new Error("Current page is null!");if(null==Hot.CurrentPage.processor)throw new Error("Current page's processor is null!");if(null==Hot.CurrentPage.processor.api)throw new Error("Current page's processor api is null! Did you forget to set the API name or URL?");return null!=Hot.CurrentPage.processor.api&&(o=yield Hot.CurrentPage.processor.api.makeCall(t,e,n,r)),o}))}static jsonRequest(t,e=null,n="POST"){return __awaiter(this,void 0,void 0,(function*(){try{let r={method:n,headers:{Accept:"application/json","Content-Type":"application/json"}};"POST"===n&&(r.body=JSON.stringify(e));let o=yield(0,node_fetch_1.default)(t,r);if(!1===o.ok)throw new Error(`${o.status}: ${o.statusText}`);return yield o.json()}catch(e){return JSON.stringify({error:`${e.message} - Could not fetch ${t}`})}}))}static httpRequest(t,e){return __awaiter(this,void 0,void 0,(function*(){return yield(0,node_fetch_1.default)(t,e)}))}static echo(t){Hot.Output+=t}static displayCSS(){for(let t=0;t<Hot.CSS.length;t++){let e=Hot.CSS[t],n=Hot.cssStr;n=n.replace(/\%CSS_FILE\%/g,e),Hot.echo(n)}}static displayJSFiles(){for(let t=0;t<Hot.JSFiles.length;t++){let e=Hot.JSFiles[t],n=Hot.jsFileStr;n=n.replace(/\%JS_FILE\%/g,e),Hot.echo(n)}}static displayJSScripts(){for(let t=0;t<Hot.JSScripts.length;t++){let e=Hot.JSScripts[t],n=Hot.jsScriptsStr;n=n.replace(/\%JS_CODE\%/g,e),Hot.echo(n)}}}exports.Hot=Hot,Hot.CurrentPage=null,Hot.Arguments=null,Hot.DeveloperMode=DeveloperMode,Hot.HotTestElement=HotTestElement_1.HotTestElement,Hot.Mode=DeveloperMode.Production,Hot.API=null,Hot.TesterAPI=null,Hot.Output="",Hot.Data={},Hot.Cookies=js_cookie_1.default,Hot.PublicKeys={},Hot.cssStr='<link rel = "stylesheet" href = "%CSS_FILE%" />',Hot.CSS=[],Hot.JSFiles=[],Hot.JSScripts=[],Hot.jsFileStr='<script type = "text/javascript" src = "%JS_FILE%"><\/script>',Hot.jsScriptsStr='<script type = "text/javascript">%JS_CODE%<\/script>'},233:function(t,e,n){"use strict";var r=this&&this.__awaiter||function(t,e,n,r){return new(n||(n=Promise))((function(o,s){function i(t){try{a(r.next(t))}catch(t){s(t)}}function l(t){try{a(r.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?o(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((r=r.apply(t,e||[])).next())}))},o=this&&this.__importDefault||function(t){return t&&t.__esModule?t:{default:t}};Object.defineProperty(e,"__esModule",{value:!0}),e.HotAPI=e.EventExecutionType=void 0;const s=o(n(300)),i=o(n(230)),l=n(810),a=n(761),u=n(70);var h;!function(t){t[t.HotRoute=0]="HotRoute",t[t.HotMethod=1]="HotMethod",t[t.HotAPI=2]="HotAPI"}(h=e.EventExecutionType||(e.EventExecutionType={})),e.HotAPI=class{constructor(t,e=null,n=null){this.connection=e,this.description="",this.baseUrl=t,this.createFunctions=!0,this.executeEventsUsing=h.HotRoute,this.db=n,this.authCredentials=null,this.userAuth=null,this.routes={},this.onPreRegister=null,this.onPostRegister=null}setDBSchema(t){if(null==this.connection.api)throw new Error("No API has been set!");if(null==this.connection.api.db)throw new Error(`No database has been set for API base url ${this.connection.api.baseUrl}`);this.connection.api.db.schema=t}getDB(){if(null==this.connection.api.db)throw new Error(`No database has been set for API base url ${this.connection.api.baseUrl}`);return this.connection.api.db}getDBSchema(){if(null==this.connection.api.db)throw new Error(`No database has been set for API base url ${this.connection.api.baseUrl}`);return this.connection.api.db.schema}addRoute(t,e=null,n=null){let r="";if(t instanceof a.HotRoute?(r=t.route,this.routes[t.route]=t):(r=t,null==this.routes[r]&&(this.routes[r]=new a.HotRoute(this.connection,r)),e instanceof u.HotRouteMethod?this.routes[r].addMethod(e):this.routes[r].addMethod(new u.HotRouteMethod(this.routes[r],e,n))),this.routes[r].connection=this.connection,!0===this.createFunctions){let t=this[r];null==t&&(t={});for(let e=0;e<this.routes[r].methods.length;e++){let n=this.routes[r],o=this.routes[r].methods[e];t[o.name]=(t,e)=>{let r=o.type,s="";""!==n.version&&(s+=`/${n.version}`),""!==n.route&&(s+=`/${n.route}`),""!==o.name&&(s+=`/${o.name}`);let i=null;if(null!=this.authCredentials&&(i=this.authCredentials),null!=o.authCredentials?i=o.authCredentials:null!=o.route.authCredentials&&(i=o.route.authCredentials),null==i&&"undefined"!=typeof Hot&&null!=Hot&&null!=Hot.API&&null!=Hot.API[n.route]&&null!=Hot.API[n.route].authCredentials&&(i=Hot.API[n.route].authCredentials),null!=i)for(let e in i){let n=i[e];null==t[e]&&(t[e]=n)}let l=[s,t,r,e];return this.makeCall.apply(this,l)}}this[r]=t}}registerRoute(t){return r(this,void 0,void 0,(function*(){this.connection instanceof l.HotServer&&(yield this.connection.registerRoute(t))}))}registerRoutes(){return r(this,void 0,void 0,(function*(){for(let t in this.routes){let e=this.routes[t];yield this.registerRoute(e)}}))}makeCall(t,e,n="POST",o={}){return r(this,void 0,void 0,(function*(){let l=this.baseUrl;if(n=n.toUpperCase(),"/"===l[l.length-1]&&(l=l.substr(0,l.length-1)),"/"!==t[0]&&(l+="/"),l+=t,Object.keys(o).length>0){if("POST"!==n)throw new Error("To upload files, you must set the httpMethod to POST.");const r=new i.default;for(let t in o)r.append(t,o[t]);let a=yield(0,s.default)(l,{method:"POST",body:r}),u=yield a.json();return null==e.hotstaq&&(e.hotstaq={}),null==e.hotstaq.uploads&&(e.hotstaq.uploads={}),e.hotstaq.uploads.uploadId=u.hotstaq.uploads.uploadId,yield this.makeCall(t,e,n)}let a={method:n,headers:{Accept:"application/json","Content-Type":"application/json"}};return"GET"!==n&&"HEAD"!==n&&(a.body=JSON.stringify(e)),new Promise(((t,e)=>{(0,s.default)(l,a).then((e=>r(this,void 0,void 0,(function*(){let n=yield e.json();t(n)})))).catch((t=>{throw new Error(`${l}: ${t.message}`)}))}))}))}}},957:(t,e,n)=>{"use strict";Object.defineProperty(e,"__esModule",{value:!0}),e.HotClient=void 0;const r=n(810);e.HotClient=class{constructor(t){this.processor=t,this.api=null,this.testerAPI=null,this.type=r.HotServerType.HTTP,this.logger=t.logger}}},260:function(t,e,n){"use strict";var r=this&&this.__awaiter||function(t,e,n,r){return new(n||(n=Promise))((function(o,s){function i(t){try{a(r.next(t))}catch(t){s(t)}}function l(t){try{a(r.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?o(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((r=r.apply(t,e||[])).next())}))};Object.defineProperty(e,"__esModule",{value:!0}),e.HotComponent=void 0;const o=n(243);e.HotComponent=class{constructor(t,e=null){t instanceof o.HotStaq||null==t?(this.processor=t,this.htmlElements=[],this.name="",this.tag="",this.api=null,this.elementOptions=void 0,this.observedAttributes=[],this.type="",this.value=null,this.inner=null,this.events={}):(this.processor=t.processor,this.htmlElements=t.htmlElements||[],this.name=t.name||"",this.tag=t.tag||this.name,this.api=t.api||null,this.elementOptions=t.elementOptions||void 0,this.observedAttributes=t.observedAttributes||[],this.type=t.type||"",this.value=t.value||null,this.inner=t.inner||null,this.events={}),null!=e&&(this.api=e)}onCreated(t){return r(this,void 0,void 0,(function*(){return t}))}}},732:function(t,e,n){"use strict";var r=this&&this.__createBinding||(Object.create?function(t,e,n,r){void 0===r&&(r=n),Object.defineProperty(t,r,{enumerable:!0,get:function(){return e[n]}})}:function(t,e,n,r){void 0===r&&(r=n),t[r]=e[n]}),o=this&&this.__setModuleDefault||(Object.create?function(t,e){Object.defineProperty(t,"default",{enumerable:!0,value:e})}:function(t,e){t.default=e}),s=this&&this.__importStar||function(t){if(t&&t.__esModule)return t;var e={};if(null!=t)for(var n in t)"default"!==n&&Object.prototype.hasOwnProperty.call(t,n)&&r(e,t,n);return o(e,t),e},i=this&&this.__awaiter||function(t,e,n,r){return new(n||(n=Promise))((function(o,s){function i(t){try{a(r.next(t))}catch(t){s(t)}}function l(t){try{a(r.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?o(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((r=r.apply(t,e||[])).next())}))},l=this&&this.__importDefault||function(t){return t&&t.__esModule?t:{default:t}};Object.defineProperty(e,"__esModule",{value:!0}),e.HotFile=void 0;const a=s(n(351)),u=l(n(300)),h=n(507);class c{constructor(t={}){this.page=t.page||null,this.name=t.name||"",this.url=t.url||"",this.localFile=t.localFile||"",this.content=t.content||"",this.throwAllErrors=t.throwAllErrors||!1}setContent(t){this.content=t}getContent(){return this.content}static httpGet(t){return i(this,void 0,void 0,(function*(){try{let e=yield(0,u.default)(t);if(!1===e.ok)throw new Error(`${e.status}: ${e.statusText}`);return yield e.text()}catch(e){return JSON.stringify({error:`${e.message} - Could not fetch ${t}`})}}))}loadUrl(){return i(this,void 0,void 0,(function*(){return this.content=yield c.httpGet(this.url),this.content}))}loadLocalFile(){return i(this,void 0,void 0,(function*(){return new Promise(((t,e)=>{a.readFile(this.localFile,((e,n)=>{if(null!=e)throw e;let r=n.toString();this.content=r,t(this.content)}))}))}))}load(){return i(this,void 0,void 0,(function*(){let t="";return""!==this.url&&(t=yield this.loadUrl()),""!==this.localFile&&(t=yield this.loadLocalFile()),t}))}static processContent(t,e,n,r,o=2,s=2){let i=e.exec(t),l=0,a="";for(;null!=i;){let u=i.index-o,h=e.lastIndex+s,c=t.substr(l,u-l);l=h,a+=r(c),a+=n(i[0]),i=e.exec(t)}return a+=r(t.substr(l)),a}static processNestedContent(t,e,n,r,o,s,i=2,l=1){let a=t.indexOf(e),u=0,h=t.indexOf(r,a),c="";for(;a>-1;){let p=t.indexOf(n,a),d=0;if(""!==r){let e=t.lastIndexOf(r,p-l);for(;e>-1&&e!==h;)e=t.lastIndexOf(r,e-l),d++}if(d>0){let r=t.indexOf(n,p+l),o=r;for(;r>-1&&d>0&&!(o<0)&&!(t.lastIndexOf(e,o-l)>r);)r=o,o=t.indexOf(n,r+l),d--;p=r}c+=s(t.substr(u,a-u)),c+=o(t.substr(a+i,p-(a+i))),a=t.indexOf(e,p+l),h=t.indexOf(r,a),u=p+l}return c+=s(t.substr(u)),c}static parseContent(t,e,n=null){null==n&&(n={outputCommands:!0,allowStringify:!0});let r="JSON.stringify (",o=")";return!1===n.allowStringify&&(r="",o=""),c.processContent(t,new RegExp("(?=\\<\\*)([\\s\\S]*?)(?=\\*\\>)","g"),(t=>`${t=t.substr(2)}`),(t=>{if(""===t)return"";let s=c.processNestedContent(t,"!{","}","{",(t=>`*&&%*%@#@!${t}*&!#%@!@*!`),(t=>t)),i=c.processNestedContent(s,"STR{","}","{",(t=>{let s="";return s=!0===n.outputCommands?`*&&%*%@#@!echoOutput (JSON.stringify(${t}), ${e});*&!#%@!@*!`:`*&&%*%@#@!${r}${t}${o}, ${e});*&!#%@!@*!`,s}),(t=>t),4,1),l=c.processNestedContent(i,"${","}","{",(t=>{let r="";return!0===n.outputCommands?(r=`*&&%*%@#@!try { Hot.echo (${t}); }catch (ex){Hot.echo ("");}*&!#%@!@*!`,!0===e&&(r=`*&&%*%@#@!Hot.echo (${t});*&!#%@!@*!`)):r=`*&&%*%@#@!${t}*&!#%@!@*!`,r}),(t=>t)),a="";h.Hot.Mode===h.DeveloperMode.Production&&(a=c.processNestedContent(l,"?(",")","(",(t=>""),(t=>t))),h.Hot.Mode===h.DeveloperMode.Development&&(a=c.processNestedContent(l,"?(",")","(",(t=>{let e="";try{JSON.parse(t),e=!0===n.allowStringify?JSON.stringify(t):`${t}`}catch(n){e=`${t}`}let r="";if(!0===n.outputCommands)r=`*&&%*%@#@!{\nconst testElm = createTestElement (${e});\nHot.echo (\`data-test-object-name = "\${testElm.name}" data-test-object-func = "\${testElm.func}" data-test-object-value = "\${testElm.value}"\`);\n}*&!#%@!@*!\n`;else{let t=t=>{let e=null;try{let n=t;if("string"==typeof t&&(n=JSON.parse(t)),"string"==typeof n&&(e=new h.Hot.HotTestElement(n)),n instanceof Array&&(e=new h.Hot.HotTestElement(n[0],n[1],n[2])),null!=n.name&&(e=new h.Hot.HotTestElement(n)),null!=h.Hot.CurrentPage.testElements[e.name])throw new Error(`Test element ${e.name} already exists!`)}catch(e){throw new Error(`Error processing test element ${t} in ${h.Hot.CurrentPage.name}. Error: ${e.message}`)}return e};const n=t(e);r=`*&&%*%@#@!data-test-object-name = "${n.name}" data-test-object-func = "${n.func}" data-test-object-value = "${n.value}"*&!#%@!@*!`}return r}),(t=>t)));let u=c.processNestedContent(a,"*&&%*%@#@!","*&!#%@!@*!","*&&%*%@#@!",(t=>t),(t=>{let r="";r=!0===n.allowStringify?JSON.stringify(t):t;let o="";return o=!0===n.outputCommands?`echoOutput (${r}, ${e});\n`:r,o}),"*&&%*%@#@!".length,"*&!#%@!@*!".length);return u=u.replace(/\*\&\&\%\*\%\@\#\@\!/g,""),u=u.replace(/\*\&\!\#\%\@\!\@\*\!/g,""),u}),0)}process(t=null){return i(this,void 0,void 0,(function*(){let e=this.content;h.Hot.Mode=this.page.processor.mode,h.Hot.Arguments=t,h.Hot.CurrentPage=this.page,h.Hot.PublicKeys=this.page.processor.publicKeys,h.Hot.API=this.page.getAPI(),h.Hot.TesterAPI=this.page.getTesterAPI();let n=c.parseContent(e,this.throwAllErrors),r=null;try{let e="\n\t\t\tvar Hot = arguments[0];\n\t\t\tvar PassedHotFile = arguments[1];\n\n\t\t\t";if("string"==typeof t)throw new Error("The passing arguments cannot be a string!");for(let n in t){let r="",o=t[n];r=`var ${n} = ${JSON.stringify(o)};\n`,e+=r}let o=this.name;""===o&&(o=this.localFile),""===o&&(o=this.url),e+='\n\n\t\t\tfunction echoOutput (content, throwErrors)\n\t\t\t{\n\t\t\t\tif (throwErrors == null)\n\t\t\t\t\tthrowErrors = true;\n\n\t\t\t\tif (throwErrors === true)\n\t\t\t\t{\n\t\t\t\t\tHot.echo (content);\n\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\ttry\n\t\t\t\t{\n\t\t\t\t\tHot.echo (content);\n\t\t\t\t}\n\t\t\t\tcatch (ex)\n\t\t\t\t{\n\t\t\t\t\tHot.echo ("");\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tfunction createTestElement (foundStr)\n\t\t\t{\n\t\t\t\tlet testElm = null;\n\n\t\t\t\ttry\n\t\t\t\t{\n\t\t\t\t\tlet obj = foundStr;\n\n\t\t\t\t\tif (typeof (foundStr) === "string")\n\t\t\t\t\t\tobj = JSON.parse (foundStr);\n\n\t\t\t\t\tif (typeof (obj) === "string")\n\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj);\n\n\t\t\t\t\tif (obj instanceof Array)\n\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj[0], obj[1], obj[2]);\n\n\t\t\t\t\tif (obj["name"] != null)\n\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj);\n\n\t\t\t\t\tif (Hot.CurrentPage.testElements[testElm.name] != null)\n\t\t\t\t\t\tthrow new Error (`Test element ${testElm.name} already exists!`);\n\n\t\t\t\t\tHot.CurrentPage.addTestElement (testElm);\n\t\t\t\t}\n\t\t\t\tcatch (ex)\n\t\t\t\t{\n\t\t\t\t\tthrow new Error (\n\t\t\t`Error processing test element ${foundStr} in ${Hot.CurrentPage.name}. Error: ${ex.message}`\n\t\t\t\t\t\t);\n\t\t\t\t}\n\n\t\t\t\treturn (testElm);\n\t\t\t}\n\n\t\t\tasync function runContent (CurrentHotFile)\n\t\t\t{\n',e+=n,e+="\n\t\t\t}\n\n\t\t\treturn (runContent (PassedHotFile).then (() =>\n\t\t\t{\n\t\t\t\treturn ({\n\t\t\t\t\t\thot: Hot,\n\t\t\t\t\t\toutput: Hot.Output,\n\t\t\t\t\t\tdata: JSON.stringify (Hot.Data)\n\t\t\t\t\t});\n\t\t\t}));";let s=new Function(e);r=yield s.apply(this,[h.Hot,this])}catch(t){throw SyntaxError,t}h.Hot.Data=r.hot.Data;let o=r.output;return h.Hot.Output="",o}))}}e.HotFile=c},802:(t,e)=>{"use strict";var n;Object.defineProperty(e,"__esModule",{value:!0}),e.HotLog=e.HotLogLevel=void 0,function(t){t[t.Info=0]="Info",t[t.Warning=1]="Warning",t[t.Error=2]="Error",t[t.Verbose=3]="Verbose",t[t.All=4]="All",t[t.None=5]="None"}(n=e.HotLogLevel||(e.HotLogLevel={})),e.HotLog=class{constructor(t=n.All){this.logLevel=t}log(t,e){this.logLevel===n.Verbose&&(t===n.Error&&this.error(e),t===n.Warning&&this.warning(e),t!==n.Info&&t!==n.Verbose||this.info(e)),this.logLevel===n.All&&(t===n.Error&&this.error(e),t===n.Warning&&this.warning(e),t===n.Info&&this.info(e)),this.logLevel===n.Error&&t===n.Error&&this.error(e),this.logLevel===n.Warning&&t===n.Warning&&this.warning(e),this.logLevel===n.Info&&t===n.Info&&this.info(e)}verbose(t){this.logLevel===n.Verbose&&console.info(t)}info(t){this.logLevel!==n.All&&this.logLevel!==n.Verbose&&this.logLevel!==n.Info||console.info(t)}warning(t){this.logLevel!==n.All&&this.logLevel!==n.Verbose&&this.logLevel!==n.Warning||console.warn(t)}error(t){if(this.logLevel===n.All||this.logLevel===n.Verbose||this.logLevel===n.Error){let e="";"string"==typeof t?e=t:(null!=t.message&&(e=t.message),null!=t.stack&&(e=t.stack)),console.error(e)}}}},136:function(t,e,n){"use strict";var r=this&&this.__awaiter||function(t,e,n,r){return new(n||(n=Promise))((function(o,s){function i(t){try{a(r.next(t))}catch(t){s(t)}}function l(t){try{a(r.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?o(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((r=r.apply(t,e||[])).next())}))};Object.defineProperty(e,"__esModule",{value:!0}),e.HotPage=void 0;const o=n(507),s=n(243);e.HotPage=class{constructor(t){t instanceof s.HotStaq?(this.processor=t,this.name="",this.testerName="",this.testerMap="",this.route="",this.files=[],this.testElements={},this.testPaths={}):(this.processor=t.processor,this.name=t.name||"",this.testerName=t.testerName||"",this.testerMap=t.testerMap||"",this.route=t.route||"",this.files=t.files||[],this.testElements=t.testElements||{},this.testPaths=t.testPaths||{})}addFile(t){return r(this,void 0,void 0,(function*(){t.page=this,this.files.push(t)}))}getAPI(){return this.processor.api}getTesterAPI(){return this.processor.testerAPI}load(t){return r(this,void 0,void 0,(function*(){for(let t=0;t<this.files.length;t++){let e=this.files[t];yield e.load()}}))}process(t=null){return r(this,void 0,void 0,(function*(){let e="";for(let n=0;n<this.files.length;n++){let r=this.files[n];o.Hot.Output="",r.page=this,e+=(yield r.process(t))}return e}))}addTestElement(t){if(null!=this.testElements[t.name])throw new Error(`Test element ${t.name} already exists!`);this.testElements[t.name]=t}getTestElement(t){if(null==this.testElements[t])throw new Error(`Test element ${t} doest not exist!`);return this.testElements[t]}createTestPath(t,e){if(null!=this.testPaths[t])throw new Error(`Test path ${t} already exists!`);this.testPaths[t]=e}}},761:(t,e,n)=>{"use strict";Object.defineProperty(e,"__esModule",{value:!0}),e.HotRoute=void 0;const r=n(70);class o{constructor(t,e,n=[]){this.onPreRegister=null,this.onRegister=null,this.onPostRegister=null,this.onAuthorizeUser=null,this.connection=t,this.logger=null,null!=this.connection&&null!=this.connection.processor&&(this.logger=this.connection.processor.logger),this.route=e,this.description="",this.version="v1",this.prefix="",this.authCredentials=null,this.methods=n,this.errors={not_authorized:o.createError("Not authorized."),no_server_execute_function:o.createError("Missing server execute function.")}}static createError(t){return{error:t}}addMethod(t,e=null,n=r.HTTPMethod.POST,o=null){"string"==typeof t&&(t=new r.HotRouteMethod(this,t,e,n,null,null,null,o)),t instanceof r.HotRouteMethod||(null==t.route&&(t.route=this),t=new r.HotRouteMethod(t)),this.methods.push(t)}getMethod(t){let e=null;for(let n=0;n<this.methods.length;n++){let r=this.methods[n];if(r.name===t){e=r;break}}return e}}e.HotRoute=o},70:(t,e,n)=>{"use strict";Object.defineProperty(e,"__esModule",{value:!0}),e.HotRouteMethod=e.HTTPMethod=void 0;const r=n(507),o=n(761),s=n(810);var i;!function(t){t.GET="get",t.POST="post",t.FILE_UPLOAD="file_upload_then_post_json"}(i=e.HTTPMethod||(e.HTTPMethod={})),e.HotRouteMethod=class{constructor(t,e="",n=null,l=i.POST,a=null,u=null,h=null,c=null){let p=null;if(t instanceof o.HotRoute)p=t;else{if(p=t.route,null!=t.type&&(l=t.type),null!=t.name&&(e=t.name),null!=t.description&&(this.description=t.description),null!=t.returns&&("string"==typeof t.returns?this.returns={type:"string",required:!0,description:t.returns}:this.returns=t.returns),null!=t.parameters){this.parameters={};for(let e in t.parameters){let n=t.parameters[e];"string"==typeof n?this.parameters[e]={type:n,required:!1,description:""}:(null==n.type&&(n.type="string"),this.parameters[e]=n)}}null!=t.authCredentials&&(h=t.authCredentials),null!=t.onServerExecute&&(n=t.onServerExecute),null!=t.onServerAuthorize&&(a=t.onServerAuthorize),null!=t.onRegister&&(u=t.onRegister),null!=t.onPostRegister&&(this.onPostRegister=t.onPostRegister),null!=t.onServerExecute&&(this.onServerExecute=t.onServerExecute),null!=t.onClientExecute&&(this.onClientExecute=t.onClientExecute),null!=t.testCases&&(c=t.testCases)}if(""===e)throw new Error("All route methods must have a name!");if(this.route=p,this.name=e,this.type=l,this.isRegistered=!1,this.executeSetup=!1,this.authCredentials=h,this.onServerAuthorize=a,this.onRegister=u,this.testCases={},this.route.connection.processor.mode===r.DeveloperMode.Development&&null!=c)if(c instanceof Array)for(let t=0;t<c.length;t++){let e=c[t];if("string"==typeof e){const n=e,r=c[t+1];this.addTestCase(n,r),t++}else this.addTestCase(e)}else for(let t in c){let e=c[t];this.addTestCase(e)}this.route.connection instanceof s.HotServer&&(this.onServerExecute=n)}addTestCase(t,e=null){if("string"==typeof t){const n=t,r=e;return void(this.testCases[n]={name:n,func:r})}if("function"==typeof t){const e=Object.keys(this.testCases).length,n=`${this.route.route}/${this.name} test case ${e}`,r=t;return void(this.testCases[n]={name:n,func:r})}const n=t;this.testCases[n.name]=n}}},810:function(t,e,n){"use strict";var r=this&&this.__awaiter||function(t,e,n,r){return new(n||(n=Promise))((function(o,s){function i(t){try{a(r.next(t))}catch(t){s(t)}}function l(t){try{a(r.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?o(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((r=r.apply(t,e||[])).next())}))};Object.defineProperty(e,"__esModule",{value:!0}),e.HotServer=e.HotServerType=void 0;const o=n(243);var s;!function(t){t[t.HTTP=0]="HTTP",t[t.WebSockets=1]="WebSockets",t[t.Generate=2]="Generate"}(s=e.HotServerType||(e.HotServerType={})),e.HotServer=class{constructor(t){t instanceof o.HotStaq?(this.processor=t,this.serverType="Server",this.api=null,this.listenAddress="0.0.0.0",this.ports={http:80,https:443},this.ssl={cert:"",key:"",ca:""},this.redirectHTTPtoHTTPS=!0,this.type=s.HTTP,this.logger=t.logger,this.secrets={}):(this.processor=t.processor,this.serverType=t.serverType||"Server",this.api=t.api||null,this.listenAddress=t.listenAddress||"0.0.0.0",this.ports=t.ports||{http:80,https:443},this.ssl=t.ssl||{cert:"",key:"",ca:""},this.redirectHTTPtoHTTPS=null==t.redirectHTTPtoHTTPS||t.redirectHTTPtoHTTPS,this.type=t.type||s.HTTP,this.logger=t.logger,this.secrets=t.secrets||{})}setAPI(t){return r(this,void 0,void 0,(function*(){this.processor.api=t,this.api=t}))}}},243:function(__unused_webpack_module,exports,__webpack_require__){"use strict";var __createBinding=this&&this.__createBinding||(Object.create?function(t,e,n,r){void 0===r&&(r=n),Object.defineProperty(t,r,{enumerable:!0,get:function(){return e[n]}})}:function(t,e,n,r){void 0===r&&(r=n),t[r]=e[n]}),__setModuleDefault=this&&this.__setModuleDefault||(Object.create?function(t,e){Object.defineProperty(t,"default",{enumerable:!0,value:e})}:function(t,e){t.default=e}),__importStar=this&&this.__importStar||function(t){if(t&&t.__esModule)return t;var e={};if(null!=t)for(var n in t)"default"!==n&&Object.prototype.hasOwnProperty.call(t,n)&&__createBinding(e,t,n);return __setModuleDefault(e,t),e},__awaiter=this&&this.__awaiter||function(t,e,n,r){return new(n||(n=Promise))((function(o,s){function i(t){try{a(r.next(t))}catch(t){s(t)}}function l(t){try{a(r.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?o(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((r=r.apply(t,e||[])).next())}))},__importDefault=this&&this.__importDefault||function(t){return t&&t.__esModule?t:{default:t}};Object.defineProperty(exports,"__esModule",{value:!0}),exports.HotStaq=void 0;const fs=__importStar(__webpack_require__(351)),ppath=__importStar(__webpack_require__(606)),node_fetch_1=__importDefault(__webpack_require__(300)),validate_npm_package_name_1=__importDefault(__webpack_require__(651)),HotPage_1=__webpack_require__(136),HotFile_1=__webpack_require__(732),HotComponent_1=__webpack_require__(260),HotLog_1=__webpack_require__(802),Hot_1=__webpack_require__(507),HotClient_1=__webpack_require__(957),HotTesterAPI_1=__webpack_require__(414),HotTestMap_1=__webpack_require__(985);var HotTesterMocha=null,HotTesterMochaSelenium=null,HotTestSeleniumDriver=null;class HotStaq{constructor(t={}){this.api=t.api||null,this.testerAPI=t.testerAPI||null,this.mode=t.mode||Hot_1.DeveloperMode.Production,this.pages=t.pages||{},this.components=t.components||{},this.files=t.files||{},this.hotSite=t.hotSite||null,this.apiContent="\n\t\t\t{\n\t\t\t\tvar %api_name% = %api_exported_name%.%api_name%;\n\t\t\t\tvar newHotClient = new HotClient (processor);\n\t\t\t\tvar newapi = new %api_name% (%base_url%, newHotClient);\n\t\t\t\tnewHotClient.api = newapi;\n\t\t\t\tprocessor.api = newapi;\n\t\t\t}",this.testerApiContent="\n\t\t\tvar HotTesterAPI = HotStaqWeb.HotTesterAPI;\n\t\t\tvar newHotTesterClient = new HotClient (processor);\n\t\t\tvar newtesterapi = new HotTesterAPI (%base_tester_url%, newHotTesterClient);\n\t\t\tnewHotTesterClient.testerAPI = newtesterapi;\n\t\t\tprocessor.testerAPI = newtesterapi;",this.pageContent='<!DOCTYPE html>\n<html>\n\n<head>\n\t<title>%title%</title>\n\n\t<script type = "text/javascript" src = "%hotstaq_js_src%"><\/script>\n\t<script type = "text/javascript">\n\t\twindow.HotStaq = HotStaqWeb.HotStaq;\n\t\twindow.HotClient = HotStaqWeb.HotClient;\n\t\twindow.HotAPI = HotStaqWeb.HotAPI;\n\t\twindow.Hot = HotStaqWeb.Hot;\n\t<\/script>\n\n%apis_to_load%\n\n\t<script type = "text/javascript">\n\t\tfunction hotstaq_startApp ()\n\t\t{\n\t\t\tlet tempMode = 0;\n\n\t\t\tif (window["Hot"] != null)\n\t\t\t\ttempMode = Hot.Mode;\n\n\t\t\t%load_hot_site%\n\n\t\t\tvar processor = new HotStaq ();\n\t\t\tvar promises = [];\n\t\t\t%developer_mode%\n\n\t\t\t%api_code%\n\n\t\t\t%public_secrets%\n\t\t\t%tester_api%\n\t\t\t%load_files%\n\n\t\t\tprocessor.mode = tempMode;\n\n\t\t\tPromise.all (promises).then (function ()\n\t\t\t\t{\n\t\t\t\t\tHotStaq.displayUrl ({\n\t\t\t\t\t\t\turl: "%url%",\n\t\t\t\t\t\t\tname: "%title%",\n\t\t\t\t\t\t\tprocessor: processor,\n\t\t\t\t\t\t\targs: %args%,\n\t\t\t\t\t\t\ttesterName: %tester_name%,\n\t\t\t\t\t\t\ttesterMap: %tester_map%,\n\t\t\t\t\t\t\ttesterAPIBaseUrl: %tester_api_base_url%,\n\t\t\t\t\t\t\ttesterLaunchpadUrl: %tester_launchpad_url%\n\t\t\t\t\t\t});\n\t\t\t\t});\n\t\t}\n\n\t\thotstaq_startApp ();\n\t<\/script>\n</head>\n\n<body>\n</body>\n\n</html>',this.logger=new HotLog_1.HotLog(HotLog_1.HotLogLevel.None),this.publicKeys={},this.testers={}}static parseBoolean(t){if("true"===(t=t.toLowerCase()))return!0;if("false"===t)return!1;if("yes"===t)return!0;if("no"===t)return!1;if("yep"===t)return!0;if("nah"===t)return!1;try{if(0!=parseInt(t))return!0}catch(t){}return!1}static getParam(t,e,n=!0,r=!0){let o=e[t];if(null==o&&!0===n&&!0===r)throw new Error(`Missing required parameter ${t}.`);if("string"==typeof o&&!0===n&&""===o&&!0===r)throw new Error(`Missing required parameter ${t}.`);return o}static getParamDefault(t,e,n){let r=e[t];return null==r||"string"==typeof r&&""===r?n:r}static executeError(t){if(null!=HotStaq.errors[t]){let e=HotStaq.errors[t].redirectToUrl;if(null!=e&&""!==e)return void(window.location.href=e);let n=HotStaq.errors[t].func;null!=n&&n(t)}}static wait(t){return __awaiter(this,void 0,void 0,(function*(){return yield new Promise(((e,n)=>{setTimeout((()=>{e()}),t)}))}))}addPage(t){this.pages[t.name]=t}getPage(t){return this.pages[t]}addFile(t){let e=t.name;""===e&&(e=t.localFile),""===e&&(e=t.url),this.files[e]=t}getFile(t){if(null==this.files[t])throw new Error(`Unable to find file ${t}`);return this.files[t]}static keepContext(t,e,n){return function(){var r=Array.prototype.slice.call(arguments);return null!=n&&r.push(n),null==e?t.apply(this,r):t.apply(e,r)}}addComponent(t,e=null,n){let r=this.api;null!=e&&(r=e);let o=new t(this,r);if(null!=this.components[o.tag])throw new Error(`Component ${o.tag} already exists!`);this.components[o.tag]={componentType:t,processor:this,api:r},this.registerComponent(o.tag,n)}static fixHTML(t){const e=t.replace(/ \/>/g,">").replace(/(<(area|base|br|col|command|embed|hr|img|input|keygen|link|meta|param|source|track|wbr).*?>)/g,"$1</$2>"),n=(new DOMParser).parseFromString(`<xml>${e}</xml>`,"text/xml");let r="";if(n.documentElement.children.length>0){const e=n.documentElement.children[0].tagName.toLowerCase();"tr"===e&&(t=`<table>${t}</table>`,r="tbody"),"th"===e&&(t=`<table>${t}</table>`,r=e)}return{fixedStr:t,querySelector:r}}registerComponent(t,e){if(null==t||""===t)throw new Error("All components must have a tag!");if(void 0!==customElements.get(t))return;let n=this.components;customElements.define(t,class extends HTMLElement{constructor(){super();let e=n[t];this.component=new e.componentType(e.processor,e.api)}looseParseFromString(t,e){e=e.replace(/ \/>/g,">").replace(/(<(area|base|br|col|command|embed|hr|img|input|keygen|link|meta|param|source|track|wbr).*?>)/g,"$1</$2>");const n=t.parseFromString("<xml>"+e+"</xml>","text/xml"),r=t.parseFromString("","text/html");for(let t of Array.from(n.documentElement.children))r.body.appendChild(t);for(let t of Array.from(r.querySelectorAll("area,base,br,col,command,embed,hr,img,input,keygen,link,meta,param,source,track,wbr")))t.outerHTML="<"+t.outerHTML.slice(1).split("<")[0];return r}get observedAttributes(){return this.component.observedAttributes}connectedCallback(){return __awaiter(this,void 0,void 0,(function*(){if(this.hotComponent=this.component,this.component.htmlElements=[this],this.component.inner=this.innerHTML,null!=this.component.handleAttributes)yield this.component.handleAttributes(this.attributes);else for(let t=0;t<this.attributes.length;t++){const e=this.attributes[t],n=e.name.toLowerCase(),r=e.value;if("id"===n&&(this.component.name=r),"name"===n&&(this.component.name=r),"value"===n&&(this.component.value=r),n.indexOf("hot-")>-1){const t=n.substring(4);this.component[t]=r}}if(null!=this.component.onPreOutput&&!1===(yield this.component.onPreOutput()))return;let t=yield this.component.output();null!=this.component.onPostOutput&&(t=yield this.component.onPostOutput(t));let e=[];"string"==typeof t?e.push({html:t}):e=t instanceof Array?t:[t];for(let t=0;t<e.length;t++){let n=e[t],r=n.html,o="";null!=n.addFunctionsTo&&(o=n.addFunctionsTo);let s=HotFile_1.HotFile.parseContent(r,!0,{outputCommands:!1});null!=this.component.onParsed&&(s=yield this.component.onParsed(s));let i={fixedStr:"",querySelector:""};i=null!=this.component.onFixHTML?yield this.component.onFixHTML(s):HotStaq.fixHTML(s);let l=null,a=null;if(l=null!=this.component.onParseDOM?yield this.component.onParseDOM(i.fixedStr):(new DOMParser).parseFromString(i.fixedStr,"text/html"),l.body.children.length<1)throw new Error(`No component output from ${this.component.name}`);if(l.body.children.length>1){let t=!0;for(let e=0;e<l.body.children.length;e++){let n=l.body.children[e];if(n instanceof HTMLElement&&"parsererror"===n.tagName.toLowerCase()){a=n,t=!1;break}}if(!0===t)throw new Error(`Only a single html element can come from component ${this.component.name}, multiple elements were detected.`)}a=""===i.querySelector?l.body.children[0]:l.querySelector(i.querySelector);let u=[];for(let t=this.children.length-1;t>-1;t--){let e=this.children[t];u.push(this.removeChild(e))}this.replaceWith(a),null!=this.component.click&&(a.onclick=this.component.click.bind(this.component));for(let t in this.component.events){let e=this.component.events[t];a.addEventListener(e.type,e.func,e.options)}let h=Object.getOwnPropertyNames(this.component.constructor.prototype);for(let t=0;t<h.length;t++){let e=h[t];if("constructor"!==e&&"function"==typeof this.component[e]){let t=!0;for(let n in HotComponent_1.HotComponent.prototype)if(e===n){t=!1;break}!0===t&&(a[e]=HotStaq.keepContext(this.component[e],this.component),""!==o)&&(document.querySelector(o)[e]=HotStaq.keepContext(this.component[e],this.component))}}null!=this.component.onPrePlace&&(a=yield this.component.onPrePlace(a));let c=yield this.component.onCreated(a);if(null!=this.component.onParentPlace&&(c.onParentPlace=this.component.onParentPlace),c.hotComponent=this.component,this.component.htmlElements.push(c),null!=n.parentSelector){let t=document.querySelector(n.parentSelector);c.parentElement.removeChild(c),t.appendChild(c),null!=c.onParentPlace&&(yield c.hotComponent.onParentPlace(t,c))}if(null!=n.placeHereParent){let t=c.parentNode,e=0;for(;e<10&&null!=t&&!(t instanceof HTMLHtmlElement);){let r=t.querySelectorAll(`hot-place-here[name="${n.placeHereParent}"]`);if(r.length>0){let t=r[0];c.parentNode.removeChild(c),t.appendChild(c),null!=c.onParentPlace&&(yield c.hotComponent.onParentPlace(t,c));break}if(r.length<1){let e=t.querySelectorAll(`[hot-place-here="${n.placeHereParent}"]`);if(e.length>0){let t=e[0];c.parentNode.removeChild(c),t.appendChild(c),null!=c.onParentPlace&&(yield c.hotComponent.onParentPlace(t,c));break}}t=t.parentNode,e++}}for(let t=0;t<u.length;t++){const e=u[t];c.appendChild(e),null!=e.onParentPlace&&(yield e.hotComponent.onParentPlace(c,e))}null!=this.component.onPostPlace&&(c=yield this.component.onPostPlace(c.parentNode,c))}}))}},e)}static addHtml(t,e){let n=null;if(n="string"==typeof t?document.querySelector(t):t,null==n)throw new Error(`Unable to find parent ${t}!`);let r=null;if("string"==typeof e){let t=HotStaq.fixHTML(e),r=(new DOMParser).parseFromString(t.fixedStr,"text/html"),o=null;o=""===t.querySelector?r.body.children:r.querySelector(t.querySelector).children;let s=[];for(let t=0;t<o.length;t++){let e=o[t];s.push(n.appendChild(e))}return s}return r=n.appendChild(e),r}static checkHotSiteName(t,e=!1){let n=(0,validate_npm_package_name_1.default)(t);return null!=n.errors&&n.errors.length>0&&(()=>{if(!0===e)throw new Error(`HotSite ${t} has an invalid name! The name cannot be empty and must have a valid NPM module name.`)})(),!0}static replaceKey(t,e,n){return t.replace(new RegExp(`\\$\\{${e}\\}`,"g"),n)}static getValueFromHotSiteObj(t,e){let n=null;if(null!=t){let r=t;for(let t=0;t<e.length;t++){let n=e[t];if(null==r[n]){r=null;break}r=r[n]}null!=r&&(n=r)}return n}processHotSite(){return __awaiter(this,void 0,void 0,(function*(){HotStaq.checkHotSiteName(this.hotSite.name,!0);let routes=this.hotSite.routes,testerUrl="http://127.0.0.1:8182",tester=null,driver=null;if(!1===HotStaq.isWeb&&this.mode===Hot_1.DeveloperMode.Development&&null!=this.hotSite.testing){let t=t=>{let e=!0;null!=t.createNewTester&&(e=t.createNewTester);let n="Tester";if(null!=t.tester&&(n=t.tester),null!=t.testerName&&(n=t.testerName),!0===e?(HotTesterMocha=Object(function(){var t=new Error("Cannot find module './HotTesterMocha'");throw t.code="MODULE_NOT_FOUND",t}()),HotTesterMochaSelenium=Object(function(){var t=new Error("Cannot find module './HotTesterMochaSelenium'");throw t.code="MODULE_NOT_FOUND",t}()),HotTestSeleniumDriver=Object(function(){var t=new Error("Cannot find module './HotTestSeleniumDriver'");throw t.code="MODULE_NOT_FOUND",t}()),""===t.testerAPIUrl&&(testerUrl=t.testerAPIUrl),"HotTestSeleniumDriver"===t.driver&&(driver=new HotTestSeleniumDriver),"HotTesterMocha"===t.tester&&(tester=new HotTesterMocha(this,n,testerUrl,driver)),"HotTesterMochaSelenium"===t.tester&&(tester=new HotTesterMochaSelenium(this,n,testerUrl))):tester=this.testers[n],null==tester.driver)throw new Error(`Tester ${n} does not have a driver set!`);null!=t.commandDelay&&(tester.driver.commandDelay=t.commandDelay)};null!=this.hotSite.testing.web&&t(this.hotSite.testing.web),null!=this.hotSite.testing.api&&t(this.hotSite.testing.api)}if(null!=routes)for(let t in routes){let e=routes[t],n=new HotFile_1.HotFile(e),r=new HotPage_1.HotPage({processor:this,name:e.name||"",route:t,files:[n]});if(null!=tester&&this.mode===Hot_1.DeveloperMode.Development){let t=e.name,n=null;if(null!=e.map){if("string"==typeof e.map){if(null==tester.testMaps[e.map])throw new Error(`Test map ${e.map} does not exist!`);tester.testMaps[t]=tester.testMaps[e.map]}else{n=new HotTestMap_1.HotTestMap;let t=null;if(e.map instanceof Array){t=[];for(let n=0;n<e.map.length;n++){let r=e.map[n];t.push(new HotTestMap_1.HotTestDestination(r))}}else{t={};for(let n in e.map){let r=e.map[n];t[n]=new HotTestMap_1.HotTestDestination(r)}}n.destinations=t}tester.testMaps[t]=n}null!=e.destinationOrder&&(tester.testMaps[t].destinationOrder=e.destinationOrder)}this.addPage(r)}if(null!=this.hotSite.apis)for(let t in this.hotSite.apis){let e=this.hotSite.apis[t];if(null!=e.map&&!1===HotStaq.isWeb&&this.mode===Hot_1.DeveloperMode.Development){let n=t,r=new HotTestMap_1.HotTestMap;r.destinations=[];for(let t=0;t<e.map.length;t++){let n=e.map[t];r.destinations.push(new HotTestMap_1.HotTestDestination(n))}if(null==tester)throw new Error("A tester was not created first! You must specify one in the CLI or in HotSite.json.");tester.testMaps[n]=r}}if(!0===HotStaq.isWeb)for(let key in this.hotSite.components){let component=this.hotSite.components[key],componentUrl=component.url,res=yield(0,node_fetch_1.default)(componentUrl),ComponentClass=eval(res);this.addComponent(ComponentClass)}null==this.hotSite.routes&&(this.hotSite.routes={});let disableFileLoading=!1;null!=this.hotSite.disableFileLoading&&(disableFileLoading=this.hotSite.disableFileLoading),!1===disableFileLoading?yield this.loadHotFiles(this.hotSite.files):this.logger.verbose("Hotsite has file loading disabled..."),null!=tester&&this.addTester(tester)}))}loadHotSite(t){return __awaiter(this,void 0,void 0,(function*(){let e="";if(!0===HotStaq.isWeb){this.logger.info(`Downloading HotSite ${t}`);let n=yield(0,node_fetch_1.default)(t);this.logger.info(`Downloaded site ${t}`),e=n.text()}else t=ppath.normalize(t),this.logger.info(`Accessing HotSite ${t}`),e=yield new Promise(((e,n)=>{fs.readFile(t,((n,r)=>{if(null!=n)throw n;let o=r.toString();this.logger.info(`Accessed site ${t}`),e(o)}))}));this.hotSite=JSON.parse(e),this.hotSite.hotsitePath=t}))}loadHotFiles(t,e=!1){return __awaiter(this,void 0,void 0,(function*(){this.logger.verbose("Loading Hott files...");for(let n in t){let r=t[n],o=null;HotStaq.isWeb,o=new HotFile_1.HotFile({name:n}),null!=r.url&&(o.url=r.url),!1===HotStaq.isWeb&&null!=r.localFile&&(o.localFile=r.localFile);let s=!0;if(null!=r.content&&(o.content=r.content,s=!1),!0===e&&(s=!0),!0===s){let t="";""!==o.url&&(t=o.url),""!==o.localFile&&(t=o.localFile),this.logger.verbose(`Loading Hott file: ${t}`),yield o.load(),this.logger.verbose(`Finished loading Hott file: ${t}`)}this.addFile(o)}this.logger.verbose("Finished loading Hott files...")}))}generateContent(t,e="",n="./",r="./js/HotStaq.min.js",o=!0,s=null){let i="",l="",a="";if(null!=this.hotSite){if(null!=this.hotSite.server.globalApi&&""!==this.hotSite.server.globalApi){const t=this.hotSite.apis[this.hotSite.server.globalApi];if(null==t)this.logger.warning(`API with name ${this.hotSite.server.globalApi} doesn't exist!`);else{let e=!0;if(null==t.jsapi&&(e=!1,this.logger.warning(`API with name ${this.hotSite.server.globalApi} doesn't have a jsapi set. Will not send js content to client.`)),null==t.libraryName&&(e=!1,this.logger.warning(`API with name ${this.hotSite.server.globalApi} doesn't have a libraryName set. Will not send js content to client.`)),null==t.apiName&&(e=!1,this.logger.warning(`API with name ${this.hotSite.server.globalApi} doesn't have a apiName set. Will not send js content to client.`)),!0===e){i+=`\t<script type = "text/javascript" src = "${t.jsapi}"><\/script>\n`;let e='""';null!=t.url&&(e=`"${t.url}"`),null!=this.api&&(e=`"${this.api.baseUrl}"`);let n=this.apiContent;n=n.replace(/\%api\_name\%/g,t.apiName),n=n.replace(/\%api\_exported\_name\%/g,t.libraryName),n=n.replace(/\%base\_url\%/g,e),l+=n}}}if(null!=this.hotSite.apis){let e=this.hotSite.routes[t];if(null!=e&&null!=e.api){let t=this.hotSite.apis[e.api];if(null==t)throw new Error(`Unable to find API ${e.api}`);let n=!0;if(null==t.jsapi&&(n=!1,this.logger.warning(`API with name ${e.api} doesn't have a jsapi set. Will not send js content to client.`)),null==t.libraryName&&(n=!1,this.logger.warning(`API with name ${e.api} doesn't have a libraryName set. Will not send js content to client.`)),null==t.apiName&&(n=!1,this.logger.warning(`API with name ${e.api} doesn't have a apiName set. Will not send js content to client.`)),!0===n){let e=t.jsapi;i+=`\t<script type = "text/javascript" src = "${e}"><\/script>\n`;let n='""';null!=t.url&&(n=`"${t.url}"`),null!=this.api&&(n=`"${this.api.baseUrl}"`);let r=this.apiContent;r=r.replace(/\%api\_name\%/g,t.apiName),r=r.replace(/\%api\_exported\_name\%/g,t.libraryName),r=r.replace(/\%base\_url\%/g,n),l+=r}}}if(null!=this.hotSite.server&&null!=this.hotSite.server.jsSrcPath&&(r=this.hotSite.server.jsSrcPath),null!=this.hotSite.publicKeys)for(let t in this.hotSite.publicKeys){let e,n=this.hotSite.publicKeys[t];if("string"==typeof n)e=JSON.stringify(n);else if(!1===HotStaq.isWeb){if(null!=this.api){if(null==this.api.connection)throw new Error("Cannot pass secrets from the API if there's no connection!");let r=this.api.connection;null!=n.passSecretFromAPI&&(e=JSON.stringify(r.secrets[t]))}if(null!=n.env){const t=n.env;e=JSON.stringify(process.env[t])}}a+=`processor.publicKeys["${t}"] = ${e};\n`}}let u=this.pageContent;return u=(u=>{let h="",c="";this.mode===Hot_1.DeveloperMode.Development&&(h="tempMode = HotStaqWeb.DeveloperMode.Development;",null!=this.hotSite&&null!=this.hotSite.testing&&null!=this.hotSite.testing.web&&(c=this.testerApiContent,null==this.hotSite.testing.web.testerAPIUrl&&(this.hotSite.testing.web.testerAPIUrl="http://127.0.0.1:8182"),c=c.replace(/\%base\_tester\_url\%/g,`"${this.hotSite.testing.web.testerAPIUrl}"`)));let p="";if(Object.keys(this.files).length>0){p+="var files = {};\n\n";for(let t in this.files){let e=this.files[t],n=`"${e.url}"`,r="";if(""!==e.content){let t=JSON.stringify(e.content);t=t.replace(new RegExp("\\<script","gmi"),'<scr" + "ipt'),t=t.replace(new RegExp("\\<\\/script","gmi"),'</scr" + "ipt'),r=`, "content": ${t}`}p+=`\t\t\tfiles["${t}"] = { "url": ${n}${r} };\n`}p+="\t\t\tpromises.push (processor.loadHotFiles (files));\n"}u=u.replace(/\%title\%/g,e),!0===o&&(u=u.replace(/\%args\%/g,"Hot.Arguments")),null!=s&&(u=u.replace(/\%args\%/g,JSON.stringify(s)));let d=t,m="",f="",g="Tester";return null!=this.hotSite&&(null!=this.hotSite.testing&&null!=this.hotSite.testing.web&&(null!=this.hotSite.testing.web.tester&&(g=this.hotSite.testing.web.tester),null!=this.hotSite.testing.web.testerName&&(g=this.hotSite.testing.web.testerName),null!=this.hotSite.testing.web.testerAPIUrl&&(m=this.hotSite.testing.web.testerAPIUrl),null!=this.hotSite.testing.web.launchpadUrl&&(f=this.hotSite.testing.web.launchpadUrl)),null!=this.hotSite.routes&&null!=this.hotSite.routes[t])&&(d=this.hotSite.routes[t].name),(u=(u=(u=(u=(u=(u=(u=(u=(u=(u=(u=(u=u.replace(/\%hotstaq\_js\_src\%/g,r)).replace(/\%developer\_mode\%/g,h)).replace(/\%tester\_api\%/g,c)).replace(/\%apis\_to\_load\%/g,i)).replace(/\%load\_hot\_site\%/g,"")).replace(/\%load\_files\%/g,p)).replace(/\%api\_code\%/g,l)).replace(/\%public\_secrets\%/g,a)).replace(/\%url\%/g,n)).replace(/\%tester\_name\%/g,`"${g}"`)).replace(/\%tester\_map\%/g,`"${d}"`)).replace(/\%tester\_api\_base\_url\%/g,`"${m}"`)).replace(/\%tester\_launchpad\_url\%/g,`"${f}"`)})(u),u}createExpressRoutes(t,e="./js/HotStaq.min.js"){for(let n in this.pages){let r=this.pages[n];const o=this.generateContent(r.route,r.name,r.files[0].url,e);t.get(r.route,((t,e)=>{e.send(o)}))}}addTester(t){this.testers[t.name]=t}getWebTestingMaps(){if(null==this.hotSite)throw new Error("No HotSite was loaded!");if(null==this.hotSite.testing)throw new Error("The HotSite does not have a testing object!");if(null==this.hotSite.testing.web)throw new Error("The HotSite does not have a testing web object!");if(null==this.hotSite.testing.web.maps)throw new Error("The HotSite testing object does not have any maps!");return this.hotSite.testing.web.maps}getAPITestingMaps(){if(null==this.hotSite)throw new Error("No HotSite was loaded!");if(null==this.hotSite.testing)throw new Error("The HotSite does not have a testing object!");if(null==this.hotSite.testing.api)throw new Error("The HotSite does not have a testing api object!");if(null==this.hotSite.testing.api.maps)throw new Error("The HotSite testing object does not have any maps!");return this.hotSite.testing.api.maps}getRouteKeyFromName(t){let e="";if(null!=this.hotSite&&null!=this.hotSite.routes)for(let n in this.hotSite.routes)if(this.hotSite.routes[n].name===t){e=n;break}return e}getRouteFromName(t){let e=null,n=this.getRouteKeyFromName(t);return""!==n&&(e=this.hotSite.routes[n]),e}executeTests(t,e){return __awaiter(this,void 0,void 0,(function*(){let n=this.testers[t];if(null==n)throw new Error(`Unable to execute tests. Tester ${t} does not exist!`);return n.execute(e)}))}executeAllWebTests(t){return __awaiter(this,void 0,void 0,(function*(){let e=this.getWebTestingMaps();if(null==this.testers[t])throw new Error(`Unable to execute tests. Tester ${t} does not exist!`);for(let n=0;n<e.length;n++){let r=e[n];yield this.executeTests(t,r)}}))}executeAllAPITests(t){return __awaiter(this,void 0,void 0,(function*(){let e=this.getAPITestingMaps();if(null==this.testers[t])throw new Error(`Unable to execute tests. Tester ${t} does not exist!`);for(let n=0;n<e.length;n++){let r=e[n];yield this.executeTests(t,r)}}))}process(t,e=null){return __awaiter(this,void 0,void 0,(function*(){let n=this.getPage(t);return yield n.process(e)}))}static processLocalFile(t,e=t,n=null){return __awaiter(this,void 0,void 0,(function*(){let r=new HotStaq,o=new HotFile_1.HotFile({localFile:t});yield o.load();let s=new HotPage_1.HotPage({processor:r,name:e,files:[o]});return r.addPage(s),yield r.process(e,n)}))}static processUrl(t){return __awaiter(this,void 0,void 0,(function*(){let e=new HotFile_1.HotFile({url:t.url});yield e.load();let n=new HotPage_1.HotPage({processor:t.processor,name:t.name,files:[e],testerName:t.testerName,testerMap:t.testerMap});return t.processor.addPage(n),yield t.processor.process(t.name,t.args)}))}static processContent(t){return __awaiter(this,void 0,void 0,(function*(){let e=new HotFile_1.HotFile({content:t.content});yield e.load();let n=new HotPage_1.HotPage({processor:t.processor,name:t.name,files:[e]});return t.processor.addPage(n),yield t.processor.process(t.name,t.args)}))}static onReady(t){"complete"===document.readyState||"interactive"===document.readyState?t():window.addEventListener("load",t)}static useOutput(t){return __awaiter(this,void 0,void 0,(function*(){let e=(new DOMParser).parseFromString(t,"text/html");document.getElementsByTagName("html")[0].innerHTML=e.getElementsByTagName("html")[0].innerHTML;let n=document.getElementsByTagName("script");if(n.length>0){let t=[];for(let e=0;e<n.length;e++)t.push(n[e]);for(let e=0;e<t.length;e++){let n=document.createElement("script");t[e].parentNode.appendChild(n),t[e].parentNode.removeChild(t[e]),yield new Promise(((r,o)=>{n.onload=()=>{r()};let s=!1;null!=t[e].getAttribute("src")&&""!==t[e].getAttribute("src")&&(n.setAttribute("src",t[e].getAttribute("src")),s=!0),null!=t[e].getAttribute("type")&&""!==t[e].getAttribute("type")&&n.setAttribute("type",t[e].getAttribute("type")),n.innerHTML=t[e].innerHTML,!1===s&&r()}))}}}))}static waitForTesters(){return __awaiter(this,void 0,void 0,(function*(){for(;!1===HotStaq.isReadyForTesting;)yield HotStaq.wait(10);null!=HotStaq.onReadyForTesting&&(yield HotStaq.onReadyForTesting())}))}static setupTesters(t,e){if(t.mode===Hot_1.DeveloperMode.Development&&null==t.testerAPI){null==e.testerAPIBaseUrl&&(e.testerAPIBaseUrl=""),""===e.testerAPIBaseUrl&&(e.testerAPIBaseUrl="http://127.0.0.1:8182");let n=new HotClient_1.HotClient(t),r=new HotTesterAPI_1.HotTesterAPI(e.testerAPIBaseUrl,n);r.connection.api=r,t.testerAPI=r}}static setupClientTesters(t){let e="";return t.mode===Hot_1.DeveloperMode.Development&&(e+='<script type = "text/javascript">\nfunction hotstaq_isDocumentReady ()\n{\nif (window["Hot"] != null)\n{\nif (Hot.Mode === HotStaqWeb.DeveloperMode.Development)\n{\nlet func = function ()\n\t{\n\t\tif (Hot.TesterAPI != null)\n\t\t{\n\t\t\tlet testPaths = {};\n\t\t\tlet testElements = JSON.stringify (Hot.CurrentPage.testElements);\n\t\t\tlet testMaps = JSON.stringify (Hot.CurrentPage.testMaps);\n\n\t\t\tfor (let key in Hot.CurrentPage.testPaths)\n\t\t\t{\n\t\t\t\tlet testPath = Hot.CurrentPage.testPaths[key];\n\n\t\t\t\ttestPaths[key] = testPath.toString ();\n\t\t\t}\n\n\t\t\tlet testPathsStr = JSON.stringify (testPaths);\n\n\t\t\tHot.TesterAPI.tester.pageLoaded ({\n\t\t\t\t\ttesterName: Hot.CurrentPage.testerName,\n\t\t\t\t\ttesterMap: Hot.CurrentPage.testerMap,\n\t\t\t\t\tpageName: Hot.CurrentPage.name,\n\t\t\t\t\ttestElements: testElements,\n\t\t\t\t\ttestPaths: testPathsStr\n\t\t\t\t}).then (function (resp)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (resp.error != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (resp.error !== "")\n\t\t\t\t\t\t\t\tthrow new Error (resp.error);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tHotStaqWeb.HotStaq.isReadyForTesting = true;\n\t\t\t\t\t});\n\t\t}\n\t};\n\nif ((document.readyState === "complete") || (document.readyState === "interactive"))\n\tfunc ();\nelse\n\tdocument.addEventListener ("DOMContentLoaded", func);\n}\n}\n}\n\nhotstaq_isDocumentReady ();\n<\/script>'),e}static displayUrl(t,e=null,n=null,r=null){return __awaiter(this,void 0,void 0,(function*(){return new Promise(((o,s)=>{HotStaq.onReady((()=>__awaiter(this,void 0,void 0,(function*(){let s={url:""};s.name=null==e?"string"==typeof t?t:t.name:e,""===s.name&&(s.name="string"==typeof t?t:t.name),"string"==typeof t?s.url=t:(s.url=t.url,null==n&&null!=t.processor&&(n=t.processor),null==r&&null!=t.args&&(r=t.args),null!=t.testerMap&&(s.testerMap=t.testerMap),null!=t.testerName&&(s.testerName=t.testerName),null!=t.testerAPIBaseUrl&&(s.testerAPIBaseUrl=t.testerAPIBaseUrl)),null==n&&(n=new HotStaq),HotStaq.setupTesters(n,s),s.processor=n,s.args=r,s.url.indexOf("hstqserve")<0&&(s.url+="?hstqserve=nahfam");let i=yield HotStaq.processUrl(s);i+=HotStaq.setupClientTesters(n),yield HotStaq.useOutput(i),o(n)}))))}))}))}static displayContent(t,e=null,n=null,r=null){return __awaiter(this,void 0,void 0,(function*(){return new Promise(((o,s)=>{HotStaq.onReady((()=>__awaiter(this,void 0,void 0,(function*(){let s={content:""};s.name=null==e?"string"==typeof t?"":t.name:e,""===s.name&&(s.name="string"==typeof t?"":t.name),"string"==typeof t?s.content=t:(s.content=t.content,null==n&&null!=t.processor&&(n=t.processor),null==r&&null!=t.args&&(r=t.args),null!=t.testerMap&&(s.testerMap=t.testerMap),null!=t.testerName&&(s.testerName=t.testerName),null!=t.testerAPIBaseUrl&&(s.testerAPIBaseUrl=t.testerAPIBaseUrl)),null==n&&(n=new HotStaq),HotStaq.setupTesters(n,s),s.processor=n,s.args=r;let i=yield HotStaq.processContent(s);yield HotStaq.useOutput(i),o(n)}))))}))}))}}if(exports.HotStaq=HotStaq,HotStaq.version="0.6.23",HotStaq.isWeb=!1,HotStaq.isReadyForTesting=!1,HotStaq.onReadyForTesting=null,HotStaq.errors={},"undefined"!=typeof document){let loadHotStaqSite=function(){let hotstaqElms=document.getElementsByTagName("hotstaq");if(HotStaq.isWeb=!0,void 0!==HotStaqWeb&&(window.HotStaq=HotStaqWeb.HotStaq,window.HotClient=HotStaqWeb.HotClient,window.HotAPI=HotStaqWeb.HotAPI,window.Hot=HotStaqWeb.Hot,window.HotComponent=HotStaqWeb.HotComponent),hotstaqElms.length>0){let hotstaqElm=hotstaqElms[0];setTimeout((function(){return __awaiter(this,void 0,void 0,(function*(){let getAttr=(t,e)=>{for(let n=0;n<e.length;n++){let r=e[n];if(null!=t.getAttribute(r))return t.getAttribute(r);if(null!=t.getAttribute(`data-${r}`))return t.getAttribute(`data-${r}`)}},loadPage=getAttr(hotstaqElm,["load-page","loadPage","src"])||"",router=getAttr(hotstaqElm,["router"])||"",name=getAttr(hotstaqElm,["name"])||"default",args=getAttr(hotstaqElm,["args"])||null,apiLibrary=getAttr(hotstaqElm,["api-library","apiLibrary"])||null,apiName=getAttr(hotstaqElm,["api-name","apiName"])||null,apiUrl=getAttr(hotstaqElm,["api-url","apiUrl"])||null,testerName=getAttr(hotstaqElm,["tester-name","testerName"])||"HotTesterMochaSelenium",testerMap=getAttr(hotstaqElm,["tester-map","testerMap"])||null,testerApiBaseUrl=getAttr(hotstaqElm,["tester-api-base-url","testerApiBaseUrl"])||null,testerLaunchpadUrl=getAttr(hotstaqElm,["tester-launchpad-url","testerLaunchpadUrl"])||null,dontReuseProcessor=!1,passRawUrl=!1,htmlSource=hotstaqElm.innerHTML||"",routerManager={},routerWildcards=[],search=new URLSearchParams(window.location.search);null!=getAttr(hotstaqElm,["src"])&&(loadPage=getAttr(hotstaqElm,["src"])),null!=getAttr(hotstaqElm,["passRawUrl"])&&(passRawUrl=!0),null!=getAttr(hotstaqElm,["dont-reuse-processor","dontReuseProcessor"])&&(dontReuseProcessor=!0);let hstqbaseredirect=search.get("hstqbaseredirect");null!=hstqbaseredirect&&(hstqbaseredirect=decodeURI(hstqbaseredirect),window.history.replaceState("","",hstqbaseredirect),loadPage=hstqbaseredirect);let hotstaqErrors=document.getElementsByTagName("hotstaq-error");for(let t=0;t<hotstaqErrors.length;t++){let e=hotstaqErrors[t],n=getAttr(e,["status"]),r=getAttr(e,["unsupported-browser-redirect"]);null!=r?HotStaq.errors.unsupportedBrowser={redirectToUrl:r}:HotStaq.errors[`${n}`]={redirectToUrl:r}}try{eval("async () => {}")}catch(t){HotStaq.executeError("unsupportedBrowser")}if(""!==router){let t=document.getElementsByTagName("hotstaq-router");for(let e=0;e<t.length;e++){let n=t[e],r=getAttr(n,["name"]),o=getAttr(n,["serve-local","serveLocally"]);if(r===router){for(let t=0;t<n.childNodes.length;t++){let e=n.childNodes[t];if(e instanceof HTMLElement&&"ROUTE"===e.tagName.toUpperCase()){let t=getAttr(e,["path"]),n=getAttr(e,["redirect"]),r=getAttr(e,["base-redirect","baseRedirect"]),o=getAttr(e,["base"]),s=getAttr(e,["src"]);t.indexOf("*")>-1&&routerWildcards.push(t),routerManager[t]={redirect:n||void 0,baseRedirect:r||void 0,base:o||void 0,src:s||void 0}}}let t=window.location.pathname,e=window.location.pathname;if(null!=o){const n=o.toLowerCase();if("true"===n||"yes"===n||"1"===n){const n=t.lastIndexOf("/");n>-1&&(t=t.substring(n),e=e.substring(n))}}if(routerWildcards.length>0)for(let e=0;e<routerWildcards.length;e++){let n=routerWildcards[e],r=n.replace("*","");if(t.indexOf(r)>-1){t=n;break}}if(null!=routerManager[t]){if(null!=routerManager[t].redirect)return void(window.location.href=routerManager[t].redirect);if(null!=routerManager[t].baseRedirect)return void(window.location.href=`${routerManager[t].baseRedirect}?hstqbaseredirect=${encodeURI(e)}`);null!=routerManager[t].src&&(loadPage=routerManager[t].src)}break}}}args=null!=args?JSON.parse(args):Hot_1.Hot.Arguments;let hasHtmlSource=!1;if(""!==htmlSource){const t=htmlSource.replace(/\s/g,"");""!==t&&(hasHtmlSource=!0)}let tempMode=0;null!=window.Hot&&(tempMode=Hot_1.Hot.Mode);let processor=null;!1===dontReuseProcessor&&void 0!==Hot_1.Hot&&null!=Hot_1.Hot.CurrentPage&&null!=Hot_1.Hot.CurrentPage.processor&&(processor=Hot_1.Hot.CurrentPage.processor),null==processor&&(processor=new HotStaq),processor.mode=tempMode;let options={name,processor,args};if(""!==loadPage&&(!1===passRawUrl&&loadPage.indexOf("hstqserve")<0&&(loadPage+="?hstqserve=nahfam"),options.url=loadPage),null!=testerMap&&(options.testerMap=testerMap,options.testerName=testerName),null!=testerName&&(options.testerName=testerName),null!=testerApiBaseUrl&&(options.testerAPIBaseUrl=testerApiBaseUrl),null!=testerLaunchpadUrl&&(options.testerLaunchpadUrl=testerLaunchpadUrl),null!=apiName){let t=new HotClient_1.HotClient(processor);if(""===apiUrl)throw new Error("api-url was not set!");let e=window;null!=apiLibrary&&(e=window[apiLibrary]);let n=new e[apiName](apiUrl,t);n.connection.api=n,processor.api=n}if(!1===hasHtmlSource){if(""===loadPage)throw new Error("The hotstaq tag must have a src, HTML contents inside it, or a router set.");HotStaq.displayUrl(options)}else HotStaq.displayContent(options)}))}),50)}};window.ethereum22=window.ethereum,window.addEventListener("load",loadHotStaqSite)}},628:function(t,e,n){"use strict";var r=this&&this.__awaiter||function(t,e,n,r){return new(n||(n=Promise))((function(o,s){function i(t){try{a(r.next(t))}catch(t){s(t)}}function l(t){try{a(r.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?o(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((r=r.apply(t,e||[])).next())}))};Object.defineProperty(e,"__esModule",{value:!0}),e.HotTestDriver=void 0;const o=n(243),s=n(967);e.HotTestDriver=class{constructor(t,e=null){this.processor=t,this.page=e,this.commandDelay=20,this.persistentData={}}parseTestObject(t){let e=t.indexOf("*"),n="";e>-1&&(t=t.replace(/\*/,""),n="*");let r=`[data-test-object-name${n}='${t}']`;return e=t.indexOf(">"),e>-1&&(r=t=t.replace(/\>/,"")),r}wait(t){return r(this,void 0,void 0,(function*(){return yield new Promise(((e,n)=>{setTimeout((()=>{e()}),t)}))}))}print(t){return r(this,void 0,void 0,(function*(){process.stdout.write(t)}))}println(t){return r(this,void 0,void 0,(function*(){yield this.print(`${t}\n`)}))}assert(t,e=""){return r(this,void 0,void 0,(function*(){if(!t)throw new Error(e)}))}run(t){return r(this,void 0,void 0,(function*(){let e=[];for(let n=0;n<t.length;n++){let r=t[n],i=null,l="",a="";if("string"==typeof r){if(i=this.page.testElements[r],null==i)throw new Error(`HotTestDriver: Unable to find test element ${r}`);l=i.func,a=i.value}if(r instanceof Array){let t=r[0];i=this.page.testElements[t],null==i?(i=new s.HotTestElement(t),l=r[1],a=r[2]):(l=i.func,a=i.value,r.length>1&&(l=r[1]),r.length>2&&(a=r[2]))}i.func=l,i.value=a;let u=yield this.runCommand(i);yield o.HotStaq.wait(this.commandDelay),e.push(u)}return e}))}}},967:(t,e)=>{"use strict";Object.defineProperty(e,"__esModule",{value:!0}),e.HotTestElement=e.HotTestElementOptions=void 0,e.HotTestElementOptions=class{constructor(t={}){this.mustBeVisible=t.mustBeVisible||!0,this.ignoreMissingElementError=t.ignoreMissingElementError||!1}},e.HotTestElement=class{constructor(t,e="",n=null){"string"==typeof t?(this.name=t,this.func=e,this.value=n):(this.name=t.name,this.func=t.func||e,this.value=t.value||n)}}},985:(t,e)=>{"use strict";Object.defineProperty(e,"__esModule",{value:!0}),e.HotTestMap=e.HotTestDestination=void 0;class n{constructor(t="",e=!0){"string"==typeof t?(this.destination=t,this.autoStart=e):t instanceof n?(this.destination=t.destination,this.autoStart=t.autoStart):(this.destination=t.path,this.autoStart=t.autoStart)}}e.HotTestDestination=n,e.HotTestMap=class{constructor(t=[],e={},r=[]){if(t instanceof Array){this.destinations=[];for(let e=0;e<t.length;e++){let r=t[e];this.destinations.push(new n(r))}}else{this.destinations={};for(let e in t){let r=t[e];this.destinations[e]=new n(r)}}this.destinationOrder=r,this.pages=e}}},677:function(t,e,n){"use strict";var r=this&&this.__awaiter||function(t,e,n,r){return new(n||(n=Promise))((function(o,s){function i(t){try{a(r.next(t))}catch(t){s(t)}}function l(t){try{a(r.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?o(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((r=r.apply(t,e||[])).next())}))};Object.defineProperty(e,"__esModule",{value:!0}),e.HotTester=void 0;const o=n(243);class s{constructor(t,e,n,r,o={}){this.processor=t,this.name=e,this.baseUrl=n,this.testMaps=o,this.driver=r,this.finishedLoading=!1,this.hasBeenSetup=!1,this.hasBeenDestroyed=!1}waitForData(){return r(this,void 0,void 0,(function*(){for(;!1===this.finishedLoading;)yield o.HotStaq.wait(10)}))}getTestPage(t){return this.testMaps[t.mapName].pages[t.page]}getTestPath(t,e){return this.testMaps[t.mapName].pages[t.page].testPaths[e]}static interpretDestination(t,e){let n={mapName:t,page:"",api:"",paths:[]},r=e.destination.split(/\-\>/g),o=r[0];if(o.length<2)return null;if("/"===o[0]&&"/"===o[1])return null;let s=(t,e)=>{let n=t.indexOf(e),r="";return n>-1&&(r=t.substr(n+e.length),r=r.trim()),r};n.page=s(o,"page:"),n.api=s(o,"api:");for(let t=1;t<r.length;t++){let e=r[t],o={cmd:"",dest:"",path:""};e=e.trim(),o.dest=s(e,"dest:"),o.cmd=s(e,"cmd:"),o.path=s(e,"path:"),""==o.dest&&""==o.cmd&&""==o.path&&(o.path=e),n.paths.push(o)}return n}executeTestAPIPath(t,e,n,o=!1,s=!1){return r(this,void 0,void 0,(function*(){let r=!0;if(null==e)throw new Error(`Trying to access null method on destination map ${t.mapName}.`);!1===o&&null!=this.onTestAPIPathStart&&(r=yield this.onTestAPIPathStart(t,e,n,s));let i=null;if(!0===r){let t=e.testCases[n];if(null==t)throw new Error(`HotTester: Test case object ${n} does not exist!`);i=yield t.func(this.driver)}return!1===o&&null!=this.onTestAPIPathEnd&&(yield this.onTestAPIPathEnd(t,e,n,i,s)),i}))}executeTestAPIPaths(t){return r(this,void 0,void 0,(function*(){let e=[];if(null==this.testMaps[t.mapName])throw new Error(`HotTester: Map ${t.mapName} does not exist!`);if(null==this.processor.api)throw new Error("HotTester: Associated processor does not have an API!");let n=this.processor.api.routes[t.api];if(null==n)throw new Error(`HotTester: API does not have route ${t.api}!`);for(let r=0;r<t.paths.length;r+=2){let o=t.paths[r].path,s=n.getMethod(o),i=t.paths[r+1].path;if(null==s)throw new Error(`Unable to find method related to path ${o} in map ${t.mapName}`);let l=yield this.executeTestAPIPath(t,s,i);e.push(l)}return e}))}executeTestPagePath(t,e,n=!1,o=!1){return r(this,void 0,void 0,(function*(){let r=!0,s=this.testMaps[t.mapName];if(null==s)throw new Error(`HotTester: Map ${t.mapName} does not exist!`);let i=s.pages[t.page];if(null==i)throw new Error(`HotTester: Page ${t.page} does not exist!`);this.driver.page=i;let l=e.path,a=i.testPaths[l];!1===n&&null!=this.onTestPagePathStart&&(r=yield this.onTestPagePathStart(t,i,e,o));let u=null;if(!0===r){if(null==a)throw new Error(`HotTester: Test path ${l} does not have a function!`);u=yield a(this.driver)}return!1===n&&null!=this.onTestPagePathEnd&&(yield this.onTestPagePathEnd(t,a,u,o)),u}))}executeCommand(t,e,n,s){return r(this,void 0,void 0,(function*(){let i=(t,e,r)=>{let o=!1;n.cmd===e&&(o=!0);const s=n.cmd.indexOf("(");return s>-1&&n.cmd.substr(0,s)===e&&(o=!0),o},l=t=>{let e=[],n=t.match(/(?=\()(.*?)(?=\))/g);if(null!=n){let t=n[0];t=t.substr(2,t.length),e.push(t)}if(e.length<1)throw new Error(`HotTester: Command ${t} requires arguments, but none were supplied.`);return e},a=null,u=[];if(!0===i(n.cmd,"waitForTesterAPIData")&&(a=t=>r(this,void 0,void 0,(function*(){this.finishedLoading=!1,yield this.waitForData()}))),!0===i(n.cmd,"wait")&&(u=l(n.cmd),a=t=>r(this,void 0,void 0,(function*(){let e=parseInt(t[0]);yield o.HotStaq.wait(e)}))),!0===i(n.cmd,"url")&&(u=l(n.cmd),a=t=>r(this,void 0,void 0,(function*(){let e=t[0];yield this.driver.navigateToUrl(e)}))),!0===i(n.cmd,"print")&&(u=l(n.cmd),a=t=>r(this,void 0,void 0,(function*(){let e=t[0];yield this.driver.print(e)}))),!0===i(n.cmd,"println")&&(u=l(n.cmd),a=t=>r(this,void 0,void 0,(function*(){let e=t[0];yield this.driver.println(e)}))),!0===i(n.cmd,"waitForTestObject")&&(u=l(n.cmd),a=t=>r(this,void 0,void 0,(function*(){let e=JSON.parse(t[0]);yield this.driver.waitForTestElement(e)}))),null==a)throw new Error(`HotTester: Command ${n.cmd} does not exist!`);yield this.onCommand(t,e,n,s,u,a)}))}executeTestPagePaths(t,e=!1){return r(this,void 0,void 0,(function*(){let n=[],r=this.testMaps[t.mapName];if(null==r)throw new Error(`HotTester: Map ${t.mapName} does not exist!`);for(let o=0;o<t.paths.length;o++){let i=t.paths[o],l=null,a=r.pages[t.page];if(null==a)throw new Error(`HotTester: Page ${t.page} does not exist!`);if(""!==i.dest){if(r.destinations instanceof Array)throw new Error(`HotTester: When using type 'dest' in a destination string, all destinations in map ${t.mapName} must be named.`);let e=r.destinations[i.dest],n=s.interpretDestination(t.mapName,e);null!=n&&(l=yield this.executeTestPagePaths(n))}""!==i.cmd&&(yield this.executeCommand(t,a,i,i.cmd)),""!==i.path&&(l=yield this.executeTestPagePath(t,i,!1,e)),n.push(l)}return n}))}execute(t){return r(this,void 0,void 0,(function*(){let e=this.testMaps[t];if(null==e)throw new Error(`HotTester: Map ${t} does not exist!`);let n=this.processor.getRouteKeyFromName(t),o="";""!==n&&(o=`${this.baseUrl}${n}`);let i=(e,n="")=>r(this,void 0,void 0,(function*(){if(!1===e.autoStart)return;let r=s.interpretDestination(t,e),i=!1,l=!0;""!==r.page&&(i=!0),null!=this.setup&&!1===this.hasBeenSetup&&(yield this.setup(i,o,n),this.hasBeenSetup=!0,this.hasBeenDestroyed=!1),null!=this.onTestStart&&(l=yield this.onTestStart(r,o,n)),!0===l&&(""!==r.page&&(yield this.executeTestPagePaths(r)),""!==r.api&&(yield this.executeTestAPIPaths(r))),null!=this.onTestEnd&&(yield this.onTestEnd(r)),null!=this.destroy&&!1===this.hasBeenDestroyed&&(yield this.destroy(),this.hasBeenDestroyed=!0,this.hasBeenSetup=!1)}));if(e.destinations instanceof Array)for(let t=0;t<e.destinations.length;t++){let n=e.destinations[t];yield i(n)}else if(e.destinationOrder.length>0){let t=[];for(let n=0;n<e.destinationOrder.length;n++){let r=e.destinationOrder[n],o=e.destinations[r];if(null==o)throw new Error(`HotTester: Destination ${r} does not exist!`);t.push(r),yield i(o,r)}for(let n in e.destinations){let r=!0;for(let e=0;e<t.length;e++)if(t[e]===n){r=!1;break}if(!0===r){let t=e.destinations[n];yield i(t,n)}}}else for(let t in e.destinations){let n=e.destinations[t];yield i(n,t)}}))}}e.HotTester=s},414:function(__unused_webpack_module,exports,__webpack_require__){"use strict";var __awaiter=this&&this.__awaiter||function(t,e,n,r){return new(n||(n=Promise))((function(o,s){function i(t){try{a(r.next(t))}catch(t){s(t)}}function l(t){try{a(r.throw(t))}catch(t){s(t)}}function a(t){var e;t.done?o(t.value):(e=t.value,e instanceof n?e:new n((function(t){t(e)}))).then(i,l)}a((r=r.apply(t,e||[])).next())}))};Object.defineProperty(exports,"__esModule",{value:!0}),exports.HotTesterAPI=void 0;const HotAPI_1=__webpack_require__(233),HotRoute_1=__webpack_require__(761),HotRouteMethod_1=__webpack_require__(70);class HotTesterAPI extends HotAPI_1.HotAPI{constructor(t,e=null,n=null){super(t,e,n),this.executeEventsUsing=HotAPI_1.EventExecutionType.HotAPI;let r=new HotRoute_1.HotRoute(e,"tester");r.addMethod({name:"pageLoaded",onServerExecute:this.pageLoaded,parameters:{testerName:{required:!0,type:"string",description:"The name of the tester executing the test."},testerMap:{required:!0,type:"string",description:"The tester map executing the test."},pageName:{required:!0,type:"string",description:"The name of the page executing the test."},testElements:{required:!0,type:"array",description:"The test elements on the page."},testPaths:{required:!0,type:"array",description:"The test paths on the page."}},returns:"Returns true as an acknowledgement."}),r.addMethod({name:"executeTests",onServerExecute:this.executeTests,parameters:{testerName:{required:!0,type:"string",description:"The name of the tester executing the test."},testerMap:{required:!0,type:"object",description:"The tester map to execute."}},returns:"Returns true when tests are complete."}),r.addMethod({name:"heartbeat",type:HotRouteMethod_1.HTTPMethod.GET,onServerExecute:this.heartbeat,returns:"Returns true as an acknowledgement."}),this.addRoute(r)}pageLoaded(req,res,authorizedValue,jsonObj,queryObj){return __awaiter(this,void 0,void 0,(function*(){let testerObj={testerName:jsonObj.testerName,testerMap:jsonObj.testerMap,pageName:jsonObj.pageName,testElements:jsonObj.testElements,testPathsStrs:jsonObj.testPaths};for(let t in testerObj){let e=testerObj[t],n=!1;if(null==e&&(n=!0),""!=testerObj.testerName&&""!==testerObj.testerMap&&""!==testerObj.testElements&&""!==testerObj.testPathsStrs||(n=!0),!0===n)throw new Error(`TesterAPI: Object ${t} was not passed.`)}testerObj.testElements=JSON.parse(testerObj.testElements),testerObj.testPathsStrs=JSON.parse(testerObj.testPathsStrs);let testPaths={};for(let key in testerObj.testPathsStrs){let testPath=eval(testerObj.testPathsStrs[key]);testPaths[key]=testPath}let tester=this.connection.processor.testers[testerObj.testerName];if(null==tester)throw new Error(`TesterAPI: Tester ${testerObj.testerMap} does not exist!`);let testMap=tester.testMaps[testerObj.testerMap];if(null==testMap)throw new Error(`TesterAPI: Tester map ${testerObj.testerMap} does not exist!`);return testMap.pages[testerObj.pageName]={testElements:{},testPaths:{}},testMap.pages[testerObj.pageName].testElements=testerObj.testElements,testMap.pages[testerObj.pageName].testPaths=testPaths,tester.finishedLoading=!0,null!=tester.onFinishedLoading&&(yield tester.onFinishedLoading()),!0}))}executeTests(t,e,n,r,o){return __awaiter(this,void 0,void 0,(function*(){let t=r.testerName,e=r.testerMap;if(null==t||null==e)throw new Error("TesterAPI: Not all required json objects were passed.");if(""===t||""===e)throw new Error("TesterAPI: Not all required json objects were passed.");let n=this.connection;return null!=n.executeTests&&(yield n.executeTests(t,e)),!0}))}heartbeat(t,e,n,r,o){return __awaiter(this,void 0,void 0,(function*(){return!0}))}}exports.HotTesterAPI=HotTesterAPI},859:(t,e,n)=>{"use strict";Object.defineProperty(e,"__esModule",{value:!0});const r=n(243),o=n(507),s=n(260),i=n(732),l=n(802),a=n(136),u=n(233),h=n(761),c=n(70),p=n(810),d=n(957),m=n(628),f=n(967),g=n(677),_=n(414),H=n(985);r.HotStaq.isWeb=!0,t.exports.HotStaq=r.HotStaq,t.exports.Hot=o.Hot,t.exports.DeveloperMode=o.DeveloperMode,t.exports.HotComponent=s.HotComponent,t.exports.HotAPI=u.HotAPI,t.exports.EventExecutionType=u.EventExecutionType,t.exports.HotFile=i.HotFile,t.exports.HotLog=l.HotLog,t.exports.HotLogLevel=l.HotLogLevel,t.exports.HotPage=a.HotPage,t.exports.HotRoute=h.HotRoute,t.exports.HotRouteMethod=c.HotRouteMethod,t.exports.HTTPMethod=c.HTTPMethod,t.exports.HotServer=p.HotServer,t.exports.HotServerType=p.HotServerType,t.exports.HotClient=d.HotClient,t.exports.HotTester=g.HotTester,t.exports.HotTesterAPI=_.HotTesterAPI,t.exports.HotTestMap=H.HotTestMap,t.exports.HotTestDestination=H.HotTestDestination,t.exports.HotTestElement=f.HotTestElement,t.exports.HotTestElementOptions=f.HotTestElementOptions,t.exports.HotTestDriver=m.HotTestDriver},651:t=>{"use strict";t.exports={}},351:()=>{},606:()=>{}},__webpack_module_cache__={};function __webpack_require__(t){var e=__webpack_module_cache__[t];if(void 0!==e)return e.exports;var n=__webpack_module_cache__[t]={exports:{}};return __webpack_modules__[t].call(n.exports,n,n.exports,__webpack_require__),n.exports}var __webpack_exports__=__webpack_require__(859);HotStaqWeb=__webpack_exports__})();
2
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"HotStaq.js","mappings":"sDACAA,EAAOC,QAAyB,iBAARC,KAAmBA,KAAKC,SAAWC,OAAOD,Q,kBCEDH,EAAOC,QAOhE,WAAe,aAGrB,SAASI,EAAQC,GACf,IAAK,IAAIC,EAAI,EAAGA,EAAIC,UAAUC,OAAQF,IAAK,CACzC,IAAIG,EAASF,UAAUD,GACvB,IAAK,IAAII,KAAOD,EACdJ,EAAOK,GAAOD,EAAOC,EAEzB,CACA,OAAOL,CACT,CA2HA,OArGA,SAASM,EAAMC,EAAWC,GACxB,SAASC,EAAKJ,EAAKK,EAAOC,GACxB,GAAwB,oBAAbC,SAAX,CAMkC,iBAFlCD,EAAaZ,EAAO,CAAC,EAAGS,EAAmBG,IAErBE,UACpBF,EAAWE,QAAU,IAAIC,KAAKA,KAAKC,MAA6B,MAArBJ,EAAWE,UAEpDF,EAAWE,UACbF,EAAWE,QAAUF,EAAWE,QAAQG,eAG1CX,EAAMY,mBAAmBZ,GACtBa,QAAQ,uBAAwBC,oBAChCD,QAAQ,QAASE,QAEpB,IAAIC,EAAwB,GAC5B,IAAK,IAAIC,KAAiBX,EACnBA,EAAWW,KAIhBD,GAAyB,KAAOC,GAEE,IAA9BX,EAAWW,KAWfD,GAAyB,IAAMV,EAAWW,GAAeC,MAAM,KAAK,KAGtE,OAAQX,SAASY,OACfnB,EAAM,IAAME,EAAUkB,MAAMf,EAAOL,GAAOgB,CAtC5C,CAuCF,CA4BA,OAAOK,OAAOC,OACZ,CACElB,IAAKA,EACLmB,IA7BJ,SAAcvB,GACZ,GAAwB,oBAAbO,YAA6BV,UAAUC,QAAWE,GAA7D,CAQA,IAFA,IAAIwB,EAAUjB,SAASY,OAASZ,SAASY,OAAOD,MAAM,MAAQ,GAC1DO,EAAM,CAAC,EACF7B,EAAI,EAAGA,EAAI4B,EAAQ1B,OAAQF,IAAK,CACvC,IAAI8B,EAAQF,EAAQ5B,GAAGsB,MAAM,KACzBb,EAAQqB,EAAMC,MAAM,GAAGC,KAAK,KAEhC,IACE,IAAIC,EAAWf,mBAAmBY,EAAM,IAGxC,GAFAD,EAAII,GAAY3B,EAAU4B,KAAKzB,EAAOwB,GAElC7B,IAAQ6B,EACV,KAES,CAAX,MAAOE,GAAI,CACf,CAEA,OAAO/B,EAAMyB,EAAIzB,GAAOyB,CApBxB,CAqBF,EAMIO,OAAQ,SAAUhC,EAAKM,GACrBF,EACEJ,EACA,GACAN,EAAO,CAAC,EAAGY,EAAY,CACrBE,SAAU,IAGhB,EACAyB,eAAgB,SAAU3B,GACxB,OAAOL,EAAKiC,KAAKhC,UAAWR,EAAO,CAAC,EAAGwC,KAAK5B,WAAYA,GAC1D,EACA6B,cAAe,SAAUjC,GACvB,OAAOD,EAAKP,EAAO,CAAC,EAAGwC,KAAKhC,UAAWA,GAAYgC,KAAK5B,WAC1D,GAEF,CACEA,WAAY,CAAED,MAAOgB,OAAOe,OAAOjC,IACnCD,UAAW,CAAEG,MAAOgB,OAAOe,OAAOlC,KAGxC,CAEUD,CApHa,CACrB6B,KAAM,SAAUzB,GAId,MAHiB,MAAbA,EAAM,KACRA,EAAQA,EAAMsB,MAAM,GAAI,IAEnBtB,EAAMQ,QAAQ,mBAAoBC,mBAC3C,EACAM,MAAO,SAAUf,GACf,OAAOO,mBAAmBP,GAAOQ,QAC/B,2CACAC,mBAEJ,GAwG+B,CAAEuB,KAAM,KAK1C,CA/IiFC,E,2BCAlF,IAUIC,EAVY,WAIf,GAAoB,oBAAThD,KAAwB,OAAOA,KAC1C,GAAsB,oBAAXE,OAA0B,OAAOA,OAC5C,QAAsB,IAAX8C,EAA0B,OAAOA,EAC5C,MAAM,IAAIC,MAAM,iCACjB,CAEaC,GAEbpD,EAAOC,QAAUA,EAAUiD,EAAOG,MAG9BH,EAAOG,QACVpD,EAAA,QAAkBiD,EAAOG,MAAMC,KAAKJ,IAGrCjD,EAAQsD,QAAUL,EAAOK,QACzBtD,EAAQuD,QAAUN,EAAOM,QACzBvD,EAAQwD,SAAWP,EAAOO,Q,ulBCxB1B,yCAEA,mCAEA,0CAEA,sDACA,uDAKA,IAAYC,eAAZ,SAAYA,GAMX,+BAKA,gCACA,CAZD,CAAYA,cAAA,QAAAA,gBAAA,QAAAA,cAAa,KAgCzB,MAAaC,IAmFZC,eAAsBC,EAAwBC,EAAc,M,iDAE3D,IAAsB,IAAlB,UAAAC,QAAQC,OAEW,iBAAX,EACX,CACC,MAAMC,EAAoBJ,EAAKK,cAI3BD,EAAUE,QAAS,UAAY,GAE9BF,EAAUE,QAAS,UAAY,IAClCN,GAAQ,oB,CAKZF,IAAIS,WAAYT,IAAIU,QAASR,EAAMC,GACpC,G,CAKAF,iBAAwBC,M,iDAEvB,MAAMS,UAAYX,IAAIY,YAAaV,MAC7BW,aAAuBF,IAAIG,QAEX,IAAlB,UAAAV,QAAQC,MACXU,KAAKC,MAAOvE,OAAQ,CAACoE,SAErBE,KAAMF,OACR,G,CAKAZ,eAAsBgB,EAAkBd,EAAc,M,iDAErD,IAGIe,EAHgBlB,IAAImB,YAAYC,UAAUV,QAASO,GAKvDC,EAASG,KAAOnC,KAAKiC,YACrB,IAAIG,QAAwBJ,EAASK,QAASpB,GAE9CH,IAAIS,KAAMa,EACX,G,CAKArB,eAAsBZ,EAAwBc,EAAc,M,iDAE3D,IAAIe,EAAoB,KAmBxB,MAjBsB,iBAAX,GAEVA,EAAW,IAAI,UAAAM,SAEO,IAAlB,UAAApB,QAAQC,MACXa,EAASO,IAAMpC,EAEf6B,EAASQ,UAAYrC,GAGtB6B,EAAW7B,QAEN6B,EAASS,OAEfT,EAASG,KAAOnC,KAAKiC,kBACOD,EAASK,QAASpB,EAG/C,G,CAKAF,eAAsB2B,EAAeC,EAAY,KAChDC,EAAqB,OACrBC,EAAiC,CAAC,G,iDAElC,IAAIC,EAAc,KAElB,GAAuB,MAAnBhC,IAAImB,YACP,MAAM,IAAI3B,MAAO,yBAElB,GAAiC,MAA7BQ,IAAImB,YAAYC,UACnB,MAAM,IAAI5B,MAAO,qCAElB,GAAqC,MAAjCQ,IAAImB,YAAYC,UAAUa,IAC7B,MAAM,IAAIzC,MAAO,oFAQlB,OANqC,MAAjCQ,IAAImB,YAAYC,UAAUa,MAE7BD,QAAehC,IAAImB,YAAYC,UAAUa,IAAIC,SAAUN,EACnDC,EAAMC,EAAYC,IAGhB,CACR,G,CAWA9B,mBAA0BwB,EAAaI,EAAY,KAAMC,EAAqB,Q,iDAE7E,IAEC,IAAIK,EAAW,CACd,OAAUL,EACV,QAAW,CACT,OAAU,mBACV,eAAgB,qBAIA,SAAfA,IAGHK,EAAe,KAAIC,KAAKC,UAAWR,IAGpC,IAAIlB,QAAY,wBAAOc,EAAKU,GAE5B,IAAe,IAAXxB,EAAI2B,GACP,MAAM,IAAI9C,MAAO,GAAGmB,EAAI4B,WAAW5B,EAAI6B,cAIxC,aAFwB7B,EAAI8B,M,CAI7B,MAAOC,GAEN,OAAQN,KAAKC,UAAW,CAAE,MAAS,GAAGK,EAAGC,6BAA6BlB,K,CAExE,G,CAUAxB,mBAA0BwB,EAAamB,G,iDAItC,aAFgB,wBAAOnB,EAAKmB,EAG7B,G,CAKA3C,YAAa0C,GAEZ3C,IAAI6C,QAAUF,CACf,CAKA1C,oBAEC,IAAK,IAAI6C,EAAO,EAAGA,EAAO9C,IAAI+C,IAAIjG,OAAQgG,IAC1C,CACC,IAAIE,EAAkBhD,IAAI+C,IAAID,GAC1BG,EAAiBjD,IAAIkD,OAEzBD,EAASA,EAAOpF,QAAS,gBAAiBmF,GAE1ChD,IAAIS,KAAMwC,E,CAEZ,CAKAhD,wBAEC,IAAK,IAAI6C,EAAO,EAAGA,EAAO9C,IAAImD,QAAQrG,OAAQgG,IAC9C,CACC,IAAIM,EAAiBpD,IAAImD,QAAQL,GAC7BO,EAAoBrD,IAAIsD,UAE5BD,EAAYA,EAAUxF,QAAS,eAAgBuF,GAE/CpD,IAAIS,KAAM4C,E,CAEZ,CAKApD,0BAEC,IAAK,IAAI6C,EAAO,EAAGA,EAAO9C,IAAIuD,UAAUzG,OAAQgG,IAChD,CACC,IAAIU,EAAmBxD,IAAIuD,UAAUT,GACjCW,EAAsBzD,IAAI0D,aAE9BD,EAAcA,EAAY5F,QAAS,eAAgB2F,GAEnDxD,IAAIS,KAAMgD,E,CAEZ,EA3SD,gBAKQ,IAAAtC,YAAuB,KAIvB,IAAAwC,UAAiB,KAKjB,IAAA5D,cAAgBA,cAKhBC,IAAA4D,eAAiB,iBAAAA,eAKjB,IAAAC,KAAsB9D,cAAc+D,WAIpC,IAAAC,IAAc,KAId,IAAAC,UAAoB,KAIpB,IAAAnB,OAAiB,GAIjB,IAAAoB,KAAY,CAAC,EAIb,IAAAC,QAAiC,oBAIjC,IAAAC,WAAkB,CAAC,EAInB,IAAAjB,OAAiB,kDAOjB,IAAAH,IAAgB,GAOhB,IAAAI,QAAiB,GAIjB,IAAAI,UAAmB,GAInB,IAAAD,UAAoB,gEAIpB,IAAAI,aAAuB,sD,ygBC1H/B,kBACA,YAEA,SACA,SAEA,QAgBA,IAAYU,GAAZ,SAAYA,GAEX,2BACA,6BACA,sBACA,CALD,CAAYA,EAAA,EAAAA,qBAAA,EAAAA,mBAAkB,KAU9B,eAgECC,YAAaC,EAAiBC,EAAoC,KAAMC,EAAY,MAEnFtF,KAAKqF,WAAaA,EAClBrF,KAAKuF,YAAc,GACnBvF,KAAKoF,QAAUA,EACfpF,KAAKwF,iBAAkB,EACvBxF,KAAKyF,mBAAqBP,EAAmBQ,SAC7C1F,KAAKsF,GAAKA,EACVtF,KAAK2F,gBAAkB,KACvB3F,KAAK4F,SAAW,KAChB5F,KAAK6F,OAAS,CAAC,EACf7F,KAAK8F,cAAgB,KACrB9F,KAAK+F,eAAiB,IACvB,CAKAC,YAAaC,GAEZ,GAA2B,MAAvBjG,KAAKqF,WAAWtC,IACnB,MAAM,IAAIzC,MAAO,wBAElB,GAA8B,MAA1BN,KAAKqF,WAAWtC,IAAIuC,GACvB,MAAM,IAAIhF,MAAO,6CAA6CN,KAAKqF,WAAWtC,IAAIqC,WAEnFpF,KAAKqF,WAAWtC,IAAIuC,GAAGW,OAASA,CACjC,CAKAC,QAEC,GAA8B,MAA1BlG,KAAKqF,WAAWtC,IAAIuC,GACvB,MAAM,IAAIhF,MAAO,6CAA6CN,KAAKqF,WAAWtC,IAAIqC,WAEnF,OAAQpF,KAAKqF,WAAWtC,IAAM,EAC/B,CAKAoD,cAEC,GAA8B,MAA1BnG,KAAKqF,WAAWtC,IAAIuC,GACvB,MAAM,IAAIhF,MAAO,6CAA6CN,KAAKqF,WAAWtC,IAAIqC,WAEnF,OAAQpF,KAAKqF,WAAWtC,IAAIuC,GAAS,MACtC,CAyCAc,SACC1D,EACA2D,EAAuC,KACvCC,EAA2G,MAG3G,IAAIC,EAAoB,GA0BxB,GAxBI7D,aAAiB,EAAAgD,UAEpBa,EAAY7D,EAAMA,MAClB1C,KAAK6F,OAAOnD,EAAMA,OAASA,IAI3B6D,EAAY7D,EAEkB,MAA1B1C,KAAK6F,OAAOU,KACfvG,KAAK6F,OAAOU,GAAa,IAAI,EAAAb,SAAU1F,KAAKqF,WAAYkB,IAErDF,aAAuB,EAAAG,eAC1BxG,KAAK6F,OAAOU,GAAWE,UAAWJ,GAGlCrG,KAAK6F,OAAOU,GAAWE,UAAW,IAAI,EAAAD,eACrCxG,KAAK6F,OAAOU,GAAYF,EAAaC,KAIxCtG,KAAK6F,OAAOU,GAAWlB,WAAarF,KAAKqF,YAGZ,IAAzBrF,KAAKwF,gBACT,CAEC,IAAIkB,EAAyC1G,KAAKuG,GAElC,MAAZG,IACHA,EAAW,CAAC,GAEb,IAAK,IAAI9C,EAAO,EAAGA,EAAO5D,KAAK6F,OAAOU,GAAWI,QAAQ/I,OAAQgG,IACjE,CACC,IAAIgD,EAAyB5G,KAAK6F,OAAOU,GACrCM,EAAiC7G,KAAK6F,OAAOU,GAAWI,QAAQ/C,GAmBlE8C,EAASG,EAAeC,MAAQ,CAACnE,EAAWE,KAE1C,IAAID,EAAqBiE,EAAeE,KAEpCC,EAAmB,GAEM,KAAzBJ,EAAaK,UAChBD,GAAY,IAAIJ,EAAaK,WAEH,KAAvBL,EAAalE,QAChBsE,GAAY,IAAIJ,EAAalE,SAEF,KAAxBmE,EAAeC,OAClBE,GAAY,IAAIH,EAAeC,QAEhC,IAAInB,EAAuB,KA+C3B,GAxC4B,MAAxB3F,KAAK2F,kBACRA,EAAkB3F,KAAK2F,iBAKc,MAAlCkB,EAAelB,gBAClBA,EAAkBkB,EAAelB,gBAGW,MAAxCkB,EAAenE,MAAMiD,kBACxBA,EAAkBkB,EAAenE,MAAMiD,iBAGlB,MAAnBA,GAGkB,oBAAV,KAGC,MAAP7E,KAGY,MAAXA,IAAI+D,KAG4B,MAA/B/D,IAAI+D,IAAI+B,EAAalE,QAG2B,MAA/C5B,IAAI+D,IAAI+B,EAAalE,OAAOiD,kBAG/BA,EAAkB7E,IAAI+D,IAAI+B,EAAalE,OAAOiD,iBAQ7B,MAAnBA,EAGH,IAAK,IAAI7H,KAAO6H,EAChB,CACC,IAAIuB,EAAsBvB,EAAgB7H,GAIzB,MAAb6E,EAAK7E,KACR6E,EAAK7E,GAAOoJ,E,CAIf,IAAIjG,EAAc,CAAC+F,EAAUrE,EAAMC,EAAYC,GAE/C,OAAQ7C,KAAKgD,SAASlB,MAAO9B,KAAMiB,EAAM,C,CAO9CjB,KAAKuG,GAAaG,C,CAEpB,CAKMS,cAAezE,G,yCAEhB1C,KAAKqF,sBAAsB,EAAA+B,kBACxBpH,KAAKqF,WAAW8B,cAAezE,GACvC,G,CAKM2E,iB,yCAEL,IAAK,IAAIvJ,KAAOkC,KAAK6F,OACrB,CACC,IAAInD,EAAkB1C,KAAK6F,OAAO/H,SAE5BkC,KAAKmH,cAAezE,E,CAE5B,G,CAKMM,SAAUN,EAAeC,EAAWC,EAAqB,OAC9DC,EAAiC,CAAC,G,yCAElC,IAAIN,EAAcvC,KAAKoF,QAcvB,GAZAxC,EAAaA,EAAW0E,cAEM,MAA1B/E,EAAKA,EAAI3E,OAAS,KACrB2E,EAAMA,EAAIgF,OAAQ,EAAIhF,EAAI3E,OAAS,IAEnB,MAAb8E,EAAM,KACTH,GAAO,KAERA,GAAOG,EAEkBvD,OAAOqI,KAAM3E,GAAOjF,OAE9B,EACf,CACC,GAAmB,SAAfgF,EACH,MAAM,IAAItC,MAAO,yDAElB,MAAMmH,EAAqB,IAAI,UAE/B,IAAK,IAAI3J,KAAO+E,EACf4E,EAASC,OAAQ5J,EAAK+E,EAAM/E,IAE7B,IAAI2D,QAAY,aAAOc,EAAK,CAC1BoF,OAAQ,OAERC,KAAMH,IAEJI,QAAqBpG,EAAI8B,OAc7B,OAZuB,MAAnBZ,EAAc,UACjBA,EAAc,QAAI,CAAC,GAEc,MAA9BA,EAAc,QAAW,UAC5BA,EAAc,QAAW,QAAI,CAAC,GAE/BA,EAAc,QAAW,QAAY,SACnCkF,EAAiB,QAAW,QAAY,eAGhB7H,KAAKgD,SAAUN,EAAOC,EAAMC,E,CAKvD,IAAIK,EAAgB,CAClB0E,OAAQ/E,EACRkF,QAAS,CACP,OAAU,mBACV,eAAgB,qBAwBpB,MApBoB,QAAflF,GACY,SAAfA,IAEDK,EAAe,KAAIC,KAAKC,UAAWR,IAGtB,IAAIoF,SAAS,CAACC,EAASC,MAEnC,aAAO1F,EAAKU,GAAUiF,MAAazG,GAAO,kCAExC,IAAI0G,QAAqB1G,EAAI8B,OAE7ByE,EAASG,EACV,MACCC,OAAQC,IAER,MAAM,IAAI/H,MAAO,GAAGiC,MAAQ8F,EAAO5E,UAAU,GAC5C,GAIN,G,mGCtbD,eAMA,kBAuBC0B,YAAajD,GAEZlC,KAAKkC,UAAYA,EACjBlC,KAAK+C,IAAM,KACX/C,KAAKsI,UAAY,KACjBtI,KAAK+G,KAAO,EAAAwB,cAAcC,KAC1BxI,KAAKyI,OAASvG,EAAUuG,MACzB,E,0aCrCD,eA+HA,qBA4FCtD,YAAauD,EAA+B3F,EAAc,MAEpD2F,aAAgB,EAAAxH,SAAqB,MAARwH,GAGjC1I,KAAKkC,UAAYwG,EACjB1I,KAAK2I,aAAe,GACpB3I,KAAK8G,KAAO,GACZ9G,KAAK4I,IAAM,GACX5I,KAAK+C,IAAM,KACX/C,KAAK6I,oBAAiBC,EACtB9I,KAAK+I,mBAAqB,GAC1B/I,KAAK+G,KAAO,GACZ/G,KAAK7B,MAAQ,KACb6B,KAAKgJ,MAAQ,KACbhJ,KAAKiJ,OAAS,CAAC,IAIfjJ,KAAKkC,UAAYwG,EAAKxG,UACtBlC,KAAK2I,aAAeD,EAAKC,cAAgB,GACzC3I,KAAK8G,KAAO4B,EAAK5B,MAAQ,GACzB9G,KAAK4I,IAAMF,EAAKE,KAAO5I,KAAK8G,KAC5B9G,KAAK+C,IAAM2F,EAAK3F,KAAO,KACvB/C,KAAK6I,eAAiBH,EAAKG,qBAAkBC,EAC7C9I,KAAK+I,mBAAqBL,EAAKK,oBAAsB,GACrD/I,KAAK+G,KAAO2B,EAAK3B,MAAQ,GACzB/G,KAAK7B,MAAQuK,EAAKvK,OAAS,KAC3B6B,KAAKgJ,MAAQN,EAAKM,OAAS,KAC3BhJ,KAAKiJ,OAAS,CAAC,GAGL,MAAPlG,IACH/C,KAAK+C,IAAMA,EACb,CAOMmG,UAAWC,G,yCAEhB,OAAO,CACR,G,ghCCxQD,kBAEA,YAEA,SAqDA,MAAa7G,EA2BZ6C,YAAauD,EAAiB,CAAC,GAE9B1I,KAAKmC,KAAOuG,EAAKvG,MAAQ,KACzBnC,KAAK8G,KAAO4B,EAAK5B,MAAQ,GACzB9G,KAAKuC,IAAMmG,EAAKnG,KAAO,GACvBvC,KAAKwC,UAAYkG,EAAKlG,WAAa,GACnCxC,KAAKoC,QAAUsG,EAAKtG,SAAW,GAC/BpC,KAAKoJ,eAAiBV,EAAKU,iBAAkB,CAC9C,CAKAC,WAAYjH,GAEXpC,KAAKoC,QAAUA,CAChB,CAKAkH,aAEC,OAAQtJ,KAAY,OACrB,CAKAe,eAAsBwB,G,yCAErB,IAEC,IAAId,QAAY,aAAOc,GAEvB,IAAe,IAAXd,EAAI2B,GACP,MAAM,IAAI9C,MAAO,GAAGmB,EAAI4B,WAAW5B,EAAI6B,cAIxC,aAF4B7B,EAAIG,M,CAIjC,MAAO4B,GAEN,OAAQN,KAAKC,UAAW,CAAE,MAAS,GAAGK,EAAGC,6BAA6BlB,K,CAExE,G,CAKMgH,U,yCAIL,OAFAvJ,KAAKoC,cAAgBE,EAAQkH,QAASxJ,KAAKuC,KAEnCvC,KAAY,OACrB,G,CAKMyJ,gB,yCAiBL,OAf+B,IAAI1B,SAClC,CAACC,EAAcC,KAEdyB,EAAGC,SAAU3J,KAAKwC,WAAW,CAACoH,EAA4BjH,KAExD,GAAW,MAAPiH,EACH,MAAMA,EAEP,IAAIxH,EAAkBO,EAAKkH,WAC3B7J,KAAKoC,QAAUA,EAEf4F,EAAShI,KAAKoC,QAAQ,GACrB,GAIN,G,CAKMK,O,yCAEL,IAAIL,EAAkB,GAQtB,MANiB,KAAbpC,KAAKuC,MACRH,QAAgBpC,KAAKuJ,WAEC,KAAnBvJ,KAAKwC,YACRJ,QAAgBpC,KAAKyJ,iBAEf,CACR,G,CAkBA1I,sBAAuBqB,EAAiB0H,EACvCC,EACAC,EACAC,EAAiC,EACjCC,EAA2B,GAE3B,IAAIpH,EAA0BgH,EAAaK,KAAM/H,GAC7CgI,EAAwB,EACxBzI,EAAiB,GAErB,KAAiB,MAAVmB,GACP,CACC,IAAIuH,EAAgBvH,EAAOwH,MAAQL,EAC/BM,EAAcT,EAAaU,UAAYN,EAGvCO,EAAsBrI,EAAQmF,OAAQ6C,EAAgBC,EAAQD,GAClEA,EAAgBG,EAEhB5I,GAAUqI,EAAqBS,GAI/B9I,GAAUoI,EADiBjH,EAAO,IAIlCA,EAASgH,EAAaK,KAAM/H,E,CAQ7B,OAFAT,GAAUqI,EAFgB5H,EAAQmF,OAAQ6C,IAInC,CACR,CAuBArJ,4BAA6BqB,EAAiBsI,EAAoBC,EACjEC,EAAqBb,EACrBC,EACAC,EAAiC,EACjCC,EAA2B,GAE3B,IAAIW,EAAczI,EAAQd,QAASoJ,GAC/BN,EAAwB,EACxBU,EAA0B1I,EAAQd,QAASsJ,EAAaC,GACxDlJ,EAAiB,GAErB,KAAOkJ,GAAO,GACd,CACC,IAAIN,EAAcnI,EAAQd,QAASqJ,EAAUE,GACzCE,EAAwB,EAE5B,GAAoB,KAAhBH,EACJ,CAGC,IAAII,EAAe5I,EAAQ6I,YAAaL,EAAaL,EAAML,GAE3D,KAAOc,GAAQ,GAEVA,IAASF,GAGbE,EAAO5I,EAAQ6I,YAAaL,EAAaI,EAAOd,GAChDa,G,CAMF,GAAIA,EAAgB,EACpB,CACC,IAAIG,EAAe9I,EAAQd,QAASqJ,EAAUJ,EAAML,GAChDiB,EAAmBD,EAEvB,KAAQA,GAAQ,GAAOH,EAAgB,KAElCI,EAAW,MAImB/I,EAAQ6I,YAAaP,EAAYS,EAAWjB,GAEpDgB,IAG1BA,EAAOC,EAEPA,EAAW/I,EAAQd,QAASqJ,EAAUO,EAAOhB,GAC7Ca,IAGDR,EAAMW,C,CAIPvJ,GAAUqI,EADkB5H,EAAQmF,OAAQ6C,EAAgBS,EAAMT,IAKlEzI,GAAUoI,EAFiB3H,EAAQmF,OAClCsD,EAAMZ,EAAyBM,GAAOM,EAAMZ,KAI7CY,EAAMzI,EAAQd,QAASoJ,EAAYH,EAAML,GACzCY,EAAkB1I,EAAQd,QAASsJ,EAAaC,GAChDT,EAAgBG,EAAML,C,CAQvB,OAFAvI,GAAUqI,EAFgB5H,EAAQmF,OAAQ6C,IAInC,CACR,CASArJ,oBAAqBqK,EAAqBhC,EAAyBiC,EAA+B,MAE5E,MAAjBA,IAEHA,EAAgB,CACdC,gBAAgB,EAChBC,gBAAgB,IAInB,IAAIC,EAA0B,mBAC1BC,EAAwB,IA4N5B,OA1NqC,IAAjCJ,EAAcE,iBAEjBC,EAAkB,GAClBC,EAAgB,IAMInJ,EAAQoJ,eAAgBN,EAC5C,IAAIO,OAAQ,mCAAoC,MAC/CC,GAKO,GAFPA,EAAaA,EAAWrE,OAAQ,OAIhCsE,IAEA,GAAmB,KAAfA,EACH,MAAO,GAER,IAAIC,EAAqBxJ,EAAQyJ,qBAChCF,EAAY,KAAM,IAAK,KACtBG,GAEkB,aAAaA,gBAI/BC,GAEO,IAELC,EAAsB5J,EAAQyJ,qBACjCD,EAAY,OAAQ,IAAK,KACxBE,IAEA,IAAIG,EAAc,GAOlB,OAJCA,GADoC,IAAjCd,EAAcC,eACX,wCAAwCU,OAAiB5C,gBAEzD,aAAaoC,IAAkBQ,IAAcP,MAAkBrC,gBAE/D,CAAK,IAEZ6C,GAEO,GACL,EAAG,GACHG,EAAsB9J,EAAQyJ,qBACjCG,EAAa,KAAM,IAAK,KACvBF,IAEA,IAAIG,EAAc,GAYlB,OAVqC,IAAjCd,EAAcC,gBAEjBa,EAAM,6BAA6BH,6CAEZ,IAAnB5C,IACH+C,EAAM,uBAAuBH,kBAG9BG,EAAM,aAAaH,cAEb,CAAK,IAEZC,GAEO,IAOLI,EAAsB,GAEtB,EAAAvL,IAAI6D,OAAS,EAAA9D,cAAc+D,aAE9ByH,EAAc/J,EAAQyJ,qBACrBK,EAAa,KAAM,IAAK,KACvBJ,GAEO,KAEPC,GAEO,KAON,EAAAnL,IAAI6D,OAAS,EAAA9D,cAAcyL,cAE9BD,EAAc/J,EAAQyJ,qBACrBK,EAAa,KAAM,IAAK,KACvBJ,IAEA,IAAIO,EAAmB,GAEvB,IAGCrJ,KAAKsJ,MAAOR,GAGXO,GADoC,IAAjClB,EAAcE,eACNrI,KAAKC,UAAW6I,GAEhB,GAAGA,G,CAEhB,MAAOxI,GAKN+I,EAAW,GAAGP,G,CAIf,IAAIG,EAAc,GAElB,IAAqC,IAAjCd,EAAcC,eAEjBa,EAAM,mDACuBI,6KAK9B,CACC,IAAIE,EAAqBC,IAExB,IAAIC,EAAU,KAEd,IAEC,IAAIC,EAAMF,EAcV,GAZ2B,iBAAhB,IACVE,EAAM1J,KAAKsJ,MAAOE,IAEE,iBAAV,IACVC,EAAU,IAAI,EAAA7L,IAAI4D,eAAgBkI,IAE/BA,aAAeC,QAClBF,EAAU,IAAI,EAAA7L,IAAI4D,eAAgBkI,EAAI,GAAIA,EAAI,GAAIA,EAAI,KAEpC,MAAfA,EAAU,OACbD,EAAU,IAAI,EAAA7L,IAAI4D,eAAgBkI,IAEe,MAA9C,EAAA9L,IAAImB,YAAY6K,aAAaH,EAAQ7F,MACxC,MAAM,IAAIxG,MAAO,gBAAgBqM,EAAQ7F,uB,CAE3C,MAAOtD,GAEN,MAAM,IAAIlD,MACZ,iCAAiCoM,QAAgB,EAAA5L,IAAImB,YAAY6E,gBAAgBtD,EAAGC,U,CAInF,OAAO,CAAS,EAGjB,MAAMkJ,EAAUF,EAAmBF,GACnCJ,EAAM,sCAAsCQ,EAAQ7F,kCAAkC6F,EAAQI,mCAAmCJ,EAAQxO,kB,CAG1I,OAAO,CAAK,IAEZ8N,GAEO,KAOV,IAAIe,EAAsB1K,EAAQyJ,qBACjCM,EAAa,aAAc,aAAc,cACxCT,GAEO,IAEPC,IAEA,IAAIoB,EAAyB,GAG5BA,GADoC,IAAjC5B,EAAcE,eACArI,KAAKC,UAAW0I,GAEhBA,EAElB,IAAIM,EAAc,GAOlB,OAJCA,GADoC,IAAjCd,EAAcC,eACX,eAAe2B,MAAmB7D,QAElC6D,EAEA,CAAK,GAEb,aAAarP,OAAQ,aAAaA,QAOnC,OAHAoP,EAAcA,EAAYrO,QAAS,wBAAyB,IAC5DqO,EAAcA,EAAYrO,QAAS,wBAAyB,IAErD,CAAa,GAClB,EAGL,CAWM0D,QAASpB,EAAY,M,yCAE1B,IAAImK,EAAsBpL,KAAKoC,QAE/B,EAAAtB,IAAI6D,KAAO3E,KAAKmC,KAAKD,UAAUgL,KAC/B,EAAApM,IAAI2D,UAAYxD,EAChB,EAAAH,IAAImB,YAAcjC,KAAKmC,KACvB,EAAArB,IAAImE,WAAajF,KAAKmC,KAAKD,UAAUiL,WACrC,EAAArM,IAAI+D,IAAM7E,KAAKmC,KAAKiL,SACpB,EAAAtM,IAAIgE,UAAY9E,KAAKmC,KAAKkL,eAE1B,IAAI1L,EAAiBW,EAAQgL,aAAclC,EAAapL,KAAKoJ,gBAGzDmE,EAAsB,KAE1B,IAEC,IAAIC,EAA2B,qFAO/B,GAAsB,iBAAX,EACV,MAAM,IAAIlN,MAAO,6CAElB,IAAK,IAAIxC,KAAOmD,EAChB,CACC,IAAIwM,EAAiB,GACjBC,EAAmBzM,EAAKnD,GAG5B2P,EAAS,OAAO3P,OAFaoF,KAAKC,UAAWuK,QAI7CF,GAAoBC,C,CAGrB,IAAIE,EAAsB3N,KAAK8G,KAEX,KAAhB6G,IACHA,EAAc3N,KAAKwC,WAEA,KAAhBmL,IACHA,EAAc3N,KAAKuC,KAEpBiL,GAAoB,24CA6DpBA,GAAoB7L,EACpB6L,GAAoB,gOAapB,IAAIT,EAAiB,IAAIa,SAAUJ,GACnCD,QAAuBR,EAAKjL,MAAO9B,KAAM,CAAC,EAAAc,IAAKd,M,CAEhD,MAAOwD,GAEN,MAAkBqK,YAMXrK,C,CAMR,EAAA1C,IAAIiE,KAAOwI,EAAeO,IAAI/I,KAC9B,IAAIgJ,EAAsBR,EAAe5L,OAGzC,OAFA,EAAAb,IAAI6C,OAAS,GAEN,CACR,G,EAhqBD,W,2BCtDA,IAAYqK,E,+EAAZ,SAAYA,GAKX,mBAIA,yBAIA,qBAIA,yBAIA,iBAIA,kBACA,CA1BD,CAAYA,EAAA,EAAAA,cAAA,EAAAA,YAAW,KA+BvB,eAOC7I,YAAa8I,EAAwBD,EAAYE,KAEhDlO,KAAKiO,SAAWA,CACjB,CAKAE,IAAKC,EAAoB3K,GAEpBzD,KAAKiO,WAAaD,EAAYK,UAE7BD,IAAUJ,EAAY1N,OACzBN,KAAKsO,MAAO7K,GAET2K,IAAUJ,EAAYO,SACzBvO,KAAKwO,QAAS/K,GAEV2K,IAAUJ,EAAYS,MACzBL,IAAUJ,EAAYK,SAEvBrO,KAAK0O,KAAMjL,IAITzD,KAAKiO,WAAaD,EAAYE,MAE7BE,IAAUJ,EAAY1N,OACzBN,KAAKsO,MAAO7K,GAET2K,IAAUJ,EAAYO,SACzBvO,KAAKwO,QAAS/K,GAEX2K,IAAUJ,EAAYS,MACzBzO,KAAK0O,KAAMjL,IAGTzD,KAAKiO,WAAaD,EAAY1N,OAE7B8N,IAAUJ,EAAY1N,OACzBN,KAAKsO,MAAO7K,GAGVzD,KAAKiO,WAAaD,EAAYO,SAE7BH,IAAUJ,EAAYO,SACzBvO,KAAKwO,QAAS/K,GAGZzD,KAAKiO,WAAaD,EAAYS,MAE7BL,IAAUJ,EAAYS,MACzBzO,KAAK0O,KAAMjL,EAEd,CAKAkL,QAASlL,GAEJzD,KAAKiO,WAAaD,EAAYK,SACjCO,QAAQF,KAAMjL,EAChB,CAKAiL,KAAMjL,GAEAzD,KAAKiO,WAAaD,EAAYE,KACjClO,KAAKiO,WAAaD,EAAYK,SAC9BrO,KAAKiO,WAAaD,EAAYS,MAE/BG,QAAQF,KAAMjL,EAEhB,CAKA+K,QAAS/K,GAEHzD,KAAKiO,WAAaD,EAAYE,KACjClO,KAAKiO,WAAaD,EAAYK,SAC9BrO,KAAKiO,WAAaD,EAAYO,SAE/BK,QAAQC,KAAMpL,EAEhB,CAKA6K,MAAO7K,GAEN,GAAKzD,KAAKiO,WAAaD,EAAYE,KACjClO,KAAKiO,WAAaD,EAAYK,SAC9BrO,KAAKiO,WAAaD,EAAY1N,MAChC,CACC,IAAIwO,EAAc,GAEO,iBAAd,EACVA,EAAMrL,GAGiB,MAAnBA,EAAQA,UACXqL,EAAMrL,EAAQA,SAEM,MAAjBA,EAAQsL,QACXD,EAAMrL,EAAQsL,QAGhBH,QAAQN,MAAOQ,E,CAEjB,E,qaC5JD,eAEA,SAgDA,gBAoCC3J,YAAauD,GAERA,aAAgB,EAAAxH,SAEnBlB,KAAKkC,UAAYwG,EACjB1I,KAAK8G,KAAO,GACZ9G,KAAKgP,WAAa,GAClBhP,KAAKiP,UAAY,GACjBjP,KAAK0C,MAAQ,GACb1C,KAAK6C,MAAQ,GACb7C,KAAK8M,aAAe,CAAC,EACrB9M,KAAKkP,UAAY,CAAC,IAIlBlP,KAAKkC,UAAYwG,EAAKxG,UACtBlC,KAAK8G,KAAO4B,EAAK5B,MAAQ,GACzB9G,KAAKgP,WAAatG,EAAKsG,YAAc,GACrChP,KAAKiP,UAAYvG,EAAKuG,WAAa,GACnCjP,KAAK0C,MAAQgG,EAAKhG,OAAS,GAC3B1C,KAAK6C,MAAQ6F,EAAK7F,OAAS,GAC3B7C,KAAK8M,aAAepE,EAAKoE,cAAgB,CAAC,EAC1C9M,KAAKkP,UAAYxG,EAAKwG,WAAa,CAAC,EAEtC,CAMMC,QAASnO,G,yCAEdA,EAAKmB,KAAOnC,KAEZA,KAAK6C,MAAMuM,KAAMpO,EAClB,G,CAKAoM,SAEC,OAAQpN,KAAKkC,UAAa,GAC3B,CAKAmL,eAEC,OAAQrN,KAAKkC,UAAmB,SACjC,CAMMO,KAAMzB,G,yCAEX,IAAK,IAAI4C,EAAO,EAAGA,EAAO5D,KAAK6C,MAAMjF,OAAQgG,IAC7C,CACC,IAAI5C,EAAgBhB,KAAK6C,MAAMe,SAEzB5C,EAAKyB,M,CAEb,G,CAKMJ,QAASpB,EAAY,M,yCAE1B,IAAIU,EAAiB,GAErB,IAAK,IAAIiC,EAAO,EAAGA,EAAO5D,KAAK6C,MAAMjF,OAAQgG,IAC7C,CACC,IAAI5C,EAAgBhB,KAAK6C,MAAMe,GAE/B,EAAA9C,IAAI6C,OAAS,GACb3C,EAAKmB,KAAOnC,KAEZ2B,UAAgBX,EAAKqB,QAASpB,G,CAG/B,OAAO,CACR,G,CAKAoO,eAAgBC,GAEf,GAAmC,MAA/BtP,KAAK8M,aAAawC,EAAIxI,MACzB,MAAM,IAAIxG,MAAO,gBAAgBgP,EAAIxI,wBAEtC9G,KAAK8M,aAAawC,EAAIxI,MAAQwI,CAC/B,CAKAC,eAAgBzI,GAEf,GAA+B,MAA3B9G,KAAK8M,aAAahG,GACrB,MAAM,IAAIxG,MAAO,gBAAgBwG,sBAElC,OAAQ9G,KAAK8M,aAAahG,EAC3B,CAKA0I,eAAgBC,EAAkBC,GAEjC,GAAgC,MAA5B1P,KAAKkP,UAAUO,GAClB,MAAM,IAAInP,MAAO,aAAamP,qBAE/BzP,KAAKkP,UAAUO,GAAYC,CAC5B,E,gGC3MD,cAOA,MAAahK,EA0CZP,YAAaE,EAAmC3C,EAAeiE,EAA4B,IAoF3F,KAAAb,cAAqC,KAKrC,KAAA6J,WAAqC,KAIrC,KAAA5J,eAAsC,KAQtC,KAAA6J,gBAAwD,KAnGvD5P,KAAKqF,WAAaA,EAClBrF,KAAKyI,OAAS,KAES,MAAnBzI,KAAKqF,YAEyB,MAA7BrF,KAAKqF,WAAWnD,YACnBlC,KAAKyI,OAASzI,KAAKqF,WAAWnD,UAAUuG,QAG1CzI,KAAK0C,MAAQA,EACb1C,KAAKuF,YAAc,GACnBvF,KAAKiH,QAAU,KACfjH,KAAK6P,OAAS,GACd7P,KAAK2F,gBAAkB,KACvB3F,KAAK2G,QAAUA,EACf3G,KAAK8P,OAAS,CACZ,eAAkBpK,EAASqK,YAAa,mBACxC,2BAA8BrK,EAASqK,YAAa,oCAEvD,CAKAhP,mBAAoB0C,GAEnB,MAAO,CAAG6K,MAAO7K,EAClB,CAQAgD,UACCkB,EACArB,EAA2C,KAC3CS,EAAmB,EAAAiJ,WAAWC,KAC9BC,EAAmF,MAG3D,iBAAb,IACVvI,EAAS,IAAI,EAAAnB,eAAgBxG,KAAM2H,EAAQrB,EAAiBS,EAAM,KAAM,KAAM,KAAMmJ,IAEjFvI,aAAkB,EAAAnB,iBAID,MAAhBmB,EAAOjF,QACViF,EAAOjF,MAAQ1C,MAEhB2H,EAAS,IAAI,EAAAnB,eAAgBmB,IAN7B3H,KAAK2G,QAAQyI,KAAMzH,EASrB,CAKAwI,UAAWrJ,GAEV,IAAIsJ,EAA8B,KAElC,IAAK,IAAIxM,EAAO,EAAGA,EAAO5D,KAAK2G,QAAQ/I,OAAQgG,IAC/C,CACC,IAAI+D,EAAyB3H,KAAK2G,QAAQ/C,GAE1C,GAAI+D,EAAOb,OAASA,EACpB,CACCsJ,EAAczI,EAEd,K,EAIF,OAAO,CACR,EAzHD,Y,kHCRA,eAEA,SACA,SAKA,IAAYqI,GAAZ,SAAYA,GAKX,YAIA,cAIA,0CACA,CAdD,CAAYA,EAAA,EAAAA,aAAA,EAAAA,WAAU,KA4KtB,uBAwFC7K,YAAazC,EAAmCoE,EAAe,GAC9DuJ,EAA+D,KAC/DtJ,EAAmBiJ,EAAWC,KAAMK,EAAiD,KACrFX,EAAyC,KAAMhK,EAAuB,KACtEuK,EAAyH,MAEzH,IAAIxJ,EAAqB,KAEzB,GAAIhE,aAAiB,EAAAgD,SACpBgB,EAAWhE,MAEZ,CA0BC,GAzBAgE,EAAWhE,EAAMA,MAEC,MAAdA,EAAMqE,OACTA,EAAOrE,EAAMqE,MAEI,MAAdrE,EAAMoE,OACTA,EAAOpE,EAAMoE,MAEW,MAArBpE,EAAM6C,cACTvF,KAAKuF,YAAc7C,EAAM6C,aAEL,MAAjB7C,EAAM6N,UAEsB,iBAAnB7N,EAAa,QAExB1C,KAAKuQ,QAAU,CACb,KAAQ,SACR,UAAY,EACZ,YAAe7N,EAAM6N,SAIvBvQ,KAAKuQ,QAAU7N,EAAM6N,SAGC,MAApB7N,EAAM8N,WACV,CACCxQ,KAAKwQ,WAAa,CAAC,EAEnB,IAAK,IAAI1S,KAAO4E,EAAM8N,WACtB,CACC,IAAIC,EAAQ/N,EAAM8N,WAAW1S,GAEN,iBAAZ,EAEVkC,KAAKwQ,WAAW1S,GAAO,CACrB,KAAQ2S,EACR,UAAY,EACZ,YAAe,KAKC,MAAdA,EAAM1J,OACT0J,EAAM1J,KAAO,UAEd/G,KAAKwQ,WAAW1S,GAAO2S,E,EAKG,MAAzB/N,EAAMiD,kBACTA,EAAkBjD,EAAMiD,iBAEI,MAAzBjD,EAAMgO,kBACTL,EAAY3N,EAAMgO,iBAEY,MAA3BhO,EAAM4N,oBACTA,EAAoB5N,EAAM4N,mBAEH,MAApB5N,EAAMiN,aACTA,EAAajN,EAAMiN,YAEQ,MAAxBjN,EAAMqD,iBACT/F,KAAK+F,eAAiBrD,EAAMqD,gBAEA,MAAzBrD,EAAMgO,kBACT1Q,KAAK0Q,gBAAkBhO,EAAMgO,iBAED,MAAzBhO,EAAMiO,kBACT3Q,KAAK2Q,gBAAkBjO,EAAMiO,iBAEP,MAAnBjO,EAAMwN,YACTA,EAAYxN,EAAMwN,U,CAGpB,GAAa,KAATpJ,EACH,MAAM,IAAIxG,MAAO,uCAYlB,GAVAN,KAAK0C,MAAQgE,EACb1G,KAAK8G,KAAOA,EACZ9G,KAAK+G,KAAOA,EACZ/G,KAAK4Q,cAAe,EACpB5Q,KAAK6Q,cAAe,EACpB7Q,KAAK2F,gBAAkBA,EACvB3F,KAAKsQ,kBAAoBA,EACzBtQ,KAAK2P,WAAaA,EAClB3P,KAAKkQ,UAAY,CAAC,EAEdlQ,KAAK0C,MAAM2C,WAAWnD,UAAUgL,OAAS,EAAArM,cAAcyL,aAEzC,MAAb4D,EAEH,GAAIA,aAAqBrD,MAExB,IAAK,IAAIjJ,EAAO,EAAGA,EAAOsM,EAAUtS,OAAQgG,IAC5C,CACC,IAAIgJ,EAAMsD,EAAUtM,GAEpB,GAAqB,iBAAV,EACX,CACC,MAAMkN,EAAuBlE,EACvBG,EAA4CmD,EAAUtM,EAAO,GAEnE5D,KAAK+Q,YAAaD,EAAc/D,GAChCnJ,G,MAGA5D,KAAK+Q,YAAanE,E,MAKpB,IAAK,IAAI9O,KAAOoS,EAChB,CACC,IAAItD,EAAMsD,EAAUpS,GAEpBkC,KAAK+Q,YAAanE,E,CAMlB5M,KAAK0C,MAAM2C,sBAAsB,EAAA+B,YACpCpH,KAAK0Q,gBAAkBL,EAGzB,CAKAU,YAAaC,EACXC,EAAqC,MAEtC,GAA6B,iBAAlB,EACX,CACC,MAAMnK,EAAekK,EACfjE,EAAyBkE,EAO/B,YALAjR,KAAKkQ,UAAUpJ,GAAQ,CACrBA,KAAMA,EACNiG,KAAMA,G,CAMT,GAA6B,mBAAlB,EACX,CACC,MAAMmE,EAAqB/R,OAAOqI,KAAMxH,KAAKkQ,WAAWtS,OAClDkJ,EAAe,GAAG9G,KAAK0C,MAAMA,SAAS1C,KAAK8G,kBAAkBoK,IAC7DnE,EAA4CiE,EAOlD,YALAhR,KAAKkQ,UAAUpJ,GAAQ,CACrBA,KAAMA,EACNiG,KAAMA,G,CAMT,MAAMoE,EAA4CH,EAClDhR,KAAKkQ,UAAUiB,EAASrK,MAAQqK,CACjC,E,ubC5bD,eAQA,IAAY5I,GAAZ,SAAYA,GAEX,mBACA,+BACA,0BACA,CALD,CAAYA,EAAA,EAAAA,gBAAA,EAAAA,cAAa,KAyEzB,kBA2DCpD,YAAajD,GAERA,aAAqB,EAAAhB,SAExBlB,KAAKkC,UAAYA,EACjBlC,KAAKoR,WAAa,SAClBpR,KAAK+C,IAAM,KACX/C,KAAKqR,cAAgB,UACrBrR,KAAKsR,MAAQ,CACXC,KAAM,GACNC,MAAO,KAETxR,KAAKyR,IAAM,CACTC,KAAM,GACN5T,IAAK,GACL6T,GAAI,IAEN3R,KAAK4R,qBAAsB,EAC3B5R,KAAK+G,KAAOwB,EAAcC,KAC1BxI,KAAKyI,OAASvG,EAAUuG,OACxBzI,KAAK6R,QAAU,CAAC,IAIhB7R,KAAKkC,UAAYA,EAAUA,UAC3BlC,KAAKoR,WAAalP,EAAUkP,YAAc,SAC1CpR,KAAK+C,IAAMb,EAAUa,KAAO,KAC5B/C,KAAKqR,cAAgBnP,EAAUmP,eAAiB,UAChDrR,KAAKsR,MAAQpP,EAAUoP,OAAS,CAC9BC,KAAM,GACNC,MAAO,KAETxR,KAAKyR,IAAMvP,EAAUuP,KAAO,CAC1BC,KAAM,GACN5T,IAAK,GACL6T,GAAI,IAEN3R,KAAK4R,oBAAuD,MAAjC1P,EAAU0P,qBAA8B1P,EAAU0P,oBAC7E5R,KAAK+G,KAAO7E,EAAU6E,MAAQwB,EAAcC,KAC5CxI,KAAKyI,OAASvG,EAAUuG,OACxBzI,KAAK6R,QAAU3P,EAAU2P,SAAW,CAAC,EAEvC,CAMMC,OAAQ/O,G,yCAEb/C,KAAKkC,UAAUa,IAAMA,EACrB/C,KAAK+C,IAAMA,CAIZ,G,yqCCnMD,gDACA,6CAEA,uDACA,sEAEA,mCACA,mCAEA,wCACA,kCAGA,+BACA,qCAGA,wCAEA,sCAGA,IAAIgP,eAAsB,KACtBC,uBAA8B,KAC9BC,sBAA6B,KAqZjC,MAAa/Q,QAiFZiE,YAAauD,EAAiB,CAAC,GAE9B1I,KAAK+C,IAAM2F,EAAK3F,KAAO,KACvB/C,KAAKsI,UAAYI,EAAKJ,WAAa,KACnCtI,KAAKkN,KAAOxE,EAAKwE,MAAQ,MAAArM,cAAc+D,WACvC5E,KAAKkS,MAAQxJ,EAAKwJ,OAAS,CAAC,EAC5BlS,KAAKmS,WAAazJ,EAAKyJ,YAAc,CAAC,EACtCnS,KAAK6C,MAAQ6F,EAAK7F,OAAS,CAAC,EAC5B7C,KAAKoS,QAAU1J,EAAK0J,SAAW,KAC/BpS,KAAKqS,WAAa,4QAQlBrS,KAAKsS,iBAAmB,oSAMxBtS,KAAKuS,YACP,+xCA6DEvS,KAAKyI,OAAS,IAAI,SAAA+J,OAAQ,SAAAxE,YAAYyE,MACtCzS,KAAKmN,WAAa,CAAC,EACnBnN,KAAK0S,QAAU,CAAC,CACjB,CAKA3R,oBAAqB5C,GAIpB,GAAc,UAFdA,EAAQA,EAAMkD,eAGb,OAAO,EAER,GAAc,UAAVlD,EACH,OAAO,EAER,GAAc,QAAVA,EACH,OAAO,EAER,GAAc,OAAVA,EACH,OAAO,EAER,GAAc,QAAVA,EACH,OAAO,EAER,GAAc,QAAVA,EACH,OAAO,EAER,IAEC,GAAwB,GAApBwU,SAAUxU,GACb,OAAO,C,CAET,MAAOqF,G,CAIP,OAAO,CACR,CAKAzC,gBAAiB+F,EAAc8L,EAAmBC,GAAoB,EAAMC,GAA0B,GAErG,IAAI3U,EAAayU,EAAa9L,GAE9B,GAAa,MAAT3I,IAEc,IAAb0U,IAEoB,IAAnBC,EACH,MAAM,IAAIxS,MAAO,8BAA8BwG,MAIlD,GAAuB,iBAAZ,IAEO,IAAb+L,GAEW,KAAV1U,IAEoB,IAAnB2U,EACH,MAAM,IAAIxS,MAAO,8BAA8BwG,MAKnD,OAAO,CACR,CAMA/F,uBAAwB+F,EAAc8L,EAAmBG,GAExD,IAAI5U,EAAayU,EAAa9L,GAE9B,OAAa,MAAT3I,GAGmB,iBAAZ,GAEI,KAAVA,EAJG,EAQD,CACR,CAKA4C,oBAAqBiS,GAEpB,GAA+B,MAA3B9R,QAAQ4O,OAAOkD,GACnB,CACC,IAAIzQ,EAAcrB,QAAQ4O,OAAOkD,GAASC,cAE1C,GAAW,MAAP1Q,GAES,KAARA,EAIH,YAFAhF,OAAO2V,SAASC,KAAO5Q,GAMzB,IAAIwK,EAAO7L,QAAQ4O,OAAOkD,GAASjG,KAEvB,MAARA,GACHA,EAAMiG,E,CAET,CAKAjS,YAAmBqS,G,iDAElB,aAAc,IAAIrL,SAAS,CAACC,EAASC,KAEnCoL,YAAY,KAEVrL,GAAU,GACRoL,EAAgB,GAEvB,G,CAKAE,QAASnR,GAERnC,KAAKkS,MAAM/P,EAAK2E,MAAQ3E,CACzB,CAKAoR,QAASC,GAER,OAAQxT,KAAKkS,MAAMsB,EACpB,CAKArE,QAASnO,GAER,IAAI8F,EAAe9F,EAAK8F,KAEX,KAATA,IACHA,EAAO9F,EAAKwB,WAEA,KAATsE,IACHA,EAAO9F,EAAKuB,KAEbvC,KAAK6C,MAAMiE,GAAQ9F,CACpB,CAKAQ,QAASsF,GAER,GAAwB,MAApB9G,KAAK6C,MAAMiE,GACd,MAAM,IAAIxG,MAAO,uBAAuBwG,KAEzC,OAAQ9G,KAAK6C,MAAMiE,EACpB,CAUA/F,mBAAmBgM,EAAgB0G,EAAcC,GAehD,OAbgB,WAEd,IAAIC,EAAU9G,MAAM+G,UAAUnU,MAAMoU,KAAKlW,WAKzC,OAHWmL,MAAP4K,GACHC,EAAQvE,KAAKsE,GAEC,MAAXD,EACI1G,EAAKjL,MAAM9B,KAAM2T,GAEjB5G,EAAKjL,MAAM2R,EAASE,EAC7B,CAGF,CAKAG,aAAcC,EAAqFhR,EAAc,KAChH8F,GAEA,IAAImL,EAAUhU,KAAK+C,IAER,MAAPA,IACHiR,EAAUjR,GAEX,IAAIkR,EAAmB,IAAIF,EAAe/T,KAAMgU,GAEhD,GAA6C,MAAzChU,KAAKmS,WAAW8B,EAAiBrL,KACpC,MAAM,IAAItI,MAAO,aAAa2T,EAAiBrL,uBAEhD5I,KAAKmS,WAAW8B,EAAiBrL,KAAO,CAAEsL,cAAeH,EAAe7R,UAAWlC,KAAM+C,IAAKiR,GAC9FhU,KAAKmU,kBAAmBF,EAAiBrL,IAAKC,EAC/C,CAKU9H,eAAgBqT,GAGzB,MAAMC,EAAkBD,EAAIzV,QAAQ,QAAS,KAAKA,QACjD,gGAAiG,WAC5F2V,GAAY,IAAIC,WAAaC,gBAAiB,QAAQH,UAAiB,YAC7E,IAAII,EAAwB,GAE5B,GAAIH,EAAUI,gBAAgBC,SAAS/W,OAAS,EAChD,CACC,MAAMgX,EAAkBN,EAAUI,gBAAgBC,SAAS,GAAGC,QAAQvT,cAEtD,OAAZuT,IAEHR,EAAM,UAAUA,YAChBK,EAAgB,SAGD,OAAZG,IAEHR,EAAM,UAAUA,YAChBK,EAAgBG,E,CAIlB,MAAO,CAAGC,SAAUT,EAAKK,cAAeA,EACzC,CAKUN,kBAAmBvL,EAAaC,GAEzC,GAAY,MAAPD,GAAyB,KAARA,EACrB,MAAM,IAAItI,MAAO,mCAElB,QAAiCwI,IAA7BgM,eAAezV,IAAKuJ,GAIvB,OAGD,IAAImM,EAAsB/U,KAAKmS,WAE/B2C,eAAeE,OAAQpM,EAAK,cAAcqM,YAOxC9P,cAEC+P,QAEA,IAAIC,EAAgBJ,EAAoBnM,GACxC5I,KAAKoV,UAAY,IAAID,EAAcjB,cAAeiB,EAAcjT,UAAWiT,EAAcpS,IAC1F,CAWUsS,qBAAsBC,EAAmBlB,GAClDA,EAAMA,EAAIzV,QAAQ,QAAS,KAAKA,QAAQ,gGAAiG,WACzI,MAAM4W,EAAOD,EAAOd,gBAAgB,QAAQJ,EAAI,SAAU,YACpDoB,EAAOF,EAAOd,gBAAgB,GAAI,aACxC,IAAK,IAAIiB,KAAQ5I,MAAM6I,KAAKH,EAAKb,gBAAgBC,UAEhDa,EAAK5N,KAAK+N,YAAYF,GAEvB,IAAK,IAAIA,KAAQ5I,MAAM6I,KAAKF,EAAKI,iBAAiB,wFAEjDH,EAAKI,UAAY,IAAIJ,EAAKI,UAAUpW,MAAM,GAAGT,MAAM,KAAK,GAEzD,OAAOwW,CACR,CAEIzM,yBAEH,OAAQ/I,KAAKoV,UAA4B,kBAC1C,CAEMU,oB,iDAUL,GARsB9V,KAGN+V,aAAe/V,KAAKoV,UAEpCpV,KAAKoV,UAAUzM,aAAe,CALR3I,MAMtBA,KAAKoV,UAAUpM,MAAQhJ,KAAKgW,UAEW,MAAnChW,KAAKoV,UAAUa,uBACZjW,KAAKoV,UAAUa,iBAAkBjW,KAAK5B,iBAG5C,IAAK,IAAIwF,EAAO,EAAGA,EAAO5D,KAAK5B,WAAWR,OAAQgG,IAClD,CACC,MAAMsS,EAAalW,KAAK5B,WAAWwF,GAC7BuS,EAAmBD,EAAKpP,KAAKzF,cAC7B+U,EAAoBF,EAAK/X,MAW/B,GATiB,OAAbgY,IACHnW,KAAKoV,UAAUtO,KAAOsP,GAEN,SAAbD,IACHnW,KAAKoV,UAAUtO,KAAOsP,GAEN,UAAbD,IACHnW,KAAKoV,UAAUjX,MAAQiY,GAEpBD,EAAS7U,QAAS,SAAW,EACjC,CACC,MAAM+U,EAAuBF,EAASG,UAAW,GAGjDtW,KAAKoV,UAAUiB,GAAgBD,C,EAKlC,GAAkC,MAA9BpW,KAAKoV,UAAUmB,cAE0B,WAAlCvW,KAAKoV,UAAUmB,eACxB,OAGF,IAAIC,QAAgBxW,KAAKoV,UAAUzT,SAEA,MAA/B3B,KAAKoV,UAAUqB,eAClBD,QAAgBxW,KAAKoV,UAAUqB,aAAcD,IAE9C,IAAIE,EAAyC,GAEpB,iBAAd,EACVA,EAAiBtH,KAAM,CAAEuH,KAAMH,IAI9BE,EADGF,aAAmB3J,MACH2J,EAEA,CAACA,GAGtB,IAAK,IAAII,EAAO,EAAGA,EAAOF,EAAiB9Y,OAAQgZ,IACnD,CACC,IAAIjV,EAAS+U,EAAiBE,GAC1BC,EAAkBlV,EAAOgV,KACzBG,EAAyB,GAEA,MAAzBnV,EAAOmV,iBACVA,EAAiBnV,EAAOmV,gBAEzB,IAAI1C,EAAc,UAAA9R,QAAQgL,aAAcuJ,GAAS,EAAM,CAAE,gBAAkB,IAE5C,MAA3B7W,KAAKoV,UAAU2B,WAClB3C,QAAYpU,KAAKoV,UAAU2B,SAAU3C,IAEtC,IAAI4C,EAA4D,CAAEnC,SAAU,GAAIJ,cAAe,IAG9FuC,EAD+B,MAA5BhX,KAAKoV,UAAU6B,gBACEjX,KAAKoV,UAAU6B,UAAW7C,GAEhClT,QAAQgW,QAAS9C,GAEhC,IAAI+C,EAAmB,KACnBC,EAAsB,KAW1B,GARCD,EADgC,MAA7BnX,KAAKoV,UAAUiC,iBACHrX,KAAKoV,UAAUiC,WAAYL,EAAYnC,WAK7C,IAAIN,WAAaC,gBAAiBwC,EAAYnC,SAAU,aAG9DsC,EAAOvP,KAAK+M,SAAS/W,OAAS,EACjC,MAAM,IAAI0C,MAAO,4BAA4BN,KAAKoV,UAAUtO,QAE7D,GAAIqQ,EAAOvP,KAAK+M,SAAS/W,OAAS,EAClC,CACC,IAAI0Z,GAAoB,EAExB,IAAK,IAAI1T,EAAO,EAAGA,EAAOuT,EAAOvP,KAAK+M,SAAS/W,OAAQgG,IACvD,CACC,IAAI2T,EAAQJ,EAAOvP,KAAK+M,SAAS/Q,GAEjC,GAAI2T,aAAiBtC,aAEiB,gBAAjCsC,EAAM3C,QAAQvT,cAClB,CACC+V,EAASG,EACTD,GAAW,EAEX,K,EAKH,IAAiB,IAAbA,EACH,MAAM,IAAIhX,MAAO,sDAAsDN,KAAKoV,UAAUtO,yC,CAIvFsQ,EADiC,KAA9BJ,EAAYvC,cACQ0C,EAAOvP,KAAK+M,SAAS,GAEnCwC,EAAO1C,cAAeuC,EAAYvC,eAE5C,IAAI+C,EAA0B,GAG9B,IAAK,IAAI5T,EAAQ5D,KAAK2U,SAAS/W,OAAS,EAAIgG,GAAQ,EAAGA,IACvD,CACC,IAAI2T,EAAcvX,KAAK2U,SAAS/Q,GAEhC4T,EAAgBpI,KAAMpP,KAAKyX,YAAaF,G,CAGzCvX,KAAK0X,YAAaN,GAEU,MAAxBpX,KAAKoV,UAAUuC,QAClBP,EAAOQ,QAAU5X,KAAKoV,UAAUuC,MAAMlX,KAAMT,KAAKoV,YAElD,IAAK,IAAItX,KAAOkC,KAAKoV,UAAUnM,OAC/B,CACC,IAAI4O,EAAQ7X,KAAKoV,UAAUnM,OAAOnL,GAGlCsZ,EAAOU,iBAAkBD,EAAM9Q,KAAM8Q,EAAM9K,KAAM8K,EAAME,Q,CAGxD,IAAIC,EAA4B7Y,OAAO8Y,oBAAqBjY,KAAKoV,UAAUjQ,YAAYyO,WAGvF,IAAK,IAAIhQ,EAAO,EAAGA,EAAOoU,EAAgBpa,OAAQgG,IAClD,CACC,IAAIsU,EAAkBF,EAAgBpU,GAEtC,GAAgB,gBAAZsU,GAMkB,mBAFXlY,KAAKoV,UAAU8C,GAG1B,CACC,IAAIC,GAAyB,EAM7B,IAAK,IAAIC,KAAQ,eAAAC,aAAazE,UAE7B,GAAIsE,IAAYE,EAChB,CACCD,GAAgB,EAEhB,K,EAIoB,IAAlBA,IAGHf,EAAOc,GAAWhX,QAAQoX,YAAatY,KAAKoV,UAAU8C,GAAUlY,KAAKoV,WAE9C,KAAnB0B,KAEsBzY,SAASoW,cAAeqC,GAG3CoB,GAAWhX,QAAQoX,YAAatY,KAAKoV,UAAU8C,GAAUlY,KAAKoV,W,EAMvC,MAA7BpV,KAAKoV,UAAUmD,aAClBnB,QAAepX,KAAKoV,UAAUmD,WAAYnB,IAE3C,IAAIoB,QAAsCxY,KAAKoV,UAAUlM,UAAWkO,GAYpE,GAVoC,MAAhCpX,KAAKoV,UAAUqD,gBAGlBD,EAAiBC,cAAgBzY,KAAKoV,UAAUqD,eAIjDD,EAAiBzC,aAAe/V,KAAKoV,UACrCpV,KAAKoV,UAAUzM,aAAayG,KAAMoJ,GAEL,MAAzB7W,EAAO+W,eACX,CACC,IAAIC,EAAmBta,SAASoW,cAAe9S,EAAO+W,gBAEtDF,EAAiBI,cAAcnB,YAAae,GAC5CG,EAAWhD,YAAa6C,GAGc,MAAlCA,EAAiBC,sBAGdD,EAAiBzC,aAAa0C,cAAeE,EAAYH,G,CAIjE,GAA8B,MAA1B7W,EAAOkX,gBACX,CACC,IAAIC,EAAoBN,EAAiBG,WACrCI,EAAiC,EAErC,KAAOA,EAAyB,IAEN,MAArBD,KAGAA,aAA6BE,kBAJlC,CASC,IAAIC,EAAiBH,EAAkBlD,iBAAkB,wBAAwBjU,EAAOkX,qBAExF,GAAII,EAAerb,OAAS,EAC5B,CACC,IAAIsb,EAAYD,EAAe,GAE/BT,EAAiBG,WAAWlB,YAAae,GACzCU,EAAUvD,YAAa6C,GAGe,MAAlCA,EAAiBC,sBAGdD,EAAiBzC,aAAa0C,cAAeS,EAAWV,IAG/D,K,CAGD,GAAIS,EAAerb,OAAS,EAC5B,CACC,IAAIub,EAAqBL,EAAkBlD,iBAAkB,oBAAoBjU,EAAOkX,qBAExF,GAAIM,EAAmBvb,OAAS,EAChC,CACC,IAAIsb,EAAYC,EAAmB,GACnCX,EAAiBG,WAAWlB,YAAae,GACzCU,EAAUvD,YAAa6C,GAGe,MAAlCA,EAAiBC,sBAGdD,EAAiBzC,aAAa0C,cAAeS,EAAWV,IAG/D,K,EAIFM,EAAoBA,EAAkBH,WACtCI,G,EAKF,IAAK,IAAInV,EAAO,EAAGA,EAAO4T,EAAgB5Z,OAAQgG,IAClD,CACC,MAAM2T,EAAcC,EAAgB5T,GAEpC4U,EAAiB7C,YAAa4B,GAGH,MAAvBA,EAAMkB,sBAGHlB,EAAMxB,aAAa0C,cAAeD,EAAkBjB,G,CAI1B,MAA9BvX,KAAKoV,UAAUgE,cAGlBZ,QAAyBxY,KAAKoV,UAAUgE,YAAaZ,EAAiBG,WAAYH,G,CAGrF,G,GACE3P,EACL,CAKA9H,eAAgBsY,EAA8B1C,GAE7C,IAAI2C,EAA2B,KAO/B,GAJCA,EADuB,iBAAb,EACIjb,SAASoW,cAAe4E,GAExBA,EAEI,MAAfC,EACH,MAAM,IAAIhZ,MAAO,yBAAyB+Y,MAE3C,IAAIvW,EAAsB,KAE1B,GAAsB,iBAAX,EACX,CACC,IAAIkU,EAAc9V,QAAQgW,QAASP,GAC/BQ,GAAmB,IAAI5C,WAAaC,gBAAiBwC,EAAYnC,SAAU,aAC3EF,EAAgB,KAGnBA,EADiC,KAA9BqC,EAAYvC,cACJ0C,EAAOvP,KAAK+M,SAEZwC,EAAO1C,cAAeuC,EAAYvC,eAAeE,SAE7D,IAAI4E,EAAyB,GAE7B,IAAK,IAAI3V,EAAO,EAAGA,EAAO+Q,EAAS/W,OAAQgG,IAC3C,CACC,IAAI2T,EAAmC5C,EAAS/Q,GAEhD2V,EAAQnK,KAAMkK,EAAY3D,YAAa4B,G,CAGxC,OAAO,C,CAKR,OAFCzU,EAASwW,EAAY3D,YAAagB,GAE5B,CACR,CAKA5V,wBAAyByY,EAAqB1G,GAA0B,GAEvE,IAMIyG,GAAU,uCAAoBC,GAQlC,OANsB,MAAlBD,EAAQzJ,QAEPyJ,EAAQzJ,OAAOlS,OAAS,GAVL,MAEtB,IAAuB,IAAnBkV,EACH,MAAM,IAAIxS,MAAO,WAAWkZ,yFAAmG,EAQhIC,IAGK,CACR,CAOA1Y,kBAAmBqB,EAAiBtE,EAAaK,GAIhD,OAFyBiE,EAAQzD,QAAS,IAAIgN,OAAQ,SAAS7N,OAAU,KAAMK,EAGhF,CAOA4C,8BAA+B2Y,EAAkBC,GAEhD,IAAIxb,EAAa,KAEjB,GAAe,MAAXub,EACJ,CACC,IAAIE,EAAiBF,EAIrB,IAAK,IAAI9V,EAAO,EAAGA,EAAO+V,EAAO/b,OAAQgG,IACzC,CACC,IAAI6M,EAAgBkJ,EAAO/V,GAE3B,GAAwB,MAApBgW,EAAUnJ,GACd,CACCmJ,EAAY,KAEZ,K,CAGDA,EAAYA,EAAUnJ,E,CAGN,MAAbmJ,IACHzb,EAAQyb,E,CAGV,OAAO,CACR,CAKMC,iB,iDAEL3Y,QAAQ4Y,iBAAkB9Z,KAAKoS,QAAQtL,MAAM,GAE7C,IAAIjB,OAAS7F,KAAKoS,QAAQvM,OACtBkU,UAAoB,wBACpBC,OAAoB,KACpBC,OAAwB,KAE5B,IAAsB,IAAlB/Y,QAAQC,OAEPnB,KAAKkN,OAAS,MAAArM,cAAcyL,aAEH,MAAxBtM,KAAKoS,QAAQ8H,QACjB,CACC,IAAIC,EAAeC,IAEjB,IAAIC,GAA2B,EAEE,MAA7BD,EAAUC,kBACbA,EAAkBD,EAAUC,iBAE7B,IAAIrL,EAAqB,SA+BzB,GA7BwB,MAApBoL,EAAUJ,SACbhL,EAAaoL,EAAUJ,QAEI,MAAxBI,EAAUpL,aACbA,EAAaoL,EAAUpL,aAEA,IAApBqL,GAIHtI,eAAiB,iHACjBC,uBAAyB,yHACzBC,sBAAwB,wHAEO,KAA3BmI,EAAUE,eACbP,UAAYK,EAAUE,cAEE,0BAArBF,EAAUH,SACbA,OAAS,IAAIhI,uBAEW,mBAArBmI,EAAUJ,SACbA,OAAS,IAAIjI,eAAgB/R,KAAMgP,EAAY+K,UAAWE,SAElC,2BAArBG,EAAUJ,SACbA,OAAS,IAAIhI,uBAAwBhS,KAAMgP,EAAY+K,aAGxDC,OAASha,KAAK0S,QAAQ1D,GAEF,MAAjBgL,OAAOC,OACV,MAAM,IAAI3Z,MAAO,UAAU0O,iCAEE,MAA1BoL,EAAUG,eACbP,OAAOC,OAAOM,aAAeH,EAAUG,aAAY,EAGtB,MAA5Bva,KAAKoS,QAAQ8H,QAAQM,KACxBL,EAAana,KAAKoS,QAAQ8H,QAAQM,KAEH,MAA5Bxa,KAAKoS,QAAQ8H,QAAQnX,KACxBoX,EAAana,KAAKoS,QAAQ8H,QAAQnX,I,CAKtC,GAAc,MAAV8C,OAEH,IAAK,IAAI/H,KAAO+H,OAChB,CACC,IAAInD,EAAsBmD,OAAO/H,GAC7BkD,EAAgB,IAAI,UAAAsB,QAASI,GAC7BP,EAAgB,IAAI,UAAAsY,QAAS,CAC/BvY,UAAWlC,KACX8G,KAAMpE,EAAMoE,MAAQ,GACpBpE,MAAO5E,EACP+E,MAAO,CAAC7B,KAGV,GAAc,MAAVgZ,QAECha,KAAKkN,OAAS,MAAArM,cAAcyL,YAChC,CACC,IAAIoO,EAAkBhY,EAAMoE,KACxB6T,EAAsB,KAE1B,GAAiB,MAAbjY,EAAMkY,IACV,CACC,GAA2B,iBAAflY,EAAS,IACrB,CACC,GAAkC,MAA9BsX,OAAOa,SAASnY,EAAMkY,KACzB,MAAM,IAAIta,MAAO,YAAYoC,EAAMkY,uBAEpCZ,OAAOa,SAASH,GAAWV,OAAOa,SAASnY,EAAMkY,I,KAGlD,CACCD,EAAU,IAAI,aAAAG,WACd,IAAIC,EAA8E,KAElF,GAAIrY,EAAMkY,eAAe/N,MACzB,CACCkO,EAAe,GAEf,IAAK,IAAInX,EAAO,EAAGA,EAAOlB,EAAMkY,IAAIhd,OAAQgG,IAC5C,CACC,IAAIoX,EAAOtY,EAAMkY,IAAIhX,GAErBmX,EAAa3L,KAAM,IAAI,aAAA6L,mBAAoBD,G,MAI7C,CACCD,EAAe,CAAC,EAEhB,IAAK,IAAI3C,KAAQ1V,EAAMkY,IACvB,CACC,IAAII,EAAOtY,EAAMkY,IAAIxC,GAErB2C,EAAa3C,GAAQ,IAAI,aAAA6C,mBAAoBD,E,EAI/CL,EAAQI,aAAeA,C,CAGxBf,OAAOa,SAASH,GAAWC,C,CAGE,MAA1BjY,EAAMwY,mBACTlB,OAAOa,SAASH,GAASQ,iBAAmBxY,EAAMwY,iB,CAIrDlb,KAAKsT,QAASnR,E,CAIhB,GAAyB,MAArBnC,KAAKoS,QAAQ+I,KAEhB,IAAK,IAAIrd,KAAOkC,KAAKoS,QAAQ+I,KAC7B,CACC,IAAIpY,EAAM/C,KAAKoS,QAAQ+I,KAAKrd,GAE5B,GAAe,MAAXiF,EAAI6X,MAGc,IAAlB1Z,QAAQC,OAEPnB,KAAKkN,OAAS,MAAArM,cAAcyL,YAChC,CACC,IAAIoO,EAAkB5c,EAClB6c,EAAsB,IAAI,aAAAG,WAE9BH,EAAQI,aAAe,GAEvB,IAAK,IAAInX,EAAO,EAAGA,EAAOb,EAAI6X,IAAIhd,OAAQgG,IAC1C,CACC,IAAIgX,EAAc7X,EAAI6X,IAAIhX,GAE1B+W,EAAQI,aAAa3L,KAAM,IAAI,aAAA6L,mBAAoBL,G,CAGpD,GAAc,MAAVZ,OACH,MAAM,IAAI1Z,MAAO,uFAElB0Z,OAAOa,SAASH,GAAWC,C,EAO/B,IAAsB,IAAlBzZ,QAAQC,MAEX,IAAK,IAAIrD,OAAOkC,KAAKoS,QAAQD,WAC7B,CACC,IAAIiD,UAAYpV,KAAKoS,QAAQD,WAAWrU,KACpCsd,aAAuBhG,UAAU7S,IAGjCd,UAAiB,wBAAO2Z,cACxBC,eAAiBxZ,KAAMJ,KAE3BzB,KAAK8T,aAAcuH,e,CAIM,MAAvBrb,KAAKoS,QAAQvM,SAChB7F,KAAKoS,QAAQvM,OAAS,CAAC,GAExB,IAAIyV,oBAA8B,EAEK,MAAnCtb,KAAKoS,QAAQkJ,qBAChBA,mBAAqBtb,KAAKoS,QAAQkJ,qBAER,IAAvBA,yBACGtb,KAAKub,aAAcvb,KAAKoS,QAAQvP,OAEtC7C,KAAKyI,OAAOkG,QAAS,wCAER,MAAVqL,QACHha,KAAKwb,UAAWxB,OAClB,G,CAMMyB,YAAatb,G,iDAElB,IAAIub,EAAkB,GAEtB,IAAsB,IAAlBxa,QAAQC,MACZ,CACCnB,KAAKyI,OAAOiG,KAAM,uBAAuBvO,KAEzC,IAAIsB,QAAiB,wBAAOtB,GAE5BH,KAAKyI,OAAOiG,KAAM,mBAAmBvO,KAErCub,EAAUja,EAAIG,M,MAIdzB,EAAOwb,MAAMC,UAAWzb,GAExBH,KAAKyI,OAAOiG,KAAM,qBAAqBvO,KAEvCub,QAAgB,IAAI3T,SACnB,CAACC,EAAcC,KAEdyB,GAAGC,SAAUxJ,GAAM,CAACyJ,EAA4BjH,KAE9C,GAAW,MAAPiH,EACH,MAAMA,EAEP,IAAIxH,EAAkBO,EAAKkH,WAE3B7J,KAAKyI,OAAOiG,KAAM,iBAAiBvO,KAEnC6H,EAAS5F,EAAQ,GAChB,IAINpC,KAAKoS,QAAUlP,KAAKsJ,MAAOkP,GAC3B1b,KAAKoS,QAAQyJ,YAAc1b,CAC5B,G,CAMMob,aAAc1Y,EAClBiZ,GAA+B,G,iDAEhC9b,KAAKyI,OAAOkG,QAAS,yBAErB,IAAK,IAAI7Q,KAAO+E,EAChB,CACC,IAAI7B,EAAO6B,EAAM/E,GACbie,EAAmB,KAEnB7a,QAAQC,MAEX4a,EAAU,IAAI,UAAAzZ,QAAS,CACrB,KAAQxE,IAUK,MAAZkD,EAAKuB,MACRwZ,EAAQxZ,IAAMvB,EAAKuB,MAEE,IAAlBrB,QAAQC,OAEW,MAAlBH,EAAKwB,YACRuZ,EAAQvZ,UAAYxB,EAAKwB,WAG3B,IAAIwZ,GAAuB,EAW3B,GAToB,MAAhBhb,EAAKoB,UAER2Z,EAAQ3Z,QAAUpB,EAAKoB,QACvB4Z,GAAc,IAGa,IAAxBF,IACHE,GAAc,IAEK,IAAhBA,EACJ,CACC,IAAIC,EAAmB,GAEH,KAAhBF,EAAQxZ,MACX0Z,EAAWF,EAAQxZ,KAEM,KAAtBwZ,EAAQvZ,YACXyZ,EAAWF,EAAQvZ,WAEpBxC,KAAKyI,OAAOkG,QAAS,sBAAsBsN,WACrCF,EAAQtZ,OACdzC,KAAKyI,OAAOkG,QAAS,+BAA+BsN,I,CAGrDjc,KAAKmP,QAAS4M,E,CAGf/b,KAAKyI,OAAOkG,QAAS,iCACtB,G,CAKAuN,gBAAiBC,EAAkBrV,EAAe,GAAIvE,EAAc,KAClE6Z,EAAoB,sBAAuBC,GAAoB,EAC/Dpb,EAAY,MAEb,IAAIqb,EAAqB,GACrBC,EAAkB,GAClBpP,EAAqB,GAKzB,GAAoB,MAAhBnN,KAAKoS,QACT,CACC,GAAqC,MAAjCpS,KAAKoS,QAAQoK,OAAOC,WAEe,KAAlCzc,KAAKoS,QAAQoK,OAAOC,UACxB,CACC,MAAMA,EAAYzc,KAAKoS,QAAQ+I,KAAKnb,KAAKoS,QAAQoK,OAAOC,WAExD,GAAiB,MAAbA,EACHzc,KAAKyI,OAAO+F,QAAS,iBAAiBxO,KAAKoS,QAAQoK,OAAOC,gCAE3D,CACC,IAAIC,GAAyB,EAoB7B,GAlBuB,MAAnBD,EAAUE,QAEbD,GAAgB,EAChB1c,KAAKyI,OAAO+F,QAAS,iBAAiBxO,KAAKoS,QAAQoK,OAAOC,4EAG9B,MAAzBA,EAAUG,cAEbF,GAAgB,EAChB1c,KAAKyI,OAAO+F,QAAS,iBAAiBxO,KAAKoS,QAAQoK,OAAOC,kFAGlC,MAArBA,EAAUI,UAEbH,GAAgB,EAChB1c,KAAKyI,OAAO+F,QAAS,iBAAiBxO,KAAKoS,QAAQoK,OAAOC,+EAGrC,IAAlBC,EACJ,CACCJ,GAAc,6CAA6CG,EAAUE,sBAErE,IAAIvX,EAAkB,KAED,MAAjBqX,EAAUla,MACb6C,EAAU,IAAKqX,EAAUla,QAEV,MAAZvC,KAAK+C,MACRqC,EAAU,IAAKpF,KAAK+C,IAAIqC,YAEzB,IAAI0X,EAAyB9c,KAAKqS,WAClCyK,EAAiBA,EAAene,QAAS,iBAAkB8d,EAAUI,SACrEC,EAAiBA,EAAene,QAAS,2BAA4B8d,EAAUG,aAC/EE,EAAiBA,EAAene,QAAS,iBAAkByG,GAE3DmX,GAAWO,C,GAMf,GAAyB,MAArB9c,KAAKoS,QAAQ+I,KACjB,CACC,IAAIzY,EAAQ1C,KAAKoS,QAAQvM,OAAOsW,GAEhC,GAAa,MAATzZ,GAEc,MAAbA,EAAMK,IACV,CACC,IAAIA,EAAM/C,KAAKoS,QAAQ+I,KAAKzY,EAAMK,KAElC,GAAW,MAAPA,EACH,MAAM,IAAIzC,MAAO,sBAAsBoC,EAAMK,OAE9C,IAAI2Z,GAAyB,EAoB7B,GAlBiB,MAAb3Z,EAAI4Z,QAEPD,GAAgB,EAChB1c,KAAKyI,OAAO+F,QAAS,iBAAiB9L,EAAMK,sEAGtB,MAAnBA,EAAI6Z,cAEPF,GAAgB,EAChB1c,KAAKyI,OAAO+F,QAAS,iBAAiB9L,EAAMK,4EAG1B,MAAfA,EAAI8Z,UAEPH,GAAgB,EAChB1c,KAAKyI,OAAO+F,QAAS,iBAAiB9L,EAAMK,yEAGvB,IAAlB2Z,EACJ,CACC,IAAIK,EAAYha,EAAI4Z,MACpBL,GAAc,6CAA6CS,kBAE3D,IAAI3X,EAAkB,KAEP,MAAXrC,EAAIR,MACP6C,EAAU,IAAKrC,EAAIR,QAEJ,MAAZvC,KAAK+C,MACRqC,EAAU,IAAKpF,KAAK+C,IAAIqC,YAEzB,IAAI0X,EAAyB9c,KAAKqS,WAClCyK,EAAiBA,EAAene,QAAS,iBAAkBoE,EAAI8Z,SAC/DC,EAAiBA,EAAene,QAAS,2BAA4BoE,EAAI6Z,aACzEE,EAAiBA,EAAene,QAAS,iBAAkByG,GAE3DmX,GAAWO,C,GAYf,GAN2B,MAAvB9c,KAAKoS,QAAQoK,QAEqB,MAAjCxc,KAAKoS,QAAQoK,OAAOJ,YACvBA,EAAYpc,KAAKoS,QAAQoK,OAAOJ,WAGH,MAA3Bpc,KAAKoS,QAAQjF,WAEhB,IAAK,IAAIrP,KAAOkC,KAAKoS,QAAQjF,WAC7B,CACC,IACIhP,EADA6e,EAAShd,KAAKoS,QAAQjF,WAAWrP,GAGrC,GAAwB,iBAAb,EACVK,EAAQ+E,KAAKC,UAAW6Z,QAGxB,IAAsB,IAAlB9b,QAAQC,MACZ,CACC,GAAgB,MAAZnB,KAAK+C,IACT,CACC,GAA2B,MAAvB/C,KAAK+C,IAAIsC,WACZ,MAAM,IAAI/E,MAAO,8DAElB,IAAI2c,EAAoCjd,KAAK+C,IAAIsC,WAEjB,MAA5B2X,EAAOE,oBACV/e,EAAQ+E,KAAKC,UAAW8Z,EAAWpL,QAAQ/T,I,CAG7C,GAAkB,MAAdkf,EAAOG,IACX,CAIC,MAAMC,EAAiBJ,EAAOG,IAE9Bhf,EAAQ+E,KAAKC,UAAWd,QAAQ8a,IAAIC,G,EAKvCjQ,GAAc,yBAAyBrP,SAAWK,M,EAKrD,IAAIiE,EAAkBpC,KAAKuS,YAsH3B,OAFAnQ,EAnHiB,CAACib,IAEhB,IAAIC,EAA2B,GAC3BC,EAAuB,GAEvBvd,KAAKkN,OAAS,MAAArM,cAAcyL,cAE/BgR,EAAmB,mDAEC,MAAhBtd,KAAKoS,SAEoB,MAAxBpS,KAAKoS,QAAQ8H,SAEgB,MAA5Bla,KAAKoS,QAAQ8H,QAAQM,MAExB+C,EAAevd,KAAKsS,iBAEyB,MAAzCtS,KAAKoS,QAAQ8H,QAAQM,IAAIF,eAC5Bta,KAAKoS,QAAQ8H,QAAQM,IAAIF,aAAe,yBAEzCiD,EAAeA,EAAa5e,QAAS,yBAA0B,IAAKqB,KAAKoS,QAAQ8H,QAAQM,IAAIF,mBAMjG,IAAIkD,EAAoB,GAExB,GAAIre,OAAOqI,KAAMxH,KAAK6C,OAAOjF,OAAS,EACtC,CACC4f,GAAa,sBAEb,IAAK,IAAI1f,KAAOkC,KAAK6C,MACrB,CACC,IAAI7B,EAAOhB,KAAK6C,MAAM/E,GAClB2f,EAAkB,IAAIzc,EAAKuB,OAC3Bmb,EAAsB,GAE1B,GAAqB,KAAjB1c,EAAKoB,QACT,CACC,IAAI6K,EAAyB/J,KAAKC,UAAWnC,EAAKoB,SAIlD6K,EAAiBA,EAAetO,QAAS,IAAIgN,OAAQ,YAAa,OAAQ,gBAC1EsB,EAAiBA,EAAetO,QAAS,IAAIgN,OAAQ,eAAgB,OAAQ,iBAE7E+R,EAAc,gBAAgBzQ,G,CAG/BuQ,GAAa,gBAAgB1f,kBAAoB2f,IAAUC,Q,CAG5DF,GAAa,yD,CAGdH,EAAcA,EAAY1e,QAAS,aAAcmI,IAEhC,IAAbuV,IACHgB,EAAcA,EAAY1e,QAAS,YAAa,kBAErC,MAARsC,IACHoc,EAAcA,EAAY1e,QAAS,YAAauE,KAAKC,UAAWlC,KAEjE,IAAIgO,EAAoBkN,EACpBpC,EAAoB,GACpB4D,EAA6B,GAC7B3O,EAAqB,SA8CzB,OA5CoB,MAAhBhP,KAAKoS,UAEoB,MAAxBpS,KAAKoS,QAAQ8H,SAEgB,MAA5Bla,KAAKoS,QAAQ8H,QAAQM,MAEe,MAAnCxa,KAAKoS,QAAQ8H,QAAQM,IAAIR,SAC5BhL,EAAahP,KAAKoS,QAAQ8H,QAAQM,IAAIR,QAEI,MAAvCha,KAAKoS,QAAQ8H,QAAQM,IAAIxL,aAC5BA,EAAahP,KAAKoS,QAAQ8H,QAAQM,IAAIxL,YAEM,MAAzChP,KAAKoS,QAAQ8H,QAAQM,IAAIF,eAC5BP,EAAY/Z,KAAKoS,QAAQ8H,QAAQM,IAAIF,cAEO,MAAzCta,KAAKoS,QAAQ8H,QAAQM,IAAIoD,eAC5BD,EAAqB3d,KAAKoS,QAAQ8H,QAAQM,IAAIoD,eAItB,MAAvB5d,KAAKoS,QAAQvM,QAEqB,MAAjC7F,KAAKoS,QAAQvM,OAAOsW,MAGvBlN,EADYjP,KAAKoS,QAAQvM,OAAOsW,GACdrV,OAgBrBuW,GADAA,GADAA,GADAA,GADAA,GADAA,GADAA,GADAA,GADAA,GADAA,GADAA,GADAA,EAAcA,EAAY1e,QAAS,wBAAyByd,IAClCzd,QAAS,uBAAwB2e,IACjC3e,QAAS,mBAAoB4e,IAC7B5e,QAAS,sBAAuB2d,IAChC3d,QAAS,uBAAwB,KACjCA,QAAS,mBAAoB6e,IAC7B7e,QAAS,iBAAkB4d,IAC3B5d,QAAS,uBAAwBwO,IACjCxO,QAAS,WAAY4D,IACrB5D,QAAS,oBAAqB,IAAIqQ,OAClCrQ,QAAS,mBAAoB,IAAIsQ,OACjCtQ,QAAS,8BAA+B,IAAIob,OAC5Cpb,QAAS,8BAA+B,IAAIgf,KAElD,EAEZE,CAAYzb,GAEf,CACR,CASA0b,oBAAqBC,EAAiB3B,EAAoB,uBAEzD,IAAK,IAAIte,KAAOkC,KAAKkS,MACrB,CACC,IAAI/P,EAAgBnC,KAAKkS,MAAMpU,GAC/B,MAAMsE,EAAkBpC,KAAKkc,gBAAiB/Z,EAAKO,MAAOP,EAAK2E,KAAM3E,EAAKU,MAAM,GAAGN,IAAK6Z,GAExF2B,EAAW1e,IAAK8C,EAAKO,OAAO,CAACsb,EAAUvc,KAErCA,EAAIwc,KAAM7b,EAAQ,G,CAGtB,CAKAoZ,UAAWxB,GAEVha,KAAK0S,QAAQsH,EAAOlT,MAAQkT,CAC7B,CAKAkE,oBAEC,GAAoB,MAAhBle,KAAKoS,QACR,MAAM,IAAI9R,MAAO,0BAElB,GAA4B,MAAxBN,KAAKoS,QAAQ8H,QAChB,MAAM,IAAI5Z,MAAO,+CAElB,GAAgC,MAA5BN,KAAKoS,QAAQ8H,QAAQM,IACxB,MAAM,IAAIla,MAAO,mDAElB,GAAqC,MAAjCN,KAAKoS,QAAQ8H,QAAQM,IAAI2D,KAC5B,MAAM,IAAI7d,MAAO,sDAElB,OAAQN,KAAKoS,QAAQ8H,QAAQM,IAAQ,IACtC,CAKA4D,oBAEC,GAAoB,MAAhBpe,KAAKoS,QACR,MAAM,IAAI9R,MAAO,0BAElB,GAA4B,MAAxBN,KAAKoS,QAAQ8H,QAChB,MAAM,IAAI5Z,MAAO,+CAElB,GAAgC,MAA5BN,KAAKoS,QAAQ8H,QAAQnX,IACxB,MAAM,IAAIzC,MAAO,mDAElB,GAAqC,MAAjCN,KAAKoS,QAAQ8H,QAAQnX,IAAIob,KAC5B,MAAM,IAAI7d,MAAO,sDAElB,OAAQN,KAAKoS,QAAQ8H,QAAQnX,IAAQ,IACtC,CAKAsb,oBAAqBvX,GAEpB,IAAInH,EAAmB,GAEvB,GAAoB,MAAhBK,KAAKoS,SAEmB,MAAvBpS,KAAKoS,QAAQvM,OAEhB,IAAK,IAAI/H,KAAOkC,KAAKoS,QAAQvM,OAI5B,GAF0B7F,KAAKoS,QAAQvM,OAAO/H,GAEpCgJ,OAASA,EACnB,CACCnH,EAAW7B,EAEX,K,CAMJ,OAAO,CACR,CAKAwgB,iBAAkBxX,GAEjB,IAAIyX,EAA2B,KAC3B5e,EAAmBK,KAAKqe,oBAAqBvX,GAKjD,MAHiB,KAAbnH,IACH4e,EAAave,KAAKoS,QAAQvM,OAAOlG,IAE3B,CACR,CAQM6e,aAAcxP,EAAoB0L,G,iDAEvC,IAAIV,EAAoBha,KAAK0S,QAAQ1D,GAErC,GAAc,MAAVgL,EACH,MAAM,IAAI1Z,MAAO,mCAAmC0O,qBAErD,OAAQgL,EAAOyE,QAAS/D,EACzB,G,CAOMgE,mBAAoB1P,G,iDAEzB,IAAImP,EAAiBne,KAAKke,oBAG1B,GAAc,MAFUle,KAAK0S,QAAQ1D,GAGpC,MAAM,IAAI1O,MAAO,mCAAmC0O,qBAErD,IAAK,IAAIpL,EAAO,EAAGA,EAAOua,EAAKvgB,OAAQgG,IACvC,CACC,IAAI8W,EAAkByD,EAAKva,SAErB5D,KAAKwe,aAAcxP,EAAY0L,E,CAEvC,G,CAOMiE,mBAAoB3P,G,iDAEzB,IAAImP,EAAiBne,KAAKoe,oBAG1B,GAAc,MAFUpe,KAAK0S,QAAQ1D,GAGpC,MAAM,IAAI1O,MAAO,mCAAmC0O,qBAErD,IAAK,IAAIpL,EAAO,EAAGA,EAAOua,EAAKvgB,OAAQgG,IACvC,CACC,IAAI8W,EAAkByD,EAAKva,SAErB5D,KAAKwe,aAAcxP,EAAY0L,E,CAEvC,G,CAKMrY,QAASmR,EAAkBvS,EAAY,M,iDAE5C,IAAIkB,EAAgBnC,KAAKuT,QAASC,GAGlC,aAF2BrR,EAAKE,QAASpB,EAG1C,G,CAKAF,wBAA+B6d,EAAuB9X,EAAe8X,EAAe3d,EAAY,M,iDAE/F,IAAIiB,EAAqB,IAAIhB,QACzBF,EAAgB,IAAI,UAAAsB,QAAS,CAChC,UAAasc,UAER5d,EAAKyB,OACX,IAAIN,EAAgB,IAAI,UAAAsY,QAAS,CAC/B,UAAavY,EACb,KAAQ4E,EACR,MAAS,CAAC9F,KAKZ,OAHAkB,EAAUoR,QAASnR,SACQD,EAAUG,QAASyE,EAAM7F,EAGrD,G,CAKAF,kBAAyBgX,G,iDAExB,IAAI/W,EAAgB,IAAI,UAAAsB,QAAS,CAChC,IAAOyV,EAAQxV,YAGVvB,EAAKyB,OACX,IAAIN,EAAgB,IAAI,UAAAsY,QAAS,CAC/B,UAAa1C,EAAQ7V,UACrB,KAAQ6V,EAAQjR,KAChB,MAAS,CAAC9F,GACV,WAAc+W,EAAQ/I,WACtB,UAAa+I,EAAQ9I,YAKvB,OAHA8I,EAAQ7V,UAAUoR,QAASnR,SACA4V,EAAQ7V,UAAUG,QAAS0V,EAAQjR,KAAMiR,EAAQ9W,KAG7E,G,CAKAF,sBAA6BgX,G,iDAE5B,IAAI/W,EAAgB,IAAI,UAAAsB,QAAS,CAChC,QAAWyV,EAAQ3V,gBAEdpB,EAAKyB,OACX,IAAIN,EAAgB,IAAI,UAAAsY,QAAS,CAC/B,UAAa1C,EAAQ7V,UACrB,KAAQ6V,EAAQjR,KAChB,MAAS,CAAC9F,KAKZ,OAHA+W,EAAQ7V,UAAUoR,QAASnR,SACA4V,EAAQ7V,UAAUG,QAAS0V,EAAQjR,KAAMiR,EAAQ9W,KAG7E,G,CAMAF,eAAgB8d,GAEc,aAAxBxgB,SAASygB,YAAuD,gBAAxBzgB,SAASygB,WACrDD,IAEAthB,OAAOua,iBAAkB,OAAQ+G,EACnC,CAMA9d,iBAAwBY,G,iDAEtB,IACI4V,GADS,IAAIhD,WACEC,gBAAiB7S,EAAQ,aACbtD,SAAS0gB,qBAAqB,QAAQ,GAE7D/I,UAAYuB,EAAMwH,qBAAqB,QAAQ,GAAG/I,UAI1D,IAAIgJ,EAAa3gB,SAAS0gB,qBAAqB,UAC/C,GAAIC,EAAWphB,OAAS,EAAG,CAG1B,IAAIqhB,EAA+B,GACnC,IAAK,IAAIvhB,EAAI,EAAGA,EAAIshB,EAAWphB,OAAQF,IACtCuhB,EAAQ7P,KAAK4P,EAAWthB,IAIzB,IAAK,IAAIA,EAAI,EAAGA,EAAIuhB,EAAQrhB,OAAQF,IAAK,CACxC,IAAIwhB,EAAuB7gB,SAAS8gB,cAAc,UAGlDF,EAAQvhB,GAAGib,WAAWhD,YAAYuJ,GAGlCD,EAAQvhB,GAAGib,WAAWlB,YAAYwH,EAAQvhB,UAEpC,IAAIqK,SAAe,CAACqX,EAAUC,KAElCH,EAAEI,OAAS,KAETF,GAAW,EAGb,IAAIG,GAAkB,EAEiB,MAAnCN,EAAQvhB,GAAG8hB,aAAc,QAEY,KAApCP,EAAQvhB,GAAG8hB,aAAc,SAE5BN,EAAEO,aAAc,MAAOR,EAAQvhB,GAAG8hB,aAAc,QAChDD,GAAS,GAI6B,MAApCN,EAAQvhB,GAAG8hB,aAAc,SAEa,KAArCP,EAAQvhB,GAAG8hB,aAAc,SAC5BN,EAAEO,aAAc,OAAQR,EAAQvhB,GAAG8hB,aAAc,SAGnDN,EAAElJ,UAAYiJ,EAAQvhB,GAAGsY,WAEV,IAAXuJ,GACHH,GAAW,G,EAIlB,G,CAQAre,wB,iDAEC,MAAqC,IAA9BG,QAAQwe,yBACRxe,QAAQye,KAAM,IAEY,MAA7Bze,QAAQ0e,0BACL1e,QAAQ0e,oBAChB,G,CAKA7e,oBAAqBmB,EAAoB6V,GAExC,GAAI7V,EAAUgL,OAAS,MAAArM,cAAcyL,aAET,MAAvBpK,EAAUoG,UACd,CACiC,MAA5ByP,EAAQ8H,mBACX9H,EAAQ8H,iBAAmB,IAEK,KAA7B9H,EAAQ8H,mBACX9H,EAAQ8H,iBAAmB,yBAE5B,IAAIC,EAAoB,IAAI,YAAAC,UAAW7d,GACnCoG,EAA0B,IAAI,eAAA0X,aAAcjI,EAAQ8H,iBAAkBC,GAC1ExX,EAAUjD,WAAWtC,IAAMuF,EAC3BpG,EAAUoG,UAAYA,C,CAGzB,CAKAvH,0BAA2BmB,GAE1B,IAAIP,EAAiB,GA4DrB,OA1DIO,EAAUgL,OAAS,MAAArM,cAAcyL,cAEpC3K,GACH,m3CAuDS,CACR,CAMAZ,kBAAyBwB,EAA+BuE,EAAe,KACtE5E,EAAqB,KAAMjB,EAAY,M,iDAEvC,OAAO,IAAK8G,SAAkB,CAACC,EAASC,KAEtC/G,QAAQ+e,SAAS,IAAW,0CAE1B,IAAIlI,EAA2B,CAC7B,IAAO,IAMPA,EAAQjR,KAHE,MAARA,EAEkB,iBAAV,EACKvE,EAEAA,EAAIuE,KAGLA,EAEK,KAAjBiR,EAAQjR,OAGViR,EAAQjR,KADY,iBAAV,EACKvE,EAEAA,EAAIuE,MAGA,iBAAV,EACViR,EAAQxV,IAAMA,GAGdwV,EAAQxV,IAAMA,EAAIA,IAED,MAAbL,GAEkB,MAAjBK,EAAIL,YACPA,EAAYK,EAAIL,WAGN,MAARjB,GAEa,MAAZsB,EAAItB,OACPA,EAAOsB,EAAItB,MAGQ,MAAjBsB,EAAI0M,YACP8I,EAAQ9I,UAAY1M,EAAI0M,WAEH,MAAlB1M,EAAIyM,aACP+I,EAAQ/I,WAAazM,EAAIyM,YAEE,MAAxBzM,EAAIsd,mBACP9H,EAAQ8H,iBAAmBtd,EAAIsd,mBAGhB,MAAb3d,IACHA,EAAY,IAAIhB,SAEjBA,QAAQgf,aAAche,EAAW6V,GAEjCA,EAAQ7V,UAAYA,EACpB6V,EAAQ9W,KAAOA,EAEX8W,EAAQxV,IAAIjB,QAAS,aAAe,IACvCyW,EAAQxV,KAAO,qBAEhB,IAAIZ,QAAuBT,QAAQif,WAAYpI,GAE/CpW,GAAUT,QAAQkf,mBAAoBle,SAEhChB,QAAQmf,UAAW1e,GACzBqG,EAAS9F,EACV,KAAE,GAEN,G,CAMAnB,sBAA6BqB,EAAmC0E,EAAe,KAC7E5E,EAAqB,KAAMjB,EAAY,M,iDAExC,OAAO,IAAK8G,SAAkB,CAACC,EAASC,KAEtC/G,QAAQ+e,SAAS,IAAW,0CAE1B,IAAIlI,EAA2B,CAC7B,QAAW,IAMXA,EAAQjR,KAHE,MAARA,EAEsB,iBAAd,EACK,GAEA1E,EAAQ0E,KAGTA,EAEK,KAAjBiR,EAAQjR,OAGViR,EAAQjR,KADgB,iBAAd,EACK,GAEA1E,EAAQ0E,MAGA,iBAAd,EACViR,EAAQ3V,QAAUA,GAGlB2V,EAAQ3V,QAAUA,EAAQA,QAET,MAAbF,GAEsB,MAArBE,EAAQF,YACXA,EAAYE,EAAQF,WAGV,MAARjB,GAEiB,MAAhBmB,EAAQnB,OACXA,EAAOmB,EAAQnB,MAGQ,MAArBmB,EAAQ6M,YACX8I,EAAQ9I,UAAY7M,EAAQ6M,WAEH,MAAtB7M,EAAQ4M,aACX+I,EAAQ/I,WAAa5M,EAAQ4M,YAEE,MAA5B5M,EAAQyd,mBACX9H,EAAQ8H,iBAAmBzd,EAAQyd,mBAGpB,MAAb3d,IACHA,EAAY,IAAIhB,SAEjBA,QAAQgf,aAAche,EAAW6V,GAEjCA,EAAQ7V,UAAYA,EACpB6V,EAAQ9W,KAAOA,EAEf,IAAIU,QAAuBT,QAAQwK,eAAgBqM,SAE7C7W,QAAQmf,UAAW1e,GACzBqG,EAAS9F,EACV,KAAE,GAEN,G,EAGD,GAjkEA,wBAKQ,QAAA+E,QAAkB,SAIlB,QAAA9F,OAAiB,EAIjB,QAAAue,mBAA6B,EAI7B,QAAAE,kBAAyC,KAIzC,QAAA9P,OAA6F,CAAC,EA4iE5E,oBAAf,SACX,CACC,IAAIwQ,gBAAkB,WAErB,IAAIC,YAAcliB,SAAS0gB,qBAAsB,WAoBjD,GAjBA7d,QAAQC,OAAQ,OAGY,IAAjB,aAGV5D,OAAO2D,QAAUsf,WAAWtf,QAE5B3D,OAAOwiB,UAAYS,WAAWT,UAE9BxiB,OAAOkjB,OAASD,WAAWC,OAE3BljB,OAAOuD,IAAM0f,WAAW1f,IAExBvD,OAAO8a,aAAemI,WAAWnI,cAG9BkI,YAAY3iB,OAAS,EACzB,CAEC,IAAI8iB,WAA0BH,YAAY,GAE1ClN,YAAY,W,iDAEV,IAAIsN,QAAU,CAACrR,EAAkBsR,KAE/B,IAAK,IAAIhd,EAAO,EAAGA,EAAOgd,EAAUhjB,OAAQgG,IAC5C,CACC,IAAIuS,EAAmByK,EAAUhd,GAEjC,GAAmC,MAA/B0L,EAAIkQ,aAAcrJ,GACrB,OAAQ7G,EAAIkQ,aAAcrJ,GAE3B,GAA6C,MAAzC7G,EAAIkQ,aAAc,QAAQrJ,KAC7B,OAAQ7G,EAAIkQ,aAAc,QAAQrJ,I,CAGlB,EAGhB0K,SAAmBF,QAASD,WAAY,CAAC,YAAa,WAAY,SAAW,GAC7EI,OAAiBH,QAASD,WAAY,CAAC,YAAc,GACrD5Z,KAAe6Z,QAASD,WAAY,CAAC,UAAY,UACjDzf,KAAe0f,QAASD,WAAY,CAAC,UAAY,KACjDK,WAAqBJ,QAASD,WAAY,CAAC,cAAe,gBAAkB,KAC5E7D,QAAkB8D,QAASD,WAAY,CAAC,WAAY,aAAe,KACnEM,OAAiBL,QAASD,WAAY,CAAC,UAAW,YAAc,KAChE1R,WAAqB2R,QAASD,WAAY,CAAC,cAAe,gBAAkB,yBAC5EzR,UAAoB0R,QAASD,WAAY,CAAC,aAAc,eAAiB,KACzEO,iBAA2BN,QAASD,WAAY,CAAC,sBAAuB,sBAAwB,KAChG/C,mBAA6BgD,QAASD,WAAY,CAAC,uBAAwB,wBAA0B,KACrGQ,oBAA8B,EAC9BC,YAAsB,EACtBC,WAAqBV,WAAW1K,WAAa,GAC7CqL,cAA4G,CAAC,EAC7GC,gBAA4B,GAC5BC,OAA0B,IAAIC,gBAAiBjkB,OAAO2V,SAASqO,QAE9B,MAAjCZ,QAASD,WAAY,CAAC,UACzBG,SAAWF,QAASD,WAAY,CAAC,SAEU,MAAxCC,QAASD,WAAY,CAAC,iBACzBS,YAAa,GAE8D,MAAxER,QAASD,WAAY,CAAC,uBAAwB,yBACjDQ,oBAAqB,GAEtB,IAAIO,iBAA2BF,OAAOliB,IAAK,oBAEnB,MAApBoiB,mBAEHA,iBAAmBC,UAAWD,kBAC9BlkB,OAAOokB,QAAQC,aAAc,GAAI,GAAIH,kBACrCZ,SAAWY,kBAGZ,IAAII,cAAgBxjB,SAAS0gB,qBAAsB,iBAEnD,IAAK,IAAInb,EAAO,EAAGA,EAAOie,cAAcjkB,OAAQgG,IAChD,CAEC,IAAIke,EAA+BD,cAAcje,GAC7Cme,EAAsBpB,QAASmB,EAAiB,CAAC,WACjDE,EAA6BrB,QAASmB,EAAiB,CAAC,iCAElC,MAAtBE,EACH9gB,QAAQ4O,OAA2B,mBAAI,CAAEmD,cAAe+O,GAExD9gB,QAAQ4O,OAAO,GAAGiS,KAAiB,CAAE9O,cAAe+O,E,CAItD,IAECngB,KAAM,iB,CAEP,MAAO2B,GAENtC,QAAQ+gB,aAAc,qB,CAGvB,GAAe,KAAXnB,OACJ,CACC,IAAIoB,EAAoB7jB,SAAS0gB,qBAAsB,kBAEvD,IAAK,IAAInb,EAAO,EAAGA,EAAOse,EAAkBtkB,OAAQgG,IACpD,CAEC,IAAIue,EAAgCD,EAAkBte,GAClDwe,EAAqBzB,QAASwB,EAAkB,CAAC,SACjDE,EAAuB1B,QAASwB,EAAkB,CAAC,cAAe,iBAGtE,GAAIC,IAAetB,OACnB,CAEC,IAAK,IAAIwB,EAAO,EAAGA,EAAOH,EAAiBI,WAAW3kB,OAAQ0kB,IAC9D,CAEC,IAAIE,EAAyBL,EAAiBI,WAAWD,GAEzD,GAAIE,aAAqBvN,aAEiB,UAArCuN,EAAU5N,QAAQtN,cACtB,CACC,IAAImb,EAAqB9B,QAAS6B,EAAW,CAAC,SAC1CE,EAAmB/B,QAAS6B,EAAW,CAAC,aACxCG,EAAuBhC,QAAS6B,EAAW,CAAC,gBAAiB,iBAC7DI,EAAejC,QAAS6B,EAAW,CAAC,SACpCK,EAAoBlC,QAAS6B,EAAW,CAAC,QAEzCC,EAAWnhB,QAAS,MAAQ,GAC/BggB,gBAAgBlS,KAAMqT,GAEvBpB,cAAcoB,GAAc,CAC1BC,SAAUA,QAAY5Z,EACtB6Z,aAAcA,QAAgB7Z,EAC9B8Z,KAAMA,QAAQ9Z,EACdga,IAAKD,QAAa/Z,E,EAMvB,IAAIia,EAAoBxlB,OAAO2V,SAAS8P,SACpCC,EAAmB1lB,OAAO2V,SAAS8P,SAEvC,GAAoB,MAAhBX,EACJ,CACC,MAAMa,EAA4Bb,EAAahhB,cAE/C,GAA2B,SAAtB6hB,GACmB,QAAtBA,GACsB,MAAtBA,EACF,CACC,MAAMC,EAAuBJ,EAAU9X,YAAa,KAEhDkY,GAAgB,IAEnBJ,EAAYA,EAAUzM,UAAW6M,GACjCF,EAAWA,EAAS3M,UAAW6M,G,EAKlC,GAAI7B,gBAAgB1jB,OAAS,EAI5B,IAAK,IAAI0kB,EAAO,EAAGA,EAAOhB,gBAAgB1jB,OAAQ0kB,IAClD,CACC,IAAIc,EAAwB9B,gBAAgBgB,GACxCe,EAA4BD,EAAczkB,QAAS,IAAK,IAE5D,GAAIokB,EAAUzhB,QAAS+hB,IAAsB,EAC7C,CAECN,EAAYK,EAEZ,K,EAMH,GAAgC,MAA5B/B,cAAc0B,GAClB,CACC,GAAyC,MAArC1B,cAAc0B,GAAWL,SAI5B,YAFAnlB,OAAO2V,SAASC,KAAOkO,cAAc0B,GAAWL,UAKjD,GAA6C,MAAzCrB,cAAc0B,GAAWJ,aAI5B,YAFAplB,OAAO2V,SAASC,KAAO,GAAGkO,cAAc0B,GAAWJ,iCAAiCW,UAAWL,MAK5D,MAAhC5B,cAAc0B,GAAWD,MAC5BjC,SAAWQ,cAAc0B,GAAWD,I,CAGtC,K,GAMF7hB,KADW,MAARA,KACIiC,KAAKsJ,MAAOvL,MAEZ,MAAAH,IAAI2D,UAEZ,IAAI8e,eAAyB,EAE7B,GAAmB,KAAfnC,WACJ,CACC,MAAMoC,EAA0BpC,WAAWziB,QAAS,MAAM,IAElC,KAApB6kB,IACHD,eAAgB,E,CAGlB,IAAIE,SAAW,EAGM,MAAjBlmB,OAAY,MACfkmB,SAAW,MAAA3iB,IAAI6D,MAEhB,IAAIzC,UAAqB,MAEE,IAAvBgf,yBAEkB,IAAT,MAAG,KAES,MAAnB,MAAApgB,IAAImB,aAE0B,MAA7B,MAAAnB,IAAImB,YAAYC,YACnBA,UAAY,MAAApB,IAAImB,YAAYC,WAKf,MAAbA,YACHA,UAAY,IAAIhB,SAEjBgB,UAAUgL,KAAOuW,SAEjB,IAAI1L,QAA2B,CAC7BjR,KACA5E,UACAjB,MA6BF,GA1BiB,KAAb4f,YAEgB,IAAfM,YAECN,SAASvf,QAAS,aAAe,IACpCuf,UAAY,qBAGd9I,QAAQxV,IAAMse,UAGE,MAAb5R,YAEH8I,QAAQ9I,UAAYA,UACpB8I,QAAQ/I,WAAaA,YAGJ,MAAdA,aACH+I,QAAQ/I,WAAaA,YAEE,MAApBiS,mBACHlJ,QAAQ8H,iBAAmBoB,kBAEF,MAAtBtD,qBACH5F,QAAQ4F,mBAAqBA,oBAEf,MAAXd,QACJ,CACC,IAAIiD,EAAS,IAAI,YAAAC,UAAW7d,WAE5B,GAAe,KAAX8e,OACH,MAAM,IAAI1gB,MAAO,wBAElB,IAAIojB,EAAiBnmB,OAEH,MAAdwjB,aAGH2C,EAAYnmB,OAAOwjB,aAGpB,IAAI4C,EAAS,IAAID,EAAU7G,SAAUmE,OAAQlB,GAC7C6D,EAAOte,WAAWtC,IAAM4gB,EACxBzhB,UAAUa,IAAM4gB,C,CAGjB,IAAsB,IAAlBJ,cACJ,CACC,GAAiB,KAAb1C,SACH,MAAM,IAAIvgB,MAAO,8EAElBY,QAAQ0iB,WAAY7L,Q,MAIpB7W,QAAQ2iB,eAAgB9L,QAE1B,G,GAAG,G,CAEN,EAGAxa,OAAOumB,WAAavmB,OAAOwmB,SAC3BxmB,OAAOua,iBAAkB,OAAQwI,gB,4aCpzFlC,eACA,SAMA,sBAmBCnb,YAAajD,EAAoBC,EAAoB,MAEpDnC,KAAKkC,UAAYA,EACjBlC,KAAKmC,KAAOA,EACZnC,KAAKua,aAAe,GACpBva,KAAKgkB,eAAiB,CAAC,CACxB,CAOAC,gBAAiBnd,GAEhB,IAAI+D,EAAc/D,EAAKxF,QAAS,KAC5B4iB,EAAmB,GAEnBrZ,GAAO,IAEV/D,EAAOA,EAAKnI,QAAS,KAAM,IAC3BulB,EAAW,KAGZ,IAAIC,EAAmB,yBAAyBD,MAAapd,MAS7D,OARA+D,EAAM/D,EAAKxF,QAAS,KAEhBuJ,GAAO,IAGVsZ,EADArd,EAAOA,EAAKnI,QAAS,KAAM,KAIrB,CACR,CAKMghB,KAAMvM,G,yCAEX,aAAc,IAAIrL,SAAS,CAACC,EAASC,KAEnCoL,YAAY,KAEVrL,GAAU,GACRoL,EAAgB,GAEvB,G,CAKMgR,MAAO3gB,G,yCAEZpB,QAAQgiB,OAAOnlB,MAAOuE,EACvB,G,CAKM6gB,QAAS7gB,G,+CAERzD,KAAKokB,MAAO,GAAG3gB,MACtB,G,CA+BM8gB,OAAQpmB,EAAYqmB,EAAuB,I,yCAEhD,IAAM,EACL,MAAM,IAAIlkB,MAAOkkB,EACnB,G,CAKMC,IAAKC,G,yCAEV,IAAInL,EAAiB,GAErB,IAAK,IAAI3V,EAAO,EAAGA,EAAO8gB,EAAW9mB,OAAQgG,IAC7C,CACC,IAAI+gB,EAAiBD,EAAW9gB,GAC5B+I,EAA0B,KAC1BI,EAAe,GACf5O,EAAgB,GAEpB,GAA2B,iBAAhB,EACX,CAIC,GAHAwO,EAAU3M,KAAKmC,KAAK2K,aAAa6X,GAGlB,MAAXhY,EACH,MAAM,IAAIrM,MAAO,8CAA8CqkB,KAEhE5X,EAAOJ,EAAQI,KACf5O,EAAQwO,EAAQxO,K,CAGjB,GAAIwmB,aAAqB9X,MACzB,CACC,IAAI/F,EAAe6d,EAAU,GAC7BhY,EAAU3M,KAAKmC,KAAK2K,aAAahG,GAGlB,MAAX6F,GAEHA,EAAU,IAAI,EAAAjI,eAAgBoC,GAC9BiG,EAAO4X,EAAU,GACjBxmB,EAAQwmB,EAAU,KAIlB5X,EAAOJ,EAAQI,KACf5O,EAAQwO,EAAQxO,MAEZwmB,EAAU/mB,OAAS,IACtBmP,EAAO4X,EAAU,IAEdA,EAAU/mB,OAAS,IACtBO,EAAQwmB,EAAU,I,CAIrBhY,EAAQI,KAAOA,EACfJ,EAAQxO,MAAQA,EAEhB,IAAI2E,QAAe9C,KAAK4kB,WAAYjY,SAE9B,EAAAzL,QAAQye,KAAM3f,KAAKua,cAEzBhB,EAAQnK,KAAMtM,E,CAGf,OAAO,CACR,G,8HCxKD,8BAcCqC,YAAauD,EAA+B,CAAC,GAE5C1I,KAAK6kB,cAAgBnc,EAAKmc,gBAAiB,EAC3C7kB,KAAK8kB,0BAA4Bpc,EAAKoc,4BAA6B,CACpE,GA0BD,uBAgBC3f,YAAa2B,EAAgCiG,EAAe,GAAI5O,EAAa,MAEtD,iBAAX,GAEV6B,KAAK8G,KAAOA,EACZ9G,KAAK+M,KAAOA,EACZ/M,KAAK7B,MAAQA,IAIb6B,KAAK8G,KAAOA,EAAKA,KACjB9G,KAAK+M,KAAOjG,EAAKiG,MAAQA,EACzB/M,KAAK7B,MAAQ2I,EAAK3I,OAASA,EAE7B,E,qHCnFD,MAAa8c,EAYZ9V,YAAa4f,EAA4D,GAAIC,GAAqB,GAEpE,iBAAlB,GAEVhlB,KAAK+kB,YAAcA,EACnB/kB,KAAKglB,UAAYA,GAIbD,aAAuB9J,GAE1Bjb,KAAK+kB,YAAcA,EAAYA,YAC/B/kB,KAAKglB,UAAYD,EAAYC,YAI7BhlB,KAAK+kB,YAAcA,EAAY5kB,KAC/BH,KAAKglB,UAAYD,EAAYC,UAGhC,EAhCD,uBAqDA,mBAkCC7f,YAAa4V,EAAmG,GAC/G7I,EAAyC,CAAC,EAAGgJ,EAA6B,IAG1E,GAAIH,aAAwBlO,MAC5B,CACC7M,KAAK+a,aAAe,GAEpB,IAAK,IAAInX,EAAO,EAAGA,EAAOmX,EAAand,OAAQgG,IAC/C,CACC,IAAIoX,EAAOD,EAAanX,GAExB5D,KAAK+a,aAAa3L,KAAM,IAAI6L,EAAoBD,G,MAIlD,CACChb,KAAK+a,aAAe,CAAC,EAErB,IAAK,IAAIjd,KAAOid,EAChB,CACC,IAAIC,EAAOD,EAAajd,GAExBkC,KAAK+a,aAAajd,GAAO,IAAImd,EAAoBD,E,EAInDhb,KAAKkb,iBAAmBA,EACxBlb,KAAKkS,MAAQA,CACd,E,uaChID,eAgEA,MAAsB+S,EAmCrB9f,YAAajD,EAAoB4E,EAAc1B,EAC9C6U,EAAuBY,EAA4C,CAAC,GAEpE7a,KAAKkC,UAAYA,EACjBlC,KAAK8G,KAAOA,EACZ9G,KAAKoF,QAAUA,EACfpF,KAAK6a,SAAWA,EAChB7a,KAAKia,OAASA,EACdja,KAAKklB,iBAAkB,EACvBllB,KAAKmlB,cAAe,EACpBnlB,KAAKolB,kBAAmB,CACzB,CA6DMC,c,yCAEL,MAAgC,IAAzBrlB,KAAKklB,uBACL,EAAAhkB,QAAQye,KAAM,GACtB,G,CAKA2F,YAAaP,GAIZ,OAFW/kB,KAAK6a,SAASkK,EAAYrK,SAASxI,MAAM6S,EAAY5iB,KAGjE,CAKAojB,YAAaR,EAA6BtV,GAIzC,OAFWzP,KAAK6a,SAASkK,EAAYrK,SAASxI,MAAM6S,EAAY5iB,MAEnD+M,UAAUO,EACxB,CAKA1O,4BAA6B2Z,EAAiB8K,GAE7C,IACIC,EAAiC,CACnC/K,QAASA,EACTvY,KAAM,GACNY,IAAK,GACL2iB,MAAO,IAELC,EAPsBH,EAAST,YAOF/lB,MAAO,SACpC+H,EAAe4e,EAAK,GAExB,GAAI5e,EAAKnJ,OAAS,EACjB,OAAO,KAER,GAAiB,MAAZmJ,EAAK,IAA4B,MAAZA,EAAK,GAC9B,OAAO,KAER,IAAI6e,EACH,CAACC,EAAiBC,KAEjB,IAAIjb,EAAcgb,EAAQvkB,QAASwkB,GAC/BC,EAAoB,GAQxB,OANIlb,GAAO,IAEVkb,EAAYF,EAAQte,OAAQsD,EAAMib,EAAcloB,QAChDmoB,EAAYA,EAAUC,QAGhB,CAAW,EAGpBP,EAAetjB,KAAOyjB,EAAS7e,EAAM,SACrC0e,EAAe1iB,IAAM6iB,EAAS7e,EAAM,QAEpC,IAAK,IAAInD,EAAO,EAAGA,EAAO+hB,EAAK/nB,OAAQgG,IACvC,CACC,IAAIqiB,EAAqBN,EAAK/hB,GAC1BsiB,EAAuB,CACzBC,IAAK,GACLnL,KAAM,GACN7a,KAAM,IAGR8lB,EAAaA,EAAWD,OACxBE,EAAQlL,KAAO4K,EAASK,EAAY,SACpCC,EAAQC,IAAMP,EAASK,EAAY,QACnCC,EAAQ/lB,KAAOylB,EAASK,EAAY,SAEf,IAAhBC,EAAQlL,MAA+B,IAAfkL,EAAQC,KAA+B,IAAhBD,EAAQ/lB,OAC3D+lB,EAAQ/lB,KAAO8lB,GAEhBR,EAAeC,MAAMtW,KAAM8W,E,CAG5B,OAAO,CACR,CAKME,mBAAoBrB,EAA6Bpd,EACtD0e,EAAkBC,GAA0B,EAAOC,GAAsC,G,yCAEzF,IAAIC,GAAuB,EAE3B,GAAc,MAAV7e,EACH,MAAM,IAAIrH,MAAO,mDAAmDykB,EAAYrK,aAG1D,IAAnB4L,GAE4B,MAA3BtmB,KAAKymB,qBACRD,QAAoBxmB,KAAKymB,mBAAoB1B,EAAapd,EAAQ0e,EAAUE,IAG9E,IAAIzjB,EAAc,KAElB,IAAoB,IAAhB0jB,EACJ,CACC,IAAIE,EAAiC/e,EAAOuI,UAAUmW,GAEtD,GAAsB,MAAlBK,EACH,MAAM,IAAIpmB,MAAO,+BAA+B+lB,qBAEjDvjB,QAAe4jB,EAAe3Z,KAAM/M,KAAKia,O,CAS1C,OANuB,IAAnBqM,GAE0B,MAAzBtmB,KAAK2mB,yBACF3mB,KAAK2mB,iBAAkB5B,EAAapd,EAAQ0e,EAAUvjB,EAAQyjB,IAG/D,CACR,G,CAOMK,oBAAqB7B,G,yCAE1B,IAAIxL,EAAiB,GAGrB,GAAe,MAFWvZ,KAAK6a,SAASkK,EAAYrK,SAGnD,MAAM,IAAIpa,MAAO,kBAAkBykB,EAAYrK,2BAEhD,GAA0B,MAAtB1a,KAAKkC,UAAUa,IAClB,MAAM,IAAIzC,MAAO,yDAElB,IAAIoC,EAAkB1C,KAAKkC,UAAUa,IAAI8C,OAAOkf,EAAYhiB,KAE5D,GAAa,MAATL,EACH,MAAM,IAAIpC,MAAO,sCAAsCykB,EAAYhiB,QAGpE,IAAK,IAAIa,EAAO,EAAGA,EAAOmhB,EAAYW,MAAM9nB,OAAQgG,GAAQ,EAC5D,CACC,IACI6L,EADoBsV,EAAYW,MAAM9hB,GACdzD,KACxBwH,EAAyBjF,EAAMyN,UAAWV,GAE1C4W,EADwBtB,EAAYW,MAAM9hB,EAAO,GACrBzD,KAEhC,GAAc,MAAVwH,EACH,MAAM,IAAIrH,MAAO,yCAAyCmP,YAAmBsV,EAAYrK,WAE1F,IAAI5X,QAAoB9C,KAAKomB,mBAAoBrB,EAAapd,EAAQ0e,GAEtE9M,EAAQnK,KAAMtM,E,CAGf,OAAO,CACR,G,CAKM+jB,oBAAqB9B,EAA6B+B,EACvDR,GAA0B,EAAOC,GAAsC,G,yCAEvE,IAAIC,GAAuB,EACvB7L,EAAsB3a,KAAK6a,SAASkK,EAAYrK,SAGpD,GAAe,MAAXC,EACH,MAAM,IAAIra,MAAO,kBAAkBykB,EAAYrK,2BAEhD,IAAIvY,EAAoBwY,EAAQzI,MAAM6S,EAAY5iB,MAElD,GAAY,MAARA,EACH,MAAM,IAAI7B,MAAO,mBAAmBykB,EAAY5iB,wBAEjDnC,KAAKia,OAAO9X,KAAOA,EAEnB,IAAI4kB,EAAuBD,EAAK3mB,KAC5B6mB,EAAwB7kB,EAAK+M,UAAU6X,IAGpB,IAAnBT,GAE6B,MAA5BtmB,KAAKinB,sBACRT,QAAoBxmB,KAAKinB,oBAAqBlC,EAAa5iB,EAAM2kB,EAAMP,IAGzE,IAAIzjB,EAAc,KAElB,IAAoB,IAAhB0jB,EACJ,CACC,GAAgB,MAAZQ,EAEH,MAAM,IAAI1mB,MAAO,wBAAwBymB,+BAG1CjkB,QAAekkB,EAAUhnB,KAAKia,O,CAS/B,OANuB,IAAnBqM,GAE2B,MAA1BtmB,KAAKknB,0BACFlnB,KAAKknB,kBAAmBnC,EAAaiC,EAAUlkB,EAAQyjB,IAGxD,CACR,G,CAKMY,eAAgBpC,EAA6B5iB,EAAmB2kB,EAAmBX,G,yCAKxF,IAAIiB,EACH,CAACC,EAAelB,EAAamB,KAE5B,IAAIxkB,GAAkB,EAElBgkB,EAAKX,MAAQA,IAChBrjB,GAAS,GAEV,MAAM+H,EAAcic,EAAKX,IAAI7kB,QAAS,KAWtC,OARIuJ,GAAO,GAEaic,EAAKX,IAAI5e,OAAQ,EAAGsD,KAE1Bsb,IAChBrjB,GAAS,GAGJ,CAAQ,EAQbykB,EACFF,IAEA,IAAI9N,EAAoB,GACpBiO,EAAUH,EAAMI,MAAO,sBAE3B,GAAe,MAAXD,EACJ,CACC,IAAIE,EAAYF,EAAQ,GAGxBE,EAAYA,EAAUngB,OAAQ,EAAGmgB,EAAU9pB,QAE3C2b,EAAQnK,KAAMsY,E,CAGf,GAAInO,EAAQ3b,OAAS,EACpB,MAAM,IAAI0C,MAAO,sBAAsB+mB,iDAExC,OAAO,CAAS,EAGdM,EAAkD,KAClD1mB,EAAiB,GAuErB,IArEyD,IAArDmmB,EAAQN,EAAKX,IAAK,0BAErBwB,EAAiBC,GAAoC,kCAEnD5nB,KAAKklB,iBAAkB,QACjBllB,KAAKqlB,aACZ,MAGsC,IAApC+B,EAAQN,EAAKX,IAAK,UAErBllB,EAAOsmB,EAAYT,EAAKX,KAExBwB,EAAiBC,GAAoC,kCAEnD,IAAIxU,EAA0BT,SAAUiV,EAAQ,UAE1C,EAAA1mB,QAAQye,KAAMvM,EACrB,MAGqC,IAAnCgU,EAAQN,EAAKX,IAAK,SAErBllB,EAAOsmB,EAAYT,EAAKX,KAExBwB,EAAiBC,GAAoC,kCAEnD,IAAIP,EAAgBO,EAAQ,SAEtB5nB,KAAKia,OAAO4N,cAAeR,EAClC,MAGuC,IAArCD,EAAQN,EAAKX,IAAK,WAErBllB,EAAOsmB,EAAYT,EAAKX,KAExBwB,EAAiBC,GAAoC,kCAEnD,IAAIP,EAAgBO,EAAQ,SAEtB5nB,KAAKia,OAAOmK,MAAOiD,EAC1B,MAGyC,IAAvCD,EAAQN,EAAKX,IAAK,aAErBllB,EAAOsmB,EAAYT,EAAKX,KAExBwB,EAAiBC,GAAoC,kCAEnD,IAAIP,EAAgBO,EAAQ,SAEtB5nB,KAAKia,OAAOqK,QAAS+C,EAC5B,MAGmD,IAAjDD,EAAQN,EAAKX,IAAK,uBAErBllB,EAAOsmB,EAAYT,EAAKX,KAExBwB,EAAiBC,GAAoC,kCAEnD,IAAIE,EAAqB5kB,KAAKsJ,MAAOob,EAAQ,UAEvC5nB,KAAKia,OAAO8N,mBAAoBD,EACvC,KAGa,MAAXH,EACH,MAAM,IAAIrnB,MAAO,sBAAsBwmB,EAAKX,6BAEvCnmB,KAAKgoB,UAAWjD,EAAa5iB,EAAM2kB,EAAMX,EAAKllB,EAAM0mB,EAC3D,G,CAKMM,qBAAsBlD,EAA6BwB,GAAsC,G,yCAE9F,IAAIhN,EAAiB,GACjBoB,EAAsB3a,KAAK6a,SAASkK,EAAYrK,SAGpD,GAAe,MAAXC,EACH,MAAM,IAAIra,MAAO,kBAAkBykB,EAAYrK,2BAGhD,IAAK,IAAI9W,EAAO,EAAGA,EAAOmhB,EAAYW,MAAM9nB,OAAQgG,IACpD,CACC,IAAIkjB,EAAoB/B,EAAYW,MAAM9hB,GACtCd,EAAc,KACdX,EAAoBwY,EAAQzI,MAAM6S,EAAY5iB,MAElD,GAAY,MAARA,EACH,MAAM,IAAI7B,MAAO,mBAAmBykB,EAAY5iB,wBAEjD,GAAkB,KAAd2kB,EAAK9L,KACT,CACC,GAAIL,EAAQI,wBAAwBlO,MACnC,MAAM,IAAIvM,MAAO,sFAAsFykB,EAAYrK,0BAEpH,IAAI8K,EAA+B7K,EAAQI,aAAa+L,EAAK9L,MACzDyK,EAAiCR,EAAUiD,qBACtCnD,EAAYrK,QAAS8K,GAER,MAAlBC,IACH3iB,QAAe9C,KAAKioB,qBAAsBxC,G,CAG3B,KAAbqB,EAAKX,YACFnmB,KAAKmnB,eAAgBpC,EAAa5iB,EAAM2kB,EAAMA,EAAKX,MAExC,KAAdW,EAAK3mB,OACR2C,QAAe9C,KAAK6mB,oBAAqB9B,EAAa+B,GAAM,EAAOP,IAEpEhN,EAAQnK,KAAMtM,E,CAGf,OAAO,CACR,G,CAKM2b,QAAS/D,G,yCAEd,IAAIE,EAAkB5a,KAAK6a,SAASH,GAEpC,GAAW,MAAPE,EACH,MAAM,IAAIta,MAAO,kBAAkBoa,qBAGpC,IAAIyB,EAAmBnc,KAAKkC,UAAUmc,oBAAqB3D,GACvDnY,EAAc,GAED,KAAb4Z,IACH5Z,EAAM,GAAGvC,KAAKoF,UAAU+W,KAEzB,IAAIgM,EACH,CAAO3C,EAA8B4C,EAAyB,KAAM,kCAEnE,IAA2B,IAAvB5C,EAASR,UACZ,OAED,IAAID,EAA8BE,EAAUiD,qBAAsBxN,EAAS8K,GACvE6C,GAAsB,EACtBC,GAAwB,EAEH,KAArBvD,EAAY5iB,OACfkmB,GAAa,GAEI,MAAdroB,KAAKuoB,QAEkB,IAAtBvoB,KAAKmlB,qBAEFnlB,KAAKuoB,MAAOF,EAAY9lB,EAAK6lB,GACnCpoB,KAAKmlB,cAAe,EACpBnlB,KAAKolB,kBAAmB,GAIF,MAApBplB,KAAKwoB,cACRF,QAAqBtoB,KAAKwoB,YAAazD,EAAaxiB,EAAK6lB,KAErC,IAAjBE,IAEsB,KAArBvD,EAAY5iB,aACTnC,KAAKioB,qBAAsBlD,IAEV,KAApBA,EAAYhiB,YACT/C,KAAK4mB,oBAAqB7B,KAGZ,MAAlB/kB,KAAKyoB,kBACFzoB,KAAKyoB,UAAW1D,IAEH,MAAhB/kB,KAAK0oB,UAEsB,IAA1B1oB,KAAKolB,yBAEFplB,KAAK0oB,UACX1oB,KAAKolB,kBAAmB,EACxBplB,KAAKmlB,cAAe,EAGvB,IAGD,GAAIvK,EAAIG,wBAAwBlO,MAE/B,IAAK,IAAIjJ,EAAO,EAAGA,EAAOgX,EAAIG,aAAand,OAAQgG,IACnD,CACC,IAAI4hB,EAA+B5K,EAAIG,aAAanX,SAE9CukB,EAAoB3C,E,MAM3B,GAAI5K,EAAIM,iBAAiBtd,OAAS,EAClC,CACC,IAAI+qB,EAA4B,GAGhC,IAAK,IAAI/kB,EAAO,EAAGA,EAAOgX,EAAIM,iBAAiBtd,OAAQgG,IACvD,CACC,IAAIglB,EAAmBhO,EAAIM,iBAAiBtX,GACxC4hB,EAA+B5K,EAAIG,aAAa6N,GAEpD,GAAgB,MAAZpD,EACH,MAAM,IAAIllB,MAAO,0BAA0BsoB,qBAE5CD,EAAgBvZ,KAAMwZ,SAChBT,EAAoB3C,EAAUoD,E,CAIrC,IAAK,IAAI9qB,KAAO8c,EAAIG,aACpB,CACC,IAAI8N,GAAuB,EAE3B,IAAK,IAAIjlB,EAAO,EAAGA,EAAO+kB,EAAgB/qB,OAAQgG,IAIjD,GAF0B+kB,EAAgB/kB,KAEtB9F,EACpB,CACC+qB,GAAc,EAEd,K,CAIF,IAAoB,IAAhBA,EACJ,CACC,IAAIrD,EAA+B5K,EAAIG,aAAajd,SAE9CqqB,EAAoB3C,EAAU1nB,E,QAOtC,IAAK,IAAIA,KAAO8c,EAAIG,aACpB,CACC,IAAIyK,EAA+B5K,EAAIG,aAAajd,SAE9CqqB,EAAoB3C,EAAU1nB,E,CASxC,G,EAroBD,a,4eChEA,wCACA,oCAMA,yCAEA,MAAakiB,qBAAqB,SAAAS,OAEjCtb,YAAaC,EAAiBC,EAAoC,KAAMC,EAAU,MAEjF4P,MAAM9P,EAASC,EAAYC,GAE3BtF,KAAKyF,mBAAqB,SAAAP,mBAAmBub,OAE7C,IAAI/d,EAAkB,IAAI,WAAAgD,SAAUL,EAAY,UAChD3C,EAAM+D,UAAW,CACf,KAAQ,aACR,gBAAmBzG,KAAK8oB,WACxB,WAAc,CACZ,WAAc,CACb,UAAY,EACZ,KAAQ,SACR,YAAe,8CAEhB,UAAa,CACZ,UAAY,EACZ,KAAQ,SACR,YAAe,sCAEhB,SAAY,CACX,UAAY,EACZ,KAAQ,SACR,YAAe,4CAEhB,aAAgB,CACf,UAAY,EACZ,KAAQ,QACR,YAAe,kCAEhB,UAAa,CACZ,UAAY,EACZ,KAAQ,QACR,YAAe,gCAGlB,QAAW,wCAEbpmB,EAAM+D,UAAW,CACf,KAAQ,eACR,gBAAmBzG,KAAKwe,aACxB,WAAc,CACb,WAAc,CACb,UAAY,EACZ,KAAQ,SACR,YAAe,8CAEhB,UAAa,CACZ,UAAY,EACZ,KAAQ,SACR,YAAe,+BAGjB,QAAW,0CAEb9b,EAAM+D,UAAW,CACf,KAAQ,YACR,KAAQ,iBAAAuJ,WAAW+Y,IACnB,gBAAmB/oB,KAAKgpB,UACxB,QAAW,wCAEbhpB,KAAKoG,SAAU1D,EAChB,CAKMomB,WAAY9K,IAAUvc,IAAUwnB,gBAAsB9gB,QAAc+gB,U,iDAEzE,IAAIC,UAMC,CACHna,WAAY7G,QAAoB,WAChC8G,UAAW9G,QAAmB,UAC9BqL,SAAUrL,QAAkB,SAC5B2E,aAAc3E,QAAsB,aACpCihB,cAAejhB,QAAmB,WAGpC,IAAK,IAAIrK,KAAOqrB,UAChB,CAEC,IAAIE,EAAeF,UAAUrrB,GACzBwrB,GAAsB,EAa1B,GAXe,MAAXD,IACHC,GAAa,GAEe,IAAxBH,UAAUna,YACW,KAAxBma,UAAUla,WACiB,KAA3Bka,UAAUrc,cACkB,KAA5Bqc,UAAUC,gBAEXE,GAAa,IAGK,IAAfA,EACH,MAAM,IAAIhpB,MAAO,qBAAqBxC,oB,CAGxCqrB,UAAUrc,aAAe5J,KAAKsJ,MAAO2c,UAAUrc,cAC/Cqc,UAAUC,cAAgBlmB,KAAKsJ,MAAO2c,UAAUC,eAEhD,IAAIla,UAA8C,CAAC,EAEnD,IAAK,IAAIpR,OAAOqrB,UAAUC,cAC1B,CACC,IAAIpC,SACHnlB,KAAMsnB,UAAUC,cAActrB,MAE/BoR,UAAUpR,KAAOkpB,Q,CAGlB,IAAIhN,OAAoBha,KAAKqF,WAAWnD,UAAUwQ,QAAQyW,UAAUna,YAEpE,GAAc,MAAVgL,OACH,MAAM,IAAI1Z,MAAO,qBAAqB6oB,UAAUla,6BAEjD,IAAI0L,QAAsBX,OAAOa,SAASsO,UAAUla,WAEpD,GAAe,MAAX0L,QACH,MAAM,IAAIra,MAAO,yBAAyB6oB,UAAUla,6BAcrD,OAZA0L,QAAQzI,MAAMiX,UAAU3V,UAAY,CAClC,aAAgB,CAAC,EACjB,UAAa,CAAC,GAEhBmH,QAAQzI,MAAMiX,UAAU3V,UAAU1G,aAAeqc,UAAUrc,aAC3D6N,QAAQzI,MAAMiX,UAAU3V,UAAUtE,UAAYA,UAE9C8K,OAAOkL,iBAAkB,EAEO,MAA5BlL,OAAOuP,0BACJvP,OAAOuP,sBAEP,CACR,G,CAKM/K,aAAcR,EAAUvc,EAAUwnB,EAAsB9gB,EAAc+gB,G,iDAE3E,IAAIla,EAAqB7G,EAAoB,WACzC8G,EAAoB9G,EAAmB,UAE3C,GAAmB,MAAd6G,GAAqC,MAAbC,EAC5B,MAAM,IAAI3O,MAAO,yDAElB,GAAoB,KAAf0O,GAAqC,KAAdC,EAC3B,MAAM,IAAI3O,MAAO,yDAElB,IAAIkc,EAAgCxc,KAAKqF,WASzC,OAN2B,MAAvBmX,EAAOgC,qBAGJhC,EAAOgC,aAAcxP,EAAYC,KAGjC,CACR,G,CAKM+Z,UAAWhL,EAAUvc,EAAUwnB,EAAsB9gB,EAAc+gB,G,iDAExE,OAAO,CACR,G,EAjLD,iC,8ECTA,eAEA,SACA,SACA,SACA,SACA,SAGA,SACA,SACA,QACA,SACA,SAGA,SACA,SACA,SACA,SACA,SAEA,EAAAhoB,QAAQC,OAAQ,EAIhBhE,EAAOC,QAAP,QAA4B,EAAA8D,QAC5B/D,EAAOC,QAAP,IAAwB,EAAA0D,IACxB3D,EAAOC,QAAP,cAAkC,EAAAyD,cAClC1D,EAAOC,QAAP,aAAiC,EAAAib,aACjClb,EAAOC,QAAP,OAA2B,EAAAqjB,OAC3BtjB,EAAOC,QAAP,mBAAuC,EAAA8H,mBACvC/H,EAAOC,QAAP,QAA4B,EAAAkF,QAC5BnF,EAAOC,QAAP,OAA2B,EAAAoV,OAC3BrV,EAAOC,QAAP,YAAgC,EAAA4Q,YAChC7Q,EAAOC,QAAP,QAA4B,EAAAqd,QAC5Btd,EAAOC,QAAP,SAA6B,EAAAsI,SAC7BvI,EAAOC,QAAP,eAAmC,EAAAoJ,eACnCrJ,EAAOC,QAAP,WAA+B,EAAA4S,WAC/B7S,EAAOC,QAAP,UAA8B,EAAAgK,UAC9BjK,EAAOC,QAAP,cAAkC,EAAAmL,cAClCpL,EAAOC,QAAP,UAA8B,EAAA2iB,UAC9B5iB,EAAOC,QAAP,UAA8B,EAAA6nB,UAC9B9nB,EAAOC,QAAP,aAAiC,EAAA4iB,aACjC7iB,EAAOC,QAAP,WAA+B,EAAA0d,WAC/B3d,EAAOC,QAAP,mBAAuC,EAAA6d,mBACvC9d,EAAOC,QAAP,eAAmC,EAAAsH,eACnCvH,EAAOC,QAAP,sBAA0C,EAAAosB,sBAC1CrsB,EAAOC,QAAP,cAAkC,EAAAqsB,a,uBChDlCtsB,EAAOC,QAAU,CAAC,C,yBCCdssB,yBAA2B,CAAC,EAGhC,SAASC,oBAAoBC,GAE5B,IAAIC,EAAeH,yBAAyBE,GAC5C,QAAqB9gB,IAAjB+gB,EACH,OAAOA,EAAazsB,QAGrB,IAAID,EAASusB,yBAAyBE,GAAY,CAGjDxsB,QAAS,CAAC,GAOX,OAHA0sB,oBAAoBF,GAAU/V,KAAK1W,EAAOC,QAASD,EAAQA,EAAOC,QAASusB,qBAGpExsB,EAAOC,OACf,CCnBA,IAAI2sB,oBAAsBJ,oBAAoB,K","sources":["webpack://HotStaqWeb/./node_modules/form-data/lib/browser.js","webpack://HotStaqWeb/./node_modules/js-cookie/dist/js.cookie.js","webpack://HotStaqWeb/./node_modules/node-fetch/browser.js","webpack://HotStaqWeb/./src/Hot.ts","webpack://HotStaqWeb/./src/HotAPI.ts","webpack://HotStaqWeb/./src/HotClient.ts","webpack://HotStaqWeb/./src/HotComponent.ts","webpack://HotStaqWeb/./src/HotFile.ts","webpack://HotStaqWeb/./src/HotLog.ts","webpack://HotStaqWeb/./src/HotPage.ts","webpack://HotStaqWeb/./src/HotRoute.ts","webpack://HotStaqWeb/./src/HotRouteMethod.ts","webpack://HotStaqWeb/./src/HotServer.ts","webpack://HotStaqWeb/./src/HotStaq.ts","webpack://HotStaqWeb/./src/HotTestDriver.ts","webpack://HotStaqWeb/./src/HotTestElement.ts","webpack://HotStaqWeb/./src/HotTestMap.ts","webpack://HotStaqWeb/./src/HotTester.ts","webpack://HotStaqWeb/./src/HotTesterAPI.ts","webpack://HotStaqWeb/./src/api-web.ts","webpack://HotStaqWeb/external var \"{}\"","webpack://HotStaqWeb/webpack/bootstrap","webpack://HotStaqWeb/webpack/startup"],"sourcesContent":["/* eslint-env browser */\nmodule.exports = typeof self == 'object' ? self.FormData : window.FormData;\n","/*! js-cookie v3.0.1 | MIT */\n;\n(function (global, factory) {\n  typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :\n  typeof define === 'function' && define.amd ? define(factory) :\n  (global = global || self, (function () {\n    var current = global.Cookies;\n    var exports = global.Cookies = factory();\n    exports.noConflict = function () { global.Cookies = current; return exports; };\n  }()));\n}(this, (function () { 'use strict';\n\n  /* eslint-disable no-var */\n  function assign (target) {\n    for (var i = 1; i < arguments.length; i++) {\n      var source = arguments[i];\n      for (var key in source) {\n        target[key] = source[key];\n      }\n    }\n    return target\n  }\n  /* eslint-enable no-var */\n\n  /* eslint-disable no-var */\n  var defaultConverter = {\n    read: function (value) {\n      if (value[0] === '\"') {\n        value = value.slice(1, -1);\n      }\n      return value.replace(/(%[\\dA-F]{2})+/gi, decodeURIComponent)\n    },\n    write: function (value) {\n      return encodeURIComponent(value).replace(\n        /%(2[346BF]|3[AC-F]|40|5[BDE]|60|7[BCD])/g,\n        decodeURIComponent\n      )\n    }\n  };\n  /* eslint-enable no-var */\n\n  /* eslint-disable no-var */\n\n  function init (converter, defaultAttributes) {\n    function set (key, value, attributes) {\n      if (typeof document === 'undefined') {\n        return\n      }\n\n      attributes = assign({}, defaultAttributes, attributes);\n\n      if (typeof attributes.expires === 'number') {\n        attributes.expires = new Date(Date.now() + attributes.expires * 864e5);\n      }\n      if (attributes.expires) {\n        attributes.expires = attributes.expires.toUTCString();\n      }\n\n      key = encodeURIComponent(key)\n        .replace(/%(2[346B]|5E|60|7C)/g, decodeURIComponent)\n        .replace(/[()]/g, escape);\n\n      var stringifiedAttributes = '';\n      for (var attributeName in attributes) {\n        if (!attributes[attributeName]) {\n          continue\n        }\n\n        stringifiedAttributes += '; ' + attributeName;\n\n        if (attributes[attributeName] === true) {\n          continue\n        }\n\n        // Considers RFC 6265 section 5.2:\n        // ...\n        // 3.  If the remaining unparsed-attributes contains a %x3B (\";\")\n        //     character:\n        // Consume the characters of the unparsed-attributes up to,\n        // not including, the first %x3B (\";\") character.\n        // ...\n        stringifiedAttributes += '=' + attributes[attributeName].split(';')[0];\n      }\n\n      return (document.cookie =\n        key + '=' + converter.write(value, key) + stringifiedAttributes)\n    }\n\n    function get (key) {\n      if (typeof document === 'undefined' || (arguments.length && !key)) {\n        return\n      }\n\n      // To prevent the for loop in the first place assign an empty array\n      // in case there are no cookies at all.\n      var cookies = document.cookie ? document.cookie.split('; ') : [];\n      var jar = {};\n      for (var i = 0; i < cookies.length; i++) {\n        var parts = cookies[i].split('=');\n        var value = parts.slice(1).join('=');\n\n        try {\n          var foundKey = decodeURIComponent(parts[0]);\n          jar[foundKey] = converter.read(value, foundKey);\n\n          if (key === foundKey) {\n            break\n          }\n        } catch (e) {}\n      }\n\n      return key ? jar[key] : jar\n    }\n\n    return Object.create(\n      {\n        set: set,\n        get: get,\n        remove: function (key, attributes) {\n          set(\n            key,\n            '',\n            assign({}, attributes, {\n              expires: -1\n            })\n          );\n        },\n        withAttributes: function (attributes) {\n          return init(this.converter, assign({}, this.attributes, attributes))\n        },\n        withConverter: function (converter) {\n          return init(assign({}, this.converter, converter), this.attributes)\n        }\n      },\n      {\n        attributes: { value: Object.freeze(defaultAttributes) },\n        converter: { value: Object.freeze(converter) }\n      }\n    )\n  }\n\n  var api = init(defaultConverter, { path: '/' });\n  /* eslint-enable no-var */\n\n  return api;\n\n})));\n","\"use strict\";\n\n// ref: https://github.com/tc39/proposal-global\nvar getGlobal = function () {\n\t// the only reliable means to get the global object is\n\t// `Function('return this')()`\n\t// However, this causes CSP violations in Chrome apps.\n\tif (typeof self !== 'undefined') { return self; }\n\tif (typeof window !== 'undefined') { return window; }\n\tif (typeof global !== 'undefined') { return global; }\n\tthrow new Error('unable to locate global object');\n}\n\nvar global = getGlobal();\n\nmodule.exports = exports = global.fetch;\n\n// Needed for TypeScript and Webpack.\nif (global.fetch) {\n\texports.default = global.fetch.bind(global);\n}\n\nexports.Headers = global.Headers;\nexports.Request = global.Request;\nexports.Response = global.Response;","import { HotFile } from \"./HotFile\";\nimport { HotPage } from \"./HotPage\";\nimport { HotStaq } from \"./HotStaq\";\nimport { HotAPI } from \"./HotAPI\";\nimport { HotTestElement } from \"./HotTestElement\";\n\nimport Cookies from \"js-cookie\";\nimport fetch from \"node-fetch\";\n\n/**\n * The available developer modes.\n */\nexport enum DeveloperMode\n{\n\t/**\n\t * The default developer mode. No tests will be executed and \n\t * any test related data will be ignored.\n\t */\n\tProduction,\n\t/**\n\t * For use during development/debugging. All test data will \n\t * be collected and executed if necessary.\n\t */\n\tDevelopment\n}\n\n/**\n * A CSS object to embed.\n */\nexport interface CSSObject\n{\n\t/**\n\t * The url to the CSS file to embed.\n\t */\n\turl: string;\n\t/**\n\t * The integrity hash to generate during initial compilation.\n\t */\n\tintegrityHash: string;\n}\n\n/**\n * The api used during processing.\n */\nexport class Hot\n{\n\t/**\n\t * The currently generated page being displayed. This is cleared between every file processed.\n\t */\n\tstatic CurrentPage: HotPage = null;\n\t/**\n\t * The arguments passed.\n\t */\n\tstatic Arguments: any = null;\n\t/**\n\t * The mode in which this application is running. If it's set to development mode, all testing\n\t * related data will be collected, parsed, and executed if necessary.\n\t */\n\tstatic DeveloperMode = DeveloperMode;\n\t/**\n\t * The mode in which this application is running. If it's set to development mode, all testing\n\t * related data will be collected, parsed, and executed if necessary.\n\t */\n\tstatic HotTestElement = HotTestElement;\n\t/**\n\t * The mode in which this application is running. If it's set to development mode, all testing\n\t * related data will be collected, parsed, and executed if necessary.\n\t */\n\tstatic Mode: DeveloperMode = DeveloperMode.Production;\n\t/**\n\t *The current API used on this page. This is cleared between every file processed.\n\t */\n\tstatic API: HotAPI = null;\n\t/**\n\t * The API being used by the tester.\n\t */\n\tstatic TesterAPI: HotAPI = null;\n\t/**\n\t * Contains the buffer to output. This is cleared between every file processed.\n\t */\n\tstatic Output: string = \"\";\n\t/**\n\t * The data to share across all the different files and pages. This data will be public.\n\t */\n\tstatic Data: any = {};\n\t/**\n\t * The cookies to use between pages.\n\t */\n\tstatic Cookies: Cookies.CookiesStatic = Cookies;\n\t/**\n\t * Any public keys that need to be shown. These can be passed from HotSite.json.\n\t */\n\tstatic PublicKeys: any = {};\n\t/**\n\t * The CSS string to use when echoing out the CSS files.\n\t */\n\tstatic cssStr: string = `<link rel = \"stylesheet\" href = \"%CSS_FILE%\" />`;\n\t/**\n\t * The CSS files to use in the current page being generated.\n\t * \n\t * @todo Make this a \"string | CSSObject\" data type so it can also include \n\t * the integrity hashes as well.\n\t */\n\tstatic CSS: string[] = [];\n\t/**\n\t * The JavaScript files to use in the current page being generated.\n\t * \n\t * @todo Make this a \"string | JSFileObject\" data type so it can also include \n\t * the integrity hashes as well.\n\t */\n\tstatic JSFiles: any[] = [];\n\t/**\n\t * The JavaScript inline code to use in the current page being generated.\n\t */\n\tstatic JSScripts: any[] = [];\n\t/**\n\t * The JavaScript string to use when echoing out the Scripts files.\n\t */\n\tstatic jsFileStr: string = `<script type = \"text/javascript\" src = \"%JS_FILE%\"></script>`;\n\t/**\n\t * The JavaScript string to use when echoing out the Scripts files.\n\t */\n\tstatic jsScriptsStr: string = `<script type = \"text/javascript\">%JS_CODE%</script>`;\n\n\t/**\n\t * Retrieve a file and echo out it's contents.\n\t */\n\tstatic async include (file: HotFile | string, args: any[] = null): Promise<void>\n\t{\n\t\tif (HotStaq.isWeb === true)\n\t\t{\n\t\t\tif (typeof (file) === \"string\")\n\t\t\t{\n\t\t\t\tconst lowerFile: string = file.toLowerCase ();\n\n\t\t\t\t// If the file to be included does not have a nahfam, add it. This \n\t\t\t\t// will ensure the server sends only the file content.\n\t\t\t\tif (lowerFile.indexOf (\".hott\") > -1)\n\t\t\t\t{\n\t\t\t\t\tif (lowerFile.indexOf (\"nahfam\") < 0)\n\t\t\t\t\t\tfile += \"?hstqserve=nahfam\";\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tHot.echo (await Hot.getFile (file, args));\n\t}\n\n\t/**\n\t * Include and execute JavaScript for use when running the preprocessor.\n\t */\n\tstatic async includeJS (file: string): Promise<any>\n\t{\n\t\tconst res = await Hot.httpRequest (file);\n\t\tconst output: string = await res.text ();\n\n\t\tif (HotStaq.isWeb === true)\n\t\t\teval.apply (window, [output]);\n\t\telse\n\t\t\teval (output);\n\t}\n\n\t/**\n\t * Run an already loaded file and echo out it's contents.\n\t */\n\tstatic async runFile (fileName: string, args: any[] = null): Promise<void>\n\t{\n\t\tlet file: HotFile = Hot.CurrentPage.processor.getFile (fileName);\n\t\t/// @fixme Does the file need to be deep cloned first?\n\t\t//let clonedFile: HotFile = new HotFile (Object.assign ({}, file));\n\t\tlet tempFile: HotFile = file;\n\n\t\ttempFile.page = this.CurrentPage;\n\t\tlet content: string = await tempFile.process (args);\n\n\t\tHot.echo (content);\n\t}\n\n\t/**\n\t * Get the content of a file.\n\t */\n\tstatic async getFile (path: HotFile | string, args: any[] = null): Promise<string>\n\t{\n\t\tlet tempFile: HotFile = null;\n\n\t\tif (typeof (path) === \"string\")\n\t\t{\n\t\t\ttempFile = new HotFile ();\n\n\t\t\tif (HotStaq.isWeb === true)\n\t\t\t\ttempFile.url = path;\n\t\t\telse\n\t\t\t\ttempFile.localFile = path;\n\t\t}\n\t\telse\n\t\t\ttempFile = path;\n\n\t\tawait tempFile.load ();\n\n\t\ttempFile.page = this.CurrentPage;\n\t\tlet content: string = await tempFile.process (args);\n\n\t\treturn (content);\n\t}\n\n\t/**\n\t * Make an api call.\n\t */\n\tstatic async apiCall (route: string, data: any = null, \n\t\thttpMethod: string = \"POST\", \n\t\tfiles: { [name: string]: any } = {}): Promise<any>\n\t{\n\t\tlet result: any = null;\n\n\t\tif (Hot.CurrentPage == null)\n\t\t\tthrow new Error (\"Current page is null!\");\n\n\t\tif (Hot.CurrentPage.processor == null)\n\t\t\tthrow new Error (\"Current page's processor is null!\");\n\n\t\tif (Hot.CurrentPage.processor.api == null)\n\t\t\tthrow new Error (\"Current page's processor api is null! Did you forget to set the API name or URL?\");\n\n\t\tif (Hot.CurrentPage.processor.api != null)\n\t\t{\n\t\t\tresult = await Hot.CurrentPage.processor.api.makeCall (route, \n\t\t\t\t\t\t\tdata, httpMethod, files);\n\t\t}\n\n\t\treturn (result);\n\t}\n\n\t/**\n\t * Make a HTTP JSON request.\n\t * \n\t * @param url The full url to make the HTTP call.\n\t * @param data The data to JSON.stringify and send.\n\t * @param httpMethod The HTTP method to use to send the data.\n\t * \n\t * @returns The parsed JSON object.\n\t */\n\tstatic async jsonRequest (url: string, data: any = null, httpMethod: string = \"POST\"): Promise<any>\n\t{\n\t\ttry\n\t\t{\n\t\t\tlet fetchObj = {\n\t\t\t\t\"method\": httpMethod,\n\t\t\t\t\"headers\": {\n\t\t\t\t\t\t\"Accept\": \"application/json\",\n\t\t\t\t\t\t\"Content-Type\": \"application/json\"\n\t\t\t\t\t}\n\t\t\t}\n\n\t\t\tif (httpMethod === \"POST\")\n\t\t\t{\n\t\t\t\t/// @ts-ignore\n\t\t\t\tfetchObj[\"body\"] = JSON.stringify (data);\n\t\t\t}\n\n\t\t\tlet res = await fetch (url, fetchObj);\n\n\t\t\tif (res.ok === false)\n\t\t\t\tthrow new Error (`${res.status}: ${res.statusText}`);\n\n\t\t\tlet result: any = await res.json ();\n\n\t\t\treturn (result);\n\t\t}\n\t\tcatch (ex)\n\t\t{\n\t\t\treturn (JSON.stringify ({ \"error\": `${ex.message} - Could not fetch ${url}` }));\n\t\t}\n\t}\n\n\t/**\n\t * Make a HTTP request. This is basically just a wrapper for fetch.\n\t * \n\t * @param {string} url The full url to make the HTTP call.\n\t * @param {RequestInit} requestInit The request parameters to send.\n\t * \n\t * @returns The HTTP response.\n\t */\n\tstatic async httpRequest (url: string, requestInit: any = undefined): Promise<any>\n\t{\n\t\tlet res = await fetch (url, requestInit);\n\n\t\treturn (res);\n\t}\n\n\t/**\n\t * Echo out some output.\n\t */\n\tstatic echo (message: string): void\n\t{\n\t\tHot.Output += message;\n\t}\n\n\t/**\n\t * Echo out the CSS for the current page being generated.\n\t */\n\tstatic displayCSS (): void\n\t{\n\t\tfor (let iIdx = 0; iIdx < Hot.CSS.length; iIdx++)\n\t\t{\n\t\t\tlet cssFile: string = Hot.CSS[iIdx];\n\t\t\tlet cssOut: string = Hot.cssStr;\n\n\t\t\tcssOut = cssOut.replace (/\\%CSS_FILE\\%/g, cssFile);\n\n\t\t\tHot.echo (cssOut);\n\t\t}\n\t}\n\n\t/**\n\t * Echo out the JS files for the current page being generated.\n\t */\n\tstatic displayJSFiles (): void\n\t{\n\t\tfor (let iIdx = 0; iIdx < Hot.JSFiles.length; iIdx++)\n\t\t{\n\t\t\tlet jsFile: string = Hot.JSFiles[iIdx];\n\t\t\tlet jsFileOut: string = Hot.jsFileStr;\n\n\t\t\tjsFileOut = jsFileOut.replace (/\\%JS_FILE\\%/g, jsFile);\n\n\t\t\tHot.echo (jsFileOut);\n\t\t}\n\t}\n\n\t/**\n\t * Echo out the JS scripts for the current page being generated.\n\t */\n\tstatic displayJSScripts (): void\n\t{\n\t\tfor (let iIdx = 0; iIdx < Hot.JSScripts.length; iIdx++)\n\t\t{\n\t\t\tlet jsScript: string = Hot.JSScripts[iIdx];\n\t\t\tlet jsScriptOut: string = Hot.jsScriptsStr;\n\n\t\t\tjsScriptOut = jsScriptOut.replace (/\\%JS_CODE\\%/g, jsScript);\n\n\t\t\tHot.echo (jsScriptOut);\n\t\t}\n\t}\n}","import fetch from \"node-fetch\";\nimport FormData from \"form-data\";\n\nimport { HotServer } from \"./HotServer\";\nimport { HotRoute } from \"./HotRoute\";\nimport { HotClient } from \"./HotClient\";\nimport { HotRouteMethod, ServerAuthorizationFunction } from \"./HotRouteMethod\";\nimport { HotDB } from \"./HotDB\";\n\nimport { HotDBSchema } from \"./schemas/HotDBSchema\";\n\n/**\n * The API to load.\n */\nexport type APItoLoad = {\n\texportedClassName: string;\n\tpath: string;\n };\n\n/**\n * The type of object to use during event executions.\n */\nexport enum EventExecutionType\n{\n\tHotRoute,\n\tHotMethod,\n\tHotAPI\n}\n\n/**\n * The API to use.\n */\nexport abstract class HotAPI\n{\n\t/**\n\t * The server connection.\n\t */\n\tconnection: HotServer | HotClient;\n\t/**\n\t * The description of the API.\n\t */\n\tdescription: string;\n\t/**\n\t * The base url for the server.\n\t */\n\tbaseUrl: string;\n\t/**\n\t * If set, this will create the route variables and functions for \n\t * easy client/server calling.\n\t */\n\tcreateFunctions: boolean;\n\t/**\n\t * The database connection.\n\t */\n\texecuteEventsUsing: EventExecutionType;\n\t/**\n\t * The database connection.\n\t */\n\tdb: HotDB;\n\t/**\n\t * The authorization credentials to use throughout the application. If using \n\t * basic authentication for HTTP requests, be sure to override the \n\t * `toAuthorizationHeaderString` in this object.\n\t * \n\t * Example of this object:\n\t * ```ts\n\t * this.authCredentials = {\n\t * \t\tuser: \"username\",\n\t * \t\tpassword: \"password\",\n\t * \t\ttoAuthorizationHeaderString: function ()\n\t * \t\t\t{\n\t * \t\t\t\treturn (btoa (this.user + \":\" + this.password));\n\t * \t\t\t}\n\t * \t};\n\t * ```\n\t */\n\tauthCredentials: any;\n\t/**\n\t * The function used for user authentication.\n\t */\n\tuserAuth: ServerAuthorizationFunction;\n\t/**\n\t * The database connection.\n\t */\n\troutes: { [name: string]: HotRoute };\n\t/**\n\t * Executed when the API is about to start registering routes. If \n\t * this function returns false, the server will not start.\n\t */\n\tonPreRegister: () => Promise<boolean>;\n\t/**\n\t * Executed when the API has finished registering routes. If \n\t * this function returns false, the server will not start.\n\t */\n\tonPostRegister: () => Promise<boolean>;\n\n\tconstructor (baseUrl: string, connection: HotServer | HotClient = null, db: HotDB = null)\n\t{\n\t\tthis.connection = connection;\n\t\tthis.description = \"\";\n\t\tthis.baseUrl = baseUrl;\n\t\tthis.createFunctions = true;\n\t\tthis.executeEventsUsing = EventExecutionType.HotRoute;\n\t\tthis.db = db;\n\t\tthis.authCredentials = null;\n\t\tthis.userAuth = null;\n\t\tthis.routes = {};\n\t\tthis.onPreRegister = null;\n\t\tthis.onPostRegister = null;\n\t}\n\n\t/**\n\t * Set the database schema for use.\n\t */\n\tsetDBSchema (schema: HotDBSchema): void\n\t{\n\t\tif (this.connection.api == null)\n\t\t\tthrow new Error (`No API has been set!`);\n\n\t\tif (this.connection.api.db == null)\n\t\t\tthrow new Error (`No database has been set for API base url ${this.connection.api.baseUrl}`);\n\n\t\tthis.connection.api.db.schema = schema;\n\t}\n\n\t/**\n\t * Get the database being used.\n\t */\n\tgetDB (): HotDB\n\t{\n\t\tif (this.connection.api.db == null)\n\t\t\tthrow new Error (`No database has been set for API base url ${this.connection.api.baseUrl}`);\n\n\t\treturn (this.connection.api.db);\n\t}\n\n\t/**\n\t * Get the database schema being used.\n\t */\n\tgetDBSchema (): HotDBSchema\n\t{\n\t\tif (this.connection.api.db == null)\n\t\t\tthrow new Error (`No database has been set for API base url ${this.connection.api.baseUrl}`);\n\n\t\treturn (this.connection.api.db.schema);\n\t}\n\n\t/**\n\t * Add a route. If this.createFunctions is set to true, this will take the incoming \n\t * route and create an object in this HotAPI object using the name of the route. If there's \n\t * any HotRouteMethods inside of the incoming HotRoute, it will create the methods \n\t * and attach them to the newly created HotRoute object.\n\t * \n\t * Example:\n\t * ```\n\t * export class Users extends HotRoute\n\t * {\n\t * \t\tconstructor (api: FreeLightAPI)\n\t * \t\t{\n\t * \t\t\tsuper (api.connection, \"user\");\n\t * \n\t * \t\t\tthis.addMethod (\"create\", this._create, HTTPMethod.POST);\n\t * \t\t}\n\t * \n\t * \t\tprotected async _create (req: any, res: any, authorizedValue: any, jsonObj: any, queryObj: any): Promise<any>\n\t * \t\t{\n\t * \t\t\treturn (true);\n\t * \t\t}\n\t * }\n\t * ```\n\t * \n\t * This in turn could be used like so:\n\t * ```\n\t * Hot.API.user.create ({});\n\t * ```\n\t * \n\t * Additionally it would create the endpoint: ```http://127.0.0.1:8080/v1/user/create```\n\t * \n\t * @param route The route to add. Can be either a full HotRoute object, or just \n\t * the route's name. If a HotRoute object is supplied, the rest of the parameters \n\t * will be ignored.\n\t * @param routeMethod The route's method to add. If the route parameter is a string, \n\t * it will be interpreted as the route's name, and this will be the method added to \n\t * the new route.\n\t * @param executeFunction The function to execute when routeMethod is called by the API.\n\t */\n\taddRoute (\n\t\troute: HotRoute | string,\n\t\trouteMethod: HotRouteMethod | string = null,\n\t\texecuteFunction: (req: any, res: any, authorizedValue: any, jsonObj: any, queryObj: any) => Promise<any> = null\n\t\t): void\n\t{\n\t\tlet routeName: string = \"\";\n\n\t\tif (route instanceof HotRoute)\n\t\t{\n\t\t\trouteName = route.route;\n\t\t\tthis.routes[route.route] = route;\n\t\t}\n\t\telse\n\t\t{\n\t\t\trouteName = route;\n\n\t\t\tif (this.routes[routeName] == null)\n\t\t\t\tthis.routes[routeName] = new HotRoute (this.connection, routeName);\n\n\t\t\tif (routeMethod instanceof HotRouteMethod)\n\t\t\t\tthis.routes[routeName].addMethod (routeMethod);\n\t\t\telse\n\t\t\t{\n\t\t\t\tthis.routes[routeName].addMethod (new HotRouteMethod (\n\t\t\t\t\tthis.routes[routeName], routeMethod, executeFunction));\n\t\t\t}\n\t\t}\n\n\t\tthis.routes[routeName].connection = this.connection;\n\n\t\t// Create the route functions for the server/client.\n\t\tif (this.createFunctions === true)\n\t\t{\n\t\t\t// @ts-ignore\n\t\t\tlet newRoute: { [name: string]: Function } = this[routeName];\n\n\t\t\tif (newRoute == null)\n\t\t\t\tnewRoute = {};\n\n\t\t\tfor (let iIdx = 0; iIdx < this.routes[routeName].methods.length; iIdx++)\n\t\t\t{\n\t\t\t\tlet currentRoute: HotRoute = this.routes[routeName];\n\t\t\t\tlet newRouteMethod: HotRouteMethod = this.routes[routeName].methods[iIdx];\n\n\t\t\t\t/*\n\t\t\t\t/// @fixme Is this really necessary? A HTTP call is much more preferable, \n\t\t\t\t/// especially for accruate testing.\n\t\t\t\tif (this.connection instanceof HotServer)\n\t\t\t\t{\n\t\t\t\t\tif (newRouteMethod.onServerExecute != null)\n\t\t\t\t\t\tnewRoute[newRouteMethod.name] = newRouteMethod.onServerExecute;\n\t\t\t\t}\n\t\t\t\telse*/\n\t\t\t\t{\n\t\t\t\t\t/*\n\t\t\t\t\t/// @fixme Is onClientExecute necessary? I'm thinking the dev can just simply create \n\t\t\t\t\t/// their own function to call.\n\t\t\t\t\tif (newRouteMethod.onClientExecute != null)\n\t\t\t\t\t\tnewRoute[newRouteMethod.name] = newRouteMethod.onClientExecute;\n\t\t\t\t\telse\n\t\t\t\t\t{*/\n\t\t\t\t\t\tnewRoute[newRouteMethod.name] = (data: any, files: any): any =>\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tlet httpMethod: string = newRouteMethod.type;\n\t\t\t\t\t\t\t\t// Construct the url here. Base + route + route method\n\t\t\t\t\t\t\t\tlet routeStr: string = \"\";\n\n\t\t\t\t\t\t\t\tif (currentRoute.version !== \"\")\n\t\t\t\t\t\t\t\t\trouteStr += `/${currentRoute.version}`;\n\n\t\t\t\t\t\t\t\tif (currentRoute.route !== \"\")\n\t\t\t\t\t\t\t\t\trouteStr += `/${currentRoute.route}`;\n\n\t\t\t\t\t\t\t\tif (newRouteMethod.name !== \"\")\n\t\t\t\t\t\t\t\t\trouteStr += `/${newRouteMethod.name}`;\n\n\t\t\t\t\t\t\t\tlet authCredentials: any = null;\n\n\t\t\t\t\t\t\t\t// Getting the authorization credentials from the API is the lowest \n\t\t\t\t\t\t\t\t// priority for getting credentials. The priorities are in this order: \n\t\t\t\t\t\t\t\t// 1. HotRouteMethod\n\t\t\t\t\t\t\t\t// 2. HotRoute\n\t\t\t\t\t\t\t\t// 3. HotAPI\n\t\t\t\t\t\t\t\tif (this.authCredentials != null)\n\t\t\t\t\t\t\t\t\tauthCredentials = this.authCredentials;\n\n\t\t\t\t\t\t\t\t// Find the authorization credentials. Prioritize them when they're \n\t\t\t\t\t\t\t\t// in the method. Only add the ones from the route if the ones from \n\t\t\t\t\t\t\t\t// the method are missing.\n\t\t\t\t\t\t\t\tif (newRouteMethod.authCredentials != null)\n\t\t\t\t\t\t\t\t\tauthCredentials = newRouteMethod.authCredentials;\n\t\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tif (newRouteMethod.route.authCredentials != null)\n\t\t\t\t\t\t\t\t\t\tauthCredentials = newRouteMethod.route.authCredentials;\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tif (authCredentials == null)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\tif (typeof (Hot) !== \"undefined\")\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\tif (Hot != null)\n\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\t\tif (Hot.API != null)\n\t\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\t\t\tif (Hot.API[currentRoute.route] != null)\n\t\t\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\t\t\t\tif (Hot.API[currentRoute.route].authCredentials != null)\n\t\t\t\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\t\t\t\t\tauthCredentials = Hot.API[currentRoute.route].authCredentials;\n\t\t\t\t\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tif (authCredentials != null)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t// Add the authorization credentials to the data being sent.\n\t\t\t\t\t\t\t\t\tfor (let key in authCredentials)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tlet authCredential: any = authCredentials[key];\n\n\t\t\t\t\t\t\t\t\t\t// Do not overwrite any existing keys in the data about \n\t\t\t\t\t\t\t\t\t\t// to be sent.\n\t\t\t\t\t\t\t\t\t\tif (data[key] == null)\n\t\t\t\t\t\t\t\t\t\t\tdata[key] = authCredential;\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tlet args: any[] = [routeStr, data, httpMethod, files];\n\n\t\t\t\t\t\t\t\treturn (this.makeCall.apply (this, args));\n\t\t\t\t\t\t\t};\n\t\t\t\t\t//}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// @ts-ignore\n\t\t\tthis[routeName] = newRoute;\n\t\t}\n\t}\n\n\t/**\n\t * Register a route with the server.\n\t */\n\tasync registerRoute (route: HotRoute): Promise<void>\n\t{\n\t\tif (this.connection instanceof HotServer)\n\t\t\tawait this.connection.registerRoute (route);\n\t}\n\n\t/**\n\t * Register all routes with the server.\n\t */\n\tasync registerRoutes (): Promise<void>\n\t{\n\t\tfor (let key in this.routes)\n\t\t{\n\t\t\tlet route: HotRoute = this.routes[key];\n\n\t\t\tawait this.registerRoute (route);\n\t\t}\n\t}\n\n\t/**\n\t * Make a call to the API.\n\t */\n\tasync makeCall (route: string, data: any, httpMethod: string = \"POST\", \n\t\tfiles: { [name: string]: any } = {}): Promise<any>\n\t{\n\t\tlet url: string = this.baseUrl;\n\n\t\thttpMethod = httpMethod.toUpperCase ();\n\n\t\tif (url[(url.length - 1)] === \"/\")\n\t\t\turl = url.substr (0, (url.length - 1));\n\n\t\tif (route[0] !== \"/\")\n\t\t\turl += \"/\";\n\n\t\turl += route;\n\n\t\tconst numFiles: number = Object.keys (files).length;\n\n\t\tif (numFiles > 0)\n\t\t{\n\t\t\tif (httpMethod !== \"POST\")\n\t\t\t\tthrow new Error (`To upload files, you must set the httpMethod to POST.`);\n\n\t\t\tconst formData: FormData = new FormData ();\n\n\t\t\tfor (let key in files)\n\t\t\t\tformData.append (key, files[key]);\n\n\t\t\tlet res = await fetch (url, {\n\t\t\t\t\tmethod: \"POST\",\n\t\t\t\t\t// @ts-ignore\n\t\t\t\t\tbody: formData\n\t\t\t\t});\n\t\t\tlet jsonRes: any = await res.json ();\n\n\t\t\tif (data[\"hotstaq\"] == null)\n\t\t\t\tdata[\"hotstaq\"] = {};\n\n\t\t\tif (data[\"hotstaq\"][\"uploads\"] == null)\n\t\t\t\tdata[\"hotstaq\"][\"uploads\"] = {};\n\n\t\t\tdata[\"hotstaq\"][\"uploads\"][\"uploadId\"] = \n\t\t\t\t\tjsonRes[\"hotstaq\"][\"uploads\"][\"uploadId\"];\n\n\t\t\t// After the upload, make the actual JSON call. Do not pass files again.\n\t\t\tconst result: any = await this.makeCall (route, data, httpMethod);\n\n\t\t\treturn (result);\n\t\t}\n\n\t\tlet fetchObj: any = {\n\t\t\t\tmethod: httpMethod,\n\t\t\t\theaders: {\n\t\t\t\t\t\t\"Accept\": \"application/json\",\n\t\t\t\t\t\t\"Content-Type\": \"application/json\"\n\t\t\t\t\t}\n\t\t\t};\n\n\t\tif ((httpMethod !== \"GET\") && \n\t\t\t(httpMethod !== \"HEAD\"))\n\t\t{\n\t\t\tfetchObj[\"body\"] = JSON.stringify (data);\n\t\t}\n\n\t\tlet promise = new Promise ((resolve, reject) => \n\t\t\t{\n\t\t\t\tfetch (url, fetchObj).then (async (res) =>\n\t\t\t\t\t{\n\t\t\t\t\t\tlet jsonObj: any = await res.json ();\n\t\t\n\t\t\t\t\t\tresolve (jsonObj);\n\t\t\t\t\t})\n\t\t\t\t\t.catch ((reason: any) =>\n\t\t\t\t\t{\n\t\t\t\t\t\tthrow new Error (`${url}: ${reason.message}`);\n\t\t\t\t\t});\n\t\t\t});\n\n\t\treturn (promise);\n\t}\n}\n","import { HotStaq } from \"./HotStaq\";\nimport { HotAPI } from \"./HotAPI\";\nimport { HotServerType } from \"./HotServer\";\nimport { HotLog } from \"./HotLog\";\n\n/**\n * A client connected to a server.\n */\nexport class HotClient\n{\n\t/**\n\t * The processor to use.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The API to use.\n\t */\n\tapi: HotAPI;\n\t/**\n\t * The tester API to use.\n\t */\n\ttesterAPI: HotAPI;\n\t/**\n\t * The type of server.\n\t */\n\ttype: HotServerType;\n\t/**\n\t * The logger.\n\t */\n\tlogger: HotLog;\n\n\tconstructor (processor: HotStaq)\n\t{\n\t\tthis.processor = processor;\n\t\tthis.api = null;\n\t\tthis.testerAPI = null;\n\t\tthis.type = HotServerType.HTTP;\n\t\tthis.logger = processor.logger;\n\t}\n}","import { HotAPI } from \"./HotAPI\";\nimport { HotStaq } from \"./HotStaq\";\n\nexport interface HotComponentOutput\n{\n\t/**\n\t * The HTML to output.\n\t */\n\thtml: string;\n\t/**\n\t * The query selector to add this component's functions to.\n\t * \n\t * @example #objectId\n\t */\n\taddFunctionsTo?: string;\n\t/**\n\t * The place here parent name to attach this html to.\n\t * \n\t * @example top\n\t */\n\tplaceHereParent?: string;\n\t/**\n\t * Append the output to an existing element.\n\t * \n\t * @example div .testClass\n\t */\n\tparentSelector?: string;\n}\n\n/**\n * A component to preprocess.\n */\nexport interface IHotComponent\n{\n\t/**\n\t * The processor to use.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The associated HTMLElements.\n\t */\n\thtmlElements?: HTMLElement[];\n\t/**\n\t * The name of the page.\n\t */\n\tname?: string;\n\t/**\n\t * The name of the tag.\n\t */\n\ttag?: string;\n\t/**\n\t * The connected API.\n\t */\n\tapi?: HotAPI;\n\t/**\n\t * The options to include with registering this component.\n\t */\n\telementOptions?: ElementDefinitionOptions;\n\t/**\n\t * Any extra attributes to register.\n\t */\n\tobservedAttributes?: string[];\n\t/**\n\t * The type of component.\n\t */\n\ttype?: string;\n\t/**\n\t * The value of the component.\n\t */\n\tvalue?: any;\n\t/**\n\t * The inner HTML or value of the component.\n\t */\n\tinner: any;\n\t/**\n\t * The events to trigger.\n\t */\n\tevents?: {\n\t\t\t[name: string]: {\n\t\t\t\ttype: string;\n\t\t\t\tfunc: Function;\n\t\t\t\toptions?: any;\n\t\t\t}\n\t\t};\n\t/**\n\t * Execute prior to output.\n\t * \n\t * @returns If set to false, the component will not be registered.\n\t */\n\tonPreOutput?: () => Promise<boolean>;\n\t/**\n\t * Execute after getting the output, but before the DOM parsing.\n\t * \n\t * @param output The output from the component to register. Can be manipulated one last time prior to \n\t * being parsed into a DOM element.\n\t * \n\t * @returns The final output to be parsed as a DOM element.\n\t */\n\tonPostOutput?: (output: (string | HotComponentOutput[])) => Promise<(string | HotComponentOutput[])>;\n\t/**\n\t * Execute when its time to fix the HTML prior to DOM parsing. This will skip the HotStaq default fixing.\n\t */\n\tonFixHTML?: (output: string) => Promise<{ fixedStr: string, querySelector: string; }>;\n\t/**\n\t * Execute a custom DOM parser.\n\t */\n\tonParseDOM?: (output: string) => Promise<Document>;\n\t/**\n\t * Execute after the output has been parsed and is ready to be placed into the DOM.\n\t */\n\tonParsed?: (output: string) => Promise<string>;\n\t/**\n\t * Execute prior to placing the new DOM element.\n\t */\n\tonPrePlace?: (htmlElement: HTMLElement) => Promise<HTMLElement>;\n\t/**\n\t * Execute after placing the new DOM element. Can be manipulated one final time prior to being rendered.\n\t */\n\tonPostPlace?: (parentHtmlElement: HTMLElement, htmlElement: HTMLElement) => Promise<HTMLElement>;\n\t/**\n\t * Execute after placing the DOM element onto the newly created parent.\n\t */\n\tonParentPlace?: (parentHtmlElement: HTMLElement, htmlElement: HTMLElement) => Promise<void>;\n}\n\n/**\n * A component to preprocess.\n */\nexport abstract class HotComponent implements IHotComponent\n{\n\t/**\n\t * The processor to use.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The associated HTMLElements.\n\t */\n\thtmlElements: HTMLElement[];\n\t/**\n\t * The name of the component.\n\t */\n\tname: string;\n\t/**\n\t * The name of the tag.\n\t */\n\ttag: string;\n\t/**\n\t * The connected API.\n\t */\n\tapi: HotAPI;\n\t/**\n\t * The options to include with registering this component.\n\t */\n\telementOptions: ElementDefinitionOptions;\n\t/**\n\t * Any extra attributes to register.\n\t */\n\tobservedAttributes: string[];\n\t/**\n\t * The type of component.\n\t */\n\ttype: string;\n\t/**\n\t * The value of the component.\n\t */\n\tvalue: any;\n\t/**\n\t * The inner HTML or value of the component.\n\t */\n\tinner: any;\n\t/**\n\t * The events to trigger.\n\t */\n\tevents: {\n\t\t[name: string]: {\n\t\t\t\ttype: string;\n\t\t\t\tfunc: Function;\n\t\t\t\toptions: any;\n\t\t\t}\n\t\t};\n\t/**\n\t * Execute prior to output.\n\t * \n\t * @returns If set to false, the component will not be registered.\n\t */\n\tonPreOutput?: () => Promise<boolean>;\n\t/**\n\t * Execute after getting the output, but before the DOM parsing.\n\t * \n\t * @param output The output from the component to register. Can be manipulated one last time prior to \n\t * being parsed into a DOM element.\n\t * \n\t * @returns The final output to be parsed as a DOM element.\n\t */\n\tonPostOutput?: (output: (string | HotComponentOutput[])) => Promise<(string | HotComponentOutput[])>;\n\t/**\n\t * Execute when its time to fix the HTML prior to DOM parsing. This will skip the HotStaq default fixing.\n\t */\n\tonFixHTML?: (output: string) => Promise<{ fixedStr: string, querySelector: string; }>;\n\t/**\n\t * Execute a custom DOM parser.\n\t */\n\tonParseDOM?: (output: string) => Promise<Document>;\n\t/**\n\t * Execute after the output has been parsed and is ready to be placed into the DOM.\n\t */\n\tonParsed?: (output: string) => Promise<string>;\n\t/**\n\t * Execute prior to placing the new DOM element.\n\t */\n\tonPrePlace?: (htmlElement: HTMLElement) => Promise<HTMLElement>;\n\t/**\n\t * Execute after placing the new DOM element. Can be manipulated one final time prior to being rendered.\n\t */\n\tonPostPlace?: (parentHtmlElement: HTMLElement, htmlElement: HTMLElement) => Promise<HTMLElement>;\n\t/**\n\t * Execute after placing the DOM element onto the newly created parent.\n\t */\n\tonParentPlace?: (parentHtmlElement: HTMLElement, htmlElement: HTMLElement) => Promise<void>;\n\n\tconstructor (copy: IHotComponent | HotStaq, api: HotAPI = null)\n\t{\n\t\tif ((copy instanceof HotStaq) || (copy == null))\n\t\t{\n\t\t\t// @ts-ignore\n\t\t\tthis.processor = copy;\n\t\t\tthis.htmlElements = [];\n\t\t\tthis.name = \"\";\n\t\t\tthis.tag = \"\";\n\t\t\tthis.api = null;\n\t\t\tthis.elementOptions = undefined;\n\t\t\tthis.observedAttributes = [];\n\t\t\tthis.type = \"\";\n\t\t\tthis.value = null;\n\t\t\tthis.inner = null;\n\t\t\tthis.events = {};\n\t\t}\n\t\telse\n\t\t{\n\t\t\tthis.processor = copy.processor;\n\t\t\tthis.htmlElements = copy.htmlElements || [];\n\t\t\tthis.name = copy.name || \"\";\n\t\t\tthis.tag = copy.tag || this.name;\n\t\t\tthis.api = copy.api || null;\n\t\t\tthis.elementOptions = copy.elementOptions || undefined;\n\t\t\tthis.observedAttributes = copy.observedAttributes || [];\n\t\t\tthis.type = copy.type || \"\";\n\t\t\tthis.value = copy.value || null;\n\t\t\tthis.inner = copy.inner || null;\n\t\t\tthis.events = {};\n\t\t}\n\n\t\tif (api != null)\n\t\t\tthis.api = api;\n\t}\n\n\t/**\n\t * Event that's called when this component's DOM element has been created.\n\t * \n\t * @returns The modified DOM element.\n\t */\n\tasync onCreated (element: HTMLElement): Promise<any>\n\t{\n\t\treturn (element);\n\t}\n\n\t/**\n\t * Handle the attributes manually.\n\t */\n\tasync handleAttributes? (attributes: NamedNodeMap): Promise<void>;\n\n\t/**\n\t * Handle a click event.\n\t */\n\tabstract click? (): Promise<void>;\n\n\t/**\n\t * Output the component.\n\t */\n\tabstract output (): Promise<string | HotComponentOutput[]>;\n}","import * as fs from \"fs\";\n\nimport fetch from \"node-fetch\";\n\nimport { DeveloperMode, Hot } from \"./Hot\";\nimport { HotPage } from \"./HotPage\";\nimport { HotTestElement } from \"./HotTestElement\";\n\n/**\n * A file to process.\n */\nexport interface IHotFile\n{\n\t/**\n\t * The parent page.\n\t */\n\tpage?: HotPage;\n\t/**\n\t * The name of the file.\n\t */\n\tname?: string;\n\t/**\n\t * The url to the file to get.\n\t */\n\turl?: string;\n\t/**\n\t * The path to the local file to get.\n\t */\n\tlocalFile?: string;\n\t/**\n\t * The content of the file to process.\n\t */\n\tcontent?: string;\n\t/**\n\t * Force all errors to be thrown.\n\t */\n\tthrowAllErrors?: boolean;\n}\n\n/**\n * Parser options for when processing a string or file.\n */\nexport interface ParserOptions\n{\n\t/**\n\t * Output the commands generated from processing. Default: true\n\t */\n\toutputCommands?: boolean;\n\t/**\n\t * Allow JSON.stringify to be used during processing. Default: true\n\t */\n\tallowStringify?: boolean;\n}\n\n/**\n * A file to process.\n */\nexport class HotFile implements IHotFile\n{\n\t/**\n\t * The parent page.\n\t */\n\tpage: HotPage;\n\t/**\n\t * The name of the file.\n\t */\n\tname: string;\n\t/**\n\t * The url to the file to get.\n\t */\n\turl: string;\n\t/**\n\t * The path to the local file to get.\n\t */\n\tlocalFile: string;\n\t/**\n\t * The content of the file to process.\n\t */\n\tcontent: string;\n\t/**\n\t * Force all errors to be thrown.\n\t */\n\tthrowAllErrors: boolean;\n\n\tconstructor (copy: IHotFile = {})\n\t{\n\t\tthis.page = copy.page || null;\n\t\tthis.name = copy.name || \"\";\n\t\tthis.url = copy.url || \"\";\n\t\tthis.localFile = copy.localFile || \"\";\n\t\tthis.content = copy.content || \"\";\n\t\tthis.throwAllErrors = copy.throwAllErrors || false;\n\t}\n\n\t/**\n\t * Set the content of this file.\n\t */\n\tsetContent (content: string): void\n\t{\n\t\tthis.content = content;\n\t}\n\n\t/**\n\t * Get the content of this file.\n\t */\n\tgetContent (): string\n\t{\n\t\treturn (this.content);\n\t}\n\n\t/**\n\t * Make a HTTP get request.\n\t */\n\tstatic async httpGet (url: string): Promise<string>\n\t{\n\t\ttry\n\t\t{\n\t\t\tlet res = await fetch (url);\n\n\t\t\tif (res.ok === false)\n\t\t\t\tthrow new Error (`${res.status}: ${res.statusText}`);\n\n\t\t\tlet content: string = await res.text ();\n\n\t\t\treturn (content);\n\t\t}\n\t\tcatch (ex)\n\t\t{\n\t\t\treturn (JSON.stringify ({ \"error\": `${ex.message} - Could not fetch ${url}` }));\n\t\t}\n\t}\n\n\t/**\n\t * Load content from a url.\n\t */\n\tasync loadUrl (): Promise<string>\n\t{\n\t\tthis.content = await HotFile.httpGet (this.url);\n\n\t\treturn (this.content);\n\t}\n\n\t/**\n\t * Load content from a local file.\n\t */\n\tasync loadLocalFile (): Promise<string>\n\t{\n\t\tlet promise: Promise<string> = new Promise (\n\t\t\t(resolve: any, reject: any): void =>\n\t\t\t{\n\t\t\t\tfs.readFile (this.localFile, (err: NodeJS.ErrnoException, data: Buffer): void =>\n\t\t\t\t\t{\n\t\t\t\t\t\tif (err != null)\n\t\t\t\t\t\t\tthrow err;\n\n\t\t\t\t\t\tlet content: string = data.toString ();\n\t\t\t\t\t\tthis.content = content;\n\n\t\t\t\t\t\tresolve (this.content);\n\t\t\t\t\t});\n\t\t\t});\n\n\t\treturn (promise);\n\t}\n\n\t/**\n\t * Load the contents of the file.\n\t */\n\tasync load (): Promise<string>\n\t{\n\t\tlet content: string = \"\";\n\n\t\tif (this.url !== \"\")\n\t\t\tcontent = await this.loadUrl ();\n\n\t\tif (this.localFile !== \"\")\n\t\t\tcontent = await this.loadLocalFile ();\n\n\t\treturn (content);\n\t}\n\n\t/**\n\t * Process string content. This will take in a regular expression and \n\t * parse the content based on the regex. When the regex content is found \n\t * contentProcessor will be executed with the regex content found. When \n\t * the regex content is not found, offContentProcessor will be called with \n\t * the content outside of the regex.\n\t * \n\t * @param content The content to parse.\n\t * @param contentRegex The regex to use to parse the content.\n\t * @param contentProcessor The content found inside the regex.\n\t * @param offContentProcessor The content found outside of the regex.\n\t * @param numRemoveFromBeginning The number of characters to remove from the \n\t * beginning of the found content.\n\t * @param numRemoveFromEnd The number of characters to remove from the end of \n\t * the found content.\n\t */\n\tstatic processContent (content: string, contentRegex: RegExp,\n\t\tcontentProcessor: (regexFound: string) => string,\n\t\toffContentProcessor: (offContent: string) => string,\n\t\tnumRemoveFromBeginning: number = 2,\n\t\tnumRemoveFromEnd: number = 2): string\n\t{\n\t\tlet result: RegExpExecArray = contentRegex.exec (content);\n\t\tlet previousIndex: number = 0;\n\t\tlet output: string = \"\";\n\n\t\twhile (result != null)\n\t\t{\n\t\t\tlet start: number = result.index - numRemoveFromBeginning;\n\t\t\tlet end: number = contentRegex.lastIndex + numRemoveFromEnd;\n\n\t\t\t// Get the previous section.\n\t\t\tlet prevContent: string = content.substr (previousIndex, (start - previousIndex));\n\t\t\tpreviousIndex = end;\n\n\t\t\toutput += offContentProcessor (prevContent);\n\n\t\t\t// Process the content found from the regex\n\t\t\tlet contentFound: string = result[0];\n\t\t\toutput += contentProcessor (contentFound);\n\n\t\t\t// Move on to the next section to parse.\n\t\t\tresult = contentRegex.exec (content);\n\t\t}\n\n\t\t// Append whatever else is after the last parsed section.\n\t\tlet lastContent: string = content.substr (previousIndex);\n\n\t\toutput += offContentProcessor (lastContent);\n\n\t\treturn (output);\n\t}\n\n\t/**\n\t * Process any content that could have nested values. This will \n\t * take in a regular expression and \n\t * parse the content based on the regex. When the regex content is found \n\t * contentProcessor will be executed with the regex content found. When \n\t * the regex content is not found, offContentProcessor will be called with \n\t * the content outside of the regex.\n\t * \n\t * @fixme Needs to be able to ignore any characters found inside comments \n\t * or a string. For example, if the following is used ```${\"Test }\"}``` It \n\t * will error out.\n\t * \n\t * @param content The content to parse.\n\t * @param contentRegex The regex to use to parse the content.\n\t * @param contentProcessor The content found inside the regex.\n\t * @param offContentProcessor The content found outside of the regex.\n\t * @param numRemoveFromBeginning The number of characters to remove from the \n\t * beginning of the found content.\n\t * @param numRemoveFromEnd The number of characters to remove from the end of \n\t * the found content.\n\t */\n\tstatic processNestedContent (content: string, startChars: string, endChars: string, \n\t\ttriggerChar: string, contentProcessor: (regexFound: string) => string,\n\t\toffContentProcessor: (offContent: string) => string,\n\t\tnumRemoveFromBeginning: number = 2,\n\t\tnumRemoveFromEnd: number = 1): string\n\t{\n\t\tlet pos: number = content.indexOf (startChars);\n\t\tlet previousIndex: number = 0;\n\t\tlet startTriggerPos: number = content.indexOf (triggerChar, pos);\n\t\tlet output: string = \"\";\n\n\t\twhile (pos > -1)\n\t\t{\n\t\t\tlet end: number = content.indexOf (endChars, pos);\n\t\t\tlet nestedCounter: number = 0;\n\n\t\t\tif (triggerChar !== \"\")\n\t\t\t{\n\t\t\t\t// Reverse search the trigger characters and count the number of \n\t\t\t\t// occurrences.\n\t\t\t\tlet rpos: number = content.lastIndexOf (triggerChar, end - numRemoveFromEnd);\n\n\t\t\t\twhile (rpos > -1)\n\t\t\t\t{\n\t\t\t\t\tif (rpos === startTriggerPos)\n\t\t\t\t\t\tbreak;\n\n\t\t\t\t\trpos = content.lastIndexOf (triggerChar, rpos - numRemoveFromEnd);\n\t\t\t\t\tnestedCounter++;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\t// If there's nested trigger characters, get the last occurrence of \n\t\t\t// the end character.\n\t\t\tif (nestedCounter > 0)\n\t\t\t{\n\t\t\t\tlet epos: number = content.indexOf (endChars, end + numRemoveFromEnd);\n\t\t\t\tlet tempepos: number = epos;\n\n\t\t\t\twhile ((epos > -1) && (nestedCounter > 0))\n\t\t\t\t{\n\t\t\t\t\tif (tempepos < 0)\n\t\t\t\t\t\tbreak;\n\n\t\t\t\t\t// Make sure we aren't discovering endChars that we shouldn't be.\n\t\t\t\t\tlet posOutsideOfContent: number = content.lastIndexOf (startChars, tempepos - numRemoveFromEnd);\n\n\t\t\t\t\tif (posOutsideOfContent > epos)\n\t\t\t\t\t\tbreak;\n\n\t\t\t\t\tepos = tempepos;\n\n\t\t\t\t\ttempepos = content.indexOf (endChars, epos + numRemoveFromEnd);\n\t\t\t\t\tnestedCounter--;\n\t\t\t\t}\n\n\t\t\t\tend = epos;\n\t\t\t}\n\n\t\t\tlet offContentStr: string = content.substr (previousIndex, (pos - previousIndex));\n\t\t\toutput += offContentProcessor (offContentStr);\n\n\t\t\tlet foundContent: string = content.substr (\n\t\t\t\tpos + numRemoveFromBeginning, (end - (pos + numRemoveFromBeginning)));\n\t\t\toutput += contentProcessor (foundContent);\n\n\t\t\t// Get the next content\n\t\t\tpos = content.indexOf (startChars, end + numRemoveFromEnd);\n\t\t\tstartTriggerPos = content.indexOf (triggerChar, pos);\n\t\t\tpreviousIndex = end + numRemoveFromEnd;\n\t\t}\n\n\t\t// Append whatever else is after the last parsed section.\n\t\tlet lastContent: string = content.substr (previousIndex);\n\n\t\toutput += offContentProcessor (lastContent);\n\n\t\treturn (output);\n\t}\n\n\t/**\n\t * Parse content.\n\t * \n\t * @param thisContent The content to parse.\n\t * @param throwAllErrors If set to true, any error that occurs will be thrown as an exception.\n\t * @param options The options to use when parsing the content.\n\t */\n\tstatic parseContent (thisContent: string, throwAllErrors: boolean, parserOptions: ParserOptions = null): string\n\t{\n\t\tif (parserOptions == null)\n\t\t{\n\t\t\tparserOptions = {\n\t\t\t\t\toutputCommands: true,\n\t\t\t\t\tallowStringify: true\n\t\t\t\t};\n\t\t}\n\n\t\tlet STRINGIFY_START: string = \"JSON.stringify (\";\n\t\tlet STRINGIFY_END: string = \")\";\n\n\t\tif (parserOptions.allowStringify === false)\n\t\t{\n\t\t\tSTRINGIFY_START = \"\";\n\t\t\tSTRINGIFY_END = \"\";\n\t\t}\n\n\t\t// Assemble the JS to evaluate. This will take all content outside of \n\t\t// <* and *> and wrap a Hot.echo around it. Any JS found inside of the \n\t\t// <* and *> will be executed as is.\n\t\tlet output: string = HotFile.processContent (thisContent, \n\t\t\tnew RegExp (\"(?=\\\\<\\\\*)([\\\\s\\\\S]*?)(?=\\\\*\\\\>)\", \"g\"), \n\t\t\t(regexFound: string): string =>\n\t\t\t{\n\t\t\t\t// A little hack, since I suck at Regex :(\n\t\t\t\tregexFound = regexFound.substr (2);\n\n\t\t\t\treturn (`${regexFound}`);\n\t\t\t}, \n\t\t\t(offContent: string): string =>\n\t\t\t{\n\t\t\t\tif (offContent === \"\")\n\t\t\t\t\treturn (\"\");\n\n\t\t\t\tlet tempOutput: string = HotFile.processNestedContent (\n\t\t\t\t\toffContent, \"!{\", \"}\", \"{\", \n\t\t\t\t\t(regexFound2: string): string =>\n\t\t\t\t\t{\n\t\t\t\t\t\tlet out: string = `*&&%*%@#@!${regexFound2}*&!#%@!@*!`;\n\n\t\t\t\t\t\treturn (out);\n\t\t\t\t\t}, \n\t\t\t\t\t(offContent3: string): string =>\n\t\t\t\t\t{\n\t\t\t\t\t\treturn (offContent3);\n\t\t\t\t\t});\n\t\t\t\tlet tempOutput2: string = HotFile.processNestedContent (\n\t\t\t\t\ttempOutput, \"STR{\", \"}\", \"{\", \n\t\t\t\t\t(regexFound2: string): string =>\n\t\t\t\t\t{\n\t\t\t\t\t\tlet out: string = \"\";\n\n\t\t\t\t\t\tif (parserOptions.outputCommands === true)\n\t\t\t\t\t\t\tout = `*&&%*%@#@!echoOutput (JSON.stringify(${regexFound2}), ${throwAllErrors});*&!#%@!@*!`;\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tout = `*&&%*%@#@!${STRINGIFY_START}${regexFound2}${STRINGIFY_END}, ${throwAllErrors});*&!#%@!@*!`;\n\n\t\t\t\t\t\treturn (out);\n\t\t\t\t\t}, \n\t\t\t\t\t(offContent3: string): string =>\n\t\t\t\t\t{\n\t\t\t\t\t\treturn (offContent3);\n\t\t\t\t\t}, 4, 1);\n\t\t\t\tlet tempOutput3: string = HotFile.processNestedContent (\n\t\t\t\t\ttempOutput2, \"${\", \"}\", \"{\", \n\t\t\t\t\t(regexFound2: string): string =>\n\t\t\t\t\t{\n\t\t\t\t\t\tlet out: string = \"\";\n\n\t\t\t\t\t\tif (parserOptions.outputCommands === true)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tout = `*&&%*%@#@!try { Hot.echo (${regexFound2}); }catch (ex){Hot.echo (\"\");}*&!#%@!@*!`;\n\n\t\t\t\t\t\t\tif (throwAllErrors === true)\n\t\t\t\t\t\t\t\tout = `*&&%*%@#@!Hot.echo (${regexFound2});*&!#%@!@*!`;\n\t\t\t\t\t\t}\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tout = `*&&%*%@#@!${regexFound2}*&!#%@!@*!`;\n\n\t\t\t\t\t\treturn (out);\n\t\t\t\t\t}, \n\t\t\t\t\t(offContent3: string): string =>\n\t\t\t\t\t{\n\t\t\t\t\t\treturn (offContent3);\n\t\t\t\t\t\t/*let escapedContent: string = JSON.stringify (offContent3);\n\t\t\t\t\t\tlet out: string = `echoOutput (${escapedContent}, ${throwAllErrors});\\n`;\n\n\t\t\t\t\t\treturn (out);*/\n\t\t\t\t\t});\n\n\t\t\t\tlet tempOutput4: string = \"\";\n\n\t\t\t\tif (Hot.Mode === DeveloperMode.Production)\n\t\t\t\t{\n\t\t\t\t\ttempOutput4 = HotFile.processNestedContent (\n\t\t\t\t\t\ttempOutput3, \"?(\", \")\", \"(\", \n\t\t\t\t\t\t(regexFound2: string): string =>\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\treturn (\"\");\n\t\t\t\t\t\t}, \n\t\t\t\t\t\t(offContent3: string): string =>\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\treturn (offContent3);\n\t\t\t\t\t\t\t/*let out: string = `echoOutput (${offContent3}, ${throwAllErrors});\\n`;\n\n\t\t\t\t\t\t\treturn (out);*/\n\t\t\t\t\t\t});\n\t\t\t\t}\n\n\t\t\t\tif (Hot.Mode === DeveloperMode.Development)\n\t\t\t\t{\n\t\t\t\t\ttempOutput4 = HotFile.processNestedContent (\n\t\t\t\t\t\ttempOutput3, \"?(\", \")\", \"(\", \n\t\t\t\t\t\t(regexFound2: string): string =>\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet foundStr: string = \"\";\n\n\t\t\t\t\t\t\ttry\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t// Check to see if it be parsed. If so, stringify it.\n\t\t\t\t\t\t\t\tJSON.parse (regexFound2);\n\n\t\t\t\t\t\t\t\tif (parserOptions.allowStringify === true)\n\t\t\t\t\t\t\t\t\tfoundStr = JSON.stringify (regexFound2);\n\t\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\t\tfoundStr = `${regexFound2}`;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tcatch (ex)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t// If valid JSON is not received, don't worry about it, pass it \n\t\t\t\t\t\t\t\t// along to the function below for it to be parsed in the page.\n\t\t\t\t\t\t\t\t// The exception should be thrown there instead.\n\t\t\t\t\t\t\t\tfoundStr = `${regexFound2}`;\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t/// @fixme Make this a callable function and pass foundStr, etc.\n\t\t\t\t\t\t\tlet out: string = \"\";\n\n\t\t\t\t\t\t\tif (parserOptions.outputCommands === true)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tout = `*&&%*%@#@!{\nconst testElm = createTestElement (${foundStr});\nHot.echo (\\`data-test-object-name = \"\\${testElm.name}\" data-test-object-func = \"\\${testElm.func}\" data-test-object-value = \"\\${testElm.value}\"\\`);\n}*&!#%@!@*!\\n`;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tlet createTestElement = (foundStr2: any) =>\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tlet testElm = null;\n\n\t\t\t\t\t\t\t\t\ttry\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tlet obj = foundStr2;\n\n\t\t\t\t\t\t\t\t\t\tif (typeof (foundStr2) === \"string\")\n\t\t\t\t\t\t\t\t\t\t\tobj = JSON.parse (foundStr2);\n\n\t\t\t\t\t\t\t\t\t\tif (typeof (obj) === \"string\")\n\t\t\t\t\t\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj);\n\n\t\t\t\t\t\t\t\t\t\tif (obj instanceof Array)\n\t\t\t\t\t\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj[0], obj[1], obj[2]);\n\n\t\t\t\t\t\t\t\t\t\tif (obj[\"name\"] != null)\n\t\t\t\t\t\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj);\n\n\t\t\t\t\t\t\t\t\t\tif (Hot.CurrentPage.testElements[testElm.name] != null)\n\t\t\t\t\t\t\t\t\t\t\tthrow new Error (`Test element ${testElm.name} already exists!`);\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t\tcatch (ex)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tthrow new Error (\n\t\t\t\t\t\t\t\t`Error processing test element ${foundStr2} in ${Hot.CurrentPage.name}. Error: ${ex.message}`\n\t\t\t\t\t\t\t\t\t\t\t);\n\t\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\t\treturn (testElm);\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tconst testElm = createTestElement (foundStr);\n\t\t\t\t\t\t\t\tout = `*&&%*%@#@!data-test-object-name = \"${testElm.name}\" data-test-object-func = \"${testElm.func}\" data-test-object-value = \"${testElm.value}\"*&!#%@!@*!`;\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\treturn (out);\n\t\t\t\t\t\t}, \n\t\t\t\t\t\t(offContent3: string): string =>\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\treturn (offContent3);\n\t\t\t\t\t\t\t/*let out: string = `echoOutput (${offContent3}, ${throwAllErrors});\\n`;\n\n\t\t\t\t\t\t\treturn (out);*/\n\t\t\t\t\t\t});\n\t\t\t\t}\n\n\t\t\t\tlet tempOutput5: string = HotFile.processNestedContent (\n\t\t\t\t\ttempOutput4, \"*&&%*%@#@!\", \"*&!#%@!@*!\", \"*&&%*%@#@!\", \n\t\t\t\t\t(regexFound: string): string =>\n\t\t\t\t\t{\n\t\t\t\t\t\treturn (regexFound);\n\t\t\t\t\t}, \n\t\t\t\t\t(offContent: string): string =>\n\t\t\t\t\t{\n\t\t\t\t\t\tlet escapedContent: string = \"\";\n\n\t\t\t\t\t\tif (parserOptions.allowStringify === true)\n\t\t\t\t\t\t\tescapedContent = JSON.stringify (offContent);\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tescapedContent = offContent;\n\n\t\t\t\t\t\tlet out: string = \"\";\n\t\t\t\t\t\t\n\t\t\t\t\t\tif (parserOptions.outputCommands === true)\n\t\t\t\t\t\t\tout = `echoOutput (${escapedContent}, ${throwAllErrors});\\n`;\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tout = escapedContent;\n\n\t\t\t\t\t\treturn (out);\n\t\t\t\t\t}, \n\t\t\t\t\t\"*&&%*%@#@!\".length, \"*&!#%@!@*!\".length);\n\n\t\t\t\t/// @fixme Temporary hack. These delimiters should be removed from tempOutput when \n\t\t\t\t/// executing processNestedContent.\n\t\t\t\ttempOutput5 = tempOutput5.replace (/\\*\\&\\&\\%\\*\\%\\@\\#\\@\\!/g, \"\");\n\t\t\t\ttempOutput5 = tempOutput5.replace (/\\*\\&\\!\\#\\%\\@\\!\\@\\*\\!/g, \"\");\n\n\t\t\t\treturn (tempOutput5);\n\t\t\t}, 0);\n\n\t\treturn (output);\n\t}\n\n\t/**\n\t * Process the content in this file. This treats each file as one large JavaScript\n\t * file. Any text outside of the <* *> areas will be treated as:\n\t * \n\t * \t\tHot.echo (\"text\");\n\t * \n\t * @fixme The regex's in the offContent functions need to be fixed. There's several \n\t * test cases where they will fail.\n\t */\n\tasync process (args: any = null): Promise<string>\n\t{\n\t\tlet thisContent: string = this.content;\n\n\t\tHot.Mode = this.page.processor.mode;\n\t\tHot.Arguments = args;\n\t\tHot.CurrentPage = this.page;\n\t\tHot.PublicKeys = this.page.processor.publicKeys;\n\t\tHot.API = this.page.getAPI ();\n\t\tHot.TesterAPI = this.page.getTesterAPI ();\n\n\t\tlet output: string = HotFile.parseContent (thisContent, this.throwAllErrors);\n\n\t\t// Execute the assembled JS file.\n\t\tlet returnedOutput: any = null;\n\n\t\ttry\n\t\t{\n\t\t\tlet executionContent: string = `\n\t\t\tvar Hot = arguments[0];\n\t\t\tvar PassedHotFile = arguments[1];\n\n\t\t\t`;\n\n\t\t\t// Output the arguments so it's usable in the entire document.\n\t\t\tif (typeof (args) === \"string\")\n\t\t\t\tthrow new Error (`The passing arguments cannot be a string!`);\n\n\t\t\tfor (let key in args)\n\t\t\t{\n\t\t\t\tlet newVar: string = \"\";\n\t\t\t\tlet newVarValue: any = args[key];\n\t\t\t\tlet newVarValueStr: string = JSON.stringify (newVarValue);\n\n\t\t\t\tnewVar = `var ${key} = ${newVarValueStr};\\n`;\n\n\t\t\t\texecutionContent += newVar;\n\t\t\t}\n\n\t\t\tlet contentName: string = this.name;\n\n\t\t\tif (contentName === \"\")\n\t\t\t\tcontentName = this.localFile;\n\n\t\t\tif (contentName === \"\")\n\t\t\t\tcontentName = this.url;\n\n\t\t\texecutionContent += `\n\n\t\t\tfunction echoOutput (content, throwErrors)\n\t\t\t{\n\t\t\t\tif (throwErrors == null)\n\t\t\t\t\tthrowErrors = true;\n\n\t\t\t\tif (throwErrors === true)\n\t\t\t\t{\n\t\t\t\t\tHot.echo (content);\n\n\t\t\t\t\treturn;\n\t\t\t\t}\n\n\t\t\t\ttry\n\t\t\t\t{\n\t\t\t\t\tHot.echo (content);\n\t\t\t\t}\n\t\t\t\tcatch (ex)\n\t\t\t\t{\n\t\t\t\t\tHot.echo (\"\");\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tfunction createTestElement (foundStr)\n\t\t\t{\n\t\t\t\tlet testElm = null;\n\n\t\t\t\ttry\n\t\t\t\t{\n\t\t\t\t\tlet obj = foundStr;\n\n\t\t\t\t\tif (typeof (foundStr) === \"string\")\n\t\t\t\t\t\tobj = JSON.parse (foundStr);\n\n\t\t\t\t\tif (typeof (obj) === \"string\")\n\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj);\n\n\t\t\t\t\tif (obj instanceof Array)\n\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj[0], obj[1], obj[2]);\n\n\t\t\t\t\tif (obj[\"name\"] != null)\n\t\t\t\t\t\ttestElm = new Hot.HotTestElement (obj);\n\n\t\t\t\t\tif (Hot.CurrentPage.testElements[testElm.name] != null)\n\t\t\t\t\t\tthrow new Error (\\`Test element \\${testElm.name} already exists!\\`);\n\n\t\t\t\t\tHot.CurrentPage.addTestElement (testElm);\n\t\t\t\t}\n\t\t\t\tcatch (ex)\n\t\t\t\t{\n\t\t\t\t\tthrow new Error (\n\t\t\t\\`Error processing test element \\${foundStr} in \\${Hot.CurrentPage.name}. Error: \\${ex.message}\\`\n\t\t\t\t\t\t);\n\t\t\t\t}\n\n\t\t\t\treturn (testElm);\n\t\t\t}\n\n\t\t\tasync function runContent (CurrentHotFile)\n\t\t\t{\\n`;\n\t\t\texecutionContent += output;\n\t\t\texecutionContent += `\n\t\t\t}\n\n\t\t\treturn (runContent (PassedHotFile).then (() =>\n\t\t\t{\n\t\t\t\treturn ({\n\t\t\t\t\t\thot: Hot,\n\t\t\t\t\t\toutput: Hot.Output,\n\t\t\t\t\t\tdata: JSON.stringify (Hot.Data)\n\t\t\t\t\t});\n\t\t\t}));`;\n\n\t\t\t/// @fixme Prior to execution compile any TypeScript and make it ES5 compatible.\n\t\t\tlet func: Function = new Function (executionContent);\n\t\t\treturnedOutput = await func.apply (this, [Hot, this]);\n\t\t}\n\t\tcatch (ex)\n\t\t{\n\t\t\tif (ex instanceof SyntaxError)\n\t\t\t{\n\t\t\t\t/// @fixme Put what's in the content variable into a prev content variable?\n\t\t\t\t/// Then once there's no longer any syntax errors being thrown, execute the \n\t\t\t\t/// code? This would also require saving any HTML outside of the *> and <* \n\t\t\t\t/// then echoing it out. The throw below would have to be removed as well.\n\t\t\t\tthrow ex;\n\t\t\t}\n\t\t\telse\n\t\t\t\tthrow ex;\n\t\t}\n\n\t\tHot.Data = returnedOutput.hot.Data;\n\t\tlet finalOutput: string = returnedOutput.output;\n\t\tHot.Output = \"\";\n\n\t\treturn (finalOutput);\n\t}\n}","/**\n * The logging level.\n */\nexport enum HotLogLevel\n{\n\t/**\n\t * Prints only info messages.\n\t */\n\tInfo,\n\t/**\n\t * Prints only warning messages.\n\t */\n\tWarning,\n\t/**\n\t * Prints only error messages.\n\t */\n\tError,\n\t/**\n\t * Prints all messages.\n\t */\n\tVerbose,\n\t/**\n\t * Prints all messages, except verbose.\n\t */\n\tAll,\n\t/**\n\t * Doesn't print any message.\n\t */\n\tNone\n}\n\n/**\n * The logger.\n */\nexport class HotLog\n{\n\t/**\n\t * The logging level.\n\t */\n\tlogLevel: HotLogLevel;\n\n\tconstructor (logLevel: HotLogLevel = HotLogLevel.All)\n\t{\n\t\tthis.logLevel = logLevel;\n\t}\n\n\t/**\n\t * Log a message.\n\t */\n\tlog (level: HotLogLevel, message: string)\n\t{\n\t\tif (this.logLevel === HotLogLevel.Verbose)\n\t\t{\n\t\t\tif (level === HotLogLevel.Error)\n\t\t\t\tthis.error (message);\n\n\t\t\tif (level === HotLogLevel.Warning)\n\t\t\t\tthis.warning (message);\n\n\t\t\tif ((level === HotLogLevel.Info) || \n\t\t\t\t(level === HotLogLevel.Verbose))\n\t\t\t{\n\t\t\t\tthis.info (message);\n\t\t\t}\n\t\t}\n\n\t\tif (this.logLevel === HotLogLevel.All)\n\t\t{\n\t\t\tif (level === HotLogLevel.Error)\n\t\t\t\tthis.error (message);\n\n\t\t\tif (level === HotLogLevel.Warning)\n\t\t\t\tthis.warning (message);\n\n\t\t\tif (level === HotLogLevel.Info)\n\t\t\t\tthis.info (message);\n\t\t}\n\n\t\tif (this.logLevel === HotLogLevel.Error)\n\t\t{\n\t\t\tif (level === HotLogLevel.Error)\n\t\t\t\tthis.error (message);\n\t\t}\n\n\t\tif (this.logLevel === HotLogLevel.Warning)\n\t\t{\n\t\t\tif (level === HotLogLevel.Warning)\n\t\t\t\tthis.warning (message);\n\t\t}\n\n\t\tif (this.logLevel === HotLogLevel.Info)\n\t\t{\n\t\t\tif (level === HotLogLevel.Info)\n\t\t\t\tthis.info (message);\n\t\t}\n\t}\n\n\t/**\n\t * Log a verbose message.\n\t */\n\tverbose (message: string)\n\t{\n\t\tif (this.logLevel === HotLogLevel.Verbose)\n\t\t\tconsole.info (message);\n\t}\n\n\t/**\n\t * Log a message.\n\t */\n\tinfo (message: string)\n\t{\n\t\tif ((this.logLevel === HotLogLevel.All) || \n\t\t\t(this.logLevel === HotLogLevel.Verbose) || \n\t\t\t(this.logLevel === HotLogLevel.Info))\n\t\t{\n\t\t\tconsole.info (message);\n\t\t}\n\t}\n\n\t/**\n\t * Log a warning.\n\t */\n\twarning (message: string)\n\t{\n\t\tif ((this.logLevel === HotLogLevel.All) || \n\t\t\t(this.logLevel === HotLogLevel.Verbose) || \n\t\t\t(this.logLevel === HotLogLevel.Warning))\n\t\t{\n\t\t\tconsole.warn (message);\n\t\t}\n\t}\n\n\t/**\n\t * Log an error message.\n\t */\n\terror (message: string | Error)\n\t{\n\t\tif ((this.logLevel === HotLogLevel.All) || \n\t\t\t(this.logLevel === HotLogLevel.Verbose) || \n\t\t\t(this.logLevel === HotLogLevel.Error))\n\t\t{\n\t\t\tlet msg: string = \"\";\n\n\t\t\tif (typeof (message) === \"string\")\n\t\t\t\tmsg = message;\n\t\t\telse\n\t\t\t{\n\t\t\t\tif (message.message != null)\n\t\t\t\t\tmsg = message.message;\n\n\t\t\t\tif (message.stack != null)\n\t\t\t\t\tmsg = message.stack;\n\t\t\t}\n\n\t\t\tconsole.error (msg);\n\t\t}\n\t}\n}","import { Hot } from \"./Hot\";\nimport { HotFile } from \"./HotFile\";\nimport { HotStaq } from \"./HotStaq\";\nimport { HotAPI } from \"./HotAPI\";\nimport { HotTestElement } from \"./HotTestElement\";\nimport { HotTestMap, HotTestPath } from \"./HotTestMap\";\n\n/**\n * A page to preprocess.\n */\nexport interface IHotPage\n{\n\t/**\n\t * The processor to use.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The name of the page.\n\t */\n\tname?: string;\n\t/**\n\t * The route used to get to this page.\n\t */\n\troute?: string;\n\t/**\n\t * The name of the page. File ordering matters here.\n\t * Every file is processed incrementally.\n\t */\n\tfiles?: HotFile[];\n\t/**\n\t * The associated tester name.\n\t */\n\ttesterName?: string;\n\t/**\n\t * The associated tester map.\n\t */\n\ttesterMap?: string;\n\t/**\n\t * The elements to test on this page.\n\t */\n\ttestElements?: { [name: string]: HotTestElement; };\n\t/**\n\t * The test paths to test on this page.\n\t */\n\ttestPaths?: { [name: string]: HotTestPath; };\n}\n\n/**\n * A page to preprocess.\n */\nexport class HotPage implements IHotPage\n{\n\t/**\n\t * The processor to use.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The name of the page.\n\t */\n\tname: string;\n\t/**\n\t * The route used to get to this page.\n\t */\n\troute: string;\n\t/**\n\t * The name of the page. File ordering matters here.\n\t * Every file is processed incrementally.\n\t */\n\tfiles: HotFile[];\n\t/**\n\t * The associated tester name.\n\t */\n\ttesterName: string;\n\t/**\n\t * The associated tester map.\n\t */\n\ttesterMap: string;\n\t/**\n\t * The elements to test on this page.\n\t */\n\ttestElements: { [name: string]: HotTestElement; };\n\t/**\n\t * The test paths to test on this page.\n\t */\n\ttestPaths: { [name: string]: HotTestPath; };\n\n\tconstructor (copy: IHotPage | HotStaq)\n\t{\n\t\tif (copy instanceof HotStaq)\n\t\t{\n\t\t\tthis.processor = copy;\n\t\t\tthis.name = \"\";\n\t\t\tthis.testerName = \"\";\n\t\t\tthis.testerMap = \"\";\n\t\t\tthis.route = \"\";\n\t\t\tthis.files = [];\n\t\t\tthis.testElements = {};\n\t\t\tthis.testPaths = {};\n\t\t}\n\t\telse\n\t\t{\n\t\t\tthis.processor = copy.processor;\n\t\t\tthis.name = copy.name || \"\";\n\t\t\tthis.testerName = copy.testerName || \"\";\n\t\t\tthis.testerMap = copy.testerMap || \"\";\n\t\t\tthis.route = copy.route || \"\";\n\t\t\tthis.files = copy.files || [];\n\t\t\tthis.testElements = copy.testElements || {};\n\t\t\tthis.testPaths = copy.testPaths || {};\n\t\t}\n\t}\n\n\t/**\n\t * Add a file to process. It's recommend to load the file prior to \n\t * adding it to a page if it's about to be used.\n\t */\n\tasync addFile (file: HotFile): Promise<void>\n\t{\n\t\tfile.page = this;\n\n\t\tthis.files.push (file);\n\t}\n\n\t/**\n\t * Get the API associated with this page.\n\t */\n\tgetAPI (): HotAPI\n\t{\n\t\treturn (this.processor.api);\n\t}\n\n\t/**\n\t * Get the tester API associated with this page.\n\t */\n\tgetTesterAPI (): HotAPI\n\t{\n\t\treturn (this.processor.testerAPI);\n\t}\n\n\t/**\n\t * Add all files in the page. Could decrease page loading performance.\n\t * It's recommend to load the file prior to adding it to a page.\n\t */\n\tasync load (file: HotFile): Promise<void>\n\t{\n\t\tfor (let iIdx = 0; iIdx < this.files.length; iIdx++)\n\t\t{\n\t\t\tlet file: HotFile = this.files[iIdx];\n\n\t\t\tawait file.load ();\n\t\t}\n\t}\n\n\t/**\n\t * Process a page and get the result.\n\t */\n\tasync process (args: any = null): Promise<string>\n\t{\n\t\tlet output: string = \"\";\n\n\t\tfor (let iIdx = 0; iIdx < this.files.length; iIdx++)\n\t\t{\n\t\t\tlet file: HotFile = this.files[iIdx];\n\n\t\t\tHot.Output = \"\";\n\t\t\tfile.page = this;\n\n\t\t\toutput += await file.process (args);\n\t\t}\n\n\t\treturn (output);\n\t}\n\n\t/**\n\t * Add a test element.\n\t */\n\taddTestElement (elm: HotTestElement): void\n\t{\n\t\tif (this.testElements[elm.name] != null)\n\t\t\tthrow new Error (`Test element ${elm.name} already exists!`);\n\n\t\tthis.testElements[elm.name] = elm;\n\t}\n\n\t/**\n\t * Get a test element.\n\t */\n\tgetTestElement (name: string): HotTestElement\n\t{\n\t\tif (this.testElements[name] == null)\n\t\t\tthrow new Error (`Test element ${name} doest not exist!`);\n\n\t\treturn (this.testElements[name]);\n\t}\n\n\t/**\n\t * Create a test path.\n\t */\n\tcreateTestPath (pathName: string, driverFunc: HotTestPath): void\n\t{\n\t\tif (this.testPaths[pathName] != null)\n\t\t\tthrow new Error (`Test path ${pathName} already exists!`);\n\n\t\tthis.testPaths[pathName] = driverFunc;\n\t}\n}","import { HotServer } from \"./HotServer\";\nimport { HotRouteMethod, HTTPMethod, IHotRouteMethod, ServerExecutionFunction, TestCaseFunction, TestCaseObject } from \"./HotRouteMethod\";\nimport { HotClient } from \"./HotClient\";\nimport { HotLog } from \"./HotLog\";\n\n/**\n * The route to use.\n */\nexport class HotRoute\n{\n\t/**\n\t * The server that maintains the connections.\n\t */\n\tconnection: HotServer | HotClient;\n\t/**\n\t * The associated logger.\n\t */\n\tlogger: HotLog;\n\t/**\n\t * The route.\n\t */\n\troute: string;\n\t/**\n\t * The description of the route.\n\t */\n\tdescription: string;\n\t/**\n\t * The version.\n\t */\n\tversion: string;\n\t/**\n\t * The prefix to add to the beginning of each route method.\n\t */\n\tprefix: string;\n\t/**\n\t * The authorization credentials to be used by the client \n\t * when connecting to the server.\n\t */\n\tauthCredentials: any;\n\t/**\n\t * The calls that can be made.\n\t */\n\tmethods: HotRouteMethod[];\n\t/**\n\t * The errors and their JSON that can be thrown. Can be:\n\t * * not_authorized\n\t * * no_server_execute_function\n\t */\n\terrors: { [error: string]: any };\n\n\tconstructor (connection: HotServer | HotClient, route: string, methods: HotRouteMethod[] = [])\n\t{\n\t\tthis.connection = connection;\n\t\tthis.logger = null;\n\n\t\tif (this.connection != null)\n\t\t{\n\t\t\tif (this.connection.processor != null)\n\t\t\t\tthis.logger = this.connection.processor.logger;\n\t\t}\n\n\t\tthis.route = route;\n\t\tthis.description = \"\";\n\t\tthis.version = \"v1\";\n\t\tthis.prefix = \"\";\n\t\tthis.authCredentials = null;\n\t\tthis.methods = methods;\n\t\tthis.errors = {\n\t\t\t\t\"not_authorized\": HotRoute.createError (\"Not authorized.\"),\n\t\t\t\t\"no_server_execute_function\": HotRoute.createError (\"Missing server execute function.\"),\n\t\t\t};\n\t}\n\n\t/**\n\t * Create an error JSON object.\n\t */\n\tstatic createError (message: string): any\n\t{\n\t\treturn ({ error: message });\n\t}\n\n\t/**\n\t * Add an API method to this route.\n\t * \n\t * @param method The name of the method to add. If a HotRouteMethod is supplied, the \n\t * rest of the arguments supplied will be ignored.\n\t */\n\taddMethod (\n\t\tmethod: HotRouteMethod | IHotRouteMethod | string,\n\t\texecuteFunction: ServerExecutionFunction = null,\n\t\ttype: HTTPMethod = HTTPMethod.POST,\n\t\ttestCases: (string | TestCaseFunction)[] | TestCaseFunction[] | TestCaseObject[] = null\n\t\t): void\n\t{\n\t\tif (typeof (method) === \"string\")\n\t\t\tmethod = new HotRouteMethod (this, method, executeFunction, type, null, null, null, testCases);\n\n\t\tif (method instanceof HotRouteMethod)\n\t\t\tthis.methods.push (method);\n\t\telse\n\t\t{\n\t\t\tif (method.route == null)\n\t\t\t\tmethod.route = this;\n\n\t\t\tmethod = new HotRouteMethod (method);\n\t\t\tthis.methods.push ((<HotRouteMethod>method));\n\t\t}\n\t}\n\n\t/**\n\t * Get a method by it's name.\n\t */\n\tgetMethod (name: string): HotRouteMethod\n\t{\n\t\tlet foundMethod: HotRouteMethod = null;\n\n\t\tfor (let iIdx = 0; iIdx < this.methods.length; iIdx++)\n\t\t{\n\t\t\tlet method: HotRouteMethod = this.methods[iIdx];\n\n\t\t\tif (method.name === name)\n\t\t\t{\n\t\t\t\tfoundMethod = method;\n\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\n\t\treturn (foundMethod);\n\t}\n\n\t/**\n\t * Executes before all routes have been registered.\n\t */\n\tonPreRegister: () => Promise<void> = null;\n\t/**\n\t * Executes when first registering this route with Express. If \n\t * this returns false, the route will not be registered.\n\t */\n\tonRegister: () => Promise<boolean> = null;\n\t/**\n\t * Executes after all routes have been registered.\n\t */\n\tonPostRegister: () => Promise<void> = null;\n\n\t/**\n\t * Executes when authorizing a called method.\n\t * The value returned from here will be passed to onExecute in the \n\t * called HotRouteMethod. Undefined returning from here will mean \n\t * the authorization failed.\n\t */\n\tonAuthorizeUser: (req: any, res: any) => Promise<any> = null;\n}","import { DeveloperMode } from \"./Hot\";\nimport { HotTestDriver } from \"./HotTestDriver\";\nimport { HotRoute } from \"./HotRoute\";\nimport { HotServer } from \"./HotServer\";\n\n/**\n * Available HTTP methods.\n */\nexport enum HTTPMethod\n{\n\t/**\n\t * A HTTP GET request.\n\t */\n\tGET = \"get\",\n\t/**\n\t * A HTTP POST request.\n\t */\n\tPOST = \"post\",\n\t/**\n\t * This will upload a file, then post the json request afterwards.\n\t */\n\tFILE_UPLOAD = \"file_upload_then_post_json\"\n}\n\n/**\n * A function that will be executed by the server when first registering with Express.\n * If this returns false, this route method will not be registered.\n */\nexport type ServerRegistrationFunction = () => Promise<boolean>;\n/**\n * A function that will be executed by the server.\n */\nexport type ServerExecutionFunction = \n\t(req: any, res: any, authorizedValue: any, jsonObj: any, queryObj: any, files?: any) => Promise<any>;\n/**\n * A function that will be executed by the client.\n */\nexport type ClientExecutionFunction = (...args: any[]) => Promise<any>;\n/**\n * A function that will be executed by the server for authorization. Any value \n * returned from this function will be passed to the ServerExecutionFunction.\n * If an undefined value is returned, this indicates the server was not able \n * to authenticate the user, so the ServerExecutionFunction will not be \n * executed.\n */\nexport type ServerAuthorizationFunction = (req: any, res: any, jsonObj: any, queryObj: any) => Promise<any>;\n/**\n * The test case function to execute.\n */\nexport type TestCaseFunction = ((driver: HotTestDriver) => Promise<any>) | ((driver: HotTestDriver) => any);\n/**\n * The test case object to pass.\n */\nexport interface TestCaseObject\n{\n\t/**\n\t * The name of the test case.\n\t */\n\tname: string;\n\t/**\n\t * The function to execute.\n\t */\n\tfunc: TestCaseFunction;\n}\n\n/**\n * A method parameter.\n */\nexport interface HotRouteMethodParameter\n{\n\t/**\n\t * The type of parameter. Default: string\n\t * Can be:\n\t * * string\n\t * * integer\n\t * * number\n\t * * boolean\n\t * * array\n\t * * object\n\t */\n\ttype?: string;\n\t/**\n\t * The description of the parameter. Default: \"\"\n\t */\n\tdescription?: string;\n\t/**\n\t * Is this parameter required? Default: false\n\t */\n\trequired?: boolean;\n\t/**\n\t * The parameters in the object.\n\t */\n\tparameters?: { [name: string]: string | HotRouteMethodParameter; };\n}\n\n/**\n * An API method to make.\n */\nexport interface IHotRouteMethod\n{\n\t/**\n\t * The parent route.\n\t */\n\troute?: HotRoute;\n\t/**\n\t * The api call name.\n\t */\n\tname: string;\n\t/**\n\t * The description of the api method.\n\t */\n\tdescription?: string;\n\t/**\n\t * The description of what returns from the api method.\n\t */\n\treturns?: string | HotRouteMethodParameter;\n\t/**\n\t * The parameters in the api method.\n\t */\n\tparameters?: { [name: string]: string | HotRouteMethodParameter; };\n\t/**\n\t * The api call name.\n\t */\n\ttype?: HTTPMethod;\n\t/**\n\t * The authorization credentials to be used by the client \n\t * when connecting to the server.\n\t */\n\tauthCredentials?: any;\n\t/**\n\t * The test case objects to execute during tests.\n\t */\n\ttestCases?: {\n\t\t\t[name: string]: TestCaseObject;\n\t\t} | (string | TestCaseFunction)[] | TestCaseFunction[] | TestCaseObject[];\n\t/**\n\t * Executes before all routes have been registered.\n\t */\n\tonPreRegister?: () => Promise<void>;\n\t/**\n\t * Executes when first registering this method with Express. If \n\t * this returns false, the method will not be registered.\n\t */\n\tonRegister?: ServerRegistrationFunction;\n\t/**\n\t * Executes after all routes have been registered.\n\t */\n\tonPostRegister?: () => Promise<void>;\n\n\t/**\n\t * Executes when authorizing a called method. If this method \n\t * is set, this will not call onAuthorize for the parent HotRoute.\n\t * The value returned from here will be passed to onExecute. \n\t * Undefined returning from here will mean the authorization failed.\n\t * If any exceptions are thrown from this function, they will be sent \n\t * to the server as an { error: string; } object with the exception \n\t * message as the error.\n\t */\n\tonServerAuthorize?: ServerAuthorizationFunction;\n\n\t/**\n\t * Executes when executing a called method from the server side. \n\t * This will stringify any JSON object and send it as a JSON response. \n\t * If undefined is returned no response will be sent to the server. \n\t * So the developer would have to send a response using \"res\".\n\t * If any exceptions are thrown from this function, they will be sent \n\t * to the server as an { error: string; } object with the exception \n\t * message as the error.\n\t */\n\tonServerExecute?: ServerExecutionFunction;\n\t/**\n\t * Executes when executing a called method from the client side.\n\t * @fixme Is this necessary?\n\t */\n\tonClientExecute?: ClientExecutionFunction;\n}\n\n/**\n * An API method to make.\n */\nexport class HotRouteMethod implements IHotRouteMethod\n{\n\t/**\n\t * The parent route.\n\t */\n\troute: HotRoute;\n\t/**\n\t * The api call name.\n\t */\n\tname: string;\n\t/**\n\t * The description of the api method.\n\t */\n\tdescription: string;\n\t/**\n\t * The description of what returns from the api method.\n\t */\n\treturns: HotRouteMethodParameter;\n\t/**\n\t * The parameters in the api method.\n\t */\n\tparameters: { [name: string]: HotRouteMethodParameter; };\n\t/**\n\t * The api call name.\n\t */\n\ttype: HTTPMethod;\n\t/**\n\t * Has this method been registered with the server? This \n\t * prevents the method from being reregistered.\n\t */\n\tisRegistered: boolean;\n\t/**\n\t * Has this method been registered with the server? This \n\t * prevents the method from being reregistered.\n\t */\n\texecuteSetup: boolean;\n\t/**\n\t * The authorization credentials to be used by the client \n\t * when connecting to the server.\n\t */\n\tauthCredentials: any;\n\t/**\n\t * The test case objects to execute during tests.\n\t */\n\ttestCases: {\n\t\t\t[name: string]: TestCaseObject;\n\t\t};\n\t/**\n\t * Executes before all routes have been registered.\n\t */\n\tonPreRegister?: () => Promise<void>;\n\t/**\n\t * Executes when first registering this method with Express. If \n\t * this returns false, the method will not be registered.\n\t */\n\tonRegister?: ServerRegistrationFunction;\n\t/**\n\t * Executes after all routes have been registered.\n\t */\n\tonPostRegister?: () => Promise<void>;\n\n\t/**\n\t * Executes when authorizing a called method. If this method \n\t * is set, this will not call onAuthorize for the parent HotRoute.\n\t * The value returned from here will be passed to onExecute. \n\t * Undefined returning from here will mean the authorization failed.\n\t * If any exceptions are thrown from this function, they will be sent \n\t * to the server as an { error: string; } object with the exception \n\t * message as the error.\n\t */\n\tonServerAuthorize?: ServerAuthorizationFunction;\n\n\t/**\n\t * Executes when executing a called method from the server side. \n\t * This will stringify any JSON object and send it as a JSON response. \n\t * If undefined is returned no response will be sent to the server. \n\t * So the developer would have to send a response using \"res\".\n\t * If any exceptions are thrown from this function, they will be sent \n\t * to the server as an { error: string; } object with the exception \n\t * message as the error.\n\t */\n\tonServerExecute?: ServerExecutionFunction;\n\t/**\n\t * Executes when executing a called method from the client side.\n\t * @fixme Is this necessary?\n\t */\n\tonClientExecute?: ClientExecutionFunction;\n\n\tconstructor (route: HotRoute | IHotRouteMethod, name: string = \"\", \n\t\tonExecute: ServerExecutionFunction | ClientExecutionFunction = null, \n\t\ttype: HTTPMethod = HTTPMethod.POST, onServerAuthorize: ServerAuthorizationFunction = null, \n\t\tonRegister: ServerRegistrationFunction = null, authCredentials: any = null, \n\t\ttestCases: { [name: string]: TestCaseObject; } | (string | TestCaseFunction)[] | TestCaseFunction[] | TestCaseObject[] = null)\n\t{\n\t\tlet newRoute: HotRoute = null;\n\n\t\tif (route instanceof HotRoute)\n\t\t\tnewRoute = route;\n\t\telse\n\t\t{\n\t\t\tnewRoute = route.route;\n\n\t\t\tif (route.type != null)\n\t\t\t\ttype = route.type;\n\n\t\t\tif (route.name != null)\n\t\t\t\tname = route.name;\n\n\t\t\tif (route.description != null)\n\t\t\t\tthis.description = route.description;\n\n\t\t\tif (route.returns != null)\n\t\t\t{\n\t\t\t\tif (typeof (route.returns) === \"string\")\n\t\t\t\t{\n\t\t\t\t\tthis.returns = {\n\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\"required\": true,\n\t\t\t\t\t\t\t\"description\": route.returns\n\t\t\t\t\t\t};\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t\tthis.returns = route.returns;\n\t\t\t}\n\n\t\t\tif (route.parameters != null)\n\t\t\t{\n\t\t\t\tthis.parameters = {};\n\n\t\t\t\tfor (let key in route.parameters)\n\t\t\t\t{\n\t\t\t\t\tlet param = route.parameters[key];\n\n\t\t\t\t\tif (typeof (param) === \"string\")\n\t\t\t\t\t{\n\t\t\t\t\t\tthis.parameters[key] = {\n\t\t\t\t\t\t\t\t\"type\": param,\n\t\t\t\t\t\t\t\t\"required\": false,\n\t\t\t\t\t\t\t\t\"description\": \"\"\n\t\t\t\t\t\t\t};\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tif (param.type == null)\n\t\t\t\t\t\t\tparam.type = \"string\";\n\n\t\t\t\t\t\tthis.parameters[key] = param;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif (route.authCredentials != null)\n\t\t\t\tauthCredentials = route.authCredentials;\n\n\t\t\tif (route.onServerExecute != null)\n\t\t\t\tonExecute = route.onServerExecute;\n\n\t\t\tif (route.onServerAuthorize != null)\n\t\t\t\tonServerAuthorize = route.onServerAuthorize;\n\n\t\t\tif (route.onRegister != null)\n\t\t\t\tonRegister = route.onRegister;\n\n\t\t\tif (route.onPostRegister != null)\n\t\t\t\tthis.onPostRegister = route.onPostRegister;\n\n\t\t\tif (route.onServerExecute != null)\n\t\t\t\tthis.onServerExecute = route.onServerExecute;\n\n\t\t\tif (route.onClientExecute != null)\n\t\t\t\tthis.onClientExecute = route.onClientExecute;\n\n\t\t\tif (route.testCases != null)\n\t\t\t\ttestCases = route.testCases;\n\t\t}\n\n\t\tif (name === \"\")\n\t\t\tthrow new Error (`All route methods must have a name!`);\n\n\t\tthis.route = newRoute;\n\t\tthis.name = name;\n\t\tthis.type = type;\n\t\tthis.isRegistered = false;\n\t\tthis.executeSetup = false;\n\t\tthis.authCredentials = authCredentials;\n\t\tthis.onServerAuthorize = onServerAuthorize;\n\t\tthis.onRegister = onRegister;\n\t\tthis.testCases = {};\n\n\t\tif (this.route.connection.processor.mode === DeveloperMode.Development)\n\t\t{\n\t\t\tif (testCases != null)\n\t\t\t{\n\t\t\t\tif (testCases instanceof Array)\n\t\t\t\t{\n\t\t\t\t\tfor (let iIdx = 0; iIdx < testCases.length; iIdx++)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet obj = testCases[iIdx];\n\n\t\t\t\t\t\tif (typeof (obj) === \"string\")\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tconst testCaseName: string = obj;\n\t\t\t\t\t\t\tconst func: TestCaseFunction = (<TestCaseFunction>testCases[iIdx + 1]);\n\n\t\t\t\t\t\t\tthis.addTestCase (testCaseName, func);\n\t\t\t\t\t\t\tiIdx++;\n\t\t\t\t\t\t}\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tthis.addTestCase (obj);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\tfor (let key in testCases)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet obj = testCases[key];\n\n\t\t\t\t\t\tthis.addTestCase (obj);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tif (this.route.connection instanceof HotServer)\n\t\t\tthis.onServerExecute = onExecute;\n\t\t//else\n\t\t\t//this.onClientExecute = onExecute;\n\t}\n\n\t/**\n\t * Add a new test case.\n\t */\n\taddTestCase (newTestCase: TestCaseObject | string | TestCaseFunction, \n\t\t\ttestCaseFunction: TestCaseFunction = null): void\n\t{\n\t\tif (typeof (newTestCase) === \"string\")\n\t\t{\n\t\t\tconst name: string = newTestCase;\n\t\t\tconst func: TestCaseFunction = testCaseFunction;\n\n\t\t\tthis.testCases[name] = {\n\t\t\t\t\tname: name,\n\t\t\t\t\tfunc: func\n\t\t\t\t};\n\n\t\t\treturn;\n\t\t}\n\n\t\tif (typeof (newTestCase) === \"function\")\n\t\t{\n\t\t\tconst testCaseId: number = Object.keys (this.testCases).length;\n\t\t\tconst name: string = `${this.route.route}/${this.name} test case ${testCaseId}`;\n\t\t\tconst func: TestCaseFunction = (<TestCaseFunction>newTestCase);\n\n\t\t\tthis.testCases[name] = {\n\t\t\t\t\tname: name,\n\t\t\t\t\tfunc: func\n\t\t\t\t};\n\n\t\t\treturn;\n\t\t}\n\n\t\tconst testCase: TestCaseObject = (<TestCaseObject>newTestCase);\n\t\tthis.testCases[testCase.name] = testCase;\n\t}\n}","import { HotStaq } from \"./HotStaq\";\nimport { HotLog } from \"./HotLog\";\nimport { HotAPI } from \"./HotAPI\";\nimport { HotRoute } from \"./HotRoute\";\n\n/**\n * The type of server.\n */\nexport enum HotServerType\n{\n\tHTTP,\n\tWebSockets,\n\tGenerate\n}\n\n/**\n * The server.\n */\nexport interface IHotServer\n{\n\t/**\n\t * The processor to use.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The server type.\n\t */\n\tserverType: string;\n\t/**\n\t * The API to use.\n\t */\n\tapi: HotAPI;\n\t/**\n\t * The network address to listen on.\n\t */\n\tlistenAddress: string;\n\t/**\n\t * The ports to use.\n\t */\n\tports: {\n\t\t\thttp: number;\n\t\t\thttps: number;\n\t\t};\n\t/**\n\t * SSL settings.\n\t */\n\tssl: {\n\t\t\t/**\n\t\t\t * The SSL certificate to use.\n\t\t\t */\n\t\t\tcert: string;\n\t\t\t/**\n\t\t\t * The SSL certificate key to use.\n\t\t\t */\n\t\t\tkey: string;\n\t\t\t/**\n\t\t\t * The SSL certificate CA to use.\n\t\t\t */\n\t\t\tca: string;\n\t\t};\n\t/**\n\t * Redirect HTTP traffic to HTTPS.\n\t */\n\tredirectHTTPtoHTTPS: boolean;\n\t/**\n\t * The type of server.\n\t */\n\ttype: HotServerType;\n\t/**\n\t * The logger.\n\t */\n\tlogger: HotLog;\n\t/**\n\t * Any secrets associated with this server.\n\t */\n\tsecrets: any;\n}\n\n/**\n * The server.\n */\nexport class HotServer implements IHotServer\n{\n\t/**\n\t * The processor to use.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The server type.\n\t */\n\tserverType: string;\n\t/**\n\t * The API to use.\n\t */\n\tapi: HotAPI;\n\t/**\n\t * The network address to listen on.\n\t */\n\tlistenAddress: string;\n\t/**\n\t * The ports to use.\n\t */\n\tports: {\n\t\t\thttp: number;\n\t\t\thttps: number;\n\t\t};\n\t/**\n\t * SSL settings.\n\t */\n\tssl: {\n\t\t\t/**\n\t\t\t * The SSL certificate to use.\n\t\t\t */\n\t\t\tcert: string;\n\t\t\t/**\n\t\t\t * The SSL certificate key to use.\n\t\t\t */\n\t\t\tkey: string;\n\t\t\t/**\n\t\t\t * The SSL certificate CA to use.\n\t\t\t */\n\t\t\tca: string;\n\t\t};\n\t/**\n\t * Redirect HTTP traffic to HTTPS.\n\t */\n\tredirectHTTPtoHTTPS: boolean;\n\t/**\n\t * The type of server.\n\t */\n\ttype: HotServerType;\n\t/**\n\t * The logger.\n\t */\n\tlogger: HotLog;\n\t/**\n\t * Any secrets associated with this server.\n\t */\n\tsecrets: any;\n\n\tconstructor (processor: HotStaq | HotServer)\n\t{\n\t\tif (processor instanceof HotStaq)\n\t\t{\n\t\t\tthis.processor = processor;\n\t\t\tthis.serverType = \"Server\";\n\t\t\tthis.api = null;\n\t\t\tthis.listenAddress = \"0.0.0.0\";\n\t\t\tthis.ports = {\n\t\t\t\t\thttp: 80,\n\t\t\t\t\thttps: 443\n\t\t\t\t};\n\t\t\tthis.ssl = {\n\t\t\t\t\tcert: \"\",\n\t\t\t\t\tkey: \"\",\n\t\t\t\t\tca: \"\"\n\t\t\t\t};\n\t\t\tthis.redirectHTTPtoHTTPS = true;\n\t\t\tthis.type = HotServerType.HTTP;\n\t\t\tthis.logger = processor.logger;\n\t\t\tthis.secrets = {};\n\t\t}\n\t\telse\n\t\t{\n\t\t\tthis.processor = processor.processor;\n\t\t\tthis.serverType = processor.serverType || \"Server\";\n\t\t\tthis.api = processor.api || null;\n\t\t\tthis.listenAddress = processor.listenAddress || \"0.0.0.0\";\n\t\t\tthis.ports = processor.ports || {\n\t\t\t\t\thttp: 80,\n\t\t\t\t\thttps: 443\n\t\t\t\t};\n\t\t\tthis.ssl = processor.ssl || {\n\t\t\t\t\tcert: \"\",\n\t\t\t\t\tkey: \"\",\n\t\t\t\t\tca: \"\"\n\t\t\t\t};\n\t\t\tthis.redirectHTTPtoHTTPS = processor.redirectHTTPtoHTTPS != null ? processor.redirectHTTPtoHTTPS : true;\n\t\t\tthis.type = processor.type || HotServerType.HTTP;\n\t\t\tthis.logger = processor.logger;\n\t\t\tthis.secrets = processor.secrets || {};\n\t\t}\n\t}\n\n\t/**\n\t * Set an API to this server. This will also set the associated \n\t * processor to this API as well.\n\t */\n\tasync setAPI (api: HotAPI): Promise<void>\n\t{\n\t\tthis.processor.api = api;\n\t\tthis.api = api;\n\n\t\t//if (registerRoutes === true)\n\t\t\t//await this.api.registerRoutes ();\n\t}\n\n\t/**\n\t * Register a route with the server.\n\t */\n\tasync registerRoute? (route: HotRoute): Promise<void>;\n\n\t/**\n\t * Start listening for requests.\n\t */\n\tasync listen? (): Promise<void>;\n\n\t/**\n\t * Shutdown the server.\n\t */\n\tasync shutdown? (): Promise<void>;\n}","import * as fs from \"fs\";\nimport * as ppath from \"path\";\n\nimport fetch from \"node-fetch\";\nimport validateModuleName from \"validate-npm-package-name\";\n\nimport { HotPage } from \"./HotPage\";\nimport { HotFile } from \"./HotFile\";\n\nimport { HotComponent, HotComponentOutput, IHotComponent } from \"./HotComponent\";\nimport { HotLog, HotLogLevel } from \"./HotLog\";\nimport { HotAPI } from \"./HotAPI\";\nimport { HotServer } from \"./HotServer\";\nimport { DeveloperMode, Hot } from \"./Hot\";\nimport { HotClient } from \"./HotClient\";\n\nimport { HotTester } from \"./HotTester\";\nimport { HotTesterAPI } from \"./HotTesterAPI\";\nimport { HotTestDriver } from \"./HotTestDriver\";\nimport { HotTestDestination, HotTestMap } from \"./HotTestMap\";\nimport { ServableFileExtension } from \"./HotHTTPServer\";\n\nvar HotTesterMocha: any = null;\nvar HotTesterMochaSelenium: any = null;\nvar HotTestSeleniumDriver: any = null;\n\n/**\n * A map path for testing.\n */\nexport interface HotSiteMapPath\n{\n\t/**\n\t * If set to true, this will start automatically when tests start.\n\t * The default is true.\n\t */\n\tautoStart?: boolean;\n\t/**\n\t * The path to the \n\t */\n\tpath?: string;\n}\n\n/**\n * A route used in a HotSite.\n */\nexport interface HotSiteRoute\n{\n\t/**\n\t * The name of the route. Will appear in the title.\n\t */\n\tname: string;\n\t/**\n\t * The url to the file to load.\n\t */\n\turl: string;\n\t/**\n\t * The name of the API to interface with.\n\t */\n\tapi?: string;\n\t/**\n\t * The order in which destinations are supposed to execute. This is \n\t * ignored if the destinations are an array.\n\t */\n\tdestinationOrder?: string[];\n\t/**\n\t * The HotTesterMap to use. This can be the name of an \n\t * existing one attached to the selected tester, or \n\t * can be an array of destinations that will be used to \n\t * create a new map.\n\t */\n\tmap?: string | string[] | { [name: string]: string | HotSiteMapPath; } | HotSiteMapPath[];\n}\n\n/**\n * A HotSite to load. This SHOULD NOT contain any private secret keys, passwords, \n * or database connection information related to the server. As such, future \n * versions of the HotSite interface should not contain any database related \n * connection info.\n */\nexport interface HotSite\n{\n\t/**\n\t * The name of this HotSite.\n\t */\n\tname: string;\n\t/**\n\t * The version of this HotSite.\n\t */\n\tversion?: string;\n\t/**\n\t * The description of this HotSite.\n\t */\n\tdescription?: string;\n\t/**\n\t * The path to the current HotSite. This is filled in during parsing.\n\t */\n\thotsitePath?: string;\n\t/**\n\t * Additional web server configuration.\n\t */\n\tserver?: {\n\t\t\t/**\n\t\t\t * The default name for a served Hott file.\n\t\t\t */\n\t\t\tname?: string;\n\t\t\t/**\n\t\t\t * If set to true, this will serve ALL files, including potentially secret files.\n\t\t\t */\n\t\t\tserveSecretFiles?: boolean;\n\t\t\t/**\n\t\t\t * Serve the following file extensions when requested.\n\t\t\t */\n\t\t\tserveFileExtensions?: (string | ServableFileExtension)[];\n\t\t\t/**\n\t\t\t * The name of the API to interface with across all pages.\n\t\t\t */\n\t\t\tglobalApi?: string;\n\t\t\t/**\n\t\t\t * The base url for the server.\n\t\t\t */\n\t\t\turl?: string;\n\t\t\t/**\n\t\t\t * The JavaScript source path.\n\t\t\t */\n\t\t\tjsSrcPath?: string;\n\t\t\t/**\n\t\t\t * The ports to use.\n\t\t\t */\n\t\t\tports?: {\n\t\t\t\t\t/**\n\t\t\t\t\t * The web HTTP port to serve on.\n\t\t\t\t\t */\n\t\t\t\t\thttp?: number;\n\t\t\t\t\t/**\n\t\t\t\t\t * The web HTTPS port to serve on.\n\t\t\t\t\t */\n\t\t\t\t\thttps?: number;\n\t\t\t\t\t/**\n\t\t\t\t\t * If set to true, this will redirect from HTTP to HTTPS for a web and web-api server.\n\t\t\t\t\t */\n\t\t\t\t\tredirectHTTPtoHTTPS?: boolean;\n\t\t\t\t\t/**\n\t\t\t\t\t * The api HTTP port to serve on.\n\t\t\t\t\t */\n\t\t\t\t\tapiHttp?: number;\n\t\t\t\t\t/**\n\t\t\t\t\t * The api HTTPS port to serve on.\n\t\t\t\t\t */\n\t\t\t\t\tapiHttps?: number;\n\t\t\t\t\t/**\n\t\t\t\t\t * If set to true, this will redirect from HTTP to HTTPS for an api server.\n\t\t\t\t\t */\n\t\t\t\t\tapiRedirectHTTPtoHTTPS?: boolean;\n\t\t\t\t};\n\t\t\t/**\n\t\t\t * The list of directory to serve to the client from the server.\n\t\t\t */\n\t\t\tserveDirectories?: {\n\t\t\t\t\t/**\n\t\t\t\t\t * The web route to take.\n\t\t\t\t\t */\n\t\t\t\t\troute: string;\n\t\t\t\t\t/**\n\t\t\t\t\t * The local filesystem path to serve pages from.\n\t\t\t\t\t */\n\t\t\t\t\tlocalPath: string;\n\t\t\t\t}[];\n\t\t\t/**\n\t\t\t * How to handle errors.\n\t\t\t */\n\t\t\terrors?: {\n\t\t\t\t/**\n\t\t\t\t * On a 404, serve a local file.\n\t\t\t\t */\n\t\t\t\ton404?: string;\n\t\t\t\t/**\n\t\t\t\t * On an error other than a 404, serve a local file.\n\t\t\t\t */\n\t\t\t\tonOther?: string;\n\t\t\t};\n\t\t};\n\t/**\n\t * Testing related functionality.\n\t */\n\ttesting?: {\n\t\t\tweb?: {\n\t\t\t\t/**\n\t\t\t\t * The tester class to use. EX: HotTesterMochaSelenium\n\t\t\t\t */\n\t\t\t\ttester?: string;\n\t\t\t\t/**\n\t\t\t\t * The name of the tester to use.\n\t\t\t\t */\n\t\t\t\ttesterName?: string;\n\t\t\t\t/**\n\t\t\t\t * If set to true, this will create a new tester.\n\t\t\t\t * Default Value: true\n\t\t\t\t */\n\t\t\t\tcreateNewTester?: boolean;\n\t\t\t\t/**\n\t\t\t\t * The url that connects to the tester api server.\n\t\t\t\t */\n\t\t\t\ttesterAPIUrl?: string;\n\t\t\t\t/**\n\t\t\t\t * The name of the test driver to use.\n\t\t\t\t */\n\t\t\t\tdriver?: string;\n\t\t\t\t/**\n\t\t\t\t * The number of milliseconds to wait before executing the next command.\n\t\t\t\t * Default is set to 20.\n\t\t\t\t */\n\t\t\t\tcommandDelay?: number;\n\t\t\t\t/**\n\t\t\t\t * The url to the html that loads the hott files.\n\t\t\t\t */\n\t\t\t\tlaunchpadUrl?: string;\n\t\t\t\t/**\n\t\t\t\t * The maps to test in order.\n\t\t\t\t */\n\t\t\t\tmaps?: string[];\n\t\t\t},\n\t\t\tapi?: {\n\t\t\t\t/**\n\t\t\t\t * The tester class to use. EX: HotTesterMocha\n\t\t\t\t */\n\t\t\t\ttester?: string;\n\t\t\t\t/**\n\t\t\t\t * The name of the tester to use.\n\t\t\t\t */\n\t\t\t\ttesterName?: string;\n\t\t\t\t/**\n\t\t\t\t * If set to true, this will create a new tester.\n\t\t\t\t * Default Value: true\n\t\t\t\t */\n\t\t\t\tcreateNewTester?: boolean;\n\t\t\t\t/**\n\t\t\t\t * The url that connects to the tester api server.\n\t\t\t\t */\n\t\t\t\ttesterAPIUrl?: string;\n\t\t\t\t/**\n\t\t\t\t * The name of the test driver to use.\n\t\t\t\t */\n\t\t\t\tdriver?: string;\n\t\t\t\t/**\n\t\t\t\t * The url to the html that loads the hott files.\n\t\t\t\t */\n\t\t\t\tlaunchpadUrl?: string;\n\t\t\t\t/**\n\t\t\t\t * The maps to test in order.\n\t\t\t\t */\n\t\t\t\tmaps?: string[];\n\t\t\t}\n\t\t};\n\t/**\n\t * The routes to load.\n\t */\n\troutes?: {\n\t\t\t[routeName: string]: HotSiteRoute;\n\t\t};\n\t/**\n\t * The available APIs on the server. The server must already have these \n\t * loaded.\n\t */\n\tapis?: {\n\t\t\t[name: string]: {\n\t\t\t\t\t/**\n\t\t\t\t\t * The JS API file to load.\n\t\t\t\t\t */\n\t\t\t\t\tjsapi?: string;\n\t\t\t\t\t/**\n\t\t\t\t\t * The exported JS library name to use.\n\t\t\t\t\t */\n\t\t\t\t\tlibraryName?: string;\n\t\t\t\t\t/**\n\t\t\t\t\t * The name of the api to use.\n\t\t\t\t\t */\n\t\t\t\t\tapiName?: string;\n\t\t\t\t\t/**\n\t\t\t\t\t * The port to use.\n\t\t\t\t\t */\n\t\t\t\t\tport?: number;\n\t\t\t\t\t/**\n\t\t\t\t\t * The public base url for the api.\n\t\t\t\t\t */\n\t\t\t\t\turl?: string;\n\t\t\t\t\t/**\n\t\t\t\t\t * The server-side filepath for the api.\n\t\t\t\t\t */\n\t\t\t\t\tfilepath?: string;\n\t\t\t\t\t/**\n\t\t\t\t\t * The maps to test in order.\n\t\t\t\t\t */\n\t\t\t\t\tmap?: string[];\n\t\t\t\t};\n\t\t};\n\t/**\n\t * Public keys that are embedded into the page.\n\t */\n\tpublicKeys?: {\n\t\t\t[name: string]: string | {\n\t\t\t\t\t/**\n\t\t\t\t\t * The key of an API secret to pass to the site to \n\t\t\t\t\t * be used publicly.\n\t\t\t\t\t */\n\t\t\t\t\tpassSecretFromAPI?: string;\n\t\t\t\t\t/**\n\t\t\t\t\t * Get the public secret from an environment variable.\n\t\t\t\t\t */\n\t\t\t\t\tenv?: string;\n\t\t\t\t};\n\t\t};\n\t/**\n\t * The components to load and register.\n\t */\n\tcomponents?: {\n\t\t\t[name: string]: {\n\t\t\t\t\t/**\n\t\t\t\t\t * The url to the component to load and register.\n\t\t\t\t\t */\n\t\t\t\t\turl: string;\n\t\t\t\t};\n\t\t};\n\t/**\n\t * The files to load and save in memory.\n\t */\n\tfiles?: {\n\t\t\t[name: string]: {\n\t\t\t\t\t/**\n\t\t\t\t\t * The url to the file to load.\n\t\t\t\t\t */\n\t\t\t\t\turl: string;\n\t\t\t\t};\n\t\t};\n\t/**\n\t * If set to true, this will disable loading files into memory.\n\t */\n\tdisableFileLoading?: boolean;\n}\n\n/**\n * The options to use when starting a page.\n */\nexport interface HotStartOptions\n{\n\t/**\n\t * The Hott site to load.\n\t */\n\turl?: string;\n\t/**\n\t * The content to display.\n\t */\n\tcontent?: string;\n\t/**\n\t * The name of the page to load.\n\t */\n\tname?: string;\n\t/**\n\t * The processor to use to load the page.\n\t */\n\tprocessor?: HotStaq;\n\t/**\n\t * Any arguments to pass to the new page.\n\t */\n\targs?: any;\n\t/**\n\t * The name of the tester to use.\n\t */\n\ttesterName?: string;\n\t/**\n\t * The name of the tester map to use.\n\t */\n\ttesterMap?: string;\n\t/**\n\t * The base url for the tester api.\n\t */\n\ttesterAPIBaseUrl?: string;\n\t/**\n\t * The url to the html that loads the hott file that's \n\t * pointed at the url above.\n\t */\n\ttesterLaunchpadUrl?: string;\n}\n\n/**\n * The main class that handles all HTML preprocessing, then outputs the \n * results.\n */\nexport interface IHotStaq\n{\n\t/**\n\t * The api that's used to communicate with.\n\t */\n\tapi?: HotAPI;\n\t/**\n\t * The tester api that's used to communicate with.\n\t */\n\ttesterAPI?: HotAPI;\n\t/**\n\t * Indicates what type of execution this is.\n\t */\n\tmode?: DeveloperMode;\n\t/**\n\t * The pages that can be constructed.\n\t */\n\tpages?: { [name: string]: HotPage };\n\t/**\n\t * The components that can be constructed.\n\t */\n\tcomponents?: {\n\t\t[tagName: string]: {\n\t\t\t\tcomponentType: (new  (copy: IHotComponent | HotStaq, api?: HotAPI) => HotComponent), \n\t\t\t\tprocessor: HotStaq, \n\t\t\t\tapi: HotAPI\n\t\t\t}\n\t\t};\n\t/**\n\t * The files that can be stored for later use.\n\t */\n\tfiles?: { [name: string]: HotFile };\n\t/**\n\t * The loaded hotsite.\n\t */\n\thotSite?: HotSite;\n}\n\n/**\n * The main class that handles all HTML preprocessing, then outputs the \n * results.\n */\nexport class HotStaq implements IHotStaq\n{\n\t/**\n\t * The current version of HotStaq.\n\t */\n\tstatic version: string = \"0.6.23\";\n\t/**\n\t * Indicates if this is a web build.\n\t */\n\tstatic isWeb: boolean = false;\n\t/**\n\t * Indicates if this is ready for testing.\n\t */\n\tstatic isReadyForTesting: boolean = false;\n\t/**\n\t * Executes this event when this page is ready for testing.\n\t */\n\tstatic onReadyForTesting: () => Promise<void> = null;\n\t/**\n\t * Errors to execute when something goes wrong.\n\t */\n\tstatic errors: { [name: string]: { redirectToUrl?: string; func?: (errType: string) => void; }; } = {};\n\t/**\n\t * Indicates what type of execution this is.\n\t */\n\tmode: DeveloperMode;\n\t/**\n\t * The api that's used to communicate with.\n\t */\n\tapi: HotAPI;\n\t/**\n\t * The tester api that's used to communicate with.\n\t */\n\ttesterAPI: HotAPI;\n\t/**\n\t * The pages that can be constructed.\n\t */\n\tpages: { [name: string]: HotPage };\n\t/**\n\t * The components that can be constructed.\n\t */\n\tcomponents: {\n\t\t[tagName: string]: {\n\t\t\t\tcomponentType: (new  (copy: IHotComponent | HotStaq, api?: HotAPI) => HotComponent), \n\t\t\t\tprocessor: HotStaq, \n\t\t\t\tapi: HotAPI\n\t\t\t}\n\t\t};\n\t/**\n\t * The files that can be stored for later use.\n\t */\n\tfiles: { [name: string]: HotFile };\n\t/**\n\t * The loaded hotsite.\n\t */\n\thotSite: HotSite;\n\t/**\n\t * The api content to use when about to load HotStaq.\n\t */\n\tapiContent: string;\n\t/**\n\t * The tester api content to use when about to load HotStaq.\n\t */\n\ttesterApiContent: string;\n\t/**\n\t * The page content to use when about to load HotStaq.\n\t */\n\tpageContent: string;\n\t/**\n\t * The logger.\n\t */\n\tlogger: HotLog;\n\t/**\n\t * The public keys to be exposed.\n\t */\n\tpublicKeys: any;\n\t/**\n\t * The testers that will be used to execute tests.\n\t */\n\ttesters: { [name: string]: HotTester };\n\n\tconstructor (copy: IHotStaq = {})\n\t{\n\t\tthis.api = copy.api || null;\n\t\tthis.testerAPI = copy.testerAPI || null;\n\t\tthis.mode = copy.mode || DeveloperMode.Production;\n\t\tthis.pages = copy.pages || {};\n\t\tthis.components = copy.components || {};\n\t\tthis.files = copy.files || {};\n\t\tthis.hotSite = copy.hotSite || null;\n\t\tthis.apiContent = `\n\t\t\t{\n\t\t\t\tvar %api_name% = %api_exported_name%.%api_name%;\n\t\t\t\tvar newHotClient = new HotClient (processor);\n\t\t\t\tvar newapi = new %api_name% (%base_url%, newHotClient);\n\t\t\t\tnewHotClient.api = newapi;\n\t\t\t\tprocessor.api = newapi;\n\t\t\t}`;\n\t\tthis.testerApiContent = `\n\t\t\tvar HotTesterAPI = HotStaqWeb.HotTesterAPI;\n\t\t\tvar newHotTesterClient = new HotClient (processor);\n\t\t\tvar newtesterapi = new HotTesterAPI (%base_tester_url%, newHotTesterClient);\n\t\t\tnewHotTesterClient.testerAPI = newtesterapi;\n\t\t\tprocessor.testerAPI = newtesterapi;`;\n\t\tthis.pageContent = \n`<!DOCTYPE html>\n<html>\n\n<head>\n\t<title>%title%</title>\n\n\t<script type = \"text/javascript\" src = \"%hotstaq_js_src%\"></script>\n\t<script type = \"text/javascript\">\n\t\twindow.HotStaq = HotStaqWeb.HotStaq;\n\t\twindow.HotClient = HotStaqWeb.HotClient;\n\t\twindow.HotAPI = HotStaqWeb.HotAPI;\n\t\twindow.Hot = HotStaqWeb.Hot;\n\t</script>\n\n%apis_to_load%\n\n\t<script type = \"text/javascript\">\n\t\tfunction hotstaq_startApp ()\n\t\t{\n\t\t\tlet tempMode = 0;\n\n\t\t\tif (window[\"Hot\"] != null)\n\t\t\t\ttempMode = Hot.Mode;\n\n\t\t\t%load_hot_site%\n\n\t\t\tvar processor = new HotStaq ();\n\t\t\tvar promises = [];\n\t\t\t%developer_mode%\n\n\t\t\t%api_code%\n\n\t\t\t%public_secrets%\n\t\t\t%tester_api%\n\t\t\t%load_files%\n\n\t\t\tprocessor.mode = tempMode;\n\n\t\t\tPromise.all (promises).then (function ()\n\t\t\t\t{\n\t\t\t\t\tHotStaq.displayUrl ({\n\t\t\t\t\t\t\turl: \"%url%\",\n\t\t\t\t\t\t\tname: \"%title%\",\n\t\t\t\t\t\t\tprocessor: processor,\n\t\t\t\t\t\t\targs: %args%,\n\t\t\t\t\t\t\ttesterName: %tester_name%,\n\t\t\t\t\t\t\ttesterMap: %tester_map%,\n\t\t\t\t\t\t\ttesterAPIBaseUrl: %tester_api_base_url%,\n\t\t\t\t\t\t\ttesterLaunchpadUrl: %tester_launchpad_url%\n\t\t\t\t\t\t});\n\t\t\t\t});\n\t\t}\n\n\t\thotstaq_startApp ();\n\t</script>\n</head>\n\n<body>\n</body>\n\n</html>`;\n\t\tthis.logger = new HotLog (HotLogLevel.None);\n\t\tthis.publicKeys = {};\n\t\tthis.testers = {};\n\t}\n\n\t/**\n\t * Parse a boolean value.\n\t */\n\tstatic parseBoolean (value: string): boolean\n\t{\n\t\tvalue = value.toLowerCase ();\n\n\t\tif (value === \"true\")\n\t\t\treturn (true);\n\n\t\tif (value === \"false\")\n\t\t\treturn (false);\n\n\t\tif (value === \"yes\")\n\t\t\treturn (true);\n\n\t\tif (value === \"no\")\n\t\t\treturn (false);\n\n\t\tif (value === \"yep\")\n\t\t\treturn (true);\n\n\t\tif (value === \"nah\")\n\t\t\treturn (false);\n\n\t\ttry\n\t\t{\n\t\t\tif (parseInt (value) != 0)\n\t\t\t\treturn (true);\n\t\t}\n\t\tcatch (ex)\n\t\t{\n\t\t}\n\n\t\treturn (false);\n\t}\n\n\t/**\n\t * Check if a required parameter exists inside an object. If it exists, return the value.\n\t */\n\tstatic getParam (name: string, objWithParam: any, required: boolean = true, throwException: boolean = true): any\n\t{\n\t\tlet value: any = objWithParam[name];\n\n\t\tif (value == null)\n\t\t{\n\t\t\tif (required === true)\n\t\t\t{\n\t\t\t\tif (throwException === true)\n\t\t\t\t\tthrow new Error (`Missing required parameter ${name}.`);\n\t\t\t}\n\t\t}\n\n\t\tif (typeof (value) === \"string\")\n\t\t{\n\t\t\tif (required === true)\n\t\t\t{\n\t\t\t\tif (value === \"\")\n\t\t\t\t{\n\t\t\t\t\tif (throwException === true)\n\t\t\t\t\t\tthrow new Error (`Missing required parameter ${name}.`);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn (value);\n\t}\n\n\t/**\n\t * Check if a required parameter exists inside an object. If it exists, return the value.\n\t * If it does not exist, return a default value instead.\n\t */\n\tstatic getParamDefault (name: string, objWithParam: any, defaultValue: any): any\n\t{\n\t\tlet value: any = objWithParam[name];\n\n\t\tif (value == null)\n\t\t\treturn (defaultValue);\n\n\t\tif (typeof (value) === \"string\")\n\t\t{\n\t\t\tif (value === \"\")\n\t\t\t\treturn (defaultValue);\n\t\t}\n\n\t\treturn (value);\n\t}\n\n\t/**\n\t * Execute an error. This cannot be an async function due to the nature of how this works.\n\t */\n\tstatic executeError (errType: string)\n\t{\n\t\tif (HotStaq.errors[errType] != null)\n\t\t{\n\t\t\tlet url: string = HotStaq.errors[errType].redirectToUrl;\n\n\t\t\tif (url != null)\n\t\t\t{\n\t\t\t\tif (url !== \"\")\n\t\t\t\t{\n\t\t\t\t\twindow.location.href = url;\n\n\t\t\t\t\treturn;\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tlet func = HotStaq.errors[errType].func;\n\n\t\t\tif (func != null)\n\t\t\t\tfunc (errType);\n\t\t}\n\t}\n\n\t/**\n\t * Wait for a number of milliseconds.\n\t */\n\tstatic async wait (numMilliseconds: number): Promise<void>\n\t{\n\t\treturn (await new Promise ((resolve, reject) =>\n\t\t\t{\n\t\t\t\tsetTimeout (() =>\n\t\t\t\t\t{\n\t\t\t\t\t\tresolve ();\n\t\t\t\t\t}, numMilliseconds);\n\t\t\t}));\n\t}\n\n\t/**\n\t * Add a page.\n\t */\n\taddPage (page: HotPage): void\n\t{\n\t\tthis.pages[page.name] = page;\n\t}\n\n\t/**\n\t * Get a page to process.\n\t */\n\tgetPage (pageName: string): HotPage\n\t{\n\t\treturn (this.pages[pageName]);\n\t}\n\n\t/**\n\t * Add a file.\n\t */\n\taddFile (file: HotFile): void\n\t{\n\t\tlet name: string = file.name;\n\n\t\tif (name === \"\")\n\t\t\tname = file.localFile;\n\n\t\tif (name === \"\")\n\t\t\tname = file.url;\n\n\t\tthis.files[name] = file;\n\t}\n\n\t/**\n\t * Get a file.\n\t */\n\tgetFile (name: string): HotFile\n\t{\n\t\tif (this.files[name] == null)\n\t\t\tthrow new Error (`Unable to find file ${name}`);\n\n\t\treturn (this.files[name]);\n\t}\n\n\t/** \n\t * Keep the context the object is currently in.\n\t * \n\t * @param func The document element's id.\n\t * @param context The object to remain in context.\n\t * @param [val=undefined] An additional value to pass to the context.\n\t * @return The returned result from the function func.\n\t */\n\tstatic keepContext(func: Function, context: any, val?: any): any\n\t{\n\t\tvar objReturn = function()\n\t\t\t{\n\t\t\t\tvar aryArgs = Array.prototype.slice.call(arguments);\n\n\t\t\t\tif (val != undefined)\n\t\t\t\t\taryArgs.push(val);\n\n\t\t\t\tif (context == null)\n\t\t\t\t\treturn func.apply(this, aryArgs);\n\t\t\t\telse\n\t\t\t\t\treturn func.apply(context, aryArgs);\n\t\t\t};\n\n\t\treturn objReturn;\n\t}\n\n\t/**\n\t * Add and register a component.\n\t */\n\taddComponent (ComponentType: (new  (copy: IHotComponent | HotStaq, api?: HotAPI) => HotComponent), api: HotAPI = null, \n\t\telementOptions: ElementDefinitionOptions = undefined): void\n\t{\n\t\tlet tempApi = this.api\n\t\t\n\t\tif (api != null)\n\t\t\ttempApi = api;\n\n\t\tlet tempComponentObj = new ComponentType (this, tempApi);\n\n\t\tif (this.components[tempComponentObj.tag] != null)\n\t\t\tthrow new Error (`Component ${tempComponentObj.tag} already exists!`);\n\n\t\tthis.components[tempComponentObj.tag] = { componentType: ComponentType, processor: this, api: tempApi };\n\t\tthis.registerComponent (tempComponentObj.tag, elementOptions);\n\t}\n\n\t/**\n\t * Correct any HTML prior to DOM parsing. This only accounts for <tr> currently.\n\t */\n\tprotected static fixHTML (str: string): { fixedStr: string, querySelector: string; }\n\t{\n\t\t// Take into account the difference between XML and HTML.\n\t\tconst tempStr: string = str.replace(/ \\/>/g, '>').replace(\n\t\t\t/(<(area|base|br|col|command|embed|hr|img|input|keygen|link|meta|param|source|track|wbr).*?>)/g, '$1</$2>');\n\t\tconst parsedXML = new DOMParser ().parseFromString (`<xml>${tempStr}</xml>`, \"text/xml\");\n\t\tlet querySelector: string = \"\";\n\n\t\tif (parsedXML.documentElement.children.length > 0)\n\t\t{\n\t\t\tconst tagName: string = parsedXML.documentElement.children[0].tagName.toLowerCase ();\n\n\t\t\tif (tagName === \"tr\")\n\t\t\t{\n\t\t\t\tstr = `<table>${str}</table>`;\n\t\t\t\tquerySelector = \"tbody\";\n\t\t\t}\n\n\t\t\tif (tagName === \"th\")\n\t\t\t{\n\t\t\t\tstr = `<table>${str}</table>`;\n\t\t\t\tquerySelector = tagName;\n\t\t\t}\n\t\t}\n\n\t\treturn ({ fixedStr: str, querySelector: querySelector });\n\t}\n\n\t/**\n\t * Register a component for use as a HTML tag.\n\t */\n\tprotected registerComponent (tag: string, elementOptions: ElementDefinitionOptions = undefined): void\n\t{\n\t\tif ((tag == null) || (tag === \"\"))\n\t\t\tthrow new Error (`All components must have a tag!`);\n\n\t\tif (customElements.get (tag) !== undefined)\n\t\t{\n\t\t\t/// @fixme This element has already been defined. Should this throw an error or warning? I don't think it should...\n\n\t\t\treturn;\n\t\t}\n\n\t\tlet processorComponents = this.components;\n\n\t\tcustomElements.define (tag, class extends HTMLElement\n\t\t\t{\n\t\t\t\t/**\n\t\t\t\t * The connected HotComponent.\n\t\t\t\t */\n\t\t\t\tcomponent: HotComponent;\n\t\t\t\n\t\t\t\tconstructor ()\n\t\t\t\t{\n\t\t\t\t\tsuper ();\n\n\t\t\t\t\tlet componentInfo = processorComponents[tag];\n\t\t\t\t\tthis.component = new componentInfo.componentType (componentInfo.processor, componentInfo.api);\n\t\t\t\t}\n\t\t\t\n\t\t\t\t/**\n\t\t\t\t * This helps parse <tr> and other tags that do not have a parent.\n\t\t\t\t * \n\t\t\t\t * Thanks Brandon McConnell!\n\t\t\t\t * \n\t\t\t\t * From: https://stackoverflow.com/questions/67313479/make-parsefromstring-parse-without-validation\n\t\t\t\t * \n\t\t\t\t * @todo May remove this as it does not seem to work well in a lot of edge cases.\n\t\t\t\t */\n\t\t\t\tprotected looseParseFromString (parser: DOMParser, str: string) {\n\t\t\t\t\tstr = str.replace(/ \\/>/g, '>').replace(/(<(area|base|br|col|command|embed|hr|img|input|keygen|link|meta|param|source|track|wbr).*?>)/g, '$1</$2>');\n\t\t\t\t\tconst xdom = parser.parseFromString('<xml>'+str+'</xml>', 'text/xml');\n\t\t\t\t\tconst hdom = parser.parseFromString('', 'text/html');\n\t\t\t\t\tfor (let elem of Array.from(xdom.documentElement.children)) {\n\t\t\t\t\t\t/// @ts-ignore\n\t\t\t\t\t\thdom.body.appendChild(elem);\n\t\t\t\t\t}\n\t\t\t\t\tfor (let elem of Array.from(hdom.querySelectorAll('area,base,br,col,command,embed,hr,img,input,keygen,link,meta,param,source,track,wbr'))) {\n\t\t\t\t\t\t/// @ts-ignore\n\t\t\t\t\t\telem.outerHTML = '<'+elem.outerHTML.slice(1).split('<')[0];\n\t\t\t\t\t}\n\t\t\t\t\treturn hdom;\n\t\t\t\t}\n\n\t\t\t\tget observedAttributes(): string[] /// @fixme Does this REALLY have to be static? Awful if it does...\n\t\t\t\t{\n\t\t\t\t\treturn (this.component.observedAttributes);\n\t\t\t\t}\n\t\t\t\n\t\t\t\tasync connectedCallback ()\n\t\t\t\t{\n\t\t\t\t\tlet compHtmlElement = this;\n\t\t\t\n\t\t\t\t\t// @ts-ignore\n\t\t\t\t\tcompHtmlElement.hotComponent = this.component;\n\t\t\t\n\t\t\t\t\tthis.component.htmlElements = [compHtmlElement];\n\t\t\t\t\tthis.component.inner = this.innerHTML;\n\t\t\t\n\t\t\t\t\tif (this.component.handleAttributes != null)\n\t\t\t\t\t\tawait this.component.handleAttributes (this.attributes);\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tfor (let iIdx = 0; iIdx < this.attributes.length; iIdx++)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tconst attr: Attr = this.attributes[iIdx];\n\t\t\t\t\t\t\tconst attrName: string = attr.name.toLowerCase ();\n\t\t\t\t\t\t\tconst attrValue: string = attr.value;\n\t\t\t\n\t\t\t\t\t\t\tif (attrName === \"id\")\n\t\t\t\t\t\t\t\tthis.component.name = attrValue;\n\t\t\t\n\t\t\t\t\t\t\tif (attrName === \"name\")\n\t\t\t\t\t\t\t\tthis.component.name = attrValue;\n\t\t\t\n\t\t\t\t\t\t\tif (attrName === \"value\")\n\t\t\t\t\t\t\t\tthis.component.value = attrValue;\n\t\t\t\n\t\t\t\t\t\t\tif (attrName.indexOf (\"hot-\") > -1)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tconst attrTempName: string = attrName.substring (4);\n\t\t\t\n\t\t\t\t\t\t\t\t/// @ts-ignore\n\t\t\t\t\t\t\t\tthis.component[attrTempName] = attrValue;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\tif (this.component.onPreOutput != null)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (await this.component.onPreOutput () === false)\n\t\t\t\t\t\t\treturn;\n\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\tlet outputs = await this.component.output ();\n\t\t\t\n\t\t\t\t\tif (this.component.onPostOutput != null)\n\t\t\t\t\t\toutputs = await this.component.onPostOutput (outputs);\n\t\t\t\n\t\t\t\t\tlet componentOutputs: HotComponentOutput[] = [];\n\t\t\t\n\t\t\t\t\tif (typeof (outputs) === \"string\")\n\t\t\t\t\t\tcomponentOutputs.push ({ html: outputs });\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tif (outputs instanceof Array)\n\t\t\t\t\t\t\tcomponentOutputs = outputs;\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tcomponentOutputs = [outputs];\n\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\tfor (let iKdx = 0; iKdx < componentOutputs.length; iKdx++)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet output = componentOutputs[iKdx];\n\t\t\t\t\t\tlet htmlStr: string = output.html;\n\t\t\t\t\t\tlet addFunctionsTo: string = \"\";\n\t\t\t\n\t\t\t\t\t\tif (output.addFunctionsTo != null)\n\t\t\t\t\t\t\taddFunctionsTo = output.addFunctionsTo;\n\t\t\t\n\t\t\t\t\t\tlet str: string = HotFile.parseContent (htmlStr, true, { \"outputCommands\": false });\n\t\t\t\n\t\t\t\t\t\tif (this.component.onParsed != null)\n\t\t\t\t\t\t\tstr = await this.component.onParsed (str);\n\t\t\t\n\t\t\t\t\t\tlet htmlHandler: { fixedStr: string, querySelector: string; } = { fixedStr: \"\", querySelector: \"\" };\n\t\t\t\n\t\t\t\t\t\tif (this.component.onFixHTML != null)\n\t\t\t\t\t\t\thtmlHandler = await this.component.onFixHTML (str);\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\thtmlHandler = HotStaq.fixHTML (str);\n\t\t\t\n\t\t\t\t\t\tlet newDOM: Document = null;\n\t\t\t\t\t\tlet newObj: HTMLElement = null;\n\t\t\t\n\t\t\t\t\t\tif (this.component.onParseDOM != null)\n\t\t\t\t\t\t\tnewDOM = await this.component.onParseDOM (htmlHandler.fixedStr);\n\t\t\t\t\t\telse\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\t/// @ts-ignore\n\t\t\t\t\t\t\t//newDOM = this.looseParseFromString (new DOMParser (), str);\n\t\t\t\t\t\t\tnewDOM = new DOMParser ().parseFromString (htmlHandler.fixedStr, \"text/html\");\n\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\tif (newDOM.body.children.length < 1)\n\t\t\t\t\t\t\tthrow new Error (`No component output from ${this.component.name}`);\n\t\t\t\n\t\t\t\t\t\tif (newDOM.body.children.length > 1)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet throwErr: boolean = true;\n\t\t\t\n\t\t\t\t\t\t\tfor (let iIdx = 0; iIdx < newDOM.body.children.length; iIdx++)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tlet child = newDOM.body.children[iIdx];\n\t\t\t\n\t\t\t\t\t\t\t\tif (child instanceof HTMLElement)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tif (child.tagName.toLowerCase () === \"parsererror\")\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tnewObj = child;\n\t\t\t\t\t\t\t\t\t\tthrowErr = false;\n\t\t\t\n\t\t\t\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\t\tif (throwErr === true)\n\t\t\t\t\t\t\t\tthrow new Error (`Only a single html element can come from component ${this.component.name}, multiple elements were detected.`);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (htmlHandler.querySelector === \"\")\n\t\t\t\t\t\t\tnewObj = (<HTMLElement>newDOM.body.children[0]);\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tnewObj = newDOM.querySelector (htmlHandler.querySelector);\n\n\t\t\t\t\t\tlet childrenToReadd: Node[] = [];\n\t\t\t\n\t\t\t\t\t\t// Save the children from being replaced.\n\t\t\t\t\t\tfor (let iIdx = (this.children.length - 1); iIdx > -1; iIdx--)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet child: Node = this.children[iIdx];\n\t\t\t\n\t\t\t\t\t\t\tchildrenToReadd.push (this.removeChild (child));\n\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\tthis.replaceWith (newObj);\n\t\t\t\n\t\t\t\t\t\tif (this.component.click != null)\n\t\t\t\t\t\t\tnewObj.onclick = this.component.click.bind (this.component);\n\t\t\t\n\t\t\t\t\t\tfor (let key in this.component.events)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet event = this.component.events[key];\n\t\t\t\n\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\tnewObj.addEventListener (event.type, event.func, event.options);\n\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\tlet objectFunctions: string[] = Object.getOwnPropertyNames (this.component.constructor.prototype);\n\t\t\t\n\t\t\t\t\t\t// Associate any functions to the newly created element.\n\t\t\t\t\t\tfor (let iIdx = 0; iIdx < objectFunctions.length; iIdx++)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet objFunc: string = objectFunctions[iIdx];\n\t\t\t\n\t\t\t\t\t\t\tif (objFunc === \"constructor\")\n\t\t\t\t\t\t\t\tcontinue;\n\t\t\t\n\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\tlet prop = this.component[objFunc];\n\t\t\t\n\t\t\t\t\t\t\tif (typeof (prop) === \"function\")\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tlet isNewFunction: boolean = true;\n\t\t\t\n\t\t\t\t\t\t\t\t// Go through each function in the base HotComponent and see \n\t\t\t\t\t\t\t\t// if there's any matches. If there's a match, that means \n\t\t\t\t\t\t\t\t// we're trying to add an existing function, and we don't\n\t\t\t\t\t\t\t\t// wanna do that. Skip it.\n\t\t\t\t\t\t\t\tfor (let key2 in HotComponent.prototype)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tif (objFunc === key2)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tisNewFunction = false;\n\t\t\t\n\t\t\t\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\t\t\tif (isNewFunction === true)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\tnewObj[objFunc] = HotStaq.keepContext (this.component[objFunc], this.component);\n\t\t\t\n\t\t\t\t\t\t\t\t\tif (addFunctionsTo !== \"\")\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tlet query: HTMLElement = document.querySelector (addFunctionsTo);\n\t\t\t\n\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\tquery[objFunc] = HotStaq.keepContext (this.component[objFunc], this.component);\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\tif (this.component.onPrePlace != null)\n\t\t\t\t\t\t\tnewObj = await this.component.onPrePlace (newObj);\n\t\t\t\n\t\t\t\t\t\tlet compHtmlElement2: HTMLElement = await this.component.onCreated (newObj);\n\t\t\t\n\t\t\t\t\t\tif (this.component.onParentPlace != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\tcompHtmlElement2.onParentPlace = this.component.onParentPlace;\n\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\tcompHtmlElement2.hotComponent = this.component;\n\t\t\t\t\t\tthis.component.htmlElements.push (compHtmlElement2);\n\t\t\t\n\t\t\t\t\t\tif (output.parentSelector != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet parentNode: Node = document.querySelector (output.parentSelector);\n\t\t\t\n\t\t\t\t\t\t\tcompHtmlElement2.parentElement.removeChild (compHtmlElement2);\n\t\t\t\t\t\t\tparentNode.appendChild (compHtmlElement2);\n\t\t\t\n\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\tif (compHtmlElement2.onParentPlace != null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\tawait compHtmlElement2.hotComponent.onParentPlace (parentNode, compHtmlElement2);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\tif (output.placeHereParent != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet parentNodeToCheck = compHtmlElement2.parentNode;\n\t\t\t\t\t\t\tlet parentNodeCheckCounter: number = 0;\n\t\t\t\n\t\t\t\t\t\t\twhile (parentNodeCheckCounter < 10) /// @todo Make this controllable with a variable from the component.\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (parentNodeToCheck == null)\n\t\t\t\t\t\t\t\t\tbreak;\n\t\t\t\n\t\t\t\t\t\t\t\tif (parentNodeToCheck instanceof HTMLHtmlElement)\n\t\t\t\t\t\t\t\t\tbreak;\n\t\t\t\n\t\t\t\t\t\t\t\t// If the hot-place-here exists, place the children there. If not, place it under the \n\t\t\t\t\t\t\t\t// new element.\n\t\t\t\t\t\t\t\tlet placeHereArray = parentNodeToCheck.querySelectorAll (`hot-place-here[name=\"${output.placeHereParent}\"]`);\n\t\t\t\n\t\t\t\t\t\t\t\tif (placeHereArray.length > 0)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tlet placeHere = placeHereArray[0];\n\t\t\t\n\t\t\t\t\t\t\t\t\tcompHtmlElement2.parentNode.removeChild (compHtmlElement2);\n\t\t\t\t\t\t\t\t\tplaceHere.appendChild (compHtmlElement2);\n\t\t\t\n\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\tif (compHtmlElement2.onParentPlace != null)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\tawait compHtmlElement2.hotComponent.onParentPlace (placeHere, compHtmlElement2);\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\t\t\tif (placeHereArray.length < 1)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tlet placeHereAttrArray = parentNodeToCheck.querySelectorAll (`[hot-place-here=\"${output.placeHereParent}\"]`);\n\t\t\t\n\t\t\t\t\t\t\t\t\tif (placeHereAttrArray.length > 0)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tlet placeHere = placeHereAttrArray[0];\n\t\t\t\t\t\t\t\t\t\tcompHtmlElement2.parentNode.removeChild (compHtmlElement2);\n\t\t\t\t\t\t\t\t\t\tplaceHere.appendChild (compHtmlElement2);\n\t\t\t\n\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\tif (compHtmlElement2.onParentPlace != null)\n\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\t\t\tawait compHtmlElement2.hotComponent.onParentPlace (placeHere, compHtmlElement2);\n\t\t\t\t\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\t\t\tparentNodeToCheck = parentNodeToCheck.parentNode;\n\t\t\t\t\t\t\t\tparentNodeCheckCounter++;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\t// Append the children to the newly created HTML element.\n\t\t\t\t\t\tfor (let iIdx = 0; iIdx < childrenToReadd.length; iIdx++)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tconst child: Node = childrenToReadd[iIdx];\n\t\t\t\n\t\t\t\t\t\t\tcompHtmlElement2.appendChild (child);\n\t\t\t\n\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\tif (child.onParentPlace != null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\tawait child.hotComponent.onParentPlace (compHtmlElement2, child);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\n\t\t\t\t\t\tif (this.component.onPostPlace != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\t/// @ts-ignore\n\t\t\t\t\t\t\tcompHtmlElement2 = await this.component.onPostPlace (compHtmlElement2.parentNode, compHtmlElement2);\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}, elementOptions);\n\t}\n\n\t/**\n\t * Add a new HTML element(s) to the current document.\n\t */\n\tstatic addHtml (parent: string | HTMLElement, html: string | HTMLElement): HTMLElement | HTMLElement[]\n\t{\n\t\tlet foundParent: HTMLElement = null;\n\n\t\tif (typeof (parent) === \"string\")\n\t\t\tfoundParent = document.querySelector (parent);\n\t\telse\n\t\t\tfoundParent = parent;\n\n\t\tif (foundParent == null)\n\t\t\tthrow new Error (`Unable to find parent ${parent}!`);\n\n\t\tlet result: HTMLElement = null;\n\n\t\tif (typeof (html) === \"string\")\n\t\t{\n\t\t\tlet htmlHandler = HotStaq.fixHTML (html);\n\t\t\tlet newDOM: Document = new DOMParser ().parseFromString (htmlHandler.fixedStr, \"text/html\");\n\t\t\tlet children: any = null;\n\n\t\t\tif (htmlHandler.querySelector === \"\")\n\t\t\t\tchildren = newDOM.body.children;\n\t\t\telse\n\t\t\t\tchildren = newDOM.querySelector (htmlHandler.querySelector).children;\n\n\t\t\tlet results: HTMLElement[] = [];\n\n\t\t\tfor (let iIdx = 0; iIdx < children.length; iIdx++)\n\t\t\t{\n\t\t\t\tlet child: HTMLElement = (<HTMLElement>children[iIdx]);\n\n\t\t\t\tresults.push (foundParent.appendChild (child));\n\t\t\t}\n\n\t\t\treturn (results);\n\t\t}\n\t\telse\n\t\t\tresult = foundParent.appendChild (html);\n\n\t\treturn (result);\n\t}\n\n\t/**\n\t * Check if a HotSite's name is valid.\n\t */\n\tstatic checkHotSiteName (hotsiteName: string, throwException: boolean = false): boolean\n\t{\n\t\tlet throwTheException = () =>\n\t\t\t{\n\t\t\t\tif (throwException === true)\n\t\t\t\t\tthrow new Error (`HotSite ${hotsiteName} has an invalid name! The name cannot be empty and must have a valid NPM module name.`);\n\t\t\t};\n\n\t\tlet results = validateModuleName (hotsiteName);\n\n\t\tif (results.errors != null)\n\t\t{\n\t\t\tif (results.errors.length > 0)\n\t\t\t\tthrowTheException ();\n\t\t}\n\n\t\treturn (true);\n\t}\n\n\t/**\n\t * In the supplied content, replace a key in a ${KEY} with a value.\n\t * \n\t * @returns The content with the correct values.\n\t */\n\tstatic replaceKey (content: string, key: string, value: string): string\n\t{\n\t\tconst finalStr: string = content.replace (new RegExp (`\\\\$\\\\{${key}\\\\}`, \"g\"), value);\n\n\t\treturn (finalStr);\n\t}\n\n\t/**\n\t * Get a value from a HotSite object.\n\t * \n\t * @returns Returns the value from the hotsite object. Returns null if it doesn't exist.\n\t */\n\tstatic getValueFromHotSiteObj (hotsite: HotSite, params: string[]): any\n\t{\n\t\tlet value: any = null;\n\n\t\tif (hotsite != null)\n\t\t{\n\t\t\tlet prevValue: any = hotsite;\n\n\t\t\t// Go through each object in the list of parameters and \n\t\t\t// get the value of the final parameter.\n\t\t\tfor (let iIdx = 0; iIdx < params.length; iIdx++)\n\t\t\t{\n\t\t\t\tlet param: string = params[iIdx];\n\n\t\t\t\tif (prevValue[param] == null)\n\t\t\t\t{\n\t\t\t\t\tprevValue = null;\n\n\t\t\t\t\tbreak;\n\t\t\t\t}\n\n\t\t\t\tprevValue = prevValue[param];\n\t\t\t}\n\n\t\t\tif (prevValue != null)\n\t\t\t\tvalue = prevValue;\n\t\t}\n\n\t\treturn (value);\n\t}\n\n\t/**\n\t * Process a HotSite.\n\t */\n\tasync processHotSite (): Promise<void>\n\t{\n\t\tHotStaq.checkHotSiteName (this.hotSite.name, true);\n\n\t\tlet routes = this.hotSite.routes;\n\t\tlet testerUrl: string = \"http://127.0.0.1:8182\";\n\t\tlet tester: HotTester = null;\n\t\tlet driver: HotTestDriver = null;\n\n\t\tif (HotStaq.isWeb === false)\n\t\t{\n\t\t\tif (this.mode === DeveloperMode.Development)\n\t\t\t{\n\t\t\t\tif (this.hotSite.testing != null)\n\t\t\t\t{\n\t\t\t\t\tlet setupTester = (parentObj: any) => \n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet createNewTester: boolean = true;\n\t\t\n\t\t\t\t\t\t\tif (parentObj.createNewTester != null)\n\t\t\t\t\t\t\t\tcreateNewTester = parentObj.createNewTester;\n\t\t\n\t\t\t\t\t\t\tlet testerName: string = \"Tester\";\n\t\t\n\t\t\t\t\t\t\tif (parentObj.tester != null)\n\t\t\t\t\t\t\t\ttesterName = parentObj.tester;\n\t\t\n\t\t\t\t\t\t\tif (parentObj.testerName != null)\n\t\t\t\t\t\t\t\ttesterName = parentObj.testerName;\n\t\t\n\t\t\t\t\t\t\tif (createNewTester === true)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t/// @fixme Find a way to securely allow devs to use their own drivers and testers...\n\t\t\t\t\t\t\t\t/// @fixme Hack for dealing with WebPack's bs.\n\t\t\t\t\t\t\t\tHotTesterMocha = require (\"./HotTesterMocha\").HotTesterMocha;\n\t\t\t\t\t\t\t\tHotTesterMochaSelenium = require (\"./HotTesterMochaSelenium\").HotTesterMochaSelenium;\n\t\t\t\t\t\t\t\tHotTestSeleniumDriver = require (\"./HotTestSeleniumDriver\").HotTestSeleniumDriver;\n\t\t\n\t\t\t\t\t\t\t\tif (parentObj.testerAPIUrl === \"\")\n\t\t\t\t\t\t\t\t\ttesterUrl = parentObj.testerAPIUrl;\n\t\t\n\t\t\t\t\t\t\t\tif (parentObj.driver === \"HotTestSeleniumDriver\")\n\t\t\t\t\t\t\t\t\tdriver = new HotTestSeleniumDriver ();\n\t\t\n\t\t\t\t\t\t\t\tif (parentObj.tester === \"HotTesterMocha\")\n\t\t\t\t\t\t\t\t\ttester = new HotTesterMocha (this, testerName, testerUrl, driver);\n\t\t\n\t\t\t\t\t\t\t\tif (parentObj.tester === \"HotTesterMochaSelenium\")\n\t\t\t\t\t\t\t\t\ttester = new HotTesterMochaSelenium (this, testerName, testerUrl);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\ttester = this.testers[testerName];\n\n\t\t\t\t\t\t\tif (tester.driver == null)\n\t\t\t\t\t\t\t\tthrow new Error (`Tester ${testerName} does not have a driver set!`);\n\n\t\t\t\t\t\t\tif (parentObj.commandDelay != null)\n\t\t\t\t\t\t\t\ttester.driver.commandDelay = parentObj.commandDelay;\n\t\t\t\t\t\t};\n\n\t\t\t\t\tif (this.hotSite.testing.web != null)\n\t\t\t\t\t\tsetupTester (this.hotSite.testing.web);\n\n\t\t\t\t\tif (this.hotSite.testing.api != null)\n\t\t\t\t\t\tsetupTester (this.hotSite.testing.api);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tif (routes != null)\n\t\t{\n\t\t\tfor (let key in routes)\n\t\t\t{\n\t\t\t\tlet route: HotSiteRoute = routes[key];\n\t\t\t\tlet file: HotFile = new HotFile (route);\n\t\t\t\tlet page: HotPage = new HotPage ({\n\t\t\t\t\t\tprocessor: this,\n\t\t\t\t\t\tname: route.name || \"\",\n\t\t\t\t\t\troute: key,\n\t\t\t\t\t\tfiles: [file]\n\t\t\t\t\t});\n\n\t\t\t\tif (tester != null)\n\t\t\t\t{\n\t\t\t\t\tif (this.mode === DeveloperMode.Development)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet mapName: string = route.name;\n\t\t\t\t\t\tlet testMap: HotTestMap = null;\n\n\t\t\t\t\t\tif (route.map != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (typeof (route.map) === \"string\")\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (tester.testMaps[route.map] == null)\n\t\t\t\t\t\t\t\t\tthrow new Error (`Test map ${route.map} does not exist!`);\n\n\t\t\t\t\t\t\t\ttester.testMaps[mapName] = tester.testMaps[route.map];\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\ttestMap = new HotTestMap ();\n\t\t\t\t\t\t\t\tlet destinations: HotTestDestination[] | { [name: string]: HotTestDestination } = null;\n\n\t\t\t\t\t\t\t\tif (route.map instanceof Array)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tdestinations = [];\n\n\t\t\t\t\t\t\t\t\tfor (let iIdx = 0; iIdx < route.map.length; iIdx++)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tlet dest = route.map[iIdx];\n\n\t\t\t\t\t\t\t\t\t\tdestinations.push (new HotTestDestination (dest));\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tdestinations = {};\n\n\t\t\t\t\t\t\t\t\tfor (let key2 in route.map)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tlet dest = route.map[key2];\n\n\t\t\t\t\t\t\t\t\t\tdestinations[key2] = new HotTestDestination (dest);\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\ttestMap.destinations = destinations;\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\ttester.testMaps[mapName] = testMap;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (route.destinationOrder != null)\n\t\t\t\t\t\t\ttester.testMaps[mapName].destinationOrder = route.destinationOrder;\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tthis.addPage (page);\n\t\t\t}\n\t\t}\n\n\t\tif (this.hotSite.apis != null)\n\t\t{\n\t\t\tfor (let key in this.hotSite.apis)\n\t\t\t{\n\t\t\t\tlet api = this.hotSite.apis[key];\n\n\t\t\t\tif (api.map == null)\n\t\t\t\t\tcontinue;\n\n\t\t\t\tif (HotStaq.isWeb === false)\n\t\t\t\t{\n\t\t\t\t\tif (this.mode === DeveloperMode.Development)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet mapName: string = key;\n\t\t\t\t\t\tlet testMap: HotTestMap = new HotTestMap ();\n\n\t\t\t\t\t\ttestMap.destinations = [];\n\n\t\t\t\t\t\tfor (let iIdx = 0; iIdx < api.map.length; iIdx++)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet map: string = api.map[iIdx];\n\n\t\t\t\t\t\t\ttestMap.destinations.push (new HotTestDestination (map));\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (tester == null)\n\t\t\t\t\t\t\tthrow new Error (`A tester was not created first! You must specify one in the CLI or in HotSite.json.`);\n\n\t\t\t\t\t\ttester.testMaps[mapName] = testMap;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\t/// @fixme Allow this to work for server-side as well...\n\t\tif (HotStaq.isWeb === true)\n\t\t{\n\t\t\tfor (let key in this.hotSite.components)\n\t\t\t{\n\t\t\t\tlet component = this.hotSite.components[key];\n\t\t\t\tlet componentUrl: string = component.url;\n\n\t\t\t\t/// @fixme Create unit test for fetching, loading, and registering.\n\t\t\t\tlet res: any = await fetch (componentUrl);\n\t\t\t\tlet ComponentClass = eval (res);\n\n\t\t\t\tthis.addComponent (ComponentClass);\n\t\t\t}\n\t\t}\n\n\t\tif (this.hotSite.routes == null)\n\t\t\tthis.hotSite.routes = {};\n\n\t\tlet disableFileLoading: boolean = false;\n\n\t\tif (this.hotSite.disableFileLoading != null)\n\t\t\tdisableFileLoading = this.hotSite.disableFileLoading;\n\n\t\tif (disableFileLoading === false)\n\t\t\tawait this.loadHotFiles (this.hotSite.files);\n\t\telse\n\t\t\tthis.logger.verbose (`Hotsite has file loading disabled...`);\n\n\t\tif (tester != null)\n\t\t\tthis.addTester (tester);\n\t}\n\n\t/**\n\t * Load from a HotSite.json file. Be sure to load and attach any testers before \n\t * loading a HotSite.\n\t */\n\tasync loadHotSite (path: string): Promise<void>\n\t{\n\t\tlet jsonStr: string = \"\";\n\n\t\tif (HotStaq.isWeb === true)\n\t\t{\n\t\t\tthis.logger.info (`Downloading HotSite ${path}`);\n\n\t\t\tlet res: any = await fetch (path);\n\n\t\t\tthis.logger.info (`Downloaded site ${path}`);\n\n\t\t\tjsonStr = res.text ();\n\t\t}\n\t\telse\n\t\t{\n\t\t\tpath = ppath.normalize (path);\n\n\t\t\tthis.logger.info (`Accessing HotSite ${path}`);\n\n\t\t\tjsonStr = await new Promise (\n\t\t\t\t(resolve: any, reject: any): void =>\n\t\t\t\t{\n\t\t\t\t\tfs.readFile (path, (err: NodeJS.ErrnoException, data: Buffer): void =>\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (err != null)\n\t\t\t\t\t\t\t\tthrow err;\n\t\n\t\t\t\t\t\t\tlet content: string = data.toString ();\n\n\t\t\t\t\t\t\tthis.logger.info (`Accessed site ${path}`);\n\t\n\t\t\t\t\t\t\tresolve (content);\n\t\t\t\t\t\t});\n\t\t\t\t});\n\t\t}\n\n\t\tthis.hotSite = JSON.parse (jsonStr);\n\t\tthis.hotSite.hotsitePath = path;\n\t}\n\n\t/**\n\t * Load an array of files. If a file already has content, it will not be reloaded \n\t * unless forceContentLoading is set to true.\n\t */\n\tasync loadHotFiles (files: { [name: string]: { url?: string; localFile?: string; content?: string; } }, \n\t\t\tforceContentLoading: boolean = false): Promise<void>\n\t{\n\t\tthis.logger.verbose (`Loading Hott files...`);\n\n\t\tfor (let key in files)\n\t\t{\n\t\t\tlet file = files[key];\n\t\t\tlet newFile: HotFile = null;\n\n\t\t\tif (HotStaq.isWeb === true)\n\t\t\t{\n\t\t\t\tnewFile = new HotFile ({\n\t\t\t\t\t\t\"name\": key\n\t\t\t\t\t});\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\tnewFile = new HotFile ({\n\t\t\t\t\t\t\"name\": key\n\t\t\t\t\t});\n\t\t\t}\n\n\t\t\tif (file.url != null)\n\t\t\t\tnewFile.url = file.url;\n\n\t\t\tif (HotStaq.isWeb === false)\n\t\t\t{\n\t\t\t\tif (file.localFile != null)\n\t\t\t\t\tnewFile.localFile = file.localFile;\n\t\t\t}\n\n\t\t\tlet loadContent: boolean = true;\n\n\t\t\tif (file.content != null)\n\t\t\t{\n\t\t\t\tnewFile.content = file.content;\n\t\t\t\tloadContent = false;\n\t\t\t}\n\n\t\t\tif (forceContentLoading === true)\n\t\t\t\tloadContent = true;\n\n\t\t\tif (loadContent === true)\n\t\t\t{\n\t\t\t\tlet filepath: string = \"\";\n\n\t\t\t\tif (newFile.url !== \"\")\n\t\t\t\t\tfilepath = newFile.url;\n\n\t\t\t\tif (newFile.localFile !== \"\")\n\t\t\t\t\tfilepath = newFile.localFile;\n\n\t\t\t\tthis.logger.verbose (`Loading Hott file: ${filepath}`);\n\t\t\t\tawait newFile.load ();\n\t\t\t\tthis.logger.verbose (`Finished loading Hott file: ${filepath}`);\n\t\t\t}\n\n\t\t\tthis.addFile (newFile);\n\t\t}\n\n\t\tthis.logger.verbose (`Finished loading Hott files...`);\n\t}\n\n\t/**\n\t * Generate the content to send to a client.\n\t */\n\tgenerateContent (routeKey: string, name: string = \"\", url: string = \"./\",\n\t\t\tjsSrcPath: string = \"./js/HotStaq.min.js\", passArgs: boolean = true, \n\t\t\targs: any = null): string\n\t{\n\t\tlet apiScripts: string = \"\";\n\t\tlet apiCode: string = \"\";\n\t\tlet publicKeys: string = \"\";\n\n\t\t/// @todo Optimize this function as much as possible.\n\n\t\t// Load the API string.\n\t\tif (this.hotSite != null)\n\t\t{\n\t\t\tif (this.hotSite.server.globalApi != null)\n\t\t\t{\n\t\t\t\tif (this.hotSite.server.globalApi !== \"\")\n\t\t\t\t{\n\t\t\t\t\tconst globalApi = this.hotSite.apis[this.hotSite.server.globalApi];\n\n\t\t\t\t\tif (globalApi == null)\n\t\t\t\t\t\tthis.logger.warning (`API with name ${this.hotSite.server.globalApi} doesn't exist!`);\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tlet sendJSContent: boolean = true;\n\n\t\t\t\t\t\tif (globalApi.jsapi == null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tsendJSContent = false;\n\t\t\t\t\t\t\tthis.logger.warning (`API with name ${this.hotSite.server.globalApi} doesn't have a jsapi set. Will not send js content to client.`);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (globalApi.libraryName == null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tsendJSContent = false;\n\t\t\t\t\t\t\tthis.logger.warning (`API with name ${this.hotSite.server.globalApi} doesn't have a libraryName set. Will not send js content to client.`);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (globalApi.apiName == null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tsendJSContent = false;\n\t\t\t\t\t\t\tthis.logger.warning (`API with name ${this.hotSite.server.globalApi} doesn't have a apiName set. Will not send js content to client.`);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (sendJSContent === true)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tapiScripts += `\\t<script type = \"text/javascript\" src = \"${globalApi.jsapi}\"></script>\\n`;\n\n\t\t\t\t\t\t\tlet baseUrl: string = \"\\\"\\\"\";\n\n\t\t\t\t\t\t\tif (globalApi.url != null)\n\t\t\t\t\t\t\t\tbaseUrl = `\\\"${globalApi.url}\\\"`;\n\n\t\t\t\t\t\t\tif (this.api != null)\n\t\t\t\t\t\t\t\tbaseUrl = `\\\"${this.api.baseUrl}\\\"`;\n\n\t\t\t\t\t\t\tlet tempAPIContent: string = this.apiContent;\n\t\t\t\t\t\t\ttempAPIContent = tempAPIContent.replace (/\\%api\\_name\\%/g, globalApi.apiName);\n\t\t\t\t\t\t\ttempAPIContent = tempAPIContent.replace (/\\%api\\_exported\\_name\\%/g, globalApi.libraryName);\n\t\t\t\t\t\t\ttempAPIContent = tempAPIContent.replace (/\\%base\\_url\\%/g, baseUrl);\n\n\t\t\t\t\t\t\tapiCode += tempAPIContent;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif (this.hotSite.apis != null)\n\t\t\t{\n\t\t\t\tlet route = this.hotSite.routes[routeKey];\n\n\t\t\t\tif (route != null)\n\t\t\t\t{\n\t\t\t\t\tif (route.api != null)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet api = this.hotSite.apis[route.api];\n\n\t\t\t\t\t\tif (api == null)\n\t\t\t\t\t\t\tthrow new Error (`Unable to find API ${route.api}`);\n\n\t\t\t\t\t\tlet sendJSContent: boolean = true;\n\n\t\t\t\t\t\tif (api.jsapi == null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tsendJSContent = false;\n\t\t\t\t\t\t\tthis.logger.warning (`API with name ${route.api} doesn't have a jsapi set. Will not send js content to client.`);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (api.libraryName == null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tsendJSContent = false;\n\t\t\t\t\t\t\tthis.logger.warning (`API with name ${route.api} doesn't have a libraryName set. Will not send js content to client.`);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (api.apiName == null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tsendJSContent = false;\n\t\t\t\t\t\t\tthis.logger.warning (`API with name ${route.api} doesn't have a apiName set. Will not send js content to client.`);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (sendJSContent === true)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet jsapipath = api.jsapi;\n\t\t\t\t\t\t\tapiScripts += `\\t<script type = \"text/javascript\" src = \"${jsapipath}\"></script>\\n`;\n\n\t\t\t\t\t\t\tlet baseUrl: string = \"\\\"\\\"\";\n\n\t\t\t\t\t\t\tif (api.url != null)\n\t\t\t\t\t\t\t\tbaseUrl = `\\\"${api.url}\\\"`;\n\n\t\t\t\t\t\t\tif (this.api != null)\n\t\t\t\t\t\t\t\tbaseUrl = `\\\"${this.api.baseUrl}\\\"`;\n\n\t\t\t\t\t\t\tlet tempAPIContent: string = this.apiContent;\n\t\t\t\t\t\t\ttempAPIContent = tempAPIContent.replace (/\\%api\\_name\\%/g, api.apiName);\n\t\t\t\t\t\t\ttempAPIContent = tempAPIContent.replace (/\\%api\\_exported\\_name\\%/g, api.libraryName);\n\t\t\t\t\t\t\ttempAPIContent = tempAPIContent.replace (/\\%base\\_url\\%/g, baseUrl);\n\n\t\t\t\t\t\t\tapiCode += tempAPIContent;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tif (this.hotSite.server != null)\n\t\t\t{\n\t\t\t\tif (this.hotSite.server.jsSrcPath != null)\n\t\t\t\t\tjsSrcPath = this.hotSite.server.jsSrcPath;\n\t\t\t}\n\n\t\t\tif (this.hotSite.publicKeys != null)\n\t\t\t{\n\t\t\t\tfor (let key in this.hotSite.publicKeys)\n\t\t\t\t{\n\t\t\t\t\tlet secret = this.hotSite.publicKeys[key];\n\t\t\t\t\tlet value: string = undefined;\n\n\t\t\t\t\tif (typeof (secret) === \"string\")\n\t\t\t\t\t\tvalue = JSON.stringify (secret);\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tif (HotStaq.isWeb === false)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (this.api != null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (this.api.connection == null)\n\t\t\t\t\t\t\t\t\tthrow new Error (`Cannot pass secrets from the API if there's no connection!`);\n\n\t\t\t\t\t\t\t\tlet serverConn: HotServer = (<HotServer>this.api.connection);\n\n\t\t\t\t\t\t\t\tif (secret.passSecretFromAPI != null)\n\t\t\t\t\t\t\t\t\tvalue = JSON.stringify (serverConn.secrets[key]);\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tif (secret.env != null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t/// @fixme @secvul Is this a security vulnerability? Need to verify that \n\t\t\t\t\t\t\t\t/// only the server has access to this. At this point, I think only the \n\t\t\t\t\t\t\t\t/// server has access.\n\t\t\t\t\t\t\t\tconst envKey: string = secret.env;\n\n\t\t\t\t\t\t\t\tvalue = JSON.stringify (process.env[envKey]);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tpublicKeys += `processor.publicKeys[\"${key}\"] = ${value};\\n`;\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tlet content: string = this.pageContent;\n\t\tlet fixContent = (tempContent: string) =>\n\t\t\t{\n\t\t\t\tlet developerModeStr: string = \"\";\n\t\t\t\tlet testerAPIStr: string = \"\";\n\n\t\t\t\tif (this.mode === DeveloperMode.Development)\n\t\t\t\t{\n\t\t\t\t\tdeveloperModeStr = `tempMode = HotStaqWeb.DeveloperMode.Development;`;\n\n\t\t\t\t\tif (this.hotSite != null)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (this.hotSite.testing != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (this.hotSite.testing.web != null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\ttesterAPIStr = this.testerApiContent;\n\n\t\t\t\t\t\t\t\tif (this.hotSite.testing.web.testerAPIUrl == null)\n\t\t\t\t\t\t\t\t\tthis.hotSite.testing.web.testerAPIUrl = \"http://127.0.0.1:8182\";\n\n\t\t\t\t\t\t\t\ttesterAPIStr = testerAPIStr.replace (/\\%base\\_tester\\_url\\%/g, `\\\"${this.hotSite.testing.web.testerAPIUrl}\\\"`);\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\tlet loadFiles: string = \"\";\n\n\t\t\t\tif (Object.keys (this.files).length > 0)\n\t\t\t\t{\n\t\t\t\t\tloadFiles += `var files = {};\\n\\n`;\n\n\t\t\t\t\tfor (let key in this.files)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet file = this.files[key];\n\t\t\t\t\t\tlet fileUrl: string = `\"${file.url}\"`;\n\t\t\t\t\t\tlet fileContent: string = \"\";\n\n\t\t\t\t\t\tif (file.content !== \"\")\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet escapedContent: string = JSON.stringify (file.content);\n\n\t\t\t\t\t\t\t// Find any script tags and interrupt them so the HTML parsers \n\t\t\t\t\t\t\t// don't get confused.\n\t\t\t\t\t\t\tescapedContent = escapedContent.replace (new RegExp (\"\\\\<script\", \"gmi\"), \"<scr\\\" + \\\"ipt\");\n\t\t\t\t\t\t\tescapedContent = escapedContent.replace (new RegExp (\"\\\\<\\\\/script\", \"gmi\"), \"</scr\\\" + \\\"ipt\");\n\n\t\t\t\t\t\t\tfileContent = `, \"content\": ${escapedContent}`;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tloadFiles += `\\t\\t\\tfiles[\"${key}\"] = { \"url\": ${fileUrl}${fileContent} };\\n`;\n\t\t\t\t\t}\n\n\t\t\t\t\tloadFiles += `\\t\\t\\tpromises.push (processor.loadHotFiles (files));\\n`;\n\t\t\t\t}\n\n\t\t\t\ttempContent = tempContent.replace (/\\%title\\%/g, name);\n\n\t\t\t\tif (passArgs === true)\n\t\t\t\t\ttempContent = tempContent.replace (/\\%args\\%/g, \"Hot.Arguments\");\n\n\t\t\t\tif (args != null)\n\t\t\t\t\ttempContent = tempContent.replace (/\\%args\\%/g, JSON.stringify (args));\n\n\t\t\t\tlet testerMap: string = routeKey;\n\t\t\t\tlet testerUrl: string = \"\";\n\t\t\t\tlet testerLaunchpadUrl: string = \"\";\n\t\t\t\tlet testerName: string = \"Tester\";\n\n\t\t\t\tif (this.hotSite != null)\n\t\t\t\t{\n\t\t\t\t\tif (this.hotSite.testing != null)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (this.hotSite.testing.web != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (this.hotSite.testing.web.tester != null)\n\t\t\t\t\t\t\t\ttesterName = this.hotSite.testing.web.tester;\n\n\t\t\t\t\t\t\tif (this.hotSite.testing.web.testerName != null)\n\t\t\t\t\t\t\t\ttesterName = this.hotSite.testing.web.testerName;\n\n\t\t\t\t\t\t\tif (this.hotSite.testing.web.testerAPIUrl != null)\n\t\t\t\t\t\t\t\ttesterUrl = this.hotSite.testing.web.testerAPIUrl;\n\n\t\t\t\t\t\t\tif (this.hotSite.testing.web.launchpadUrl != null)\n\t\t\t\t\t\t\t\ttesterLaunchpadUrl = this.hotSite.testing.web.launchpadUrl;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tif (this.hotSite.routes != null)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (this.hotSite.routes[routeKey] != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tlet route = this.hotSite.routes[routeKey];\n\t\t\t\t\t\t\ttesterMap = route.name;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\t\t\t\t}\n\n\t\t\t\ttempContent = tempContent.replace (/\\%hotstaq\\_js\\_src\\%/g, jsSrcPath);\n\t\t\t\ttempContent = tempContent.replace (/\\%developer\\_mode\\%/g, developerModeStr);\n\t\t\t\ttempContent = tempContent.replace (/\\%tester\\_api\\%/g, testerAPIStr);\n\t\t\t\ttempContent = tempContent.replace (/\\%apis\\_to\\_load\\%/g, apiScripts);\n\t\t\t\ttempContent = tempContent.replace (/\\%load\\_hot\\_site\\%/g, \"\"); /// @fixme Should this only be done server-side?\n\t\t\t\ttempContent = tempContent.replace (/\\%load\\_files\\%/g, loadFiles);\n\t\t\t\ttempContent = tempContent.replace (/\\%api\\_code\\%/g, apiCode);\n\t\t\t\ttempContent = tempContent.replace (/\\%public\\_secrets\\%/g, publicKeys);\n\t\t\t\ttempContent = tempContent.replace (/\\%url\\%/g, url);\n\t\t\t\ttempContent = tempContent.replace (/\\%tester\\_name\\%/g, `\"${testerName}\"`);\n\t\t\t\ttempContent = tempContent.replace (/\\%tester\\_map\\%/g, `\"${testerMap}\"`);\n\t\t\t\ttempContent = tempContent.replace (/\\%tester\\_api\\_base\\_url\\%/g, `\"${testerUrl}\"`);\n\t\t\t\ttempContent = tempContent.replace (/\\%tester\\_launchpad\\_url\\%/g, `\"${testerLaunchpadUrl}\"`);\n\n\t\t\t\treturn (tempContent);\n\t\t\t};\n\t\tcontent = fixContent (content);\n\n\t\treturn (content);\n\t}\n\n\t/**\n\t * Create the Express routes from the given pages. Be sure to load the \n\t * pages first before doing this. This method is meant to be used for \n\t * customized Express applications. If you wish to use the loaded routes \n\t * from this HotStaq object with HotHTTPServer, be sure to use \n\t * the loadHotSite method in HotHTTPServer.\n\t */\n\tcreateExpressRoutes (expressApp: any, jsSrcPath: string = \"./js/HotStaq.min.js\"): void\n\t{\n\t\tfor (let key in this.pages)\n\t\t{\n\t\t\tlet page: HotPage = this.pages[key];\n\t\t\tconst content: string = this.generateContent (page.route, page.name, page.files[0].url, jsSrcPath);\n\n\t\t\texpressApp.get (page.route, (req: any, res: any) =>\n\t\t\t\t{\n\t\t\t\t\tres.send (content);\n\t\t\t\t});\n\t\t}\n\t}\n\n\t/**\n\t * Add a tester for use later.\n\t */\n\taddTester (tester: HotTester): void\n\t{\n\t\tthis.testers[tester.name] = tester;\n\t}\n\n\t/**\n\t * Get the list of maps for testing from the HotSite.\n\t */\n\tgetWebTestingMaps (): string[]\n\t{\n\t\tif (this.hotSite == null)\n\t\t\tthrow new Error (\"No HotSite was loaded!\");\n\n\t\tif (this.hotSite.testing == null)\n\t\t\tthrow new Error (\"The HotSite does not have a testing object!\");\n\n\t\tif (this.hotSite.testing.web == null)\n\t\t\tthrow new Error (\"The HotSite does not have a testing web object!\");\n\n\t\tif (this.hotSite.testing.web.maps == null)\n\t\t\tthrow new Error (\"The HotSite testing object does not have any maps!\");\n\n\t\treturn (this.hotSite.testing.web.maps);\n\t}\n\n\t/**\n\t * Get the list of maps for testing from the HotSite.\n\t */\n\tgetAPITestingMaps (): string[]\n\t{\n\t\tif (this.hotSite == null)\n\t\t\tthrow new Error (\"No HotSite was loaded!\");\n\n\t\tif (this.hotSite.testing == null)\n\t\t\tthrow new Error (\"The HotSite does not have a testing object!\");\n\n\t\tif (this.hotSite.testing.api == null)\n\t\t\tthrow new Error (\"The HotSite does not have a testing api object!\");\n\n\t\tif (this.hotSite.testing.api.maps == null)\n\t\t\tthrow new Error (\"The HotSite testing object does not have any maps!\");\n\n\t\treturn (this.hotSite.testing.api.maps);\n\t}\n\n\t/**\n\t * Get a route's key from a route's name.\n\t */\n\tgetRouteKeyFromName (name: string): string\n\t{\n\t\tlet foundKey: string = \"\";\n\n\t\tif (this.hotSite != null)\n\t\t{\n\t\t\tif (this.hotSite.routes != null)\n\t\t\t{\n\t\t\t\tfor (let key in this.hotSite.routes)\n\t\t\t\t{\n\t\t\t\t\tlet route: HotSiteRoute = this.hotSite.routes[key];\n\n\t\t\t\t\tif (route.name === name)\n\t\t\t\t\t{\n\t\t\t\t\t\tfoundKey = key;\n\n\t\t\t\t\t\tbreak;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\treturn (foundKey);\n\t}\n\n\t/**\n\t * Get a route from a route's name.\n\t */\n\tgetRouteFromName (name: string): HotSiteRoute\n\t{\n\t\tlet foundRoute: HotSiteRoute = null;\n\t\tlet foundKey: string = this.getRouteKeyFromName (name);\n\n\t\tif (foundKey !== \"\")\n\t\t\tfoundRoute = this.hotSite.routes[foundKey];\n\n\t\treturn (foundRoute);\n\t}\n\n\t/**\n\t * Execute tests.\n\t * \n\t * @param testerName The tester to use to execute tests.\n\t * @param mapName The map or maps to use to navigate through tests.\n\t */\n\tasync executeTests (testerName: string, mapName: string): Promise<void>\n\t{\n\t\tlet tester: HotTester = this.testers[testerName];\n\n\t\tif (tester == null)\n\t\t\tthrow new Error (`Unable to execute tests. Tester ${testerName} does not exist!`);\n\n\t\treturn (tester.execute (mapName));\n\t}\n\n\t/**\n\t * Execute all web tests from the HotSite testing web object.\n\t * \n\t * @param testerName The tester to use to execute tests.\n\t */\n\tasync executeAllWebTests (testerName: string): Promise<void>\n\t{\n\t\tlet maps: string[] = this.getWebTestingMaps ();\n\t\tlet tester: HotTester = this.testers[testerName];\n\n\t\tif (tester == null)\n\t\t\tthrow new Error (`Unable to execute tests. Tester ${testerName} does not exist!`);\n\n\t\tfor (let iIdx = 0; iIdx < maps.length; iIdx++)\n\t\t{\n\t\t\tlet mapName: string = maps[iIdx];\n\n\t\t\tawait this.executeTests (testerName, mapName);\n\t\t}\n\t}\n\n\t/**\n\t * Execute all api tests from the HotSite testing api object.\n\t * \n\t * @param testerName The tester to use to execute tests.\n\t */\n\tasync executeAllAPITests (testerName: string): Promise<void>\n\t{\n\t\tlet maps: string[] = this.getAPITestingMaps ();\n\t\tlet tester: HotTester = this.testers[testerName];\n\n\t\tif (tester == null)\n\t\t\tthrow new Error (`Unable to execute tests. Tester ${testerName} does not exist!`);\n\n\t\tfor (let iIdx = 0; iIdx < maps.length; iIdx++)\n\t\t{\n\t\t\tlet mapName: string = maps[iIdx];\n\n\t\t\tawait this.executeTests (testerName, mapName);\n\t\t}\n\t}\n\n\t/**\n\t * Process a page and get the result.\n\t */\n\tasync process (pageName: string, args: any = null): Promise<string>\n\t{\n\t\tlet page: HotPage = this.getPage (pageName);\n\t\tlet result: string = await page.process (args);\n\n\t\treturn (result);\n\t}\n\n\t/**\n\t * Process a local file and get the result.\n\t */\n\tstatic async processLocalFile (localFilepath: string, name: string = localFilepath, args: any = null): Promise<string>\n\t{\n\t\tlet processor: HotStaq = new HotStaq ();\n\t\tlet file: HotFile = new HotFile ({\n\t\t\t\"localFile\": localFilepath\n\t\t});\n\t\tawait file.load ();\n\t\tlet page: HotPage = new HotPage ({\n\t\t\t\t\"processor\": processor,\n\t\t\t\t\"name\": name,\n\t\t\t\t\"files\": [file]\n\t\t\t});\n\t\tprocessor.addPage (page);\n\t\tlet result: string = await processor.process (name, args);\n\n\t\treturn (result);\n\t}\n\n\t/**\n\t * Process a url and get the result.\n\t */\n\tstatic async processUrl (options: HotStartOptions): Promise<string>\n\t{\n\t\tlet file: HotFile = new HotFile ({\n\t\t\t\"url\": options.url\n\t\t});\n\n\t\tawait file.load ();\n\t\tlet page: HotPage = new HotPage ({\n\t\t\t\t\"processor\": options.processor,\n\t\t\t\t\"name\": options.name,\n\t\t\t\t\"files\": [file],\n\t\t\t\t\"testerName\": options.testerName,\n\t\t\t\t\"testerMap\": options.testerMap\n\t\t\t});\n\t\toptions.processor.addPage (page);\n\t\tlet result: string = await options.processor.process (options.name, options.args);\n\n\t\treturn (result);\n\t}\n\n\t/**\n\t * Process content and get the result.\n\t */\n\tstatic async processContent (options: HotStartOptions): Promise<string>\n\t{\n\t\tlet file: HotFile = new HotFile ({\n\t\t\t\"content\": options.content\n\t\t});\n\t\tawait file.load ();\n\t\tlet page: HotPage = new HotPage ({\n\t\t\t\t\"processor\": options.processor,\n\t\t\t\t\"name\": options.name,\n\t\t\t\t\"files\": [file]\n\t\t\t});\n\t\toptions.processor.addPage (page);\n\t\tlet result: string = await options.processor.process (options.name, options.args);\n\n\t\treturn (result);\n\t}\n\n\t/**\n\t * When the window has finished loading, execute the function.\n\t * This is meant for web browser use only.\n\t */\n\tstatic onReady (readyFunc: () => void): void\n\t{\n\t\tif ((document.readyState === \"complete\") || (document.readyState === \"interactive\"))\n\t\t\treadyFunc ();\n\t\telse\n\t\t\twindow.addEventListener (\"load\", readyFunc);\n\t}\n\n\t/**\n\t * Replace the current HTML page with the output.\n\t * This is meant for web browser use only.\n\t */\n\tstatic async useOutput (output: string): Promise<void>\n\t{\n\t\t\tlet parser = new DOMParser ();\n\t\t\tlet child = parser.parseFromString (output, \"text/html\");\n\t\t\tlet htmlObj: HTMLHtmlElement = document.getElementsByTagName('html')[0];\n\n\t\t\thtmlObj.innerHTML = child.getElementsByTagName('html')[0].innerHTML;\n\n\t\t\t// Thanks to newfurniturey at: \n\t\t\t// https://stackoverflow.com/questions/22945884/domparser-appending-script-tags-to-head-body-but-not-executing\n\t\t\tlet tmpScripts = document.getElementsByTagName('script');\n\t\t\tif (tmpScripts.length > 0) {\n\t\t\t\t// push all of the document's script tags into an array\n\t\t\t\t// (to prevent dom manipulation while iterating over dom nodes)\n\t\t\t\tlet scripts: HTMLScriptElement[] = [];\n\t\t\t\tfor (let i = 0; i < tmpScripts.length; i++) {\n\t\t\t\t\tscripts.push(tmpScripts[i]);\n\t\t\t\t}\n\n\t\t\t\t// iterate over all script tags and create duplicate tags for each\n\t\t\t\tfor (let i = 0; i < scripts.length; i++) {\n\t\t\t\t\tlet s: HTMLScriptElement = document.createElement('script');\n\n\t\t\t\t\t// add the new node to the page\n\t\t\t\t\tscripts[i].parentNode.appendChild(s);\n\n\t\t\t\t\t// remove the original (non-executing) node from the page\n\t\t\t\t\tscripts[i].parentNode.removeChild(scripts[i]);\n\n\t\t\t\t\tawait new Promise<void> ((resolve2, reject2) =>\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\ts.onload = () =>\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tresolve2 ();\n\t\t\t\t\t\t\t\t};\n\n\t\t\t\t\t\t\tlet hasSrc: boolean = false;\n\n\t\t\t\t\t\t\tif (scripts[i].getAttribute (\"src\") != null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (scripts[i].getAttribute (\"src\") !== \"\")\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\ts.setAttribute (\"src\", scripts[i].getAttribute (\"src\"));\n\t\t\t\t\t\t\t\t\thasSrc = true;\n\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tif (scripts[i].getAttribute (\"type\") != null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (scripts[i].getAttribute (\"type\") !== \"\")\n\t\t\t\t\t\t\t\t\ts.setAttribute (\"type\", scripts[i].getAttribute (\"type\"));\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\ts.innerHTML = scripts[i].innerHTML;\n\n\t\t\t\t\t\t\tif (hasSrc === false)\n\t\t\t\t\t\t\t\tresolve2 ();\n\t\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t}\n\t}\n\n\t/**\n\t * Wait for testers to load.\n\t * \n\t * @fixme This does not wait for ALL testers to finish loading. Only \n\t * the first one.\n\t */\n\tstatic async waitForTesters (): Promise<void>\n\t{\n\t\twhile (HotStaq.isReadyForTesting === false)\n\t\t\tawait HotStaq.wait (10);\n\n\t\tif (HotStaq.onReadyForTesting != null)\n\t\t\tawait HotStaq.onReadyForTesting ();\n\t}\n\n\t/**\n\t * Setup the testers api, if any.\n\t */\n\tstatic setupTesters (processor: HotStaq, options: HotStartOptions)\n\t{\n\t\tif (processor.mode === DeveloperMode.Development)\n\t\t{\n\t\t\tif (processor.testerAPI == null)\n\t\t\t{\n\t\t\t\tif (options.testerAPIBaseUrl == null)\n\t\t\t\t\toptions.testerAPIBaseUrl = \"\";\n\n\t\t\t\tif (options.testerAPIBaseUrl === \"\")\n\t\t\t\t\toptions.testerAPIBaseUrl = \"http://127.0.0.1:8182\";\n\n\t\t\t\tlet client: HotClient = new HotClient (processor);\n\t\t\t\tlet testerAPI: HotTesterAPI = new HotTesterAPI (options.testerAPIBaseUrl, client);\n\t\t\t\ttesterAPI.connection.api = testerAPI;\n\t\t\t\tprocessor.testerAPI = testerAPI;\n\t\t\t}\n\t\t}\n\t}\n\n\t/**\n\t * Setup the testers api on the client, if needed.\n\t */\n\tstatic setupClientTesters (processor: HotStaq): string\n\t{\n\t\tlet output: string = \"\";\n\n\t\tif (processor.mode === DeveloperMode.Development)\n\t\t{\n\t\t\toutput += \n`<script type = \"text/javascript\">\nfunction hotstaq_isDocumentReady ()\n{\nif (window[\"Hot\"] != null)\n{\nif (Hot.Mode === HotStaqWeb.DeveloperMode.Development)\n{\nlet func = function ()\n\t{\n\t\tif (Hot.TesterAPI != null)\n\t\t{\n\t\t\tlet testPaths = {};\n\t\t\tlet testElements = JSON.stringify (Hot.CurrentPage.testElements);\n\t\t\tlet testMaps = JSON.stringify (Hot.CurrentPage.testMaps);\n\n\t\t\tfor (let key in Hot.CurrentPage.testPaths)\n\t\t\t{\n\t\t\t\tlet testPath = Hot.CurrentPage.testPaths[key];\n\n\t\t\t\ttestPaths[key] = testPath.toString ();\n\t\t\t}\n\n\t\t\tlet testPathsStr = JSON.stringify (testPaths);\n\n\t\t\tHot.TesterAPI.tester.pageLoaded ({\n\t\t\t\t\ttesterName: Hot.CurrentPage.testerName,\n\t\t\t\t\ttesterMap: Hot.CurrentPage.testerMap,\n\t\t\t\t\tpageName: Hot.CurrentPage.name,\n\t\t\t\t\ttestElements: testElements,\n\t\t\t\t\ttestPaths: testPathsStr\n\t\t\t\t}).then (function (resp)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (resp.error != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (resp.error !== \"\")\n\t\t\t\t\t\t\t\tthrow new Error (resp.error);\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tHotStaqWeb.HotStaq.isReadyForTesting = true;\n\t\t\t\t\t});\n\t\t}\n\t};\n\nif ((document.readyState === \"complete\") || (document.readyState === \"interactive\"))\n\tfunc ();\nelse\n\tdocument.addEventListener (\"DOMContentLoaded\", func);\n}\n}\n}\n\nhotstaq_isDocumentReady ();\n</script>`;\n\t\t}\n\n\t\treturn (output);\n\t}\n\n\t/**\n\t * Process and replace the current HTML page with the hott script from the given url.\n\t * This is meant for web browser use only.\n\t */\n\tstatic async displayUrl (url: string | HotStartOptions, name: string = null, \n\t\tprocessor: HotStaq = null, args: any = null): Promise<HotStaq>\n\t{\n\t\treturn (new Promise<HotStaq> ((resolve, reject) =>\n\t\t\t{\n\t\t\t\tHotStaq.onReady (async () =>\n\t\t\t\t\t{\n\t\t\t\t\t\tlet options: HotStartOptions = {\n\t\t\t\t\t\t\t\t\"url\": \"\"\n\t\t\t\t\t\t\t};\n\n\t\t\t\t\t\tif (name == null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (typeof (url) === \"string\")\n\t\t\t\t\t\t\t\toptions.name = url;\n\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\toptions.name = url.name;\n\t\t\t\t\t\t}\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\toptions.name = name;\n\n\t\t\t\t\t\tif (options.name === \"\")\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (typeof (url) === \"string\")\n\t\t\t\t\t\t\t\toptions.name = url;\n\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\toptions.name = url.name;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (typeof (url) === \"string\")\n\t\t\t\t\t\t\toptions.url = url;\n\t\t\t\t\t\telse\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\toptions.url = url.url;\n\n\t\t\t\t\t\t\tif (processor == null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (url.processor != null)\n\t\t\t\t\t\t\t\t\tprocessor = url.processor;\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tif (args == null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (url.args != null)\n\t\t\t\t\t\t\t\t\targs = url.args;\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tif (url.testerMap != null)\n\t\t\t\t\t\t\t\toptions.testerMap = url.testerMap;\n\n\t\t\t\t\t\t\tif (url.testerName != null)\n\t\t\t\t\t\t\t\toptions.testerName = url.testerName;\n\n\t\t\t\t\t\t\tif (url.testerAPIBaseUrl != null)\n\t\t\t\t\t\t\t\toptions.testerAPIBaseUrl = url.testerAPIBaseUrl;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (processor == null)\n\t\t\t\t\t\t\tprocessor = new HotStaq ();\n\n\t\t\t\t\t\tHotStaq.setupTesters (processor, options);\n\n\t\t\t\t\t\toptions.processor = processor;\n\t\t\t\t\t\toptions.args = args;\n\n\t\t\t\t\t\tif (options.url.indexOf (\"hstqserve\") < 0)\n\t\t\t\t\t\t\toptions.url += \"?hstqserve=nahfam\";\n\n\t\t\t\t\t\tlet output: string = await HotStaq.processUrl (options);\n\n\t\t\t\t\t\toutput += HotStaq.setupClientTesters (processor);\n\n\t\t\t\t\t\tawait HotStaq.useOutput (output);\n\t\t\t\t\t\tresolve (processor);\n\t\t\t\t\t});\n\t\t\t}));\n\t}\n\n\t/**\n\t * Process and replace the current HTML page with the hott script.\n\t * This is meant for web browser use only.\n\t */\n\tstatic async displayContent (content: string | HotStartOptions, name: string = null, \n\t\t\tprocessor: HotStaq = null, args: any = null): Promise<HotStaq>\n\t{\n\t\treturn (new Promise<HotStaq> ((resolve, reject) =>\n\t\t\t{\n\t\t\t\tHotStaq.onReady (async () =>\n\t\t\t\t\t{\n\t\t\t\t\t\tlet options: HotStartOptions = {\n\t\t\t\t\t\t\t\t\"content\": \"\"\n\t\t\t\t\t\t\t};\n\n\t\t\t\t\t\tif (name == null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (typeof (content) === \"string\")\n\t\t\t\t\t\t\t\toptions.name = \"\";\n\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\toptions.name = content.name;\n\t\t\t\t\t\t}\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\toptions.name = name;\n\n\t\t\t\t\t\tif (options.name === \"\")\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (typeof (content) === \"string\")\n\t\t\t\t\t\t\t\toptions.name = \"\"; /// @fixme Is this ok to do?\n\t\t\t\t\t\t\telse\n\t\t\t\t\t\t\t\toptions.name = content.name;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (typeof (content) === \"string\")\n\t\t\t\t\t\t\toptions.content = content;\n\t\t\t\t\t\telse\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\toptions.content = content.content;\n\n\t\t\t\t\t\t\tif (processor == null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (content.processor != null)\n\t\t\t\t\t\t\t\t\tprocessor = content.processor;\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tif (args == null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (content.args != null)\n\t\t\t\t\t\t\t\t\targs = content.args;\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\tif (content.testerMap != null)\n\t\t\t\t\t\t\t\toptions.testerMap = content.testerMap;\n\n\t\t\t\t\t\t\tif (content.testerName != null)\n\t\t\t\t\t\t\t\toptions.testerName = content.testerName;\n\n\t\t\t\t\t\t\tif (content.testerAPIBaseUrl != null)\n\t\t\t\t\t\t\t\toptions.testerAPIBaseUrl = content.testerAPIBaseUrl;\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tif (processor == null)\n\t\t\t\t\t\t\tprocessor = new HotStaq ();\n\n\t\t\t\t\t\tHotStaq.setupTesters (processor, options);\n\n\t\t\t\t\t\toptions.processor = processor;\n\t\t\t\t\t\toptions.args = args;\n\n\t\t\t\t\t\tlet output: string = await HotStaq.processContent (options);\n\n\t\t\t\t\t\tawait HotStaq.useOutput (output);\n\t\t\t\t\t\tresolve (processor);\n\t\t\t\t\t});\n\t\t\t}));\n\t}\n}\n\nif (typeof (document) !== \"undefined\")\n{\n\tlet loadHotStaqSite = function ()\n\t{\n\t\tlet hotstaqElms = document.getElementsByTagName (\"hotstaq\");\n\n\t\t// Set this to true, just in case...\n\t\tHotStaq.isWeb = true;\n\n\t\t\t// @ts-ignore\n\t\tif (typeof (HotStaqWeb) !== \"undefined\")\n\t\t{\n\t\t\t// @ts-ignore\n\t\t\twindow.HotStaq = HotStaqWeb.HotStaq;\n\t\t\t// @ts-ignore\n\t\t\twindow.HotClient = HotStaqWeb.HotClient;\n\t\t\t// @ts-ignore\n\t\t\twindow.HotAPI = HotStaqWeb.HotAPI;\n\t\t\t// @ts-ignore\n\t\t\twindow.Hot = HotStaqWeb.Hot;\n\t\t\t// @ts-ignore\n\t\t\twindow.HotComponent = HotStaqWeb.HotComponent;\n\t\t}\n\n\t\tif (hotstaqElms.length > 0)\n\t\t{\n\t\t\t// @ts-ignore\n\t\t\tlet hotstaqElm: HTMLElement = hotstaqElms[0];\n\n\t\t\tsetTimeout (async function ()\n\t\t\t\t{\n\t\t\t\t\tlet getAttr = (elm: HTMLElement, attrNames: string[]) =>\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tfor (let iIdx = 0; iIdx < attrNames.length; iIdx++)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tlet attrName: string = attrNames[iIdx];\n\n\t\t\t\t\t\t\t\tif (elm.getAttribute (attrName) != null)\n\t\t\t\t\t\t\t\t\treturn (elm.getAttribute (attrName));\n\n\t\t\t\t\t\t\t\tif (elm.getAttribute (`data-${attrName}`) != null)\n\t\t\t\t\t\t\t\t\treturn (elm.getAttribute (`data-${attrName}`));\n\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\treturn (undefined);\n\t\t\t\t\t\t};\n\n\t\t\t\t\tlet loadPage: string = getAttr (hotstaqElm, [\"load-page\", \"loadPage\", \"src\"]) || \"\";\n\t\t\t\t\tlet router: string = getAttr (hotstaqElm, [\"router\"]) || \"\";\n\t\t\t\t\tlet name: string = getAttr (hotstaqElm, [\"name\"]) || \"default\";\n\t\t\t\t\tlet args: string = getAttr (hotstaqElm, [\"args\"]) || null;\n\t\t\t\t\tlet apiLibrary: string = getAttr (hotstaqElm, [\"api-library\", \"apiLibrary\"]) || null;\n\t\t\t\t\tlet apiName: string = getAttr (hotstaqElm, [\"api-name\", \"apiName\"]) || null;\n\t\t\t\t\tlet apiUrl: string = getAttr (hotstaqElm, [\"api-url\", \"apiUrl\"]) || null;\n\t\t\t\t\tlet testerName: string = getAttr (hotstaqElm, [\"tester-name\", \"testerName\"]) || \"HotTesterMochaSelenium\";\n\t\t\t\t\tlet testerMap: string = getAttr (hotstaqElm, [\"tester-map\", \"testerMap\"]) || null;\n\t\t\t\t\tlet testerApiBaseUrl: string = getAttr (hotstaqElm, [\"tester-api-base-url\", \"testerApiBaseUrl\"]) || null;\n\t\t\t\t\tlet testerLaunchpadUrl: string = getAttr (hotstaqElm, [\"tester-launchpad-url\", \"testerLaunchpadUrl\"]) || null;\n\t\t\t\t\tlet dontReuseProcessor: boolean = false;\n\t\t\t\t\tlet passRawUrl: boolean = false;\n\t\t\t\t\tlet htmlSource: string = hotstaqElm.innerHTML || \"\";\n\t\t\t\t\tlet routerManager: { [path: string]: { redirect: string; baseRedirect: string; base: string; src: string; } } = {};\n\t\t\t\t\tlet routerWildcards: string[] = [];\n\t\t\t\t\tlet search: URLSearchParams = new URLSearchParams (window.location.search);\n\n\t\t\t\t\tif (getAttr (hotstaqElm, [\"src\"]) != null)\n\t\t\t\t\t\tloadPage = getAttr (hotstaqElm, [\"src\"]);\n\n\t\t\t\t\tif (getAttr (hotstaqElm, [\"passRawUrl\"]) != null)\n\t\t\t\t\t\tpassRawUrl = true;\n\n\t\t\t\t\tif (getAttr (hotstaqElm, [\"dont-reuse-processor\", \"dontReuseProcessor\"]) != null)\n\t\t\t\t\t\tdontReuseProcessor = true;\n\n\t\t\t\t\tlet hstqbaseredirect: string = search.get (\"hstqbaseredirect\");\n\n\t\t\t\t\tif (hstqbaseredirect != null)\n\t\t\t\t\t{\n\t\t\t\t\t\thstqbaseredirect = decodeURI (hstqbaseredirect);\n\t\t\t\t\t\twindow.history.replaceState (\"\", \"\", hstqbaseredirect);\n\t\t\t\t\t\tloadPage = hstqbaseredirect;\n\t\t\t\t\t}\n\n\t\t\t\t\tlet hotstaqErrors = document.getElementsByTagName (\"hotstaq-error\");\n\n\t\t\t\t\tfor (let iIdx = 0; iIdx < hotstaqErrors.length; iIdx++)\n\t\t\t\t\t{\n\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\tlet hotstaqErrorElm: HTMLElement = hotstaqErrors[iIdx];\n\t\t\t\t\t\tlet errorStatus: string = getAttr (hotstaqErrorElm, [\"status\"]);\n\t\t\t\t\t\tlet unsupportedBrowser: string = getAttr (hotstaqErrorElm, [\"unsupported-browser-redirect\"]);\n\n\t\t\t\t\t\tif (unsupportedBrowser != null)\n\t\t\t\t\t\t\tHotStaq.errors[\"unsupportedBrowser\"] = { redirectToUrl: unsupportedBrowser };\n\t\t\t\t\t\telse\n\t\t\t\t\t\t\tHotStaq.errors[`${errorStatus}`] = { redirectToUrl: unsupportedBrowser };\n\t\t\t\t\t}\n\n\t\t\t\t\t// Check if async/await is available.\n\t\t\t\t\ttry\n\t\t\t\t\t{\n\t\t\t\t\t\teval (\"async () => {}\");\n\t\t\t\t\t}\n\t\t\t\t\tcatch (ex)\n\t\t\t\t\t{\n\t\t\t\t\t\tHotStaq.executeError (\"unsupportedBrowser\");\n\t\t\t\t\t}\n\n\t\t\t\t\tif (router !== \"\")\n\t\t\t\t\t{\n\t\t\t\t\t\tlet hotstaqRouterElms = document.getElementsByTagName (\"hotstaq-router\");\n\n\t\t\t\t\t\tfor (let iIdx = 0; iIdx < hotstaqRouterElms.length; iIdx++)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\tlet hotstaqRouterElm: HTMLElement = hotstaqRouterElms[iIdx];\n\t\t\t\t\t\t\tlet routerName: string = getAttr (hotstaqRouterElm, [\"name\"]);\n\t\t\t\t\t\t\tlet serveLocally: string = getAttr (hotstaqRouterElm, [\"serve-local\", \"serveLocally\"]);\n\n\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\tif (routerName === router)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t// Load all routes from the router.\n\t\t\t\t\t\t\t\tfor (let iJdx = 0; iJdx < hotstaqRouterElm.childNodes.length; iJdx++)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\t\t\tlet routerElm: HTMLElement = hotstaqRouterElm.childNodes[iJdx];\n\n\t\t\t\t\t\t\t\t\tif (routerElm instanceof HTMLElement)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tif (routerElm.tagName.toUpperCase () === \"ROUTE\")\n\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\tlet routerPath: string = getAttr (routerElm, [\"path\"]);\n\t\t\t\t\t\t\t\t\t\t\tlet redirect: string = getAttr (routerElm, [\"redirect\"]);\n\t\t\t\t\t\t\t\t\t\t\tlet baseRedirect: string = getAttr (routerElm, [\"base-redirect\", \"baseRedirect\"]);\n\t\t\t\t\t\t\t\t\t\t\tlet base: string = getAttr (routerElm, [\"base\"]);\n\t\t\t\t\t\t\t\t\t\t\tlet routerSrc: string = getAttr (routerElm, [\"src\"]);\n\n\t\t\t\t\t\t\t\t\t\t\tif (routerPath.indexOf (\"*\") > -1)\n\t\t\t\t\t\t\t\t\t\t\t\trouterWildcards.push (routerPath);\n\n\t\t\t\t\t\t\t\t\t\t\trouterManager[routerPath] = {\n\t\t\t\t\t\t\t\t\t\t\t\t\tredirect: redirect || undefined, \n\t\t\t\t\t\t\t\t\t\t\t\t\tbaseRedirect: baseRedirect || undefined, \n\t\t\t\t\t\t\t\t\t\t\t\t\tbase: base || undefined, \n\t\t\t\t\t\t\t\t\t\t\t\t\tsrc: routerSrc || undefined\n\t\t\t\t\t\t\t\t\t\t\t\t};\n\t\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tlet checkPath: string = window.location.pathname;\n\t\t\t\t\t\t\t\tlet gotoPath: string = window.location.pathname;\n\n\t\t\t\t\t\t\t\tif (serveLocally != null)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tconst lowerServeLocally: string = serveLocally.toLowerCase ();\n\n\t\t\t\t\t\t\t\t\tif ((lowerServeLocally === \"true\") ||\n\t\t\t\t\t\t\t\t\t\t(lowerServeLocally === \"yes\") ||\n\t\t\t\t\t\t\t\t\t\t(lowerServeLocally === \"1\"))\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tconst lastSlashPos: number = checkPath.lastIndexOf (\"/\");\n\n\t\t\t\t\t\t\t\t\t\tif (lastSlashPos > -1)\n\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\tcheckPath = checkPath.substring (lastSlashPos);\n\t\t\t\t\t\t\t\t\t\t\tgotoPath = gotoPath.substring (lastSlashPos);\n\t\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tif (routerWildcards.length > 0)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t// Serve locally doesn't really work with wildcards\n\t\t\t\t\t\t\t\t\t/// @fixme This isn't actually working like a wildcard should. This needs to be improved.\n\t\t\t\t\t\t\t\t\tfor (let iJdx = 0; iJdx < routerWildcards.length; iJdx++)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\tlet routeWildcard: string = routerWildcards[iJdx];\n\t\t\t\t\t\t\t\t\t\tlet tempRouteWildcard: string = routeWildcard.replace (\"*\", \"\");\n\n\t\t\t\t\t\t\t\t\t\tif (checkPath.indexOf (tempRouteWildcard) > -1)\n\t\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\t\t// This simply returns the key in the routerManager to access.\n\t\t\t\t\t\t\t\t\t\t\tcheckPath = routeWildcard;\n\n\t\t\t\t\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\t// Find the correct route and load it.\n\t\t\t\t\t\t\t\tif (routerManager[checkPath] != null)\n\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\tif (routerManager[checkPath].redirect != null)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\twindow.location.href = routerManager[checkPath].redirect;\n\n\t\t\t\t\t\t\t\t\t\treturn;\n\t\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\t\tif (routerManager[checkPath].baseRedirect != null)\n\t\t\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\t\t\twindow.location.href = `${routerManager[checkPath].baseRedirect}?hstqbaseredirect=${encodeURI (gotoPath)}`;\n\n\t\t\t\t\t\t\t\t\t\treturn;\n\t\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\t\tif (routerManager[checkPath].src != null)\n\t\t\t\t\t\t\t\t\t\tloadPage = routerManager[checkPath].src;\n\t\t\t\t\t\t\t\t}\n\n\t\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tif (args != null)\n\t\t\t\t\t\targs = JSON.parse (args);\n\t\t\t\t\telse\n\t\t\t\t\t\targs = Hot.Arguments;\n\n\t\t\t\t\tlet hasHtmlSource: boolean = false;\n\n\t\t\t\t\tif (htmlSource !== \"\")\n\t\t\t\t\t{\n\t\t\t\t\t\tconst htmlSourceCheck: string = htmlSource.replace (/\\s/g,'');\n\n\t\t\t\t\t\tif (htmlSourceCheck !== \"\")\n\t\t\t\t\t\t\thasHtmlSource = true;\n\t\t\t\t\t}\n\n\t\t\t\t\tlet tempMode = 0;\n\n\t\t\t\t\t// @ts-ignore\n\t\t\t\t\tif (window[\"Hot\"] != null)\n\t\t\t\t\t\ttempMode = Hot.Mode;\n\t\t\t\n\t\t\t\t\tlet processor: HotStaq = null;\n\n\t\t\t\t\tif (dontReuseProcessor === false)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (typeof (Hot) !== \"undefined\")\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (Hot.CurrentPage != null)\n\t\t\t\t\t\t\t{\n\t\t\t\t\t\t\t\tif (Hot.CurrentPage.processor != null)\n\t\t\t\t\t\t\t\t\tprocessor = Hot.CurrentPage.processor;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tif (processor == null)\n\t\t\t\t\t\tprocessor = new HotStaq ();\n\n\t\t\t\t\tprocessor.mode = tempMode;\n\n\t\t\t\t\tlet options: HotStartOptions = {\n\t\t\t\t\t\t\tname: name,\n\t\t\t\t\t\t\tprocessor: processor,\n\t\t\t\t\t\t\targs: args\n\t\t\t\t\t\t};\n\n\t\t\t\t\tif (loadPage !== \"\")\n\t\t\t\t\t{\n\t\t\t\t\t\tif (passRawUrl === false)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\tif (loadPage.indexOf (\"hstqserve\") < 0)\n\t\t\t\t\t\t\t\tloadPage += \"?hstqserve=nahfam\";\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\toptions.url = loadPage;\n\t\t\t\t\t}\n\n\t\t\t\t\tif (testerMap != null)\n\t\t\t\t\t{\n\t\t\t\t\t\toptions.testerMap = testerMap;\n\t\t\t\t\t\toptions.testerName = testerName;\n\t\t\t\t\t}\n\n\t\t\t\t\tif (testerName != null)\n\t\t\t\t\t\toptions.testerName = testerName;\n\n\t\t\t\t\tif (testerApiBaseUrl != null)\n\t\t\t\t\t\toptions.testerAPIBaseUrl = testerApiBaseUrl;\n\n\t\t\t\t\tif (testerLaunchpadUrl != null)\n\t\t\t\t\t\toptions.testerLaunchpadUrl = testerLaunchpadUrl;\n\n\t\t\t\t\tif (apiName != null)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet client = new HotClient (processor);\n\n\t\t\t\t\t\tif (apiUrl === \"\")\n\t\t\t\t\t\t\tthrow new Error (`api-url was not set!`);\n\n\t\t\t\t\t\tlet parentLib: any = window;\n\n\t\t\t\t\t\tif (apiLibrary != null)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\t// @ts-ignore\n\t\t\t\t\t\t\tparentLib = window[apiLibrary];\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\tlet newAPI = new parentLib[apiName] (apiUrl, client);\n\t\t\t\t\t\tnewAPI.connection.api = newAPI;\n\t\t\t\t\t\tprocessor.api = newAPI;\n\t\t\t\t\t}\n\n\t\t\t\t\tif (hasHtmlSource === false)\n\t\t\t\t\t{\n\t\t\t\t\t\tif (loadPage === \"\")\n\t\t\t\t\t\t\tthrow new Error (`The hotstaq tag must have a src, HTML contents inside it, or a router set.`);\n\n\t\t\t\t\t\tHotStaq.displayUrl (options);\n\t\t\t\t\t}\n\t\t\t\t\telse\n\t\t\t\t\t{\n\t\t\t\t\t\tHotStaq.displayContent (options);\n\t\t\t\t\t}\n\t\t\t\t}, 50);\n\t\t}\n\t};\n\n\t/// @ts-ignore\n\twindow.ethereum22 = window.ethereum;\n\twindow.addEventListener (\"load\", loadHotStaqSite);\n}\n","import { HotStaq } from \"./HotStaq\";\nimport { HotTestElement, HotTestElementOptions } from \"./HotTestElement\";\nimport { HotTestPage } from \"./HotTestMap\";\n\n/**\n * This actually executes the tests.\n */\nexport abstract class HotTestDriver\n{\n\t/**\n\t * The current page.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The current page.\n\t */\n\tpage: HotTestPage;\n\t/**\n\t * The delay in milliseconds between each executed command.\n\t */\n\tcommandDelay: number;\n\t/**\n\t * Any data that needs to be saved between the different testing runs.\n\t */\n\tpersistentData: any;\n\n\tconstructor (processor: HotStaq, page: HotTestPage = null)\n\t{\n\t\tthis.processor = processor;\n\t\tthis.page = page;\n\t\tthis.commandDelay = 20;\n\t\tthis.persistentData = {};\n\t}\n\n\t/**\n\t * Get a test object by it's name. If a * is used, it will be used as a \n\t * wildcard for the object's name. If a > is used, then the name will \n\t * be treated as a CSS selector.\n\t */\n\tparseTestObject (name: string): string\n\t{\n\t\tlet pos: number = name.indexOf (\"*\");\n\t\tlet wildcard: string = \"\";\n\n\t\tif (pos > -1)\n\t\t{\n\t\t\tname = name.replace (/\\*/, \"\");\n\t\t\twildcard = \"*\";\n\t\t}\n\n\t\tlet selector: string = `[data-test-object-name${wildcard}='${name}']`;\n\t\tpos = name.indexOf (\">\");\n\n\t\tif (pos > -1)\n\t\t{\n\t\t\tname = name.replace (/\\>/, \"\");\n\t\t\tselector = name;\n\t\t}\n\n\t\treturn (selector);\n\t}\n\n\t/**\n\t * Wait for a number of milliseconds.\n\t */\n\tasync wait (numMilliseconds: number): Promise<void>\n\t{\n\t\treturn (await new Promise ((resolve, reject) =>\n\t\t\t{\n\t\t\t\tsetTimeout (() =>\n\t\t\t\t\t{\n\t\t\t\t\t\tresolve ();\n\t\t\t\t\t}, numMilliseconds);\n\t\t\t}));\n\t}\n\n\t/**\n\t * Print a message.\n\t */\n\tasync print (message: string): Promise<void>\n\t{\n\t\tprocess.stdout.write (message);\n\t}\n\n\t/**\n\t * Print a message line.\n\t */\n\tasync println (message: string): Promise<void>\n\t{\n\t\tawait this.print (`${message}\\n`);\n\t}\n\n\t/**\n\t * Disconnect this server or destroy anything associated with this HotTestDriver.\n\t */\n\tabstract destroy (): Promise<void>;\n\n\t/**\n\t * Navigate to a url.\n\t */\n\tabstract navigateToUrl (url: string): Promise<void>;\n\t/**\n\t * Wait for a HotTestElement to load.\n\t */\n\tabstract waitForTestElement (name: string | HotTestElement, options?: HotTestElementOptions): Promise<any>;\n\t/**\n\t * Find a HotTestElement to utilize.\n\t */\n\tabstract findTestElement (name: string | HotTestElement, options?: HotTestElementOptions): Promise<any>;\n\t/**\n\t * Run a HotTestElement command.\n\t */\n\tabstract runCommand (testElm: string | HotTestElement, funcName?: string, valueStr?: string): Promise<any>;\n\t/**\n\t * An expression to test.\n\t */\n\tabstract assertElementValue (name: string | HotTestElement, value: any, \n\t\terrorMessage?: string, options?: HotTestElementOptions): Promise<any>;\n\t/**\n\t * An expression to test.\n\t */\n\tasync assert (value: any, errorMessage: string = \"\"): Promise<any>\n\t{\n\t\tif (! (value))\n\t\t\tthrow new Error (errorMessage);\n\t}\n\n\t/**\n\t * Run a series of test elements.\n\t */\n\tasync run (executions: string[] | string[][]): Promise<any[]>\n\t{\n\t\tlet results: any[] = [];\n\n\t\tfor (let iIdx = 0; iIdx < executions.length; iIdx++)\n\t\t{\n\t\t\tlet execution: any = executions[iIdx];\n\t\t\tlet testElm: HotTestElement = null;\n\t\t\tlet func: string = \"\";\n\t\t\tlet value: string = \"\";\n\n\t\t\tif (typeof (execution) === \"string\")\n\t\t\t{\n\t\t\t\ttestElm = this.page.testElements[execution];\n\n\t\t\t\t/// @fixme This is going to wreck selecting test elements by wildcards.\n\t\t\t\tif (testElm == null)\n\t\t\t\t\tthrow new Error (`HotTestDriver: Unable to find test element ${execution}`);\n\n\t\t\t\tfunc = testElm.func;\n\t\t\t\tvalue = testElm.value;\n\t\t\t}\n\n\t\t\tif (execution instanceof Array)\n\t\t\t{\n\t\t\t\tlet name: string = execution[0];\n\t\t\t\ttestElm = this.page.testElements[name];\n\n\t\t\t\t// This null catch is specifically to help find wildcard test elements.\n\t\t\t\tif (testElm == null)\n\t\t\t\t{\n\t\t\t\t\ttestElm = new HotTestElement (name);\n\t\t\t\t\tfunc = execution[1];\n\t\t\t\t\tvalue = execution[2];\n\t\t\t\t}\n\t\t\t\telse\n\t\t\t\t{\n\t\t\t\t\tfunc = testElm.func;\n\t\t\t\t\tvalue = testElm.value;\n\n\t\t\t\t\tif (execution.length > 1)\n\t\t\t\t\t\tfunc = execution[1];\n\n\t\t\t\t\tif (execution.length > 2)\n\t\t\t\t\t\tvalue = execution[2];\n\t\t\t\t}\n\t\t\t}\n\n\t\t\ttestElm.func = func;\n\t\t\ttestElm.value = value;\n\n\t\t\tlet result = await this.runCommand (testElm);\n\n\t\t\tawait HotStaq.wait (this.commandDelay);\n\n\t\t\tresults.push (result);\n\t\t}\n\n\t\treturn (results);\n\t}\n}","/**\n * Hot test element options.\n */\nexport interface IHotTestElementOptions\n{\n\t/**\n\t * Indicates that the test element must be visible in \n\t * order to select it.\n\t */\n\tmustBeVisible?: boolean;\n\t/**\n\t * If the test element is missing, ignore the error. This \n\t * will cause the rest of the function to return immediately \n\t * without any exceptions being thrown.\n\t */\n\tignoreMissingElementError?: boolean;\n}\n\n/**\n * Hot test element options.\n */\nexport class HotTestElementOptions implements IHotTestElementOptions\n{\n\t/**\n\t * Indicates that the test element must be visible in \n\t * order to select it.\n\t */\n\tmustBeVisible: boolean;\n\t/**\n\t * If the test element is missing, ignore the error. This \n\t * will cause the rest of the function to return immediately \n\t * without any exceptions being thrown.\n\t */\n\tignoreMissingElementError: boolean;\n\n\tconstructor (copy: IHotTestElementOptions = {})\n\t{\n\t\tthis.mustBeVisible = copy.mustBeVisible || true;\n\t\tthis.ignoreMissingElementError = copy.ignoreMissingElementError || false;\n\t}\n}\n\n/**\n * A test element.\n */\nexport interface IHotTestElement\n{\n\t/**\n\t * The name of the element.\n\t */\n\tname: string;\n\t/**\n\t * The name of the function to execute \n\t * while executing the test.\n\t */\n\tfunc?: string;\n\t/**\n\t * The value to use.\n\t */\n\tvalue?: any;\n}\n\n/**\n * A test element.\n */\nexport class HotTestElement implements IHotTestElement\n{\n\t/**\n\t * The name of the element.\n\t */\n\tname: string;\n\t/**\n\t * The name of the function to execute \n\t * while executing the test.\n\t */\n\tfunc: string;\n\t/**\n\t * The value to use.\n\t */\n\tvalue: any;\n\n\tconstructor (name: string | IHotTestElement, func: string = \"\", value: any = null)\n\t{\n\t\tif (typeof (name) === \"string\")\n\t\t{\n\t\t\tthis.name = name;\n\t\t\tthis.func = func;\n\t\t\tthis.value = value;\n\t\t}\n\t\telse\n\t\t{\n\t\t\tthis.name = name.name;\n\t\t\tthis.func = name.func || func;\n\t\t\tthis.value = name.value || value;\n\t\t}\n\t}\n}","import { HotTestElement } from \"./HotTestElement\";\nimport { HotTestDriver } from \"./HotTestDriver\";\nimport { HotSiteMapPath } from \"./HotStaq\";\n\n/**\n * Create a test path for later execution.\n */\nexport type HotTestPath = (driver: HotTestDriver, ...args: any) => Promise<any>;\n\n/**\n * The destination to take in a map.\n */\nexport class HotTestDestination\n{\n\t/**\n\t * The destination to take.\n\t */\n\tdestination: string;\n\t/**\n\t * If set to true, this will automatically start executing it's \n\t * tests when it's time.\n\t */\n\tautoStart: boolean;\n\n\tconstructor (destination: string | HotTestDestination | HotSiteMapPath = \"\", autoStart: boolean = true)\n\t{\n\t\tif (typeof (destination) === \"string\")\n\t\t{\n\t\t\tthis.destination = destination;\n\t\t\tthis.autoStart = autoStart;\n\t\t}\n\t\telse\n\t\t{\n\t\t\tif (destination instanceof HotTestDestination)\n\t\t\t{\n\t\t\t\tthis.destination = destination.destination;\n\t\t\t\tthis.autoStart = destination.autoStart;\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\tthis.destination = destination.path;\n\t\t\t\tthis.autoStart = destination.autoStart;\n\t\t\t}\n\t\t}\n\t}\n}\n\n/**\n * A page containing only test related info.\n */\nexport interface HotTestPage\n{\n\t/**\n\t * The elements to test on this map.\n\t */\n\ttestElements: { [name: string]: HotTestElement; };\n\t/**\n\t * The test paths to test on this map.\n\t */\n\ttestPaths: { [name: string]: HotTestPath; };\n}\n\n/**\n * Maps the paths that are taken to complete a test.\n */\nexport class HotTestMap\n{\n\t/**\n\t * The order in which paths are to be taken. Each destination is a string \n\t * in a type -> path order. The type could be either a page or api route. \n\t * For example:\n\t * ```\n\t * [\n\t *      \"page:signin_page -> signin_path\",\n\t *      \"page:account_page -> change_username_path\",\n\t *      \"page:account_page -> change_password_path\",\n\t *      \"page:account_page -> change_name_path -> change_address_path\",\n\t * \t\t\"page:account_page -> signout_path\",\n\t * \t\t\"api:account_api_route -> signout_route_method -> signout_test_path\"\n\t * ]\n\t * ```\n\t * \n\t * The first string to the left of the -> will always be the type, such as a \n\t * page or an api route. Any strings to the right of the -> will be a path, even \n\t * when chaining addtional ->'s.\n\t */\n\tdestinations: HotTestDestination[] | { [name: string]: HotTestDestination; };\n\t/**\n\t * The order in which destinations are supposed to execute. This is \n\t * ignored if the destinations are an array.\n\t */\n\tdestinationOrder: string[];\n\t/**\n\t * The test pages to execute.\n\t */\n\tpages: {\n\t\t\t[name: string]: HotTestPage\n\t\t};\n\n\tconstructor (destinations: string[] | HotTestDestination[] | { [name: string]: string | HotTestDestination; } = [], \n\t\tpages: { [name: string]: HotTestPage } = {}, destinationOrder: string[] = [])\n\t{\n\t\t// Go through and convert any strings into HotTestDestinations.\n\t\tif (destinations instanceof Array)\n\t\t{\n\t\t\tthis.destinations = [];\n\n\t\t\tfor (let iIdx = 0; iIdx < destinations.length; iIdx++)\n\t\t\t{\n\t\t\t\tlet dest = destinations[iIdx];\n\n\t\t\t\tthis.destinations.push (new HotTestDestination (dest));\n\t\t\t}\n\t\t}\n\t\telse\n\t\t{\n\t\t\tthis.destinations = {};\n\n\t\t\tfor (let key in destinations)\n\t\t\t{\n\t\t\t\tlet dest = destinations[key];\n\n\t\t\t\tthis.destinations[key] = new HotTestDestination (dest);\n\t\t\t}\n\t\t}\n\n\t\tthis.destinationOrder = destinationOrder;\n\t\tthis.pages = pages;\n\t}\n}","import { HotStaq } from \"./HotStaq\";\nimport { HotRoute } from \"./HotRoute\";\nimport { HotRouteMethod, TestCaseObject } from \"./HotRouteMethod\";\nimport { HotTestDriver } from \"./HotTestDriver\";\nimport { HotTestDestination, HotTestMap, HotTestPage, HotTestPath } from \"./HotTestMap\";\n\n/**\n * The test stop that is executed as either a destination or \n * a path.\n */\nexport interface HotTestStop\n{\n\t/**\n\t * A command to execute. Can be:\n\t * * print(x)\n\t *   * Print a message to the server's console.\n\t * * println(x)\n\t *   * Print a message with a new line to the server's console.\n\t * * url(x)\n\t *   * Open a url. Must be an absolute url.\n\t * * waitForTesterAPIData\n\t *   * This will wait for the tester API to receive data.\n\t * * wait(x)\n\t *   * This will wait for x number of milliseconds.\n\t * * waitForTestObject(x)\n\t *   * This will wait for a test object to be loaded.\n\t */\n\tcmd: string;\n\t/**\n\t * The destination to execute.\n\t */\n\tdest: string;\n\t/**\n\t * The path to execute.\n\t */\n\tpath: string;\n}\n\n/**\n * The destination for a test to take.\n */\nexport interface HotDestination\n{\n\t/**\n\t * The name of the map.\n\t */\n\tmapName: string;\n\t/**\n\t * The page to start at.\n\t */\n\tpage: string;\n\t/**\n\t * The API route to start using.\n\t */\n\tapi: string;\n\t/**\n\t * The paths to take on the page.\n\t */\n\tpaths: HotTestStop[];\n}\n\n/**\n * Executes tests.\n */\nexport abstract class HotTester\n{\n\t/**\n\t * The tester name.\n\t */\n\tname: string;\n\t/**\n\t * The base url that will construct future urls.\n\t */\n\tbaseUrl: string;\n\t/**\n\t * The associated processor.\n\t */\n\tprocessor: HotStaq;\n\t/**\n\t * The test maps to test.\n\t */\n\ttestMaps: { [name: string]: HotTestMap; };\n\t/**\n\t * The driver to use when running tests.\n\t */\n\tdriver: HotTestDriver;\n\t/**\n\t * Has this tester finished loading?\n\t */\n\tfinishedLoading: boolean;\n\t/**\n\t * Has this tester finished setting up?\n\t */\n\thasBeenSetup: boolean;\n\t/**\n\t * Has this tester finished setting up?\n\t */\n\thasBeenDestroyed: boolean;\n\n\tconstructor (processor: HotStaq, name: string, baseUrl: string, \n\t\tdriver: HotTestDriver, testMaps: { [name: string]: HotTestMap; } = {})\n\t{\n\t\tthis.processor = processor;\n\t\tthis.name = name;\n\t\tthis.baseUrl = baseUrl;\n\t\tthis.testMaps = testMaps;\n\t\tthis.driver = driver;\n\t\tthis.finishedLoading = false;\n\t\tthis.hasBeenSetup = false;\n\t\tthis.hasBeenDestroyed = false;\n\t}\n\n\t/**\n\t * Executed when setting up the tester.\n\t */\n\tabstract setup (isWebRoute: boolean, url: string, destinationKey?: string): Promise<void>;\n\t/**\n\t * Executed when destroying up the tester.\n\t */\n\tabstract destroy (): Promise<void>;\n\n\t/**\n\t * Executed when tests are started. If this returns true, it will \n\t * continue and execute all test paths. If this returns it will \n\t * skip all test paths and execute onTestEnd instead.\n\t */\n\tasync onTestStart? (destination: HotDestination, url: string, destinationKey?: string): Promise<boolean>;\n\t/**\n\t * Executed when an API test path has started. If this returns false, \n\t * the testPath will not be immediately executed afterwards.\n\t */\n\tasync onTestAPIPathStart? (destination: HotDestination, method: HotRouteMethod, \n\t\ttestName: string, continueWhenTestIsComplete?: boolean): Promise<boolean>;\n\t/**\n\t * Executed when an API test path has ended.\n\t */\n\tasync onTestAPIPathEnd? (destination: HotDestination, method: HotRouteMethod, \n\t\ttestName: string, result: any, continueWhenTestIsComplete?: boolean): Promise<void>;\n\t/**\n\t * Executed when page tests are started. If this returns false, the testPath will not be \n\t * immediately executed afterwards.\n\t */\n\tasync onTestPagePathStart? (destination: HotDestination, page: HotTestPage, \n\t\tstop: HotTestStop, continueWhenTestIsComplete?: boolean): Promise<boolean>;\n\t/**\n\t * Executed when a page test has ended.\n\t */\n\tasync onTestPagePathEnd? (destination: HotDestination, testPath: HotTestPath, \n\t\tresult: any, continueWhenTestIsComplete?: boolean): Promise<void>;\n\t/**\n\t * Executed when a command is executed.\n\t */\n\tasync onCommand? (destination: HotDestination, page: HotTestPage, stop: HotTestStop, \n\t\tcmd: string, args: string[], cmdFunc: ((cmdArgs: string[]) => Promise<void>)): Promise<void>;\n\t/**\n\t * Executed when tests are finished.\n\t */\n\tasync onTestEnd? (destination: HotDestination): Promise<void>;\n\n\t/**\n\t * Executed when this tester has been executed from the API.\n\t */\n\tasync onExecute? (): Promise<void>;\n\t/**\n\t * Executed when this tester has finished loading all data from the API.\n\t */\n\tasync onFinishedLoading? (): Promise<void>;\n\n\t/**\n\t * Waits for the API to finish loading all data.\n\t */\n\tasync waitForData (): Promise<void>\n\t{\n\t\twhile (this.finishedLoading === false)\n\t\t\tawait HotStaq.wait (10);\n\t}\n\n\t/**\n\t * Get a test page.\n\t */\n\tgetTestPage (destination: HotDestination): HotTestPage\n\t{\n\t\tlet page = this.testMaps[destination.mapName].pages[destination.page];\n\n\t\treturn (page);\n\t}\n\n\t/**\n\t * Get a test path.\n\t */\n\tgetTestPath (destination: HotDestination, pathName: string): HotTestPath\n\t{\n\t\tlet page = this.testMaps[destination.mapName].pages[destination.page];\n\n\t\treturn (page.testPaths[pathName]);\n\t}\n\n\t/**\n\t * Get a destination JSON object to use.\n\t */\n\tstatic interpretDestination (mapName: string, testDest: HotTestDestination): HotDestination\n\t{\n\t\tlet destination: string = testDest.destination;\n\t\tlet newDestination: HotDestination = {\n\t\t\t\tmapName: mapName,\n\t\t\t\tpage: \"\",\n\t\t\t\tapi: \"\",\n\t\t\t\tpaths: []\n\t\t\t};\n\t\tlet strs: string[] = destination.split (/\\-\\>/g);\n\t\tlet type: string = strs[0];\n\n\t\tif (type.length < 2)\n\t\t\treturn (null);\n\n\t\tif ((type[0] === \"/\") && (type[1] === \"/\"))\n\t\t\treturn (null);\n\n\t\tlet getType: (typeStr: string, typeDelimiter: string) => string = \n\t\t\t(typeStr: string, typeDelimiter: string): string =>\n\t\t\t{\n\t\t\t\tlet pos: number = typeStr.indexOf (typeDelimiter);\n\t\t\t\tlet typeValue: string = \"\";\n\t\t\n\t\t\t\tif (pos > -1)\n\t\t\t\t{\n\t\t\t\t\ttypeValue = typeStr.substr (pos + typeDelimiter.length);\n\t\t\t\t\ttypeValue = typeValue.trim ();\n\t\t\t\t}\n\n\t\t\t\treturn (typeValue);\n\t\t\t};\n\n\t\tnewDestination.page = getType (type, \"page:\");\n\t\tnewDestination.api = getType (type, \"api:\");\n\n\t\tfor (let iIdx = 1; iIdx < strs.length; iIdx++)\n\t\t{\n\t\t\tlet newPathStr: string = strs[iIdx];\n\t\t\tlet newPath: HotTestStop = {\n\t\t\t\t\tcmd: \"\",\n\t\t\t\t\tdest: \"\",\n\t\t\t\t\tpath: \"\"\n\t\t\t\t};\n\n\t\t\tnewPathStr = newPathStr.trim ();\n\t\t\tnewPath.dest = getType (newPathStr, \"dest:\");\n\t\t\tnewPath.cmd = getType (newPathStr, \"cmd:\");\n\t\t\tnewPath.path = getType (newPathStr, \"path:\");\n\n\t\t\tif ((newPath.dest == \"\") && (newPath.cmd == \"\") && (newPath.path == \"\"))\n\t\t\t\tnewPath.path = newPathStr;\n\n\t\t\tnewDestination.paths.push (newPath);\n\t\t}\n\n\t\treturn (newDestination);\n\t}\n\n\t/**\n\t * Execute an API's test path.\n\t */\n\tasync executeTestAPIPath (destination: HotDestination, method: HotRouteMethod, \n\t\ttestName: string, skipEventCalls: boolean = false, continueWhenTestIsComplete: boolean = false): Promise<any>\n\t{\n\t\tlet runTestPath: boolean = true;\n\n\t\tif (method == null)\n\t\t\tthrow new Error (`Trying to access null method on destination map ${destination.mapName}.`);\n\n\t\t// A dumb hack to prevent any recursion that could occur.\n\t\tif (skipEventCalls === false)\n\t\t{\n\t\t\tif (this.onTestAPIPathStart != null)\n\t\t\t\trunTestPath = await this.onTestAPIPathStart (destination, method, testName, continueWhenTestIsComplete);\n\t\t}\n\n\t\tlet result: any = null;\n\n\t\tif (runTestPath === true)\n\t\t{\n\t\t\tlet testCaseObject: TestCaseObject = method.testCases[testName];\n\n\t\t\tif (testCaseObject == null)\n\t\t\t\tthrow new Error (`HotTester: Test case object ${testName} does not exist!`);\n\n\t\t\tresult = await testCaseObject.func (this.driver);\n\t\t}\n\n\t\tif (skipEventCalls === false)\n\t\t{\n\t\t\tif (this.onTestAPIPathEnd != null)\n\t\t\t\tawait this.onTestAPIPathEnd (destination, method, testName, result, continueWhenTestIsComplete);\n\t\t}\n\n\t\treturn (result);\n\t}\n\n\t/**\n\t * Execute all test paths in an API route.\n\t * \n\t * @fixme This needs a better implementation...\n\t */\n\tasync executeTestAPIPaths (destination: HotDestination): Promise<any[]>\n\t{\n\t\tlet results: any[] = [];\n\t\tlet testMap: HotTestMap = this.testMaps[destination.mapName];\n\n\t\tif (testMap == null)\n\t\t\tthrow new Error (`HotTester: Map ${destination.mapName} does not exist!`);\n\n\t\tif (this.processor.api == null)\n\t\t\tthrow new Error (`HotTester: Associated processor does not have an API!`);\n\n\t\tlet route: HotRoute = this.processor.api.routes[destination.api];\n\n\t\tif (route == null)\n\t\t\tthrow new Error (`HotTester: API does not have route ${destination.api}!`);\n\n\t\t// Iterate through each path in the destination until complete.\n\t\tfor (let iIdx = 0; iIdx < destination.paths.length; iIdx += 2)\n\t\t{\n\t\t\tlet stop: HotTestStop = destination.paths[iIdx];\n\t\t\tlet pathName: string = stop.path;\n\t\t\tlet method: HotRouteMethod = route.getMethod (pathName);\n\t\t\tlet nextStop: HotTestStop = destination.paths[iIdx + 1];\n\t\t\tlet testName: string = nextStop.path;\n\n\t\t\tif (method == null)\n\t\t\t\tthrow new Error (`Unable to find method related to path ${pathName} in map ${destination.mapName}`);\n\n\t\t\tlet result: any = await this.executeTestAPIPath (destination, method, testName);\n\n\t\t\tresults.push (result);\n\t\t}\n\n\t\treturn (results);\n\t}\n\n\t/**\n\t * Execute a test page path.\n\t */\n\tasync executeTestPagePath (destination: HotDestination, stop: HotTestStop, \n\t\tskipEventCalls: boolean = false, continueWhenTestIsComplete: boolean = false): Promise<any>\n\t{\n\t\tlet runTestPath: boolean = true;\n\t\tlet testMap: HotTestMap = this.testMaps[destination.mapName];\n\n\t\t/// @fixme For some reason the errors being thrown here are not being thrown.\n\t\tif (testMap == null)\n\t\t\tthrow new Error (`HotTester: Map ${destination.mapName} does not exist!`);\n\n\t\tlet page: HotTestPage = testMap.pages[destination.page];\n\n\t\tif (page == null)\n\t\t\tthrow new Error (`HotTester: Page ${destination.page} does not exist!`);\n\n\t\tthis.driver.page = page;\n\n\t\tlet testPathName: string = stop.path;\n\t\tlet testPath: HotTestPath = page.testPaths[testPathName];\n\n\t\t// A dumb hack to prevent any recursion that could occur.\n\t\tif (skipEventCalls === false)\n\t\t{\n\t\t\tif (this.onTestPagePathStart != null)\n\t\t\t\trunTestPath = await this.onTestPagePathStart (destination, page, stop, continueWhenTestIsComplete);\n\t\t}\n\n\t\tlet result: any = null;\n\n\t\tif (runTestPath === true)\n\t\t{\n\t\t\tif (testPath == null)\n\t\t\t{\n\t\t\t\tthrow new Error (`HotTester: Test path ${testPathName} does not have a function!`);\n\t\t\t}\n\n\t\t\tresult = await testPath (this.driver);\n\t\t}\n\n\t\tif (skipEventCalls === false)\n\t\t{\n\t\t\tif (this.onTestPagePathEnd != null)\n\t\t\t\tawait this.onTestPagePathEnd (destination, testPath, result, continueWhenTestIsComplete);\n\t\t}\n\n\t\treturn (result);\n\t}\n\n\t/**\n\t * Execute a command.\n\t */\n\tasync executeCommand (destination: HotDestination, page: HotTestPage, stop: HotTestStop, cmd: string): Promise<void>\n\t{\n\t\t/**\n\t\t * Check if the input command matches.\n\t\t */\n\t\tlet hasCmd: (input: string, cmd: string, hasArguments: boolean) => boolean = \n\t\t\t(input: string, cmd: string, hasArguments: boolean): boolean =>\n\t\t\t{\n\t\t\t\tlet result: boolean = false;\n\n\t\t\t\tif (stop.cmd === cmd)\n\t\t\t\t\tresult = true;\n\n\t\t\t\tconst pos: number = stop.cmd.indexOf (\"(\");\n\n\t\t\t\t// If there's parenthesis, get the incoming command.\n\t\t\t\tif (pos > -1)\n\t\t\t\t{\n\t\t\t\t\tlet inputCmd: string = stop.cmd.substr (0, pos);\n\n\t\t\t\t\tif (inputCmd === cmd)\n\t\t\t\t\t\tresult = true;\n\t\t\t\t}\n\n\t\t\t\treturn (result);\n\t\t\t};\n\t\t/**\n\t\t * Get the arguments in a command. This will only return a \n\t\t * single argument for now.\n\t\t * \n\t\t * @fixme Add support for multiple arguments.\n\t\t */\n\t\tlet getCmdArgs: (input: string) => string[] = \n\t\t\t(input: string): string[] =>\n\t\t\t{\n\t\t\t\tlet results: string[] = [];\n\t\t\t\tlet matches = input.match (/(?=\\()(.*?)(?=\\))/g);\n\n\t\t\t\tif (matches != null)\n\t\t\t\t{\n\t\t\t\t\tlet tempMatch = matches[0];\n\n\t\t\t\t\t// A little hack, since I suck at Regex :(\n\t\t\t\t\ttempMatch = tempMatch.substr (2, tempMatch.length);\n\n\t\t\t\t\tresults.push (tempMatch);\n\t\t\t\t}\n\n\t\t\t\tif (results.length < 1)\n\t\t\t\t\tthrow new Error (`HotTester: Command ${input} requires arguments, but none were supplied.`);\n\n\t\t\t\treturn (results);\n\t\t\t};\n\n\t\tlet cmdFunc: ((cmdArgs: string[]) => Promise<void>) = null;\n\t\tlet args: string[] = [];\n\n\t\tif (hasCmd (stop.cmd, \"waitForTesterAPIData\", false) === true)\n\t\t{\n\t\t\tcmdFunc = async (cmdArgs: string[]): Promise<void> =>\n\t\t\t\t{\n\t\t\t\t\tthis.finishedLoading = false;\n\t\t\t\t\tawait this.waitForData ();\n\t\t\t\t};\n\t\t}\n\n\t\tif (hasCmd (stop.cmd, \"wait\", true) === true)\n\t\t{\n\t\t\targs = getCmdArgs (stop.cmd);\n\n\t\t\tcmdFunc = async (cmdArgs: string[]): Promise<void> =>\n\t\t\t\t{\n\t\t\t\t\tlet numMilliseconds: number = parseInt (cmdArgs[0]);\n\n\t\t\t\t\tawait HotStaq.wait (numMilliseconds);\n\t\t\t\t};\n\t\t}\n\n\t\tif (hasCmd (stop.cmd, \"url\", true) === true)\n\t\t{\n\t\t\targs = getCmdArgs (stop.cmd);\n\n\t\t\tcmdFunc = async (cmdArgs: string[]): Promise<void> =>\n\t\t\t\t{\n\t\t\t\t\tlet input: string = cmdArgs[0];\n\n\t\t\t\t\tawait this.driver.navigateToUrl (input);\n\t\t\t\t};\n\t\t}\n\n\t\tif (hasCmd (stop.cmd, \"print\", true) === true)\n\t\t{\n\t\t\targs = getCmdArgs (stop.cmd);\n\n\t\t\tcmdFunc = async (cmdArgs: string[]): Promise<void> =>\n\t\t\t\t{\n\t\t\t\t\tlet input: string = cmdArgs[0];\n\n\t\t\t\t\tawait this.driver.print (input);\n\t\t\t\t};\n\t\t}\n\n\t\tif (hasCmd (stop.cmd, \"println\", true) === true)\n\t\t{\n\t\t\targs = getCmdArgs (stop.cmd);\n\n\t\t\tcmdFunc = async (cmdArgs: string[]): Promise<void> =>\n\t\t\t\t{\n\t\t\t\t\tlet input: string = cmdArgs[0];\n\n\t\t\t\t\tawait this.driver.println (input);\n\t\t\t\t};\n\t\t}\n\n\t\tif (hasCmd (stop.cmd, \"waitForTestObject\", true) === true)\n\t\t{\n\t\t\targs = getCmdArgs (stop.cmd);\n\n\t\t\tcmdFunc = async (cmdArgs: string[]): Promise<void> =>\n\t\t\t\t{\n\t\t\t\t\tlet testObject: string = JSON.parse (cmdArgs[0]);\n\n\t\t\t\t\tawait this.driver.waitForTestElement (testObject);\n\t\t\t\t};\n\t\t}\n\n\t\tif (cmdFunc == null)\n\t\t\tthrow new Error (`HotTester: Command ${stop.cmd} does not exist!`);\n\n\t\tawait this.onCommand (destination, page, stop, cmd, args, cmdFunc);\n\t}\n\n\t/**\n\t * Execute all test paths in a page.\n\t */\n\tasync executeTestPagePaths (destination: HotDestination, continueWhenTestIsComplete: boolean = false): Promise<any[]>\n\t{\n\t\tlet results: any[] = [];\n\t\tlet testMap: HotTestMap = this.testMaps[destination.mapName];\n\n\t\t/// @fixme For some reason the errors being thrown here are not being thrown.\n\t\tif (testMap == null)\n\t\t\tthrow new Error (`HotTester: Map ${destination.mapName} does not exist!`);\n\n\t\t// Iterate through each path in the destination until complete.\n\t\tfor (let iIdx = 0; iIdx < destination.paths.length; iIdx++)\n\t\t{\n\t\t\tlet stop: HotTestStop = destination.paths[iIdx];\n\t\t\tlet result: any = null;\n\t\t\tlet page: HotTestPage = testMap.pages[destination.page];\n\t\n\t\t\tif (page == null)\n\t\t\t\tthrow new Error (`HotTester: Page ${destination.page} does not exist!`);\n\n\t\t\tif (stop.dest !== \"\")\n\t\t\t{\n\t\t\t\tif (testMap.destinations instanceof Array)\n\t\t\t\t\tthrow new Error (`HotTester: When using type 'dest' in a destination string, all destinations in map ${destination.mapName} must be named.`);\n\n\t\t\t\tlet testDest: HotTestDestination = testMap.destinations[stop.dest];\n\t\t\t\tlet newDestination: HotDestination = HotTester.interpretDestination (\n\t\t\t\t\t\t\t\t\t\t\t\t\tdestination.mapName, testDest);\n\n\t\t\t\tif (newDestination != null)\n\t\t\t\t\tresult = await this.executeTestPagePaths (newDestination);\n\t\t\t}\n\n\t\t\tif (stop.cmd !== \"\")\n\t\t\t\tawait this.executeCommand (destination, page, stop, stop.cmd);\n\n\t\t\tif (stop.path !== \"\")\n\t\t\t\tresult = await this.executeTestPagePath (destination, stop, false, continueWhenTestIsComplete);\n\n\t\t\tresults.push (result);\n\t\t}\n\n\t\treturn (results);\n\t}\n\n\t/**\n\t * Execute the tests.\n\t */\n\tasync execute (mapName: string): Promise<void>\n\t{\n\t\tlet map: HotTestMap = this.testMaps[mapName];\n\n\t\tif (map == null)\n\t\t\tthrow new Error (`HotTester: Map ${mapName} does not exist!`);\n\n\t\t// Process routes testing first.\n\t\tlet routeKey: string = this.processor.getRouteKeyFromName (mapName);\n\t\tlet url: string = \"\";\n\n\t\tif (routeKey !== \"\")\n\t\t\turl = `${this.baseUrl}${routeKey}`;\n\n\t\tlet executeDestination: (testDest: HotTestDestination, destinationKey?: string) => Promise<void> = \n\t\t\tasync (testDest: HotTestDestination, destinationKey: string = \"\") =>\n\t\t\t{\n\t\t\t\tif (testDest.autoStart === false)\n\t\t\t\t\treturn;\n\n\t\t\t\tlet destination: HotDestination = HotTester.interpretDestination (mapName, testDest);\n\t\t\t\tlet isWebRoute: boolean = false;\n\t\t\t\tlet runTestPaths: boolean = true;\n\n\t\t\t\tif (destination.page !== \"\")\n\t\t\t\t\tisWebRoute = true;\n\t\n\t\t\t\tif (this.setup != null)\n\t\t\t\t{\n\t\t\t\t\tif (this.hasBeenSetup === false)\n\t\t\t\t\t{\n\t\t\t\t\t\tawait this.setup (isWebRoute, url, destinationKey);\n\t\t\t\t\t\tthis.hasBeenSetup = true;\n\t\t\t\t\t\tthis.hasBeenDestroyed = false;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\n\t\t\t\tif (this.onTestStart != null)\n\t\t\t\t\trunTestPaths = await this.onTestStart (destination, url, destinationKey);\n\t\n\t\t\t\tif (runTestPaths === true)\n\t\t\t\t{\n\t\t\t\t\tif (destination.page !== \"\")\n\t\t\t\t\t\tawait this.executeTestPagePaths (destination);\n\t\n\t\t\t\t\tif (destination.api !== \"\")\n\t\t\t\t\t\tawait this.executeTestAPIPaths (destination);\n\t\t\t\t}\n\t\t\n\t\t\t\tif (this.onTestEnd != null)\n\t\t\t\t\tawait this.onTestEnd (destination);\n\t\n\t\t\t\tif (this.destroy != null)\n\t\t\t\t{\n\t\t\t\t\tif (this.hasBeenDestroyed === false)\n\t\t\t\t\t{\n\t\t\t\t\t\tawait this.destroy ();\n\t\t\t\t\t\tthis.hasBeenDestroyed = true;\n\t\t\t\t\t\tthis.hasBeenSetup = false;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t};\n\n\t\t// If the map destinations are in an array, just execute those in order.\n\t\tif (map.destinations instanceof Array)\n\t\t{\n\t\t\tfor (let iIdx = 0; iIdx < map.destinations.length; iIdx++)\n\t\t\t{\n\t\t\t\tlet testDest: HotTestDestination = map.destinations[iIdx];\n\n\t\t\t\tawait executeDestination (testDest);\n\t\t\t}\n\t\t}\n\t\telse\n\t\t{\n\t\t\t// If there's a destination order, use that.\n\t\t\tif (map.destinationOrder.length > 0)\n\t\t\t{\n\t\t\t\tlet hasExecutedKeys: string[] = [];\n\n\t\t\t\t// Go through the destination order and execute each one.\n\t\t\t\tfor (let iIdx = 0; iIdx < map.destinationOrder.length; iIdx++)\n\t\t\t\t{\n\t\t\t\t\tlet orderKey: string = map.destinationOrder[iIdx];\n\t\t\t\t\tlet testDest: HotTestDestination = map.destinations[orderKey];\n\n\t\t\t\t\tif (testDest == null)\n\t\t\t\t\t\tthrow new Error (`HotTester: Destination ${orderKey} does not exist!`);\n\n\t\t\t\t\thasExecutedKeys.push (orderKey);\n\t\t\t\t\tawait executeDestination (testDest, orderKey);\n\t\t\t\t}\n\n\t\t\t\t// Execute the rest of the destinations that have not been executed yet.\n\t\t\t\tfor (let key in map.destinations)\n\t\t\t\t{\n\t\t\t\t\tlet executeDest: boolean = true;\n\n\t\t\t\t\tfor (let iIdx = 0; iIdx < hasExecutedKeys.length; iIdx++)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet executedKey: string = hasExecutedKeys[iIdx];\n\n\t\t\t\t\t\tif (executedKey === key)\n\t\t\t\t\t\t{\n\t\t\t\t\t\t\texecuteDest = false;\n\n\t\t\t\t\t\t\tbreak;\n\t\t\t\t\t\t}\n\t\t\t\t\t}\n\n\t\t\t\t\tif (executeDest === true)\n\t\t\t\t\t{\n\t\t\t\t\t\tlet testDest: HotTestDestination = map.destinations[key];\n\n\t\t\t\t\t\tawait executeDestination (testDest, key);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t\telse\n\t\t\t{\n\t\t\t\t// Execute the destinations in any order.\n\t\t\t\tfor (let key in map.destinations)\n\t\t\t\t{\n\t\t\t\t\tlet testDest: HotTestDestination = map.destinations[key];\n\n\t\t\t\t\tawait executeDestination (testDest, key);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\t// End of routes testing\n\n\t\t// Start of API testing\n\n\t}\n}","import { EventExecutionType, HotAPI } from \"./HotAPI\";\nimport { HotRoute } from \"./HotRoute\";\nimport { HotClient } from \"./HotClient\";\nimport { HotServer } from \"./HotServer\";\nimport { HotTestDriver } from \"./HotTestDriver\";\nimport { HotTester } from \"./HotTester\";\nimport { HotTestMap, HotTestPath, HotTestPage } from \"./HotTestMap\";\nimport { HTTPMethod } from \"./HotRouteMethod\";\n\nexport class HotTesterAPI extends HotAPI\n{\n\tconstructor (baseUrl: string, connection: HotServer | HotClient = null, db: any = null)\n\t{\n\t\tsuper(baseUrl, connection, db);\n\n\t\tthis.executeEventsUsing = EventExecutionType.HotAPI;\n\n\t\tlet route: HotRoute = new HotRoute (connection, \"tester\");\n\t\troute.addMethod ({\n\t\t\t\t\"name\": \"pageLoaded\", \n\t\t\t\t\"onServerExecute\": this.pageLoaded,\n\t\t\t\t\"parameters\": {\n\t\t\t\t\t\t\"testerName\": {\n\t\t\t\t\t\t\t\"required\": true,\n\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\"description\": \"The name of the tester executing the test.\"\n\t\t\t\t\t\t},\n\t\t\t\t\t\t\"testerMap\": {\n\t\t\t\t\t\t\t\"required\": true,\n\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\"description\": \"The tester map executing the test.\"\n\t\t\t\t\t\t},\n\t\t\t\t\t\t\"pageName\": {\n\t\t\t\t\t\t\t\"required\": true,\n\t\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\t\"description\": \"The name of the page executing the test.\"\n\t\t\t\t\t\t},\n\t\t\t\t\t\t\"testElements\": {\n\t\t\t\t\t\t\t\"required\": true,\n\t\t\t\t\t\t\t\"type\": \"array\",\n\t\t\t\t\t\t\t\"description\": \"The test elements on the page.\"\n\t\t\t\t\t\t},\n\t\t\t\t\t\t\"testPaths\": {\n\t\t\t\t\t\t\t\"required\": true,\n\t\t\t\t\t\t\t\"type\": \"array\",\n\t\t\t\t\t\t\t\"description\": \"The test paths on the page.\"\n\t\t\t\t\t\t}\n\t\t\t\t\t},\n\t\t\t\t\"returns\": \"Returns true as an acknowledgement.\"\n\t\t\t});\n\t\troute.addMethod ({\n\t\t\t\t\"name\": \"executeTests\",\n\t\t\t\t\"onServerExecute\": this.executeTests,\n\t\t\t\t\"parameters\": {\n\t\t\t\t\t\"testerName\": {\n\t\t\t\t\t\t\"required\": true,\n\t\t\t\t\t\t\"type\": \"string\",\n\t\t\t\t\t\t\"description\": \"The name of the tester executing the test.\"\n\t\t\t\t\t},\n\t\t\t\t\t\"testerMap\": {\n\t\t\t\t\t\t\"required\": true,\n\t\t\t\t\t\t\"type\": \"object\",\n\t\t\t\t\t\t\"description\": \"The tester map to execute.\"\n\t\t\t\t\t}\n\t\t\t\t},\n\t\t\t\t\"returns\": \"Returns true when tests are complete.\"\n\t\t\t});\n\t\troute.addMethod ({\n\t\t\t\t\"name\": \"heartbeat\",\n\t\t\t\t\"type\": HTTPMethod.GET,\n\t\t\t\t\"onServerExecute\": this.heartbeat,\n\t\t\t\t\"returns\": \"Returns true as an acknowledgement.\"\n\t\t\t});\n\t\tthis.addRoute (route);\n\t}\n\n\t/**\n\t * This is called when the page has finished loading in development mode.\n\t */\n\tasync pageLoaded (req: any, res: any, authorizedValue: any, jsonObj: any, queryObj: any): Promise<any>\n\t{\n\t\tlet testerObj: {\n\t\t\t\ttesterName: string;\n\t\t\t\ttesterMap: string;\n\t\t\t\tpageName: string;\n\t\t\t\ttestElements: any;\n\t\t\t\ttestPathsStrs: any;\n\t\t\t} = {\n\t\t\t\ttesterName: jsonObj[\"testerName\"],\n\t\t\t\ttesterMap: jsonObj[\"testerMap\"],\n\t\t\t\tpageName: jsonObj[\"pageName\"],\n\t\t\t\ttestElements: jsonObj[\"testElements\"],\n\t\t\t\ttestPathsStrs: jsonObj[\"testPaths\"]\n\t\t\t};\n\n\t\tfor (let key in testerObj)\n\t\t{\n\t\t\t// @ts-ignore\n\t\t\tlet testObj: any = testerObj[key];\n\t\t\tlet throwError: boolean = false;\n\n\t\t\tif (testObj == null)\n\t\t\t\tthrowError = true;\n\n\t\t\tif ((testerObj.testerName == \"\") || \n\t\t\t\t(testerObj.testerMap === \"\") || \n\t\t\t\t(testerObj.testElements === \"\") || \n\t\t\t\t(testerObj.testPathsStrs === \"\"))\n\t\t\t{\n\t\t\t\tthrowError = true;\n\t\t\t}\n\n\t\t\tif (throwError === true)\n\t\t\t\tthrow new Error (`TesterAPI: Object ${key} was not passed.`);\n\t\t}\n\n\t\ttesterObj.testElements = JSON.parse (testerObj.testElements);\n\t\ttesterObj.testPathsStrs = JSON.parse (testerObj.testPathsStrs);\n\n\t\tlet testPaths: { [name: string]: HotTestPath; } = {};\n\n\t\tfor (let key in testerObj.testPathsStrs)\n\t\t{\n\t\t\tlet testPath: (driver: HotTestDriver, ...args: any) => Promise<any> = \n\t\t\t\teval (testerObj.testPathsStrs[key]);\n\n\t\t\ttestPaths[key] = testPath;\n\t\t}\n\n\t\tlet tester: HotTester = this.connection.processor.testers[testerObj.testerName];\n\n\t\tif (tester == null)\n\t\t\tthrow new Error (`TesterAPI: Tester ${testerObj.testerMap} does not exist!`);\n\n\t\tlet testMap: HotTestMap = tester.testMaps[testerObj.testerMap];\n\n\t\tif (testMap == null)\n\t\t\tthrow new Error (`TesterAPI: Tester map ${testerObj.testerMap} does not exist!`);\n\n\t\ttestMap.pages[testerObj.pageName] = {\n\t\t\t\t\"testElements\": {},\n\t\t\t\t\"testPaths\": {}\n\t\t\t};\n\t\ttestMap.pages[testerObj.pageName].testElements = testerObj.testElements;\n\t\ttestMap.pages[testerObj.pageName].testPaths = testPaths;\n\n\t\ttester.finishedLoading = true;\n\n\t\tif (tester.onFinishedLoading != null)\n\t\t\tawait tester.onFinishedLoading ();\n\n\t\treturn (true);\n\t}\n\n\t/**\n\t * Execute the tests for a page.\n\t */\n\tasync executeTests (req: any, res: any, authorizedValue: any, jsonObj: any, queryObj: any): Promise<any>\n\t{\n\t\tlet testerName: string = jsonObj[\"testerName\"];\n\t\tlet testerMap: string = jsonObj[\"testerMap\"];\n\n\t\tif ((testerName == null) || (testerMap == null))\n\t\t\tthrow new Error (\"TesterAPI: Not all required json objects were passed.\");\n\n\t\tif ((testerName === \"\") || (testerMap === \"\"))\n\t\t\tthrow new Error (\"TesterAPI: Not all required json objects were passed.\");\n\n\t\tlet server: HotServer = (<HotServer>this.connection);\n\n\t\t// @ts-ignore\n\t\tif (server.executeTests != null)\n\t\t{\n\t\t\t// @ts-ignore\n\t\t\tawait server.executeTests (testerName, testerMap);\n\t\t}\n\n\t\treturn (true);\n\t}\n\n\t/**\n\t * Responds with true to heartbeat requests.\n\t */\n\tasync heartbeat (req: any, res: any, authorizedValue: any, jsonObj: any, queryObj: any): Promise<any>\n\t{\n\t\treturn (true);\n\t}\n}","import { HotStaq, HotStartOptions, IHotStaq, \n\tHotSite, HotSiteRoute, HotSiteMapPath } from \"./HotStaq\";\nimport { Hot, DeveloperMode } from \"./Hot\";\nimport { HotComponent, IHotComponent } from \"./HotComponent\";\nimport { HotFile } from \"./HotFile\";\nimport { HotLog, HotLogLevel } from \"./HotLog\";\nimport { HotPage } from \"./HotPage\";\n\n// Server stuff\nimport { HotAPI, EventExecutionType, APItoLoad } from \"./HotAPI\";\nimport { HotRoute } from \"./HotRoute\";\nimport { HotRouteMethod, HTTPMethod, ServerAuthorizationFunction, ServerExecutionFunction } from \"./HotRouteMethod\";\nimport { HotServer, HotServerType } from \"./HotServer\";\nimport { HotClient } from \"./HotClient\";\n\n// Testing stuff\nimport { HotTestDriver } from \"./HotTestDriver\";\nimport { IHotTestElement, HotTestElement, HotTestElementOptions, IHotTestElementOptions } from \"./HotTestElement\";\nimport { HotTester, HotTestStop, HotDestination } from \"./HotTester\";\nimport { HotTesterAPI } from \"./HotTesterAPI\";\nimport { HotTestMap, HotTestDestination, HotTestPath, HotTestPage } from \"./HotTestMap\";\n\nHotStaq.isWeb = true;\n\n// Can't export interfaces from here :(\n\nmodule.exports[\"HotStaq\"] = HotStaq;\nmodule.exports[\"Hot\"] = Hot;\nmodule.exports[\"DeveloperMode\"] = DeveloperMode;\nmodule.exports[\"HotComponent\"] = HotComponent;\nmodule.exports[\"HotAPI\"] = HotAPI;\nmodule.exports[\"EventExecutionType\"] = EventExecutionType;\nmodule.exports[\"HotFile\"] = HotFile;\nmodule.exports[\"HotLog\"] = HotLog;\nmodule.exports[\"HotLogLevel\"] = HotLogLevel;\nmodule.exports[\"HotPage\"] = HotPage;\nmodule.exports[\"HotRoute\"] = HotRoute;\nmodule.exports[\"HotRouteMethod\"] = HotRouteMethod;\nmodule.exports[\"HTTPMethod\"] = HTTPMethod;\nmodule.exports[\"HotServer\"] = HotServer;\nmodule.exports[\"HotServerType\"] = HotServerType;\nmodule.exports[\"HotClient\"] = HotClient;\nmodule.exports[\"HotTester\"] = HotTester;\nmodule.exports[\"HotTesterAPI\"] = HotTesterAPI;\nmodule.exports[\"HotTestMap\"] = HotTestMap;\nmodule.exports[\"HotTestDestination\"] = HotTestDestination;\nmodule.exports[\"HotTestElement\"] = HotTestElement;\nmodule.exports[\"HotTestElementOptions\"] = HotTestElementOptions;\nmodule.exports[\"HotTestDriver\"] = HotTestDriver;","module.exports = {};","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","// startup\n// Load entry module and return exports\n// This entry module used 'module' so it can't be inlined\nvar __webpack_exports__ = __webpack_require__(859);\n"],"names":["module","exports","self","FormData","window","assign","target","i","arguments","length","source","key","init","converter","defaultAttributes","set","value","attributes","document","expires","Date","now","toUTCString","encodeURIComponent","replace","decodeURIComponent","escape","stringifiedAttributes","attributeName","split","cookie","write","Object","create","get","cookies","jar","parts","slice","join","foundKey","read","e","remove","withAttributes","this","withConverter","freeze","path","factory","global","Error","getGlobal","fetch","bind","Headers","Request","Response","DeveloperMode","Hot","static","file","args","HotStaq","isWeb","lowerFile","toLowerCase","indexOf","echo","getFile","res","httpRequest","output","text","eval","apply","fileName","tempFile","CurrentPage","processor","page","content","process","HotFile","url","localFile","load","route","data","httpMethod","files","result","api","makeCall","fetchObj","JSON","stringify","ok","status","statusText","json","ex","message","requestInit","Output","iIdx","CSS","cssFile","cssOut","cssStr","JSFiles","jsFile","jsFileOut","jsFileStr","JSScripts","jsScript","jsScriptOut","jsScriptsStr","Arguments","HotTestElement","Mode","Production","API","TesterAPI","Data","Cookies","PublicKeys","EventExecutionType","constructor","baseUrl","connection","db","description","createFunctions","executeEventsUsing","HotRoute","authCredentials","userAuth","routes","onPreRegister","onPostRegister","setDBSchema","schema","getDB","getDBSchema","addRoute","routeMethod","executeFunction","routeName","HotRouteMethod","addMethod","newRoute","methods","currentRoute","newRouteMethod","name","type","routeStr","version","authCredential","registerRoute","HotServer","registerRoutes","toUpperCase","substr","keys","formData","append","method","body","jsonRes","headers","Promise","resolve","reject","then","jsonObj","catch","reason","testerAPI","HotServerType","HTTP","logger","copy","htmlElements","tag","elementOptions","undefined","observedAttributes","inner","events","onCreated","element","throwAllErrors","setContent","getContent","loadUrl","httpGet","loadLocalFile","fs","readFile","err","toString","contentRegex","contentProcessor","offContentProcessor","numRemoveFromBeginning","numRemoveFromEnd","exec","previousIndex","start","index","end","lastIndex","prevContent","startChars","endChars","triggerChar","pos","startTriggerPos","nestedCounter","rpos","lastIndexOf","epos","tempepos","thisContent","parserOptions","outputCommands","allowStringify","STRINGIFY_START","STRINGIFY_END","processContent","RegExp","regexFound","offContent","tempOutput","processNestedContent","regexFound2","offContent3","tempOutput2","out","tempOutput3","tempOutput4","Development","foundStr","parse","createTestElement","foundStr2","testElm","obj","Array","testElements","func","tempOutput5","escapedContent","mode","publicKeys","getAPI","getTesterAPI","parseContent","returnedOutput","executionContent","newVar","newVarValue","contentName","Function","SyntaxError","hot","finalOutput","HotLogLevel","logLevel","All","log","level","Verbose","error","Warning","warning","Info","info","verbose","console","warn","msg","stack","testerName","testerMap","testPaths","addFile","push","addTestElement","elm","getTestElement","createTestPath","pathName","driverFunc","onRegister","onAuthorizeUser","prefix","errors","createError","HTTPMethod","POST","testCases","getMethod","foundMethod","onExecute","onServerAuthorize","returns","parameters","param","onServerExecute","onClientExecute","isRegistered","executeSetup","testCaseName","addTestCase","newTestCase","testCaseFunction","testCaseId","testCase","serverType","listenAddress","ports","http","https","ssl","cert","ca","redirectHTTPtoHTTPS","secrets","setAPI","HotTesterMocha","HotTesterMochaSelenium","HotTestSeleniumDriver","pages","components","hotSite","apiContent","testerApiContent","pageContent","HotLog","None","testers","parseInt","objWithParam","required","throwException","defaultValue","errType","redirectToUrl","location","href","numMilliseconds","setTimeout","addPage","getPage","pageName","context","val","aryArgs","prototype","call","addComponent","ComponentType","tempApi","tempComponentObj","componentType","registerComponent","str","tempStr","parsedXML","DOMParser","parseFromString","querySelector","documentElement","children","tagName","fixedStr","customElements","processorComponents","define","HTMLElement","super","componentInfo","component","looseParseFromString","parser","xdom","hdom","elem","from","appendChild","querySelectorAll","outerHTML","connectedCallback","hotComponent","innerHTML","handleAttributes","attr","attrName","attrValue","attrTempName","substring","onPreOutput","outputs","onPostOutput","componentOutputs","html","iKdx","htmlStr","addFunctionsTo","onParsed","htmlHandler","onFixHTML","fixHTML","newDOM","newObj","onParseDOM","throwErr","child","childrenToReadd","removeChild","replaceWith","click","onclick","event","addEventListener","options","objectFunctions","getOwnPropertyNames","objFunc","isNewFunction","key2","HotComponent","keepContext","onPrePlace","compHtmlElement2","onParentPlace","parentSelector","parentNode","parentElement","placeHereParent","parentNodeToCheck","parentNodeCheckCounter","HTMLHtmlElement","placeHereArray","placeHere","placeHereAttrArray","onPostPlace","parent","foundParent","results","hotsiteName","throwTheException","hotsite","params","prevValue","processHotSite","checkHotSiteName","testerUrl","tester","driver","testing","setupTester","parentObj","createNewTester","testerAPIUrl","commandDelay","web","HotPage","mapName","testMap","map","testMaps","HotTestMap","destinations","dest","HotTestDestination","destinationOrder","apis","componentUrl","ComponentClass","disableFileLoading","loadHotFiles","addTester","loadHotSite","jsonStr","ppath","normalize","hotsitePath","forceContentLoading","newFile","loadContent","filepath","generateContent","routeKey","jsSrcPath","passArgs","apiScripts","apiCode","server","globalApi","sendJSContent","jsapi","libraryName","apiName","tempAPIContent","jsapipath","secret","serverConn","passSecretFromAPI","env","envKey","tempContent","developerModeStr","testerAPIStr","loadFiles","fileUrl","fileContent","testerLaunchpadUrl","launchpadUrl","fixContent","createExpressRoutes","expressApp","req","send","getWebTestingMaps","maps","getAPITestingMaps","getRouteKeyFromName","getRouteFromName","foundRoute","executeTests","execute","executeAllWebTests","executeAllAPITests","localFilepath","readyFunc","readyState","getElementsByTagName","tmpScripts","scripts","s","createElement","resolve2","reject2","onload","hasSrc","getAttribute","setAttribute","isReadyForTesting","wait","onReadyForTesting","testerAPIBaseUrl","client","HotClient","HotTesterAPI","onReady","setupTesters","processUrl","setupClientTesters","useOutput","loadHotStaqSite","hotstaqElms","HotStaqWeb","HotAPI","hotstaqElm","getAttr","attrNames","loadPage","router","apiLibrary","apiUrl","testerApiBaseUrl","dontReuseProcessor","passRawUrl","htmlSource","routerManager","routerWildcards","search","URLSearchParams","hstqbaseredirect","decodeURI","history","replaceState","hotstaqErrors","hotstaqErrorElm","errorStatus","unsupportedBrowser","executeError","hotstaqRouterElms","hotstaqRouterElm","routerName","serveLocally","iJdx","childNodes","routerElm","routerPath","redirect","baseRedirect","base","routerSrc","src","checkPath","pathname","gotoPath","lowerServeLocally","lastSlashPos","routeWildcard","tempRouteWildcard","encodeURI","hasHtmlSource","htmlSourceCheck","tempMode","parentLib","newAPI","displayUrl","displayContent","ethereum22","ethereum","persistentData","parseTestObject","wildcard","selector","print","stdout","println","assert","errorMessage","run","executions","execution","runCommand","mustBeVisible","ignoreMissingElementError","destination","autoStart","HotTester","finishedLoading","hasBeenSetup","hasBeenDestroyed","waitForData","getTestPage","getTestPath","testDest","newDestination","paths","strs","getType","typeStr","typeDelimiter","typeValue","trim","newPathStr","newPath","cmd","executeTestAPIPath","testName","skipEventCalls","continueWhenTestIsComplete","runTestPath","onTestAPIPathStart","testCaseObject","onTestAPIPathEnd","executeTestAPIPaths","executeTestPagePath","stop","testPathName","testPath","onTestPagePathStart","onTestPagePathEnd","executeCommand","hasCmd","input","hasArguments","getCmdArgs","matches","match","tempMatch","cmdFunc","cmdArgs","navigateToUrl","testObject","waitForTestElement","onCommand","executeTestPagePaths","interpretDestination","executeDestination","destinationKey","isWebRoute","runTestPaths","setup","onTestStart","onTestEnd","destroy","hasExecutedKeys","orderKey","executeDest","pageLoaded","GET","heartbeat","authorizedValue","queryObj","testerObj","testPathsStrs","testObj","throwError","onFinishedLoading","HotTestElementOptions","HotTestDriver","__webpack_module_cache__","__webpack_require__","moduleId","cachedModule","__webpack_modules__","__webpack_exports__"],"sourceRoot":""}